Room.inMemoryDatabaseBuilder() не найден в инструментальном тесте

Я пытаюсь создать инструментальный тест для Room DB, который есть в моем приложении, но я получаю эту ошибку:

java.lang.NoSuchMethodError: No static method inMemoryDatabaseBuilder(Landroid/content/Context;Ljava/lang/Class;)Landroid/arch/persistence/room/RoomDatabase$Builder; in class Landroid/arch/persistence/room/Room; or its super classes (declaration of 'android.arch.persistence.room.Room' appears in /data/app/com.example.android-nhsbta60PW8T0UpasgtR0B==/base.apk)
at com.example.android.FooDaoTest.<init>(FooDaoTest.kt:40)
at java.lang.reflect.Constructor.newInstance0(Native Method)
at java.lang.reflect.Constructor.newInstance(Constructor.java:334)
at org.junit.runners.BlockJUnit4ClassRunner.createTest(BlockJUnit4ClassRunner.java:217)
at org.junit.runners.BlockJUnit4ClassRunner$1.runReflectiveCall(BlockJUnit4ClassRunner.java:266)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.BlockJUnit4ClassRunner.methodBlock(BlockJUnit4ClassRunner.java:263)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:78)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:57)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.junit.runners.Suite.runChild(Suite.java:128)
at org.junit.runners.Suite.runChild(Suite.java:27)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
at org.junit.runner.JUnitCore.run(JUnitCore.java:115)
at android.support.test.internal.runner.TestExecutor.execute(TestExecutor.java:58)
at android.support.test.runner.AndroidJUnitRunner.onStart(AndroidJUnitRunner.java:375)
at android.app.Instrumentation$InstrumentationThread.run(Instrumentation.java:2074)

Это моя конфигурация сборки Gradle:

проект:

buildscript {
    ext {
        kotlinVersion = '1.2.0'

        supportLibVersion = '26.1.0'
        archLibVersion = '1.0.0'

        ...

        // Test libs
        androidTestLibVersion = '1.0.1'
        espressoVersion = '3.0.1'
        powerMockVersion = '1.7.1'
    }

    repositories {
        google()
        jcenter()
    }

    dependencies {
        classpath 'com.android.tools.build:gradle:3.0.1'
        classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlinVersion"
    }
}

allprojects {
    repositories {
        google()
        jcenter()
    }
}

приложение:

android {
    compileSdkVersion 26
    buildToolsVersion '26.0.3'

    defaultConfig {
        applicationId "com.example.android"

        minSdkVersion 17
        targetSdkVersion 26

        versionCode 1
        versionName "0.1"

        ...

        multiDexEnabled true

        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
    }

    buildTypes {
        debug {
            minifyEnabled true
            useProguard false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
        release {
            minifyEnabled true
            shrinkResources true
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])

    implementation 'com.android.support:multidex:1.0.2'

    implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk7:$kotlinVersion"

    ...

    implementation "android.arch.lifecycle:extensions:$archLibVersion"
    implementation "android.arch.persistence.room:runtime:$archLibVersion"
    kapt "android.arch.lifecycle:compiler:$archLibVersion"
    kapt "android.arch.persistence.room:compiler:$archLibVersion"

    ...

    // Test libs

    testImplementation 'junit:junit:4.12'
    testImplementation 'org.mockito:mockito-core:2.8.47'
    testImplementation "org.powermock:powermock-module-junit4:$powerMockVersion"
    testImplementation "org.powermock:powermock-api-mockito2:$powerMockVersion"
    testImplementation 'org.hamcrest:hamcrest-library:1.3'
    testImplementation "android.arch.core:core-testing:$archLibVersion"
    testImplementation "com.squareup.okhttp3:mockwebserver:3.9.1"

    androidTestImplementation "com.android.support.test:runner:$androidTestLibVersion"
    androidTestImplementation "com.android.support.test:rules:$androidTestLibVersion"
    androidTestImplementation("com.android.support.test.espresso:espresso-core:$espressoVersion", {
        exclude group: 'com.android.support', module: 'support-annotations'
    })
    androidTestImplementation "com.android.support.test.espresso:espresso-contrib:$espressoVersion"
    androidTestImplementation "com.android.support.test.espresso:espresso-intents:$espressoVersion"
    androidTestImplementation "com.android.support.test.espresso:espresso-idling-resource:$espressoVersion"
    androidTestImplementation 'com.android.support.test.uiautomator:uiautomator-v18:2.1.3'

    androidTestUtil "com.android.support.test:orchestrator:$androidTestLibVersion"
}

И это мой инструментальный тест:

@RunWith(AndroidJUnit4::class)
class FooDaoTest {

    private lateinit var db: AppDatabase

    private lateinit var fooDao: FooDao


    @Before
    fun setUp() {
        db = Room.inMemoryDatabaseBuilder(InstrumentationRegistry.getContext(), AppDatabase::class.java).build()

        fooDao = db.fooDao()
    }

    @After
    fun closeDb() {
        // db.close()
    }

    @Test
    fun save() {
        ...
    }
}

Я предполагаю, что ошибка связана с мультидексом, но я не могу понять, почему. Кто-нибудь знает, почему это происходит и как я могу заставить этот тест работать?

1 ответ

Решение

Кажется, это была какая-то проблема со сборкой, которую я имел, я выкладываю то, что исправило это для меня на случай, если это поможет другим:

Я много раз пытался очистить проект от Android Studio, но проблема не исчезла, когда я очистил проект из командной строки, используя:

./gradlew clean

и затем запустите тест снова, это сработало.

ОБНОВИТЬ:

После еще одного тестирования я обнаружил, что настоящая проблема была с моей конфигурацией Proguard. Я использую Proguard в сборках отладки, которая удаляет все неиспользуемые классы / методы (Room.inMemoryDatabaseBuilder() будучи одним из них). Как только я отключил Proguard для отладочных сборок и очистил проект, как указано выше, он начал работать. Итак, чтобы подвести итог, решение:

  1. Отключите Proguard для отладочной сборки (или добавьте тестовую конфигурацию Proguard в отладочную сборку):

Опция 1:

buildTypes {
//      debug {
//          minifyEnabled true
//          useProguard false
//          proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
//      }
        ...
}

Вариант 2:

buildTypes {
        debug {
            minifyEnabled true
            useProguard false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'

            testProguardFiles 'proguard-rules-test.pro'
        }
        ...
}

Содержание proguard-rules-test.pro будет зависеть от того, что вам нужно для вашего приложения.

  1. Очистите проект из командной строки:

    ./gradlew clean

Другие вопросы по тегам