blob: 91f1655bf464cfb39fc458c0f904cdf5dac3733c [file] [log] [blame]
/*
* Copyright 2022 Google LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* 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 com.google.android.libraries.mobiledatadownload.file.common.testing;
import static org.robolectric.shadow.api.Shadow.directlyOn;
import android.app.Application;
import android.content.Context;
import android.os.Build;
import android.os.Environment;
import androidx.test.core.app.ApplicationProvider;
import java.io.File;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import org.robolectric.annotation.Implementation;
import org.robolectric.annotation.Implements;
import org.robolectric.annotation.RealObject;
import org.robolectric.shadow.api.Shadow;
import org.robolectric.shadows.ShadowStatFs;
/** Common helper utilities that extend the Robolectric Shadow API. */
public final class ShadowUtils {
/**
* Adds an external dir to the Robolectric {@code Environment} and {@code Context}. In order to
* use this method, the test class must be configured to use the custom {@link ShadowContextImpl}
* and {@link ShadowEnvironment}.
*/
public static File addExternalDir(String path, boolean isEmulated, boolean isMounted) {
File dir = ShadowEnvironment.addExternalDir(path);
String storageState = isMounted ? Environment.MEDIA_MOUNTED : Environment.MEDIA_REMOVED;
ShadowEnvironment.setExternalStorageEmulated(dir, isEmulated);
ShadowEnvironment.setExternalStorageState(dir, storageState);
// The shadow implementation of LOLLIPOP storage APIs doesn't fully handle subdirectories, so
// the best we can do is to set the same storage properties on each directory we're adding.
ShadowContextImpl shadow =
Shadow.extract(
((Application) ApplicationProvider.getApplicationContext()).getBaseContext());
List<File> packageDirs = shadow.addExternalPackageDirs(dir);
for (File packageDir : packageDirs) {
ShadowEnvironment.setExternalStorageEmulated(packageDir, isEmulated);
ShadowEnvironment.setExternalStorageState(packageDir, storageState);
}
// Configure primary storage APIs if this is the first external dir
if (shadow.getExternalFilesDirs(null).length == 1) {
ShadowEnvironment.setExternalStorageDirectory(dir);
ShadowEnvironment.setIsExternalStorageEmulated(isEmulated);
ShadowEnvironment.setExternalStorageState(storageState);
}
return dir;
}
/**
* Configures the information returned by the Robolectric {@code StatsFs}.
*
* @param dir The file under which {@code StatFs} should return the specified stats
* @param totalBytes Total number of bytes on the filesystem
* @param freeBytes Number of unused bytes on the filesystem
*/
public static void setStatFs(File dir, int totalBytes, int freeBytes) {
int blockCount = totalBytes / ShadowStatFs.BLOCK_SIZE;
int freeBlocks = freeBytes / ShadowStatFs.BLOCK_SIZE;
int availableBlocks = freeBlocks;
ShadowStatFs.registerStats(dir, blockCount, freeBlocks, availableBlocks);
}
/** Extends the stock Robolectric {@code Context} shadow to support multiple externalFilesDirs. */
@Implements(className = org.robolectric.shadows.ShadowContextImpl.CLASS_NAME)
public static class ShadowContextImpl extends org.robolectric.shadows.ShadowContextImpl {
@RealObject private Context realObject;
private final List<File> externalFilesDirs = new ArrayList<>();
private final List<File> externalCacheDirs = new ArrayList<>();
// Used to simulate a race condition failure on pre-N devices. See getFilesDir.
private boolean getFilesDirRunAlready = false;
/**
* Adds package-private /files and /cache subdirectories to the Robolectric {@code Context}
* under the named external storage {@code partition}, then returns those new subdirectories.
*/
List<File> addExternalPackageDirs(File partition) {
File filesDir = new File(partition, "Android/data/com.google.android.storage.test/files");
File cacheDir = new File(partition, "Android/data/com.google.android.storage.test/cache");
externalFilesDirs.add(filesDir);
externalCacheDirs.add(cacheDir);
return Arrays.asList(filesDir, cacheDir);
}
@Override
@Implementation
public File getExternalFilesDir(String type) {
return !externalFilesDirs.isEmpty() ? externalFilesDirs.get(0) : null;
}
@Override
@Implementation
public File[] getExternalFilesDirs(String type) {
return externalFilesDirs.toArray(new File[externalFilesDirs.size()]);
}
@Implementation
public File getExternalCacheDir() {
return !externalCacheDirs.isEmpty() ? externalCacheDirs.get(0) : null;
}
@Implementation
public File[] getExternalCacheDirs() {
return externalCacheDirs.toArray(new File[externalCacheDirs.size()]);
}
/**
* See b/70255835. The first call (or first few calls) of getFilesDir may return null on pre-N
* devices. We simulate this by returning null only on the first call here.
*/
@Implementation
public File getFilesDir() {
if (getFilesDirRunAlready || Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
return directlyOn(realObject, ShadowContextImpl.CLASS_NAME, "getFilesDir");
}
getFilesDirRunAlready = true;
return null;
}
}
/**
* Extends the stock Robolectric {@code Environment} shadow to better emulate external storage
* APIs on lower sdk levels.
*/
@Implements(Environment.class)
public static class ShadowEnvironment extends org.robolectric.shadows.ShadowEnvironment {
private static File externalStorageDirectory;
/**
* Sets the value returned by {@code getExternalStorageDirectory}, which should be the same path
* as the first directory added via {@link ShadowEnvironment#addExternalDir}. This is necessary
* because by default, Robolectric returns a fixed value for {@code getExternalStorageDirectory}
* that doesn't reflect calls to the other shadow APIs.
*/
static void setExternalStorageDirectory(File dir) {
externalStorageDirectory = dir;
}
@Implementation
public static File getExternalStorageDirectory() {
return externalStorageDirectory;
}
}
private ShadowUtils() {}
}