deemix-js/deemix/decryption.js

128 lines
3.9 KiB
JavaScript
Raw Normal View History

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')
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)
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}
let chunkLength = start
let complete = 0
let itemName = `[${track.mainArtist.name} - ${track.title}]`
2021-04-12 18:40:39 +00:00
let response = got.stream(track.downloadURL, {
headers: headers,
2021-04-18 11:02:47 +00:00
retry: 3
}).on('response', (response)=>{
2021-04-12 17:32:57 +00:00
complete = parseInt(response.headers["content-length"])
if (complete == 0) throw new DownloadEmpty
if (start != 0){
2021-04-12 17:32:57 +00:00
let responseRange = response.headers["content-range"]
console.log(`${itemName} downloading range ${responseRange}`)
}else {
console.log(`${itemName} downloading ${complete} bytes`)
}
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-05-13 16:11:43 +00:00
downloadObject.updateProgress(listener)
}
})
2021-04-12 18:40:39 +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
}
}
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
}