前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >JS逆向,前端加密暴力破解(小白无痛学习)

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

作者头像
红队蓝军
发布2024-08-20 20:47:08
2140
发布2024-08-20 20:47:08
举报
文章被收录于专栏:红队蓝军

网站运行的时间轴

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

代码语言:javascript
复制
// ==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", {
set: function (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

代码语言:javascript
复制
// ==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(可选)使用

代码语言:javascript
复制
// ==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

代码语言:javascript
复制
# win 运行
sekiro.bat
# linux 运行
bin/sekiro.sh
# mac 运行
bin/sekiro

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

jsrpc.js
代码语言:javascript
复制
// ==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);
});
})();

配置完成后启用,

代码语言:javascript
复制
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逆向学习比较浅,即使看到加密代码,还是不太清楚,可以尝试是使用这个工具

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2024-08-19,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 红队蓝军 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 网站运行的时间轴
  • 浏览器的调试功能
  • 控制台
  • 网络(Network)
  • 那么在js文件中我们如何快速定位到加密函数
  • 利用加解密函数
  • 工具推荐
    • phantomjs
      • HookScript.js
        • NoDebugger.js
          • jsrpc.js
            • JSFinder
              • 二开JSFinder
                • 7eTeamTools v1.0
                  • 发 条js_JS-ED_1.9
                    • FaCai_V1.0.0_jdk8_zangcc
                    • 实战
                      • HookScript.js无痛寻找加密位置
                        • jsrpc调用jsrpc.js
                          • 启动sekiro
                          • jsrpc.js
                      相关产品与服务
                      验证码
                      腾讯云新一代行为验证码(Captcha),基于十道安全栅栏, 为网页、App、小程序开发者打造立体、全面的人机验证。最大程度保护注册登录、活动秒杀、点赞发帖、数据保护等各大场景下业务安全的同时,提供更精细化的用户体验。
                      领券
                      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档