JS中注入eval, Function等系統(tǒng)函數(shù)截獲動態(tài)代碼
現(xiàn)在很多網(wǎng)站都上了各種前端反爬手段,無論手段如何,最重要的是要把包含反爬手段的前端javascript代碼加密隱藏起來,然后在運(yùn)行時實時解密動態(tài)執(zhí)行。
動態(tài)執(zhí)行js代碼無非兩種方法,即eval和Function。那么,不管網(wǎng)站加密代碼寫的多牛,我們只要將這兩個方法hook住,即可獲取到解密后的可執(zhí)行js代碼。
注意,有些網(wǎng)站會檢測eval和Function這兩個方法是否原生,因此需要一些小花招來忽悠過去。
掛鉤代碼
首先是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操作都會在控制臺打印輸出將要執(zhí)行的js源碼。
同理可以寫出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是個變長參數(shù)的構(gòu)造方法,需要處理this
另外,有些網(wǎng)站還會用類似的機(jī)制加密頁面內(nèi)容,然后通過document.write輸出動態(tài)解密的內(nèi)容,因此同樣可以掛鉤document.write,掛鉤方法類似eval,這里就不重復(fù)了。
注入方式
另外,還有個問題需要關(guān)注,就是掛鉤代碼的注入方法。
最簡單的就是F12調(diào)出控制臺,直接執(zhí)行上面的代碼,但這樣只能hook住之后的調(diào)用,如果希望從頁面剛加載時就注入,那么可以用以下幾種方式:
- 油猴注入,油猴可以監(jiān)聽文檔加載的幾種不同狀態(tài),并在特定時刻執(zhí)行js代碼。我沒有太多研究,具體請參見油猴手冊
- 代理注入,修改應(yīng)答數(shù)據(jù),在<head>標(biāo)簽內(nèi)的第一個位置插入<script>節(jié)點,確保在其它js加載執(zhí)行前注入;Fiddler, anyproxy等都可以編寫外部規(guī)則,具體請參見代理工具的手冊
- 使用chrome-devtools-protocol, 通過Page.addScriptToEvaluateOnNewDocument注入外部js代碼
Fiddler代理規(guī)則
不少人沒用過代理規(guī)則,這里寫一下Fiddler的規(guī)則編寫方法:
Fiddler菜單里Rules > Customize Rules 打開腳本編輯器
在腳本編輯器里找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
}
這樣就會在所有html文檔頭部自動添加js代碼了
總結(jié)
以上所述是小編給大家介紹的JS中注入eval, Function等系統(tǒng)函數(shù)截獲動態(tài)代碼,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
相關(guān)文章
javascript實例--教你實現(xiàn)撲克牌洗牌功能
我們一般都會按照順序把隨機(jī)摸過來的牌從小到大的順序在手上理整齊(記得小時候打牌兩副牌手都抓不過來),這篇隨筆就是想通過實現(xiàn)這個功能來熟悉下js中排序數(shù)組等相關(guān)知識。2014-05-05
JS+DIV實現(xiàn)鼠標(biāo)劃過切換層效果的實例代碼
這篇文章主要是對JS+DIV實現(xiàn)鼠標(biāo)劃過切換層效果的實例代碼進(jìn)行了詳細(xì)的介紹,需要的朋友可以過來參考下,希望對大家有所幫助2013-11-11
BootStrap中Datetimepicker和uploadify插件應(yīng)用實例小結(jié)
這篇文章主要介紹了BootStrap中Datetimepicker和uploadify插件應(yīng)用實例小結(jié)的相關(guān)資料,非常不錯具有參考借鑒價值,需要的朋友可以參考下2016-05-05
如何用uni-app實現(xiàn)頂部導(dǎo)航欄顯示按鈕和搜索框
本文介紹了如何用uni-app實現(xiàn)頂部導(dǎo)航欄顯示按鈕和搜索框,感興趣的同學(xué),可以參考下,并且試驗一下。2021-06-06
js實現(xiàn)增加數(shù)字顯示的環(huán)形進(jìn)度條效果
本文主要分享了js實現(xiàn)增加數(shù)字顯示的環(huán)形進(jìn)度條效果的示例代碼。具有一定的參考價值,下面跟著小編一起來看下吧2017-02-02
微信小程序?qū)崿F(xiàn)多行文字超出部分省略號顯示功能
這篇文章主要介紹了微信小程序?qū)崿F(xiàn)多行文字 超出部分省略號顯示功能,比如設(shè)置只顯示2行,超出部分省略號顯示,本文通過實例代碼給大家介紹,需要的朋友可以參考下2019-10-10
解決微信內(nèi)置瀏覽器返回上一頁強(qiáng)制刷新問題方法
微信內(nèi)置瀏覽器在返回上一頁面,且上一頁面包含AJAX代碼時,頁面就會被強(qiáng)制刷新,極度影響用戶體驗。而我們想要的效果是:返回上一頁面時,頁面還停留在原來的狀態(tài),AJAX獲取到的數(shù)據(jù)還在,滾動條也在原來的位置。下面跟著小編一起來看下吧2017-02-02
解決JS浮點數(shù)運(yùn)算出現(xiàn)Bug的方法
解決JS浮點數(shù)運(yùn)算出現(xiàn)Bug的方法,需要的朋友可以參考一下2013-03-03

