SlideShare uma empresa Scribd logo
Indo Além com Automação e Testes de Apps
Android
Eduardo Carrara
@DuCarrara
Você já se sentiu
assim em seus
projetos?
Infelizmente não há bala de prata...
Precisamos de segurança,
para Iterações Rápidas.
Testes Automatizados podem ajudar
Então o que vamos explorar hoje?
Recursos Externos & Test Flakiness
Simplificação de Dependências com Mocks
UI Tests e as Camadas de Testes
Recursos Externos e Test Flakiness
“Non-deterministic tests have two problems,
firstly they are useless, secondly they are a
virulent infection that can completely ruin your
entire test suite.” - Martin Fowler
Test Flakiness
Falham e passam aleatoriamente
Diminuem a confiança na suite de testes
Escondem bugs
São difíceis de corrigir
Principais Causas
Concorrência
Dependência de comportamentos indefinidos ou não determinísticos
Código de terceiros que falham e funcionam aleatoriamente
Problemas de infraestrutura
<adicione sua causa aqui>
O que acontece no Android?
Rede
Recursos de Hardware do telefone (ex.: sensores)
Interação com outros apps
Dependências de serviços, apis e sdks externos
Exemplificando
Caso comum: chamadas a serviços REST
Dependência de recursos de rede, disponibilidade de serviços etc
Como resolver o flakiness no teste da chamada de um serviço?
Resposta: Desacoplar e Isolar
Desacoplando e Isolando
Presentation Domain Data
Activity
Fragment
Views
Use Cases
Repositories
Biz Entities
Data Sources
Data Entities
RestAPI
Network
Latency
Service Errs.
etc...
Origem do flakiness
Desacoplando e Isolando
Data
Layer
RestAPI
Server
RestApi
Connection
RestApi
Interface
API Calls
Retrofit
OkHttp
etc
<<builds>>
<<network calls>>
Produção
App
Config
Base URL
<<provides>>
<<injects>>
Desacoplando e Isolando
Test
Mock Local
Server
RestApi
Connection
RestApi
Interface
Retrofit
OkHttp
etc
<<builds>>
<<local calls>>
Teste
Test URL
Test Responses
<<provides>>
<<injects>>
<<creates>>
E o código...
Com Retrofit + OkHttp + MockWebServer
dependencies {
testCompile 'com.squareup.okhttp3:mockwebserver:3.3.0'
}
E o código...
@Before public void setUp() throws IOException {
prepareExpectedTestData();
server = new MockWebServer();
server.enqueue(
new MockResponse()
.setBody(TestServiceResponseFor20150907.getJson())
);
server.start();
restApi = new RestApi(server.url("/").toString());
}
@After public void tearDown() throws IOException {
server.shutdown();
}
Exemplo em NEOM on Github: ListNearEarthObjectsTest.java
E o código...
@Test
public void testSuccessfulListNearEarthObjects() throws Exception {
List<NearEarthObjectJsonEntity> nearEarthObjectJsonEntityList =
restApi.listNearEarthObjectsFor(requestDate);
assertThat(nearEarthObjectJsonEntityList,
is(expectedNearEarthObjectJsonEntityList));
}
Exemplo em NEOM on Github: ListNearEarthObjectsTest.java
Em nosso exemplo...
Isolamos a chamada à nossos serviços
Substituímos o servidor por um componente sob nosso controle
Podemos testar esta camada com testes unitários
Tomamos controle sobre um recurso que poderia ser Flaky
Simplificação com Mocks
Dependências
+ Dependências = + Acoplamento
Alto acoplamento dificulta a criação de testes e a automação
Sistemas 100% desacoplados são raros
Mocking
Objetos dublês
Controle sobre o comportamento e estado
Isolamento da unidade a ser testada
Deve nos ajudar a lidar com as dependências de maneira simples
O que acontece no Android?
Dependência de componentes do framework é frequente
Ex.: Context, Dados de Sensores, Services, Activities etc
Complexidade dos testes aumenta muito
Mockito
API fácil de entender
Configuração de comportamentos e Stubs
Permite a verificação de comportamentos
Mas como funciona?
Exemplo em NEOM on Github: NearEarthObjectListPresenterTest.java
@RunWith(MockitoJUnitRunner. class)
public class NearEarthObjectListPresenterTest {
NearEarthObjectListPresenter presenter;
@Mock private ObservableInteractor<List<NearEarthObject>> mockInteractor;
@Mock private NearEarthObjectsListView mockView;
@Captor
private ArgumentCaptor<Observer<List<NearEarthObject>>> observerArgumentCaptor;
@Before private void setUp() {
presenter = new NearEarthObjectListPresenter( mockInteractor);
}
}
Mas como funciona?
Exemplo em NEOM on Github: NearEarthObjectListPresenterTest.java
@Test public void testSuccessfulPresenterInitialization() {
presenter.attachTo(mockView);
verify(mockView).hideRetry();
verify(mockView).showLoading();
// Here we capture the observer registration to mock the update call
verify(mockInteractor, times(1))
.addObserver(observerArgumentCaptor.capture());
verify(mockInteractor).execute();
observerArgumentCaptor.getValue()
.update(mockInteractor, new ArrayList<NearEarthObject>());
verify(mockView).renderNearEarthObjectsList(any(List.class));
verify(mockView).hideLoading();
}
Robolectric
Testes que dependem do Android na JVM
Provê mocks e stubs ao Android SDK
Execução mais rápida de testes
Demora para disponibilizar versões mais recentes do SDK
Mas como funciona?
Exemplo em NEOM on Github: NearEarthObjectsListActivityTest.java
@RunWith(RobolectricGradleTestRunner.class)
@Config(constants = BuildConfig.class, sdk = Build.VERSION_CODES.LOLLIPOP)
public class NearEarthObjectsListActivityTest {
private NearEarthObjectsListActivity nearEarthObjectsListActivity;
@Before
public void setup() {
nearEarthObjectsListActivity =
Robolectric.setupActivity(NearEarthObjectsListActivity.class);
}
}
Mas como funciona?
Exemplo em NEOM on Github: NearEarthObjectsListActivityTest.java
@Test
public void testErrorMessageDisplay() {
final String expectedErrorMessage = "Error message";
nearEarthObjectsListActivity.showError(expectedErrorMessage);
View errorView = findById(nearEarthObjectsListActivity, R.id.error_view);
TextView errorTextView = findById(nearEarthObjectsListActivity,
R.id.error_message_text_view);
assertThat(errorView.getVisibility(), is(VISIBLE));
assertThat(errorTextView.getText().toString(), is(expectedErrorMessage));
}
+
Combinação Poderosa
Testes Rodando na JVM Local = Produtividade
Ajuda com dependências indiretas do Android Framework
Permite o mocking e uso de classes do Android na JVM Local
Exemplificando
Caso: Uso de dados obtidos de um Content Provider
O Content Provider nos provê um Cursor
Como testar um componente que dependa de um Cursor?
Resposta: Mocks*
* A classe MockCursor do Android Framework foi Deprecated na API Level 24. Uma das sugestões é usar o Mockito com ATSL.
Mocking
Component
Content
Provider
Content
Resolver
<<creates>><<uses>>
Produção
<<queries>>
Cursor
<<queries>>
Mocking
Component
Content
Provider
Content
Resolver
<<creates>><<uses>>
Teste
<<queries>>
Cursor
<<queries>>
Componente
a ser Testado
Boilerplate
Mocking
Component
<<mocks>>
<<uses>>
Teste
Cursor
Test
<<tests>>
E o código...
Com Mockito + Robolectric
dependencies {
testCompile 'org.robolectric:robolectric:3.0'
testCompile 'org.mockito:mockito-core:2.+'
}
E o código...
@RunWith(RobolectricGradleTestRunner.class)
@Config(constants = BuildConfig.class)
public class BookBorrowingContentProviderMapperTest {
@Before public void prepareTest() {
this.bookBorrowingContentProviderMapper = new BookBorrowingContentProviderMapper();
}
@Test public void testTransformEmptyCursorToBookBorrowing() throws Exception {
Cursor cursor = mock(Cursor.class);
when(cursor.moveToFirst()).thenReturn(false);
List<BookBorrowing> bookBorrowings =
this.bookBorrowingContentProviderMapper.transform(cursor);
assertThat(bookBorrowings, is(empty()));
}
}
Exemplo em Vilibra on Github: BookBorrowingContentProviderMapperTest.java
Em nosso exemplo...
Isolamos o componente a ser testado de dependências boilerplate
Usamos mocks para criar as situações de teste desejadas
Podemos rodar nossos testes como testes unitários na JVM
Usando mocks lembre-se...
Se você cria mocks para tudo você não testa nada
Mocks nos ajudam a isolar o componente a ser testado e
a ter controle sobre dependências e comportamentos externos
Podem ajudar a reduzir flakiness
UI Tests e as Camadas de Testes
Fonte: Layers of Test Automation by Stuart Ashman
Camadas de Testes
Estratégia para automação
de testes
Onde investir nossos
recursos para automação?
Layers of Test by Mike Cohn
Criação e
Manutenção
Execução e
Criação
Custo Tempo
$
$$
$$$
Testes Unitários
Testes de
Integração
UI
Testes Manuais e
Exploratórios $$$$
Testes de Interface do Usuário (UI)
Devem focar em funcionalidades, user stories, casos de uso etc
Testes do ponto de vista do usuário (UI)
Retira o trabalho repetitivo do time
QA pode se dedicar a testes mais complexos
ATST
Algumas Ferramentas
Android Testing Support Library
Execução relativamente rápida
Bem próximo da plataforma e exige conhecimentos de dev.
Utiliza as mesmas ferramentas que usamos para criar as apps
Difícil para não desenvolvedores
Android Testing Support Library
Ferramentas específicas para testes de componentes Android
AndroidJUnitRunner + JUnit4 Rules + Espresso + UIAutomator
Em geral compatível com Api Level 8 ou superior
UIAutomator é compatível com Api Level 18 ou superior
UI Tests
App
External
Dependency
<<mocks>>
Uma abordagem possível
External
DependencyExternal
Dependency
External
Dependency
Test
<<instruments>>
<<tests>>
Espresso
UIAutomator
<<uses>>
Um exemplo simples...
android {
defaultConfig{
testInstrumentationRunner ”android.support.test.runner.AndroidJUnitRunner”
}
}
dependencies {
androidTestCompile 'com.android.support.test:runner:0.5'
androidTestCompile 'com.android.support.test:rules:0.5'
androidTestCompile 'com.android.support.test.espresso:espresso-core:2.2.2'
androidTestCompile 'com.android.support.test.uiautomator:uiautomator-v18:2.1.2'
}
Exemplo em Vilibra on Github: TestEmptyLoanedBooksList.java
Um exemplo simples...
Exemplo em Vilibra on Github: TestEmptyLoanedBooksList.java
@RunWith(AndroidJUnit4.class)
public class TestEmptyLoanedBooksList {
private Context context;
@Rule public ActivityTestRule<BookListActivity> activityRule =
new ActivityTestRule<>(BookListActivity.class);
@Before public void setUp() {
this.context = getTargetContext();
clearTestData();
}
@Test public void simpleTestEmptyStateOfLoanedBooksList() {
onView(withId(R.id.loaned_book_list_view)).check(matches(not(isDisplayed())));
onView(withId(R.id.empty_book_list_icon)).check(matches(isDisplayed()));
}
}
Testes de Interface do Usuário (UI)
São difíceis de criar e manter
Sua execução geralmente é mais lenta, deixe-os para o final
Selecione a ferramenta adequada aos conhecimentos do time
Quer saber mais? Dê uma olhada em minhas outras apresentações!
Pensamentos Finais
Mantenha o Test Flakiness sob controle
Mocks podem aumentar sua produtividade e controlar o flakiness
Testes de UI são difíceis de criar e manter, use-os sabiamente
Cuide de seus códigos de testes assim como você
cuida de seus códigos de produção!
“… if you are afraid to
change something it is
clearly poorly designed.”
- Martin Fowler
Perguntas?
http://goo.gl/oms4Hs
Material em:
Eduardo Carrara
@DuCarrara
Obrigado!
github.com/ecarrara-araujo
Referências
1. Building Effective Unit Tests by Android Documentation
2. How to Deal With and Eliminate Flaky Tests
3. Flaky Tests at Google and How We Mitigate Them
4. Eradicating Non-Determinism in Tests by Martin Fowler
5. Retrofit 2 – Mocking HTTP Responses
6. The Difference Between Mocks and Stubs by Martin Fowler
7. Mockito
8. Robolectric
9. Layers of Test Automation
10. Android Test Support Library
11. Espresso Test Recorder Docs

Mais conteúdo relacionado

PDF
Testes: Por onde Começar?
PDF
Android DevConference - Refactoring for RxJava
PDF
Android: testes automatizados e TDD
PDF
better faster stronger dagger
PDF
TDC2016POA | Trilha Android - Testes no Android
PDF
Desenvolvimento Dirigido por Testes com Junit
PPTX
Introdução a testes unitários com jUnit
PDF
Testes de Unidade com Junit
Testes: Por onde Começar?
Android DevConference - Refactoring for RxJava
Android: testes automatizados e TDD
better faster stronger dagger
TDC2016POA | Trilha Android - Testes no Android
Desenvolvimento Dirigido por Testes com Junit
Introdução a testes unitários com jUnit
Testes de Unidade com Junit

Mais procurados (18)

PDF
Palestra Testes Unidade Com JUnit
PDF
Testes de Unidade com JUnit
PPT
Junit 4.0
PPT
PDF
Palestra Testes De Unidade Com JUnit
PPTX
Power mock
PDF
Implementando Testes Unitários em Java - Manoel Pimentel
PPTX
Teste unitário
PDF
Mock objects - Teste de código com dependências
PPTX
Testes Unitários/Integrados
PPT
Testes Automatizados de Software
PDF
TDD com Python (Completo)
PPTX
PDF
Testes de Integração
PPTX
Por que automatizar testes de software?
PDF
Teste de Integração - Unidade III
Palestra Testes Unidade Com JUnit
Testes de Unidade com JUnit
Junit 4.0
Palestra Testes De Unidade Com JUnit
Power mock
Implementando Testes Unitários em Java - Manoel Pimentel
Teste unitário
Mock objects - Teste de código com dependências
Testes Unitários/Integrados
Testes Automatizados de Software
TDD com Python (Completo)
Testes de Integração
Por que automatizar testes de software?
Teste de Integração - Unidade III
Anúncio

Semelhante a Indo além com Automação de Testes de Apps Android (20)

PDF
TDC2016POA | Trilha Android - Testes no Android
PDF
Android DevConference - Automatizando testes sem sofrimento
PDF
Mobile Conf 2015 - Testes de Unidade com Robolectric
PDF
Mobile Conf 2015 - Testes de Unidade com Robolectric
PDF
Testes de Unidade - Unidade II
PDF
Melhorando seu App com Kotlin e Testes
PDF
TDC 2015 - Testes de Unidade com Robolectric
PDF
TDC 2015 Floripa - Testes de Unidade com Robolectric
PDF
Automatize seus testes de UI com a Espresso!
PDF
Teste sua app e aumente as chances de mantê-la no mercado
PDF
Automação de Teste com Robotium - Tche Mobile 2014
PDF
Scrum Gathering Rio 2015 - Testando na nuvem a UI e o JavaScript de sua aplic...
PDF
Agile Brazil 2015 - Testando na Nuvem a UI e o JavaScript de sua aplicação
KEY
Android testing PT-BR
PPTX
Android - Frameworks de Testes
PPTX
Xamarin + TDD - Reagindo rápido à mudanças no mundo mobile
PPTX
Automação de Teste para REST, Web e Mobile
PDF
Paralelize seus testes web e mobile para ter feedbacks mais rápidos
PPTX
Curso: Desenvolvimento de aplicativos híbridos (dia 2)
ODP
Palestra Mocks - AgileBrazil 2010
TDC2016POA | Trilha Android - Testes no Android
Android DevConference - Automatizando testes sem sofrimento
Mobile Conf 2015 - Testes de Unidade com Robolectric
Mobile Conf 2015 - Testes de Unidade com Robolectric
Testes de Unidade - Unidade II
Melhorando seu App com Kotlin e Testes
TDC 2015 - Testes de Unidade com Robolectric
TDC 2015 Floripa - Testes de Unidade com Robolectric
Automatize seus testes de UI com a Espresso!
Teste sua app e aumente as chances de mantê-la no mercado
Automação de Teste com Robotium - Tche Mobile 2014
Scrum Gathering Rio 2015 - Testando na nuvem a UI e o JavaScript de sua aplic...
Agile Brazil 2015 - Testando na Nuvem a UI e o JavaScript de sua aplicação
Android testing PT-BR
Android - Frameworks de Testes
Xamarin + TDD - Reagindo rápido à mudanças no mundo mobile
Automação de Teste para REST, Web e Mobile
Paralelize seus testes web e mobile para ter feedbacks mais rápidos
Curso: Desenvolvimento de aplicativos híbridos (dia 2)
Palestra Mocks - AgileBrazil 2010
Anúncio

Mais de Eduardo Carrara de Araujo (18)

PDF
Só um appzinho aê!? - O guia de sobrevivência para o dev da ideia inovadora a...
PDF
Android apps ci
PDF
2016 - Por que mobile?
PDF
Android ndk: Entering the native world
PDF
Android NDK: Entrando no Mundo Nativo
PDF
Implementation of a Participatory Sensing Solution to Collect Data About Pave...
PDF
GDG ABC - Aventura 2015
PDF
Android Test Automation Workshop
PDF
Android M - Getting Started
PDF
Testando Sua App Android na Nuvem
PDF
Utilizando Espresso e UIAutomator no Teste de Apps Android
PDF
Começando com Android (#AndroidOnIntel)
PDF
Android Auto Basics
PDF
Debugging in Android
PDF
Android 101: Do Plano ao Play
PDF
Testing Your App in the Cloud
PPTX
Android 101: Do Plano ao Play em 30 minutos
Só um appzinho aê!? - O guia de sobrevivência para o dev da ideia inovadora a...
Android apps ci
2016 - Por que mobile?
Android ndk: Entering the native world
Android NDK: Entrando no Mundo Nativo
Implementation of a Participatory Sensing Solution to Collect Data About Pave...
GDG ABC - Aventura 2015
Android Test Automation Workshop
Android M - Getting Started
Testando Sua App Android na Nuvem
Utilizando Espresso e UIAutomator no Teste de Apps Android
Começando com Android (#AndroidOnIntel)
Android Auto Basics
Debugging in Android
Android 101: Do Plano ao Play
Testing Your App in the Cloud
Android 101: Do Plano ao Play em 30 minutos

Indo além com Automação de Testes de Apps Android

  • 1. Indo Além com Automação e Testes de Apps Android Eduardo Carrara @DuCarrara
  • 2. Você já se sentiu assim em seus projetos?
  • 3. Infelizmente não há bala de prata...
  • 4. Precisamos de segurança, para Iterações Rápidas.
  • 5. Testes Automatizados podem ajudar Então o que vamos explorar hoje? Recursos Externos & Test Flakiness Simplificação de Dependências com Mocks UI Tests e as Camadas de Testes
  • 6. Recursos Externos e Test Flakiness “Non-deterministic tests have two problems, firstly they are useless, secondly they are a virulent infection that can completely ruin your entire test suite.” - Martin Fowler
  • 7. Test Flakiness Falham e passam aleatoriamente Diminuem a confiança na suite de testes Escondem bugs São difíceis de corrigir
  • 8. Principais Causas Concorrência Dependência de comportamentos indefinidos ou não determinísticos Código de terceiros que falham e funcionam aleatoriamente Problemas de infraestrutura <adicione sua causa aqui>
  • 9. O que acontece no Android? Rede Recursos de Hardware do telefone (ex.: sensores) Interação com outros apps Dependências de serviços, apis e sdks externos
  • 10. Exemplificando Caso comum: chamadas a serviços REST Dependência de recursos de rede, disponibilidade de serviços etc Como resolver o flakiness no teste da chamada de um serviço? Resposta: Desacoplar e Isolar
  • 11. Desacoplando e Isolando Presentation Domain Data Activity Fragment Views Use Cases Repositories Biz Entities Data Sources Data Entities RestAPI Network Latency Service Errs. etc... Origem do flakiness
  • 12. Desacoplando e Isolando Data Layer RestAPI Server RestApi Connection RestApi Interface API Calls Retrofit OkHttp etc <<builds>> <<network calls>> Produção App Config Base URL <<provides>> <<injects>>
  • 13. Desacoplando e Isolando Test Mock Local Server RestApi Connection RestApi Interface Retrofit OkHttp etc <<builds>> <<local calls>> Teste Test URL Test Responses <<provides>> <<injects>> <<creates>>
  • 14. E o código... Com Retrofit + OkHttp + MockWebServer dependencies { testCompile 'com.squareup.okhttp3:mockwebserver:3.3.0' }
  • 15. E o código... @Before public void setUp() throws IOException { prepareExpectedTestData(); server = new MockWebServer(); server.enqueue( new MockResponse() .setBody(TestServiceResponseFor20150907.getJson()) ); server.start(); restApi = new RestApi(server.url("/").toString()); } @After public void tearDown() throws IOException { server.shutdown(); } Exemplo em NEOM on Github: ListNearEarthObjectsTest.java
  • 16. E o código... @Test public void testSuccessfulListNearEarthObjects() throws Exception { List<NearEarthObjectJsonEntity> nearEarthObjectJsonEntityList = restApi.listNearEarthObjectsFor(requestDate); assertThat(nearEarthObjectJsonEntityList, is(expectedNearEarthObjectJsonEntityList)); } Exemplo em NEOM on Github: ListNearEarthObjectsTest.java
  • 17. Em nosso exemplo... Isolamos a chamada à nossos serviços Substituímos o servidor por um componente sob nosso controle Podemos testar esta camada com testes unitários Tomamos controle sobre um recurso que poderia ser Flaky
  • 19. Dependências + Dependências = + Acoplamento Alto acoplamento dificulta a criação de testes e a automação Sistemas 100% desacoplados são raros
  • 20. Mocking Objetos dublês Controle sobre o comportamento e estado Isolamento da unidade a ser testada Deve nos ajudar a lidar com as dependências de maneira simples
  • 21. O que acontece no Android? Dependência de componentes do framework é frequente Ex.: Context, Dados de Sensores, Services, Activities etc Complexidade dos testes aumenta muito
  • 22. Mockito API fácil de entender Configuração de comportamentos e Stubs Permite a verificação de comportamentos
  • 23. Mas como funciona? Exemplo em NEOM on Github: NearEarthObjectListPresenterTest.java @RunWith(MockitoJUnitRunner. class) public class NearEarthObjectListPresenterTest { NearEarthObjectListPresenter presenter; @Mock private ObservableInteractor<List<NearEarthObject>> mockInteractor; @Mock private NearEarthObjectsListView mockView; @Captor private ArgumentCaptor<Observer<List<NearEarthObject>>> observerArgumentCaptor; @Before private void setUp() { presenter = new NearEarthObjectListPresenter( mockInteractor); } }
  • 24. Mas como funciona? Exemplo em NEOM on Github: NearEarthObjectListPresenterTest.java @Test public void testSuccessfulPresenterInitialization() { presenter.attachTo(mockView); verify(mockView).hideRetry(); verify(mockView).showLoading(); // Here we capture the observer registration to mock the update call verify(mockInteractor, times(1)) .addObserver(observerArgumentCaptor.capture()); verify(mockInteractor).execute(); observerArgumentCaptor.getValue() .update(mockInteractor, new ArrayList<NearEarthObject>()); verify(mockView).renderNearEarthObjectsList(any(List.class)); verify(mockView).hideLoading(); }
  • 25. Robolectric Testes que dependem do Android na JVM Provê mocks e stubs ao Android SDK Execução mais rápida de testes Demora para disponibilizar versões mais recentes do SDK
  • 26. Mas como funciona? Exemplo em NEOM on Github: NearEarthObjectsListActivityTest.java @RunWith(RobolectricGradleTestRunner.class) @Config(constants = BuildConfig.class, sdk = Build.VERSION_CODES.LOLLIPOP) public class NearEarthObjectsListActivityTest { private NearEarthObjectsListActivity nearEarthObjectsListActivity; @Before public void setup() { nearEarthObjectsListActivity = Robolectric.setupActivity(NearEarthObjectsListActivity.class); } }
  • 27. Mas como funciona? Exemplo em NEOM on Github: NearEarthObjectsListActivityTest.java @Test public void testErrorMessageDisplay() { final String expectedErrorMessage = "Error message"; nearEarthObjectsListActivity.showError(expectedErrorMessage); View errorView = findById(nearEarthObjectsListActivity, R.id.error_view); TextView errorTextView = findById(nearEarthObjectsListActivity, R.id.error_message_text_view); assertThat(errorView.getVisibility(), is(VISIBLE)); assertThat(errorTextView.getText().toString(), is(expectedErrorMessage)); }
  • 28. + Combinação Poderosa Testes Rodando na JVM Local = Produtividade Ajuda com dependências indiretas do Android Framework Permite o mocking e uso de classes do Android na JVM Local
  • 29. Exemplificando Caso: Uso de dados obtidos de um Content Provider O Content Provider nos provê um Cursor Como testar um componente que dependa de um Cursor? Resposta: Mocks* * A classe MockCursor do Android Framework foi Deprecated na API Level 24. Uma das sugestões é usar o Mockito com ATSL.
  • 33. E o código... Com Mockito + Robolectric dependencies { testCompile 'org.robolectric:robolectric:3.0' testCompile 'org.mockito:mockito-core:2.+' }
  • 34. E o código... @RunWith(RobolectricGradleTestRunner.class) @Config(constants = BuildConfig.class) public class BookBorrowingContentProviderMapperTest { @Before public void prepareTest() { this.bookBorrowingContentProviderMapper = new BookBorrowingContentProviderMapper(); } @Test public void testTransformEmptyCursorToBookBorrowing() throws Exception { Cursor cursor = mock(Cursor.class); when(cursor.moveToFirst()).thenReturn(false); List<BookBorrowing> bookBorrowings = this.bookBorrowingContentProviderMapper.transform(cursor); assertThat(bookBorrowings, is(empty())); } } Exemplo em Vilibra on Github: BookBorrowingContentProviderMapperTest.java
  • 35. Em nosso exemplo... Isolamos o componente a ser testado de dependências boilerplate Usamos mocks para criar as situações de teste desejadas Podemos rodar nossos testes como testes unitários na JVM
  • 36. Usando mocks lembre-se... Se você cria mocks para tudo você não testa nada Mocks nos ajudam a isolar o componente a ser testado e a ter controle sobre dependências e comportamentos externos Podem ajudar a reduzir flakiness
  • 37. UI Tests e as Camadas de Testes
  • 38. Fonte: Layers of Test Automation by Stuart Ashman Camadas de Testes Estratégia para automação de testes Onde investir nossos recursos para automação? Layers of Test by Mike Cohn
  • 39. Criação e Manutenção Execução e Criação Custo Tempo $ $$ $$$ Testes Unitários Testes de Integração UI Testes Manuais e Exploratórios $$$$
  • 40. Testes de Interface do Usuário (UI) Devem focar em funcionalidades, user stories, casos de uso etc Testes do ponto de vista do usuário (UI) Retira o trabalho repetitivo do time QA pode se dedicar a testes mais complexos
  • 42. Android Testing Support Library Execução relativamente rápida Bem próximo da plataforma e exige conhecimentos de dev. Utiliza as mesmas ferramentas que usamos para criar as apps Difícil para não desenvolvedores
  • 43. Android Testing Support Library Ferramentas específicas para testes de componentes Android AndroidJUnitRunner + JUnit4 Rules + Espresso + UIAutomator Em geral compatível com Api Level 8 ou superior UIAutomator é compatível com Api Level 18 ou superior
  • 44. UI Tests App External Dependency <<mocks>> Uma abordagem possível External DependencyExternal Dependency External Dependency Test <<instruments>> <<tests>> Espresso UIAutomator <<uses>>
  • 45. Um exemplo simples... android { defaultConfig{ testInstrumentationRunner ”android.support.test.runner.AndroidJUnitRunner” } } dependencies { androidTestCompile 'com.android.support.test:runner:0.5' androidTestCompile 'com.android.support.test:rules:0.5' androidTestCompile 'com.android.support.test.espresso:espresso-core:2.2.2' androidTestCompile 'com.android.support.test.uiautomator:uiautomator-v18:2.1.2' } Exemplo em Vilibra on Github: TestEmptyLoanedBooksList.java
  • 46. Um exemplo simples... Exemplo em Vilibra on Github: TestEmptyLoanedBooksList.java @RunWith(AndroidJUnit4.class) public class TestEmptyLoanedBooksList { private Context context; @Rule public ActivityTestRule<BookListActivity> activityRule = new ActivityTestRule<>(BookListActivity.class); @Before public void setUp() { this.context = getTargetContext(); clearTestData(); } @Test public void simpleTestEmptyStateOfLoanedBooksList() { onView(withId(R.id.loaned_book_list_view)).check(matches(not(isDisplayed()))); onView(withId(R.id.empty_book_list_icon)).check(matches(isDisplayed())); } }
  • 47. Testes de Interface do Usuário (UI) São difíceis de criar e manter Sua execução geralmente é mais lenta, deixe-os para o final Selecione a ferramenta adequada aos conhecimentos do time Quer saber mais? Dê uma olhada em minhas outras apresentações!
  • 48. Pensamentos Finais Mantenha o Test Flakiness sob controle Mocks podem aumentar sua produtividade e controlar o flakiness Testes de UI são difíceis de criar e manter, use-os sabiamente Cuide de seus códigos de testes assim como você cuida de seus códigos de produção!
  • 49. “… if you are afraid to change something it is clearly poorly designed.” - Martin Fowler Perguntas? http://goo.gl/oms4Hs Material em:
  • 51. Referências 1. Building Effective Unit Tests by Android Documentation 2. How to Deal With and Eliminate Flaky Tests 3. Flaky Tests at Google and How We Mitigate Them 4. Eradicating Non-Determinism in Tests by Martin Fowler 5. Retrofit 2 – Mocking HTTP Responses 6. The Difference Between Mocks and Stubs by Martin Fowler 7. Mockito 8. Robolectric 9. Layers of Test Automation 10. Android Test Support Library 11. Espresso Test Recorder Docs