You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
129 lines
3.7 KiB
Go
129 lines
3.7 KiB
Go
4 years ago
|
package qmc
|
||
|
|
||
|
import (
|
||
|
"encoding/base64"
|
||
|
"encoding/binary"
|
||
|
"errors"
|
||
4 years ago
|
"github.com/unlock-music/cli/algo/common"
|
||
4 years ago
|
)
|
||
|
|
||
|
var (
|
||
|
ErrQmcFileLength = errors.New("invalid qmc file length")
|
||
|
ErrQmcKeyDecodeFailed = errors.New("base64 decode qmc key failed")
|
||
|
ErrQmcKeyLength = errors.New("unexpected decoded qmc key length")
|
||
|
)
|
||
|
|
||
|
type Decoder struct {
|
||
|
file []byte
|
||
|
maskDetector func(encodedData []byte) (*Key256Mask, error)
|
||
|
mask *Key256Mask
|
||
|
audioExt string
|
||
|
key []byte
|
||
|
audio []byte
|
||
|
}
|
||
|
|
||
4 years ago
|
func NewMflac256Decoder(data []byte) common.Decoder {
|
||
4 years ago
|
return &Decoder{file: data, maskDetector: detectMflac256Mask, audioExt: "flac"}
|
||
|
}
|
||
|
|
||
4 years ago
|
func NewMgg256Decoder(data []byte) common.Decoder {
|
||
4 years ago
|
return &Decoder{file: data, maskDetector: detectMgg256Mask, audioExt: "ogg"}
|
||
|
}
|
||
|
|
||
4 years ago
|
func (d *Decoder) Validate() error {
|
||
4 years ago
|
if nil != d.mask {
|
||
4 years ago
|
return nil
|
||
4 years ago
|
}
|
||
|
if nil != d.maskDetector {
|
||
|
if err := d.validateKey(); err != nil {
|
||
4 years ago
|
return err
|
||
4 years ago
|
}
|
||
4 years ago
|
var err error
|
||
|
d.mask, err = d.maskDetector(d.file)
|
||
|
return err
|
||
4 years ago
|
}
|
||
4 years ago
|
return errors.New("no mask or mask detector found")
|
||
4 years ago
|
}
|
||
|
|
||
|
func (d *Decoder) validateKey() error {
|
||
|
lenData := len(d.file)
|
||
|
if lenData < 4 {
|
||
|
return ErrQmcFileLength
|
||
|
}
|
||
|
|
||
|
keyLen := binary.LittleEndian.Uint32(d.file[lenData-4:])
|
||
|
if lenData < int(keyLen+4) {
|
||
|
return ErrQmcFileLength
|
||
|
}
|
||
|
var err error
|
||
|
d.key, err = base64.StdEncoding.DecodeString(
|
||
|
string(d.file[lenData-4-int(keyLen) : lenData-4]))
|
||
|
if err != nil {
|
||
|
return ErrQmcKeyDecodeFailed
|
||
|
}
|
||
|
|
||
|
if len(d.key) != 272 {
|
||
|
return ErrQmcKeyLength
|
||
|
}
|
||
|
d.file = d.file[:lenData-4-int(keyLen)]
|
||
|
return nil
|
||
|
|
||
|
}
|
||
|
|
||
|
func (d *Decoder) Decode() error {
|
||
|
d.audio = d.mask.Decrypt(d.file)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (d Decoder) GetCoverImage() []byte {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (d Decoder) GetAudioData() []byte {
|
||
|
return d.audio
|
||
|
}
|
||
|
|
||
|
func (d Decoder) GetAudioExt() string {
|
||
4 years ago
|
if d.audioExt != "" {
|
||
|
return "." + d.audioExt
|
||
|
}
|
||
|
return ""
|
||
4 years ago
|
}
|
||
|
|
||
|
func (d Decoder) GetMeta() common.Meta {
|
||
|
return nil
|
||
|
}
|
||
4 years ago
|
|
||
4 years ago
|
func DecoderFuncWithExt(ext string) common.NewDecoderFunc {
|
||
|
return func(file []byte) common.Decoder {
|
||
4 years ago
|
return &Decoder{file: file, audioExt: ext, mask: getDefaultMask()}
|
||
4 years ago
|
}
|
||
|
}
|
||
|
|
||
|
//goland:noinspection SpellCheckingInspection
|
||
4 years ago
|
func init() {
|
||
3 years ago
|
common.RegisterDecoder("qmc0", false, DecoderFuncWithExt("mp3")) //QQ Music Mp3
|
||
|
common.RegisterDecoder("qmc3", false, DecoderFuncWithExt("mp3")) //QQ Music Mp3
|
||
4 years ago
|
|
||
3 years ago
|
common.RegisterDecoder("qmc2", false, DecoderFuncWithExt("m4a")) //QQ Music M4A
|
||
|
common.RegisterDecoder("qmc4", false, DecoderFuncWithExt("m4a")) //QQ Music M4A
|
||
|
common.RegisterDecoder("qmc6", false, DecoderFuncWithExt("m4a")) //QQ Music M4A
|
||
|
common.RegisterDecoder("qmc8", false, DecoderFuncWithExt("m4a")) //QQ Music M4A
|
||
4 years ago
|
|
||
3 years ago
|
common.RegisterDecoder("qmcflac", false, DecoderFuncWithExt("flac")) //QQ Music Flac
|
||
|
common.RegisterDecoder("qmcogg", false, DecoderFuncWithExt("ogg")) //QQ Music Ogg
|
||
|
common.RegisterDecoder("tkm", false, DecoderFuncWithExt("m4a")) //QQ Music Accompaniment M4a
|
||
4 years ago
|
|
||
3 years ago
|
common.RegisterDecoder("bkcmp3", false, DecoderFuncWithExt("mp3")) //Moo Music Mp3
|
||
|
common.RegisterDecoder("bkcflac", false, DecoderFuncWithExt("flac")) //Moo Music Flac
|
||
4 years ago
|
|
||
3 years ago
|
common.RegisterDecoder("666c6163", false, DecoderFuncWithExt("flac")) //QQ Music Weiyun Flac
|
||
|
common.RegisterDecoder("6d7033", false, DecoderFuncWithExt("mp3")) //QQ Music Weiyun Mp3
|
||
|
common.RegisterDecoder("6f6767", false, DecoderFuncWithExt("ogg")) //QQ Music Weiyun Ogg
|
||
|
common.RegisterDecoder("6d3461", false, DecoderFuncWithExt("m4a")) //QQ Music Weiyun M4a
|
||
|
common.RegisterDecoder("776176", false, DecoderFuncWithExt("wav")) //QQ Music Weiyun Wav
|
||
4 years ago
|
|
||
3 years ago
|
common.RegisterDecoder("mgg", false, NewMgg256Decoder) //QQ Music New Ogg
|
||
|
common.RegisterDecoder("mflac", false, NewMflac256Decoder) //QQ Music New Flac
|
||
4 years ago
|
}
|