arthas最底层的java agent原理

\color{red}{欢迎关注我的微信公众号:进阶者euj}
上一篇文章不停机不更新代码线上调试BUG的工具
教会了大家如果使用arthas是定位系统线上问题
这篇文章教大家关于arthas的原理
arthas其实也是利用是了java agent。
这个是jvm提供出来的接口,在类加载之前会触发一个方法,让大家自定义自己想要切入的业务;

如何写一个自己的java agent

1、新建一个springboot的工程,用来模拟自己的app应用,应用每5秒会调用一个方法

package com.eujian.arthaslearn.controller;

public class MyService {

    public String send(){
        System.out.println("send被调用了");
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        return "send";
    }
}

github地址 https://github.com/hd-eujian/arthas-learn.git
码云地址 https://gitee.com/guoeryyj/arthas-learn.git

2、新建 agent工程,打包成jar包,去做class文件的aop
pom文件引入依赖

 <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-assembly-plugin</artifactId>
                <configuration>
                    <descriptorRefs>
                        <descriptorRef>jar-with-dependencies</descriptorRef>
                    </descriptorRefs>
                    <archive>
                        <manifestEntries>
                            <Premain-Class>com.eujian.agent.PreMainTraceAgent</Premain-Class>
                            <Agent-Class>com.eujian.agent.PreMainTraceAgent</Agent-Class>
                            <Can-Redefine-Classes>true</Can-Redefine-Classes>
                            <Can-Retransform-Classes>true</Can-Retransform-Classes>
                        </manifestEntries>
                    </archive>
                </configuration>

            </plugin>
        </plugins>
    </build>

新建文件PreMainTraceAgent

package com.eujian.agent;

import java.lang.instrument.Instrumentation;

public class PreMainTraceAgent {
    public static void agentmain (String agentArgs, Instrumentation inst) throws Exception {
        System.out.println("agent begin agentArgs="+agentArgs);
        MyClassFileTransformer myClassFileTransformer;

        //如果入参是1就清除aop
        if("1".equals(agentArgs)){
            myClassFileTransformer = new MyClassFileTransformer(true);
            inst.addTransformer(myClassFileTransformer,true);
        }else {
            myClassFileTransformer = new MyClassFileTransformer();
            inst.addTransformer(myClassFileTransformer,true);
        }
        System.out.println("agent end");

        Class[] allLoadedClasses = inst.getAllLoadedClasses();
        for (Class clazz : allLoadedClasses){
            if(clazz.getName().contains("com.eujian.arthaslearn.controller.MyService")){
                inst.retransformClasses(clazz);
                System.out.println("重新加载"+clazz);
            }
        }
    }
}

新建文件MyClassFileTransformer实现接口ClassFileTransformer

package com.eujian.agent;

import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtMethod;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.lang.instrument.ClassFileTransformer;
import java.lang.instrument.IllegalClassFormatException;
import java.security.ProtectionDomain;

public class MyClassFileTransformer implements ClassFileTransformer {

    private boolean isReLoad = false;

    public MyClassFileTransformer() {
    }

    public MyClassFileTransformer(boolean isReLoad) {
        this.isReLoad = isReLoad;
    }

    @Override
    public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined, ProtectionDomain protectionDomain, byte[] classfileBuffer) throws IllegalClassFormatException {

        if (!className.equals("com/eujian/arthaslearn/controller/MyService")) {
            return null;
        }
        try {

            System.out.println("进入 isReLoad:"+isReLoad);
            System.out.println("进入 className:"+className);
            System.out.println("进入 loader:"+loader);
            System.out.println("进入 classBeingRedefined:"+classBeingRedefined);
            CtClass cl = null;
            ClassPool classPool = ClassPool.getDefault();
            cl = classPool.getCtClass("com.eujian.arthaslearn.controller.MyService");

            System.out.println("cl.isFrozen()+"+cl.isFrozen());
            if(isReLoad){
                //重新加载本地class文件
                cl.defrost();
                return readStream(ClassLoader.getSystemResourceAsStream(className.replace('.', '/') + ".class"), true);
            }
            CtMethod method = cl.getDeclaredMethod("send");

            //插入你想要的代码
            method.insertBefore("System.out.println(\"send-begin\");");
            method.insertAfter("System.out.println(\"send-end\");");

            byte[] transformed = cl.toBytecode();

            cl.detach();
            return transformed;

        }catch (Exception e){
            e.printStackTrace();
        }
        return null;
    }
    private static byte[] readStream(InputStream inputStream, boolean close) throws IOException {
        if(inputStream == null) {
            throw new IOException("Class not found");
        } else {
            try {
                ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
                byte[] data = new byte[4096];

                int bytesRead;
                while((bytesRead = inputStream.read(data, 0, data.length)) != -1) {
                    outputStream.write(data, 0, bytesRead);
                }

                outputStream.flush();
                byte[] var5 = outputStream.toByteArray();
                return var5;
            } finally {
                if(close) {
                    inputStream.close();
                }

            }
        }
    }
}

执行命令打包mvn clean assembly:assembly
github 地址:https://github.com/hd-eujian/agent.git
码云地址: https://gitee.com/guoeryyj/agent.git

新建一个main函数的工程,工程主要作用是把打包的jar包注入到运行的app中
代码如下

import com.sun.tools.attach.VirtualMachine;
import com.sun.tools.attach.VirtualMachineDescriptor;

import java.util.List;

public class Main {

    public static void main(String[] args) throws Exception{
        System.out.println("running JVM start ");
        String jarUrl = "/xxx/agent/target/agent-1.0-SNAPSHOT-jar-with-dependencies.jar";
        List<VirtualMachineDescriptor> list = VirtualMachine.list();
        for (VirtualMachineDescriptor vmd : list) {
            if (vmd.displayName().contains("ArthasLearnApplication")) {
                VirtualMachine virtualMachine = VirtualMachine.attach(vmd.id());

                //这个是加载jar包增强class文件
                virtualMachine.loadAgent(jarUrl);

                //这个是恢复class文件
//                virtualMachine.loadAgent(jarUrl,"1");
                virtualMachine.detach();
            }
        }
    }
}

实操环节

1、先启动arthas-learn工程


arthas-learn工程

2、agent工程打包
mvn clean assembly:assembly

3、执行main函数


注入jar包后的效果

\color{red}{欢迎关注我的微信公众号:进阶者euj}

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