瀏覽代碼

feat: added admin button to request a password reset for a user

Kristian Vos 3 年之前
父節點
當前提交
8c23b6650d
共有 2 個文件被更改,包括 89 次插入0 次删除
  1. 77 0
      backend/logic/actions/users.js
  2. 12 0
      frontend/src/components/modals/EditUser.vue

+ 77 - 0
backend/logic/actions/users.js

@@ -2546,6 +2546,83 @@ export default {
 		);
 	},
 
+	/**
+	 * Requests a password reset for a a user as an admin
+	 *
+	 * @param {object} session - the session object automatically added by the websocket
+	 * @param {string} email - the email of the user for which the password reset is intended
+	 * @param {Function} cb - gets called with the result
+	 */
+	 adminRequestPasswordReset: isAdminRequired(async function adminRequestPasswordReset(session, userId, cb) {
+		const code = await UtilsModule.runJob("GENERATE_RANDOM_STRING", { length: 8 }, this);
+		const userModel = await DBModule.runJob("GET_MODEL", { modelName: "user" }, this);
+
+		const resetPasswordRequestSchema = await MailModule.runJob(
+			"GET_SCHEMA",
+			{ schemaName: "resetPasswordRequest" },
+			this
+		);
+
+		async.waterfall(
+			[
+				next => {
+					return userModel.findOne({ "_id": userId }, next);
+				},
+
+				(user, next) => {
+					if (!user) return next("User not found.");
+					if (!user.services.password || !user.services.password.password)
+						return next("User does not have a password set, and probably uses GitHub to log in.");
+					return next();
+				},
+
+				(next) => {
+					const expires = new Date();
+					expires.setDate(expires.getDate() + 1);
+					userModel.findOneAndUpdate(
+						{ _id: userId },
+						{
+							$set: {
+								"services.password.reset": {
+									code,
+									expires
+								}
+							}
+						},
+						{ runValidators: true },
+						next
+					);
+				},
+
+				(user, next) => {
+					resetPasswordRequestSchema(user.email.address, user.username, code, next);
+				}
+			],
+			async err => {
+				if (err && err !== true) {
+					err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
+					this.log(
+						"ERROR",
+						"ADMINREQUEST_PASSWORD_RESET",
+						`User '${userId}' failed to get a password reset. '${err}'`
+					);
+					return cb({ status: "error", message: err });
+				}
+
+				this.log(
+					"SUCCESS",
+					"ADMIN_REQUEST_PASSWORD_RESET",
+					`User '${userId}' successfully got sent a password reset.`
+				);
+
+				return cb({
+					status: "success",
+					message: "Successfully requested password reset for user."
+				});
+			}
+		);
+	}),
+
 	/**
 	 * Verifies a reset code
 	 *

+ 12 - 0
frontend/src/components/modals/EditUser.vue

@@ -79,6 +79,9 @@
 				<confirm @confirm="resendVerificationEmail()">
 					<a class="button is-warning"> Resend verification email </a>
 				</confirm>
+				<confirm @confirm="requestPasswordReset()">
+					<a class="button is-warning"> Request password reset </a>
+				</confirm>
 				<confirm @confirm="removeSessions()">
 					<a class="button is-warning"> Remove all sessions </a>
 				</confirm>
@@ -244,6 +247,15 @@ export default {
 				}
 			);
 		},
+		requestPasswordReset() {
+			this.socket.dispatch(
+				`users.adminRequestPasswordReset`,
+				this.user._id,
+				res => {
+					new Toast(res.message);
+				}
+			);
+		},
 		removeAccount() {
 			this.socket.dispatch(`users.adminRemove`, this.user._id, res => {
 				new Toast(res.message);