index.js 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. const jwtSecret = '4atikJgznM'
  2. const jwt = require('jsonwebtoken')
  3. const express = require('express');
  4. const express_graphql = require('express-graphql');
  5. const { buildSchema, printSchema, GraphQLString } = require('graphql');
  6. const expand = require('mm-graphql/expand')
  7. const fs = require('fs')
  8. const uploadPath = `${__dirname}/public/images/`;
  9. const upload = require('multer')({ dest: uploadPath })
  10. ;(async () => {
  11. const {Savable, slice, getModels} = await require('./models.js')()
  12. const { jwtGQLAnon, jwtCheck } = require('mm-graphql/jwt')
  13. let schema = buildSchema(`
  14. type User {
  15. _id: ID
  16. createdAt: String
  17. login: String
  18. nick : String
  19. acl: [String]
  20. avatar: Media
  21. chats: [Chat]
  22. }
  23. input UserInput {
  24. _id: ID
  25. login: String
  26. nick : String
  27. password: String
  28. acl: [String]
  29. avatar: MediaInput
  30. chats: [ChatInput]
  31. }
  32. type Message {
  33. _id: ID
  34. createdAt: String
  35. owner: User
  36. chat: Chat
  37. text: String
  38. media: [Media]
  39. replies: [Message]
  40. replyTo: Message
  41. forwarded: Message
  42. forwardWith: [Message]
  43. }
  44. input MessageInput {
  45. _id: ID
  46. text: String
  47. chat: ChatInput
  48. media: [MediaInput]
  49. replies: [MessageInput]
  50. replyTo: MessageInput
  51. forwarded: MessageInput
  52. forwardWith: [MessageInput]
  53. }
  54. type Chat {
  55. _id: ID
  56. createdAt: String
  57. lastModified: String
  58. owner: User
  59. title: String
  60. members: [User]
  61. messages: [Message]
  62. avatar: Media
  63. }
  64. input ChatInput {
  65. _id: ID
  66. title: String
  67. members: [UserInput]
  68. messages: [MessageInput]
  69. }
  70. type Media {
  71. _id: ID,
  72. createdAt: String
  73. owner: User
  74. text: String,
  75. url: String,
  76. originalFileName: String,
  77. type: String
  78. userAvatar: User
  79. chatAvatars: [Chat]
  80. messages: [Message]
  81. }
  82. input MediaInput {
  83. _id: ID,
  84. text: String,
  85. userAvatar: UserInput,
  86. chatAvatars: [ChatInput]
  87. messages: [MessageInput]
  88. }
  89. `);
  90. schema = expand(schema, {
  91. login:{
  92. type: GraphQLString,
  93. args: {login: {type: GraphQLString},
  94. password: {type: GraphQLString},
  95. },
  96. async resolve(root, {login, password}, context, info){
  97. const Savable = context.models.Savable
  98. if (!login || !password) return null;
  99. const user = await Savable.m.User.findOne({login, password})
  100. console.log(user, {login, password})
  101. if (!user)
  102. return null;
  103. const token = jwt.sign({ sub: {id: user._id, login, acl: user.acl}}, jwtSecret); //подписывам токен нашим ключем
  104. return token
  105. }
  106. }
  107. })
  108. console.log(printSchema(schema))
  109. const app = express();
  110. const http = require('http').Server(app);
  111. const io = require('socket.io')(http);
  112. app.use(require('cors')())
  113. app.use(express.static('public'));
  114. app.use('/graphql', express_graphql(jwtGQLAnon({schema, createContext: decoded => getModels(decoded, messageWatcher, chatWatcher), graphiql: true, secret: jwtSecret})))
  115. app.post('/upload', upload.single('media'), async (req, res, next) => {
  116. let decoded;
  117. if (decoded = jwtCheck(req, jwtSecret)){
  118. console.log('SOME UPLOAD', decoded, req.file)
  119. let {models: {Media }} = await getModels(decoded.sub)
  120. let media = await Media.fromFileData(req.file)
  121. res.end(JSON.stringify({_id: media._id, url: media.url}))
  122. }
  123. else {
  124. res.status(503).send('permission denied')
  125. }
  126. })
  127. app.use(express.static('public'));
  128. const sockets = {}
  129. async function messageWatcher(msg){
  130. console.log('NEW MSG', msg.text, msg.chat.title, msg.owner.login)
  131. for (let [id,{user, socket}] of Object.entries(sockets)){
  132. if (user.chats.some(chat => chat._id.toString() === msg.chat._id.toString())){
  133. let {_id, createdAt, text, chat, media, replyTo, forwarded, owner} = msg
  134. await chat;
  135. await Promise.all(media || []);
  136. await replyTo;
  137. await forwarded;
  138. owner = await owner;
  139. await owner.avatar
  140. console.log('MEDIA', media)
  141. socket.emit('msg', {_id, createdAt: createdAt.getTime(), text, chat: chat && {_id: chat._id, title: chat.title},
  142. media: media && media.map(media => ({_id: media._id, url: media.url, type: media.type})),
  143. replyTo: replyTo && {_id: replyTo._id, text: replyTo.text},
  144. forwarded: forwarded && {_id: forwarded._id, text: forwarded.text},
  145. owner: {_id: owner._id, login: owner.login, nick: owner.nick, avatar: owner.avatar && {url: owner.avatar.url}}})
  146. }
  147. }
  148. }
  149. async function chatWatcher({_loadRelations, members, _id, createdAt, title, messages, avatar, lastModified}){
  150. console.log('CHAT SAVE', _id, title)
  151. for (let [id,socketData] of Object.entries(sockets)){
  152. const {user, socket} = socketData
  153. const isMember = members.some(member => member._id.toString() === user._id.toString())
  154. const wasMember =(_loadRelations.members && _loadRelations.members.some(member => member._id.toString() === user._id.toString()))
  155. if (isMember || wasMember){
  156. //await chat;
  157. //await media;
  158. //await replyTo;
  159. //await forwarded;
  160. await avatar;
  161. await Promise.all(members)
  162. //owner = await owner;
  163. console.log('EMIT', _id, createdAt.getTime(), title)
  164. socket.emit(isMember ? 'chat' : 'chat_left', {_id, createdAt: createdAt.getTime(), title, avatar: avatar && {_id: avatar._id, url: avatar.url}, members: members.map(({_id, login, nick}) => ({_id, login, nick})), lastModified})
  165. socketData.user = (await getModels(socketData.decoded.sub)).thisUser //re-read chat permissions
  166. }
  167. }
  168. }
  169. io.on('connection', socket => {
  170. console.log('connect', socket.id)
  171. socket.on('disconnect', () => {
  172. console.log('disconnect', socket.id)
  173. delete sockets[socket.id]
  174. })
  175. socket.on('jwt', async token =>{
  176. let decoded;
  177. try {
  178. decoded = jwt.verify(token, jwtSecret)
  179. }
  180. catch (e){
  181. socket.emit('jwt_fail', e)
  182. return
  183. }
  184. socket.emit('jwt_ok', decoded)
  185. //sockets[socket.id] = decoded
  186. sockets[socket.id] = {decoded, user: (await getModels(decoded.sub)).thisUser, socket}
  187. })
  188. })
  189. let socketPath = 5001
  190. http.listen(socketPath, () => {
  191. console.log(`Express GraphQL Server Now Running On ${socketPath}/graphql`);
  192. });
  193. })()