在 Java 代理中重新定位 AspectJ 包

2022-09-04 23:17:00

我正在使用AspectJ来监视字段访问和字段修改。我有一个 gradle 项目,它编译了这两个方面,并使用 gradle 阴影插件将该 jar 与 aspectjrt 和 aspectjweaver 一起打包到一个阴影 jar 中。代理仍然是 org.aspectj.weaver.loadtime.Agent。一切都很好,但是当我尝试重新定位aspectaj软件包时,我得到一个错误。

影子插件配置如下:

shadowJar {
    relocate 'org.aspectj', 'shadow.org.aspectj'
    relocate 'aj.org.objectweb.asm', 'shadow.aj.org.objectweb.asm'
}

清单 :

jar {
    manifest {
        attributes("Premain-Class": "shadow.org.aspectj.weaver.loadtime.Agent",
                "Can-Redefine-Classes": true,
                "Can-Retransform-Classes":true)
    }
}

这是反编译的方面类,因此它似乎是正确的:

package com.vfunction.singletonanalysis;

import shadow.org.aspectj.lang.JoinPoint;
import shadow.org.aspectj.lang.NoAspectBoundException;
import shadow.org.aspectj.lang.annotation.Aspect;
import shadow.org.aspectj.lang.annotation.Before;

@Aspect
public class StaticFieldBeforeAccessAspect extends AbstractFieldAccessAspect {
    public StaticFieldBeforeAccessAspect() {
    }

    @Before("callAt()")
    public void before(JoinPoint joinPoint) throws Throwable {
        this.printJoinPoint(joinPoint);
    }

    public static StaticFieldBeforeAccessAspect aspectOf() {
        if (ajc$perSingletonInstance == null) {
            throw new NoAspectBoundException("com.vfunction.singletonanalysis.StaticFieldBeforeAccessAspect", ajc$initFailureCause);
        } else {
            return ajc$perSingletonInstance;
        }
    }

    public static boolean hasAspect() {
        return ajc$perSingletonInstance != null;
    }

    static {
        try {
            ajc$postClinit();
        } catch (Throwable var1) {
            ajc$initFailureCause = var1;
        }

    }
}

但是,在尝试运行测试程序时,我仍然收到错误,指出找到的类型不是一个方面:

[AppClassLoader@18b4aac2] info AspectJ Weaver Version 1.8.12 built on Friday Oct 20, 2017 at 21:58:11 GMT
[AppClassLoader@18b4aac2] info register classloader sun.misc.Launcher$AppClassLoader@18b4aac2
[AppClassLoader@18b4aac2] info using configuration file:***/workspace/singleton-analysis/agent/build/libs/agent-1.0.0-SNAPSHOT-all.jar!/META-INF/aop.xml
[AppClassLoader@18b4aac2] info register aspect com.vfunction.singletonanalysis.StaticFieldModifyAspect
[AppClassLoader@18b4aac2] error The specified aspect 'com.vfunction.singletonanalysis.StaticFieldModifyAspect' cannot be found
[AppClassLoader@18b4aac2] info register aspect com.vfunction.singletonanalysis.StaticFieldAccessAspect
[AppClassLoader@18b4aac2] error The specified aspect 'com.vfunction.singletonanalysis.StaticFieldAccessAspect' cannot be found
[AppClassLoader@18b4aac2] info register aspect com.vfunction.singletonanalysis.StaticFieldBeforeAccessAspect
[AppClassLoader@18b4aac2] error Cannot register 'com.vfunction.singletonanalysis.StaticFieldBeforeAccessAspect' because the type found with that name is not an aspect

答案 1

检查阴影罐内的内容。方面类名称是否隐藏?我怀疑它没有给出错误消息。aop.xml

如文档中所述,您可以使 Shadow 插件转换 XML 文件,如下所示:

shadowJar {
    tranform(XmlAppendingTransformer.class) {
        resource = 'aop.xml'
    }
}

答案 2

我不认为这是问题所在。我不重新定位我的 aspect 类,只重新定位像 aspectj 库这样的依赖项。所以我的方面类看起来像这样:

import com.vfunction.jni.Callbacks;
import java.lang.reflect.Field;
import vshadow.org.aspectj.lang.JoinPoint;
import vshadow.org.aspectj.lang.annotation.Aspect;
import vshadow.org.aspectj.lang.annotation.Before;
import vshadow.org.aspectj.lang.annotation.Pointcut;

@Aspect
public class StaticFieldBeforeModifyAspect extends AbstractFieldAccessAspect

正如你所看到的,它导入vshadow.org.aspectj.lang.annotation.Aspect,该类存在。问题是aspectjweaver代码通过检查类是否具有org.aspectj.lang.annotation.Aspect注释来检查该类是否是有效的方面,但它没有,它有一个vshadow.org.aspectj.lang.annotation.Aspect注释。aspectjweaver 使用字符串常量检查注释。它发生在org.aspectj.weaver.bcel.BcelWeaver#addLibraryAspect中,并且常量是这样使用的,例如:

public final static UnresolvedType ASPECT_ANNOTATION = UnresolvedType.forSignature("Lorg/aspectj/lang/annotation/Aspect;")

并且影子插件不会重新定位它。

我很确定这就是问题所在。


推荐