Skip to content

GrinGraz/marvelapp-android

Repository files navigation

Marvel Catalog [WIP]

Repository of the Marvel Catalog Android App (iOS soon). Is a totally functional app build in Kotlin following the SOLID principles and clean architecture, using the android guidelines and development tools in jetpack. Is fully modularized using Gradle as build system.

Features

As an app is a simple listing of the official Marvel API with two sections, the character listing itself and a specific character search, the items in both lists opens a character detail view with a little description and a brief list of comics, stories, series and events of the heroe or villain.

Screenshots

Listing Search Detail

Development Environment

The repository is ready to import in Android Studio, configured with Java 11 and Gradle. In order to request the Marvel API you should add an auth.properties file (ignored by VCS) to the root project directory containing two fields privateKey and publicKey, both keys provided by the Marvel developer platform. Also the keys can be configured as system enviroment variables, the env vars configuration is mandatory for the CI system, Github Actions in this case, to be able to run.

** An internal tools is use for feature flag,

Modularization

The project modularization has some key characteristics in mind loose coupling and high cohesion, easy to start collaborating, testability and low build times. The app components are splited in 3 main modules, app, features and core.

The app module is a simple container with the navigation graph and minimum configurations, it depends on all feature modules and in none or more core modules.

The feature modules represents a unit of functionality for the user that involves a view. These do not know anything about other feature modules or the app module and can have zero or more core modules as dependencies.

Finally core modules contains components to resolve common task as network request, analytics or databases.

There is an experimental branch where even the clean architecture layers are independant modules inside the feature module, that bring the posibility of share a functionality without presentation in a backen-ish way.

There is a buildScr module to provide gradle dependencies and configurations to the entire project. There are some experimental dependency implementations in the build.gradle files.

imagen

Architecture

The Marvel Catalog app follows the SOLID and clean architecture principles. Each feature module contains 3 main layers data, domain and presentation that comunicate between them throught interfaces using manual dependency injection, the module graph is internal and is created and requested in runtime from the Fragment to provide the ViewModelProvider.Factory to the viewModels delegate, all other depedencies are file private and passed by contructor.

imagen

Data layer

The app has only a remote data source by feature module that use the core:network module. It is represented by a Retrofit interface requested by the data source suspended functions implementations where the raw Retrofit Response is processed to return it corresponding model or error wrapped in the Arrow Either data type.

The repository implementation orchestrates the data sources and maps the response model to a domain model with the key properties for the feature still wrapped in Either.

This layer is requested by the repository interface defined in the domain layer.

This layer do not have any android dependencies.

Domain layer

The domain layer does not depend on any layer. It contains the repository definition that expect to returns domain models, also has the app business actions, like get the characters or search for one, these are simple commands because of that they are kotlin functional interfaces.

This layer is requested by the use case interface.

This layer do not have any android dependencies.

Presentation layer

The app has a single activity as entry point that launch the :feature:... modules. Inside the modules the user interfaces are managed by an Android Fragment with a bare minimun logic to change elements visibilitiy reactively based on the state and data returned by the viewmodel in the form of a StateFlow.

From this layer all the requests are executed via Kotlin Coroutines using the android specific CoroutineScope finally the ViewModel is resposible of wraps the domain model to the expected flow.

Navigation is executed via deeplinks.

The app UI supports a simply dark mode based on device configuration.

Stub Activitys, build and Manifest implementations are commented for independent module execution.

Linting and Testing

Linting

Detekt and Ktlint tasks are executed in the CI to ensure the style and good practices.

Unit Test

For these test the main goal besides making a safe net, is to explicit each action inside of the software components what are being tested, as seen in TDD and BDD, using the basis of the Gherkin language.

To achieve this, the unit tests were leveraged on Junit5 and Mockk, in a mix of mocks and fakes. The component under test is a spy that is executed with its real behavior and its dependencies are mocks that return fakes. All the tests have assertions besides exhaustive verifications and confirmations.

The tests run for each push and pull request in the CI.

The di, data, domain and presentation layer are 100% covered in each feature module.

Screenshots

Detail Listing

Instrumented Test

The Fragments happy path in each module are tested independendly using Junit4, androidx fragment testing tools and a Espresso wrapper library called Barista, that simplify the assertions and interactions with the UI elements.

Next Steps

Development Environment

  • Create product flavors.
  • Improve Gihub Actions CI (releases and tags creation).
  • Add Benchmarks.

Features

  • Implement favorites view.
  • Improve search view with filters.
  • Add SnackBars for show connections/server issues and retry.

Architecture And Modularization

  • Add a Compose version of UI and presentation layer.
  • Add Dagger/Hilt for dependency injection.
  • Review the :feature:common module to evaluate its scalability.
  • Review the design of complex use cases.
  • Creation of missing :core modules (navigation, test, desing system).
  • Creation of local data sources (Favorites feature).

Test and linting

  • Add Spotless.
  • Add Jacoco for full code coverage reports.
  • Add app instumented test.
  • Add navigation test.
  • Add mock web server.
  • Add Git hooks.
  • Add Danger.

Technical Debt

  • Add internal visibility modifier to :core and :feature module components.
  • Fix inconsistent behavior loading the character list when change of section before being loaded.
  • Add network state listeners to avoid false requests
  • Replace ListAdapter
  • Clean up gradle dependencies experiments.
  • Management of string resources in view models.
  • Transform :feature:common into a Android library.
  • Remove error response model from core:network.
  • Improve poor test definitions and add unhappy path for instrumented test.
  • Add full suit of API error tests.
  • Instrumented test runs with real requests.
  • Improve resources definitions (themes, string, styles, etc.).
  • Add missing docs and reference links.

Contributions

If you find any flawed spot, please feel free to open an issue with your suggestion or fix.

License

Marvel Catalog is distributed under the terms of the Apache License (Version 2.0). See the license for more information.