SpringBoot自定義maven-plugin插件整合asm代碼插樁
背景
公司開發(fā)框架增加了web系統(tǒng)license授權(quán)證書校驗模塊,實行一臺機器一個授權(quán)證書,初步方案是增加攔截器針對全局請求進行攔截校驗,評估后認為校驗方式單一,應該增加重要工具類,業(yè)務service實現(xiàn)中每個方法的進行校驗,因為涉及代碼量較大硬編碼工作困難,故選擇通過自定義maven插件在編譯期間進行動態(tài)代碼插樁操作
項目配置
新建maven項目設置打包方式
<packaging>maven-plugin</packaging>
增加依賴項
?? ??? ? <!--使用doc的方式--> ? ? ? ? <dependency> ? ? ? ? ? ? <groupId>org.apache.maven</groupId> ? ? ? ? ? ? <artifactId>maven-plugin-api</artifactId> ? ? ? ? ? ? <version>3.5.2</version> ? ? ? ? </dependency> ?? ??? ?<!--使用注解的方式--> ? ? ? ? <dependency> ? ? ? ? ? ? <groupId>org.apache.maven.plugin-tools</groupId> ? ? ? ? ? ? <artifactId>maven-plugin-annotations</artifactId> ? ? ? ? ? ? <version>3.5.2</version> ? ? ? ? ? ? <scope>provided</scope> ? ? ? ? </dependency> ? ? ? ? <dependency> ? ? ? ? ? ? <groupId>org.apache.maven</groupId> ? ? ? ? ? ? <artifactId>maven-project</artifactId> ? ? ? ? ? ? <version>2.2.1</version> ? ? ? ? </dependency> ? ? ? ? <dependency> ? ? ? ? ? ? <groupId>org.ow2.asm</groupId> ? ? ? ? ? ? <artifactId>asm</artifactId> ? ? ? ? ? ? <version>9.0</version> ? ? ? ? </dependency> ?
build內(nèi)容配置
?? ? <plugins> ? ? ? ? ? ? <plugin> ? ? ? ? ? ? ? ? <groupId>org.apache.maven.plugins</groupId> ? ? ? ? ? ? ? ? <artifactId>maven-plugin-plugin</artifactId> ? ? ? ? ? ? ? ? <version>3.5</version> ? ? ? ? ? ? </plugin> ? ? ? ? ? ? <plugin> ? ? ? ? ? ? ? ? <groupId>org.apache.maven.plugins</groupId> ? ? ? ? ? ? ? ? <artifactId>maven-compiler-plugin</artifactId> ? ? ? ? ? ? ? ? <version>3.6.1</version> ? ? ? ? ? ? ? ? <configuration> ? ? ? ? ? ? ? ? ? ? <source>1.8</source> ? ? ? ? ? ? ? ? ? ? <target>1.8</target> ? ? ? ? ? ? ? ? </configuration> ? ? ? ? ? ? </plugin> ? ? ? ? </plugins>
編譯攔截
創(chuàng)建編譯操作類FramePlugin,繼承AbstractMojo并使用Mojo注解標注,output參數(shù)是class文件編譯后路徑
@Mojo(name = "deepcompile", defaultPhase = LifecyclePhase.COMPILE)
public class FramePlugin extends AbstractMojo {
@Parameter(name = "output", defaultValue = "${project.build.directory}")
private File output;
public void execute() throws MojoExecutionException {
File f = ;
if (!f.exists()) {
f.mkdirs();
}
try {
insertPile(f);
} catch (Exception e) {
exceptioncount++;
e.printStackTrace();
}
}ASM插樁
新建ClassVisitor重寫visitMethod方法來過濾訪問需要插樁的方法,需要排除自帶的init方法
public class MethodCoverageClassVisitor extends ClassVisitor {
? ? public MethodCoverageClassVisitor(ClassVisitor classVisitor) {
? ? ? ? super(Opcodes.ASM9, classVisitor);
? ? }
? ? @Override
? ? public MethodVisitor visitMethod(int access, String name, String descriptor, String signature,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?String[] exceptions) {
? ? ? ? final MethodVisitor methodVisitor = super.visitMethod(access, name, descriptor, signature, exceptions);
? ? ? ? if (name.equals("<init>")) {
? ? ? ? ? ? return methodVisitor;
? ? ? ? }
? ? ? ? return new MethodCoverageMethodVisitor(Opcodes.ASM9, methodVisitor);
? ? }
}新建MethodVisitor重寫visitCode方法針對方法內(nèi)部字節(jié)碼進行自定義操作,這里是使用框架內(nèi)部封裝好的一個靜態(tài)方法來校驗license證書
public class MethodCoverageMethodVisitor extends MethodVisitor {
? ? public MethodCoverageMethodVisitor(int api, MethodVisitor methodVisitor) {
? ? ? ? super(api, methodVisitor);
? ? }
? ? @Override
? ? public void visitCode() {
? ? ? ? mv.visitFieldInsn(Opcodes.INVOKESTATIC, "com/xxxx/frame/common/utils/ComplieSDK", "checkLicense", "()V");
? ? }
}最后在execute中進行文件遞歸查找調(diào)用,就是將已經(jīng)編譯的class文件讀取/自定義操作后保存
?private void insertPile(File root) throws IOException {
? ? ? ? if (root.isDirectory()) {
? ? ? ? ? ? for (File file : root.listFiles()) {
? ? ? ? ? ? ? ? insertPile(file);
? ? ? ? ? ? }
? ? ? ? }
? ? ? ? String className = root.getName().replace(".class", "");
? ? ? ? if (root.getName().endsWith(".class")) {
? ? ? ? ? ? //class篩選
? ? ? ? ? ? boolean flag = false;
? ? ? ? ??? ?//自定義的class文件篩選條件代碼
? ? ? ? ? ? if (flag) {
? ? ? ? ? ? ? ? System.out.println("【insertPile】:" + className);
? ? ? ? ? ? ? ? FileOutputStream fos = null;
? ? ? ? ? ? ? ? try {
? ? ? ? ? ? ? ? ? ? final byte[] instrumentBytes = doInsertPile(root);
? ? ? ? ? ? ? ? ? ? fos = new FileOutputStream(root);
? ? ? ? ? ? ? ? ? ? fos.write(instrumentBytes);
? ? ? ? ? ? ? ? ? ? fos.flush();
? ? ? ? ? ? ? ? } catch (MojoExecutionException e) {
? ? ? ? ? ? ? ? ? ? System.out.println("【insertPile-exception】:" + className);
? ? ? ? ? ? ? ? ? ? e.printStackTrace();
? ? ? ? ? ? ? ? } finally {
? ? ? ? ? ? ? ? ? ? if (fos != null) {
? ? ? ? ? ? ? ? ? ? ? ? fos.close();
? ? ? ? ? ? ? ? ? ? }
? ? ? ? ? ? ? ? }
? ? ? ? ? ? }
? ? ? ? }
? ? }項目使用
maven-plugin項目執(zhí)行mvn install安裝到本地倉庫
框架項目配置自定義maven插件進行打包,配置執(zhí)行的聲明周期為complie(編譯),這里goal自定義命令名稱需要和mojo注解標注類中指定的name名稱一致
??? ??? ? <plugin> ? ? ? ? ? ? ? ? <groupId>com.xxxxx</groupId> ? ? ? ? ? ? ? ? <artifactId>frame-maven-plugin</artifactId> ? ? ? ? ? ? ? ? <version>1.2.5</version> ? ? ? ? ? ? ? ? <executions> ? ? ? ? ? ? ? ? ? ? <execution> ? ? ? ? ? ? ? ? ? ? ? ? <goals> ? ? ? ? ? ? ? ? ? ? ? ? ? ? <!-- 執(zhí)行目標 --> ? ? ? ? ? ? ? ? ? ? ? ? ? ? <goal>deepcompile</goal> ? ? ? ? ? ? ? ? ? ? ? ? </goals> ? ? ? ? ? ? ? ? ? ? ? ? <!-- 執(zhí)行這個目標所在的生命周期 --> ? ? ? ? ? ? ? ? ? ? ? ? <phase>compile</phase> ? ? ? ? ? ? ? ? ? ? </execution> ? ? ? ? ? ? ? ? </executions> ? ? ? ? ? ? </plugin>
到此這篇關(guān)于SpringBoot自定義maven-plugin插件整合asm代碼插樁的文章就介紹到這了,更多相關(guān)maven-plugin asm代碼插樁內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java?Dubbo服務調(diào)用擴展點Filter使用教程
Dubbo是阿里巴巴公司開源的一個高性能優(yōu)秀的服務框架,使得應用可通過高性能的RPC實現(xiàn)服務的輸出和輸入功能,可以和Spring框架無縫集成2022-12-12
簡單了解springboot中的配置文件相關(guān)知識
這篇文章主要介紹了簡單了解springboot中的配置文件相關(guān)知識,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2019-11-11

