/*
 * Copyright 2012-2019 the original author or 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
 *
 *      https://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 org.springframework.boot.configurationprocessor;

import java.util.Map;

import javax.lang.model.element.AnnotationMirror;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.Modifier;
import javax.lang.model.element.TypeElement;
import javax.lang.model.element.VariableElement;
import javax.lang.model.type.TypeMirror;

import org.springframework.boot.configurationprocessor.metadata.ItemDeprecation;

A PropertyDescriptor for a Lombok field.
Author:Stephane Nicoll
/** * A {@link PropertyDescriptor} for a Lombok field. * * @author Stephane Nicoll */
class LombokPropertyDescriptor extends PropertyDescriptor<VariableElement> { private static final String LOMBOK_DATA_ANNOTATION = "lombok.Data"; private static final String LOMBOK_GETTER_ANNOTATION = "lombok.Getter"; private static final String LOMBOK_SETTER_ANNOTATION = "lombok.Setter"; private static final String LOMBOK_ACCESS_LEVEL_PUBLIC = "PUBLIC"; LombokPropertyDescriptor(TypeElement typeElement, ExecutableElement factoryMethod, VariableElement field, String name, TypeMirror type, ExecutableElement getter, ExecutableElement setter) { super(typeElement, factoryMethod, field, name, type, field, getter, setter); } @Override protected boolean isProperty(MetadataGenerationEnvironment env) { if (!hasLombokPublicAccessor(env, true)) { return false; } boolean isCollection = env.getTypeUtils().isCollectionOrMap(getType()); return !env.isExcluded(getType()) && (hasSetter(env) || isCollection); } @Override protected Object resolveDefaultValue(MetadataGenerationEnvironment environment) { return environment.getFieldDefaultValue(getOwnerElement(), getName()); } @Override protected boolean isNested(MetadataGenerationEnvironment environment) { if (!hasLombokPublicAccessor(environment, true)) { return false; } return super.isNested(environment); } @Override protected ItemDeprecation resolveItemDeprecation(MetadataGenerationEnvironment environment) { boolean deprecated = environment.isDeprecated(getField()) || environment.isDeprecated(getGetter()) || environment.isDeprecated(getFactoryMethod()); return deprecated ? environment.resolveItemDeprecation(getGetter()) : null; } private boolean hasSetter(MetadataGenerationEnvironment env) { boolean nonFinalPublicField = !getField().getModifiers().contains(Modifier.FINAL) && hasLombokPublicAccessor(env, false); return getSetter() != null || nonFinalPublicField; }
Determine if the current field defines a public accessor using lombok annotations.
Params:
Returns:true if this field has a public accessor of the specified type
/** * Determine if the current {@link #getField() field} defines a public accessor using * lombok annotations. * @param env the {@link MetadataGenerationEnvironment} * @param getter {@code true} to look for the read accessor, {@code false} for the * write accessor * @return {@code true} if this field has a public accessor of the specified type */
private boolean hasLombokPublicAccessor(MetadataGenerationEnvironment env, boolean getter) { String annotation = (getter ? LOMBOK_GETTER_ANNOTATION : LOMBOK_SETTER_ANNOTATION); AnnotationMirror lombokMethodAnnotationOnField = env.getAnnotation(getField(), annotation); if (lombokMethodAnnotationOnField != null) { return isAccessLevelPublic(env, lombokMethodAnnotationOnField); } AnnotationMirror lombokMethodAnnotationOnElement = env.getAnnotation(getOwnerElement(), annotation); if (lombokMethodAnnotationOnElement != null) { return isAccessLevelPublic(env, lombokMethodAnnotationOnElement); } return (env.getAnnotation(getOwnerElement(), LOMBOK_DATA_ANNOTATION) != null); } private boolean isAccessLevelPublic(MetadataGenerationEnvironment env, AnnotationMirror lombokAnnotation) { Map<String, Object> values = env.getAnnotationElementValues(lombokAnnotation); Object value = values.get("value"); return (value == null || value.toString().equals(LOMBOK_ACCESS_LEVEL_PUBLIC)); } }