Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: add e2e tests with maestro #100

Open
wants to merge 1 commit into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
141 changes: 104 additions & 37 deletions .github/workflows/build.yml
Original file line number Diff line number Diff line change
Expand Up @@ -7,79 +7,146 @@ on:
branches: [main]

jobs:
android:
name: Build Android App
runs-on: ubuntu-latest
ios-build:
name: iOS Build
runs-on: macos-latest
defaults:
run:
working-directory: example

env:
SKIP_YARN_COREPACK_CHECK: 0

steps:
- uses: actions/checkout@v4

- name: Setup Java
uses: actions/setup-java@v4
with:
distribution: 'zulu'
java-version: '17'
- name: Checkout repository
uses: actions/checkout@v4

- name: Setup Node.js
uses: actions/setup-node@v4
with:
node-version: 'lts/*'
node-version: '22'
cache: 'yarn'
cache-dependency-path: example/yarn.lock

- name: Setup Ruby
uses: ruby/setup-ruby@v1
with:
ruby-version: '3.2'
bundler-cache: true

- name: Cache CocoaPods
uses: actions/cache@v4
with:
path: |
example/swift/Pods
key: ${{ runner.os }}-pods-${{ hashFiles('example/swift/Podfile.lock') }}
restore-keys: |
${{ runner.os }}-pods-

- name: Install root dependencies
run: yarn install

- name: Install example dependencies
run: yarn install

- name: Install pods
run: |
cd example
yarn install
cd ..
cd swift
pod install

- name: Build Android App
- name: Install Maestro CLI
run: |
cd example/kotlin
./gradlew assembleDebug
curl -Ls "https://get.maestro.mobile.dev" | bash
brew tap facebook/fb
brew install facebook/fb/idb-companion

ios:
name: Build iOS App
runs-on: macos-latest
- name: Add Maestro to path
run: echo "${HOME}/.maestro/bin" >> $GITHUB_PATH

- name: Start packager
run: yarn start &

- name: Build iOS App (Release)
run: |
cd swift
xcodebuild -workspace SwiftExample.xcworkspace -scheme SwiftExample -configuration Release -sdk iphonesimulator -destination 'platform=iOS Simulator,name=iPhone 15' build CODE_SIGNING_ALLOWED=NO

- name: Setup iOS simulator
run: |
UDID=$(xcrun simctl list devices | grep "iPhone" | grep "Booted" | head -1 | grep -E -o -i "([0-9a-f]{8}-([0-9a-f]{4}-){3}[0-9a-f]{12})")
if [ -z "$UDID" ]; then
UDID=$(xcrun simctl list devices available | grep "iPhone" | head -1 | grep -E -o -i "([0-9a-f]{8}-([0-9a-f]{4}-){3}[0-9a-f]{12})")
xcrun simctl boot "${UDID}"
fi
open -a Simulator
xcrun simctl launch "${UDID}" com.callstack.reactnativebrownfield.SwiftExample

- name: Run iOS tests
run: |
export MAESTRO_DRIVER_STARTUP_TIMEOUT=1500000
export MAESTRO_WAIT_TIMEOUT=10000
yarn test:e2e:ios

android-build:
name: Android Build
runs-on: ubuntu-latest
defaults:
run:
working-directory: example

env:
SKIP_YARN_COREPACK_CHECK: 0

steps:
- uses: actions/checkout@v4
- name: Checkout repository
uses: actions/checkout@v4

- name: Setup Node.js
uses: actions/setup-node@v4
with:
node-version: 'lts/*'
node-version: '22'
cache: 'yarn'
cache-dependency-path: example/yarn.lock

- name: Setup Ruby
uses: ruby/setup-ruby@v1
- name: Setup Java
uses: actions/setup-java@v4
with:
ruby-version: '3.2'
bundler-cache: true
distribution: 'zulu'
java-version: '17'

- name: Install root dependencies
run: yarn install

- name: Install example dependencies
run: |
cd example
yarn install
cd ..
run: yarn install

- name: Install pods
run: |
cd example/swift
pod install
- name: Start packager
run: yarn start &

- name: Build iOS App
- name: Install Maestro CLI
run: |
cd example/swift
xcodebuild -workspace SwiftExample.xcworkspace -scheme SwiftExample -configuration Debug -sdk iphonesimulator -destination 'platform=iOS Simulator,name=iPhone 15' build CODE_SIGNING_ALLOWED=NO
curl -Ls "https://get.maestro.mobile.dev" | bash

- name: Add Maestro to path
run: echo "${HOME}/.maestro/bin" >> $GITHUB_PATH

- name: Create AVD and generate snapshot for caching
uses: reactivecircus/android-emulator-runner@v2
with:
target: aosp_atd
api-level: 30
arch: x86
ram-size: 4096M
channel: canary
profile: pixel
avd-name: Pixel_3a_API_30_AOSP
force-avd-creation: false
emulator-options: -no-window -gpu swiftshader_indirect -noaudio -no-boot-anim -camera-back none
emulator-boot-timeout: 12000
disable-animations: false
working-directory: example/kotlin
script: |
chmod +x ./gradlew
./gradlew :app:installRelease
cd ..
yarn test:e2e:android
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,7 @@ fun HomeScreen(
verticalArrangement = Arrangement.Center
) {
Text(
"React Native Brownfield Example",
"React Native Brownfield",
fontSize = 30.sp,
modifier = Modifier.padding(15.dp),
textAlign = TextAlign.Center
Expand Down
37 changes: 37 additions & 0 deletions example/maestro-tests/android_test.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
appId: com.callstack.kotlinexample

---
- launchApp
- assertVisible: "React Native Brownfield"
- tapOn: "Navigate to React Native Fragment with Compose"

- assertVisible: "React Native Screen"

- tapOn: "Push next screen"

- assertVisible: "React Native Screen"

- tapOn: "Go back"

- assertVisible: "React Native Screen"

- tapOn: "Go back"

- assertVisible: "React Native Brownfield"

- tapOn: "Navigate to React Native Fragment Activity"

- assertVisible: "React Native Screen"

- tapOn: "Push next screen"

- assertVisible: "React Native Screen"

- tapOn: "Go back"

- assertVisible: "React Native Screen"

- tapOn: "Go back"

- assertVisible: "React Native Brownfield"

21 changes: 21 additions & 0 deletions example/maestro-tests/ios_test.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
appId: com.callstack.reactnativebrownfield.SwiftExample

---
- launchApp

- assertVisible: "React Native Brownfield"
- tapOn: "Push React Native screen"

- assertVisible: "React Native Screen"

- tapOn: "Push next screen"

- assertVisible: "React Native Screen"

- tapOn: "Go back"

- assertVisible: "React Native Screen"

- tapOn: "Go back"

- assertVisible: "React Native Brownfield"
4 changes: 3 additions & 1 deletion example/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,9 @@
"scripts": {
"android": "react-native run-android",
"ios": "react-native run-ios",
"start": "react-native start"
"start": "react-native start",
"test:e2e:ios": "maestro test ./maestro-tests/ios_test.yml",
"test:e2e:android": "maestro test ./maestro-tests/android_test.yml"
},
"dependencies": {
"react": "19.0.0",
Expand Down
2 changes: 1 addition & 1 deletion example/swift/App.swift
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ struct ContentView: View {
var body: some View {
NavigationView {
VStack {
Text("React Native Brownfield App")
Text("React Native Brownfield")
.font(.title)
.bold()
.padding()
Expand Down
Loading