forked from um/web
Use Universal Decoder for Qmc,Mgg,Mflac
This commit is contained in:
parent
47cea6eae9
commit
41a45176be
@ -1,8 +1,6 @@
|
|||||||
const NcmDecrypt = require("./ncm");
|
const NcmDecrypt = require("./ncm");
|
||||||
const QmcDecrypt = require("./qmc");
|
const QmcDecrypt = require("./qmc");
|
||||||
const RawDecrypt = require("./raw");
|
const RawDecrypt = require("./raw");
|
||||||
const MFlacDecrypt = require("./mflac");
|
|
||||||
const MggDecrypt = require("./mgg");
|
|
||||||
const TmDecrypt = require("./tm");
|
const TmDecrypt = require("./tm");
|
||||||
|
|
||||||
|
|
||||||
@ -31,13 +29,9 @@ export async function CommonDecrypt(file) {
|
|||||||
case "tkm"://QQ Music Accompaniment M4a
|
case "tkm"://QQ Music Accompaniment M4a
|
||||||
case "bkcmp3"://Moo Music Mp3
|
case "bkcmp3"://Moo Music Mp3
|
||||||
case "bkcflac"://Moo Music Flac
|
case "bkcflac"://Moo Music Flac
|
||||||
rt_data = await QmcDecrypt.Decrypt(file.raw, raw_filename, raw_ext);
|
|
||||||
break;
|
|
||||||
case "mflac"://QQ Music Desktop Flac
|
case "mflac"://QQ Music Desktop Flac
|
||||||
rt_data = await MFlacDecrypt.Decrypt(file.raw, raw_filename, raw_ext);
|
case "mgg": //QQ Music Desktop Ogg
|
||||||
break;
|
rt_data = await QmcDecrypt.Decrypt(file.raw, raw_filename, raw_ext);
|
||||||
case "mgg":
|
|
||||||
rt_data = await MggDecrypt.Decrypt(file.raw, raw_filename, raw_ext);
|
|
||||||
break;
|
break;
|
||||||
case "tm2":// QQ Music IOS M4a
|
case "tm2":// QQ Music IOS M4a
|
||||||
case "tm6":// QQ Music IOS M4a
|
case "tm6":// QQ Music IOS M4a
|
||||||
|
@ -1,55 +0,0 @@
|
|||||||
const musicMetadata = require("music-metadata-browser");
|
|
||||||
import {GetArrayBuffer, GetCoverURL, GetFileInfo} from "./util"
|
|
||||||
|
|
||||||
import * as mask from "./qmcmask"
|
|
||||||
|
|
||||||
export async function Decrypt(file, raw_filename, raw_ext) {
|
|
||||||
// 获取扩展名
|
|
||||||
if (raw_ext !== "mflac") return {
|
|
||||||
status: false,
|
|
||||||
message: "File type is incorrect!",
|
|
||||||
};
|
|
||||||
// 读取文件
|
|
||||||
const fileBuffer = await GetArrayBuffer(file);
|
|
||||||
const audioData = new Uint8Array(fileBuffer.slice(0, -0x170));
|
|
||||||
//const audioDataLen = audioData.length;
|
|
||||||
|
|
||||||
// 转换数据
|
|
||||||
const seed = mask.QmcMaskDetectMflac(audioData);
|
|
||||||
if (seed === undefined) return {
|
|
||||||
status: false,
|
|
||||||
message: "此音乐无法解锁,目前mflac格式不提供完整支持",
|
|
||||||
};
|
|
||||||
const dec = seed.Decrypt(audioData);
|
|
||||||
// 导出
|
|
||||||
const musicData = new Blob([dec], {type: "audio/flac"});
|
|
||||||
|
|
||||||
// 读取Meta
|
|
||||||
let tag = await musicMetadata.parseBlob(musicData);
|
|
||||||
const info = GetFileInfo(tag.common.artist, tag.common.title, raw_filename);
|
|
||||||
//reportKeyInfo(new Uint8Array(fileBuffer.slice(-0x170)), seed.mask128,
|
|
||||||
// info.artist, info.title, tag.common.album, raw_filename);
|
|
||||||
|
|
||||||
// 返回
|
|
||||||
return {
|
|
||||||
status: true,
|
|
||||||
title: info.title,
|
|
||||||
artist: info.artist,
|
|
||||||
ext: 'flac',
|
|
||||||
album: tag.common.album,
|
|
||||||
picture: GetCoverURL(tag),
|
|
||||||
file: URL.createObjectURL(musicData),
|
|
||||||
mime: "audio/flac"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function reportKeyInfo(keyData, maskData, artist, title, album, filename) {
|
|
||||||
fetch("https://stats.ixarea.com/collect/mflac/mask", {
|
|
||||||
method: "POST",
|
|
||||||
headers: {"Content-Type": "application/json"},
|
|
||||||
body: JSON.stringify({
|
|
||||||
Mask: Array.from(maskData), Key: Array.from(keyData),
|
|
||||||
Artist: artist, Title: title, Album: album, Filename: filename
|
|
||||||
}),
|
|
||||||
}).then().catch()
|
|
||||||
}
|
|
@ -1,60 +0,0 @@
|
|||||||
const musicMetadata = require("music-metadata-browser");
|
|
||||||
const util = require("./util");
|
|
||||||
|
|
||||||
import * as mask from "./qmcmask"
|
|
||||||
|
|
||||||
//todo: combine qmc mflac mgg
|
|
||||||
export async function Decrypt(file, raw_filename, raw_ext) {
|
|
||||||
// 获取扩展名
|
|
||||||
if (raw_ext !== "mgg") return {
|
|
||||||
status: false,
|
|
||||||
message: "File type is incorrect!",
|
|
||||||
};
|
|
||||||
// 读取文件
|
|
||||||
const fileBuffer = await util.GetArrayBuffer(file);
|
|
||||||
const audioData = new Uint8Array(fileBuffer.slice(0, -0x170));
|
|
||||||
const audioDataLen = audioData.length;
|
|
||||||
const keyData = new Uint8Array(fileBuffer.slice(-0x170));
|
|
||||||
const headData = new Uint8Array(fileBuffer.slice(0, 170));
|
|
||||||
let seed = mask.QmcMaskDetectMgg(headData);
|
|
||||||
if (seed === undefined) {
|
|
||||||
return {
|
|
||||||
status: false,
|
|
||||||
message: "此音乐无法解锁,目前mgg格式仅提供试验性支持",
|
|
||||||
};
|
|
||||||
/*try {
|
|
||||||
let resp = await queryKeyInfo(keyData, headData, raw_filename);
|
|
||||||
let data = await resp.json();
|
|
||||||
seed = mask.QmcMaskCreate128(data.Mask);
|
|
||||||
} catch (e) {}*/
|
|
||||||
}
|
|
||||||
const dec = seed.Decrypt(audioData);
|
|
||||||
// 导出
|
|
||||||
const musicData = new Blob([dec], {type: "audio/ogg"});
|
|
||||||
|
|
||||||
// 读取Meta
|
|
||||||
let tag = await musicMetadata.parseBlob(musicData);
|
|
||||||
const info = util.GetFileInfo(tag.common.artist, tag.common.title, raw_filename);
|
|
||||||
|
|
||||||
// 返回
|
|
||||||
return {
|
|
||||||
status: true,
|
|
||||||
title: info.title,
|
|
||||||
artist: info.artist,
|
|
||||||
ext: 'ogg',
|
|
||||||
album: tag.common.album,
|
|
||||||
picture: util.GetCoverURL(tag),
|
|
||||||
file: URL.createObjectURL(musicData),
|
|
||||||
mime: "audio/ogg"
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
function queryKeyInfo(keyData, headData, filename) {
|
|
||||||
return fetch("http://127.0.0.1:6580/mgg/query", {
|
|
||||||
method: "POST",
|
|
||||||
headers: {"Content-Type": "application/json"},
|
|
||||||
body: JSON.stringify({Key: Array.from(keyData), Head: Array.from(headData), Filename: filename}),
|
|
||||||
})
|
|
||||||
}
|
|
@ -1,46 +1,78 @@
|
|||||||
const musicMetadata = require("music-metadata-browser");
|
import {AudioMimeType, GetArrayBuffer, GetCoverURL, GetFileInfo} from "./util";
|
||||||
const util = require("./util");
|
import * as mask from "./qmcMask"
|
||||||
import * as mask from "./qmcmask"
|
|
||||||
|
|
||||||
const OriginalExtMap = {
|
const musicMetadata = require("music-metadata-browser");
|
||||||
"qmc0": "mp3",
|
|
||||||
"qmc3": "mp3",
|
const HandlerMap = {
|
||||||
"qmcogg": "ogg",
|
"mgg": {handler: mask.QmcMaskDetectMgg, ext: "ogg", detect: true},
|
||||||
"qmcflac": "flac",
|
"mflac": {handler: mask.QmcMaskDetectMflac, ext: "flac", detect: true},
|
||||||
"bkcmp3": "mp3",
|
"qmc0": {handler: mask.QmcMaskGetDefault, ext: "mp3", detect: false},
|
||||||
"bkcflac": "flac",
|
"qmc3": {handler: mask.QmcMaskGetDefault, ext: "mp3", detect: false},
|
||||||
"tkm": "m4a"
|
"qmcogg": {handler: mask.QmcMaskGetDefault, ext: "ogg", detect: false},
|
||||||
|
"qmcflac": {handler: mask.QmcMaskGetDefault, ext: "flac", detect: false},
|
||||||
|
"bkcmp3": {handler: mask.QmcMaskGetDefault, ext: "mp3", detect: false},
|
||||||
|
"bkcflac": {handler: mask.QmcMaskGetDefault, ext: "flac", detect: false},
|
||||||
|
"tkm": {handler: mask.QmcMaskGetDefault, ext: "m4a", detect: false}
|
||||||
};
|
};
|
||||||
|
|
||||||
//todo: use header to detect media type
|
//todo: use header to detect media type
|
||||||
export async function Decrypt(file, raw_filename, raw_ext) {
|
export async function Decrypt(file, raw_filename, raw_ext) {
|
||||||
// 获取扩展名
|
if (!(raw_ext in HandlerMap)) return {status: false, message: "File type is incorrect!"};
|
||||||
if (!(raw_ext in OriginalExtMap)) {
|
const handler = HandlerMap[raw_ext];
|
||||||
return {status: false, message: "File type is incorrect!"}
|
|
||||||
}
|
|
||||||
const new_ext = OriginalExtMap[raw_ext];
|
|
||||||
const mime = util.AudioMimeType[new_ext];
|
|
||||||
// 读取文件
|
|
||||||
const fileBuffer = await util.GetArrayBuffer(file);
|
|
||||||
const audioData = new Uint8Array(fileBuffer);
|
|
||||||
// 转换数据
|
|
||||||
const seed = mask.QmcMaskGetDefault();
|
|
||||||
const dec = seed.Decrypt(audioData);
|
|
||||||
// 导出
|
|
||||||
const musicData = new Blob([dec], {type: mime});
|
|
||||||
// 读取Meta
|
|
||||||
const tag = await musicMetadata.parseBlob(musicData);
|
|
||||||
const info = util.GetFileInfo(tag.common.artist, tag.common.title, raw_filename);
|
|
||||||
|
|
||||||
// 返回
|
const fileData = new Uint8Array(await GetArrayBuffer(file));
|
||||||
|
let audioData, seed, keyData;
|
||||||
|
if (handler.detect) {
|
||||||
|
audioData = fileData.slice(0, -0x170);
|
||||||
|
seed = handler.handler(audioData);
|
||||||
|
keyData = fileData.slice(-0x170);
|
||||||
|
if (seed === undefined) seed = await queryKeyInfo(keyData, raw_filename, raw_ext);
|
||||||
|
if (seed === undefined) return {status: false, message: "此格式仅提供实验性支持!"};
|
||||||
|
} else {
|
||||||
|
audioData = fileData;
|
||||||
|
seed = handler.handler(audioData);
|
||||||
|
}
|
||||||
|
const dec = seed.Decrypt(audioData);
|
||||||
|
|
||||||
|
const mime = AudioMimeType[handler.ext];
|
||||||
|
const musicData = new Blob([dec], {type: mime});
|
||||||
|
|
||||||
|
const tag = await musicMetadata.parseBlob(musicData);
|
||||||
|
const info = GetFileInfo(tag.common.artist, tag.common.title, raw_filename);
|
||||||
|
if (handler.detect) reportKeyUsage(keyData, seed.Matrix128,
|
||||||
|
info.artist, info.title, tag.common.album, raw_filename, raw_ext);
|
||||||
return {
|
return {
|
||||||
status: true,
|
status: true,
|
||||||
title: info.title,
|
title: info.title,
|
||||||
artist: info.artist,
|
artist: info.artist,
|
||||||
ext: new_ext,
|
ext: handler.ext,
|
||||||
album: tag.common.album,
|
album: tag.common.album,
|
||||||
picture: util.GetCoverURL(tag),
|
picture: GetCoverURL(tag),
|
||||||
file: URL.createObjectURL(musicData),
|
file: URL.createObjectURL(musicData),
|
||||||
mime: mime
|
mime: mime
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function reportKeyUsage(keyData, maskData, artist, title, album, filename, format) {
|
||||||
|
fetch("https://stats.ixarea.com/collect/qmcmask/usage", {
|
||||||
|
method: "POST",
|
||||||
|
headers: {"Content-Type": "application/json"},
|
||||||
|
body: JSON.stringify({
|
||||||
|
Mask: Array.from(maskData), Key: Array.from(keyData),
|
||||||
|
Artist: artist, Title: title, Album: album, Filename: filename, Format: format
|
||||||
|
}),
|
||||||
|
}).then().catch()
|
||||||
|
}
|
||||||
|
|
||||||
|
async function queryKeyInfo(keyData, filename, format) {
|
||||||
|
try {
|
||||||
|
const resp = await fetch("https://stats.ixarea.com/collect/qmcmask/query", {
|
||||||
|
method: "POST",
|
||||||
|
headers: {"Content-Type": "application/json"},
|
||||||
|
body: JSON.stringify({Format: format, Key: Array.from(keyData), Filename: filename}),
|
||||||
|
});
|
||||||
|
let data = await resp.json();
|
||||||
|
return mask.QmcMaskCreate58(data.Matrix58, data.Super58A, data.Super58B);
|
||||||
|
} catch (e) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user