blob: d9dd943fcf75411299dd65b1cc0908c1cb7231a6 [file] [log] [blame]
/**
* Copyright (C) 2006 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.inject;
import static com.google.inject.Asserts.assertContains;
import static com.google.inject.Asserts.assertEqualsBothWays;
import static com.google.inject.Asserts.assertNotSerializable;
import static com.google.inject.Asserts.awaitClear;
import static java.lang.annotation.RetentionPolicy.RUNTIME;
import com.google.inject.name.Named;
import com.google.inject.name.Names;
import com.google.inject.spi.Dependency;
import com.google.inject.util.Types;
import junit.framework.TestCase;
import java.io.IOException;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.Target;
import java.lang.ref.WeakReference;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.lang.reflect.TypeVariable;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.concurrent.atomic.AtomicReference;
/**
* @author crazybob@google.com (Bob Lee)
*/
public class KeyTest extends TestCase {
public void foo(List<String> a, List<String> b) {}
public void bar(Provider<List<String>> a) {}
@Foo String baz;
List<? extends CharSequence> wildcardExtends;
public void testOfType() {
Key<Object> k = Key.get(Object.class, Foo.class);
Key<Integer> ki = k.ofType(Integer.class);
assertEquals(Integer.class, ki.getRawType());
assertEquals(Foo.class, ki.getAnnotationType());
}
public void testKeyEquality() {
Key<List<String>> a = new Key<List<String>>(Foo.class) {};
Key<List<String>> b = Key.get(new TypeLiteral<List<String>>() {}, Foo.class);
assertEqualsBothWays(a, b);
}
public void testProviderKey() throws NoSuchMethodException {
Key<?> actual = Key.get(getClass().getMethod("foo", List.class, List.class)
.getGenericParameterTypes()[0]).providerKey();
Key<?> expected = Key.get(getClass().getMethod("bar", Provider.class)
.getGenericParameterTypes()[0]);
assertEqualsBothWays(expected, actual);
assertEquals(expected.toString(), actual.toString());
}
public void testTypeEquality() throws Exception {
Method m = getClass().getMethod("foo", List.class, List.class);
Type[] types = m.getGenericParameterTypes();
assertEquals(types[0], types[1]);
Key<List<String>> k = new Key<List<String>>() {};
assertEquals(types[0], k.getTypeLiteral().getType());
assertFalse(types[0].equals(
new Key<List<Integer>>() {}.getTypeLiteral().getType()));
}
/**
* Key canonicalizes {@link int.class} to {@code Integer.class}, and
* won't expose wrapper types.
*/
public void testPrimitivesAndWrappersAreEqual() {
Class[] primitives = new Class[] {
boolean.class, byte.class, short.class, int.class, long.class,
float.class, double.class, char.class, void.class
};
Class[] wrappers = new Class[] {
Boolean.class, Byte.class, Short.class, Integer.class, Long.class,
Float.class, Double.class, Character.class, Void.class
};
for (int t = 0; t < primitives.length; t++) {
@SuppressWarnings("unchecked")
Key primitiveKey = Key.get(primitives[t]);
@SuppressWarnings("unchecked")
Key wrapperKey = Key.get(wrappers[t]);
assertEquals(primitiveKey, wrapperKey);
assertEquals(wrappers[t], primitiveKey.getRawType());
assertEquals(wrappers[t], wrapperKey.getRawType());
assertEquals(wrappers[t], primitiveKey.getTypeLiteral().getType());
assertEquals(wrappers[t], wrapperKey.getTypeLiteral().getType());
}
Key<Integer> integerKey = Key.get(Integer.class);
Key<Integer> integerKey2 = Key.get(Integer.class, Named.class);
Key<Integer> integerKey3 = Key.get(Integer.class, Names.named("int"));
Class<Integer> intClassLiteral = int.class;
assertEquals(integerKey, Key.get(intClassLiteral));
assertEquals(integerKey2, Key.get(intClassLiteral, Named.class));
assertEquals(integerKey3, Key.get(intClassLiteral, Names.named("int")));
Type intType = int.class;
assertEquals(integerKey, Key.get(intType));
assertEquals(integerKey2, Key.get(intType, Named.class));
assertEquals(integerKey3, Key.get(intType, Names.named("int")));
TypeLiteral<Integer> intTypeLiteral = TypeLiteral.get(int.class);
assertEquals(integerKey, Key.get(intTypeLiteral));
assertEquals(integerKey2, Key.get(intTypeLiteral, Named.class));
assertEquals(integerKey3, Key.get(intTypeLiteral, Names.named("int")));
}
public void testSerialization() throws IOException, NoSuchFieldException {
assertNotSerializable(Key.get(B.class));
assertNotSerializable(Key.get(B.class, Names.named("bee")));
assertNotSerializable(Key.get(B.class, Named.class));
assertNotSerializable(Key.get(B[].class));
assertNotSerializable(Key.get(new TypeLiteral<Map<List<B>, B>>() {}));
assertNotSerializable(Key.get(new TypeLiteral<List<B[]>>() {}));
assertNotSerializable(Key.get(Types.listOf(Types.subtypeOf(CharSequence.class))));
}
public void testEqualityOfAnnotationTypesAndInstances() throws NoSuchFieldException {
Foo instance = getClass().getDeclaredField("baz").getAnnotation(Foo.class);
Key<String> keyWithInstance = Key.get(String.class, instance);
Key<String> keyWithLiteral = Key.get(String.class, Foo.class);
assertEqualsBothWays(keyWithInstance, keyWithLiteral);
}
public void testNonBindingAnnotationOnKey() {
try {
Key.get(String.class, Deprecated.class);
fail();
} catch (IllegalArgumentException expected) {
assertContains(expected.getMessage(), "java.lang.Deprecated is not a binding annotation. ",
"Please annotate it with @BindingAnnotation.");
}
}
public void testBindingAnnotationWithoutRuntimeRetention() {
try {
Key.get(String.class, Bar.class);
fail();
} catch (IllegalArgumentException expected) {
assertContains(expected.getMessage(), Bar.class.getName() + " is not retained at runtime.",
"Please annotate it with @Retention(RUNTIME).");
}
}
<T> void parameterizedWithVariable(List<T> typeWithVariables) {}
/** Test for issue 186 */
public void testCannotCreateKeysWithTypeVariables() throws NoSuchMethodException {
ParameterizedType listOfTType = (ParameterizedType) getClass().getDeclaredMethod(
"parameterizedWithVariable", List.class).getGenericParameterTypes()[0];
TypeLiteral<?> listOfT = TypeLiteral.get(listOfTType);
try {
Key.get(listOfT);
fail("Guice should not allow keys for java.util.List<T>");
} catch (ConfigurationException e) {
assertContains(e.getMessage(),
"java.util.List<T> cannot be used as a key; It is not fully specified.");
}
TypeVariable tType = (TypeVariable) listOfTType.getActualTypeArguments()[0];
TypeLiteral<?> t = TypeLiteral.get(tType);
try {
Key.get(t);
fail("Guice should not allow keys for T");
} catch (ConfigurationException e) {
assertContains(e.getMessage(),
"T cannot be used as a key; It is not fully specified.");
}
}
public void testCannotGetKeyWithUnspecifiedTypeVariables() {
TypeLiteral<Integer> typeLiteral = KeyTest.createTypeLiteral();
try {
Key.get(typeLiteral);
fail("Guice should not allow keys for T");
} catch (ConfigurationException e) {
assertContains(e.getMessage(),
"T cannot be used as a key; It is not fully specified.");
}
}
private static <T> TypeLiteral<T> createTypeLiteral() {
return new TypeLiteral<T>() {};
}
public void testCannotCreateKeySubclassesWithUnspecifiedTypeVariables() {
try {
KeyTest.<Integer>createKey();
fail("Guice should not allow keys for T");
} catch (ConfigurationException e) {
assertContains(e.getMessage(),
"T cannot be used as a key; It is not fully specified.");
}
}
private static <T> Key<T> createKey() {
return new Key<T>() {};
}
interface B {}
@Retention(RUNTIME)
@Target({ ElementType.FIELD, ElementType.PARAMETER, ElementType.METHOD })
@BindingAnnotation @interface Foo {}
@Target({ ElementType.FIELD, ElementType.PARAMETER, ElementType.METHOD })
@BindingAnnotation @interface Bar {}
class HasTypeParameters<A, B extends List<A> & Runnable, C extends Runnable> {
A a; B b; C c;
}
public void testKeysWithDefaultAnnotations() {
AllDefaults allDefaults = HasAnnotations.class.getAnnotation(AllDefaults.class);
assertEquals(Key.get(Foo.class, allDefaults), Key.get(Foo.class, AllDefaults.class));
Marker marker = HasAnnotations.class.getAnnotation(Marker.class);
assertEquals(Key.get(Foo.class, marker), Key.get(Foo.class, Marker.class));
Key<?> noDefaults = Key.get(Foo.class, NoDefaults.class);
assertNull(noDefaults.getAnnotation());
assertEquals(NoDefaults.class, noDefaults.getAnnotationType());
Key<?> someDefaults = Key.get(Foo.class, SomeDefaults.class);
assertNull(someDefaults.getAnnotation());
assertEquals(SomeDefaults.class, someDefaults.getAnnotationType());
}
@Retention(RUNTIME)
@BindingAnnotation @interface AllDefaults {
int v1() default 1;
String v2() default "foo";
}
@Retention(RUNTIME)
@BindingAnnotation @interface SomeDefaults {
int v1() default 1;
String v2() default "foo";
Class<?> clazz();
}
@Retention(RUNTIME)
@BindingAnnotation @interface NoDefaults {
int value();
}
@Retention(RUNTIME)
@BindingAnnotation @interface Marker {
}
@AllDefaults
@Marker
class HasAnnotations {}
public void testAnonymousClassesDontHoldRefs() {
final AtomicReference<Provider<List<String>>> stringProvider =
new AtomicReference<Provider<List<String>>>();
final AtomicReference<Provider<List<Integer>>> intProvider =
new AtomicReference<Provider<List<Integer>>>();
final Object foo = new Object() {
@SuppressWarnings("unused") @Inject List<String> list;
};
Module module = new AbstractModule() {
@Override protected void configure() {
bind(new Key<List<String>>() {}).toInstance(new ArrayList<String>());
bind(new TypeLiteral<List<Integer>>() {}).toInstance(new ArrayList<Integer>());
stringProvider.set(getProvider(new Key<List<String>>() {}));
intProvider.set(binder().getProvider(Dependency.get(new Key<List<Integer>>() {})));
binder().requestInjection(new TypeLiteral<Object>() {}, foo);
}
};
WeakReference<Module> moduleRef = new WeakReference<Module>(module);
final Injector injector = Guice.createInjector(module);
module = null;
awaitClear(moduleRef); // Make sure anonymous keys & typeliterals don't hold the module.
Runnable runner = new Runnable() {
@Override public void run() {
injector.getInstance(new Key<Typed<String>>() {});
injector.getInstance(Key.get(new TypeLiteral<Typed<Integer>>() {}));
}
};
WeakReference<Runnable> runnerRef = new WeakReference<Runnable>(runner);
runner.run();
runner = null;
awaitClear(runnerRef); // also make sure anonymous keys & typeliterals don't hold for JITs
}
static class Typed<T> {}
}