当前位置: 代码网 > it编程>编程语言>其他编程 > Jvm sandbox mock机制的实践过程

Jvm sandbox mock机制的实践过程

2025年05月28日 其他编程 我要评论
一、背景jvm sandbox沙箱机制,是一种实现不重启、无侵入改变目标应用返回值的面向切面编程解决方案。测试方面来说,对于rpc接口、http接口都适用。如果需要开发一个比较全面的mock平台,不仅

一、背景

jvm sandbox沙箱机制,是一种实现不重启、无侵入改变目标应用返回值的面向切面编程解决方案。测试方面来说,对于rpc接口、http接口都适用。

如果需要开发一个比较全面的mock平台,不仅仅是简单的http接口mock,则可以考虑该方案。

本次主要介绍使用官网的案例,进行实践测试效果,后续会介绍如何利用jvm sandbox搭建mock平台。

二、定义一个损坏的钟

下面直接以linux服务器上的实操举例,mac同理。

我们先创建一个springboot工程,工程中定义一个controller类定义url入口,调用clock类,clock类为官网案例:一个损坏了的钟。

1、 springboot工程中创建一个clock类

代码如下:

package com.taobao.demo;

/**
 * 报时的钟
 */
public class clock {

    // 日期格式化
    private final java.text.simpledateformat clockdateformat
            = new java.text.simpledateformat("yyyy-mm-dd hh:mm:ss");

    /**
     * 状态检查
     */
    final void checkstate() {
        throw new illegalstateexception("state error!");
    }

    /**
     * 获取当前时间
     *
     * @return 当前时间
     */
    final java.util.date now() {
        return new java.util.date();
    }

    /**
     * 报告时间
     *
     * @return 报告时间
     */
    final string report() {
        checkstate();
        return clockdateformat.format(now());
    }

    /**
     * 循环播报时间
     */
    final void loopreport() throws interruptedexception {
        while (true) {
            try {
                system.out.println(report());
            } catch (throwable cause) {
                cause.printstacktrace();
            }
            thread.sleep(1000);
        }
    }

    public static void main(string... args) throws interruptedexception {
        new clock().loopreport();
    }

}

本地运行之后,会看到目前一直在报异常:

java.lang.illegalstateexception: state error!
    at com.taobao.demo.clock.checkstate(clock.java:16)
    at com.taobao.demo.clock.report(clock.java:34)
    at com.taobao.demo.clock.loopreport(clock.java:44)
    at com.taobao.demo.clock.main(clock.java:53)
java.lang.illegalstateexception: state error!
    at com.taobao.demo.clock.checkstate(clock.java:16)
    at com.taobao.demo.clock.report(clock.java:34)
    at com.taobao.demo.clock.loopreport(clock.java:44)
    at com.taobao.demo.clock.main(clock.java:53)

2、 添加一个controller类

指向运行clock类中的loopreport方法:

import com.taobao.demo.service.clock;
import lombok.extern.slf4j.slf4j;
import org.springframework.beans.factory.annotation.autowired;
import org.springframework.web.bind.annotation.requestmapping;
import org.springframework.web.bind.annotation.restcontroller;

@slf4j
@restcontroller
@requestmapping("clock")
public class clockcontroller {

    @autowired
    clock clock;

    @requestmapping("/start")
    public void start() throws interruptedexception {
        log.info("start clock");
        clock.loopreport();
    }
}

3、 部署该springboot项目到linux服务器上运行

访问定义好的接口:http://服务器ip地址:8080/clock/start, 会看到目前一直在报错:

三、修复这个损坏的钟

1、 编写一个模块修复损坏的钟

创建一个maven工程:clock-tinker, pom.xml添加“sandbox-module-starter” maven依赖:

<parent>
    <groupid>com.alibaba.jvm.sandbox</groupid>
    <artifactid>sandbox-module-starter</artifactid>
    <version>1.2.0</version>
</parent> 

编写“brokenclocktinkermodule ”类,修复损坏的钟代码:

package com.alibaba.jvm.sandbox.demo;

import com.alibaba.jvm.sandbox.api.information;
import com.alibaba.jvm.sandbox.api.module;
import com.alibaba.jvm.sandbox.api.processcontroller;
import com.alibaba.jvm.sandbox.api.annotation.command;
import com.alibaba.jvm.sandbox.api.listener.ext.advice;
import com.alibaba.jvm.sandbox.api.listener.ext.advicelistener;
import com.alibaba.jvm.sandbox.api.listener.ext.eventwatchbuilder;
import com.alibaba.jvm.sandbox.api.resource.moduleeventwatcher;
import org.kohsuke.metainfservices;

import javax.annotation.resource;

@metainfservices(module.class)
@information(id = "broken-clock-tinker")
public class brokenclocktinkermodule implements module {

    @resource
    private moduleeventwatcher moduleeventwatcher;

    @command("repaircheckstate")
    public void repaircheckstate() {

        new eventwatchbuilder(moduleeventwatcher)
                .onclass("com.taobao.demo.clock")
                .onbehavior("checkstate")
                .onwatch(new advicelistener() {

                    /**
                     * 拦截{@code com.taobao.demo.clock#checkstate()}方法,当这个方法抛出异常时将会被
                     * advicelistener#afterthrowing()所拦截
                     */
                    @override
                    protected void afterthrowing(advice advice) throws throwable {
                        
                        // 在此,你可以通过processcontroller来改变原有方法的执行流程
                        // 这里的代码意义是:改变原方法抛出异常的行为,变更为立即返回;void返回值用null表示
                        processcontroller.returnimmediately(null);
                    }
                });
    }
}

编译部署clock-tinker工程:

mvn clean package

项目工程的target目录下会生成“clock-tinker-1.0-snapshot-jar-with-dependencies.jar” 包:

2、 下载并安装jvm-sandbox

下载地址:https://ompc.oss.aliyuncs.com/jvm-sandbox/release/sandbox-stable-bin.zip

使用命令wget下载后,解压:

unzip sandbox-stable-bin.zipcd sandbox

目录结构如下:

3、 将修复的jar包复制到sandbox-module目录下

cp target/clock-tinker-1.0-snapshot-jar-with-dependencies.jar ~/.sandbox-module/

4、 启动sandbox

使用命令:ps -ef | grep java , 查到服务器上启动的被损坏的钟clock的java 进程号,比如进程号是:1。

进入到解压后的sandbox的bin目录下,使用以下命令启动sandbox:

./sandbox.sh -p 1

使用以下命令查看模块:

./sandbox.sh -p 1 -l

可以看到broken-clock-tinker模块已经正确被沙箱所加载。

5、 修复clock#checkstate()方法

接下来就是重头戏,如何在不影响目标应用的情况下,无声无息的修复这个故障!

触发broken-clock-tinker模块的repaircheckstate(),让修复逻辑生效。

执行命令:触发brokenclocktinkermodule#repaircheckstate()方法执行

./sandbox.sh -p 1 -d 'broken-clock-tinker/repaircheckstate'

模块生效完成,你就会发现原本一直抛异常的钟已经开始在刷新时间了:

6、 恢复被修复的check()方法

停止sandbox的命令:

./sandbox.sh -p 64229 -s

会看到提示信息:

jvm-sandbox[default] shutdown finished.

你就会发现原本已经被修复好的钟,又开始继续报错了。

原因是原来通过clock-tinker模块修复的checkstate()方法随着沙箱的卸载又恢复成原来错误的代码流程:

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持代码网。

(0)

相关文章:

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论

验证码:
Copyright © 2017-2025  代码网 保留所有权利. 粤ICP备2024248653号
站长QQ:2386932994 | 联系邮箱:2386932994@qq.com