Jvm?sandbox?mock機(jī)制的實(shí)踐過(guò)程
一、背景
Jvm sandbox沙箱機(jī)制,是一種實(shí)現(xiàn)不重啟、無(wú)侵入改變目標(biāo)應(yīng)用返回值的面向切面編程解決方案。測(cè)試方面來(lái)說(shuō),對(duì)于RPC接口、HTTP接口都適用。
如果需要開(kāi)發(fā)一個(gè)比較全面的mock平臺(tái),不僅僅是簡(jiǎn)單的http接口mock,則可以考慮該方案。
本次主要介紹使用官網(wǎng)的案例,進(jìn)行實(shí)踐測(cè)試效果,后續(xù)會(huì)介紹如何利用Jvm sandbox搭建Mock平臺(tái)。
二、定義一個(gè)損壞的鐘
下面直接以Linux服務(wù)器上的實(shí)操舉例,Mac同理。
我們先創(chuàng)建一個(gè)Springboot工程,工程中定義一個(gè)Controller類定義URL入口,調(diào)用Clock類,Clock類為官網(wǎng)案例:一個(gè)損壞了的鐘。
1、 Springboot工程中創(chuàng)建一個(gè)Clock類
代碼如下:
package com.taobao.demo;
/**
* 報(bào)時(shí)的鐘
*/
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)前時(shí)間
*
* @return 當(dāng)前時(shí)間
*/
final java.util.Date now() {
return new java.util.Date();
}
/**
* 報(bào)告時(shí)間
*
* @return 報(bào)告時(shí)間
*/
final String report() {
checkState();
return clockDateFormat.format(now());
}
/**
* 循環(huán)播報(bào)時(shí)間
*/
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();
}
}本地運(yùn)行之后,會(huì)看到目前一直在報(bào)異常:
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、 添加一個(gè)Controller類
指向運(yùn)行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項(xiàng)目到linux服務(wù)器上運(yùn)行
訪問(wèn)定義好的接口:http://服務(wù)器IP地址:8080/clock/start, 會(huì)看到目前一直在報(bào)錯(cuò):

三、修復(fù)這個(gè)損壞的鐘
1、 編寫(xiě)一個(gè)模塊修復(fù)損壞的鐘
創(chuàng)建一個(gè)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> 編寫(xiě)“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)這個(gè)方法拋出異常時(shí)將會(huì)被
* AdviceListener#afterThrowing()所攔截
*/
@Override
protected void afterThrowing(Advice advice) throws Throwable {
// 在此,你可以通過(guò)ProcessController來(lái)改變?cè)蟹椒ǖ膱?zhí)行流程
// 這里的代碼意義是:改變?cè)椒⊕伋霎惓5男袨?,變更為立即返回;void返回值用null表示
ProcessController.returnImmediately(null);
}
});
}
}編譯部署clock-tinker工程:
mvn clean package
項(xiàng)目工程的target目錄下會(huì)生成“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、 啟動(dòng)sandbox
使用命令:ps -ef | grep java , 查到服務(wù)器上啟動(dòng)的被損壞的鐘clock的java 進(jìn)程號(hào),比如進(jìn)程號(hào)是:1。
進(jìn)入到解壓后的sandbox的bin目錄下,使用以下命令啟動(dòng)sandbox:
./sandbox.sh -p 1

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

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

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

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

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

總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
深入了解SpringBoot中@ControllerAdvice的介紹及三種用法
這篇文章主要為大家詳細(xì)介紹了SpringBoot中@ControllerAdvice的介紹及三種用法,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-02-02
java遠(yuǎn)程連接Linux執(zhí)行命令的3種方式完整代碼
在一些Java應(yīng)用程序中需要執(zhí)行一些Linux系統(tǒng)命令,例如服務(wù)器資源查看、文件操作等,這篇文章主要給大家介紹了關(guān)于java遠(yuǎn)程連接Linux執(zhí)行命令的3種方式,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-06-06
springboot實(shí)現(xiàn)瀏覽器截屏并添加文字
大家好,本篇文章主要講的是springboot實(shí)現(xiàn)瀏覽器截屏并添加文字,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下2022-02-02
Spring Boot利用JSR303實(shí)現(xiàn)參數(shù)驗(yàn)證的方法實(shí)例
這篇文章主要給大家介紹了關(guān)于Spring Boot利用JSR303實(shí)現(xiàn)參數(shù)驗(yàn)證的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Spring Boot具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05
Java解決同時(shí)出庫(kù)入庫(kù)訂單號(hào)自動(dòng)獲取問(wèn)題解決
在Java中,處理多線程環(huán)境下的訂單號(hào)生成問(wèn)題可以采用多種策略,如使用AtomicLong保證線程安全,通過(guò)定義訂單號(hào)生成器并利用線程模擬出庫(kù)和入庫(kù)操作,每個(gè)線程從訂單號(hào)生成器中獲取唯一訂單號(hào),感興趣的朋友一起看看吧2024-09-09
SpringBoot?ScheduledTaskRegistrar解決動(dòng)態(tài)定時(shí)任務(wù)思路詳解
本文將從問(wèn)題出發(fā),詳細(xì)介紹ScheduledTaskRegistrar類是如何解決動(dòng)態(tài)調(diào)整定時(shí)任務(wù)的思路,并給出關(guān)鍵的代碼示例,幫助大家快速地上手學(xué)習(xí)2023-02-02

