一、背景
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()方法随着沙箱的卸载又恢复成原来错误的代码流程:
总结
以上为个人经验,希望能给大家一个参考,也希望大家多多支持代码网。
发表评论