是不是在为ncm下载的格式烦恼,下载之后本地还没办法播放,ncm加密格式导致许多地方都没办法使用,不可以设置为铃声,剪辑视频也无法使用等等,这份代码解决你的问题。
一、环境配置以及库的安装
安装库 Crypto是最麻烦的,首先需要把之前相关的库删除若是有的话
pip uninstall crypto
pip uninstall pycryptodome
接下来安装这一个库:
pip install pycryptodome
下载之后如果引用没有出现错误,则可以忽略这一节的下一步。
如果出现了引用错误, 恭喜你,接下来你手得酸了,需要手动把所有大写C全部改为小写c。
运行的时候如果报错未找到Crypto,点进去报错位置,将Crypto改为crypyo即可以解决了,这个过程会比较难熬,坚持住也就十几个文件。
二、代码文件
话不多说,直接上代码:
import binascii
import struct
import base64
import json
import os
from crypto.Cipher import AES # 下载之后可能会出现一些错误,将所有的库内文件from Crypto 改为crypto
# 另外还需要将下载的包,通常是:c:\Python36\Lib\site-packages\Crypto同样改成小写可以解决
def dump(file_path):
# 十六进制转字符串
core_key = binascii.a2b_hex("687A4852416D736F356B496E62617857")
meta_key = binascii.a2b_hex("2331346C6A6B5F215C5D2630553C2728")
unpad = lambda s: s[0:-(s[-1] if type(s[-1]) == int else ord(s[-1]))]
f = open(file_path, 'rb')
header = f.read(8)
# 字符串转十六进制
assert binascii.b2a_hex(header) == b'4354454e4644414d'
f.seek(2, 1)
key_length = f.read(4)
key_length = struct.unpack('<I', bytes(key_length))[0]
key_data = f.read(key_length)
key_data_array = bytearray(key_data)
for i in range(0, len(key_data_array)):
key_data_array[i] ^= 0x64
key_data = bytes(key_data_array)
cryptor = AES.new(core_key, AES.MODE_ECB)
key_data = unpad(cryptor.decrypt(key_data))[17:]
key_length = len(key_data)
key_data = bytearray(key_data)
key_box = bytearray(range(256))
c = 0
last_byte = 0
key_offset = 0
for i in range(256):
swap = key_box[i]
c = (swap + last_byte + key_data[key_offset]) & 0xff
key_offset += 1
if key_offset >= key_length:
key_offset = 0
key_box[i] = key_box[c]
key_box[c] = swap
last_byte = c
meta_length = f.read(4)
meta_length = struct.unpack('<I', bytes(meta_length))[0]
meta_data = f.read(meta_length)
meta_data_array = bytearray(meta_data)
for i in range(0, len(meta_data_array)):
meta_data_array[i] ^= 0x63
meta_data = bytes(meta_data_array)
meta_data = base64.b64decode(meta_data[22:])
cryptor = AES.new(meta_key, AES.MODE_ECB)
meta_data = unpad(cryptor.decrypt(meta_data)).decode('utf-8')[6:]
meta_data = json.loads(meta_data)
crc32 = f.read(4)
crc32 = struct.unpack('<I', bytes(crc32))[0]
f.seek(5, 1)
image_size = f.read(4)
image_size = struct.unpack('<I', bytes(image_size))[0]
image_data = f.read(image_size)
file_name = f.name.split("/")[-1].split(".ncm")[0] + '.' + meta_data['format']
m = open(os.path.join(os.path.split(file_path)[0], file_name), 'wb')
chunk = bytearray()
while True:
chunk = bytearray(f.read(0x8000))
chunk_length = len(chunk)
if not chunk:
break
for i in range(1, chunk_length + 1):
j = i & 0xff
chunk[i - 1] ^= key_box[(key_box[j] + key_box[(key_box[j] + j) & 0xff]) & 0xff]
m.write(chunk)
m.close()
f.close()
return file_name
if __name__ == '__main__':
ncm_list = r'D:\CloudMusic\VipSongsDownload'
mp3_list = r'D:\CloudMusic\VipSongsDownload\mp3'
# 遍历目录
for root, dirs, files in os.walk(ncm_list):
for file in files:
# 检查文件扩展名是否为 .ncm
if file.endswith('.ncm'):
# 拼接完整的文件路径
full_path = os.path.join(root, file)
print(f"{file}正在转换为mp3格式") # 打印文件的完整路
MP3_name = dump(full_path)
base, mp3_file = os.path.split(MP3_name)
os.rename(MP3_name, os.path.join(mp3_list, mp3_file)) # 不想把两种格式文件放在一起,放入专门的MP3文件夹
使用的时候在main中只需要更改ncm_list和mp3_list的地址即可,这些地址都是要真实存在且可以访问到的,会将该ncm_list目录下的所有ncm文件全部转换为MP3文件,且放入mp3_list之中。
希望可以帮助到大家。