boss直聘zp_stoken逆向分析源码放送

1,923 阅读5分钟

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%成功 在这里插入图片描述 剩余步骤请关注公众号获取 在这里插入图片描述