name: e2e-Android on: workflow_dispatch: concurrency: group: ${{ github.workflow }}-${{ github.ref }} cancel-in-progress: true jobs: checksecret: name: check for oauth client runs-on: macos-latest outputs: is_SECRETS_PRESENT_set: ${{ steps.checksecret_job.outputs.is_SECRETS_PRESENT_set }} steps: - name: Check whether unity activation requests should be done id: checksecret_job env: SECRETS_PRESENT: ${{ secrets.OAUTH_CLIENT_SECRET }} run: | echo "is_SECRETS_PRESENT_set: ${{ env.SECRETS_PRESENT != '' }}" echo "::set-output name=is_SECRETS_PRESENT_set::${{ env.SECRETS_PRESENT != '' }}" test: needs: checksecret if: needs.checksecret.outputs.is_SECRETS_PRESENT_set == 'true' runs-on: macos-latest # Kill the task if not finished after 60 minutes timeout-minutes: 60 steps: - name: Check out Git repository uses: actions/checkout@v4 with: fetch-depth: 1 - name: Install Node.js, NPM and Yarn uses: actions/setup-node@v4 with: node-version: 22 # Use the Android command line tools to download an AOSP emulator image, and setup new avd # The name of the device for testing has to be the same as on the .detoxrc.js file (even if it is not a Pixel) # The mac-os latest runner has 3 different versions of Java pre-installed (8,11,17), and there were errors when setting to use either 11 or 17 explicitly here # I am assuming (but haven't tested) it uses 8 and then this step works - name: Download Android Emulator Image run: | echo "y" | $ANDROID_HOME/tools/bin/sdkmanager --install "system-images;android-31;default;x86_64" echo "no" | $ANDROID_HOME/tools/bin/avdmanager create avd --force --name Pixel_5_API_31_AOSP --device "Nexus 5X" -k 'system-images;android-31;default;x86_64' $ANDROID_HOME/emulator/emulator -list-avds - name: Cache node modules uses: actions/cache@v4 id: cache with: path: node_modules key: node-modules-${{ hashFiles('**/package-lock.json') }} - name: Rebuild detox from cache if: steps.cache.outputs.cache-hit == 'true' # Currently, (Detox 20) those two commands are macOS only run: npx detox clean-framework-cache && npx detox build-framework-cache # supposedly our current cache includes native modules like Realm # that have compiled binaries specific to the environment where they were installed # which might not be the same as the github actions environment # so we need to rebuild them - name: Rebuild native modules if: steps.cache.outputs.cache-hit == 'true' run: npm rebuild - name: Install Dependencies if: steps.cache.outputs.cache-hit != 'true' run: npm install # Generate the secret files needed for a release build - name: Create .env file env: OAUTH_CLIENT_ID: ${{ secrets.OAUTH_CLIENT_ID }} OAUTH_CLIENT_SECRET: ${{ secrets.OAUTH_CLIENT_SECRET }} E2E_TEST_USERNAME: ${{ secrets.E2E_TEST_USERNAME }} E2E_TEST_PASSWORD: ${{ secrets.E2E_TEST_PASSWORD }} JWT_ANONYMOUS_API_SECRET: ${{ secrets.JWT_ANONYMOUS_API_SECRET }} GMAPS_API_KEY: ${{ secrets.GMAPS_API_KEY }} run: printf 'API_URL=https://stagingapi.inaturalist.org/v2\nOAUTH_API_URL=https://staging.inaturalist.org\nJWT_ANONYMOUS_API_SECRET=%s\nOAUTH_CLIENT_ID=%s\nOAUTH_CLIENT_SECRET=%s\nE2E_TEST_USERNAME=%s\nE2E_TEST_PASSWORD=%s\nGMAPS_API_KEY=%s\nANDROID_MODEL_FILE_NAME=INatVision_Small_2_fact256_8bit.tflite\nANDROID_TAXONOMY_FILE_NAME=taxonomy.csv\nANDROID_GEOMODEL_FILE_NAME=INatGeomodel_Small_2_8bit.tflite\nIOS_MODEL_FILE_NAME=INatVision_Small_2_fact256_8bit.mlmodel\nIOS_TAXONOMY_FILE_NAME=taxonomy.json\nIOS_GEOMODEL_FILE_NAME=INatGeomodel_Small_2_8bit.mlmodel\nCV_MODEL_VERSION=small_2\n' "$JWT_ANONYMOUS_API_SECRET" "$OAUTH_CLIENT_ID" "$OAUTH_CLIENT_SECRET" "$E2E_TEST_USERNAME" "$E2E_TEST_PASSWORD" "$GMAPS_API_KEY" > .env - name: Create keystore.properties file env: ANDROID_KEY_STORE_PASSWORD: ${{ secrets.ANDROID_KEY_STORE_PASSWORD }} ANDROID_KEY_PASSWORD: ${{ secrets.ANDROID_KEY_PASSWORD }} ANDROID_ALIAS: ${{ secrets.ANDROID_ALIAS }} run: printf 'storePassword=%s\nkeyPassword=%s\nkeyAlias=%s\nstoreFile=release.keystore' "$ANDROID_KEY_STORE_PASSWORD" "$ANDROID_KEY_PASSWORD" "$ANDROID_ALIAS" > android/keystore.properties - name: Generate release keystore env: ANDROID_ALIAS: ${{ secrets.ANDROID_ALIAS }} ANDROID_KEY_STORE_PASSWORD: ${{ secrets.ANDROID_KEY_STORE_PASSWORD }} ANDROID_KEY_PASSWORD: ${{ secrets.ANDROID_KEY_PASSWORD }} run: | keytool -genkeypair -v -noprompt -storetype PKCS12 -keystore release.keystore -alias "$ANDROID_ALIAS" -keyalg RSA -keysize 2048 -validity 10000 -storepass "$ANDROID_KEY_STORE_PASSWORD" -keypass "$ANDROID_KEY_PASSWORD" -dname "CN=mqttserver.ibm.com, OU=ID, O=IBM, L=Hursley, S=Hants, C=GB" - name: Move keystore run: mv release.keystore android/app/release.keystore # Download the example model otherwise an error alert will be shown on app start, requires .env - name: Download the small example cv and geomodel run: npm run add-example-model -- -f=main # Macos-latest runner has 3 Java versions pre-installed, if not specified as here, the build step errors with requiring at least Java 11 or higher # So, this step is needed for the apk build step, but somehow this is breaking emulator setup, so it is placed here - name: Set up JDK 11 uses: actions/setup-java@v4 with: java-version: '11' distribution: 'zulu' # This is by far the longest step in this job, currently we are building the apk everytime, maybe there should be a more specific trigger for the entire job - name: Build for detox run: npm run e2e:build:android # Starts the avd previously set-up by name - name: Android Emulator timeout-minutes: 10 continue-on-error: true run: | echo "Starting emulator" nohup $ANDROID_HOME/emulator/emulator -avd Pixel_5_API_31_AOSP -no-audio -no-snapshot -no-window -no-boot-anim -timezone America/Los_Angeles & $ANDROID_HOME/platform-tools/adb wait-for-device shell 'while [[ -z $(getprop sys.boot_completed | tr -d '\r') ]]; do sleep 1; done; input keyevent 82' $ANDROID_HOME/platform-tools/adb devices $ANDROID_HOME/platform-tools/adb emu geo fix -121.45356 46.51119 4392 12 echo "Emulator started" - name: Ensure servers are running run: | # is rails running? curl -I --fail "https://staging.inaturalist.org/ping" # is node running & is ES working? curl -I --fail "https://stagingapi.inaturalist.org/v2/taxa" # Start the Android e2e tests with extensive logging and screen captures for failing tests - name: Android Detox run: npm run e2e:test:android -- --take-screenshots failing --record-videos failing -l debug # The artifacts for the failing tests are available for download on github.com on the page of the individual actions run - name: Store Detox artifacts on test failure uses: actions/upload-artifact@v4 if: failure() with: name: detox-artifacts path: artifacts notify: name: Notify Slack needs: test if: ${{ success() || failure() }} runs-on: macos-latest steps: - uses: iRoachie/slack-github-actions@v2.3.0 if: env.SLACK_WEBHOOK_URL != null env: SLACK_WEBHOOK_URL: ${{ secrets.SLACK_BUILDS_WEBHOOK_URL }} GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}