青梅1219
港岛妹妹

前言:

最近想通过pr剪辑几个视频,然后发现pr导入不了ncm格式的音频。于是想把ncm格式转化下。

一、环境:

Python版本:3.7.9

Python库:Crypto

安装Crypto:pip install Crypto

我安装后,Crypto库默认为crypto

检查crypto中是否存在Cipher,如果没有需要安装pycrypto:执行pip install pycrypto

需要把crypto文件夹名改为Crypto

Ps:

Crypto:加密算法库

二、代码

 1 import binascii
 2 import struct
 3 import base64
 4 import json
 5 import os,sys
 6 from Crypto.Cipher import AES
 7 
 8 
 9 def dump(file_path):
10     core_key = binascii.a2b_hex("687A4852416D736F356B496E62617857")
11     meta_key = binascii.a2b_hex("2331346C6A6B5F215C5D2630553C2728")
12     unpad = lambda s : s[0:-(s[-1] if type(s[-1]) == int else ord(s[-1]))]
13     f = open(file_path,'rb')
14     header = f.read(8)
15     assert binascii.b2a_hex(header) == b'4354454e4644414d'
16     f.seek(2, 1)
17     key_length = f.read(4)
18     key_length = struct.unpack('<I', bytes(key_length))[0]
19     key_data = f.read(key_length)
20     key_data_array = bytearray(key_data)
21     for i in range (0,len(key_data_array)): key_data_array[i] ^= 0x64
22     key_data = bytes(key_data_array)
23     cryptor = AES.new(core_key, AES.MODE_ECB)
24     key_data = unpad(cryptor.decrypt(key_data))[17:]
25     key_length = len(key_data)
26     key_data = bytearray(key_data)
27     key_box = bytearray(range(256))
28     c = 0
29     last_byte = 0
30     key_offset = 0
31     for i in range(256):
32         swap = key_box[i]
33         c = (swap + last_byte + key_data[key_offset]) & 0xff
34         key_offset += 1
35         if key_offset >= key_length: key_offset = 0
36         key_box[i] = key_box[c]
37         key_box[c] = swap
38         last_byte = c
39     meta_length = f.read(4)
40     meta_length = struct.unpack('<I', bytes(meta_length))[0]
41     meta_data = f.read(meta_length)
42     meta_data_array = bytearray(meta_data)
43     for i in range(0,len(meta_data_array)): meta_data_array[i] ^= 0x63
44     meta_data = bytes(meta_data_array)
45     meta_data = base64.b64decode(meta_data[22:])
46     cryptor = AES.new(meta_key, AES.MODE_ECB)
47     meta_data = unpad(cryptor.decrypt(meta_data)).decode('utf-8')[6:]
48     meta_data = json.loads(meta_data)
49     crc32 = f.read(4)
50     crc32 = struct.unpack('<I', bytes(crc32))[0]
51     f.seek(5, 1)
52     image_size = f.read(4)
53     image_size = struct.unpack('<I', bytes(image_size))[0]
54     image_data = f.read(image_size)
55     file_name = meta_data['musicName'] + '.' + meta_data['format']
56     m = open(os.path.join(os.path.split(file_path)[0],file_name),'wb')
57     chunk = bytearray()
58     while True:
59         chunk = bytearray(f.read(0x8000))
60         chunk_length = len(chunk)
61         if not chunk:
62             break
63         for i in range(1,chunk_length+1):
64             j = i & 0xff;
65             chunk[i-1] ^= key_box[(key_box[j] + key_box[(key_box[j] + j) & 0xff]) & 0xff]
66         m.write(chunk)
67     m.close()
68     f.close()
69 
70 
71 def file_name(file_dir):
72     for root, dirs, files in os.walk(file_dir):
73         for file in files:
74             if os.path.splitext(file)[1] == '.ncm':
75                 dump(file)
76                 print(file)
77 
78 
79 if __name__ == '__main__':
80     try:
81         # os.getcwdb获取该文件所在的路径
82         print(os.getcwd())
83         file_name(os.getcwd())
84     except Exception as e:
85         print(e)

三、使用

将该文件放在于ncm同一个文件夹下,然后执行脚本

四、格式介绍

有损音乐格式主要有:MP3、WMA、AAC等等,当年周杰伦时代我们手里拿的纽曼mp3随身听,见到最多的格式就是这三种,一首大概3M左右。对文件进行了不可逆向的压缩,从频谱上看上半部分都被切掉了。 用比如spek这类专业的音频分析软件,可以验证出一首无损音乐是真无损还是败絮其中的假无损。

无损音乐格式主要有WAV、AIFF、FLAC、APE、ALAC等等,WAV、AIFF是CD的数字记录,基本上是由CD转成数字形式所得到的文件;而APE、FLAC等是对WAV文件进行无损压缩后得到的,即再加工得到。所以,wav是无损音乐文件,而APE、FLAC等则是无损压缩音乐文件。AIFF是苹果自研,WAV是微软自研。

五、展示

 

6.参考

https://github.com/lianglixin/ncmdump

 

posted on 2022-06-10 17:31  青梅1219  阅读(1297)  评论(0编辑  收藏  举报