JS逆向,前端加密暴力破解(小白无痛学习)

网站运行的时间轴

url–>加载html–>加载js–>运行js初始化–>用户触发某个事件–调用了某段js–>明文数据–>加密函数–>加密后的 数据–>send(给服务器发信息{XHR–SEND}) –>接收到服务器数据–>解密函数–>刷新函数–>刷新网页渲染

浏览器的调试功能

Chrome高阶调试指南:https://zhuanlan.zhihu.com/p/62177097

调试时使用最多的功能页面是:元素(ELements)、控制台(Console)、源代码(Sources)、网络(Network)

  • 元素(Elements):用于查看或修改HTML元素的属性、CSS属性、监听事件、断点(DOM断点:在JavaScript调试中,我们经常使用到断点调试,其实在DOM结构的调试中,我们也可以使用断点方法,这就是DOM Breakpoint(DOM 断点))

  • 控制台(Console):控制台一般用于执行一次性代码,查看JavaScript对象,查看调试日志信息或异常信息。

  • 源代码(Sources):该页面用于查看页面的HTML文件源代码、JavaScript源代码、CSS源代码,此外最重要的是可以调试JavaScript源代码,可以给JS代码添加断点等。

  • 网络(Network):网络页面主要用于查看header等与网络连接相关的信息。

  • 资源(Resources):查看本地资源信息的(cookie,local storage,session storage等)

定位到具体位置,可以右键进行编辑

这个修改也仅对当前的页面渲染生效,不会修改服务器的源代码

右边的侧栏个功能的介绍

控制台

作用是

  • 查看JS对象的及其属性

  • 执行JS语句

  • 查看控制台日志:当网页的JS代码中使用了console.log()函数时,该函数输出的日志信息会在控制台中显示。日志信息一般在开发调试时启用,而当正式上线后,一般会将该函数去掉。

小知识(tab键补全这里也是支持的),建议大家dom树学好

调试快捷键

  • F10,跳过当前方法(如果执行到一个自定义方法,不进入方法内部)

  • F11,进入当前方法(如果当前方法是一个自定义方法,进入方法内部)

  • SHIFT+F11 跳出当前方法

  • F8,跳到下一个断点

网络(Network)

  • Header:面板列出资源的请求url、HTTP方法、响应状态码、请求头和响应头及它们各自的值、请求参数等等

  • Preview:预览面板,用于资源的预览。

  • Response:响应信息面板包含资源还未进行格式处理的内容

  • Timing:资源请求的详细信息花费时间

查看Network基本信息,请求了哪些地址及每个URL的网络相关请求信息都可以看的到:URL,响应状态码,响应数据类型,响应数据大小,响应时间

请求URL可进行筛选和分类:选择不同分类,查看请求URL,方便查找

也可以直接Filter搜索查询相关URL,可以输入关键字或者正则表达式进行查询

那么在js文件中我们如何快速定位到加密函数

  • 搜索关键字:登陆时的uri、passwd、Encrypt、Decrypt、…..

  • 使用一个神奇的脚本提高效率:https://github.com/Cha111Ng1/Tampermonkey_cha11/blob/main/HookScript.js

利用加解密函数

  • 复原原加密逻辑

  • 抠出原有js

  • rpc主动调用

工具推荐

phantomjs

https://phantomjs.org/download.html

PhantomJS是一个基于webkit的javaScript API。它使用QtWebKit作为它核心浏览器的功能,使用webkit来编译解释执行javaScript代码。任何你可以基于在webkit浏览器做的事情,它都能做到。它不仅是个隐性的浏览器,提供了诸如css选择器、支持wen标准、DOM操作、json、HTML5等,同时也提供了处理文件I/O的操作,从而使你可以向操作系统读写文件等。phantomJS的用处可谓非常广泛诸如网络监测、网页截屏、无需浏览器的wen测试、页面访问自动化等。

在前端渗透过程中,常会遇到需要进行爆破,但密码字段使用了自定义加密算法加密的情况。此时可以使用Burp配合jsEncrypter插件自定义加密算法进行爆破。流程图

项目地址:

https://github.com/c0ny1/jsEncrypter

这里面自带靶场,直接利用小p搭建就可以

选取rsa加密,进行

找到公钥(ps:现在几乎没有公钥泄露了,几乎都被提交了)

加密js下载下来

先将rsa.js文件保存到本地,重命名为rsa.js,然后修加密JSphantomjs_server.js

开启端口

最后在Burp使用插件连接phantomjs_server.js中开启的webserver

然后用常用的字典爆破,

成功

项目地址:https://github.com/Cha111Ng1/Tampermonkey_cha11

一个渗透测试油猴脚本库,整理常用脚本

HookScript.js

一些用于hook的常用断点+禁用无限debug

// ==UserScript==
// @name 「Hook Script」fuck断点
// @namespace http://tampermonkey.net/
// @version 0.1
// @description 一些用于hook的常用断点,禁用无限debug
// @author Cha111Ng1
// @match http*://*/*
// @icon 
// @grant none
// @run-at document-start
// ==/UserScript==

(function () {
// 取消vm debug
(function(){}).constructor === Function;
Function.prototype.constructor = function(){};

'use strict';
console.log('# ++++++++++++++++++++++++++++++++++++++++++')
console.log('# + 微信公众号:星火sec 作者zero +')
console.log('# ++++++++++++++++++++++++++++++++++++++++++')

var userChoices = prompt(" ━━━━━━️☠─────── n请选择要启动的插件(多选用逗号分隔)n ━━━━━━️☠─────── n1. JSON.parse调用断点n2. 每当尝试设置document.cookie时断点n3.每当调用XMLHttpRequest时断点n4.当发送POST请求时断点");

if (userChoices) {
var choicesArray = userChoices.split(',').map(function (choice) {
return choice.trim();
});

choicesArray.forEach(function (chosenPlugin) {
switch (chosenPlugin) {
case '1':
//alert("JSON.parse调用断点");
var _parse = JSON.parse;
JSON.parse = function (arg) {
console.log("[+]「油猴」JSON.parse调用断点:",arg);
debugger;
return _parse(arg);
};
break;

case '2':
//alert("每当尝试设置document.cookie时断点");
Object.defineProperty(document, "cookie", {
setfunction (a) {
debugger;
}
});
break;
case '3':
//alert("每当调用XMLHttpRequest.prototype.open时断点");
var _open = XMLHttpRequest.prototype.open;
XMLHttpRequest.prototype.open = function (method, url, async, user, password) {
console.log("[+]「油猴」调用XMLHttpRequest.prototype.open断点:", method, url);
debugger;
return _open.apply(this, arguments);
};
break;
case '4':
//alert("每当调用XMLHttpRequest.prototype.send时断点");
var _send = XMLHttpRequest.prototype.send;

XMLHttpRequest.prototype.send = function (data) {
console.log("[+]「油猴」POST请求发送数据:", data);
debugger;
_send.apply(this, arguments);
};
break;

default:
alert("未知插件:" + chosenPlugin);
break;
}
});
else {
//alert("没有选择插件");
console.log("[x]没有选择插件")
}
})();

NoDebugger.js

代码中禁用无限Debugger

// ==UserScript==
// @name 代码中禁用无限Debugger
// @namespace http://tampermonkey.net/
// @version 0.1
// @description 禁用无限Debugger
// @author Cha111Ng1
// @match http*://*/*
// @icon 
// @grant none
// @run-at document-start
// ==/UserScript==

(function () {
// 破解无限Debugger
var constructorHook = constructor;
Function.prototype.constructor = function(s) {
if (s == "debugger") {
return function() {}
}
return constructorHook(s);
}
const setInterval = window.setInterval;
window.setInterval = function(fun, time) {
// console.log(time, 'ddddd', fun.toString());
if (fun && fun.toString) {
var funString = fun.toString();
if (funString.indexOf('debugger') > -1) return;
if (funString.indexOf('window.close') > -1) return;
}

return setInterval(fun, time);
}
})()

jsrpc.js

加解密jsrpc自动化脚本,配合sekiro、Mitmproxy(可选)使用

// ==UserScript==
// @name jsrpc模版
// @namespace http://tampermonkey.net/
// @version 0.1
// @description 加解密jsrpc自动化脚本
// @author Cha111Ng1
// @match https://此处修改为需要使用的域名/*
// @icon 
// @grant none
// @run-at document-start
// ==/UserScript==



// 需配合Sekiro、Mitmproxy(自行百度),需要找到加密位置,并把加密函数设置为全局函数,本例子为window._h(data),在源码加密函数下方插入if (!window._h){window._h = h}
// 例如:
// 加密
// http://127.0.0.1:5612/business-demo/invoke?group=test_cha11&action=encrypt&data=
// 解密
// http://127.0.0.1:5612/business-demo/invoke?group=test_cha11&action=decode&data=



(function() {
// 取消debug
(function(){}).constructor === Function;
Function.prototype.constructor = function(){};

'use strict';
console.log('# ++++++++++++++++++++++++++++++++++++++++++')
console.log('# + 微信公众号:星火sec 作者zero +')
console.log('# ++++++++++++++++++++++++++++++++++++++++++')
// Your code here...
function SekiroClient(e){if(this.wsURL=e,this.handlers={},this.socket={},!e)throw new Error("wsURL can not be empty!!");this.webSocketFactory=this.resolveWebSocketFactory(),this.connect()}SekiroClient.prototype.resolveWebSocketFactory=function(){if("object"==typeof window){var e=window.WebSocket?window.WebSocket:window.MozWebSocket;return function(o){function t(o){this.mSocket=new e(o)}return t.prototype.close=function(){this.mSocket.close()},t.prototype.onmessage=function(e){this.mSocket.onmessage=e},t.prototype.onopen=function(e){this.mSocket.onopen=e},t.prototype.onclose=function(e){this.mSocket.onclose=e},t.prototype.send=function(e){this.mSocket.send(e)},new t(o)}}if("object"==typeof weex)try{console.log("test webSocket for weex");var o=weex.requireModule("webSocket");return console.log("find webSocket for weex:"+o),function(e){try{o.close()}catch(e){}return o.WebSocket(e,""),o}}catch(e){console.log(e)}if("object"==typeof WebSocket)return function(o){return new e(o)};throw new Error("the js environment do not support websocket")},SekiroClient.prototype.connect=function(){console.log("sekiro: begin of connect to wsURL: "+this.wsURL);var e=this;try{this.socket=this.webSocketFactory(this.wsURL)}catch(o){return console.log("sekiro: create connection failed,reconnect after 2s:"+o),void setTimeout(function(){e.connect()},2e3)}this.socket.onmessage(function(o){e.handleSekiroRequest(o.data)}),this.socket.onopen(function(e){console.log("sekiro: open a sekiro client connection")}),this.socket.onclose(function(o){console.log("sekiro: disconnected ,reconnection after 2s"),setTimeout(function(){e.connect()},2e3)})},SekiroClient.prototype.handleSekiroRequest=function(e){console.log("receive sekiro request: "+e);var o=JSON.parse(e),t=o.__sekiro_seq__;if(o.action){var n=o.action;if(this.handlers[n]){var s=this.handlers[n],i=this;try{s(o,function(e){try{i.sendSuccess(t,e)}catch(e){i.sendFailed(t,"e:"+e)}},function(e){i.sendFailed(t,e)})}catch(e){console.log("error: "+e),i.sendFailed(t,":"+e)}}else this.sendFailed(t,"no action handler: "+n+" defined")}else this.sendFailed(t,"need request param {action}")},SekiroClient.prototype.sendSuccess=function(e,o){var t;if("string"==typeof o)try{t=JSON.parse(o)}catch(e){(t={}).data=o}else"object"==typeof o?t=o:(t={}).data=o;(Array.isArray(t)||"string"==typeof t)&&(t={data:t,code:0}),t.code?t.code=0:(t.status,t.status=0),t.__sekiro_seq__=e;var n=JSON.stringify(t);console.log("response :"+n),this.socket.send(n)},SekiroClient.prototype.sendFailed=function(e,o){"string"!=typeof o&&(o=JSON.stringify(o));var t={};t.message=o,t.status=-1,t.__sekiro_seq__=e;var n=JSON.stringify(t);console.log("sekiro: response :"+n),this.socket.send(n)},SekiroClient.prototype.registerAction=function(e,o){if("string"!=typeof e)throw new Error("an action must be string");if("function"!=typeof o)throw new Error("a handler must be function");return console.log("sekiro: register action: "+e),this.handlers[e]=o,this};
var client = new SekiroClient("ws://127.0.0.1:5612/business-demo/register?group=test_cha11&clientId=" + Math.random());

// 加密接口1
client.registerAction("encrypt",function(request, resolve, reject){
var data = request['data']
console.log('[+]「油猴加密」明文:',data)
var cha = window._v(data)
console.log('[+]「油猴加密」密文:',cha)
resolve(cha);
});

// 解密接口1
client.registerAction("decode",function(request, resolve, reject){
var data = request['data']
console.log('[+]「油猴解密」密文:',data)
var chaa = window._h(data)
console.log('[+]「油猴解密」明文:',chaa)
resolve(chaa);
});
})();

JSFinder

https://github.com/kacakb/jsfinder

二开JSFinder

https://github.com/MiNi39/JSSCAN

7eTeamTools v1.0

https://github.com/0x7eTeam/0x7eTeamTools

如果大家不喜欢在网页上调试这里也有个工具可以调试

发 条js_JS-ED_1.9

链接:https://pan.baidu.com/s/19V0N-RC1hGl5zZalkEb9MA?pwd=32em 提取码:32em –来自百度网盘超级会员V5的分享

FaCai_V1.0.0_jdk8_zangcc

https://github.com/zangcc

开发软件的作者也写了一篇文章讲软件了https://blog.csdn.net/weixin_43847838/article/details/125841581?spm=1001.2014.3001.5501

实战

HookScript.js无痛寻找加密位置

我们去输入一个admin,123456

看看他加密函数,和加密逻辑在哪里

我们可以看到,直接定位到userInfo,然后加密出来像是一串base64字符串

然后我们全局搜索一下userInfo,验证一下

验证一下,我们下个断点

然后明白了,就是admin,123456,验证码,三个变量的值,一起加密base64.就是前端加密的思路,那我们再进行前端弱口令爆破,就变的容易,小白超级容易上手

jsrpc调用jsrpc.js

启动sekiro

项目地址:https://sekiro.iinti.cn/sekiro-doc

这是没有打包的https://github.com/yint-tech/sekiro-samples/releases/tag/v3.20240124

打包好了的

https://github.com/yint-tech/sekiro-open/releases/tag/v3.20240311

# win 运行
sekiro.bat
# linux 运行
bin/sekiro.sh
# mac 运行
bin/sekiro

浏览器加载脚本,为方便,所以写成了油猴脚本jsrpc.js,加载油猴脚本

jsrpc.js

// ==UserScript==
// @name jsrpc模版
// @namespace http://tampermonkey.net/
// @version 0.1
// @description 加解密jsrpc自动化脚本
// @author Cha111Ng1
// @match https://此处修改为需要使用的域名/*
// @icon 
// @grant none
// @run-at document-start
// ==/UserScript==



// 需配合Sekiro、Mitmproxy(自行百度),需要找到加密位置,并把加密函数设置为全局函数,本例子为window._h(data),在源码加密函数下方插入if (!window._h){window._h = h}
// 例如:
// 加密
// http://127.0.0.1:5612/business-demo/invoke?group=test_cha11&action=encrypt&data=
// 解密
// http://127.0.0.1:5612/business-demo/invoke?group=test_cha11&action=decode&data=



(function() {
// 取消debug
(function(){}).constructor === Function;
Function.prototype.constructor = function(){};

'use strict';
console.log('# ++++++++++++++++++++++++++++++++++++++++++')
console.log('# + 微信公众号:攻有道 By:Cha111Ng1 +')
console.log('# ++++++++++++++++++++++++++++++++++++++++++')
// Your code here...
function SekiroClient(e){if(this.wsURL=e,this.handlers={},this.socket={},!e)throw new Error("wsURL can not be empty!!");this.webSocketFactory=this.resolveWebSocketFactory(),this.connect()}SekiroClient.prototype.resolveWebSocketFactory=function(){if("object"==typeof window){var e=window.WebSocket?window.WebSocket:window.MozWebSocket;return function(o){function t(o){this.mSocket=new e(o)}return t.prototype.close=function(){this.mSocket.close()},t.prototype.onmessage=function(e){this.mSocket.onmessage=e},t.prototype.onopen=function(e){this.mSocket.onopen=e},t.prototype.onclose=function(e){this.mSocket.onclose=e},t.prototype.send=function(e){this.mSocket.send(e)},new t(o)}}if("object"==typeof weex)try{console.log("test webSocket for weex");var o=weex.requireModule("webSocket");return console.log("find webSocket for weex:"+o),function(e){try{o.close()}catch(e){}return o.WebSocket(e,""),o}}catch(e){console.log(e)}if("object"==typeof WebSocket)return function(o){return new e(o)};throw new Error("the js environment do not support websocket")},SekiroClient.prototype.connect=function(){console.log("sekiro: begin of connect to wsURL: "+this.wsURL);var e=this;try{this.socket=this.webSocketFactory(this.wsURL)}catch(o){return console.log("sekiro: create connection failed,reconnect after 2s:"+o),void setTimeout(function(){e.connect()},2e3)}this.socket.onmessage(function(o){e.handleSekiroRequest(o.data)}),this.socket.onopen(function(e){console.log("sekiro: open a sekiro client connection")}),this.socket.onclose(function(o){console.log("sekiro: disconnected ,reconnection after 2s"),setTimeout(function(){e.connect()},2e3)})},SekiroClient.prototype.handleSekiroRequest=function(e){console.log("receive sekiro request: "+e);var o=JSON.parse(e),t=o.__sekiro_seq__;if(o.action){var n=o.action;if(this.handlers[n]){var s=this.handlers[n],i=this;try{s(o,function(e){try{i.sendSuccess(t,e)}catch(e){i.sendFailed(t,"e:"+e)}},function(e){i.sendFailed(t,e)})}catch(e){console.log("error: "+e),i.sendFailed(t,":"+e)}}else this.sendFailed(t,"no action handler: "+n+" defined")}else this.sendFailed(t,"need request param {action}")},SekiroClient.prototype.sendSuccess=function(e,o){var t;if("string"==typeof o)try{t=JSON.parse(o)}catch(e){(t={}).data=o}else"object"==typeof o?t=o:(t={}).data=o;(Array.isArray(t)||"string"==typeof t)&&(t={data:t,code:0}),t.code?t.code=0:(t.status,t.status=0),t.__sekiro_seq__=e;var n=JSON.stringify(t);console.log("response :"+n),this.socket.send(n)},SekiroClient.prototype.sendFailed=function(e,o){"string"!=typeof o&&(o=JSON.stringify(o));var t={};t.message=o,t.status=-1,t.__sekiro_seq__=e;var n=JSON.stringify(t);console.log("sekiro: response :"+n),this.socket.send(n)},SekiroClient.prototype.registerAction=function(e,o){if("string"!=typeof e)throw new Error("an action must be string");if("function"!=typeof o)throw new Error("a handler must be function");return console.log("sekiro: register action: "+e),this.handlers[e]=o,this};
var client = new SekiroClient("ws://127.0.0.1:5612/business-demo/register?group=test_cha11&clientId=" + Math.random());

// 加密接口1
client.registerAction("encrypt",function(request, resolve, reject){
var data = request['data']
console.log('[+]「油猴加密」明文:',data)
var cha = window._v(data)
console.log('[+]「油猴加密」密文:',cha)
resolve(cha);
});

// 解密接口1
client.registerAction("decode",function(request, resolve, reject){
var data = request['data']
console.log('[+]「油猴解密」密文:',data)
var chaa = window._h(data)
console.log('[+]「油猴解密」明文:',chaa)
resolve(chaa);
});
})();

配置完成后启用,

POST /business-demo/invoke?group=test_cha11&action=login HTTP/1.1
Content-Type: application/json
Host: 127.0.0.1:5612

{"userInfo""admin,123456,8946"

这里我的验证码需要每次批量获取,所以,用这个工具生成加密不太实用,但是如果你遇到没有验证码,然后他的加密是md5加盐这种,可能小白对js逆向学习比较浅,即使看到加密代码,还是不太清楚,可以尝试是使用这个工具

首发于奇安信攻防社区:https://forum.butian.net/share/3647

加下方wx,拉你一起进群学习

© 版权声明
THE END
喜欢就支持一下吧, 不喜欢也支持一下吧
点赞11 分享
评论 抢沙发
头像
欢迎您留下宝贵的见解!
提交
头像

昵称

取消
昵称表情代码图片

    暂无评论内容