playlists.js 38 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455
  1. import async from "async";
  2. import { isLoginRequired } from "./hooks";
  3. import moduleManager from "../../index";
  4. const DBModule = moduleManager.modules.db;
  5. const UtilsModule = moduleManager.modules.utils;
  6. const IOModule = moduleManager.modules.io;
  7. const SongsModule = moduleManager.modules.songs;
  8. const CacheModule = moduleManager.modules.cache;
  9. const PlaylistsModule = moduleManager.modules.playlists;
  10. const YouTubeModule = moduleManager.modules.youtube;
  11. const ActivitiesModule = moduleManager.modules.activities;
  12. CacheModule.runJob("SUB", {
  13. channel: "playlist.create",
  14. cb: playlist => {
  15. IOModule.runJob("SOCKETS_FROM_USER", { userId: playlist.createdBy }, this).then(response => {
  16. response.sockets.forEach(socket => {
  17. socket.emit("event:playlist.create", playlist);
  18. });
  19. });
  20. if (playlist.privacy === "public")
  21. IOModule.runJob("EMIT_TO_ROOM", {
  22. room: `profile-${playlist.createdBy}`,
  23. args: ["event:playlist.create", playlist]
  24. });
  25. }
  26. });
  27. CacheModule.runJob("SUB", {
  28. channel: "playlist.delete",
  29. cb: res => {
  30. IOModule.runJob("SOCKETS_FROM_USER", { userId: res.userId }, this).then(response => {
  31. response.sockets.forEach(socket => {
  32. socket.emit("event:playlist.delete", res.playlistId);
  33. });
  34. });
  35. IOModule.runJob("EMIT_TO_ROOM", {
  36. room: `profile-${res.userId}`,
  37. args: ["event:playlist.delete", res.playlistId]
  38. });
  39. }
  40. });
  41. CacheModule.runJob("SUB", {
  42. channel: "playlist.moveSongToTop",
  43. cb: res => {
  44. IOModule.runJob("SOCKETS_FROM_USER", { userId: res.userId }, this).then(response => {
  45. response.sockets.forEach(socket => {
  46. socket.emit("event:playlist.moveSongToTop", {
  47. playlistId: res.playlistId,
  48. songId: res.songId
  49. });
  50. });
  51. });
  52. }
  53. });
  54. CacheModule.runJob("SUB", {
  55. channel: "playlist.moveSongToBottom",
  56. cb: res => {
  57. IOModule.runJob("SOCKETS_FROM_USER", { userId: res.userId }, this).then(response => {
  58. response.sockets.forEach(socket => {
  59. socket.emit("event:playlist.moveSongToBottom", {
  60. playlistId: res.playlistId,
  61. songId: res.songId
  62. });
  63. });
  64. });
  65. }
  66. });
  67. CacheModule.runJob("SUB", {
  68. channel: "playlist.addSong",
  69. cb: res => {
  70. IOModule.runJob("SOCKETS_FROM_USER", { userId: res.userId }, this).then(response => {
  71. response.sockets.forEach(socket => {
  72. socket.emit("event:playlist.addSong", {
  73. playlistId: res.playlistId,
  74. song: res.song
  75. });
  76. });
  77. });
  78. if (res.privacy === "public")
  79. IOModule.runJob("EMIT_TO_ROOM", {
  80. room: `profile-${res.userId}`,
  81. args: [
  82. "event:playlist.addSong",
  83. {
  84. playlistId: res.playlistId,
  85. song: res.song
  86. }
  87. ]
  88. });
  89. }
  90. });
  91. CacheModule.runJob("SUB", {
  92. channel: "playlist.removeSong",
  93. cb: res => {
  94. IOModule.runJob("SOCKETS_FROM_USER", { userId: res.userId }, this).then(response => {
  95. response.sockets.forEach(socket => {
  96. socket.emit("event:playlist.removeSong", {
  97. playlistId: res.playlistId,
  98. songId: res.songId
  99. });
  100. });
  101. });
  102. if (res.privacy === "public")
  103. IOModule.runJob("EMIT_TO_ROOM", {
  104. room: `profile-${res.userId}`,
  105. args: [
  106. "event:playlist.removeSong",
  107. {
  108. playlistId: res.playlistId,
  109. songId: res.songId
  110. }
  111. ]
  112. });
  113. }
  114. });
  115. CacheModule.runJob("SUB", {
  116. channel: "playlist.updateDisplayName",
  117. cb: res => {
  118. IOModule.runJob("SOCKETS_FROM_USER", { userId: res.userId }, this).then(response => {
  119. response.sockets.forEach(socket => {
  120. socket.emit("event:playlist.updateDisplayName", {
  121. playlistId: res.playlistId,
  122. displayName: res.displayName
  123. });
  124. });
  125. });
  126. if (res.privacy === "public")
  127. IOModule.runJob("EMIT_TO_ROOM", {
  128. room: `profile-${res.userId}`,
  129. args: [
  130. "event:playlist.updateDisplayName",
  131. {
  132. playlistId: res.playlistId,
  133. displayName: res.displayName
  134. }
  135. ]
  136. });
  137. }
  138. });
  139. CacheModule.runJob("SUB", {
  140. channel: "playlist.updatePrivacy",
  141. cb: res => {
  142. IOModule.runJob("SOCKETS_FROM_USER", { userId: res.userId }, this).then(response => {
  143. response.sockets.forEach(socket => {
  144. socket.emit("event:playlist.updatePrivacy");
  145. });
  146. });
  147. if (res.playlist.privacy === "public")
  148. return IOModule.runJob("EMIT_TO_ROOM", {
  149. room: `profile-${res.userId}`,
  150. args: ["event:playlist.create", res.playlist]
  151. });
  152. return IOModule.runJob("EMIT_TO_ROOM", {
  153. room: `profile-${res.userId}`,
  154. args: ["event:playlist.delete", res.playlist._id]
  155. });
  156. }
  157. });
  158. export default {
  159. /**
  160. * Gets the first song from a private playlist
  161. *
  162. * @param {object} session - the session object automatically added by socket.io
  163. * @param {string} playlistId - the id of the playlist we are getting the first song from
  164. * @param {Function} cb - gets called with the result
  165. */
  166. getFirstSong: isLoginRequired(function getFirstSong(session, playlistId, cb) {
  167. async.waterfall(
  168. [
  169. next => {
  170. PlaylistsModule.runJob("GET_PLAYLIST", { playlistId }, this)
  171. .then(playlist => {
  172. next(null, playlist);
  173. })
  174. .catch(next);
  175. },
  176. (playlist, next) => {
  177. if (!playlist || playlist.createdBy !== session.userId) return next("Playlist not found.");
  178. return next(null, playlist.songs[0]);
  179. }
  180. ],
  181. async (err, song) => {
  182. if (err) {
  183. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  184. this.log(
  185. "ERROR",
  186. "PLAYLIST_GET_FIRST_SONG",
  187. `Getting the first song of playlist "${playlistId}" failed for user "${session.userId}". "${err}"`
  188. );
  189. return cb({ status: "failure", message: err });
  190. }
  191. this.log(
  192. "SUCCESS",
  193. "PLAYLIST_GET_FIRST_SONG",
  194. `Successfully got the first song of playlist "${playlistId}" for user "${session.userId}".`
  195. );
  196. return cb({
  197. status: "success",
  198. song
  199. });
  200. }
  201. );
  202. }),
  203. /**
  204. * Gets a list of all the playlists for a specific user
  205. *
  206. * @param {object} session - the session object automatically added by socket.io
  207. * @param {string} userId - the user id in question
  208. * @param {Function} cb - gets called with the result
  209. */
  210. indexForUser: async function indexForUser(session, userId, cb) {
  211. const playlistModel = await DBModule.runJob("GET_MODEL", { modelName: "playlist" }, this);
  212. const userModel = await DBModule.runJob("GET_MODEL", { modelName: "user" }, this);
  213. async.waterfall(
  214. [
  215. next => {
  216. userModel.findById(userId).select({ "preferences.orderOfPlaylists": -1 }).exec(next);
  217. },
  218. ({ preferences }, next) => {
  219. const { orderOfPlaylists } = preferences;
  220. const match = {
  221. createdBy: userId
  222. };
  223. // if a playlist order exists
  224. if (orderOfPlaylists > 0) match._id = { $in: orderOfPlaylists };
  225. playlistModel
  226. .aggregate()
  227. .match(match)
  228. .addFields({
  229. weight: { $indexOfArray: [orderOfPlaylists, "$_id"] }
  230. })
  231. .sort({ weight: 1 })
  232. .exec(next);
  233. },
  234. (playlists, next) => {
  235. if (session.userId === userId) return next(null, playlists); // user requesting playlists is the owner of the playlists
  236. const filteredPlaylists = [];
  237. return async.each(
  238. playlists,
  239. (playlist, nextPlaylist) => {
  240. if (playlist.privacy === "public") filteredPlaylists.push(playlist);
  241. return nextPlaylist();
  242. },
  243. () => next(null, filteredPlaylists)
  244. );
  245. }
  246. ],
  247. async (err, playlists) => {
  248. if (err) {
  249. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  250. this.log(
  251. "ERROR",
  252. "PLAYLIST_INDEX_FOR_USER",
  253. `Indexing playlists for user "${userId}" failed. "${err}"`
  254. );
  255. return cb({ status: "failure", message: err });
  256. }
  257. this.log("SUCCESS", "PLAYLIST_INDEX_FOR_USER", `Successfully indexed playlists for user "${userId}".`);
  258. return cb({
  259. status: "success",
  260. data: playlists
  261. });
  262. }
  263. );
  264. },
  265. /**
  266. * Gets all playlists for the user requesting it
  267. *
  268. * @param {object} session - the session object automatically added by socket.io
  269. * @param {boolean} showNonModifiablePlaylists - whether or not to show non modifiable playlists e.g. liked songs
  270. * @param {Function} cb - gets called with the result
  271. */
  272. indexMyPlaylists: isLoginRequired(async function indexMyPlaylists(session, showNonModifiablePlaylists, cb) {
  273. const playlistModel = await DBModule.runJob("GET_MODEL", { modelName: "playlist" }, this);
  274. const userModel = await DBModule.runJob("GET_MODEL", { modelName: "user" }, this);
  275. async.waterfall(
  276. [
  277. next => {
  278. userModel.findById(session.userId).select({ "preferences.orderOfPlaylists": -1 }).exec(next);
  279. },
  280. ({ preferences }, next) => {
  281. const { orderOfPlaylists } = preferences;
  282. const match = {
  283. createdBy: session.userId
  284. };
  285. // if non modifiable playlists should be shown as well
  286. if (!showNonModifiablePlaylists) match.isUserModifiable = true;
  287. // if a playlist order exists
  288. if (orderOfPlaylists > 0) match._id = { $in: orderOfPlaylists };
  289. playlistModel
  290. .aggregate()
  291. .match(match)
  292. .addFields({
  293. weight: { $indexOfArray: [orderOfPlaylists, "$_id"] }
  294. })
  295. .sort({ weight: 1 })
  296. .exec(next);
  297. }
  298. ],
  299. async (err, playlists) => {
  300. if (err) {
  301. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  302. this.log(
  303. "ERROR",
  304. "PLAYLIST_INDEX_FOR_ME",
  305. `Indexing playlists for user "${session.userId}" failed. "${err}"`
  306. );
  307. return cb({ status: "failure", message: err });
  308. }
  309. this.log(
  310. "SUCCESS",
  311. "PLAYLIST_INDEX_FOR_ME",
  312. `Successfully indexed playlists for user "${session.userId}".`
  313. );
  314. return cb({
  315. status: "success",
  316. data: playlists
  317. });
  318. }
  319. );
  320. }),
  321. /**
  322. * Creates a new private playlist
  323. *
  324. * @param {object} session - the session object automatically added by socket.io
  325. * @param {object} data - the data for the new private playlist
  326. * @param {Function} cb - gets called with the result
  327. */
  328. create: isLoginRequired(async function create(session, data, cb) {
  329. const playlistModel = await DBModule.runJob("GET_MODEL", { modelName: "playlist" }, this);
  330. const userModel = await DBModule.runJob("GET_MODEL", { modelName: "user" }, this);
  331. const blacklist = ["liked songs", "likedsongs", "disliked songs", "dislikedsongs"];
  332. async.waterfall(
  333. [
  334. next => (data ? next() : cb({ status: "failure", message: "Invalid data" })),
  335. next => {
  336. const { displayName, songs } = data;
  337. if (blacklist.indexOf(displayName.toLowerCase()) !== -1)
  338. return next("That playlist name is blacklisted. Please use a different name.");
  339. return playlistModel.create(
  340. {
  341. displayName,
  342. songs,
  343. createdBy: session.userId,
  344. createdAt: Date.now()
  345. },
  346. next
  347. );
  348. },
  349. (playlist, next) => {
  350. userModel.updateOne(
  351. { _id: session.userId },
  352. { $push: { "preferences.orderOfPlaylists": playlist._id } },
  353. err => {
  354. if (err) return next(err);
  355. return next(null, playlist);
  356. }
  357. );
  358. }
  359. ],
  360. async (err, playlist) => {
  361. if (err) {
  362. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  363. this.log(
  364. "ERROR",
  365. "PLAYLIST_CREATE",
  366. `Creating private playlist failed for user "${session.userId}". "${err}"`
  367. );
  368. return cb({ status: "failure", message: err });
  369. }
  370. CacheModule.runJob("PUB", {
  371. channel: "playlist.create",
  372. value: playlist
  373. });
  374. ActivitiesModule.runJob("ADD_ACTIVITY", {
  375. userId: session.userId,
  376. activityType: "created_playlist",
  377. payload: [playlist._id]
  378. });
  379. this.log(
  380. "SUCCESS",
  381. "PLAYLIST_CREATE",
  382. `Successfully created private playlist for user "${session.userId}".`
  383. );
  384. return cb({
  385. status: "success",
  386. message: "Successfully created playlist",
  387. data: {
  388. _id: playlist._id
  389. }
  390. });
  391. }
  392. );
  393. }),
  394. /**
  395. * Gets a playlist from id
  396. *
  397. * @param {object} session - the session object automatically added by socket.io
  398. * @param {string} playlistId - the id of the playlist we are getting
  399. * @param {Function} cb - gets called with the result
  400. */
  401. getPlaylist: isLoginRequired(function getPlaylist(session, playlistId, cb) {
  402. async.waterfall(
  403. [
  404. next => {
  405. PlaylistsModule.runJob("GET_PLAYLIST", { playlistId }, this)
  406. .then(playlist => {
  407. next(null, playlist);
  408. })
  409. .catch(next);
  410. },
  411. (playlist, next) => {
  412. if (!playlist || playlist.createdBy !== session.userId) return next("Playlist not found");
  413. return next(null, playlist);
  414. }
  415. ],
  416. async (err, playlist) => {
  417. if (err) {
  418. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  419. this.log(
  420. "ERROR",
  421. "PLAYLIST_GET",
  422. `Getting private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`
  423. );
  424. return cb({ status: "failure", message: err });
  425. }
  426. this.log(
  427. "SUCCESS",
  428. "PLAYLIST_GET",
  429. `Successfully got private playlist "${playlistId}" for user "${session.userId}".`
  430. );
  431. return cb({
  432. status: "success",
  433. data: playlist
  434. });
  435. }
  436. );
  437. }),
  438. /**
  439. * Obtains basic metadata of a playlist in order to format an activity
  440. *
  441. * @param {object} session - the session object automatically added by socket.io
  442. * @param {string} playlistId - the playlist id
  443. * @param {Function} cb - callback
  444. */
  445. getPlaylistForActivity(session, playlistId, cb) {
  446. async.waterfall(
  447. [
  448. next => {
  449. PlaylistsModule.runJob("GET_PLAYLIST", { playlistId }, this)
  450. .then(playlist => {
  451. next(null, playlist);
  452. })
  453. .catch(next);
  454. }
  455. ],
  456. async (err, playlist) => {
  457. if (err) {
  458. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  459. this.log(
  460. "ERROR",
  461. "PLAYLISTS_GET_PLAYLIST_FOR_ACTIVITY",
  462. `Failed to obtain metadata of playlist ${playlistId} for activity formatting. "${err}"`
  463. );
  464. return cb({ status: "failure", message: err });
  465. }
  466. this.log(
  467. "SUCCESS",
  468. "PLAYLISTS_GET_PLAYLIST_FOR_ACTIVITY",
  469. `Obtained metadata of playlist ${playlistId} for activity formatting successfully.`
  470. );
  471. return cb({
  472. status: "success",
  473. data: {
  474. title: playlist.displayName
  475. }
  476. });
  477. }
  478. );
  479. },
  480. // TODO Remove this
  481. /**
  482. * Updates a private playlist
  483. *
  484. * @param {object} session - the session object automatically added by socket.io
  485. * @param {string} playlistId - the id of the playlist we are updating
  486. * @param {object} playlist - the new private playlist object
  487. * @param {Function} cb - gets called with the result
  488. */
  489. update: isLoginRequired(async function update(session, playlistId, playlist, cb) {
  490. const playlistModel = await DBModule.runJob(
  491. "GET_MODEL",
  492. {
  493. modelName: "playlist"
  494. },
  495. this
  496. );
  497. async.waterfall(
  498. [
  499. next => {
  500. playlistModel.updateOne(
  501. { _id: playlistId, createdBy: session.userId },
  502. playlist,
  503. { runValidators: true },
  504. next
  505. );
  506. },
  507. (res, next) => {
  508. PlaylistsModule.runJob("UPDATE_PLAYLIST", { playlistId }, this)
  509. .then(playlist => {
  510. next(null, playlist);
  511. })
  512. .catch(next);
  513. }
  514. ],
  515. async (err, playlist) => {
  516. if (err) {
  517. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  518. this.log(
  519. "ERROR",
  520. "PLAYLIST_UPDATE",
  521. `Updating private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`
  522. );
  523. return cb({ status: "failure", message: err });
  524. }
  525. this.log(
  526. "SUCCESS",
  527. "PLAYLIST_UPDATE",
  528. `Successfully updated private playlist "${playlistId}" for user "${session.userId}".`
  529. );
  530. return cb({
  531. status: "success",
  532. data: playlist
  533. });
  534. }
  535. );
  536. }),
  537. /**
  538. * Updates a private playlist
  539. *
  540. * @param {object} session - the session object automatically added by socket.io
  541. * @param {string} playlistId - the id of the playlist we are updating
  542. * @param {Function} cb - gets called with the result
  543. */
  544. shuffle: isLoginRequired(async function shuffle(session, playlistId, cb) {
  545. const playlistModel = await DBModule.runJob(
  546. "GET_MODEL",
  547. {
  548. modelName: "playlist"
  549. },
  550. this
  551. );
  552. async.waterfall(
  553. [
  554. next => {
  555. if (!playlistId) return next("No playlist id.");
  556. return playlistModel.findById(playlistId, next);
  557. },
  558. (playlist, next) => {
  559. if (!playlist.isUserModifiable) return next("Playlist cannot be shuffled.");
  560. return UtilsModule.runJob("SHUFFLE", { array: playlist.songs }, this)
  561. .then(result => next(null, result.array))
  562. .catch(next);
  563. },
  564. (songs, next) => {
  565. playlistModel.updateOne({ _id: playlistId }, { $set: { songs } }, { runValidators: true }, next);
  566. },
  567. (res, next) => {
  568. PlaylistsModule.runJob("UPDATE_PLAYLIST", { playlistId }, this)
  569. .then(playlist => {
  570. next(null, playlist);
  571. })
  572. .catch(next);
  573. }
  574. ],
  575. async (err, playlist) => {
  576. if (err) {
  577. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  578. this.log(
  579. "ERROR",
  580. "PLAYLIST_SHUFFLE",
  581. `Updating private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`
  582. );
  583. return cb({ status: "failure", message: err });
  584. }
  585. this.log(
  586. "SUCCESS",
  587. "PLAYLIST_SHUFFLE",
  588. `Successfully updated private playlist "${playlistId}" for user "${session.userId}".`
  589. );
  590. return cb({
  591. status: "success",
  592. message: "Successfully shuffled playlist.",
  593. data: playlist
  594. });
  595. }
  596. );
  597. }),
  598. /**
  599. * Adds a song to a private playlist
  600. *
  601. * @param {object} session - the session object automatically added by socket.io
  602. * @param {boolean} isSet - is the song part of a set of songs to be added
  603. * @param {string} songId - the id of the song we are trying to add
  604. * @param {string} playlistId - the id of the playlist we are adding the song to
  605. * @param {Function} cb - gets called with the result
  606. */
  607. addSongToPlaylist: isLoginRequired(async function addSongToPlaylist(session, isSet, songId, playlistId, cb) {
  608. const playlistModel = await DBModule.runJob(
  609. "GET_MODEL",
  610. {
  611. modelName: "playlist"
  612. },
  613. this
  614. );
  615. async.waterfall(
  616. [
  617. next => {
  618. PlaylistsModule.runJob("GET_PLAYLIST", { playlistId }, this)
  619. .then(playlist => {
  620. if (!playlist || playlist.createdBy !== session.userId)
  621. return next("Something went wrong when trying to get the playlist");
  622. return async.each(
  623. playlist.songs,
  624. (song, next) => {
  625. if (song.songId === songId) return next("That song is already in the playlist");
  626. return next();
  627. },
  628. next
  629. );
  630. })
  631. .catch(next);
  632. },
  633. next => {
  634. SongsModule.runJob("GET_SONG", { id: songId }, this)
  635. .then(response => {
  636. const { song } = response;
  637. next(null, {
  638. _id: song._id,
  639. songId,
  640. title: song.title,
  641. duration: song.duration
  642. });
  643. })
  644. .catch(() => {
  645. YouTubeModule.runJob("GET_SONG", { songId }, this)
  646. .then(response => next(null, response.song))
  647. .catch(next);
  648. });
  649. },
  650. (newSong, next) => {
  651. playlistModel.updateOne(
  652. { _id: playlistId },
  653. { $push: { songs: newSong } },
  654. { runValidators: true },
  655. err => {
  656. if (err) return next(err);
  657. return PlaylistsModule.runJob("UPDATE_PLAYLIST", { playlistId }, this)
  658. .then(playlist => next(null, playlist, newSong))
  659. .catch(next);
  660. }
  661. );
  662. }
  663. ],
  664. async (err, playlist, newSong) => {
  665. if (err) {
  666. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  667. this.log(
  668. "ERROR",
  669. "PLAYLIST_ADD_SONG",
  670. `Adding song "${songId}" to private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`
  671. );
  672. return cb({ status: "failure", message: err });
  673. }
  674. this.log(
  675. "SUCCESS",
  676. "PLAYLIST_ADD_SONG",
  677. `Successfully added song "${songId}" to private playlist "${playlistId}" for user "${session.userId}".`
  678. );
  679. if (!isSet)
  680. ActivitiesModule.runJob("ADD_ACTIVITY", {
  681. userId: session.userId,
  682. activityType: "added_song_to_playlist",
  683. payload: [{ songId, playlistId }]
  684. });
  685. CacheModule.runJob("PUB", {
  686. channel: "playlist.addSong",
  687. value: {
  688. playlistId: playlist._id,
  689. song: newSong,
  690. userId: session.userId,
  691. privacy: playlist.privacy
  692. }
  693. });
  694. return cb({
  695. status: "success",
  696. message: "Song has been successfully added to the playlist",
  697. data: playlist.songs
  698. });
  699. }
  700. );
  701. }),
  702. /**
  703. * Adds a set of songs to a private playlist
  704. *
  705. * @param {object} session - the session object automatically added by socket.io
  706. * @param {string} url - the url of the the YouTube playlist
  707. * @param {string} playlistId - the id of the playlist we are adding the set of songs to
  708. * @param {boolean} musicOnly - whether to only add music to the playlist
  709. * @param {Function} cb - gets called with the result
  710. */
  711. addSetToPlaylist: isLoginRequired(function addSetToPlaylist(session, url, playlistId, musicOnly, cb) {
  712. let videosInPlaylistTotal = 0;
  713. let songsInPlaylistTotal = 0;
  714. let addSongsStats = null;
  715. const addedSongs = [];
  716. async.waterfall(
  717. [
  718. next => {
  719. YouTubeModule.runJob(
  720. "GET_PLAYLIST",
  721. {
  722. url,
  723. musicOnly
  724. },
  725. this
  726. ).then(response => {
  727. if (response.filteredSongs) {
  728. videosInPlaylistTotal = response.songs.length;
  729. songsInPlaylistTotal = response.filteredSongs.length;
  730. } else {
  731. songsInPlaylistTotal = videosInPlaylistTotal = response.songs.length;
  732. }
  733. next(null, response.songs);
  734. });
  735. },
  736. (songIds, next) => {
  737. let successful = 0;
  738. let failed = 0;
  739. let alreadyInPlaylist = 0;
  740. if (songIds.length === 0) next();
  741. async.eachLimit(
  742. songIds,
  743. 1,
  744. (songId, next) => {
  745. IOModule.runJob(
  746. "RUN_ACTION2",
  747. {
  748. session,
  749. namespace: "playlists",
  750. action: "addSongToPlaylist",
  751. args: [true, songId, playlistId]
  752. },
  753. this
  754. )
  755. .then(res => {
  756. if (res.status === "success") {
  757. successful += 1;
  758. addedSongs.push(songId);
  759. } else failed += 1;
  760. if (res.message === "That song is already in the playlist") alreadyInPlaylist += 1;
  761. })
  762. .catch(() => {
  763. failed += 1;
  764. })
  765. .finally(() => {
  766. next();
  767. });
  768. },
  769. () => {
  770. addSongsStats = { successful, failed, alreadyInPlaylist };
  771. next(null);
  772. }
  773. );
  774. },
  775. next => {
  776. PlaylistsModule.runJob("GET_PLAYLIST", { playlistId }, this)
  777. .then(playlist => {
  778. next(null, playlist);
  779. })
  780. .catch(next);
  781. },
  782. (playlist, next) => {
  783. if (!playlist || playlist.createdBy !== session.userId) return next("Playlist not found.");
  784. if (!playlist.isUserModifiable) return next("Playlist cannot be modified.");
  785. return next(null, playlist);
  786. }
  787. ],
  788. async (err, playlist) => {
  789. if (err) {
  790. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  791. this.log(
  792. "ERROR",
  793. "PLAYLIST_IMPORT",
  794. `Importing a YouTube playlist to private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`
  795. );
  796. return cb({ status: "failure", message: err });
  797. }
  798. ActivitiesModule.runJob("ADD_ACTIVITY", {
  799. userId: session.userId,
  800. activityType: "added_songs_to_playlist",
  801. payload: addedSongs
  802. });
  803. this.log(
  804. "SUCCESS",
  805. "PLAYLIST_IMPORT",
  806. `Successfully imported a YouTube playlist to private playlist "${playlistId}" for user "${session.userId}". Videos in playlist: ${videosInPlaylistTotal}, songs in playlist: ${songsInPlaylistTotal}, songs successfully added: ${addSongsStats.successful}, songs failed: ${addSongsStats.failed}, already in playlist: ${addSongsStats.alreadyInPlaylist}.`
  807. );
  808. return cb({
  809. status: "success",
  810. message: `Playlist has been imported. ${addSongsStats.successful} were added successfully, ${addSongsStats.failed} failed (${addSongsStats.alreadyInPlaylist} were already in the playlist)`,
  811. data: playlist.songs,
  812. stats: {
  813. videosInPlaylistTotal,
  814. songsInPlaylistTotal
  815. }
  816. });
  817. }
  818. );
  819. }),
  820. /**
  821. * Removes a song from a private playlist
  822. *
  823. * @param {object} session - the session object automatically added by socket.io
  824. * @param {string} songId - the id of the song we are removing from the private playlist
  825. * @param {string} playlistId - the id of the playlist we are removing the song from
  826. * @param {Function} cb - gets called with the result
  827. */
  828. removeSongFromPlaylist: isLoginRequired(async function removeSongFromPlaylist(session, songId, playlistId, cb) {
  829. const playlistModel = await DBModule.runJob(
  830. "GET_MODEL",
  831. {
  832. modelName: "playlist"
  833. },
  834. this
  835. );
  836. async.waterfall(
  837. [
  838. next => {
  839. if (!songId || typeof songId !== "string") return next("Invalid song id.");
  840. if (!playlistId) return next("Invalid playlist id.");
  841. return next();
  842. },
  843. next => {
  844. PlaylistsModule.runJob("GET_PLAYLIST", { playlistId }, this)
  845. .then(playlist => {
  846. next(null, playlist);
  847. })
  848. .catch(next);
  849. },
  850. (playlist, next) => {
  851. if (!playlist || playlist.createdBy !== session.userId) return next("Playlist not found");
  852. return playlistModel.updateOne({ _id: playlistId }, { $pull: { songs: { songId } } }, next);
  853. },
  854. (res, next) => {
  855. PlaylistsModule.runJob("UPDATE_PLAYLIST", { playlistId }, this)
  856. .then(playlist => {
  857. next(null, playlist);
  858. })
  859. .catch(next);
  860. }
  861. ],
  862. async (err, playlist) => {
  863. if (err) {
  864. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  865. this.log(
  866. "ERROR",
  867. "PLAYLIST_REMOVE_SONG",
  868. `Removing song "${songId}" from private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`
  869. );
  870. return cb({ status: "failure", message: err });
  871. }
  872. this.log(
  873. "SUCCESS",
  874. "PLAYLIST_REMOVE_SONG",
  875. `Successfully removed song "${songId}" from private playlist "${playlistId}" for user "${session.userId}".`
  876. );
  877. CacheModule.runJob("PUB", {
  878. channel: "playlist.removeSong",
  879. value: {
  880. playlistId: playlist._id,
  881. songId,
  882. userId: session.userId,
  883. privacy: playlist.privacy
  884. }
  885. });
  886. return cb({
  887. status: "success",
  888. message: "Song has been successfully removed from playlist",
  889. data: playlist.songs
  890. });
  891. }
  892. );
  893. }),
  894. /**
  895. * Updates the displayName of a private playlist
  896. *
  897. * @param {object} session - the session object automatically added by socket.io
  898. * @param {string} playlistId - the id of the playlist we are updating the displayName for
  899. * @param {Function} cb - gets called with the result
  900. */
  901. updateDisplayName: isLoginRequired(async function updateDisplayName(session, playlistId, displayName, cb) {
  902. const playlistModel = await DBModule.runJob(
  903. "GET_MODEL",
  904. {
  905. modelName: "playlist"
  906. },
  907. this
  908. );
  909. async.waterfall(
  910. [
  911. next => {
  912. PlaylistsModule.runJob("GET_PLAYLIST", { playlistId }, this)
  913. .then(playlist => next(null, playlist))
  914. .catch(next);
  915. },
  916. (playlist, next) => {
  917. if (!playlist.isUserModifiable) return next("Playlist cannot be modified.");
  918. return next(null);
  919. },
  920. next => {
  921. playlistModel.updateOne(
  922. { _id: playlistId, createdBy: session.userId },
  923. { $set: { displayName } },
  924. { runValidators: true },
  925. next
  926. );
  927. },
  928. (res, next) => {
  929. PlaylistsModule.runJob("UPDATE_PLAYLIST", { playlistId }, this)
  930. .then(playlist => {
  931. next(null, playlist);
  932. })
  933. .catch(next);
  934. }
  935. ],
  936. async (err, playlist) => {
  937. if (err) {
  938. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  939. this.log(
  940. "ERROR",
  941. "PLAYLIST_UPDATE_DISPLAY_NAME",
  942. `Updating display name to "${displayName}" for private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`
  943. );
  944. return cb({ status: "failure", message: err });
  945. }
  946. this.log(
  947. "SUCCESS",
  948. "PLAYLIST_UPDATE_DISPLAY_NAME",
  949. `Successfully updated display name to "${displayName}" for private playlist "${playlistId}" for user "${session.userId}".`
  950. );
  951. CacheModule.runJob("PUB", {
  952. channel: "playlist.updateDisplayName",
  953. value: {
  954. playlistId,
  955. displayName,
  956. userId: session.userId,
  957. privacy: playlist.privacy
  958. }
  959. });
  960. return cb({
  961. status: "success",
  962. message: "Playlist has been successfully updated"
  963. });
  964. }
  965. );
  966. }),
  967. /**
  968. * Moves a song to the top of the list in a private playlist
  969. *
  970. * @param {object} session - the session object automatically added by socket.io
  971. * @param {string} playlistId - the id of the playlist we are moving the song to the top from
  972. * @param {string} songId - the id of the song we are moving to the top of the list
  973. * @param {Function} cb - gets called with the result
  974. */
  975. moveSongToTop: isLoginRequired(async function moveSongToTop(session, playlistId, songId, cb) {
  976. const playlistModel = await DBModule.runJob(
  977. "GET_MODEL",
  978. {
  979. modelName: "playlist"
  980. },
  981. this
  982. );
  983. async.waterfall(
  984. [
  985. next => {
  986. PlaylistsModule.runJob("GET_PLAYLIST", { playlistId }, this)
  987. .then(playlist => {
  988. next(null, playlist);
  989. })
  990. .catch(next);
  991. },
  992. (playlist, next) => {
  993. if (!playlist || playlist.createdBy !== session.userId) return next("Playlist not found");
  994. if (!playlist.isUserModifiable) return next("Playlist cannot be modified.");
  995. return async.each(
  996. playlist.songs,
  997. (song, next) => {
  998. if (song.songId === songId) return next(song);
  999. return next();
  1000. },
  1001. err => {
  1002. if (err && err.songId) return next(null, err);
  1003. return next("Song not found");
  1004. }
  1005. );
  1006. },
  1007. (song, next) => {
  1008. playlistModel.updateOne({ _id: playlistId }, { $pull: { songs: { songId } } }, err => {
  1009. if (err) return next(err);
  1010. return next(null, song);
  1011. });
  1012. },
  1013. (song, next) => {
  1014. playlistModel.updateOne(
  1015. { _id: playlistId },
  1016. {
  1017. $push: {
  1018. songs: {
  1019. $each: [song],
  1020. $position: 0
  1021. }
  1022. }
  1023. },
  1024. next
  1025. );
  1026. },
  1027. (res, next) => {
  1028. PlaylistsModule.runJob("UPDATE_PLAYLIST", { playlistId }, this)
  1029. .then(playlist => {
  1030. next(null, playlist);
  1031. })
  1032. .catch(next);
  1033. }
  1034. ],
  1035. async err => {
  1036. if (err) {
  1037. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  1038. this.log(
  1039. "ERROR",
  1040. "PLAYLIST_MOVE_SONG_TO_TOP",
  1041. `Moving song "${songId}" to the top for private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`
  1042. );
  1043. return cb({ status: "failure", message: err });
  1044. }
  1045. this.log(
  1046. "SUCCESS",
  1047. "PLAYLIST_MOVE_SONG_TO_TOP",
  1048. `Successfully moved song "${songId}" to the top for private playlist "${playlistId}" for user "${session.userId}".`
  1049. );
  1050. CacheModule.runJob("PUB", {
  1051. channel: "playlist.moveSongToTop",
  1052. value: {
  1053. playlistId,
  1054. songId,
  1055. userId: session.userId
  1056. }
  1057. });
  1058. return cb({
  1059. status: "success",
  1060. message: "Playlist has been successfully updated"
  1061. });
  1062. }
  1063. );
  1064. }),
  1065. /**
  1066. * Moves a song to the bottom of the list in a private playlist
  1067. *
  1068. * @param {object} session - the session object automatically added by socket.io
  1069. * @param {string} playlistId - the id of the playlist we are moving the song to the bottom from
  1070. * @param {string} songId - the id of the song we are moving to the bottom of the list
  1071. * @param {Function} cb - gets called with the result
  1072. */
  1073. moveSongToBottom: isLoginRequired(async function moveSongToBottom(session, playlistId, songId, cb) {
  1074. const playlistModel = await DBModule.runJob(
  1075. "GET_MODEL",
  1076. {
  1077. modelName: "playlist"
  1078. },
  1079. this
  1080. );
  1081. async.waterfall(
  1082. [
  1083. next => {
  1084. PlaylistsModule.runJob("GET_PLAYLIST", { playlistId }, this)
  1085. .then(playlist => {
  1086. next(null, playlist);
  1087. })
  1088. .catch(next);
  1089. },
  1090. (playlist, next) => {
  1091. if (!playlist || playlist.createdBy !== session.userId) return next("Playlist not found");
  1092. if (!playlist.isUserModifiable) return next("Playlist cannot be modified.");
  1093. return async.each(
  1094. playlist.songs,
  1095. (song, next) => {
  1096. if (song.songId === songId) return next(song);
  1097. return next();
  1098. },
  1099. err => {
  1100. if (err && err.songId) return next(null, err);
  1101. return next("Song not found");
  1102. }
  1103. );
  1104. },
  1105. (song, next) => {
  1106. playlistModel.updateOne({ _id: playlistId }, { $pull: { songs: { songId } } }, err => {
  1107. if (err) return next(err);
  1108. return next(null, song);
  1109. });
  1110. },
  1111. (song, next) => {
  1112. playlistModel.updateOne(
  1113. { _id: playlistId },
  1114. {
  1115. $push: {
  1116. songs: song
  1117. }
  1118. },
  1119. next
  1120. );
  1121. },
  1122. (res, next) => {
  1123. PlaylistsModule.runJob("UPDATE_PLAYLIST", { playlistId }, this)
  1124. .then(playlist => {
  1125. next(null, playlist);
  1126. })
  1127. .catch(next);
  1128. }
  1129. ],
  1130. async err => {
  1131. if (err) {
  1132. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  1133. this.log(
  1134. "ERROR",
  1135. "PLAYLIST_MOVE_SONG_TO_BOTTOM",
  1136. `Moving song "${songId}" to the bottom for private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`
  1137. );
  1138. return cb({ status: "failure", message: err });
  1139. }
  1140. this.log(
  1141. "SUCCESS",
  1142. "PLAYLIST_MOVE_SONG_TO_BOTTOM",
  1143. `Successfully moved song "${songId}" to the bottom for private playlist "${playlistId}" for user "${session.userId}".`
  1144. );
  1145. CacheModule.runJob("PUB", {
  1146. channel: "playlist.moveSongToBottom",
  1147. value: {
  1148. playlistId,
  1149. songId,
  1150. userId: session.userId
  1151. }
  1152. });
  1153. return cb({
  1154. status: "success",
  1155. message: "Playlist has been successfully updated"
  1156. });
  1157. }
  1158. );
  1159. }),
  1160. /**
  1161. * Removes a private playlist
  1162. *
  1163. * @param {object} session - the session object automatically added by socket.io
  1164. * @param {string} playlistId - the id of the playlist we are moving the song to the top from
  1165. * @param {Function} cb - gets called with the result
  1166. */
  1167. remove: isLoginRequired(async function remove(session, playlistId, cb) {
  1168. const stationModel = await DBModule.runJob("GET_MODEL", { modelName: "station" }, this);
  1169. const userModel = await DBModule.runJob("GET_MODEL", { modelName: "user" }, this);
  1170. async.waterfall(
  1171. [
  1172. next => {
  1173. PlaylistsModule.runJob("GET_PLAYLIST", { playlistId }, this)
  1174. .then(playlist => next(null, playlist))
  1175. .catch(next);
  1176. },
  1177. (playlist, next) => {
  1178. if (!playlist.isUserModifiable) return next("Playlist cannot be removed.");
  1179. return next(null, playlist);
  1180. },
  1181. (playlist, next) => {
  1182. userModel.updateOne(
  1183. { _id: playlist.createdBy },
  1184. { $pull: { "preferences.orderOfPlaylists": playlist._id } },
  1185. err => {
  1186. if (err) return next(err);
  1187. return next(null);
  1188. }
  1189. );
  1190. },
  1191. next => {
  1192. PlaylistsModule.runJob("DELETE_PLAYLIST", { playlistId }, this).then(next).catch(next);
  1193. },
  1194. next => {
  1195. stationModel.find({ privatePlaylist: playlistId }, (err, res) => {
  1196. next(err, res);
  1197. });
  1198. },
  1199. (stations, next) => {
  1200. async.each(
  1201. stations,
  1202. (station, next) => {
  1203. async.waterfall(
  1204. [
  1205. next => {
  1206. stationModel.updateOne(
  1207. { _id: station._id },
  1208. { $set: { privatePlaylist: null } },
  1209. { runValidators: true },
  1210. next
  1211. );
  1212. },
  1213. (res, next) => {
  1214. if (!station.partyMode) {
  1215. moduleManager.modules.stations
  1216. .runJob(
  1217. "UPDATE_STATION",
  1218. {
  1219. stationId: station._id
  1220. },
  1221. this
  1222. )
  1223. .then(station => next(null, station))
  1224. .catch(next);
  1225. CacheModule.runJob("PUB", {
  1226. channel: "privatePlaylist.selected",
  1227. value: {
  1228. playlistId: null,
  1229. stationId: station._id
  1230. }
  1231. });
  1232. } else next();
  1233. }
  1234. ],
  1235. () => {
  1236. next();
  1237. }
  1238. );
  1239. },
  1240. () => {
  1241. next();
  1242. }
  1243. );
  1244. }
  1245. ],
  1246. async err => {
  1247. if (err) {
  1248. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  1249. this.log(
  1250. "ERROR",
  1251. "PLAYLIST_REMOVE",
  1252. `Removing private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`
  1253. );
  1254. return cb({ status: "failure", message: err });
  1255. }
  1256. this.log(
  1257. "SUCCESS",
  1258. "PLAYLIST_REMOVE",
  1259. `Successfully removed private playlist "${playlistId}" for user "${session.userId}".`
  1260. );
  1261. CacheModule.runJob("PUB", {
  1262. channel: "playlist.delete",
  1263. value: {
  1264. userId: session.userId,
  1265. playlistId
  1266. }
  1267. });
  1268. ActivitiesModule.runJob("ADD_ACTIVITY", {
  1269. userId: session.userId,
  1270. activityType: "deleted_playlist",
  1271. payload: [playlistId]
  1272. });
  1273. return cb({
  1274. status: "success",
  1275. message: "Playlist successfully removed"
  1276. });
  1277. }
  1278. );
  1279. }),
  1280. /**
  1281. * Updates the privacy of a private playlist
  1282. *
  1283. * @param {object} session - the session object automatically added by socket.io
  1284. * @param {string} playlistId - the id of the playlist we are updating the privacy for
  1285. * @param {string} privacy - what the new privacy of the playlist should be e.g. public
  1286. * @param {Function} cb - gets called with the result
  1287. */
  1288. updatePrivacy: isLoginRequired(async function updatePrivacy(session, playlistId, privacy, cb) {
  1289. const playlistModel = await DBModule.runJob(
  1290. "GET_MODEL",
  1291. {
  1292. modelName: "playlist"
  1293. },
  1294. this
  1295. );
  1296. async.waterfall(
  1297. [
  1298. next => {
  1299. playlistModel.updateOne(
  1300. { _id: playlistId, createdBy: session.userId },
  1301. { $set: { privacy } },
  1302. { runValidators: true },
  1303. next
  1304. );
  1305. },
  1306. (res, next) => {
  1307. PlaylistsModule.runJob("UPDATE_PLAYLIST", { playlistId }, this)
  1308. .then(playlist => {
  1309. next(null, playlist);
  1310. })
  1311. .catch(next);
  1312. }
  1313. ],
  1314. async (err, playlist) => {
  1315. if (err) {
  1316. err = await UtilsModule.runJob("GET_ERROR", { error: err }, this);
  1317. this.log(
  1318. "ERROR",
  1319. "PLAYLIST_UPDATE_PRIVACY",
  1320. `Updating privacy to "${privacy}" for private playlist "${playlistId}" failed for user "${session.userId}". "${err}"`
  1321. );
  1322. return cb({ status: "failure", message: err });
  1323. }
  1324. this.log(
  1325. "SUCCESS",
  1326. "PLAYLIST_UPDATE_PRIVACY",
  1327. `Successfully updated privacy to "${privacy}" for private playlist "${playlistId}" for user "${session.userId}".`
  1328. );
  1329. CacheModule.runJob("PUB", {
  1330. channel: "playlist.updatePrivacy",
  1331. value: {
  1332. userId: session.userId,
  1333. playlist
  1334. }
  1335. });
  1336. return cb({
  1337. status: "success",
  1338. message: "Playlist has been successfully updated"
  1339. });
  1340. }
  1341. );
  1342. })
  1343. };