RuntimeException:无法创建输出目录:/storage/emulated/0/app_spoon-screenshots
RuntimeException: Unable to create output dir: /storage/emulated/0/app_spoon-screenshots
我正在使用 Spoon 2.0.0 快照为 Android UI 测试设置 Spoon。我的项目是使用 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
目录中添加一个清单,该目录授予读取和写入外部存储(有和没有 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(不确定如何检查测试应用程序的清单)。
我已经尝试通过 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")
这会输出一个 Permission denied
到 Logcat 并导致上面相同的异常。
如果我尝试利用 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-screenshots
运行 在 Android Studio 中或在命令行上使用 gradle-spoon-plugin
不会影响上述任何内容。
查看我的应用程序和测试应用程序在“设置”中授予的权限,我看到我的应用程序具有存储权限,但我的测试应用程序 (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 也无济于事。
首先,尝试在Pixel2(API27)上找到SystemSetting->YOUR_APP->Permission
,检查外部存储是否被禁用。
在 API23 (Android 6+) 之后的运行时需要 EXTERNAL_STORAGE
(WRITE
和 READ
)的权限。您应该检查权限并通知用户在系统配置中启用。参见 Request App Permissions
当然,普通应用程序无权像往常一样使用pm grant ...
授予他们的权限。
一个不推荐的解决方案是使整个应用程序以 API22 为目标。但正确的方法也是检查权限并通知用户。
我的问题是由外部库合并到具有 WRITE_EXTERNAL_STORAGE 权限的 maxSdkVersion 引起的。为了解决这个问题,我删除了该属性并重新添加了 WRITE_EXTERNAL_STORAGE 权限。这仅适用于我的应用程序清单,而不适用于测试应用程序清单。
<!-- 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
命令将失败。确保这些命令成功!
我正在使用 Spoon 2.0.0 快照为 Android UI 测试设置 Spoon。我的项目是使用 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
目录中添加一个清单,该目录授予读取和写入外部存储(有和没有 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(不确定如何检查测试应用程序的清单)。
我已经尝试通过 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")
这会输出一个 Permission denied
到 Logcat 并导致上面相同的异常。
如果我尝试利用 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-screenshots
运行 在 Android Studio 中或在命令行上使用 gradle-spoon-plugin
不会影响上述任何内容。
查看我的应用程序和测试应用程序在“设置”中授予的权限,我看到我的应用程序具有存储权限,但我的测试应用程序 (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 也无济于事。
首先,尝试在Pixel2(API27)上找到SystemSetting->YOUR_APP->Permission
,检查外部存储是否被禁用。
在 API23 (Android 6+) 之后的运行时需要 EXTERNAL_STORAGE
(WRITE
和 READ
)的权限。您应该检查权限并通知用户在系统配置中启用。参见 Request App Permissions
当然,普通应用程序无权像往常一样使用pm grant ...
授予他们的权限。
一个不推荐的解决方案是使整个应用程序以 API22 为目标。但正确的方法也是检查权限并通知用户。
我的问题是由外部库合并到具有 WRITE_EXTERNAL_STORAGE 权限的 maxSdkVersion 引起的。为了解决这个问题,我删除了该属性并重新添加了 WRITE_EXTERNAL_STORAGE 权限。这仅适用于我的应用程序清单,而不适用于测试应用程序清单。
<!-- 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
命令将失败。确保这些命令成功!