songs.js 36 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477
  1. import async from "async";
  2. import mongoose from "mongoose";
  3. import CoreClass from "../core";
  4. let SongsModule;
  5. let CacheModule;
  6. let DBModule;
  7. let UtilsModule;
  8. let YouTubeModule;
  9. let StationsModule;
  10. let PlaylistsModule;
  11. class ErrorWithData extends Error {
  12. /**
  13. * @param {string} message - the error message
  14. * @param {object} data - the error data
  15. */
  16. constructor(message, data) {
  17. super(message);
  18. this.data = data;
  19. }
  20. }
  21. class _SongsModule extends CoreClass {
  22. // eslint-disable-next-line require-jsdoc
  23. constructor() {
  24. super("songs");
  25. SongsModule = this;
  26. }
  27. /**
  28. * Initialises the songs module
  29. *
  30. * @returns {Promise} - returns promise (reject, resolve)
  31. */
  32. async initialize() {
  33. this.setStage(1);
  34. CacheModule = this.moduleManager.modules.cache;
  35. DBModule = this.moduleManager.modules.db;
  36. UtilsModule = this.moduleManager.modules.utils;
  37. YouTubeModule = this.moduleManager.modules.youtube;
  38. StationsModule = this.moduleManager.modules.stations;
  39. PlaylistsModule = this.moduleManager.modules.playlists;
  40. this.SongModel = await DBModule.runJob("GET_MODEL", { modelName: "song" });
  41. this.SongSchemaCache = await CacheModule.runJob("GET_SCHEMA", { schemaName: "song" });
  42. this.setStage(2);
  43. return new Promise((resolve, reject) => {
  44. async.waterfall(
  45. [
  46. next => {
  47. this.setStage(2);
  48. CacheModule.runJob("HGETALL", { table: "songs" })
  49. .then(songs => {
  50. next(null, songs);
  51. })
  52. .catch(next);
  53. },
  54. (songs, next) => {
  55. this.setStage(3);
  56. if (!songs) return next();
  57. const youtubeIds = Object.keys(songs);
  58. return async.each(
  59. youtubeIds,
  60. (youtubeId, next) => {
  61. SongsModule.SongModel.findOne({ youtubeId }, (err, song) => {
  62. if (err) next(err);
  63. else if (!song)
  64. CacheModule.runJob("HDEL", {
  65. table: "songs",
  66. key: youtubeId
  67. })
  68. .then(() => next())
  69. .catch(next);
  70. else next();
  71. });
  72. },
  73. next
  74. );
  75. },
  76. next => {
  77. this.setStage(4);
  78. SongsModule.SongModel.find({}, next);
  79. },
  80. (songs, next) => {
  81. this.setStage(5);
  82. async.each(
  83. songs,
  84. (song, next) => {
  85. CacheModule.runJob("HSET", {
  86. table: "songs",
  87. key: song.youtubeId,
  88. value: SongsModule.SongSchemaCache(song)
  89. })
  90. .then(() => next())
  91. .catch(next);
  92. },
  93. next
  94. );
  95. }
  96. ],
  97. async err => {
  98. if (err) {
  99. err = await UtilsModule.runJob("GET_ERROR", { error: err });
  100. reject(new Error(err));
  101. } else resolve();
  102. }
  103. );
  104. });
  105. }
  106. /**
  107. * Gets a song by id from the cache or Mongo, and if it isn't in the cache yet, adds it the cache
  108. *
  109. * @param {object} payload - object containing the payload
  110. * @param {string} payload.songId - the id of the song we are trying to get
  111. * @returns {Promise} - returns a promise (resolve, reject)
  112. */
  113. GET_SONG(payload) {
  114. return new Promise((resolve, reject) => {
  115. async.waterfall(
  116. [
  117. next => {
  118. if (!mongoose.Types.ObjectId.isValid(payload.songId))
  119. return next("songId is not a valid ObjectId.");
  120. return CacheModule.runJob("HGET", { table: "songs", key: payload.songId }, this)
  121. .then(song => next(null, song))
  122. .catch(next);
  123. },
  124. (song, next) => {
  125. if (song) return next(true, song);
  126. return SongsModule.SongModel.findOne({ _id: payload.songId }, next);
  127. },
  128. (song, next) => {
  129. if (song) {
  130. CacheModule.runJob(
  131. "HSET",
  132. {
  133. table: "songs",
  134. key: payload.songId,
  135. value: song
  136. },
  137. this
  138. ).then(song => next(null, song));
  139. } else next("Song not found.");
  140. }
  141. ],
  142. (err, song) => {
  143. if (err && err !== true) return reject(new Error(err));
  144. return resolve({ song });
  145. }
  146. );
  147. });
  148. }
  149. /**
  150. * Gets songs by id from Mongo
  151. *
  152. * @param {object} payload - object containing the payload
  153. * @param {string} payload.songIds - the ids of the songs we are trying to get
  154. * @param {string} payload.properties - the properties to return
  155. * @returns {Promise} - returns a promise (resolve, reject)
  156. */
  157. GET_SONGS(payload) {
  158. return new Promise((resolve, reject) => {
  159. async.waterfall(
  160. [
  161. next => {
  162. if (!payload.songIds.every(songId => mongoose.Types.ObjectId.isValid(songId)))
  163. next("One or more songIds are not a valid ObjectId.");
  164. else next();
  165. },
  166. next => {
  167. const includeProperties = {};
  168. payload.properties.forEach(property => {
  169. includeProperties[property] = true;
  170. });
  171. return SongsModule.SongModel.find(
  172. {
  173. _id: { $in: payload.songIds }
  174. },
  175. includeProperties,
  176. next
  177. );
  178. }
  179. ],
  180. (err, songs) => {
  181. if (err && err !== true) return reject(new Error(err));
  182. return resolve({ songs });
  183. }
  184. );
  185. });
  186. }
  187. /**
  188. * Makes sure that if a song is not currently in the songs db, to add it
  189. *
  190. * @param {object} payload - an object containing the payload
  191. * @param {string} payload.youtubeId - the youtube song id of the song we are trying to ensure is in the songs db
  192. * @param {string} payload.userId - the youtube song id of the song we are trying to ensure is in the songs db
  193. * @param {string} payload.automaticallyRequested - whether the song was automatically requested or not
  194. * @returns {Promise} - returns a promise (resolve, reject)
  195. */
  196. ENSURE_SONG_EXISTS_BY_YOUTUBE_ID(payload) {
  197. return new Promise((resolve, reject) => {
  198. async.waterfall(
  199. [
  200. next => {
  201. SongsModule.SongModel.findOne({ youtubeId: payload.youtubeId }, next);
  202. },
  203. (song, next) => {
  204. if (song && song.duration > 0) next(true, song);
  205. else {
  206. YouTubeModule.runJob("GET_SONG", { youtubeId: payload.youtubeId }, this)
  207. .then(response => {
  208. next(null, song, response.song);
  209. })
  210. .catch(next);
  211. }
  212. },
  213. (song, youtubeSong, next) => {
  214. if (song && song.duration <= 0) {
  215. song.duration = youtubeSong.duration;
  216. song.save({ validateBeforeSave: true }, err => {
  217. if (err) return next(err, song);
  218. return next(null, song);
  219. });
  220. } else {
  221. const song = new SongsModule.SongModel({
  222. ...youtubeSong,
  223. requestedBy: payload.userId,
  224. requestedAt: Date.now()
  225. });
  226. song.save({ validateBeforeSave: true }, err => {
  227. if (err) return next(err, song);
  228. return next(null, song);
  229. });
  230. }
  231. }
  232. ],
  233. (err, song) => {
  234. if (err && err !== true) return reject(new Error(err));
  235. return resolve({ song });
  236. }
  237. );
  238. });
  239. }
  240. /**
  241. * Gets a song by youtube id
  242. *
  243. * @param {object} payload - an object containing the payload
  244. * @param {string} payload.youtubeId - the youtube id of the song we are trying to get
  245. * @returns {Promise} - returns a promise (resolve, reject)
  246. */
  247. GET_SONG_FROM_YOUTUBE_ID(payload) {
  248. return new Promise((resolve, reject) => {
  249. async.waterfall(
  250. [
  251. next => {
  252. SongsModule.SongModel.findOne({ youtubeId: payload.youtubeId }, next);
  253. }
  254. ],
  255. (err, song) => {
  256. if (err && err !== true) return reject(new Error(err));
  257. return resolve({ song });
  258. }
  259. );
  260. });
  261. }
  262. /**
  263. * Create song
  264. *
  265. * @param {object} payload - an object containing the payload
  266. * @param {string} payload.song - the song object
  267. * @returns {Promise} - returns a promise (resolve, reject)
  268. */
  269. CREATE_SONG(payload) {
  270. return new Promise((resolve, reject) => {
  271. async.waterfall(
  272. [
  273. next => {
  274. const song = new SongsModule.SongModel(payload.song);
  275. song.save({ validateBeforeSave: true }, err => {
  276. if (err) return next(err, song);
  277. return next(null, song);
  278. });
  279. },
  280. (song, next) => {
  281. SongsModule.runJob("UPDATE_SONG", { songId: song._id });
  282. return next(null, song);
  283. }
  284. ],
  285. (err, song) => {
  286. if (err && err !== true) return reject(new Error(err));
  287. return resolve({ song });
  288. }
  289. );
  290. });
  291. }
  292. /**
  293. * Gets a song from id from Mongo and updates the cache with it
  294. *
  295. * @param {object} payload - an object containing the payload
  296. * @param {string} payload.songId - the id of the song we are trying to update
  297. * @param {string} payload.oldStatus - old status of song being updated (optional)
  298. * @returns {Promise} - returns a promise (resolve, reject)
  299. */
  300. UPDATE_SONG(payload) {
  301. return new Promise((resolve, reject) => {
  302. async.waterfall(
  303. [
  304. next => {
  305. SongsModule.SongModel.findOne({ _id: payload.songId }, next);
  306. },
  307. (song, next) => {
  308. if (!song) {
  309. CacheModule.runJob("HDEL", {
  310. table: "songs",
  311. key: payload.songId
  312. });
  313. return next("Song not found.");
  314. }
  315. return CacheModule.runJob(
  316. "HSET",
  317. {
  318. table: "songs",
  319. key: payload.songId,
  320. value: song
  321. },
  322. this
  323. )
  324. .then(song => {
  325. next(null, song);
  326. })
  327. .catch(next);
  328. },
  329. (song, next) => {
  330. const { _id, youtubeId, title, artists, thumbnail, duration, verified } = song;
  331. const trimmedSong = {
  332. _id,
  333. youtubeId,
  334. title,
  335. artists,
  336. thumbnail,
  337. duration,
  338. verified
  339. };
  340. this.log("INFO", `Going to update playlists now for song ${_id}`);
  341. DBModule.runJob("GET_MODEL", { modelName: "playlist" }, this)
  342. .then(playlistModel => {
  343. playlistModel.updateMany(
  344. { "songs._id": song._id },
  345. { $set: { "songs.$": trimmedSong } },
  346. err => {
  347. if (err) next(err);
  348. else
  349. playlistModel.find({ "songs._id": song._id }, (err, playlists) => {
  350. if (err) next(err);
  351. else {
  352. async.eachLimit(
  353. playlists,
  354. 1,
  355. (playlist, next) => {
  356. PlaylistsModule.runJob(
  357. "UPDATE_PLAYLIST",
  358. {
  359. playlistId: playlist._id
  360. },
  361. this
  362. )
  363. .then(() => {
  364. next();
  365. })
  366. .catch(err => {
  367. next(err);
  368. });
  369. },
  370. err => {
  371. if (err) next(err);
  372. else next(null, song);
  373. }
  374. );
  375. }
  376. });
  377. }
  378. );
  379. })
  380. .catch(err => {
  381. next(err);
  382. });
  383. },
  384. (song, next) => {
  385. const { _id, youtubeId, title, artists, thumbnail, duration, verified } = song;
  386. this.log("INFO", `Going to update stations now for song ${_id}`);
  387. DBModule.runJob("GET_MODEL", { modelName: "station" }, this)
  388. .then(stationModel => {
  389. stationModel.updateMany(
  390. { "queue._id": song._id },
  391. {
  392. $set: {
  393. "queue.$.youtubeId": youtubeId,
  394. "queue.$.title": title,
  395. "queue.$.artists": artists,
  396. "queue.$.thumbnail": thumbnail,
  397. "queue.$.duration": duration,
  398. "queue.$.verified": verified
  399. }
  400. },
  401. err => {
  402. if (err) this.log("ERROR", err);
  403. else
  404. stationModel.find({ "queue._id": song._id }, (err, stations) => {
  405. if (err) next(err);
  406. else {
  407. async.eachLimit(
  408. stations,
  409. 1,
  410. (station, next) => {
  411. StationsModule.runJob(
  412. "UPDATE_STATION",
  413. { stationId: station._id },
  414. this
  415. )
  416. .then(() => {
  417. next();
  418. })
  419. .catch(err => {
  420. next(err);
  421. });
  422. },
  423. err => {
  424. if (err) next(err);
  425. else next(null, song);
  426. }
  427. );
  428. }
  429. });
  430. }
  431. );
  432. })
  433. .catch(err => {
  434. next(err);
  435. });
  436. },
  437. (song, next) => {
  438. async.eachLimit(
  439. song.genres,
  440. 1,
  441. (genre, next) => {
  442. PlaylistsModule.runJob("AUTOFILL_GENRE_PLAYLIST", { genre }, this)
  443. .then(() => {
  444. next();
  445. })
  446. .catch(err => next(err));
  447. },
  448. err => {
  449. next(err, song);
  450. }
  451. );
  452. }
  453. ],
  454. (err, song) => {
  455. if (err && err !== true) return reject(new Error(err));
  456. if (!payload.oldStatus) payload.oldStatus = null;
  457. CacheModule.runJob("PUB", {
  458. channel: "song.updated",
  459. value: { songId: song._id, oldStatus: payload.oldStatus }
  460. });
  461. return resolve(song);
  462. }
  463. );
  464. });
  465. }
  466. /**
  467. * Gets multiple songs from id from Mongo and updates the cache with it
  468. *
  469. * @param {object} payload - an object containing the payload
  470. * @param {Array} payload.songIds - the ids of the songs we are trying to update
  471. * @param {string} payload.oldStatus - old status of song being updated (optional)
  472. * @returns {Promise} - returns a promise (resolve, reject)
  473. */
  474. async UPDATE_SONGS(payload) {
  475. const playlistModel = await DBModule.runJob("GET_MODEL", { modelName: "playlist" }, this);
  476. const stationModel = await DBModule.runJob("GET_MODEL", { modelName: "station" }, this);
  477. return new Promise((resolve, reject) => {
  478. async.waterfall(
  479. [
  480. // Get songs from Mongo
  481. next => {
  482. const { songIds } = payload;
  483. SongsModule.SongModel.find({ _id: songIds }, next);
  484. },
  485. // Any songs that were not in Mongo, remove from cache, if they're in the cache
  486. (songs, next) => {
  487. const { songIds } = payload;
  488. async.eachLimit(
  489. songIds,
  490. 1,
  491. (songId, next) => {
  492. if (songs.findIndex(song => song._id.toString() === songId) === -1) {
  493. // NOTE: could be made lower priority
  494. CacheModule.runJob("HDEL", {
  495. table: "songs",
  496. key: songId
  497. });
  498. next();
  499. } else next();
  500. },
  501. () => {
  502. next(null, songs);
  503. }
  504. );
  505. },
  506. // Adds/updates all songs in the cache
  507. (songs, next) => {
  508. async.eachLimit(
  509. songs,
  510. 1,
  511. (song, next) => {
  512. CacheModule.runJob(
  513. "HSET",
  514. {
  515. table: "songs",
  516. key: song._id,
  517. value: song
  518. },
  519. this
  520. )
  521. .then(() => {
  522. next();
  523. })
  524. .catch(next);
  525. },
  526. () => {
  527. next(null, songs);
  528. }
  529. );
  530. },
  531. // Updates all playlists that the songs are in by setting the new trimmed song
  532. (songs, next) => {
  533. const trimmedSongs = songs.map(song => {
  534. const { _id, youtubeId, title, artists, thumbnail, duration, verified } = song;
  535. return {
  536. _id,
  537. youtubeId,
  538. title,
  539. artists,
  540. thumbnail,
  541. duration,
  542. verified
  543. };
  544. });
  545. const playlistsToUpdate = new Set();
  546. async.eachLimit(
  547. trimmedSongs,
  548. 1,
  549. (trimmedSong, next) => {
  550. async.waterfall(
  551. [
  552. next => {
  553. playlistModel.updateMany(
  554. { "songs._id": trimmedSong._id },
  555. { $set: { "songs.$": trimmedSong } },
  556. next
  557. );
  558. },
  559. (res, next) => {
  560. playlistModel.find({ "songs._id": trimmedSong._id }, next);
  561. },
  562. (playlists, next) => {
  563. playlists.forEach(playlist => {
  564. playlistsToUpdate.add(playlist._id.toString());
  565. });
  566. next();
  567. }
  568. ],
  569. next
  570. );
  571. },
  572. err => {
  573. next(err, songs, playlistsToUpdate);
  574. }
  575. );
  576. },
  577. // Updates all playlists that the songs are in
  578. (songs, playlistsToUpdate, next) => {
  579. async.eachLimit(
  580. playlistsToUpdate,
  581. 1,
  582. (playlistId, next) => {
  583. PlaylistsModule.runJob(
  584. "UPDATE_PLAYLIST",
  585. {
  586. playlistId
  587. },
  588. this
  589. )
  590. .then(() => {
  591. next();
  592. })
  593. .catch(err => {
  594. next(err);
  595. });
  596. },
  597. err => {
  598. next(err, songs);
  599. }
  600. );
  601. },
  602. // Updates all station queues that the songs are in by setting the new trimmed song
  603. (songs, next) => {
  604. const stationsToUpdate = new Set();
  605. async.eachLimit(
  606. songs,
  607. 1,
  608. (song, next) => {
  609. async.waterfall(
  610. [
  611. next => {
  612. const { youtubeId, title, artists, thumbnail, duration, verified } = song;
  613. stationModel.updateMany(
  614. { "queue._id": song._id },
  615. {
  616. $set: {
  617. "queue.$.youtubeId": youtubeId,
  618. "queue.$.title": title,
  619. "queue.$.artists": artists,
  620. "queue.$.thumbnail": thumbnail,
  621. "queue.$.duration": duration,
  622. "queue.$.verified": verified
  623. }
  624. },
  625. next
  626. );
  627. },
  628. (res, next) => {
  629. stationModel.find({ "queue._id": song._id }, next);
  630. },
  631. (stations, next) => {
  632. stations.forEach(station => {
  633. stationsToUpdate.add(station._id.toString());
  634. });
  635. next();
  636. }
  637. ],
  638. next
  639. );
  640. },
  641. err => {
  642. next(err, songs, stationsToUpdate);
  643. }
  644. );
  645. },
  646. // Updates all playlists that the songs are in
  647. (songs, stationsToUpdate, next) => {
  648. async.eachLimit(
  649. stationsToUpdate,
  650. 1,
  651. (stationId, next) => {
  652. StationsModule.runJob(
  653. "UPDATE_STATION",
  654. {
  655. stationId
  656. },
  657. this
  658. )
  659. .then(() => {
  660. next();
  661. })
  662. .catch(err => {
  663. next(err);
  664. });
  665. },
  666. err => {
  667. next(err, songs);
  668. }
  669. );
  670. },
  671. // Autofill the genre playlists of all genres of all songs
  672. (songs, next) => {
  673. const genresToAutofill = new Set();
  674. songs.forEach(song => {
  675. song.genres.forEach(genre => {
  676. genresToAutofill.add(genre);
  677. });
  678. });
  679. async.eachLimit(
  680. genresToAutofill,
  681. 1,
  682. (genre, next) => {
  683. PlaylistsModule.runJob("AUTOFILL_GENRE_PLAYLIST", { genre }, this)
  684. .then(() => {
  685. next();
  686. })
  687. .catch(err => next(err));
  688. },
  689. err => {
  690. next(err, songs);
  691. }
  692. );
  693. },
  694. // Send event that the song was updated
  695. (songs, next) => {
  696. async.eachLimit(
  697. songs,
  698. 1,
  699. (song, next) => {
  700. CacheModule.runJob("PUB", {
  701. channel: "song.updated",
  702. value: { songId: song._id, oldStatus: null }
  703. });
  704. next();
  705. },
  706. () => {
  707. next();
  708. }
  709. );
  710. }
  711. ],
  712. err => {
  713. if (err && err !== true) return reject(new Error(err));
  714. return resolve();
  715. }
  716. );
  717. });
  718. }
  719. /**
  720. * Updates all songs
  721. *
  722. * @returns {Promise} - returns a promise (resolve, reject)
  723. */
  724. UPDATE_ALL_SONGS() {
  725. return new Promise((resolve, reject) => {
  726. async.waterfall(
  727. [
  728. next => {
  729. SongsModule.SongModel.find({}, next);
  730. },
  731. (songs, next) => {
  732. let index = 0;
  733. const { length } = songs;
  734. async.eachLimit(
  735. songs,
  736. 2,
  737. (song, next) => {
  738. index += 1;
  739. console.log(`Updating song #${index} out of ${length}: ${song._id}`);
  740. SongsModule.runJob("UPDATE_SONG", { songId: song._id }, this)
  741. .then(() => {
  742. next();
  743. })
  744. .catch(err => {
  745. next(err);
  746. });
  747. },
  748. err => {
  749. next(err);
  750. }
  751. );
  752. }
  753. ],
  754. err => {
  755. if (err && err !== true) return reject(new Error(err));
  756. return resolve();
  757. }
  758. );
  759. });
  760. }
  761. // /**
  762. // * Deletes song from id from Mongo and cache
  763. // *
  764. // * @param {object} payload - returns an object containing the payload
  765. // * @param {string} payload.songId - the song id of the song we are trying to delete
  766. // * @returns {Promise} - returns a promise (resolve, reject)
  767. // */
  768. // DELETE_SONG(payload) {
  769. // return new Promise((resolve, reject) =>
  770. // async.waterfall(
  771. // [
  772. // next => {
  773. // SongsModule.SongModel.deleteOne({ _id: payload.songId }, next);
  774. // },
  775. // next => {
  776. // CacheModule.runJob(
  777. // "HDEL",
  778. // {
  779. // table: "songs",
  780. // key: payload.songId
  781. // },
  782. // this
  783. // )
  784. // .then(() => next())
  785. // .catch(next);
  786. // },
  787. // next => {
  788. // this.log("INFO", `Going to update playlists and stations now for deleted song ${payload.songId}`);
  789. // DBModule.runJob("GET_MODEL", { modelName: "playlist" }).then(playlistModel => {
  790. // playlistModel.find({ "songs._id": song._id }, (err, playlists) => {
  791. // if (err) this.log("ERROR", err);
  792. // else {
  793. // playlistModel.updateMany(
  794. // { "songs._id": payload.songId },
  795. // { $pull: { "songs.$._id": payload.songId} },
  796. // err => {
  797. // if (err) this.log("ERROR", err);
  798. // else {
  799. // playlists.forEach(playlist => {
  800. // PlaylistsModule.runJob("UPDATE_PLAYLIST", {
  801. // playlistId: playlist._id
  802. // });
  803. // });
  804. // }
  805. // }
  806. // );
  807. // }
  808. // });
  809. // });
  810. // DBModule.runJob("GET_MODEL", { modelName: "station" }).then(stationModel => {
  811. // stationModel.find({ "queue._id": payload.songId }, (err, stations) => {
  812. // stationModel.updateMany(
  813. // { "queue._id": payload.songId },
  814. // {
  815. // $pull: { "queue._id": }
  816. // },
  817. // err => {
  818. // if (err) this.log("ERROR", err);
  819. // else {
  820. // stations.forEach(station => {
  821. // StationsModule.runJob("UPDATE_STATION", { stationId: station._id });
  822. // });
  823. // }
  824. // }
  825. // );
  826. // });
  827. // });
  828. // }
  829. // ],
  830. // err => {
  831. // if (err && err !== true) return reject(new Error(err));
  832. // return resolve();
  833. // }
  834. // )
  835. // );
  836. // }
  837. /**
  838. * Searches through songs
  839. *
  840. * @param {object} payload - object that contains the payload
  841. * @param {string} payload.query - the query
  842. * @param {string} payload.includeUnverified - include unverified songs
  843. * @param {string} payload.includeVerified - include verified songs
  844. * @param {string} payload.trimmed - include trimmed songs
  845. * @param {string} payload.page - page (default 1)
  846. * @returns {Promise} - returns promise (reject, resolve)
  847. */
  848. SEARCH(payload) {
  849. return new Promise((resolve, reject) => {
  850. async.waterfall(
  851. [
  852. next => {
  853. const isVerified = [];
  854. if (payload.includeUnverified) isVerified.push(false);
  855. if (payload.includeVerified) isVerified.push(true);
  856. if (isVerified.length === 0) return next("No verified status has been included.");
  857. let { query } = payload;
  858. const isRegex =
  859. query.length > 2 && query.indexOf("/") === 0 && query.lastIndexOf("/") === query.length - 1;
  860. if (isRegex) query = query.slice(1, query.length - 1);
  861. else query = query.replaceAll(/[.*+?^${}()|[\]\\]/g, "\\$&");
  862. const filterArray = [
  863. {
  864. title: new RegExp(`${query}`, "i"),
  865. verified: { $in: isVerified }
  866. },
  867. {
  868. artists: new RegExp(`${query}`, "i"),
  869. verified: { $in: isVerified }
  870. }
  871. ];
  872. return next(null, filterArray);
  873. },
  874. (filterArray, next) => {
  875. const page = payload.page ? payload.page : 1;
  876. const pageSize = 15;
  877. const skipAmount = pageSize * (page - 1);
  878. SongsModule.SongModel.find({ $or: filterArray }).count((err, count) => {
  879. if (err) next(err);
  880. else {
  881. SongsModule.SongModel.find({ $or: filterArray })
  882. .skip(skipAmount)
  883. .limit(pageSize)
  884. .exec((err, songs) => {
  885. if (err) next(err);
  886. else {
  887. next(null, {
  888. songs,
  889. page,
  890. pageSize,
  891. skipAmount,
  892. count
  893. });
  894. }
  895. });
  896. }
  897. });
  898. },
  899. (data, next) => {
  900. if (data.songs.length === 0) next("No songs found");
  901. else if (payload.trimmed) {
  902. next(null, {
  903. songs: data.songs.map(song => {
  904. const { _id, youtubeId, title, artists, thumbnail, duration, verified } = song;
  905. return {
  906. _id,
  907. youtubeId,
  908. title,
  909. artists,
  910. thumbnail,
  911. duration,
  912. verified
  913. };
  914. }),
  915. ...data
  916. });
  917. } else next(null, data);
  918. }
  919. ],
  920. (err, data) => {
  921. if (err && err !== true) return reject(new Error(err));
  922. return resolve(data);
  923. }
  924. );
  925. });
  926. }
  927. /**
  928. * Recalculates dislikes and likes for a song
  929. *
  930. * @param {object} payload - returns an object containing the payload
  931. * @param {string} payload.youtubeId - the youtube id of the song
  932. * @param {string} payload.songId - the song id of the song
  933. * @returns {Promise} - returns a promise (resolve, reject)
  934. */
  935. async RECALCULATE_SONG_RATINGS(payload) {
  936. const playlistModel = await DBModule.runJob("GET_MODEL", { modelName: "playlist" }, this);
  937. return new Promise((resolve, reject) => {
  938. async.waterfall(
  939. [
  940. next => {
  941. playlistModel.countDocuments(
  942. { songs: { $elemMatch: { _id: payload.songId } }, type: "user-liked" },
  943. (err, likes) => {
  944. if (err) return next(err);
  945. return next(null, likes);
  946. }
  947. );
  948. },
  949. (likes, next) => {
  950. playlistModel.countDocuments(
  951. { songs: { $elemMatch: { _id: payload.songId } }, type: "user-disliked" },
  952. (err, dislikes) => {
  953. if (err) return next(err);
  954. return next(err, { likes, dislikes });
  955. }
  956. );
  957. },
  958. ({ likes, dislikes }, next) => {
  959. SongsModule.SongModel.updateOne(
  960. { _id: payload.songId },
  961. {
  962. $set: {
  963. likes,
  964. dislikes
  965. }
  966. },
  967. err => next(err, { likes, dislikes })
  968. );
  969. }
  970. ],
  971. (err, { likes, dislikes }) => {
  972. if (err) return reject(new Error(err));
  973. return resolve({ likes, dislikes });
  974. }
  975. );
  976. });
  977. }
  978. /**
  979. * Recalculates dislikes and likes for all songs
  980. *
  981. * @returns {Promise} - returns a promise (resolve, reject)
  982. */
  983. RECALCULATE_ALL_SONG_RATINGS() {
  984. return new Promise((resolve, reject) => {
  985. async.waterfall(
  986. [
  987. next => {
  988. SongsModule.SongModel.find({}, { _id: true }, next);
  989. },
  990. (songs, next) => {
  991. async.eachLimit(
  992. songs,
  993. 2,
  994. (song, next) => {
  995. SongsModule.runJob("RECALCULATE_SONG_RATINGS", { songId: song._id }, this)
  996. .then(() => {
  997. next();
  998. })
  999. .catch(err => {
  1000. next(err);
  1001. });
  1002. },
  1003. err => {
  1004. next(err);
  1005. }
  1006. );
  1007. }
  1008. ],
  1009. err => {
  1010. if (err) return reject(new Error(err));
  1011. return resolve();
  1012. }
  1013. );
  1014. });
  1015. }
  1016. /**
  1017. * Gets an array of all genres
  1018. *
  1019. * @returns {Promise} - returns a promise (resolve, reject)
  1020. */
  1021. GET_ALL_GENRES() {
  1022. return new Promise((resolve, reject) => {
  1023. async.waterfall(
  1024. [
  1025. next => {
  1026. SongsModule.SongModel.find({ verified: true }, { genres: 1, _id: false }, next);
  1027. },
  1028. (songs, next) => {
  1029. let allGenres = [];
  1030. songs.forEach(song => {
  1031. allGenres = allGenres.concat(song.genres);
  1032. });
  1033. const lowerCaseGenres = allGenres.map(genre => genre.toLowerCase());
  1034. const uniqueGenres = lowerCaseGenres.filter(
  1035. (value, index, self) => self.indexOf(value) === index
  1036. );
  1037. next(null, uniqueGenres);
  1038. }
  1039. ],
  1040. (err, genres) => {
  1041. if (err && err !== true) return reject(new Error(err));
  1042. return resolve({ genres });
  1043. }
  1044. );
  1045. });
  1046. }
  1047. /**
  1048. * Gets an array of all songs with a specific genre
  1049. *
  1050. * @param {object} payload - returns an object containing the payload
  1051. * @param {string} payload.genre - the genre
  1052. * @returns {Promise} - returns a promise (resolve, reject)
  1053. */
  1054. GET_ALL_SONGS_WITH_GENRE(payload) {
  1055. return new Promise((resolve, reject) => {
  1056. async.waterfall(
  1057. [
  1058. next => {
  1059. SongsModule.SongModel.find(
  1060. {
  1061. verified: true,
  1062. genres: { $regex: new RegExp(`^${payload.genre.toLowerCase()}$`, "i") }
  1063. },
  1064. next
  1065. );
  1066. }
  1067. ],
  1068. (err, songs) => {
  1069. if (err && err !== true) return reject(new Error(err));
  1070. return resolve({ songs });
  1071. }
  1072. );
  1073. });
  1074. }
  1075. // runjob songs GET_ORPHANED_PLAYLIST_SONGS {}
  1076. /**
  1077. * Gets a orphaned playlist songs
  1078. *
  1079. * @returns {Promise} - returns promise (reject, resolve)
  1080. */
  1081. GET_ORPHANED_PLAYLIST_SONGS() {
  1082. return new Promise((resolve, reject) => {
  1083. DBModule.runJob("GET_MODEL", { modelName: "playlist" }, this).then(playlistModel => {
  1084. playlistModel.find({}, (err, playlists) => {
  1085. if (err) reject(new Error(err));
  1086. else {
  1087. SongsModule.SongModel.find({}, { _id: true, youtubeId: true }, (err, songs) => {
  1088. if (err) reject(new Error(err));
  1089. else {
  1090. const songIds = songs.map(song => song._id.toString());
  1091. const orphanedYoutubeIds = new Set();
  1092. async.eachLimit(
  1093. playlists,
  1094. 1,
  1095. (playlist, next) => {
  1096. playlist.songs.forEach(song => {
  1097. if (
  1098. (!song._id || songIds.indexOf(song._id.toString() === -1)) &&
  1099. !orphanedYoutubeIds.has(song.youtubeId)
  1100. ) {
  1101. orphanedYoutubeIds.add(song.youtubeId);
  1102. }
  1103. });
  1104. next();
  1105. },
  1106. () => {
  1107. resolve({ youtubeIds: Array.from(orphanedYoutubeIds) });
  1108. }
  1109. );
  1110. }
  1111. });
  1112. }
  1113. });
  1114. });
  1115. });
  1116. }
  1117. /**
  1118. * Requests a song, adding it to the DB
  1119. *
  1120. * @param {object} payload - The payload
  1121. * @param {string} payload.youtubeId - The YouTube song id of the song
  1122. * @param {string} payload.userId - The user id of the person requesting the song
  1123. * @returns {Promise} - returns promise (reject, resolve)
  1124. */
  1125. REQUEST_SONG(payload) {
  1126. return new Promise((resolve, reject) => {
  1127. const { youtubeId, userId } = payload;
  1128. const requestedAt = Date.now();
  1129. async.waterfall(
  1130. [
  1131. next => {
  1132. DBModule.runJob("GET_MODEL", { modelName: "user" }, this)
  1133. .then(UserModel => {
  1134. UserModel.findOne({ _id: userId }, { "preferences.anonymousSongRequests": 1 }, next);
  1135. })
  1136. .catch(next);
  1137. },
  1138. (user, next) => {
  1139. SongsModule.SongModel.findOne({ youtubeId }, (err, song) => next(err, user, song));
  1140. },
  1141. // Get YouTube data from id
  1142. (user, song, next) => {
  1143. if (song) return next("This song is already in the database.", song);
  1144. // TODO Add err object as first param of callback
  1145. return YouTubeModule.runJob("GET_SONG", { youtubeId }, this)
  1146. .then(response => {
  1147. const { song } = response;
  1148. song.artists = [];
  1149. song.genres = [];
  1150. song.skipDuration = 0;
  1151. song.explicit = false;
  1152. song.requestedBy = user.preferences.anonymousSongRequests ? null : userId;
  1153. song.requestedAt = requestedAt;
  1154. song.verified = false;
  1155. next(null, song);
  1156. })
  1157. .catch(next);
  1158. },
  1159. (newSong, next) => {
  1160. const song = new SongsModule.SongModel(newSong);
  1161. song.save({ validateBeforeSave: false }, err => {
  1162. if (err) return next(err, song);
  1163. return next(null, song);
  1164. });
  1165. },
  1166. (song, next) => {
  1167. DBModule.runJob("GET_MODEL", { modelName: "user" }, this)
  1168. .then(UserModel => {
  1169. UserModel.findOne({ _id: userId }, (err, user) => {
  1170. if (err) return next(err);
  1171. if (!user) return next(null, song);
  1172. user.statistics.songsRequested += 1;
  1173. return user.save(err => {
  1174. if (err) return next(err);
  1175. return next(null, song);
  1176. });
  1177. });
  1178. })
  1179. .catch(next);
  1180. }
  1181. ],
  1182. async (err, song) => {
  1183. if (err && err !== "This song is already in the database.") return reject(err);
  1184. const { _id, youtubeId, title, artists, thumbnail, duration, verified } = song;
  1185. const trimmedSong = {
  1186. _id,
  1187. youtubeId,
  1188. title,
  1189. artists,
  1190. thumbnail,
  1191. duration,
  1192. verified
  1193. };
  1194. if (err && err === "This song is already in the database.")
  1195. return reject(new ErrorWithData(err, { song: trimmedSong }));
  1196. SongsModule.runJob("UPDATE_SONG", { songId: song._id });
  1197. return resolve({ song: trimmedSong });
  1198. }
  1199. );
  1200. });
  1201. }
  1202. // runjob songs REQUEST_ORPHANED_PLAYLIST_SONGS {}
  1203. /**
  1204. * Requests all orphaned playlist songs, adding them to the database
  1205. *
  1206. * @returns {Promise} - returns promise (reject, resolve)
  1207. */
  1208. REQUEST_ORPHANED_PLAYLIST_SONGS() {
  1209. return new Promise((resolve, reject) => {
  1210. DBModule.runJob("GET_MODEL", { modelName: "playlist" })
  1211. .then(playlistModel => {
  1212. SongsModule.runJob("GET_ORPHANED_PLAYLIST_SONGS", {}, this).then(response => {
  1213. const { youtubeIds } = response;
  1214. const playlistsToUpdate = new Set();
  1215. async.eachLimit(
  1216. youtubeIds,
  1217. 1,
  1218. (youtubeId, next) => {
  1219. async.waterfall(
  1220. [
  1221. next => {
  1222. console.log(
  1223. youtubeId,
  1224. `this is song ${youtubeIds.indexOf(youtubeId) + 1}/${youtubeIds.length}`
  1225. );
  1226. setTimeout(next, 150);
  1227. },
  1228. next => {
  1229. SongsModule.runJob(
  1230. "ENSURE_SONG_EXISTS_BY_SONG_ID",
  1231. { youtubeId, automaticallyRequested: true },
  1232. this
  1233. )
  1234. .then(() => next())
  1235. .catch(next);
  1236. },
  1237. next => {
  1238. console.log(444, youtubeId);
  1239. SongsModule.SongModel.findOne({ youtubeId }, next);
  1240. },
  1241. (song, next) => {
  1242. const { _id, title, artists, thumbnail, duration, verified } = song;
  1243. const trimmedSong = {
  1244. _id,
  1245. youtubeId,
  1246. title,
  1247. artists,
  1248. thumbnail,
  1249. duration,
  1250. verified
  1251. };
  1252. playlistModel.updateMany(
  1253. { "songs.youtubeId": song.youtubeId },
  1254. { $set: { "songs.$": trimmedSong } },
  1255. err => {
  1256. next(err, song);
  1257. }
  1258. );
  1259. },
  1260. (song, next) => {
  1261. playlistModel.find({ "songs._id": song._id }, next);
  1262. },
  1263. (playlists, next) => {
  1264. playlists.forEach(playlist => {
  1265. playlistsToUpdate.add(playlist._id.toString());
  1266. });
  1267. next();
  1268. }
  1269. ],
  1270. next
  1271. );
  1272. },
  1273. err => {
  1274. if (err) reject(err);
  1275. else {
  1276. async.eachLimit(
  1277. Array.from(playlistsToUpdate),
  1278. 1,
  1279. (playlistId, next) => {
  1280. PlaylistsModule.runJob(
  1281. "UPDATE_PLAYLIST",
  1282. {
  1283. playlistId
  1284. },
  1285. this
  1286. )
  1287. .then(() => {
  1288. next();
  1289. })
  1290. .catch(next);
  1291. },
  1292. err => {
  1293. if (err) reject(err);
  1294. else resolve();
  1295. }
  1296. );
  1297. }
  1298. }
  1299. );
  1300. });
  1301. })
  1302. .catch(reject);
  1303. });
  1304. }
  1305. /**
  1306. * Gets a list of all genres
  1307. *
  1308. * @returns {Promise} - returns promise (reject, resolve)
  1309. */
  1310. GET_GENRES() {
  1311. return new Promise((resolve, reject) => {
  1312. async.waterfall(
  1313. [
  1314. next => {
  1315. SongsModule.SongModel.distinct("genres", next);
  1316. }
  1317. ],
  1318. (err, genres) => {
  1319. if (err) reject(err);
  1320. resolve({ genres });
  1321. }
  1322. );
  1323. });
  1324. }
  1325. /**
  1326. * Gets a list of all artists
  1327. *
  1328. * @returns {Promise} - returns promise (reject, resolve)
  1329. */
  1330. GET_ARTISTS() {
  1331. return new Promise((resolve, reject) => {
  1332. async.waterfall(
  1333. [
  1334. next => {
  1335. SongsModule.SongModel.distinct("artists", next);
  1336. }
  1337. ],
  1338. (err, artists) => {
  1339. if (err) reject(err);
  1340. resolve({ artists });
  1341. }
  1342. );
  1343. });
  1344. }
  1345. /**
  1346. * Gets a list of all tags
  1347. *
  1348. * @returns {Promise} - returns promise (reject, resolve)
  1349. */
  1350. GET_TAGS() {
  1351. return new Promise((resolve, reject) => {
  1352. async.waterfall(
  1353. [
  1354. next => {
  1355. SongsModule.SongModel.distinct("tags", next);
  1356. }
  1357. ],
  1358. (err, tags) => {
  1359. if (err) reject(err);
  1360. resolve({ tags });
  1361. }
  1362. );
  1363. });
  1364. }
  1365. }
  1366. export default new _SongsModule();