From 3bdd76b7ffa75c02f58204ea39c4071972812e93 Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Fri, 15 Nov 2024 10:36:01 -0800 Subject: [PATCH 01/15] replace App Center with BrowserStack --- ...appcenter.yml => android-browserstack.yml} | 68 +++++---- .github/workflows/android-perf.yml | 37 +++-- ...ios-appcenter.yml => ios-browserstack.yml} | 49 ++++--- .github/workflows/ios-perf.yml | 46 +++--- script/automation/browserstack.py | 131 ++++++++++++++++++ 5 files changed, 240 insertions(+), 91 deletions(-) rename .github/workflows/{android-appcenter.yml => android-browserstack.yml} (63%) rename .github/workflows/{ios-appcenter.yml => ios-browserstack.yml} (51%) create mode 100644 script/automation/browserstack.py diff --git a/.github/workflows/android-appcenter.yml b/.github/workflows/android-browserstack.yml similarity index 63% rename from .github/workflows/android-appcenter.yml rename to .github/workflows/android-browserstack.yml index 290a7586..98da2cec 100644 --- a/.github/workflows/android-appcenter.yml +++ b/.github/workflows/android-browserstack.yml @@ -1,18 +1,18 @@ -name: Android AppCenter Tests +name: Android BrowserStack Tests on: workflow_dispatch: push: branches: [ master ] paths: - - '.github/workflows/android-appcenter.yml' + - '.github/workflows/android-browserstack.yml' - 'binding/android/RhinoTestApp/**' - 'resources/.test/**' - 'resources/audio_samples/**' pull_request: branches: [ master, 'v[0-9]+.[0-9]+' ] paths: - - '.github/workflows/android-appcenter.yml' + - '.github/workflows/android-browserstack.yml' - 'binding/android/RhinoTestApp/**' - 'resources/.test/**' - 'resources/audio_samples/**' @@ -23,19 +23,18 @@ defaults: jobs: build: - name: Run Android Tests on AppCenter + name: Run Android Tests on BrowserStack runs-on: ubuntu-latest steps: - uses: actions/checkout@v3 - - name: Set up Node.js LTS - uses: actions/setup-node@v3 + - name: Installing Python + uses: actions/setup-python@v5 with: - node-version: lts/* - - - name: Install AppCenter CLI - run: npm install -g appcenter-cli + python-version: '3.10' + - run: + pip3 install requests - name: set up JDK 11 uses: actions/setup-java@v3 @@ -68,30 +67,29 @@ jobs: - name: Build androidTest run: ./gradlew assembleEnDebugAndroidTest - - name: Run tests on AppCenter - run: appcenter test run espresso - --token ${{secrets.APPCENTERAPITOKEN}} - --app "Picovoice/Rhino-Android-Activity" - --devices "Picovoice/android-min-max" - --app-path rhino-test-app/build/outputs/apk/en/debug/rhino-test-app-en-debug.apk - --test-series "rhino-android" - --locale "en_US" - --build-dir rhino-test-app/build/outputs/apk/androidTest/en/debug + - name: Run tests on BrowserStack + run: python3 ../../../script/automation/browserstack.py + --type espresso + --username "${{secrets.BROWSERSTACK_USERNAME}}" + --access_key "${{secrets.BROWSERSTACK_ACCESS_KEY}}" + --project_name "Rhino-Android" + --devices "android-min-max" + --app_path "rhino-test-app/build/outputs/apk/debug/rhino-test-app-debug.apk" + --test_path "rhino-test-app/build/outputs/apk/androidTest/debug/rhino-test-app-debug-androidTest.apk" build-integ: - name: Run Android Integration Tests on AppCenter + name: Run Android Integration Tests on BrowserStack runs-on: ubuntu-latest steps: - uses: actions/checkout@v3 - - name: Set up Node.js LTS - uses: actions/setup-node@v3 + - name: Installing Python + uses: actions/setup-python@v5 with: - node-version: lts/* - - - name: Install AppCenter CLI - run: npm install -g appcenter-cli + python-version: '3.10' + - run: + pip3 install requests - name: set up JDK 11 uses: actions/setup-java@v3 @@ -124,12 +122,12 @@ jobs: - name: Build androidTest run: ./gradlew assembleEnReleaseAndroidTest -DtestBuildType=integ - - name: Run tests on AppCenter - run: appcenter test run espresso - --token ${{secrets.APPCENTERAPITOKEN}} - --app "Picovoice/Rhino-Android-Activity" - --devices "Picovoice/android-min-max" - --app-path rhino-test-app/build/outputs/apk/en/release/rhino-test-app-en-release.apk - --test-series "rhino-android" - --locale "en_US" - --build-dir rhino-test-app/build/outputs/apk/androidTest/en/release + - name: Run tests on BrowserStack + run: python3 ../../../script/automation/browserstack.py + --type espresso + --username "${{secrets.BROWSERSTACK_USERNAME}}" + --access_key "${{secrets.BROWSERSTACK_ACCESS_KEY}}" + --project_name "Rhino-Android-Integration" + --devices "android-min-max" + --app_path "rhino-test-app/build/outputs/apk/release/rhino-test-app-release.apk" + --test_path "rhino-test-app/build/outputs/apk/androidTest/release/rhino-test-app-release-androidTest.apk" diff --git a/.github/workflows/android-perf.yml b/.github/workflows/android-perf.yml index 7300e2a8..0a704f26 100644 --- a/.github/workflows/android-perf.yml +++ b/.github/workflows/android-perf.yml @@ -19,28 +19,25 @@ defaults: jobs: build: - name: Run Android Speed Tests on AppCenter + name: Run Android Speed Tests on BrowserStack runs-on: ubuntu-latest strategy: matrix: - device: [single-android, 32bit-android] + device: [ android-perf ] include: - - device: single-android + - device: android-perf performanceThresholdSec: 0.2 - - device: 32bit-android - performanceThresholdSec: 1.0 steps: - uses: actions/checkout@v3 - - name: Set up Node.js LTS - uses: actions/setup-node@v3 + - name: Installing Python + uses: actions/setup-python@v5 with: - node-version: lts/* - - - name: Install AppCenter CLI - run: npm install -g appcenter-cli + python-version: '3.10' + - run: + pip3 install requests - name: set up JDK 11 uses: actions/setup-java@v3 @@ -79,12 +76,12 @@ jobs: - name: Build androidTest run: ./gradlew assembleEnDebugAndroidTest -DtestBuildType=perf - - name: Run tests on AppCenter - run: appcenter test run espresso - --token ${{secrets.APPCENTERAPITOKEN}} - --app "Picovoice/Rhino-Android-Activity" - --devices "Picovoice/${{ matrix.device }}" - --app-path rhino-test-app/build/outputs/apk/en/debug/rhino-test-app-en-debug.apk - --test-series "rhino-android" - --locale "en_US" - --build-dir rhino-test-app/build/outputs/apk/androidTest/en/debug + - name: Run tests on BrowserStack + run: python3 ../../../script/automation/browserstack.py + --type espresso + --username "${{secrets.BROWSERSTACK_USERNAME}}" + --access_key "${{secrets.BROWSERSTACK_ACCESS_KEY}}" + --project_name "Rhino-Android-Performance" + --devices "${{ matrix.device }}" + --app_path "rhino-test-app/build/outputs/apk/debug/rhino-test-app-debug.apk" + --test_path "rhino-test-app/build/outputs/apk/androidTest/debug/rhino-test-app-debug-androidTest.apk" diff --git a/.github/workflows/ios-appcenter.yml b/.github/workflows/ios-browserstack.yml similarity index 51% rename from .github/workflows/ios-appcenter.yml rename to .github/workflows/ios-browserstack.yml index 97cc38a0..50f16486 100644 --- a/.github/workflows/ios-appcenter.yml +++ b/.github/workflows/ios-browserstack.yml @@ -1,18 +1,18 @@ -name: iOS AppCenter Tests +name: iOS BrowserStack Tests on: workflow_dispatch: push: branches: [ master ] paths: - - '.github/workflows/ios-appcenter.yml' + - '.github/workflows/ios-browserstack.yml' - 'binding/ios/RhinoAppTest/**' - 'resources/.test/**' - 'resources/audio_samples/**' pull_request: branches: [ master, 'v[0-9]+.[0-9]+' ] paths: - - '.github/workflows/ios-appcenter.yml' + - '.github/workflows/ios-browserstack.yml' - 'binding/ios/RhinoAppTest/**' - 'resources/.test/**' - 'resources/audio_samples/**' @@ -23,24 +23,23 @@ defaults: jobs: build: - name: Run iOS Tests on AppCenter + name: Run iOS Tests on BrowserStack runs-on: macos-latest steps: - name: Checkout uses: actions/checkout@v3 - - name: Set up Node.js LTS - uses: actions/setup-node@v3 + - name: Installing Python + uses: actions/setup-python@v5 with: - node-version: lts/* + python-version: '3.10' + - run: + pip3 install requests - name: Install Cocoapods run: gem install cocoapods - - name: Install AppCenter CLI - run: npm install -g appcenter-cli - - name: Make build dir run: mkdir ddp @@ -55,7 +54,7 @@ jobs: - name: Run Cocoapods run: pod install - - name: Inject AppID + - name: Inject AccessKey run: sed -i '.bak' 's:{TESTING_ACCESS_KEY_HERE}:${{secrets.PV_VALID_ACCESS_KEY}}:' RhinoAppTestUITests/BaseTest.swift @@ -68,11 +67,23 @@ jobs: -derivedDataPath ddp CODE_SIGNING_ALLOWED=NO - - name: Run Tests on AppCenter - run: appcenter test run xcuitest - --token ${{secrets.APPCENTERAPITOKEN}} - --app "Picovoice/Rhino-iOS" - --devices "Picovoice/ios-min-max" - --test-series "rhino-ios" - --locale "en_US" - --build-dir ddp/Build/Products/Debug-iphoneos + - name: Generating ipa + run: cd ddp/Build/Products/Debug-iphoneos/ && + mkdir Payload && + cp -r RhinoAppTest.app Payload && + zip --symlinks -r RhinoAppTest.ipa Payload && + rm -r Payload + + - name: Zipping Tests + run: cd ddp/Build/Products/Debug-iphoneos/ && + zip --symlinks -r RhinoAppTestUITests.zip RhinoAppTestUITests-Runner.app + + - name: Run tests on BrowserStack + run: python3 ../../../script/automation/browserstack.py + --type xcuitest + --username "${{secrets.BROWSERSTACK_USERNAME}}" + --access_key "${{secrets.BROWSERSTACK_ACCESS_KEY}}" + --project_name "Rhino-iOS" + --devices "ios-min-max" + --app_path "ddp/Build/Products/Debug-iphoneos/RhinoAppTest.ipa" + --test_path "ddp/Build/Products/Debug-iphoneos/RhinoAppTestUITests.zip" diff --git a/.github/workflows/ios-perf.yml b/.github/workflows/ios-perf.yml index e9d12ab9..805f9ce3 100644 --- a/.github/workflows/ios-perf.yml +++ b/.github/workflows/ios-perf.yml @@ -23,12 +23,12 @@ defaults: jobs: build: - name: Run iOS Tests on AppCenter + name: Run iOS Tests on BrowserStack runs-on: macos-latest strategy: matrix: - device: [ios-perf] + device: [ ios-perf ] include: - device: ios-perf performanceThresholdSec: 0.2 @@ -37,17 +37,16 @@ jobs: - name: Checkout uses: actions/checkout@v3 - - name: Set up Node.js LTS - uses: actions/setup-node@v3 + - name: Installing Python + uses: actions/setup-python@v5 with: - node-version: lts/* + python-version: '3.10' + - run: + pip3 install requests - name: Install Cocoapods run: gem install cocoapods - - name: Install AppCenter CLI - run: npm install -g appcenter-cli - - name: Make build dir run: mkdir ddp @@ -71,7 +70,8 @@ jobs: PerformanceTest/PerformanceTest.swift - name: Inject Performance Threshold - run: sed -i '.bak' 's:{PERFORMANCE_THRESHOLD_SEC}:${{ matrix.performanceThresholdSec }}:' + run: sed -i '.bak' + '1,/{PERFORMANCE_THRESHOLD_SEC}/s/{PERFORMANCE_THRESHOLD_SEC}/${{ matrix.performanceThresholdSec }}/' PerformanceTest/PerformanceTest.swift - name: XCode Build @@ -83,11 +83,23 @@ jobs: -derivedDataPath ddp CODE_SIGNING_ALLOWED=NO - - name: Run Tests on AppCenter - run: appcenter test run xcuitest - --token ${{secrets.APPCENTERAPITOKEN}} - --app "Picovoice/Rhino-iOS" - --devices "Picovoice/${{ matrix.device }}" - --test-series "rhino-ios" - --locale "en_US" - --build-dir ddp/Build/Products/Debug-iphoneos + - name: Generating ipa + run: cd ddp/Build/Products/Debug-iphoneos/ && + mkdir Payload && + cp -r RhinoAppTest.app Payload && + zip --symlinks -r RhinoAppTest.ipa Payload && + rm -r Payload + + - name: Zipping Tests + run: cd ddp/Build/Products/Debug-iphoneos/ && + zip --symlinks -r PerformanceTest.zip PerformanceTest-Runner.app + + - name: Run tests on BrowserStack + run: python3 ../../../script/automation/browserstack.py + --type xcuitest + --username "${{secrets.BROWSERSTACK_USERNAME}}" + --access_key "${{secrets.BROWSERSTACK_ACCESS_KEY}}" + --project_name "Rhino-iOS-Performance" + --devices "${{ matrix.device }}" + --app_path "ddp/Build/Products/Debug-iphoneos/RhinoAppTest.ipa" + --test_path "ddp/Build/Products/Debug-iphoneos/PerformanceTest.zip" diff --git a/script/automation/browserstack.py b/script/automation/browserstack.py new file mode 100644 index 00000000..3cc5e98e --- /dev/null +++ b/script/automation/browserstack.py @@ -0,0 +1,131 @@ +import argparse +import requests +import time + +APP_URI = 'https://api-cloud.browserstack.com/app-automate/{}/v2/app' +TEST_URI = 'https://api-cloud.browserstack.com/app-automate/{}/v2/test-suite' +BUILD_URI = 'https://api-cloud.browserstack.com/app-automate/{}/v2/build' +STATUS_URI = 'https://api-cloud.browserstack.com/app-automate/{}/v2/builds/{}' + +devices_dict = { + 'android-min-max': [ + 'Samsung Galaxy S8-7.0', + 'Samsung Galaxy M52-11.0', + 'Google Pixel 9-15.0' + ], + 'android-perf': [ + 'Google Pixel 6 Pro-15.0' + ], + 'ios-min-max': [ + 'iPhone SE 2020-13', + 'iPhone 14 Pro-16', + 'iPhone 14-18' + ], + 'ios-perf': [ + 'iPhone 13-18', + ] +} + +def main(args: argparse.Namespace) -> None: + app_files = { + 'file': open(args.app_path, 'rb') + } + + app_response = requests.post( + APP_URI.format(args.type), + files=app_files, + auth=(args.username, args.access_key) + ) + app_response_json = app_response.json() + + if not app_response.ok: + print('App Upload Failed', app_response_json) + exit(1) + + test_files = { + 'file': open(args.test_path, 'rb') + } + test_response = requests.post( + TEST_URI.format(args.type), + files=test_files, + auth=(args.username, args.access_key) + ) + test_response_json = test_response.json() + + if not test_response.ok: + print('Test Upload Failed', test_response_json) + exit(1) + + build_headers = { + 'Content-Type': 'application/json' + } + build_data = { + 'app': app_response_json['app_url'], + 'testSuite': test_response_json['test_suite_url'], + 'project': args.project_name, + 'devices': devices_dict[args.devices] + } + + while True: + build_response = requests.post( + BUILD_URI.format(args.type), + headers=build_headers, + json=build_data, + auth=(args.username, args.access_key) + ) + if (build_response is not None and 'message' in build_response.json() and '[BROWSERSTACK_ALL_PARALLELS_IN_USE]' + in build_response.json()['message']): + print('Parallel threads limit reached. Waiting...', flush=True) + time.sleep(60) + else: + break + + if build_response is None: + print('Build Failed') + exit(1) + + build_response_json = build_response.json() + + if not build_response.ok: + print('Build Failed', build_response.json()) + exit(1) + + if build_response_json['message'] != 'Success': + print('Build Unsuccessful') + exit(1) + + print('View build results at https://app-automate.browserstack.com/dashboard/v2/builds/{}'.format(build_response_json['build_id'])) + + while True: + time.sleep(60) + status_response = requests.get( + STATUS_URI.format(args.type, build_response_json['build_id']), + auth=(args.username, args.access_key) + ) + status_response_json = status_response.json() + status = status_response_json['status'] + + if not status_response.ok: + print('Status Request Failed', status_response_json) + exit(1) + + if status != 'queued' and status != 'running': + break + + print('Status:', status) + if status != 'passed': + exit(1) + +if __name__ == '__main__': + parser = argparse.ArgumentParser() + parser.add_argument('--type', choices=['espresso', 'xcuitest'], required=True) + parser.add_argument('--username', required=True) + parser.add_argument('--access_key', required=True) + + parser.add_argument('--project_name', required=True) + parser.add_argument('--devices', choices=devices_dict.keys(), required=True) + parser.add_argument('--app_path', required=True) + parser.add_argument('--test_path', required=True) + args = parser.parse_args() + + main(args) \ No newline at end of file From 5a0162379a86e3b69aa3f37b7e975763bdb254db Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Fri, 15 Nov 2024 10:55:26 -0800 Subject: [PATCH 02/15] fix android paths --- .github/workflows/android-browserstack.yml | 8 ++++---- .github/workflows/android-perf.yml | 4 ++-- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/.github/workflows/android-browserstack.yml b/.github/workflows/android-browserstack.yml index 98da2cec..69c8c7e5 100644 --- a/.github/workflows/android-browserstack.yml +++ b/.github/workflows/android-browserstack.yml @@ -74,8 +74,8 @@ jobs: --access_key "${{secrets.BROWSERSTACK_ACCESS_KEY}}" --project_name "Rhino-Android" --devices "android-min-max" - --app_path "rhino-test-app/build/outputs/apk/debug/rhino-test-app-debug.apk" - --test_path "rhino-test-app/build/outputs/apk/androidTest/debug/rhino-test-app-debug-androidTest.apk" + --app_path "rhino-test-app/build/outputs/apk/en/debug/rhino-test-app-en-debug.apk" + --test_path "rhino-test-app/build/outputs/apk/androidTest/en/debug/rhino-test-app-en-debug-androidTest.apk" build-integ: name: Run Android Integration Tests on BrowserStack @@ -129,5 +129,5 @@ jobs: --access_key "${{secrets.BROWSERSTACK_ACCESS_KEY}}" --project_name "Rhino-Android-Integration" --devices "android-min-max" - --app_path "rhino-test-app/build/outputs/apk/release/rhino-test-app-release.apk" - --test_path "rhino-test-app/build/outputs/apk/androidTest/release/rhino-test-app-release-androidTest.apk" + --app_path "rhino-test-app/build/outputs/apk/en/release/rhino-test-app-en-release.apk" + --test_path "rhino-test-app/build/outputs/apk/androidTest/en/release/rhino-test-app-en-release-androidTest.apk" diff --git a/.github/workflows/android-perf.yml b/.github/workflows/android-perf.yml index 0a704f26..2e1f729d 100644 --- a/.github/workflows/android-perf.yml +++ b/.github/workflows/android-perf.yml @@ -83,5 +83,5 @@ jobs: --access_key "${{secrets.BROWSERSTACK_ACCESS_KEY}}" --project_name "Rhino-Android-Performance" --devices "${{ matrix.device }}" - --app_path "rhino-test-app/build/outputs/apk/debug/rhino-test-app-debug.apk" - --test_path "rhino-test-app/build/outputs/apk/androidTest/debug/rhino-test-app-debug-androidTest.apk" + --app_path "rhino-test-app/build/outputs/apk/en/debug/rhino-test-app-en-debug.apk" + --test_path "rhino-test-app/build/outputs/apk/androidTest/en/debug/rhino-test-app-en-debug-androidTest.apk" From d0398081ad1a2e9a2fa21c3a60354ca227a7aa7f Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Fri, 15 Nov 2024 11:00:39 -0800 Subject: [PATCH 03/15] spelling --- resources/.lint/spell-check/dict.txt | 1 + 1 file changed, 1 insertion(+) diff --git a/resources/.lint/spell-check/dict.txt b/resources/.lint/spell-check/dict.txt index 5462be85..b6138bb5 100644 --- a/resources/.lint/spell-check/dict.txt +++ b/resources/.lint/spell-check/dict.txt @@ -137,4 +137,5 @@ xcodeproj xcscheme xcschemes xcshareddata +xcuitest xcworkspace From f10ae826c5139a6fdb8073311bb84412cf0e726d Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Tue, 19 Nov 2024 13:00:01 -0800 Subject: [PATCH 04/15] rm cocoapods from ios actions --- .github/workflows/ios-browserstack.yml | 6 ------ .github/workflows/ios-demos.yml | 6 ------ .github/workflows/ios-perf.yml | 6 ------ 3 files changed, 18 deletions(-) diff --git a/.github/workflows/ios-browserstack.yml b/.github/workflows/ios-browserstack.yml index 50f16486..5589b28d 100644 --- a/.github/workflows/ios-browserstack.yml +++ b/.github/workflows/ios-browserstack.yml @@ -37,9 +37,6 @@ jobs: - run: pip3 install requests - - name: Install Cocoapods - run: gem install cocoapods - - name: Make build dir run: mkdir ddp @@ -51,9 +48,6 @@ jobs: - name: Copy test_resources run: ./copy_test_resources.sh - - name: Run Cocoapods - run: pod install - - name: Inject AccessKey run: sed -i '.bak' 's:{TESTING_ACCESS_KEY_HERE}:${{secrets.PV_VALID_ACCESS_KEY}}:' RhinoAppTestUITests/BaseTest.swift diff --git a/.github/workflows/ios-demos.yml b/.github/workflows/ios-demos.yml index bcf84ff4..fe35bb51 100644 --- a/.github/workflows/ios-demos.yml +++ b/.github/workflows/ios-demos.yml @@ -29,18 +29,12 @@ jobs: with: node-version: lts/* - - name: Install Cocoapods - run: gem install cocoapods - - name: Install AppCenter CLI run: npm install -g appcenter-cli - name: Make build dir run: mkdir ddp - - name: Run Cocoapods - run: pod install - - name: Build English run: xcrun xcodebuild build -configuration Debug diff --git a/.github/workflows/ios-perf.yml b/.github/workflows/ios-perf.yml index 805f9ce3..8705b803 100644 --- a/.github/workflows/ios-perf.yml +++ b/.github/workflows/ios-perf.yml @@ -44,9 +44,6 @@ jobs: - run: pip3 install requests - - name: Install Cocoapods - run: gem install cocoapods - - name: Make build dir run: mkdir ddp @@ -58,9 +55,6 @@ jobs: - name: Copy test_resources run: ./copy_test_resources.sh - - name: Run Cocoapods - run: pod install - - name: Inject AppID run: sed -i '.bak' 's:{TESTING_ACCESS_KEY_HERE}:${{secrets.PV_VALID_ACCESS_KEY}}:' PerformanceTest/PerformanceTest.swift From ed0900f195064b9d96fe2405e254d39274f09eab Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Tue, 19 Nov 2024 14:24:32 -0800 Subject: [PATCH 05/15] revert cocoapods, cleanup android --- .github/workflows/ios-browserstack.yml | 11 ++++++----- .github/workflows/ios-demos.yml | 7 +++++-- .github/workflows/ios-perf.yml | 13 +++++++------ .../RhinoTestApp/rhino-test-app/build.gradle | 1 - .../java/ai/picovoice/rhino/testapp/BaseTest.java | 12 ------------ .../ai/picovoice/rhino/testapp/IntegrationTest.java | 11 ----------- binding/ios/RhinoAppTest/copy_test_resources.sh | 3 --- resources/.lint/spell-check/dict.txt | 2 -- 8 files changed, 18 insertions(+), 42 deletions(-) diff --git a/.github/workflows/ios-browserstack.yml b/.github/workflows/ios-browserstack.yml index 5589b28d..69c5ec90 100644 --- a/.github/workflows/ios-browserstack.yml +++ b/.github/workflows/ios-browserstack.yml @@ -37,17 +37,18 @@ jobs: - run: pip3 install requests + - name: Install Cocoapods + run: gem install cocoapods + - name: Make build dir run: mkdir ddp - - name: Install resource script dependency - run: | - brew update - brew install convmv - - name: Copy test_resources run: ./copy_test_resources.sh + - name: Run Cocoapods + run: pod install + - name: Inject AccessKey run: sed -i '.bak' 's:{TESTING_ACCESS_KEY_HERE}:${{secrets.PV_VALID_ACCESS_KEY}}:' RhinoAppTestUITests/BaseTest.swift diff --git a/.github/workflows/ios-demos.yml b/.github/workflows/ios-demos.yml index fe35bb51..7156b1e9 100644 --- a/.github/workflows/ios-demos.yml +++ b/.github/workflows/ios-demos.yml @@ -29,12 +29,15 @@ jobs: with: node-version: lts/* - - name: Install AppCenter CLI - run: npm install -g appcenter-cli + - name: Install Cocoapods + run: gem install cocoapods - name: Make build dir run: mkdir ddp + - name: Run Cocoapods + run: pod install + - name: Build English run: xcrun xcodebuild build -configuration Debug diff --git a/.github/workflows/ios-perf.yml b/.github/workflows/ios-perf.yml index 8705b803..ae43049b 100644 --- a/.github/workflows/ios-perf.yml +++ b/.github/workflows/ios-perf.yml @@ -44,18 +44,19 @@ jobs: - run: pip3 install requests + - name: Install Cocoapods + run: gem install cocoapods + - name: Make build dir run: mkdir ddp - - name: Install resource script dependency - run: | - brew update - brew install convmv - - name: Copy test_resources run: ./copy_test_resources.sh - - name: Inject AppID + - name: Run Cocoapods + run: pod install + + - name: Inject AccessKey run: sed -i '.bak' 's:{TESTING_ACCESS_KEY_HERE}:${{secrets.PV_VALID_ACCESS_KEY}}:' PerformanceTest/PerformanceTest.swift diff --git a/binding/android/RhinoTestApp/rhino-test-app/build.gradle b/binding/android/RhinoTestApp/rhino-test-app/build.gradle index c1b60b68..12b68bd3 100644 --- a/binding/android/RhinoTestApp/rhino-test-app/build.gradle +++ b/binding/android/RhinoTestApp/rhino-test-app/build.gradle @@ -153,7 +153,6 @@ dependencies { exclude group: 'com.android.support', module: 'support-annotations' }) - androidTestImplementation('com.microsoft.appcenter:espresso-test-extension:1.4') androidTestImplementation('androidx.test.espresso:espresso-intents:3.5.1') } diff --git a/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/BaseTest.java b/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/BaseTest.java index 20335d1c..5f2c1027 100644 --- a/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/BaseTest.java +++ b/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/BaseTest.java @@ -8,12 +8,7 @@ import androidx.test.platform.app.InstrumentationRegistry; -import com.microsoft.appcenter.espresso.Factory; -import com.microsoft.appcenter.espresso.ReportHelper; - -import org.junit.After; import org.junit.Before; -import org.junit.Rule; import java.io.BufferedInputStream; import java.io.BufferedOutputStream; @@ -32,19 +27,12 @@ public class BaseTest { - @Rule - public ReportHelper reportHelper = Factory.getReportHelper(); Context testContext; Context appContext; AssetManager assetManager; String testResourcesPath; String accessKey; - @After - public void TearDown() { - reportHelper.label("Stopping App"); - } - @Before public void Setup() throws IOException { testContext = InstrumentationRegistry.getInstrumentation().getContext(); diff --git a/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/IntegrationTest.java b/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/IntegrationTest.java index 30a65305..dcca9e41 100644 --- a/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/IntegrationTest.java +++ b/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/IntegrationTest.java @@ -16,9 +16,6 @@ import androidx.test.ext.junit.rules.ActivityScenarioRule; import androidx.test.ext.junit.runners.AndroidJUnit4; -import com.microsoft.appcenter.espresso.Factory; -import com.microsoft.appcenter.espresso.ReportHelper; - import org.hamcrest.Matcher; import org.junit.After; import org.junit.Before; @@ -74,9 +71,6 @@ public void perform(UiController uiController, View view) { @RunWith(AndroidJUnit4.class) public class IntegrationTest { - @Rule - public ReportHelper reportHelper = Factory.getReportHelper(); - @Rule public ActivityScenarioRule activityScenarioRule = new ActivityScenarioRule<>(MainActivity.class); @@ -91,11 +85,6 @@ public void intentsTeardown() { Intents.release(); } - @After - public void TearDown() { - reportHelper.label("Stopping App"); - } - @Test public void testRhino() { onView(withId(R.id.testButton)).perform(click()); diff --git a/binding/ios/RhinoAppTest/copy_test_resources.sh b/binding/ios/RhinoAppTest/copy_test_resources.sh index 75c5415d..c049f687 100755 --- a/binding/ios/RhinoAppTest/copy_test_resources.sh +++ b/binding/ios/RhinoAppTest/copy_test_resources.sh @@ -29,6 +29,3 @@ cp ${LIB_DIR}/common/*.pv ${ASSETS_DIR}/model_files echo "Copying test data file..." cp ${RESOURCE_DIR}/.test/test_data.json ${ASSETS_DIR} - -echo "Fixing filename encodings for Appcenter compatibility" -convmv --notest -f utf8 -t utf8 --nfd -r ${ASSETS_DIR} diff --git a/resources/.lint/spell-check/dict.txt b/resources/.lint/spell-check/dict.txt index b6138bb5..21848e70 100644 --- a/resources/.lint/spell-check/dict.txt +++ b/resources/.lint/spell-check/dict.txt @@ -1,7 +1,6 @@ aarch Alexa androidx -Appcenter armeabi armv armv7l @@ -22,7 +21,6 @@ Colour commandline Compat constraintlayout -convmv cozinha crhino cstring From 84f6b67fe924404289095bbc61c3ff6c24f0d5cf Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Wed, 20 Nov 2024 18:08:50 -0800 Subject: [PATCH 06/15] try without Samsung Galaxy S8 --- script/automation/browserstack.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/script/automation/browserstack.py b/script/automation/browserstack.py index 3cc5e98e..735ad87d 100644 --- a/script/automation/browserstack.py +++ b/script/automation/browserstack.py @@ -9,7 +9,7 @@ devices_dict = { 'android-min-max': [ - 'Samsung Galaxy S8-7.0', + # 'Samsung Galaxy S8-7.0', Slow and causes BrowserStack timeout of 2 hours. 'Samsung Galaxy M52-11.0', 'Google Pixel 9-15.0' ], From a341c9f28ca91e530938601fc9bf0b9565e69188 Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Thu, 21 Nov 2024 13:06:50 -0800 Subject: [PATCH 07/15] bump gradle --- binding/android/RhinoTestApp/build.gradle | 12 +++++++----- binding/android/RhinoTestApp/gradle.properties | 3 +++ .../gradle/wrapper/gradle-wrapper.properties | 2 +- .../android/RhinoTestApp/rhino-test-app/build.gradle | 3 ++- .../rhino-test-app/src/main/AndroidManifest.xml | 3 +-- 5 files changed, 14 insertions(+), 9 deletions(-) diff --git a/binding/android/RhinoTestApp/build.gradle b/binding/android/RhinoTestApp/build.gradle index bdc4b580..80c9b76c 100644 --- a/binding/android/RhinoTestApp/build.gradle +++ b/binding/android/RhinoTestApp/build.gradle @@ -1,6 +1,6 @@ // Top-level build file where you can add configuration options common to all sub-projects/modules. ext { - defaultTargetSdkVersion = 31 + defaultTargetSdkVersion = 33 } buildscript { @@ -10,14 +10,16 @@ buildscript { } dependencies { - classpath 'com.android.tools.build:gradle:7.2.2' + classpath 'com.android.tools.build:gradle:8.4.2' } } allprojects { gradle.projectsEvaluated { - tasks.withType(JavaCompile) { - options.compilerArgs << "-Xlint:deprecation" + tasks.withType(JavaCompile).tap { + configureEach { + options.compilerArgs << "-Xlint:deprecation" + } } } @@ -27,6 +29,6 @@ allprojects { } } -task clean(type: Delete) { +tasks.register('clean', Delete) { delete rootProject.buildDir } diff --git a/binding/android/RhinoTestApp/gradle.properties b/binding/android/RhinoTestApp/gradle.properties index d546deaf..16a11ee0 100644 --- a/binding/android/RhinoTestApp/gradle.properties +++ b/binding/android/RhinoTestApp/gradle.properties @@ -6,7 +6,10 @@ # http://www.gradle.org/docs/current/userguide/build_environment.html # Specifies the JVM arguments used for the daemon process. # The setting is particularly useful for tweaking memory settings. +android.defaults.buildfeatures.buildconfig=true android.enableJetifier=true +android.nonFinalResIds=false +android.nonTransitiveRClass=false android.useAndroidX=true org.gradle.jvmargs=-Xmx1536m # When configured, Gradle will run in incubating parallel mode. diff --git a/binding/android/RhinoTestApp/gradle/wrapper/gradle-wrapper.properties b/binding/android/RhinoTestApp/gradle/wrapper/gradle-wrapper.properties index 0d698460..a4cafe43 100644 --- a/binding/android/RhinoTestApp/gradle/wrapper/gradle-wrapper.properties +++ b/binding/android/RhinoTestApp/gradle/wrapper/gradle-wrapper.properties @@ -3,4 +3,4 @@ distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists -distributionUrl=https\://services.gradle.org/distributions/gradle-7.3.3-bin.zip +distributionUrl=https\://services.gradle.org/distributions/gradle-8.6-bin.zip diff --git a/binding/android/RhinoTestApp/rhino-test-app/build.gradle b/binding/android/RhinoTestApp/rhino-test-app/build.gradle index 12b68bd3..a642f990 100644 --- a/binding/android/RhinoTestApp/rhino-test-app/build.gradle +++ b/binding/android/RhinoTestApp/rhino-test-app/build.gradle @@ -30,7 +30,7 @@ if (rootProject.file("local.properties").exists()) { } android { - compileSdkVersion defaultTargetSdkVersion + compileSdk defaultTargetSdkVersion defaultConfig { applicationId "ai.picovoice.rhino.testapp" @@ -138,6 +138,7 @@ android { lint { abortOnError false } + namespace 'ai.picovoice.rhino.testapp' } dependencies { diff --git a/binding/android/RhinoTestApp/rhino-test-app/src/main/AndroidManifest.xml b/binding/android/RhinoTestApp/rhino-test-app/src/main/AndroidManifest.xml index e05c06c1..0e8c87b0 100644 --- a/binding/android/RhinoTestApp/rhino-test-app/src/main/AndroidManifest.xml +++ b/binding/android/RhinoTestApp/rhino-test-app/src/main/AndroidManifest.xml @@ -1,6 +1,5 @@ - + From f9f7df44aa683f983cb271e668d41a78d0655796 Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Thu, 21 Nov 2024 16:46:59 -0800 Subject: [PATCH 08/15] try fix tests - use separate files instead of @RunWith(Enclosed.class) --- binding/android/RhinoTestApp/build.gradle | 2 +- .../gradle/wrapper/gradle-wrapper.properties | 2 +- .../RhinoTestApp/rhino-test-app/build.gradle | 8 +- .../testapp/LanguageOutOfContextTests.java | 97 ++++ .../testapp/LanguageWithinContextTests.java | 123 +++++ .../ai/picovoice/rhino/testapp/RhinoTest.java | 465 ------------------ .../rhino/testapp/StandardTests.java | 298 +++++++++++ binding/ios/RhinoAppTest/Podfile.lock | 2 +- .../RhinoAppTest.xcodeproj/project.pbxproj | 43 +- script/automation/browserstack.py | 15 +- 10 files changed, 559 insertions(+), 496 deletions(-) create mode 100644 binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/LanguageOutOfContextTests.java create mode 100644 binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/LanguageWithinContextTests.java delete mode 100644 binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/RhinoTest.java create mode 100644 binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/StandardTests.java diff --git a/binding/android/RhinoTestApp/build.gradle b/binding/android/RhinoTestApp/build.gradle index 80c9b76c..65629d5f 100644 --- a/binding/android/RhinoTestApp/build.gradle +++ b/binding/android/RhinoTestApp/build.gradle @@ -10,7 +10,7 @@ buildscript { } dependencies { - classpath 'com.android.tools.build:gradle:8.4.2' + classpath 'com.android.tools.build:gradle:8.2.2' } } diff --git a/binding/android/RhinoTestApp/gradle/wrapper/gradle-wrapper.properties b/binding/android/RhinoTestApp/gradle/wrapper/gradle-wrapper.properties index a4cafe43..b0912934 100644 --- a/binding/android/RhinoTestApp/gradle/wrapper/gradle-wrapper.properties +++ b/binding/android/RhinoTestApp/gradle/wrapper/gradle-wrapper.properties @@ -3,4 +3,4 @@ distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists -distributionUrl=https\://services.gradle.org/distributions/gradle-8.6-bin.zip +distributionUrl=https\://services.gradle.org/distributions/gradle-8.2-bin.zip diff --git a/binding/android/RhinoTestApp/rhino-test-app/build.gradle b/binding/android/RhinoTestApp/rhino-test-app/build.gradle index a642f990..c062ede9 100644 --- a/binding/android/RhinoTestApp/rhino-test-app/build.gradle +++ b/binding/android/RhinoTestApp/rhino-test-app/build.gradle @@ -123,10 +123,14 @@ android { androidTest { java { if (System.getProperty("testBuildType", "debug") == "perf") { - exclude "**/RhinoTest.java" + exclude "**/StandardTests.java" + exclude "**/LanguageOutOfContextTests.java" + exclude "**/LanguageWithinContextTests.java" exclude "**/IntegrationTest.java" } else if (System.getProperty("testBuildType", "debug") == "integ") { - exclude "**/RhinoTest.java" + exclude "**/StandardTests.java" + exclude "**/LanguageOutOfContextTests.java" + exclude "**/LanguageWithinContextTests.java" exclude "**/PerformanceTest.java" } else { exclude "**/IntegrationTest.java" diff --git a/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/LanguageOutOfContextTests.java b/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/LanguageOutOfContextTests.java new file mode 100644 index 00000000..8b58cdbb --- /dev/null +++ b/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/LanguageOutOfContextTests.java @@ -0,0 +1,97 @@ +/* + Copyright 2022-2024 Picovoice Inc. + + You may not use this file except in compliance with the license. A copy of the license is + located in the "LICENSE" file accompanying this source. + + Unless required by applicable law or agreed to in writing, software distributed under the + License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either + express or implied. See the License for the specific language governing permissions and + limitations under the License. +*/ + +package ai.picovoice.rhino.testapp; + +import static org.junit.Assert.assertFalse; + +import com.google.gson.JsonArray; +import com.google.gson.JsonObject; +import com.google.gson.JsonParser; + +import org.junit.Test; +import org.junit.runner.RunWith; +import org.junit.runners.Parameterized; + +import java.io.File; +import java.io.IOException; +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; +import java.util.Objects; + +import ai.picovoice.rhino.Rhino; +import ai.picovoice.rhino.RhinoInference; + +@RunWith(Parameterized.class) +public class LanguageOutOfContextTests extends BaseTest { + + @Parameterized.Parameter(value = 0) + public String modelFile; + + @Parameterized.Parameter(value = 1) + public String contextFile; + + @Parameterized.Parameter(value = 2) + public String testAudioFile; + + @Parameterized.Parameters(name = "{2}") + public static Collection initParameters() throws IOException { + String testDataJsonString = getTestDataString(); + + JsonObject testDataJson = JsonParser.parseString(testDataJsonString).getAsJsonObject(); + JsonArray outOfContextDataJson = testDataJson.getAsJsonObject("tests").getAsJsonArray("out_of_context"); + + List parameters = new ArrayList<>(); + for (int i = 0; i < outOfContextDataJson.size(); i++) { + JsonObject testData = outOfContextDataJson.get(i).getAsJsonObject(); + String language = testData.get("language").getAsString(); + String contextName = testData.get("context_name").getAsString(); + + String modelFile = String.format("model_files/rhino_params_%s.pv", language); + String contextFile = String.format("context_files/%s/%s_android.rhn", language, contextName); + String audioFile = String.format("audio_samples/test_out_of_context_%s.wav", language); + + if (Objects.equals(language, "en")) { + modelFile = "model_files/rhino_params.pv"; + audioFile = "audio_samples/test_out_of_context.wav"; + } + + parameters.add(new Object[] { + modelFile, + contextFile, + audioFile, + }); + } + + return parameters; + } + + @Test + public void testOutOfContext() throws Exception { + + String modelPath = new File(testResourcesPath, modelFile).getAbsolutePath(); + String contextPath = new File(testResourcesPath, contextFile).getAbsolutePath(); + + Rhino r = new Rhino.Builder() + .setAccessKey(accessKey) + .setModelPath(modelPath) + .setContextPath(contextPath) + .build(appContext); + + File testAudio = new File(testResourcesPath, testAudioFile); + + RhinoInference inference = processTestAudio(r, testAudio); + assertFalse(inference.getIsUnderstood()); + r.delete(); + } +} diff --git a/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/LanguageWithinContextTests.java b/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/LanguageWithinContextTests.java new file mode 100644 index 00000000..0b20db6a --- /dev/null +++ b/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/LanguageWithinContextTests.java @@ -0,0 +1,123 @@ +/* + Copyright 2022-2024 Picovoice Inc. + + You may not use this file except in compliance with the license. A copy of the license is + located in the "LICENSE" file accompanying this source. + + Unless required by applicable law or agreed to in writing, software distributed under the + License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either + express or implied. See the License for the specific language governing permissions and + limitations under the License. +*/ + +package ai.picovoice.rhino.testapp; + +import static org.junit.Assert.assertEquals; + +import com.google.gson.JsonArray; +import com.google.gson.JsonElement; +import com.google.gson.JsonObject; +import com.google.gson.JsonParser; + +import org.junit.Test; +import org.junit.runner.RunWith; +import org.junit.runners.Parameterized; + +import java.io.File; +import java.io.IOException; +import java.util.ArrayList; +import java.util.Collection; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Objects; + +import ai.picovoice.rhino.Rhino; +import ai.picovoice.rhino.RhinoInference; + +@RunWith(Parameterized.class) +public class LanguageWithinContextTests extends BaseTest { + + @Parameterized.Parameter(value = 0) + public String modelFile; + + @Parameterized.Parameter(value = 1) + public String contextFile; + + @Parameterized.Parameter(value = 2) + public String testAudioFile; + + @Parameterized.Parameter(value = 3) + public boolean expectedIsUnderstood; + + @Parameterized.Parameter(value = 4) + public String expectedIntent; + + @Parameterized.Parameter(value = 5) + public Map expectedSlots; + + @Parameterized.Parameters(name = "{2}") + public static Collection initParameters() throws IOException { + String testDataJsonString = getTestDataString(); + + JsonObject testDataJson = JsonParser.parseString(testDataJsonString).getAsJsonObject(); + JsonArray withinContextDataJson = testDataJson.getAsJsonObject("tests").getAsJsonArray("within_context"); + + List parameters = new ArrayList<>(); + for (int i = 0; i < withinContextDataJson.size(); i++) { + JsonObject testData = withinContextDataJson.get(i).getAsJsonObject(); + String language = testData.get("language").getAsString(); + String contextName = testData.get("context_name").getAsString(); + JsonObject inferenceJson = testData.getAsJsonObject("inference"); + + String modelFile = String.format("model_files/rhino_params_%s.pv", language); + String contextFile = String.format("context_files/%s/%s_android.rhn", language, contextName); + String audioFile = String.format("audio_samples/test_within_context_%s.wav", language); + + String intent = inferenceJson.get("intent").getAsString(); + HashMap slots = new HashMap(); + for (Map.Entry entry : inferenceJson.getAsJsonObject("slots").asMap().entrySet()) { + slots.put(entry.getKey(), entry.getValue().getAsString()); + } + + if (Objects.equals(language, "en")) { + modelFile = "model_files/rhino_params.pv"; + audioFile = "audio_samples/test_within_context.wav"; + } + + parameters.add(new Object[] { + modelFile, + contextFile, + audioFile, + true, + intent, + slots, + }); + } + + return parameters; + } + + @Test + public void testWithinContext() throws Exception { + + String modelPath = new File(testResourcesPath, modelFile).getAbsolutePath(); + String contextPath = new File(testResourcesPath, contextFile).getAbsolutePath(); + + Rhino r = new Rhino.Builder() + .setAccessKey(accessKey) + .setModelPath(modelPath) + .setContextPath(contextPath) + .build(appContext); + + File testAudio = new File(testResourcesPath, testAudioFile); + + RhinoInference inference = processTestAudio(r, testAudio); + assertEquals(expectedIsUnderstood, inference.getIsUnderstood()); + assertEquals(expectedIntent, inference.getIntent()); + + Map slots = inference.getSlots(); + assertEquals(expectedSlots, slots); + r.delete(); + } +} diff --git a/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/RhinoTest.java b/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/RhinoTest.java deleted file mode 100644 index 65a17920..00000000 --- a/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/RhinoTest.java +++ /dev/null @@ -1,465 +0,0 @@ -/* - Copyright 2022-2023 Picovoice Inc. - - You may not use this file except in compliance with the license. A copy of the license is - located in the "LICENSE" file accompanying this source. - - Unless required by applicable law or agreed to in writing, software distributed under the - License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either - express or implied. See the License for the specific language governing permissions and - limitations under the License. -*/ - -package ai.picovoice.rhino.testapp; - -import static org.junit.Assert.assertEquals; -import static org.junit.Assert.assertFalse; -import static org.junit.Assert.assertTrue; - -import com.google.gson.JsonArray; -import com.google.gson.JsonElement; -import com.google.gson.JsonObject; -import com.google.gson.JsonParser; - -import org.junit.Test; -import org.junit.experimental.runners.Enclosed; -import org.junit.runner.RunWith; -import org.junit.runners.Parameterized; - -import java.io.File; -import java.io.IOException; -import java.util.ArrayList; -import java.util.Collection; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.Objects; - -import ai.picovoice.rhino.Rhino; -import ai.picovoice.rhino.RhinoException; -import ai.picovoice.rhino.RhinoInference; - - -@RunWith(Enclosed.class) -public class RhinoTest { - - public static class StandardTests extends BaseTest { - - @Test - public void testInitSuccessSimple() throws RhinoException { - File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); - Rhino r = new Rhino.Builder() - .setAccessKey(accessKey) - .setContextPath(contextPath.getAbsolutePath()) - .build(appContext); - - assertTrue(r.getVersion() != null && !r.getVersion().equals("")); - assertTrue(r.getFrameLength() > 0); - assertTrue(r.getSampleRate() > 0); - assertTrue(r.getContextInformation() != null && !r.getContextInformation().equals("")); - - r.delete(); - } - - @Test - public void testInitSuccessWithCustomModelPath() throws RhinoException { - File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); - File modelPath = new File(testResourcesPath, "model_files/rhino_params.pv"); - Rhino r = new Rhino.Builder() - .setAccessKey(accessKey) - .setContextPath(contextPath.getAbsolutePath()) - .setModelPath(modelPath.getAbsolutePath()) - .build(appContext); - - assertTrue(r.getContextInformation() != null && !r.getContextInformation().equals("")); - r.delete(); - } - - @Test - public void testInitSuccessWithCustomSensitivity() throws RhinoException { - File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); - Rhino r = new Rhino.Builder() - .setAccessKey(accessKey) - .setContextPath(contextPath.getAbsolutePath()) - .setSensitivity(0.7f) - .build(appContext); - - assertTrue(r.getContextInformation() != null && !r.getContextInformation().equals("")); - r.delete(); - } - - @Test - public void testInitSuccessWithCustomEndpointDurationSec() throws RhinoException { - File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); - Rhino r = new Rhino.Builder() - .setAccessKey(accessKey) - .setContextPath(contextPath.getAbsolutePath()) - .setEndpointDurationSec(3.0f) - .build(appContext); - - assertTrue(r.getContextInformation() != null && !r.getContextInformation().equals("")); - r.delete(); - } - - @Test - public void testInitSuccessWithRequireEndpointFalse() throws RhinoException { - File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); - Rhino r = new Rhino.Builder() - .setAccessKey(accessKey) - .setContextPath(contextPath.getAbsolutePath()) - .setRequireEndpoint(false) - .build(appContext); - - assertTrue(r.getContextInformation() != null && !r.getContextInformation().equals("")); - r.delete(); - } - - @Test - public void testInitFailWithMismatchedLanguage() { - boolean didFail = false; - File contextPath = new File(testResourcesPath, "context_files/de/beleuchtung_android.rhn"); - File modelPath = new File(testResourcesPath, "model_files/rhino_params.pv"); - try { - new Rhino.Builder() - .setAccessKey(accessKey) - .setContextPath(contextPath.getAbsolutePath()) - .setModelPath(modelPath.getAbsolutePath()) - .build(appContext); - } catch (RhinoException e) { - didFail = true; - } - - assertTrue(didFail); - } - - @Test - public void testInitFailWithNoAccessKey() { - boolean didFail = false; - File contextPath = new File(testResourcesPath, "context_files/fr/éclairage_intelligent_android.rhn"); - try { - new Rhino.Builder() - .setContextPath(contextPath.getAbsolutePath()) - .build(appContext); - } catch (RhinoException e) { - didFail = true; - } - - assertTrue(didFail); - } - - @Test - public void testInitFailWithNoContext() { - boolean didFail = false; - try { - new Rhino.Builder() - .setAccessKey(accessKey) - .build(appContext); - } catch (RhinoException e) { - didFail = true; - } - - assertTrue(didFail); - } - - @Test - public void testInitFailWithInvalidContextPath() { - boolean didFail = false; - File contextPath = new File(testResourcesPath, "bad_path/bad_path.rhn"); - try { - new Rhino.Builder() - .setAccessKey(accessKey) - .setContextPath(contextPath.getAbsolutePath()) - .build(appContext); - } catch (RhinoException e) { - didFail = true; - } - - assertTrue(didFail); - } - - @Test - public void testInitFailWithInvalidModelPath() { - boolean didFail = false; - File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); - File modelPath = new File(testResourcesPath, "bad_path/bad_path.pv"); - try { - new Rhino.Builder() - .setAccessKey(accessKey) - .setContextPath(contextPath.getAbsolutePath()) - .setModelPath(modelPath.getAbsolutePath()) - .build(appContext); - } catch (RhinoException e) { - didFail = true; - } - - assertTrue(didFail); - } - - @Test - public void testInitFailWithInvalidSensitivity() { - boolean didFail = false; - File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); - try { - new Rhino.Builder() - .setAccessKey(accessKey) - .setContextPath(contextPath.getAbsolutePath()) - .setSensitivity(10) - .build(appContext); - } catch (RhinoException e) { - didFail = true; - } - - assertTrue(didFail); - } - - @Test - public void testInitFailWithInvalidEndpointDurationSec() { - boolean didFail = false; - File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); - try { - new Rhino.Builder() - .setAccessKey(accessKey) - .setContextPath(contextPath.getAbsolutePath()) - .setEndpointDurationSec(10.0f) - .build(appContext); - } catch (RhinoException e) { - didFail = true; - } - - assertTrue(didFail); - } - - @Test - public void testInitFailWithWrongPlatform() { - boolean didFail = false; - File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_linux.rhn"); - try { - new Rhino.Builder() - .setAccessKey(accessKey) - .setContextPath(contextPath.getAbsolutePath()) - .build(appContext); - } catch (RhinoException e) { - didFail = true; - } - - assertTrue(didFail); - } - - - @Test - public void testInitWithNonAsciiModelName() throws RhinoException { - File contextPath = new File(testResourcesPath, "context_files/es/iluminación_inteligente_android.rhn"); - File modelPath = new File(testResourcesPath, "model_files/rhino_params_es.pv"); - Rhino r = new Rhino.Builder() - .setAccessKey(accessKey) - .setContextPath(contextPath.getAbsolutePath()) - .setModelPath(modelPath.getAbsolutePath()) - .build(appContext); - - assertTrue(r.getContextInformation() != null && !r.getContextInformation().equals("")); - r.delete(); - } - - @Test - public void testReset() throws Exception { - File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); - - Rhino r = new Rhino.Builder() - .setAccessKey(accessKey) - .setContextPath(contextPath.getAbsolutePath()) - .build(appContext); - - File testAudio = new File(testResourcesPath, "audio_samples/test_within_context.wav"); - boolean isFinalized = processFileHelper(r, testAudio, 15); - assertFalse(isFinalized); - - r.reset(); - isFinalized = processFileHelper(r, testAudio, -1); - assertTrue(isFinalized); - - RhinoInference inference = r.getInference(); - assertTrue(inference.getIsUnderstood()); - r.delete(); - } - - @Test - public void testErrorStack() { - File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); - - String[] error = {}; - try { - new Rhino.Builder() - .setAccessKey("invalid") - .setContextPath(contextPath.getAbsolutePath()) - .build(appContext); - } catch (RhinoException e) { - error = e.getMessageStack(); - } - - assertTrue(0 < error.length); - assertTrue(error.length <= 8); - - try { - new Rhino.Builder() - .setAccessKey("invalid") - .setContextPath(contextPath.getAbsolutePath()) - .build(appContext); - } catch (RhinoException e) { - for (int i = 0; i < error.length; i++) { - assertEquals(e.getMessageStack()[i], error[i]); - } - } - } - } - - @RunWith(Parameterized.class) - public static class LanguageWithinContextTests extends BaseTest { - - @Parameterized.Parameter(value = 0) - public String modelFile; - - @Parameterized.Parameter(value = 1) - public String contextFile; - - @Parameterized.Parameter(value = 2) - public String testAudioFile; - - @Parameterized.Parameter(value = 3) - public boolean expectedIsUnderstood; - - @Parameterized.Parameter(value = 4) - public String expectedIntent; - - @Parameterized.Parameter(value = 5) - public Map expectedSlots; - - @Parameterized.Parameters(name = "{2}") - public static Collection initParameters() throws IOException { - String testDataJsonString = getTestDataString(); - - JsonObject testDataJson = JsonParser.parseString(testDataJsonString).getAsJsonObject(); - JsonArray withinContextDataJson = testDataJson.getAsJsonObject("tests").getAsJsonArray("within_context"); - - List parameters = new ArrayList<>(); - for (int i = 0; i < withinContextDataJson.size(); i++) { - JsonObject testData = withinContextDataJson.get(i).getAsJsonObject(); - String language = testData.get("language").getAsString(); - String contextName = testData.get("context_name").getAsString(); - JsonObject inferenceJson = testData.getAsJsonObject("inference"); - - String modelFile = String.format("model_files/rhino_params_%s.pv", language); - String contextFile = String.format("context_files/%s/%s_android.rhn", language, contextName); - String audioFile = String.format("audio_samples/test_within_context_%s.wav", language); - - String intent = inferenceJson.get("intent").getAsString(); - HashMap slots = new HashMap(); - for (Map.Entry entry : inferenceJson.getAsJsonObject("slots").asMap().entrySet()) { - slots.put(entry.getKey(), entry.getValue().getAsString()); - } - - if (Objects.equals(language, "en")) { - modelFile = "model_files/rhino_params.pv"; - audioFile = "audio_samples/test_within_context.wav"; - } - - parameters.add(new Object[] { - modelFile, - contextFile, - audioFile, - true, - intent, - slots, - }); - } - - return parameters; - } - - @Test - public void testWithinContext() throws Exception { - - String modelPath = new File(testResourcesPath, modelFile).getAbsolutePath(); - String contextPath = new File(testResourcesPath, contextFile).getAbsolutePath(); - - Rhino r = new Rhino.Builder() - .setAccessKey(accessKey) - .setModelPath(modelPath) - .setContextPath(contextPath) - .build(appContext); - - File testAudio = new File(testResourcesPath, testAudioFile); - - RhinoInference inference = processTestAudio(r, testAudio); - assertEquals(expectedIsUnderstood, inference.getIsUnderstood()); - assertEquals(expectedIntent, inference.getIntent()); - - Map slots = inference.getSlots(); - assertEquals(expectedSlots, slots); - r.delete(); - } - } - - @RunWith(Parameterized.class) - public static class LanguageOutOfContextTests extends BaseTest { - - @Parameterized.Parameter(value = 0) - public String modelFile; - - @Parameterized.Parameter(value = 1) - public String contextFile; - - @Parameterized.Parameter(value = 2) - public String testAudioFile; - - @Parameterized.Parameters(name = "{2}") - public static Collection initParameters() throws IOException { - String testDataJsonString = getTestDataString(); - - JsonObject testDataJson = JsonParser.parseString(testDataJsonString).getAsJsonObject(); - JsonArray outOfContextDataJson = testDataJson.getAsJsonObject("tests").getAsJsonArray("out_of_context"); - - List parameters = new ArrayList<>(); - for (int i = 0; i < outOfContextDataJson.size(); i++) { - JsonObject testData = outOfContextDataJson.get(i).getAsJsonObject(); - String language = testData.get("language").getAsString(); - String contextName = testData.get("context_name").getAsString(); - - String modelFile = String.format("model_files/rhino_params_%s.pv", language); - String contextFile = String.format("context_files/%s/%s_android.rhn", language, contextName); - String audioFile = String.format("audio_samples/test_out_of_context_%s.wav", language); - - if (Objects.equals(language, "en")) { - modelFile = "model_files/rhino_params.pv"; - audioFile = "audio_samples/test_out_of_context.wav"; - } - - parameters.add(new Object[] { - modelFile, - contextFile, - audioFile, - }); - } - - return parameters; - } - - @Test - public void testOutOfContext() throws Exception { - - String modelPath = new File(testResourcesPath, modelFile).getAbsolutePath(); - String contextPath = new File(testResourcesPath, contextFile).getAbsolutePath(); - - Rhino r = new Rhino.Builder() - .setAccessKey(accessKey) - .setModelPath(modelPath) - .setContextPath(contextPath) - .build(appContext); - - File testAudio = new File(testResourcesPath, testAudioFile); - - RhinoInference inference = processTestAudio(r, testAudio); - assertFalse(inference.getIsUnderstood()); - r.delete(); - } - } -} diff --git a/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/StandardTests.java b/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/StandardTests.java new file mode 100644 index 00000000..c5b17993 --- /dev/null +++ b/binding/android/RhinoTestApp/rhino-test-app/src/androidTest/java/ai/picovoice/rhino/testapp/StandardTests.java @@ -0,0 +1,298 @@ +/* + Copyright 2022-2024 Picovoice Inc. + + You may not use this file except in compliance with the license. A copy of the license is + located in the "LICENSE" file accompanying this source. + + Unless required by applicable law or agreed to in writing, software distributed under the + License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either + express or implied. See the License for the specific language governing permissions and + limitations under the License. +*/ + +package ai.picovoice.rhino.testapp; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertTrue; + +import org.junit.Test; +import org.junit.runner.RunWith; + +import java.io.File; + +import ai.picovoice.rhino.Rhino; +import ai.picovoice.rhino.RhinoException; +import ai.picovoice.rhino.RhinoInference; + +import androidx.test.ext.junit.runners.AndroidJUnit4; + +@RunWith(AndroidJUnit4.class) +public class StandardTests extends BaseTest { + + @Test + public void testInitSuccessSimple() throws RhinoException { + File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); + Rhino r = new Rhino.Builder() + .setAccessKey(accessKey) + .setContextPath(contextPath.getAbsolutePath()) + .build(appContext); + + assertTrue(r.getVersion() != null && !r.getVersion().equals("")); + assertTrue(r.getFrameLength() > 0); + assertTrue(r.getSampleRate() > 0); + assertTrue(r.getContextInformation() != null && !r.getContextInformation().equals("")); + + r.delete(); + } + + @Test + public void testInitSuccessWithCustomModelPath() throws RhinoException { + File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); + File modelPath = new File(testResourcesPath, "model_files/rhino_params.pv"); + Rhino r = new Rhino.Builder() + .setAccessKey(accessKey) + .setContextPath(contextPath.getAbsolutePath()) + .setModelPath(modelPath.getAbsolutePath()) + .build(appContext); + + assertTrue(r.getContextInformation() != null && !r.getContextInformation().equals("")); + r.delete(); + } + + @Test + public void testInitSuccessWithCustomSensitivity() throws RhinoException { + File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); + Rhino r = new Rhino.Builder() + .setAccessKey(accessKey) + .setContextPath(contextPath.getAbsolutePath()) + .setSensitivity(0.7f) + .build(appContext); + + assertTrue(r.getContextInformation() != null && !r.getContextInformation().equals("")); + r.delete(); + } + + @Test + public void testInitSuccessWithCustomEndpointDurationSec() throws RhinoException { + File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); + Rhino r = new Rhino.Builder() + .setAccessKey(accessKey) + .setContextPath(contextPath.getAbsolutePath()) + .setEndpointDurationSec(3.0f) + .build(appContext); + + assertTrue(r.getContextInformation() != null && !r.getContextInformation().equals("")); + r.delete(); + } + + @Test + public void testInitSuccessWithRequireEndpointFalse() throws RhinoException { + File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); + Rhino r = new Rhino.Builder() + .setAccessKey(accessKey) + .setContextPath(contextPath.getAbsolutePath()) + .setRequireEndpoint(false) + .build(appContext); + + assertTrue(r.getContextInformation() != null && !r.getContextInformation().equals("")); + r.delete(); + } + + @Test + public void testInitFailWithMismatchedLanguage() { + boolean didFail = false; + File contextPath = new File(testResourcesPath, "context_files/de/beleuchtung_android.rhn"); + File modelPath = new File(testResourcesPath, "model_files/rhino_params.pv"); + try { + new Rhino.Builder() + .setAccessKey(accessKey) + .setContextPath(contextPath.getAbsolutePath()) + .setModelPath(modelPath.getAbsolutePath()) + .build(appContext); + } catch (RhinoException e) { + didFail = true; + } + + assertTrue(didFail); + } + + @Test + public void testInitFailWithNoAccessKey() { + boolean didFail = false; + File contextPath = new File(testResourcesPath, "context_files/fr/éclairage_intelligent_android.rhn"); + try { + new Rhino.Builder() + .setContextPath(contextPath.getAbsolutePath()) + .build(appContext); + } catch (RhinoException e) { + didFail = true; + } + + assertTrue(didFail); + } + + @Test + public void testInitFailWithNoContext() { + boolean didFail = false; + try { + new Rhino.Builder() + .setAccessKey(accessKey) + .build(appContext); + } catch (RhinoException e) { + didFail = true; + } + + assertTrue(didFail); + } + + @Test + public void testInitFailWithInvalidContextPath() { + boolean didFail = false; + File contextPath = new File(testResourcesPath, "bad_path/bad_path.rhn"); + try { + new Rhino.Builder() + .setAccessKey(accessKey) + .setContextPath(contextPath.getAbsolutePath()) + .build(appContext); + } catch (RhinoException e) { + didFail = true; + } + + assertTrue(didFail); + } + + @Test + public void testInitFailWithInvalidModelPath() { + boolean didFail = false; + File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); + File modelPath = new File(testResourcesPath, "bad_path/bad_path.pv"); + try { + new Rhino.Builder() + .setAccessKey(accessKey) + .setContextPath(contextPath.getAbsolutePath()) + .setModelPath(modelPath.getAbsolutePath()) + .build(appContext); + } catch (RhinoException e) { + didFail = true; + } + + assertTrue(didFail); + } + + @Test + public void testInitFailWithInvalidSensitivity() { + boolean didFail = false; + File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); + try { + new Rhino.Builder() + .setAccessKey(accessKey) + .setContextPath(contextPath.getAbsolutePath()) + .setSensitivity(10) + .build(appContext); + } catch (RhinoException e) { + didFail = true; + } + + assertTrue(didFail); + } + + @Test + public void testInitFailWithInvalidEndpointDurationSec() { + boolean didFail = false; + File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); + try { + new Rhino.Builder() + .setAccessKey(accessKey) + .setContextPath(contextPath.getAbsolutePath()) + .setEndpointDurationSec(10.0f) + .build(appContext); + } catch (RhinoException e) { + didFail = true; + } + + assertTrue(didFail); + } + + @Test + public void testInitFailWithWrongPlatform() { + boolean didFail = false; + File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_linux.rhn"); + try { + new Rhino.Builder() + .setAccessKey(accessKey) + .setContextPath(contextPath.getAbsolutePath()) + .build(appContext); + } catch (RhinoException e) { + didFail = true; + } + + assertTrue(didFail); + } + + + @Test + public void testInitWithNonAsciiModelName() throws RhinoException { + File contextPath = new File(testResourcesPath, "context_files/es/iluminación_inteligente_android.rhn"); + File modelPath = new File(testResourcesPath, "model_files/rhino_params_es.pv"); + Rhino r = new Rhino.Builder() + .setAccessKey(accessKey) + .setContextPath(contextPath.getAbsolutePath()) + .setModelPath(modelPath.getAbsolutePath()) + .build(appContext); + + assertTrue(r.getContextInformation() != null && !r.getContextInformation().equals("")); + r.delete(); + } + + @Test + public void testReset() throws Exception { + File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); + + Rhino r = new Rhino.Builder() + .setAccessKey(accessKey) + .setContextPath(contextPath.getAbsolutePath()) + .build(appContext); + + File testAudio = new File(testResourcesPath, "audio_samples/test_within_context.wav"); + boolean isFinalized = processFileHelper(r, testAudio, 15); + assertFalse(isFinalized); + + r.reset(); + isFinalized = processFileHelper(r, testAudio, -1); + assertTrue(isFinalized); + + RhinoInference inference = r.getInference(); + assertTrue(inference.getIsUnderstood()); + r.delete(); + } + + @Test + public void testErrorStack() { + File contextPath = new File(testResourcesPath, "context_files/en/coffee_maker_android.rhn"); + + String[] error = {}; + try { + new Rhino.Builder() + .setAccessKey("invalid") + .setContextPath(contextPath.getAbsolutePath()) + .build(appContext); + } catch (RhinoException e) { + error = e.getMessageStack(); + } + + assertTrue(0 < error.length); + assertTrue(error.length <= 8); + + try { + new Rhino.Builder() + .setAccessKey("invalid") + .setContextPath(contextPath.getAbsolutePath()) + .build(appContext); + } catch (RhinoException e) { + for (int i = 0; i < error.length; i++) { + assertEquals(e.getMessageStack()[i], error[i]); + } + } + } +} diff --git a/binding/ios/RhinoAppTest/Podfile.lock b/binding/ios/RhinoAppTest/Podfile.lock index c72bb35d..453bc9dd 100644 --- a/binding/ios/RhinoAppTest/Podfile.lock +++ b/binding/ios/RhinoAppTest/Podfile.lock @@ -17,4 +17,4 @@ SPEC CHECKSUMS: PODFILE CHECKSUM: befcf72816c8d591c1514da9266b761003ddcca5 -COCOAPODS: 1.11.3 +COCOAPODS: 1.16.2 diff --git a/binding/ios/RhinoAppTest/RhinoAppTest.xcodeproj/project.pbxproj b/binding/ios/RhinoAppTest/RhinoAppTest.xcodeproj/project.pbxproj index bda118e7..d722497e 100644 --- a/binding/ios/RhinoAppTest/RhinoAppTest.xcodeproj/project.pbxproj +++ b/binding/ios/RhinoAppTest/RhinoAppTest.xcodeproj/project.pbxproj @@ -43,7 +43,6 @@ /* End PBXContainerItemProxy section */ /* Begin PBXFileReference section */ - 18FAB8FC8D086FF2500A0A6A /* Pods-RhinoAppTestUITests.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RhinoAppTestUITests.debug.xcconfig"; path = "Target Support Files/Pods-RhinoAppTestUITests/Pods-RhinoAppTestUITests.debug.xcconfig"; sourceTree = ""; }; 1E1A57CE27CEC84000A05F37 /* RhinoAppTest.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = RhinoAppTest.app; sourceTree = BUILT_PRODUCTS_DIR; }; 1E1A57D127CEC84000A05F37 /* AppDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = ""; }; 1E1A57D527CEC84000A05F37 /* ViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ViewController.swift; sourceTree = ""; }; @@ -59,16 +58,17 @@ 1E5B7AD82800AC8F00F8BDDB /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; 1E5B7AD92800ADCD00F8BDDB /* coffee_maker_ios.rhn */ = {isa = PBXFileReference; lastKnownFileType = file; name = coffee_maker_ios.rhn; path = ../../../../resources/contexts/ios/coffee_maker_ios.rhn; sourceTree = ""; }; 1E5B7ADB2800ADD800F8BDDB /* test_within_context.wav */ = {isa = PBXFileReference; lastKnownFileType = audio.wav; name = test_within_context.wav; path = ../../../../resources/audio_samples/test_within_context.wav; sourceTree = ""; }; - 2B1742C8ACAFE68554B2DFD4 /* Pods-RhinoAppTest.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RhinoAppTest.release.xcconfig"; path = "Target Support Files/Pods-RhinoAppTest/Pods-RhinoAppTest.release.xcconfig"; sourceTree = ""; }; + 2531CB2545CE07CA2D9EEF02 /* Pods-RhinoAppTestUITests.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RhinoAppTestUITests.debug.xcconfig"; path = "Target Support Files/Pods-RhinoAppTestUITests/Pods-RhinoAppTestUITests.debug.xcconfig"; sourceTree = ""; }; + 294C62EF15A326500BCC3DA5 /* Pods-RhinoAppTest.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RhinoAppTest.debug.xcconfig"; path = "Target Support Files/Pods-RhinoAppTest/Pods-RhinoAppTest.debug.xcconfig"; sourceTree = ""; }; + 3BF8058B5843026A2D8F16DE /* Pods-RhinoAppTestUITests.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RhinoAppTestUITests.release.xcconfig"; path = "Target Support Files/Pods-RhinoAppTestUITests/Pods-RhinoAppTestUITests.release.xcconfig"; sourceTree = ""; }; + 4EA4D8381C4AEB0E8E83ECD5 /* Pods-PerformanceTest.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-PerformanceTest.debug.xcconfig"; path = "Target Support Files/Pods-PerformanceTest/Pods-PerformanceTest.debug.xcconfig"; sourceTree = ""; }; 5AA66651E907F4A1B2E5D84A /* libPods-RhinoAppTestUITests.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; includeInIndex = 0; path = "libPods-RhinoAppTestUITests.a"; sourceTree = BUILT_PRODUCTS_DIR; }; 5C8B41DCF24254E027C2B30C /* libPods-RhinoAppTest.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; includeInIndex = 0; path = "libPods-RhinoAppTest.a"; sourceTree = BUILT_PRODUCTS_DIR; }; - 5D3D0649948C17F6FF2875C0 /* Pods-RhinoAppTestUITests.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RhinoAppTestUITests.release.xcconfig"; path = "Target Support Files/Pods-RhinoAppTestUITests/Pods-RhinoAppTestUITests.release.xcconfig"; sourceTree = ""; }; - 900FCBC5CECD2C71CA548677 /* Pods-RhinoAppTest.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RhinoAppTest.debug.xcconfig"; path = "Target Support Files/Pods-RhinoAppTest/Pods-RhinoAppTest.debug.xcconfig"; sourceTree = ""; }; + 802E178A8F623E2D063F5669 /* Pods-PerformanceTest.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-PerformanceTest.release.xcconfig"; path = "Target Support Files/Pods-PerformanceTest/Pods-PerformanceTest.release.xcconfig"; sourceTree = ""; }; A52A723849AED2DCDE7E4FF2 /* BaseTest.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = BaseTest.swift; sourceTree = ""; }; A52A7ACF80691513299515C9 /* RhinoLanguageTests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = RhinoLanguageTests.swift; sourceTree = ""; }; + BC7411F9CDE2C60219C5AA1F /* Pods-RhinoAppTest.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-RhinoAppTest.release.xcconfig"; path = "Target Support Files/Pods-RhinoAppTest/Pods-RhinoAppTest.release.xcconfig"; sourceTree = ""; }; C71D40F5297A12EC00489C75 /* test_resources */ = {isa = PBXFileReference; lastKnownFileType = folder; path = test_resources; sourceTree = ""; }; - F10FFBFDE9AD2034FA8C0429 /* Pods-PerformanceTest.debug.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-PerformanceTest.debug.xcconfig"; path = "Target Support Files/Pods-PerformanceTest/Pods-PerformanceTest.debug.xcconfig"; sourceTree = ""; }; - F679D59C9F02B60A47B90200 /* Pods-PerformanceTest.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-PerformanceTest.release.xcconfig"; path = "Target Support Files/Pods-PerformanceTest/Pods-PerformanceTest.release.xcconfig"; sourceTree = ""; }; F7F2642FE16C549581638C57 /* libPods-PerformanceTest.a */ = {isa = PBXFileReference; explicitFileType = archive.ar; includeInIndex = 0; path = "libPods-PerformanceTest.a"; sourceTree = BUILT_PRODUCTS_DIR; }; /* End PBXFileReference section */ @@ -107,8 +107,8 @@ 1E1A57D027CEC84000A05F37 /* RhinoAppTest */, 1E1A57F127CEC84000A05F37 /* RhinoAppTestUITests */, 1E1A57CF27CEC84000A05F37 /* Products */, - 286CC2DF6F709F2C33610374 /* Pods */, DAAB5662628E2FB49E4EDDC1 /* Frameworks */, + B085AB16554F34C9EF0CA542 /* Pods */, ); sourceTree = ""; }; @@ -158,16 +158,17 @@ path = PerformanceTest; sourceTree = ""; }; - 286CC2DF6F709F2C33610374 /* Pods */ = { + B085AB16554F34C9EF0CA542 /* Pods */ = { isa = PBXGroup; children = ( - 900FCBC5CECD2C71CA548677 /* Pods-RhinoAppTest.debug.xcconfig */, - 2B1742C8ACAFE68554B2DFD4 /* Pods-RhinoAppTest.release.xcconfig */, - 18FAB8FC8D086FF2500A0A6A /* Pods-RhinoAppTestUITests.debug.xcconfig */, - 5D3D0649948C17F6FF2875C0 /* Pods-RhinoAppTestUITests.release.xcconfig */, - F10FFBFDE9AD2034FA8C0429 /* Pods-PerformanceTest.debug.xcconfig */, - F679D59C9F02B60A47B90200 /* Pods-PerformanceTest.release.xcconfig */, - ); + 4EA4D8381C4AEB0E8E83ECD5 /* Pods-PerformanceTest.debug.xcconfig */, + 802E178A8F623E2D063F5669 /* Pods-PerformanceTest.release.xcconfig */, + 294C62EF15A326500BCC3DA5 /* Pods-RhinoAppTest.debug.xcconfig */, + BC7411F9CDE2C60219C5AA1F /* Pods-RhinoAppTest.release.xcconfig */, + 2531CB2545CE07CA2D9EEF02 /* Pods-RhinoAppTestUITests.debug.xcconfig */, + 3BF8058B5843026A2D8F16DE /* Pods-RhinoAppTestUITests.release.xcconfig */, + ); + name = Pods; path = Pods; sourceTree = ""; }; @@ -671,7 +672,7 @@ }; 1E1A57F927CEC84000A05F37 /* Debug */ = { isa = XCBuildConfiguration; - baseConfigurationReference = 900FCBC5CECD2C71CA548677 /* Pods-RhinoAppTest.debug.xcconfig */; + baseConfigurationReference = 294C62EF15A326500BCC3DA5 /* Pods-RhinoAppTest.debug.xcconfig */; buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor; @@ -701,7 +702,7 @@ }; 1E1A57FA27CEC84000A05F37 /* Release */ = { isa = XCBuildConfiguration; - baseConfigurationReference = 2B1742C8ACAFE68554B2DFD4 /* Pods-RhinoAppTest.release.xcconfig */; + baseConfigurationReference = BC7411F9CDE2C60219C5AA1F /* Pods-RhinoAppTest.release.xcconfig */; buildSettings = { ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon; ASSETCATALOG_COMPILER_GLOBAL_ACCENT_COLOR_NAME = AccentColor; @@ -731,7 +732,7 @@ }; 1E1A57FF27CEC84000A05F37 /* Debug */ = { isa = XCBuildConfiguration; - baseConfigurationReference = 18FAB8FC8D086FF2500A0A6A /* Pods-RhinoAppTestUITests.debug.xcconfig */; + baseConfigurationReference = 2531CB2545CE07CA2D9EEF02 /* Pods-RhinoAppTestUITests.debug.xcconfig */; buildSettings = { CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 1; @@ -754,7 +755,7 @@ }; 1E1A580027CEC84000A05F37 /* Release */ = { isa = XCBuildConfiguration; - baseConfigurationReference = 5D3D0649948C17F6FF2875C0 /* Pods-RhinoAppTestUITests.release.xcconfig */; + baseConfigurationReference = 3BF8058B5843026A2D8F16DE /* Pods-RhinoAppTestUITests.release.xcconfig */; buildSettings = { CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 1; @@ -777,7 +778,7 @@ }; 1E5B7AD32800ABDE00F8BDDB /* Debug */ = { isa = XCBuildConfiguration; - baseConfigurationReference = F10FFBFDE9AD2034FA8C0429 /* Pods-PerformanceTest.debug.xcconfig */; + baseConfigurationReference = 4EA4D8381C4AEB0E8E83ECD5 /* Pods-PerformanceTest.debug.xcconfig */; buildSettings = { CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 1; @@ -800,7 +801,7 @@ }; 1E5B7AD42800ABDE00F8BDDB /* Release */ = { isa = XCBuildConfiguration; - baseConfigurationReference = F679D59C9F02B60A47B90200 /* Pods-PerformanceTest.release.xcconfig */; + baseConfigurationReference = 802E178A8F623E2D063F5669 /* Pods-PerformanceTest.release.xcconfig */; buildSettings = { CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 1; diff --git a/script/automation/browserstack.py b/script/automation/browserstack.py index 735ad87d..f9e711e0 100644 --- a/script/automation/browserstack.py +++ b/script/automation/browserstack.py @@ -26,6 +26,7 @@ ] } + def main(args: argparse.Namespace) -> None: app_files = { 'file': open(args.app_path, 'rb') @@ -63,7 +64,8 @@ def main(args: argparse.Namespace) -> None: 'app': app_response_json['app_url'], 'testSuite': test_response_json['test_suite_url'], 'project': args.project_name, - 'devices': devices_dict[args.devices] + 'devices': devices_dict[args.devices], + 'deviceLogs': true } while True: @@ -75,8 +77,8 @@ def main(args: argparse.Namespace) -> None: ) if (build_response is not None and 'message' in build_response.json() and '[BROWSERSTACK_ALL_PARALLELS_IN_USE]' in build_response.json()['message']): - print('Parallel threads limit reached. Waiting...', flush=True) - time.sleep(60) + print('Parallel threads limit reached. Waiting...', flush=True) + time.sleep(60) else: break @@ -94,7 +96,9 @@ def main(args: argparse.Namespace) -> None: print('Build Unsuccessful') exit(1) - print('View build results at https://app-automate.browserstack.com/dashboard/v2/builds/{}'.format(build_response_json['build_id'])) + print( + 'View build results at https://app-automate.browserstack.com/dashboard/v2/builds/{}' + .format(build_response_json['build_id'])) while True: time.sleep(60) @@ -116,6 +120,7 @@ def main(args: argparse.Namespace) -> None: if status != 'passed': exit(1) + if __name__ == '__main__': parser = argparse.ArgumentParser() parser.add_argument('--type', choices=['espresso', 'xcuitest'], required=True) @@ -128,4 +133,4 @@ def main(args: argparse.Namespace) -> None: parser.add_argument('--test_path', required=True) args = parser.parse_args() - main(args) \ No newline at end of file + main(args) From 3b34386fcb63755bf750f39340597cdc5317d1d6 Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Thu, 21 Nov 2024 16:51:05 -0800 Subject: [PATCH 09/15] use java 17 --- .github/workflows/android-browserstack.yml | 4 ++-- .github/workflows/android-demos.yml | 4 ++-- .github/workflows/android-perf.yml | 4 ++-- 3 files changed, 6 insertions(+), 6 deletions(-) diff --git a/.github/workflows/android-browserstack.yml b/.github/workflows/android-browserstack.yml index 69c8c7e5..1239dcf2 100644 --- a/.github/workflows/android-browserstack.yml +++ b/.github/workflows/android-browserstack.yml @@ -36,10 +36,10 @@ jobs: - run: pip3 install requests - - name: set up JDK 11 + - name: set up JDK 17 uses: actions/setup-java@v3 with: - java-version: '11' + java-version: '17' distribution: 'temurin' - name: Copy test_resources diff --git a/.github/workflows/android-demos.yml b/.github/workflows/android-demos.yml index 5cdfb11f..81f87285 100644 --- a/.github/workflows/android-demos.yml +++ b/.github/workflows/android-demos.yml @@ -25,10 +25,10 @@ jobs: steps: - uses: actions/checkout@v3 - - name: set up JDK 11 + - name: set up JDK 17 uses: actions/setup-java@v3 with: - java-version: '11' + java-version: '17' distribution: 'temurin' - name: Build diff --git a/.github/workflows/android-perf.yml b/.github/workflows/android-perf.yml index 2e1f729d..fd8c9fdd 100644 --- a/.github/workflows/android-perf.yml +++ b/.github/workflows/android-perf.yml @@ -39,10 +39,10 @@ jobs: - run: pip3 install requests - - name: set up JDK 11 + - name: set up JDK 17 uses: actions/setup-java@v3 with: - java-version: '11' + java-version: '17' distribution: 'temurin' - name: Copy test_resources From da5c3945fef8058f6c75a73ad6691f264f9c2808 Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Thu, 21 Nov 2024 16:56:25 -0800 Subject: [PATCH 10/15] fix --- .github/workflows/android-browserstack.yml | 4 ++-- script/automation/browserstack.py | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/.github/workflows/android-browserstack.yml b/.github/workflows/android-browserstack.yml index 1239dcf2..630a3f96 100644 --- a/.github/workflows/android-browserstack.yml +++ b/.github/workflows/android-browserstack.yml @@ -91,10 +91,10 @@ jobs: - run: pip3 install requests - - name: set up JDK 11 + - name: set up JDK 17 uses: actions/setup-java@v3 with: - java-version: '11' + java-version: '17' distribution: 'temurin' - name: Copy test_resources diff --git a/script/automation/browserstack.py b/script/automation/browserstack.py index f9e711e0..70dbd447 100644 --- a/script/automation/browserstack.py +++ b/script/automation/browserstack.py @@ -65,7 +65,7 @@ def main(args: argparse.Namespace) -> None: 'testSuite': test_response_json['test_suite_url'], 'project': args.project_name, 'devices': devices_dict[args.devices], - 'deviceLogs': true + 'deviceLogs': True } while True: From 73a0b3a3451ae304ada742e113f0a70eb7628cd6 Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Thu, 21 Nov 2024 17:41:50 -0800 Subject: [PATCH 11/15] try fix integ build --- binding/android/RhinoTestApp/rhino-test-app/build.gradle | 3 +++ script/automation/browserstack.py | 2 +- 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/binding/android/RhinoTestApp/rhino-test-app/build.gradle b/binding/android/RhinoTestApp/rhino-test-app/build.gradle index c062ede9..0fed967f 100644 --- a/binding/android/RhinoTestApp/rhino-test-app/build.gradle +++ b/binding/android/RhinoTestApp/rhino-test-app/build.gradle @@ -172,5 +172,8 @@ afterEvaluate { tasks."merge${flavor.name.capitalize()}ReleaseAssets".dependsOn "${flavor.name}ContextName" tasks."merge${flavor.name.capitalize()}DebugAssets".dependsOn "${flavor.name}CopyAudio" tasks."merge${flavor.name.capitalize()}ReleaseAssets".dependsOn "${flavor.name}CopyAudio" + tasks."generate${flavor.name.capitalize()}ReleaseLintVitalReportModel".dependsOn "${flavor.name}CopyParams" + tasks."generate${flavor.name.capitalize()}ReleaseLintVitalReportModel".dependsOn "${flavor.name}CopyContext" + tasks."generate${flavor.name.capitalize()}ReleaseLintVitalReportModel".dependsOn "${flavor.name}CopyAudio" } } diff --git a/script/automation/browserstack.py b/script/automation/browserstack.py index 70dbd447..dd5a0d2b 100644 --- a/script/automation/browserstack.py +++ b/script/automation/browserstack.py @@ -9,7 +9,7 @@ devices_dict = { 'android-min-max': [ - # 'Samsung Galaxy S8-7.0', Slow and causes BrowserStack timeout of 2 hours. + 'Samsung Galaxy S8-7.0', 'Samsung Galaxy M52-11.0', 'Google Pixel 9-15.0' ], From cabc9fda759500ab6ba3a9d5dea4452602bb5e31 Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Thu, 21 Nov 2024 17:49:42 -0800 Subject: [PATCH 12/15] try fix --- binding/android/RhinoTestApp/rhino-test-app/build.gradle | 3 +++ 1 file changed, 3 insertions(+) diff --git a/binding/android/RhinoTestApp/rhino-test-app/build.gradle b/binding/android/RhinoTestApp/rhino-test-app/build.gradle index 0fed967f..28a06716 100644 --- a/binding/android/RhinoTestApp/rhino-test-app/build.gradle +++ b/binding/android/RhinoTestApp/rhino-test-app/build.gradle @@ -175,5 +175,8 @@ afterEvaluate { tasks."generate${flavor.name.capitalize()}ReleaseLintVitalReportModel".dependsOn "${flavor.name}CopyParams" tasks."generate${flavor.name.capitalize()}ReleaseLintVitalReportModel".dependsOn "${flavor.name}CopyContext" tasks."generate${flavor.name.capitalize()}ReleaseLintVitalReportModel".dependsOn "${flavor.name}CopyAudio" + tasks."lintVitalAnalyze${flavor.name.capitalize()}Release".dependsOn "${flavor.name}CopyParams" + tasks."lintVitalAnalyze${flavor.name.capitalize()}Release".dependsOn "${flavor.name}CopyContext" + tasks."lintVitalAnalyze${flavor.name.capitalize()}Release".dependsOn "${flavor.name}CopyAudio" } } From cc17281e927060db0f4b767f13ad49d4adb3f34a Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Fri, 22 Nov 2024 11:06:50 -0800 Subject: [PATCH 13/15] try fix --- .../android/RhinoTestApp/rhino-test-app/proguard-rules.pro | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/binding/android/RhinoTestApp/rhino-test-app/proguard-rules.pro b/binding/android/RhinoTestApp/rhino-test-app/proguard-rules.pro index 158caf35..2d07c7d2 100644 --- a/binding/android/RhinoTestApp/rhino-test-app/proguard-rules.pro +++ b/binding/android/RhinoTestApp/rhino-test-app/proguard-rules.pro @@ -20,4 +20,7 @@ # hide the original source file name. #-renamesourcefileattribute SourceFile -keep class com.google.** { *; } --keep class com.microsoft.** { *; } \ No newline at end of file +-keep class com.microsoft.** { *; } + +-dontwarn com.google.errorprone.annotations.CheckReturnValue +-dontwarn com.google.errorprone.annotations.MustBeClosed \ No newline at end of file From 7d484be5ffeb4a71ca5dd616bcd706640b1ec976 Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Fri, 22 Nov 2024 11:20:20 -0800 Subject: [PATCH 14/15] try fix --- binding/android/RhinoTestApp/rhino-test-app/build.gradle | 1 + binding/android/RhinoTestApp/rhino-test-app/proguard-rules.pro | 3 ++- 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/binding/android/RhinoTestApp/rhino-test-app/build.gradle b/binding/android/RhinoTestApp/rhino-test-app/build.gradle index 28a06716..7e7535bc 100644 --- a/binding/android/RhinoTestApp/rhino-test-app/build.gradle +++ b/binding/android/RhinoTestApp/rhino-test-app/build.gradle @@ -150,6 +150,7 @@ dependencies { implementation 'androidx.appcompat:appcompat:1.3.0' implementation 'androidx.constraintlayout:constraintlayout:2.1.4' implementation 'com.google.code.gson:gson:2.10' + implementation 'com.google.errorprone:error_prone_annotations:2.36.0' implementation 'ai.picovoice:rhino-android:3.0.1' // Espresso UI Testing diff --git a/binding/android/RhinoTestApp/rhino-test-app/proguard-rules.pro b/binding/android/RhinoTestApp/rhino-test-app/proguard-rules.pro index 2d07c7d2..280693da 100644 --- a/binding/android/RhinoTestApp/rhino-test-app/proguard-rules.pro +++ b/binding/android/RhinoTestApp/rhino-test-app/proguard-rules.pro @@ -23,4 +23,5 @@ -keep class com.microsoft.** { *; } -dontwarn com.google.errorprone.annotations.CheckReturnValue --dontwarn com.google.errorprone.annotations.MustBeClosed \ No newline at end of file +-dontwarn com.google.errorprone.annotations.MustBeClosed +-dontwarn javax.lang.model.element.Modifier \ No newline at end of file From 811f90456b184b72b4b768b0a1d981991584a2e3 Mon Sep 17 00:00:00 2001 From: Albert Ho Date: Fri, 22 Nov 2024 15:03:08 -0800 Subject: [PATCH 15/15] try different ios devices --- script/automation/browserstack.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/script/automation/browserstack.py b/script/automation/browserstack.py index dd5a0d2b..54c0e77a 100644 --- a/script/automation/browserstack.py +++ b/script/automation/browserstack.py @@ -17,8 +17,8 @@ 'Google Pixel 6 Pro-15.0' ], 'ios-min-max': [ - 'iPhone SE 2020-13', - 'iPhone 14 Pro-16', + 'iPhone SE 2022-15', + 'iPhone 14 Plus-16', 'iPhone 14-18' ], 'ios-perf': [