Jvm?sandbox?mock機制的實踐過程
一、背景
Jvm sandbox沙箱機制,是一種實現(xiàn)不重啟、無侵入改變目標應(yīng)用返回值的面向切面編程解決方案。測試方面來說,對于RPC接口、HTTP接口都適用。
如果需要開發(fā)一個比較全面的mock平臺,不僅僅是簡單的http接口mock,則可以考慮該方案。
本次主要介紹使用官網(wǎng)的案例,進行實踐測試效果,后續(xù)會介紹如何利用Jvm sandbox搭建Mock平臺。
二、定義一個損壞的鐘
下面直接以Linux服務(wù)器上的實操舉例,Mac同理。
我們先創(chuàng)建一個Springboot工程,工程中定義一個Controller類定義URL入口,調(diào)用Clock類,Clock類為官網(wǎng)案例:一個損壞了的鐘。
1、 Springboot工程中創(chuàng)建一個Clock類
代碼如下:
package com.taobao.demo;
/**
* 報時的鐘
*/
public class Clock {
// 日期格式化
private final java.text.SimpleDateFormat clockDateFormat
= new java.text.SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
/**
* 狀態(tài)檢查
*/
final void checkState() {
throw new IllegalStateException("STATE ERROR!");
}
/**
* 獲取當(dāng)前時間
*
* @return 當(dāng)前時間
*/
final java.util.Date now() {
return new java.util.Date();
}
/**
* 報告時間
*
* @return 報告時間
*/
final String report() {
checkState();
return clockDateFormat.format(now());
}
/**
* 循環(huán)播報時間
*/
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服務(wù)器上運行
訪問定義好的接口:http://服務(wù)器IP地址:8080/clock/start, 會看到目前一直在報錯:

三、修復(fù)這個損壞的鐘
1、 編寫一個模塊修復(fù)損壞的鐘
創(chuàng)建一個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 ”類,修復(fù)損壞的鐘代碼:
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()}方法,當(dāng)這個方法拋出異常時將會被
* AdviceListener#afterThrowing()所攔截
*/
@Override
protected void afterThrowing(Advice advice) throws Throwable {
// 在此,你可以通過ProcessController來改變原有方法的執(zhí)行流程
// 這里的代碼意義是:改變原方法拋出異常的行為,變更為立即返回;void返回值用null表示
ProcessController.returnImmediately(null);
}
});
}
}編譯部署clock-tinker工程:
mvn clean package
項目工程的target目錄下會生成“clock-tinker-1.0-SNAPSHOT-jar-with-dependencies.jar” 包:

2、 下載并安裝jvm-sandbox
下載地址:
使用命令wget下載后,解壓:
unzip sandbox-stable-bin.zipcd sandbox
目錄結(jié)構(gòu)如下:

3、 將修復(fù)的jar包復(fù)制到sandbox-module目錄下
cp target/clock-tinker-1.0-SNAPSHOT-jar-with-dependencies.jar ~/.sandbox-module/
4、 啟動sandbox
使用命令:ps -ef | grep java , 查到服務(wù)器上啟動的被損壞的鐘clock的java 進程號,比如進程號是:1。
進入到解壓后的sandbox的bin目錄下,使用以下命令啟動sandbox:
./sandbox.sh -p 1

使用以下命令查看模塊:
./sandbox.sh -p 1 -l

可以看到broken-clock-tinker模塊已經(jīng)正確被沙箱所加載。
5、 修復(fù)clock#checkState()方法
接下來就是重頭戲,如何在不影響目標應(yīng)用的情況下,無聲無息的修復(fù)這個故障!
觸發(fā)broken-clock-tinker模塊的repairCheckState(),讓修復(fù)邏輯生效。
執(zhí)行命令:觸發(fā)BrokenClockTinkerModule#repairCheckState()方法執(zhí)行
./sandbox.sh -p 1 -d 'broken-clock-tinker/repairCheckState'

模塊生效完成,你就會發(fā)現(xiàn)原本一直拋異常的鐘已經(jīng)開始在刷新時間了:

6、 恢復(fù)被修復(fù)的check()方法
停止sandbox的命令:
./sandbox.sh -p 64229 -S

會看到提示信息:
jvm-sandbox[default] shutdown finished.
你就會發(fā)現(xiàn)原本已經(jīng)被修復(fù)好的鐘,又開始繼續(xù)報錯了。
原因是原來通過clock-tinker模塊修復(fù)的checkState()方法隨著沙箱的卸載又恢復(fù)成原來錯誤的代碼流程:

總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
深入了解SpringBoot中@ControllerAdvice的介紹及三種用法
這篇文章主要為大家詳細介紹了SpringBoot中@ControllerAdvice的介紹及三種用法,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-02-02
java遠程連接Linux執(zhí)行命令的3種方式完整代碼
在一些Java應(yīng)用程序中需要執(zhí)行一些Linux系統(tǒng)命令,例如服務(wù)器資源查看、文件操作等,這篇文章主要給大家介紹了關(guān)于java遠程連接Linux執(zhí)行命令的3種方式,文中通過代碼介紹的非常詳細,需要的朋友可以參考下2024-06-06
Spring Boot利用JSR303實現(xiàn)參數(shù)驗證的方法實例
這篇文章主要給大家介紹了關(guān)于Spring Boot利用JSR303實現(xiàn)參數(shù)驗證的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家學(xué)習(xí)或者使用Spring Boot具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05
SpringBoot?ScheduledTaskRegistrar解決動態(tài)定時任務(wù)思路詳解
本文將從問題出發(fā),詳細介紹ScheduledTaskRegistrar類是如何解決動態(tài)調(diào)整定時任務(wù)的思路,并給出關(guān)鍵的代碼示例,幫助大家快速地上手學(xué)習(xí)2023-02-02

