- 100% Kotlin based + Coroutines + Flow for asynchronous.
- This app uses A single-activity pattern, using the Navigation component to manage fragment operations.
- Reactive UIs using LiveData observables and Data Binding.
- It consists of one fragment which is fully tested by Espresso.
Sample | Description |
---|---|
master | The base for the rest of the other branches. Uses Kotlin, Architecture Components, Coroutines + Flow, Dagger, Retrofit Data Binding, etc. |
room-cache | Same like master branch but it uses Room db for caching data implementing single source of truth |
paging3-network-db-livedata | Added Paging3 library, It uses RemoteMediator with Room DAO + PagingSource as single source of truth, exposes data as LiveData |
paging3-network-db-flow | Added Paging3 library, It uses RemoteMediator with Room DAO + PagingSource as single source of truth, exposes data as Flow |
- The interesting thing about this repository is that it implements the Paging 3 Library that has so many features that simplified complicated process creating RecyclerView with paging.
- Loading small chunk of data that reduces usage of network bandwidth and system resources.
- Built-in support for error handling, including refresh and retry capabilities.
- Built-in separator, header, and footer support.
- Automatically requests the correct page when the user has scrolled to the end of the list.
- Ensures that multiple requests are not triggered at the same time.
- 100% Kotlin
- MVVM Architecture
- Architecture Components (Lifecycle, LiveData, ViewModel, Paging, Navigation Component, DataBinding, Room)
- Coroutines For handling concurrent tasks
- Dagger2 for dependency injection
- Retrofit2 for REST API
- Glide for loading images
- LeakCanary for detecting memory leak
- Mockito-kotlin for Junit mock test
- Espresso for UI testing
is an architectural pattern in programming. The main task of the pattern is to separate presentation logic from business logic. The most important component in MVVM is the ViewModel, which behaves more like a model and less like a view. It is responsible for converting and delegating data objects to presentation logic that displays those objects to the user on the device screen.
-
Avoid references to Views in ViewModels. The ViewModel should never have any view or UI controllers (activities and fragments) references because ViewModels have different scopes than activities/fragments. While a ViewModel is alive and running, an activity can be in any of its lifecycle states. Activities/fragments can be destroyed and created again while the ViewModel is unaware
-
Instead of pushing data to the UI, let the UI observe changes (For example using LiveData) Fragments/Activities are intended to display UI data, react to user actions, or handle operating system communication, they shouldn't know how the data is coming they only observe it, ViewModel prepares data for the UI. ViewModel is retained during configuration changes, like rotating the phone.
-
Add a data repository as the single-point entry to your data. Each component in MVVM depends only on the component one level below it. For example, activities and fragments depend only on a view model. The repository is the only class that depends on multiple other classes; For example, the repository depends on a persistent data model and a remote backend data source. In my case I am using only remote backend data source.
-
Expose information about the state of your data using a wrapper or another LiveData. In my app I have used Wrapper named Resource that can expose three States: LOADING, SUCCESS, ERROR.
I have used Retrofit which is a REST Client for Java and Android that makes it easy to retrieve and upload JSON (or other structured data) via a REST based webservice.
- As it is known we can configure which converter is used for the data serialization by specifying the factory. In my case I have used: GsonConverterFactory. (MoshiConverterFactory can also be used) We can also create our own CallAdapter that converts the Call into a ApiResponse it is Common class used by API responses representing three types of:
- ApiSuccessResponse represents the Success response.
- ApiFailureResponse represent tow type of errors: Error response (e.g. server error), Exception Response (e.g. network connection error) Then I have created a factory of that CallAdapter to be used when creating the Retrofit service.
@GET("/api/community_{page}.json")
suspend fun getCommunityMembers(@Path("page") page: Int): ApiResponse<CommunityResponse>
There is a good article about CallAdapter in here
Dependency injection is a technique in which an object receives other objects that it depends on. These other objects are called dependencies. In the typical "using" relationship the receiving object is called a client and the passed (that is, "injected") object is called a service.
-
Dependency injection provides the flexibility of configurable.
-
Dependency injection makes it much easier to unit test in isolation using stubs or mock objects that simulate other objects not under test.
-
Dependency injection allows a client to remove all knowledge of a concrete implementation that it needs to use. This helps isolate the client from the impact of design changes and defects. It promotes reusability, testability and maintainability
-
Dependency Injection decreases coupling between a class and its dependency
-
In this app I have used Dagger2 for dependency injection Dagger is a fully static, compile-time dependency injection framework for Java, Kotlin, and Android. It is an adaptation of an earlier version created by Square and now maintained by Google. I have created a AppInjector that has a method
fun init(tandemApp: TandemApp)
where tandemApp is an instance of the Application passed during runtime. This Method is called in onCreate of TandemApp as follows:
class TandemApp : Application(), HasAndroidInjector {
@Inject
lateinit var androidInjector: DispatchingAndroidInjector<Any>
override fun onCreate() {
super.onCreate()
if (BuildConfig.DEBUG) {
Timber.plant(Timber.DebugTree())
}
AppInjector.init(this)
}
override fun androidInjector(): AndroidInjector<Any> = androidInjector
}
- Note: I could have used Dagger Hilt which is much easier than Dagger2 and it basically gets rid of a lot of boilerplate code. But the only problem with it, as u can see from Hitl Doc is that Hilt does not currently support FragmentScenario for testing fragments, however there is still a workaround.
They're different tools with different strengths. Like a tank and a cannon, they have a lot of overlap but are more or less desirable under different circumstances.
- Coroutines Are light weight threads for asynchronous programming.
- RX-Kotlin/RX-Java is functional reactive programming, its core pattern relay on observer design pattern, so you can use it to handle user interaction with UI while you still using coroutines as the main core for background work.
- Creating a coroutine is a lot cheaper than creating a thread.
- Coroutines: When we have concurrent tasks like you would fetch data from Remote service, database, any background processes, sure you can use RX in such cases too, but it looks like you use a tank to kill an ant.
- RX-Kotlin: When you would to handle a stream of UI actions like user scrolling, clicks, update UI upon some events ... etc.
- Writing asynchronous code is a sequential manner.
- Costing of creating coroutines is much cheaper to create threads.
- Don't be over-engineered to use observable pattern, when no need to use it.
- parent coroutine can automatically manage the life cycle of its child coroutines for you.
Repository modules are responsible for handle data operations. By ensuring this, Repository modules can provide a clean API to the rest of the app and simplify the job of the consumer ViewModel. Repository modules should know where to get the data from and what API calls to make when data is updated if necessary. They can be considered as mediators between different data sources (REST services, Databases ..)
- I have used Flow in my app, A flow is an asynchronous version of a Sequence, a flow produces each value on-demand whenever the value is needed.
-
Flow includes full support for coroutines. That means you can build, transform, and consume a Flow using coroutines.
-
You can also control concurrency, which means coordinating the execution of several coroutines declaratively with Flow.
-
Flow can be used in a fully-reactive programming style. like RxJava.
-
Flow also supports suspending functions on most operators. By using suspending operations inside of a flow, it often results in shorter and easier to read code than the equivalent code in a fully-reactive style.
-
My repository looks like:
@OpenForTesting
class CommunityRepository @Inject constructor(
private val service: TandemService,
private val dispatcherIO: CoroutineDispatcher
) {
@WorkerThread
suspend fun getCommunityMembers(
page: Int
): Flow<Resource<List<Member>>> {
return flow {
emit(Resource.loading(null))
service.getCommunityMembers(page).apply {
this.onSuccessSuspend {
data?.let {
emit(Resource.success(it.members, it.members.size != 20))
}
}
// handle the case when the API request gets an error response.
// e.g. internal server error.
}.onErrorSuspend {
emit(Resource.error(message(), null))
// handle the case when the API request gets an exception response.
// e.g. network connection error.
}.onExceptionSuspend {
emit(Resource.error(message(), null))
}
}.flowOn(dispatcherIO)
}
}
- onSuccessSuspend, onErrorSuspend, onExceptionSuspend are nothing but suspend extension functions for ApiResponse. something like:
@SuspensionFunction
suspend fun <T> ApiResponse<T>.onSuccessSuspend(
onResult: suspend ApiResponse.ApiSuccessResponse<T>.() -> Unit
): ApiResponse<T> {
if (this is ApiResponse.ApiSuccessResponse) {
onResult(this)
}
return this
}
-
As we can notice I am specifying the Dispatcher by
flowOn(dispatcherIO)
which is the Dispatcher that this Flow is going to work on and it is basically Dispatchers.IO but it is injected (Provided by Dagger) for better testability (I will talk about it later). -
Also
service: TandemService
is being injected for better testability, so that when we testCommunityRepository
we just need mockservice
and expect values to ensure we are testing in separation. -
For success responses we emit
Resource.success(it.members, it.members.size != 20))
-
For error responses we emit
(Resource.error(message(), null))
-
As you can notice I have handled the last page condition in the resource by adding the statement
it.members.size != 20
So for each request we have if it is a success we check if we are on the last page and in the UI when observing the data, we can react accordingly. (That could also have been done in the ApiResponse)
The ViewModel class is designed to store and manage UI-related data in a lifecycle conscious way. The ViewModel class allows data to survive configuration changes such as screen rotations.
- My viewModel looks like:
@OpenForTesting
class CommunityViewModel @Inject constructor(
private val repository: CommunityRepository,
private val dispatcherIO: CoroutineDispatcher
) :
ViewModel() {
private val pageLiveData: MutableLiveData<Int> = MutableLiveData()
private var pageNumber = 1
init {
pageLiveData.postValue(1)
}
val membersListLiveData = pageLiveData.switchMap { pageNumber ->
liveData(viewModelScope.coroutineContext + dispatcherIO) {
val members = repository.getCommunityMembers(pageNumber).asLiveData()
emitSource(members)
}
}
fun loadMore() {
pageNumber++
pageLiveData.postValue(pageNumber)
}
fun refresh() {
pageLiveData.value?.let {
pageLiveData.value = it
}
}
}
-
repository: CommunityRepository
anddispatcherIO: CoroutineDispatcher
are injected and provided by Dagger for better testability. -
The ViewModel exposes
membersListLiveData
that is going to be observed by the UI My fragment and changing the UI accordingly with the Livedata changes (Recommended to exposes only LiveData not MutableLiveData (it should be used internally)) -
The way it works is when changing the value of
pageLiveData
,repository.getCommunityMembers(pageNumber)
will be called returning Flow that is going to be converted to LiveData. -
The Most important thing here is that I am using
viewModelScope
which is using to launch the coroutines, and it will destroy it when the ViewModel is destroyed (onCleared() is called) with having to destroy it ourselves (Not like RX-Java and disposables)
- In the fragment I am using Databinding to bind LOADING Resource to the progressbar visibility, ERROR Resource to a snackbar showing the error message and the data of each member in the community using BindingAdapter. The
onViewCreated
in my fragment looks like:
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
with(binding) {
lifecycleOwner = this@CommunityFragment
vm = viewModel
callback = object : RetryCallback {
override fun retry() {
viewModel.refresh()
}
}
}
initRecyclerView()
subscribeUi()
}
- I am observing the liveData as follows:
private fun subscribeUi() {
viewModel.membersListLiveData.observe(viewLifecycleOwner, { result ->
if (result.status == Status.SUCCESS && !result.data.isNullOrEmpty()) {
adapter.submitList(result.data)
}
})
}
-
It is very important to pass
viewLifecycleOwner
as the LifeCycleOwner because when fragment gets detached it won't be destroyed but theonDestroyView
gets called so here the Fragment survives and in case passing the fragment is LifeCycleOwner the observer will not be destroyed and we would haveonChanged
called twice.
- As it is know all Kotlin classes are final that means we cannot mock them or stub them the way I’ve handled this is by using the same mechanism that was used in GithubBrowserSample The idea is that making kotlin classes open in our test build but also keep them final in our releases builds, then we can mock in our tests without having to make classes always open, that can be done with the Kotlin compiler plugin kotlin-allopen. What we need to do is creating two annotations one of them is put in src/debug which has OpenClass and the other one is put in src/release but without OpenClass:
/**
* This annotation allows us to open some classes for mocking purposes while they are final in
* release builds.
*/
@Target(AnnotationTarget.ANNOTATION_CLASS)
annotation class OpenClass
/**
* Annotate a class with [OpenForTesting] if you want it to be extendable in debug builds.
*/
@OpenClass
@Target(AnnotationTarget.CLASS)
annotation class OpenForTesting
@Target(AnnotationTarget.CLASS)
annotation class OpenForTesting
Then Add the classpath to the classpath "org.jetbrains.kotlin:kotlin-allopen:$kotlin_version"
root build.gradle file.
Then in the app build.gradle apply the plugin and specify the annotation class.
apply plugin: 'kotlin-allopen'
allOpen {
// allows mocking for classes w/o directly opening them for release builds
annotation 'com.mustafa.tandem.testing.OpenClass'
}
Finally, I just need to add @OpenForTesting for the classes I need to be open in Debug build.
- When testing with Coroutines the Dispatchers.Main requires main Looper which is not available in the unit tests. So the solution is to override Dispatchers.Main with TestCoroutineDispatcher, That is why we always need to write a general rule to use TestCoroutineDispatcher, We simply need to set the main dispatcher to TestCoroutineDispatcher whenever a test is started.
- TestCoroutineDispatcher has runBlockingTest which will immediately progress delays with a virtual clock. It can be used for faster tests instead of runBlocking.
- when you start a new coroutine with other dispatchers, that coroutine will be executed in a different thread This is why we need to inject the dispatchers.
- Here is an example of a Junit test for
CommunityRepository
:
@Test
fun getCommunityMembers_onlyOneMember() = coroutinesRule.testDispatcher.runBlockingTest {
val mockMember = Member("Mustafa", listOf("SP", "HU"), listOf("EN", "DE"), "URL", 1, "TOPIC")
val members = listOf(mockMember)
val mockResponse = CommunityResponse(null, members, "success")
val mockData = mockResponse.members
val call = successCall(mockResponse)
whenever(service.getCommunityMembers(1)).thenReturn(call)
repository.getCommunityMembers(1).collectIndexed { index, resource ->
if (index == 0) assertThat(resource.status, `is`(Status.LOADING))
if (index == 1) {
assertThat(resource.status, `is`(Status.SUCCESS))
resource.data?.let { members ->
assertThat(members[0].firstName, `is`("Mustafa"))
assertThat(members[0].topic, `is`("TOPIC"))
assertThat(members[0].referenceCnt, `is`(1))
assertThat(members[0].learns[0], `is`("SP"))
assertThat(members[0].natives[0], `is`("EN"))
assertThat(members[0].pictureUrl, `is`("URL"))
assertThat(members, `is`(mockData))
}
assertThat(resource.message, `is`(nullValue()))
// Since we have only one members, when the count of the response members is less than 20, that means we are in the last page
assertThat(resource.isLastPage, `is`(true))
}
}
verify(service, times(1)).getCommunityMembers(1)
verifyNoMoreInteractions(service)
}
-
After mocking the service and expecting it to return a fake response we test our
getCommunityMembers
with page and that can be done in different ways as Flow as really cool operators, one of them iscollectIndexed
which provides the index and the data not having to use delay function. So in the above method, I am asserting that the first resource is Loading and then the second resource is the data that was fetched successfully. -
For testing the viewModel we should mock the repository, and then expecting the repository to return fake Flows, then observing the Livedata on a mock observer, changing the liveData value, and finally asserting for example:
@Test
fun getCommunityMembersSuccessTest() = coroutinesRule.testDispatcher.runBlockingTest {
val observer = mock<Observer<Resource<List<Member>>>>()
val members = MockTestUtil.createMembers(20)
val resourceSuccess = Resource.success(members, false)
val flow = flow {
emit(resourceSuccess)
}
whenever(repository.getCommunityMembers(anyInt())).thenReturn(flow)
viewModel.membersListLiveData.observeForever(observer)
// I am changing pageLiveData with pageLiveData.postValue(1) in init block
verify(repository).getCommunityMembers(1)
verify(observer).onChanged(resourceSuccess)
assertThat(viewModel.membersListLiveData.value?.data?.size, `is`(members.size))
assertThat(
viewModel.membersListLiveData.value?.data?.get(0)?.firstName, `is`(members[0].firstName)
)
assertThat(viewModel.membersListLiveData.value?.data, `is`(members))
// Or resourceSuccess.status
assertThat(viewModel.membersListLiveData.value?.status, `is`(Status.SUCCESS))
assertThat(viewModel.membersListLiveData.value, `is`(resourceSuccess))
viewModel.membersListLiveData.removeObserver(observer)
}
- One of the main reasons why I have chosen to use Navigation component is the ease of testing as we don't have to create TestActivity and then attach the fragment to it.
- We only need to use FragmentScenario so the init method in CommunityFragmentTest looks like:
@Before
fun init() {
viewModel = mock()
whenever(viewModel.membersListLiveData).thenReturn(results)
mockBindingAdapter = Mockito.mock(FragmentBindingAdapters::class.java)
val scenario = launchFragmentInContainer(themeResId = R.style.AppTheme) {
CommunityFragment().apply {
viewModelFactory = ViewModelUtil.createFor(viewModel)
dataBindingComponent = object : DataBindingComponent {
override fun getFragmentBindingAdapters(): FragmentBindingAdapters {
return mockBindingAdapter
}
}
}
}
dataBindingIdlingResourceRule.monitorFragment(scenario)
// Set the navigation graph to the NavHostController
navController.setGraph(R.navigation.main)
scenario.onFragment { fragment ->
Navigation.setViewNavController(fragment.requireView(), navController)
fragment.disableProgressBarAnimations()
}
}
- Then for testing the UI we just need to post valus to he liveData and assert the views for example:
@Test
fun testErrorResult_RetryButton_SnackBar() {
onView(withId(R.id.progress_bar))
.check(matches(not(isDisplayed())))
onView(withId(R.id.button_retry))
.check(matches(not(isDisplayed())))
results.postValue(Resource.error("ERROR"))
onView(withId(R.id.button_retry))
.check(matches(isDisplayed()))
onView(withId(com.google.android.material.R.id.snackbar_text))
.check(matches(withText("ERROR")))
}
@Test
fun loadCommunityMembers() {
//Given
val oldMemberList = MockTestUtil.createMembers(3)
// Action
results.postValue(Resource.success(oldMemberList, true))
// Assert
onView(withId(R.id.members_recycler_view)).check(matches(isDisplayed()))
onView(listMatcher().atPosition(0)).check(matches(isDisplayed()))
onView(listMatcher().atPosition(1)).check(matches(isDisplayed()))
onView(listMatcher().atPosition(2)).check(matches(isDisplayed()))
//if your ViewHolder uses ViewGroup, wrap withText() with a hasDescendant() like:
onView(listMatcher().atPosition(0)).check(matches(hasDescendant(withText("Mustafa0"))))
onView(listMatcher().atPosition(0)).check(matches(hasDescendant(withText("DE0"))))
onView(listMatcher().atPosition(0)).check(matches(hasDescendant(withText("EN0"))))
onView(listMatcher().atPosition(0)).check(matches(hasDescendant(withText("TOPIC0"))))
onView(listMatcher().atPosition(1)).check(matches(hasDescendant(withText("Mustafa1"))))
onView(listMatcher().atPosition(1)).check(matches(hasDescendant(withText("DE1"))))
onView(listMatcher().atPosition(1)).check(matches(hasDescendant(withText("EN1"))))
onView(listMatcher().atPosition(1)).check(matches(hasDescendant(withText("TOPIC1"))))
onView(listMatcher().atPosition(2)).check(matches(hasDescendant(withText("Mustafa2"))))
onView(listMatcher().atPosition(2)).check(matches(hasDescendant(withText("DE2"))))
onView(listMatcher().atPosition(2)).check(matches(hasDescendant(withText("EN2"))))
onView(listMatcher().atPosition(2)).check(matches(hasDescendant(withText("TOPIC2"))))
}
-
For UI testing when using DataBinding is important to use an espresso idling resource implementation that reports idle status for all data binding layouts since Data Binding uses a mechanism to post messages which Espresso doesn't track yet.
-
This App could also be done by using RX-Java by using Observable as the return type in the API service and then using RxJava2CallAdapter
.addCallAdapterFactory(RxJava2CallAdapterFactory.create())
when creating Retrofit instance. the method in the ViewModel would look similar to the following:
fun getCommunityMembers(page: Int) {
subscription = repository.getCommunityMembers(page)
.subscribeOn(schedulerProvider.io())
.observeOn(schedulerProvider.ui())
.doOnSubscribe {
membersLiveDate.value = Resource.loading(null)
}
.subscribe(
{ result ->
membersLiveDate.value = Resource.success(result.members, result.members.size != 20))
},
{
membersLiveDate.value = Resource.error(it.localizedMessage, null)
}
)
}
Then we will have to clear the subscription as follows:
override fun onCleared() {
super.onCleared()
if (!subscription.isDisposed) {
subscription.dispose()
}
}
- It can also be done by Using coroutines and liveData in the Repository without using Flow something similar to:
suspend fun getCommunityMembers(page: Int): LiveData<List<Member>> {
val membersLiveDate = MutableLiveData<Resource<List<Member>>>()
val members: LiveData<Resource<List<Member>>> = membersLiveDate
withContext(Dispatchers.IO) {
val response = tandemService.getCommunityMembers(page).apply { response ->
response.onSuccess {
data?.let { result ->
mutableLiveDate.postValue(Resource.success(result.members, result.members.size != 20))
}
}.onError {
mutableLiveDate.postValue(Resource.error(message(), null))
}.onException {
mutableLiveDate.postValue(Resource.error(message(), null))
}
}
}
return members
}
Then in the ViewModel:
val membersLiveData = pageLiveData.switchMap { pageNumber ->
liveData(context = viewModelScope.coroutineContext + Dispatchers.IO) {
val members = repository.getCommunityMembers(pageNumber)
emitSource(members)
}
}
- If I had to create a real production App, I would have cashed the data using Room following "single source of truth pattern".
- Using the Paging Library, The idea is when building production apps we would want to query the whole data (old data and new queried data) each time
- For that, we should use ListAdapter with diffUtil (better performance), but the user usually interacts with only a small chunk of data at a time.
- The Paging Library helps you load and display small chunks of data at a time. Loading partial data on-demand reduces the usage of network bandwidth and system resources.