mirror of
				https://github.com/NeteaseCloudMusicApiEnhanced/api-enhanced.git
				synced 2025-10-25 09:23:11 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			136 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			136 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| const CryptoJS = require('crypto-js')
 | ||
| const forge = require('node-forge')
 | ||
| const iv = '0102030405060708'
 | ||
| const presetKey = '0CoJUm6Qyw8W8jud'
 | ||
| const linuxapiKey = 'rFgB&h#%2?^eDg:Q'
 | ||
| const base62 = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789'
 | ||
| const publicKey = `-----BEGIN PUBLIC KEY-----
 | ||
| MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQDgtQn2JZ34ZC28NWYpAUd98iZ37BUrX/aKzmFbt7clFSs6sXqHauqKWqdtLkF2KexO40H1YTX8z2lSgBBOAxLsvaklV8k4cBFK9snQXE9/DDaFt6Rr7iVZMldczhC0JNgTz+SHXT6CBHuX3e9SdB1Ua44oncaTWz7OBGLbCiK45wIDAQAB
 | ||
| -----END PUBLIC KEY-----`
 | ||
| const eapiKey = 'e82ckenh8dichen8'
 | ||
| 
 | ||
| const aesEncrypt = (text, mode, key, iv, format = 'base64') => {
 | ||
|   let encrypted = CryptoJS.AES.encrypt(
 | ||
|     CryptoJS.enc.Utf8.parse(text),
 | ||
|     CryptoJS.enc.Utf8.parse(key),
 | ||
|     {
 | ||
|       iv: CryptoJS.enc.Utf8.parse(iv),
 | ||
|       mode: CryptoJS.mode[mode.toUpperCase()],
 | ||
|       padding: CryptoJS.pad.Pkcs7,
 | ||
|     },
 | ||
|   )
 | ||
|   if (format === 'base64') {
 | ||
|     return encrypted.toString()
 | ||
|   }
 | ||
| 
 | ||
|   return encrypted.ciphertext.toString().toUpperCase()
 | ||
| }
 | ||
| const aesDecrypt = (ciphertext, key, iv, format = 'base64') => {
 | ||
|   let bytes
 | ||
|   if (format === 'base64') {
 | ||
|     bytes = CryptoJS.AES.decrypt(ciphertext, CryptoJS.enc.Utf8.parse(key), {
 | ||
|       iv: CryptoJS.enc.Utf8.parse(iv),
 | ||
|       mode: CryptoJS.mode.ECB,
 | ||
|       padding: CryptoJS.pad.Pkcs7,
 | ||
|     })
 | ||
|   } else {
 | ||
|     bytes = CryptoJS.AES.decrypt(
 | ||
|       { ciphertext: CryptoJS.enc.Hex.parse(ciphertext) },
 | ||
|       CryptoJS.enc.Utf8.parse(key),
 | ||
|       {
 | ||
|         iv: CryptoJS.enc.Utf8.parse(iv),
 | ||
|         mode: CryptoJS.mode.ECB,
 | ||
|         padding: CryptoJS.pad.Pkcs7,
 | ||
|       },
 | ||
|     )
 | ||
|   }
 | ||
|   return bytes.toString(CryptoJS.enc.Utf8)
 | ||
| }
 | ||
| const rsaEncrypt = (str, key) => {
 | ||
|   const forgePublicKey = forge.pki.publicKeyFromPem(key)
 | ||
|   const encrypted = forgePublicKey.encrypt(str, 'NONE')
 | ||
|   return forge.util.bytesToHex(encrypted)
 | ||
| }
 | ||
| 
 | ||
| const weapi = (object) => {
 | ||
|   const text = JSON.stringify(object)
 | ||
|   let secretKey = ''
 | ||
|   for (let i = 0; i < 16; i++) {
 | ||
|     secretKey += base62.charAt(Math.round(Math.random() * 61))
 | ||
|   }
 | ||
|   return {
 | ||
|     params: aesEncrypt(
 | ||
|       aesEncrypt(text, 'cbc', presetKey, iv),
 | ||
|       'cbc',
 | ||
|       secretKey,
 | ||
|       iv,
 | ||
|     ),
 | ||
|     encSecKey: rsaEncrypt(secretKey.split('').reverse().join(''), publicKey),
 | ||
|   }
 | ||
| }
 | ||
| 
 | ||
| const linuxapi = (object) => {
 | ||
|   const text = JSON.stringify(object)
 | ||
|   return {
 | ||
|     eparams: aesEncrypt(text, 'ecb', linuxapiKey, '', 'hex'),
 | ||
|   }
 | ||
| }
 | ||
| 
 | ||
| const eapi = (url, object) => {
 | ||
|   const text = typeof object === 'object' ? JSON.stringify(object) : object
 | ||
|   const message = `nobody${url}use${text}md5forencrypt`
 | ||
|   const digest = CryptoJS.MD5(message).toString()
 | ||
|   const data = `${url}-36cd479b6b5-${text}-36cd479b6b5-${digest}`
 | ||
|   return {
 | ||
|     params: aesEncrypt(data, 'ecb', eapiKey, '', 'hex'),
 | ||
|   }
 | ||
| }
 | ||
| const eapiResDecrypt = (encryptedParams) => {
 | ||
|   // 使用aesDecrypt解密参数
 | ||
|   try {
 | ||
|     const decryptedData = aesDecrypt(encryptedParams, eapiKey, '', 'hex')
 | ||
|     return JSON.parse(decryptedData)
 | ||
|   } catch (error) {
 | ||
|     console.log('eapiResDecrypt error:', error)
 | ||
|     return null
 | ||
|   }
 | ||
| }
 | ||
| const eapiReqDecrypt = (encryptedParams) => {
 | ||
|   // 使用aesDecrypt解密参数
 | ||
|   const decryptedData = aesDecrypt(encryptedParams, eapiKey, '', 'hex')
 | ||
|   // 使用正则表达式解析出URL和数据
 | ||
|   const match = decryptedData.match(/(.*?)-36cd479b6b5-(.*?)-36cd479b6b5-(.*)/)
 | ||
|   if (match) {
 | ||
|     const url = match[1]
 | ||
|     const data = JSON.parse(match[2])
 | ||
|     return { url, data }
 | ||
|   }
 | ||
| 
 | ||
|   // 如果没有匹配到,返回null
 | ||
|   return null
 | ||
| }
 | ||
| const decrypt = (cipher) => {
 | ||
|   const decipher = CryptoJS.AES.decrypt(
 | ||
|     {
 | ||
|       ciphertext: CryptoJS.enc.Hex.parse(cipher),
 | ||
|     },
 | ||
|     eapiKey,
 | ||
|     {
 | ||
|       mode: CryptoJS.mode.ECB,
 | ||
|     },
 | ||
|   )
 | ||
|   const decryptedBytes = CryptoJS.enc.Utf8.stringify(decipher)
 | ||
|   return decryptedBytes
 | ||
| }
 | ||
| 
 | ||
| module.exports = {
 | ||
|   weapi,
 | ||
|   linuxapi,
 | ||
|   eapi,
 | ||
|   decrypt,
 | ||
|   aesEncrypt,
 | ||
|   aesDecrypt,
 | ||
|   eapiReqDecrypt,
 | ||
|   eapiResDecrypt,
 | ||
| }
 | 
