All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.auto.value.extension.toprettystring.processor.ToPrettyStringExtension Maven / Gradle / Ivy

There is a newer version: 2.0.32
Show newest version
/*
 * Copyright 2021 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.extension.toprettystring.processor;

import static com.google.auto.common.MoreElements.getLocalAndInheritedMethods;
import static com.google.auto.common.MoreStreams.toImmutableList;
import static com.google.auto.common.MoreTypes.asTypeElement;
import static com.google.auto.value.extension.toprettystring.processor.ExtensionClassTypeSpecBuilder.extensionClassTypeSpecBuilder;
import static com.google.auto.value.extension.toprettystring.processor.ToPrettyStringMethods.toPrettyStringMethod;
import static com.google.auto.value.extension.toprettystring.processor.ToPrettyStringMethods.toPrettyStringMethods;
import static com.google.common.collect.Iterables.getLast;
import static com.google.common.collect.Iterables.getOnlyElement;
import static com.google.common.collect.Sets.intersection;
import static com.squareup.javapoet.MethodSpec.methodBuilder;
import static javax.lang.model.element.Modifier.FINAL;
import static javax.lang.model.element.Modifier.PRIVATE;
import static javax.lang.model.element.Modifier.PROTECTED;
import static javax.lang.model.element.Modifier.PUBLIC;
import static javax.lang.model.element.Modifier.STATIC;

import com.google.auto.common.MoreTypes;
import com.google.auto.service.AutoService;
import com.google.auto.value.extension.AutoValueExtension;
import com.google.auto.value.extension.toprettystring.processor.ToPrettyStringExtension.PrettyPrintableKind.KindVisitor;
import com.google.common.base.Equivalence;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.squareup.javapoet.ClassName;
import com.squareup.javapoet.CodeBlock;
import com.squareup.javapoet.JavaFile;
import com.squareup.javapoet.MethodSpec;
import com.squareup.javapoet.TypeName;
import com.squareup.javapoet.TypeSpec;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.function.Supplier;
import javax.annotation.processing.ProcessingEnvironment;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.Modifier;
import javax.lang.model.element.TypeElement;
import javax.lang.model.type.ArrayType;
import javax.lang.model.type.DeclaredType;
import javax.lang.model.type.PrimitiveType;
import javax.lang.model.type.TypeMirror;
import javax.lang.model.util.Elements;
import javax.lang.model.util.SimpleTypeVisitor8;
import javax.lang.model.util.Types;

/**
 * Generates implementations of {@link
 * com.google.auto.value.extension.toprettystring.ToPrettyString} annotated methods in {@link
 * com.google.auto.value.AutoValue} types.
 */
@AutoService(AutoValueExtension.class)
public final class ToPrettyStringExtension extends AutoValueExtension {
  private static final ImmutableSet INHERITED_VISIBILITY_MODIFIERS =
      ImmutableSet.of(PUBLIC, PROTECTED);
  private static final String INDENT = "  ";
  private static final String INDENT_METHOD_NAME = "$indent";
  private static final CodeBlock KEY_VALUE_SEPARATOR = CodeBlock.of("$S", ": ");

  @Override
  public String generateClass(
      Context context, String className, String classToExtend, boolean isFinal) {
    TypeSpec type =
        extensionClassTypeSpecBuilder(context, className, classToExtend, isFinal)
            .addMethods(toPrettyStringMethodSpecs(context))
            .build();
    return JavaFile.builder(context.packageName(), type)
        .skipJavaLangImports(true)
        .build()
        .toString();
  }

  private ImmutableList toPrettyStringMethodSpecs(Context context) {
    ExecutableElement toPrettyStringMethod = getOnlyElement(toPrettyStringMethods(context));
    MethodSpec.Builder method =
        methodBuilder(toPrettyStringMethod.getSimpleName().toString())
            .addAnnotation(Override.class)
            .returns(ClassName.get(String.class))
            .addModifiers(FINAL)
            .addModifiers(
                intersection(toPrettyStringMethod.getModifiers(), INHERITED_VISIBILITY_MODIFIERS));

    method.addCode("return $S", context.autoValueClass().getSimpleName() + " {");
    ToPrettyStringImplementation implementation = ToPrettyStringImplementation.create(context);
    method.addCode(implementation.toStringCodeBlock.build());

    if (!context.properties().isEmpty()) {
      method.addCode(" + $S", "\n");
    }
    method.addCode(" + $S;\n", "}");

    return ImmutableList.builder()
        .add(method.build())
        .addAll(implementation.delegateMethods.values())
        .add(indentMethod())
        .build();
  }

  private static MethodSpec indentMethod() {
    return methodBuilder(INDENT_METHOD_NAME)
        .addModifiers(PRIVATE, STATIC)
        .returns(ClassName.get(String.class))
        .addParameter(TypeName.INT, "level")
        .addStatement("$1T builder = new $1T()", StringBuilder.class)
        .beginControlFlow("for (int i = 0; i < level; i++)")
        .addStatement("builder.append($S)", INDENT)
        .endControlFlow()
        .addStatement("return builder.toString()")
        .build();
  }

  private static class ToPrettyStringImplementation {
    private final Types types;
    private final Elements elements;

    private final CodeBlock.Builder toStringCodeBlock = CodeBlock.builder();
    private final Map, MethodSpec> delegateMethods =
        new LinkedHashMap<>();
    private final Set methodNames = new HashSet<>();

    private ToPrettyStringImplementation(Context context) {
      this.types = context.processingEnvironment().getTypeUtils();
      this.elements = context.processingEnvironment().getElementUtils();
      // do not submit: what about "inherited" static methods?
      getLocalAndInheritedMethods(context.autoValueClass(), types, elements)
          .forEach(method -> methodNames.add(method.getSimpleName().toString()));
    }

    static ToPrettyStringImplementation create(Context context) {
      ToPrettyStringImplementation implemention = new ToPrettyStringImplementation(context);
      context
          .propertyTypes()
          .forEach(
              (propertyName, type) -> {
                String methodName =
                    context.properties().get(propertyName).getSimpleName().toString();
                implemention.toStringCodeBlock.add(
                    "\n + $S + $L + $S",
                    String.format("\n%s%s = ", INDENT, propertyName),
                    implemention.format(CodeBlock.of("$N()", methodName), CodeBlock.of("1"), type),
                    ",");
              });
      return implemention;
    }

    /**
     * Returns {@code propertyAccess} formatted for use within the {@link
     * com.google.auto.value.extension.toprettystring.ToPrettyString} implementation.
     *
     * 

If a helper method is necessary for formatting, a {@link MethodSpec} will be added to * {@link #delegateMethods}. * * @param propertyAccess a reference to the variable that should be formatted. * @param indentAccess a reference to an {@code int} representing how many indent levels should * be used for this property. * @param type the type of the {@code propertyAccess}. */ private CodeBlock format(CodeBlock propertyAccess, CodeBlock indentAccess, TypeMirror type) { PrettyPrintableKind printableKind = type.accept(new KindVisitor(types, elements), null); DelegateMethod delegateMethod = new DelegateMethod(propertyAccess, indentAccess); switch (printableKind) { case PRIMITIVE: return propertyAccess; case REGULAR_OBJECT: return delegateMethod .methodName("format") .invocation( elements.getTypeElement("java.lang.Object").asType(), () -> reindent("toString")); case HAS_TO_PRETTY_STRING_METHOD: ExecutableElement method = toPrettyStringMethod(asTypeElement(type), types, elements).get(); return delegateMethod.invocation(type, () -> reindent(method.getSimpleName())); case ARRAY: TypeMirror componentType = MoreTypes.asArray(type).getComponentType(); return delegateMethod.invocation(type, () -> forEachLoopMethodBody(componentType)); case COLLECTION: TypeMirror elementType = getOnlyElement(resolvedTypeParameters(type, "java.util.Collection")); return delegateMethod.invocation( collectionOf(elementType), () -> forEachLoopMethodBody(elementType)); case IMMUTABLE_PRIMITIVE_ARRAY: return delegateMethod.invocation(type, this::forLoopMethodBody); case OPTIONAL: case GUAVA_OPTIONAL: TypeMirror optionalType = getOnlyElement(MoreTypes.asDeclared(type).getTypeArguments()); return delegateMethod.invocation( type, () -> optionalMethodBody(optionalType, printableKind)); case MAP: return formatMap(type, delegateMethod); case MULTIMAP: return formatMultimap(type, delegateMethod); } throw new AssertionError(printableKind); } private CodeBlock formatMap(TypeMirror type, DelegateMethod delegateMethod) { ImmutableList typeParameters = resolvedTypeParameters(type, "java.util.Map"); TypeMirror keyType = typeParameters.get(0); TypeMirror valueType = typeParameters.get(1); return delegateMethod.invocation( mapOf(keyType, valueType), () -> mapMethodBody(keyType, valueType)); } private CodeBlock formatMultimap(TypeMirror type, DelegateMethod delegateMethod) { ImmutableList typeParameters = resolvedTypeParameters(type, "com.google.common.collect.Multimap"); TypeMirror keyType = typeParameters.get(0); TypeMirror valueType = typeParameters.get(1); return delegateMethod.invocation( multimapOf(keyType, valueType), () -> multimapMethodBody(keyType, collectionOf(valueType))); } /** * Parameter object to simplify the branches of {@link #format(CodeBlock, CodeBlock, * TypeMirror)} that call a delegate method. */ private class DelegateMethod { private final CodeBlock propertyAccess; private final CodeBlock indentAccess; private Optional methodName = Optional.empty(); DelegateMethod(CodeBlock propertyAccess, CodeBlock indentAccess) { this.propertyAccess = propertyAccess; this.indentAccess = indentAccess; } DelegateMethod methodName(String methodName) { this.methodName = Optional.of(methodName); return this; } CodeBlock invocation(TypeMirror parameterType, Supplier methodBody) { Equivalence.Wrapper key = MoreTypes.equivalence().wrap(parameterType); // This doesn't use putIfAbsent because the methodBody supplier could recursively create // new delegate methods. Map.putIfAbsent doesn't support reentrant calls. if (!delegateMethods.containsKey(key)) { delegateMethods.put( key, createMethod( methodName.orElseGet(() -> newDelegateMethodName(parameterType)), parameterType, methodBody)); } return CodeBlock.of( "$N($L, $L)", delegateMethods.get(key).name, propertyAccess, indentAccess); } private String newDelegateMethodName(TypeMirror type) { String prefix = "format" + nameForType(type); String methodName = prefix; for (int i = 2; !methodNames.add(methodName); i++) { methodName = prefix + i; } return methodName; } private MethodSpec createMethod( String methodName, TypeMirror type, Supplier methodBody) { return methodBuilder(methodName) .addModifiers(PRIVATE, STATIC) .returns(ClassName.get(String.class)) .addParameter(TypeName.get(type), "value") .addParameter(TypeName.INT, "indentLevel") .beginControlFlow("if (value == null)") .addStatement("return $S", "null") .endControlFlow() .addCode(methodBody.get()) .build(); } } private CodeBlock reindent(CharSequence methodName) { return CodeBlock.builder() .addStatement( "return value.$1N().replace($2S, $2S + $3N(indentLevel))", methodName, "\n", INDENT_METHOD_NAME) .build(); } private CodeBlock forEachLoopMethodBody(TypeMirror elementType) { return loopMethodBody( "[", "]", CodeBlock.of("for ($T element : value)", elementType), format(CodeBlock.of("element"), CodeBlock.of("indentLevel + 1"), elementType)); } private CodeBlock forLoopMethodBody() { return loopMethodBody( "[", "]", CodeBlock.of("for (int i = 0; i < value.length(); i++)"), CodeBlock.of("value.get(i)")); } private CodeBlock mapMethodBody(TypeMirror keyType, TypeMirror valueType) { return forEachMapEntryMethodBody(keyType, valueType, "value"); } private CodeBlock multimapMethodBody(TypeMirror keyType, TypeMirror valueType) { return forEachMapEntryMethodBody(keyType, valueType, "value.asMap()"); } private CodeBlock forEachMapEntryMethodBody( TypeMirror keyType, TypeMirror valueType, String propertyAccess) { CodeBlock entryType = CodeBlock.of("$T<$T, $T>", Map.Entry.class, keyType, valueType); return loopMethodBody( "{", "}", CodeBlock.of("for ($L entry : $L.entrySet())", entryType, propertyAccess), format(CodeBlock.of("entry.getKey()"), CodeBlock.of("indentLevel + 1"), keyType), KEY_VALUE_SEPARATOR, format(CodeBlock.of("entry.getValue()"), CodeBlock.of("indentLevel + 1"), valueType)); } private CodeBlock loopMethodBody( String openSymbol, String closeSymbol, CodeBlock loopDeclaration, CodeBlock... appendedValues) { ImmutableList allAppendedValues = ImmutableList.builder() .add(CodeBlock.of("$S", "\n")) .add(CodeBlock.of("$N(indentLevel + 1)", INDENT_METHOD_NAME)) .add(appendedValues) .add(CodeBlock.of("$S", ",")) .build(); return CodeBlock.builder() .addStatement("$1T builder = new $1T().append($2S)", StringBuilder.class, openSymbol) .addStatement("boolean hasElements = false") .beginControlFlow("$L", loopDeclaration) .addStatement( "builder$L", allAppendedValues.stream() .map(value -> CodeBlock.of(".append($L)", value)) .collect(CodeBlock.joining(""))) .addStatement("hasElements = true") .endControlFlow() .beginControlFlow("if (hasElements)") .addStatement("builder.append($S).append($N(indentLevel))", "\n", INDENT_METHOD_NAME) .endControlFlow() .addStatement("return builder.append($S).toString()", closeSymbol) .build(); } private CodeBlock optionalMethodBody( TypeMirror optionalType, PrettyPrintableKind printableKind) { return CodeBlock.builder() .addStatement( "return (value.isPresent() ? $L : $S)", format(CodeBlock.of("value.get()"), CodeBlock.of("indentLevel"), optionalType), printableKind.equals(PrettyPrintableKind.OPTIONAL) ? "" : "") .build(); } private ImmutableList resolvedTypeParameters( TypeMirror propertyType, String interfaceName) { return elements.getTypeElement(interfaceName).getTypeParameters().stream() .map(p -> types.asMemberOf(MoreTypes.asDeclared(propertyType), p)) .collect(toImmutableList()); } private DeclaredType collectionOf(TypeMirror elementType) { return types.getDeclaredType(elements.getTypeElement("java.util.Collection"), elementType); } private DeclaredType mapOf(TypeMirror keyType, TypeMirror valueType) { return types.getDeclaredType(elements.getTypeElement("java.util.Map"), keyType, valueType); } private DeclaredType multimapOf(TypeMirror keyType, TypeMirror valueType) { return types.getDeclaredType( elements.getTypeElement("com.google.common.collect.Multimap"), keyType, valueType); } /** Returns a valid Java identifier for method or variable of type {@code type}. */ private String nameForType(TypeMirror type) { return type.accept( new SimpleTypeVisitor8() { @Override public String visitDeclared(DeclaredType type, Void v) { String simpleName = simpleNameForType(type); if (type.getTypeArguments().isEmpty()) { return simpleName; } ImmutableList typeArgumentNames = type.getTypeArguments().stream() .map(t -> simpleNameForType(t)) .collect(toImmutableList()); if (isMapOrMultimap(type) && typeArgumentNames.size() == 2) { return String.format( "%sOf%sTo%s", simpleName, typeArgumentNames.get(0), typeArgumentNames.get(1)); } List parts = new ArrayList<>(); parts.add(simpleName); parts.add("Of"); parts.addAll(typeArgumentNames.subList(0, typeArgumentNames.size() - 1)); if (typeArgumentNames.size() > 1) { parts.add("And"); } parts.add(getLast(typeArgumentNames)); return String.join("", parts); } @Override protected String defaultAction(TypeMirror type, Void v) { return simpleNameForType(type); } }, null); } boolean isMapOrMultimap(TypeMirror type) { TypeMirror mapType = elements.getTypeElement("java.util.Map").asType(); if (types.isAssignable(type, types.erasure(mapType))) { return true; } TypeElement multimapElement = elements.getTypeElement("com.google.common.collect.Multimap"); return multimapElement != null && types.isAssignable(type, types.erasure(multimapElement.asType())); } private String simpleNameForType(TypeMirror type) { return type.accept( new SimpleTypeVisitor8() { @Override public String visitPrimitive(PrimitiveType primitiveType, Void v) { return types.boxedClass(primitiveType).getSimpleName().toString(); } @Override public String visitArray(ArrayType arrayType, Void v) { return arrayType.getComponentType().accept(this, null) + "Array"; } @Override public String visitDeclared(DeclaredType declaredType, Void v) { return declaredType.asElement().getSimpleName().toString(); } @Override protected String defaultAction(TypeMirror typeMirror, Void v) { throw new AssertionError(typeMirror); } }, null); } } enum PrettyPrintableKind { HAS_TO_PRETTY_STRING_METHOD, REGULAR_OBJECT, PRIMITIVE, COLLECTION, ARRAY, IMMUTABLE_PRIMITIVE_ARRAY, OPTIONAL, GUAVA_OPTIONAL, MAP, MULTIMAP, ; private static final ImmutableMap KINDS_BY_EXACT_TYPE = ImmutableMap.of( "java.util.Optional", OPTIONAL, "com.google.common.base.Optional", GUAVA_OPTIONAL, "com.google.common.primitives.ImmutableIntArray", IMMUTABLE_PRIMITIVE_ARRAY, "com.google.common.primitives.ImmutableLongArray", IMMUTABLE_PRIMITIVE_ARRAY, "com.google.common.primitives.ImmutableDoubleArray", IMMUTABLE_PRIMITIVE_ARRAY); private static final ImmutableMap KINDS_BY_SUPERTYPE = ImmutableMap.of( "java.util.Collection", COLLECTION, "java.util.Map", MAP, "com.google.common.collect.Multimap", MULTIMAP); static class KindVisitor extends SimpleTypeVisitor8 { private final Elements elements; private final Types types; KindVisitor(Types types, Elements elements) { this.types = types; this.elements = elements; } @Override public PrettyPrintableKind visitPrimitive(PrimitiveType primitiveType, Void v) { return PRIMITIVE; } @Override public PrettyPrintableKind visitArray(ArrayType arrayType, Void v) { return ARRAY; } @Override public PrettyPrintableKind visitDeclared(DeclaredType declaredType, Void v) { TypeElement typeElement = asTypeElement(declaredType); if (toPrettyStringMethod(typeElement, types, elements).isPresent()) { return HAS_TO_PRETTY_STRING_METHOD; } PrettyPrintableKind byExactType = KINDS_BY_EXACT_TYPE.get(typeElement.getQualifiedName().toString()); if (byExactType != null) { return byExactType; } for (Map.Entry entry : KINDS_BY_SUPERTYPE.entrySet()) { TypeElement supertypeElement = elements.getTypeElement(entry.getKey()); if (supertypeElement != null && types.isAssignable(declaredType, types.erasure(supertypeElement.asType()))) { return entry.getValue(); } } return REGULAR_OBJECT; } } } @Override public boolean applicable(Context context) { return toPrettyStringMethods(context).size() == 1; } @Override public ImmutableSet consumeMethods(Context context) { return toPrettyStringMethods(context); } @Override public IncrementalExtensionType incrementalType(ProcessingEnvironment processingEnvironment) { return IncrementalExtensionType.ISOLATING; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy