Skip to content

Project Architecture (EN)

Thanh-Son-Philippe Lam edited this page Feb 17, 2019 · 44 revisions

The project has three main modules:

Module Description
android The android module contains code used for the Android app.
ios The ios module contains code used for the iOS app.
shared The shared module contains common code used by the android module and the ios module. To access platform APIs, the code in this module make use of the expect/actual mechanism provided by Kotlin. The code in this module rely on multiplatform libraries such as SQLDelight and Kotlin Coroutines.

Architecture Diagram

ViewModel

ViewModels provide the data for the UI.

Repository

The repository module serves as a clean API so that that the rest of the app can retrieve data easily. It acts as a mediator for between the DB and the API modules. The module handle data operations by using Coroutines Suspend Functions.

The repository creates and returns a Coroutine Channel that the app "subscribes" to in order to be notified of further updates. The Channel will be updated with relevant data and a status (loading, success or failure). The status allows the UI to update its state accordingly. For example, for the loading status, the UI can display a loading bar.

Here are the operations that are generally done by the repository module:

  1. The process starts when the rest of the app calls-in the repository.
  2. The repository starts by calling the DB module which will fetch the data stored in the DB.
  3. The DB module returns a Query which is a a listenable, typed query generated by SQLDelight. Notice that, if needed, this object can be easily converted to a Coroutine Channel by using the asChannel extension function.

Here is an example of usage with the map operator:

        val query = queries.selectAll()

        query.asChannel().map(context) { it.executeAsList() }
  1. The Query finishes and the channel is updated with cached data and a "loading" status. The app should have displayed a loading state by now.
  2. The repository calls the API module to start fetching data from the webservice.
  3. The webservice returns a response. If the call succeeded, the repository updates the DB with the new data. The Channel is also updated with a "success status" and new data coming from the DB. If the call failed, the Channel is updated with a "failure" status.

How to deliver the data and its status?

To do so, use the Resource class. It encapsulates both the data and its state.

class Resource<T> private constructor(val status: Status, val data: T?, val message: String?) {
    enum class Status {
        SUCCESS,
        ERROR,
        LOADING
    }

    companion object {
        fun <T> success(data: T): Resource<T> = Resource(Status.SUCCESS, data, null)
        fun <T> error(msg: String, data: T?): Resource<T> = Resource(Status.ERROR, data, msg)
        fun <T> loading(data: T?): Resource<T> = Resource(Status.LOADING, data, null)
    }

    ...
}

Use one of the static methods to create a Resource instance. For example, to encapsulate a data called "myData" and a loading status, you can do like so:

Resource.loading(myData)

DB

The database module uses SQLDelight to persist the app's data. The library let you write write SQL statements from which their Gradle plugin will generate APIs to run your queries. Please refer to their documentation for more information.

API

Platform-Specific Modules

Android

A viewModelScope extension was added for the ViewModel class. We can use it to launch coroutines inside our ViewModels.

iOS