mirror of
				https://git.unlock-music.dev/um/web.git
				synced 2025-11-04 21:53:28 +08:00 
			
		
		
		
	Unlock Kuwo .kwm Files!
This commit is contained in:
		
							parent
							
								
									7233fdc707
								
							
						
					
					
						commit
						71a8d9fab7
					
				@ -1,4 +1,5 @@
 | 
				
			|||||||
const NcmDecrypt = require("./ncm");
 | 
					const NcmDecrypt = require("./ncm");
 | 
				
			||||||
 | 
					const KwmDecrypt = require("./kwm");
 | 
				
			||||||
const QmcDecrypt = require("./qmc");
 | 
					const QmcDecrypt = require("./qmc");
 | 
				
			||||||
const RawDecrypt = require("./raw");
 | 
					const RawDecrypt = require("./raw");
 | 
				
			||||||
const TmDecrypt = require("./tm");
 | 
					const TmDecrypt = require("./tm");
 | 
				
			||||||
@ -11,6 +12,9 @@ export async function CommonDecrypt(file) {
 | 
				
			|||||||
        case "ncm":// Netease Mp3/Flac
 | 
					        case "ncm":// Netease Mp3/Flac
 | 
				
			||||||
            rt_data = await NcmDecrypt.Decrypt(file.raw, raw_filename, raw_ext);
 | 
					            rt_data = await NcmDecrypt.Decrypt(file.raw, raw_filename, raw_ext);
 | 
				
			||||||
            break;
 | 
					            break;
 | 
				
			||||||
 | 
					        case "kwm"://Kuwo Mp3/Flac
 | 
				
			||||||
 | 
					            rt_data = await KwmDecrypt.Decrypt(file.raw, raw_filename, raw_ext);
 | 
				
			||||||
 | 
					            break
 | 
				
			||||||
        case "mp3":// Raw Mp3
 | 
					        case "mp3":// Raw Mp3
 | 
				
			||||||
        case "flac"://Raw Flac
 | 
					        case "flac"://Raw Flac
 | 
				
			||||||
        case "m4a":// Raw M4a
 | 
					        case "m4a":// Raw M4a
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										89
									
								
								src/decrypt/kwm.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								src/decrypt/kwm.js
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,89 @@
 | 
				
			|||||||
 | 
					import {
 | 
				
			||||||
 | 
					    AudioMimeType,
 | 
				
			||||||
 | 
					    DetectAudioExt, FLAC_HEADER,
 | 
				
			||||||
 | 
					    GetArrayBuffer,
 | 
				
			||||||
 | 
					    GetFileInfo,
 | 
				
			||||||
 | 
					    GetMetaCoverURL,
 | 
				
			||||||
 | 
					    IsBytesEqual,
 | 
				
			||||||
 | 
					    RequestJsonp
 | 
				
			||||||
 | 
					} from "./util";
 | 
				
			||||||
 | 
					import {QmcMaskCreate58, QmcMaskDetectMflac, QmcMaskDetectMgg, QmcMaskGetDefault} from "./qmcMask";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import {decode} from "iconv-lite"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const musicMetadata = require("music-metadata-browser");
 | 
				
			||||||
 | 
					const MagicHeader = [
 | 
				
			||||||
 | 
					    0x79, 0x65, 0x65, 0x6C, 0x69, 0x6F, 0x6E, 0x2D,
 | 
				
			||||||
 | 
					    0x6B, 0x75, 0x77, 0x6F, 0x2D, 0x74, 0x6D, 0x65,
 | 
				
			||||||
 | 
					]
 | 
				
			||||||
 | 
					const PreDefinedKey = "MoOtOiTvINGwd2E6n0E1i7L5t2IoOoNk"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export async function Decrypt(file, raw_filename, raw_ext) {
 | 
				
			||||||
 | 
					    const oriData = new Uint8Array(await GetArrayBuffer(file));
 | 
				
			||||||
 | 
					    if (!IsBytesEqual(MagicHeader, oriData.slice(0, 0x10)))
 | 
				
			||||||
 | 
					        return {status: false, message: "Not a valid kwm file!"}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    let fileKey = oriData.slice(0x18, 0x20)
 | 
				
			||||||
 | 
					    let mask = createMaskFromKey(fileKey)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    function Uint8ArrayToString(fileData) {
 | 
				
			||||||
 | 
					        var dataString = "";
 | 
				
			||||||
 | 
					        for (var i = 0; i < fileData.length; i++) {
 | 
				
			||||||
 | 
					            dataString += String.fromCharCode(fileData[i]);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        return dataString
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    console.log(Uint8ArrayToString(oriData.slice(0x30, 0x38)))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    let audioData = oriData.slice(0x400);
 | 
				
			||||||
 | 
					    let lenAudioData = audioData.length;
 | 
				
			||||||
 | 
					    for (let cur = 0; cur < lenAudioData; ++cur)
 | 
				
			||||||
 | 
					        audioData[cur] ^= mask[cur % 0x20];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const ext = DetectAudioExt(audioData, "mp3");
 | 
				
			||||||
 | 
					    const mime = AudioMimeType[ext];
 | 
				
			||||||
 | 
					    let musicBlob = new Blob([audioData], {type: mime});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const musicMeta = await musicMetadata.parseBlob(musicBlob);
 | 
				
			||||||
 | 
					    const info = GetFileInfo(musicMeta.common.artist, musicMeta.common.title, raw_filename);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const imgUrl = GetMetaCoverURL(musicMeta);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return {
 | 
				
			||||||
 | 
					        status: true,
 | 
				
			||||||
 | 
					        title: info.title,
 | 
				
			||||||
 | 
					        artist: info.artist,
 | 
				
			||||||
 | 
					        ext: ext,
 | 
				
			||||||
 | 
					        album: musicMeta.common.album,
 | 
				
			||||||
 | 
					        picture: imgUrl,
 | 
				
			||||||
 | 
					        file: URL.createObjectURL(musicBlob),
 | 
				
			||||||
 | 
					        mime: mime
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function createMaskFromKey(keyBytes) {
 | 
				
			||||||
 | 
					    let keyView = new DataView(keyBytes.buffer)
 | 
				
			||||||
 | 
					    let keyStr = keyView.getBigUint64(0, true).toString()
 | 
				
			||||||
 | 
					    let keyStrTrim = trimKey(keyStr)
 | 
				
			||||||
 | 
					    let key = new Uint8Array(32)
 | 
				
			||||||
 | 
					    for (let i = 0; i < 32; i++) {
 | 
				
			||||||
 | 
					        key[i] = PreDefinedKey[i].charCodeAt() ^ keyStrTrim[i].charCodeAt()
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    return key
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					function trimKey(keyRaw) {
 | 
				
			||||||
 | 
					    let lenRaw = keyRaw.length;
 | 
				
			||||||
 | 
					    let out = keyRaw;
 | 
				
			||||||
 | 
					    if (lenRaw > 32) {
 | 
				
			||||||
 | 
					        out = keyRaw.slice(0, 32)
 | 
				
			||||||
 | 
					    } else if (lenRaw < 32) {
 | 
				
			||||||
 | 
					        out = keyRaw.padEnd(32, keyRaw)
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    return out
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@ -3,6 +3,10 @@ export const FLAC_HEADER = [0x66, 0x4C, 0x61, 0x43];
 | 
				
			|||||||
export const MP3_HEADER = [0x49, 0x44, 0x33];
 | 
					export const MP3_HEADER = [0x49, 0x44, 0x33];
 | 
				
			||||||
export const OGG_HEADER = [0x4F, 0x67, 0x67, 0x53];
 | 
					export const OGG_HEADER = [0x4F, 0x67, 0x67, 0x53];
 | 
				
			||||||
export const M4A_HEADER = [0x66, 0x74, 0x79, 0x70];
 | 
					export const M4A_HEADER = [0x66, 0x74, 0x79, 0x70];
 | 
				
			||||||
 | 
					export const WMA_HEADER = [
 | 
				
			||||||
 | 
					    0x30, 0x26, 0xB2, 0x75, 0x8E, 0x66, 0xCF, 0x11,
 | 
				
			||||||
 | 
					    0xA6, 0xD9, 0x00, 0xAA, 0x00, 0x62, 0xCE, 0x6C,
 | 
				
			||||||
 | 
					]
 | 
				
			||||||
export const AudioMimeType = {
 | 
					export const AudioMimeType = {
 | 
				
			||||||
    mp3: "audio/mpeg",
 | 
					    mp3: "audio/mpeg",
 | 
				
			||||||
    flac: "audio/flac",
 | 
					    flac: "audio/flac",
 | 
				
			||||||
@ -63,6 +67,7 @@ export function DetectAudioExt(data, fallbackExt) {
 | 
				
			|||||||
    if (IsBytesEqual(FLAC_HEADER, data.slice(0, FLAC_HEADER.length))) return "flac";
 | 
					    if (IsBytesEqual(FLAC_HEADER, data.slice(0, FLAC_HEADER.length))) return "flac";
 | 
				
			||||||
    if (IsBytesEqual(OGG_HEADER, data.slice(0, OGG_HEADER.length))) return "ogg";
 | 
					    if (IsBytesEqual(OGG_HEADER, data.slice(0, OGG_HEADER.length))) return "ogg";
 | 
				
			||||||
    if (IsBytesEqual(M4A_HEADER, data.slice(4, 4 + M4A_HEADER.length))) return "m4a";
 | 
					    if (IsBytesEqual(M4A_HEADER, data.slice(4, 4 + M4A_HEADER.length))) return "m4a";
 | 
				
			||||||
 | 
					    if (IsBytesEqual(WMA_HEADER, data.slice(0, WMA_HEADER.length))) return "wma";
 | 
				
			||||||
    return fallbackExt;
 | 
					    return fallbackExt;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user