songs.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475
  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 StationModule;
  10. let PlaylistModule;
  11. class _SongsModule extends CoreClass {
  12. // eslint-disable-next-line require-jsdoc
  13. constructor() {
  14. super("songs");
  15. SongsModule = this;
  16. }
  17. /**
  18. * Initialises the songs module
  19. *
  20. * @returns {Promise} - returns promise (reject, resolve)
  21. */
  22. async initialize() {
  23. this.setStage(1);
  24. CacheModule = this.moduleManager.modules.cache;
  25. DBModule = this.moduleManager.modules.db;
  26. UtilsModule = this.moduleManager.modules.utils;
  27. YouTubeModule = this.moduleManager.modules.youtube;
  28. StationModule = this.moduleManager.modules.stations;
  29. PlaylistModule = this.moduleManager.modules.playlists;
  30. this.SongModel = await DBModule.runJob("GET_MODEL", { modelName: "song" });
  31. this.SongSchemaCache = await CacheModule.runJob("GET_SCHEMA", { schemaName: "song" });
  32. this.setStage(2);
  33. return new Promise((resolve, reject) =>
  34. async.waterfall(
  35. [
  36. next => {
  37. this.setStage(2);
  38. CacheModule.runJob("HGETALL", { table: "songs" })
  39. .then(songs => {
  40. next(null, songs);
  41. })
  42. .catch(next);
  43. },
  44. (songs, next) => {
  45. this.setStage(3);
  46. if (!songs) return next();
  47. const songIds = Object.keys(songs);
  48. return async.each(
  49. songIds,
  50. (songId, next) => {
  51. SongsModule.SongModel.findOne({ songId }, (err, song) => {
  52. if (err) next(err);
  53. else if (!song)
  54. CacheModule.runJob("HDEL", {
  55. table: "songs",
  56. key: songId
  57. })
  58. .then(() => next())
  59. .catch(next);
  60. else next();
  61. });
  62. },
  63. next
  64. );
  65. },
  66. next => {
  67. this.setStage(4);
  68. SongsModule.SongModel.find({}, next);
  69. },
  70. (songs, next) => {
  71. this.setStage(5);
  72. async.each(
  73. songs,
  74. (song, next) => {
  75. CacheModule.runJob("HSET", {
  76. table: "songs",
  77. key: song.songId,
  78. value: SongsModule.SongSchemaCache(song)
  79. })
  80. .then(() => next())
  81. .catch(next);
  82. },
  83. next
  84. );
  85. }
  86. ],
  87. async err => {
  88. if (err) {
  89. err = await UtilsModule.runJob("GET_ERROR", { error: err });
  90. reject(new Error(err));
  91. } else resolve();
  92. }
  93. )
  94. );
  95. }
  96. /**
  97. * Gets a song by id from the cache or Mongo, and if it isn't in the cache yet, adds it the cache
  98. *
  99. * @param {object} payload - object containing the payload
  100. * @param {string} payload.id - the id of the song we are trying to get
  101. * @returns {Promise} - returns a promise (resolve, reject)
  102. */
  103. GET_SONG(payload) {
  104. return new Promise((resolve, reject) =>
  105. async.waterfall(
  106. [
  107. next => {
  108. if (!mongoose.Types.ObjectId.isValid(payload.id)) return next("Id is not a valid ObjectId.");
  109. return CacheModule.runJob("HGET", { table: "songs", key: payload.id }, this)
  110. .then(song => next(null, song))
  111. .catch(next);
  112. },
  113. (song, next) => {
  114. if (song) return next(true, song);
  115. return SongsModule.SongModel.findOne({ _id: payload.id }, next);
  116. },
  117. (song, next) => {
  118. if (song) {
  119. CacheModule.runJob(
  120. "HSET",
  121. {
  122. table: "songs",
  123. key: payload.id,
  124. value: song
  125. },
  126. this
  127. ).then(song => next(null, song));
  128. } else next("Song not found.");
  129. }
  130. ],
  131. (err, song) => {
  132. if (err && err !== true) return reject(new Error(err));
  133. return resolve({ song });
  134. }
  135. )
  136. );
  137. }
  138. /**
  139. * Makes sure that if a song is not currently in the songs db, to add it
  140. *
  141. * @param {object} payload - an object containing the payload
  142. * @param {string} payload.songId - the youtube song id of the song we are trying to ensure is in the songs db
  143. * @returns {Promise} - returns a promise (resolve, reject)
  144. */
  145. ENSURE_SONG_EXISTS_BY_SONG_ID(payload) {
  146. return new Promise((resolve, reject) =>
  147. async.waterfall(
  148. [
  149. next => {
  150. SongsModule.SongModel.findOne({ songId: payload.songId }, next);
  151. },
  152. (song, next) => {
  153. if (song) next(true, song);
  154. else {
  155. YouTubeModule.runJob("GET_SONG", { songId: payload.songId }, this)
  156. .then(response => next(null, { ...response.song }))
  157. .catch(next);
  158. }
  159. },
  160. (_song, next) => {
  161. const song = new SongsModule.SongModel({ ..._song });
  162. song.save({ validateBeforeSave: true }, err => {
  163. if (err) return next(err, song);
  164. return next(null, song);
  165. });
  166. }
  167. ],
  168. (err, song) => {
  169. if (err && err !== true) return reject(new Error(err));
  170. return resolve({ song });
  171. }
  172. )
  173. );
  174. }
  175. /**
  176. * Gets a song by song id from the cache or Mongo, and if it isn't in the cache yet, adds it the cache
  177. *
  178. * @param {object} payload - an object containing the payload
  179. * @param {string} payload.songId - the mongo id of the song we are trying to get
  180. * @returns {Promise} - returns a promise (resolve, reject)
  181. */
  182. GET_SONG_FROM_ID(payload) {
  183. return new Promise((resolve, reject) =>
  184. async.waterfall(
  185. [
  186. next => {
  187. SongsModule.SongModel.findOne({ songId: payload.songId }, next);
  188. }
  189. ],
  190. (err, song) => {
  191. if (err && err !== true) return reject(new Error(err));
  192. return resolve({ song });
  193. }
  194. )
  195. );
  196. }
  197. /**
  198. * Gets a song from id from Mongo and updates the cache with it
  199. *
  200. * @param {object} payload - an object containing the payload
  201. * @param {string} payload.songId - the id of the song we are trying to update
  202. * @returns {Promise} - returns a promise (resolve, reject)
  203. */
  204. UPDATE_SONG(payload) {
  205. return new Promise((resolve, reject) =>
  206. async.waterfall(
  207. [
  208. next => {
  209. SongsModule.SongModel.findOne({ _id: payload.songId }, next);
  210. },
  211. (song, next) => {
  212. if (!song) {
  213. CacheModule.runJob("HDEL", {
  214. table: "songs",
  215. key: payload.songId
  216. });
  217. return next("Song not found.");
  218. }
  219. return CacheModule.runJob(
  220. "HSET",
  221. {
  222. table: "songs",
  223. key: payload.songId,
  224. value: song
  225. },
  226. this
  227. )
  228. .then(song => {
  229. next(null, song);
  230. })
  231. .catch(next);
  232. },
  233. (song, next) => {
  234. next(null, song);
  235. const { _id, songId, title, artists, thumbnail, duration, verified } = song;
  236. const trimmedSong = {
  237. _id,
  238. songId,
  239. title,
  240. artists,
  241. thumbnail,
  242. duration,
  243. verified
  244. };
  245. this.log("INFO", `Going to update playlists and stations now for song ${_id}`);
  246. DBModule.runJob("GET_MODEL", { modelName: "playlist" }).then(playlistModel => {
  247. playlistModel.updateMany(
  248. { "songs._id": song._id },
  249. { $set: { "songs.$": trimmedSong } },
  250. err => {
  251. if (err) this.log("ERROR", err);
  252. else
  253. playlistModel.find({ "songs._id": song._id }, (err, playlists) => {
  254. playlists.forEach(playlist => {
  255. PlaylistModule.runJob("UPDATE_PLAYLIST", {
  256. playlistId: playlist._id
  257. });
  258. });
  259. });
  260. }
  261. );
  262. });
  263. DBModule.runJob("GET_MODEL", { modelName: "station" }).then(stationModel => {
  264. stationModel.updateMany(
  265. { "queue._id": song._id },
  266. {
  267. $set: {
  268. "queue.$.songId": songId,
  269. "queue.$.title": title,
  270. "queue.$.artists": artists,
  271. "queue.$.thumbnail": thumbnail,
  272. "queue.$.duration": duration,
  273. "queue.$.verified": verified
  274. }
  275. },
  276. err => {
  277. if (err) this.log("ERROR", err);
  278. else
  279. stationModel.find({ "queue._id": song._id }, (err, stations) => {
  280. stations.forEach(station => {
  281. StationModule.runJob("UPDATE_STATION", { stationId: station._id });
  282. });
  283. });
  284. }
  285. );
  286. });
  287. }
  288. ],
  289. (err, song) => {
  290. if (err && err !== true) return reject(new Error(err));
  291. return resolve(song);
  292. }
  293. )
  294. );
  295. }
  296. /**
  297. * Deletes song from id from Mongo and cache
  298. *
  299. * @param {object} payload - returns an object containing the payload
  300. * @param {string} payload.songId - the id of the song we are trying to delete
  301. * @returns {Promise} - returns a promise (resolve, reject)
  302. */
  303. DELETE_SONG(payload) {
  304. return new Promise((resolve, reject) =>
  305. async.waterfall(
  306. [
  307. next => {
  308. SongsModule.SongModel.deleteOne({ songId: payload.songId }, next);
  309. },
  310. next => {
  311. CacheModule.runJob(
  312. "HDEL",
  313. {
  314. table: "songs",
  315. key: payload.songId
  316. },
  317. this
  318. )
  319. .then(() => next())
  320. .catch(next);
  321. }
  322. ],
  323. err => {
  324. if (err && err !== true) return reject(new Error(err));
  325. return resolve();
  326. }
  327. )
  328. );
  329. }
  330. /**
  331. * Recalculates dislikes and likes for a song
  332. *
  333. * @param {object} payload - returns an object containing the payload
  334. * @param {string} payload.musareSongId - the (musare) id of the song
  335. * @param {string} payload.songId - the (mongodb) id of the song
  336. * @returns {Promise} - returns a promise (resolve, reject)
  337. */
  338. async RECALCULATE_SONG_RATINGS(payload) {
  339. const playlistModel = await DBModule.runJob("GET_MODEL", { modelName: "playlist" }, this);
  340. return new Promise((resolve, reject) => {
  341. async.waterfall(
  342. [
  343. next => {
  344. playlistModel.countDocuments(
  345. { songs: { $elemMatch: { songId: payload.musareSongId } }, displayName: "Liked Songs" },
  346. (err, likes) => {
  347. if (err) return next(err);
  348. return next(null, likes);
  349. }
  350. );
  351. },
  352. (likes, next) => {
  353. playlistModel.countDocuments(
  354. { songs: { $elemMatch: { songId: payload.musareSongId } }, displayName: "Disliked Songs" },
  355. (err, dislikes) => {
  356. if (err) return next(err);
  357. return next(err, { likes, dislikes });
  358. }
  359. );
  360. },
  361. ({ likes, dislikes }, next) => {
  362. SongsModule.SongModel.updateOne(
  363. { _id: payload.songId },
  364. {
  365. $set: {
  366. likes,
  367. dislikes
  368. }
  369. },
  370. err => next(err, { likes, dislikes })
  371. );
  372. }
  373. ],
  374. (err, { likes, dislikes }) => {
  375. if (err) return reject(new Error(err));
  376. return resolve({ likes, dislikes });
  377. }
  378. );
  379. });
  380. }
  381. /**
  382. * Gets an array of all genres
  383. *
  384. * @returns {Promise} - returns a promise (resolve, reject)
  385. */
  386. GET_ALL_GENRES() {
  387. return new Promise((resolve, reject) =>
  388. async.waterfall(
  389. [
  390. next => {
  391. SongsModule.SongModel.find({ verified: true }, { genres: 1, _id: false }, next);
  392. },
  393. (songs, next) => {
  394. let allGenres = [];
  395. songs.forEach(song => {
  396. allGenres = allGenres.concat(song.genres);
  397. });
  398. const lowerCaseGenres = allGenres.map(genre => genre.toLowerCase());
  399. const uniqueGenres = lowerCaseGenres.filter(
  400. (value, index, self) => self.indexOf(value) === index
  401. );
  402. next(null, uniqueGenres);
  403. }
  404. ],
  405. (err, genres) => {
  406. if (err && err !== true) return reject(new Error(err));
  407. return resolve({ genres });
  408. }
  409. )
  410. );
  411. }
  412. /**
  413. * Gets an array of all songs with a specific genre
  414. *
  415. * @param {object} payload - returns an object containing the payload
  416. * @param {string} payload.genre - the genre
  417. * @returns {Promise} - returns a promise (resolve, reject)
  418. */
  419. GET_ALL_SONGS_WITH_GENRE(payload) {
  420. return new Promise((resolve, reject) =>
  421. async.waterfall(
  422. [
  423. next => {
  424. SongsModule.SongModel.find(
  425. { verified: true, genres: { $regex: new RegExp(`^${payload.genre.toLowerCase()}$`, "i") } },
  426. next
  427. );
  428. }
  429. ],
  430. (err, songs) => {
  431. if (err && err !== true) return reject(new Error(err));
  432. return resolve({ songs });
  433. }
  434. )
  435. );
  436. }
  437. }
  438. export default new _SongsModule();