2021-04-09 18:05:32 +00:00
|
|
|
const { Track, AlbumDoesntExists } = require('./types/Track.js')
|
2021-04-09 16:45:57 +00:00
|
|
|
const { streamTrack, generateStreamURL } = require('./decryption.js')
|
2021-04-18 11:02:47 +00:00
|
|
|
const { tagID3, tagFLAC } = require('./tagger.js')
|
2021-04-14 16:10:31 +00:00
|
|
|
const { USER_AGENT_HEADER, pipeline } = require('./utils/index.js')
|
|
|
|
const { DEFAULTS, OverwriteOption } = require('./settings.js')
|
2021-04-13 16:40:34 +00:00
|
|
|
const { generatePath } = require('./utils/pathtemplates.js')
|
2021-04-14 16:10:31 +00:00
|
|
|
const { TrackFormats } = require('deezer-js')
|
2021-04-09 18:05:32 +00:00
|
|
|
const got = require('got')
|
2021-04-12 17:32:57 +00:00
|
|
|
const fs = require('fs')
|
2021-04-14 16:10:31 +00:00
|
|
|
const { tmpdir } = require('os')
|
2021-04-30 10:07:45 +00:00
|
|
|
const { queue } = require('async')
|
2021-04-09 16:45:57 +00:00
|
|
|
|
2021-04-13 16:40:34 +00:00
|
|
|
const extensions = {
|
2021-04-14 16:10:31 +00:00
|
|
|
[TrackFormats.FLAC]: '.flac',
|
|
|
|
[TrackFormats.LOCAL]: '.mp3',
|
|
|
|
[TrackFormats.MP3_320]: '.mp3',
|
|
|
|
[TrackFormats.MP3_128]: '.mp3',
|
|
|
|
[TrackFormats.DEFAULT]: '.mp3',
|
|
|
|
[TrackFormats.MP4_RA3]: '.mp4',
|
|
|
|
[TrackFormats.MP4_RA2]: '.mp4',
|
|
|
|
[TrackFormats.MP4_RA1]: '.mp4'
|
|
|
|
}
|
|
|
|
|
|
|
|
const TEMPDIR = tmpdir()+`/deemix-imgs`
|
|
|
|
fs.mkdirSync(TEMPDIR, { recursive: true })
|
|
|
|
|
|
|
|
async function downloadImage(url, path, overwrite){
|
2021-04-21 17:00:51 +00:00
|
|
|
if (fs.existsSync(path) && ![OverwriteOption.OVERWRITE, OverwriteOption.ONLY_TAGS, OverwriteOption.KEEP_BOTH].includes(overwrite)) return path
|
2021-04-14 16:10:31 +00:00
|
|
|
|
|
|
|
const downloadStream = got.stream(url, { headers: {'User-Agent': USER_AGENT_HEADER}, timeout: 30000})
|
|
|
|
const fileWriterStream = fs.createWriteStream(path)
|
|
|
|
|
|
|
|
await pipeline(downloadStream, fileWriterStream)
|
|
|
|
return path
|
2021-04-13 16:40:34 +00:00
|
|
|
}
|
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
async function getPreferredBitrate(track, bitrate, shouldFallback, uuid, listener){
|
2021-04-09 16:45:57 +00:00
|
|
|
bitrate = parseInt(bitrate)
|
|
|
|
if (track.localTrack) { return TrackFormats.LOCAL }
|
|
|
|
|
|
|
|
let falledBack = false
|
|
|
|
|
|
|
|
const formats_non_360 = {
|
2021-04-13 16:40:34 +00:00
|
|
|
[TrackFormats.FLAC]: "FLAC",
|
|
|
|
[TrackFormats.MP3_320]: "MP3_320",
|
|
|
|
[TrackFormats.MP3_128]: "MP3_128"
|
2021-04-09 16:45:57 +00:00
|
|
|
}
|
|
|
|
const formats_360 = {
|
2021-04-13 16:40:34 +00:00
|
|
|
[TrackFormats.MP4_RA3]: "MP4_RA3",
|
|
|
|
[TrackFormats.MP4_RA2]: "MP4_RA2",
|
|
|
|
[TrackFormats.MP4_RA1]: "MP4_RA1"
|
2021-04-09 16:45:57 +00:00
|
|
|
}
|
|
|
|
|
2021-04-21 17:00:51 +00:00
|
|
|
const is360Format = Object.keys(formats_360).includes(bitrate)
|
2021-04-09 16:45:57 +00:00
|
|
|
let formats
|
|
|
|
if (!shouldFallback){
|
|
|
|
formats = {...formats_360, ...formats_non_360}
|
|
|
|
}else if (is360Format){
|
|
|
|
formats = {...formats_360}
|
|
|
|
}else{
|
|
|
|
formats = {...formats_non_360}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (let i = 0; i < Object.keys(formats).length; i++){
|
2021-04-18 11:02:47 +00:00
|
|
|
let formatNumber = Object.keys(formats).reverse()[i]
|
2021-04-13 16:40:34 +00:00
|
|
|
let formatName = formats[formatNumber]
|
2021-04-09 16:45:57 +00:00
|
|
|
|
2021-04-09 21:05:24 +00:00
|
|
|
if (formatNumber > bitrate) { continue }
|
2021-04-21 17:00:51 +00:00
|
|
|
if (Object.keys(track.filesizes).includes(`FILESIZE_${formatName}`)){
|
2021-04-09 16:45:57 +00:00
|
|
|
if (parseInt(track.filesizes[`FILESIZE_${formatName}`]) != 0) return formatNumber
|
|
|
|
if (!track.filesizes[`FILESIZE_${formatName}_TESTED`]){
|
2021-04-09 21:05:24 +00:00
|
|
|
let request
|
2021-04-09 18:05:32 +00:00
|
|
|
try {
|
2021-04-09 21:05:24 +00:00
|
|
|
request = got.get(
|
2021-04-09 18:05:32 +00:00
|
|
|
generateStreamURL(track.id, track.MD5, track.mediaVersion, formatNumber),
|
|
|
|
{ headers: {'User-Agent': USER_AGENT_HEADER}, timeout: 30000 }
|
|
|
|
).on("response", (response)=>{
|
2021-04-09 21:05:24 +00:00
|
|
|
track.filesizes[`FILESIZE_${formatName}`] = response.headers["content-length"]
|
2021-04-09 18:05:32 +00:00
|
|
|
track.filesizes[`FILESIZE_${formatName}_TESTED`] = true
|
|
|
|
request.cancel()
|
|
|
|
}).on("error", (e)=>{
|
|
|
|
throw e
|
|
|
|
})
|
|
|
|
|
2021-04-09 21:05:24 +00:00
|
|
|
await request
|
|
|
|
} catch (e){
|
|
|
|
if (e.isCanceled) { return formatNumber }
|
|
|
|
console.error(e)
|
|
|
|
throw e
|
|
|
|
}
|
2021-04-09 16:45:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!shouldFallback){
|
|
|
|
throw new PreferredBitrateNotFound
|
|
|
|
}else if (!falledBack){
|
|
|
|
falledBack = true
|
|
|
|
if (listener && uuid){
|
|
|
|
listener.send("queueUpdate", {
|
|
|
|
uuid,
|
|
|
|
bitrateFallback: true,
|
|
|
|
data:{
|
|
|
|
id: track.id,
|
|
|
|
title: track.title,
|
|
|
|
artist: track.mainArtist.name
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
}
|
2021-04-09 16:45:57 +00:00
|
|
|
if (is360Format) throw new TrackNot360
|
|
|
|
return TrackFormats.DEFAULT
|
|
|
|
}
|
2021-04-08 17:37:31 +00:00
|
|
|
|
|
|
|
class Downloader {
|
|
|
|
constructor(dz, downloadObject, settings, listener){
|
|
|
|
this.dz = dz
|
|
|
|
this.downloadObject = downloadObject
|
2021-04-13 16:40:34 +00:00
|
|
|
this.settings = settings || DEFAULTS
|
2021-04-08 17:37:31 +00:00
|
|
|
this.bitrate = downloadObject.bitrate
|
|
|
|
this.listener = listener
|
|
|
|
|
|
|
|
this.extrasPath = null
|
|
|
|
this.playlistCoverName = null
|
|
|
|
this.playlistURLs = []
|
|
|
|
}
|
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
async start(){
|
2021-04-09 16:45:57 +00:00
|
|
|
if (this.downloadObject.__type__ === "Single"){
|
2021-04-29 19:14:38 +00:00
|
|
|
await this.downloadWrapper({
|
2021-04-09 16:45:57 +00:00
|
|
|
trackAPI_gw: this.downloadObject.single.trackAPI_gw,
|
|
|
|
trackAPI: this.downloadObject.single.trackAPI,
|
|
|
|
albumAPI: this.downloadObject.single.albumAPI
|
|
|
|
})
|
|
|
|
} else if (this.downloadObject.__type__ === "Collection") {
|
|
|
|
let tracks = []
|
2021-04-30 10:07:45 +00:00
|
|
|
let q = queue(async (data) => {
|
|
|
|
let {track, pos} = data
|
2021-04-29 19:14:38 +00:00
|
|
|
tracks[pos] = await this.downloadWrapper({
|
2021-04-09 16:45:57 +00:00
|
|
|
trackAPI_gw: track,
|
|
|
|
albumAPI: this.downloadObject.collection.albumAPI,
|
|
|
|
playlistAPI: this.downloadObject.collection.playlistAPI
|
|
|
|
})
|
2021-04-30 10:07:45 +00:00
|
|
|
}, this.settings.queueConcurrency)
|
|
|
|
for (let pos = 0; pos < this.downloadObject.collection.tracks_gw.length; pos++){
|
|
|
|
let track = this.downloadObject.collection.tracks_gw[pos]
|
|
|
|
q.push({track, pos})
|
2021-04-27 20:24:57 +00:00
|
|
|
}
|
2021-04-30 10:07:45 +00:00
|
|
|
await q.drain()
|
2021-04-08 17:37:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (this.listener) this.listener.send("finishedDownload", this.downloadObject.uuid)
|
|
|
|
}
|
|
|
|
|
2021-04-09 16:45:57 +00:00
|
|
|
async download(extraData, track){
|
|
|
|
const { trackAPI_gw, trackAPI, albumAPI, playlistAPI } = extraData
|
2021-04-08 17:37:31 +00:00
|
|
|
if (trackAPI_gw.SNG_ID == "0") throw new DownloadFailed("notOnDeezer")
|
|
|
|
|
2021-04-29 19:14:38 +00:00
|
|
|
let itemName = `[${trackAPI_gw.ART_NAME} - ${trackAPI_gw.SNG_TITLE.trim()}]`
|
|
|
|
|
2021-04-08 17:37:31 +00:00
|
|
|
// Generate track object
|
|
|
|
if (!track){
|
2021-04-09 21:05:24 +00:00
|
|
|
track = new Track()
|
2021-04-29 19:14:38 +00:00
|
|
|
console.log(`${itemName} Getting tags`)
|
2021-04-09 18:05:32 +00:00
|
|
|
try{
|
|
|
|
await track.parseData(
|
|
|
|
this.dz,
|
2021-04-09 21:05:24 +00:00
|
|
|
trackAPI_gw.SNG_ID,
|
2021-04-09 18:05:32 +00:00
|
|
|
trackAPI_gw,
|
|
|
|
trackAPI,
|
2021-04-27 20:24:57 +00:00
|
|
|
null, // albumAPI_gw
|
2021-04-09 18:05:32 +00:00
|
|
|
albumAPI,
|
|
|
|
playlistAPI
|
|
|
|
)
|
|
|
|
} catch (e){
|
2021-04-29 19:14:38 +00:00
|
|
|
if (e instanceof AlbumDoesntExists) { throw new DownloadFailed('albumDoesntExists') }
|
2021-04-09 18:05:32 +00:00
|
|
|
console.error(e)
|
|
|
|
throw e
|
|
|
|
}
|
2021-04-08 17:37:31 +00:00
|
|
|
}
|
|
|
|
|
2021-04-29 19:14:38 +00:00
|
|
|
itemName = `[${track.mainArtist.name} - ${track.title}]`
|
|
|
|
|
2021-04-08 17:37:31 +00:00
|
|
|
// Check if the track is encoded
|
|
|
|
if (track.MD5 === "") throw new DownloadFailed("notEncoded", track)
|
|
|
|
|
|
|
|
// Check the target bitrate
|
2021-04-29 19:14:38 +00:00
|
|
|
console.log(`${itemName} Getting bitrate`)
|
2021-04-09 18:05:32 +00:00
|
|
|
let selectedFormat
|
|
|
|
try{
|
2021-04-09 21:05:24 +00:00
|
|
|
selectedFormat = await getPreferredBitrate(
|
2021-04-09 18:05:32 +00:00
|
|
|
track,
|
|
|
|
this.bitrate,
|
2021-04-09 21:05:24 +00:00
|
|
|
true, // fallbackBitrate
|
2021-04-09 18:05:32 +00:00
|
|
|
this.downloadObject.uuid, this.listener
|
|
|
|
)
|
|
|
|
}catch (e){
|
|
|
|
if (e instanceof PreferredBitrateNotFound) { throw new DownloadFailed("wrongBitrate", track) }
|
|
|
|
if (e instanceof TrackNot360) { throw new DownloadFailed("no360RA") }
|
|
|
|
console.error(e)
|
|
|
|
throw e
|
|
|
|
}
|
2021-04-08 17:37:31 +00:00
|
|
|
track.bitrate = selectedFormat
|
|
|
|
track.album.bitrate = selectedFormat
|
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
// Apply Settings
|
2021-04-13 16:40:34 +00:00
|
|
|
track.applySettings(this.settings)
|
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
// Generate filename and filepath from metadata
|
2021-04-13 16:40:34 +00:00
|
|
|
let {
|
|
|
|
filename,
|
|
|
|
filepath,
|
|
|
|
artistPath,
|
|
|
|
coverPath,
|
|
|
|
extrasPath
|
|
|
|
} = generatePath(track, this.downloadObject, this.settings)
|
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
// Make sure the filepath exsists
|
2021-04-13 16:40:34 +00:00
|
|
|
fs.mkdirSync(filepath, { recursive: true })
|
2021-04-14 15:11:56 +00:00
|
|
|
let extension = extensions[track.bitrate]
|
|
|
|
let writepath = `${filepath}/${filename}${extension}`
|
2021-04-13 16:40:34 +00:00
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
// Save extrasPath
|
2021-04-13 16:40:34 +00:00
|
|
|
if (extrasPath && !this.extrasPath) this.extrasPath = extrasPath
|
|
|
|
|
|
|
|
// Generate covers URLs
|
2021-04-14 16:10:31 +00:00
|
|
|
let embeddedImageFormat = `jpg-${this.settings.jpegImageQuality}`
|
|
|
|
if (this.settings.embeddedArtworkPNG) embeddedImageFormat = 'png'
|
|
|
|
|
|
|
|
track.album.embeddedCoverURL = track.album.pic.getURL(this.settings.embeddedArtworkSize, embeddedImageFormat)
|
|
|
|
let ext = track.album.embeddedCoverURL.slice(-4)
|
|
|
|
if (ext.charAt(0) != '.') ext = '.jpg'
|
|
|
|
track.album.embeddedCoverPath = `${TEMPDIR}/${track.album.isPlaylist ? 'pl'+track.playlist.id : 'alb'+track.album.id}_${this.settings.embeddedArtworkSize}${ext}`
|
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
// Download and cache the coverart
|
2021-04-14 16:10:31 +00:00
|
|
|
track.album.embeddedCoverPath = await downloadImage(track.album.embeddedCoverURL, track.album.embeddedCoverPath)
|
2021-04-29 19:14:38 +00:00
|
|
|
console.log(`${itemName} Albumart downloaded`)
|
2021-04-14 16:10:31 +00:00
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
// Save local album art
|
|
|
|
// Save artist art
|
|
|
|
// Save playlist art
|
|
|
|
// Save lyrics in lrc file
|
|
|
|
// Check for overwrite settings
|
|
|
|
|
2021-04-08 17:37:31 +00:00
|
|
|
// Download the track
|
2021-04-29 19:14:38 +00:00
|
|
|
console.log(`${itemName} Downloading`)
|
2021-04-09 16:45:57 +00:00
|
|
|
track.downloadURL = generateStreamURL(track.id, track.MD5, track.mediaVersion, track.bitrate)
|
2021-04-13 16:40:34 +00:00
|
|
|
let stream = fs.createWriteStream(writepath)
|
2021-04-29 19:14:38 +00:00
|
|
|
try {
|
|
|
|
await streamTrack(stream, track, 0, this.downloadObject, this.listener)
|
|
|
|
} catch (e){
|
|
|
|
fs.unlinkSync(writepath)
|
|
|
|
if (e instanceof got.HTTPError) throw new DownloadFailed('notAvailable', track)
|
|
|
|
throw e
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
console.log(`${itemName} Tagging file`)
|
2021-04-09 18:05:32 +00:00
|
|
|
// Adding tags
|
2021-04-14 15:11:56 +00:00
|
|
|
if (extension == '.mp3'){
|
|
|
|
tagID3(writepath, track, this.settings.tags)
|
2021-04-18 11:02:47 +00:00
|
|
|
} else if (extension == '.flac'){
|
|
|
|
tagFLAC(writepath, track, this.settings.tags)
|
2021-04-14 15:11:56 +00:00
|
|
|
}
|
2021-04-29 19:14:38 +00:00
|
|
|
|
|
|
|
return {}
|
|
|
|
}
|
|
|
|
|
|
|
|
async downloadWrapper(extraData, track){
|
|
|
|
const { trackAPI_gw } = extraData
|
|
|
|
// Temp metadata to generate logs
|
|
|
|
let tempTrack = {
|
|
|
|
id: trackAPI_gw.SNG_ID,
|
|
|
|
title: trackAPI_gw.SNG_TITLE.trim(),
|
|
|
|
artist: trackAPI_gw.ART_NAME
|
|
|
|
}
|
|
|
|
if (trackAPI_gw.VERSION && trackAPI_gw.SNG_TITLE.includes(trackAPI_gw.VERSION))
|
|
|
|
tempTrack.title += ` ${trackAPI_gw.VERSION.trim()}`
|
|
|
|
|
|
|
|
let itemName = `[${tempTrack.artist} - ${tempTrack.title}]`
|
|
|
|
let result
|
|
|
|
try {
|
|
|
|
result = await this.download(extraData, track)
|
|
|
|
} catch (e){
|
|
|
|
if (e instanceof DownloadFailed){
|
|
|
|
if (e.track){
|
|
|
|
let track = e.track
|
|
|
|
if (track.fallbackID != 0){
|
|
|
|
console.warn(`${itemName} ${e.message} Using fallback id.`)
|
|
|
|
let newTrack = await this.dz.gw.get_track_with_fallback(track.fallbackID)
|
|
|
|
track.parseEssentialData(newTrack)
|
|
|
|
track.retriveFilesizes(this.dz)
|
|
|
|
return await this.downloadWrapper(extraData, track)
|
|
|
|
}
|
|
|
|
if (!track.searched && this.settings.fallbackSearch){
|
|
|
|
console.warn(`${itemName} ${e.message} Searching for alternative.`)
|
|
|
|
let searchedID = this.dz.api.get_track_id_from_metadata(track.mainArtist.name, track.title, track.album.title)
|
|
|
|
if (searchedID != "0"){
|
|
|
|
let newTrack = await this.dz.gw.get_track_with_fallback(track.fallbackID)
|
|
|
|
track.parseEssentialData(newTrack)
|
|
|
|
track.retriveFilesizes(this.dz)
|
|
|
|
track.searched = true
|
|
|
|
if (this.listener) this.listener.send('queueUpdate', {
|
|
|
|
uuid: this.downloadObject.uuid,
|
|
|
|
searchFallback: true,
|
|
|
|
data: {
|
|
|
|
id: track.id,
|
|
|
|
title: track.title,
|
|
|
|
artist: track.mainArtist.name
|
|
|
|
}
|
|
|
|
})
|
|
|
|
return await this.downloadWrapper(extraData, track)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
e.errid += "NoAlternative"
|
|
|
|
e.message = errorMessages[e.errid]
|
|
|
|
}
|
|
|
|
console.error(`${itemName} ${e.message}`)
|
|
|
|
result = {error:{
|
|
|
|
message: e.message,
|
|
|
|
errid: e.errid,
|
|
|
|
data: tempTrack
|
|
|
|
}}
|
|
|
|
} else {
|
|
|
|
console.error(`${itemName} ${e.message}`)
|
|
|
|
result = {error:{
|
|
|
|
message: e.message,
|
|
|
|
data: tempTrack
|
|
|
|
}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (result.error){
|
|
|
|
this.downloadObject.completeTrackProgress(this.interface)
|
|
|
|
this.downloadObject.failed += 1
|
|
|
|
this.downloadObject.errors.push(result.error)
|
|
|
|
if (this.interface){
|
|
|
|
let error = result.error
|
|
|
|
this.interface.send("updateQueue", {
|
|
|
|
uuid: this.downloadObject.uuid,
|
|
|
|
failed: true,
|
|
|
|
data: error.data,
|
|
|
|
error: error.message,
|
|
|
|
errid: error.errid || null
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result
|
2021-04-09 16:45:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class DownloadError extends Error {
|
2021-04-29 19:14:38 +00:00
|
|
|
constructor() {
|
|
|
|
super()
|
2021-04-09 16:45:57 +00:00
|
|
|
this.name = "DownloadError"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-29 19:14:38 +00:00
|
|
|
const errorMessages = {
|
|
|
|
notOnDeezer: "Track not available on Deezer!",
|
|
|
|
notEncoded: "Track not yet encoded!",
|
|
|
|
notEncodedNoAlternative: "Track not yet encoded and no alternative found!",
|
|
|
|
wrongBitrate: "Track not found at desired bitrate.",
|
|
|
|
wrongBitrateNoAlternative: "Track not found at desired bitrate and no alternative found!",
|
|
|
|
no360RA: "Track is not available in Reality Audio 360.",
|
|
|
|
notAvailable: "Track not available on deezer's servers!",
|
|
|
|
notAvailableNoAlternative: "Track not available on deezer's servers and no alternative found!",
|
|
|
|
noSpaceLeft: "No space left on target drive, clean up some space for the tracks.",
|
|
|
|
albumDoesntExists: "Track's album does not exsist, failed to gather info."
|
|
|
|
}
|
|
|
|
|
2021-04-09 16:45:57 +00:00
|
|
|
class DownloadFailed extends DownloadError {
|
|
|
|
constructor(errid, track) {
|
2021-04-29 19:14:38 +00:00
|
|
|
super()
|
|
|
|
this.errid = errid
|
|
|
|
this.message = errorMessages[errid]
|
|
|
|
this.name = "DownloadFailed"
|
2021-04-09 16:45:57 +00:00
|
|
|
this.track = track
|
2021-04-08 17:37:31 +00:00
|
|
|
}
|
|
|
|
}
|
2021-04-09 16:45:57 +00:00
|
|
|
|
|
|
|
class TrackNot360 extends Error {
|
2021-04-29 19:14:38 +00:00
|
|
|
constructor() {
|
|
|
|
super()
|
2021-04-09 16:45:57 +00:00
|
|
|
this.name = "TrackNot360"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class PreferredBitrateNotFound extends Error {
|
2021-04-29 19:14:38 +00:00
|
|
|
constructor() {
|
|
|
|
super()
|
2021-04-09 16:45:57 +00:00
|
|
|
this.name = "PreferredBitrateNotFound"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
Downloader,
|
|
|
|
DownloadError,
|
2021-04-09 18:05:32 +00:00
|
|
|
DownloadFailed,
|
|
|
|
getPreferredBitrate,
|
|
|
|
TrackNot360,
|
|
|
|
PreferredBitrateNotFound
|
2021-04-09 16:45:57 +00:00
|
|
|
}
|