| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555 | 'use strict';const async   = require('async'),	  request = require('request'),	  config  = require('config');const io = require('../io');const db = require('../db');const cache = require('../cache');const notifications = require('../notifications');const utils = require('../utils');const stations = require('../stations');const songs = require('../songs');const hooks = require('./hooks');cache.sub('station.pause', stationId => {	io.io.to(`station.${stationId}`).emit("event:stations.pause");});cache.sub('station.resume', stationId => {	stations.getStation(stationId, (err, station) => {		io.io.to(`station.${stationId}`).emit("event:stations.resume", {timePaused: station.timePaused});	});});cache.sub('station.queueUpdate', stationId => {	stations.getStation(stationId, (err, station) => {		if (!err) {			io.io.to(`station.${stationId}`).emit("event:queue.update", station.queue);		}	});});cache.sub('station.create', stationId => {	stations.initializeStation(stationId, (err, station) => {		console.log("*************", err, station);		//TODO Emit to admin station page		// TODO If community, check if on whitelist		console.log("*************", station.privacy);		if (station.privacy === 'public') io.io.to('home').emit("event:stations.created", station);		else {			let sockets = io.io.to('home').sockets;			console.log("*************", sockets.length);			for (let socketId in sockets) {				let socket = sockets[socketId];				let session = sockets[socketId].session;				console.log("*************", session);				if (session.sessionId) {					cache.hget('sessions', session.sessionId, (err, session) => {						console.log("*************", err, session);						if (!err && session) {							console.log("*************");							db.models.user.findOne({_id: session.userId}, (err, user) => {								console.log("*************", err, user.role, station.type, station.owner, session.userId);								if (user.role === 'admin') socket.emit("event:stations.created", station);								else if (station.type === "community" && station.owner === session.userId) socket.emit("event:stations.created", station);							});						}					});				}			}		}	});});module.exports = {	/**	 * Get a list of all the stations	 *	 * @param session	 * @param cb	 * @return {{ status: String, stations: Array }}	 */	index: (session, cb) => {		cache.hgetall('stations', (err, stations) => {			if (err && err !== true) {				return cb({					status: 'error',					message: 'An error occurred while obtaining the stations'				});			}			let arr = [];			let done = 0;			for (let prop in stations) {				// TODO If community, check if on whitelist				let station = stations[prop];				console.log(station)				if (station.privacy === 'public') add(true, station);				else if (!session.sessionId) add(false);				else {					cache.hget('sessions', session.sessionId, (err, session) => {						if (err || !session) {							add(false);						} else {							db.models.user.findOne({_id: session.userId}, (err, user) => {								if (err || !user) add(false);								else if (user.role === 'admin') add(true, station);								else if (station.type === 'official') add(false);								else if (station.owner === session.userId) add(true, station);								else add(false);							});						}					});				}			}			function add(add, station) {				console.log("ADD!", add, station);				if (add) arr.push(station);				done++;				if (done === Object.keys(stations).length) {					console.log("DONE!", done);					cb({ status: 'success', stations: arr });				}			}		});	},	getPlaylist: (session, stationId, cb) => {		let playlist = [];		stations.getStation(stationId, (err, station) => {			for (let s = 1; s < station.playlist.length; s++) {				songs.getSong(station.playlist[s], (err, song) => {					playlist.push(song);				});			}		});		cb({ status: 'success', data: playlist })	},	/**	 * Joins the station by its id	 *	 * @param session	 * @param stationId - the station id	 * @param cb	 * @return {{ status: String, userCount: Integer }}	 */	join: (session, stationId, cb) => {		stations.getStation(stationId, (err, station) => {			if (err && err !== true) return cb({ status: 'error', message: 'An error occurred while joining the station' });			if (station) {				if (station.privacy !== 'private') {					func();				} else {					// TODO If community, check if on whitelist					if (!session.userId) return cb({ status: 'error', message: 'An error occurred while joining the station1' });					db.models.user.findOne({_id: session.userId}, (err, user) => {						if (err || !user) return cb({ status: 'error', message: 'An error occurred while joining the station2' });						if (user.role === 'admin') return func();						if (station.type === 'official') return cb({ status: 'error', message: 'An error occurred while joining the station3' });						if (station.owner === session.userId) return func();						return cb({ status: 'error', message: 'An error occurred while joining the station4' });					});				}				function func() {					utils.socketJoinRoom(session.socketId, `station.${stationId}`);					if (station.currentSong) {						utils.socketJoinSongRoom(session.socketId, `song.${station.currentSong._id}`);						//TODO Emit to cache, listen on cache						songs.getSong(station.currentSong._id, (err, song) => {							if (!err && song) {								station.currentSong.likes = song.likes;								station.currentSong.dislikes = song.dislikes;							} else {								station.currentSong.likes = -1;								station.currentSong.dislikes = -1;							}							cb({								status: 'success',								data: {									type: station.type,									currentSong: station.currentSong,									startedAt: station.startedAt,									paused: station.paused,									timePaused: station.timePaused,									description: station.description,									displayName: station.displayName,									privacy: station.privacy,									partyMode: station.partyMode,									owner: station.owner,									privatePlaylist: station.privatePlaylist								}							});						});					} else {						cb({							status: 'success',							data: {								type: station.type,								currentSong: null,								startedAt: station.startedAt,								paused: station.paused,								timePaused: station.timePaused,								description: station.description,								displayName: station.displayName,								privacy: station.privacy,								owner: station.owner							}						});					}				}			} else {				cb({ status: 'failure', message: `That station doesn't exist` });			}		});	},	/**	 * Skips the users current station	 *	 * @param session	 * @param stationId - the station id	 * @param cb	 * @return {{ status: String, skipCount: Integer }}	 */	/*skip: (session, stationId, cb) => {		if (!session) return cb({ status: 'failure', message: 'You must be logged in to skip a song!' });		stations.getStation(stationId, (err, station) => {						if (err && err !== true) {				return cb({ status: 'error', message: 'An error occurred while skipping the station' });			}			if (station) {				cache.client.hincrby('station.skipCounts', session.stationId, 1, (err, skipCount) => {					session.skippedSong = true;					if (err) return cb({ status: 'error', message: 'An error occurred while skipping the station' });					cache.hget('station.userCounts', session.stationId, (err, userCount) => {						cb({ status: 'success', skipCount });					});				});			}			else {				cb({ status: 'failure', message: `That station doesn't exist` });			}		});	},*/	forceSkip: hooks.ownerRequired((session, stationId, cb) => {		stations.getStation(stationId, (err, station) => {			if (err && err !== true) {				return cb({ status: 'error', message: 'An error occurred while skipping the station' });			}			if (station) {				notifications.unschedule(`stations.nextSong?id=${stationId}`);				//notifications.schedule(`stations.nextSong?id=${stationId}`, 100);				stations.skipStation(stationId)();			}			else {				cb({ status: 'failure', message: `That station doesn't exist` });			}		});	}),	/**	 * Leaves the users current station	 *	 * @param session	 * @param stationId	 * @param cb	 * @return {{ status: String, userCount: Integer }}	 */	leave: (session, stationId, cb) => {		stations.getStation(stationId, (err, station) => {			if (err && err !== true) {				return cb({ status: 'error', message: 'An error occurred while leaving the station' });			}			if (session) session.stationId = null;			else if (station) {				cache.client.hincrby('station.userCounts', stationId, -1, (err, userCount) => {					if (err) return cb({ status: 'error', message: 'An error occurred while leaving the station' });					utils.socketLeaveRooms(session);					cb({ status: 'success', userCount });				});			} else {				cb({ status: 'failure', message: `That station doesn't exist, it may have been deleted` });			}		});	},	updateDisplayName: hooks.ownerRequired((session, stationId, newDisplayName, cb) => {		db.models.station.update({_id: stationId}, {$set: {displayName: newDisplayName}}, (err) => {			if (err) return cb({ status: 'failure', message: 'Something went wrong when saving the station.' });			stations.updateStation(stationId, () => {				//TODO Pub/sub for displayName change				cb({ status: 'success', message: 'Successfully updated the display name.' });			})		});	}),	updateDescription: hooks.ownerRequired((session, stationId, newDescription, cb) => {		db.models.station.update({_id: stationId}, {$set: {description: newDescription}}, (err) => {			if (err) return cb({ status: 'failure', message: 'Something went wrong when saving the station.' });			stations.updateStation(stationId, () => {				//TODO Pub/sub for description change				cb({ status: 'success', message: 'Successfully updated the description.' });			})		});	}),	updatePrivacy: hooks.ownerRequired((session, stationId, newPrivacy, cb) => {		db.models.station.update({_id: stationId}, {$set: {privacy: newPrivacy}}, (err) => {			if (err) return cb({ status: 'failure', message: 'Something went wrong when saving the station.' });			stations.updateStation(stationId, () => {				//TODO Pub/sub for privacy change				cb({ status: 'success', message: 'Successfully updated the privacy.' });			})		});	}),	updatePartyMode: hooks.ownerRequired((session, stationId, newPartyMode, cb) => {		stations.getStation(stationId, (err, station) => {			if (err) return cb({ status: 'failure', message: err });			if (station.partyMode === newPartyMode) return cb({ status: 'failure', message: 'The party mode was already ' + ((newPartyMode) ? 'enabled.' : 'disabled.') });			db.models.station.update({_id: stationId}, {$set: {partyMode: newPartyMode}}, (err) => {				if (err) return cb({ status: 'failure', message: 'Something went wrong when saving the station.' });				stations.updateStation(stationId, () => {					//TODO Pub/sub for privacy change					stations.skipStation(stationId)();					cb({ status: 'success', message: 'Successfully updated the party mode.' });				})			});		});	}),	pause: hooks.ownerRequired((session, stationId, cb) => {		stations.getStation(stationId, (err, station) => {			if (err && err !== true) {				return cb({ status: 'error', message: 'An error occurred while pausing the station' });			} else if (station) {				if (!station.paused) {					station.paused = true;					station.pausedAt = Date.now();					db.models.station.update({_id: stationId}, {$set: {paused: true, pausedAt: Date.now()}}, () => {						if (err) return cb({ status: 'failure', message: 'An error occurred while pausing the station.' });						stations.updateStation(stationId, () => {							cache.pub('station.pause', stationId);							notifications.unschedule(`stations.nextSong?id=${stationId}`);							cb({ status: 'success' });						});					});				} else {					cb({ status: 'failure', message: 'That station was already paused.' });				}				cb({ status: 'success' });			} else {				cb({ status: 'failure', message: `That station doesn't exist, it may have been deleted` });			}		});	}),	resume: hooks.ownerRequired((session, stationId, cb) => {		stations.getStation(stationId, (err, station) => {			if (err && err !== true) {				return cb({ status: 'error', message: 'An error occurred while resuming the station' });			} else if (station) {				if (station.paused) {					station.paused = false;					station.timePaused += (Date.now() - station.pausedAt);					console.log("&&&", station.timePaused, station.pausedAt, Date.now(), station.timePaused);					db.models.station.update({_id: stationId}, {$set: {paused: false}, $inc: {timePaused: Date.now() - station.pausedAt}}, () => {						stations.updateStation(stationId, (err, station) => {							cache.pub('station.resume', stationId);							cb({ status: 'success' });						});					});				} else {					cb({ status: 'failure', message: 'That station is not paused.' });				}			} else {				cb({ status: 'failure', message: `That station doesn't exist, it may have been deleted` });			}		});	}),	remove: hooks.ownerRequired((session, stationId, cb) => {		db.models.station.remove({ _id: stationId }, (err) => {			console.log(err, stationId);			if (err) return cb({status: 'failure', message: 'Something went wrong when deleting that station.'});			cache.hdel('stations', stationId, () => {				return cb({ status: 'success', message: 'Station successfully removed' });			});		});	}),	create: hooks.loginRequired((session, data, cb) => {		async.waterfall([			(next) => {				return (data) ? next() : cb({ 'status': 'failure', 'message': 'Invalid data' });			},			// check the cache for the station			(next) => cache.hget('stations', data._id, next),			// if the cached version exist			(station, next) => {				if (station) return next({ 'status': 'failure', 'message': 'A station with that id already exists' });				db.models.station.findOne({ _id: data._id }, next);			},			(station, next) => {				if (station) return next({ 'status': 'failure', 'message': 'A station with that id already exists' });				const { _id, displayName, description, genres, playlist, type } = data;				if (type == 'official') {					db.models.station.create({						_id,						displayName,						description,						type,						privacy: 'private',						playlist,						genres,						currentSong: stations.defaultSong					}, next);				} else if (type == 'community') {					cache.hget('sessions', session.sessionId, (err, session) => {						db.models.station.create({							_id,							displayName,							description,							type,							privacy: 'private',							owner: session.userId,							queue: [],							currentSong: null						}, next);					});				}			}		], (err, station) => {			if (err) console.log(err); return cb({ 'status': 'failure', 'message': 'Something went wrong.'});			return cb(null, { 'status': 'success', 'message': 'Successfully created station.' });			cache.pub('station.create', data._id);		});	}),		addToQueue: hooks.loginRequired((session, stationId, songId, cb, userId) => {		stations.getStation(stationId, (err, station) => {			if (err) return cb(err);			if (station.type === 'community') {				let has = false;				station.queue.forEach((queueSong) => {					if (queueSong._id === songId) {						has = true;					}				});				if (has) return cb({'status': 'failure', 'message': 'That song has already been added to the queue.'});				if (station.currentSong && station.currentSong._id === songId) return cb({'status': 'failure', 'message': 'That song is currently playing.'});				songs.getSong(songId, (err, song) => {					if (err) {						utils.getSongFromYouTube(songId, (song) => {							song.artists = [];							song.skipDuration = 0;							song.likes = -1;							song.dislikes = -1;							song.thumbnail = "empty";							song.explicit = false;							cont(song);						});					} else cont(song);					function cont(song) {						db.models.station.update({_id: stationId}, {$push: {queue: song}}, (err) => {							console.log(err);							if (err) return cb({'status': 'failure', 'message': 'Something went wrong.'});							stations.updateStation(stationId, (err, station) => {								if (err) return cb(err);								cache.pub('station.queueUpdate', stationId);								cb({'status': 'success', 'message': 'Added that song to the queue.'});							});						});					}				});			} else cb({'status': 'failure', 'message': 'That station is not a community station.'});		});	}),	removeFromQueue: hooks.ownerRequired((session, stationId, songId, cb, userId) => {		stations.getStation(stationId, (err, station) => {			if (err) return cb(err);			if (station.type === 'community') {				let has = false;				station.queue.forEach((queueSong) => {					if (queueSong._id === songId) {						has = true;					}				});				if (!has) return cb({'status': 'failure', 'message': 'That song is not in the queue.'});				db.models.update({_id: stationId}, {$pull: {queue: {songId: songId}}}, (err) => {					if (err) return cb({'status': 'failure', 'message': 'Something went wrong.'});					stations.updateStation(stationId, (err, station) => {						if (err) return cb(err);						cache.pub('station.queueUpdate', stationId);					});				});			} else cb({'status': 'failure', 'message': 'That station is not a community station.'});		});	}),	getQueue: hooks.adminRequired((session, stationId, cb) => {		stations.getStation(stationId, (err, station) => {			if (err) return cb(err);			if (!station) return cb({'status': 'failure', 'message': 'Station not found.'});			if (station.type === 'community') {				cb({'status': 'success', queue: station.queue});			} else cb({'status': 'failure', 'message': 'That station is not a community station.'});		});	}),	selectPrivatePlaylist: hooks.csOwnerRequired((session, stationId, playlistId, cb, userId) => {		stations.getStation(stationId, (err, station) => {			if (err) return cb(err);			if (station.type === 'community') {				if (station.privatePlaylist === playlistId) return cb({'status': 'failure', 'message': 'That playlist is already selected.'});				db.models.playlist.findOne({_id: playlistId}, (err, playlist) => {					if (err) return cb(err);					if (playlist) {						db.models.station.update({_id: stationId}, {$set: {privatePlaylist: playlistId, currentSongIndex: 0}}, (err) => {							if (err) return cb(err);							stations.updateStation(stationId, (err, station) => {								if (err) return cb(err);								stations.skipStation(stationId)();								cb({'status': 'success', 'message': 'Playlist selected.'});							});						});					} else cb({'status': 'failure', 'message': 'Playlist not found.'});				});			} else cb({'status': 'failure', 'message': 'That station is not a community station.'});		});	}),};
 |