android - 运行时异常 : Unable to create output dir:/storage/emulated/0/app_spoon-screenshots

标签 android ui-testing spoon barista

我正在使用 gradle-spoon-plugin 为 Android UI 测试设置 Spoon使用 Spoon 2.0.0 快照。我的项目是使用 Android Gradle 插件 3.0.1 设置的。

当通过 spoonRule.screenshot(activity, "hello") 截屏时,我得到这个 RuntimeException:

java.lang.RuntimeException: Unable to create output dir: /storage/emulated/0/app_spoon-screenshots
at com.squareup.spoon.SpoonRule.createDir(SpoonRule.java:167)
at com.squareup.spoon.SpoonRule.createDir(SpoonRule.java:164)
at com.squareup.spoon.SpoonRule.createDir(SpoonRule.java:164)
at com.squareup.spoon.SpoonRule.obtainDirectory(SpoonRule.java:108)
at com.squareup.spoon.SpoonRule.screenshot(SpoonRule.java:66)

如果我在 Nexus 4 API 19 模拟器上运行它,一切正常,但它在 Pixel 2 API 27 模拟器上不起作用。权限已从 19 更改为 27,因此这并非完全出乎意料。

我已经尝试了目前可用的大部分建议,包括在我的 androidTest 目录中添加一个 list ,该 list 授予读取和写入外部存储的权限(有和没有 maxSdkVersion) :

<?xml version="1.0" encoding="utf-8"?>
<manifest
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    package="tv.twitch.android.test">

    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" android:maxSdkVersion="18"/>
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" android:maxSdkVersion="18"/>

    <uses-sdk tools:overrideLibrary="android.support.test.uiautomator.v18"/>

</manifest>

在这两种情况下,我都看到这些权限被合并到我的应用程序 AndroidManifest 的最终 list 中(不确定如何检查测试应用程序的 list )。

我已经尝试通过 UIAutomator 向应用和测试包授予权限:

val device = UiDevice.getInstance(getInstrumentation())
device.executeShellCommand("pm grant tv.twitch.android.test android.permission.READ_EXTERNAL_STORAGE")
device.executeShellCommand("pm grant tv.twitch.android.debug android.permission.READ_EXTERNAL_STORAGE")
device.executeShellCommand("pm grant tv.twitch.android.test android.permission.WRITE_EXTERNAL_STORAGE")
device.executeShellCommand("pm grant tv.twitch.android.debug android.permission.WRITE_EXTERNAL_STORAGE")

这会向 Logcat 输出 Permission denied 并导致上述相同的异常。

如果我尝试利用 GrantPermissionRule,例如:

@get:Rule var runtimePermissionRule = GrantPermissionRule.grant(Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE)

我得到一个不同的异常:

junit.framework.AssertionFailedError: Failed to grant permissions, see logcat for details
at junit.framework.Assert.fail(Assert.java:50)
at android.support.test.runner.permission.PermissionRequester.requestPermissions(PermissionRequester.java:110)
at android.support.test.rule.GrantPermissionRule$RequestPermissionStatement.evaluate(GrantPermissionRule.java:108)

GrantPermissionCallable: Permission: android.permission.WRITE_EXTERNAL_STORAGE cannot be granted!

删除 Manifest.permission.WRITE_EXTERNAL_STORAGE 并只留下读取的内容让我们回到原来的异常:java.lang.RuntimeException: Unable to create output dir:/storage/emulated/0/app_spoon-截图

使用 gradle-spoon-plugin 从 Android Studio 中运行或在命令行上运行不会影响上述任何内容。

查看我的应用程序和测试应用程序在“设置”中授予的权限,我看到我的应用程序具有存储权限,但我的测试应用程序 (tv.twitch.android.test) 没有请求任何权限。

我还尝试使用 Barista图书馆:

PermissionGranter.allowPermissionsIfNeeded(Manifest.permission.WRITE_EXTERNAL_STORAGE)

他们也没有运气。

更新:

我尝试让我的测试应用程序以 SDK 版本 22 为目标,希望它能获得写入权限。

<?xml version="1.0" encoding="utf-8"?>
<manifest
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    package="tv.twitch.android.test">

    <uses-sdk
        android:minSdkVersion="16"
        android:targetSdkVersion="22"
        tools:overrideLibrary="android.support.test.uiautomator.v18"/>

    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

</manifest>

通过 Gradle (./gradlew spoon) 我尝试配置 gradle-spoon-plugin 来请求所有权限:

spoon {
    // Grant all runtime permissions during installation on Marshmallow and above devices.
    grantAll = true
}

运气不好。我什至尝试使用 Spoon 库版本 1.3.1 也无济于事。

最佳答案

我的问题是由外部库合并到具有 WRITE_EXTERNAL_STORAGE 权限的 maxSdkVersion 引起的。为了解决这个问题,我删除了该属性并重新添加了 WRITE_EXTERNAL_STORAGE 权限。这仅适用于我的应用程序 list ,而不是测试应用程序 list 。

<!-- Strip away maxSdkVersion -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"
                 tools:remove="android:maxSdkVersion"/>

<!-- Add the permission with no maxSdkVersion defined -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

然后您可以构建、授予权限并运行测试:

# Uninstall existing APKs and install our app APK and test APK
./gradlew uninstallAll installDebug installDebugAndroidTest

# List all APKs installed with adb shell 'pm list packages -f'
# Grant the app APK write and read to external storage permissions
adb shell pm grant gg.mark.debug android.permission.WRITE_EXTERNAL_STORAGE
adb shell pm grant gg.mark.debug android.permission.READ_EXTERNAL_STORAGE

export APK=build/outputs/apk/debug/debug.apk
export TEST_APK=build/outputs/apk/androidTest/debug/debug-androidTest.apk

# TEST_APK and APK are positional arguments so keep them in this order
# Disable GIF generation because it's slow
java -jar spoon-runner-2.0.0.jar --debug --disable-gif "$TEST_APK" "$APK"

如果您的应用的权限未在合并的 AndroidManifest.xml 中正确设置,则 pm grant 命令将失败。确保这些命令成功!

关于android - 运行时异常 : Unable to create output dir:/storage/emulated/0/app_spoon-screenshots,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49040071/

相关文章:

ios - 设置 UI 测试 Xcode 7 后单元测试从不运行

android - 空测试套件。 Espresso

java - 通过 spoon-gradle 插件从特定包运行仪器测试

android - 如何将 libgdx 添​​加到现有项目

android - 如何检查用户是否使用 YouTube api 观看了整个视频

Android 网络服务数据集

ios - UI 测试 - 访问相机

postgresql - 当我将数据从 DB2 复制到 Postgres 时,如何将字符代码从 Shift-JIS 更改为 UTF-8?

Pentaho ETL : Modified Javascript Step "SKIP_TRANSFORMATION" transformation Constant work logic

java - 安卓工作室;底部导航 View = 空