secant-android-wallet/docs/Architecture.md

7.9 KiB

Design and Architecture

Note: This document will continue to be updated as the app is implemented.

Gradle

  • Versions are declared in gradle.properties. There's still enough inconsistency in how versions are handled in Gradle, that this is as close as we can get to a universal system. A version catalog is used for dependencies and is configured in settings.gradle.kts, but other versions like Gradle Plug-ins, the NDK version, Java version, and Android SDK versions don't fit into the version catalog model and are read directly from the properties
  • Much of the Gradle configuration lives in build-convention to prevent repetitive configuration as additional modules are added to the project
  • Build scripts are written in Kotlin, so that a single language is used across build and the app code bases
  • Only Gradle, Google, and JetBrains plug-ins are included in the critical path. Third party plug-ins can be used, but they're outside the critical path. For example, the Gradle Versions Plugin could be removed and wouldn't negatively impact local building, testing, or releasing the app
  • Repository restrictions are enabled in build-convention, settings.gradle.kts, and build.gradle.kts to reduce likelihood of pulling in an incorrect dependency. If adding a new dependency, these restrictions may need to be changed otherwise an error that the dependency cannot be found will be displayed

Multiplatform

While this repository is for an Android application, efforts are made to give multiplatform flexibility in the future. Specific adaptions that are being made:

  • Where possible, common code is extracted into multiplatform modules. This sometimes means that additional modules with an -android suffix exist to add Android-specific extensions. In the future, we would like to move towards multiplatform modules with source directories along the lines of commonJvmMain, jvmMain, and androidMain.
  • In UI state management code, Kotlin Flow is often preferred over Android LiveData and Compose State to grant future flexibility
  • Saver is preferred over @Parcelize for objects in the SDK

Note: Test coverage for multiplatform modules behaves differently than coverage for Android modules. Coverage is enabled by default for the JVM target when running ./gradlew check.

App

The main entrypoints of the application are:

  • AppImpl.kt - The root Application object defined in the app module
  • MainActivity.kt - The main Activity, defined in ui-lib. Note that the Activity is NOT exported. Instead, the app module defines an activity-alias in the AndroidManifest which is what presents the actual icon on the Android home screen.

Modules

The logical components of the app are implemented as a number of Gradle modules.

  • app — Compiles all of the modules together into the final application. This module contains minimal actual code. Note that the Java package structure for this module is under co.electriccoin.zcash.app while the Android package name is co.electriccoin.zcash.
  • build-info-lib — Collects information from the build environment (e.g. Git SHA, Git commit count) and compiles them into the application. Can also be used for injection of API keys or other secrets.
  • crash — For collecting and reporting exceptions and crashes
    • crash-lib — Common crash collection logic for Kotlin and JVM. This is not fully-featured by itself, but the long-term plan is multiplatform support.
    • crash-android-lib — Android-specific crash collection logic, built on top of the common and JVM implementation in crash-lib
  • ui
    • ui-design — Contains UI theme elements only. Besides offering modularization, this allows for hiding of some Material Design components behind our own custom components.
    • ui-lib — User interface that the user interacts with. This contains 99% of the UI code, along with localizations, icons, and other assets.
    • ui-integration-test-lib — Is dedicated for integration tests only. It has Android Test Orchestrator turned on — it allows us to run each of our tests within its own invocation of Instrumentation, and thus brings us benefits for the testing environment (minimal shared state, crashes are isolated, permissions are reset).
  • preference
    • preference-api-lib — Multiplatform interfaces for key-value storage of preferences.
    • preference-impl-android-lib — Android-specific implementation for preference storage.
  • sdk
    • sdk-ext-lib — Contains extensions on top of the to the Zcash SDK. Some of these extensions might be migrated into the SDK eventually, while others might represent Android-centric idioms. Depending on how this module evolves, it could adopt another name such as wallet-lib or be split into two.
    • sdk-ext-ui — Place for Zcash SDK components (same as sdk-ext-lib), which are related to the UI (e.g. depend on user locale and thus need to be translated via strings.xml).
  • spackle — Random utilities, to fill in the cracks in the frameworks.
    • spackle-lib — Multiplatform implementation for Kotlin and JVM
    • spackle-android-lib — Android-specific additions.

The following diagram shows a rough depiction of dependencies between the modules. Two notes on this diagram:

  • sdk-lib is in a different repository
  • Although effort goes into ensuring this diagram stays up-to-date, Gradle build files are the authoritative source on dependencies
  flowchart TB;
      subgraph sdk
          sdkLib[[sdk-lib]];
          sdkExtLib[[sdk-ext-lib]];
          sdkExtUi[[sdk-ext-ui]];
      end
      sdkLib[[sdk-lib]] --> sdkExtLib[[sdk-ext-lib]] --> sdkExtUi[[sdk-ext-ui]];
      subgraph preference
          preferenceApiLib[[preference-api-lib]];
          preferenceImplAndroidLib[[preference-impl-android-lib]];
      end
      preferenceApiLib[[preference-api-lib]] --> preferenceImplAndroidLib[[preference-impl-android-lib]];
      subgraph crash
          crashLib[[crash-lib]];
          crashAndroidLib[[crash-android-lib]];
      end
      crashLib[[crash-lib]] --> crashAndroidLib[[crash-android-lib]];
      subgraph ui
          uiDesignLib[[ui-design-lib]];
          uiLib[[ui-lib]];
          uiIntegrationTestLib[[ui-integration-test-lib]];
      end
      uiDesignLib[[ui-design-lib]] --> uiLib[[ui-lib]];
      uiLib[[ui-lib]] --> uiIntegrationTestLib[[ui-integration-test-lib]];
      uiDesignLib[[ui-design-lib]] -- > uiIntegrationTestLib[[ui-integration-test-lib]];
      subgraph spackle
          spackleLib[[spackle-lib]];
          spackleAndroidLib[[spackle-android-lib]];
      end
      spackleLib[[spackle-lib]] --> spackleAndroidLib[[spackle-android-lib]];
      preference --> ui[[ui]];
      sdk --> ui[[ui]];
      spackle[[spackle]] --> ui[[ui]];
      ui[[ui]] --> app{app};
      crash[[crash]] --> app{app};

Test Fixtures

Until the Kotlin adopts support for fixtures, fixtures live within the main source modules. These fixtures make it easy to write automated tests, as well as create Compose previews. Although these fixtures are compiled into the main application, they should be removed by R8 in release builds.

Shared Resources

There are some app-wide resources that share a common namespace, and these should be documented here to make it easy to ensure there are no collisions.

  • SharedPreferences
    • "co.electriccoin.zcash.encrypted" is defined as a preference file in EncryptedPreferenceSingleton.kt
  • Databases
    • Some databases are defined by the SDK
  • Notification IDs
    • No notification IDs are currently defined
  • Notification Channels
    • No notification channels are currently defined
  • WorkManager Tags
    • "co.electriccoin.zcash.background_sync" is defined in WorkIds.kt