feat: 新增助眠解压相关模块 && bump 4.32.0

This commit is contained in:
ElyPrism 2026-04-17 20:41:31 +08:00
parent 2c36daf07a
commit ba9c6deaee
No known key found for this signature in database
16 changed files with 341 additions and 51 deletions

View File

@ -0,0 +1,9 @@
// 跑步漫游
const createOption = require('../util/option.js')
module.exports = (query, request) => {
const data = {
bpm: query.bpm || 50,
}
return request(`/api/radio/sport/get`, data, createOption(query))
}

View File

@ -2,6 +2,7 @@ const CryptoJS = require('crypto-js')
const path = require('path') const path = require('path')
const fs = require('fs') const fs = require('fs')
const ID_XOR_KEY_1 = '3go8&$8*3*3h0k(2)2' const ID_XOR_KEY_1 = '3go8&$8*3*3h0k(2)2'
const logger = require('../util/logger.js')
const createOption = require('../util/option.js') const createOption = require('../util/option.js')
const { generateDeviceId } = require('../util/index') const { generateDeviceId } = require('../util/index')
@ -23,7 +24,7 @@ function cloudmusic_dll_encode_id(some_id) {
module.exports = async (query, request) => { module.exports = async (query, request) => {
const deviceId = generateDeviceId() const deviceId = generateDeviceId()
console.log(`[register_anonimous] deviceId: ${deviceId}`) logger.info(`Successfully registered anonimous token, deviceId: ${deviceId}`)
global.deviceId = deviceId global.deviceId = deviceId
const encodedId = CryptoJS.enc.Base64.stringify( const encodedId = CryptoJS.enc.Base64.stringify(
CryptoJS.enc.Utf8.parse( CryptoJS.enc.Utf8.parse(

View File

@ -0,0 +1,11 @@
// 助眠解压 - 获取标签下资源列表
const createOption = require('../util/option.js')
module.exports = (query, request) => {
const data = {
tag: query.tag,
firstQuery: false,
}
return request(`/api/voice/sati/resource/list`, data, createOption(query))
}

View File

@ -0,0 +1,13 @@
// 助眠解压 - 查看同类推荐
const createOption = require('../util/option.js')
module.exports = (query, request) => {
const data = {
id: query.id,
}
return request(
`/api/voice/sati/resource/list/more/v1`,
data,
createOption(query),
)
}

View File

@ -0,0 +1,10 @@
// 助眠解压 - 收藏
const createOption = require('../util/option.js')
module.exports = (query, request) => {
const data = {
id: query.id,
cancel: query.cancel || false,
}
return request(`/api/voice/sati/resource/sub`, data, createOption(query))
}

View File

@ -0,0 +1,7 @@
// 助眠解压 - 收藏列表
const createOption = require('../util/option.js')
module.exports = (query, request) => {
const data = {}
return request(`/api/voice/sati/resource/sub/list`, data, createOption(query))
}

7
module/sati_tag_list.js Normal file
View File

@ -0,0 +1,7 @@
// 助眠解压 - 标签列表
const createOption = require('../util/option.js')
module.exports = (query, request) => {
const data = {}
return request(`/api/voice/sati/tag/list`, data, createOption(query))
}

View File

@ -0,0 +1,13 @@
// 助眠解压 - 特定时间场景下的推荐资源
const createOption = require('../util/option.js')
module.exports = (query, request) => {
const data = {
firstQuery: false,
}
return request(
`/api/voice/sati/timescene/resources/get`,
data,
createOption(query),
)
}

9
module/song_creators.js Normal file
View File

@ -0,0 +1,9 @@
// 歌曲创作者信息
const createOption = require('../util/option.js')
module.exports = (query, request) => {
const data = {
songId: query.id,
}
return request(`/api/song/creators`, data, createOption(query))
}

View File

@ -1,6 +1,6 @@
{ {
"name": "@neteasecloudmusicapienhanced/api", "name": "@neteasecloudmusicapienhanced/api",
"version": "4.31.0", "version": "4.32.0",
"description": "全网最全的网易云音乐API接口 || A revival project for NeteaseCloudMusicApi Node.js Services (Half Refactor & Enhanced) || 网易云音乐 API 备份 + 增强 || 本项目自原版v4.28.0版本后开始自行维护", "description": "全网最全的网易云音乐API接口 || A revival project for NeteaseCloudMusicApi Node.js Services (Half Refactor & Enhanced) || 网易云音乐 API 备份 + 增强 || 本项目自原版v4.28.0版本后开始自行维护",
"scripts": { "scripts": {
"dev": "nodemon app.js", "dev": "nodemon app.js",

View File

@ -188,6 +188,35 @@
document.getElementById('result').value = 'Request failed: ' + error.message; document.getElementById('result').value = 'Request failed: ' + error.message;
} }
} }
(function fillFromQuery() {
const params = new URLSearchParams(window.location.search);
if (!params.toString()) return;
const uri = params.get('uri');
const crypto = params.get('crypto');
const dataParam = params.get('data');
if (uri) {
document.getElementById('uri').value = uri;
}
if (crypto) {
const cryptoSelect = document.getElementById('crypto');
if ([...cryptoSelect.options].some((opt) => opt.value === crypto)) {
cryptoSelect.value = crypto;
}
}
if (dataParam) {
const decoded = dataParam;
try {
document.getElementById('data').value = JSON.stringify(
JSON.parse(decoded),
null,
2,
);
} catch (error) {
document.getElementById('data').value = decoded;
}
}
})();
</script> </script>
</body> </body>
</html> </html>

View File

@ -209,6 +209,12 @@ $ sudo docker build . -t netease-music-api
$ sudo docker run -d -p 3000:3000 netease-music-api $ sudo docker run -d -p 3000:3000 netease-music-api
``` ```
## 调试工具
- `eapi` 请求参数或返回内容可在 `/eapi_decrypt.html` 里解析
- 请求参数模式下, 解密结果可直接带到 `/api.html` 继续调试
- 需要返回值加密时, 可传 `e_r=1`, `weapi``eapi` 都支持
## 接口文档 ## 接口文档
### 调用前须知 ### 调用前须知
@ -5031,6 +5037,21 @@ let data = encodeURIComponent(
**调用例子:** `/vip/sign/info` **调用例子:** `/vip/sign/info`
### 广播电台 - 收藏/取消收藏电台
说明: 登录后调用此接口, 传入电台 id, 可收藏或取消收藏广播电台
**必选参数:**
`id`: 电台 id
`t`: 操作类型, `1` 为收藏, 其余值为取消收藏
**接口地址:** `/broadcast/sub`
**调用例子:** `/broadcast/sub?id=5&t=1`
### 用户的创建歌单列表 ### 用户的创建歌单列表
说明 : 调用此接口, 传入用户id, 获取用户的创建歌单列表 说明 : 调用此接口, 传入用户id, 获取用户的创建歌单列表
@ -5209,6 +5230,92 @@ let data = encodeURIComponent(
**调用例子:** `/dj/difm/playing/tracks/list?source=0&channelId=1012` **调用例子:** `/dj/difm/playing/tracks/list?source=0&channelId=1012`
### 助眠解压 - 特定时间场景下的推荐资源
说明: 调用此接口, 获取特定时间场景下的推荐资源
**接口地址:** `/sati/timescene/resources/get`
**调用例子:** `/sati/timescene/resources/get`
### 助眠解压 - 标签列表
说明: 调用此接口, 获取标签列表
**接口地址:** `/sati/tag/list`
**调用例子:** `/sati/tag/list`
### 助眠解压 - 获取标签下资源列表
说明: 调用此接口, 获取标签下资源列表; 接口返回的`trackId`可以用于请求`/song/url/v1`接口,用于获取声音的下载地址
**必选参数 :**
`tag`: 标签, 由标签列表接口得到
**接口地址:** `/sati/resource/list`
**调用例子:** `/sati/resource/list?tag=naturalMusic`
### 助眠解压 - 查看同类推荐
说明: 调用此接口, 查看同类推荐
**必选参数 :**
`id`: id, `/sati/tag/list`接口返回的`trackId`
**接口地址:** `/sati/resource/list/more`
**调用例子:** `/sati/resource/list/more?id=167003`
### 助眠解压 - 收藏列表
说明: 调用此接口, 获取收藏列表
**接口地址:** `/sati/resource/sub/list`
**调用例子:** `/sati/resource/sub/list`
### 助眠解压 - 收藏
说明: 调用此接口, 收藏声音
**必选参数 :**
`id`: id, `/sati/tag/list`接口返回的`trackId`
**可选参数 :**
`cancel`: 是否取消收藏, 默认不取消
**接口地址:** `/sati/resource/sub`
**调用例子:** `/sati/resource/sub?id=167003`
### 跑步漫游
说明: 调用此接口,获取跑步漫游的歌曲信息
**必选参数:**
`bpm`: 步频
**接口地址:** `/radio/sport/get`
**调用例子:** `/radio/sport/get?bpm=50`
### 歌曲创作者信息
说明 : 调用此接口, 传入音乐 id 可获得对应音乐的创作者信息
**必选参数 :** `id`: 音乐 id
**接口地址 :** `/song/creators`
**调用例子 :** `/song/creators?id=33894312`
## 离线访问此文档 ## 离线访问此文档
此文档同时也是 Progressive Web Apps(PWA), 加入了 serviceWorker, 可离线访问 此文档同时也是 Progressive Web Apps(PWA), 加入了 serviceWorker, 可离线访问

View File

@ -96,6 +96,10 @@
transition: background 0.2s ease; transition: background 0.2s ease;
} }
button + button {
margin-left: 12px;
}
button:hover { button:hover {
background: #555; background: #555;
} }
@ -166,6 +170,7 @@
</div> </div>
<button @click="decrypt">解密</button> <button @click="decrypt">解密</button>
<button @click="sendToApi" :disabled="!canSend" :class="[{ 'opacity-50 cursor-not-allowed pointer-events-none': !canSend },]">填入 API 调试</button>
<div class="result-section"> <div class="result-section">
<label>解密结果:</label> <label>解密结果:</label>
@ -194,12 +199,29 @@
mounted() { mounted() {
this.decrypt() this.decrypt()
}, },
methods: { computed: {
formatResult(result) { isRequestMode() {
return this.isReq === true || this.isReq === 'true'
},
canSend() {
if (!this.isRequestMode) return false
if (!this.result || this.result === '{}' || this.result === 'null') return false
try { try {
return JSON.stringify(JSON.parse(result), null, 2) JSON.parse(this.result)
} catch (e) { return true
return result } catch (error) {
return false
}
},
},
methods: {
formatResult(value) {
if (value == null || value === '') return ''
try {
const parsed = typeof value === 'string' ? JSON.parse(value) : value
return JSON.stringify(parsed, null, 2)
} catch (error) {
return String(value)
} }
}, },
async decrypt() { async decrypt() {
@ -215,9 +237,25 @@
console.log(res.data); console.log(res.data);
} catch (error) { } catch (error) {
console.error(error) console.error(error)
alert(error?.response?.data?.message || '解密失败数据格式错误') alert(error?.response?.data?.message || '解密失败,数据格式错误')
} }
} },
sendToApi() {
if (!this.canSend) return
const payload = JSON.parse(this.result)
const params = new URLSearchParams()
params.set('uri', payload.uri || payload.url || payload.path || '')
params.set('crypto', 'eapi')
const data =
payload.params ||
payload.data ||
payload.body ||
payload.payload ||
payload.request ||
{}
params.set('data', JSON.stringify(data))
window.open(`/api.html?${params.toString()}`, '_blank')
},
} }
}) })
app.mount('#app') app.mount('#app')

View File

@ -156,6 +156,29 @@ function getCorsAllowOrigin(allowOrigins, requestOrigin) {
return null return null
} }
function createConsoleSpinner(message = '启动中') {
if (!process.stdout.isTTY) {
return {
stop() {},
}
}
const frames = ['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏']
let index = 0
process.stdout.write(`${frames[index]} ${message}...`)
const timer = setInterval(() => {
index = (index + 1) % frames.length
process.stdout.write(`\r${frames[index]} ${message}...`)
}, 80)
return {
stop() {
clearInterval(timer)
process.stdout.write(`\r${message} 完成。\n`)
},
}
}
/** /**
* Construct the server of NCM API. * Construct the server of NCM API.
* *
@ -387,6 +410,8 @@ async function serveNcmApi(options) {
const port = Number(options.port || process.env.PORT || '3000') const port = Number(options.port || process.env.PORT || '3000')
const host = options.host || process.env.HOST || '' const host = options.host || process.env.HOST || ''
const spinner = createConsoleSpinner('服务启动中')
const checkVersionSubmission = const checkVersionSubmission =
options.checkVersion && options.checkVersion &&
checkVersion().then(({ npmVersion, ourVersion, status }) => { checkVersion().then(({ npmVersion, ourVersion, status }) => {
@ -403,21 +428,15 @@ async function serveNcmApi(options) {
constructServerSubmission, constructServerSubmission,
]) ])
spinner.stop()
/** @type {import('express').Express & ExpressExtension} */ /** @type {import('express').Express & ExpressExtension} */
const appExt = app const appExt = app
appExt.server = app.listen(port, host, () => { appExt.server = app.listen(port, host, () => {
console.log(` console.log(`
_ _ _____ __ __
| \\ | |/ ____| \\/ |
| \\| | | | \\ / |
| . \` | | | |\\/| |
| |\\ | |____| | | |
|_| \\_|\\_____|_| |_|
`)
console.log(`
`) `)
logger.info(` logger.info(`
- Server started successfully @ http://${host ? host : 'localhost'}:${port} - Server started successfully @ http://${host ? host : 'localhost'}:${port}

View File

@ -1,5 +1,6 @@
const CryptoJS = require('crypto-js') const CryptoJS = require('crypto-js')
const forge = require('node-forge') const forge = require('node-forge')
const zlib = require('zlib')
const iv = '0102030405060708' const iv = '0102030405060708'
const presetKey = '0CoJUm6Qyw8W8jud' const presetKey = '0CoJUm6Qyw8W8jud'
const linuxapiKey = 'rFgB&h#%2?^eDg:Q' const linuxapiKey = 'rFgB&h#%2?^eDg:Q'
@ -44,7 +45,7 @@ const aesDecrypt = (ciphertext, key, iv, format = 'base64') => {
}, },
) )
} }
return bytes.toString(CryptoJS.enc.Utf8) return bytes
} }
const rsaEncrypt = (str, key) => { const rsaEncrypt = (str, key) => {
const forgePublicKey = forge.pki.publicKeyFromPem(key) const forgePublicKey = forge.pki.publicKeyFromPem(key)
@ -85,20 +86,37 @@ const eapi = (url, object) => {
params: aesEncrypt(data, 'ecb', eapiKey, '', 'hex'), params: aesEncrypt(data, 'ecb', eapiKey, '', 'hex'),
} }
} }
const eapiResDecrypt = (encryptedParams) => { const eapiResDecrypt = (encryptedParams, aeapi = false) => {
// 使用aesDecrypt解密参数 // 使用aesDecrypt解密参数
try { try {
const decryptedData = aesDecrypt(encryptedParams, eapiKey, '', 'hex') const decrypted = aesDecrypt(encryptedParams, eapiKey, '', 'hex') // WordArray
return JSON.parse(decryptedData)
if (aeapi) {
// 带压缩的解密:先转 Base64 再解压
const decryptedBuffer = Buffer.from(
decrypted.toString(CryptoJS.enc.Base64),
'base64',
)
const decompressed = zlib.gunzipSync(decryptedBuffer)
return JSON.parse(decompressed.toString())
} else {
// 普通解密:直接转 UTF-8 字符串
return JSON.parse(decrypted.toString(CryptoJS.enc.Utf8))
}
} catch (error) { } catch (error) {
console.log('eapiResDecrypt error:', error) console.log(`eapiResDecrypt error:`, error)
return null return null
} }
} }
const eapiReqDecrypt = (encryptedParams) => { const eapiReqDecrypt = (encryptedParams) => {
// 使用aesDecrypt解密参数 // 使用 aesDecrypt 解密参数
const decryptedData = aesDecrypt(encryptedParams, eapiKey, '', 'hex') const decryptedData = aesDecrypt(
// 使用正则表达式解析出URL和数据 encryptedParams,
eapiKey,
'',
'hex',
).toString(CryptoJS.enc.Utf8)
// 使用正则表达式解析出 URL 和数据
const match = decryptedData.match(/(.*?)-36cd479b6b5-(.*?)-36cd479b6b5-(.*)/) const match = decryptedData.match(/(.*?)-36cd479b6b5-(.*?)-36cd479b6b5-(.*)/)
if (match) { if (match) {
const url = match[1] const url = match[1]
@ -106,7 +124,7 @@ const eapiReqDecrypt = (encryptedParams) => {
return { url, data } return { url, data }
} }
// 如果没有匹配到,返回null // 如果没有匹配到,返回 null
return null return null
} }
const decrypt = (cipher) => { const decrypt = (cipher) => {

View File

@ -3,7 +3,6 @@ const encrypt = require('./crypto')
const CryptoJS = require('crypto-js') const CryptoJS = require('crypto-js')
const { default: axios } = require('axios') const { default: axios } = require('axios')
const { PacProxyAgent } = require('pac-proxy-agent') const { PacProxyAgent } = require('pac-proxy-agent')
const logger = require('./logger')
const http = require('http') const http = require('http')
const https = require('https') const https = require('https')
const tunnel = require('tunnel') const tunnel = require('tunnel')
@ -160,10 +159,8 @@ const createRequest = (uri, data, options) => {
return new Promise((resolve, reject) => { return new Promise((resolve, reject) => {
// 变量声明和初始化 // 变量声明和初始化
const headers = options.headers ? { ...options.headers } : {} const headers = options.headers ? { ...options.headers } : {}
const ip = const ip = options.realIP || options.ip || ''
options.realIP ||
options.ip ||
(options.randomCNIP ? generateRandomChineseIP() : '')
// IP头设置 // IP头设置
if (ip) { if (ip) {
headers['X-Real-IP'] = ip headers['X-Real-IP'] = ip
@ -191,6 +188,13 @@ const createRequest = (uri, data, options) => {
const answer = { status: 500, body: {}, cookie: [] } const answer = { status: 500, body: {}, cookie: [] }
data.e_r = toBoolean(
options.e_r !== undefined
? options.e_r
: data.e_r !== undefined
? data.e_r
: ENCRYPT_RESPONSE,
)
// 根据加密方式处理 // 根据加密方式处理
switch (crypto) { switch (crypto) {
case 'weapi': case 'weapi':
@ -242,13 +246,7 @@ const createRequest = (uri, data, options) => {
if (crypto === 'eapi') { if (crypto === 'eapi') {
// headers['x-aeapi'] = true // 服务器会使用gzip压缩返回值 // headers['x-aeapi'] = true // 服务器会使用gzip压缩返回值
data.header = header data.header = header
data.e_r = toBoolean(
options.e_r !== undefined
? options.e_r
: data.e_r !== undefined
? data.e_r
: ENCRYPT_RESPONSE,
)
encryptData = encrypt.eapi(uri, data) encryptData = encrypt.eapi(uri, data)
url = (options.domain || API_DOMAIN) + '/eapi/' + uri.substr(5) url = (options.domain || API_DOMAIN) + '/eapi/' + uri.substr(5)
} else if (crypto === 'api') { } else if (crypto === 'api') {
@ -258,10 +256,10 @@ const createRequest = (uri, data, options) => {
break break
default: default:
logger.error('Unknown Crypto:', crypto) console.log('[ERR]', 'Unknown Crypto:', crypto)
break break
} }
// logger.info(url); // console.log(url);
// settings创建 // settings创建
let settings = { let settings = {
method: 'POST', method: 'POST',
@ -272,8 +270,9 @@ const createRequest = (uri, data, options) => {
httpsAgent: createHttpsAgent(), httpsAgent: createHttpsAgent(),
} }
// e_r处理 // 使用返回值加密
if (data.e_r) { const use_e_r = (crypto === 'eapi' || crypto === 'weapi') && data.e_r
if (use_e_r) {
settings.encoding = null settings.encoding = null
settings.responseType = 'arraybuffer' settings.responseType = 'arraybuffer'
} }
@ -303,16 +302,16 @@ const createRequest = (uri, data, options) => {
settings.httpAgent = agent settings.httpAgent = agent
settings.proxy = false settings.proxy = false
} else { } else {
logger.error('代理配置无效,不使用代理') console.error('代理配置无效,不使用代理')
} }
} catch (e) { } catch (e) {
logger.error('代理URL解析失败:', e.message) console.error('代理URL解析失败:', e.message)
} }
} }
} else { } else {
settings.proxy = false settings.proxy = false
} }
// logger.info(settings.headers); // console.log(settings.headers);
axios(settings) axios(settings)
.then((res) => { .then((res) => {
const body = res.data const body = res.data
@ -321,7 +320,7 @@ const createRequest = (uri, data, options) => {
) )
try { try {
if (crypto === 'eapi' && data.e_r) { if (use_e_r) {
answer.body = encrypt.eapiResDecrypt( answer.body = encrypt.eapiResDecrypt(
body.toString('hex').toUpperCase(), body.toString('hex').toUpperCase(),
headers['x-aeapi'], headers['x-aeapi'],
@ -352,14 +351,14 @@ const createRequest = (uri, data, options) => {
if (answer.status === 200) { if (answer.status === 200) {
resolve(answer) resolve(answer)
} else { } else {
logger.error(answer) console.log('[ERR]', answer)
reject(answer) reject(answer)
} }
}) })
.catch((err) => { .catch((err) => {
answer.status = 502 answer.status = 502
answer.body = { code: 502, msg: err.message || err } answer.body = { code: 502, msg: err.message || err }
logger.error(answer) console.log('[ERR]', answer)
reject(answer) reject(answer)
}) })
}) })