| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553 | import async from "async";import CoreClass from "../core";let PlaylistsModule;let SongsModule;let CacheModule;let DBModule;let UtilsModule;class _PlaylistsModule extends CoreClass {	// eslint-disable-next-line require-jsdoc	constructor() {		super("playlists");		PlaylistsModule = this;	}	/**	 * Initialises the playlists module	 *	 * @returns {Promise} - returns promise (reject, resolve)	 */	async initialize() {		this.setStage(1);		CacheModule = this.moduleManager.modules.cache;		DBModule = this.moduleManager.modules.db;		UtilsModule = this.moduleManager.modules.utils;		SongsModule = this.moduleManager.modules.songs;		this.playlistModel = await DBModule.runJob("GET_MODEL", { modelName: "playlist" });		this.playlistSchemaCache = await CacheModule.runJob("GET_SCHEMA", { schemaName: "playlist" });		this.setStage(2);		return new Promise((resolve, reject) =>			async.waterfall(				[					next => {						this.setStage(3);						CacheModule.runJob("HGETALL", { table: "playlists" })							.then(playlists => {								next(null, playlists);							})							.catch(next);					},					(playlists, next) => {						this.setStage(4);						if (!playlists) return next();						const playlistIds = Object.keys(playlists);						return async.each(							playlistIds,							(playlistId, next) => {								PlaylistsModule.playlistModel.findOne({ _id: playlistId }, (err, playlist) => {									if (err) next(err);									else if (!playlist) {										CacheModule.runJob("HDEL", {											table: "playlists",											key: playlistId										})											.then(() => next())											.catch(next);									} else next();								});							},							next						);					},					next => {						this.setStage(5);						PlaylistsModule.playlistModel.find({}, next);					},					(playlists, next) => {						this.setStage(6);						async.each(							playlists,							(playlist, cb) => {								CacheModule.runJob("HSET", {									table: "playlists",									key: playlist._id,									value: PlaylistsModule.playlistSchemaCache(playlist)								})									.then(() => cb())									.catch(next);							},							next						);					}				],				async err => {					if (err) {						const formattedErr = await UtilsModule.runJob("GET_ERROR", {							error: err						});						reject(new Error(formattedErr));					} else resolve();				}			)		);	}	/**	 * Creates a playlist that is not generated or editable by a user e.g. liked songs playlist	 *	 * @param {object} payload - object that contains the payload	 * @param {string} payload.userId - the id of the user to create the playlist for	 * @param {string} payload.displayName - the display name of the playlist	 * @returns {Promise} - returns promise (reject, resolve)	 */	CREATE_READ_ONLY_PLAYLIST(payload) {		return new Promise((resolve, reject) => {			PlaylistsModule.playlistModel.create(				{					isUserModifiable: false,					displayName: payload.displayName,					songs: [],					createdBy: payload.userId,					createdAt: Date.now(),					createdFor: null,					type: payload.type				},				(err, playlist) => {					if (err) return reject(new Error(err));					return resolve(playlist._id);				}			);		});	}	/**	 * Creates a playlist that contains all songs of a specific genre	 *	 * @param {object} payload - object that contains the payload	 * @param {string} payload.genre - the genre	 * @returns {Promise} - returns promise (reject, resolve)	 */	CREATE_GENRE_PLAYLIST(payload) {		return new Promise((resolve, reject) => {			PlaylistsModule.runJob("GET_GENRE_PLAYLIST", { genre: payload.genre.toLowerCase() }, this)				.then(() => {					reject(new Error("Playlist already exists"));				})				.catch(err => {					if (err.message === "Playlist not found") {						PlaylistsModule.playlistModel.create(							{								isUserModifiable: false,								displayName: `Genre - ${payload.genre}`,								songs: [],								createdBy: "Musare",								createdFor: `${payload.genre.toLowerCase()}`,								createdAt: Date.now(),								type: "genre"							},							(err, playlist) => {								if (err) return reject(new Error(err));								return resolve(playlist._id);							}						);					} else reject(new Error(err));				});		});	}	/**	 * Gets all genre playlists	 *	 * @param {object} payload - object that contains the payload	 * @param {string} payload.includeSongs - include the songs	 * @returns {Promise} - returns promise (reject, resolve)	 */	GET_ALL_GENRE_PLAYLISTS(payload) {		return new Promise((resolve, reject) => {			const includeObject = payload.includeSongs ? null : { songs: false };			PlaylistsModule.playlistModel.find({ type: "genre" }, includeObject, (err, playlists) => {				if (err) reject(new Error(err));				else resolve({ playlists });			});		});	}	/**	 * Gets a genre playlist	 *	 * @param {object} payload - object that contains the payload	 * @param {string} payload.genre - the genre	 * @param {string} payload.includeSongs - include the songs	 * @returns {Promise} - returns promise (reject, resolve)	 */	GET_GENRE_PLAYLIST(payload) {		return new Promise((resolve, reject) => {			const includeObject = payload.includeSongs ? null : { songs: false };			PlaylistsModule.playlistModel.findOne(				{ type: "genre", createdFor: payload.genre },				includeObject,				(err, playlist) => {					if (err) reject(new Error(err));					else if (!playlist) reject(new Error("Playlist not found"));					else resolve({ playlist });				}			);		});	}	/**	 * Adds a song to a playlist	 *	 * @param {object} payload - object that contains the payload	 * @param {string} payload.playlistId - the playlist id	 * @param {string} payload.song - the song	 * @returns {Promise} - returns promise (reject, resolve)	 */	ADD_SONG_TO_PLAYLIST(payload) {		return new Promise((resolve, reject) => {			const song = {				_id: payload.song._id,				songId: payload.song.songId,				title: payload.song.title,				duration: payload.song.duration			};			PlaylistsModule.playlistModel.updateOne(				{ _id: payload.playlistId },				{ $push: { songs: song } },				{ runValidators: true },				err => {					if (err) reject(new Error(err));					else {						PlaylistsModule.runJob("UPDATE_PLAYLIST", { playlistId: payload.playlistId }, this)							.then(() => resolve())							.catch(err => {								reject(new Error(err));							});					}				}			);		});	}	/**	 * Deletes a song from a playlist based on the songId	 *	 * @param {object} payload - object that contains the payload	 * @param {string} payload.playlistId - the playlist id	 * @param {string} payload.songId - the songId	 * @returns {Promise} - returns promise (reject, resolve)	 */	DELETE_SONG_FROM_PLAYLIST_BY_SONGID(payload) {		return new Promise((resolve, reject) => {			PlaylistsModule.playlistModel.updateOne(				{ _id: payload.playlistId },				{ $pull: { songs: { songId: payload.songId } } },				err => {					if (err) reject(new Error(err));					else {						PlaylistsModule.runJob("UPDATE_PLAYLIST", { playlistId: payload.playlistId }, this)							.then(() => resolve())							.catch(err => {								reject(new Error(err));							});					}				}			);		});	}	/**	 * Fills a genre playlist with songs	 *	 * @param {object} payload - object that contains the payload	 * @param {string} payload.genre - the genre	 * @returns {Promise} - returns promise (reject, resolve)	 */	AUTOFILL_GENRE_PLAYLIST(payload) {		return new Promise((resolve, reject) => {			async.waterfall(				[					next => {						PlaylistsModule.runJob(							"GET_GENRE_PLAYLIST",							{ genre: payload.genre.toLowerCase(), includeSongs: true },							this						)							.then(response => {								next(null, { playlist: response.playlist });							})							.catch(err => {								if (err.message === "Playlist not found") {									PlaylistsModule.runJob("CREATE_GENRE_PLAYLIST", { genre: payload.genre }, this)										.then(playlistId => {											next(null, { playlist: { _id: playlistId, songs: [] } });										})										.catch(err => {											next(err);										});								} else next(err);							});					},					(data, next) => {						SongsModule.runJob("GET_ALL_SONGS_WITH_GENRE", { genre: payload.genre }, this)							.then(response => {								data.songs = response.songs;								next(null, data);							})							.catch(err => {								console.log(err);								next(err);							});					},					(data, next) => {						data.songsToDelete = [];						data.songsToAdd = [];						data.playlist.songs.forEach(playlistSong => {							const found = data.songs.find(song => playlistSong.songId === song.songId);							if (!found) data.songsToDelete.push(playlistSong);						});						data.songs.forEach(song => {							const found = data.playlist.songs.find(playlistSong => song.songId === playlistSong.songId);							if (!found) data.songsToAdd.push(song);						});						next(null, data);					},					(data, next) => {						const promises = [];						data.songsToAdd.forEach(song => {							promises.push(								PlaylistsModule.runJob(									"ADD_SONG_TO_PLAYLIST",									{ playlistId: data.playlist._id, song },									this								)							);						});						data.songsToDelete.forEach(song => {							promises.push(								PlaylistsModule.runJob(									"DELETE_SONG_FROM_PLAYLIST_BY_SONGID",									{										playlistId: data.playlist._id,										songId: song.songId									},									this								)							);						});						Promise.allSettled(promises)							.then(() => {								next();							})							.catch(err => {								next(err);							});					}				],				err => {					if (err && err !== true) return reject(new Error(err));					return resolve({});				}			);		});	}	/**	 * Gets a playlist by id from the cache or Mongo, and if it isn't in the cache yet, adds it the cache	 *	 * @param {object} payload - object that contains the payload	 * @param {string} payload.playlistId - the id of the playlist we are trying to get	 * @returns {Promise} - returns promise (reject, resolve)	 */	GET_PLAYLIST(payload) {		return new Promise((resolve, reject) =>			async.waterfall(				[					next => {						CacheModule.runJob("HGETALL", { table: "playlists" }, this)							.then(playlists => next(null, playlists))							.catch(next);					},					(playlists, next) => {						if (!playlists) return next();						const playlistIds = Object.keys(playlists);						return async.each(							playlistIds,							(playlistId, next) => {								PlaylistsModule.playlistModel.findOne({ _id: playlistId }, (err, playlist) => {									if (err) next(err);									else if (!playlist) {										CacheModule.runJob(											"HDEL",											{												table: "playlists",												key: playlistId											},											this										)											.then(() => next())											.catch(next);									} else next();								});							},							next						);					},					next => {						CacheModule.runJob(							"HGET",							{								table: "playlists",								key: payload.playlistId							},							this						)							.then(playlist => next(null, playlist))							.catch(next);					},					(playlist, next) => {						if (playlist) return next(true, playlist);						return PlaylistsModule.playlistModel.findOne({ _id: payload.playlistId }, next);					},					(playlist, next) => {						if (playlist) {							CacheModule.runJob(								"HSET",								{									table: "playlists",									key: payload.playlistId,									value: playlist								},								this							)								.then(playlist => {									next(null, playlist);								})								.catch(next);						} else next("Playlist not found");					}				],				(err, playlist) => {					if (err && err !== true) return reject(new Error(err));					return resolve(playlist);				}			)		);	}	/**	 * Gets a playlist from id from Mongo and updates the cache with it	 *	 * @param {object} payload - object that contains the payload	 * @param {string} payload.playlistId - the id of the playlist we are trying to update	 * @returns {Promise} - returns promise (reject, resolve)	 */	UPDATE_PLAYLIST(payload) {		// playlistId, cb		return new Promise((resolve, reject) =>			async.waterfall(				[					next => {						PlaylistsModule.playlistModel.findOne({ _id: payload.playlistId }, next);					},					(playlist, next) => {						if (!playlist) {							CacheModule.runJob("HDEL", {								table: "playlists",								key: payload.playlistId							});							return next("Playlist not found");						}						return CacheModule.runJob(							"HSET",							{								table: "playlists",								key: payload.playlistId,								value: playlist							},							this						)							.then(playlist => {								next(null, playlist);							})							.catch(next);					}				],				(err, playlist) => {					if (err && err !== true) return reject(new Error(err));					return resolve(playlist);				}			)		);	}	/**	 * Deletes playlist from id from Mongo and cache	 *	 * @param {object} payload - object that contains the payload	 * @param {string} payload.playlistId - the id of the playlist we are trying to delete	 * @returns {Promise} - returns promise (reject, resolve)	 */	DELETE_PLAYLIST(payload) {		// playlistId, cb		return new Promise((resolve, reject) =>			async.waterfall(				[					next => {						PlaylistsModule.playlistModel.deleteOne({ _id: payload.playlistId }, next);					},					(res, next) => {						CacheModule.runJob(							"HDEL",							{								table: "playlists",								key: payload.playlistId							},							this						)							.then(() => next())							.catch(next);					}				],				err => {					if (err && err !== true) return reject(new Error(err));					return resolve();				}			)		);	}}export default new _PlaylistsModule();
 |