Source: admin/index.js

  1. /**
  2. * @module admin
  3. */
  4. const adminRepo = require("../../repositories/admin");
  5. const jose = require("../jose");
  6. const digest = require("../digest");
  7. const config = require("../../config").jose.aud;
  8. const mail = require("../mail");
  9. /**
  10. * Temporary logs of bugreports.
  11. * Solely used for weekly notification of admins
  12. * and cleared afterwards to ensure privacy of users.
  13. */
  14. const bugReports = [];
  15. /**
  16. * Temporary logs contact requests.
  17. * Solely used for weekly notification of admins
  18. * and cleared afterwards to ensure privacy of users.
  19. */
  20. const contactRequests = [];
  21. /**
  22. * Attempt to log in admin and generate a JWE token
  23. * if successful.
  24. * Throws an error on failure with appropriate message.
  25. * @param {String} email
  26. * @param {String} password
  27. * @return {Object} JWE token on success
  28. * @throws {Error} on invalid email or password
  29. */
  30. const logInAdmin = async (email, password) => {
  31. const findResult = await adminRepo.findByEmail(email);
  32. if (findResult.rows.length === 0) {
  33. throw new Error("Invalid admin email.");
  34. }
  35. const admin = findResult.rows[0];
  36. if (digest.hashPassWithSaltInHex(password, admin.salt) === admin.passwordHash) {
  37. return jose.signAndEncrypt({
  38. sub: admin.id.toString(),
  39. name: admin.username,
  40. email: email,
  41. });
  42. } else {
  43. throw new Error("Invalid admin password.");
  44. }
  45. };
  46. /**
  47. * Attempt to authenticate admin
  48. * by JWE. Upon successful verification,
  49. * the admin's (user)name is returned.
  50. * Otherwise, a JWE or JWT error is thrown.
  51. * @param {Object} jwe
  52. * @return {string} name of admin
  53. */
  54. const authenticateAdmin = (jwe) => {
  55. return jose.decryptAndVerify(jwe, config).name;
  56. };
  57. /**
  58. * Log a copy of a bugreport in
  59. * weekly temp storage.
  60. * @param {Object} bugreport
  61. */
  62. const logBugReport = (bugreport) => {
  63. bugReports.push(bugreport);
  64. };
  65. /**
  66. * Log a copy of a contact request in
  67. * weekly temp storage.
  68. * @param {Object} contactRequest
  69. */
  70. const logContactRequest = (contactRequest) => {
  71. contactRequests.push(contactRequest);
  72. };
  73. /**
  74. * Send weekly summary of bugreports and contact
  75. * requests to admins, then clear temp storage
  76. * to ensure privacy of users.
  77. */
  78. const sendSummary = () => {
  79. mail.sendEmail(
  80. process.env.DEFAULT_EMAIL_ADDRESS,
  81. process.env.SUMMARY_EMAIL_ADDRESS,
  82. "Your RDFmapped weekly summary",
  83. `Hello,\nHere is your rdfmapped.com weekly summary.\n\n` +
  84. `Number of new bugreports: ${bugReports.length}\nNumber of new contact requests: ${contactRequests.length}\n\n\n` +
  85. `Bug reports: ${bugReports.toString()}\n\n` +
  86. `Contact requests: ${contactRequests.toString()}\n\n`,
  87. `<p>Hello,</p><p>Here is your rdfmapped.com weekly summary.<br></p>` +
  88. `<p>Number of new bugreports: <b>${bugReports.length}</b></p>` +
  89. `<p>Number of new contact requests: <b>${contactRequests.length}</b><br></p>` +
  90. `<p>Bug reports: <code>${JSON.stringify(bugReports)}</code><br></p>` +
  91. `<p>Contact requests: <code>${JSON.stringify(contactRequests)}</code></p>`);
  92. bugReports.length = 0;
  93. contactRequests.length = 0;
  94. };
  95. module.exports = {
  96. logInAdmin,
  97. authenticateAdmin,
  98. logBugReport,
  99. logContactRequest,
  100. sendSummary,
  101. };