前期准备
- 网址:
aHR0cHM6Ly9ycy5qc2hyc3MuamlhbmdzdS5nb3YuY24vaW5kZXgv
- 目标:
- 接口请求内容加密和响应内容解密
- 请求头加密参数
Web-Encrypt-Response-Encrypt-Key
和Web-Encrypt-Sign
- 涉及内容:
- js
worker
多线程通信 axios
网络请求库- 国密
sm2
、sm3
、sm4
- js
Axios
网址:Axios
这个网站用到了 axios,可以先了解一下 axios 这个网络请求库,主要是interceptor
这块
data:image/s3,"s3://crabby-images/c591a/c591a54f2168966a094fd246f7dc36800b6163d3" alt=""
拦截器
在请求或响应被 then 或 catch 处理前拦截它们。
jsx
// 添加请求拦截器
axios.interceptors.request.use(function (config) {
// 在发送请求之前做些什么
return config;
},function (error) {
// 对请求错误做些什么
return Promise.reject(error);
});
// 添加响应拦截器
axios.interceptors.response.use(function (response) {
// 2xx 范围内的状态码都会触发该函数。
// 对响应数据做点什么
return response;
},function (error) {
// 超出 2xx 范围的状态码都会触发该函数。
// 对响应错误做点什么
return Promise.reject(error);
});
分析流程
先来看一下要逆向的内容
data:image/s3,"s3://crabby-images/58455/58455c4e4ef67269fa2afbf2e8a7faf0c715c04d" alt=""
data:image/s3,"s3://crabby-images/17b6a/17b6a490141715f0f249c8e0f8c33c906672418f" alt=""
如上图所示,获取考试列表(getExamTaskIM
)这个接口的请求和返回数据都是加密的
请求加密分析
老规矩直接看堆栈,进去第一个下个断点,刷新页面
data:image/s3,"s3://crabby-images/65c94/65c94687b3d3372ff69ec75b35b48b59b1b1b7ef" alt=""
成功断下后发现 xhr send 发送已经完成了加密
data:image/s3,"s3://crabby-images/3503a/3503a4e3e400ef81b3e8790a53f1bfa35c2996c6" alt=""
往上找一下堆栈,发现一个axios
的请求拦截器,直接下断点刷新
data:image/s3,"s3://crabby-images/d2866/d286618c7311028ff709d37e2e2f12ac4ef3debc" alt=""
data:image/s3,"s3://crabby-images/b3fdd/b3fddeae6e4b7f1e3f90f6397532831abbea4e05" alt=""
断住之后可以追进去看一下
data:image/s3,"s3://crabby-images/5c1d6/5c1d645bebeb49aacdf1b1afbfccc51fd32a1f4b" alt=""
data:image/s3,"s3://crabby-images/61fc6/61fc6e60cf79951b9177a4994b94fd92f67d73e0" alt=""
有四个拦截器,分别进去下一个断点去查看,最终在最后一个拦截器发现关键数据
data:image/s3,"s3://crabby-images/6ac01/6ac01c4e9873bc17138dedfd6e23928afcec3600" alt=""
data:image/s3,"s3://crabby-images/e35e5/e35e593716ce0764411ca37de47fbdc3f452494f" alt=""
这是一个与worker
进行通信的异步函数,我们往下看看其他逻辑
data:image/s3,"s3://crabby-images/ec79c/ec79c60b53ff7d9ffb61efac8c74692f34b5f6dc" alt=""
发现有一个onmessage
的回调方法,我猜测加密流程是这样的:
- 主线程发送明文数据(
postMessage
) - work 线程接收并开始加密(
onMessage
) - work 线程加密完后发送加密数据给主线程(
postMessage
) - 主线程接收加密数据(
onMessage
)
异步方法大都有一个回调的过程
下图主线程为postMessage
一方,其它线程则需要使用onMessage
接受数据
data:image/s3,"s3://crabby-images/eb787/eb7878ca92ec94dd196152cbe9646e3860bea6a7" alt=""
data:image/s3,"s3://crabby-images/003ea/003ea7c2ef1c5f90fe3f2974c7fcd7bc2f96ecf2" alt=""
接着分析流程,我们先在这个onmessage
回调方法内下段,再跟堆栈就容易找到调用位置了
data:image/s3,"s3://crabby-images/62aa9/62aa912513b5a2963cdfbb6ed438d6b0c3053d4e" alt=""
data:image/s3,"s3://crabby-images/39488/39488ad284dcf73ca90e4db81be54aea065fcb0e" alt=""
找到postMessage
后往上看看就能发现关键加密逻辑
data:image/s3,"s3://crabby-images/6c19c/6c19c66cf80b88b596e1589d13a6083ee7160403" alt=""
- 源流程代码,大量的 promise
jsx
self.onmessage = function(t) {
var r, e, o, h, a, f, c;
(r = t.data.prefix,
e = t.data.data,
o = e._s1,
h = e._s2,
a = e.requestData,
f = Math.floor(Date.now() / 1e3).toString(),
c = function() {
return 2 === s
}
,
u.timestamp = f,
new Promise((function(t, e) {
new Promise((function(t, e) {
try {
t({
key: r + i.default._s2EN(o, h, 0)
})
...
}
...
new Promise((function(t, e) {
try {
var o = r + i.default._s2EN(JSON.stringify(a), h, 0)
, s = f + o;
t({
content: o,
signature: (0,
n.default)(s)
})
...
}
...
}
...
}
进到这里了直接单步调试就是了,直接跟到_s2EN 函数里,这里就是加密函数了,加密函数用到了两次,一次加密 sign,一次加密 payload
data:image/s3,"s3://crabby-images/c744a/c744ad5598aa52c0c24029cce9e121a63d24ac92" alt=""
这个其实就是sm2
标准算法,看方法名就能猜出个大概了,就懒得写过程了,直接给出个大概的代码吧
jsx
var prefix = "04";
// 解密response的key
var _s1 = "862344dec7e0907a2b215c37a57caf95";
// 加密data的publicKey
var _s2 = "04fc439405f925df23510517e1e5a8078d19b23b24d62190c40e632f1d0bcd784fc6fcf1a8c3b5cf7f422815c6b322176e89f56f781ccd3c36aa02e5d31400090a";
// 计算key
key = prefix + sm2(_s1, _s2, 0);
// 加密payload,计算sign
payload = {
"bge304": 202432990000309,
"bge316": "320199"
}
var timestamp = Math.floor(Date.now() / 1000).toString();
var content = prefix + sm2(JSON.stringify(payload), _s2, 0)
var sign = sm3(timestamp + o);
headers= {
"Web-Encrypt-Response-Encrypt-Key": key,
"Web-Encrypt-Sign": sign ,
"Web-Encrypt-Timestamp": timestamp
}
sign 加密过程忘记截图了,实际上就是个
sm3
加密
完成以上请求头和请求数据的加密后,就可以正常请求拿到返回数据了
响应解密分析
跟找加密过程差不多,直接找到响应拦截器就行
data:image/s3,"s3://crabby-images/db72e/db72e6cc561d3b070d30229ba2a552773088687a" alt=""
挨个下断点,发现第一个很像了
data:image/s3,"s3://crabby-images/e9524/e9524970d4ce18833f20919dd190496b965633a6" alt=""
这个web-encrypt-sign
是响应头里的一个值,往下翻翻会发现熟悉的postMessage
和onMessage
data:image/s3,"s3://crabby-images/d88ee/d88ee67325821290533c558ca40d723eb0d79bc8" alt=""
这里过程跟加密差不多就不详细写过程了,直接找关键点
data:image/s3,"s3://crabby-images/d96a9/d96a957bb6d3b49061f34d54bd9a28aae23d33fc" alt=""
data:image/s3,"s3://crabby-images/c367c/c367c017386c6192df62106dd201da7314574bd0" alt=""
看方法名能猜出个大概是sm4
解密,解密的 key 就是之前传入的_s1
data:image/s3,"s3://crabby-images/c270e/c270e62b90e7999c71eefaf192247192f6061e1f" alt=""
data:image/s3,"s3://crabby-images/b3190/b31907ec91046d78836fdc682de1f27d5111c848" alt=""
随便抓个包解密试一下,可以看到正常解密成功,解密出来的 response 还需要 html 实体解码一下才行
Python 还原实现
- 代码:
data:image/s3,"s3://crabby-images/5b212/5b212373147e5896c258af705b1f89d75b028c49" alt=""
- 成果:
data:image/s3,"s3://crabby-images/2b0b7/2b0b7f9d0d1b5a2bf810058bc7a297c2a6bb184f" alt=""
大功告成!
微信公众号
公众号更新比较快,欢迎关注!