theme: channing-cyan
关键词
JS 混淆
、源码乱码、参数动态加密
逆向目标
题目1:抓取所有(5页)机票的价格,并计算所有机票价格的平均值,填入答案。
逆向过程
解决无限debug
打开 F12 刷新页面,傻眼了。页面无法继续下翻...
陷入了无限debug
!!
解决:Never pause here 不在此处下断
在debugger
位置,点击行号,右键Never pause here
,永远不在此处断下,然后重新刷新就可以过这第一个小坑。
抓包分析
接着我们需要找到目标url
。
上下翻页,我们很快能够找到具体数据响应的接口。
请求详情:
响应:
参数page
无需多言,关键是有个不知道哪里冒出来的参数m
?
将
curl
命令转换为Python
、JavaScript
等: https://curlconverter.com/
此时,如果我们构造requests
请求,并不带上m
查询参数,执行会发现响应结果为:
{'error': 'token failed'}
接下来就是要分析下这个m
是怎么带上的?
调用堆栈
很容易猜想后半部分大概率是一个时间戳,前半部分是被加密的字符串,中间用中文的|
连接。
由于
params
参数是封装到请求体中发到服务器的,所以,参数m
的生成必然是在请求前,因此可以直接来到请求时的这个栈中进行查看调试。
在点进去显示的那一行打断点刷新调试一下,会发现在_0xb89747
已经是加密完成的参数,肯定加密过程是在这之前完成的。
所以往回找,先根据_0xb89747
看看是从哪里来的,发现是:
var _0xb89747 = _0x5d83a3;
同理再往回找_0x5d83a3
从哪里来,发现_0x5d83a3
是定义的对象,然后往里面塞了什么东西进去组成的:
const _0x5d83a3 = {};
_0x5d83a3['\x70\x61\x67\x65'] = window['\x70\x61\x67\x65'],
_0x5d83a3['\x6d'] = _0x57feae + '\u4e28' + _0x2268f9 / (-1 * 3483 + -9059 + 13542);
根据_0xb89747
为{"page": 4, "m": "8d596e76013033e75c3e23015c553c55丨1715257368"}
能够得出参数里面m
的组成:
_0x57feae
就是前半部分被加密完成的东西。_0x2268f9
就是后半部分。
是不是真相快被找到了,再往回找这两个是哪里来的:
var _0x2268f9 = Date['\x70\x61\x72\x73\x65'](new Date()) + (16798545 + -72936737 + 156138192)
, _0x57feae = oo0O0(_0x2268f9['\x74\x6f\x53\x74\x72' + '\x69\x6e\x67']()) + window['\x66'];
在线网站解混淆
通过堆栈跟踪,我们已经找到m
参数加密入口,但是代码是混淆的,看起来并不那么清晰。
找在线网站,稍微解一下混淆:
https://deobfuscator.kuizuo.cn/
function a() {
var _0x2268f9 = Date.parse(new Date()) + 100000000;
var _0x57feae = oo0O0(_0x2268f9.toString()) + window.f;
const _0x5d83a3 = {
page: window.page,
m: _0x57feae + "丨" + _0x2268f9 / 1000
};
var _0xb89747 = _0x5d83a3;
$.ajax({
url: window.url,
dataType: "json",
async: false,
data: _0xb89747,
type: "GET",
beforeSend: function (_0x4c488e) {},
success: function (_0x131e59) {
_0x131e59 = _0x131e59.data;
let _0x354583 = "";
let _0x1b89ba = "中国联合航空KN5911波音737(中)13:50
大兴国际机场
3小时40分钟17:30
宝安机场
¥price_sole
收起
";
let _0x548377 = ["中国南方航空", "吉祥航空", "奥凯航空", "九元航空", "长龙航空", "东方航空", "中国国际航空", "深圳航空", "海南航空", "春秋航空", "上海航空", "西部航空", "重庆航空", "西藏航空", "中国联合航空", "云南祥鹏航空", "云南英安航空", "厦门航空", "天津航空", "山东航空", "四川航空", "华夏航空", "长城航空", "成都航空有", "北京首都航空", "中华航空", "意大利国家航空公司", "印度百捷航空", "越南航空", "远东航空", "印度航空公司", "印度捷特航空有限公司", "以色列航空公司", "意大利航空", "伊朗航空公司", "印度尼西亚鹰航空公司", "英国航空公司", "西方天空航空", "西捷航空", "西班牙欧洲航空公司", "西班牙航空公司", "中国南方航空", "吉祥航空", "奥凯航空", "九元航空", "长龙航空", "东方航空", "中国国际航空", "深圳航空", "海南航空", "春秋航空", "上海航空", "西部航空", "重庆航空", "西藏航空", "中国联合航空", "云南祥鹏航空", "云南英安航空", "厦门航空", "天津航空", "山东航空", "四川航空", "华夏航空", "长城航空", "成都航空有", "北京首都航空", "中华航空", "意大利国家航空公司", "印度百捷航空", "越南航空", "远东航空", "印度航空公司", "印度捷特航空有限公司", "以色列航空公司", "意大利航空", "伊朗航空公司", "印度尼西亚鹰航空公司", "英国航空公司", "西方天空航空", "西捷航空", "西班牙欧洲航空公司", "西班牙航空公司"];
let _0x5286d2 = 1;
let _0xa24ff9 = ["北京首都国际机场", "上海虹桥国际机场", "上海浦东国际机场", "天津滨海国际机场", "太原武宿机场", "呼和浩特白塔机场", "沈阳桃仙国际机场", "大连周水子国际机场", "长春大房身机场", "哈尔滨阎家岗国际机场", "齐齐哈尔三家子机场", "佳木斯东郊机场", "厦门高崎国际机场", "福州长乐国际机场", "杭州萧山国际机场", "合肥骆岗机场", "宁波栎社机场", "南京禄口国际机场", "广州白云国际机场", "深圳宝安国际机场", "长沙黄花机场", "海口美亚机场", "武汉天河机场", "济南遥墙机场", "青岛流亭机场", "南宁吴墟机场", "三亚凤凰国际机场", "重庆江北国际机场", "成都双流国际机场", "昆明巫家坝国际机场", "昆明长水国际机场", "桂林两江国际机场", "西安咸阳国际机场", "兰州中川机场", "贵阳龙洞堡机场", "拉萨贡嘎机场", "乌鲁木齐地窝堡机场", "南昌向塘机场", "郑州新郑机场", "北京首都国际机场", "上海虹桥国际机场", "上海浦东国际机场", "天津滨海国际机场", "太原武宿机场", "呼和浩特白塔机场", "沈阳桃仙国际机场", "大连周水子国际机场", "长春大房身机场", "哈尔滨阎家岗国际机场", "齐齐哈尔三家子机场", "佳木斯东郊机场", "厦门高崎国际机场", "福州长乐国际机场", "杭州萧山国际机场", "合肥骆岗机场", "宁波栎社机场", "南京禄口国际机场", "广州白云国际机场", "深圳宝安国际机场", "长沙黄花机场", "海口美亚机场", "武汉天河机场", "济南遥墙机场", "青岛流亭机场", "南宁吴墟机场", "三亚凤凰国际机场", "重庆江北国际机场", "成都双流国际机场", "昆明巫家坝国际机场", "昆明长水国际机场", "桂林两江国际机场", "西安咸阳国际机场", "兰州中川机场", "贵阳龙洞堡机场", "拉萨贡嘎机场", "乌鲁木齐地窝堡机场", "南昌向塘机场", "郑州新郑机场"];
if (window.page) {} else {
window.page = 1;
}
$.each(_0x131e59, function (_0x282f1d, _0x4e0853) {
_0x354583 += _0x1b89ba.replace("price_sole", _0x4e0853.value).replace("中国联合航空", _0x548377[_0x5286d2 * window.page]).replace("大兴国际", _0xa24ff9[parseInt(_0x5286d2 * window.page / 2) + 1]).replace("宝安机场", _0xa24ff9[_0xa24ff9.length - parseInt(_0x5286d2 * window.page / 2) - 1]);
_0x5286d2 += 1;
});
$(".m-airfly-lst").text("").append(_0x354583);
},
complete: function () {},
error: function () {
alert("数据拉取失败。可能是触发了风控系统,若您是正常访问,请使用谷歌浏览器无痕模式,并且校准电脑的系统时间重新尝试");
alert("生而为虫,我很抱歉,请刷新页面,查看问题是否存在");
$(".page-message").eq(0).addClass("active");
$(".page-message").removeClass("active");
}
});
}
通过解混淆后的 JS 代码也是进一步验证了我们跟踪堆栈的正确性。
我们可以看到m
字符串是通过oo0O0()函数 + window.f + ‘|’ + 时间处理
构成的,接下来我们来找oo0O0()
这个函数。
跟栈找到oo0O0()函数
的定义:
抠出function oo0O0(mw)
函数,使用在线 HTML 美化工具美化。
HTML 美化代码工具分析: https://beautifier.io/
function oo0O0(mw)
{
window.b = '';
for (var i = 0, len = window.a.length; i < len; i++)
{
console.log(window.a[i]);
window.b += String[document.e + document.g](window.a[i][document.f +
document.h
]() - i - window.c)
}
var U = ['W5r5W6VdIHZcT8kU', 'WQ8CWRaxWQirAW=='];
var J = function (o, E)
{
o = o - 0x0;
var N = U[o];
if (J['bSSGte'] === undefined)
{
var Y = function (w)
{
var m =
'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=',
T = String(w)['replace'](/=+$/, '');
var A = '';
for (var C = 0x0, b, W, l = 0x0; W = T['charAt'](l++); ~
W && (b = C % 0x4 ? b * 0x40 + W : W, C++ % 0x4) ?
A += String['fromCharCode'](0xff & b >> (-0x2 * C &
0x6)) : 0x0)
{
W = m['indexOf'](W)
}
return A
};
var t = function (w, m)
{
var T = [],
A = 0x0,
C, b = '',
W = '';
w = Y(w);
for (var R = 0x0, v = w['length']; R < v; R++)
{
W += '%' + ('00' + w['charCodeAt'](R)['toString'](
0x10))['slice'](-0x2)
}
w = decodeURIComponent(W);
var l;
for (l = 0x0; l < 0x100; l++)
{
T[l] = l
}
for (l = 0x0; l < 0x100; l++)
{
A = (A + T[l] + m['charCodeAt'](l % m['length'])) %
0x100, C = T[l], T[l] = T[A], T[A] = C
}
l = 0x0, A = 0x0;
for (var L = 0x0; L < w['length']; L++)
{
l = (l + 0x1) % 0x100, A = (A + T[l]) % 0x100, C =
T[l], T[l] = T[A], T[A] = C, b += String[
'fromCharCode'](w['charCodeAt'](L) ^ T[(T[
l] + T[A]) % 0x100])
}
return b
};
J['luAabU'] = t, J['qlVPZg'] = {}, J['bSSGte'] = !![]
}
var H = J['qlVPZg'][o];
return H === undefined ? (J['TUDBIJ'] === undefined && (J[
'TUDBIJ'] = !![]), N = J['luAabU'](N, E), J['qlVPZg'][
o] = N) : N = H, N
};
eval(atob(window['b'])[J('0x0', ']dQW')](J('0x1', 'GTu!'), '\x27' + mw +
'\x27'));
return ''
}
这里我们会发现,oo0O0()
这个函数只是执行了eval
,而函数本身返回空值""
。
优秀啊,年轻人真是会玩!
接下来分析这条代码:
> atob(window['b'])
> 'var hexcase=0;var b64pad="";var chrsz=16;function hex_md5(a){return binl2hex(core_md5(str2binl(a),a.length*chrsz))}function b64_md5(a){return binl2b64(core_md5(str2binl(a),a.length*chrsz))}function str_md5(a){return binl2str(core_md5(str2binl(a),a.length*chrsz))}function hex_hmac_md5(a,b){return binl2hex(core_hmac_md5(a,b))}function b64_hmac_md5(a,b){return binl2b64(core_hmac_md5(a,b))}function str_hmac_md5(a,b){return binl2str(core_hmac_md5(a,b))}function md5_vm_test(){return hex_md5("abc")=="900150983cd24fb0d6963f7d28e17f72"}function core_md5(p,k){p[k>>5]|=128<<((k)%32);p[(((k+64)>>>9)<<4)+14]=k;var o=1732584193;var n=-271733879;var m=-1732584194;var l=271733878;for(var g=0;g16){e=core_md5(e,c.length*chrsz)}var a=Array(16),d=Array(16);for(var b=0;b<16;b++){a[b]=e[b]^909522486;d[b]=e[b]^1549556828}var g=core_md5(a.concat(str2binl(f)),512+f.length*chrsz);return core_md5(d.concat(g),512+128)}function safe_add(a,d){var c=(a&65535)+(d&65535);var b=(a>>16)+(d>>16)+(c>>16);return(b<<16)|(c&65535)}function bit_rol(a,b){return(a<>>(32-b))}function str2binl(d){var c=Array();var a=(1<>5]|=(d.charCodeAt(b/chrsz)&a)<<(b%32)}return c}function binl2str(c){var d="";var a=(1<>5]>>>(b%32))&a)}return d}function binl2hex(c){var b=hexcase?"0123456789ABCDEF":"0123456789abcdef";var d="";for(var a=0;a>2]>>((a%4)*8+4))&15)+b.charAt((c[a>>2]>>((a%4)*8))&15)}return d}function binl2b64(d){var c="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";var f="";for(var b=0;b>2]>>8*(b%4))&255)<<16)|(((d[b+1>>2]>>8*((b+1)%4))&255)<<8)|((d[b+2>>2]>>8*((b+2)%4))&255);for(var a=0;a<4;a++){if(b*8+a*6>d.length*32){f+=b64pad}else{f+=c.charAt((e>>6*(3-a))&63)}}}return f};window.f = hex_md5(mwqqppz)'
atob()
函数是JavaScript
中的一个内置函数,用于将base64
编码的字符串解码为原始字符串。
使用美化工具(**https://beautifier.io/**)美化得到下面可读的`JavaScript`代码:
var hexcase = 0; var b64pad = ""; var chrsz = 16;
function hex_md5(a)
{
return binl2hex(core_md5(str2binl(a), a.length * chrsz))
}function b64_md5(a)
{
return binl2b64(core_md5(str2binl(a), a.length * chrsz))
}function str_md5(a)
{
return binl2str(core_md5(str2binl(a), a.length * chrsz))
}function hex_hmac_md5(a, b)
{
return binl2hex(core_hmac_md5(a, b))
}function b64_hmac_md5(a, b)
{
return binl2b64(core_hmac_md5(a, b))
}function str_hmac_md5(a, b)
{
return binl2str(core_hmac_md5(a, b))
}function md5_vm_test()
{
return hex_md5(“abc”) == “900150983cd24fb0d6963f7d28e17f72”
}function core_md5(p, k)
{
p[k >> 5] |= 128 << ((k) % 32);
p[(((k + 64) >>> 9) << 4) + 14] = k;
var o = 1732584193;
var n = -271733879;
var m = -1732584194;
var l = 271733878;
for (var g = 0; g < p.length; g += 16)
{
var j = o;
var h = n;
var f = m;
var e = l;
o = md5_ff(o, n, m, l, p[g + 0], 7, -680976936);
l = md5_ff(l, o, n, m, p[g + 1], 12, -389564586);
m = md5_ff(m, l, o, n, p[g + 2], 17, 606105819);
n = md5_ff(n, m, l, o, p[g + 3], 22, -1044525330);
o = md5_ff(o, n, m, l, p[g + 4], 7, -176418897);
l = md5_ff(l, o, n, m, p[g + 5], 12, 1200080426);
m = md5_ff(m, l, o, n, p[g + 6], 17, -1473231341);
n = md5_ff(n, m, l, o, p[g + 7], 22, -45705983);
o = md5_ff(o, n, m, l, p[g + 8], 7, 1770035416);
l = md5_ff(l, o, n, m, p[g + 9], 12, -1958414417);
m = md5_ff(m, l, o, n, p[g + 10], 17, -42063);
n = md5_ff(n, m, l, o, p[g + 11], 22, -1990404162);
o = md5_ff(o, n, m, l, p[g + 12], 7, 1804660682);
l = md5_ff(l, o, n, m, p[g + 13], 12, -40341101);
m = md5_ff(m, l, o, n, p[g + 14], 17, -1502002290);
n = md5_ff(n, m, l, o, p[g + 15], 22, 1236535329);
o = md5_gg(o, n, m, l, p[g + 1], 5, -165796510);
l = md5_gg(l, o, n, m, p[g + 6], 9, -1069501632);
m = md5_gg(m, l, o, n, p[g + 11], 14, 643717713);
n = md5_gg(n, m, l, o, p[g + 0], 20, -373897302);
o = md5_gg(o, n, m, l, p[g + 5], 5, -701558691);
l = md5_gg(l, o, n, m, p[g + 10], 9, 38016083);
m = md5_gg(m, l, o, n, p[g + 15], 14, -660478335);
n = md5_gg(n, m, l, o, p[g + 4], 20, -405537848);
o = md5_gg(o, n, m, l, p[g + 9], 5, 568446438);
l = md5_gg(l, o, n, m, p[g + 14], 9, -1019803690);
m = md5_gg(m, l, o, n, p[g + 3], 14, -187363961);
n = md5_gg(n, m, l, o, p[g + 8], 20, 1163531501);
o = md5_gg(o, n, m, l, p[g + 13], 5, -1444681467);
l = md5_gg(l, o, n, m, p[g + 2], 9, -51403784);
m = md5_gg(m, l, o, n, p[g + 7], 14, 1735328473);
n = md5_gg(n, m, l, o, p[g + 12], 20, -1921207734);
o = md5_hh(o, n, m, l, p[g + 5], 4, -378558);
l = md5_hh(l, o, n, m, p[g + 8], 11, -2022574463);
m = md5_hh(m, l, o, n, p[g + 11], 16, 1839030562);
n = md5_hh(n, m, l, o, p[g + 14], 23, -35309556);
o = md5_hh(o, n, m, l, p[g + 1], 4, -1530992060);
l = md5_hh(l, o, n, m, p[g + 4], 11, 1272893353);
m = md5_hh(m, l, o, n, p[g + 7], 16, -155497632);
n = md5_hh(n, m, l, o, p[g + 10], 23, -1094730640);
o = md5_hh(o, n, m, l, p[g + 13], 4, 681279174);
l = md5_hh(l, o, n, m, p[g + 0], 11, -358537222);
m = md5_hh(m, l, o, n, p[g + 3], 16, -722881979);
n = md5_hh(n, m, l, o, p[g + 6], 23, 76029189);
o = md5_hh(o, n, m, l, p[g + 9], 4, -640364487);
l = md5_hh(l, o, n, m, p[g + 12], 11, -421815835);
m = md5_hh(m, l, o, n, p[g + 15], 16, 530742520);
n = md5_hh(n, m, l, o, p[g + 2], 23, -995338651);
o = md5_ii(o, n, m, l, p[g + 0], 6, -198630844);
l = md5_ii(l, o, n, m, p[g + 7], 10, 11261161415);
m = md5_ii(m, l, o, n, p[g + 14], 15, -1416354905);
n = md5_ii(n, m, l, o, p[g + 5], 21, -57434055);
o = md5_ii(o, n, m, l, p[g + 12], 6, 1700485571);
l = md5_ii(l, o, n, m, p[g + 3], 10, -1894446606);
m = md5_ii(m, l, o, n, p[g + 10], 15, -1051523);
n = md5_ii(n, m, l, o, p[g + 1], 21, -2054922799);
o = md5_ii(o, n, m, l, p[g + 8], 6, 1873313359);
l = md5_ii(l, o, n, m, p[g + 15], 10, -30611744);
m = md5_ii(m, l, o, n, p[g + 6], 15, -1560198380);
n = md5_ii(n, m, l, o, p[g + 13], 21, 1309151649);
o = md5_ii(o, n, m, l, p[g + 4], 6, -145523070);
l = md5_ii(l, o, n, m, p[g + 11], 10, -1120210379);
m = md5_ii(m, l, o, n, p[g + 2], 15, 718787259);
n = md5_ii(n, m, l, o, p[g + 9], 21, -343485551);
o = safe_add(o, j);
n = safe_add(n, h);
m = safe_add(m, f);
l = safe_add(l, e)
}
return Array(o, n, m, l)
}function md5_cmn(h, e, d, c, g, f)
{
return safe_add(bit_rol(safe_add(safe_add(e, h), safe_add(c, f)), g), d)
}function md5_ff(g, f, k, j, e, i, h)
{
return md5_cmn((f & k) | ((~f) & j), g, f, e, i, h)
}function md5_gg(g, f, k, j, e, i, h)
{
return md5_cmn((f & j) | (k & (~j)), g, f, e, i, h)
}function md5_hh(g, f, k, j, e, i, h)
{
return md5_cmn(f ^ k ^ j, g, f, e, i, h)
}function md5_ii(g, f, k, j, e, i, h)
{
return md5_cmn(k ^ (f | (~j)), g, f, e, i, h)
}function core_hmac_md5(c, f)
{
var e = str2binl(c);
if (e.length > 16)
{
e = core_md5(e, c.length * chrsz)
}
var a = Array(16),
d = Array(16);
for (var b = 0; b < 16; b++)
{
a[b] = e[b] ^ 909522486;
d[b] = e[b] ^ 1549556828
}
var g = core_md5(a.concat(str2binl(f)), 512 + f.length * chrsz);
return core_md5(d.concat(g), 512 + 128)
}function safe_add(a, d)
{
var c = (a & 65535) + (d & 65535);
var b = (a >> 16) + (d >> 16) + (c >> 16);
return (b << 16) | (c & 65535)
}function bit_rol(a, b)
{
return (a << b) | (a >>> (32 - b))
}function str2binl(d)
{
var c = Array();
var a = (1 << chrsz) - 1;
for (var b = 0; b < d.length * chrsz; b += chrsz)
{
c[b >> 5] |= (d.charCodeAt(b / chrsz) & a) << (b % 32)
}
return c
}function binl2str(c)
{
var d = “”;
var a = (1 << chrsz) - 1;
for (var b = 0; b < c.length * 32; b += chrsz)
{
d += String.fromCharCode((c[b >> 5] >>> (b % 32)) & a)
}
return d
}function binl2hex(c)
{
var b = hexcase ? “0123456789ABCDEF” : “0123456789abcdef”;
var d = “”;
for (var a = 0; a < c.length * 4; a++)
{
d += b.charAt((c[a >> 2] >> ((a % 4) * 8 + 4)) & 15) + b.charAt((c[a >>
2] >> ((a % 4) * 8)) & 15)
}
return d
}
function binl2b64(d)
{
var c = “ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/”;
var f = “”;
for (var b = 0; b < d.length * 4; b += 3)
{
var e = (((d[b >> 2] >> 8 * (b % 4)) & 255) << 16) | (((d[b + 1 >> 2] >>
8 * ((b + 1) % 4)) & 255) << 8) | ((d[b + 2 >> 2] >> 8 * ((b +
2) % 4)) & 255);
for (var a = 0; a < 4; a++)
{
if (b * 8 + a * 6 > d.length * 32)
{
f += b64pad
}
else
{
f += c.charAt((e >> 6 * (3 - a)) & 63)
}
}
}
return f
};
window.f = hex_md5(mwqqppz)
我们发现window.f = hex_md5(mwqqppz)
,是通过很多函数加密的,最重要的就是这最后一条,接下来我们来寻找mwqqppz
参数。
回到调用oo0O0()函数
的部分:
var _0x2268f9 = Date.parse(new Date()) + 100000000;
var _0x57feae = oo0O0(_0x2268f9.toString()) + window.f;
const _0x5d83a3 = {
page: window.page,
m: _0x57feae + "丨" + _0x2268f9 / 1000
};
以及eval()
那一条代码,分析传入的参数。
eval(atob(window['b'])[J('0x0', ']dQW')](J('0x1', 'GTu!'), '\x27' + mw + '\x27'));
通过分析,我们发现J
函数就在oo0O0
函数内定义,我们将J
函数抠出:
var U = ['W5r5W6VdIHZcT8kU', 'WQ8CWRaxWQirAW==']; var J = function (o, E) { o = o - 0x0; var N = U[o]; if (J['bSSGte'] === undefined) { var Y = function (w) { var m = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=', T = String(w)['replace'](/=+$/, ''); var A = ''; for (var C = 0x0, b, W, l = 0x0; W = T['charAt'](l++); ~ W && (b = C % 0x4 ? b * 0x40 + W : W, C++ % 0x4) ? A += String['fromCharCode'](0xff & b >> (-0x2 * C & 0x6)) : 0x0) { W = m['indexOf'](W) } return A }; var t = function (w, m) { var T = [], A = 0x0, C, b = '', W = ''; w = Y(w); for (var R = 0x0, v = w['length']; R < v; R++) { W += '%' + ('00' + w['charCodeAt'](R)['toString']( 0x10))['slice'](-0x2) } w = decodeURIComponent(W); var l; for (l = 0x0; l < 0x100; l++) { T[l] = l } for (l = 0x0; l < 0x100; l++) { A = (A + T[l] + m['charCodeAt'](l % m['length'])) % 0x100, C = T[l], T[l] = T[A], T[A] = C } l = 0x0, A = 0x0; for (var L = 0x0; L < w['length']; L++) { l = (l + 0x1) % 0x100, A = (A + T[l]) % 0x100, C = T[l], T[l] = T[A], T[A] = C, b += String[ 'fromCharCode'](w['charCodeAt'](L) ^ T[(T[ l] + T[A]) % 0x100]) } return b }; J['luAabU'] = t, J['qlVPZg'] = {}, J['bSSGte'] = !![] } var H = J['qlVPZg'][o]; return H === undefined ? (J['TUDBIJ'] === undefined && (J[ 'TUDBIJ'] = !![]), N = J['luAabU'](N, E), J['qlVPZg'][ o] = N) : N = H, N };
console.log(J(‘0x0’, ‘]dQW’))
console.log(J(‘0x1’, ‘GTu!’))
并在浏览器执行:
得到:
J('0x0', ']dQW')
的值为:replace
J('0x1', 'GTu!')
的值为:mwqqppz
又'\x27' + mw + '\x27'
的值为1715257368000
,也就是说这里就是某一个字符串进行replace
操作,将mwqqppz
替换成了时间戳mw
,最终这个字符串就是window['b']
的值,是一个base64
的编码。
eval()
部分的代码即可替换为:
eval(atob(window['b'])["replace"]("mwqqppz","1715257368000"));
通过以上分析,我们可以知道window.f = hex_md5(mwqqppz)
中mwqqppz
就是时间戳mw
,这个例子中时间戳mw
的值为1715257368000
。
mwqqppz
字符串本身只是用于标识需要被替换的占位字符串,无特殊含义,然后通过eval
执行。
我们手动替换下oo0O0_eval.js
中的window.f = hex_md5(mwqqppz)
为console.log(hex_md5("1715257368000"))
后执行,得到8d596e76013033e75c3e23015c553c55
,正是我们之前在控制台分析的_0x57feae
的值。
另外,我们通过猴子脚本hook
一下window.f
也能得到加密算法。
hook
脚本:
设置:
翻页调试:
查看第一个箭头所指的栈,清晰的看到加密算法运行时的代码:// ... // 跟我们解混淆后的代码不能说是一样,只能说是完全相同。 // 此处省略。
function binl2b64(d) {
var c = “ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/”;
var f = “”;
for (var b = 0; b < d.length * 4; b += 3) {
var e = (((d[b >> 2] >> 8 * (b % 4)) & 255) << 16) | (((d[b + 1 >> 2] >> 8 * ((b + 1) % 4)) & 255) << 8) | ((d[b + 2 >> 2] >> 8 * ((b + 2) % 4)) & 255);
for (var a = 0; a < 4; a++) {
if (b * 8 + a * 6 > d.length * 32) {
f += b64pad
} else {
f += c.charAt((e >> 6 * (3 - a)) & 63)
}
}
}
return f
}
;window.f = hex_md5(‘1715257368000’)
完整 JS 加密代码如下:
var window =global
var hexcase = 0;
var b64pad = “”;
var chrsz = 16;function hex_md5(a) {
return binl2hex(core_md5(str2binl(a), a.length * chrsz))
}function b64_md5(a) {
return binl2b64(core_md5(str2binl(a), a.length * chrsz))
}function str_md5(a) {
return binl2str(core_md5(str2binl(a), a.length * chrsz))
}function hex_hmac_md5(a, b) {
return binl2hex(core_hmac_md5(a, b))
}function b64_hmac_md5(a, b) {
return binl2b64(core_hmac_md5(a, b))
}function str_hmac_md5(a, b) {
return binl2str(core_hmac_md5(a, b))
}function md5_vm_test() {
return hex_md5(“abc”) == “900150983cd24fb0d6963f7d28e17f72”
}function core_md5(p, k) {
p[k >> 5] |= 128 << ((k) % 32);
p[(((k + 64) >>> 9) << 4) + 14] = k;
var o = 1732584193;
var n = -271733879;
var m = -1732584194;
var l = 271733878;
for (var g = 0; g < p.length; g += 16) {
var j = o;
var h = n;
var f = m;
var e = l;
o = md5_ff(o, n, m, l, p[g + 0], 7, -680976936);
l = md5_ff(l, o, n, m, p[g + 1], 12, -389564586);
m = md5_ff(m, l, o, n, p[g + 2], 17, 606105819);
n = md5_ff(n, m, l, o, p[g + 3], 22, -1044525330);
o = md5_ff(o, n, m, l, p[g + 4], 7, -176418897);
l = md5_ff(l, o, n, m, p[g + 5], 12, 1200080426);
m = md5_ff(m, l, o, n, p[g + 6], 17, -1473231341);
n = md5_ff(n, m, l, o, p[g + 7], 22, -45705983);
o = md5_ff(o, n, m, l, p[g + 8], 7, 1770035416);
l = md5_ff(l, o, n, m, p[g + 9], 12, -1958414417);
m = md5_ff(m, l, o, n, p[g + 10], 17, -42063);
n = md5_ff(n, m, l, o, p[g + 11], 22, -1990404162);
o = md5_ff(o, n, m, l, p[g + 12], 7, 1804660682);
l = md5_ff(l, o, n, m, p[g + 13], 12, -40341101);
m = md5_ff(m, l, o, n, p[g + 14], 17, -1502002290);
n = md5_ff(n, m, l, o, p[g + 15], 22, 1236535329);
o = md5_gg(o, n, m, l, p[g + 1], 5, -165796510);
l = md5_gg(l, o, n, m, p[g + 6], 9, -1069501632);
m = md5_gg(m, l, o, n, p[g + 11], 14, 643717713);
n = md5_gg(n, m, l, o, p[g + 0], 20, -373897302);
o = md5_gg(o, n, m, l, p[g + 5], 5, -701558691);
l = md5_gg(l, o, n, m, p[g + 10], 9, 38016083);
m = md5_gg(m, l, o, n, p[g + 15], 14, -660478335);
n = md5_gg(n, m, l, o, p[g + 4], 20, -405537848);
o = md5_gg(o, n, m, l, p[g + 9], 5, 568446438);
l = md5_gg(l, o, n, m, p[g + 14], 9, -1019803690);
m = md5_gg(m, l, o, n, p[g + 3], 14, -187363961);
n = md5_gg(n, m, l, o, p[g + 8], 20, 1163531501);
o = md5_gg(o, n, m, l, p[g + 13], 5, -1444681467);
l = md5_gg(l, o, n, m, p[g + 2], 9, -51403784);
m = md5_gg(m, l, o, n, p[g + 7], 14, 1735328473);
n = md5_gg(n, m, l, o, p[g + 12], 20, -1921207734);
o = md5_hh(o, n, m, l, p[g + 5], 4, -378558);
l = md5_hh(l, o, n, m, p[g + 8], 11, -2022574463);
m = md5_hh(m, l, o, n, p[g + 11], 16, 1839030562);
n = md5_hh(n, m, l, o, p[g + 14], 23, -35309556);
o = md5_hh(o, n, m, l, p[g + 1], 4, -1530992060);
l = md5_hh(l, o, n, m, p[g + 4], 11, 1272893353);
m = md5_hh(m, l, o, n, p[g + 7], 16, -155497632);
n = md5_hh(n, m, l, o, p[g + 10], 23, -1094730640);
o = md5_hh(o, n, m, l, p[g + 13], 4, 681279174);
l = md5_hh(l, o, n, m, p[g + 0], 11, -358537222);
m = md5_hh(m, l, o, n, p[g + 3], 16, -722881979);
n = md5_hh(n, m, l, o, p[g + 6], 23, 76029189);
o = md5_hh(o, n, m, l, p[g + 9], 4, -640364487);
l = md5_hh(l, o, n, m, p[g + 12], 11, -421815835);
m = md5_hh(m, l, o, n, p[g + 15], 16, 530742520);
n = md5_hh(n, m, l, o, p[g + 2], 23, -995338651);
o = md5_ii(o, n, m, l, p[g + 0], 6, -198630844);
l = md5_ii(l, o, n, m, p[g + 7], 10, 11261161415);
m = md5_ii(m, l, o, n, p[g + 14], 15, -1416354905);
n = md5_ii(n, m, l, o, p[g + 5], 21, -57434055);
o = md5_ii(o, n, m, l, p[g + 12], 6, 1700485571);
l = md5_ii(l, o, n, m, p[g + 3], 10, -1894446606);
m = md5_ii(m, l, o, n, p[g + 10], 15, -1051523);
n = md5_ii(n, m, l, o, p[g + 1], 21, -2054922799);
o = md5_ii(o, n, m, l, p[g + 8], 6, 1873313359);
l = md5_ii(l, o, n, m, p[g + 15], 10, -30611744);
m = md5_ii(m, l, o, n, p[g + 6], 15, -1560198380);
n = md5_ii(n, m, l, o, p[g + 13], 21, 1309151649);
o = md5_ii(o, n, m, l, p[g + 4], 6, -145523070);
l = md5_ii(l, o, n, m, p[g + 11], 10, -1120210379);
m = md5_ii(m, l, o, n, p[g + 2], 15, 718787259);
n = md5_ii(n, m, l, o, p[g + 9], 21, -343485551);
o = safe_add(o, j);
n = safe_add(n, h);
m = safe_add(m, f);
l = safe_add(l, e)
}
return Array(o, n, m, l)
}function md5_cmn(h, e, d, c, g, f) {
return safe_add(bit_rol(safe_add(safe_add(e, h), safe_add(c, f)), g), d)
}function md5_ff(g, f, k, j, e, i, h) {
return md5_cmn((f & k) | ((~f) & j), g, f, e, i, h)
}function md5_gg(g, f, k, j, e, i, h) {
return md5_cmn((f & j) | (k & (~j)), g, f, e, i, h)
}function md5_hh(g, f, k, j, e, i, h) {
return md5_cmn(f ^ k ^ j, g, f, e, i, h)
}function md5_ii(g, f, k, j, e, i, h) {
return md5_cmn(k ^ (f | (~j)), g, f, e, i, h)
}function core_hmac_md5(c, f) {
var e = str2binl(c);
if (e.length > 16) {
e = core_md5(e, c.length * chrsz)
}
var a = Array(16),
d = Array(16);
for (var b = 0; b < 16; b++) {
a[b] = e[b] ^ 909522486;
d[b] = e[b] ^ 1549556828
}
var g = core_md5(a.concat(str2binl(f)), 512 + f.length * chrsz);
return core_md5(d.concat(g), 512 + 128)
}function safe_add(a, d) {
var c = (a & 65535) + (d & 65535);
var b = (a >> 16) + (d >> 16) + (c >> 16);
return (b << 16) | (c & 65535)
}function bit_rol(a, b) {
return (a << b) | (a >>> (32 - b))
}function str2binl(d) {
var c = Array();
var a = (1 << chrsz) - 1;
for (var b = 0; b < d.length * chrsz; b += chrsz) {
c[b >> 5] |= (d.charCodeAt(b / chrsz) & a) << (b % 32)
}
return c
}function binl2str(c) {
var d = “”;
var a = (1 << chrsz) - 1;
for (var b = 0; b < c.length * 32; b += chrsz) {
d += String.fromCharCode((c[b >> 5] >>> (b % 32)) & a)
}
return d
}function binl2hex(c) {
var b = hexcase ? “0123456789ABCDEF” : “0123456789abcdef”;
var d = “”;
for (var a = 0; a < c.length * 4; a++) {
d += b.charAt((c[a >> 2] >> ((a % 4) * 8 + 4)) & 15) + b.charAt((c[a >>
2] >> ((a % 4) * 8)) & 15)
}
return d
}function binl2b64(d) {
var c = “ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/”;
var f = “”;
for (var b = 0; b < d.length * 4; b += 3) {
var e = (((d[b >> 2] >> 8 * (b % 4)) & 255) << 16) | (((d[b + 1 >> 2] >>
8 * ((b + 1) % 4)) & 255) << 8) | ((d[b + 2 >> 2] >> 8 * ((b +
2) % 4)) & 255);
for (var a = 0; a < 4; a++) {
if (b * 8 + a * 6 > d.length * 32) {
f += b64pad
} else {
f += c.charAt((e >> 6 * (3 - a)) & 63)
}
}
}
return f
};// window.f = hex_md5(mwqqppz)
// console.log(hex_md5(“1715257368000”))
// 整合hex_md5的参数
function get_m_value() {
var _0x2268f9 = Date[“parse”](new Date()) + 100000000;
// var _0x2268f9 = 1715257368000
var mwqqppz = _0x2268f9"toString";
var m = hex_md5(mwqqppz);
var m_value = m + ‘丨’ + _0x2268f9 / 1000;
return m_value;
};
console.log(get_m_value())
在pycharm
中运行结果如下:
和浏览器分析时的m
值一致,JS逆向过程结束。
大功告成
逆向结束,剩下的编码就是模拟请求,解析结果了。
将
curl
命令转换为Python
、JavaScript
等: https://curlconverter.com/
老规矩,借用curl
命令转代码工具快速生成代码,略作修改即可,重点是逆向过程分析。
#! -*-conding=: UTF-8 -*-
import execjs
import requestscookies = {
‘tk’: ‘改成自己的tk’,
‘sessionid’: ‘改成自己的sessionid’,
}headers = {
‘authority’: ‘match.yuanrenxue.cn’,
‘accept’: ‘application/json, text/javascript, /; q=0.01’,
‘accept-language’: ‘zh-CN,zh;q=0.9’,
‘cache-control’: ‘no-cache’,
# ‘cookie’: ‘tk=7699535563564611254; sessionid=0qtsyidunti3daajuz7l2qkqy3m0knmd’,
‘pragma’: ‘no-cache’,
‘referer’: ‘第一题 js混淆源码乱码 - 猿人学’,
‘sec-ch-ua’: ‘“Not A(Brand”;v=“99”, “Google Chrome”;v=“121”, “Chromium”;v=“121”’,
‘sec-ch-ua-mobile’: ‘?0’,
‘sec-ch-ua-platform’: ‘“Windows”’,
‘sec-fetch-dest’: ‘empty’,
‘sec-fetch-mode’: ‘cors’,
‘sec-fetch-site’: ‘same-origin’,
‘user-agent’: ‘Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/121.0.0.0 Safari/537.36’,
‘x-requested-with’: ‘XMLHttpRequest’,
}params = {
‘page’: 1,
‘m’: ‘’,
}def calculate_m_value():
with open(‘_m.js’, mode=‘r’, encoding=‘utf-8’) as f:
JsData = f.read()
m_value = execjs.compile(JsData).call(‘get_m_value’)
# m_value_process = m_value.replace(“丨”, “%E4%B8%A8”)
return m_valuedef main() -> int:
_sum = 0
times = 0
for page_number in range(1, 6):
params[“page”] = page_number
params[“m”] = calculate_m_value()
print(params)
response = requests.get(‘https://match.yuanrenxue.cn/api/match/1’, params=params, cookies=cookies,
headers=headers).json()for data in response["data"]: _sum += data["value"] times += 1 if not times: return 0 return int(_sum / times)
if name == “main”:
result = main()
print(int(result))
结语
至此,这次逆向之旅终于结束了。
如果觉得不错,点赞
,在看
,关注
安排起来吧。
这是一个从 https://juejin.cn/post/7369023361954332698 下的原始话题分离的讨论话题