Skip to content

Commit

Permalink
Merge remote-tracking branch 'origin/master'
Browse files Browse the repository at this point in the history
  • Loading branch information
jairrab committed Oct 7, 2019
2 parents 4ce7760 + b183983 commit bac81ce
Show file tree
Hide file tree
Showing 2 changed files with 93 additions and 29 deletions.
122 changes: 93 additions & 29 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,48 +1,60 @@
# Mapbox App
A sample mobile app that implements a MapBox map
* App will download an array of Pin data from a given *http* address that will be saved on a persistent storage.
* Display the user’s location on the map.
* Display the Pin data on the map.
* Display the Pin data in a list.
A sample Android app that implements a MapBox map using *Clean Archecture* principles with *MVVM* design pattern.
# Technical Objectives
The app requirements is very basic, but my objective is not to come up with the simplest/easiest implementation. Although seemingly over-engineered (and it is for this case), my objective was to design the app with a modern architecture that is scalable and flexible (future growth and requirements) and compliant with a team development setting. Thus, I have set the following goals for the app
* Targeted towards the latest Android API (29) and Kotlin language version
* Architecture Goals
* Adheres to *Clean Architecture* design principles
* Modular, scalable, flexible
* Multi-module design composed of independent modules that can be replaced with changing the entire system.
* For instance, the UI module can be changed without affacting the business and data layers of the application
* Modules and classes with well defined responsibility and separation of concerns
* Highly testable
* Utlizies the latest technology stack
The app requirements are very basic, but my objective is not to come up with the simplest/easiest implementation. Although seemingly over-engineered (and it is for this case), my objective was to design it with a modern architecture that is very compliant with a team development setting. Thus, I have set the following goals for the app
* Adheres to *Clean Architecture* design principles
* Modular, scalable, flexible (adaptable to future growth and requirements)
* Multi-module design composed of independent modules that can be replaced without changing the entire system.
* For instance, the UI module can be changed without affacting the business and data layers of the application
* Modules and classes with well defined responsibility and separation of concerns
* Highly testable
* Uses the latest technology stack where applicable

# App Overview/Features

![alt text](resources/images/app_screenshots.png)

* When the app is opened for the first time, it will ask the user to provide access to *location permission request*. This is required to detect and display the user's current location.
* If a user grants permission, the app navigates to the user's current location. It will query the Mapbox API for the closest name of the current location (*screenshot A*).
* The app downloads map geo information in JSON format from the internet.
* The app stores this information on the local phone database for cache purposes.
* The app also stores a timestamp of when this information was saved.
* The geo information is displayed as a list inside a `bottomsheet` slider panel, shown with its distance/bearing to your current location. It is collapsed by default but can be expanded by sliding up (*screenshot B*)
* The geo list is also displayed in the Map using *location pin* icons with a text underneath that shows the item name.
* Clicking on an item list causes the map to navigate to the item location.
* Clicking on a pin icon shows the pin's description, GPS coordinates, and distance/bearing from your current location (*screenshot D*).
* Clicking on the GPS icon navigates the user back to his current location on the map. The user's coordinate is also presented.
* If the user opened the app and it failed to download geo information, the app will check it's local database for the most recent information and use it instead to display the table and map information described previously.
* It will also display how much time has elapsed since the cached information was saved (*screenshot C*).
* The app handles orientation changes gracefully, using `ViewModel` to retain states.

# App Architecture
The app architecture is organized into several layers, namely the *presentation*, *domain*, and the *data* layers.

![alt text](resources/images/app_layer_architecture.png)

The app architecture is organized into several layers, namely the *presentation*, *domain*, and the *data* layers.

* **Presentation**- contains the View and Presentation modules of the app. The view module contains the activities and fragments that are coordinated through the Presenter/ViewModel. I am using the Android ViewModel class (Android Jetpack) which is designed to store and manage UI-related data in a lifecycle conscious way. This ViewModel class allows data to survive configuration changes such as screen rotations. This layer depends on the Domain layer.
* **Domain**- is the core of our application, that contains the business logic. It should not depend on how the data will be presented or where the data is coming from, as such, this is the most inner part that has no dependencies to the outer layers. It defines Use Cases which define operations that can be performed. It is purely a Kotlin library with no Android dependencies. The domain calls for subscribing it's observers into the main UI thread, for this, I created an abstraction for the main thread that will be implemented in the UI module (via *AndroidSchedulers.mainThread*) so that the domain layer can be free of the Android framework
* **Data**- contains the repository implementations, and is also responsible for coordinating data from the two data sources being used by the app- namely the network data source (i.e. SWAPI.co api) and the local storage data source. The network data will be setup and accessed using the Retrofit library. The local data source will be setup and accessed using the Room Library (Android JetPack).
* **Data**- contains the repository implementations, and is also responsible for coordinating data from the two data sources being used by the app- namely the network data source and the local storage data source. The network data will be setup and accessed using the Retrofit library. The local data source will be setup and accessed using the Room Library (JetPack).

In general, this architecture is probably best described as an **MVVM pattern with Clean Architecture**. Key advantages of this are:
In general, this architecture is probably best described as an **MVVM pattern with Clean Architecture**. Key advantages of these are:
* Enables future stability as more features or functions are added later on. For example, when adding a new query required on a ViewModel controller, a new user case can be added on the domain layer, which will then be implemented by someone (or the same person) working on the data layer.
* Helps eliminate frictions between teams working on a project, promotes accountability in a collaborative way.
* Allows each modular layers to be separately tested leading to a more robust and high quality product.

## Use of reactive programming
The app adopts principles of reactive programming through use of *observables* and *observers* and *schedulers*. For example, the presentation layer waits to be notified by the domain layer when a network call is made. The UI layer waits to be notified by `livedata` objects to display view changes. Many of this calls has to be performed asynchronously and schedulers helps with thread management, such as using RxJava's `observeOn` to tell observers which thread they should run on. My goal was to end up with a more cleaner, readable, and structured code base and this helped a lot.

# App Operation
* When the app is opened for the first time, it will ask the user to provide access to *location permission request*. This is required to detect and display the user's current location.
* If a user grants permission, the app navigates to the user's current location.
* If the user does not grant permission, it proceeds to the next step.
* The app downloads map pins information from the internet.
* if...

# Libraries Used

## Mapbox SDK
Used to download street maps, feature information for the user's current location and plotting a list of geolocation information inside the map.

## Dagger 2
The app use Dagger with *AndroidInjector* module for dependency injection. The app module is responsible for the concrete creation of the Dagger objects outside of their own modules. The modules are organized into the following:
The app use Dagger with *AndroidInjector* module for dependency injection. The app module is responsible for the concrete creation of the Dagger objects outside of their own modules. As a general practice, all dependencies are injected into classes, and object creation inside classes are avoided. This loose coupling approach also helps improve the testability of the app.

The modules are organized into the following:
* *AppModule* - Responsible for creating application context dependent objects, such as SharedPreferences
* *CacheModule* - Responsible for creating the Room Database object, as well as the local database access object
* *DataModule* - Responsible for creating the concrete implementaion of the data repository object defined on the domain layer
Expand All @@ -52,21 +64,70 @@ The app use Dagger with *AndroidInjector* module for dependency injection. The a
* *ViewModelModule* - Responsible for concrete instantiation of ViewModel classes
* *ViewModelFactoryModule* - responsible for creating the ViewModelFactory required to instantiate and inject dependencies into Android ViewModels
## Retrofit
I am using Retrofit as client service to accessing server data. Retrofit makes it relatively easy to retrieve JSON via a REST based webservice. Although the actual Json data required for this app is very simple, using Retrofit allows the app to scale more convenienty as more data structures or server requests are required.
I am using Retrofit as client service to accessing server data. Retrofit makes it relatively easy to retrieve JSON via a REST based webservice. Although the actual Json data required for this app is very simple, using Retrofit allows the app to scale more convenienty as more data structures or server requests are required. There are 2 Retrofit clients on this app:
* Test Server client - for obtaining the geo location information that is displayed on the map and the slider panel
* Mapbox client - for obtaining Mapbox feature information for the detected user GPS coordinates.
## RxJava
RxJava is used extensively for composing asynchronous calls to the remote/local data sources by using observable chains and sequences. In this app, using RxJava, a call is chained to:
RxJava is used extensively for composing asynchronous calls to the remote/local data sources by using observable chains and sequences. For instance, during the process of displaying a pre-defined list of geo-information to the user, the app implements the following flow:
* get data from the remote server
* store the data into the local database
* store the timestamp for the data into the local atabase
* check the local database for the most recent location information during connectivity issues
* if cached information is being used, also display how much time has elapsed since the test server was succesully queried.
* notify the UI observers when the information is readily available, or an error has been detected

The code below illustrates how RxJava observables are chained to implement the flow described above.
```kotlin
override fun getMapPoints(): Observable<MapInformation> {
return dataStore.getRemoteData(true).getMapPoints()
.map { list ->
MapInformation(
mapPoints = list.map { mapper.mapToDomain(it) },
source = Source.REMOTE,
timeStamp = timeUtils.currentTime
)
}
.map { information ->
val list = information.mapPoints.map { mapper.mapToData(it) }

val updateLastLocationTimeStamp = dataStore.getRemoteData(false)
.updateLastLocationTimeStamp(information.timeStamp)

dataStore.getRemoteData(false)
.saveMapPoints(list)
.andThen(updateLastLocationTimeStamp)
.andThen(Observable.just(information))
}
.onErrorResumeNext { t: Throwable ->
Observable.zip(
dataStore.getRemoteData(false).getLastLocationTimeStamp(),
dataStore.getRemoteData(false).getMapPoints(),
BiFunction<Long, List<MapPointData>, Pair<Long, List<MapPointData>>> { t1, t2 ->
Pair(t1, t2)
})
.map { pair ->
Observable.just(
MapInformation(
mapPoints = pair.second.map { mapper.mapToDomain(it) },
source = Source.CACHE,
timeStamp = pair.first
)
)
}
}
.flatMap { it }
}
```
## Unit Testing
JUnit4, Mockito, Room-in-memory testing was used for tests included. For example, to unit test the RXJava call above, the data repositories are mocked using Mockito. Due to dependency injection, there are very loose coupling between classes which makes it very suitable for comprehensive unit testing coverage.
## LeakCanary
LeakCanary was used to detect and anticipate potential memory leaks. The app yielded 0% memory leaks through extensive debug testing.
## Android Jetpack
Jetpack is a suite of libraries, tools, and guidance to help developers write high-quality apps easier. Many Jetpack libraries are used in this application.
### Room Library
The app uses Room Library (Jetpack) to store and access SQLite database information on the device. SQLite is being used to cache the map information information- which will be accessed when the device cannot connect to the network to obtain map information.
### Data Binding
Used to bind observable data to UI elements directly in the XML layouts, minimizing boilerplate code in the fragments or activities employing them. It also eliminates the need to setup click listener callbacks in some casses, such as when passing click events and data binding variables to receiving methods, such as `mapInfo` embeded on the RecyclerView `viewholders`.
Data binding is leveraged heavily. It was used to bind observable data to UI elements directly in the XML layouts, minimizing boilerplate code in the fragments or activities employing them. It also eliminates the need to setup click listener callbacks in some casses, such as when passing click events and data binding variables to receiving methods, such as `mapInfo` embeded on the RecyclerView `viewholders`.
### Lifecycle-Aware Components
Helped keep UI controllers (activities and fragments) as lean as possible. For instance, the MapBox library which is heavily reliant on lifecycle states (`onCreate`, `onStart`, `onDestroy`, etc.) was able to employ the use of lifecycle observers allowing them to be cleanly separated on its own.
### ViewModel
Expand All @@ -75,5 +136,8 @@ Helps decouple the control logic from the UI elements in a lifecycle conscious w
It helped in supporting the reactive programming approach that supported the clean architecture design of the app. The live data was used as *observables* that notifies its *observers* (view elements) when something needs to be updated.
### Navigation
Specifically navigation graph was used to place content inside the main container, called a `NavHost`. The app only has a single view, but as more views are added, a 'NavController' can be used to orchestrate the swapping content in the `NavHost`.
## Other Libraries used
* Kotlin Coroutines
* Kotlin Serialization


Binary file added resources/images/app_screenshots.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.

0 comments on commit bac81ce

Please sign in to comment.