关注微信公众号:K哥爬虫,继续分享爬虫进阶、JS/安卓逆向等技术干货!

申明

本文章中所有内容仅供学习交换,抓包内容、敏感网址、数据接口均已做脱敏解决,严禁用于商业用途和非法用处,否则由此产生的所有结果均与作者无关,若有侵权,请分割我立刻删除!

逆向指标

  • 指标:网洛者反反爬虫练习平台第四题:JSFuck 加密
  • 链接:http://spider.wangluozhe.com/...
  • 简介:本题依然是要求采集100页的全副数字,并计算所有数据加和,须要抠出源码进行计算,次要应用了 JSFuck 加密

JSFuck 简介

JSFuck、AAEncode、JJEncode 都是同一个作者,JSFuck 由日本的 Yosuke HASEGAWA 在 2010 发明,它能够将任意 JavaScript 编码为仅应用 6 个符号的混同模式 []()!+,2012 年,Martin Kleppe 在 GitHub 上创立了一个 jsfuck 我的项目和一个 JSFuck.com 网站,其中蕴含应用该编码器实现的 Web 应用程序。JSFuck 可用于绕过对网站上提交的恶意代码的检测,例如跨站点脚本(XSS)攻打。JSFuck 的另一个潜在用处在于代码混同,目前的 jQuery 就曾经有通过 JSFuck 混同后的功能齐全的版本。

在线体验地址:https://utf-8.jp/public/jsfuc... http://www.jsfuck.com/

失常的一段 JS 代码:

alert(1)

通过 JSFuck 混同之后的代码相似于:

[][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]][([][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]]+[])[!+[]+!+[]+!+[]]+(!![]+[][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]])[+!+[]+[+[]]]+([][[]]+[])[+!+[]]+(![]+[])[!+[]+!+[]+!+[]]+(!![]+[])[+[]]+(!![]+[])[+!+[]]+([][[]]+[])[+[]]+([][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]]+[])[!+[]+!+[]+!+[]]+(!![]+[])[+[]]+(!![]+[][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]])[+!+[]+[+[]]]+(!![]+[])[+!+[]]]((!![]+[])[+!+[]]+(!![]+[])[!+[]+!+[]+!+[]]+(!![]+[])[+[]]+([][[]]+[])[+[]]+(!![]+[])[+!+[]]+([][[]]+[])[+!+[]]+(+[![]]+[][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]])[+!+[]+[+!+[]]]+(!![]+[])[!+[]+!+[]+!+[]]+(+(!+[]+!+[]+!+[]+[+!+[]]))[(!![]+[])[+[]]+(!![]+[][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]])[+!+[]+[+[]]]+([]+[])[([][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]]+[])[!+[]+!+[]+!+[]]+(!![]+[][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]])[+!+[]+[+[]]]+([][[]]+[])[+!+[]]+(![]+[])[!+[]+!+[]+!+[]]+(!![]+[])[+[]]+(!![]+[])[+!+[]]+([][[]]+[])[+[]]+([][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]]+[])[!+[]+!+[]+!+[]]+(!![]+[])[+[]]+(!![]+[][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]])[+!+[]+[+[]]]+(!![]+[])[+!+[]]][([][[]]+[])[+!+[]]+(![]+[])[+!+[]]+((+[])[([][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]]+[])[!+[]+!+[]+!+[]]+(!![]+[][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]])[+!+[]+[+[]]]+([][[]]+[])[+!+[]]+(![]+[])[!+[]+!+[]+!+[]]+(!![]+[])[+[]]+(!![]+[])[+!+[]]+([][[]]+[])[+[]]+([][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]]+[])[!+[]+!+[]+!+[]]+(!![]+[])[+[]]+(!![]+[][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]])[+!+[]+[+[]]]+(!![]+[])[+!+[]]]+[])[+!+[]+[+!+[]]]+(!![]+[])[!+[]+!+[]+!+[]]]](!+[]+!+[]+!+[]+[!+[]+!+[]])+(![]+[])[+!+[]]+(![]+[])[!+[]+!+[]])()((![]+[])[+!+[]]+(![]+[])[!+[]+!+[]]+(!![]+[])[!+[]+!+[]+!+[]]+(!![]+[])[+!+[]]+(!![]+[])[+[]]+([][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]]+[])[+!+[]+[!+[]+!+[]+!+[]]]+[+!+[]]+([+[]]+![]+[][(![]+[])[+[]]+(![]+[])[!+[]+!+[]]+(![]+[])[+!+[]]+(!![]+[])[+[]]])[!+[]+!+[]+[+[]]])

JSFuck 中常见的元素、数字、符号转换如下表,更多元素可参考 JSFuck 官网 GitHub 或 JSFuck 维基百科:

ValueJSFuck
false![]
true!![] or !+[]
NaN+[![]]
undefined[][[]]
Infinity+(+!+[]+(!+[]+[])[!+[]+!+[]+!+[]]+[+!+[]]+[+[]]+[+[]]+[+[]])
Array[]
Number+[]
String[]+[]
Boolean![]
Function[]["filter"]
eval[]["filter"]["constructor"]( CODE )()
window[]["filter"]["constructor"]("return this")()
+(+(+!+[]+(!+[]+[])[!+[]+!+[]+!+[]]+[+!+[]]+[+[]]+[+[]])+[])[!+[]+!+[]]
.(+(+!+[]+[+!+[]]+(!![]+[])[!+[]+!+[]+!+[]]+[!+[]+!+[]]+[+[]])+[])[+!+[]]
0+[]
1+!![] or +!+[]
2!![]+!![] or !+[]+!+[]
3!![]+!![]+!![] or !+[]+!+[]+!+[]
a(![]+[])[+!+[]]
d([][[]]+[])[!+[]+!+[]]
e(!![]+[])[!+[]+!+[]+!+[]]
f(![]+[])[+[]]

咱们以字母 a 为例,来演示一遍其混同的流程:

  1. "false"[1]:字母 a 取自字符串 false,在 false 中,a 的索引值是 1;
  2. (false+[])[1]:false 能够写成 false+[],即布尔常量 false 加上一个空数组;
  3. (![]+[])[1]:false 又能够写成 ![],即否定利用于空数组;
  4. (![]+[])[+true]:1 是一个数字,咱们能够把它写成 +true;
  5. (![]+[])[+!![]]:因为 false 是 ![],所以 true 就是 !![],生成最终混同代码。

JSFuck 解混同办法

JSFuck 在调用办法时通常都是通过 Function(xxx)() 和 eval(xxx) 的模式来执行,因而 JSFuck 常见解混同的形式如下:

  1. 应用在线工具间接解密,比方:https://lelinhtinh.github.io/... ;
  2. 针对 Function 的状况,复制代码最外层倒数第二个括号内的内容,放到浏览器外面去间接执行就能够看到源码;
  3. 针对 eval 的状况,复制代码最外层最初一个括号内的内容,放到浏览器外面去间接执行就能够看到源码;
  4. 应用 Hook 的形式,别离 Hook Function 和 eval,打印输出源码;
  5. 应用 AST 进行解混同,AST 的教程 K 哥后续也会写,本文不具体介绍。

如后面 alert(1) 的混同代码,复制最外层最初一个括号内的内容到浏览器,就能够看到源代码:

逆向参数

逆向的指标次要是翻页接口 _signature 参数,调用的加密办法依然是 window.get_sign(),和后面几题是一样的,本文不再赘述,不分明的能够去看 K 哥上期的文章。

持续跟进,会发现是一个 JSFuck 混同:

咱们将这段代码复制进去,放到编辑器外面,这里以 PyCharm 为例,因为咱们要选中匹配括号里的内容,所以咱们能够设置一下 PyCharm 括号匹配高亮为红色,便于咱们查找,顺次点击 File - Settings - Editor - Color Scheme - General - Code - Matched brace,设置 Background 为显眼的色彩:

此时咱们选中最初一个括号,往上找,就能够非常明显地看到与之匹配的另一个括号,如下图所示:

咱们将括号外面的内容复制进去(能够蕴含括号,也能够不蕴含),放到浏览器控制台运行一下,就能够看到源码了:

除了这种办法以外,咱们还能够应用 Hook 的形式,间接捕捉源码而后打印输出,留神到这段混同代码最初没有 () 括号,那就是 eval 的形式执行的,咱们编写 Hook eval 代码如下:

eval_ = eval;eval = function (a){    debugger;    return eval_()}// 另外提供一个 Hook Function 的代码// Function.prototype.constructor_ = Function.prototype.constructor;// Function.prototype.constructor = function (a) {//     debugger;//     return Function.prototype.constructor_(a);// };

刷新网页,间接断下,此时 a 的值就是源码:

将源码复制下来,本地剖析一下:

(function () {    let time_tmp = Date.now();    let date = Date.parse(new Date());    window = {};    let click = window.document.onclick;    let key_tmp;    let iv_tmp;    if (!click) {        key_tmp = date * 1234;    } else {        key_tmp = date * 1244;    }    if (time_tmp - window.time < 1000) {        iv_tmp = date * 4321;    } else {        iv_tmp = date * 4311;    }    const key = CryptoJS.enc.Utf8.parse(key_tmp);    var iv = CryptoJS.enc.Utf8.parse(iv_tmp);    (function tmp(date, key, iv) {        function Encrypt(word) {            let srcs = CryptoJS.enc.Utf8.parse(word);            let encrypted = CryptoJS.AES.encrypt(srcs, key, {                iv: iv,                mode: CryptoJS.mode.CBC,                padding: CryptoJS.pad.Pkcs7            });            return encrypted.ciphertext.toString().toUpperCase();        }        window.sign = Encrypt(date);    })(date, key, iv);})();

能够看到就是一个 AES 加密,这里次要留神有两个 if-else 语句,第一个判断是否存在 window.document.onclick,第二个是时间差的判断,咱们能够在控制台去尝试取一下 window.document.onclickwindow.time,看一下到底走的是 if 还是 else,在本地把这两个值也补全即可,实际上通过K哥测试 window.document.onclick 为 null,而后不论是走 if 还是 else 都是能够拿到后果的,所以对于本题来说,两个 window 对象都无所谓,间接去掉,key_tmpiv_tmp 任意取值都能够。

自此本题剖析结束,本地改写之后,配合 Python 代码携带 _signature 挨个计算每一页的数据,最终提交胜利:

残缺代码

GitHub 关注 K 哥爬虫,继续分享爬虫相干代码!欢送 star !https://github.com/kgepachong/

以下只演示局部要害代码,不能间接运行! 残缺代码仓库地址:https://github.com/kgepachong...

JavaScript 加密代码

/* ==================================# @Time    : 2021-12-13# @Author  : 微信公众号:K哥爬虫# @FileName: challenge_4.js# @Software: PyCharm# ================================== */var CryptoJS = require('crypto-js')let date = Date.parse(new Date());window = {};let key_tmp = date * 1234;// let key_tmp = date * 1244;let iv_tmp = date * 4321;// let iv_tmp = date * 4311;const key = CryptoJS.enc.Utf8.parse(key_tmp);var iv = CryptoJS.enc.Utf8.parse(iv_tmp);(function tmp(date, key, iv) {    function Encrypt(word) {        let srcs = CryptoJS.enc.Utf8.parse(word);        let encrypted = CryptoJS.AES.encrypt(srcs, key, {            iv: iv,            mode: CryptoJS.mode.CBC,            padding: CryptoJS.pad.Pkcs7        });        return encrypted.ciphertext.toString().toUpperCase();    }    window.sign = Encrypt(date);})(date, key, iv);function getSign() {    return window.sign}// 测试输入// console.log(getSign())

Python 计算要害代码

# ==================================# --*-- coding: utf-8 --*--# @Time    : 2021-12-13# @Author  : 微信公众号:K哥爬虫# @FileName: challenge_4.py# @Software: PyCharm# ==================================import execjsimport requestschallenge_api = "http://spider.wangluozhe.com/challenge/api/4"headers = {    "Content-Type": "application/x-www-form-urlencoded; charset=UTF-8",    "Cookie": "将 cookie 值改为你本人的!",    "Host": "spider.wangluozhe.com",    "Origin": "http://spider.wangluozhe.com",    "Referer": "http://spider.wangluozhe.com/challenge/4",    "User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.45 Safari/537.36",    "X-Requested-With": "XMLHttpRequest"}def get_signature():    with open('challenge_4.js', 'r', encoding='utf-8') as f:        ppdai_js = execjs.compile(f.read())    signature = ppdai_js.call("getSign")    print("signature: ", signature)    return signaturedef main():    result = 0    for page in range(1, 101):        data = {            "page": page,            "count": 10,            "_signature": get_signature()        }        response = requests.post(url=challenge_api, headers=headers, data=data).json()        for d in response["data"]:            result += d["value"]    print("后果为: ", result)if __name__ == '__main__':    main()