2022-12-04 16:06:38 +00:00
|
|
|
package qmc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"runtime"
|
|
|
|
|
2022-12-05 01:43:33 +00:00
|
|
|
"github.com/samber/lo"
|
2022-12-05 00:54:40 +00:00
|
|
|
"go.uber.org/zap"
|
2022-12-05 01:43:33 +00:00
|
|
|
"golang.org/x/exp/slices"
|
2022-12-05 03:04:57 +00:00
|
|
|
"golang.org/x/text/unicode/norm"
|
2024-02-14 08:59:05 +00:00
|
|
|
"unlock-music.dev/mmkv"
|
2022-12-04 16:06:38 +00:00
|
|
|
)
|
|
|
|
|
2022-12-04 23:24:03 +00:00
|
|
|
var streamKeyVault mmkv.Vault
|
2022-12-04 16:06:38 +00:00
|
|
|
|
2022-12-05 00:54:40 +00:00
|
|
|
// TODO: move to factory
|
|
|
|
func readKeyFromMMKV(file string, logger *zap.Logger) ([]byte, error) {
|
2022-12-04 16:06:38 +00:00
|
|
|
if file == "" {
|
|
|
|
return nil, errors.New("file path is required while reading key from mmkv")
|
|
|
|
}
|
|
|
|
|
|
|
|
//goland:noinspection GoBoolExpressions
|
|
|
|
if runtime.GOOS != "darwin" {
|
|
|
|
return nil, errors.New("mmkv vault not supported on this platform")
|
|
|
|
}
|
|
|
|
|
|
|
|
if streamKeyVault == nil {
|
|
|
|
mmkvDir, err := getRelativeMMKVDir(file)
|
|
|
|
if err != nil {
|
|
|
|
mmkvDir, err = getDefaultMMKVDir()
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("mmkv key valut not found: %w", err)
|
|
|
|
}
|
|
|
|
}
|
2022-12-05 00:54:40 +00:00
|
|
|
|
2022-12-04 23:24:03 +00:00
|
|
|
mgr, err := mmkv.NewManager(mmkvDir)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("init mmkv manager: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
streamKeyVault, err = mgr.OpenVault("MMKVStreamEncryptId")
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("open mmkv vault: %w", err)
|
|
|
|
}
|
2022-12-05 00:54:40 +00:00
|
|
|
|
|
|
|
logger.Debug("mmkv vault opened", zap.Strings("keys", streamKeyVault.Keys()))
|
2022-12-04 16:06:38 +00:00
|
|
|
}
|
|
|
|
|
2022-12-05 01:43:33 +00:00
|
|
|
_, partName := filepath.Split(file)
|
2022-12-05 03:04:57 +00:00
|
|
|
partName = normalizeUnicode(partName)
|
2022-12-05 01:43:33 +00:00
|
|
|
|
2022-12-05 03:04:57 +00:00
|
|
|
buf, err := streamKeyVault.GetBytes(file)
|
2022-12-05 01:43:33 +00:00
|
|
|
if buf == nil {
|
|
|
|
filePaths := streamKeyVault.Keys()
|
2022-12-05 03:04:57 +00:00
|
|
|
fileNames := lo.Map(filePaths, func(filePath string, _ int) string {
|
|
|
|
_, name := filepath.Split(filePath)
|
|
|
|
return normalizeUnicode(name)
|
|
|
|
})
|
|
|
|
|
|
|
|
for _, key := range fileNames { // fallback: match filename only
|
|
|
|
if key != partName {
|
2022-12-04 23:24:03 +00:00
|
|
|
continue
|
|
|
|
}
|
2022-12-05 03:04:57 +00:00
|
|
|
idx := slices.Index(fileNames, key)
|
|
|
|
buf, err = streamKeyVault.GetBytes(filePaths[idx])
|
2022-12-04 23:24:03 +00:00
|
|
|
if err != nil {
|
2022-12-05 03:04:57 +00:00
|
|
|
logger.Warn("read key from mmkv", zap.String("key", filePaths[idx]), zap.Error(err))
|
2022-12-04 16:06:38 +00:00
|
|
|
}
|
|
|
|
}
|
2022-12-05 01:43:33 +00:00
|
|
|
|
2022-12-04 16:06:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(buf) == 0 {
|
|
|
|
return nil, errors.New("key not found in mmkv vault")
|
|
|
|
}
|
|
|
|
|
|
|
|
return deriveKey(buf)
|
|
|
|
}
|
|
|
|
|
2024-02-12 16:58:13 +00:00
|
|
|
func OpenMMKV(vaultPath string, vaultKey string, logger *zap.Logger) error {
|
|
|
|
filePath, fileName := filepath.Split(vaultPath)
|
|
|
|
mgr, err := mmkv.NewManager(filepath.Dir(filePath))
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("init mmkv manager: %w", err)
|
|
|
|
}
|
2024-02-05 00:41:40 +00:00
|
|
|
|
2024-02-12 16:58:13 +00:00
|
|
|
streamKeyVault, err = mgr.OpenVaultCrypto(fileName, vaultKey)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("open mmkv vault: %w", err)
|
|
|
|
}
|
2024-02-05 00:41:40 +00:00
|
|
|
|
2024-02-12 16:58:13 +00:00
|
|
|
logger.Debug("mmkv vault opened", zap.Strings("keys", streamKeyVault.Keys()))
|
|
|
|
return nil
|
|
|
|
}
|
2024-02-05 00:41:40 +00:00
|
|
|
|
2024-02-12 18:41:42 +00:00
|
|
|
func readKeyFromMMKVCustom(mid string) ([]byte, error) {
|
2024-02-12 16:58:13 +00:00
|
|
|
if streamKeyVault == nil {
|
|
|
|
return nil, fmt.Errorf("mmkv vault not loaded")
|
2024-02-05 00:41:40 +00:00
|
|
|
}
|
|
|
|
|
2024-02-12 18:41:42 +00:00
|
|
|
// get ekey from mmkv vault
|
|
|
|
eKey, err := streamKeyVault.GetBytes(mid)
|
2024-02-05 00:41:40 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("get eKey error: %w", err)
|
|
|
|
}
|
2024-02-14 08:59:05 +00:00
|
|
|
return deriveKey(eKey)
|
2024-02-05 00:41:40 +00:00
|
|
|
}
|
|
|
|
|
2022-12-04 16:06:38 +00:00
|
|
|
func getRelativeMMKVDir(file string) (string, error) {
|
|
|
|
mmkvDir := filepath.Join(filepath.Dir(file), "../mmkv")
|
|
|
|
if _, err := os.Stat(mmkvDir); err != nil {
|
|
|
|
return "", fmt.Errorf("stat default mmkv dir: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
keyFile := filepath.Join(mmkvDir, "MMKVStreamEncryptId")
|
|
|
|
if _, err := os.Stat(keyFile); err != nil {
|
|
|
|
return "", fmt.Errorf("stat default mmkv file: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return mmkvDir, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getDefaultMMKVDir() (string, error) {
|
|
|
|
homeDir, err := os.UserHomeDir()
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("get user home dir: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
mmkvDir := filepath.Join(
|
|
|
|
homeDir,
|
|
|
|
"Library/Containers/com.tencent.QQMusicMac/Data", // todo: make configurable
|
|
|
|
"Library/Application Support/QQMusicMac/mmkv",
|
|
|
|
)
|
|
|
|
if _, err := os.Stat(mmkvDir); err != nil {
|
|
|
|
return "", fmt.Errorf("stat default mmkv dir: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
keyFile := filepath.Join(mmkvDir, "MMKVStreamEncryptId")
|
|
|
|
if _, err := os.Stat(keyFile); err != nil {
|
|
|
|
return "", fmt.Errorf("stat default mmkv file: %w", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return mmkvDir, nil
|
|
|
|
}
|
2022-12-05 03:04:57 +00:00
|
|
|
|
2022-12-05 03:10:40 +00:00
|
|
|
// normalizeUnicode normalizes unicode string to NFC.
|
|
|
|
// since macOS may change some characters in the file name.
|
|
|
|
// e.g. "ぜ"(e3 81 9c) -> "ぜ"(e3 81 9b e3 82 99)
|
2022-12-05 03:04:57 +00:00
|
|
|
func normalizeUnicode(str string) string {
|
|
|
|
return norm.NFC.String(str)
|
|
|
|
}
|