- Overview
- Architecture
- Setup
- Running the mobile wallet
- Debugging & App Profiling
- Testing
- Building APKs / Bundles
- Other
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.
The app uses React Native.
You must have the monorepo successfully set up and built before setting up and running the mobile wallet. To do this, follow the setup instructions.
# On a mac
brew install watchman
brew install jq
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.)
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 needs to be disabled. You can do this by setting
FIREBASE_ENABLED=false
in the.env.*
files.
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.
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 13 (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., Xcode10.3.app
and Xcode11.app
).
If you are on an M1, please read how to setup the environment on an M1 before you continue.
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
If your machine does not recognize the gem
command, you may need to download Ruby first.
- Run
yarn install
in the repository root. - Run
yarn dev:ios
in the repository root.
And the app should be running in the simulator! If you run into any issues, see below for troubleshooting.
Currently it is not possible to install the wallet natively. There are a few problems that need to be addressed before being able to run the repo on an M1:
-
The M1 comes with a preinstalled version of Ruby that doesn't work with this repository.
-
The build process that gets executed with
yarn dev:ios
is not able to finish withnvm
installed. -
Running the necessary scripts with an M1-native node version will not work.
-
It is currently not possible to run the repository in the integrated VSCode terminal.
-
It is not possible to run the repository when the containing folder is located anywhere within
Documents
orDesktop
.
- Make sure all necessary software (VS Code, Terminal/iTerm2, XCode, Simulator) is running with Rosetta. You can verify this by typing
arch
in your terminal.
$ arch
-> i386
- Reinstall
brew
with x86_64 architecture:
arch -x86_64 /bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/install.sh)"
- (optional) add your different brew installation paths as aliases to your
.zshrc
:
alias ibrew='arch -x86_64 /usr/local/bin/brew'
alias mbrew='arch -arm64e /opt/homebrew/bin/brew'
- Install
rbenv
on Intel architecture:
ibrew install rbenv
4.1. Add rbenv initialization to your shell. For instance in .zshrc
or .bashrc
:
eval "$(rbenv init -)"
- Install ruby version 2.7.6 with
rbenv
and set it as the main version:
rbenv install 2.7.6
rbenv global 2.7.6
- Install the required node version from
.nvmrc
with Intel architecture (If you have it installed under M1 architecture already, uninstall):
(nvm uninstall 16.5.0)
nvm install 16.5.0
nvm use
Verify that node is using x64 architecture:
node -e 'console.log(process.arch)'
-> x64
- The build script will fail if
node
+npm
andyarn
have been installed throughbrew
. Please uninstall them through brew (the nvm installations will still be there) and install them throughnvm
:
brew uninstall npm yarn node
which node
-> /Users/[youruser]/.nvm/versions/node/v16.15.0/bin/node
nvm install-latest-npm
which npm
-> /Users/[youruser]/.nvm/versions/node/v16.15.0/bin/npm
npm install -g yarn
which yarn
-> /Users/[youruser]/.nvm/versions/node/v16.15.0/bin/yarn
- Now verify that everything is correct for the build:
which ruby
-> /opt/homebrew/opt/rbenv/shims/ruby
which node
-> /Users/[youruser]/.nvm/versions/node/v16.15.0/bin/node
node -e 'console.log(process.arch)'
-> x64
which npm
-> /Users/[youruser]/.nvm/versions/node/v16.15.0/bin/npm
which yarn
-> /Users/[youruser]/.nvm/versions/node/v16.15.0/bin/yarn
Now follow the steps for iOS installation.
We need Java to be able to build and deploy the mobile app to Android devices. Android currently only builds correctly with Java 8. (Using OpenJDK because of Oracle being Oracle).
Install by running the following:
brew install cask
brew tap homebrew/cask-versions
brew install --cask homebrew/cask-versions/adoptopenjdk8
Optionally, install Jenv to manage multiple Java versions:
brew install jenv
eval "$(jenv init -)"
# next step assumes openjdk8 already installed
jenv add /Library/Java/JavaVirtualMachines/adoptopenjdk-8.jdk/Contents/Home/
Install by running the following:
sudo apt install openjdk-8-jdk
Install the Android SDK and platform tools:
brew install --cask android-sdk
brew install --cask android-platform-tools
Next install Android Studio and add the Android NDK (if you run into issues with the toolchain, try using version: 22.x).
Execute the following (and make sure the lines are in your ~/.bash_profile
).
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=${YOUR_ANDROID_SDK_PATH}
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
Then install the Android 29 platform:
sdkmanager 'platforms;android-29'
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
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:$PATH
Install the Android 29 system image and create an Android Virtual Device:
sdkmanager "system-images;android-29;default;x86_64"
avdmanager create avd --force --name Pixel_API_29_AOSP_x86_64 --device pixel -k "system-images;android-29;default;x86_64"
Run the emulator with:
emulator -avd Pixel_API_29_AOSP_x86_64
Another Android emulator option is Genymotion.
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
)
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
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
-
If you haven't already, run
yarn
and thenyarn build
from the repository root to install and build dependencies. -
Attach your device or start an emulated one.
-
Launch Xcode and use it to open the directory
celo.xcworkspace
. Confirm your iOS device has been detected by Xcode. -
Build the project by pressing the play button in the top left corner or selecting
Product > Build
from the Xcode menu bar. -
From the repository root directory run
yarn run dev:ios
.
-
Follow these instructions to enable Developer Options on your Android device.
-
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
) -
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. -
From the repository root directory run
yarn run dev:android
.
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
.
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.
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:
- Reactotron (Flipper -> Manage Plugins -> Install Plugins -> flipper-plugin-reactotron)
- Redux Debugger (Flipper -> Manage Plugins > Install Plugins > search redux-debugger)
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
- AsyncStorage
- App preferences
- Hermes
- and more ;)
Run yarn run react-devtools
. 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.
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.
To execute the suite of tests, run yarn test
.
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.
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
We use redux-saga-test-plan to test complex sagas.
See [src/identity/verification.test.ts
] for an example.
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"
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:
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 celo-release-key.keystore -alias celo-key-alias -storepass celoFakeReleaseStorePass -keypass celoFakeReleaseKeyPass -keyalg RSA -keysize 2048 -validity 10000 -dname "CN=Android Debug,O=Android,C=US"
export CELO_RELEASE_STORE_PASSWORD=celoFakeReleaseStorePass
export CELO_RELEASE_KEY_PASSWORD=celoFakeReleaseKeyPass
# 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.
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:
- Developers update the base strings in English (in
locales/base
) in the branch they are working on. - When the corresponding PR is merged into
main
, Crowdin integration automatically picks up changes to the base strings. - Crowdin then auto translates the new strings and opens a PR with them from the
l10n/main
branch - 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.
- 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 newl10n/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.
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 route 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.
- Go to the play console for the relevant app, Release management > App signing, and download the App signing certificate.
- Use this script to generate the hash code: https://github.com/michalbrz/sms-retriever-hash-generator
We're using GraphQL Code Generator to properly type GraphQL queries. If you make a change to a query, run yarn build:gen-graphql-types
to update the typings in the typings
directory.
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.
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?
- Run
yarn test:update-root-state-schema
. This will ensure the JSON schema is in sync with the RootState TypeScript type. - Review the changes in the schema
- Increase the schema version in src/redux/store.ts
- Add a new migration in src/redux/migrations.ts
- 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.
- Optional: if the migration is not trivial, add a test for it in src/redux/migrations.test.ts
- Commit the changes
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.
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.
We have a script to check for vulnerabilities in our dependencies.
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-runyarn 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 inpackage.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:
- run:
yarn audit --json --groups dependencies --level high | grep auditAdvisory > yarn-audit-known-issues
- commit
yarn-audit-known-issues
and open a PR
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] ✔
[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.
Make sure to follow the steps here to set up Google Cloud correctly with the wallet.
Images and icons in Valora are stored in the branding repo. When running yarn install
, the script scripts/sync_branding.sh
is run to clone this repo into branding/valora
, and these assets are then put into src/images
and src/icons
. If you do not have access to the branding repo, assets are pulled from branding/celo
, and are displayed as pink squares instead. The jest tests and CircleCI pipeline also use these default assets.
When adding new images to the branding repo, we also include the 1.5x, 2x, 3x, and 4x versions. The app will automatically download the appropriate size. After making changes to the remote repo, find the commit hash and update it in scripts/sync_branding.sh
. Make sure to also add the corresponding pink square version of the images to branding/celo/src/images
. You can do this by copying one of the existing files and renaming it.
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