app.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364
  1. const express = require('express');
  2. const app = express()
  3. const cors = require("cors");
  4. const http = require('http'); //create new http
  5. const mongoose = require('mongoose');
  6. const socket = require("socket.io");
  7. const User = require('./db/models/User');
  8. const Message = require('./db/models/Message');
  9. const jwt = require('jsonwebtoken');
  10. const bcrypt = require('bcrypt');
  11. require('dotenv').config(); // add dotnv for config
  12. const Uuid = require('uuid'); //lib for unic id generate
  13. const fileupload = require('express-fileupload');
  14. const fs = require('fs');
  15. const server = http.createServer(app);
  16. app.use(cors());
  17. app.use(express.json());
  18. app.use(fileupload())
  19. app.use(express.static('avatars')); //folder for static files
  20. const io = require("socket.io")(server, {
  21. cors: {
  22. origin: "http://localhost:3000" //client endpoint and port
  23. }
  24. });
  25. const PORT = process.env.PORT || 5000;
  26. const TOKEN_KEY = process.env.TOKEN_KEY || 'rGH4r@3DKOg06hgj';
  27. const HASH_KEY = 7;
  28. const STATIC_PATH = process.env. STATIC_PATH || 'avatars';
  29. const generateToken = (id, userName, isAdmin) => {
  30. const payload = {
  31. id,
  32. userName,
  33. isAdmin
  34. }
  35. return jwt.sign(payload, TOKEN_KEY);
  36. }
  37. const isValidUserName = (userName) => {
  38. const nameRegex = /[^A-Za-z0-9]/ ;
  39. return (!nameRegex.test(userName) && userName.trim().length > 2);
  40. }
  41. const getAllDbUsers = async (socket) => {
  42. const usersDb = await User.find({})
  43. socket.emit('allDbUsers', usersDb)
  44. }
  45. const getOneUser = async (userName) => {
  46. const userInDb = await User.findOne({userName});
  47. return userInDb;
  48. }
  49. app.post('/login', async (req, res) => {
  50. try {
  51. const {userName,password} = req.body;
  52. if (!isValidUserName(userName)){
  53. return res.status(400).json({message: 'Invalid username'})
  54. }
  55. const dbUser = await getOneUser(userName)
  56. if (dbUser?.isBanned){
  57. return res.status(401).json({message: 'Your account has been banned!!!'})
  58. }
  59. const hashPassword = await bcrypt.hash(password, HASH_KEY);
  60. if (!dbUser) {
  61. const user = new User({
  62. userName,
  63. hashPassword,
  64. isAdmin: !await User.count().exec(),
  65. isBanned: false,
  66. isMutted: false,
  67. avatar: '',
  68. messages: []
  69. });
  70. await user.save()
  71. return res.json({
  72. token: generateToken(user.id, user.userName, user.isAdmin)
  73. });
  74. }
  75. if (dbUser && !bcrypt.compareSync(password, dbUser.hashPassword)){
  76. return res.status(400).json({message: 'Invalid credantials'})
  77. }
  78. res.json({
  79. token: generateToken(dbUser.id, dbUser.userName, dbUser.isAdmin),
  80. })
  81. } catch (e) {
  82. console.log(e);
  83. res.status(500).json({message: `Error ${e}`});
  84. }
  85. })
  86. app.post('/avatar', async (req, res) => {
  87. if (!req.files || Object.keys(req.files).length === 0) {
  88. return res.status(400).json('No files were uploaded.');
  89. }
  90. try {
  91. const file = req.files.file;
  92. const user = jwt.verify(req.body.token, TOKEN_KEY);
  93. const avatarFileName = Uuid.v4() + '.jpeg';
  94. file.mv(STATIC_PATH + '\/' + avatarFileName)
  95. const userFromDb = await getOneUser(user.userName);
  96. if(userFromDb.avatar){
  97. const oldAvatar = userFromDb.avatar;
  98. fs.unlinkSync(STATIC_PATH + '\/' + oldAvatar)
  99. }
  100. await User.findOneAndUpdate({userName: user.userName},{ $set: {'avatar': avatarFileName}}, {
  101. new: true
  102. });
  103. return res.json({ message:'Avatar was uploud succesfully...', avatarUrl: avatarFileName})
  104. } catch (error) {
  105. res.status(500).json({message: `Error uppload file to serverp: ${error}`});
  106. }
  107. })
  108. app.post('/files', async (req, res) => {
  109. if (!req.files || Object.keys(req.files).length === 0) {
  110. return res.status(400).json('No files were uploaded.');
  111. }
  112. const user = jwt.verify(req.body.token, TOKEN_KEY);
  113. const oneUser = await getOneUser(user.userName);
  114. if(oneUser.isMutted){
  115. return;
  116. }
  117. const files = req.files.files;
  118. if (files.length) {
  119. for (let i = 0; i < files.length; i++) {
  120. let file = files[i]
  121. file.mv(STATIC_PATH + '\/' + file.name)
  122. const message = new Message({
  123. text: file.name,
  124. userName: user.userName,
  125. createDate: Date.now(),
  126. user: oneUser.id, //add link to other collection by id
  127. file: file.name,
  128. fileType: file.mimetype
  129. });
  130. try {
  131. await message.save();
  132. if(!oneUser.messages){
  133. await oneUser.update({ $set: {'messages': []}});
  134. }
  135. await oneUser.messages.push(message)
  136. await oneUser.save()
  137. }
  138. catch (error) {
  139. console.log('Message save to db error', error);
  140. }
  141. const newMessages = await message.populate( {path:'user'})
  142. io.emit('newmessage', newMessages);
  143. }}
  144. else {
  145. files.mv(STATIC_PATH + '\/' + files.name); //for one file
  146. const message = new Message({
  147. text: files.name,
  148. userName: user.userName,
  149. createDate: Date.now(),
  150. user: oneUser.id, //add link to other collection by id
  151. file: files.name,
  152. fileType: files.mimetype
  153. });
  154. try {
  155. await message.save();
  156. if(!oneUser.messages){
  157. await oneUser.update({ $set: {'messages': []}});
  158. }
  159. await oneUser.messages.push(message)
  160. await oneUser.save()
  161. }
  162. catch (error) {
  163. console.log('Message save to db error', error);
  164. }
  165. const newMessages = await message.populate( {path:'user'})
  166. io.emit('newmessage', newMessages);
  167. }
  168. return res.json({ message:'File was uploud succesfully...'})
  169. })
  170. io.use( async (socket, next) => {
  171. const token = socket.handshake.auth.token;
  172. const sockets = await io.fetchSockets();
  173. if(!token) {
  174. console.log('socket')
  175. socket.disconnect();
  176. return;
  177. }
  178. const usersOnline = [];
  179. sockets.map((sock) => {
  180. usersOnline.push(sock.user);
  181. })
  182. try {
  183. const user = jwt.verify(token, TOKEN_KEY);
  184. const userName = user.userName;
  185. const dbUser = await getOneUser(userName);
  186. if(dbUser.isBanned){
  187. socket.disconnect();
  188. return;
  189. }
  190. socket.user = user;
  191. const exist = sockets.find((current) => current.user.userName == socket.user.userName)
  192. if(exist) { //&& !user.isAdmin - add for two or more admins
  193. console.log(exist.userName, 'exist twice entering...')
  194. exist.disconnect();
  195. }
  196. } catch(e) {
  197. console.log(e);
  198. socket.disconnect();
  199. }
  200. next();
  201. });
  202. io.on("connection", async (socket) => {
  203. const userName = socket.user.userName;
  204. const sockets = await io.fetchSockets();
  205. const dbUser = await getOneUser(userName);
  206. io.emit('usersOnline', sockets.map(sock => sock.user)); // send array online users
  207. socket.emit('connected', dbUser); //socket.user
  208. if(socket.user.isAdmin){
  209. getAllDbUsers(socket);
  210. }//sent all users from db to admin
  211. const messagesToShow = await Message.find({}).sort({ 'createDate': -1 }).limit(20).populate( {path:'user'});
  212. socket.emit('allmessages', messagesToShow.reverse());
  213. socket.on("message", async (data) => {
  214. const dateNow = Date.now(); // for correct working latest post
  215. const post = await Message.findOne({userName}).sort({ 'createDate': -1 })
  216. const oneUser = await getOneUser(userName);
  217. if(oneUser.isMutted){ //(oneUser.isMutted || !post)
  218. return;
  219. }
  220. if(((Date.now() - Date.parse(post?.createDate)) < 1500)){
  221. console.log((Date.now() - Date.parse(post?.createDate)))// can use to show timer near by button
  222. return;
  223. }
  224. // if(!oneUser.isMutted && post){
  225. // if(((Date.now() - Date.parse(post.createDate)) > 150)){//change later 15000
  226. const message = new Message({
  227. text: data.message,
  228. userName: userName,
  229. createDate: Date.now(),
  230. user: oneUser.id, //add link to other collection by id
  231. });
  232. try {
  233. await message.save();
  234. if(!oneUser.messages){
  235. await oneUser.update({ $set: {'messages': []}});
  236. }
  237. await oneUser.messages.push(message)
  238. await oneUser.save()
  239. } catch (error) {
  240. console.log('Message save to db error', error);
  241. }
  242. const newMessages = await message.populate( {path:'user'})
  243. io.emit('newmessage', newMessages);
  244. // }
  245. // }
  246. });
  247. try {
  248. socket.on("disconnect", async () => {
  249. const sockets = await io.fetchSockets();
  250. io.emit('usersOnline', sockets.map(sock => sock.user));
  251. console.log(`user :${socket.user.userName} , disconnected to socket`);
  252. });
  253. console.log(`user :${socket.user.userName} , connected to socket`);
  254. socket.on("muteUser",async (data) => {
  255. if(!socket.user.isAdmin){
  256. return;
  257. }
  258. // if(socket.user.isAdmin){
  259. const {user, prevStatus} = data;
  260. const sockets = await io.fetchSockets();
  261. const mute = await User.updateOne({userName : user}, {$set: {isMutted :!prevStatus}});
  262. getAllDbUsers(socket);
  263. const exist = sockets.find( current => current.user.userName == user)
  264. const dbUser = await getOneUser(user);
  265. if(exist){
  266. exist.emit('connected', dbUser);
  267. }
  268. // }
  269. });
  270. socket.on("banUser",async (data) => {
  271. if(!socket.user.isAdmin){
  272. return;
  273. }
  274. // if(socket.user.isAdmin) {
  275. const {user, prevStatus} = data;
  276. const sockets = await io.fetchSockets();
  277. const ban = await User.updateOne({userName : user}, {$set: {isBanned:!prevStatus}});
  278. getAllDbUsers(socket)
  279. const exist = sockets.find( current => current.user.userName == user)
  280. if(exist){
  281. exist.emit('ban', "dbUser")
  282. exist.disconnect();
  283. }
  284. // }
  285. });
  286. socket.on('userWriting', async () => {
  287. let isTyping = true;
  288. io.emit('writing', {userName, isTyping})
  289. })
  290. } catch (e) {
  291. console.log(e);
  292. }
  293. });
  294. //server and database start
  295. const start = async () => {
  296. try {
  297. await mongoose.connect('mongodb://localhost:27017/chat')
  298. .then(() => console.log(`DB started`))
  299. server.listen(PORT, () => {
  300. console.log(`Server started. Port: ${PORT}`);
  301. })
  302. } catch (e) {
  303. console.log(e);
  304. }
  305. }
  306. start();