如何从 CI 启动 APK 测试?
How to launch tests on APK from CI?
我有一个 CI 服务器可以从 GIT 构建 APK,它运行良好。
我需要在 CI 服务器生成的构建上启动自动测试 (UI Automator)。为了实现它我:
设置 CI 服务器以通过命令 gradlew assembleDebug
测试构建 APK。我也尝试用 gradlew assembleDebug -Pandroid.injected.signing.store.file=/path/to/keystore.jks -Pandroid.injected.signing.store.password=password1 -Pandroid.injected.signing.key.alias=myapp -Pandroid.injected.signing.key.password=password2
构建它
正在尝试使用命令启动它 adb.exe shell am instrument -w -r -e debug false -e package com.mysite.myapp com.mysite.myapp.test/com.mysite.myapp.runner.MyCustomTestsRunner
运气不好。我收到一个错误:
java.lang.SecurityException: Permission Denial: starting instrumentation ComponentInfo{com.mysite.myapp.test/com.mysite.myapp.runner.MyCustomTestsRunner} from pid=4922, uid=4922 not allowed be
cause package com.mysite.myapp.test does not have a signature matching the target com.mysite.myapp
at android.os.Parcel.createException(Parcel.java:2071)
at android.os.Parcel.readException(Parcel.java:2039)
at android.os.Parcel.readException(Parcel.java:1987)
at android.app.IActivityManager$Stub$Proxy.startInstrumentation(IActivityManager.java:5441)
at com.android.commands.am.Instrument.run(Instrument.java:512)
at com.android.commands.am.Am.runInstrument(Am.java:196)
at com.android.commands.am.Am.onRun(Am.java:80)
at com.android.internal.os.BaseCommand.run(BaseCommand.java:56)
at com.android.commands.am.Am.main(Am.java:50)
at com.android.internal.os.RuntimeInit.nativeFinishInit(Native Method)
at com.android.internal.os.RuntimeInit.main(RuntimeInit.java:338)
INSTRUMENTATION_STATUS: Error=Permission Denial: starting instrumentation ComponentInfo{com.mysite.myapp.test/com.mysite.myapp.runner.MyCustomTestsRunner} from pid=4922, uid=4922 not allowed b
ecause package com.mysite.myapp.test does not have a signature matching the target com.mysite.myapp
INSTRUMENTATION_STATUS: id=ActivityManagerService
INSTRUMENTATION_STATUS_CODE: -1
Caused by: android.os.RemoteException: Remote stack trace:
at com.android.server.am.ActivityManagerService.startInstrumentation(ActivityManagerService.java:15744)
at android.app.IActivityManager$Stub.onTransact(IActivityManager.java:2350)
at com.android.server.am.ActivityManagerService.onTransact(ActivityManagerService.java:2741)
at android.os.Binder.execTransactInternal(Binder.java:1021)
at android.os.Binder.execTransact(Binder.java:994)
它是如何工作的?我想 APK 内部已经包含测试,不需要安装一些额外的代码……或者我也必须在 CI 上构建测试?如何使用 CI?
中的 APK 在本地计算机上启动测试
您的应用的调试变体可能存在签名问题,因为插桩测试默认使用调试变体。唯一的判断方法是检查应用程序模块的 build.gradle
文件并查看应用了哪些 signingConfigs
。
我的建议是尝试以下方法。因为最终 CI 只是 运行 一个命令脚本,所以在你的开发机器上测试它,看看它是否成功:
- 打开命令终端并运行以下命令将构建您的应用程序和调试变体中的Android仪器测试,然后将它们安装到连接的Android设备:
gradlew clean
gradlew assembleDebug assembleDebugAndroidTest installDebug installDebugAndroidTest
- 启动插桩测试用例:
adb.exe shell am instrument -w -r -e debug false -e package com.mysite.myapp com.mysite.myapp.test/com.mysite.myapp.runner.MyCustomTestsRunner
如果异常 java.lang.SecurityException
仍然出现,那么这应该更多地表明 APK 签名不正确。
我有一个 CI 服务器可以从 GIT 构建 APK,它运行良好。
我需要在 CI 服务器生成的构建上启动自动测试 (UI Automator)。为了实现它我:
设置 CI 服务器以通过命令
构建它gradlew assembleDebug
测试构建 APK。我也尝试用gradlew assembleDebug -Pandroid.injected.signing.store.file=/path/to/keystore.jks -Pandroid.injected.signing.store.password=password1 -Pandroid.injected.signing.key.alias=myapp -Pandroid.injected.signing.key.password=password2
正在尝试使用命令启动它
adb.exe shell am instrument -w -r -e debug false -e package com.mysite.myapp com.mysite.myapp.test/com.mysite.myapp.runner.MyCustomTestsRunner
运气不好。我收到一个错误:
java.lang.SecurityException: Permission Denial: starting instrumentation ComponentInfo{com.mysite.myapp.test/com.mysite.myapp.runner.MyCustomTestsRunner} from pid=4922, uid=4922 not allowed be
cause package com.mysite.myapp.test does not have a signature matching the target com.mysite.myapp
at android.os.Parcel.createException(Parcel.java:2071)
at android.os.Parcel.readException(Parcel.java:2039)
at android.os.Parcel.readException(Parcel.java:1987)
at android.app.IActivityManager$Stub$Proxy.startInstrumentation(IActivityManager.java:5441)
at com.android.commands.am.Instrument.run(Instrument.java:512)
at com.android.commands.am.Am.runInstrument(Am.java:196)
at com.android.commands.am.Am.onRun(Am.java:80)
at com.android.internal.os.BaseCommand.run(BaseCommand.java:56)
at com.android.commands.am.Am.main(Am.java:50)
at com.android.internal.os.RuntimeInit.nativeFinishInit(Native Method)
at com.android.internal.os.RuntimeInit.main(RuntimeInit.java:338)
INSTRUMENTATION_STATUS: Error=Permission Denial: starting instrumentation ComponentInfo{com.mysite.myapp.test/com.mysite.myapp.runner.MyCustomTestsRunner} from pid=4922, uid=4922 not allowed b
ecause package com.mysite.myapp.test does not have a signature matching the target com.mysite.myapp
INSTRUMENTATION_STATUS: id=ActivityManagerService
INSTRUMENTATION_STATUS_CODE: -1
Caused by: android.os.RemoteException: Remote stack trace:
at com.android.server.am.ActivityManagerService.startInstrumentation(ActivityManagerService.java:15744)
at android.app.IActivityManager$Stub.onTransact(IActivityManager.java:2350)
at com.android.server.am.ActivityManagerService.onTransact(ActivityManagerService.java:2741)
at android.os.Binder.execTransactInternal(Binder.java:1021)
at android.os.Binder.execTransact(Binder.java:994)
它是如何工作的?我想 APK 内部已经包含测试,不需要安装一些额外的代码……或者我也必须在 CI 上构建测试?如何使用 CI?
中的 APK 在本地计算机上启动测试您的应用的调试变体可能存在签名问题,因为插桩测试默认使用调试变体。唯一的判断方法是检查应用程序模块的 build.gradle
文件并查看应用了哪些 signingConfigs
。
我的建议是尝试以下方法。因为最终 CI 只是 运行 一个命令脚本,所以在你的开发机器上测试它,看看它是否成功:
- 打开命令终端并运行以下命令将构建您的应用程序和调试变体中的Android仪器测试,然后将它们安装到连接的Android设备:
gradlew clean
gradlew assembleDebug assembleDebugAndroidTest installDebug installDebugAndroidTest
- 启动插桩测试用例:
adb.exe shell am instrument -w -r -e debug false -e package com.mysite.myapp com.mysite.myapp.test/com.mysite.myapp.runner.MyCustomTestsRunner
如果异常 java.lang.SecurityException
仍然出现,那么这应该更多地表明 APK 签名不正确。