blob: b99adce239bdeac811bf55be6c4675d672bacbcf [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 com.google.common.collect.Iterables;
import static com.google.inject.Asserts.assertContains;
import com.google.inject.matcher.Matchers;
import com.google.inject.spi.TypeConverter;
import java.lang.annotation.Retention;
import static java.lang.annotation.RetentionPolicy.RUNTIME;
import java.text.DateFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import junit.framework.TestCase;
/**
* @author crazybob@google.com (Bob Lee)
*/
public class TypeConversionTest extends TestCase {
@Retention(RUNTIME)
@BindingAnnotation @interface NumericValue {}
@Retention(RUNTIME)
@BindingAnnotation @interface BooleanValue {}
@Retention(RUNTIME)
@BindingAnnotation @interface EnumValue {}
@Retention(RUNTIME)
@BindingAnnotation @interface ClassName {}
public static class Foo {
@Inject @BooleanValue Boolean booleanField;
@Inject @BooleanValue boolean primitiveBooleanField;
@Inject @NumericValue Byte byteField;
@Inject @NumericValue byte primitiveByteField;
@Inject @NumericValue Short shortField;
@Inject @NumericValue short primitiveShortField;
@Inject @NumericValue Integer integerField;
@Inject @NumericValue int primitiveIntField;
@Inject @NumericValue Long longField;
@Inject @NumericValue long primitiveLongField;
@Inject @NumericValue Float floatField;
@Inject @NumericValue float primitiveFloatField;
@Inject @NumericValue Double doubleField;
@Inject @NumericValue double primitiveDoubleField;
@Inject @EnumValue Bar enumField;
@Inject @ClassName Class<?> classField;
}
public enum Bar {
TEE, BAZ, BOB
}
public void testOneConstantInjection() throws CreationException {
Injector injector = Guice.createInjector(new AbstractModule() {
protected void configure() {
bindConstant().annotatedWith(NumericValue.class).to("5");
bind(Simple.class);
}
});
Simple simple = injector.getInstance(Simple.class);
assertEquals(5, simple.i);
}
static class Simple {
@Inject @NumericValue int i;
}
public void testConstantInjection() throws CreationException {
Injector injector = Guice.createInjector(new AbstractModule() {
protected void configure() {
bindConstant().annotatedWith(NumericValue.class).to("5");
bindConstant().annotatedWith(BooleanValue.class).to("true");
bindConstant().annotatedWith(EnumValue.class).to("TEE");
bindConstant().annotatedWith(ClassName.class).to(Foo.class.getName());
}
});
Foo foo = injector.getInstance(Foo.class);
checkNumbers(
foo.integerField,
foo.primitiveIntField,
foo.longField,
foo.primitiveLongField,
foo.byteField,
foo.primitiveByteField,
foo.shortField,
foo.primitiveShortField,
foo.floatField,
foo.primitiveFloatField,
foo.doubleField,
foo.primitiveDoubleField
);
assertEquals(Bar.TEE, foo.enumField);
assertEquals(Foo.class, foo.classField);
}
void checkNumbers(Number... ns) {
for (Number n : ns) {
assertEquals(5, n.intValue());
}
}
public void testInvalidInteger() throws CreationException {
Injector injector = Guice.createInjector(new AbstractModule() {
protected void configure() {
bindConstant().annotatedWith(NumericValue.class).to("invalid");
}
});
try {
injector.getInstance(InvalidInteger.class);
fail();
} catch (ProvisionException expected) {
assertContains(expected.getMessage(), "Error converting 'invalid'");
assertContains(expected.getMessage(), "bound at " + getClass().getName());
assertContains(expected.getMessage(), "to java.lang.Integer");
}
}
public static class InvalidInteger {
@Inject @NumericValue Integer integerField;
}
public void testInvalidCharacter() throws CreationException {
Injector injector = Guice.createInjector(new AbstractModule() {
protected void configure() {
bindConstant().annotatedWith(NumericValue.class).to("invalid");
}
});
try {
injector.getInstance(InvalidCharacter.class);
fail();
} catch (ProvisionException expected) {
assertContains(expected.getMessage(), "Error converting 'invalid'");
assertContains(expected.getMessage(), "bound at " + getClass().getName());
assertContains(expected.getMessage(), "to java.lang.Character");
}
}
public static class InvalidCharacter {
@Inject @NumericValue char foo;
}
public void testInvalidEnum() throws CreationException {
Injector injector = Guice.createInjector(new AbstractModule() {
protected void configure() {
bindConstant().annotatedWith(NumericValue.class).to("invalid");
}
});
try {
injector.getInstance(InvalidEnum.class);
fail();
} catch (ProvisionException expected) {
assertContains(expected.getMessage(), "Error converting 'invalid'");
assertContains(expected.getMessage(), "bound at " + getClass().getName());
assertContains(expected.getMessage(), "to " + Bar.class.getName());
}
}
public static class InvalidEnum {
@Inject @NumericValue Bar foo;
}
public void testToInstanceIsTreatedLikeConstant() throws CreationException {
Injector injector = Guice.createInjector(new AbstractModule() {
protected void configure() {
bind(String.class).toInstance("5");
bind(LongHolder.class);
}
});
assertEquals(5L, (long) injector.getInstance(LongHolder.class).foo);
}
static class LongHolder {
@Inject Long foo;
}
private final TypeConverter dateConverter = new TypeConverter() {
final DateFormat dateFormat = new SimpleDateFormat("d'-'MMM'-'yyyy");
public Object convert(String value, TypeLiteral<?> toType) {
try {
return dateFormat.parse(value);
} catch (ParseException e) {
throw new IllegalArgumentException("Unparseable date: \"" + value + "\"");
}
}
@Override public String toString() {
return "TypeConverter<Date>";
}
};
public void testCustomTypeConversion() throws CreationException {
Injector injector = Guice.createInjector(new AbstractModule() {
protected void configure() {
convertToTypes(Matchers.only(TypeLiteral.get(Date.class)) , dateConverter);
bindConstant().annotatedWith(NumericValue.class).to("15-Aug-1981");
bind(DateHolder.class);
}
});
DateHolder dateHolder = injector.getInstance(DateHolder.class);
Calendar calendar = new GregorianCalendar();
calendar.setTime(dateHolder.date);
assertEquals(15, calendar.get(Calendar.DAY_OF_MONTH));
assertEquals(7, calendar.get(Calendar.MONTH));
assertEquals(1981, calendar.get(Calendar.YEAR));
}
public void testInvalidCustomValue() throws CreationException {
Module module = new AbstractModule() {
protected void configure() {
convertToTypes(Matchers.only(TypeLiteral.get(Date.class)), dateConverter);
bindConstant().annotatedWith(NumericValue.class).to("invalid");
bind(DateHolder.class);
}
};
try {
Guice.createInjector(module);
fail();
} catch (CreationException expected) {
Throwable cause = Iterables.getOnlyElement(expected.getErrorMessages()).getCause();
assertTrue(cause instanceof IllegalArgumentException);
assertContains(expected.getMessage(),
"Error at " + DateHolder.class.getName() + ".date(TypeConversionTest.java:",
"Error converting 'invalid' (bound at ", getClass().getName(),
".configure(TypeConversionTest.java:", "to java.util.Date",
"using TypeConverter<Date> which matches only(java.util.Date) ",
"(bound at " + getClass().getName(), ".configure(TypeConversionTest.java:",
"Reason: java.lang.IllegalArgumentException: Unparseable date: \"invalid\"");
}
}
public void testNullCustomValue() {
Module module = new AbstractModule() {
protected void configure() {
convertToTypes(Matchers.only(TypeLiteral.get(Date.class)), new TypeConverter() {
public Object convert(String value, TypeLiteral<?> toType) {
return null;
}
@Override public String toString() {
return "TypeConverter<Null>";
}
});
bindConstant().annotatedWith(NumericValue.class).to("foo");
bind(DateHolder.class);
}
};
try {
Guice.createInjector(module);
fail();
} catch (CreationException expected) {
assertContains(expected.getMessage(),
"Error at " + DateHolder.class.getName() + ".date(TypeConversionTest.java:",
"Received null converting 'foo' (bound at ", getClass().getName(),
".configure(TypeConversionTest.java:", "to java.util.Date",
"using TypeConverter<Null> which matches only(java.util.Date) ",
"(bound at " + getClass().getName(), ".configure(TypeConversionTest.java:");
}
}
public void testCustomValueTypeMismatch() {
Module module = new AbstractModule() {
protected void configure() {
convertToTypes(Matchers.only(TypeLiteral.get(Date.class)), new TypeConverter() {
public Object convert(String value, TypeLiteral<?> toType) {
return -1;
}
@Override public String toString() {
return "TypeConverter<Date>";
}
});
bindConstant().annotatedWith(NumericValue.class).to("foo");
bind(DateHolder.class);
}
};
try {
Guice.createInjector(module);
fail();
} catch (CreationException expected) {
assertContains(expected.getMessage(),
"Error at " + DateHolder.class.getName() + ".date(TypeConversionTest.java:",
"Type mismatch converting 'foo' (bound at ", getClass().getName(),
".configure(TypeConversionTest.java:", "to java.util.Date",
"using TypeConverter<Date> which matches only(java.util.Date) ",
"(bound at " + getClass().getName(), ".configure(TypeConversionTest.java:",
"Converter returned -1.");
}
}
static class DateHolder {
@Inject @NumericValue Date date;
}
}