key_derive.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. package qmc
  2. import (
  3. "bytes"
  4. "encoding/base64"
  5. "errors"
  6. "fmt"
  7. "math"
  8. "golang.org/x/crypto/tea"
  9. )
  10. func simpleMakeKey(salt byte, length int) []byte {
  11. keyBuf := make([]byte, length)
  12. for i := 0; i < length; i++ {
  13. tmp := math.Tan(float64(salt) + float64(i)*0.1)
  14. keyBuf[i] = byte(math.Abs(tmp) * 100.0)
  15. }
  16. return keyBuf
  17. }
  18. const rawKeyPrefixV2 = "QQMusic EncV2,Key:"
  19. func deriveKey(rawKey []byte) ([]byte, error) {
  20. rawKeyDec := make([]byte, base64.StdEncoding.DecodedLen(len(rawKey)))
  21. n, err := base64.StdEncoding.Decode(rawKeyDec, rawKey)
  22. if err != nil {
  23. return nil, err
  24. }
  25. rawKeyDec = rawKeyDec[:n]
  26. if bytes.HasPrefix(rawKeyDec, []byte(rawKeyPrefixV2)) {
  27. rawKeyDec, err = deriveKeyV2(bytes.TrimPrefix(rawKeyDec, []byte(rawKeyPrefixV2)))
  28. if err != nil {
  29. return nil, fmt.Errorf("deriveKeyV2 failed: %w", err)
  30. }
  31. }
  32. return deriveKeyV1(rawKeyDec)
  33. }
  34. func deriveKeyV1(rawKeyDec []byte) ([]byte, error) {
  35. if len(rawKeyDec) < 16 {
  36. return nil, errors.New("key length is too short")
  37. }
  38. simpleKey := simpleMakeKey(106, 8)
  39. teaKey := make([]byte, 16)
  40. for i := 0; i < 8; i++ {
  41. teaKey[i<<1] = simpleKey[i]
  42. teaKey[i<<1+1] = rawKeyDec[i]
  43. }
  44. rs, err := decryptTencentTea(rawKeyDec[8:], teaKey)
  45. if err != nil {
  46. return nil, err
  47. }
  48. return append(rawKeyDec[:8], rs...), nil
  49. }
  50. var (
  51. deriveV2Key1 = []byte{
  52. 0x33, 0x38, 0x36, 0x5A, 0x4A, 0x59, 0x21, 0x40,
  53. 0x23, 0x2A, 0x24, 0x25, 0x5E, 0x26, 0x29, 0x28,
  54. }
  55. deriveV2Key2 = []byte{
  56. 0x2A, 0x2A, 0x23, 0x21, 0x28, 0x23, 0x24, 0x25,
  57. 0x26, 0x5E, 0x61, 0x31, 0x63, 0x5A, 0x2C, 0x54,
  58. }
  59. )
  60. func deriveKeyV2(raw []byte) ([]byte, error) {
  61. buf, err := decryptTencentTea(raw, deriveV2Key1)
  62. if err != nil {
  63. return nil, err
  64. }
  65. buf, err = decryptTencentTea(buf, deriveV2Key2)
  66. if err != nil {
  67. return nil, err
  68. }
  69. n, err := base64.StdEncoding.Decode(buf, buf)
  70. if err != nil {
  71. return nil, err
  72. }
  73. return buf[:n], nil
  74. }
  75. func decryptTencentTea(inBuf []byte, key []byte) ([]byte, error) {
  76. const saltLen = 2
  77. const zeroLen = 7
  78. if len(inBuf)%8 != 0 {
  79. return nil, errors.New("inBuf size not a multiple of the block size")
  80. }
  81. if len(inBuf) < 16 {
  82. return nil, errors.New("inBuf size too small")
  83. }
  84. blk, err := tea.NewCipherWithRounds(key, 32)
  85. if err != nil {
  86. return nil, err
  87. }
  88. destBuf := make([]byte, 8)
  89. blk.Decrypt(destBuf, inBuf)
  90. padLen := int(destBuf[0] & 0x7)
  91. outLen := len(inBuf) - 1 - padLen - saltLen - zeroLen
  92. out := make([]byte, outLen)
  93. ivPrev := make([]byte, 8)
  94. ivCur := inBuf[:8]
  95. inBufPos := 8
  96. destIdx := 1 + padLen
  97. cryptBlock := func() {
  98. ivPrev = ivCur
  99. ivCur = inBuf[inBufPos : inBufPos+8]
  100. xor8Bytes(destBuf, destBuf, inBuf[inBufPos:inBufPos+8])
  101. blk.Decrypt(destBuf, destBuf)
  102. inBufPos += 8
  103. destIdx = 0
  104. }
  105. for i := 1; i <= saltLen; {
  106. if destIdx < 8 {
  107. destIdx++
  108. i++
  109. } else if destIdx == 8 {
  110. cryptBlock()
  111. }
  112. }
  113. outPos := 0
  114. for outPos < outLen {
  115. if destIdx < 8 {
  116. out[outPos] = destBuf[destIdx] ^ ivPrev[destIdx]
  117. destIdx++
  118. outPos++
  119. } else if destIdx == 8 {
  120. cryptBlock()
  121. }
  122. }
  123. for i := 1; i <= zeroLen; i++ {
  124. if destBuf[destIdx] != ivPrev[destIdx] {
  125. return nil, errors.New("zero check failed")
  126. }
  127. }
  128. return out, nil
  129. }
  130. func xor8Bytes(dst, a, b []byte) {
  131. for i := 0; i < 8; i++ {
  132. dst[i] = a[i] ^ b[i]
  133. }
  134. }