io.micronaut.annotation.processing.JavaConfigurationMetadataBuilder Maven / Gradle / Ivy
/*
* Copyright 2017-2018 original authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package io.micronaut.annotation.processing;
import io.micronaut.context.annotation.ConfigurationReader;
import io.micronaut.context.annotation.EachProperty;
import io.micronaut.core.annotation.AnnotationMetadata;
import io.micronaut.core.util.StringUtils;
import io.micronaut.inject.configuration.ConfigurationMetadataBuilder;
import javax.lang.model.element.Element;
import javax.lang.model.element.NestingKind;
import javax.lang.model.element.TypeElement;
import javax.lang.model.type.DeclaredType;
import javax.lang.model.type.TypeMirror;
import javax.lang.model.util.Elements;
import javax.lang.model.util.Types;
import java.util.Optional;
import java.util.function.Function;
/**
* Implementation of {@link ConfigurationMetadataBuilder} for Java.
*
* @author Graeme Rocher
* @see ConfigurationMetadataBuilder
* @since 1.0
*/
public class JavaConfigurationMetadataBuilder extends ConfigurationMetadataBuilder {
private final AnnotationUtils annotationUtils;
private final ModelUtils modelUtils;
private final Elements elements;
/**
* @param elements The {@link Elements}
* @param types The {@link Types}
* @param annotationUtils The annotation utils
*/
public JavaConfigurationMetadataBuilder(Elements elements, Types types, AnnotationUtils annotationUtils) {
this.elements = elements;
this.annotationUtils = annotationUtils;
this.modelUtils = new ModelUtils(elements, types);
// ensure initialization
final TypeElement crte = elements.getTypeElement(ConfigurationReader.class.getName());
if (crte != null) {
this.annotationUtils.getAnnotationMetadata(crte);
}
final TypeElement epte = elements.getTypeElement(EachProperty.class.getName());
if (epte != null) {
this.annotationUtils.getAnnotationMetadata(epte);
}
}
/**
* @return The {@link Elements}
*/
public Elements getElements() {
return elements;
}
@Override
protected String buildPropertyPath(TypeElement owningType, TypeElement declaringType, String propertyName) {
String value = buildTypePath(owningType, declaringType);
return value + '.' + propertyName;
}
@Override
protected String buildTypePath(TypeElement owningType, TypeElement declaringType) {
String initialPath = calculateInitialPath(owningType, declaringType);
StringBuilder path = new StringBuilder(initialPath);
prependSuperclasses(declaringType, path);
if (owningType.getNestingKind() == NestingKind.MEMBER) {
// we have an inner class, so prepend inner class
Element enclosingElement = owningType.getEnclosingElement();
if (enclosingElement instanceof TypeElement) {
TypeElement enclosingType = (TypeElement) enclosingElement;
while (true) {
AnnotationMetadata enclosingTypeMetadata = annotationUtils.getAnnotationMetadata(enclosingType);
Optional parentConfig = enclosingTypeMetadata.getValue(ConfigurationReader.class, String.class);
if (parentConfig.isPresent()) {
String parentPath = pathEvaluationFunctionForMetadata(enclosingTypeMetadata).apply(parentConfig.get());
path.insert(0, parentPath + '.');
prependSuperclasses(enclosingType, path);
if (enclosingType.getNestingKind() == NestingKind.MEMBER) {
Element el = enclosingType.getEnclosingElement();
if (el instanceof TypeElement) {
enclosingType = (TypeElement) el;
} else {
break;
}
} else {
break;
}
} else {
break;
}
}
}
}
return path.toString();
}
private String calculateInitialPath(TypeElement owningType, TypeElement declaringType) {
AnnotationMetadata annotationMetadata = annotationUtils.getAnnotationMetadata(declaringType);
Function evaluatePathFunction = pathEvaluationFunctionForMetadata(annotationMetadata);
return annotationMetadata.getValue(ConfigurationReader.class, String.class)
.map(evaluatePathFunction)
.orElseGet(() -> {
AnnotationMetadata ownerMetadata = annotationUtils.getAnnotationMetadata(owningType);
return ownerMetadata
.getValue(ConfigurationReader.class, String.class)
.map(pathEvaluationFunctionForMetadata(ownerMetadata))
.orElseGet(() ->
pathEvaluationFunctionForMetadata(annotationMetadata).apply("")
);
}
);
}
private Function pathEvaluationFunctionForMetadata(AnnotationMetadata annotationMetadata) {
return path -> {
if (annotationMetadata.hasDeclaredAnnotation(EachProperty.class)) {
return path + ".*";
}
String prefix = annotationMetadata.getValue(ConfigurationReader.class, "prefix", String.class)
.orElse(null);
if (StringUtils.isNotEmpty(prefix)) {
if (StringUtils.isEmpty(path)) {
return prefix;
} else {
return prefix + "." + path;
}
} else {
return path;
}
};
}
@Override
protected String getTypeString(TypeElement type) {
return modelUtils.resolveTypeReference(type).toString();
}
private void prependSuperclasses(TypeElement declaringType, StringBuilder path) {
TypeMirror superclass = declaringType.getSuperclass();
while (superclass instanceof DeclaredType) {
DeclaredType declaredType = (DeclaredType) superclass;
Element element = declaredType.asElement();
AnnotationMetadata annotationMetadata = annotationUtils.getAnnotationMetadata(element);
Optional parentConfig = annotationMetadata.getValue(ConfigurationReader.class, String.class);
if (parentConfig.isPresent()) {
String parentPath = pathEvaluationFunctionForMetadata(annotationMetadata).apply(parentConfig.get());
path.insert(0, parentPath + '.');
superclass = ((TypeElement) element).getSuperclass();
} else {
break;
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy