JS中注入eval, Function等系統(tǒng)函數(shù)截獲動(dòng)態(tài)代碼
現(xiàn)在很多網(wǎng)站都上了各種前端反爬手段,無(wú)論手段如何,最重要的是要把包含反爬手段的前端javascript代碼加密隱藏起來(lái),然后在運(yùn)行時(shí)實(shí)時(shí)解密動(dòng)態(tài)執(zhí)行。
動(dòng)態(tài)執(zhí)行js代碼無(wú)非兩種方法,即eval和Function。那么,不管網(wǎng)站加密代碼寫(xiě)的多牛,我們只要將這兩個(gè)方法hook住,即可獲取到解密后的可執(zhí)行js代碼。
注意,有些網(wǎng)站會(huì)檢測(cè)eval和Function這兩個(gè)方法是否原生,因此需要一些小花招來(lái)忽悠過(guò)去。
掛鉤代碼
首先是eval的掛鉤代碼:
(function() {
if (window.__cr_eval) return
window.__cr_eval = window.eval
var myeval = function (src) {
console.log("================ eval begin: length=" + src.length + ",caller=" + (myeval.caller && myeval.caller.name) + " ===============")
console.log(src);
console.log("================ eval end ================")
return window.__cr_eval(src)
}
var _myeval = myeval.bind(null)
_myeval.toString = window.__cr_eval.toString
Object.defineProperty(window, 'eval', { value: _myeval })
console.log(">>>>>>>>>>>>>> eval injected: " + document.location + " <<<<<<<<<<<<<<<<<<<")
})();
這段代碼執(zhí)行后,之后所有的eval操作都會(huì)在控制臺(tái)打印輸出將要執(zhí)行的js源碼。
同理可以寫(xiě)出Function的掛鉤代碼:
(function() {
if (window.__cr_fun) return
window.__cr_fun = window.Function
var myfun = function () {
var args = Array.prototype.slice.call(arguments, 0, -1).join(","), src = arguments[arguments.length - 1]
console.log("================ Function begin: args=" + args + ", length=" + src.length + ",caller=" + (myfun.caller && myfun.caller.name) + " ===============")
console.log(src);
console.log("================ Function end ================")
return window.__cr_fun.apply(this, arguments)
}
myfun.toString = function() { return window.__cr_fun + "" }
Object.defineProperty(window, 'Function', { value: myfun })
console.log(">>>>>>>>>>>>>> Function injected: " + document.location + " <<<<<<<<<<<<<<<<<<<")
})();
注意和eval不同,F(xiàn)unction是個(gè)變長(zhǎng)參數(shù)的構(gòu)造方法,需要處理this
另外,有些網(wǎng)站還會(huì)用類(lèi)似的機(jī)制加密頁(yè)面內(nèi)容,然后通過(guò)document.write輸出動(dòng)態(tài)解密的內(nèi)容,因此同樣可以掛鉤document.write,掛鉤方法類(lèi)似eval,這里就不重復(fù)了。
注入方式
另外,還有個(gè)問(wèn)題需要關(guān)注,就是掛鉤代碼的注入方法。
最簡(jiǎn)單的就是F12調(diào)出控制臺(tái),直接執(zhí)行上面的代碼,但這樣只能hook住之后的調(diào)用,如果希望從頁(yè)面剛加載時(shí)就注入,那么可以用以下幾種方式:
- 油猴注入,油猴可以監(jiān)聽(tīng)文檔加載的幾種不同狀態(tài),并在特定時(shí)刻執(zhí)行js代碼。我沒(méi)有太多研究,具體請(qǐng)參見(jiàn)油猴手冊(cè)
- 代理注入,修改應(yīng)答數(shù)據(jù),在<head>標(biāo)簽內(nèi)的第一個(gè)位置插入<script>節(jié)點(diǎn),確保在其它js加載執(zhí)行前注入;Fiddler, anyproxy等都可以編寫(xiě)外部規(guī)則,具體請(qǐng)參見(jiàn)代理工具的手冊(cè)
- 使用chrome-devtools-protocol, 通過(guò)Page.addScriptToEvaluateOnNewDocument注入外部js代碼
Fiddler代理規(guī)則
不少人沒(méi)用過(guò)代理規(guī)則,這里寫(xiě)一下Fiddler的規(guī)則編寫(xiě)方法:
Fiddler菜單里Rules > Customize Rules 打開(kāi)腳本編輯器
在腳本編輯器里找OnBeforeResponse方法,方法內(nèi)添加下面代碼:
if (oSession.oResponse.headers.ExistsAndContains("Content-Type", "html")){
oSession.utilDecodeResponse(); // Remove any compression or chunking
var b = System.Text.Encoding.UTF8.GetString(oSession.responseBodyBytes);
var r = /<head[^>]*>/i;
var js = "..."; // 要注入的js源碼
b = b.replace(r, "$0<script>" + js + "</script>");
oSession.utilSetResponseBody(b); // Set the response body back
}
這樣就會(huì)在所有html文檔頭部自動(dòng)添加js代碼了
總結(jié)
以上所述是小編給大家介紹的JS中注入eval, Function等系統(tǒng)函數(shù)截獲動(dòng)態(tài)代碼,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
基于js?+?html2canvas實(shí)現(xiàn)網(wǎng)頁(yè)放大鏡功能
最近接到任務(wù),需實(shí)現(xiàn)【網(wǎng)頁(yè)】放大鏡的效果,百度搜索?【js?放大鏡】關(guān)鍵字,千篇一律的都是一些仿淘寶/京東等電商網(wǎng)站中查看規(guī)格大圖的效果實(shí)現(xiàn),根本無(wú)法滿足我的需求,于是自己花了點(diǎn)時(shí)間調(diào)研實(shí)現(xiàn),在這里分享給大家,感興趣的朋友可以參考下2023-12-12
javascript關(guān)于“時(shí)間”的一次探索
這篇文章主要介紹了javascript關(guān)于“時(shí)間”的一次探索,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-07-07
JavaScript取得鼠標(biāo)絕對(duì)位置程序代碼介紹
文章介紹了關(guān)于JavaScript在不同瀏覽器下取得鼠標(biāo)絕對(duì)位置相關(guān)函數(shù)用法及兼容性介紹,有需要的同學(xué)可參考一下2012-09-09
JS動(dòng)態(tài)圖片的實(shí)現(xiàn)方法完整示例
這篇文章主要介紹了JS動(dòng)態(tài)圖片的實(shí)現(xiàn)方法,結(jié)合完整實(shí)例形式分析了JavaScript鼠標(biāo)響應(yīng)與頁(yè)面元素屬性動(dòng)態(tài)變換相關(guān)操作技巧,需要的朋友可以參考下2020-01-01
JavaScript實(shí)現(xiàn)添加及刪除事件的方法小結(jié)
這篇文章主要介紹了JavaScript實(shí)現(xiàn)添加及刪除事件的方法,實(shí)例總結(jié)了javascript對(duì)事件的添加及刪除的技巧,涉及javascript事件綁定的方法及瀏覽器兼容的相關(guān)注意事項(xiàng),需要的朋友可以參考下2015-08-08
帶你使用webpack快速構(gòu)建web項(xiàng)目的方法
這篇文章主要介紹了帶你使用webpack快速構(gòu)建web項(xiàng)目的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11
JavaScript中的定時(shí)器之Item23的合理使用
window對(duì)象提供了兩個(gè)方法來(lái)實(shí)現(xiàn)定時(shí)器的效果,分別是window.setTimeout()和window.setInterval。其中前者可以使一段代碼在指定時(shí)間后運(yùn)行;而后者則可以使一段代碼每過(guò)指定時(shí)間就運(yùn)行一次2015-10-10

