2021-04-09 16:45:57 +00:00
|
|
|
const crypto = require('crypto')
|
2021-04-12 18:40:39 +00:00
|
|
|
const got = require('got')
|
2021-04-12 18:52:15 +00:00
|
|
|
|
2021-04-14 16:10:31 +00:00
|
|
|
const { USER_AGENT_HEADER, pipeline } = require('./utils/index.js')
|
2021-04-09 16:45:57 +00:00
|
|
|
|
|
|
|
function _md5 (data, type = 'binary') {
|
|
|
|
let md5sum = crypto.createHash('md5')
|
|
|
|
md5sum.update(Buffer.from(data, type))
|
|
|
|
return md5sum.digest('hex')
|
|
|
|
}
|
|
|
|
|
|
|
|
function _ecbCrypt (key, data) {
|
|
|
|
let cipher = crypto.createCipheriv("aes-128-ecb", Buffer.from(key), Buffer.from(""));
|
|
|
|
return Buffer.concat([cipher.update(data, 'binary'), cipher.final()]).toString("hex").toLowerCase();
|
|
|
|
}
|
|
|
|
|
|
|
|
function _ecbDecrypt (key, data) {
|
|
|
|
let cipher = crypto.createDecipheriv("aes-128-ecb", Buffer.from(key), Buffer.from(""));
|
|
|
|
return Buffer.concat([cipher.update(data, 'binary'), cipher.final()]).toString("hex").toLowerCase();
|
|
|
|
}
|
|
|
|
|
|
|
|
function generateStreamPath(sngID, md5, mediaVersion, format){
|
|
|
|
let urlPart = md5+"¤"+format+"¤"+sngID+"¤"+mediaVersion
|
|
|
|
let md5val = _md5(urlPart)
|
|
|
|
let step2 = md5val+"¤"+urlPart+"¤"
|
|
|
|
step2 += ('.' * (16 - (step2.length % 16)))
|
|
|
|
urlPart = _ecbCrypt('jo6aey6haid2Teih', step2)
|
|
|
|
return urlPart
|
|
|
|
}
|
|
|
|
|
|
|
|
function reverseStreamPath(urlPart){
|
|
|
|
let step2 = _ecbDecrypt('jo6aey6haid2Teih', urlPart)
|
|
|
|
let [, md5, format, sngID, mediaVersion] = step2.split("¤")
|
|
|
|
return [sngID, md5, mediaVersion, format]
|
|
|
|
}
|
|
|
|
|
|
|
|
function generateStreamURL(sngID, md5, mediaVersion, format){
|
|
|
|
let urlPart = generateStreamPath(sngID, md5, mediaVersion, format)
|
|
|
|
return "https://e-cdns-proxy-" + md5[0] + ".dzcdn.net/api/1/" + urlPart
|
|
|
|
}
|
|
|
|
|
|
|
|
function reverseStreamURL(url){
|
2021-04-21 17:00:51 +00:00
|
|
|
let urlPart = url.slice(url.find("/1/")+3)
|
2021-04-09 16:45:57 +00:00
|
|
|
return reverseStreamPath(urlPart)
|
|
|
|
}
|
|
|
|
|
2021-04-12 18:40:39 +00:00
|
|
|
async function streamTrack(outputStream, track, start=0, downloadObject, listener){
|
|
|
|
let headers = {'User-Agent': USER_AGENT_HEADER}
|
2021-04-09 16:45:57 +00:00
|
|
|
let chunkLength = start
|
|
|
|
let complete = 0
|
|
|
|
|
2021-04-29 19:14:38 +00:00
|
|
|
let itemName = `[${track.mainArtist.name} - ${track.title}]`
|
|
|
|
|
2021-04-12 18:40:39 +00:00
|
|
|
let response = got.stream(track.downloadURL, {
|
2021-04-09 16:45:57 +00:00
|
|
|
headers: headers,
|
2021-04-18 11:02:47 +00:00
|
|
|
retry: 3
|
2021-04-09 16:45:57 +00:00
|
|
|
}).on('response', (response)=>{
|
2021-04-12 17:32:57 +00:00
|
|
|
complete = parseInt(response.headers["content-length"])
|
2021-04-09 16:45:57 +00:00
|
|
|
if (complete == 0) throw new DownloadEmpty
|
|
|
|
if (start != 0){
|
2021-04-12 17:32:57 +00:00
|
|
|
let responseRange = response.headers["content-range"]
|
2021-04-29 19:14:38 +00:00
|
|
|
console.log(`${itemName} downloading range ${responseRange}`)
|
2021-04-09 16:45:57 +00:00
|
|
|
}else {
|
2021-04-29 19:14:38 +00:00
|
|
|
console.log(`${itemName} downloading ${complete} bytes`)
|
2021-04-09 16:45:57 +00:00
|
|
|
}
|
2021-05-13 16:11:43 +00:00
|
|
|
}).on('data', function(chunk){
|
|
|
|
chunkLength += chunk.length
|
2021-04-12 18:40:39 +00:00
|
|
|
|
2021-05-13 16:11:43 +00:00
|
|
|
if (downloadObject){
|
|
|
|
let chunkProgres
|
|
|
|
if (downloadObject.__type__ === "Single"){
|
|
|
|
chunkProgres = (chunkLength / (complete + start)) * 100
|
|
|
|
downloadObject.progressNext = chunkProgres
|
|
|
|
}else{
|
|
|
|
chunkProgres = (chunk.length / (complete + start)) / downloadObject.size * 100
|
|
|
|
downloadObject.progressNext += chunkProgres
|
2021-04-09 16:45:57 +00:00
|
|
|
}
|
2021-05-13 16:11:43 +00:00
|
|
|
downloadObject.updateProgress(listener)
|
2021-04-09 16:45:57 +00:00
|
|
|
}
|
|
|
|
})
|
2021-04-12 18:40:39 +00:00
|
|
|
|
2021-04-29 19:14:38 +00:00
|
|
|
try {
|
|
|
|
await pipeline(response, outputStream)
|
|
|
|
} catch (e){
|
|
|
|
if (e instanceof got.ReadError || e instanceof got.TimeoutError)
|
|
|
|
await streamTrack(outputStream, track, chunkLength, downloadObject, listener)
|
|
|
|
else throw e
|
|
|
|
}
|
2021-04-09 16:45:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
class DownloadEmpty extends Error {
|
|
|
|
constructor(message) {
|
|
|
|
super(message);
|
|
|
|
this.name = "DownloadEmpty"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
function generateBlowfishKey(trackId) {
|
|
|
|
const SECRET = 'g4el58wc0zvf9na1';
|
|
|
|
const idMd5 = _md5(trackId.toString(), 'ascii')
|
|
|
|
let bfKey = ''
|
|
|
|
for (let i = 0; i < 16; i++) {
|
|
|
|
bfKey += String.fromCharCode(idMd5.charCodeAt(i) ^ idMd5.charCodeAt(i + 16) ^ SECRET.charCodeAt(i))
|
|
|
|
}
|
|
|
|
return bfKey;
|
|
|
|
}
|
|
|
|
|
|
|
|
function generateCryptedStreamURL(sngID, md5, mediaVersion, format){
|
|
|
|
let urlPart = generateStreamPath(sngID, md5, mediaVersion, format)
|
|
|
|
return "https://e-cdns-proxy-" + md5[0] + ".dzcdn.net/mobile/1/" + urlPart
|
|
|
|
}
|
|
|
|
|
|
|
|
function decryptChunk(chunk, blowFishKey){
|
|
|
|
var cipher = crypto.createDecipheriv('bf-cbc', blowFishKey, Buffer.from([0, 1, 2, 3, 4, 5, 6, 7]))
|
|
|
|
cipher.setAutoPadding(false)
|
|
|
|
return cipher.update(chunk, 'binary', 'binary') + cipher.final()
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
module.exports = {
|
|
|
|
generateStreamPath,
|
|
|
|
generateStreamURL,
|
|
|
|
reverseStreamPath,
|
|
|
|
reverseStreamURL,
|
|
|
|
streamTrack
|
|
|
|
}
|