diff --git a/core-processor/src/main/java/io/micronaut/inject/ast/FieldElement.java b/core-processor/src/main/java/io/micronaut/inject/ast/FieldElement.java index 8689aaeffc3..2bb9e1d9824 100644 --- a/core-processor/src/main/java/io/micronaut/inject/ast/FieldElement.java +++ b/core-processor/src/main/java/io/micronaut/inject/ast/FieldElement.java @@ -26,6 +26,30 @@ */ public interface FieldElement extends TypedElement, MemberElement { + /** + * Returns the value of this variable if this is a {@code final} + * field initialized to a compile-time constant. Returns {@code + * null} otherwise. The value will be of a primitive type or a + * {@code String}. If the value is of a primitive type, it is + * wrapped in the appropriate wrapper class (such as {@link + * Integer}). + * + *
Note that not all {@code final} fields will have + * constant values. In particular, {@code enum} constants are + * not considered to be compile-time constants. To have a + * constant value, a field's type must be either a primitive type + * or {@code String}. + * + * @return the value of this variable if this is a {@code final} + * field initialized to a compile-time constant, or {@code null} + * otherwise + * + * @since 4.5.0 + */ + default Object getConstantValue() { + return null; + } + /** * Obtain the generic type with the associated annotation metadata for the field. * diff --git a/gradle.properties b/gradle.properties index 41276cbc523..6da77fc45d6 100644 --- a/gradle.properties +++ b/gradle.properties @@ -1,4 +1,4 @@ -projectVersion=4.4.2-SNAPSHOT +projectVersion=4.5.0-SNAPSHOT projectGroupId=io.micronaut projectDesc=Core components supporting the Micronaut Framework title=Micronaut Core diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index 5bd1e8d517d..b5e30577ebd 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -31,7 +31,7 @@ jazzer = "0.22.1" jcache = "1.1.1" junit5 = "5.10.2" junit-platform="1.10.2" -logback = "1.5.3" +logback = "1.5.4" logbook-netty = "2.16.0" log4j = "2.22.1" micronaut-aws = "4.5.0" diff --git a/inject-groovy/src/main/groovy/io/micronaut/ast/groovy/visitor/GroovyFieldElement.java b/inject-groovy/src/main/groovy/io/micronaut/ast/groovy/visitor/GroovyFieldElement.java index 403619a0653..04e645fdb76 100644 --- a/inject-groovy/src/main/groovy/io/micronaut/ast/groovy/visitor/GroovyFieldElement.java +++ b/inject-groovy/src/main/groovy/io/micronaut/ast/groovy/visitor/GroovyFieldElement.java @@ -25,6 +25,7 @@ import io.micronaut.inject.ast.annotation.ElementAnnotationMetadataFactory; import org.codehaus.groovy.ast.ClassNode; import org.codehaus.groovy.ast.FieldNode; +import org.codehaus.groovy.ast.expr.ConstantExpression; import java.lang.reflect.Modifier; import java.util.Map; @@ -149,6 +150,15 @@ public boolean isPackagePrivate() { return !Modifier.isPublic(fieldNode.getModifiers()) && !Modifier.isProtected(fieldNode.getModifiers()) && !Modifier.isPrivate(fieldNode.getModifiers()); } + @Override + public Object getConstantValue() { + if (fieldNode.hasInitialExpression() + && fieldNode.getInitialValueExpression() instanceof ConstantExpression constExpression) { + return constExpression.getValue(); + } + return null; + } + @NonNull @Override public ClassElement getType() { diff --git a/inject-groovy/src/test/groovy/io/micronaut/ast/groovy/visitor/GroovyEnumElementSpec.groovy b/inject-groovy/src/test/groovy/io/micronaut/ast/groovy/visitor/GroovyEnumElementSpec.groovy index e3d8ed3ec41..e61330dc171 100644 --- a/inject-groovy/src/test/groovy/io/micronaut/ast/groovy/visitor/GroovyEnumElementSpec.groovy +++ b/inject-groovy/src/test/groovy/io/micronaut/ast/groovy/visitor/GroovyEnumElementSpec.groovy @@ -76,4 +76,36 @@ enum MyEnum { enumConstantAnnotation.stringValue().get() == "C" } } + + void "get enum constantValue for final fields"() { + given: + def element = (EnumElement) buildClassElement(""" +package test + +enum MyEnum { + + ENUM_VAL1(10), + ENUM_VAL2(11), + UNRECOGNIZED(-1), + ; + + public static final int ENUM_VAL1_VALUE = 10 + public static final int ENUM_VAL2_VALUE = 11 + + private final int value + + MyEnum(int value) { + this.value = value + } +} +""") + expect: + for (def field : element.fields) { + if (field.name == 'ENUM_VAL1_VALUE') { + field.constantValue == 10 + } else if (field.name == 'ENUM_VAL2_VALUE') { + field.constantValue == 11 + } + } + } } diff --git a/inject-java/src/main/java/io/micronaut/annotation/processing/visitor/JavaFieldElement.java b/inject-java/src/main/java/io/micronaut/annotation/processing/visitor/JavaFieldElement.java index d319ed0a66c..a2ec2de4fc9 100644 --- a/inject-java/src/main/java/io/micronaut/annotation/processing/visitor/JavaFieldElement.java +++ b/inject-java/src/main/java/io/micronaut/annotation/processing/visitor/JavaFieldElement.java @@ -86,6 +86,11 @@ public FieldElement withAnnotationMetadata(AnnotationMetadata annotationMetadata return (FieldElement) super.withAnnotationMetadata(annotationMetadata); } + @Override + public Object getConstantValue() { + return variableElement.getConstantValue(); + } + @NonNull @Override public ClassElement getType() { diff --git a/inject-java/src/test/groovy/io/micronaut/annotation/JavaEnumElementSpec.groovy b/inject-java/src/test/groovy/io/micronaut/annotation/JavaEnumElementSpec.groovy index 5565e82ad91..52e2d5c9d6b 100644 --- a/inject-java/src/test/groovy/io/micronaut/annotation/JavaEnumElementSpec.groovy +++ b/inject-java/src/test/groovy/io/micronaut/annotation/JavaEnumElementSpec.groovy @@ -71,4 +71,36 @@ enum MyEnum { expect: element.values().size() == 2; } + + void "get enum constantValue for final fields"() { + given: + def element = (EnumElement) buildClassElement(""" +package test; + +enum MyEnum { + + ENUM_VAL1(10), + ENUM_VAL2(11), + UNRECOGNIZED(-1), + ; + + public static final int ENUM_VAL1_VALUE = 10; + public static final int ENUM_VAL2_VALUE = 11; + + private final int value; + + MyEnum(int value) { + this.value = value; + } +} +""") + expect: + for (def field : element.fields) { + if (field.name == 'ENUM_VAL1_VALUE') { + field.constantValue == 10 + } else if (field.name == 'ENUM_VAL2_VALUE') { + field.constantValue == 11 + } + } + } } diff --git a/inject-kotlin/src/test/groovy/io/micronaut/kotlin/processing/ast/visitor/KotlinEnumElementSpec.groovy b/inject-kotlin/src/test/groovy/io/micronaut/kotlin/processing/ast/visitor/KotlinEnumElementSpec.groovy new file mode 100644 index 00000000000..5cc8e477a99 --- /dev/null +++ b/inject-kotlin/src/test/groovy/io/micronaut/kotlin/processing/ast/visitor/KotlinEnumElementSpec.groovy @@ -0,0 +1,73 @@ +package io.micronaut.kotlin.processing.ast.visitor + +import io.micronaut.annotation.processing.test.AbstractKotlinCompilerSpec +import io.micronaut.inject.ast.ClassElement +import io.micronaut.inject.ast.ElementQuery +import io.micronaut.inject.ast.EnumElement +import spock.lang.Ignore +import spock.lang.PendingFeature + +class KotlinEnumElementSpec extends AbstractKotlinCompilerSpec { + + void "test is enum"() { + given: + def element = buildClassElement("test.MyEnum", """ +package test + +enum class MyEnum { + A, B +} +""") + expect: + element.isEnum() + element.getPrimaryConstructor().get().getDeclaringType().isEnum() + } + + void "test inner enum is enum"() { + given: + def element = buildClassElement("test.Foo",""" +package test + +class Foo { + + enum class MyEnum { + A, B + } +} +""") + expect: + element.getEnclosedElement(ElementQuery.of(ClassElement.class)).get().getPrimaryConstructor().get().getDeclaringType().isEnum() + } + + @Ignore + @PendingFeature(reason = "constantValue not available with KSP") + void "get enum constantValue for final fields"() { + given: + def element = (EnumElement) buildClassElement("test.MyEnum", """ +package test + +enum class MyEnum( + value: Int +) { + + ENUM_VAL1(10), + ENUM_VAL2(11), + UNRECOGNIZED(-1), + ; + + companion object { + const val ENUM_VAL1_VALUE = 10 + const val ENUM_VAL2_VALUE = 11 + } +} +""") + expect: + for (def field : element.fields) { + if (field.name == 'ENUM_VAL1_VALUE') { + field.constantValue == 10 + } else if (field.name == 'ENUM_VAL2_VALUE') { + field.constantValue == 11 + } + } + } +}