deemix-js/deemix/downloader.js

289 lines
8.6 KiB
JavaScript
Raw Normal View History

const { Track, AlbumDoesntExists } = require('./types/Track.js')
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')
const { generatePath } = require('./utils/pathtemplates.js')
2021-04-14 16:10:31 +00:00
const { TrackFormats } = require('deezer-js')
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')
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
}
async function getPreferredBitrate(track, bitrate, shouldFallback, uuid, listener){
bitrate = parseInt(bitrate)
if (track.localTrack) { return TrackFormats.LOCAL }
let falledBack = false
const formats_non_360 = {
[TrackFormats.FLAC]: "FLAC",
[TrackFormats.MP3_320]: "MP3_320",
[TrackFormats.MP3_128]: "MP3_128"
}
const formats_360 = {
[TrackFormats.MP4_RA3]: "MP4_RA3",
[TrackFormats.MP4_RA2]: "MP4_RA2",
[TrackFormats.MP4_RA1]: "MP4_RA1"
}
2021-04-21 17:00:51 +00:00
const is360Format = Object.keys(formats_360).includes(bitrate)
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]
let formatName = formats[formatNumber]
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}`)){
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
try {
2021-04-09 21:05:24 +00:00
request = got.get(
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"]
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
}
}
}
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
}
})
}
}
}
if (is360Format) throw new TrackNot360
return TrackFormats.DEFAULT
}
class Downloader {
constructor(dz, downloadObject, settings, listener){
this.dz = dz
this.downloadObject = downloadObject
this.settings = settings || DEFAULTS
this.bitrate = downloadObject.bitrate
this.listener = listener
this.extrasPath = null
this.playlistCoverName = null
this.playlistURLs = []
}
async start(){
if (this.downloadObject.__type__ === "Single"){
await this.download({
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-27 20:24:57 +00:00
for (let pos = 0; pos < this.downloadObject.collection.tracks_gw.length; pos++){
let track = this.downloadObject.collection.tracks_gw[pos]
tracks[pos] = await this.download({
trackAPI_gw: track,
albumAPI: this.downloadObject.collection.albumAPI,
playlistAPI: this.downloadObject.collection.playlistAPI
})
2021-04-27 20:24:57 +00:00
}
}
if (this.listener) this.listener.send("finishedDownload", this.downloadObject.uuid)
}
async download(extraData, track){
const { trackAPI_gw, trackAPI, albumAPI, playlistAPI } = extraData
if (trackAPI_gw.SNG_ID == "0") throw new DownloadFailed("notOnDeezer")
// Generate track object
if (!track){
2021-04-09 21:05:24 +00:00
track = new Track()
console.log("Getting tags")
try{
await track.parseData(
this.dz,
2021-04-09 21:05:24 +00:00
trackAPI_gw.SNG_ID,
trackAPI_gw,
trackAPI,
2021-04-27 20:24:57 +00:00
null, // albumAPI_gw
albumAPI,
playlistAPI
)
} catch (e){
if (e instanceof AlbumDoesntExists) { throw new DownloadError('albumDoesntExists') }
console.error(e)
throw e
}
}
// Check if the track is encoded
if (track.MD5 === "") throw new DownloadFailed("notEncoded", track)
// Check the target bitrate
console.log("Getting bitrate")
let selectedFormat
try{
2021-04-09 21:05:24 +00:00
selectedFormat = await getPreferredBitrate(
track,
this.bitrate,
2021-04-09 21:05:24 +00:00
true, // fallbackBitrate
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
}
track.bitrate = selectedFormat
track.album.bitrate = selectedFormat
// Apply Settings
track.applySettings(this.settings)
// Generate filename and filepath from metadata
let {
filename,
filepath,
artistPath,
coverPath,
extrasPath
} = generatePath(track, this.downloadObject, this.settings)
// Make sure the filepath exsists
fs.mkdirSync(filepath, { recursive: true })
2021-04-14 15:11:56 +00:00
let extension = extensions[track.bitrate]
let writepath = `${filepath}/${filename}${extension}`
// Save extrasPath
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}`
// Download and cache the coverart
2021-04-14 16:10:31 +00:00
track.album.embeddedCoverPath = await downloadImage(track.album.embeddedCoverURL, track.album.embeddedCoverPath)
console.log("Albumart downloaded")
// Save local album art
// Save artist art
// Save playlist art
// Save lyrics in lrc file
// Check for overwrite settings
// Download the track
console.log("Downloading")
track.downloadURL = generateStreamURL(track.id, track.MD5, track.mediaVersion, track.bitrate)
let stream = fs.createWriteStream(writepath)
2021-04-12 18:40:39 +00:00
await streamTrack(stream, track, 0, this.downloadObject, this.listener)
console.log(filename)
// 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
}
}
}
class DownloadError extends Error {
constructor(message) {
super(message);
this.name = "DownloadError"
}
}
class DownloadFailed extends DownloadError {
constructor(errid, track) {
super(errid);
this.name = "ISRCnotOnDeezer"
this.track = track
}
}
class TrackNot360 extends Error {
constructor(message) {
super(message);
this.name = "TrackNot360"
}
}
class PreferredBitrateNotFound extends Error {
constructor(message) {
super(message);
this.name = "PreferredBitrateNotFound"
}
}
module.exports = {
Downloader,
DownloadError,
DownloadFailed,
getPreferredBitrate,
TrackNot360,
PreferredBitrateNotFound
}