Compare commits
5 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
0c23249d76 | ||
![]() |
6502d2d067 | ||
![]() |
24e1d33642 | ||
![]() |
a72804544f | ||
![]() |
2aeb60d0a9 |
12
package-lock.json
generated
12
package-lock.json
generated
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "unlock-music",
|
"name": "unlock-music",
|
||||||
"version": "1.3.3",
|
"version": "1.3.4",
|
||||||
"lockfileVersion": 1,
|
"lockfileVersion": 1,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
@@ -1074,7 +1074,7 @@
|
|||||||
},
|
},
|
||||||
"@types/debug": {
|
"@types/debug": {
|
||||||
"version": "4.1.5",
|
"version": "4.1.5",
|
||||||
"resolved": "https://registry.npm.taobao.org/@types/debug/download/@types/debug-4.1.5.tgz?cache=0&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2F%40types%2Fdebug%2Fdownload%2F%40types%2Fdebug-4.1.5.tgz",
|
"resolved": "https://registry.npm.taobao.org/@types/debug/download/@types/debug-4.1.5.tgz",
|
||||||
"integrity": "sha1-sU76iFK3do2JiQZhPCP2iHE+As0="
|
"integrity": "sha1-sU76iFK3do2JiQZhPCP2iHE+As0="
|
||||||
},
|
},
|
||||||
"@types/events": {
|
"@types/events": {
|
||||||
@@ -8929,9 +8929,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"register-service-worker": {
|
"register-service-worker": {
|
||||||
"version": "1.6.2",
|
"version": "1.7.0",
|
||||||
"resolved": "https://registry.npm.taobao.org/register-service-worker/download/register-service-worker-1.6.2.tgz",
|
"resolved": "https://registry.npm.taobao.org/register-service-worker/download/register-service-worker-1.7.0.tgz?cache=0&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Fregister-service-worker%2Fdownload%2Fregister-service-worker-1.7.0.tgz",
|
||||||
"integrity": "sha1-kpflTCBcNxxuSb+oj2mX6N0xX0w="
|
"integrity": "sha1-tLYNh6HRF/EZv9JOzgc9zbhdrmQ="
|
||||||
},
|
},
|
||||||
"regjsgen": {
|
"regjsgen": {
|
||||||
"version": "0.5.1",
|
"version": "0.5.1",
|
||||||
@@ -10392,7 +10392,7 @@
|
|||||||
},
|
},
|
||||||
"typedarray-to-buffer": {
|
"typedarray-to-buffer": {
|
||||||
"version": "3.1.5",
|
"version": "3.1.5",
|
||||||
"resolved": "https://registry.npm.taobao.org/typedarray-to-buffer/download/typedarray-to-buffer-3.1.5.tgz",
|
"resolved": "https://registry.npm.taobao.org/typedarray-to-buffer/download/typedarray-to-buffer-3.1.5.tgz?cache=0&other_urls=https%3A%2F%2Fregistry.npm.taobao.org%2Ftypedarray-to-buffer%2Fdownload%2Ftypedarray-to-buffer-3.1.5.tgz",
|
||||||
"integrity": "sha1-qX7nqf9CaRufeD/xvFES/j/KkIA=",
|
"integrity": "sha1-qX7nqf9CaRufeD/xvFES/j/KkIA=",
|
||||||
"requires": {
|
"requires": {
|
||||||
"is-typedarray": "^1.0.0"
|
"is-typedarray": "^1.0.0"
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "unlock-music",
|
"name": "unlock-music",
|
||||||
"version": "1.3.3",
|
"version": "1.3.4",
|
||||||
"updateInfo": "支持网易云音乐DJ类型ncm;优化旧浏览器的支持",
|
"updateInfo": "修复解锁无元数据的ncm文件时会产生的错误",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"description": "Unlock encrypted music file in browser.",
|
"description": "Unlock encrypted music file in browser.",
|
||||||
"repository": {
|
"repository": {
|
||||||
@@ -20,7 +20,7 @@
|
|||||||
"crypto-js": "^4.0.0",
|
"crypto-js": "^4.0.0",
|
||||||
"element-ui": "^2.13.0",
|
"element-ui": "^2.13.0",
|
||||||
"music-metadata-browser": "^2.0.4",
|
"music-metadata-browser": "^2.0.4",
|
||||||
"register-service-worker": "^1.6.2",
|
"register-service-worker": "^1.7.0",
|
||||||
"vue": "^2.6.11"
|
"vue": "^2.6.11"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
@@ -140,8 +140,8 @@
|
|||||||
});
|
});
|
||||||
if (process.env.NODE_ENV === 'production') {
|
if (process.env.NODE_ENV === 'production') {
|
||||||
window._paq.push(["trackEvent", "Error", errInfo, filename]);
|
window._paq.push(["trackEvent", "Error", errInfo, filename]);
|
||||||
console.error(errInfo, filename);
|
|
||||||
}
|
}
|
||||||
|
console.error(errInfo, filename);
|
||||||
},
|
},
|
||||||
changePlaying(url) {
|
changePlaying(url) {
|
||||||
this.playing_url = url;
|
this.playing_url = url;
|
||||||
|
@@ -9,7 +9,7 @@ export async function CommonDecrypt(file) {
|
|||||||
let rt_data;
|
let rt_data;
|
||||||
switch (raw_ext) {
|
switch (raw_ext) {
|
||||||
case "ncm":// Netease Mp3/Flac
|
case "ncm":// Netease Mp3/Flac
|
||||||
rt_data = await NcmDecrypt.Decrypt(file.raw);
|
rt_data = await NcmDecrypt.Decrypt(file.raw, raw_filename, raw_ext);
|
||||||
break;
|
break;
|
||||||
case "mp3":// Raw Mp3
|
case "mp3":// Raw Mp3
|
||||||
case "flac"://Raw Flac
|
case "flac"://Raw Flac
|
||||||
|
@@ -2,10 +2,9 @@ const CryptoJS = require("crypto-js");
|
|||||||
const ID3Writer = require("browser-id3-writer");
|
const ID3Writer = require("browser-id3-writer");
|
||||||
const CORE_KEY = CryptoJS.enc.Hex.parse("687a4852416d736f356b496e62617857");
|
const CORE_KEY = CryptoJS.enc.Hex.parse("687a4852416d736f356b496e62617857");
|
||||||
const META_KEY = CryptoJS.enc.Hex.parse("2331346C6A6B5F215C5D2630553C2728");
|
const META_KEY = CryptoJS.enc.Hex.parse("2331346C6A6B5F215C5D2630553C2728");
|
||||||
import {AudioMimeType, DetectAudioExt, GetArrayBuffer} from "./util"
|
import {AudioMimeType, DetectAudioExt, GetArrayBuffer, GetFileInfo} from "./util"
|
||||||
|
|
||||||
export async function Decrypt(file) {
|
|
||||||
|
|
||||||
|
export async function Decrypt(file, raw_filename, raw_ext) {
|
||||||
const fileBuffer = await GetArrayBuffer(file);
|
const fileBuffer = await GetArrayBuffer(file);
|
||||||
const dataView = new DataView(fileBuffer);
|
const dataView = new DataView(fileBuffer);
|
||||||
|
|
||||||
@@ -13,32 +12,34 @@ export async function Decrypt(file) {
|
|||||||
dataView.getUint32(4, true) !== 0x4d414446)
|
dataView.getUint32(4, true) !== 0x4d414446)
|
||||||
return {status: false, message: "此ncm文件已损坏"};
|
return {status: false, message: "此ncm文件已损坏"};
|
||||||
|
|
||||||
|
|
||||||
const keyDataObj = getKeyData(dataView, fileBuffer, 10);
|
const keyDataObj = getKeyData(dataView, fileBuffer, 10);
|
||||||
const keyBox = getKeyBox(keyDataObj.data);
|
const keyBox = getKeyBox(keyDataObj.data);
|
||||||
|
|
||||||
const musicMetaObj = getMetaData(dataView, fileBuffer, keyDataObj.offset);
|
const musicMetaObj = getMetaData(dataView, fileBuffer, keyDataObj.offset);
|
||||||
const musicMeta = musicMetaObj.data;
|
const musicMeta = musicMetaObj.data;
|
||||||
|
|
||||||
let audioOffset = musicMetaObj.offset + dataView.getUint32(musicMetaObj.offset + 5, true) + 13;
|
let audioOffset = musicMetaObj.offset + dataView.getUint32(musicMetaObj.offset + 5, true) + 13;
|
||||||
let audioData = new Uint8Array(fileBuffer, audioOffset);
|
let audioData = new Uint8Array(fileBuffer, audioOffset);
|
||||||
|
|
||||||
for (let cur = 0; cur < audioData.length; ++cur) audioData[cur] ^= keyBox[cur & 0xff];
|
for (let cur = 0; cur < audioData.length; ++cur) audioData[cur] ^= keyBox[cur & 0xff];
|
||||||
|
|
||||||
if (musicMeta.format === undefined) musicMeta.format = DetectAudioExt(audioData, "mp3");
|
|
||||||
|
|
||||||
const mime = AudioMimeType[musicMeta.format];
|
if (musicMeta.album === undefined) musicMeta.album = "";
|
||||||
|
|
||||||
const artists = [];
|
const artists = [];
|
||||||
musicMeta.artist.forEach(arr => artists.push(arr[0]));
|
if (!!musicMeta.artist) musicMeta.artist.forEach(arr => artists.push(arr[0]));
|
||||||
if (musicMeta.format === "mp3")
|
const info = GetFileInfo(artists.join(" & "), musicMeta.musicName, raw_filename);
|
||||||
audioData = await writeID3(audioData, artists, musicMeta.musicName, musicMeta.album, musicMeta.albumPic);
|
if (artists.length === 0) artists.push(info.artist);
|
||||||
|
|
||||||
|
if (musicMeta.format === undefined) musicMeta.format = DetectAudioExt(audioData, "mp3");
|
||||||
|
if (musicMeta.format === "mp3")
|
||||||
|
audioData = await writeID3(audioData, artists, info.title, musicMeta.album, musicMeta.albumPic);
|
||||||
|
|
||||||
|
const mime = AudioMimeType[musicMeta.format];
|
||||||
const musicData = new Blob([audioData], {type: mime});
|
const musicData = new Blob([audioData], {type: mime});
|
||||||
return {
|
return {
|
||||||
status: true,
|
status: true,
|
||||||
title: musicMeta.musicName,
|
title: info.title,
|
||||||
artist: artists.join(" & "),
|
artist: info.artist,
|
||||||
ext: musicMeta.format,
|
ext: musicMeta.format,
|
||||||
album: musicMeta.album,
|
album: musicMeta.album,
|
||||||
picture: musicMeta.albumPic,
|
picture: musicMeta.albumPic,
|
||||||
@@ -129,9 +130,7 @@ function getKeyBox(keyData) {
|
|||||||
function getMetaData(dataView, fileBuffer, offset) {
|
function getMetaData(dataView, fileBuffer, offset) {
|
||||||
const metaDataLen = dataView.getUint32(offset, true);
|
const metaDataLen = dataView.getUint32(offset, true);
|
||||||
offset += 4;
|
offset += 4;
|
||||||
if (metaDataLen === 0) {
|
if (metaDataLen === 0) return {data: {}, offset: offset};
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
const cipherText = new Uint8Array(fileBuffer, offset, metaDataLen).map(
|
const cipherText = new Uint8Array(fileBuffer, offset, metaDataLen).map(
|
||||||
data => data ^ 0x63
|
data => data ^ 0x63
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
module.exports = {
|
module.exports = {
|
||||||
publicPath: '',
|
publicPath: '',
|
||||||
productionSourceMap: true,
|
productionSourceMap: false,
|
||||||
pwa: {
|
pwa: {
|
||||||
workboxOptions: {
|
workboxOptions: {
|
||||||
skipWaiting: true
|
skipWaiting: true
|
||||||
|
Reference in New Issue
Block a user