郑重声明
此处超星指代的是本人独立开发的自主单机游戏-》超级星星V0.1版
并非通俗意义的超星!请勿进行过度联想
本文全程为自攻自破
如有雷同纯属巧合
预览
data:image/s3,"s3://crabby-images/1171c/1171c61ef0684bd96fa452dd858f564bf826c27b" alt="图片.png"
开始
通过观察代码以及动态运行可以发现_0xdd65b4是解密函数
data:image/s3,"s3://crabby-images/68bac/68bac15c4903378fda021e16885bb7e0f5c95598" alt="图片.png"
点进去直接下个断
data:image/s3,"s3://crabby-images/52c3c/52c3ca8144de1818b9f84422f30145e8a7e07d47" alt="图片.png"
然后字符串化一下拷贝出去字符数组
data:image/s3,"s3://crabby-images/08d04/08d04125d722712cd6d53f48774cde801a444049" alt="图片.png"
在nodejs里写出来解密函数,并且parse一下拷贝的字符串
data:image/s3,"s3://crabby-images/de72f/de72fa814995bd7075028b47955d0ab78e262b5d" alt="图片.png"
观察代码发现是一个CallExpression节点
data:image/s3,"s3://crabby-images/97f38/97f381991abcf86d3bcea4f9c85335644149a167" alt="图片.png"
然后我们直接写一下代码
traverse(ast, {
CallExpression(path) {
if (path.node.callee.name === "_0xdd65b4") {
let loc = path.node.arguments[0].value;
let str_node = types.stringLiteral(crp_str(loc));
path.replaceInline(str_node);
}
},
});
可以发现已经解开一大堆字符串了
data:image/s3,"s3://crabby-images/4b6de/4b6de232a1713062e9d74990b922f69181f24a80" alt="图片.png"
然后搜索一下,发现还有很多加密的函数调用
原来是在函数头上做了一下赋值,然后由函数的引用来做加密
data:image/s3,"s3://crabby-images/d42d1/d42d107eb96bcd9f1add3146f491330fc3a16650" alt="图片.png"
我们丢到Ast看一下,可以发现是一个VariableDeclarator对象
id是赋值的名字,init是加密的字符串,那我们继续写VariableDeclarator的AST解析吧
data:image/s3,"s3://crabby-images/b5757/b57578357db4d7db7afb0f79e5e33202036a311b" alt="图片.png"
但是根据测试发现一个比较头疼的问题
他对函数进行了多次赋值,所以我们必须写一个递归遍历的东西
data:image/s3,"s3://crabby-images/7a8f7/7a8f7b4d172171123cf221a9cab3d76bb06a4a55" alt="图片.png"
直接上代码
function find_and_fuck_crp(path, func_name) {
let bind = path.scope.getBinding(func_name);
let refer_list = bind.referencePaths;
refer_list.forEach((item) => {
let parent_path = item.parentPath;
if (parent_path.node.type === "CallExpression") {
let loc = parent_path.node.arguments[0].value;
let str_node = types.stringLiteral(crp_str(loc));
parent_path.replaceInline(str_node);
} else {
if (parent_path.node.type === "VariableDeclarator") {
find_and_fuck_crp(parent_path, parent_path.node.id.name);
}
}
});
path.remove()
}
traverse(ast, {
VariableDeclarator(path) {
let func_name = path.node.id.name;
let init_name = path.node.init?.name;
if (init_name === "_0xdd65b4") {
find_and_fuck_crp(path, func_name);
}
},
});
经过递归遍历我们已经具备了一定的可读性了
data:image/s3,"s3://crabby-images/c4312/c43121e46c4589d2897c9471593cdfde86fc6731" alt="图片.png"
然后发现还有一个加密的函数引用a0_0x1f3a,直接还按刚才的方法
data:image/s3,"s3://crabby-images/7a3f2/7a3f204c8ef6f454c271de3bfaf907709c8c62a6" alt="图片.png"
处理之后删除加密的函数
再看一下,可以发现已经有了初步的样子
data:image/s3,"s3://crabby-images/1da93/1da936d1c822cf7ca1f1b98c97dcf9d34ae62d44" alt="图片.png"
但是部分字符串里还有混淆
data:image/s3,"s3://crabby-images/81893/8189396128e1794324303372515754dc506328bb" alt="图片.png"
配置一下generator函数
let code = generator(ast, {
minified: true,
jsescOption: { minimal: true },
}).code;
到这里已经具备基本的调试条件了
剩下的大家也可以根据这个思路进行微妙
结语
撒花~