| /* |
| * Copyright (C) 2018 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 androidx.room.compiler.processing.util.Source; |
| import dagger.testing.compile.CompilerTests; |
| import org.junit.Test; |
| import org.junit.runner.RunWith; |
| import org.junit.runners.JUnit4; |
| |
| /** |
| * Tests that errors are reported for invalid members injection methods and {@link |
| * dagger.MembersInjector} dependency requests. |
| */ |
| @RunWith(JUnit4.class) |
| public class MembersInjectionValidationTest { |
| @Test |
| public void membersInjectDependsOnUnboundedType() { |
| Source injectsUnboundedType = |
| CompilerTests.javaSource( |
| "test.InjectsUnboundedType", |
| "package test;", |
| "", |
| "import dagger.MembersInjector;", |
| "import java.util.ArrayList;", |
| "import javax.inject.Inject;", |
| "", |
| "class InjectsUnboundedType {", |
| " @Inject MembersInjector<ArrayList<?>> listInjector;", |
| "}"); |
| |
| CompilerTests.daggerCompiler(injectsUnboundedType) |
| .compile( |
| subject -> { |
| subject.hasErrorCount(1); |
| subject.hasErrorContaining( |
| "Cannot inject members into types with unbounded type arguments: " |
| + "java.util.ArrayList<?>") |
| .onSource(injectsUnboundedType) |
| .onLineContaining("@Inject MembersInjector<ArrayList<?>> listInjector;"); |
| }); |
| } |
| |
| @Test |
| public void membersInjectPrimitive() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "", |
| "@Component", |
| "interface TestComponent {", |
| " void inject(int primitive);", |
| "}"); |
| CompilerTests.daggerCompiler(component) |
| .compile( |
| subject -> { |
| subject.hasErrorCount(1); |
| subject.hasErrorContaining("Cannot inject members into int") |
| .onSource(component) |
| .onLineContaining("void inject(int primitive);"); |
| }); |
| } |
| |
| @Test |
| public void membersInjectArray() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "", |
| "@Component", |
| "interface TestComponent {", |
| " void inject(Object[] array);", |
| "}"); |
| CompilerTests.daggerCompiler(component) |
| .compile( |
| subject -> { |
| subject.hasErrorCount(1); |
| subject.hasErrorContaining("Cannot inject members into java.lang.Object[]") |
| .onSource(component) |
| .onLineContaining("void inject(Object[] array);"); |
| }); |
| } |
| |
| @Test |
| public void membersInjectorOfArray() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "import dagger.MembersInjector;", |
| "", |
| "@Component", |
| "interface TestComponent {", |
| " MembersInjector<Object[]> objectArrayInjector();", |
| "}"); |
| CompilerTests.daggerCompiler(component) |
| .compile( |
| subject -> { |
| subject.hasErrorCount(1); |
| subject.hasErrorContaining("Cannot inject members into java.lang.Object[]") |
| .onSource(component) |
| .onLineContaining("objectArrayInjector();"); |
| }); |
| } |
| |
| @Test |
| public void membersInjectRawType() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "import java.util.Set;", |
| "", |
| "@Component", |
| "interface TestComponent {", |
| " void inject(Set rawSet);", |
| "}"); |
| CompilerTests.daggerCompiler(component) |
| .compile( |
| subject -> { |
| subject.hasErrorCount(1); |
| subject.hasErrorContaining("Cannot inject members into raw type java.util.Set"); |
| }); |
| } |
| |
| @Test |
| public void qualifiedMembersInjector() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "import dagger.MembersInjector;", |
| "import javax.inject.Named;", |
| "", |
| "@Component", |
| "interface TestComponent {", |
| " @Named(\"foo\") MembersInjector<Object> objectInjector();", |
| "}"); |
| CompilerTests.daggerCompiler(component) |
| .compile( |
| subject -> { |
| subject.hasErrorCount(1); |
| subject.hasErrorContaining("Cannot inject members into qualified types") |
| .onSource(component) |
| .onLineContaining("objectInjector();"); |
| }); |
| } |
| |
| @Test |
| public void qualifiedMembersInjectionMethod() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "import dagger.MembersInjector;", |
| "import javax.inject.Named;", |
| "", |
| "@Component", |
| "interface TestComponent {", |
| " @Named(\"foo\") void injectObject(Object object);", |
| "}"); |
| CompilerTests.daggerCompiler(component) |
| .compile( |
| subject -> { |
| subject.hasErrorCount(1); |
| subject.hasErrorContaining("Cannot inject members into qualified types") |
| .onSource(component) |
| .onLineContaining("injectObject(Object object);"); |
| }); |
| } |
| |
| @Test |
| public void qualifiedMembersInjectionMethodParameter() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "import dagger.MembersInjector;", |
| "import javax.inject.Named;", |
| "", |
| "@Component", |
| "interface TestComponent {", |
| " void injectObject(@Named(\"foo\") Object object);", |
| "}"); |
| CompilerTests.daggerCompiler(component) |
| .compile( |
| subject -> { |
| subject.hasErrorCount(1); |
| subject.hasErrorContaining("Cannot inject members into qualified types") |
| .onSource(component) |
| .onLineContaining("injectObject(@Named(\"foo\") Object object);"); |
| }); |
| } |
| |
| @Test |
| public void staticFieldInjection() { |
| Source injected = |
| CompilerTests.javaSource( |
| "test.Injected", |
| "package test;", |
| "", |
| "import javax.inject.Inject;", |
| "", |
| "final class Injected {", |
| " @Inject static Object object;", |
| "}"); |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "", |
| "@Component", |
| "interface TestComponent {", |
| " void inject(Injected injected);", |
| "}"); |
| CompilerTests.daggerCompiler(injected, component) |
| .compile( |
| subject -> { |
| subject.hasErrorCount(2); |
| subject.hasErrorContaining("static fields") |
| .onSource(injected) |
| .onLine(6); |
| subject.hasErrorContaining( |
| "Injected cannot be provided without an @Inject constructor or an " |
| + "@Provides-annotated method."); |
| }); |
| } |
| |
| @Test |
| public void missingMembersInjectorForKotlinProperty() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "import dagger.internal.codegen.KotlinInjectedQualifier;", |
| "", |
| "@Component(modules = TestModule.class)", |
| "interface TestComponent {", |
| " void inject(KotlinInjectedQualifier injected);", |
| "}"); |
| Source module = |
| CompilerTests.javaSource( |
| "test.TestModule", |
| "package test;", |
| "", |
| "import dagger.Module;", |
| "import dagger.Provides;", |
| "import javax.inject.Named;", |
| "", |
| "@Module", |
| "class TestModule {", |
| " @Provides", |
| " @Named(\"TheString\")", |
| " String theString() { return \"\"; }", |
| "}"); |
| CompilerTests.daggerCompiler(component, module) |
| .compile( |
| subject -> { |
| switch (CompilerTests.backend(subject)) { |
| case KSP: |
| // KSP works fine in this case so we shouldn't expect any errors here. |
| subject.hasErrorCount(0); |
| break; |
| case JAVAC: |
| subject.hasErrorCount(2); |
| subject.hasErrorContaining( |
| "Unable to read annotations on an injected Kotlin property."); |
| subject.hasErrorContaining("KotlinInjectedQualifier cannot be provided"); |
| break; |
| } |
| }); |
| } |
| |
| @Test |
| public void memberInjectionForKotlinObjectFails() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "import dagger.internal.codegen.KotlinObjectWithMemberInjection;", |
| "", |
| "@Component(modules = TestModule.class)", |
| "interface TestComponent {", |
| " void inject(KotlinObjectWithMemberInjection injected);", |
| "}"); |
| CompilerTests.daggerCompiler(component, testModule) |
| .compile( |
| subject -> { |
| switch (CompilerTests.backend(subject)) { |
| case KSP: |
| subject.hasErrorCount(2); |
| break; |
| case JAVAC: |
| subject.hasErrorCount(3); |
| subject.hasErrorContaining( |
| "Dagger does not support injection into static fields"); |
| break; |
| } |
| subject.hasErrorContaining("Dagger does not support injection into Kotlin objects"); |
| subject.hasErrorContaining("KotlinObjectWithMemberInjection cannot be provided"); |
| }); |
| } |
| |
| @Test |
| public void setterMemberInjectionForKotlinObjectFails() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "import dagger.internal.codegen.KotlinObjectWithSetterMemberInjection;", |
| "", |
| "@Component(modules = TestModule.class)", |
| "interface TestComponent {", |
| " void inject(KotlinObjectWithSetterMemberInjection injected);", |
| "}"); |
| CompilerTests.daggerCompiler(component, testModule) |
| .compile( |
| subject -> { |
| subject.hasErrorCount(2); |
| subject.hasErrorContaining("Dagger does not support injection into Kotlin objects"); |
| subject.hasErrorContaining( |
| "KotlinObjectWithSetterMemberInjection cannot be provided without an " |
| + "@Provides-annotated method."); |
| }); |
| } |
| |
| @Test |
| public void memberInjectionForKotlinClassWithCompanionObjectFails() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "import dagger.internal.codegen.KotlinClassWithMemberInjectedCompanion;", |
| "", |
| "@Component(modules = TestModule.class)", |
| "interface TestComponent {", |
| " void inject(KotlinClassWithMemberInjectedCompanion injected);", |
| " void injectCompanion(KotlinClassWithMemberInjectedCompanion.Companion injected);", |
| "}"); |
| CompilerTests.daggerCompiler(component, testModule) |
| .compile( |
| subject -> { |
| switch (CompilerTests.backend(subject)) { |
| case KSP: |
| subject.hasErrorCount(4); |
| subject.hasErrorContaining( |
| "Dagger does not support injection into Kotlin objects"); |
| break; |
| case JAVAC: |
| subject.hasErrorCount(2); |
| break; |
| } |
| subject.hasErrorContaining("Dagger does not support injection into static fields"); |
| subject.hasErrorContaining( |
| "KotlinClassWithMemberInjectedCompanion cannot be provided"); |
| }); |
| } |
| |
| @Test |
| public void setterMemberInjectionForKotlinClassWithCompanionObjectFails() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "import dagger.internal.codegen.KotlinClassWithSetterMemberInjectedCompanion;", |
| "", |
| "@Component(modules = TestModule.class)", |
| "interface TestComponent {", |
| " void inject(KotlinClassWithSetterMemberInjectedCompanion.Companion injected);", |
| "}"); |
| CompilerTests.daggerCompiler(component, testModule) |
| .compile( |
| subject -> { |
| switch (CompilerTests.backend(subject)) { |
| case KSP: |
| // TODO(b/268257007): The KSP results should match KAPT once this bug is fixed. |
| subject.hasErrorCount(3); |
| subject.hasErrorContaining( |
| "Dagger does not support injection into static methods"); |
| break; |
| case JAVAC: |
| subject.hasErrorCount(2); |
| break; |
| } |
| subject.hasErrorContaining("Dagger does not support injection into Kotlin objects"); |
| subject.hasErrorContaining( |
| "KotlinClassWithSetterMemberInjectedCompanion.Companion cannot be provided"); |
| }); |
| } |
| |
| @Test |
| public void memberInjectionForKotlinClassWithNamedCompanionObjectFails() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "import dagger.internal.codegen.KotlinClassWithMemberInjectedNamedCompanion;", |
| "", |
| "@Component(modules = TestModule.class)", |
| "interface TestComponent {", |
| " void inject(KotlinClassWithMemberInjectedNamedCompanion injected);", |
| " void injectCompanion(KotlinClassWithMemberInjectedNamedCompanion.TheCompanion" |
| + " injected);", |
| "}"); |
| CompilerTests.daggerCompiler(component, testModule) |
| .compile( |
| subject -> { |
| switch (CompilerTests.backend(subject)) { |
| case KSP: |
| subject.hasErrorCount(4); |
| subject.hasErrorContaining( |
| "Dagger does not support injection into Kotlin objects"); |
| break; |
| case JAVAC: |
| subject.hasErrorCount(2); |
| break; |
| } |
| subject.hasErrorContaining("Dagger does not support injection into static fields"); |
| subject.hasErrorContaining( |
| "KotlinClassWithMemberInjectedNamedCompanion cannot be provided"); |
| }); |
| } |
| |
| @Test |
| public void setterMemberInjectionForKotlinClassWithNamedCompanionObjectFails() { |
| Source component = |
| CompilerTests.javaSource( |
| "test.TestComponent", |
| "package test;", |
| "", |
| "import dagger.Component;", |
| "import dagger.internal.codegen.KotlinClassWithSetterMemberInjectedNamedCompanion;", |
| "", |
| "@Component(modules = TestModule.class)", |
| "interface TestComponent {", |
| " void inject(", |
| " KotlinClassWithSetterMemberInjectedNamedCompanion.TheCompanion injected);", |
| "}"); |
| CompilerTests.daggerCompiler(component, testModule) |
| .compile( |
| subject -> { |
| switch (CompilerTests.backend(subject)) { |
| case KSP: |
| // TODO(b/268257007): The KSP results should match KAPT once this bug is fixed. |
| subject.hasErrorCount(3); |
| subject.hasErrorContaining( |
| "Dagger does not support injection into static methods"); |
| break; |
| case JAVAC: |
| subject.hasErrorCount(2); |
| break; |
| } |
| subject.hasErrorContaining("Dagger does not support injection into Kotlin objects"); |
| subject.hasErrorContaining( |
| "KotlinClassWithSetterMemberInjectedNamedCompanion.TheCompanion " |
| + "cannot be provided"); |
| }); |
| } |
| |
| private final Source testModule = |
| CompilerTests.javaSource( |
| "test.TestModule", |
| "package test;", |
| "", |
| "import dagger.Module;", |
| "import dagger.Provides;", |
| "", |
| "@Module", |
| "class TestModule {", |
| " @Provides", |
| " String theString() { return \"\"; }", |
| "}"); |
| } |