ncmdump/ncmcrypt.cpp

407 lines
9.6 KiB
C++
Raw Normal View History

2021-10-07 00:34:29 +08:00
#include "ncmcrypt.h"
#include "aes.h"
#include "base64.h"
#include "cJSON.h"
#define TAGLIB_STATIC
#include "taglib/toolkit/tfile.h"
#include "taglib/mpeg/mpegfile.h"
#include "taglib/flac/flacfile.h"
#include "taglib/mpeg/id3v2/frames/attachedpictureframe.h"
#include "taglib/mpeg/id3v2/id3v2tag.h"
#include "taglib/tag.h"
2021-10-07 00:34:29 +08:00
#include <stdexcept>
#include <string>
#include <filesystem>
2024-03-22 12:07:32 +08:00
const unsigned char NeteaseCrypt::sCoreKey[17] = {0x68, 0x7A, 0x48, 0x52, 0x41, 0x6D, 0x73, 0x6F, 0x35, 0x6B, 0x49, 0x6E, 0x62, 0x61, 0x78, 0x57, 0};
2021-10-07 00:34:29 +08:00
const unsigned char NeteaseCrypt::sModifyKey[17] = {0x23, 0x31, 0x34, 0x6C, 0x6A, 0x6B, 0x5F, 0x21, 0x5C, 0x5D, 0x26, 0x30, 0x55, 0x3C, 0x27, 0x28, 0};
const unsigned char NeteaseCrypt::mPng[8] = {0x89, 0x50, 0x4E, 0x47, 0x0D, 0x0A, 0x1A, 0x0A};
2024-03-22 12:07:32 +08:00
static void aesEcbDecrypt(const unsigned char *key, std::string &src, std::string &dst)
{
int n, i;
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
unsigned char out[16];
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
n = src.length() >> 4;
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
dst.clear();
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
AES aes(key);
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
for (i = 0; i < n - 1; i++)
{
aes.decrypt((unsigned char *)src.c_str() + (i << 4), out);
dst += std::string((char *)out, 16);
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
aes.decrypt((unsigned char *)src.c_str() + (i << 4), out);
char pad = out[15];
if (pad > 16)
{
pad = 0;
}
dst += std::string((char *)out, 16 - pad);
2021-10-07 00:34:29 +08:00
}
2024-03-22 12:07:32 +08:00
static void replace(std::string &str, const std::string &from, const std::string &to)
{
if (from.empty())
return;
size_t start_pos = 0;
while ((start_pos = str.find(from, start_pos)) != std::string::npos)
{
str.replace(start_pos, from.length(), to);
start_pos += to.length(); // In case 'to' contains 'from', like replacing 'x' with 'yx'
}
2021-10-07 00:34:29 +08:00
}
2024-03-22 12:07:32 +08:00
NeteaseMusicMetadata::~NeteaseMusicMetadata()
{
cJSON_Delete(mRaw);
2021-10-07 00:34:29 +08:00
}
2024-03-22 12:07:32 +08:00
NeteaseMusicMetadata::NeteaseMusicMetadata(cJSON *raw)
{
if (!raw)
{
return;
}
cJSON *swap;
int artistLen, i;
mRaw = raw;
swap = cJSON_GetObjectItem(raw, "musicName");
if (swap)
{
mName = std::string(cJSON_GetStringValue(swap));
}
swap = cJSON_GetObjectItem(raw, "album");
if (swap)
{
mAlbum = std::string(cJSON_GetStringValue(swap));
}
swap = cJSON_GetObjectItem(raw, "artist");
if (swap)
{
artistLen = cJSON_GetArraySize(swap);
i = 0;
for (i = 0; i < artistLen - 1; i++)
{
mArtist += std::string(cJSON_GetStringValue(cJSON_GetArrayItem(cJSON_GetArrayItem(swap, i), 0)));
mArtist += "/";
}
mArtist += std::string(cJSON_GetStringValue(cJSON_GetArrayItem(cJSON_GetArrayItem(swap, i), 0)));
}
swap = cJSON_GetObjectItem(raw, "bitrate");
if (swap)
{
mBitrate = swap->valueint;
}
swap = cJSON_GetObjectItem(raw, "duration");
if (swap)
{
mDuration = swap->valueint;
}
swap = cJSON_GetObjectItem(raw, "format");
if (swap)
{
mFormat = std::string(cJSON_GetStringValue(swap));
}
2021-10-07 00:34:29 +08:00
}
2024-03-22 12:07:32 +08:00
bool NeteaseCrypt::openFile(std::string const &path)
{
mFile.open(std::filesystem::u8path(path), std::ios::in | std::ios::binary);
if (!mFile.is_open())
{
return false;
}
else
{
return true;
}
2021-10-07 00:34:29 +08:00
}
2024-03-22 12:07:32 +08:00
bool NeteaseCrypt::isNcmFile()
{
unsigned int header;
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
mFile.read(reinterpret_cast<char *>(&header), sizeof(header));
if (header != (unsigned int)0x4e455443)
{
return false;
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
mFile.read(reinterpret_cast<char *>(&header), sizeof(header));
if (header != (unsigned int)0x4d414446)
{
return false;
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
return true;
2021-10-07 00:34:29 +08:00
}
2024-03-22 12:07:32 +08:00
int NeteaseCrypt::read(char *s, std::streamsize n)
{
mFile.read(s, n);
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
int gcount = mFile.gcount();
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
if (gcount <= 0)
{
throw std::invalid_argument("Can't read file");
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
return gcount;
2021-10-07 00:34:29 +08:00
}
2024-03-22 12:07:32 +08:00
void NeteaseCrypt::buildKeyBox(unsigned char *key, int keyLen)
{
int i;
for (i = 0; i < 256; ++i)
{
mKeyBox[i] = (unsigned char)i;
}
unsigned char swap = 0;
unsigned char c = 0;
unsigned char last_byte = 0;
unsigned char key_offset = 0;
for (i = 0; i < 256; ++i)
{
swap = mKeyBox[i];
c = ((swap + last_byte + key[key_offset++]) & 0xff);
if (key_offset >= keyLen)
key_offset = 0;
mKeyBox[i] = mKeyBox[c];
mKeyBox[c] = swap;
last_byte = c;
}
2021-10-07 00:34:29 +08:00
}
2024-03-22 12:07:32 +08:00
std::string NeteaseCrypt::mimeType(std::string &data)
{
if (memcmp(data.c_str(), mPng, 8) == 0)
{
return std::string("image/png");
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
return std::string("image/jpeg");
2021-10-07 00:34:29 +08:00
}
2024-03-22 12:07:32 +08:00
void NeteaseCrypt::FixMetadata()
{
TagLib::File *audioFile;
TagLib::Tag *tag;
TagLib::ByteVector vector(mImageData.c_str(), mImageData.length());
if (mFormat == NeteaseCrypt::MP3)
{
audioFile = new TagLib::MPEG::File(mDumpFilepath.c_str());
tag = dynamic_cast<TagLib::MPEG::File *>(audioFile)->ID3v2Tag(true);
if (mImageData.length() > 0)
{
TagLib::ID3v2::AttachedPictureFrame *frame = new TagLib::ID3v2::AttachedPictureFrame;
frame->setMimeType(mimeType(mImageData));
frame->setPicture(vector);
dynamic_cast<TagLib::ID3v2::Tag *>(tag)->addFrame(frame);
}
}
else if (mFormat == NeteaseCrypt::FLAC)
{
audioFile = new TagLib::FLAC::File(mDumpFilepath.c_str());
tag = audioFile->tag();
if (mImageData.length() > 0)
{
TagLib::FLAC::Picture *cover = new TagLib::FLAC::Picture;
cover->setMimeType(mimeType(mImageData));
cover->setType(TagLib::FLAC::Picture::FrontCover);
cover->setData(vector);
dynamic_cast<TagLib::FLAC::File *>(audioFile)->addPicture(cover);
}
}
if (mMetaData != NULL)
{
tag->setTitle(TagLib::String(mMetaData->name(), TagLib::String::UTF8));
tag->setArtist(TagLib::String(mMetaData->artist(), TagLib::String::UTF8));
tag->setAlbum(TagLib::String(mMetaData->album(), TagLib::String::UTF8));
}
tag->setComment(TagLib::String("Create by netease copyright protected dump tool. author 5L", TagLib::String::UTF8));
audioFile->save();
2021-10-07 00:34:29 +08:00
}
2024-03-22 12:07:32 +08:00
void NeteaseCrypt::Dump()
{
mDumpFilepath = std::filesystem::u8path(mFilepath);
std::vector<unsigned char> buffer(0x8000);
std::ofstream output;
while (!mFile.eof())
{
int n = read((char *)buffer.data(), buffer.size());
for (int i = 0; i < n; i++)
{
int j = (i + 1) & 0xff;
buffer[i] ^= mKeyBox[(mKeyBox[j] + mKeyBox[(mKeyBox[j] + j) & 0xff]) & 0xff];
}
if (!output.is_open())
{
// identify format
// ID3 format mp3
if (buffer[0] == 0x49 && buffer[1] == 0x44 && buffer[2] == 0x33)
{
mDumpFilepath = mDumpFilepath.replace_extension("mp3");
mFormat = NeteaseCrypt::MP3;
}
else
{
mDumpFilepath = mDumpFilepath.replace_extension("flac");
mFormat = NeteaseCrypt::FLAC;
}
output.open(mDumpFilepath, output.out | output.binary);
}
output.write((char *)buffer.data(), n);
}
output.flush();
output.close();
2021-10-07 00:34:29 +08:00
}
2024-03-22 12:07:32 +08:00
NeteaseCrypt::~NeteaseCrypt()
{
if (mMetaData != NULL)
{
delete mMetaData;
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
mFile.close();
2021-10-07 00:34:29 +08:00
}
2024-03-22 12:07:32 +08:00
NeteaseCrypt::NeteaseCrypt(std::string const &path)
{
if (!openFile(path))
{
throw std::invalid_argument("Can't open file");
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
if (!isNcmFile())
{
throw std::invalid_argument("Not netease protected file");
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
if (!mFile.seekg(2, mFile.cur))
{
throw std::invalid_argument("Can't seek file");
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
mFilepath = path;
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
unsigned int n;
read(reinterpret_cast<char *>(&n), sizeof(n));
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
if (n <= 0)
{
throw std::invalid_argument("Broken NCM file");
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
std::vector<char> keydata(n);
read(keydata.data(), n);
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
for (size_t i = 0; i < n; i++)
{
keydata[i] ^= 0x64;
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
std::string rawKeyData(keydata.begin(), keydata.end());
std::string mKeyData;
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
aesEcbDecrypt(sCoreKey, rawKeyData, mKeyData);
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
buildKeyBox((unsigned char *)mKeyData.c_str() + 17, mKeyData.length() - 17);
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
read(reinterpret_cast<char *>(&n), sizeof(n));
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
if (n <= 0)
{
std::cout << "[Warn] " << path << " missing metadata infomation can't fix some infomation!" << std::endl;
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
mMetaData = NULL;
}
else
{
std::vector<char> modifyData(n);
read(modifyData.data(), n);
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
for (size_t i = 0; i < n; i++)
{
modifyData[i] ^= 0x63;
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
std::string swapModifyData;
std::string modifyOutData;
std::string modifyDecryptData;
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
swapModifyData = std::string(modifyData.begin() + 22, modifyData.end());
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
// escape `163 key(Don't modify):`
Base64::Decode(swapModifyData, modifyOutData);
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
aesEcbDecrypt(sModifyKey, modifyOutData, modifyDecryptData);
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
// escape `music:`
modifyDecryptData = std::string(modifyDecryptData.begin() + 6, modifyDecryptData.end());
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
// std::cout << modifyDecryptData << std::endl;
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
mMetaData = new NeteaseMusicMetadata(cJSON_Parse(modifyDecryptData.c_str()));
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
// skip crc32 & unuse charset
if (!mFile.seekg(9, mFile.cur))
{
throw std::invalid_argument("can't seek file");
}
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
read(reinterpret_cast<char *>(&n), sizeof(n));
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
if (n > 0)
{
char *imageData = (char *)malloc(n);
read(imageData, n);
2021-10-07 00:34:29 +08:00
2024-03-22 12:07:32 +08:00
mImageData = std::string(imageData, n);
}
else
{
std::cout << "[Warn] " << path << " missing album can't fix album image!" << std::endl;
}
2021-10-07 00:34:29 +08:00
}