From bfd2d3ae8499a7337fe6dfa1172ea7b15fc767a8 Mon Sep 17 00:00:00 2001 From: Albert Ho <82767499+albho@users.noreply.github.com> Date: Mon, 25 Nov 2024 13:31:49 -0800 Subject: [PATCH] Migrate mobile testing to BrowserStack (#746) --- ...appcenter.yml => android-browserstack.yml} | 76 ++- .github/workflows/android-demos.yml | 4 +- .github/workflows/android-perf.yml | 41 +- .github/workflows/ios-appcenter.yml | 78 --- .github/workflows/ios-browserstack.yml | 84 ++++ .github/workflows/ios-demos.yml | 3 - .github/workflows/ios-perf.yml | 53 +- binding/android/RhinoTestApp/build.gradle | 12 +- .../android/RhinoTestApp/gradle.properties | 3 + .../gradle/wrapper/gradle-wrapper.properties | 2 +- .../RhinoTestApp/rhino-test-app/build.gradle | 19 +- .../rhino-test-app/proguard-rules.pro | 6 +- .../ai/picovoice/rhino/testapp/BaseTest.java | 12 - .../rhino/testapp/IntegrationTest.java | 11 - .../testapp/LanguageOutOfContextTests.java | 97 ++++ .../testapp/LanguageWithinContextTests.java | 123 +++++ .../ai/picovoice/rhino/testapp/RhinoTest.java | 465 ------------------ .../rhino/testapp/StandardTests.java | 298 +++++++++++ .../src/main/AndroidManifest.xml | 3 +- binding/ios/RhinoAppTest/Podfile.lock | 2 +- .../RhinoAppTest.xcodeproj/project.pbxproj | 43 +- .../ios/RhinoAppTest/copy_test_resources.sh | 3 - resources/.lint/spell-check/dict.txt | 3 +- script/automation/browserstack.py | 136 +++++ 24 files changed, 882 insertions(+), 695 deletions(-) rename .github/workflows/{android-appcenter.yml => android-browserstack.yml} (60%) delete mode 100644 .github/workflows/ios-appcenter.yml create mode 100644 .github/workflows/ios-browserstack.yml 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 create mode 100644 script/automation/browserstack.py diff --git a/.github/workflows/android-appcenter.yml b/.github/workflows/android-browserstack.yml similarity index 60% rename from .github/workflows/android-appcenter.yml rename to .github/workflows/android-browserstack.yml index 290a75868..630a3f964 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,24 +23,23 @@ 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/* + python-version: '3.10' + - run: + pip3 install requests - - name: Install AppCenter CLI - run: npm install -g appcenter-cli - - - 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 @@ -68,35 +67,34 @@ 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/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 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 + - name: set up JDK 17 uses: actions/setup-java@v3 with: - java-version: '11' + java-version: '17' distribution: 'temurin' - name: Copy test_resources @@ -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/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-demos.yml b/.github/workflows/android-demos.yml index 5cdfb11fd..81f872856 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 7300e2a8c..fd8c9fdd9 100644 --- a/.github/workflows/android-perf.yml +++ b/.github/workflows/android-perf.yml @@ -19,33 +19,30 @@ 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/* + python-version: '3.10' + - run: + pip3 install requests - - name: Install AppCenter CLI - run: npm install -g appcenter-cli - - - 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 @@ -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/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" diff --git a/.github/workflows/ios-appcenter.yml b/.github/workflows/ios-appcenter.yml deleted file mode 100644 index 97cc38a05..000000000 --- a/.github/workflows/ios-appcenter.yml +++ /dev/null @@ -1,78 +0,0 @@ -name: iOS AppCenter Tests - -on: - workflow_dispatch: - push: - branches: [ master ] - paths: - - '.github/workflows/ios-appcenter.yml' - - 'binding/ios/RhinoAppTest/**' - - 'resources/.test/**' - - 'resources/audio_samples/**' - pull_request: - branches: [ master, 'v[0-9]+.[0-9]+' ] - paths: - - '.github/workflows/ios-appcenter.yml' - - 'binding/ios/RhinoAppTest/**' - - 'resources/.test/**' - - 'resources/audio_samples/**' - -defaults: - run: - working-directory: binding/ios/RhinoAppTest - -jobs: - build: - name: Run iOS Tests on AppCenter - runs-on: macos-latest - - steps: - - name: Checkout - uses: actions/checkout@v3 - - - name: Set up Node.js LTS - uses: actions/setup-node@v3 - 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: 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 AppID - run: sed -i '.bak' 's:{TESTING_ACCESS_KEY_HERE}:${{secrets.PV_VALID_ACCESS_KEY}}:' - RhinoAppTestUITests/BaseTest.swift - - - name: XCode Build - run: xcrun xcodebuild build-for-testing - -configuration Debug - -workspace RhinoAppTest.xcworkspace - -sdk iphoneos - -scheme RhinoAppTest - -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 diff --git a/.github/workflows/ios-browserstack.yml b/.github/workflows/ios-browserstack.yml new file mode 100644 index 000000000..69c5ec901 --- /dev/null +++ b/.github/workflows/ios-browserstack.yml @@ -0,0 +1,84 @@ +name: iOS BrowserStack Tests + +on: + workflow_dispatch: + push: + branches: [ master ] + paths: + - '.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-browserstack.yml' + - 'binding/ios/RhinoAppTest/**' + - 'resources/.test/**' + - 'resources/audio_samples/**' + +defaults: + run: + working-directory: binding/ios/RhinoAppTest + +jobs: + build: + name: Run iOS Tests on BrowserStack + runs-on: macos-latest + + steps: + - name: Checkout + uses: actions/checkout@v3 + + - name: Installing Python + uses: actions/setup-python@v5 + with: + python-version: '3.10' + - run: + pip3 install requests + + - name: Install Cocoapods + run: gem install cocoapods + + - name: Make build dir + run: mkdir ddp + + - 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 + + - name: XCode Build + run: xcrun xcodebuild build-for-testing + -configuration Debug + -workspace RhinoAppTest.xcworkspace + -sdk iphoneos + -scheme RhinoAppTest + -derivedDataPath ddp + CODE_SIGNING_ALLOWED=NO + + - 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-demos.yml b/.github/workflows/ios-demos.yml index bcf84ff4d..7156b1e9f 100644 --- a/.github/workflows/ios-demos.yml +++ b/.github/workflows/ios-demos.yml @@ -32,9 +32,6 @@ jobs: - name: Install Cocoapods run: gem install cocoapods - - name: Install AppCenter CLI - run: npm install -g appcenter-cli - - name: Make build dir run: mkdir ddp diff --git a/.github/workflows/ios-perf.yml b/.github/workflows/ios-perf.yml index e9d12ab9f..ae43049bc 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,32 +37,26 @@ 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 - - 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 AppID + - name: Inject AccessKey run: sed -i '.bak' 's:{TESTING_ACCESS_KEY_HERE}:${{secrets.PV_VALID_ACCESS_KEY}}:' PerformanceTest/PerformanceTest.swift @@ -71,7 +65,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 +78,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/binding/android/RhinoTestApp/build.gradle b/binding/android/RhinoTestApp/build.gradle index bdc4b5806..65629d5fa 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.2.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 d546deaf0..16a11ee07 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 0d6984602..b09129340 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.2-bin.zip diff --git a/binding/android/RhinoTestApp/rhino-test-app/build.gradle b/binding/android/RhinoTestApp/rhino-test-app/build.gradle index c1b60b688..7e7535bcb 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" @@ -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" @@ -138,6 +142,7 @@ android { lint { abortOnError false } + namespace 'ai.picovoice.rhino.testapp' } dependencies { @@ -145,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 @@ -153,7 +159,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') } @@ -168,5 +173,11 @@ 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" + 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" } } diff --git a/binding/android/RhinoTestApp/rhino-test-app/proguard-rules.pro b/binding/android/RhinoTestApp/rhino-test-app/proguard-rules.pro index 158caf35e..280693dab 100644 --- a/binding/android/RhinoTestApp/rhino-test-app/proguard-rules.pro +++ b/binding/android/RhinoTestApp/rhino-test-app/proguard-rules.pro @@ -20,4 +20,8 @@ # 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 +-dontwarn javax.lang.model.element.Modifier \ No newline at end of file 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 20335d1cd..5f2c10278 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 30a653052..dcca9e417 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/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 000000000..8b58cdbb6 --- /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 000000000..0b20db6ad --- /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 65a179207..000000000 --- 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 000000000..c5b17993c --- /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/android/RhinoTestApp/rhino-test-app/src/main/AndroidManifest.xml b/binding/android/RhinoTestApp/rhino-test-app/src/main/AndroidManifest.xml index e05c06c1b..0e8c87b0c 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 @@ - + diff --git a/binding/ios/RhinoAppTest/Podfile.lock b/binding/ios/RhinoAppTest/Podfile.lock index c72bb35df..453bc9dd1 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 bda118e7c..d722497ea 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/binding/ios/RhinoAppTest/copy_test_resources.sh b/binding/ios/RhinoAppTest/copy_test_resources.sh index 75c5415d7..c049f6879 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 5462be854..21848e70c 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 @@ -137,4 +135,5 @@ xcodeproj xcscheme xcschemes xcshareddata +xcuitest xcworkspace diff --git a/script/automation/browserstack.py b/script/automation/browserstack.py new file mode 100644 index 000000000..54c0e77a0 --- /dev/null +++ b/script/automation/browserstack.py @@ -0,0 +1,136 @@ +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 2022-15', + 'iPhone 14 Plus-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], + 'deviceLogs': True + } + + 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)