Skip to content

Commit fa21dff

Browse files
committed
Restore traditional AspectJ behavior through "spring.aop.ajc.ignore=true"
Closes gh-33704
1 parent 323de12 commit fa21dff

File tree

2 files changed

+40
-6
lines changed

2 files changed

+40
-6
lines changed

framework-docs/modules/ROOT/pages/appendix.adoc

+9-5
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,12 @@ The following table lists all currently supported Spring properties.
2323
|===
2424
| Name | Description
2525

26+
| `spring.aop.ajc.ignore`
27+
| Instructs Spring to ignore ajc-compiled aspects for Spring AOP proxying, restoring traditional
28+
Spring behavior for scenarios where both weaving and AspectJ auto-proxying are enabled. See
29+
{spring-framework-api}++/aop/aspectj/annotation/AbstractAspectJAdvisorFactory.html#IGNORE_AJC_PROPERTY_NAME++[`AbstractAspectJAdvisorFactory`]
30+
for details.
31+
2632
| `spring.aot.enabled`
2733
| Indicates the application should run with AOT generated artifacts. See
2834
xref:core/aot.adoc[Ahead of Time Optimizations] and
@@ -32,7 +38,7 @@ for details.
3238
| `spring.beaninfo.ignore`
3339
| Instructs Spring to use the `Introspector.IGNORE_ALL_BEANINFO` mode when calling the
3440
JavaBeans `Introspector`. See
35-
{spring-framework-api}++/beans/StandardBeanInfoFactory.html#IGNORE_BEANINFO_PROPERTY_NAME++[`CachedIntrospectionResults`]
41+
{spring-framework-api}++/beans/StandardBeanInfoFactory.html#IGNORE_BEANINFO_PROPERTY_NAME++[`StandardBeanInfoFactory`]
3642
for details.
3743

3844
| `spring.cache.reactivestreams.ignore`
@@ -49,15 +55,13 @@ for details.
4955

5056
| `spring.context.checkpoint`
5157
| Property that specifies a common context checkpoint. See
52-
xref:integration/checkpoint-restore.adoc#_automatic_checkpointrestore_at_startup[Automatic
53-
checkpoint/restore at startup] and
58+
xref:integration/checkpoint-restore.adoc#_automatic_checkpointrestore_at_startup[Automatic checkpoint/restore at startup] and
5459
{spring-framework-api}++/context/support/DefaultLifecycleProcessor.html#CHECKPOINT_PROPERTY_NAME++[`DefaultLifecycleProcessor`]
5560
for details.
5661

5762
| `spring.context.exit`
5863
| Property for terminating the JVM when the context reaches a specific phase. See
59-
xref:integration/checkpoint-restore.adoc#_automatic_checkpointrestore_at_startup[Automatic
60-
checkpoint/restore at startup] and
64+
xref:integration/checkpoint-restore.adoc#_automatic_checkpointrestore_at_startup[Automatic checkpoint/restore at startup] and
6165
{spring-framework-api}++/context/support/DefaultLifecycleProcessor.html#EXIT_PROPERTY_NAME++[`DefaultLifecycleProcessor`]
6266
for details.
6367

spring-aop/src/main/java/org/springframework/aop/aspectj/annotation/AbstractAspectJAdvisorFactory.java

+31-1
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@
1818

1919
import java.lang.annotation.Annotation;
2020
import java.lang.reflect.Constructor;
21+
import java.lang.reflect.Field;
2122
import java.lang.reflect.Method;
2223
import java.util.Map;
2324
import java.util.StringTokenizer;
@@ -37,6 +38,7 @@
3738

3839
import org.springframework.aop.framework.AopConfigException;
3940
import org.springframework.core.ParameterNameDiscoverer;
41+
import org.springframework.core.SpringProperties;
4042
import org.springframework.core.annotation.AnnotationUtils;
4143
import org.springframework.lang.Nullable;
4244

@@ -58,6 +60,23 @@ public abstract class AbstractAspectJAdvisorFactory implements AspectJAdvisorFac
5860
private static final Class<?>[] ASPECTJ_ANNOTATION_CLASSES = new Class<?>[] {
5961
Pointcut.class, Around.class, Before.class, After.class, AfterReturning.class, AfterThrowing.class};
6062

63+
private static final String AJC_MAGIC = "ajc$";
64+
65+
/**
66+
* System property that instructs Spring to ignore ajc-compiled aspects
67+
* for Spring AOP proxying, restoring traditional Spring behavior for
68+
* scenarios where both weaving and AspectJ auto-proxying are enabled.
69+
* <p>The default is "false". Consider switching this to "true" if you
70+
* encounter double execution of your aspects in a given build setup.
71+
* Note that we recommend restructuring your AspectJ configuration to
72+
* avoid such double exposure of an AspectJ aspect to begin with.
73+
* @since 6.1.15
74+
*/
75+
public static final String IGNORE_AJC_PROPERTY_NAME = "spring.aop.ajc.ignore";
76+
77+
private static final boolean shouldIgnoreAjcCompiledAspects =
78+
SpringProperties.getFlag(IGNORE_AJC_PROPERTY_NAME);
79+
6180

6281
/** Logger available to subclasses. */
6382
protected final Log logger = LogFactory.getLog(getClass());
@@ -67,7 +86,8 @@ public abstract class AbstractAspectJAdvisorFactory implements AspectJAdvisorFac
6786

6887
@Override
6988
public boolean isAspect(Class<?> clazz) {
70-
return (AnnotationUtils.findAnnotation(clazz, Aspect.class) != null);
89+
return (AnnotationUtils.findAnnotation(clazz, Aspect.class) != null &&
90+
(!shouldIgnoreAjcCompiledAspects || !compiledByAjc(clazz)));
7191
}
7292

7393
@Override
@@ -114,6 +134,16 @@ private static AspectJAnnotation findAnnotation(Method method, Class<? extends A
114134
}
115135
}
116136

137+
private static boolean compiledByAjc(Class<?> clazz) {
138+
for (Field field : clazz.getDeclaredFields()) {
139+
System.out.println(clazz + ": " + field.getName());
140+
if (field.getName().startsWith(AJC_MAGIC)) {
141+
return true;
142+
}
143+
}
144+
return false;
145+
}
146+
117147

118148
/**
119149
* Enum for AspectJ annotation types.

0 commit comments

Comments
 (0)