com.google.auto.value.processor.PropertyBuilderClassifier Maven / Gradle / Ivy
/*
* Copyright 2016 Google LLC
*
* 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 com.google.auto.value.processor;
import static java.util.stream.Collectors.collectingAndThen;
import static java.util.stream.Collectors.joining;
import static java.util.stream.Collectors.toMap;
import com.google.auto.common.MoreElements;
import com.google.auto.common.MoreTypes;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.function.Function;
import java.util.function.Predicate;
import javax.lang.model.element.ElementKind;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.Modifier;
import javax.lang.model.element.TypeElement;
import javax.lang.model.type.DeclaredType;
import javax.lang.model.type.TypeKind;
import javax.lang.model.type.TypeMirror;
import javax.lang.model.util.ElementFilter;
import javax.lang.model.util.Elements;
import javax.lang.model.util.Types;
/**
* Classifies methods inside builder types that return builders for properties. For example, if
* {@code @AutoValue} class Foo has a method {@code ImmutableList bar()} then Foo.Builder
* can have a method {@code ImmutableList.Builder barBuilder()}. This class checks that a
* method like {@code barBuilder()} follows the rules, and if so constructs a {@link
* PropertyBuilder} instance with information about {@code barBuilder}.
*
* @author Éamonn McManus
*/
class PropertyBuilderClassifier {
private final ErrorReporter errorReporter;
private final Types typeUtils;
private final Elements elementUtils;
private final BuilderMethodClassifier> builderMethodClassifier;
private final Predicate propertyIsNullable;
private final ImmutableMap propertyTypes;
private final Nullables nullables;
PropertyBuilderClassifier(
ErrorReporter errorReporter,
Types typeUtils,
Elements elementUtils,
BuilderMethodClassifier> builderMethodClassifier,
Predicate propertyIsNullable,
ImmutableMap propertyTypes,
Nullables nullables) {
this.errorReporter = errorReporter;
this.typeUtils = typeUtils;
this.elementUtils = elementUtils;
this.builderMethodClassifier = builderMethodClassifier;
this.propertyIsNullable = propertyIsNullable;
this.propertyTypes = propertyTypes;
this.nullables = nullables;
}
/**
* Information about a property builder, referenced from the autovalue.vm template. A property
* called bar (defined by a method bar() or getBar()) can have a property builder called
* barBuilder(). For example, if {@code bar()} returns {@code ImmutableSet} then {@code
* barBuilder()} might return {@code ImmutableSet.Builder}.
*/
public static class PropertyBuilder {
private final ExecutableElement propertyBuilderMethod;
private final String name;
private final String builderType;
private final String nullableBuilderType;
private final AnnotatedTypeMirror builderAnnotatedType;
private final String build;
private final String initializer;
private final String beforeInitDefault;
private final String initDefault;
private final String builtToBuilder;
private final String copyAll;
PropertyBuilder(
ExecutableElement propertyBuilderMethod,
String builderType,
String nullableBuilderType,
AnnotatedTypeMirror builderAnnotatedType,
String build,
String initializer,
String beforeInitDefault,
String initDefault,
String builtToBuilder,
String copyAll) {
this.propertyBuilderMethod = propertyBuilderMethod;
this.name = propertyBuilderMethod.getSimpleName() + "$";
this.builderType = builderType;
this.nullableBuilderType = nullableBuilderType;
this.builderAnnotatedType = builderAnnotatedType;
this.build = build;
this.initializer = initializer;
this.beforeInitDefault = beforeInitDefault;
this.initDefault = initDefault;
this.builtToBuilder = builtToBuilder;
this.copyAll = copyAll;
}
/** The property builder method, for example {@code barBuilder()}. */
public ExecutableElement getPropertyBuilderMethod() {
return propertyBuilderMethod;
}
/** The name of the property builder method. */
public String getMethodName() {
return propertyBuilderMethod.getSimpleName().toString();
}
/** The property builder method parameters, for example {@code Comparator comparator} */
public String getPropertyBuilderMethodParameters() {
return propertyBuilderMethod.getParameters().stream()
.map(
parameter -> TypeEncoder.encode(parameter.asType()) + " " + parameter.getSimpleName())
.collect(joining(", "));
}
public String getAccess() {
return SimpleMethod.access(propertyBuilderMethod);
}
/** The name of the field to hold this builder. */
public String getName() {
return name;
}
/** The type of the builder, for example {@code ImmutableSet.Builder}. */
public String getBuilderType() {
return builderType;
}
/** The type of the builder with an appropriate {@code @Nullable} type annotation. */
public String getNullableBuilderType() {
return nullableBuilderType;
}
TypeMirror getBuilderTypeMirror() {
return builderAnnotatedType.getType();
}
/** The name of the build method, {@code build} or {@code buildOrThrow}. */
public String getBuild() {
return build;
}
/** An initializer for the builder field, for example {@code ImmutableSet.builder()}. */
public String getInitializer() {
return initializer;
}
/**
* An empty string, or a complete statement to be included before the expression returned by
* {@link #getInitDefault()}.
*/
public String getBeforeInitDefault() {
return beforeInitDefault;
}
/**
* An expression to return a default instance of the type that this builder builds. For example,
* if this is an {@code ImmutableList} then the method {@code ImmutableList.of()} will
* correctly return an empty {@code ImmutableList}, assuming the appropriate context for
* type inference. The expression here can assume that the statement from {@link
* #getBeforeInitDefault} has preceded it.
*/
public String getInitDefault() {
return initDefault;
}
/**
* A method to convert the built type back into a builder. Unfortunately Guava collections don't
* have this (you can't say {@code myImmutableMap.toBuilder()}), but for other types such as
* {@code @AutoValue} types this is {@code toBuilder()}.
*/
public String getBuiltToBuilder() {
return builtToBuilder;
}
/**
* The method to copy another collection into this builder. It is {@code addAll} for
* one-dimensional collections like {@code ImmutableList} and {@code ImmutableSet}, and it is
* {@code putAll} for two-dimensional collections like {@code ImmutableMap} and {@code
* ImmutableTable}.
*/
public String getCopyAll() {
return copyAll;
}
}
// Our @AutoValue class `Foo` has a property `Bar bar()` or `Bar getBar()` and we've encountered
// a builder method like `BarBuilder barBuilder()`. Here `BarBuilder` can have any name (its name
// doesn't have to be the name of `Bar` with `Builder` stuck on the end), but `barBuilder()` does
// have to be the name of the property with `Builder` stuck on the end. The requirements for the
// `BarBuilder` type are:
// (1) It must have an instance method called `build()` or `buildOrThrow() that returns `Bar`. If
// the type of `bar()` is `Bar` then the type of the build method must be
// `Bar`.
// (2) `BarBuilder` must have a public no-arg constructor, or `Bar` must have a static method
// `naturalOrder(), `builder()`, or `newBuilder()` that returns `BarBuilder`. The
// `naturalOrder()` case is specifically for ImmutableSortedSet and ImmutableSortedMap.
// (3) If `Foo` has a `toBuilder()` method, or if we have both `barBuilder()` and `setBar(Bar)`
// methods, then `Bar` must have an instance method `BarBuilder toBuilder()`, or `BarBuilder`
// must have an `addAll` or `putAll` method that accepts an argument of type `Bar`.
//
// This method outputs an error and returns Optional.empty() if the barBuilder() method has a
// problem.
Optional makePropertyBuilder(ExecutableElement method, String property) {
AnnotatedTypeMirror barBuilderAnnotatedType = builderMethodClassifier.builderMethodReturnType(method);
if (barBuilderAnnotatedType.getKind() != TypeKind.DECLARED) {
errorReporter.reportError(
method,
"[AutoValueOddBuilderMethod] Method looks like a property builder, but its return type"
+ " is not a class or interface");
return Optional.empty();
}
DeclaredType barBuilderDeclaredType = MoreTypes.asDeclared(barBuilderAnnotatedType.getType());
TypeElement barBuilderTypeElement = MoreTypes.asTypeElement(barBuilderDeclaredType);
Map barBuilderNoArgMethods = noArgMethodsOf(barBuilderTypeElement);
TypeMirror barTypeMirror = propertyTypes.get(property).getType();
if (barTypeMirror.getKind() != TypeKind.DECLARED) {
errorReporter.reportError(
method,
"[AutoValueBadBuilderMethod] Method looks like a property builder, but the type of"
+ " property %s is not a class or interface",
property);
return Optional.empty();
}
if (propertyIsNullable.test(property)) {
errorReporter.reportError(
method,
"[AutoValueNullBuilder] Property %s is @Nullable so it cannot have a property builder",
property);
}
TypeElement barTypeElement = MoreTypes.asTypeElement(barTypeMirror);
Map barNoArgMethods = noArgMethodsOf(barTypeElement);
// Condition (1), must have build() or buildOrThrow() method returning Bar.
ExecutableElement build = barBuilderNoArgMethods.get("buildOrThrow");
if (build == null) {
build = barBuilderNoArgMethods.get("build");
}
if (build == null || build.getModifiers().contains(Modifier.STATIC)) {
errorReporter.reportError(
method,
"[AutoValueBuilderNotBuildable] Method looks like a property builder, but it returns %s"
+ " which does not have a non-static build() or buildOrThrow() method",
barBuilderTypeElement);
return Optional.empty();
}
// We've determined that `BarBuilder` has a method `build()` or `buildOrThrow(). But it must
// return `Bar`. And if the type of `bar()` is Bar then `BarBuilder.build()` must return
// something that can be assigned to Bar.
TypeMirror buildType =
MethodSignature.asMemberOf(typeUtils, barBuilderDeclaredType, build).returnType().getType();
if (!typeUtils.isAssignable(buildType, barTypeMirror)) {
errorReporter.reportError(
method,
"[AutoValueBuilderWrongType] Property builder for %s has type %s whose %s() method"
+ " returns %s instead of %s",
property,
barBuilderTypeElement,
build.getSimpleName(),
buildType,
barTypeMirror);
return Optional.empty();
}
Optional maybeBuilderMaker;
if (method.getParameters().isEmpty()) {
maybeBuilderMaker = noArgBuilderMaker(barNoArgMethods, barBuilderTypeElement);
} else {
Map barOneArgMethods = oneArgumentMethodsOf(barTypeElement);
maybeBuilderMaker = oneArgBuilderMaker(barOneArgMethods, barBuilderTypeElement);
}
if (!maybeBuilderMaker.isPresent()) {
errorReporter.reportError(
method,
"[AutoValueCantMakePropertyBuilder] Method looks like a property builder, but its type"
+ " %s does not have a public constructor and %s does not have a static builder() or"
+ " newBuilder() method that returns %s",
barBuilderTypeElement,
barTypeElement,
barBuilderTypeElement);
return Optional.empty();
}
ExecutableElement builderMaker = maybeBuilderMaker.get();
String barBuilderType = TypeEncoder.encodeWithAnnotations(barBuilderAnnotatedType);
String nullableBarBuilderType =
TypeEncoder.encodeWithAnnotations(
barBuilderAnnotatedType, nullables.nullableTypeAnnotations());
String rawBarType = TypeEncoder.encodeRaw(barTypeMirror);
String arguments =
method.getParameters().isEmpty()
? "()"
: "(" + method.getParameters().get(0).getSimpleName() + ")";
String initializer =
(builderMaker.getKind() == ElementKind.CONSTRUCTOR)
? "new " + barBuilderType + arguments
: rawBarType + "." + builderMaker.getSimpleName() + arguments;
String builtToBuilder = null;
String copyAll = null;
ExecutableElement toBuilder = barNoArgMethods.get("toBuilder");
if (toBuilder != null
&& !toBuilder.getModifiers().contains(Modifier.STATIC)
&& typeUtils.isAssignable(
typeUtils.erasure(toBuilder.getReturnType()),
typeUtils.erasure(barBuilderDeclaredType))) {
builtToBuilder = toBuilder.getSimpleName().toString();
} else {
Optional maybeCopyAll =
addAllPutAll(barBuilderTypeElement, barBuilderDeclaredType, barTypeMirror);
if (maybeCopyAll.isPresent()) {
copyAll = maybeCopyAll.get().getSimpleName().toString();
}
}
ExecutableElement barOf = barNoArgMethods.get("of");
boolean hasOf = (barOf != null && barOf.getModifiers().contains(Modifier.STATIC));
// An expression (initDefault) to make a default one of these, plus optionally a statement
// (beforeInitDefault) that prepares the expression. For a collection, beforeInitDefault is
// empty and initDefault is (e.g.) `ImmutableList.of()`. For a nested value type,
// beforeInitDefault is (e.g.)
// `NestedAutoValueType.Builder foo$builder = NestedAutoValueType.builder();`
// and initDefault is `foo$builder.build();`. The reason for the separate statement is to
// exploit type inference rather than having to write `NestedAutoValueType.build();`.
String beforeInitDefault;
String initDefault;
if (hasOf) {
beforeInitDefault = "";
initDefault = rawBarType + ".of()";
} else {
String localBuilder = property + "$builder";
beforeInitDefault = nullableBarBuilderType + " " + localBuilder + " = " + initializer + ";";
initDefault = localBuilder + "." + build.getSimpleName() + "()";
}
PropertyBuilder propertyBuilder =
new PropertyBuilder(
method,
barBuilderType,
nullableBarBuilderType,
barBuilderAnnotatedType,
build.getSimpleName().toString(),
initializer,
beforeInitDefault,
initDefault,
builtToBuilder,
copyAll);
return Optional.of(propertyBuilder);
}
private static final ImmutableSet BUILDER_METHOD_NAMES =
ImmutableSet.of("naturalOrder", "builder", "newBuilder");
// (2) `BarBuilder` must have a public no-arg constructor, or `Bar` must have a visible static
// method `naturalOrder(), `builder()`, or `newBuilder()` that returns `BarBuilder`; or,
// if we have a foosBuilder(T) method, then `BarBuilder` must have a public constructor with
// a single parameter assignable from T, or a visible static `builder(T)` method.
private Optional noArgBuilderMaker(
Map barNoArgMethods, TypeElement barBuilderTypeElement) {
return builderMaker(BUILDER_METHOD_NAMES, barNoArgMethods, barBuilderTypeElement, 0);
}
private static final ImmutableSet ONE_ARGUMENT_BUILDER_METHOD_NAMES =
ImmutableSet.of("builder");
private Optional oneArgBuilderMaker(
Map barOneArgMethods, TypeElement barBuilderTypeElement) {
return builderMaker(
ONE_ARGUMENT_BUILDER_METHOD_NAMES, barOneArgMethods, barBuilderTypeElement, 1);
}
private Optional builderMaker(
ImmutableSet methodNamesToCheck,
Map methods,
TypeElement barBuilderTypeElement,
int argumentCount) {
Optional maybeMethod =
methodNamesToCheck.stream()
.map(methods::get)
.filter(Objects::nonNull)
.filter(method -> method.getModifiers().contains(Modifier.STATIC))
.filter(
method ->
typeUtils.isSameType(
typeUtils.erasure(method.getReturnType()),
typeUtils.erasure(barBuilderTypeElement.asType())))
.findFirst();
if (maybeMethod.isPresent()) {
// TODO(emcmanus): check visibility. We don't want to require public for @AutoValue
// builders. By not checking visibility we risk accepting something as a builder maker
// and then failing when the generated code tries to call Bar.builder(). But the risk
// seems small.
return maybeMethod;
}
return ElementFilter.constructorsIn(barBuilderTypeElement.getEnclosedElements()).stream()
.filter(c -> c.getParameters().size() == argumentCount)
.filter(c -> c.getModifiers().contains(Modifier.PUBLIC))
.findFirst();
}
private Map noArgMethodsOf(TypeElement type) {
return methodsOf(type, 0);
}
private ImmutableMap oneArgumentMethodsOf(TypeElement type) {
return methodsOf(type, 1);
}
private ImmutableMap methodsOf(TypeElement type, int argumentCount) {
return ElementFilter.methodsIn(elementUtils.getAllMembers(type)).stream()
.filter(method -> method.getParameters().size() == argumentCount)
.filter(method -> !isStaticInterfaceMethodNotIn(method, type))
.collect(
collectingAndThen(
toMap(
method -> method.getSimpleName().toString(),
Function.identity(),
(method1, method2) -> method1),
ImmutableMap::copyOf));
}
// Work around an Eclipse compiler bug: https://bugs.eclipse.org/bugs/show_bug.cgi?id=547185
// The result of Elements.getAllMembers includes static methods declared in superinterfaces.
// That's wrong because those aren't inherited. So this method checks whether the given method is
// a static interface method not in the given type.
private static boolean isStaticInterfaceMethodNotIn(ExecutableElement method, TypeElement type) {
return method.getModifiers().contains(Modifier.STATIC)
&& !method.getEnclosingElement().equals(type)
&& method.getEnclosingElement().getKind().equals(ElementKind.INTERFACE);
}
private static final ImmutableSet ADD_ALL_PUT_ALL = ImmutableSet.of("addAll", "putAll");
// We have `Bar bar()` and `Foo.Builder toBuilder()` in the @AutoValue type Foo, and we have
// `BarBuilder barBuilder()` in Foo.Builder. That means that we need to be able to make a
// `BarBuilder` from a `Bar` as part of the implementation of `Foo.toBuilder()`. We can do that
// if `Bar` has a method `BarBuilder toBuilder()`, but what if it doesn't? For example, Guava's
// `ImmutableList` doesn't have a method `ImmutableList.Builder toBuilder()`. So we also allow it
// to work if `BarBuilder` has a method `addAll(T)` or `putAll(T)`, where `Bar` is assignable to
// `T`. `ImmutableList.Builder` does have a method `addAll(Iterable extends E>)` and
// `ImmutableList` is assignable to `Iterable extends E>`, so that works.
private Optional addAllPutAll(
TypeElement barBuilderTypeElement,
DeclaredType barBuilderDeclaredType,
TypeMirror barTypeMirror) {
return MoreElements.getLocalAndInheritedMethods(barBuilderTypeElement, typeUtils, elementUtils)
.stream()
.filter(
method ->
ADD_ALL_PUT_ALL.contains(method.getSimpleName().toString())
&& method.getParameters().size() == 1)
.filter(
method -> {
TypeMirror parameterType =
MethodSignature.asMemberOf(typeUtils, barBuilderDeclaredType, method)
.parameterTypes()
.get(0)
.getType();
return typeUtils.isAssignable(barTypeMirror, parameterType);
})
.findFirst();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy