Тесты Robolectric ContentProvider

Я заметил, что время от времени возникает вопрос об использовании Robolectric для тестирования пользовательских ContentProviders. Тем не менее, никогда не было конкретного и однозначного ответа о том, как это сделать должным образом. Я наткнулся на два разных подхода:

Тем не менее, я получаю исключение java.lang.InstantiationException с обоими подходами. Были некоторые сообщения SO, в которых говорится, что это связано с тем, что SQLiteDatabase.rawQueryWithFactory(SQLiteDatabase.java) не переопределяется в Robolectric (Android + Robolectric - исключение RuntimeException/InstantiationException в queryBuilder.query() в ContentProvider).

Я предполагаю, что мой вопрос: есть ли предпочтительные обходные пути, которые делают тестирование ContentProviders возможным. Или есть другие подходы, которые лучше, чем те, которые упомянуты выше.

Ответ 1

Это тест Robolectric, который отлично работал у меня:

import android.content.ContentResolver;
import android.database.Cursor;
import android.net.Uri;

import org.joda.time.LocalDate;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.robolectric.Robolectric;
import org.robolectric.RobolectricTestRunner;
import org.robolectric.annotation.Config;
import org.robolectric.shadows.ShadowContentResolver;

import co.tomup.app.db.DbSchema;
import co.tomup.app.db.TomupContentProvider;
import co.tomup.app.db.model.CalendarDay;
import co.tomup.app.db.tables.CalendarDayTable;

import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertTrue;

@Config(emulateSdk = 18)
@RunWith(RobolectricTestRunner.class)
public class CalendarDayProviderTest {

    private ContentResolver mContentResolver;
    private ShadowContentResolver mShadowContentResolver;
    private TomupContentProvider mProvider;

    @Before
    public void setup() {
        mProvider = new TomupContentProvider();
        mContentResolver = Robolectric.application.getContentResolver();
        mShadowContentResolver = Robolectric.shadowOf(mContentResolver);
        mProvider.onCreate();
        ShadowContentResolver.registerProvider(DbSchema.AUTHORITY, mProvider);
    }

    @Test
    public void testInsertAndDelete() {
        // insert
        CalendarDay calendarDay = new CalendarDay();
        calendarDay.setId(1L);
        calendarDay.setDay(new LocalDate());
        calendarDay.setMoonPhase("new");
        calendarDay.setSunrise(1);
        calendarDay.setSunset(100);
        Uri insertionId = mContentResolver.insert(CalendarDayTable.CONTENT_URI,
                calendarDay.toSQLiteContentValues());
        Cursor cursorCheck = mShadowContentResolver.query(CalendarDayTable.CONTENT_URI,
                null, null, null, null);
        while (cursorCheck.moveToNext()) {
            CalendarDay calendarDayCheck = CalendarDay.fromSQLiteCursor(cursorCheck);
            assertEquals(calendarDay, calendarDayCheck);
        }
        assertTrue(cursorCheck.getCount() > 0);
        // delete
        mShadowContentResolver.delete(insertionId,
                null, null);
        cursorCheck = mShadowContentResolver.query(CalendarDayTable.CONTENT_URI,
                null, null, null, null);
        assertTrue(cursorCheck.getCount() == 0);
    }
}

Ответ 2

Все, что вам нужно сделать, настроено ShadowContentResolver перед тестированием, поэтому оно будет правильно связывать полномочия вашего ContentProvider с ContentProvider. Вот пример:

ShadowContentResolver.registerProvider(
        "com.example.provider",    //authority of your provider
        contentProvider    //instance of your ContentProvider (you can just use default constructor)    
);

Самый простой способ - поместить этот материал в некоторый метод аннотирования @Before. Однако более правильным (и, следовательно, лучшим в долгосрочной перспективе) способом является включение этого метода в ваш метод TestApplication#onCreate, поэтому эта конфигурация будет использоваться всеми тестами в вашем приложении.