Skip to content

Workflow file for this run

name: Native Integration Tests
env:
# Relative path from the monorepo root to the app
test_app_path: tests/test
# Package name in the monorepo
test_app_package_name: test-test
# Used for cache keys, must be unique among all workflows
app_identifier: ios-test-app
on:
push:
# pull_request:
jobs:
build-ios-test-dev-container:
name: Build iOS Test Development Container App
uses: ./.github/workflows/build-ios-test-app.yml
secrets: inherit
permissions:
contents: read
pull-requests: read
with:
configuration: Debug
build-ios-test-app:
name: Build iOS Release App
uses: ./.github/workflows/build-ios-test-app.yml
secrets: inherit
permissions:
contents: read
pull-requests: read
with:
configuration: Release
# test-ios-native:
# name: iOS Development Integration Test
# needs:
# - build-ios-test-dev-container
# runs-on: macos-14
# defaults:
# run:
# working-directory: ${{ env.test_app_path }}
# steps:
# - name: Checkout
# uses: actions/checkout@v4
# - name: Install
# uses: ./.github/actions/install
# with:
# workspace-focus: ${{ env.test_app_package_name }}
# - name: Start Dev Server
# run: node scripts/start-dev-server.mjs --no-wait
# - name: Download Dev Container App
# uses: actions/cache/restore@v4
# with:
# fail-on-cache-miss: true
# key: ${{ needs.build-ios-test-dev-container.outputs.built-app-cache-key }}
# path: ${{ needs.build-ios-test-dev-container.outputs.built-app-path }}
# - name: Get Simulator UDID
# id: get-simulator-udid
# run: |
# AVAILABLE_SIMULATORS=$(xcrun simctl list devices available --json)
# echo "Available simulators: $AVAILABLE_SIMULATORS"
# RUNTIME=$(echo $AVAILABLE_SIMULATORS | jq -r '.devices | keys | map(select(test("iOS"))) | last')
# echo "Runtime: $RUNTIME"
# SIMULATOR_INFO=$(echo $AVAILABLE_SIMULATORS | jq ".devices.\"$RUNTIME\" | map(select(.name | test(\"iPhone\"))) | last")
# SIMULATOR_UDID=$(echo $SIMULATOR_INFO | jq -r .udid)
# echo "Simulator info: $SIMULATOR_INFO"
# echo "Simulator UDID: $SIMULATOR_UDID"
# echo "simulator_udid=$SIMULATOR_UDID" >> $GITHUB_OUTPUT
# - name: Boot Simulator
# env:
# SIMULATOR_UDID: ${{ steps.get-simulator-udid.outputs.simulator_udid }}
# run: xcrun simctl boot $SIMULATOR_UDID
# - name: Install Maestro
# uses: ./.github/actions/install-maestro
# id: install-maestro
# - name: Wait for Dev Server to Be Ready
# run: node scripts/start-dev-server.mjs --check-only
# - name: Get a Copy of the Bundle
# run: |
# echo 'Fetching the React Native bundle in the background, since the first request can take a while pre-bundling dependencies and producing the first bundle without a cache. The bundle we got can also serve as troubleshooting material in case of a failure, and will be uploaded as an artifact.'
# wget -O bundle.ios.js 'http://localhost:8081/index.bundle?platform=ios' &
# - name: Install Dev Container App in Simulator
# env:
# SIMULATOR_UDID: ${{ steps.get-simulator-udid.outputs.simulator_udid }}
# APP_PATH: ${{ needs.build-ios-test-dev-container.outputs.built-app-path }}
# working-directory: .
# run: xcrun simctl install $SIMULATOR_UDID $APP_PATH
# - name: Test
# run: |
# export PATH="$PATH":"$HOME/.maestro/bin"
# export MAESTRO_DRIVER_STARTUP_TIMEOUT=180000 # 3 minutes
# maestro test maestro-flows/basic.yaml
# - name: Upload Dev Container App
# uses: actions/upload-artifact@v4.3.1
# # Only upload if the test failed to save space
# if: ${{ failure() }}
# continue-on-error: true
# with:
# name: dev-container-app
# path: |
# path: ${{ needs.build-ios-test-dev-container.outputs.built-app-path }}
# - name: Upload Server Log
# uses: actions/upload-artifact@v4.3.1
# if: ${{ always() }}
# continue-on-error: true
# with:
# name: dev-server.log
# path: |
# ${{ env.test_app_path }}/dev-server.log
# - name: Prepare Maestro Logs
# id: prepare-maestro-logs
# if: ${{ always() && steps.install-maestro.outcome == 'success' }}
# continue-on-error: true
# run: cp -r "$HOME/.maestro/tests" maestro-logs
# - name: Upload Maestro Logs
# uses: actions/upload-artifact@v4.3.1
# if: ${{ always() && steps.prepare-maestro-logs.outcome == 'success' }}
# continue-on-error: true
# with:
# name: release-maestro-logs
# path: |
# ${{ env.test_app_path }}/maestro-logs
# - name: Upload Maestro Screenshots
# uses: actions/upload-artifact@v4.3.1
# if: ${{ always() && steps.install-maestro.outcome == 'success' }}
# continue-on-error: true
# with:
# name: release-maestro-screenshots
# path: |
# ${{ env.test_app_path }}/maestro-screenshots
# - name: Upload Reference React Native Bundle
# uses: actions/upload-artifact@v4.3.1
# if: ${{ always() }}
# continue-on-error: true
# with:
# name: dev-sample-bundle.ios.js
# path: |
# ${{ env.test_app_path }}/bundle.ios.js
test-ios-native-release:
name: iOS Release Integration Test
# Note: Some parts of the native build process are not covered by this test:
# * We do not test the generated Xcode project has the Build Phase "Bundle React Native code and images" setup correctly using the standard `react-native bundle` command. This is done by the "vxrn/expo-plugin" Expo plugin which users should include in their `app.json` configuration. If that didn't work properally, Expo prebuild may setup the Xcode project to use Expo's custom bundle command that uses Metro.
# * We do not test `react-native/scripts/react-native-xcode.sh`, instead we copy parts of it into our test scripts, simulating parts of the build process without running the whole native build.
needs:
- build-ios-test-app
runs-on: macos-14
defaults:
run:
working-directory: ${{ env.test_app_path }}
steps:
- name: Checkout
uses: actions/checkout@v4
- name: Install
uses: ./.github/actions/install
with:
workspace-focus: ${{ env.test_app_package_name }}
- name: Download Built Native App
uses: actions/cache/restore@v4
with:
fail-on-cache-miss: true
key: ${{ needs.build-ios-test-app.outputs.built-app-cache-key }}
path: ${{ needs.build-ios-test-app.outputs.built-app-path }}
- name: Get Simulator UDID
id: get-simulator-udid
run: |
AVAILABLE_SIMULATORS=$(xcrun simctl list devices available --json)
echo "Available simulators: $AVAILABLE_SIMULATORS"
RUNTIME=$(echo $AVAILABLE_SIMULATORS | jq -r '.devices | keys | map(select(test("iOS"))) | last')
echo "Runtime: $RUNTIME"
SIMULATOR_INFO=$(echo $AVAILABLE_SIMULATORS | jq ".devices.\"$RUNTIME\" | map(select(.name | test(\"iPhone\"))) | last")
SIMULATOR_UDID=$(echo $SIMULATOR_INFO | jq -r .udid)
echo "Simulator info: $SIMULATOR_INFO"
echo "Simulator UDID: $SIMULATOR_UDID"
echo "simulator_udid=$SIMULATOR_UDID" >> $GITHUB_OUTPUT
- name: Boot Simulator
env:
SIMULATOR_UDID: ${{ steps.get-simulator-udid.outputs.simulator_udid }}
run: xcrun simctl boot $SIMULATOR_UDID
- name: Install Maestro
id: install-maestro
uses: ./.github/actions/install-maestro
- name: Patch Packages
run: |
yarn one patch
# Prebuild and pod install because we need hermesc which is installed by pod install
- name: Prebuild
run: |
yarn expo prebuild --platform ios --no-install # --no-install is used to skip installing dependencies, specifically `pod install` as we want to do it after the Cache Pods step
- name: Cache Pods
uses: actions/cache@v4
env:
cache-name: ${{ env.app_identifier }}-pods
with:
path: ${{ env.test_app_path }}/ios/Pods
key: ${{ runner.os }}-${{ env.cache-name }}-${{ hashFiles(format('{0}/ios/Podfile.lock', env.test_app_path)) }}
restore-keys: |
${{ runner.os }}-${{ env.cache-name }}-
- name: Pod Install
run: |
cd ios && pod install
- name: Bundle React Native code and images
run: |
mkdir test-bundle-output
yarn react-native bundle --platform ios --dev false --bundle-output test-bundle-output/main.tmp.jsbundle --assets-dest test-bundle-output
ios/Pods/hermes-engine/destroot/bin/hermesc -emit-binary -max-diagnostic-width=80 -O -output-source-map -out test-bundle-output/main.jsbundle test-bundle-output/main.tmp.jsbundle
- name: Replace JS Bundle and Assets in App
run: |
cp -r test-bundle-output/. ${{ github.workspace }}/${{ needs.build-ios-test-app.outputs.built-app-path }}/
- name: Check .app
id: check-app
run: |
if [ ! -e ${{ github.workspace }}/${{ needs.build-ios-test-app.outputs.built-app-path }} ]; then
echo "Error: .app not found"
exit 1
fi
- name: Install App in Simulator
env:
SIMULATOR_UDID: ${{ steps.get-simulator-udid.outputs.simulator_udid }}
APP_PATH: ${{ needs.build-ios-test-app.outputs.built-app-path }}
working-directory: .
run: xcrun simctl install $SIMULATOR_UDID $APP_PATH
- name: Test
run: |
export PATH="$PATH":"$HOME/.maestro/bin"
export MAESTRO_DRIVER_STARTUP_TIMEOUT=180000 # 3 minutes
maestro test maestro-flows/basic.yaml
- name: Upload App
uses: actions/upload-artifact@v4.3.1
if: ${{ always() && steps.check-app.outcome == 'success' }}
continue-on-error: true
with:
name: release-app
path: |
${{ needs.build-ios-test-app.outputs.built-app-path }}
- name: Prepare Maestro Logs
id: prepare-maestro-logs
if: ${{ always() && steps.install-maestro.outcome == 'success' }}
continue-on-error: true
run: cp -r "$HOME/.maestro/tests" maestro-logs
- name: Upload Maestro Logs
uses: actions/upload-artifact@v4.3.1
if: ${{ always() && steps.prepare-maestro-logs.outcome == 'success' }}
continue-on-error: true
with:
name: release-maestro-logs
path: |
${{ env.test_app_path }}/maestro-logs
- name: Upload Maestro Screenshots
uses: actions/upload-artifact@v4.3.1
if: ${{ always() && steps.install-maestro.outcome == 'success' }}
continue-on-error: true
with:
name: release-maestro-screenshots
path: |
${{ env.test_app_path }}/maestro-screenshots