Привет, я использовал кинжал для инъекций зависимостей сетевого модуля, ApplicationModule, DatabaseModule, Presenters и интерактора в моем приложении.
Я хочу использовать эти же классы и модуль во время модульного тестирования.
В качестве эталонного тестирования я создал AndroidTestAppComponent, используя следующий код:
@Singleton
@Component(modules = {
AndroidSupportInjectionModule.class,
AndroidTestAppModule.class,
NetworkModule.class
})
public interface AndroidTestAppComponent extends AndroidInjector<AndroidTestApplication> {
@Component.Builder
abstract class AndroidTestAppComponentBuilder extends Builder<AndroidTestApplication> {
}
}
Предоставление всего модуля выходит за рамки этого вопроса, рассмотрим AndroidTestAppModule.java below
:
public class AndroidTestAppModule {
@Provides
@Singleton
Context provideContext(AndroidTestApplication application) {
return application.getApplicationContext();
}
@Singleton
@Provides
KeyguardManager provideKeyguardManager(Context context) {
return (KeyguardManager) context.getSystemService(Context.KEYGUARD_SERVICE);
}
@Singleton
@Provides
FingerprintManagerCompat providerFingerPrintManager(Context context) {
return FingerprintManagerCompat.from(context);
}
}
Я могу сгенерировать DaggerAndroidTestAppComponent
.
Мой класс приложения, как показано ниже:
public class AndroidTestApplication extends DaggerApplication implements HasActivityInjector {
@Inject
DispatchingAndroidInjector<Activity> dispatchingActivityInjector;
AndroidInjector<AndroidTestApplication> androidInjector;
@Override
public void onCreate() {
super.onCreate();
androidInjector.inject(this);
}
@Override
protected AndroidInjector<? extends DaggerApplication> applicationInjector() {
androidInjector = DaggerAndroidTestAppComponent.builder().create(this);
return androidInjector;
}
@Override
public DispatchingAndroidInjector<Activity> activityInjector() {
return dispatchingActivityInjector;
}
}
Некоторые другие AppPref.java
класс
@Singleton
public class AppPref {
private SharedPreferences preferences;
@Inject
AppPref(Context context) {
preferences = context.getSharedPreferences("somefile", Activity.MODE_PRIVATE);
}
}
Как видно из документации: AndroidInjection#inject(T t)
t здесь берет основной модуль android, поэтому, когда я вызываю это в своей Деятельности AndroidInjection.inject(activity_reference_usually__this__)
, это работает (нормальный сценарий, реальная сборка и приложение для тестирования)
Без изменения большого количества кода, как я могу использовать эти классы в AndroidInstrumentationTest, потому что я буду изменять только реализацию теста в Test**DaggerModules
внутри пакета теста.
Пример кода для приборов приведен ниже:
@RunWith(AndroidJUnit4.class)
public class ExampleInstrumentedTest {
AndroidTestApplication application;
@Inject
AppPref appPref;
@Before
public void setUp() throws IllegalAccessException, InstantiationException, ClassNotFoundException {
Context appContext = InstrumentationRegistry.getTargetContext();
application = (AndroidTestApplication) Instrumentation.newApplication(AndroidTestApplication.class, appContext);
DaggerAndroidTestAppComponent.builder().create(application).inject(application);
}
@Test
public void useAppContext() {
// Context of the app under test.
Context appContext = InstrumentationRegistry.getTargetContext();
assertEquals("com.a.b", appContext.getPackageName());
}
@Test
public void testPreNotNUll() {
Assert.assertNotNull(appPref);
}
}
В идеале apppref имеет значение null, поскольку в методе setUp
я ввел класс AndroidTestApplication, а не в ExampleInstrumentedTest
как я могу отредактировать свой код dagger2, чтобы @Inject работал нормально, и я получил действительный объект appPref.
Спасибо.