| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173 | "use strict";const async = require("async");const hooks = require("./hooks");const moduleManager = require("../../index");const db = require("../db");const cache = require("../cache");const utils = require("../utils");const playlists = require("../playlists");const songs = require("../songs");const activities = require("../activities");cache.runJob("SUB", {    channel: "playlist.create",    cb: (playlistId) => {        playlists.runJob("GET_PLAYLIST", { playlistId }).then((playlist) => {            utils                .runJob("SOCKETS_FROM_USER", { userId: playlist.createdBy })                .then((response) => {                    response.sockets.forEach((socket) => {                        socket.emit("event:playlist.create", playlist);                    });                });        });    },});cache.runJob("SUB", {    channel: "playlist.delete",    cb: (res) => {        utils            .runJob("SOCKETS_FROM_USER", { userId: res.userId })            .then((response) => {                response.sockets.forEach((socket) => {                    socket.emit("event:playlist.delete", res.playlistId);                });            });    },});cache.runJob("SUB", {    channel: "playlist.moveSongToTop",    cb: (res) => {        utils            .runJob("SOCKETS_FROM_USER", { userId: res.userId })            .then((response) => {                response.sockets.forEach((socket) => {                    socket.emit("event:playlist.moveSongToTop", {                        playlistId: res.playlistId,                        songId: res.songId,                    });                });            });    },});cache.runJob("SUB", {    channel: "playlist.moveSongToBottom",    cb: (res) => {        utils            .runJob("SOCKETS_FROM_USER", { userId: res.userId })            .then((response) => {                response.sockets.forEach((socket) => {                    socket.emit("event:playlist.moveSongToBottom", {                        playlistId: res.playlistId,                        songId: res.songId,                    });                });            });    },});cache.runJob("SUB", {    channel: "playlist.addSong",    cb: (res) => {        utils            .runJob("SOCKETS_FROM_USER", { userId: res.userId })            .then((response) => {                response.sockets.forEach((socket) => {                    socket.emit("event:playlist.addSong", {                        playlistId: res.playlistId,                        song: res.song,                    });                });            });    },});cache.runJob("SUB", {    channel: "playlist.removeSong",    cb: (res) => {        utils            .runJob("SOCKETS_FROM_USER", { userId: res.userId })            .then((response) => {                response.sockets.forEach((socket) => {                    socket.emit("event:playlist.removeSong", {                        playlistId: res.playlistId,                        songId: res.songId,                    });                });            });    },});cache.runJob("SUB", {    channel: "playlist.updateDisplayName",    cb: (res) => {        utils            .runJob("SOCKETS_FROM_USER", { userId: res.userId })            .then((response) => {                response.sockets.forEach((socket) => {                    socket.emit("event:playlist.updateDisplayName", {                        playlistId: res.playlistId,                        displayName: res.displayName,                    });                });            });    },});let lib = {    /**     * Gets the first song from a private playlist     *     * @param {Object} session - the session object automatically added by socket.io     * @param {String} playlistId - the id of the playlist we are getting the first song from     * @param {Function} cb - gets called with the result     */    getFirstSong: hooks.loginRequired((session, playlistId, cb) => {        async.waterfall(            [                (next) => {                    playlists                        .runJob("GET_PLAYLIST", { playlistId })                        .then((playlist) => next(null, playlist))                        .catch(next);                },                (playlist, next) => {                    if (!playlist || playlist.createdBy !== session.userId)                        return next("Playlist not found.");                    next(null, playlist.songs[0]);                },            ],            async (err, song) => {                if (err) {                    err = await utils.runJob("GET_ERROR", { error: err });                    console.log(                        "ERROR",                        "PLAYLIST_GET_FIRST_SONG",                        `Getting the first song of playlist "${playlistId}" failed for user "${session.userId}". "${err}"`                    );                    return cb({ status: "failure", message: err });                }                console.log(                    "SUCCESS",                    "PLAYLIST_GET_FIRST_SONG",                    `Successfully got the first song of playlist "${playlistId}" for user "${session.userId}".`                );                cb({                    status: "success",                    song: song,                });            }        );    }),    /**     * Gets all playlists for the user requesting it     *     * @param {Object} session - the session object automatically added by socket.io     * @param {Function} cb - gets called with the result     */    indexForUser: hooks.loginRequired(async (session, cb) => {        const playlistModel = await db.runJob("GET_MODEL", {            modelName: "playlist",        });        async.waterfall(            [                (next) => {                    playlistModel.find({ createdBy: session.userId }, next);                },            ],            async (err, playlists) => {                if (err) {                    err = await utils.runJob("GET_ERROR", { error: err });                    console.log(                        "ERROR",                        "PLAYLIST_INDEX_FOR_USER",                        `Indexing playlists for user "${session.userId}" failed. "${err}"`                    );                    return cb({ status: "failure", message: err });                }                console.log(                    "SUCCESS",                    "PLAYLIST_INDEX_FOR_USER",                    `Successfully indexed playlists for user "${session.userId}".`                );                cb({                    status: "success",                    data: playlists,                });            }        );    }),    /**     * Creates a new private playlist     *     * @param {Object} session - the session object automatically added by socket.io     * @param {Object} data - the data for the new private playlist     * @param {Function} cb - gets called with the result     */    create: hooks.loginRequired(async (session, data, cb) => {        const playlistModel = await db.runJob("GET_MODEL", {            modelName: "playlist",        });        async.waterfall(            [                (next) => {                    return data                        ? next()                        : cb({ status: "failure", message: "Invalid data" });                },                (next) => {                    const { displayName, songs } = data;                    playlistModel.create(                        {                            displayName,                            songs,                            createdBy: session.userId,                            createdAt: Date.now(),                        },                        next                    );                },            ],            async (err, playlist) => {                if (err) {                    err = await utils.runJob("GET_ERROR", { error: err });                    console.log(                        "ERROR",                        "PLAYLIST_CREATE",                        `Creating private playlist failed for user "${session.userId}". "${err}"`                    );                    return cb({ status: "failure", message: err });                }                cache.runJob("PUB", {                    channel: "playlist.create",                    value: playlist._id,                });                activities.runJob("ADD_ACTIVITY", {                    userId: session.userId,                    activityType: "created_playlist",                    payload: [playlist._id],                });                console.log(                    "SUCCESS",                    "PLAYLIST_CREATE",                    `Successfully created private playlist for user "${session.userId}".`                );                cb({                    status: "success",                    message: "Successfully created playlist",                    data: {                        _id: playlist._id,                    },                });            }        );    }),    /**     * Gets a playlist from id     *     * @param {Object} session - the session object automatically added by socket.io     * @param {String} playlistId - the id of the playlist we are getting     * @param {Function} cb - gets called with the result     */    getPlaylist: hooks.loginRequired((session, playlistId, cb) => {        async.waterfall(            [                (next) => {                    playlists                        .runJob("GET_PLAYLIST", { playlistId })                        .then((playlist) => next(null, playlist))                        .catch(next);                },                (playlist, next) => {                    if (!playlist || playlist.createdBy !== session.userId)                        return next("Playlist not found");                    next(null, playlist);                },            ],            async (err, playlist) => {                if (err) {                    err = await utils.runJob("GET_ERROR", { error: err });                    console.log(                        "ERROR",                        "PLAYLIST_GET",                        `Getting private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`                    );                    return cb({ status: "failure", message: err });                }                console.log(                    "SUCCESS",                    "PLAYLIST_GET",                    `Successfully got private playlist "${playlistId}" for user "${session.userId}".`                );                cb({                    status: "success",                    data: playlist,                });            }        );    }),    /**     * Obtains basic metadata of a playlist in order to format an activity     *     * @param session     * @param playlistId - the playlist id     * @param cb     */    getPlaylistForActivity: (session, playlistId, cb) => {        async.waterfall(            [                (next) => {                    playlists                        .runJob("GET_PLAYLIST", { playlistId })                        .then((playlist) => next(null, playlist))                        .catch(next);                },            ],            async (err, playlist) => {                if (err) {                    err = await utils.runJob("GET_ERROR", { error: err });                    console.log(                        "ERROR",                        "PLAYLISTS_GET_PLAYLIST_FOR_ACTIVITY",                        `Failed to obtain metadata of playlist ${playlistId} for activity formatting. "${err}"`                    );                    return cb({ status: "failure", message: err });                } else {                    console.log(                        "SUCCESS",                        "PLAYLISTS_GET_PLAYLIST_FOR_ACTIVITY",                        `Obtained metadata of playlist ${playlistId} for activity formatting successfully.`                    );                    cb({                        status: "success",                        data: {                            title: playlist.displayName,                        },                    });                }            }        );    },    //TODO Remove this    /**     * Updates a private playlist     *     * @param {Object} session - the session object automatically added by socket.io     * @param {String} playlistId - the id of the playlist we are updating     * @param {Object} playlist - the new private playlist object     * @param {Function} cb - gets called with the result     */    update: hooks.loginRequired(async (session, playlistId, playlist, cb) => {        const playlistModel = await db.runJob("GET_MODEL", {            modelName: "playlist",        });        async.waterfall(            [                (next) => {                    playlistModel.updateOne(                        { _id: playlistId, createdBy: session.userId },                        playlist,                        { runValidators: true },                        next                    );                },                (res, next) => {                    playlists                        .runJob("UPDATE_PLAYLIST", { playlistId })                        .then((playlist) => next(null, playlist))                        .catch(next);                },            ],            async (err, playlist) => {                if (err) {                    err = await utils.runJob("GET_ERROR", { error: err });                    console.log(                        "ERROR",                        "PLAYLIST_UPDATE",                        `Updating private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`                    );                    return cb({ status: "failure", message: err });                }                console.log(                    "SUCCESS",                    "PLAYLIST_UPDATE",                    `Successfully updated private playlist "${playlistId}" for user "${session.userId}".`                );                cb({                    status: "success",                    data: playlist,                });            }        );    }),    /**     * Updates a private playlist     *     * @param {Object} session - the session object automatically added by socket.io     * @param {String} playlistId - the id of the playlist we are updating     * @param {Function} cb - gets called with the result     */    shuffle: hooks.loginRequired(async (session, playlistId, cb) => {        const playlistModel = await db.runJob("GET_MODEL", {            modelName: "playlist",        });        async.waterfall(            [                (next) => {                    if (!playlistId) return next("No playlist id.");                    playlistModel.findById(playlistId, next);                },                (playlist, next) => {                    utils                        .runJob("SHUFFLE", { array: playlist.songs })                        .then((result) => next(null, result.array))                        .catch(next);                },                (songs, next) => {                    playlistModel.updateOne(                        { _id: playlistId },                        { $set: { songs } },                        { runValidators: true },                        next                    );                },                (res, next) => {                    playlists                        .runJob("UPDATE_PLAYLIST", { playlistId })                        .then((playlist) => next(null, playlist))                        .catch(next);                },            ],            async (err, playlist) => {                if (err) {                    err = await utils.runJob("GET_ERROR", { error: err });                    console.log(                        "ERROR",                        "PLAYLIST_SHUFFLE",                        `Updating private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`                    );                    return cb({ status: "failure", message: err });                }                console.log(                    "SUCCESS",                    "PLAYLIST_SHUFFLE",                    `Successfully updated private playlist "${playlistId}" for user "${session.userId}".`                );                cb({                    status: "success",                    message: "Successfully shuffled playlist.",                    data: playlist,                });            }        );    }),    /**     * Adds a song to a private playlist     *     * @param {Object} session - the session object automatically added by socket.io     * @param {Boolean} isSet - is the song part of a set of songs to be added     * @param {String} songId - the id of the song we are trying to add     * @param {String} playlistId - the id of the playlist we are adding the song to     * @param {Function} cb - gets called with the result     */    addSongToPlaylist: hooks.loginRequired(        async (session, isSet, songId, playlistId, cb) => {            const playlistModel = await db.runJob("GET_MODEL", {                modelName: "playlist",            });            async.waterfall(                [                    (next) => {                        playlists                            .runJob("GET_PLAYLIST", { playlistId })                            .then((playlist) => {                                if (                                    !playlist ||                                    playlist.createdBy !== session.userId                                )                                    return next(                                        "Something went wrong when trying to get the playlist"                                    );                                async.each(                                    playlist.songs,                                    (song, next) => {                                        if (song.songId === songId)                                            return next(                                                "That song is already in the playlist"                                            );                                        next();                                    },                                    next                                );                            })                            .catch(next);                    },                    (next) => {                        songs                            .runJob("GET_SONG", { id: songId })                            .then((response) => {                                const song = response.song;                                next(null, {                                    _id: song._id,                                    songId: songId,                                    title: song.title,                                    duration: song.duration,                                });                            })                            .catch(() => {                                utils                                    .runJob("GET_SONG_FROM_YOUTUBE", { songId })                                    .then((response) =>                                        next(null, response.song)                                    )                                    .catch(next);                            });                    },                    (newSong, next) => {                        playlistModel.updateOne(                            { _id: playlistId },                            { $push: { songs: newSong } },                            { runValidators: true },                            (err) => {                                if (err) return next(err);                                playlists                                    .runJob("UPDATE_PLAYLIST", { playlistId })                                    .then((playlist) =>                                        next(null, playlist, newSong)                                    )                                    .catch(next);                            }                        );                    },                ],                async (err, playlist, newSong) => {                    if (err) {                        err = await utils.runJob("GET_ERROR", { error: err });                        console.log(                            "ERROR",                            "PLAYLIST_ADD_SONG",                            `Adding song "${songId}" to private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`                        );                        return cb({ status: "failure", message: err });                    } else {                        console.log(                            "SUCCESS",                            "PLAYLIST_ADD_SONG",                            `Successfully added song "${songId}" to private playlist "${playlistId}" for user "${session.userId}".`                        );                        if (!isSet)                            activities.runJob("ADD_ACTIVITY", {                                userId: session.userId,                                activityType: "added_song_to_playlist",                                payload: [{ songId, playlistId }],                            });                        cache.runJob("PUB", {                            channel: "playlist.addSong",                            value: {                                playlistId: playlist._id,                                song: newSong,                                userId: session.userId,                            },                        });                        return cb({                            status: "success",                            message:                                "Song has been successfully added to the playlist",                            data: playlist.songs,                        });                    }                }            );        }    ),    /**     * Adds a set of songs to a private playlist     *     * @param {Object} session - the session object automatically added by socket.io     * @param {String} url - the url of the the YouTube playlist     * @param {String} playlistId - the id of the playlist we are adding the set of songs to     * @param {Boolean} musicOnly - whether to only add music to the playlist     * @param {Function} cb - gets called with the result     */    addSetToPlaylist: hooks.loginRequired(        (session, url, playlistId, musicOnly, cb) => {            let videosInPlaylistTotal = 0;            let songsInPlaylistTotal = 0;            let songsSuccess = 0;            let songsFail = 0;            let addedSongs = [];            async.waterfall(                [                    (next) => {                        utils                            .runJob("GET_PLAYLIST_FROM_YOUTUBE", {                                url,                                musicOnly,                            })                            .then((response) => {                                if (response.filteredSongs) {                                    videosInPlaylistTotal =                                        response.songs.length;                                    songsInPlaylistTotal =                                        response.filteredSongs.length;                                } else {                                    songsInPlaylistTotal = videosInPlaylistTotal =                                        response.songs.length;                                }                                next(null, response.songs);                            });                    },                    (songIds, next) => {                        let processed = 0;                        function checkDone() {                            if (processed === songIds.length) next();                        }                        for (let s = 0; s < songIds.length; s++) {                            lib.addSongToPlaylist(                                session,                                true,                                songIds[s],                                playlistId,                                (res) => {                                    processed++;                                    if (res.status === "success") {                                        addedSongs.push(songIds[s]);                                        songsSuccess++;                                    } else songsFail++;                                    checkDone();                                }                            );                        }                    },                    (next) => {                        playlists                            .runJob("GET_PLAYLIST", { playlistId })                            .then((playlist) => next(null, playlist))                            .catch(next);                    },                    (playlist, next) => {                        if (!playlist || playlist.createdBy !== session.userId)                            return next("Playlist not found.");                        next(null, playlist);                    },                ],                async (err, playlist) => {                    if (err) {                        err = await utils.runJob("GET_ERROR", { error: err });                        console.log(                            "ERROR",                            "PLAYLIST_IMPORT",                            `Importing a YouTube playlist to private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`                        );                        return cb({ status: "failure", message: err });                    } else {                        activities.runJob("ADD_ACTIVITY", {                            userId: session.userId,                            activityType: "added_songs_to_playlist",                            payload: addedSongs,                        });                        console.log(                            "SUCCESS",                            "PLAYLIST_IMPORT",                            `Successfully imported a YouTube playlist to private playlist "${playlistId}" for user "${session.userId}". Videos in playlist: ${videosInPlaylistTotal}, songs in playlist: ${songsInPlaylistTotal}, songs successfully added: ${songsSuccess}, songs failed: ${songsFail}.`                        );                        cb({                            status: "success",                            message: "Playlist has been successfully imported.",                            data: playlist.songs,                            stats: {                                videosInPlaylistTotal,                                songsInPlaylistTotal,                                songsAddedSuccessfully: songsSuccess,                                songsFailedToAdd: songsFail,                            },                        });                    }                }            );        }    ),    /**     * Removes a song from a private playlist     *     * @param {Object} session - the session object automatically added by socket.io     * @param {String} songId - the id of the song we are removing from the private playlist     * @param {String} playlistId - the id of the playlist we are removing the song from     * @param {Function} cb - gets called with the result     */    removeSongFromPlaylist: hooks.loginRequired(        async (session, songId, playlistId, cb) => {            const playlistModel = await db.runJob("GET_MODEL", {                modelName: "playlist",            });            async.waterfall(                [                    (next) => {                        if (!songId || typeof songId !== "string")                            return next("Invalid song id.");                        if (!playlistId || typeof playlistId !== "string")                            return next("Invalid playlist id.");                        next();                    },                    (next) => {                        playlists                            .runJob("GET_PLAYLIST", { playlistId })                            .then((playlist) => next(null, playlist))                            .catch(next);                    },                    (playlist, next) => {                        if (!playlist || playlist.createdBy !== session.userId)                            return next("Playlist not found");                        playlistModel.updateOne(                            { _id: playlistId },                            { $pull: { songs: { songId: songId } } },                            next                        );                    },                    (res, next) => {                        playlists                            .runJob("UPDATE_PLAYLIST", { playlistId })                            .then((playlist) => next(null, playlist))                            .catch(next);                    },                ],                async (err, playlist) => {                    if (err) {                        err = await utils.runJob("GET_ERROR", { error: err });                        console.log(                            "ERROR",                            "PLAYLIST_REMOVE_SONG",                            `Removing song "${songId}" from private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`                        );                        return cb({ status: "failure", message: err });                    } else {                        console.log(                            "SUCCESS",                            "PLAYLIST_REMOVE_SONG",                            `Successfully removed song "${songId}" from private playlist "${playlistId}" for user "${session.userId}".`                        );                        cache.runJob("PUB", {                            channel: "playlist.removeSong",                            value: {                                playlistId: playlist._id,                                songId: songId,                                userId: session.userId,                            },                        });                        return cb({                            status: "success",                            message:                                "Song has been successfully removed from playlist",                            data: playlist.songs,                        });                    }                }            );        }    ),    /**     * Updates the displayName of a private playlist     *     * @param {Object} session - the session object automatically added by socket.io     * @param {String} playlistId - the id of the playlist we are updating the displayName for     * @param {Function} cb - gets called with the result     */    updateDisplayName: hooks.loginRequired(        async (session, playlistId, displayName, cb) => {            const playlistModel = await db.runJob("GET_MODEL", {                modelName: "playlist",            });            async.waterfall(                [                    (next) => {                        playlistModel.updateOne(                            { _id: playlistId, createdBy: session.userId },                            { $set: { displayName } },                            { runValidators: true },                            next                        );                    },                    (res, next) => {                        playlists                            .runJob("UPDATE_PLAYLIST", { playlistId })                            .then((playlist) => next(null, playlist))                            .catch(next);                    },                ],                async (err, playlist) => {                    if (err) {                        err = await utils.runJob("GET_ERROR", { error: err });                        console.log(                            "ERROR",                            "PLAYLIST_UPDATE_DISPLAY_NAME",                            `Updating display name to "${displayName}" for private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`                        );                        return cb({ status: "failure", message: err });                    }                    console.log(                        "SUCCESS",                        "PLAYLIST_UPDATE_DISPLAY_NAME",                        `Successfully updated display name to "${displayName}" for private playlist "${playlistId}" for user "${session.userId}".`                    );                    cache.runJob("PUB", {                        channel: "playlist.updateDisplayName",                        value: {                            playlistId: playlistId,                            displayName: displayName,                            userId: session.userId,                        },                    });                    return cb({                        status: "success",                        message: "Playlist has been successfully updated",                    });                }            );        }    ),    /**     * Moves a song to the top of the list in a private playlist     *     * @param {Object} session - the session object automatically added by socket.io     * @param {String} playlistId - the id of the playlist we are moving the song to the top from     * @param {String} songId - the id of the song we are moving to the top of the list     * @param {Function} cb - gets called with the result     */    moveSongToTop: hooks.loginRequired(        async (session, playlistId, songId, cb) => {            const playlistModel = await db.runJob("GET_MODEL", {                modelName: "playlist",            });            async.waterfall(                [                    (next) => {                        playlists                            .runJob("GET_PLAYLIST", { playlistId })                            .then((playlist) => next(null, playlist))                            .catch(next);                    },                    (playlist, next) => {                        if (!playlist || playlist.createdBy !== session.userId)                            return next("Playlist not found");                        async.each(                            playlist.songs,                            (song, next) => {                                if (song.songId === songId) return next(song);                                next();                            },                            (err) => {                                if (err && err.songId) return next(null, err);                                next("Song not found");                            }                        );                    },                    (song, next) => {                        playlistModel.updateOne(                            { _id: playlistId },                            { $pull: { songs: { songId } } },                            (err) => {                                if (err) return next(err);                                return next(null, song);                            }                        );                    },                    (song, next) => {                        playlistModel.updateOne(                            { _id: playlistId },                            {                                $push: {                                    songs: {                                        $each: [song],                                        $position: 0,                                    },                                },                            },                            next                        );                    },                    (res, next) => {                        playlists                            .runJob("UPDATE_PLAYLIST", { playlistId })                            .then((playlist) => next(null, playlist))                            .catch(next);                    },                ],                async (err, playlist) => {                    if (err) {                        err = await utils.runJob("GET_ERROR", { error: err });                        console.log(                            "ERROR",                            "PLAYLIST_MOVE_SONG_TO_TOP",                            `Moving song "${songId}" to the top for private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`                        );                        return cb({ status: "failure", message: err });                    }                    console.log(                        "SUCCESS",                        "PLAYLIST_MOVE_SONG_TO_TOP",                        `Successfully moved song "${songId}" to the top for private playlist "${playlistId}" for user "${session.userId}".`                    );                    cache.runJob("PUB", {                        channel: "playlist.moveSongToTop",                        value: {                            playlistId,                            songId,                            userId: session.userId,                        },                    });                    return cb({                        status: "success",                        message: "Playlist has been successfully updated",                    });                }            );        }    ),    /**     * Moves a song to the bottom of the list in a private playlist     *     * @param {Object} session - the session object automatically added by socket.io     * @param {String} playlistId - the id of the playlist we are moving the song to the bottom from     * @param {String} songId - the id of the song we are moving to the bottom of the list     * @param {Function} cb - gets called with the result     */    moveSongToBottom: hooks.loginRequired(        async (session, playlistId, songId, cb) => {            const playlistModel = await db.runJob("GET_MODEL", {                modelName: "playlist",            });            async.waterfall(                [                    (next) => {                        playlists                            .runJob("GET_PLAYLIST", { playlistId })                            .then((playlist) => next(null, playlist))                            .catch(next);                    },                    (playlist, next) => {                        if (!playlist || playlist.createdBy !== session.userId)                            return next("Playlist not found");                        async.each(                            playlist.songs,                            (song, next) => {                                if (song.songId === songId) return next(song);                                next();                            },                            (err) => {                                if (err && err.songId) return next(null, err);                                next("Song not found");                            }                        );                    },                    (song, next) => {                        playlistModel.updateOne(                            { _id: playlistId },                            { $pull: { songs: { songId } } },                            (err) => {                                if (err) return next(err);                                return next(null, song);                            }                        );                    },                    (song, next) => {                        playlistModel.updateOne(                            { _id: playlistId },                            {                                $push: {                                    songs: song,                                },                            },                            next                        );                    },                    (res, next) => {                        playlists                            .runJob("UPDATE_PLAYLIST", { playlistId })                            .then((playlist) => next(null, playlist))                            .catch(next);                    },                ],                async (err, playlist) => {                    if (err) {                        err = await utils.runJob("GET_ERROR", { error: err });                        console.log(                            "ERROR",                            "PLAYLIST_MOVE_SONG_TO_BOTTOM",                            `Moving song "${songId}" to the bottom for private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`                        );                        return cb({ status: "failure", message: err });                    }                    console.log(                        "SUCCESS",                        "PLAYLIST_MOVE_SONG_TO_BOTTOM",                        `Successfully moved song "${songId}" to the bottom for private playlist "${playlistId}" for user "${session.userId}".`                    );                    cache.runJob("PUB", {                        channel: "playlist.moveSongToBottom",                        value: {                            playlistId,                            songId,                            userId: session.userId,                        },                    });                    return cb({                        status: "success",                        message: "Playlist has been successfully updated",                    });                }            );        }    ),    /**     * Removes a private playlist     *     * @param {Object} session - the session object automatically added by socket.io     * @param {String} playlistId - the id of the playlist we are moving the song to the top from     * @param {Function} cb - gets called with the result     */    remove: hooks.loginRequired(async (session, playlistId, cb) => {        const stationModel = await db.runJob("GET_MODEL", {            modelName: "station",        });        async.waterfall(            [                (next) => {                    playlists                        .runJob("DELETE_PLAYLIST", { playlistId })                        .then((playlist) => next(null, playlist))                        .catch(next);                },                (next) => {                    stationModel.find({ privatePlaylist: playlistId }, next);                },                (stations, next) => {                    async.each(                        stations,                        (station, next) => {                            async.waterfall(                                [                                    (next) => {                                        stationModel.updateOne(                                            { _id: station._id },                                            { $set: { privatePlaylist: null } },                                            { runValidators: true },                                            next                                        );                                    },                                    (res, next) => {                                        if (!station.partyMode) {                                            moduleManager.modules["stations"]                                                .runJob("UPDATE_STATION", {                                                    stationId: station._id,                                                })                                                .then((station) =>                                                    next(null, station)                                                )                                                .catch(next);                                            cache.runJob("PUB", {                                                channel:                                                    "privatePlaylist.selected",                                                value: {                                                    playlistId: null,                                                    stationId: station._id,                                                },                                            });                                        } else next();                                    },                                ],                                (err) => {                                    next();                                }                            );                        },                        (err) => {                            next();                        }                    );                },            ],            async (err) => {                if (err) {                    err = await utils.runJob("GET_ERROR", { error: err });                    console.log(                        "ERROR",                        "PLAYLIST_REMOVE",                        `Removing private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`                    );                    return cb({ status: "failure", message: err });                }                console.log(                    "SUCCESS",                    "PLAYLIST_REMOVE",                    `Successfully removed private playlist "${playlistId}" for user "${session.userId}".`                );                cache.runJob("PUB", {                    channel: "playlist.delete",                    value: {                        userId: session.userId,                        playlistId,                    },                });                activities.runJob("ADD_ACTIVITY", {                    userId: session.userId,                    activityType: "deleted_playlist",                    payload: [playlistId],                });                return cb({                    status: "success",                    message: "Playlist successfully removed",                });            }        );    }),};module.exports = lib;
 |