网站运行的时间轴
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", {
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
// ==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逆向学习比较浅,即使看到加密代码,还是不太清楚,可以尝试是使用这个工具
加下方wx,拉你一起进群学习
- 1. 本站名称: 橘子缤纷乐园
- 2. 本站网址: https://blog.zuziy.cn
- 3. 本网站的文章部分内容可能来源于网络,仅供大家学习与参考,如有侵权,请联系站长进行删除处理。
- 4. 本站一切资源不代表本站立场,并不代表本站赞同其观点和对其真实性负责。
- 5. 本站一律禁止以任何方式发布或转载任何违法的相关信息,访客发现请向站长举报。
- 6. 本站附件资源、教程等内容如因时效原因失效或不可用,请评论区留言或联系站长及时更新。
暂无评论内容