2016-05-20 35 views
7

Ho bisogno di alcuni consigli su come prendere in giro una api di riposo. La mia applicazione è in architettura MVP.Unit Test di Retrofit Chiamata 2 api con Mockito

mio interfaccia per API:

public interface MyAPI { 

    @GET("{cmd}/{userName}/{password}") 
    Observable<Response> login(
     @Path("cmd") String cmd, 
     @Path("userName") String userName, 
     @Path("password") String password 
    ); 

Il mio servizio:

public class MyService implements IService { 

    private static MyService mInstance = new MyService(); 
    private MyAPI mApi; 

    public static MyService getInstance() { 
     return mInstance; 
    } 

    private MyService() { 

     OkHttpClient.Builder httpClientBuilder = new OkHttpClient.Builder(); 
     httpClientBuilder.connectTimeout(Config.DEFAULT_TIMEOUT, TimeUnit.SECONDS); 

     Retrofit retrofit = new Retrofit.Builder() 
      .baseUrl(Config.kBaseUrl) 
      .addConverterFactory(GsonConverterFactory.create()) 
      .addCallAdapterFactory(RxJavaCallAdapterFactory.create()) 
      .client(httpClientBuilder.build()) 
      .build(); 

     this.mApi = retrofit.create(MyAPI.class); 
    } 

    public void login(
     Subscriber<Response> subscriber, 
     String userName, 
     String password) { 
     mApi.login(Config.kLoginCmd,userName,password) 
      .subscribeOn(Schedulers.io()) 
      .unsubscribeOn(Schedulers.io()) 
      .observeOn(AndroidSchedulers.mainThread()) 
      .subscribe(subscriber); 
    } 

La mia classe presentatore:

public class LoginPresenter implements LoginContract.Presenter { 

    LoginContract.View mView; 
    IService mService; 
    ISession mSession; 

    public LoginPresenter(LoginContract.View loginView, IService service, ISession session) { 
     mView = loginView; 
     mService = service; 
     mSession = session; 
    } 

    @Override 
    public void login(String email, String password) { 

     Subscriber<Response> subscriber = new Subscriber<Response>() { 
      @Override 
      public void onCompleted() { 
       mView.showLoading(false); 
      } 

      @Override 
      public void onError(Throwable e) { 
       mView.showError(e.getLocalizedMessage()); 
      } 

      @Override 
      public void onNext(Response response) { 
       if (response.getResults().getStatus().equalsIgnoreCase(Config.kResultCodeOK)) { 
        mView.loginSuccess(); 
       } else { 
        mView.showError(response.getResults().getStatus().getErrmsg()); 
       } 
      } 
     }; 

     mView.showLoading(true); 
     mService.login(
      subscriber, 
      email, 
      password); 
    } 

C'è un altro modo per testare il mio presentatore scrivendo un servizio Mock . Ma non mi piace così tanto e penso che Mockito potrebbe aiutare.

Qui è la mia classe di test:

public class LoginPresenterMockTest { 

    private LoginPresenter mLoginPresenter; 

    @Mock 
    LoginContract.View view; 
    @Mock 
    IService service; 
    @Mock 
    ISession session; 

    @Before 
    public void setup() throws Exception { 
     MockitoAnnotations.initMocks(this); 
     mLoginPresenter = new LoginPresenter(view, service, session); 
    } 

    @Test 
    public void testLoginWithCorrectUserNameAndPassword() throws Exception { 
     mLoginPresenter.login("[email protected]","password"); 
     verify(view).loginSuccess(); 
    } 

} 

Quello che voglio fare è che deridere il loginSuccess chiamata dati di risposta() quando la risposta è corretta.

Ovviamente il mio test corrente non funzionerà. Ho bisogno di alcuni consigli su come deriderlo? Qualche idea? Grazie.

risposta

5

Si può fare in modo successivo:

@Test 
public void testLoginWithCorrectUserNameAndPassword() throws Exception { 
    // create or mock response object 
    when(service.login(anyString(), anyString(), anyString).thenReturn(Observable.just(response)); 
    mLoginPresenter.login("[email protected]","password"); 
    verify(view).loginSuccess(); 
} 

@Test 
public void testLoginWithIncorrectUserNameAndPassword() throws Exception { 
    // create or mock response object 
    when(service.login(anyString(), anyString(), anyString).thenReturn(Observable.<Response>error(new IOException())); 
    mLoginPresenter.login("[email protected]","password"); 
    verify(view).showError(anyString); 
} 
+0

'' 'Mockito.when (service.login (any(), anyString(), anyString()). thenReturn (Observable.just (response)); '' 'Come specificare il primo parametro? È Subscriber . Non riesco a trasmetterlo a any(). Grazie? –

+0

scusate, non ho notato che si passa abbonato. Penso che sia meglio c appendi il tuo metodo di servizio che restituisce Osservabile e iscriviti su di esso nella classe presentatore. In questo caso il codice fornito funzionerà. Altrimenti dovresti prendere in giro mApi. –

+0

Risolvi il problema con ** ArgumentCaptor ** –

3

Grazie per @Ilya Tretyakov, sono uscito questa soluzione:

private ArgumentCaptor<Subscriber<Response>> subscriberArgumentCaptor; 

@Test 
public void testLoginWithCorrectUserNameAndPassword() throws Exception { 
    mLoginPresenter.login("[email protected]","password"); 
    // create the mock Response object 
    Response response = ...... 

    verify(service, times(1)).login(
     subscriberArgumentCaptor.capture(), 
     stringUserNameCaptor.capture(), 
     stringPasswordCaptor.capture() 
    ); 

    subscriberArgumentCaptor.getValue().onNext(response); 
    verify(view).loginSuccess(); 
}