2021-05-29 08:43:09 +00:00
|
|
|
const { Track } = require('./types/Track.js')
|
2021-05-19 21:03:47 +00:00
|
|
|
const { StaticPicture } = require('./types/Picture.js')
|
2021-08-02 21:53:56 +00:00
|
|
|
const { streamTrack, generateCryptedStreamURL } = require('./decryption.js')
|
2021-05-28 10:00:40 +00:00
|
|
|
const { tagID3, tagID3v1, tagFLAC } = require('./tagger.js')
|
2021-05-31 21:25:53 +00:00
|
|
|
const { USER_AGENT_HEADER, pipeline, shellEscape } = require('./utils/index.js')
|
2021-04-14 16:10:31 +00:00
|
|
|
const { DEFAULTS, OverwriteOption } = require('./settings.js')
|
2021-05-19 21:03:47 +00:00
|
|
|
const { generatePath, generateAlbumName, generateArtistName, generateDownloadObjectName } = require('./utils/pathtemplates.js')
|
2021-08-02 21:53:56 +00:00
|
|
|
const { PreferredBitrateNotFound, TrackNot360, DownloadFailed, ErrorMessages, DownloadCanceled} = require('./errors.js')
|
2021-04-14 16:10:31 +00:00
|
|
|
const { TrackFormats } = require('deezer-js')
|
2021-07-28 10:43:17 +00:00
|
|
|
const { WrongLicense, WrongGeolocation } = require('deezer-js').errors
|
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-06-07 18:14:41 +00:00
|
|
|
const { queue, each } = require('async')
|
2021-05-31 21:25:53 +00:00
|
|
|
const { exec } = require("child_process")
|
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'
|
|
|
|
}
|
|
|
|
|
2021-07-25 09:33:16 +00:00
|
|
|
const formatsName = {
|
|
|
|
[TrackFormats.FLAC]: 'FLAC',
|
|
|
|
[TrackFormats.LOCAL]: 'MP3_MISC',
|
|
|
|
[TrackFormats.MP3_320]: 'MP3_320',
|
|
|
|
[TrackFormats.MP3_128]: 'MP3_128',
|
|
|
|
[TrackFormats.DEFAULT]: 'MP3_MISC',
|
|
|
|
[TrackFormats.MP4_RA3]: 'MP4_RA3',
|
|
|
|
[TrackFormats.MP4_RA2]: 'MP4_RA2',
|
|
|
|
[TrackFormats.MP4_RA1]: 'MP4_RA1'
|
|
|
|
}
|
|
|
|
|
2021-04-14 16:10:31 +00:00
|
|
|
const TEMPDIR = tmpdir()+`/deemix-imgs`
|
|
|
|
fs.mkdirSync(TEMPDIR, { recursive: true })
|
|
|
|
|
2021-06-07 18:14:41 +00:00
|
|
|
async function downloadImage(url, path, overwrite = OverwriteOption.DONT_OVERWRITE){
|
2021-06-17 18:05:13 +00:00
|
|
|
if (fs.existsSync(path) && ![OverwriteOption.OVERWRITE, OverwriteOption.ONLY_TAGS, OverwriteOption.KEEP_BOTH].includes(overwrite)){
|
|
|
|
let file = fs.readFileSync(path)
|
|
|
|
if (file.length != 0) return path
|
|
|
|
fs.unlinkSync(path)
|
|
|
|
}
|
2021-04-14 16:10:31 +00:00
|
|
|
|
2021-06-17 18:05:13 +00:00
|
|
|
const downloadStream = got.stream(url, { headers: {'User-Agent': USER_AGENT_HEADER}, timeout: 30000, retry: 3})
|
2021-04-14 16:10:31 +00:00
|
|
|
const fileWriterStream = fs.createWriteStream(path)
|
|
|
|
|
2021-05-29 18:22:20 +00:00
|
|
|
try {
|
|
|
|
await pipeline(downloadStream, fileWriterStream)
|
|
|
|
} catch (e){
|
|
|
|
fs.unlinkSync(path)
|
|
|
|
if (e instanceof got.HTTPError) {
|
|
|
|
if (url.includes('images.dzcdn.net')){
|
|
|
|
let urlBase = url.slice(0, url.lastIndexOf('/')+1)
|
|
|
|
let pictureURL = url.slice(urlBase.length)
|
|
|
|
let pictureSize = parseInt(pictureURL.slice(0, pictureURL.indexOf('x')))
|
|
|
|
if (pictureSize > 1200)
|
2021-06-07 18:14:41 +00:00
|
|
|
return downloadImage(urlBase+pictureURL.replace(`${pictureSize}x${pictureSize}`, '1200x1200'), path, overwrite)
|
2021-05-29 18:22:20 +00:00
|
|
|
}
|
|
|
|
return null
|
|
|
|
}
|
|
|
|
if (e instanceof got.TimeoutError) {
|
|
|
|
return downloadImage(url, path, overwrite)
|
|
|
|
}
|
|
|
|
console.trace(e)
|
|
|
|
throw e
|
|
|
|
}
|
2021-04-14 16:10:31 +00:00
|
|
|
return path
|
2021-04-13 16:40:34 +00:00
|
|
|
}
|
|
|
|
|
2021-07-28 10:43:17 +00:00
|
|
|
async function getPreferredBitrate(dz, track, preferredBitrate, shouldFallback, uuid, listener){
|
|
|
|
preferredBitrate = parseInt(preferredBitrate)
|
|
|
|
if (track.localTrack) { return TrackFormats.LOCAL}
|
2021-04-09 16:45:57 +00:00
|
|
|
|
|
|
|
let falledBack = false
|
2021-07-28 11:07:24 +00:00
|
|
|
let hasAlternative = track.fallbackID != 0
|
2021-07-28 10:43:17 +00:00
|
|
|
let isGeolocked = false
|
|
|
|
let wrongLicense = false
|
2021-04-09 16:45:57 +00:00
|
|
|
|
|
|
|
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-07-28 10:43:17 +00:00
|
|
|
const is360Format = Object.keys(formats_360).includes(preferredBitrate)
|
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}
|
|
|
|
}
|
|
|
|
|
2021-08-02 21:08:06 +00:00
|
|
|
async function testURL(track, url, formatName){
|
2021-06-01 18:11:02 +00:00
|
|
|
let request
|
2021-07-28 10:43:17 +00:00
|
|
|
try{
|
2021-06-01 18:11:02 +00:00
|
|
|
request = got.get(
|
2021-07-28 10:43:17 +00:00
|
|
|
url,
|
2021-06-01 18:11:02 +00:00
|
|
|
{ headers: {'User-Agent': USER_AGENT_HEADER}, timeout: 30000 }
|
|
|
|
).on("response", (response)=>{
|
2021-07-28 12:20:43 +00:00
|
|
|
track.filesizes[`FILESIZE_${formatName}`] = response.statusCode == 403 ? 0 : response.headers["content-length"]
|
2021-06-01 18:11:02 +00:00
|
|
|
track.filesizes[`FILESIZE_${formatName}_TESTED`] = true
|
|
|
|
request.cancel()
|
|
|
|
})
|
|
|
|
|
|
|
|
await request
|
|
|
|
} catch (e){
|
2021-06-13 12:01:09 +00:00
|
|
|
if (e.isCanceled) {
|
2021-07-28 10:43:17 +00:00
|
|
|
if (track.filesizes[`FILESIZE_${formatName}`] == 0) return false
|
|
|
|
return true
|
2021-06-13 12:01:09 +00:00
|
|
|
}
|
2021-06-01 18:11:02 +00:00
|
|
|
if (e instanceof got.ReadError || e instanceof got.TimeoutError){
|
2021-08-02 21:08:06 +00:00
|
|
|
return await testURL(track, url, formatName)
|
2021-06-01 18:11:02 +00:00
|
|
|
}
|
2021-07-28 10:43:17 +00:00
|
|
|
if (e instanceof got.HTTPError) return false
|
2021-06-01 18:11:02 +00:00
|
|
|
console.trace(e)
|
|
|
|
throw e
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-28 10:43:17 +00:00
|
|
|
async function getCorrectURL(track, formatName, formatNumber){
|
|
|
|
// Check the track with the legit method
|
|
|
|
let url
|
|
|
|
try {
|
|
|
|
url = await dz.get_track_url(track.trackToken, formatName)
|
2021-08-02 21:08:06 +00:00
|
|
|
if (await testURL(track, url, formatName, formatNumber)) return url
|
2021-07-28 12:20:43 +00:00
|
|
|
url = undefined
|
2021-07-28 10:43:17 +00:00
|
|
|
} catch (e){
|
|
|
|
wrongLicense = (e.name === "WrongLicense")
|
|
|
|
isGeolocked = (e.name === "WrongGeolocation")
|
|
|
|
}
|
|
|
|
// Fallback to old method
|
|
|
|
if (!url){
|
|
|
|
url = generateCryptedStreamURL(track.id, track.MD5, track.mediaVersion, formatNumber)
|
2021-08-02 21:08:06 +00:00
|
|
|
if (await testURL(track, url, formatName, formatNumber)) return url
|
2021-07-28 10:43:17 +00:00
|
|
|
url = undefined
|
|
|
|
}
|
|
|
|
return url
|
|
|
|
}
|
|
|
|
|
2021-04-09 16:45:57 +00:00
|
|
|
for (let i = 0; i < Object.keys(formats).length; i++){
|
2021-07-28 10:43:17 +00:00
|
|
|
// Check bitrates
|
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-07-28 10:43:17 +00:00
|
|
|
// Current bitrate is higher than preferred bitrate; skip
|
|
|
|
if (formatNumber > preferredBitrate) { continue }
|
|
|
|
|
|
|
|
let currentTrack = track
|
2021-07-28 11:07:24 +00:00
|
|
|
let url = await getCorrectURL(currentTrack, formatName, formatNumber)
|
2021-07-28 10:43:17 +00:00
|
|
|
let newTrack
|
|
|
|
do {
|
2021-07-28 11:07:24 +00:00
|
|
|
if (!url && hasAlternative){
|
2021-07-28 10:43:17 +00:00
|
|
|
newTrack = await dz.gw.get_track_with_fallback(currentTrack.fallbackID)
|
|
|
|
currentTrack = new Track()
|
|
|
|
currentTrack.parseEssentialData(newTrack)
|
2021-07-28 11:07:24 +00:00
|
|
|
hasAlternative = currentTrack.fallbackID != 0
|
2021-04-09 16:45:57 +00:00
|
|
|
}
|
2021-07-28 12:20:43 +00:00
|
|
|
if (!url) url = await getCorrectURL(currentTrack, formatName, formatNumber)
|
2021-07-28 10:43:17 +00:00
|
|
|
} while (!url && hasAlternative)
|
|
|
|
|
|
|
|
if (url) {
|
|
|
|
if (newTrack) track.parseEssentialData(newTrack)
|
|
|
|
track.urls[formatName] = url
|
|
|
|
return formatNumber
|
2021-04-09 16:45:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!shouldFallback){
|
2021-07-28 10:43:17 +00:00
|
|
|
if (wrongLicense) throw new WrongLicense(formatName)
|
|
|
|
if (isGeolocked) throw new WrongGeolocation(dz.current_user.country)
|
2021-04-09 16:45:57 +00:00
|
|
|
throw new PreferredBitrateNotFound
|
2021-07-28 10:43:17 +00:00
|
|
|
} else if (!falledBack){
|
2021-04-09 16:45:57 +00:00
|
|
|
falledBack = true
|
|
|
|
if (listener && uuid){
|
2021-08-02 21:45:19 +00:00
|
|
|
listener.send("downloadInfo", {
|
2021-04-09 16:45:57 +00:00
|
|
|
uuid,
|
2021-08-02 21:45:19 +00:00
|
|
|
state: "bitrateFallback",
|
2021-04-09 16:45:57 +00:00
|
|
|
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
|
|
|
|
|
2021-05-19 21:03:47 +00:00
|
|
|
this.playlistCovername = null
|
2021-04-08 17:37:31 +00:00
|
|
|
this.playlistURLs = []
|
2021-06-19 10:03:18 +00:00
|
|
|
|
|
|
|
this.coverQueue = {}
|
2021-04-08 17:37:31 +00:00
|
|
|
}
|
|
|
|
|
2021-06-07 18:14:41 +00:00
|
|
|
log(data, state){
|
2021-06-01 18:11:02 +00:00
|
|
|
if (this.listener)
|
2021-06-07 18:14:41 +00:00
|
|
|
this.listener.send('downloadInfo', { uuid: this.downloadObject.uuid, data, state })
|
2021-06-01 18:11:02 +00:00
|
|
|
}
|
|
|
|
|
2021-06-07 18:14:41 +00:00
|
|
|
warn(data, state, solution){
|
2021-06-01 18:11:02 +00:00
|
|
|
if (this.listener)
|
2021-06-07 18:14:41 +00:00
|
|
|
this.listener.send('downloadWarn', { uuid: this.downloadObject.uuid, data, state , solution })
|
2021-06-01 18:11:02 +00:00
|
|
|
}
|
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
async start(){
|
2021-06-08 09:11:10 +00:00
|
|
|
if (!this.downloadObject.isCanceled){
|
|
|
|
if (this.downloadObject.__type__ === "Single"){
|
|
|
|
let track = await this.downloadWrapper({
|
|
|
|
trackAPI_gw: this.downloadObject.single.trackAPI_gw,
|
|
|
|
trackAPI: this.downloadObject.single.trackAPI,
|
|
|
|
albumAPI: this.downloadObject.single.albumAPI
|
|
|
|
})
|
|
|
|
if (track) await this.afterDownloadSingle(track)
|
|
|
|
} else if (this.downloadObject.__type__ === "Collection") {
|
|
|
|
let tracks = []
|
|
|
|
|
|
|
|
let q = queue(async (data) => {
|
|
|
|
let {track, pos} = data
|
|
|
|
tracks[pos] = await this.downloadWrapper({
|
|
|
|
trackAPI_gw: track,
|
|
|
|
albumAPI: this.downloadObject.collection.albumAPI,
|
|
|
|
playlistAPI: this.downloadObject.collection.playlistAPI
|
|
|
|
})
|
|
|
|
}, this.settings.queueConcurrency)
|
|
|
|
|
2021-07-27 18:36:55 +00:00
|
|
|
if (this.downloadObject.collection.tracks_gw.length){
|
|
|
|
this.downloadObject.collection.tracks_gw.forEach((track, pos) => {
|
|
|
|
q.push({track, pos})
|
|
|
|
})
|
2021-06-07 18:14:41 +00:00
|
|
|
|
2021-07-27 18:36:55 +00:00
|
|
|
await q.drain()
|
|
|
|
}
|
2021-06-08 09:11:10 +00:00
|
|
|
await this.afterDownloadCollection(tracks)
|
|
|
|
}
|
2021-04-08 17:37:31 +00:00
|
|
|
}
|
|
|
|
|
2021-05-16 18:01:56 +00:00
|
|
|
if (this.listener){
|
2021-06-08 09:11:10 +00:00
|
|
|
if (this.downloadObject.isCanceled){
|
|
|
|
this.listener.send('currentItemCancelled', this.downloadObject.uuid)
|
|
|
|
this.listener.send("removedFromQueue", this.downloadObject.uuid)
|
|
|
|
} else {
|
|
|
|
this.listener.send("finishDownload", this.downloadObject.uuid)
|
|
|
|
}
|
2021-05-16 18:01:56 +00:00
|
|
|
}
|
2021-04-08 17:37:31 +00:00
|
|
|
}
|
|
|
|
|
2021-04-09 16:45:57 +00:00
|
|
|
async download(extraData, track){
|
2021-05-19 21:03:47 +00:00
|
|
|
let returnData = {}
|
2021-04-09 16:45:57 +00:00
|
|
|
const { trackAPI_gw, trackAPI, albumAPI, playlistAPI } = extraData
|
2021-06-08 16:53:33 +00:00
|
|
|
trackAPI_gw.SIZE = this.downloadObject.size
|
2021-05-16 18:01:56 +00:00
|
|
|
if (this.downloadObject.isCanceled) throw new DownloadCanceled
|
2021-04-08 17:37:31 +00:00
|
|
|
if (trackAPI_gw.SNG_ID == "0") throw new DownloadFailed("notOnDeezer")
|
|
|
|
|
2021-06-07 18:14:41 +00:00
|
|
|
let itemData = {
|
|
|
|
id: trackAPI_gw.SNG_ID,
|
|
|
|
title: trackAPI_gw.SNG_TITLE.trim(),
|
|
|
|
artist: trackAPI_gw.ART_NAME
|
|
|
|
}
|
2021-04-29 19:14:38 +00:00
|
|
|
|
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-06-07 18:14:41 +00:00
|
|
|
this.log(itemData, "getTags")
|
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-05-29 08:43:09 +00:00
|
|
|
if (e.name === "AlbumDoesntExists") { throw new DownloadFailed('albumDoesntExists') }
|
|
|
|
if (e.name === "MD5NotFound") { throw new DownloadFailed('notLoggedIn') }
|
2021-06-01 18:11:02 +00:00
|
|
|
console.trace(e)
|
2021-04-09 18:05:32 +00:00
|
|
|
throw e
|
|
|
|
}
|
2021-06-07 18:14:41 +00:00
|
|
|
this.log(itemData, "gotTags")
|
2021-04-08 17:37:31 +00:00
|
|
|
}
|
2021-05-16 18:01:56 +00:00
|
|
|
if (this.downloadObject.isCanceled) throw new DownloadCanceled
|
2021-04-08 17:37:31 +00:00
|
|
|
|
2021-06-07 18:14:41 +00:00
|
|
|
itemData = {
|
|
|
|
id: track.id,
|
|
|
|
title: track.title,
|
|
|
|
artist: track.mainArtist.name
|
|
|
|
}
|
2021-04-29 19:14:38 +00:00
|
|
|
|
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-06-07 18:14:41 +00:00
|
|
|
this.log(itemData, "getBitrate")
|
2021-04-09 18:05:32 +00:00
|
|
|
let selectedFormat
|
|
|
|
try{
|
2021-04-09 21:05:24 +00:00
|
|
|
selectedFormat = await getPreferredBitrate(
|
2021-07-25 11:11:23 +00:00
|
|
|
this.dz,
|
2021-04-09 18:05:32 +00:00
|
|
|
track,
|
|
|
|
this.bitrate,
|
2021-05-29 18:21:54 +00:00
|
|
|
this.settings.fallbackBitrate,
|
2021-07-28 10:43:17 +00:00
|
|
|
this.downloadObject.uuid, this.listener
|
2021-04-09 18:05:32 +00:00
|
|
|
)
|
|
|
|
}catch (e){
|
2021-07-25 09:33:33 +00:00
|
|
|
if (e.name === "WrongLicense") { throw new DownloadFailed("wrongLicense")}
|
2021-07-28 10:43:17 +00:00
|
|
|
if (e.name === "WrongGeolocation") { throw new DownloadFailed("wrongGeolocation")}
|
2021-05-29 08:43:09 +00:00
|
|
|
if (e.name === "PreferredBitrateNotFound") { throw new DownloadFailed("wrongBitrate", track) }
|
|
|
|
if (e.name === "TrackNot360") { throw new DownloadFailed("no360RA") }
|
2021-06-01 18:11:02 +00:00
|
|
|
console.trace(e)
|
2021-04-09 18:05:32 +00:00
|
|
|
throw e
|
|
|
|
}
|
2021-04-08 17:37:31 +00:00
|
|
|
track.bitrate = selectedFormat
|
|
|
|
track.album.bitrate = selectedFormat
|
2021-06-07 18:14:41 +00:00
|
|
|
this.log(itemData, "gotBitrate")
|
2021-04-08 17:37:31 +00:00
|
|
|
|
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-05-16 18:01:56 +00:00
|
|
|
if (this.downloadObject.isCanceled) throw new DownloadCanceled
|
2021-04-13 16:40:34 +00:00
|
|
|
|
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-08-02 18:52:33 +00:00
|
|
|
if (extrasPath && !this.downloadObject.extrasPath) {
|
2021-08-02 15:31:17 +00:00
|
|
|
this.downloadObject.extrasPath = extrasPath
|
|
|
|
}
|
2021-04-13 16:40:34 +00:00
|
|
|
|
|
|
|
// 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-06-07 18:14:41 +00:00
|
|
|
this.log(itemData, "getAlbumArt")
|
2021-06-19 10:03:18 +00:00
|
|
|
if (!this.coverQueue[track.album.embeddedCoverPath])
|
|
|
|
this.coverQueue[track.album.embeddedCoverPath] = downloadImage(track.album.embeddedCoverURL, track.album.embeddedCoverPath)
|
|
|
|
track.album.embeddedCoverPath = await this.coverQueue[track.album.embeddedCoverPath]
|
|
|
|
if (this.coverQueue[track.album.embeddedCoverPath]) delete this.coverQueue[track.album.embeddedCoverPath]
|
2021-06-07 18:14:41 +00:00
|
|
|
this.log(itemData, "gotAlbumArt")
|
2021-04-14 16:10:31 +00:00
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
// Save local album art
|
2021-05-19 21:03:47 +00:00
|
|
|
if (coverPath){
|
|
|
|
returnData.albumURLs = []
|
|
|
|
this.settings.localArtworkFormat.split(',').forEach((picFormat) => {
|
|
|
|
if (['png', 'jpg'].includes(picFormat)){
|
|
|
|
let extendedFormat = picFormat
|
|
|
|
if (extendedFormat == 'jpg') extendedFormat += `-${this.settings.jpegImageQuality}`
|
|
|
|
let url = track.album.pic.getURL(this.settings.localArtworkSize, extendedFormat)
|
|
|
|
// Skip non deezer pictures at the wrong format
|
|
|
|
if (track.album.pic instanceof StaticPicture && picFormat != 'jpg') return
|
|
|
|
returnData.albumURLs.push({url, ext: picFormat})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
returnData.albumPath = coverPath
|
|
|
|
returnData.albumFilename = generateAlbumName(this.settings.coverImageTemplate, track.album, this.settings, track.playlist)
|
|
|
|
}
|
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
// Save artist art
|
2021-05-19 21:03:47 +00:00
|
|
|
if (artistPath){
|
|
|
|
returnData.artistURLs = []
|
|
|
|
this.settings.localArtworkFormat.split(',').forEach((picFormat) => {
|
|
|
|
// Deezer doesn't support png artist images
|
|
|
|
if (picFormat === 'jpg'){
|
|
|
|
let extendedFormat = `${picFormat}-${this.settings.jpegImageQuality}`
|
2021-06-07 18:14:41 +00:00
|
|
|
let url = track.album.mainArtist.pic.getURL(this.settings.localArtworkSize, extendedFormat)
|
2021-05-19 21:03:47 +00:00
|
|
|
// Skip non deezer pictures at the wrong format
|
|
|
|
if (track.album.mainArtist.pic.md5 == "") return
|
|
|
|
returnData.artistURLs.push({url, ext: picFormat})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
returnData.artistPath = artistPath
|
|
|
|
returnData.artistFilename = generateArtistName(this.settings.artistImageTemplate, track.album.mainArtist, this.settings, track.album.rootArtist)
|
|
|
|
}
|
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
// Save playlist art
|
2021-05-19 21:03:47 +00:00
|
|
|
if (track.playlist){
|
|
|
|
if (this.playlistURLs.length == 0){
|
|
|
|
this.settings.localArtworkFormat.split(',').forEach((picFormat) => {
|
|
|
|
if (['png', 'jpg'].includes(picFormat)){
|
|
|
|
let extendedFormat = picFormat
|
|
|
|
if (extendedFormat == 'jpg') extendedFormat += `-${this.settings.jpegImageQuality}`
|
|
|
|
let url = track.playlist.pic.getURL(this.settings.localArtworkSize, extendedFormat)
|
|
|
|
// Skip non deezer pictures at the wrong format
|
|
|
|
if (track.playlist.pic instanceof StaticPicture && picFormat != 'jpg') return
|
|
|
|
this.playlistURLs.push({url, ext: picFormat})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
if (!this.playlistCovername){
|
|
|
|
track.playlist.bitrate = track.bitrate
|
|
|
|
track.playlist.dateString = track.playlist.date.format(this.settings.dateFormat)
|
|
|
|
this.playlistCovername = generateAlbumName(this.settings.coverImageTemplate, track.playlist, this.settings, track.playlist)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
// Save lyrics in lrc file
|
2021-05-19 21:03:47 +00:00
|
|
|
if (this.settings.syncedLyrics && track.lyrics.sync){
|
|
|
|
if (!fs.existsSync(`${filepath}/${filename}.lrc`) || [OverwriteOption.OVERWRITE, OverwriteOption.ONLY_TAGS].includes(this.settings.overwriteFile))
|
|
|
|
fs.writeFileSync(`${filepath}/${filename}.lrc`, track.lyrics.sync)
|
|
|
|
}
|
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
// Check for overwrite settings
|
2021-05-19 21:03:47 +00:00
|
|
|
let trackAlreadyDownloaded = fs.existsSync(writepath)
|
|
|
|
|
|
|
|
// Don't overwrite and don't mind extension
|
|
|
|
if (!trackAlreadyDownloaded && this.settings.overwriteFile == OverwriteOption.DONT_CHECK_EXT){
|
|
|
|
let extensions = ['.mp3', '.flac', '.opus', '.m4a']
|
|
|
|
let baseFilename = `${filepath}/${filename}`
|
|
|
|
for (let i = 0; i < extensions.length; i++){
|
|
|
|
let ext = extensions[i]
|
|
|
|
trackAlreadyDownloaded = fs.existsSync(baseFilename+ext)
|
|
|
|
if (trackAlreadyDownloaded) break
|
|
|
|
}
|
|
|
|
}
|
2021-04-09 18:05:32 +00:00
|
|
|
|
2021-05-19 21:03:47 +00:00
|
|
|
// Don't overwrite and keep both files
|
|
|
|
if (trackAlreadyDownloaded && this.settings.overwriteFile == OverwriteOption.KEEP_BOTH){
|
|
|
|
let baseFilename = `${filepath}/${filename}`
|
|
|
|
let currentFilename
|
|
|
|
let c = 0
|
|
|
|
do {
|
|
|
|
c++
|
|
|
|
currentFilename = `${baseFilename} (${c})${extension}`
|
|
|
|
} while (fs.existsSync(currentFilename))
|
|
|
|
trackAlreadyDownloaded = false
|
|
|
|
writepath = currentFilename
|
2021-04-29 19:14:38 +00:00
|
|
|
}
|
|
|
|
|
2021-05-19 21:03:47 +00:00
|
|
|
// Download the track
|
|
|
|
if (!trackAlreadyDownloaded || this.settings.overwriteFile == OverwriteOption.OVERWRITE){
|
2021-07-25 09:33:16 +00:00
|
|
|
track.downloadURL = track.urls[formatsName[track.bitrate]]
|
2021-05-19 21:03:47 +00:00
|
|
|
let stream = fs.createWriteStream(writepath)
|
|
|
|
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
|
|
|
|
}
|
2021-06-07 18:14:41 +00:00
|
|
|
this.log(itemData, "downloaded")
|
2021-05-19 21:03:47 +00:00
|
|
|
} else {
|
2021-06-07 18:14:41 +00:00
|
|
|
this.log(itemData, "alreadyDownloaded")
|
2021-05-19 21:03:47 +00:00
|
|
|
this.downloadObject.completeTrackProgress(this.listener)
|
|
|
|
}
|
2021-04-29 19:14:38 +00:00
|
|
|
|
2021-04-09 18:05:32 +00:00
|
|
|
// Adding tags
|
2021-05-19 21:03:47 +00:00
|
|
|
if (!trackAlreadyDownloaded || [OverwriteOption.ONLY_TAGS, OverwriteOption.OVERWRITE].includes(this.settings.overwriteFile) && !track.local){
|
2021-06-07 18:14:41 +00:00
|
|
|
this.log(itemData, "tagging")
|
2021-05-19 21:03:47 +00:00
|
|
|
if (extension == '.mp3'){
|
|
|
|
tagID3(writepath, track, this.settings.tags)
|
2021-05-28 10:00:40 +00:00
|
|
|
if (this.settings.tags.saveID3v1) tagID3v1(writepath, track, this.settings.tags)
|
2021-05-19 21:03:47 +00:00
|
|
|
} else if (extension == '.flac'){
|
|
|
|
tagFLAC(writepath, track, this.settings.tags)
|
|
|
|
}
|
2021-06-07 18:14:41 +00:00
|
|
|
this.log(itemData, "tagged")
|
2021-04-14 15:11:56 +00:00
|
|
|
}
|
2021-05-19 21:03:47 +00:00
|
|
|
|
|
|
|
if (track.searched) returnData.searched = true
|
|
|
|
this.downloadObject.downloaded += 1
|
2021-05-13 16:11:43 +00:00
|
|
|
this.downloadObject.files.push(String(writepath))
|
|
|
|
if (this.listener)
|
|
|
|
this.listener.send('updateQueue', {
|
|
|
|
uuid: this.downloadObject.uuid,
|
|
|
|
downloaded: true,
|
|
|
|
downloadPath: String(writepath),
|
2021-08-02 18:52:33 +00:00
|
|
|
extrasPath: String(this.downloadObject.extrasPath)
|
2021-05-13 16:11:43 +00:00
|
|
|
})
|
2021-05-19 21:03:47 +00:00
|
|
|
returnData.filename = writepath.slice(extrasPath.length+1)
|
2021-06-07 18:14:41 +00:00
|
|
|
returnData.data = itemData
|
2021-05-19 21:03:47 +00:00
|
|
|
return returnData
|
2021-04-29 19:14:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async downloadWrapper(extraData, track){
|
|
|
|
const { trackAPI_gw } = extraData
|
2021-06-08 16:53:20 +00:00
|
|
|
if (trackAPI_gw._EXTRA_TRACK){
|
|
|
|
extraData.trackAPI = {...trackAPI_gw._EXTRA_TRACK}
|
|
|
|
delete extraData.trackAPI_gw._EXTRA_TRACK
|
|
|
|
delete trackAPI_gw._EXTRA_TRACK
|
|
|
|
}
|
2021-04-29 19:14:38 +00:00
|
|
|
// Temp metadata to generate logs
|
2021-06-07 18:14:41 +00:00
|
|
|
let itemData = {
|
2021-04-29 19:14:38 +00:00
|
|
|
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))
|
2021-06-07 18:14:41 +00:00
|
|
|
itemData.title += ` ${trackAPI_gw.VERSION.trim()}`
|
2021-04-29 19:14:38 +00:00
|
|
|
|
|
|
|
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){
|
2021-06-07 18:14:41 +00:00
|
|
|
this.warn(itemData, e.errid, 'fallback')
|
2021-04-29 19:14:38 +00:00
|
|
|
let newTrack = await this.dz.gw.get_track_with_fallback(track.fallbackID)
|
|
|
|
track.parseEssentialData(newTrack)
|
2021-06-28 23:17:19 +00:00
|
|
|
await track.retriveFilesizes(this.dz)
|
2021-04-29 19:14:38 +00:00
|
|
|
return await this.downloadWrapper(extraData, track)
|
|
|
|
}
|
|
|
|
if (!track.searched && this.settings.fallbackSearch){
|
2021-06-07 18:14:41 +00:00
|
|
|
this.warn(itemData, e.errid, 'search')
|
2021-06-05 15:17:03 +00:00
|
|
|
let searchedID = await this.dz.api.get_track_id_from_metadata(track.mainArtist.name, track.title, track.album.title)
|
2021-04-29 19:14:38 +00:00
|
|
|
if (searchedID != "0"){
|
2021-06-05 15:17:03 +00:00
|
|
|
let newTrack = await this.dz.gw.get_track_with_fallback(searchedID)
|
2021-04-29 19:14:38 +00:00
|
|
|
track.parseEssentialData(newTrack)
|
2021-06-28 23:17:19 +00:00
|
|
|
await track.retriveFilesizes(this.dz)
|
2021-04-29 19:14:38 +00:00
|
|
|
track.searched = true
|
2021-08-02 21:45:19 +00:00
|
|
|
this.log(itemData, "searchFallback")
|
2021-04-29 19:14:38 +00:00
|
|
|
return await this.downloadWrapper(extraData, track)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
e.errid += "NoAlternative"
|
2021-08-02 19:33:58 +00:00
|
|
|
e.message = ErrorMessages[e.errid]
|
2021-04-29 19:14:38 +00:00
|
|
|
}
|
|
|
|
result = {error:{
|
|
|
|
message: e.message,
|
|
|
|
errid: e.errid,
|
2021-06-07 18:14:41 +00:00
|
|
|
data: itemData
|
2021-04-29 19:14:38 +00:00
|
|
|
}}
|
2021-06-17 19:59:15 +00:00
|
|
|
} else if (e instanceof DownloadCanceled){
|
|
|
|
return
|
|
|
|
} else {
|
2021-05-27 19:15:17 +00:00
|
|
|
console.trace(e)
|
2021-04-29 19:14:38 +00:00
|
|
|
result = {error:{
|
|
|
|
message: e.message,
|
2021-06-07 18:14:41 +00:00
|
|
|
data: itemData
|
2021-04-29 19:14:38 +00:00
|
|
|
}}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (result.error){
|
2021-05-19 21:03:47 +00:00
|
|
|
this.downloadObject.completeTrackProgress(this.listener)
|
2021-04-29 19:14:38 +00:00
|
|
|
this.downloadObject.failed += 1
|
|
|
|
this.downloadObject.errors.push(result.error)
|
2021-05-19 21:03:47 +00:00
|
|
|
if (this.listener){
|
2021-04-29 19:14:38 +00:00
|
|
|
let error = result.error
|
2021-05-19 21:03:47 +00:00
|
|
|
this.listener.send("updateQueue", {
|
2021-04-29 19:14:38 +00:00
|
|
|
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
|
|
|
}
|
2021-05-19 21:03:47 +00:00
|
|
|
|
|
|
|
async afterDownloadSingle(track){
|
2021-06-07 18:14:41 +00:00
|
|
|
if (!track) return
|
2021-08-02 18:52:33 +00:00
|
|
|
if (!this.downloadObject.extrasPath) {
|
|
|
|
this.downloadObject.extrasPath = this.settings.downloadLocation
|
2021-08-02 15:31:17 +00:00
|
|
|
}
|
2021-05-19 21:03:47 +00:00
|
|
|
|
|
|
|
// Save local album artwork
|
|
|
|
if (this.settings.saveArtwork && track.albumPath)
|
|
|
|
await each(track.albumURLs, async (image) => {
|
|
|
|
await downloadImage(image.url, `${track.albumPath}/${track.albumFilename}.${image.ext}`, this.settings.overwriteFile)
|
|
|
|
})
|
|
|
|
|
|
|
|
// Save local artist artwork
|
|
|
|
if (this.settings.saveArtworkArtist && track.artistPath)
|
|
|
|
await each(track.artistURLs, async (image) => {
|
|
|
|
await downloadImage(image.url, `${track.artistPath}/${track.artistFilename}.${image.ext}`, this.settings.overwriteFile)
|
|
|
|
})
|
|
|
|
|
|
|
|
// Create searched logfile
|
|
|
|
if (this.settings.logSearched && track.searched){
|
|
|
|
let filename = `${track.data.artist} - ${track.data.title}`
|
2021-08-02 18:52:33 +00:00
|
|
|
let searchedFile = fs.readFileSync(`${this.downloadObject.extrasPath}/searched.txt`).toString()
|
2021-05-19 21:03:47 +00:00
|
|
|
if (searchedFile.indexOf(filename) == -1){
|
|
|
|
if (searchedFile != "") searchedFile += "\r\n"
|
|
|
|
searchedFile += filename + "\r\n"
|
2021-08-02 18:52:33 +00:00
|
|
|
fs.writeFileSync(`${this.downloadObject.extrasPath}/searched.txt`, searchedFile)
|
2021-05-19 21:03:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Execute command after download
|
2021-05-31 21:25:53 +00:00
|
|
|
if (this.settings.executeCommand !== "")
|
2021-08-02 18:52:33 +00:00
|
|
|
exec(this.settings.executeCommand.replaceAll("%folder%", shellEscape(this.downloadObject.extrasPath)).replaceAll("%filename%", shellEscape(track.filename)))
|
2021-05-19 21:03:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
async afterDownloadCollection(tracks){
|
2021-08-02 18:52:33 +00:00
|
|
|
if (!this.downloadObject.extrasPath) {
|
|
|
|
this.downloadObject.extrasPath = this.settings.downloadLocation
|
2021-08-02 15:31:17 +00:00
|
|
|
}
|
|
|
|
|
2021-05-19 21:03:47 +00:00
|
|
|
let playlist = []
|
|
|
|
let errors = ""
|
|
|
|
let searched = ""
|
|
|
|
|
2021-06-07 18:14:41 +00:00
|
|
|
for (let i=0; i < tracks.length; i++){
|
|
|
|
let track = tracks[i]
|
2021-05-19 21:03:47 +00:00
|
|
|
if (!track) return
|
|
|
|
|
|
|
|
if (track.error){
|
|
|
|
if (!track.error.data) track.error.data = {id: "0", title: 'Unknown', artist: 'Unknown'}
|
|
|
|
errors += `${track.error.data.id} | ${track.error.data.artist} - ${track.error.data.title} | ${track.error.message}\r\n`
|
|
|
|
}
|
|
|
|
|
|
|
|
if (track.searched) searched += `${track.data.artist} - ${track.data.title}\r\n`
|
|
|
|
|
|
|
|
// Save local album artwork
|
|
|
|
if (this.settings.saveArtwork && track.albumPath)
|
|
|
|
await each(track.albumURLs, async (image) => {
|
|
|
|
await downloadImage(image.url, `${track.albumPath}/${track.albumFilename}.${image.ext}`, this.settings.overwriteFile)
|
|
|
|
})
|
|
|
|
|
|
|
|
// Save local artist artwork
|
|
|
|
if (this.settings.saveArtworkArtist && track.artistPath)
|
|
|
|
await each(track.artistURLs, async (image) => {
|
|
|
|
await downloadImage(image.url, `${track.artistPath}/${track.artistFilename}.${image.ext}`, this.settings.overwriteFile)
|
|
|
|
})
|
|
|
|
|
|
|
|
// Save filename for playlist file
|
|
|
|
playlist[i] = track.filename || ""
|
2021-06-07 18:14:41 +00:00
|
|
|
}
|
2021-05-19 21:03:47 +00:00
|
|
|
|
|
|
|
// Create errors logfile
|
|
|
|
if (this.settings.logErrors && errors != "")
|
2021-08-02 18:52:33 +00:00
|
|
|
fs.writeFileSync(`${this.downloadObject.extrasPath}/errors.txt`, errors)
|
2021-05-19 21:03:47 +00:00
|
|
|
|
|
|
|
// Create searched logfile
|
|
|
|
if (this.settings.logSearched && searched != "")
|
2021-08-02 18:52:33 +00:00
|
|
|
fs.writeFileSync(`${this.downloadObject.extrasPath}/searched.txt`, searched)
|
2021-05-19 21:03:47 +00:00
|
|
|
|
|
|
|
// Save Playlist Artwork
|
|
|
|
if (this.settings.saveArtwork && this.playlistCovername && !this.settings.tags.savePlaylistAsCompilation)
|
|
|
|
await each(this.playlistURLs, async (image) => {
|
2021-08-02 18:52:33 +00:00
|
|
|
await downloadImage(image.url, `${this.downloadObject.extrasPath}/${this.playlistCovername}.${image.ext}`, this.settings.overwriteFile)
|
2021-05-19 21:03:47 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
// Create M3U8 File
|
|
|
|
if (this.settings.createM3U8File){
|
|
|
|
let filename = generateDownloadObjectName(this.settings.playlistFilenameTemplate, this.downloadObject, this.settings) || "playlist"
|
2021-08-02 18:52:33 +00:00
|
|
|
fs.writeFileSync(`${this.downloadObject.extrasPath}/${filename}.m3u8`, playlist.join('\n'))
|
2021-05-19 21:03:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Execute command after download
|
2021-05-31 21:25:53 +00:00
|
|
|
if (this.settings.executeCommand !== "")
|
2021-08-02 18:52:33 +00:00
|
|
|
exec(this.settings.executeCommand.replaceAll("%folder%", shellEscape(this.downloadObject.extrasPath)).replaceAll("%filename%", ''))
|
2021-05-19 21:03:47 +00:00
|
|
|
}
|
2021-04-09 16:45:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
Downloader,
|
2021-06-17 18:05:13 +00:00
|
|
|
downloadImage,
|
2021-08-02 19:33:58 +00:00
|
|
|
getPreferredBitrate
|
2021-04-09 16:45:57 +00:00
|
|
|
}
|