Frida 之 Java层Hook

前言

Frida是个轻量级别的hook框架,是Python API,但JavaScript调试逻辑,它既可以hook java层也可以hook native层

Frida的核心是用C编写的,并将Google的V8引擎注入到目标进程中,在这些进程中,JS可以完全访问内存,挂钩函数甚至调用进程内的本机函数来执行。

使用Python和JS可以使用无风险的API进行快速开发。Frida可以帮助您轻松捕获JS中的错误并为您提供异常而不是崩溃。

环境

Android4.4.4
Nexus5手机(ARM)
frida12.11.18
python3.6

安装

首先使用pip安装frida和frida-tools模块

pip install frida
pip install frida-tools

我这里使用PyCharm安装

image.png

下载frida-server,官方下载地址https://github.com/frida/frida/releases,这里注意下载的版本应该和上面安装的frida版本一致。
在adb shell中查看cpu的架构

getprop ro.product.cpu.abi
image.png

所以我们下载这个版本


image.png

下载后解压重命名为frida-server 并push到手机上。
进入root权限,给其权限并运行,手机会重启


image.png

image.png

然后把端口转发到PC端:

adb forward tcp:27042 tcp:27042

adb forward tcp:27043 tcp:27043

到这里我们就把通信的手机端工作做完了

java层Hook

我们要做事情
1.hook类的普通方法
2.hook类的构造方法
3.构造和修改自定义类型对象和属性

先写一个目标app

image.png

添加SharkUtils和Student两个类
SharkUtils.java

package com.shark.fridatarget;

public class SharkUtils {
    public static String getPwd(String info) {
        return info + "shark";
    }

    public static String getPwd() {
        return "shark";
    }

    public static Student getStu() {
        return new Student("fujie", 100);
    }

    public static int getStuScore(Student student) {
        return student.getScore();
    }
}

Student.java

package com.shark.fridatarget;

public class Student {

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getScore() {
        return score;
    }

    public void setScore(int score) {
        this.score = score;
    }

    public String name;
    public int score;

    public Student(String name, int score) {
        this.name = name;
        this.score = score;
    }


}

MainActivity.java

package com.shark.fridatarget;

import androidx.appcompat.app.AppCompatActivity;

import android.os.Bundle;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity {

    TextView textView;
    TextView textView2;
    TextView textView3;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        textView = findViewById(R.id.test);
        Student student = SharkUtils.getStu();
        textView.setText("姓名:" + student.getName() + "------分数:" + student.getScore());

        textView2 = findViewById(R.id.test2);
        textView2.setText("password:"+SharkUtils.getPwd("123456"));

        textView3 = findViewById(R.id.test3);
        textView3.setText("getStuScore:"+SharkUtils.getStuScore(student));

    }
}

app就是调用SharkUtils的方法,然后将返回的信息输出到控件上
运行如下


image.png

现在就来编写Hook代码

import frida  # 导入frida模块
import sys  # 导入sys模块

jscode = """  //从此处开始定义用来Hook的javascript代码
    Java.perform(function(){  
        var student = Java.use('com.shark.fridatarget.Student'); //获得Student类
        var sharkUtils = Java.use('com.shark.fridatarget.SharkUtils'); //获得SharkUtils类
        var clazz = Java.use('java.lang.Class'); //获得Class类
         //Hook普通方法
        sharkUtils.getPwd.overload("java.lang.String").implementation = function(info){ 
            //方式一获取参数
            send("getPwd1:"+info);
            //方式二获取参数
            send("getPwd2:"+arguments[0]);
            return this.getPwd("shark"); //劫持返回值,修改为我们想要返回的字符串
        }
        
        //Hook Student的构造函数$init,用js自己实现
        student.$init.overload("java.lang.String","int").implementation = function(name,score){ 
            send('Statr! Hook!'); //发送信息,用于回调python中的函数
            return this.$init("shark",99); //调用原来的初始化方法
        }
       
       //构造和修改自定义类型对象和属性
       sharkUtils.getStuScore.overload("com.shark.fridatarget.Student").implementation = function(student){ 
            send('student:'+student); 
            
            //使用方法得到属性
            var score = student.getScore();
            send('student score:'+score); 
            //直接得到属性
            var score2 = student.score;
            send('student score2:'+score); 

             //构造一个新的student对象
            var new_stu = student.$new("shark chilli",55);

            //将com.shark.fridatarget.Student转化java.lang.Class
            var scorc_field = Java.cast(student.getClass(),clazz).getDeclaredField("score");
            //这里就是普通的反射了
            scorc_field.setAccessible(true);
            send('reflect scorc_field:'+scorc_field.get(student)); 
            scorc_field.setInt(student,999);
            return student.getScore(); 
        }
    });
"""

def on_message(message, data):  # js中执行send函数后要回调的函数
    print(message)

# 得到设备并劫持进程com.shark.fridatarget(
 # 该开始用get_usb_device函数用来获取设备,但是一直报错找不到设备,改用get_remote_device函数即可解决这个问题)
process = frida.get_remote_device().attach('com.shark.fridatarget')

script = process.create_script(jscode)  # 创建js脚本
script.on('message', on_message)  # 加载回调函数,也就是js中执行send函数规定要执行的python函数
script.load()  # 加载脚本
sys.stdin.read()
image.png

这段代码就是hook的整体逻辑使用get_remote_device来获取到设备,然后调用attach附加到app上,所以这里必须要在手机上打开这个应用。否则会找不到。然后创建js脚本,加载回调函数就是我们自己定义的on_message函数,这样js中调用send就调用了我们的on_message函数了。下面就是加载脚本了。

 Java.perform(function(){ 
  ...
 });

我们的hook逻辑都写在上面的js代码中

image.png

使用Java.use获得类的类型
image.png

Hook普通方法,直接使用要hook的方法名,overload是确认方法重载的。
上面看到想要获得参数有两种方式,一是在方法上定义参数名即可。二是通过隐含的arguments变量获取。
this指针就是被hook的对象,想要修改返回值可以直接使用return
image.png

Hook构造方法东西其实都一样,就是调用构造方法的时候使用$init

image.png

构造和修改自定义类型对象和属性,想new一个类的实例使用的是$new。其他的代码和java中的反射大同小异。注意一下的是要使用反射操作这个student实例前需要使用cast转化成java.lang.Class

运行

image.png

引用

Frida hook入门
Android逆向之旅—Hook神器家族的Frida工具使用详解
frida入门总结

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容