在 android.os.Looper 中使用 LiveData 方法 getMainLooper 进行单元测试未模拟

Iba*_*ola 5 junit android unit-testing android-looper android-livedata

在尝试进行单元测试时,我无法让 liveData.postValue 工作。我一直在谷歌检查解决方案,这是我现在的代码。

public class ProjectListViewModelTest {

    GetProjectList getProjectList = Mockito.mock(GetProjectList.class);
    ProjectModel.Project project = new ProjectModel.Project("testing",
            "this is a test",
            "https://logo.jpg",
            new ProjectModel.Company("cat"),
            "20150404",
            "active");
    List<ProjectModel.Project> projects = Arrays.asList(project);
    ProjectModel.ProjectList projectsList = new ProjectModel.ProjectList(projects);

    ProjectsListViewModel projectsListViewModel;

    private PublishSubject<ProjectModel.ProjectList> projectsListPublishSubject = PublishSubject.create();

    @Rule public InstantTaskExecutorRule instantExecutorRule = new InstantTaskExecutorRule();

    @BeforeClass
    public static void setUpRxSchedulers() {
        Scheduler immediate = new Scheduler() {
            @Override
            public Disposable scheduleDirect(@NonNull Runnable run, long delay, @NonNull TimeUnit unit) {
                return super.scheduleDirect(run, 0, unit);
            }

            @Override
            public Scheduler.Worker createWorker() {
                return new ExecutorScheduler.ExecutorWorker(Runnable::run);
            }
        };

        RxJavaPlugins.setInitIoSchedulerHandler(scheduler -> immediate);
        RxJavaPlugins.setInitComputationSchedulerHandler(scheduler -> immediate);
        RxJavaPlugins.setInitNewThreadSchedulerHandler(scheduler -> immediate);
        RxJavaPlugins.setInitSingleSchedulerHandler(scheduler -> immediate);
        RxAndroidPlugins.setInitMainThreadSchedulerHandler(scheduler -> immediate);
    }

    @Before
    @Throws(exceptionClasses = Exception.class)
    public void setUp() {
        MockitoAnnotations.initMocks(this);
        projectsListViewModel = new ProjectsListViewModel(getProjectList);
        when(getProjectList.execute()).thenReturn(projectsListPublishSubject.take(1).singleOrError());
    }

    @Test
    public void testExecuteGetProjectsListSuccess() {
        LiveData<List<ProjectModel.MapProject>> liveData = projectsListViewModel.getLiveData();
        ProjectModel.MapProject expectedResult = new ProjectModel.MapProject(
                "testing", "this is a test", "https://logo.jpg",
                "cat", "2015-04-04", "active");
        projectsListViewModel.getProjects();
        projectsListPublishSubject.onNext(projectsList);
        Assert.assertEquals(expectedResult, liveData.getValue().get(0));
    }

    @After
    public void tearDownClass(){
        RxAndroidPlugins.reset();
    }
Run Code Online (Sandbox Code Playgroud)

setUpRxSchedulers为了避免Method getMainLooper in android.os.Looper not mocked与 Rx相同的错误 ( ) ,我所拥有的代码是强制性的。但是我无法解决在调用 liveData.post(projectList) 时遇到的这个错误。在我检查过的所有论坛中,他们都说@Rule public InstantTaskExecutorRule instantExecutorRule = new InstantTaskExecutorRule();应该解决问题。但不是我的情况。

我也把视图模型放在这里,以防它可以提供帮助:

public class ProjectsListViewModel extends ViewModel {

    GetProjectList getProjectList;
    MutableLiveData<List<ProjectModel.MapProject>> liveData = new MutableLiveData<>();

    public ProjectsListViewModel(GetProjectList getProjectList){
        this.getProjectList = getProjectList;
    }

    public LiveData<List<ProjectModel.MapProject>> getLiveData(){
        return liveData;
    }

    public void getProjects(){
        getProjectList.execute()
        .subscribeOn(Schedulers.io())
        .observeOn(AndroidSchedulers.mainThread())
                .map(ProjectModel.ProjectList::getProjects)
                .toObservable().flatMapIterable(projects -> projects)
                .map(project -> project.convertToMapProject()).toList()
        .subscribe(projectsList ->
            liveData.setValue(projectsList));
    }

}
Run Code Online (Sandbox Code Playgroud)

mar*_*ino 3

使用InstantTaskExecutorRule实际上可以解决这个问题。

我认为问题在于 JUnit 5 中@Rule似乎不再支持注释(因为扩展现在是可行的方法)。代码将成功编译,但规则不会应用。

对此有(至少)两种解决方案:

使用 JUnit 4

肯定是更快,但可能不是最好的,具体取决于您需要 JUnit 5 的程度。

只需将方法的注释setup从更改@BeforeEach@Before@Test从 JUnit 4 导入注释即可完成此操作。导入应如下所示。

import org.junit.Before
import org.junit.Rule
import org.junit.Test
Run Code Online (Sandbox Code Playgroud)

实施一个InstantTaskExecutorExtension

如果您关心使用 JUnit 5,这会更好:)

这里有一篇文章讲了如何精确实现InstantTaskExecutorExtension。完成后,请记住使用@ExtendWith(InstantTaskExecutorExtension::class)注释而不是@Rule!将其应用到您的测试类。