"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.Player = void 0; const tslib_1 = require("tslib"); const discord_js_1 = require("discord.js"); const tiny_typed_emitter_1 = require("tiny-typed-emitter"); const Queue_1 = require("./Structures/Queue"); const VoiceUtils_1 = require("./VoiceInterface/VoiceUtils"); const types_1 = require("./types/types"); const Track_1 = tslib_1.__importDefault(require("./Structures/Track")); const play_dl_1 = tslib_1.__importDefault(require("play-dl")); const spotify_url_info_1 = tslib_1.__importDefault(require("spotify-url-info")); const QueryResolver_1 = require("./utils/QueryResolver"); const Util_1 = require("./utils/Util"); const PlayerError_1 = require("./Structures/PlayerError"); const Playlist_1 = require("./Structures/Playlist"); const ExtractorModel_1 = require("./Structures/ExtractorModel"); const voice_1 = require("@discordjs/voice"); class Player extends tiny_typed_emitter_1.TypedEmitter { /** * Creates new Discord Player * @param {Client} client The Discord Client * @param {PlayerInitOptions} [options] The player init options */ constructor(client, options = {}) { super(); this.options = { autoRegisterExtractor: true, connectionTimeout: 20000 }; this.queues = new discord_js_1.Collection(); this.voiceUtils = new VoiceUtils_1.VoiceUtils(); this.extractors = new discord_js_1.Collection(); this.requiredEvents = ["error", "connectionError"]; /** * The discord.js client * @type {Client} */ this.client = client; if (this.client?.options?.intents && !new discord_js_1.IntentsBitField(this.client?.options?.intents).has(discord_js_1.IntentsBitField.Flags.GuildVoiceStates)) { throw new PlayerError_1.PlayerError('client is missing "GuildVoiceStates" intent'); } /** * The extractors collection * @type {ExtractorModel} */ this.options = Object.assign(this.options, options); this.client.on("voiceStateUpdate", this._handleVoiceState.bind(this)); if (this.options?.autoRegisterExtractor) { let nv; // eslint-disable-line @typescript-eslint/no-explicit-any if ((nv = Util_1.Util.require("@discord-player/extractor"))) { ["Attachment", "Facebook", "Reverbnation", "Vimeo"].forEach((ext) => void this.use(ext, nv[ext])); } } } /** * Handles voice state update * @param {VoiceState} oldState The old voice state * @param {VoiceState} newState The new voice state * @returns {void} * @private */ _handleVoiceState(oldState, newState) { const queue = this.getQueue(oldState.guild.id); if (!queue || !queue.connection) return; if (oldState.channelId && !newState.channelId && newState.member.id === newState.guild.members.me.id) { try { queue.destroy(); } catch { /* noop */ } return void this.emit("botDisconnect", queue); } if (!oldState.channelId && newState.channelId && newState.member.id === newState.guild.members.me.id) { if (!oldState.serverMute && newState.serverMute) { // state.serverMute can be null queue.setPaused(!!newState.serverMute); } else if (!oldState.suppress && newState.suppress) { // state.suppress can be null queue.setPaused(!!newState.suppress); if (newState.suppress) { newState.guild.members.me.voice.setRequestToSpeak(true).catch(Util_1.Util.noop); } } } if (oldState.channelId === newState.channelId && newState.member.id === newState.guild.members.me.id) { if (!oldState.serverMute && newState.serverMute) { // state.serverMute can be null queue.setPaused(!!newState.serverMute); } else if (!oldState.suppress && newState.suppress) { // state.suppress can be null queue.setPaused(!!newState.suppress); if (newState.suppress) { newState.guild.members.me.voice.setRequestToSpeak(true).catch(Util_1.Util.noop); } } } if (queue.connection && !newState.channelId && oldState.channelId === queue.connection.channel.id) { if (!Util_1.Util.isVoiceEmpty(queue.connection.channel)) return; const timeout = setTimeout(() => { if (!Util_1.Util.isVoiceEmpty(queue.connection.channel)) return; if (!this.queues.has(queue.guild.id)) return; if (queue.options.leaveOnEmpty) queue.destroy(true); this.emit("channelEmpty", queue); }, queue.options.leaveOnEmptyCooldown || 0).unref(); queue._cooldownsTimeout.set(`empty_${oldState.guild.id}`, timeout); } if (queue.connection && newState.channelId && newState.channelId === queue.connection.channel.id) { const emptyTimeout = queue._cooldownsTimeout.get(`empty_${oldState.guild.id}`); const channelEmpty = Util_1.Util.isVoiceEmpty(queue.connection.channel); if (!channelEmpty && emptyTimeout) { clearTimeout(emptyTimeout); queue._cooldownsTimeout.delete(`empty_${oldState.guild.id}`); } } if (oldState.channelId && newState.channelId && oldState.channelId !== newState.channelId && newState.member.id === newState.guild.members.me.id) { if (queue.connection && newState.member.id === newState.guild.members.me.id) queue.connection.channel = newState.channel; const emptyTimeout = queue._cooldownsTimeout.get(`empty_${oldState.guild.id}`); const channelEmpty = Util_1.Util.isVoiceEmpty(queue.connection.channel); if (!channelEmpty && emptyTimeout) { clearTimeout(emptyTimeout); queue._cooldownsTimeout.delete(`empty_${oldState.guild.id}`); } else { const timeout = setTimeout(() => { if (queue.connection && !Util_1.Util.isVoiceEmpty(queue.connection.channel)) return; if (!this.queues.has(queue.guild.id)) return; if (queue.options.leaveOnEmpty) queue.destroy(true); this.emit("channelEmpty", queue); }, queue.options.leaveOnEmptyCooldown || 0).unref(); queue._cooldownsTimeout.set(`empty_${oldState.guild.id}`, timeout); } } } /** * Creates a queue for a guild if not available, else returns existing queue * @param {GuildResolvable} guild The guild * @param {PlayerOptions} queueInitOptions Queue init options * @returns {Queue} */ createQueue(guild, queueInitOptions = {}) { guild = this.client.guilds.resolve(guild); if (!guild) throw new PlayerError_1.PlayerError("Unknown Guild", PlayerError_1.ErrorStatusCode.UNKNOWN_GUILD); if (this.queues.has(guild.id)) return this.queues.get(guild.id); const _meta = queueInitOptions.metadata; delete queueInitOptions["metadata"]; queueInitOptions.volumeSmoothness ?? (queueInitOptions.volumeSmoothness = 0.08); const queue = new Queue_1.Queue(this, guild, queueInitOptions); queue.metadata = _meta; this.queues.set(guild.id, queue); return queue; } /** * Returns the queue if available * @param {GuildResolvable} guild The guild id * @returns {Queue} */ getQueue(guild) { guild = this.client.guilds.resolve(guild); if (!guild) throw new PlayerError_1.PlayerError("Unknown Guild", PlayerError_1.ErrorStatusCode.UNKNOWN_GUILD); return this.queues.get(guild.id); } /** * Deletes a queue and returns deleted queue object * @param {GuildResolvable} guild The guild id to remove * @returns {Queue} */ deleteQueue(guild) { guild = this.client.guilds.resolve(guild); if (!guild) throw new PlayerError_1.PlayerError("Unknown Guild", PlayerError_1.ErrorStatusCode.UNKNOWN_GUILD); const prev = this.getQueue(guild); try { prev.destroy(); } catch { } // eslint-disable-line no-empty this.queues.delete(guild.id); return prev; } /** * @typedef {object} PlayerSearchResult * @property {Playlist} [playlist] The playlist (if any) * @property {Track[]} tracks The tracks */ /** * Search tracks * @param {string|Track} query The search query * @param {SearchOptions} options The search options * @returns {Promise} */ async search(query, options) { if (query instanceof Track_1.default) return { playlist: query.playlist || null, tracks: [query] }; if (!options) throw new PlayerError_1.PlayerError("DiscordPlayer#search needs search options!", PlayerError_1.ErrorStatusCode.INVALID_ARG_TYPE); options.requestedBy = this.client.users.resolve(options.requestedBy); if (!("searchEngine" in options)) options.searchEngine = types_1.QueryType.AUTO; if (typeof options.searchEngine === "string" && this.extractors.has(options.searchEngine)) { const extractor = this.extractors.get(options.searchEngine); if (!extractor.validate(query)) return { playlist: null, tracks: [] }; const data = await extractor.handle(query); if (data && data.data.length) { const playlist = !data.playlist ? null : new Playlist_1.Playlist(this, { ...data.playlist, tracks: [] }); const tracks = data.data.map((m) => new Track_1.default(this, { ...m, requestedBy: options.requestedBy, duration: Util_1.Util.buildTimeCode(Util_1.Util.parseMS(m.duration)), playlist: playlist })); if (playlist) playlist.tracks = tracks; return { playlist: playlist, tracks: tracks }; } } // eslint-disable-next-line @typescript-eslint/no-unused-vars for (const [_, extractor] of this.extractors) { if (options.blockExtractor) break; if (!extractor.validate(query)) continue; const data = await extractor.handle(query); if (data && data.data.length) { const playlist = !data.playlist ? null : new Playlist_1.Playlist(this, { ...data.playlist, tracks: [] }); const tracks = data.data.map((m) => new Track_1.default(this, { ...m, requestedBy: options.requestedBy, duration: Util_1.Util.buildTimeCode(Util_1.Util.parseMS(m.duration)), playlist: playlist })); if (playlist) playlist.tracks = tracks; return { playlist: playlist, tracks: tracks }; } } const qt = options.searchEngine === types_1.QueryType.AUTO ? await QueryResolver_1.QueryResolver.resolve(query) : options.searchEngine; switch (qt) { case types_1.QueryType.YOUTUBE_VIDEO: { const info = await play_dl_1.default.video_info(query).catch(Util_1.Util.noop); if (!info) return { playlist: null, tracks: [] }; const track = new Track_1.default(this, { title: info.video_details.title, description: info.video_details.description, author: info.video_details.channel?.name, url: info.video_details.url, requestedBy: options.requestedBy, thumbnail: Util_1.Util.last(info.video_details.thumbnails)?.url, views: info.video_details.views || 0, duration: Util_1.Util.buildTimeCode(Util_1.Util.parseMS(info.video_details.durationInSec * 1000)), source: "youtube", raw: info }); return { playlist: null, tracks: [track] }; } case types_1.QueryType.YOUTUBE_SEARCH: { const videos = await play_dl_1.default.search(query, { limit: 10, source: { youtube: "video" } }).catch(Util_1.Util.noop); if (!videos) return { playlist: null, tracks: [] }; const tracks = videos.map(m => { m.source = "youtube"; // eslint-disable-line @typescript-eslint/no-explicit-any return new Track_1.default(this, { title: m.title, description: m.description, author: m.channel?.name, url: m.url, requestedBy: options.requestedBy, thumbnail: Util_1.Util.last(m.thumbnails).url, views: m.views, duration: m.durationRaw, source: "youtube", raw: m }); }); return { playlist: null, tracks, searched: true }; } case types_1.QueryType.SOUNDCLOUD_TRACK: case types_1.QueryType.SOUNDCLOUD_SEARCH: { const result = await QueryResolver_1.QueryResolver.resolve(query) === types_1.QueryType.SOUNDCLOUD_TRACK ? [{ url: query }] : await play_dl_1.default.search(query, { limit: 5, source: { soundcloud: "tracks" } }).catch(() => []); if (!result || !result.length) return { playlist: null, tracks: [] }; const res = []; for (const r of result) { const trackInfo = await play_dl_1.default.soundcloud(r.url).catch(Util_1.Util.noop); if (!trackInfo) continue; const track = new Track_1.default(this, { title: trackInfo.name, url: trackInfo.url, duration: Util_1.Util.buildTimeCode(Util_1.Util.parseMS(trackInfo.durationInMs)), description: "", thumbnail: trackInfo.user.thumbnail, views: 0, author: trackInfo.user.name, requestedBy: options.requestedBy, source: "soundcloud", engine: trackInfo }); res.push(track); } return { playlist: null, tracks: res }; } case types_1.QueryType.SPOTIFY_SONG: { const spotifyData = await (0, spotify_url_info_1.default)(await Util_1.Util.getFetch()) .getData(query) .catch(Util_1.Util.noop); if (!spotifyData) return { playlist: null, tracks: [] }; const spotifyTrack = new Track_1.default(this, { title: spotifyData.name, description: spotifyData.description ?? "", author: spotifyData.artists[0]?.name ?? "Unknown Artist", url: spotifyData.external_urls?.spotify ?? query, thumbnail: spotifyData.album?.images[0]?.url ?? spotifyData.preview_url?.length ? `https://i.scdn.co/image/${spotifyData.preview_url?.split("?cid=")[1]}` : "https://www.scdn.co/i/_global/twitter_card-default.jpg", duration: Util_1.Util.buildTimeCode(Util_1.Util.parseMS(spotifyData.duration_ms)), views: 0, requestedBy: options.requestedBy, source: "spotify" }); return { playlist: null, tracks: [spotifyTrack] }; } case types_1.QueryType.SPOTIFY_PLAYLIST: case types_1.QueryType.SPOTIFY_ALBUM: { const spotifyPlaylist = await (0, spotify_url_info_1.default)(await Util_1.Util.getFetch()) .getData(query) .catch(Util_1.Util.noop); if (!spotifyPlaylist) return { playlist: null, tracks: [] }; const playlist = new Playlist_1.Playlist(this, { title: spotifyPlaylist.name ?? spotifyPlaylist.title, description: spotifyPlaylist.description ?? "", thumbnail: spotifyPlaylist.images[0]?.url ?? "https://www.scdn.co/i/_global/twitter_card-default.jpg", type: spotifyPlaylist.type, source: "spotify", author: spotifyPlaylist.type !== "playlist" ? { name: spotifyPlaylist.artists[0]?.name ?? "Unknown Artist", url: spotifyPlaylist.artists[0]?.external_urls?.spotify ?? null } : { name: spotifyPlaylist.owner?.display_name ?? spotifyPlaylist.owner?.id ?? "Unknown Artist", url: spotifyPlaylist.owner?.external_urls?.spotify ?? null }, tracks: [], id: spotifyPlaylist.id, url: spotifyPlaylist.external_urls?.spotify ?? query, rawPlaylist: spotifyPlaylist }); if (spotifyPlaylist.type !== "playlist") { // eslint-disable-next-line @typescript-eslint/no-explicit-any playlist.tracks = spotifyPlaylist.tracks.items.map((m) => { const data = new Track_1.default(this, { title: m.name ?? "", description: m.description ?? "", author: m.artists[0]?.name ?? "Unknown Artist", url: m.external_urls?.spotify ?? query, thumbnail: spotifyPlaylist.images[0]?.url ?? "https://www.scdn.co/i/_global/twitter_card-default.jpg", duration: Util_1.Util.buildTimeCode(Util_1.Util.parseMS(m.duration_ms)), views: 0, requestedBy: options.requestedBy, playlist, source: "spotify" }); return data; }); } else { // eslint-disable-next-line @typescript-eslint/no-explicit-any playlist.tracks = spotifyPlaylist.tracks.items.map((m) => { const data = new Track_1.default(this, { title: m.track.name ?? "", description: m.track.description ?? "", author: m.track.artists?.[0]?.name ?? "Unknown Artist", url: m.track.external_urls?.spotify ?? query, thumbnail: m.track.album?.images?.[0]?.url ?? "https://www.scdn.co/i/_global/twitter_card-default.jpg", duration: Util_1.Util.buildTimeCode(Util_1.Util.parseMS(m.track.duration_ms)), views: 0, requestedBy: options.requestedBy, playlist, source: "spotify" }); return data; }); } return { playlist: playlist, tracks: playlist.tracks }; } case types_1.QueryType.SOUNDCLOUD_PLAYLIST: { const data = await play_dl_1.default.soundcloud(query).catch(Util_1.Util.noop); if (!data) return { playlist: null, tracks: [] }; const res = new Playlist_1.Playlist(this, { title: data.name, description: "", thumbnail: "https://soundcloud.com/pwa-icon-192.png", type: "playlist", source: "soundcloud", author: { name: data.user.name ?? "Unknown Owner", url: data.user.url }, tracks: [], id: `${data.id}`, url: data.url, rawPlaylist: data }); const songs = await data.all_tracks(); for (const song of songs) { const track = new Track_1.default(this, { title: song.name, description: "", author: song.publisher.name ?? "Unknown Publisher", url: song.url, thumbnail: song.thumbnail, duration: Util_1.Util.buildTimeCode(Util_1.Util.parseMS(song.durationInMs)), views: 0, requestedBy: options.requestedBy, playlist: res, source: "soundcloud", engine: song }); res.tracks.push(track); } return { playlist: res, tracks: res.tracks }; } case types_1.QueryType.YOUTUBE_PLAYLIST: { const ytpl = await play_dl_1.default.playlist_info(query, { incomplete: true }).catch(Util_1.Util.noop); if (!ytpl) return { playlist: null, tracks: [] }; const playlist = new Playlist_1.Playlist(this, { title: ytpl.title, thumbnail: ytpl.thumbnail, description: "", type: "playlist", source: "youtube", author: { name: ytpl.channel.name, url: ytpl.channel.url }, tracks: [], id: ytpl.id, url: ytpl.url, rawPlaylist: ytpl }); const videos = await ytpl.all_videos(); playlist.tracks = videos.map(video => new Track_1.default(this, { title: video.title, description: video.description, author: video.channel?.name, url: video.url, requestedBy: options.requestedBy, thumbnail: Util_1.Util.last(video.thumbnails).url, views: video.views, duration: video.durationRaw, raw: video, playlist: playlist, source: "youtube" })); return { playlist: playlist, tracks: playlist.tracks }; } default: return { playlist: null, tracks: [] }; } } /** * Registers extractor * @param {string} extractorName The extractor name * @param {ExtractorModel|any} extractor The extractor object * @param {boolean} [force=false] Overwrite existing extractor with this name (if available) * @returns {ExtractorModel} */ // eslint-disable-next-line @typescript-eslint/no-explicit-any use(extractorName, extractor, force = false) { if (!extractorName) throw new PlayerError_1.PlayerError("Cannot use unknown extractor!", PlayerError_1.ErrorStatusCode.UNKNOWN_EXTRACTOR); if (this.extractors.has(extractorName) && !force) return this.extractors.get(extractorName); if (extractor instanceof ExtractorModel_1.ExtractorModel) { this.extractors.set(extractorName, extractor); return extractor; } for (const method of ["validate", "getInfo"]) { if (typeof extractor[method] !== "function") throw new PlayerError_1.PlayerError("Invalid extractor data!", PlayerError_1.ErrorStatusCode.INVALID_EXTRACTOR); } const model = new ExtractorModel_1.ExtractorModel(extractorName, extractor); this.extractors.set(model.name, model); return model; } /** * Removes registered extractor * @param {string} extractorName The extractor name * @returns {ExtractorModel} */ unuse(extractorName) { if (!this.extractors.has(extractorName)) throw new PlayerError_1.PlayerError(`Cannot find extractor "${extractorName}"`, PlayerError_1.ErrorStatusCode.UNKNOWN_EXTRACTOR); const prev = this.extractors.get(extractorName); this.extractors.delete(extractorName); return prev; } /** * Generates a report of the dependencies used by the `@discordjs/voice` module. Useful for debugging. * @returns {string} */ scanDeps() { const line = "-".repeat(50); const depsReport = (0, voice_1.generateDependencyReport)(); const extractorReport = this.extractors .map((m) => { return `${m.name} :: ${m.version || "0.1.0"}`; }) .join("\n"); return `${depsReport}\n${line}\nLoaded Extractors:\n${extractorReport || "None"}`; } emit(eventName, ...args) { if (this.requiredEvents.includes(eventName) && !super.eventNames().includes(eventName)) { // eslint-disable-next-line no-console console.error(...args); process.emitWarning(`[DiscordPlayerWarning] Unhandled "${eventName}" event! Events ${this.requiredEvents.map((m) => `"${m}"`).join(", ")} must have event listeners!`); return false; } else { return super.emit(eventName, ...args); } } /** * Resolves queue * @param {GuildResolvable|Queue} queueLike Queue like object * @returns {Queue} */ resolveQueue(queueLike) { return this.getQueue(queueLike instanceof Queue_1.Queue ? queueLike.guild : queueLike); } *[Symbol.iterator]() { yield* Array.from(this.queues.values()); } /** * Creates `Playlist` instance * @param data The data to initialize a playlist */ createPlaylist(data) { return new Playlist_1.Playlist(this, data); } } exports.Player = Player;