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類(lèi)定義URL入口,調(diào)用Clock類(lèi),Clock類(lèi)為官網(wǎng)案例:一個(gè)損壞了的鐘。
1、 Springboot工程中創(chuàng)建一個(gè)Clock類(lèi)
代碼如下:
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類(lèi)
指向運(yùn)行Clock類(lèi)中的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 ”類(lèi),修復(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-02java遠(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關(guān)于Java類(lèi)的構(gòu)造方法詳解
這篇文章主要介紹了關(guān)于Java類(lèi)的構(gòu)造方法詳解的相關(guān)資料,需要的朋友可以參考下2023-01-01springboot實(shí)現(xiàn)瀏覽器截屏并添加文字
大家好,本篇文章主要講的是springboot實(shí)現(xiàn)瀏覽器截屏并添加文字,感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下2022-02-02Spring 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-05springboot日志切面通用類(lèi)實(shí)例詳解
這篇文章主要介紹了springboot日志切面通用類(lèi),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-01-01Java解決同時(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-09SpringBoot?ScheduledTaskRegistrar解決動(dòng)態(tài)定時(shí)任務(wù)思路詳解
本文將從問(wèn)題出發(fā),詳細(xì)介紹ScheduledTaskRegistrar類(lèi)是如何解決動(dòng)態(tài)調(diào)整定時(shí)任務(wù)的思路,并給出關(guān)鍵的代碼示例,幫助大家快速地上手學(xué)習(xí)2023-02-02