index.js 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  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 media;
  136. await replyTo;
  137. await forwarded;
  138. owner = await owner;
  139. socket.emit('msg', {_id, createdAt, text, chat: chat && {_id: chat._id, title: chat.title},
  140. media: media && {_id: media._id, url: media.url},
  141. replyTo: replyTo && {_id: replyTo._id, text: replyTo.text},
  142. forwarded: forwarded && {_id: forwarded._id, text: forwarded.text},
  143. owner: {_id: owner._id, login: owner.login, nick: owner.nick}})
  144. }
  145. }
  146. }
  147. async function chatWatcher({members, _id, createdAt, title, messages, avatar, lastModified}){
  148. console.log('CHAT SAVE', _id, title)
  149. for (let [id,{user, socket}] of Object.entries(sockets)){
  150. if (members.some(member => member._id.toString() === user._id.toString())){
  151. //await chat;
  152. //await media;
  153. //await replyTo;
  154. //await forwarded;
  155. await avatar;
  156. await Promise.all(members)
  157. //owner = await owner;
  158. console.log('EMIT', _id, createdAt, title)
  159. socket.emit('chat', {_id, createdAt, title, avatar: avatar && {_id: avatar._id, url: avatar.url}, members: members.map(({_id, login, nick}) => ({_id, login, nick})), lastModified})
  160. }
  161. }
  162. }
  163. io.on('connection', socket => {
  164. console.log('connect', socket.id)
  165. socket.on('disconnect', () => {
  166. console.log('disconnect', socket.id)
  167. delete sockets[socket.id]
  168. })
  169. socket.on('jwt', async token =>{
  170. let decoded;
  171. try {
  172. decoded = jwt.verify(token, jwtSecret)
  173. }
  174. catch (e){
  175. socket.emit('jwt_fail', e)
  176. return
  177. }
  178. socket.emit('jwt_ok', decoded)
  179. //sockets[socket.id] = decoded
  180. sockets[socket.id] = {decoded, user: (await getModels(decoded.sub)).thisUser, socket}
  181. })
  182. })
  183. let socketPath = 5001
  184. http.listen(socketPath, () => {
  185. console.log(`Express GraphQL Server Now Running On ${socketPath}/graphql`);
  186. });
  187. })()