songs.js 36 KB

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