You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

server.js 20KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573
  1. const express = require("express");
  2. const bodyParser = require("body-parser");
  3. const app = express();
  4. const bcrypt = require("bcrypt");
  5. const Datastore = require("nedb");
  6. const session = require("express-session");
  7. const NedbSessionStore = require("nedb-session-store")(session);
  8. app.listen(8080, () => {
  9. console.log("Server online on port 8080");
  10. })
  11. app.use(session({
  12. secret: process.env.SECRET || 'ENTER YOUR SECRET KEY IN ENV VARIABLES!',
  13. resave: false,
  14. saveUninitialized: false,
  15. unset: "destroy",
  16. cookie: {
  17. path: '/',
  18. httpOnly: true,
  19. maxAge: 3600000 //cookie time in ms (=1h)
  20. },
  21. store: new NedbSessionStore({
  22. filename: 'db/sessions.db'
  23. })
  24. }))
  25. app.use(express.static("public"));
  26. app.use(bodyParser.json());
  27. app.use(bodyParser.urlencoded({ extended: true }));
  28. /*--------------------------------------------------PAGE REQUESTS--------------------------------------------------*/
  29. app.get("/", (req, res) => {
  30. res.send("index.html");
  31. });
  32. app.get("/info", (req, res) => {
  33. res.sendFile(__dirname + '/public/info-page.html')
  34. });
  35. app.get("/main", (req, res) => {
  36. if(req.session.isLoggedIn && req.session.userRole === "admin"){
  37. res.sendFile(__dirname + '/public/admin-login.html')
  38. }else if(req.session.isLoggedIn && req.session.userRole === "mod") {
  39. res.sendFile(__dirname + '/public/mod-login.html')
  40. }else if(req.session.isLoggedIn && req.session.userRole === "user") {
  41. res.sendFile(__dirname + '/public/user-login.html')
  42. }else {
  43. req.session = null;
  44. res.redirect("/");
  45. }
  46. });
  47. app.get("/newPost", (req, res) => {
  48. if(req.session.isLoggedIn && req.session.userRole === "user"){
  49. res.sendFile(__dirname + '/public/new-Postv2.html')
  50. } else {
  51. req.session = null;
  52. res.redirect("/");
  53. }
  54. });
  55. app.get("/manage-posts", (req, res) => {
  56. if(req.session.isLoggedIn && (req.session.userRole === "mod" || req.session.userRole === "admin")){
  57. res.sendFile(__dirname + '/public/manage-post.html')
  58. } else {
  59. req.session = null;
  60. res.redirect("/");
  61. }
  62. });
  63. app.get("/new-password", (req, res) => {
  64. if(req.session.isLoggedIn){
  65. res.sendFile(__dirname + '/public/change-pw.html')
  66. } else {
  67. req.session = null;
  68. res.redirect("/");
  69. }
  70. });
  71. app.post("/updatePostID", (req, res) => {
  72. if(req.session.isLoggedIn){
  73. req.session.revpost = req.body.postid;
  74. res.json({suc:true});
  75. //res.sendFile(__dirname + '/public/post-review.html')
  76. } else {
  77. res.json({suc:false});
  78. }
  79. });
  80. app.get("/post-review", (req, res) => {
  81. if(req.session.isLoggedIn ){
  82. res.sendFile(__dirname + '/public/post-review.html')
  83. } else {
  84. req.session = null;
  85. res.redirect("/");
  86. }
  87. });
  88. /*--------------------------------------------------GETTER FUNCTIONS--------------------------------------------------*/
  89. app.get("/getmods", (req, res) => {
  90. if(req.session.isLoggedIn && req.session.userRole === "admin"){
  91. const db = new Datastore("db/users.db");
  92. db.loadDatabase();
  93. db.find({role:"mod"}, (err, docs) => {
  94. if(docs.length > 0) {
  95. let resArray = [];
  96. docs.forEach(element => {
  97. resArray.push({
  98. username:element.username,
  99. suspended:element.suspended
  100. })
  101. });
  102. res.json({suc:true, users:resArray})
  103. } else {
  104. res.json({suc:false});
  105. }
  106. })
  107. } else {
  108. res.json({suc:false});
  109. }
  110. });
  111. app.get("/getusers", (req, res) => {
  112. if(req.session.isLoggedIn && req.session.userRole === "mod"){
  113. const db = new Datastore("db/users.db");
  114. db.loadDatabase();
  115. db.find({role:"user" , group: req.session.userid }, (err, docs) => {
  116. if(docs.length > 0) {
  117. let resArray = [];
  118. docs.forEach(element => {
  119. resArray.push({
  120. username:element.username,
  121. suspended:element.suspended
  122. })
  123. });
  124. res.json({suc:true, users:resArray})
  125. } else {
  126. res.json({suc:false});
  127. }
  128. })
  129. } else {
  130. res.json({suc:false});
  131. }
  132. });
  133. app.get("/getReviewPost", (req, res) => {
  134. if(req.session.isLoggedIn){
  135. const db = new Datastore("db/posts.db");
  136. db.loadDatabase();
  137. db.find({_id:req.session.revpost}, (err, docs) => {
  138. if(docs.length === 1) {
  139. let post = [];
  140. post.push({
  141. postid:docs[0]._id,
  142. username:docs[0].creatorName,
  143. showpost:docs[0].showpost,
  144. title:docs[0].title,
  145. text:docs[0].text,
  146. bcolor:docs[0].bcolor,
  147. startdate:docs[0].startDate,
  148. enddate:docs[0].endDate,
  149. })
  150. res.json({suc:true, post:post})
  151. } else {
  152. res.json({suc:false});
  153. }
  154. })
  155. } else {
  156. res.json({suc:false});
  157. }
  158. });
  159. app.get("/getposts", (req, res) => {
  160. if(req.session.isLoggedIn){
  161. const db_post = new Datastore("db/posts.db");
  162. db_post.loadDatabase();
  163. switch(req.session.userRole)
  164. {
  165. case "admin":
  166. db_post.find({}, (err, docs) => {
  167. if(docs.length > 0) {
  168. let resArray = [];
  169. docs.forEach(element => {
  170. resArray.push({
  171. creator: element.creatorName,
  172. postid: element._id,
  173. title:element.title,
  174. text:element.text,
  175. showpost:element.showpost,
  176. startDate:element.startDate,
  177. endDate:element.endDate,
  178. })
  179. });
  180. res.json({suc:true, users:resArray})
  181. } else {
  182. res.json({suc:false});
  183. }
  184. });
  185. break;
  186. case "mod":
  187. db_post.find({creatorGroup:req.session.userid}, (err, docs) => {
  188. if(docs.length > 0) {
  189. let resArray = [];
  190. docs.forEach(element => {
  191. resArray.push({
  192. creator: element.creatorName,
  193. postid: element._id,
  194. title:element.title,
  195. text:element.text,
  196. showpost:element.showpost,
  197. startDate:element.startDate,
  198. endDate:element.endDate,
  199. })
  200. });
  201. res.json({suc:true, users:resArray})
  202. } else {
  203. res.json({suc:false});
  204. }
  205. });
  206. break;
  207. case "user":
  208. db_post.find({creatorId:req.session.userid}, (err, docs) => {
  209. if(docs.length > 0) {
  210. let resArray = [];
  211. docs.forEach(element => {
  212. resArray.push({
  213. title:element.title,
  214. showpost:element.showpost,
  215. postid:element._id
  216. })
  217. });
  218. res.json({suc:true, users:resArray})
  219. } else {
  220. res.json({suc:false});
  221. }});
  222. break;
  223. }
  224. }
  225. });
  226. app.get("/getliveposts", (req, res) => {
  227. const now = new Date();
  228. const db_post = new Datastore("db/posts.db");
  229. db_post.loadDatabase();
  230. const db_users = new Datastore("db/users.db");
  231. db_users.loadDatabase();
  232. db_post.find({}, (err, docs) => {
  233. if(docs.length > 0) {
  234. let resArray = [];
  235. docs.forEach(element => {
  236. if(element.showpost){
  237. var start = new Date(element.startDate) - now;
  238. var end = new Date(element.endDate) - now;
  239. if(start < 0 && end > 0){
  240. db_users.find({_id:element.creatorId}, (err2, docs2) =>{
  241. docs2.forEach(element2 => {
  242. if(!element2.suspended)
  243. {
  244. db_users.find({_id:element.creatorGroup}, (err3, docs3) =>{
  245. docs3.forEach(element3 => {
  246. if(!element3.suspended)
  247. {
  248. resArray.push({
  249. title:element.title,
  250. text:element.text,
  251. bcolor:element.bcolor
  252. })
  253. }
  254. })
  255. })
  256. }
  257. })
  258. })
  259. }
  260. }
  261. });
  262. setTimeout(function(){ res.json({suc:true, posts:resArray}); }, 100); //delay to resolve timing issue
  263. } else {
  264. res.json({suc:false});
  265. }
  266. });
  267. });
  268. /*--------------------------------------------------FUNCTIONS--------------------------------------------------*/
  269. app.post("/login", (req, res) => {
  270. const db = new Datastore("db/users.db");
  271. db.loadDatabase();
  272. const un = req.body.username;
  273. const pw = req.body.password;
  274. if(un && pw){
  275. db.find({username:un}, (err, docs) => {
  276. if(docs.length === 1){
  277. const userRole = docs[0].role;
  278. bcrypt.compare(pw, docs[0].password, (err2, result) => {
  279. if(result){
  280. req.session.userRole = userRole;
  281. req.session.userid = docs[0]._id;
  282. req.session.gr = docs[0].group;
  283. req.session.un = docs[0].username;
  284. req.session.revpost = 0;
  285. req.session.isLoggedIn = true;
  286. if(userRole === "admin") {
  287. res.json({suc:true, redirect:"admin"}) //login to admin page
  288. } else if(userRole === "mod"){
  289. res.json({suc:true, redirect:"mod"}) //login to mod page
  290. } else if(userRole === "user"){
  291. res.json({suc:true, redirect:"user"}) //login to user page
  292. }
  293. } else { res.json({suc:false}); }
  294. })
  295. } else { res.json({suc:false}); }
  296. })
  297. } else { res.json({suc:false}); }
  298. });
  299. app.get("/logout", (req, res) => {
  300. req.session = null;
  301. res.redirect("/");
  302. });
  303. app.post("/changepw", (req, res) => {
  304. const db = new Datastore("db/users.db");
  305. db.loadDatabase();
  306. const old_pw = req.body.old_password;
  307. const new_pw_1 = req.body.new_password_1;
  308. const new_pw_2 = req.body.new_password_2;
  309. if(old_pw && (new_pw_1 === new_pw_2)){
  310. db.find({_id:req.session.userid}, (err, docs) => {
  311. if(docs.length === 1){
  312. bcrypt.compare(old_pw, docs[0].password, (err2, result) => {
  313. if(result){
  314. bcrypt.hash(new_pw_1, 10, (err3, hash) => {
  315. db.update({_id: docs[0]._id}, {$set: {password: hash}}, (err4, num) =>{
  316. res.json({suc:true});
  317. });
  318. })
  319. } else { res.json({suc:false}); }
  320. })
  321. } else { res.json({suc:false}); }
  322. })
  323. } else { res.json({suc:false}); }
  324. })
  325. app.post("/newmod", (req, res) => {
  326. if(req.session.isLoggedIn && req.session.userRole === "admin"){
  327. const nun = req.body.username;
  328. const nemail = req.body.email;
  329. const db = new Datastore("db/users.db");
  330. db.loadDatabase();
  331. db.find({username:nun}, (err, docs) => {
  332. if(docs.length === 0){
  333. db.find({email:nemail}, (err2, docs2) => {
  334. if(docs2.length === 0){
  335. bcrypt.hash(nun, 10, (err3, hash) => {
  336. db.insert({email: nemail,
  337. username: nun,
  338. password: hash,
  339. role: "mod",
  340. suspended: false
  341. }, (err4, doc) => {
  342. res.json({suc:true});
  343. });
  344. });
  345. } else {
  346. res.json({suc:false});
  347. }
  348. });
  349. } else {
  350. res.json({suc:false});
  351. }
  352. });
  353. } else {
  354. res.json({suc:false});
  355. }
  356. })
  357. app.post("/newuser", (req, res) => {
  358. if(req.session.isLoggedIn && req.session.userRole === "mod"){
  359. const nun = req.body.username;
  360. const nemail = req.body.email;
  361. const db = new Datastore("db/users.db");
  362. db.loadDatabase();
  363. db.find({username:nun}, (err, docs) => {
  364. if(docs.length === 0){
  365. db.find({email:nemail}, (err2, docs2) => {
  366. if(docs2.length === 0){
  367. bcrypt.hash(nun, 10, (err3, hash) => {
  368. db.insert({
  369. email: nemail,
  370. username: nun,
  371. password: hash,
  372. role: "user",
  373. group: req.session.userid,
  374. suspended: false
  375. }, (err4, doc) => {
  376. res.json({suc:true});
  377. });
  378. });
  379. } else {
  380. res.json({suc:false});
  381. }
  382. });
  383. } else {
  384. res.json({suc:false});
  385. }
  386. });
  387. } else {
  388. res.json({suc:false});
  389. }
  390. })
  391. app.post("/newpost", (req, res) => {
  392. if(req.session.isLoggedIn && req.session.userRole === "user"){
  393. const db = new Datastore("db/posts.db");
  394. db.loadDatabase();
  395. db.insert({
  396. creatorId:req.session.userid ,
  397. creatorName:req.session.un ,
  398. creatorGroup:req.session.gr ,
  399. showpost:false ,
  400. startDate:req.body.startDate ,
  401. endDate:req.body.endDate ,
  402. title:req.body.title ,
  403. text:req.body.text,
  404. bcolor:req.body.bcolor
  405. }, (err, doc) => {
  406. res.json({suc:true});
  407. });
  408. }
  409. });
  410. app.post("/toggleSus", (req, res) => {
  411. if(req.session.isLoggedIn && (req.session.userRole === "admin" || req.session.userRole === "mod")){
  412. const db = new Datastore("db/users.db");
  413. db.loadDatabase();
  414. db.find({username:req.body.username}, (err, docs) => {
  415. if(docs.length === 1) {
  416. var now = !docs[0].suspended;
  417. db.update({_id: docs[0]._id}, {$set: {suspended: now}}, (err2, num) =>{
  418. res.json({suc:true});
  419. })
  420. } else {
  421. res.json({suc:false});
  422. }
  423. })
  424. } else {
  425. res.json({suc:false});
  426. }
  427. });
  428. app.post("/toggleShow", (req, res) => {
  429. if(req.session.isLoggedIn && (req.session.userRole === "mod" || req.session.userRole === "admin")){
  430. const db = new Datastore("db/posts.db");
  431. db.loadDatabase();
  432. db.find({_id:req.body.postid}, (err, docs) => {
  433. if(docs.length === 1) {
  434. var now = !docs[0].showpost;
  435. db.update({_id: docs[0]._id}, {$set: {showpost: now}}, (err2, num) =>{
  436. res.json({suc:true});
  437. })
  438. } else {
  439. res.json({suc:false});
  440. }
  441. })
  442. } else {
  443. res.json({suc:false});
  444. }
  445. });
  446. app.post("/deleteMod", (req, res) => {
  447. if(req.session.isLoggedIn && req.session.userRole === "admin") {
  448. const db = new Datastore("db/users.db");
  449. db.loadDatabase();
  450. db.find({username:req.body.username}, (err, docs) => {
  451. if(docs.length === 1) {
  452. deleteMod(docs[0]._id);
  453. setTimeout(function(){ res.json({suc:true}); }, 1000); //delay to resolve timing issue
  454. } else {
  455. res.json({suc:false});
  456. }
  457. })
  458. } else {
  459. res.json({suc:false});
  460. }
  461. });
  462. app.post("/deleteUser", (req, res) => {
  463. if(req.session.isLoggedIn && req.session.userRole === "mod"){
  464. const db = new Datastore("db/users.db");
  465. db.loadDatabase();
  466. db.find({username:req.body.username}, (err, docs) => {
  467. if(docs.length === 1) {
  468. deleteUser(docs[0]._id);
  469. setTimeout(function(){ res.json({suc:true}); }, 1000); //delay to resolve timing issue
  470. } else {
  471. res.json({suc:false});
  472. }
  473. })
  474. } else {
  475. res.json({suc:false});
  476. }
  477. });
  478. app.post("/deletePost", (req, res) => {
  479. if(req.session.isLoggedIn){
  480. const db = new Datastore("db/posts.db");
  481. db.loadDatabase();
  482. db.find({_id:req.body.postid}, (err, docs) => {
  483. if(docs.length === 1) {
  484. deletePost(docs[0]._id);
  485. setTimeout(function(){ res.json({suc:true}); }, 1000); //delay to resolve timing issue
  486. } else {
  487. res.json({suc:false});
  488. }
  489. })
  490. } else {
  491. res.json({suc:false});
  492. }
  493. });
  494. /*--------------------------------------------------FUNCTIONS--------------------------------------------------*/
  495. function deleteMod(modID){
  496. const db_users = new Datastore("db/users.db");
  497. db_users.loadDatabase();
  498. db_users.find({group:modID}, (err, docs) => {
  499. docs.forEach(element => {
  500. deleteUser(element._id);
  501. if(docs.indexOf(element) === docs.length-1)
  502. db_users.find({_id:modID}, (err2, docs2) => {
  503. console.log("deleted mod: ", docs2[0]._id);
  504. db_users.remove({_id: docs2[0]._id});
  505. })
  506. });
  507. })
  508. }
  509. function deleteUser(userID){
  510. const db_posts = new Datastore("db/posts.db");
  511. db_posts.loadDatabase();
  512. db_posts.find({creatorId:userID}, (err, docs) => {
  513. docs.forEach(element => {
  514. deletePost(element._id);
  515. });
  516. })
  517. const db_users = new Datastore("db/users.db");
  518. db_users.loadDatabase();
  519. db_users.find({_id:userID}, (err, docs) => {
  520. console.log("deleted user: ", docs[0]._id);
  521. db_users.remove({_id: docs[0]._id})
  522. })
  523. }
  524. function deletePost(postID){
  525. const db_posts = new Datastore("db/posts.db");
  526. db_posts.loadDatabase();
  527. db_posts.find({_id:postID}, (err, docs) => {
  528. console.log("deleted post: ", docs[0]._id);
  529. db_posts.remove({_id: docs[0]._id});
  530. })
  531. }