Browse Source

refactor: move remove user logic to module

Kristian Vos 3 months ago
parent
commit
551de65643
4 changed files with 171 additions and 375 deletions
  1. 1 0
      backend/index.js
  2. 13 375
      backend/logic/actions/users.js
  3. 19 0
      backend/logic/mail/index.js
  4. 138 0
      backend/logic/users.js

+ 1 - 0
backend/index.js

@@ -260,6 +260,7 @@ if (!config.get("migration")) {
 	moduleManager.addModule("stations");
 	moduleManager.addModule("media");
 	moduleManager.addModule("tasks");
+	moduleManager.addModule("users");
 	moduleManager.addModule("utils");
 	moduleManager.addModule("youtube");
 	if (config.get("experimental.soundcloud")) moduleManager.addModule("soundcloud");

+ 13 - 375
backend/logic/actions/users.js

@@ -20,7 +20,7 @@ const MailModule = moduleManager.modules.mail;
 const PunishmentsModule = moduleManager.modules.punishments;
 const ActivitiesModule = moduleManager.modules.activities;
 const PlaylistsModule = moduleManager.modules.playlists;
-const MediaModule = moduleManager.modules.media;
+const UsersModule = moduleManager.modules.users;
 
 CacheModule.runJob("SUB", {
 	channel: "user.updatePreferences",
@@ -331,205 +331,22 @@ export default {
 	 * @param {Function} cb - gets called with the result
 	 */
 	remove: isLoginRequired(async function remove(session, cb) {
-		const userModel = await DBModule.runJob("GET_MODEL", { modelName: "user" }, this);
-		const dataRequestModel = await DBModule.runJob("GET_MODEL", { modelName: "dataRequest" }, this);
-		const stationModel = await DBModule.runJob("GET_MODEL", { modelName: "station" }, this);
-		const playlistModel = await DBModule.runJob("GET_MODEL", { modelName: "playlist" }, this);
-		const activityModel = await DBModule.runJob("GET_MODEL", { modelName: "activity" }, this);
-
-		const dataRequestEmail = await MailModule.runJob("GET_SCHEMA", { schemaName: "dataRequest" }, this);
-
-		const songsToAdjustRatings = [];
+		const { userId } = session;
 
 		async.waterfall(
 			[
-				// activities related to the user
-				next => {
-					activityModel.deleteMany({ userId: session.userId }, next);
-				},
-
-				// user's stations
-				(res, next) => {
-					stationModel.find({ owner: session.userId }, (err, stations) => {
-						if (err) return next(err);
-
-						return async.each(
-							stations,
-							(station, callback) => {
-								// delete the station
-								stationModel.deleteOne({ _id: station._id }, err => {
-									if (err) return callback(err);
-
-									CacheModule.runJob("HDEL", { table: "stations", key: station._id });
-
-									// if applicable, delete the corresponding playlist for the station
-									if (station.playlist)
-										return PlaylistsModule.runJob("DELETE_PLAYLIST", {
-											playlistId: station.playlist
-										})
-											.then(() => callback())
-											.catch(callback);
-
-									return callback();
-								});
-							},
-							err => next(err)
-						);
-					});
-				},
-
-				// remove user as station DJ
 				next => {
-					stationModel.updateMany({ djs: session.userId }, { $pull: { djs: session.userId } }, next);
-				},
-
-				(res, next) => {
-					playlistModel.findOne({ createdBy: session.userId, type: "user-liked" }, next);
-				},
-
-				// get all liked songs (as the global rating values for these songs will need adjusted)
-				(playlist, next) => {
-					if (!playlist) return next();
-
-					playlist.songs.forEach(song =>
-						songsToAdjustRatings.push({ songId: song._id, mediaSource: song.mediaSource })
-					);
-
-					return next();
-				},
-
-				next => {
-					playlistModel.findOne({ createdBy: session.userId, type: "user-disliked" }, next);
-				},
-
-				// get all disliked songs (as the global rating values for these songs will need adjusted)
-				(playlist, next) => {
-					if (!playlist) return next();
-
-					playlist.songs.forEach(song => songsToAdjustRatings.push({ mediaSource: song.mediaSource }));
-
-					return next();
-				},
-
-				// user's playlists
-				next => {
-					playlistModel.deleteMany({ createdBy: session.userId }, next);
-				},
-
-				(res, next) => {
-					async.each(
-						songsToAdjustRatings,
-						(song, next) => {
-							const { mediaSource } = song;
-
-							MediaModule.runJob("RECALCULATE_RATINGS", { mediaSource })
-								.then(() => next())
-								.catch(next);
-						},
-						err => next(err)
-					);
-				},
-
-				// user object
-				next => {
-					userModel.deleteMany({ _id: session.userId }, next);
-				},
-
-				// session
-				(res, next) => {
-					CacheModule.runJob("PUB", {
-						channel: "user.removeSessions",
-						value: session.userId
-					});
-
-					async.waterfall(
-						[
-							next => {
-								CacheModule.runJob("HGETALL", { table: "sessions" }, this)
-									.then(sessions => {
-										next(null, sessions);
-									})
-									.catch(next);
-							},
-
-							(sessions, next) => {
-								if (!sessions) return next(null, [], {});
-
-								const keys = Object.keys(sessions);
-
-								return next(null, keys, sessions);
-							},
-
-							(keys, sessions, next) => {
-								// temp fix, need to wait properly for the SUB/PUB refactor (on wekan)
-								const { userId } = session;
-								setTimeout(
-									() =>
-										async.each(
-											keys,
-											(sessionId, callback) => {
-												const session = sessions[sessionId];
-
-												if (session && session.userId === userId) {
-													CacheModule.runJob(
-														"HDEL",
-														{
-															table: "sessions",
-															key: sessionId
-														},
-														this
-													)
-														.then(() => callback(null))
-														.catch(callback);
-												} else callback();
-											},
-											err => {
-												next(err);
-											}
-										),
-									50
-								);
-							}
-						],
-						next
-					);
-				},
-
-				// request data removal for user
-				next => {
-					dataRequestModel.create({ userId: session.userId, type: "remove" }, next);
-				},
-
-				(request, next) => {
-					WSModule.runJob("EMIT_TO_ROOM", {
-						room: "admin.users",
-						args: ["event:admin.dataRequests.created", { data: { request } }]
-					});
-
-					return next();
-				},
-
-				next => userModel.find({ role: "admin" }, next),
-
-				// send email to all admins of a data removal request
-				(users, next) => {
-					if (!config.get("sendDataRequestEmails")) return next();
-					if (users.length === 0) return next();
-
-					const to = [];
-					users.forEach(user => to.push(user.email.address));
-
-					return dataRequestEmail(to, session.userId, "remove", err => next(err));
+					UsersModule.runJob("REMOVE_USER", { userId })
+						.then(() => next())
+						.catch(err => next(err));
 				}
 			],
 			async err => {
 				if (err && err !== true) {
-					err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
-					this.log(
-						"ERROR",
-						"USER_REMOVE",
-						`Removing data and account for user "${session.userId}" failed. "${err}"`
-					);
+					err = await UtilsModule.runJob("GET_ERROR", { error: err });
+
+					this.log("ERROR", "USER_REMOVE", `Removing data and account for user "${userId}" failed. "${err}"`);
+
 					return cb({ status: "error", message: err });
 				}
 
@@ -541,7 +358,7 @@ export default {
 
 				CacheModule.runJob("PUB", {
 					channel: "user.removeAccount",
-					value: session.userId
+					value: userId
 				});
 
 				return cb({
@@ -559,196 +376,17 @@ export default {
 	 * @param {Function} cb - gets called with the result
 	 */
 	adminRemove: useHasPermission("users.remove", async function adminRemove(session, userId, cb) {
-		const userModel = await DBModule.runJob("GET_MODEL", { modelName: "user" }, this);
-		const dataRequestModel = await DBModule.runJob("GET_MODEL", { modelName: "dataRequest" }, this);
-		const stationModel = await DBModule.runJob("GET_MODEL", { modelName: "station" }, this);
-		const playlistModel = await DBModule.runJob("GET_MODEL", { modelName: "playlist" }, this);
-		const activityModel = await DBModule.runJob("GET_MODEL", { modelName: "activity" }, this);
-
-		const dataRequestEmail = await MailModule.runJob("GET_SCHEMA", { schemaName: "dataRequest" }, this);
-
-		const songsToAdjustRatings = [];
-
 		async.waterfall(
 			[
 				next => {
 					if (!userId) return next("You must provide a userId to remove.");
 					return next();
 				},
-				// activities related to the user
-				next => {
-					activityModel.deleteMany({ userId }, next);
-				},
-
-				// user's stations
-				(res, next) => {
-					stationModel.find({ owner: userId }, (err, stations) => {
-						if (err) return next(err);
-
-						return async.each(
-							stations,
-							(station, callback) => {
-								// delete the station
-								stationModel.deleteOne({ _id: station._id }, err => {
-									if (err) return callback(err);
-
-									// if applicable, delete the corresponding playlist for the station
-									if (station.playlist)
-										return PlaylistsModule.runJob("DELETE_PLAYLIST", {
-											playlistId: station.playlist
-										})
-											.then(() => callback())
-											.catch(callback);
-
-									return callback();
-								});
-							},
-							err => next(err)
-						);
-					});
-				},
-
-				// remove user as station DJ
-				next => {
-					stationModel.updateMany({ djs: userId }, { $pull: { djs: userId } }, next);
-				},
-
-				(res, next) => {
-					playlistModel.findOne({ createdBy: userId, type: "user-liked" }, next);
-				},
-
-				// get all liked songs (as the global rating values for these songs will need adjusted)
-				(playlist, next) => {
-					if (!playlist) return next();
-
-					playlist.songs.forEach(song =>
-						songsToAdjustRatings.push({ songId: song._id, mediaSource: song.mediaSource })
-					);
-
-					return next();
-				},
-
-				next => {
-					playlistModel.findOne({ createdBy: userId, type: "user-disliked" }, next);
-				},
-
-				// get all disliked songs (as the global rating values for these songs will need adjusted)
-				(playlist, next) => {
-					if (!playlist) return next();
-
-					playlist.songs.forEach(song => songsToAdjustRatings.push({ mediaSource: song.mediaSource }));
-
-					return next();
-				},
 
-				// user's playlists
 				next => {
-					playlistModel.deleteMany({ createdBy: userId }, next);
-				},
-
-				(res, next) => {
-					async.each(
-						songsToAdjustRatings,
-						(song, next) => {
-							const { mediaSource } = song;
-
-							MediaModule.runJob("RECALCULATE_RATINGS", { mediaSource })
-								.then(() => next())
-								.catch(next);
-						},
-						err => next(err)
-					);
-				},
-
-				// user object
-				next => {
-					userModel.deleteMany({ _id: userId }, next);
-				},
-
-				// session
-				(res, next) => {
-					CacheModule.runJob("PUB", {
-						channel: "user.removeSessions",
-						value: userId
-					});
-
-					async.waterfall(
-						[
-							next => {
-								CacheModule.runJob("HGETALL", { table: "sessions" }, this)
-									.then(sessions => {
-										next(null, sessions);
-									})
-									.catch(next);
-							},
-
-							(sessions, next) => {
-								if (!sessions) return next(null, [], {});
-
-								const keys = Object.keys(sessions);
-
-								return next(null, keys, sessions);
-							},
-
-							(keys, sessions, next) => {
-								// temp fix, need to wait properly for the SUB/PUB refactor (on wekan)
-								setTimeout(
-									() =>
-										async.each(
-											keys,
-											(sessionId, callback) => {
-												const session = sessions[sessionId];
-
-												if (session && session.userId === userId) {
-													CacheModule.runJob(
-														"HDEL",
-														{
-															table: "sessions",
-															key: sessionId
-														},
-														this
-													)
-														.then(() => callback(null))
-														.catch(callback);
-												} else callback();
-											},
-											err => {
-												next(err);
-											}
-										),
-									50
-								);
-							}
-						],
-						next
-					);
-				},
-
-				// request data removal for user
-				next => {
-					dataRequestModel.create({ userId, type: "remove" }, next);
-				},
-
-				(request, next) => {
-					WSModule.runJob("EMIT_TO_ROOM", {
-						room: "admin.users",
-						args: ["event:admin.dataRequests.created", { data: { request } }]
-					});
-
-					return next();
-				},
-
-				next => userModel.find({ role: "admin" }, next),
-
-				// send email to all admins of a data removal request
-				(users, next) => {
-					if (!config.get("sendDataRequestEmails")) return next();
-					if (users.length === 0) return next();
-
-					const to = [];
-					users.forEach(user => to.push(user.email.address));
-
-					return dataRequestEmail(to, userId, "remove", err => next(err));
+					UsersModule.runJob("REMOVE_USER", { userId })
+						.then(() => next())
+						.catch(err => next(err));
 				}
 			],
 			async err => {

+ 19 - 0
backend/logic/mail/index.js

@@ -83,6 +83,25 @@ class _MailModule extends CoreClass {
 			resolve(MailModule.schemas[payload.schemaName]);
 		});
 	}
+
+	/**
+	 * Returns an email schema, but using async instead of callbacks
+	 * @param {object} payload - object that contains the payload
+	 * @param {string} payload.schemaName - name of the schema to get
+	 * @returns {Promise} - returns promise (reject, resolve)
+	 */
+	async GET_SCHEMA_ASYNC(payload) {
+		const { schemaName } = payload;
+
+		return (...args) =>
+			new Promise((resolve, reject) => {
+				const cb = err => {
+					if (err) reject(err);
+					else resolve();
+				};
+				MailModule.schemas[schemaName](...args, cb);
+			});
+	}
 }
 
 export default new _MailModule();

+ 138 - 0
backend/logic/users.js

@@ -0,0 +1,138 @@
+import config from "config";
+import CoreClass from "../core";
+
+let UsersModule;
+let MailModule;
+let CacheModule;
+let DBModule;
+let PlaylistsModule;
+let WSModule;
+let MediaModule;
+
+class _UsersModule extends CoreClass {
+	// eslint-disable-next-line require-jsdoc
+	constructor() {
+		super("users");
+
+		UsersModule = this;
+	}
+
+	/**
+	 * Initialises the app module
+	 * @returns {Promise} - returns promise (reject, resolve)
+	 */
+	async initialize() {
+		DBModule = this.moduleManager.modules.db;
+		MailModule = this.moduleManager.modules.mail;
+		WSModule = this.moduleManager.modules.ws;
+		CacheModule = this.moduleManager.modules.cache;
+		MediaModule = this.moduleManager.modules.media;
+
+		this.userModel = await DBModule.runJob("GET_MODEL", { modelName: "user" });
+		this.dataRequestModel = await DBModule.runJob("GET_MODEL", { modelName: "dataRequest" });
+		this.stationModel = await DBModule.runJob("GET_MODEL", { modelName: "station" });
+		this.playlistModel = await DBModule.runJob("GET_MODEL", { modelName: "playlist" });
+		this.activityModel = await DBModule.runJob("GET_MODEL", { modelName: "activity" });
+
+		this.dataRequestEmail = await MailModule.runJob("GET_SCHEMA_ASYNC", { schemaName: "dataRequest" });
+	}
+
+	/**
+	 * Removes a user and associated data
+	 * @param {object} payload - object that contains the payload
+	 * @param {string} payload.userId - id of the user to remove
+	 * @returns {Promise} - returns a promise (resolve, reject)
+	 */
+	async REMOVE_USER(payload) {
+		const { userId } = payload;
+
+		// Create data request, in case the process fails halfway through. An admin can finish the removal manually
+		const dataRequest = await UsersModule.dataRequestModel.create({ userId, type: "remove" });
+		await WSModule.runJob(
+			"EMIT_TO_ROOM",
+			{
+				room: "admin.users",
+				args: ["event:admin.dataRequests.created", { data: { request: dataRequest } }]
+			},
+			this
+		);
+
+		if (config.get("sendDataRequestEmails")) {
+			const adminUsers = await UsersModule.userModel.find({ role: "admin" });
+			const to = adminUsers.map(adminUser => adminUser.email.address);
+			await UsersModule.dataRequestEmail(to, userId, "remove");
+		}
+
+		// Delete activities
+		await UsersModule.activityModel.deleteMany({ userId });
+
+		// Delete stations and associated data
+		const stations = await UsersModule.stationModel.find({ owner: userId });
+		const stationJobs = stations.map(station => async () => {
+			const { _id: stationId } = station;
+
+			await UsersModule.stationModel.deleteOne({ _id: stationId });
+			await CacheModule.runJob("HDEL", { table: "stations", key: stationId }, this);
+
+			if (!station.playlist) return;
+
+			await PlaylistsModule.runJob("DELETE_PLAYLIST", { playlistId: station.playlist }, this);
+		});
+		await Promise.all(stationJobs);
+
+		// Remove user as dj
+		await UsersModule.stationModel.updateMany({ djs: userId }, { $pull: { djs: userId } });
+
+		// Collect songs to adjust ratings for later
+		const likedPlaylist = await UsersModule.playlistModel.findOne({ createdBy: userId, type: "user-liked" });
+		const dislikedPlaylist = await UsersModule.playlistModel.findOne({ createdBy: userId, type: "user-disliked" });
+		const songsToAdjustRatings = [
+			...likedPlaylist.songs.map(({ mediaSource }) => mediaSource),
+			...dislikedPlaylist.songs.map(({ mediaSource }) => mediaSource)
+		];
+
+		// Delete playlists created by user
+		await UsersModule.playlistModel.deleteMany({ createdBy: userId });
+
+		// TODO Maybe we don't need to wait for this to finish?
+		// Recalculate ratings of songs the user liked/disliked
+		const recalculateRatingsJobs = songsToAdjustRatings.map(songsToAdjustRating =>
+			MediaModule.runJob("RECALCULATE_RATINGS", { mediaSource: songsToAdjustRating }, this)
+		);
+		await Promise.all(recalculateRatingsJobs);
+
+		// Delete user object
+		await UsersModule.userModel.deleteMany({ _id: userId });
+
+		// Remove sessions from Redis and MongoDB
+		await CacheModule.runJob("PUB", { channel: "user.removeSessions", value: userId }, this);
+
+		const sessions = await CacheModule.runJob("HGETALL", { table: "sessions" }, this);
+		const sessionIds = Object.keys(sessions);
+		const sessionJobs = sessionIds.map(sessionId => async () => {
+			const session = sessions[sessionId];
+			if (!session || session.userId !== userId) return;
+
+			await CacheModule.runJob("HDEL", { table: "sessions", key: sessionId }, this);
+		});
+		await Promise.all(sessionJobs);
+
+		await CacheModule.runJob(
+			"PUB",
+			{
+				channel: "user.removeAccount",
+				value: userId
+			},
+			this
+		);
+	}
+
+	// EXAMPLE_JOB() {
+	// 	return new Promise((resolve, reject) => {
+	// 		if (true) resolve({});
+	// 		else reject(new Error("Nothing changed."));
+	// 	});
+	// }
+}
+
+export default new _UsersModule();