blob: 5dfabaa54e2eeedccf1c7b0244cb27205d436613 [file] [log] [blame]
/*
* Copyright (C) 2015 Square, 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.squareup.javapoet;
import com.google.testing.compile.CompilationRule;
import java.io.Closeable;
import java.io.IOException;
import java.lang.annotation.ElementType;
import java.lang.annotation.Target;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.TimeoutException;
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.util.Elements;
import javax.lang.model.util.Types;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import static com.google.common.collect.Iterables.getOnlyElement;
import static com.google.common.truth.Truth.assertThat;
import static javax.lang.model.util.ElementFilter.methodsIn;
import static org.junit.Assert.fail;
public final class MethodSpecTest {
@Rule public final CompilationRule compilation = new CompilationRule();
private Elements elements;
private Types types;
@Before public void setUp() {
elements = compilation.getElements();
types = compilation.getTypes();
}
private TypeElement getElement(Class<?> clazz) {
return elements.getTypeElement(clazz.getCanonicalName());
}
private ExecutableElement findFirst(Collection<ExecutableElement> elements, String name) {
for (ExecutableElement executableElement : elements) {
if (executableElement.getSimpleName().toString().equals(name)) {
return executableElement;
}
}
throw new IllegalArgumentException(name + " not found in " + elements);
}
@Test public void nullAnnotationsAddition() {
try {
MethodSpec.methodBuilder("doSomething").addAnnotations(null);
fail();
} catch (IllegalArgumentException expected) {
assertThat(expected).hasMessageThat().isEqualTo("annotationSpecs == null");
}
}
@Test public void nullTypeVariablesAddition() {
try {
MethodSpec.methodBuilder("doSomething").addTypeVariables(null);
fail();
} catch (IllegalArgumentException expected) {
assertThat(expected).hasMessageThat().isEqualTo("typeVariables == null");
}
}
@Test public void nullParametersAddition() {
try {
MethodSpec.methodBuilder("doSomething").addParameters(null);
fail();
} catch (IllegalArgumentException expected) {
assertThat(expected).hasMessageThat().isEqualTo("parameterSpecs == null");
}
}
@Test public void nullExceptionsAddition() {
try {
MethodSpec.methodBuilder("doSomething").addExceptions(null);
fail();
} catch (IllegalArgumentException expected) {
assertThat(expected).hasMessageThat().isEqualTo("exceptions == null");
}
}
@Target(ElementType.PARAMETER)
@interface Nullable {
}
abstract static class Everything {
@Deprecated protected abstract <T extends Runnable & Closeable> Runnable everything(
@Nullable String thing, List<? extends T> things) throws IOException, SecurityException;
}
abstract static class Generics {
<T, R, V extends Throwable> T run(R param) throws V {
return null;
}
}
abstract static class HasAnnotation {
@Override public abstract String toString();
}
interface Throws<R extends RuntimeException> {
void fail() throws R;
}
interface ExtendsOthers extends Callable<Integer>, Comparable<ExtendsOthers>,
Throws<IllegalStateException> {
}
interface ExtendsIterableWithDefaultMethods extends Iterable<Object> {
}
final class FinalClass {
void method() {
}
}
abstract static class InvalidOverrideMethods {
final void finalMethod() {
}
private void privateMethod() {
}
static void staticMethod() {
}
}
@Test public void overrideEverything() {
TypeElement classElement = getElement(Everything.class);
ExecutableElement methodElement = getOnlyElement(methodsIn(classElement.getEnclosedElements()));
MethodSpec method = MethodSpec.overriding(methodElement).build();
assertThat(method.toString()).isEqualTo(""
+ "@java.lang.Override\n"
+ "protected <T extends java.lang.Runnable & java.io.Closeable> java.lang.Runnable "
+ "everything(\n"
+ " java.lang.String arg0, java.util.List<? extends T> arg1) throws java.io.IOException,\n"
+ " java.lang.SecurityException {\n"
+ "}\n");
}
@Test public void overrideGenerics() {
TypeElement classElement = getElement(Generics.class);
ExecutableElement methodElement = getOnlyElement(methodsIn(classElement.getEnclosedElements()));
MethodSpec method = MethodSpec.overriding(methodElement)
.addStatement("return null")
.build();
assertThat(method.toString()).isEqualTo(""
+ "@java.lang.Override\n"
+ "<T, R, V extends java.lang.Throwable> T run(R param) throws V {\n"
+ " return null;\n"
+ "}\n");
}
@Test public void overrideDoesNotCopyOverrideAnnotation() {
TypeElement classElement = getElement(HasAnnotation.class);
ExecutableElement exec = getOnlyElement(methodsIn(classElement.getEnclosedElements()));
MethodSpec method = MethodSpec.overriding(exec).build();
assertThat(method.toString()).isEqualTo(""
+ "@java.lang.Override\n"
+ "public java.lang.String toString() {\n"
+ "}\n");
}
@Test public void overrideDoesNotCopyDefaultModifier() {
TypeElement classElement = getElement(ExtendsIterableWithDefaultMethods.class);
DeclaredType classType = (DeclaredType) classElement.asType();
List<ExecutableElement> methods = methodsIn(elements.getAllMembers(classElement));
ExecutableElement exec = findFirst(methods, "spliterator");
MethodSpec method = MethodSpec.overriding(exec, classType, types).build();
assertThat(method.toString()).isEqualTo(""
+ "@java.lang.Override\n"
+ "public java.util.Spliterator<java.lang.Object> spliterator() {\n"
+ "}\n");
}
@Test public void overrideExtendsOthersWorksWithActualTypeParameters() {
TypeElement classElement = getElement(ExtendsOthers.class);
DeclaredType classType = (DeclaredType) classElement.asType();
List<ExecutableElement> methods = methodsIn(elements.getAllMembers(classElement));
ExecutableElement exec = findFirst(methods, "call");
MethodSpec method = MethodSpec.overriding(exec, classType, types).build();
assertThat(method.toString()).isEqualTo(""
+ "@java.lang.Override\n"
+ "public java.lang.Integer call() throws java.lang.Exception {\n"
+ "}\n");
exec = findFirst(methods, "compareTo");
method = MethodSpec.overriding(exec, classType, types).build();
assertThat(method.toString()).isEqualTo(""
+ "@java.lang.Override\n"
+ "public int compareTo(" + ExtendsOthers.class.getCanonicalName() + " arg0) {\n"
+ "}\n");
exec = findFirst(methods, "fail");
method = MethodSpec.overriding(exec, classType, types).build();
assertThat(method.toString()).isEqualTo(""
+ "@java.lang.Override\n"
+ "public void fail() throws java.lang.IllegalStateException {\n"
+ "}\n");
}
@Test public void overrideFinalClassMethod() {
TypeElement classElement = getElement(FinalClass.class);
List<ExecutableElement> methods = methodsIn(elements.getAllMembers(classElement));
try {
MethodSpec.overriding(findFirst(methods, "method"));
fail();
} catch (IllegalArgumentException expected) {
assertThat(expected).hasMessageThat().isEqualTo(
"Cannot override method on final class com.squareup.javapoet.MethodSpecTest.FinalClass");
}
}
@Test public void overrideInvalidModifiers() {
TypeElement classElement = getElement(InvalidOverrideMethods.class);
List<ExecutableElement> methods = methodsIn(elements.getAllMembers(classElement));
try {
MethodSpec.overriding(findFirst(methods, "finalMethod"));
fail();
} catch (IllegalArgumentException expected) {
assertThat(expected).hasMessageThat().isEqualTo("cannot override method with modifiers: [final]");
}
try {
MethodSpec.overriding(findFirst(methods, "privateMethod"));
fail();
} catch (IllegalArgumentException expected) {
assertThat(expected).hasMessageThat().isEqualTo("cannot override method with modifiers: [private]");
}
try {
MethodSpec.overriding(findFirst(methods, "staticMethod"));
fail();
} catch (IllegalArgumentException expected) {
assertThat(expected).hasMessageThat().isEqualTo("cannot override method with modifiers: [static]");
}
}
@Test public void equalsAndHashCode() {
MethodSpec a = MethodSpec.constructorBuilder().build();
MethodSpec b = MethodSpec.constructorBuilder().build();
assertThat(a.equals(b)).isTrue();
assertThat(a.hashCode()).isEqualTo(b.hashCode());
a = MethodSpec.methodBuilder("taco").build();
b = MethodSpec.methodBuilder("taco").build();
assertThat(a.equals(b)).isTrue();
assertThat(a.hashCode()).isEqualTo(b.hashCode());
TypeElement classElement = getElement(Everything.class);
ExecutableElement methodElement = getOnlyElement(methodsIn(classElement.getEnclosedElements()));
a = MethodSpec.overriding(methodElement).build();
b = MethodSpec.overriding(methodElement).build();
assertThat(a.equals(b)).isTrue();
assertThat(a.hashCode()).isEqualTo(b.hashCode());
}
@Test public void duplicateExceptionsIgnored() {
ClassName ioException = ClassName.get(IOException.class);
ClassName timeoutException = ClassName.get(TimeoutException.class);
MethodSpec methodSpec = MethodSpec.methodBuilder("duplicateExceptions")
.addException(ioException)
.addException(timeoutException)
.addException(timeoutException)
.addException(ioException)
.build();
assertThat(methodSpec.exceptions).isEqualTo(Arrays.asList(ioException, timeoutException));
assertThat(methodSpec.toBuilder().addException(ioException).build().exceptions)
.isEqualTo(Arrays.asList(ioException, timeoutException));
}
@Test public void nullIsNotAValidMethodName() {
try {
MethodSpec.methodBuilder(null);
fail("NullPointerException expected");
} catch (NullPointerException e) {
assertThat(e.getMessage()).isEqualTo("name == null");
}
}
@Test public void addModifiersVarargsShouldNotBeNull() {
try {
MethodSpec.methodBuilder("taco")
.addModifiers((Modifier[]) null);
fail("NullPointerException expected");
} catch (NullPointerException e) {
assertThat(e.getMessage()).isEqualTo("modifiers == null");
}
}
}