Skip to content

Latest commit

 

History

History
725 lines (505 loc) · 34.1 KB

WALLET.md

File metadata and controls

725 lines (505 loc) · 34.1 KB

Mobile (Valora)

Overview

This package contains the code for the Valora mobile apps for Android and iOS. Valora is a self-sovereign wallet that enables anyone to onboard onto the Celo network, manage their currencies, and send payments.

Architecture

The app uses React Native.

Setup

Prerequisites

Install Homebrew if you are on macOS.

Install NVM if you don't have any Node version manager.

Install Node version listed in .nvmrc and make it default (example for NVM):

nvm install --default

Install Yarn

npm install --global yarn

Install watchman and jq

# On a mac
brew install watchman
brew install jq

Repository secrets

For Valora employees only

This is only for Valora employees.

You will need to be added the team keyring on GCP so you can decrypt secrets in the repo. (Ask for an invite to celo-mobile-alfajores.)

Once you have access, install Google Cloud by running brew install google-cloud-sdk. Follow instructions here for logging in with Google credentials.

To test your GCP access, try running yarn keys:decrypt from the wallet repo root. You should see something like this: Encrypted files decrypted. (You will not need to run this command on an ongoing basis, since it is done automatically as part of the postinstall script.)

For External contributors

External contributors don't need to decrypt repository secrets and can successfully build and run the mobile application with the following differences:

  • the default branding will be used (some images/icons will appear in pink or will be missing)
  • Firebase related features need to be disabled. You can do this by setting FIREBASE_ENABLED=false in the .env.* files.

iOS

Enroll in the Apple Developer Program

In order to successfully set up your iOS development environment you will need to enroll in the Apple Developer Program. It is recommended that you enroll from an iOS device by downloading the Apple Developer App in the App Store. Using the app will result in the fastest processing of your enrollment.

If you are a Valora employee, please ask to be added to the Valora iOS development team.

Install Xcode

Xcode is needed to build and deploy the mobile wallet to your iOS device. If you do not have an iOS device, Xcode can be used to emulate one.

Install Xcode 15.2 (an Apple Developer Account is needed to access this link).

We do not recommend installing Xcode through the App Store as it can auto update and become incompatible with our projects.

Note that using the method above, you can have multiple versions of Xcode installed in parallel if you'd like. Simply use different names for the different version of Xcode in your computer's Applications folder (e.g., Xcode14.3.1.app and Xcode15.2.app).

Install Ruby, Cocoapods, Bundler, and download project dependencies

Install Ruby 2.7 and make it global

brew install rbenv ruby-build

# run and follow the printed instructions:
rbenv init

rbenv install 2.7.8
rbenv global 2.7.8

Make sure you are in the ios directory of the repository root before running the following:

# install cocopods and bundler if you don't already have it
gem install cocoapods
gem install bundler
# download the project dependencies in repository root
bundle install
# run inside /ios
bundle exec pod install
  1. Run yarn install in the repository root.
  2. Run yarn dev:ios in the repository root.

Install Rosetta (M1 macs only)

If you are unable to run the app in the iOS Simulator, install Rosetta:

/usr/sbin/softwareupdate --install-rosetta --agree-to-license

Android

Download and install Android Studio and the following add-ons:

  • Android SDK
  • Android SDK Platform
  • Android Virtual Device

Install the Android 13 (Tiramisu) SDK. It can be found can be installed through the SDK Manager in Android Studio.

Configure the ANDROID_HOME environment variables by adding the following lines to your ~/.zprofile or ~/.zshrc. You can find the actual location of the SDK in the Android Studio "Preferences" dialog, under Appearance & Behavior → System Settings → Android SDK.

export ANDROID_HOME=$HOME/Library/Android/sdk
export PATH=$PATH:$ANDROID_HOME/emulator
export PATH=$PATH:$ANDROID_HOME/platform-tools

MacOS

After installing Andoid Studio, add the Android NDK (if you run into issues with the toolchain, try using version: 22.x).

Make sure these lines are in your shell profile (~/.bash_profile, ~/.zshrc etc.):

Note that these paths may differ on your machine. You can find the path to the SDK and NDK via the Android Studio menu.

export ANDROID_HOME=$HOME/Library/Android/sdk
export PATH=$PATH:$ANDROID_HOME/emulator
export PATH=$PATH:$ANDROID_HOME/platform-tools
export ANDROID_NDK=$ANDROID_HOME/ndk-bundle
export ANDROID_SDK_ROOT=$ANDROID_HOME
# this is an optional gradle configuration that should make builds faster
export GRADLE_OPTS='-Dorg.gradle.daemon=true -Dorg.gradle.parallel=true -Dorg.gradle.jvmargs="-Xmx4096m -XX:+HeapDumpOnOutOfMemoryError"'
export TERM_PROGRAM=iterm  # or whatever your favorite terminal program is

(optional) You may want to install Jenv to manage multiple Java versions:

brew install jenv
eval "$(jenv init -)"
# next step assumes jdk already installed
jenv add /Library/Java/JavaVirtualMachines/zulu-11.jdk/Contents/Home

Linux

Install Java by running the following:

sudo apt install openjdk-11-jdk

You can download the complete Android Studio and SDK from the Android Developer download site.

You can find the complete instructions about how to install the tools in Linux environments in the Documentation page.

Set the following environment variables and optionally add to your shell profile (e.g., .bash_profile):

export ANDROID_HOME=/usr/local/share/android-sdk
export ANDROID_SDK_ROOT=/usr/local/share/android-sdk
# this is an optional gradle configuration that should make builds faster
export GRADLE_OPTS='-Dorg.gradle.daemon=true -Dorg.gradle.parallel=true -Dorg.gradle.jvmargs="-Xmx4096m -XX:+HeapDumpOnOutOfMemoryError"'
# this is used to launch the react native packager in its own terminal
export TERM_PROGRAM=xterm  # or whatever your favorite terminal is
Optional: Install an Android emulator
Configure an emulator using the Android SDK Manager

Set your PATH environment variable and optionally update your shell profile (e.g., .bash_profile):

export PATH=$ANDROID_HOME/emulator:$ANDROID_HOME/tools:$ANDROID_HOME/platform-tools:$ANDROID_HOME/cmdline-tools/latest/bin:$PATH

Install the Android 31 system image and create an Android Virtual Device.

For Intel chip Macs:
sdkmanager "system-images;android-31;default;x86_64"
avdmanager create avd --force --name Pixel_API_31_AOSP_x86_64 --device pixel -k "system-images;android-31;default;x86_64"
For M1 Macs:

On an M1 mac, the above commands may succeed, but when you try to run the emulator it will fail saying you have an unsupported architecture. To get around this, you can manually create the Android Virtual Device in Android Studio by doing the following:

  • Open the wallet repo in Android Studio. In the top bar, click Tools -> Device Manager. A side-bar should pop up on your screen showing your virtual devices (if any).
  • Click "Create Device", choose a device (e.g. Pixel 6 Pro), and hit next.
  • When prompted to select a system image, click the "ARM Images" tab, and choose an image where the target does not include "Google APIs". You may need to download the system image, to do that hit the download icon next to the release name. For e2e testing, choose release name "Q" a.k.a. Android 10.
  • Give the device a name, and hit finish If you are creating a virtual device for e2e testing purposes, name your device Pixel_API_29_AOSP_x86_64.

Run the emulator with:

emulator -avd <virtual-device-name (e.g. Pixel_API_31_AOSP_x86_64)>
Install Genymotion Emulator Manager

Another Android emulator option is Genymotion.

MacOS
brew install --cask genymotion

Under OSX High Sierra and later, you'll get a message that you need to approve it in System Preferences > Security & Privacy > General.

Do that, and then repeat the line above.

Then make sure the ADB path is set correctly in Genymotion — set Preferences > ADB > Use custom Android SDK tools to /usr/local/share/android-sdk (same as $ANDROID_HOME)

Linux

You can download the Linux version of Genymotion from the fun zone! (you need to sign in first).

After having the binary you only need to run the installer:

sudo ./genymotion-3.0.2-linux_x64.bin

Running the mobile wallet

The below steps should help you successfully run the mobile wallet on either a USB connected or emulated device. For additional information and troubleshooting see the React Native docs.

Note: We've seen some issues running the metro bundler from iTerm

  1. If you haven't already, run yarn and then yarn build from the repository root to install and build dependencies.

  2. Attach your device or start an emulated one.

iOS

  1. Launch Xcode and use it to open the directory MobileStack.xcworkspace. Confirm your iOS device has been detected by Xcode.

  2. Build the project by pressing the play button in the top left corner or selecting Product > Build from the Xcode menu bar.

  3. From the repository root directory run yarn run dev:ios.

Android

  1. Follow these instructions to enable Developer Options on your Android device.

  2. Unplug and replug your device. You'll be prompted to accept the connection and shown a public key (corresponding to the abd_key.pub file in ~/.android)

  3. To confirm your device is properly connected, running adb devices from the terminal should reflect your connected device. If it lists a device as "unauthorized", make sure you've accepted the prompt or troubleshoot here.

  4. From the repository root directory run yarn run dev:android.

Running on Mainnet

By default, the mobile wallet app runs on celo's testnet alfajores. To run the app on mainnet, supply an env flag, eg. yarn run dev:ios -e mainnet. The command will then run the app with the env file .env.mainnet.

Reinstalling the app without building

To test some scenarios (e.g., native permissions modals which appear only once), you may require a fresh install of the app. Instead of rebuilding the app to get a fresh install, you can drag drop the generated app into the simulator after uninstalling the app. It is typically available in the following paths:

  • For iOS: $HOME/Library/Developer/Xcode/DerivedData/MobileStack-<randomid>/Build/Products/Debug-iphonesimulator/Valora.app
  • For Android: <path-to-wallet>/android/app/build/outputs/apk/alfajoresdev/debug/app-alfajoresdev-debug.apk

Debugging & App Profiling

Debugging

Since we integrated dependencies making use of TurboModules, debugging via Chrome DevTools or React Native Debugger doesn't work anymore. As an alternative, Flipper can be used instead.

Install Flipper

Flipper is a platform for debugging iOS, Android and React Native apps. Visualize, inspect, and control your apps from a simple desktop interface. Download on the web or through brew.

brew install flipper

As of Jan 2021, Flipper is not notarized and triggers a MacOS Gatekeeper popup when trying to run it for the first time. Follow these steps to successfully launch it (only needed the very first time it's run)

The application currently makes use of 2 additional Flipper plugins to enable more detailed debugging:

  • Redux Debugger (Flipper -> Manage Plugins -> Install Plugins -> search redux-debugger)
  • React Navigation (Flipper -> Manage Plugins -> Install Plugins -> search react-navigation)

Once installed, you should be able to see them and interact with them when the wallet is running (only in dev builds).

This allows viewing / debugging the following:

  • React DevTools (Components and Profiling)
  • Network connections
  • View hierarchy
  • Redux State / Actions
  • Navigation State
  • AsyncStorage
  • App preferences
  • Hermes
  • and more ;)

If you're using an Android simulator and the device / app is not showing up, navigate to settings (gear icon in the bottom left) and ensure the Android SDK location points to the same location as the $ANDROID_HOME environment variable.

App Profiling with react-devtools

From Flipper select React DevTools Plugin while the app is running locally, or run yarn run react-devtools in the wallet root folder. It should automatically connect to the running app and includes a profiler (second tab). Start recording with the profiler, use the app and then stop recording. If running from the terminal, Flipper cannot be run at the same time.

The flame graph provides a view of each component and sub-component. The width is proportional to how long it took to load. If it is grey, it was not re-rendered at that 'commit' or DOM change. Details on the react native profiler are here. The biggest thing to look for are large number of renders when no state has changed. Reducing renders can be done via pure components in React or overloading the should component update method example here.

App Profiling with Android Profiler

Profiling in release mode is recommended because memory usage tends to be significantly higher in development builds. To create a local mainnet release build for profiling, use the following command: yarn dev:android -e mainnet -r -t. This supplies an env flag: -e <environment>, the release flag: -r and the profile flag: -t.

To analyze the app's memory, CPU, and energy usage, the Android APK Profiler is a useful tool. In Android Studio, navigate to File > Profile or Debug APK, then select the APK built in the previous step, typically located at android/app/build/outputs/apk/mainnet/release. Once both the app and Android Studio are running, attach a new profiling session by selecting your device and choosing the debuggable process, such as co.clabs.valora.

Testing

To execute the suite of tests, run yarn test.

Snapshot testing

We use Jest snapshot testing to assert that no intentional changes to the component tree have been made without explicit developer intention. See an example at [src/send/SendAmount.test.tsx]. If your snapshot is expected to deviate, you can update the snapshot with the -u or --updateSnapshot flag when running the test.

React component unit testing

We use react-native-testing-library and @testing-library/jest-native to unit test react components. It allows for deep rendering and interaction with the rendered tree to assert proper reactions to user interaction and input. See an example at [src/send/SendAmount.test.tsx] or read more about the docs.

To run a single component test file: yarn test Send.test.tsx

Saga testing

We use redux-saga-test-plan to test complex sagas. See [src/app/saga.test.ts] for an example.

End-to-End testing

We use Detox for E2E testing. In order to run the tests locally, you must have the proper emulator set up. Follow the instructions in e2e/README.md.

Once setup is done, you can build the tests with yarn e2e:build:android-release or yarn e2e:build:ios-release. Once test build is done, you can run the tests with yarn e2e:test:android-release or yarn e2e:test:ios-release. If you want to run a single e2e test: yarn e2e:test:ios-release Exchange.spec.js -t "Then Buy CELO"

Building APKs / Bundles

You can create your own custom build of the app via the command line or in Android Studio. For an exact set of commands, refer to the lanes in fastlane/FastFile. For convenience, the basic are described below:

Creating a fake keystore

If you have not yet created a keystore, one will be required to generate a release APKs / bundles:

cd android/app
keytool -genkey -v -keystore mobilestack-release-key.keystore -alias mobilestack-key-alias -storepass fakeReleaseStorePass -keypass fakeReleaseStorePass -keyalg RSA -keysize 2048 -validity 10000 -dname "CN=Android Debug,O=Android,C=US"
export RELEASE_STORE_PASSWORD=fakeReleaseStorePass

Building an APK or Bundle

# With fastlane:
bundle install
bundle exec fastlane android build_apk env:YOUR_BUILDING_VARIANT

# Or, manually
cd android/
./gradlew clean
./gradlew bundle{YOUR_BUILDING_VARIANT}JsAndAssets
# For an APK:
./gradlew assemble{YOUR_BUILDING_VARIANT} -x bundle{YOUR_BUILDING_VARIANT}JsAndAssets
# Or for a bundle:
./gradlew bundle{YOUR_BUILDING_VARIANT} -x bundle{YOUR_BUILDING_VARIANT}JsAndAssets

Where YOUR_BUILD_VARIANT can be any of the app's build variants, such as debug or release.

Other

Localization (l10n) / translation process

We are using Crowdin to manage the translation of all user facing strings in the app.

During development, developers should only update the language files in the base locale. These are the source files for Crowdin.

The main branch of this repository is automatically synced with our Crowdin project. Source files in Crowdin are updated automatically and ready translations are pushed as a pull request.

Translation process overview:

  1. Developers update the base strings in English (in locales/base) in the branch they are working on.
  2. When the corresponding PR is merged into main, Crowdin integration automatically picks up changes to the base strings.
  3. Crowdin then auto translates the new strings and opens a PR with them from the l10n/main branch
  4. We can then manually check and edit the translated strings in the Crowdin UI. The changes will be reflected in the PR after 10 mins.
  5. When we are happy with the changes, we can merge the PR and delete the related l10n/main branch to avoid possible future conflicts. Once new translations are made in Crowdin, a new l10n/main branch will be automatically created again.

When making a release, we should make sure there are no outstanding translation changes not yet merged into main. i.e. no Crowdin PR open and the translation status for all supported languages is at 100% and approved on Crowdin.

Note that Crowdin Over-The-Air (OTA) content delivery is used to push live translation updates to the app. As only target languages are included in the Crowdin OTA distribution, English is set up as a target language as well as the source. This is a necessary implementation detail to prevent bi-directional sync between Crowdin and Github. The translated English strings (in locales/en) are only to receive the OTA translations, and it is not necessary to consume or edit them otherwise within the app.

Configuring the SMS Retriever

On Android, the wallet app uses the SMS Retriever API to automatically input codes during phone number verification. When creating a new app build type this needs to be properly configured.

The service that routes SMS messages to the app needs to be configured to append this app signature to the message. The hash depends on both the bundle id and the signing certificate. Since we use Google Play signing, we need to download the certificate.

  1. Go to the play console for the relevant app, Release management > App signing, and download the App signing certificate.
  2. Use this script to generate the hash code: https://github.com/michalbrz/sms-retriever-hash-generator

Redux state migration

We're using redux-persist to persist the state of the app across launches.

Whenever we add/remove/update properties to the RootState, we need to ensure previous versions of the app can successfully migrate their persisted state to the new schema version. Otherwise it can lead to subtle bugs or crashes for existing users of the app, when their app is upgraded.

We have automated checks to ensure that the state migration is working correctly across all versions. You're probably reading this because these checks pointed you to this documentation. These checks are based on the JSON schema representation of the RootState TypeScript type. It is stored in test/RootStateSchema.json.

When is a migration or new schema version needed?

As a rule of thumb, a migration is needed whenever the RootState changes. That is whenever test/RootStateSchema.json changes.

Here we're optimizing for correctness and explicitness to avoid breaking existing users.

redux-persist can automatically handle newly added properties with its state reconcilier. However it leaves removed properties. Which is fine in the majority of the cases, but could create issues if later on a property is added again with the same name. And it only merges the initial state with the persisted state up to 2 levels of nesting (this is the autoMergeLevel2 config we are using).

So in general, if you're only adding a new reducer or adding a new property to an existing reducer, the migration can just return the input state. The state reconciler will do the right thing.

If you're deleting or updating existing properties, please implement the appropriate migration for them.

What do to when test/RootStateSchema.json needs an update?

  1. Run yarn test:update-root-state-schema. This will ensure the JSON schema is in sync with the RootState TypeScript type.
  2. Review the changes in the schema
  3. Increase the schema version in src/redux/store.ts
  4. Add a new migration in src/redux/migrations.ts
  5. Add a new test schema in test/schema.ts, with the newly added/deleted/updated properties. The test schema is useful to test migrations and show how the schema changed over time.
  6. Optional: if the migration is not trivial, add a test for it in src/redux/migrations.test.ts
  7. Commit the changes

Redux-Saga pitfalls

Error bubbling

It's important to understand how errors propagate with Redux-Saga.

Take the following example:

function* rootSaga() {
  yield spawn(mySaga)
  yield spawn(someOtherSaga)
}

function* mySaga() {
  yield takeEvery('SEND_PAYMENT', sendPayment)
  yield takeEvery('NOTIFY_USER', notifyUser)
}

function* someOtherSaga() {
  // [...]
}

If an exception is thrown from sendPayment or notifyUser, the whole mySaga will be cancelled. And won't handle SEND_PAYMENT AND NOTIFY_USER actions until the app is restarted.

Since mySaga was spawned from the root saga, someOtherSaga won't be affected though.

You may think that a good way to address this problem is to make sure sendPayment uses try/catch.

function* sendPayment() {
  try {
    // Code to send payment
    // [...]
  } catch (e) {
    Logger.error(e)
    yield put('SEND_PAYMENT_FAILED')
  }
}

However it's still possible that the catch block throws again, and we'd be back to the initial problem.

To avoid this problem, we recommend wrapping takeEvery/takeLatest/takeLeading worker sagas using the safely helper.

Note that you should still handle errors happening in your action handlers. But at least you'll have the guarantee that it won't unexpectedly stop listening to actions because of an unhandled error.

See more details https://redux-saga.js.org/docs/api#error-propagation

Why do we use http(s) provider?

Websockets (ws) would have been a better choice but we cannot use unencrypted ws provider since it would be bad to send plain-text data from a privacy perspective. Geth does not support wss by default. And Kubernetes does not support it either. This forced us to use https provider.

Helpful hints for development

We try to minimise the differences between running Valora in different modes and environments, however there are a few helpful things to know when developing the app.

  • Valora uses Crowdin Over-The-Air (OTA) content delivery to enable dynamic translation updates. The OTA translations are cached and used on subsequent app loads instead of the strings in the translation files of the app bundle. This means that during development, the app will not respond to manual changes of the translation.json files.
  • In development mode, analytics are disabled.

Vulnerabilities found in dependencies

We have a script to check for vulnerabilities in our dependencies.

The script reports all vulnerabilities found; compare its output with yarn-audit-known-issues to see which ones are new.

In case vulnerabilities are reported, check to see if they apply to production and if they have fixes available.

If they apply to production, start a discussion in our #on-call channel.

Then if they have fixes available, update the dependencies using Renovate or manually:

  • If it's a direct dependency, update the dependency in package.json.
  • If it's a transitive dependency, you can manually remove the transitive dependency in yarn.lock and re-run yarn install to see if it can use the fixed version. If the sub dependency is pinned somewhere, you'll need to use a yarn resolution in package.json to get the fixed version. Be careful with this as it can break other dependencies depending on a specific version.

If they do not have fixes and they do not apply to production, you may ignore them:

  1. run: yarn audit --json --groups dependencies --level high | grep auditAdvisory > yarn-audit-known-issues
  2. commit yarn-audit-known-issues and open a PR

Branding

Images related to the brand are stored in the src/images folder. When adding new images, we also include the 1.5x, 2x, 3x, and 4x versions. The app will automatically download the appropriate size.

Troubleshooting

Postinstall script

If you're having an error with installing packages, or secrets.json not existing:

try to run yarn postinstall in the wallet root folder after running yarn install.

If some of your assets are not loaded and you see an error running sync_branding.sh. Check if you have set up your Github connection with SSH.

A successful yarn postinstall looks like:

$ yarn postinstall
yarn run v1.22.17
$ patch-package && yarn keys:decrypt && yarn run unzipCeloClient && ./scripts/sync_branding.sh && ./scripts/copy_license_to_android_assets.sh
patch-package 6.2.2
Applying patches...
@react-native-community/[email protected][email protected][email protected][email protected][email protected][email protected][email protected][email protected] ✔
$ bash scripts/key_placer.sh decrypt
Processing encrypted files
Encrypted files decrypted
.
~/src/github.com/valora-inc/wallet/branding/valora ~/src/github.com/valora-inc/wallet
HEAD is now at ec0637b fix: update valora forum link (#9)
~/src/github.com/valora-inc/wallet
Using branding/valora
building file list ... done

sent 7697 bytes  received 20 bytes  15434.00 bytes/sec
total size is 8003608  speedup is 1037.14
building file list ... done

sent 91 bytes  received 20 bytes  222.00 bytes/sec
total size is 1465080  speedup is 13198.92
✨  Done in 12.77s.

Activity class {org.celo.mobile.staging/org.celo.mobile.MainActivity} does not exist.

From time to time the app refuses to start showing this error:

557 actionable tasks: 525 executed, 32 up-to-date
info Running /usr/local/share/android-sdk/platform-tools/adb -s PL2GARH861213542 reverse tcp:8081 tcp:8081
info Starting the app on PL2GARH861213542 (/usr/local/share/android-sdk/platform-tools/adb -s PL2GARH861213542 shell am start -n org.celo.mobile.staging/org.celo.mobile.MainActivity)...
Starting: Intent { cmp=org.celo.mobile.staging/org.celo.mobile.MainActivity }
Error type 3
Error: Activity class {org.celo.mobile.staging/org.celo.mobile.MainActivity} does not exist.

Solution:

$ adb kill-server && adb start-server
* daemon not running; starting now at tcp:5037
* daemon started successfully

Podfile changes not picked up by iOS build

Some Podfile changes may not be picked up by an iOS build (e.g., add new permissions with react-native-permissions here) and will need cleaning the XCode derived data folder.

rm -rf $HOME/Library/Developer/Xcode/DerivedData/*