1.前言
本分析过程仅供学习交流使用,切勿用于非法行为 因为前一篇文章感觉写的偏水,今天特定将整个流程重新梳理一遍,顺便开放一下源码 需要源码的关注公众号后加我微信
2.分析过程
2-1.寻找函数入口
code = (new ABC).z(seed, parseInt(ts) + (480 + (new Date).getTimezoneOffset()) * 60 * 1e3)
//这个就是函数入口
//(480 + (new Date).getTimezoneOffset()) * 60 * 1e3这一段等于0
//所以实际的函数入口就是
code = (new ABC).z(seed, parseInt(ts))
2-2.插桩暴露环境差异
插桩后在浏览器和本地运行即可看出差异
- 插桩后浏览器端的运行效果
- 插桩后本地编辑器运行的结果
有对nodejs的检测__filename,Buffer等等
在浏览器里面这个是undefined,而在nodejs里面这是个特定的模块
2-3.Js Proxy自吐环境
Js Proxy检测获取属性对象,走完这一步已经补完了60%左右的环境
window = new Proxy(window, {
get: function (x, y) {
console.log(y)
return x[y]
},
set(target, p, value, receiver) {
target[p] = value
}
})
2-4.正则匹配可疑属性
Js proxy无法完全将window的属性全部获取,利用这个正则查看可疑代码,对相关对象做调试
.*\[.*\].*\(.*\)
2-5.hook 内置函数
因为zp_stoken是动态变化的,造成变化就是这个Math和Date对象 走到这里你已经成功了95%。
old_random=Math.random
Math.random=function(){
console.log()
debugger
return old_random();
}
2-6.测试效果
通过这些操作后,以下就是补的环境,但是仍然不能实现百分之百成功,还需要操作一个细节
我们先看一下效果,会出现无法100%成功的尴尬局面,剩余补全部分请关注微信公众号获取
window = {
document: {
cookie: "",
createElement: function (tag) {
if (tag == "canvas") {
return canvas
} else if (tag == "caption") {
return {
tagName: "CAPTION"
}
}
},
getElementById: function (a, b, c, d) {
return false
},
getElementsByName: function (a, b, c, d) {
return false
},
getElementsByTagName: function (a, b, c, d) {
return false
},
getElementsByClassName: function (a, b, c, d) {
return false
},
title: ""
},
moveBy: function () {
},
moveTo: function () {
},
open: function () {
},
dispatchEvent: function () {
return true
},
screen: {
availHeight: 824,
availWidth: 1536
},
location: {
host: "www.zhipin.com",
hostname: "www.zhipin.com",
href: "https://www.zhipin.com/c101280100-p100101/",
origin: "https://www.zhipin.com",
pathname: "/c101280100-p100101/",
toString: function () {
return "https://www.zhipin.com/"
}
},
decodeURI: decodeURI,
OfflineAudioContext: function () {
this.createOscillator = function () {
return {
frequency: {
setValueAtTime: function () {
}
},
connect: function () {
},
start: function () {
},
}
},
this.createDynamicsCompressor = function () {
return {
threshold: {
setValueAtTime: function () {
},
},
setValueAtTime: function () {
},
knee: {
setValueAtTime: function () {
},
},
ratio: {
setValueAtTime: function () {
},
},
reduction: {
setValueAtTime: function () {
},
},
attack: {
setValueAtTime: function () {
},
},
release: {
setValueAtTime: function () {
},
},
connect: function () {
},
}
},
this.startRendering = function () {
}
},
history: {
"length": 2,
"scrollRestoration": "auto",
"state": null
},
// outerHeight: 824,
// innerHeight: 150,
// outerWidth: 1536,
// innerWidth: 0,
outerHeight: 28,
innerHeight: 0,
outerWidth: 160,
innerWidth: 0,
Math: Math,
Date: Date,
encodeURIComponent: encodeURIComponent,
RegExp: RegExp,
length: 0,
ScreenOrientation: function () {
},
onmessage: null,
performance: {},
speechSynthesis: {
paused: false,
pending: false,
speaking: false,
getVoices: function () {
},
speak: function () {
}
},
SourceBuffer: function () {
return {
mode: "",
appendWindowStart: "",
audioTracks: "",
buffered: "",
textTracks: "",
toString: function () {
return "function SourceBuffer() { [native code] }"
}
}
},
XMLHttpRequest: function () {
return {
readyState: "",
response: "",
responseText: "",
responseURL: "",
responseXML: "",
statusText: "",
toString: function () {
return "function XMLHttpRequest() { [native code] }"
}
}
},
SpeechSynthesisUtterance: function () {
},
toString: function () {
return "[object Window]"
},
}
scrollBy = function () {
}
scrollBy.toString=function(){
return "function scrollBy() { [native code] }"
}
window.scrollBy=scrollBy
scrollTo = function () {
}
scrollTo.toString=function(){
return "function scrollTo() { [native code] }"
}
window.scrollTo=scrollTo
window.open.toString = function () {
return "function open() { [native code] }"
}
OfflineAudioContext = window.OfflineAudioContext
canvas = {
getContext: function () {
return CanvasRenderingContext2D
},
toDataURL: function () {
// 实际为canvas画布填充了图片
return ""
},
}
CanvasRenderingContext2D = {
fillRect: function () {
},
fillText: function () {
}
}
localStorage = {
removeItem: function (key) {
delete this[key]
},
getItem: function (key) {
return this[key] ? this[key] : null;
},
setItem: function (key, value) {
this[key] = "" + value;
},
};
sessionStorage = {}
document = window.document
window.navigator = navigator = {
cookieEnabled: true,
language: "zh-CN",
appCodeName: "Mozilla",
appName: "Netscape",
userAgent: "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/83.0.4103.61 Safari/537.36",
appVersion: "5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/83.0.4103.61 Safari/537.36"
}
window.sessionStorage = sessionStorage
window.localStorage = localStorage
setInterval = window.setInterval = function () {
debugger;
}
setInterval.toString = function () {
return "function setInterval() { [native code] }"
}
setTimeout = window.setTimeout = function () {
}
setTimeout.toString = function () {
return "function setTimeout() { [native code] }"
}
clearTimeout = window.clearTimeout = function () {
}
clearTimeout.toString = function () {
return "function clearTimeout() { [native code] }"
}
top = window.top = window
self = window.self = window
window.window = window
child_process = undefined;
closed = {
__proto__: (1 >> 3 > 4)["__proto__"]
}
screenTop = {
__proto__: (2)["__proto__"]
}
Function.prototype.toString = function () {
return "function () { [native code] }";
}
Object.keys(window).forEach(property => {
try {
if (typeof global[property] === 'undefined') {
global[property] = window[property];
}
} catch (e) {
// console.log(e);
}
});
global.window = window;
global = undefined;
global = window
process.argv = undefined;
最终效果
100%成功
剩余步骤请关注公众号获取