Проблема с Android Studio InstrumentedTest с комнатой - PullRequest
0 голосов
/ 17 ноября 2018

Я не могу понять, почему мой инструментальный тестовый модуль не проходит в позиции, указанной в первом блоке кода ниже. Насколько я вижу, я следил за продемонстрированными примерами онлайн для использования библиотеки постоянства Room, с тестом, основанным на сайте разработчика Android. Тестируемый код получен из Google CodeLab "комната с видом". Тест вставляет объект и читает список этих объектов, завернутый в LiveData. Только что вставленный объект должен быть в возвращаемом списке, однако в LiveData возвращается ноль.

Контрольный пример:

@RunWith(AndroidJUnit4.class)
public class PersonReadWriteTest
{
private PersonDAO personDao;
private EventDatabase database;

@Before
public void createDb ()
{
    Context appContext = InstrumentationRegistry.getTargetContext();
    database = Room.inMemoryDatabaseBuilder(appContext, EventDatabase.class).build();
    personDao = database.personDAO();
}

@After
public void closeDb () throws IOException
{
    database.close();
}

@Test
public void writePersonAndReadInList() throws Exception
{
    Person person = TestUtil.createPerson("John", "Doe");
    personDao.insert(person);
    LiveData<List<Person>> peopleLive = personDao.getAll();
    List<Person> people = peopleLive.getValue();
    assertNotNull(people); <=============================== FAILS HERE
    assertThat(people.size(), equalTo(1));
    Person read = people.get(0);
    assertNotNull(read);
    assertThat(read, equalTo(person));
}
}

Результат теста:

java.lang.AssertionError
at org.junit.Assert.fail(Assert.java:86)
at org.junit.Assert.assertTrue(Assert.java:41)
at org.junit.Assert.assertNotNull(Assert.java:712)
at org.junit.Assert.assertNotNull(Assert.java:722)
at PersonReadWriteTest.writePersonAndReadInList(PersonReadWriteTest.java:69)
at java.lang.reflect.Method.invoke(Native Method)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at android.support.test.internal.runner.junit4.statement.RunBefores.evaluate(RunBefores.java:80)
at android.support.test.internal.runner.junit4.statement.RunAfters.evaluate(RunAfters.java:61)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
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 android.support.test.runner.AndroidJUnit4.run(AndroidJUnit4.java:101)
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:56)
at android.support.test.runner.AndroidJUnitRunner.onStart(AndroidJUnitRunner.java:384)
at android.app.Instrumentation$InstrumentationThread.run(Instrumentation.java:2106)

Физическое лицо:

@Entity(tableName="people")
public class Person implements Serializable
{
// ---------------------------------------------------------------
// Attributes

@PrimaryKey
private Long id = null;
@ColumnInfo(name="first_name")
private String firstName = null;
@ColumnInfo(name="last_name")
private String surname = null;
@ColumnInfo(name="address")
private String address = null;
@ColumnInfo(name="phone1")
private String phone1 = null;
@ColumnInfo(name="phone2")
private String phone2 = null;
@ColumnInfo(name="email")
private String email = null;

// ---------------------------------------------------------------
// Attribute access

... all the public getters and setters...

// -----------------------------------------------------------------

public Person ()
{}
}

DAO:

@Dao
public interface PersonDAO
{
@Query("SELECT * FROM people")
LiveData<List<Person>> getAll ();

@Query("SELECT * FROM people WHERE id IN (:ids)")
LiveData<List<Person>> loadAllByIds (int[] ids);

@Query("SELECT * FROM people WHERE first_name LIKE :first AND last_name LIKE :last LIMIT 1")
LiveData<Person> findByName (String first, String last);

@Insert(onConflict = OnConflictStrategy.REPLACE)
void insert (Person person);

@Delete
void delete (Person person);

@Query("DELETE from people")
void deleteAll ();
}

Класс базы данных:

@Database(entities = { Event.class, Person.class }, version = 1)
public abstract class EventDatabase extends RoomDatabase
{
public abstract EventDAO eventDAO ();
public abstract PersonDAO personDAO ();

private static volatile EventDatabase INSTANCE;

static EventDatabase getDatabase (final Context context)
{
    if (INSTANCE == null)
    {
        synchronized (EventDatabase.class)
        {
            if (INSTANCE == null)
            {
                INSTANCE = Room.databaseBuilder(context.getApplicationContext(),
                        EventDatabase.class, "event_database").
                        allowMainThreadQueries(). // SHOULD NOT BE USED IN PRODUCTION !!!
                        addCallback(sRoomDatabaseCallback).build();
            }
        }
    }
    return INSTANCE;
}

private static RoomDatabase.Callback sRoomDatabaseCallback = new RoomDatabase.Callback() {
    @Override
    public void onOpen (@NonNull SupportSQLiteDatabase db)
    {
        super.onOpen(db);
        new PopulateDbAsync(INSTANCE).execute();
    }
};

private static class PopulateDbAsync extends AsyncTask<Void, Void, Void>
{
    private final EventDAO m_EventDao;
    private final PersonDAO m_PersonDao;

    PopulateDbAsync (EventDatabase db)
    {
        m_EventDao = db.eventDAO();
        m_PersonDao = db.personDAO();
    }

    @Override
    protected Void doInBackground (final Void... params)
    {
        Event event;
        m_EventDao.deleteAll();
        m_PersonDao.deleteAll();

        ... insert objects ...
        For example:
        Person person = new Person("Joe", "Bloe");
        m_PersonDao.insert(person);

        return null;
    }
}

}

Извлечение из файла зависимостей файла приложения:

testImplementation 'junit:junit:4.12'
androidTestImplementation 'com.android.support:support-annotations:27.1.1'
androidTestImplementation 'com.android.support.test:runner:1.0.2'
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.2'

// Lifecycle components
def archLifecycleVersion = "1.1.0"
implementation "android.arch.lifecycle:extensions:$archLifecycleVersion"
annotationProcessor "android.arch.lifecycle:compiler:$archLifecycleVersion"
implementation "android.arch.lifecycle:viewmodel:$archLifecycleVersion"
implementation "android.arch.lifecycle:livedata:$archLifecycleVersion"

// Room components
def room_version = "1.1.1"
implementation "android.arch.persistence.room:runtime:$room_version"
annotationProcessor "android.arch.persistence.room:compiler:$room_version" // use kapt for Kotlin
androidTestImplementation "android.arch.persistence.room:testing:$room_version"

1 Ответ

0 голосов
/ 22 декабря 2018

для синхронной проверки содержимого LiveData необходимо использовать экземпляр InstantTaskExecutorRule, а также наблюдать за экземпляром LiveData на предмет изменений.

Вы можетесделать это, зарегистрировав observer.

Я недавно написал в блоге о хорошем решении здесь .

...