欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

從零開始使用gradle配置即可執(zhí)行的Hook庫詳解

 更新時間:2022年09月21日 12:07:42   作者:Pika  
這篇文章主要為大家介紹了從零開始使用gradle配置即可執(zhí)行的Hook庫詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

背景

有一天,老板突然找到小B說,隱私合規(guī)需要我們獲取權限前,需要明確授權來意,這個你來跟一下吧!小B此時就可愁了,因為項目權限那么多,每個自己手動加上授權來意提示的話,可能會漏掉很多,工作量也大,這可咋辦呀!老B看到小B這么愁眉苦臉,連忙說:“可以用ASM進行插樁呀!hook想要的方法”,小B聽了,興奮的去百度了一下,但是發(fā)現asm學習成本又高,短期又不可能搞完,這可咋辦呀!明明我只想搞hook一個方法交差來著!!老B:”沒事,所以本文就來了!”

本文須知

這里只是提供一個設計思路,不會涉及到太多細節(jié),需要讀者了解相關的知識,如果不清楚只想使用的話,也是有的 github.com/TestPlanB/S… 歡迎點星星或者pr噢!

當前技術背景

目前可以利用字節(jié)碼進行hook的框架有很多,比如ASM,AspectJ,javassit等等,都是可以在編譯時插入相關的字節(jié)碼,進行方法的插樁,從而達到一個hook的目的,但是這些工具好歸好,但是都有一個小問題,就是需要上手,部分hook框架上手門檻高,也有自己獨特的用法,短時間內可能很難使得開發(fā)人員上手。所以對hook庫進行一個二次封裝,也是很多公司在做的一個事情。方法有很多種,作者基于自己的理解,認為配置式的hook才是最簡單的,畢竟,Android就有gradle進行各種的項目工程配置,那么我們?yōu)槭裁淳筒荒芡ㄟ^gradle進行配置的Hook呢?基于上面的猜想,就有了本文!友情提示:閱讀本文最好對asm跟transform機制有所了解

底層選擇

為了更加通用和高效,本次采用asm作為底層,進行二次封裝,畢竟android官方的link還有比較出色的aspectj都是基于asm進行底層修改的,那我們這次也同樣使用,好了就開干!

目標流程圖

Transform

為了讓不太了解的ASM的也能夠閱讀本文,所以也會介紹部分ASM相關的信息,詳細了解還需要大家去官網閱讀噢!這里先介紹Transform機制。 Transform是android 進行編譯時,在class 文件生成 dex文件時,給我們開發(fā)者預留的一個小口,可以理解在這個階段,我們可以修改已生成的class等文件,編織入自己額外的字節(jié)碼,從而達到無需修改項目本身的源代碼就可以行為修改的機制!如果大家有留意的話,這個機制就是gradle 在build階段中,會存在一個transformClassesWithXXForXX的task,舉例子:

transformClassesWithSpiderPluginForDebug,就是在這里進行的transform修改。 當然,一個項目會存在多個transform,如圖所示

就像流水線一樣,我們的transform處理完就會交給下一個transform,共同修改生成的字節(jié)碼的行為。大家可以先簡單理解為這是一個任務,提供了接口給外部修改生成字節(jié)碼的機會,具體我們可以google相關的資料,也可以看下最后例子項目的處理

ASM

ASM是一個字節(jié)碼修改框架,他就在我們上文提到的Transform里面做了文章。關于ASM的介紹我們簡單來幾下,有個大概的認知就好,就像我們訪問一個方法/屬性一樣,jvm肯定是先加載類,然后在執(zhí)行方法或者屬性的方法,ASM的運行機制就如圖一樣

封裝開始

目標

我們的目標是建立一個基于gradle配置即可運行的hook庫,先從使用角度考慮,如果我想hook一個類是LogUtils,中的test方法的話,需要哪些參數呢?快動一下你聰明的小腦袋,emmm,比如類的名稱需要吧!方法名稱!還有捏!只靠這兩個明顯還不夠,因為我們還存在著各種重載不是嘛,那怎么表示一個特定方法呢!沒錯,還有函數簽名對吧!畢竟編譯器底層就是靠著函數簽名去識別某個方法的呀,還有嘛?找到這個方法后,我們是在方法前/方法本身/方法后 進行自定義修改呢?所以就還需要一個類似于模式一樣的東西吧!這里就稱為hook模式好了,還有嘛?找到這個方法,我們還需要自己自定義的操作吧!就定義為hook操作吧。 總結起來,我們需要hook模式,類的名稱,方法名稱,函數簽名,hook操作就可以完成一次hook某個方法的需求了對吧,就比如以下代碼所示

比如hook LogUtils類的test方法,簽名是()V,
替換為調用LogTest類的一個靜態(tài)方法test
hookMethod hookMode.Default(hook模式), 
"com/example/spider/LogUtils"(類的名稱), 
"test"(方法名稱), "()V"(函數簽名), { MethodVisitor mv ->
    mv.visitMethodInsn(Opcodes.INVOKESTATIC, 
    "com/example/spider/LogTest", "test", "()V", false)
}(hook操作)
 

實現

為了使只要通過上面的代碼就能實現hook操作,我們需要定義: asm相關的:自定義的classvisitor,methodvisitor gradle:extension參數,比如上面的“hookMethod”,用來標記我們需要哪部分進行hook操作 transform:標準transform寫法。

gradle 定義extension

我們按照上面思路,是不是需要定義一個類,包含hook模式,類的名稱,方法名稱,函數簽名,hook操作,才能將參數傳給transform,從而執(zhí)行自己的ASM操作。 所以就需要定義extension參數: 我們可以在定義plugin的時候,在apply階段通過project.extensions.create,創(chuàng)建一個自己的配置格式參數,比如Hook.class里面就有我們的參數

project.extensions.create("hook"(標識名稱), Hook.class)

使用的話就可以在任意gradle文件使用

hook{
  參數1 對象值
  參數2 對象值
}

這樣的話,我們只需要在Transform階段收集到配置信息傳給ASM即可!。

Transform階段收集信息:

gradle聲明的信息我們都可以通過project.xx(標識名稱)獲取

比如

hook.methodHooker = project.hook,就拿到了一個屬于Hook類的hook對象
后續(xù)通過hook.hook模式就可以拿到屬性是hook模式的參數了

自定義的classvisitor

我們transform階段會遍歷所有的類,但是我們只需要對特定的類進行修改對不對,所以在這里,我們需要針對只需對gradle配置的類,比如例子中的LogUtils進行處理即可,而不需要動刀其他的類! transform進行時,調用classvisitor就會調用其visit方法,我們在這里識別出我們需要hook的類即可對不對,加入我們需要hook的東西都在 hook.hookMethodList里面,我們只需要遍歷一遍,找到需要的類,然后打上一個標記

@Override
public void visit(int version, int access, String name, String signature, String superName, String[] interfaces) {
    super.visit(version, access, name, signature, superName, interfaces);
   for(遍歷hookMethodList里面){
    if 如果配置的類 == name{
        標記就為true
    }
    }
}

調用visit方法后,就代表了這個類被訪問過了,就會調用其visitmethod方法,如果標記有效,我們就采用自定義的method visitor進行方法的修改,否則就還是原本的method visitor

@Override
public MethodVisitor visitMethod(int access, String name, String descriptor, String signature, String[] exceptions) {
    MethodVisitor mv = cv.visitMethod(access, name, descriptor, signature, exceptions);
    if (標記不為true) {
        return mv;
    }
    進行我們自定義的method visitor操作
    }
    return mv;
}

自定義method visitor

如果class是我們需要hook的class,就會走到了自定義的method visitor,這里是ASM的定義

@Override
public void visitCode() {
    if hook模式是方法前{
     hook 行為執(zhí)行
     }
    super.visitCode();
}
@Override
public void visitMethodInsn(int opcode, String owner, String methodName, String descriptor, boolean isInterface) {
if hook模式是方法本身{
     hook 行為執(zhí)行
     }
    super.visitMethodInsn(opcode, owner, methodName, descriptor, isInterface);
}
@Override
public void visitInsn(int opcode) {
    if (opcode == Opcodes.ATHROW || (opcode >= Opcodes.IRETURN && opcode <= Opcodes.RETURN)) {
     if hook模式是方法后{
     hook 行為執(zhí)行
     }
    }
    super.visitInsn(opcode);
}

自定義hook操作

在配置階段,一個hook操作就可以抽象為Closure,如果用groovy語法就是Closure,如果是Kotlin就是一個函數,代表要進行的操作。 在Transform階段我們就可以織入自定義的closure,等滿足條件就觸發(fā)。幸運的是,ASM本身就提供了一個為AndroidStudio,準備的插件,叫“ASM Bytecode viewer”,通過這個插件,我們可以直接生成想要的插入代碼所對應的ASM編碼,如圖:

通過closure所傳遞的methodvisitor,我們就可以執(zhí)行配置的hook操作了。值得注意一點是,Spider不重新定義hook規(guī)則,而是在ASM基礎上,封裝比較容易編譯錯誤的點,比如Transform編寫,visitor類的編寫等等,便于實現我們自己的hook規(guī)格,而脫離框架本身,這點是需要運用Spider的開發(fā)者需要注意的點!

總結

因為ASM體系有很多細節(jié),文章是沒辦法列舉出所有細節(jié),所以只能表露一個設計思路,具體的用法大家可以移步github.com/TestPlanB/S… 上面也是Spider的設計思路,具體用法也可以看Readme噢!

以上就是從零開始使用gradle配置即可執(zhí)行的Hook庫詳解的詳細內容,更多關于gradle配置可執(zhí)行Hook庫的資料請關注腳本之家其它相關文章!

相關文章

最新評論