com.google.auto.value.processor.EclipseHack Maven / Gradle / Ivy
/*
* Copyright (C) 2013 Google, Inc.
*
* 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 com.google.auto.common.MoreTypes;
import com.google.auto.value.processor.AutoValueProcessor.Property;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableListMultimap;
import com.google.common.collect.ImmutableMap;
import java.io.FileReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;
import java.lang.reflect.Method;
import java.net.URI;
import java.net.URL;
import java.util.Collections;
import java.util.Comparator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.Callable;
import javax.annotation.processing.ProcessingEnvironment;
import javax.lang.model.element.Element;
import javax.lang.model.element.ExecutableElement;
import javax.lang.model.element.Name;
import javax.lang.model.element.TypeElement;
import javax.lang.model.type.DeclaredType;
import javax.lang.model.type.TypeMirror;
import javax.lang.model.util.ElementFilter;
import javax.lang.model.util.Elements;
import javax.lang.model.util.Types;
import javax.tools.Diagnostic;
/**
* Works around an Eclipse bug where methods are sorted into alphabetical order before being given
* to annotation processors. Unfortunately this seems to be deeply built in to the JDT compiler
* that Eclipse uses. The bug has been open for over three years with no progress.
*
* To work around the problem, we access Eclipse-specific APIs to find the original source code of
* the class with the {@code @AutoValue} annotation, and we do just enough parsing of that code to
* be able to pick out the abstract method declarations so we can determine their order. The code
* to access Eclipse-specific APIs will fail in environments other than Eclipse (for example, javac)
* and the methods will be left in the order they came in, which in these other environments should
* already be the correct order.
*
* This is obviously a giant hack, and the right thing would be for the Eclipse compiler to be
* fixed. The approach here works, but is vulnerable to future changes in the Eclipse API. If
* {@code @AutoValue} constructor calls like {@code new AutoValue_Foo(...)} suddenly start being
* redlined in a new Eclipse version then the likely cause is that the APIs have changed and this
* hack will need to be updated to track the change.
*
* @see Eclipse bug 300408
*
* @author Éamonn McManus
*/
class EclipseHack {
private final ProcessingEnvironment processingEnv;
EclipseHack(ProcessingEnvironment processingEnv) {
this.processingEnv = processingEnv;
}
/**
* Returns a map containing the real return types of the given methods, knowing that they appear
* in the given type. This means that if the given type is say
* {@code StringIterator implements Iterator} then we want the {@code next()} method
* to map to String, rather than the {@code T} that it returns as inherited from
* {@code Iterator}. This method is in EclipseHack because if it weren't for
* this Eclipse bug it would
* be trivial. Unfortunately, versions of Eclipse up to at least 4.5 have a bug where the
* {@link Types#asMemberOf} method throws IllegalArgumentException if given a method that is
* inherited from an interface. Fortunately, Eclipse's implementation of
* {@link Elements#getAllMembers} does the type substitution that {@code asMemberOf} would have
* done. But javac's implementation doesn't. So we try the way that would work if Eclipse weren't
* buggy, and only if we get IllegalArgumentException do we use {@code getAllMembers}.
*/
ImmutableMap methodReturnTypes(
Set methods, TypeElement autoValueType) {
DeclaredType autoValueTypeMirror = MoreTypes.asDeclared(autoValueType.asType());
Types typeUtils = processingEnv.getTypeUtils();
ImmutableMap.Builder map = ImmutableMap.builder();
Map noArgMethods = null;
for (ExecutableElement method : methods) {
TypeMirror returnType = null;
try {
TypeMirror methodMirror = typeUtils.asMemberOf(autoValueTypeMirror, method);
returnType = MoreTypes.asExecutable(methodMirror).getReturnType();
} catch (IllegalArgumentException e) {
if (method.getParameters().isEmpty()) {
if (noArgMethods == null) {
noArgMethods = noArgMethodsIn(autoValueType);
}
returnType = noArgMethods.get(method.getSimpleName()).getReturnType();
}
}
if (returnType == null) {
returnType = method.getReturnType();
}
map.put(method, returnType);
}
return map.build();
}
/**
* Constructs a map from name to method of the no-argument methods in the given type. We need
* this because an ExecutableElement returned by {@link Elements#getAllMembers} will not compare
* equal to the original ExecutableElement if {@code getAllMembers} substituted type parameters,
* as it does in Eclipse.
*/
private Map noArgMethodsIn(TypeElement autoValueType) {
Elements elementUtils = processingEnv.getElementUtils();
List allMethods =
ElementFilter.methodsIn(elementUtils.getAllMembers(autoValueType));
Map map = new LinkedHashMap();
for (ExecutableElement method : allMethods) {
if (method.getParameters().isEmpty()) {
map.put(method.getSimpleName(), method);
}
}
return map;
}
/**
* Reorders the properties (abstract methods) in the given list to correspond to the order found
* by parsing the source code of the given type. In environments other than Eclipse this method
* has no effect.
*/
void reorderProperties(List properties) {
// Eclipse sorts methods in each class. Because of the way we construct the list, we will see
// all the abstract property methods from a given class or interface consecutively. So we can
// fix each sublist independently.
int index = 0;
while (index < properties.size()) {
TypeElement owner = properties.get(index).getOwner();
int nextIndex = index + 1;
while (nextIndex < properties.size() && properties.get(nextIndex).getOwner().equals(owner)) {
nextIndex++;
}
List subList = properties.subList(index, nextIndex);
reorderProperties(owner, subList);
index = nextIndex;
}
}
private void reorderProperties(TypeElement type, List properties) {
PropertyOrderer propertyOrderer = getPropertyOrderer(type);
if (propertyOrderer == null) {
return;
}
final ImmutableList order;
try {
order = propertyOrderer.determinePropertyOrder();
} catch (Exception e) {
processingEnv.getMessager().printMessage(Diagnostic.Kind.NOTE, e.toString());
return;
}
// We expect that all the properties will be found, but if not then we won't try reordering.
boolean allFound = true;
for (Property property : properties) {
allFound &= order.contains(property.getGetter());
}
if (allFound) {
// We successfully found the abstract methods corresponding to all the properties, so now
// reorder the List to reflect the order of the methods.
Comparator comparator = new Comparator() {
@Override
public int compare(Property a, Property b) {
String aName = a.getGetter();
String bName = b.getGetter();
return order.indexOf(aName) - order.indexOf(bName);
}
};
Collections.sort(properties, comparator);
}
}
private PropertyOrderer getPropertyOrderer(TypeElement type) {
try {
// If we are in Eclipse, then processingEnv will be an instance of
// org.eclipse.jdt.internal.apt.pluggable.core.dispatch.IdeProcessingEnvImpl
// and we can access its getEnclosingIFile method to obtain an Eclipse
// org.eclipse.core.resources.IFile. Then we will access the
// String getCharset();
// and
// InputStream getContents();
// methods to access the whole source file that includes this class.
// If the class in question has not changed since Eclipse last succeessfully compiled it
// then the IFile will be the compiled class file rather than the source, and we will need
// to read the order of the methods out of the class file. The method
// URI getRawLocationURI();
// will tell us this because the URI will end with .class instead of .java.
// If we are not in Eclipse then the reflection here will fail and we will return null,
// which will mean that the caller won't try to reorder.
Method getEnclosingIFile =
processingEnv.getClass().getMethod("getEnclosingIFile", Element.class);
final Object iFile = getEnclosingIFile.invoke(processingEnv, type);
URI uri = (URI) iFile.getClass().getMethod("getRawLocationURI").invoke(iFile);
if (uri.getPath().endsWith(".class")) {
return new BinaryPropertyOrderer(uri);
} else {
Method getCharset = iFile.getClass().getMethod("getCharset");
final String charset = (String) getCharset.invoke(iFile);
final Method getContents = iFile.getClass().getMethod("getContents");
Callable readerProvider = new Callable() {
@Override
public Reader call() throws Exception {
InputStream inputStream = (InputStream) getContents.invoke(iFile);
return new InputStreamReader(inputStream, charset);
}
};
return new SourcePropertyOrderer(type, readerProvider);
}
} catch (Exception e) {
// The method getRawLocationURI used above exists on the Eclipse IDE environment, but not on
// the batch compiler environment. However, the file can also be obtained from the TypeElement
// through the getFileName method.
if (!type.getClass().getName().toLowerCase().contains("eclipse")) {
// Guard against the case where a non-Eclipse type happens to have a getFileName method
return null;
}
try {
final String filename = (String) type.getClass().getMethod("getFileName").invoke(type);
Callable readerProvider = new Callable() {
@Override
public Reader call() throws Exception {
return new FileReader(filename);
}
};
return new SourcePropertyOrderer(type, readerProvider);
} catch (Exception e2) {
// Reflection failed (twice), so we are presumably not in Eclipse.
return null;
}
}
}
private interface PropertyOrderer {
ImmutableList determinePropertyOrder() throws IOException;
}
private class SourcePropertyOrderer implements PropertyOrderer {
private final TypeElement type;
private final Callable readerProvider;
/**
* Constructs an object that scans the source code of the given type and returns the names of
* all abstract methods directly declared in the type (not in nested types). The type itself may
* be nested inside another class. Returns an empty list if the order could not be determined.
*
* @param type The type whose source is being scanned.
* @param readerProvider A Callable that returns a Reader that will read the source of the whole
* file in which the class is declared.
*/
SourcePropertyOrderer(TypeElement type, Callable readerProvider) {
this.type = type;
this.readerProvider = readerProvider;
}
@Override public ImmutableList determinePropertyOrder() throws IOException {
Reader sourceReader;
try {
sourceReader = readerProvider.call();
} catch (Exception e) {
return ImmutableList.of();
}
try {
String packageName = TypeSimplifier.packageNameOf(type);
String className = type.getQualifiedName().toString();
AbstractMethodExtractor extractor = new AbstractMethodExtractor();
EclipseHackTokenizer tokenizer = new EclipseHackTokenizer(sourceReader);
ImmutableListMultimap methodOrders =
extractor.abstractMethods(tokenizer, packageName);
return methodOrders.get(className);
} finally {
sourceReader.close();
}
}
}
private class BinaryPropertyOrderer implements PropertyOrderer {
private final URI classFileUri;
BinaryPropertyOrderer(URI classFileUri) {
this.classFileUri = classFileUri;
}
@Override
public ImmutableList determinePropertyOrder() throws IOException {
InputStream inputStream = null;
try {
URL classFileUrl = classFileUri.toURL();
inputStream = classFileUrl.openStream();
AbstractMethodLister lister = new AbstractMethodLister(inputStream);
return lister.abstractNoArgMethods();
} finally {
if (inputStream != null) {
inputStream.close();
}
}
}
}
}