[android]: Merge two StorageUtils classes into one

No semantic changes intended. This patch is pure refactoring.

Signed-off-by: Roman Tsisyk <roman@tsisyk.com>
This commit is contained in:
Roman Tsisyk 2021-05-16 11:58:50 +03:00
parent 86277837e5
commit 0a134c430c
5 changed files with 196 additions and 190 deletions

View file

@ -4,22 +4,21 @@ import android.app.DownloadManager;
import android.content.Context;
import android.database.Cursor;
import android.net.Uri;
import android.os.FileUtils;
import android.text.TextUtils;
import androidx.annotation.MainThread;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.annotation.WorkerThread;
import com.mapswithme.maps.MwmApplication;
import com.mapswithme.util.StorageUtils;
import com.mapswithme.util.concurrency.UiThread;
import com.mapswithme.util.log.Logger;
import com.mapswithme.util.log.LoggerFactory;
import java.io.FileOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
public class MapDownloadCompletedProcessor
{
@ -67,31 +66,17 @@ public class MapDownloadCompletedProcessor
}
String dstPath = MapManager.nativeGetFilePathByUrl(targetUriPath);
return copyFile(context, downloadedFileUri, dstPath);
}
private static boolean copyFile(@NonNull Context context, @NonNull Uri from, @NonNull String to)
{
try (InputStream in = context.getContentResolver().openInputStream(from))
try
{
if (in == null)
if (!StorageUtils.copyFile(context, downloadedFileUri, new File(dstPath)))
return false;
try (OutputStream out = new FileOutputStream(to))
{
byte[] buf = new byte[4 * 1024];
int len;
while ((len = in.read(buf)) > 0)
out.write(buf, 0, len);
context.getContentResolver().delete(from, null, null);
return true;
}
context.getContentResolver().delete(downloadedFileUri, null, null);
return true;
}
catch (IOException e)
{
LOGGER.e(TAG, "Failed to copy or delete downloaded map file from " + from.toString() +
" to " + to + ". Exception ", e);
LOGGER.e(TAG, "Failed to copy or delete downloaded map file from " + downloadedFileUri.toString() +
" to " + dstPath + ". Exception ", e);
return false;
}
}

View file

@ -11,11 +11,15 @@ import android.widget.TextView;
import androidx.appcompat.app.AlertDialog;
import com.mapswithme.maps.BuildConfig;
import com.mapswithme.maps.Framework;
import com.mapswithme.maps.R;
import com.mapswithme.maps.base.OnBackPressListener;
import com.mapswithme.util.Constants;
import com.mapswithme.util.StorageUtils;
import com.mapswithme.util.Utils;
import java.io.File;
import java.util.List;
import java.util.Locale;
@ -81,9 +85,23 @@ public class StoragePathFragment extends BaseSettingsFragment
mPathManager.stopExternalStorageWatching();
}
static long getWritableDirSize()
{
final File writableDir = new File(Framework.nativeGetWritableDir());
if (BuildConfig.DEBUG)
{
if (!writableDir.exists())
throw new IllegalStateException("Writable directory doesn't exits, can't get size.");
if (!writableDir.isDirectory())
throw new IllegalStateException("Writable directory isn't a directory, can't get size.");
}
return StorageUtils.getDirSizeRecursively(writableDir, StoragePathManager.MOVABLE_FILES_FILTER);
}
private void updateList()
{
long dirSize = StorageUtils.getWritableDirSize();
long dirSize = getWritableDirSize();
mHeader.setText(getString(R.string.maps) + ": " + getSizeString(dirSize));
if (mAdapter != null)

View file

@ -22,6 +22,7 @@ import com.mapswithme.maps.R;
import com.mapswithme.maps.dialog.DialogUtils;
import com.mapswithme.maps.downloader.MapManager;
import com.mapswithme.util.Config;
import com.mapswithme.util.StorageUtils;
import com.mapswithme.util.concurrency.ThreadPool;
import com.mapswithme.util.concurrency.UiThread;
import com.mapswithme.util.log.Logger;

View file

@ -1,165 +0,0 @@
package com.mapswithme.maps.settings;
import com.mapswithme.maps.BuildConfig;
import com.mapswithme.maps.Framework;
import com.mapswithme.util.Constants;
import com.mapswithme.util.Utils;
import com.mapswithme.util.log.Logger;
import com.mapswithme.util.log.LoggerFactory;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.FilenameFilter;
import java.io.IOException;
import java.nio.channels.FileChannel;
import java.util.ArrayList;
final class StorageUtils
{
private StorageUtils() {}
private static final Logger LOGGER = LoggerFactory.INSTANCE.getLogger(LoggerFactory.Type.STORAGE);
private static final String TAG = StorageUtils.class.getSimpleName();
/**
* Check if directory is writable. On some devices with KitKat (eg, Samsung S4) simple File.canWrite() returns
* true for some actually read only directories on sdcard.
* see https://code.google.com/p/android/issues/detail?id=66369 for details
*
* @param path path to ckeck
* @return result
*/
@SuppressWarnings("ResultOfMethodCallIgnored")
static boolean isDirWritable(String path)
{
File f = new File(path, "mapsme_test_dir");
f.mkdir();
if (!f.exists())
return false;
f.delete();
return true;
}
static long getFreeBytesAtPath(String path)
{
long size = 0;
try
{
size = new File(path).getFreeSpace();
} catch (RuntimeException e)
{
e.printStackTrace();
}
return size;
}
static void copyFile(File source, File dest) throws IOException
{
int maxChunkSize = 10 * Constants.MB; // move file by smaller chunks to avoid OOM.
FileChannel inputChannel = null, outputChannel = null;
try
{
inputChannel = new FileInputStream(source).getChannel();
outputChannel = new FileOutputStream(dest).getChannel();
long totalSize = inputChannel.size();
for (long currentPosition = 0; currentPosition < totalSize; currentPosition += maxChunkSize)
{
outputChannel.position(currentPosition);
outputChannel.transferFrom(inputChannel, currentPosition, maxChunkSize);
}
} finally
{
Utils.closeSafely(inputChannel);
Utils.closeSafely(outputChannel);
}
}
private static long getDirSizeRecursively(File file, FilenameFilter fileFilter)
{
if (file.isDirectory())
{
long dirSize = 0;
for (File child : file.listFiles())
dirSize += getDirSizeRecursively(child, fileFilter);
return dirSize;
}
if (fileFilter.accept(file.getParentFile(), file.getName()))
return file.length();
return 0;
}
static long getWritableDirSize()
{
final File writableDir = new File(Framework.nativeGetWritableDir());
if (BuildConfig.DEBUG)
{
if (!writableDir.exists())
throw new IllegalStateException("Writable directory doesn't exits, can't get size.");
if (!writableDir.isDirectory())
throw new IllegalStateException("Writable directory isn't a directory, can't get size.");
}
return getDirSizeRecursively(writableDir, StoragePathManager.MOVABLE_FILES_FILTER);
}
/**
* Recursively lists all movable files in the directory.
*/
static void listFilesRecursively(File dir, String prefix, FilenameFilter filter, ArrayList<String> relPaths)
{
File[] list = dir.listFiles();
if (list == null)
return;
for (File file : list)
{
if (file.isDirectory())
{
listFilesRecursively(file, prefix + file.getName() + File.separator, filter, relPaths);
continue;
}
String name = file.getName();
if (filter.accept(dir, name))
relPaths.add(prefix + name);
}
}
@SuppressWarnings("ResultOfMethodCallIgnored")
private static void removeEmptyDirectories(File dir)
{
for (File file : dir.listFiles())
{
if (!file.isDirectory())
continue;
removeEmptyDirectories(file);
file.delete();
}
}
@SuppressWarnings("ResultOfMethodCallIgnored")
static boolean removeFilesInDirectory(File dir, File[] files)
{
try
{
for (File file : files)
{
if (file != null)
file.delete();
}
removeEmptyDirectories(dir);
return true;
} catch (Exception e)
{
e.printStackTrace();
return false;
}
}
}

View file

@ -17,7 +17,15 @@ import com.mapswithme.util.log.Logger;
import com.mapswithme.util.log.LoggerFactory;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.FilenameFilter;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.channels.FileChannel;
import java.util.ArrayList;
public class StorageUtils
{
@ -163,4 +171,163 @@ public class StorageUtils
return FileProvider.getUriForFile(context.getApplicationContext(),
BuildConfig.FILE_PROVIDER_AUTHORITY, new File(path));
}
/**
* Copy data from a URI into a local file.
* @param context context
* @param from a source URI.
* @param to a destination file
* @return true on success and false if the provider recently crashed.
* @throws IOException - if I/O error occurs.
*/
public static boolean copyFile(@NonNull Context context, @NonNull Uri from, @NonNull File to) throws IOException
{
try (InputStream in = context.getContentResolver().openInputStream(from))
{
if (in == null)
return false;
try (OutputStream out = new FileOutputStream(to))
{
byte[] buf = new byte[4 * 1024];
int len;
while ((len = in.read(buf)) > 0)
out.write(buf, 0, len);
return true;
}
}
}
/**
* Copy the contents of the source file to the target file.
* @param from a source file
* @param to a destination file
* @throws IOException - if I/O error occurs.
*/
static void copyFile(File from, File to) throws IOException
{
int maxChunkSize = 10 * Constants.MB; // move file by smaller chunks to avoid OOM.
FileChannel inputChannel = null, outputChannel = null;
try
{
inputChannel = new FileInputStream(from).getChannel();
outputChannel = new FileOutputStream(to).getChannel();
long totalSize = inputChannel.size();
for (long currentPosition = 0; currentPosition < totalSize; currentPosition += maxChunkSize)
{
outputChannel.position(currentPosition);
outputChannel.transferFrom(inputChannel, currentPosition, maxChunkSize);
}
} finally
{
Utils.closeSafely(inputChannel);
Utils.closeSafely(outputChannel);
}
}
/**
* Recursively lists all movable files in the directory.
*/
public static void listFilesRecursively(File dir, String prefix, FilenameFilter filter, ArrayList<String> relPaths)
{
File[] list = dir.listFiles();
if (list == null)
return;
for (File file : list)
{
if (file.isDirectory())
{
listFilesRecursively(file, prefix + file.getName() + File.separator, filter, relPaths);
continue;
}
String name = file.getName();
if (filter.accept(dir, name))
relPaths.add(prefix + name);
}
}
/**
* Check if directory is writable. On some devices with KitKat (eg, Samsung S4) simple File.canWrite() returns
* true for some actually read only directories on sdcard.
* see https://code.google.com/p/android/issues/detail?id=66369 for details
*
* @param path path to ckeck
* @return result
*/
@SuppressWarnings("ResultOfMethodCallIgnored")
public static boolean isDirWritable(String path)
{
File f = new File(path, "mapsme_test_dir");
f.mkdir();
if (!f.exists())
return false;
f.delete();
return true;
}
public static long getFreeBytesAtPath(String path)
{
long size = 0;
try
{
size = new File(path).getFreeSpace();
} catch (RuntimeException e)
{
e.printStackTrace();
}
return size;
}
public static long getDirSizeRecursively(File file, FilenameFilter fileFilter)
{
if (file.isDirectory())
{
long dirSize = 0;
for (File child : file.listFiles())
dirSize += getDirSizeRecursively(child, fileFilter);
return dirSize;
}
if (fileFilter.accept(file.getParentFile(), file.getName()))
return file.length();
return 0;
}
@SuppressWarnings("ResultOfMethodCallIgnored")
public static void removeEmptyDirectories(File dir)
{
for (File file : dir.listFiles())
{
if (!file.isDirectory())
continue;
removeEmptyDirectories(file);
file.delete();
}
}
@SuppressWarnings("ResultOfMethodCallIgnored")
public static boolean removeFilesInDirectory(File dir, File[] files)
{
try
{
for (File file : files)
{
if (file != null)
file.delete();
}
removeEmptyDirectories(dir);
return true;
} catch (Exception e)
{
e.printStackTrace();
return false;
}
}
}