forked from um/web
fix: decrypt/qmc-cache
adapt: decrypt/qmc for qmc-cache
This commit is contained in:
parent
02a146e069
commit
f0875ad175
@ -22,7 +22,7 @@ interface Handler {
|
|||||||
handler(data?: Uint8Array): QmcMask | undefined
|
handler(data?: Uint8Array): QmcMask | undefined
|
||||||
}
|
}
|
||||||
|
|
||||||
const HandlerMap: { [key: string]: Handler } = {
|
export const HandlerMap: { [key: string]: Handler } = {
|
||||||
"mgg": {handler: QmcMaskDetectMgg, ext: "ogg", detect: true},
|
"mgg": {handler: QmcMaskDetectMgg, ext: "ogg", detect: true},
|
||||||
"mflac": {handler: QmcMaskDetectMflac, ext: "flac", detect: true},
|
"mflac": {handler: QmcMaskDetectMflac, ext: "flac", detect: true},
|
||||||
"mgg.cache": {handler: QmcMaskDetectMgg, ext: "ogg", detect: false},
|
"mgg.cache": {handler: QmcMaskDetectMgg, ext: "ogg", detect: false},
|
||||||
@ -42,8 +42,8 @@ const HandlerMap: { [key: string]: Handler } = {
|
|||||||
"776176": {handler: QmcMaskGetDefault, ext: "wav", detect: false}
|
"776176": {handler: QmcMaskGetDefault, ext: "wav", detect: false}
|
||||||
};
|
};
|
||||||
|
|
||||||
export async function Decrypt(file: File, raw_filename: string, raw_ext: string): Promise<DecryptResult> {
|
export async function Decrypt(file: Blob, raw_filename: string, raw_ext: string): Promise<DecryptResult> {
|
||||||
if (!(raw_ext in HandlerMap)) throw "File type is incorrect!";
|
if (!(raw_ext in HandlerMap)) throw `Qmc cannot handle type: ${raw_ext}`;
|
||||||
const handler = HandlerMap[raw_ext];
|
const handler = HandlerMap[raw_ext];
|
||||||
|
|
||||||
const fileData = new Uint8Array(await GetArrayBuffer(file));
|
const fileData = new Uint8Array(await GetArrayBuffer(file));
|
||||||
|
@ -1,34 +1,41 @@
|
|||||||
import {AudioMimeType, GetArrayBuffer, GetCoverFromFile, GetMetaFromFile, SniffAudioExt} from "@/decrypt/utils.ts";
|
import {
|
||||||
|
AudioMimeType,
|
||||||
|
GetArrayBuffer,
|
||||||
|
GetCoverFromFile,
|
||||||
|
GetMetaFromFile,
|
||||||
|
SniffAudioExt,
|
||||||
|
SplitFilename
|
||||||
|
} from "@/decrypt/utils.ts";
|
||||||
|
|
||||||
import {Decrypt as QmcDecrypt} from "@/decrypt/qmc";
|
import {Decrypt as QmcDecrypt, HandlerMap} from "@/decrypt/qmc";
|
||||||
|
|
||||||
import {DecryptResult} from "@/decrypt/entity";
|
import {DecryptResult} from "@/decrypt/entity";
|
||||||
|
|
||||||
import {parseBlob as metaParseBlob} from "music-metadata-browser";
|
import {parseBlob as metaParseBlob} from "music-metadata-browser";
|
||||||
|
|
||||||
export async function Decrypt(file: Blob, raw_filename: string, raw_ext: string, detect: boolean = true)
|
export async function Decrypt(file: Blob, raw_filename: string, _: string)
|
||||||
: Promise<DecryptResult> {
|
: Promise<DecryptResult> {
|
||||||
let ext = raw_ext;
|
|
||||||
if (detect) {
|
|
||||||
const buffer = new Uint8Array(await GetArrayBuffer(file));
|
const buffer = new Uint8Array(await GetArrayBuffer(file));
|
||||||
let length = buffer.length
|
let length = buffer.length
|
||||||
for (let i = 0; i < length; i++) {
|
for (let i = 0; i < length; i++) {
|
||||||
buffer[i] ^= 0xf4
|
buffer[i] ^= 0xf4
|
||||||
if(buffer[i] <= 0x3f)buffer[i] = buffer[i] * 4;
|
if (buffer[i] <= 0x3f) buffer[i] = buffer[i] * 4;
|
||||||
else if(buffer[i] <= 0x7f)buffer[i] = (buffer[i] - 0x40) * 4 + 1;
|
else if (buffer[i] <= 0x7f) buffer[i] = (buffer[i] - 0x40) * 4 + 1;
|
||||||
else if(buffer[i] <= 0xbf)buffer[i] = (buffer[i] - 0x80) * 4 + 2;
|
else if (buffer[i] <= 0xbf) buffer[i] = (buffer[i] - 0x80) * 4 + 2;
|
||||||
else buffer[i] = (buffer[i] - 0xc0) * 4 + 3;
|
else buffer[i] = (buffer[i] - 0xc0) * 4 + 3;
|
||||||
}
|
}
|
||||||
ext = SniffAudioExt(buffer, raw_ext);
|
let ext = SniffAudioExt(buffer, "");
|
||||||
if (ext !== raw_ext) file = new Blob([buffer], {type: AudioMimeType[ext]})
|
const newName = SplitFilename(raw_filename)
|
||||||
else {
|
let audioBlob: Blob
|
||||||
file = new Blob([buffer], {type: "application/octet-stream"})
|
if (ext !== "" || newName.ext === "mp3") {
|
||||||
let ext = raw_filename.substring(file.name.lastIndexOf(".") + 1, file.name.length).toLowerCase();
|
audioBlob = new Blob([buffer], {type: AudioMimeType[ext]})
|
||||||
if (ext !== "mgg" && ext !== "mflac") throw "不支持的QQ音乐缓存格式:" + raw_filename + ".cache";
|
} else if (newName.ext in HandlerMap) {
|
||||||
return QmcDecrypt(file, raw_filename, ext+".cache");
|
audioBlob = new Blob([buffer], {type: "application/octet-stream"})
|
||||||
|
return QmcDecrypt(audioBlob, newName.name, newName.ext);
|
||||||
|
} else {
|
||||||
|
throw "不支持的QQ音乐缓存格式"
|
||||||
}
|
}
|
||||||
}
|
const tag = await metaParseBlob(audioBlob);
|
||||||
const tag = await metaParseBlob(file);
|
|
||||||
const {title, artist} = GetMetaFromFile(raw_filename, tag.common.title, tag.common.artist)
|
const {title, artist} = GetMetaFromFile(raw_filename, tag.common.title, tag.common.artist)
|
||||||
|
|
||||||
return {
|
return {
|
||||||
@ -37,8 +44,8 @@ export async function Decrypt(file: Blob, raw_filename: string, raw_ext: string,
|
|||||||
ext,
|
ext,
|
||||||
album: tag.common.album,
|
album: tag.common.album,
|
||||||
picture: GetCoverFromFile(tag),
|
picture: GetCoverFromFile(tag),
|
||||||
file: URL.createObjectURL(file),
|
file: URL.createObjectURL(audioBlob),
|
||||||
blob: file,
|
blob: audioBlob,
|
||||||
mime: AudioMimeType[ext]
|
mime: AudioMimeType[ext]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -160,3 +160,11 @@ export function WriteMetaToFlac(audioData: Buffer, info: IMusicMeta, original: I
|
|||||||
}
|
}
|
||||||
return writer.save()
|
return writer.save()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function SplitFilename(n: string): { name: string; ext: string } {
|
||||||
|
const pos = n.lastIndexOf(".")
|
||||||
|
return {
|
||||||
|
ext: n.substring(pos + 1).toLowerCase(),
|
||||||
|
name: n.substring(0, pos)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user