QQ音乐API分析之-加密参数分析(sign计算)

article/2025/3/4 22:15:47

QQ音乐API加密参数分析

  • 1、背景
  • 2、QQ音乐sign计算
  • 3、Java代码实现
  • 4、总结

1、背景

不知道什么时候开始,各家音乐APP都开始对API进行加密,最近一段时间对六大音乐平台的加密算法进行了研究,逆向了网页端、安卓端等等,已经掌握了各家的加密算法。

平台加密算法非加密接口专属资源海外IP支持
QQMD5存在,可完全替代加密接口需要绿钻Cookie不支持,需要国内IP代理
KWDES存在,可完全替代加密接口不需要额外信息支持
KGMD5存在,不能完全替代加密接口需要豪华会员Cookie不支持,需要国内IP代理,或在请求后附加area_code=0
WYAES、RSA、MD5存在,少数接口不能使用未加密接口,如登录需要黑胶Cookie不支持,但可通过设置X-Real-IP规避
MGMD5存在,可完全替代加密接口不需要额外信息支持
XMMD5加密存在,可完全替代加密接口不需要额外信息不支持,需要国内IP代理

2、QQ音乐sign计算

首先说明,所有加密接口都有非加密的替代接口,但是以后肯定是向加密接口发展,两个最基础的API是


https://u.y.qq.com/cgi-bin/musicu.fcg  支持加密和非加密https://u.y.qq.com/cgi-bin/musics.fcg 仅支持加密

以上说多了,这篇主要是讲如何计算最新版的QQ音乐加密参数。

首先打开QQ音乐官网,F12打开开发者工具,我们随便找一条加密的链接

在这里插入图片描述

如图所示,我们可以看到这样一条链接

https://u.y.qq.com/cgi-bin/musics.fcg?-=getUCGI6892186109455234&g_tk=680034686&sign=zzadg51jzc4krod4v97bc3696a5f2740962d1e6c467957c88&loginUin=0&hostUin=0&format=json&inCharset=utf8&outCharset=utf-8&notice=0&platform=yqq.json&needNewCode=0&data=%7B%22detail%22%3A%7B%22module%22%3A%22musicToplist.ToplistInfoServer%22%2C%22method%22%3A%22GetDetail%22%2C%22param%22%3A%7B%22topId%22%3A4%2C%22offset%22%3A0%2C%22num%22%3A20%2C%22period%22%3A%222020-12-21%22%7D%7D%2C%22comm%22%3A%7B%22ct%22%3A24%2C%22cv%22%3A0%7D%7D

在链接中有个参数,我们的目标就是得到这个参数如何计算的方式

sign=zzadg51jzc4krod4v97bc3696a5f2740962d1e6c467957c88

点Initiator,找一下请求来源,点击来源的JS

在这里插入图片描述
在打开的JS里面查找sign找到如下图所示的区域,sign就是在这里得到的(格式化下js代码,否则看不清楚流程)在图中位置下断点,再刷新页面

https://y.gtimg.cn/music/portal/js/common/pkg/common_d9439fa.js?max_age=31536000:formatted

在这里插入图片描述
然后断在了这个地方,如果不是加密链接就放过,如果是,就步入
在这里插入图片描述
可以把鼠标放在变量上或者再控制台打印变量数据,如下图所示,我将请求的URL和请求参数打印了出来

//u.y.qq.com/cgi-bin/musicu.fcg?-=getUCGI4444074835099534&g_tk=680034686
{"detail":{"module":"musicToplist.ToplistInfoServer","method":"GetDetail","param":{"topId":4,"offset":0,"num":20,"period":"2020-12-21"}},"comm":{"ct":24,"cv":0}}

在这里插入图片描述
sign就是对请求参数加密之后得到的。

继续步入后,我们跳转到了下面这个JS里面,请求参数就是在这里加密的

https://y.qq.com/component/m/qmfe-security-sign/index.umd.js?max_age=2592000:formatted

在这里插入图片描述
具体的内容

!function(n, t) {"object" == typeof exports && "undefined" != typeof module ? module.exports = t() : "function" == typeof define && define.amd ? define(t) : (n = n || self).getSecuritySign = t()
}(this, function() {"use strict";var n = function() {if ("undefined" != typeof self)return self;if ("undefined" != typeof window)return window;if ("undefined" != typeof global)return global;throw new Error("unable to locate global object")}();n.__sign_hash_20200305 = function(n) {function l(n, t) {var o = (65535 & n) + (65535 & t);return (n >> 16) + (t >> 16) + (o >> 16) << 16 | 65535 & o}function r(n, t, o, e, u, p) {return l((i = l(l(t, n), l(e, p))) << (r = u) | i >>> 32 - r, o);var i, r}function g(n, t, o, e, u, p, i) {return r(t & o | ~t & e, n, t, u, p, i)}function a(n, t, o, e, u, p, i) {return r(t & e | o & ~e, n, t, u, p, i)}function s(n, t, o, e, u, p, i) {return r(t ^ o ^ e, n, t, u, p, i)}function v(n, t, o, e, u, p, i) {return r(o ^ (t | ~e), n, t, u, p, i)}function t(n) {return function(n) {var t, o = "";for (t = 0; t < 32 * n.length; t += 8)o += String.fromCharCode(n[t >> 5] >>> t % 32 & 255);return o}(function(n, t) {n[t >> 5] |= 128 << t % 32,n[14 + (t + 64 >>> 9 << 4)] = t;var o, e, u, p, i, r = 1732584193, f = -271733879, h = -1732584194, c = 271733878;for (o = 0; o < n.length; o += 16)r = g(e = r, u = f, p = h, i = c, n[o], 7, -680876936),c = g(c, r, f, h, n[o + 1], 12, -389564586),h = g(h, c, r, f, n[o + 2], 17, 606105819),f = g(f, h, c, r, n[o + 3], 22, -1044525330),r = g(r, f, h, c, n[o + 4], 7, -176418897),c = g(c, r, f, h, n[o + 5], 12, 1200080426),h = g(h, c, r, f, n[o + 6], 17, -1473231341),f = g(f, h, c, r, n[o + 7], 22, -45705983),r = g(r, f, h, c, n[o + 8], 7, 1770035416),c = g(c, r, f, h, n[o + 9], 12, -1958414417),h = g(h, c, r, f, n[o + 10], 17, -42063),f = g(f, h, c, r, n[o + 11], 22, -1990404162),r = g(r, f, h, c, n[o + 12], 7, 1804603682),c = g(c, r, f, h, n[o + 13], 12, -40341101),h = g(h, c, r, f, n[o + 14], 17, -1502002290),r = a(r, f = g(f, h, c, r, n[o + 15], 22, 1236535329), h, c, n[o + 1], 5, -165796510),c = a(c, r, f, h, n[o + 6], 9, -1069501632),h = a(h, c, r, f, n[o + 11], 14, 643717713),f = a(f, h, c, r, n[o], 20, -373897302),r = a(r, f, h, c, n[o + 5], 5, -701558691),c = a(c, r, f, h, n[o + 10], 9, 38016083),h = a(h, c, r, f, n[o + 15], 14, -660478335),f = a(f, h, c, r, n[o + 4], 20, -405537848),r = a(r, f, h, c, n[o + 9], 5, 568446438),c = a(c, r, f, h, n[o + 14], 9, -1019803690),h = a(h, c, r, f, n[o + 3], 14, -187363961),f = a(f, h, c, r, n[o + 8], 20, 1163531501),r = a(r, f, h, c, n[o + 13], 5, -1444681467),c = a(c, r, f, h, n[o + 2], 9, -51403784),h = a(h, c, r, f, n[o + 7], 14, 1735328473),r = s(r, f = a(f, h, c, r, n[o + 12], 20, -1926607734), h, c, n[o + 5], 4, -378558),c = s(c, r, f, h, n[o + 8], 11, -2022574463),h = s(h, c, r, f, n[o + 11], 16, 1839030562),f = s(f, h, c, r, n[o + 14], 23, -35309556),r = s(r, f, h, c, n[o + 1], 4, -1530992060),c = s(c, r, f, h, n[o + 4], 11, 1272893353),h = s(h, c, r, f, n[o + 7], 16, -155497632),f = s(f, h, c, r, n[o + 10], 23, -1094730640),r = s(r, f, h, c, n[o + 13], 4, 681279174),c = s(c, r, f, h, n[o], 11, -358537222),h = s(h, c, r, f, n[o + 3], 16, -722521979),f = s(f, h, c, r, n[o + 6], 23, 76029189),r = s(r, f, h, c, n[o + 9], 4, -640364487),c = s(c, r, f, h, n[o + 12], 11, -421815835),h = s(h, c, r, f, n[o + 15], 16, 530742520),r = v(r, f = s(f, h, c, r, n[o + 2], 23, -995338651), h, c, n[o], 6, -198630844),c = v(c, r, f, h, n[o + 7], 10, 1126891415),h = v(h, c, r, f, n[o + 14], 15, -1416354905),f = v(f, h, c, r, n[o + 5], 21, -57434055),r = v(r, f, h, c, n[o + 12], 6, 1700485571),c = v(c, r, f, h, n[o + 3], 10, -1894986606),h = v(h, c, r, f, n[o + 10], 15, -1051523),f = v(f, h, c, r, n[o + 1], 21, -2054922799),r = v(r, f, h, c, n[o + 8], 6, 1873313359),c = v(c, r, f, h, n[o + 15], 10, -30611744),h = v(h, c, r, f, n[o + 6], 15, -1560198380),f = v(f, h, c, r, n[o + 13], 21, 1309151649),r = v(r, f, h, c, n[o + 4], 6, -145523070),c = v(c, r, f, h, n[o + 11], 10, -1120210379),h = v(h, c, r, f, n[o + 2], 15, 718787259),f = v(f, h, c, r, n[o + 9], 21, -343485551),r = l(r, e),f = l(f, u),h = l(h, p),c = l(c, i);return [r, f, h, c]}(function(n) {var t, o = [];for (o[(n.length >> 2) - 1] = void 0,t = 0; t < o.length; t += 1)o[t] = 0;for (t = 0; t < 8 * n.length; t += 8)o[t >> 5] |= (255 & n.charCodeAt(t / 8)) << t % 32;return o}(n), 8 * n.length))}function o(n) {return t(unescape(encodeURIComponent(n)))}return function(n) {var t, o, e = "0123456789abcdef", u = "";for (o = 0; o < n.length; o += 1)t = n.charCodeAt(o),u += e.charAt(t >>> 4 & 15) + e.charAt(15 & t);return u}(o(n))},function r(f, h, c, l, g) {g = g || [[this], [{}]];for (var t = [], o = null, n = [function() {return !0}, function() {}, function() {g.length = c[h++]}, function() {g.push(c[h++])}, function() {g.pop()}, function() {var n = c[h++], t = g[g.length - 2 - n];g[g.length - 2 - n] = g.pop(),g.push(t)}, function() {g.push(g[g.length - 1])}, function() {g.push([g.pop(), g.pop()].reverse())}, function() {g.push([l, g.pop()])}, function() {g.push([g.pop()])}, function() {var n = g.pop();g.push(n[0][n[1]])}, function() {g.push(g[g.pop()[0]][0])}, function() {var n = g[g.length - 2];n[0][n[1]] = g[g.length - 1]}, function() {g[g[g.length - 2][0]][0] = g[g.length - 1]}, function() {var n = g.pop(), t = g.pop();g.push([t[0][t[1]], n])}, function() {var n = g.pop();g.push([g[g.pop()][0], n])}, function() {var n = g.pop();g.push(delete n[0][n[1]])}, function() {var n = [];for (var t in g.pop())n.push(t);g.push(n)}, function() {g[g.length - 1].length ? g.push(g[g.length - 1].shift(), !0) : g.push(void 0, !1)}, function() {var n = g[g.length - 2], t = Object.getOwnPropertyDescriptor(n[0], n[1]) || {configurable: !0,enumerable: !0};t.get = g[g.length - 1],Object.defineProperty(n[0], n[1], t)}, function() {var n = g[g.length - 2], t = Object.getOwnPropertyDescriptor(n[0], n[1]) || {configurable: !0,enumerable: !0};t.set = g[g.length - 1],Object.defineProperty(n[0], n[1], t)}, function() {h = c[h++]}, function() {var n = c[h++];g[g.length - 1] && (h = n)}, function() {throw g[g.length - 1]}, function() {var n = c[h++], t = n ? g.slice(-n) : [];g.length -= n,g.push(g.pop().apply(l, t))}, function() {var n = c[h++], t = n ? g.slice(-n) : [];g.length -= n;var o = g.pop();g.push(o[0][o[1]].apply(o[0], t))}, function() {var n = c[h++], t = n ? g.slice(-n) : [];g.length -= n,t.unshift(null),g.push(new (Function.prototype.bind.apply(g.pop(), t)))}, function() {var n = c[h++], t = n ? g.slice(-n) : [];g.length -= n,t.unshift(null);var o = g.pop();g.push(new (Function.prototype.bind.apply(o[0][o[1]], t)))}, function() {g.push(!g.pop())}, function() {g.push(~g.pop())}, function() {g.push(typeof g.pop())}, function() {g[g.length - 2] = g[g.length - 2] == g.pop()}, function() {g[g.length - 2] = g[g.length - 2] === g.pop()}, function() {g[g.length - 2] = g[g.length - 2] > g.pop()}, function() {g[g.length - 2] = g[g.length - 2] >= g.pop()}, function() {g[g.length - 2] = g[g.length - 2] << g.pop()}, function() {g[g.length - 2] = g[g.length - 2] >> g.pop()}, function() {g[g.length - 2] = g[g.length - 2] >>> g.pop()}, function() {g[g.length - 2] = g[g.length - 2] + g.pop()}, function() {g[g.length - 2] = g[g.length - 2] - g.pop()}, function() {g[g.length - 2] = g[g.length - 2] * g.pop()}, function() {g[g.length - 2] = g[g.length - 2] / g.pop()}, function() {g[g.length - 2] = g[g.length - 2] % g.pop()}, function() {g[g.length - 2] = g[g.length - 2] | g.pop()}, function() {g[g.length - 2] = g[g.length - 2] & g.pop()}, function() {g[g.length - 2] = g[g.length - 2] ^ g.pop()}, function() {g[g.length - 2] = g[g.length - 2]in g.pop()}, function() {g[g.length - 2] = g[g.length - 2]instanceof g.pop()}, function() {g[g[g.length - 1][0]] = void 0 === g[g[g.length - 1][0]] ? [] : g[g[g.length - 1][0]]}, function() {for (var e = c[h++], u = [], n = c[h++], t = c[h++], p = [], o = 0; o < n; o++)u[c[h++]] = g[c[h++]];for (var i = 0; i < t; i++)p[i] = c[h++];g.push(function n() {var t = u.slice(0);t[0] = [this],t[1] = [arguments],t[2] = [n];for (var o = 0; o < p.length && o < arguments.length; o++)0 < p[o] && (t[p[o]] = [arguments[o]]);return r(f, e, c, l, t)})}, function() {t.push([c[h++], g.length, c[h++]])}, function() {t.pop()}, function() {return !!o}, function() {o = null}, function() {g[g.length - 1] += String.fromCharCode(c[h++])}, function() {g.push("")}, function() {g.push(void 0)}, function() {g.push(null)}, function() {g.push(!0)}, function() {g.push(!1)}, function() {g.length -= c[h++]}, function() {g[g.length - 1] = c[h++]}, function() {var n = g.pop(), t = g[g.length - 1];t[0][t[1]] = g[n[0]][0]}, function() {var n = g.pop(), t = g[g.length - 1];t[0][t[1]] = n[0][n[1]]}, function() {var n = g.pop(), t = g[g.length - 1];g[t[0]][0] = g[n[0]][0]}, function() {var n = g.pop(), t = g[g.length - 1];g[t[0]][0] = n[0][n[1]]}, function() {g[g.length - 2] = g[g.length - 2] < g.pop()}, function() {g[g.length - 2] = g[g.length - 2] <= g.pop()}]; ; )try {for (; !n[c[h++]](); );if (o)throw o;return g.pop()} catch (n) {var e = t.pop();if (void 0 === e)throw n;o = n,h = e[0],g.length = e[1],e[2] && (g[e[2]][0] = o)}}(120731, 0, [21, 34, 50, 100, 57, 50, 102, 50, 98, 99, 101, 52, 54, 97, 52, 99, 55, 56, 52, 49, 57, 54, 57, 49, 56, 98, 102, 100, 100, 48, 48, 55, 55, 102, 2, 10, 3, 2, 9, 48, 61, 3, 9, 48, 61, 4, 9, 48, 61, 5, 9, 48, 61, 6, 9, 48, 61, 7, 9, 48, 61, 8, 9, 48, 61, 9, 9, 48, 4, 21, 427, 54, 2, 15, 3, 2, 9, 48, 61, 3, 9, 48, 61, 4, 9, 48, 61, 5, 9, 48, 61, 6, 9, 48, 61, 7, 9, 48, 61, 8, 9, 48, 61, 9, 9, 48, 61, 10, 9, 48, 61, 11, 9, 48, 61, 12, 9, 48, 61, 13, 9, 48, 61, 14, 9, 48, 61, 10, 9, 55, 54, 97, 54, 98, 54, 99, 54, 100, 54, 101, 54, 102, 54, 103, 54, 104, 54, 105, 54, 106, 54, 107, 54, 108, 54, 109, 54, 110, 54, 111, 54, 112, 54, 113, 54, 114, 54, 115, 54, 116, 54, 117, 54, 118, 54, 119, 54, 120, 54, 121, 54, 122, 54, 48, 54, 49, 54, 50, 54, 51, 54, 52, 54, 53, 54, 54, 54, 55, 54, 56, 54, 57, 13, 4, 61, 11, 9, 55, 54, 77, 54, 97, 54, 116, 54, 104, 8, 55, 54, 102, 54, 108, 54, 111, 54, 111, 54, 114, 14, 55, 54, 77, 54, 97, 54, 116, 54, 104, 8, 55, 54, 114, 54, 97, 54, 110, 54, 100, 54, 111, 54, 109, 14, 25, 0, 3, 4, 9, 11, 3, 3, 9, 11, 39, 3, 1, 38, 40, 3, 3, 9, 11, 38, 25, 1, 13, 4, 61, 12, 9, 55, 13, 4, 61, 13, 9, 3, 0, 13, 4, 4, 3, 13, 9, 11, 3, 11, 9, 11, 66, 22, 306, 4, 21, 422, 24, 4, 3, 14, 9, 55, 54, 77, 54, 97, 54, 116, 54, 104, 8, 55, 54, 102, 54, 108, 54, 111, 54, 111, 54, 114, 14, 55, 54, 77, 54, 97, 54, 116, 54, 104, 8, 55, 54, 114, 54, 97, 54, 110, 54, 100, 54, 111, 54, 109, 14, 25, 0, 3, 10, 9, 55, 54, 108, 54, 101, 54, 110, 54, 103, 54, 116, 54, 104, 15, 10, 40, 25, 1, 13, 4, 61, 12, 9, 6, 11, 3, 10, 9, 3, 14, 9, 11, 15, 10, 38, 13, 4, 61, 13, 9, 6, 11, 6, 5, 1, 5, 0, 3, 1, 38, 13, 4, 61, 0, 5, 0, 43, 4, 21, 291, 61, 3, 12, 9, 11, 0, 3, 9, 9, 49, 72, 0, 2, 3, 4, 13, 4, 61, 8, 9, 21, 721, 3, 2, 8, 3, 2, 9, 48, 61, 3, 9, 48, 61, 4, 9, 48, 61, 5, 9, 48, 61, 6, 9, 48, 61, 7, 9, 48, 4, 55, 54, 115, 54, 101, 54, 108, 54, 102, 8, 10, 30, 55, 54, 117, 54, 110, 54, 100, 54, 101, 54, 102, 54, 105, 54, 110, 54, 101, 54, 100, 32, 28, 22, 510, 4, 21, 523, 22, 4, 55, 54, 115, 54, 101, 54, 108, 54, 102, 8, 10, 0, 55, 54, 119, 54, 105, 54, 110, 54, 100, 54, 111, 54, 119, 8, 10, 30, 55, 54, 117, 54, 110, 54, 100, 54, 101, 54, 102, 54, 105, 54, 110, 54, 101, 54, 100, 32, 28, 22, 566, 4, 21, 583, 3, 4, 55, 54, 119, 54, 105, 54, 110, 54, 100, 54, 111, 54, 119, 8, 10, 0, 55, 54, 103, 54, 108, 54, 111, 54, 98, 54, 97, 54, 108, 8, 10, 30, 55, 54, 117, 54, 110, 54, 100, 54, 101, 54, 102, 54, 105, 54, 110, 54, 101, 54, 100, 32, 28, 22, 626, 4, 21, 643, 25, 4, 55, 54, 103, 54, 108, 54, 111, 54, 98, 54, 97, 54, 108, 8, 10, 0, 55, 54, 69, 54, 114, 54, 114, 54, 111, 54, 114, 8, 55, 54, 117, 54, 110, 54, 97, 54, 98, 54, 108, 54, 101, 54, 32, 54, 116, 54, 111, 54, 32, 54, 108, 54, 111, 54, 99, 54, 97, 54, 116, 54, 101, 54, 32, 54, 103, 54, 108, 54, 111, 54, 98, 54, 97, 54, 108, 54, 32, 54, 111, 54, 98, 54, 106, 54, 101, 54, 99, 54, 116, 27, 1, 23, 56, 0, 49, 444, 0, 0, 24, 0, 13, 4, 61, 8, 9, 55, 54, 95, 54, 95, 54, 103, 54, 101, 54, 116, 54, 83, 54, 101, 54, 99, 54, 117, 54, 114, 54, 105, 54, 116, 54, 121, 54, 83, 54, 105, 54, 103, 54, 110, 15, 21, 1126, 49, 2, 14, 3, 2, 9, 48, 61, 3, 9, 48, 61, 4, 9, 48, 61, 5, 9, 48, 61, 6, 9, 48, 61, 7, 9, 48, 61, 8, 9, 48, 61, 9, 9, 48, 61, 10, 9, 48, 61, 11, 9, 48, 61, 9, 9, 55, 54, 108, 54, 111, 54, 99, 54, 97, 54, 116, 54, 105, 54, 111, 54, 110, 8, 10, 30, 55, 54, 117, 54, 110, 54, 100, 54, 101, 54, 102, 54, 105, 54, 110, 54, 101, 54, 100, 32, 28, 22, 862, 21, 932, 21, 4, 55, 54, 108, 54, 111, 54, 99, 54, 97, 54, 116, 54, 105, 54, 111, 54, 110, 8, 55, 54, 104, 54, 111, 54, 115, 54, 116, 14, 55, 54, 105, 54, 110, 54, 100, 54, 101, 54, 120, 54, 79, 54, 102, 14, 55, 54, 121, 54, 46, 54, 113, 54, 113, 54, 46, 54, 99, 54, 111, 54, 109, 25, 1, 3, 0, 3, 1, 39, 32, 22, 963, 4, 55, 54, 67, 54, 74, 54, 66, 54, 80, 54, 65, 54, 67, 54, 114, 54, 82, 54, 117, 54, 78, 54, 121, 54, 55, 21, 974, 50, 4, 3, 12, 9, 11, 3, 8, 3, 10, 24, 2, 13, 4, 61, 10, 9, 3, 13, 9, 55, 54, 95, 54, 95, 54, 115, 54, 105, 54, 103, 54, 110, 54, 95, 54, 104, 54, 97, 54, 115, 54, 104, 54, 95, 54, 50, 54, 48, 54, 50, 54, 48, 54, 48, 54, 51, 54, 48, 54, 53, 15, 10, 22, 1030, 21, 1087, 22, 4, 3, 13, 9, 55, 54, 95, 54, 95, 54, 115, 54, 105, 54, 103, 54, 110, 54, 95, 54, 104, 54, 97, 54, 115, 54, 104, 54, 95, 54, 50, 54, 48, 54, 50, 54, 48, 54, 48, 54, 51, 54, 48, 54, 53, 15, 3, 9, 9, 11, 3, 3, 9, 11, 38, 25, 1, 13, 4, 61, 11, 9, 3, 12, 9, 11, 3, 10, 3, 53, 3, 37, 39, 24, 2, 13, 4, 4, 55, 54, 122, 54, 122, 54, 97, 3, 11, 9, 11, 38, 3, 10, 9, 11, 38, 0, 49, 771, 2, 1, 12, 9, 13, 8, 3, 12, 4, 4, 56, 0], n);var t = n.__getSecuritySign;return delete n.__getSecuritySign,t
});

然后在进入函数的返回出下断点,断在如下位置后,在控制台执行

r(f, e, c, l, t)

在这里插入图片描述

然后打印参数t,我们可以看到参数t里面出现了几个很特别的参数

CJBPACrRuNy797bc3696a5f2740962d1e6c467957c88{"detail":{"module":"musicToplist.ToplistInfoServer","method":"GetDetail","param":{"topId":4,"offset":0,"num":20,"period":"2020-12-21"}},"comm":{"ct":24,"cv":0}}

看到97bc3696a5f2740962d1e6c467957c88 这样的数字,又是32位,可以大胆地猜测是md5,事实上,97bc3696a5f2740962d1e6c467957c88就是 把固定的字符串CJBPACrRuNy7与请求参数拼接起来之后的md5值

CJBPACrRuNy7{"detail":{"module":"musicToplist.ToplistInfoServer","method":"GetDetail","param":{"topId":4,"offset":0,"num":20,"period":"2020-12-21"}},"comm":{"ct":24,"cv":0}}

这个JS文件的作用就是用JS实现了一段MD5加密算法。

多次探索之后,我发现sign由以下规则生成

固定头zza+一段随机的小写字符串,由小写字母和数字组成,长度为10-16位+对固定字符串和请求参数取md5。这三部分拼接而成。

下面回到我们调试的地方,取消所有断电,放开运行,之后我们就得到了一个网址

https://u.y.qq.com/cgi-bin/musics.fcg?-=getUCGI5601160880585316&g_tk=680034686&sign=zza4bb58zhof1tkhk8t97bc3696a5f2740962d1e6c467957c88&loginUin=0&hostUin=0&format=json&inCharset=utf8&outCharset=utf-8&notice=0&platform=yqq.json&needNewCode=0&data=%7B%22detail%22%3A%7B%22module%22%3A%22musicToplist.ToplistInfoServer%22%2C%22method%22%3A%22GetDetail%22%2C%22param%22%3A%7B%22topId%22%3A4%2C%22offset%22%3A0%2C%22num%22%3A20%2C%22period%22%3A%222020-12-21%22%7D%7D%2C%22comm%22%3A%7B%22ct%22%3A24%2C%22cv%22%3A0%7D%7D

计算出了sign后面就可以正常请求了。

3、Java代码实现

附上一段无名音乐中计算sign的代码,用Java实现

public class QqEncrypt {private static final String encNonce="CJBPACrRuNy7";private static final String signPrxfix="zza";private static final char[] dir="0234567890abcdefghijklmnopqrstuvwxyz".toCharArray();/****@param encParams 需要加密的参数,这是一段请求体数据,为json字符串格式,例如下面的格式,可以抓包获取*                 {"comm":{"ct":24,"cv":0},"vip":{"module":"userInfo…baseinfo_v2","param":{"vec_uin":["3011429848"]}}}** @return 加密的方式为固定字串 zza加上一个10-16位的随机字符串再加上 固定字串 CJBPACrRuNy7加上请求数据拼接的 MD5值*/public static String getSign(String encParams){return signPrxfix+uuidGenerate()+ Md5Encrypt.convertToMd5(encNonce+encParams);}private static String uuidGenerate(){int minLen=10;int maxLen=16;Random ran=new Random(System.currentTimeMillis());int ranLen=ran.nextInt(maxLen-minLen)+minLen;StringBuilder sb=new StringBuilder(ranLen);for (int i=0;i<ranLen;i++){sb.append(dir[ran.nextInt(dir.length)]);}return sb.toString();}
}public class Md5Encrypt {//必须要重视编码问题了!折腾了好多天才发现MD5要用UTF-8形式加密public static String convertToMd5(String plainText) {byte[] secretBytes = null;try {secretBytes = MessageDigest.getInstance("md5").digest(plainText.getBytes("utf-8"));} catch (Exception e) {throw new RuntimeException("没有这个md5算法!");}String md5code = new BigInteger(1, secretBytes).toString(16);for (int i = 0; i < 32 - md5code.length(); i++) {md5code = "0" + md5code;}return md5code;}
}

下面是封装了一段请求代码

public class RequestModule {/*** musicu支持加密请求和不加密请求,没有对请求进行限制* musics仅支持加密请求*/private static final String[] apiServer={"https://u.y.qq.com/cgi-bin/musicu.fcg?","https://u.y.qq.com/cgi-bin/musics.fcg?"};private static final Map<String,String> headerMap=new HashMap<>();static {headerMap.put("user-agent","Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/87.0.4280.88 Safari/537.36 Edg/87.0.664.66");headerMap.put("accept-encoding","application/json");headerMap.put("cache-control","max-age=0");}/*** 请求示例:{"area":-100,"sex":-100,"genre":-100,"index":15,"sin":0,"cur_page":1}* @param area 地区 -100 不限* @param genre 流派 全部:-100,流行:1 ,摇滚:2,民谣:3,电子:4,爵士:5,嘻哈:6 .....* @param index 索引 热门:-100  范围 0-26 对应歌手名称拼音A-Z开头* @param sex 性别 不限:-100  男:0 女:1 组合:2* @param sin 偏移(第sin位)* @param currPage 当前页,第n页,默认每页最多80人* @return*/public static String getSingerList(int area,int genre,int index,int sex,int sin,int currPage){RequestTemplate template=new RequestTemplate("Music.SingerListServer","get_singer_list","{\"area\":"+area+",\"sex\":"+sex+",\"genre\":"+genre+",\"index\":"+index+",\"sin\":"+sin+",\"cur_page\":"+currPage+"}");RequestTemplateBuilder templateBuilder=new RequestTemplateBuilder();templateBuilder.add("singerList",template);String sign= QqEncrypt.getSign(templateBuilder.toString());Random ran=new Random(System.currentTimeMillis());return HttpRequestHelper.downloadWebSiteUsePost(apiServer[ran.nextInt(apiServer.length)]+"sign="+sign,templateBuilder.toString(),headerMap);}public static String getHotSingerList(){return getSingerList(-100,-100,-100,-100,0,1);}/**** @param songId 统一规范,指的是songmid* @return*/public static String getSongDetail(String songId){//{"songinfo":{"module":"music.pf_song_detail_svr","method":"get_song_detail","param":{"song_mid":"112344"}}}RequestTemplate template=new RequestTemplate("music.pf_song_detail_svr","get_song_detail","{\"song_mid\":\""+songId+"\"}");RequestTemplateBuilder templateBuilder=new RequestTemplateBuilder();templateBuilder.add("songdetail",template);String sign= QqEncrypt.getSign(templateBuilder.toString());Random ran=new Random(System.currentTimeMillis());return HttpRequestHelper.downloadWebSiteUsePost(apiServer[ran.nextInt(apiServer.length)] +"sign="+sign,templateBuilder.toString(),headerMap);}}

实体类

public class RequestTemplate{public String module="";public String method="";public Object param=new Object();public RequestTemplate() {}public RequestTemplate(String module, String method, Object param) {this.module = module;this.method = method;this.param = param;}public String getModule() {return module;}public void setModule(String module) {this.module = module;}public String getMethod() {return method;}public void setMethod(String method) {this.method = method;}public Object getParam() {return param;}public void setParam(Object param) {this.param = param;}@Overridepublic String toString() {return "{" +"\"module\":\"" + module + "\"" +",\"method\":\"" + method + "\"" +",\"param\":" + param.toString()  +"}";}
}/*** 构造请求结构,所有请求都可以是这样的** {*     "comm": {*         "ct": 24,*         "cv": 0*     },*     "singerList": {*         "module": "Music.SingerListServer",*         "method": "get_singer_list",*         "param": {*             "area": -100,*             "sex": -100,*             "genre": -100,*             "index": -100,*             "sin": 0,*             "cur_page": 1*         }*     }* }*/
public class RequestTemplateBuilder {private Map<String,Object> requestTemplateMap=new HashMap<>();public RequestTemplateBuilder add(String name, Object req){requestTemplateMap.put(name,req);return this;}@Overridepublic String toString() {StringBuilder sb=new StringBuilder();for (String key: requestTemplateMap.keySet()){sb.append("\"").append(key).append("\":").append(requestTemplateMap.get(key).toString()).append(",");}sb.substring(0,sb.length()-1);return "{"+sb.substring(0,sb.length()-1)+"}";}
}

4、总结

核心的加密函数在以下JS文件中实现,这段JS主要就是实现了MD5算法,由于以前没接触过如何实现MD5算法,在分析这段代码上还是遇到了很大困难的,不过我们在实现过程中完全不必自己实现MD5加密算法,我们可以直接使用各个语言上的加密库实现MD5加密。

https://y.qq.com/component/m/qmfe-security-sign/index.umd.js?max_age=2592000:formatted

http://chatgpt.dhexx.cn/article/JITG8vE7.shtml

相关文章

微信小程序使用QQ音乐API完整实例

微信小程序使用QQ音乐API完整实例 一、QQ音乐常用API接口&#xff1a;1.1、音乐搜索接口&#xff1a;1.2、最新音乐排行榜top1001.3. 随机推荐 二、请求数据格式&#xff08;以搜索为例&#xff09;三、播放链接&#xff08;重点&#xff09;3.1、获取歌曲信息3.2 获取token3.3…

音乐播放器之QQ音乐最新api,亲测可用

大家好&#xff0c;前段时间重写了自己的音乐播放器&#xff0c;源码放在github上&#xff0c;源码地址和项目地址下面都有&#xff0c;如果喜欢记得star一下哈。 由于之前给大家分享的api虽然可以用&#xff0c;但是版本太旧了&#xff0c;很多也没有了歌词&#xff0c;今天博…

步数精灵v3.0运动安卓版

软件介绍: 一键修改微信运动、支付宝运动的步数&#xff0c;让你在家足不出户也能走几万步&#xff0c;天天霸占好友排行榜首位&#xff01; 软件预览图&#xff1a; 软件版本号&#xff1a;3.0 软件是否收费&#xff1a;免费软件/共享软件 运行环境&#xff1a;安卓 软件…

艾出行广告精灵挂机源码+对接码支付即时到账+充值系统+推广下级系统+封装app

简介&#xff1a; 安装说明&#xff1a;请使用win系统 apache php5.6 mysql5.6 进行安装。 1、上传到网站根目录 2、用phpMyadmin导入数据库文件.sql 3、修改数据库链接文件 /APP/Conf/config.php &#xff08;记得不要用记事本修改&#xff0c;否则可能会出现验证码显示不了问…

支付平台架构:终端安全技术实现

前蚂蚁集团宣布即将IPO之后&#xff0c;9月11日晚间&#xff0c;以金融支付起家的京东数科也要上市了。近年来&#xff0c;第三方支付业务的资金规模不断扩大&#xff0c;支付业务量稳步增长&#xff0c;“第三方支付”及“移动支付”已成为年度搜索热词&#xff0c;支付平台作…

键盘精灵_键盘

键盘精灵 They folks over at DasKeyboard loaned me a reviewers copy of their totally blank keyboard about three weeks ago and Ive been typing on it since. 大约在三周前&#xff0c; DasKeyboard的同事们借给了我审阅他们完全空白的键盘的副本&#xff0c;此后我一直…

Telink ble mesh天猫精灵应用

文章目录 1 前言2 完美对接天猫精灵平台2.1 创建新产品2.1.1 添加产品类型2.1.2 添加产品信息2.1.3 添加产品功能 2.2 设置人机交互2.2.1 设置控制口令2.2.2 添加产品展示图2.2.3 添加配网指导2.2.4 添加升级指导2.2.5 添加控制面板 2.3 选择品牌模组2.4 下载并保管好三元组2.5…

[超详细] 2021支付宝集五福【攻略】来了-附自动化脚本

一年一度的支付宝集五福在2月1号的凌晨开始了&#xff0c;虽然一年比一年瓜分的少&#xff0c;但不知不觉集福卡成了过年最佳的消遣娱乐方式了。活动从2月1日开始至1月11日结束。 扫下面这个福字&#xff0c;必得沾福卡&#xff01; 零、额外福利 额外福卡二维码扫福器下载地址…

小程序图标-精灵图

文章来源&#xff1a;刘俊涛的博客 欢迎关注&#xff0c;有问题一起学习欢迎留言、评论。

python调用按键精灵插件_【按键精灵】三分钟教你实现自动解决各种验证码

金猪脚本(原飞猪脚本)以按键精灵教学为主,涉及UiBot&#xff0c;Python,Lua等脚本编程语言,教学包括全自动办公脚本,游戏辅助脚本,引流脚本,网页脚本,安卓脚本,IOS脚本,注册脚本,点赞脚本,阅读脚本以及网赚脚本等各个领域。想制作脚本和学习按键精灵的朋友可以添加按键精灵学习…

python调用按键精灵插件_【按键精灵教程】此帖在手,打码不愁

金猪脚本(原飞猪脚本)以按键精灵教学为主,涉及UiBot&#xff0c;Python,Lua等脚本编程语言,教学包括全自动办公脚本,游戏辅助脚本,引流脚本,网页脚本,安卓脚本,IOS脚本,注册脚本,点赞脚本,阅读脚本以及网赚脚本等各个领域。想制作脚本和学习按键精灵的朋友可以添加按键精灵学习…

python调用按键精灵插件_按键精灵教程打码平台接入

金猪脚本(原飞猪脚本)以按键精灵教学为主,涉及UiBot&#xff0c;Python,Lua等脚本编程语言,教学包括全自动办公脚本,游戏辅助脚本,引流脚本,网页脚本,安卓脚本,IOS脚本,注册脚本,点赞脚本,阅读脚本以及网赚脚本等各个领域。想制作脚本和学习按键精灵的朋友可以添加按键精灵学习…

手机端APP调起支付宝客户端支付!!!!

需要商户登录支付宝后台&#xff0c;在“账户中心-密钥管理”下配置MAPI网关产品密钥&#xff0c;用支付宝官方提供的openssl-1.0.2m工具生成一对私钥和公钥&#xff0c;把公钥配置上去就可以了。不清楚的也可以咨询支付宝的在线客服技术人员&#xff0c;很好的客服 赋代码&am…

支付宝蚂蚁森林php自动,auto.js蚂蚁森林智能脚本使用方法分享 支付宝蚂蚁森林自动辅助...

auto.js蚂蚁森林智能脚本使用方法分享 支付宝蚂蚁森林自动辅助 这个项目是 github上的一个开源项目&#xff0c;作者应该是 SuperMonster &#xff0c;此人在52也有ID 不管怎么说是一个长期维护的开源项目&#xff0c;安全性 跟可用性完全不用担心了 重点就是你玩不玩蚂蚁森林了…

按键精灵初接触

初衷 当目前想模拟人工操作的时候&#xff0c;而且不想去操作底层协议去攻破各大系统漏洞&#xff0c;还想自动化的操作&#xff0c;加流量&#xff0c;加粉丝&#xff0c;加关注&#xff0c;自动挂机等等。解放双手&#xff0c;按键精灵就是你需要的。 目前按键精灵类似软件…

MacOS 按键精灵推荐

前段时间本人被反复枯燥的工作折磨&#xff0c;后突发奇想使用按键精灵。 按键精灵是一款功能强大的自动化操作工具&#xff0c;可以用来完成一系列重复性的操作&#xff0c;比如键盘快捷键的模拟、文本自动输入、鼠标操作的模拟等等&#xff0c;可以大大提高工作效率。 工具…

支付宝集五福攻略

新年到了&#xff0c;支付宝又开启了一年一度的集五福活动 &#xff08;说实话一般情况下得不到多少钱&#xff09;&#xff0c;主要还是图个喜庆&#xff0c;图个高兴。下面小编为你奉上集五福攻略。 AR扫福这张图片大概率可以获得沾沾卡 第二是支付宝分别搜&#xff1a;滴滴…

1.3 JavaScript 输入与输出

1.3 JavaScript 输入与输出 输入 从HTML与用户的交互中输入信息&#xff0c;例如通过input、textarea等标签获取用户的键盘输入&#xff0c;通过click、hover等事件获取用户的鼠标输入。通过Ajax与WebSocket从服务器端获取输入标准输入&#xff0c;参考AcWing 1. A B let fs…

js控制文本框只能输入数字 及 常用字符对应ASCII码值

方法一: < INPUT TYPE =text NAME =text onkeypress ="a()" > < script language =javascript > ... function a()...{ var k = window.event.keyCode;if ( (k > 47 && k < 58) ||

js-确定输入字符串是否有效

// 给定一个仅包含字符 (, ), {, }, [ 和 ] 的字符串 s&#xff0c;// 确定输入字符串是否有效。// 输入字符串在以下情况下有效&#xff1a; // Input: s "()[]{}"// Output: true// Input: s "(]"// Output: false // Input: s "([)]"// Ou…