songs.js 38 KB

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