blob: 3c066bab2e31f7a8175b899d67b08e1c61e0b5ed [file] [log] [blame]
/*
* Copyright (C) 2015 The Dagger Authors.
*
* 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 dagger.internal.codegen;
import static com.google.testing.compile.CompilationSubject.assertThat;
import static dagger.internal.codegen.Compilers.daggerCompiler;
import static dagger.internal.codegen.TestUtils.message;
import com.google.testing.compile.Compilation;
import com.google.testing.compile.JavaFileObjects;
import javax.tools.JavaFileObject;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
@RunWith(JUnit4.class)
public class MissingBindingSuggestionsTest {
private static JavaFileObject injectable(String className, String constructorParams) {
return JavaFileObjects.forSourceLines("test." + className,
"package test;",
"",
"import javax.inject.Inject;",
"",
"class " + className +" {",
" @Inject " + className + "(" + constructorParams + ") {}",
"}");
}
private static JavaFileObject emptyInterface(String interfaceName) {
return JavaFileObjects.forSourceLines("test." + interfaceName,
"package test;",
"",
"import javax.inject.Inject;",
"",
"interface " + interfaceName +" {}");
}
@Test public void suggestsBindingInSeparateComponent() {
JavaFileObject fooComponent = JavaFileObjects.forSourceLines("test.FooComponent",
"package test;",
"",
"import dagger.Subcomponent;",
"",
"@Subcomponent",
"interface FooComponent {",
" Foo getFoo();",
"}");
JavaFileObject barModule = JavaFileObjects.forSourceLines("test.BarModule",
"package test;",
"",
"import dagger.Provides;",
"import javax.inject.Inject;",
"",
"@dagger.Module",
"final class BarModule {",
" @Provides Bar provideBar() {return null;}",
"}");
JavaFileObject barComponent = JavaFileObjects.forSourceLines("test.BarComponent",
"package test;",
"",
"import dagger.Subcomponent;",
"",
"@Subcomponent(modules = {BarModule.class})",
"interface BarComponent {",
" Bar getBar();",
"}");
JavaFileObject foo = injectable("Foo", "Bar bar");
JavaFileObject bar = emptyInterface("Bar");
JavaFileObject topComponent = JavaFileObjects.forSourceLines("test.TopComponent",
"package test;",
"",
"import dagger.Component;",
"",
"@Component",
"interface TopComponent {",
" FooComponent getFoo();",
" BarComponent getBar(BarModule barModule);",
"}");
Compilation compilation =
daggerCompiler().compile(fooComponent, barComponent, topComponent, foo, bar, barModule);
assertThat(compilation).failed();
assertThat(compilation).hadErrorCount(1);
assertThat(compilation)
.hadErrorContaining("A binding with matching key exists in component: BarComponent");
}
@Test public void suggestsBindingInNestedSubcomponent() {
JavaFileObject fooComponent = JavaFileObjects.forSourceLines("test.FooComponent",
"package test;",
"",
"import dagger.Subcomponent;",
"",
"@Subcomponent",
"interface FooComponent {",
" Foo getFoo();",
"}");
JavaFileObject barComponent = JavaFileObjects.forSourceLines("test.BarComponent",
"package test;",
"",
"import dagger.Subcomponent;",
"",
"@Subcomponent()",
"interface BarComponent {",
" BazComponent getBaz();",
"}");
JavaFileObject bazModule = JavaFileObjects.forSourceLines("test.BazModule",
"package test;",
"",
"import dagger.Provides;",
"import javax.inject.Inject;",
"",
"@dagger.Module",
"final class BazModule {",
" @Provides Baz provideBaz() {return null;}",
"}");
JavaFileObject bazComponent = JavaFileObjects.forSourceLines("test.BazComponent",
"package test;",
"",
"import dagger.Subcomponent;",
"",
"@Subcomponent(modules = {BazModule.class})",
"interface BazComponent {",
" Baz getBaz();",
"}");
JavaFileObject foo = injectable("Foo", "Baz baz");
JavaFileObject baz = emptyInterface("Baz");
JavaFileObject topComponent = JavaFileObjects.forSourceLines("test.TopComponent",
"package test;",
"",
"import dagger.Component;",
"",
"@Component",
"interface TopComponent {",
" FooComponent getFoo();",
" BarComponent getBar();",
"}");
Compilation compilation =
daggerCompiler()
.compile(fooComponent, barComponent, bazComponent, topComponent, foo, baz, bazModule);
assertThat(compilation).failed();
assertThat(compilation).hadErrorCount(1);
assertThat(compilation)
.hadErrorContaining("A binding with matching key exists in component: BazComponent");
}
@Test
public void missingBindingInParentComponent() {
JavaFileObject parent =
JavaFileObjects.forSourceLines(
"Parent",
"import dagger.Component;",
"",
"@Component",
"interface Parent {",
" Foo foo();",
" Bar bar();",
" Child child();",
"}");
JavaFileObject child =
JavaFileObjects.forSourceLines(
"Child",
"import dagger.Subcomponent;",
"",
"@Subcomponent(modules=BazModule.class)",
"interface Child {",
" Foo foo();",
" Baz baz();",
"}");
JavaFileObject foo =
JavaFileObjects.forSourceLines(
"Foo",
"import javax.inject.Inject;",
"",
"class Foo {",
" @Inject Foo(Bar bar) {}",
"}");
JavaFileObject bar =
JavaFileObjects.forSourceLines(
"Bar",
"import javax.inject.Inject;",
"",
"class Bar {",
" @Inject Bar(Baz baz) {}",
"}");
JavaFileObject baz = JavaFileObjects.forSourceLines("Baz", "class Baz {}");
JavaFileObject bazModule = JavaFileObjects.forSourceLines(
"BazModule",
"import dagger.Module;",
"import dagger.Provides;",
"import javax.inject.Inject;",
"",
"@Module",
"final class BazModule {",
" @Provides Baz provideBaz() {return new Baz();}",
"}");
Compilation compilation = daggerCompiler().compile(parent, child, foo, bar, baz, bazModule);
assertThat(compilation).failed();
assertThat(compilation).hadErrorCount(1);
assertThat(compilation)
.hadErrorContaining(
message(
"\033[1;31m[Dagger/MissingBinding]\033[0m Baz cannot be provided without an "
+ "@Inject constructor or an @Provides-annotated method.",
"A binding with matching key exists in component: Child",
" Baz is injected at",
" Bar(baz)",
" Bar is requested at",
" Parent.bar()",
"The following other entry points also depend on it:",
" Parent.foo()",
" Child.foo() [Parent → Child]"))
.inFile(parent)
.onLineContaining("interface Parent");
}
@Test
public void missingBindingInSiblingComponent() {
JavaFileObject parent =
JavaFileObjects.forSourceLines(
"Parent",
"import dagger.Component;",
"",
"@Component",
"interface Parent {",
" Foo foo();",
" Bar bar();",
" Child1 child1();",
" Child2 child2();",
"}");
JavaFileObject child1 =
JavaFileObjects.forSourceLines(
"Child1",
"import dagger.Subcomponent;",
"",
"@Subcomponent",
"interface Child1 {",
" Foo foo();",
" Baz baz();",
"}");
JavaFileObject child2 =
JavaFileObjects.forSourceLines(
"Child2",
"import dagger.Subcomponent;",
"",
"@Subcomponent(modules = BazModule.class)",
"interface Child2 {",
" Foo foo();",
" Baz baz();",
"}");
JavaFileObject foo =
JavaFileObjects.forSourceLines(
"Foo",
"import javax.inject.Inject;",
"",
"class Foo {",
" @Inject Foo(Bar bar) {}",
"}");
JavaFileObject bar =
JavaFileObjects.forSourceLines(
"Bar",
"import javax.inject.Inject;",
"",
"class Bar {",
" @Inject Bar(Baz baz) {}",
"}");
JavaFileObject baz = JavaFileObjects.forSourceLines("Baz", "class Baz {}");
JavaFileObject bazModule = JavaFileObjects.forSourceLines(
"BazModule",
"import dagger.Module;",
"import dagger.Provides;",
"import javax.inject.Inject;",
"",
"@Module",
"final class BazModule {",
" @Provides Baz provideBaz() {return new Baz();}",
"}");
Compilation compilation =
daggerCompiler().compile(parent, child1, child2, foo, bar, baz, bazModule);
assertThat(compilation).failed();
assertThat(compilation).hadErrorCount(1);
assertThat(compilation)
.hadErrorContaining(
message(
"\033[1;31m[Dagger/MissingBinding]\033[0m Baz cannot be provided without an "
+ "@Inject constructor or an @Provides-annotated method.",
"A binding with matching key exists in component: Child2",
" Baz is injected at",
" Bar(baz)",
" Bar is requested at",
" Parent.bar()",
"The following other entry points also depend on it:",
" Parent.foo()",
" Child1.foo() [Parent → Child1]",
" Child2.foo() [Parent → Child2]",
" Child1.baz() [Parent → Child1]"))
.inFile(parent)
.onLineContaining("interface Parent");
}
}