2
0
mirror of synced 2025-01-11 22:54:12 +00:00

[android][storage] storage implementation using react native promises

This commit is contained in:
Salakar 2017-03-29 17:50:25 +01:00
parent b78ebc7de2
commit e16b788ad7

View File

@ -7,6 +7,7 @@ import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.List;
import java.util.Map;
import java.util.HashMap;
@ -15,13 +16,12 @@ import android.database.Cursor;
import android.provider.MediaStore;
import android.support.annotation.NonNull;
import com.facebook.react.bridge.Callback;
import com.facebook.react.bridge.Arguments;
import com.facebook.react.bridge.Promise;
import com.facebook.react.bridge.WritableArray;
import com.facebook.react.bridge.WritableMap;
import com.facebook.react.bridge.ReactMethod;
import com.facebook.react.bridge.ReadableMap;
import com.facebook.react.bridge.ReactContext;
import com.facebook.react.bridge.ReactApplicationContext;
import com.facebook.react.bridge.ReactContextBaseJavaModule;
@ -29,8 +29,6 @@ import com.google.android.gms.tasks.Task;
import com.google.android.gms.tasks.OnFailureListener;
import com.google.android.gms.tasks.OnSuccessListener;
import com.google.firebase.database.DataSnapshot;
import com.google.firebase.database.DatabaseError;
import com.google.firebase.storage.StorageException;
import com.google.firebase.storage.StorageTask;
import com.google.firebase.storage.StreamDownloadTask;
@ -54,21 +52,17 @@ public class RNFirebaseStorage extends ReactContextBaseJavaModule {
private static final String PicturesDirectoryPath = "PICTURES_DIRECTORY_PATH";
private static final String TemporaryDirectoryPath = "TEMPORARY_DIRECTORY_PATH";
private static final String CachesDirectoryPath = "CACHES_DIRECTORY_PATH";
private static final String DocumentDirectory = "DOCUMENT_DIRECTORY_PATH";
private static final String FileTypeRegular = "FILETYPE_REGULAR";
private static final String FileTypeDirectory = "FILETYPE_DIRECTORY";
private static final String STORAGE_EVENT = "storage_event";
private static final String STORAGE_ERROR = "storage_error";
private static final String STORAGE_STATE_CHANGED = "state_changed";
private static final String STORAGE_UPLOAD_SUCCESS = "upload_success";
private static final String STORAGE_UPLOAD_FAILURE = "upload_failure";
private static final String STORAGE_DOWNLOAD_SUCCESS = "download_success";
private static final String STORAGE_DOWNLOAD_FAILURE = "download_failure";
private ReactContext mReactContext;
public RNFirebaseStorage(ReactApplicationContext reactContext) {
super(reactContext);
@ -81,35 +75,65 @@ public class RNFirebaseStorage extends ReactContextBaseJavaModule {
}
/**
* Check if we can write to storage, usually false if no permission set on manifest
*
* @return
*/
public boolean isExternalStorageWritable() {
boolean mExternalStorageAvailable;
boolean mExternalStorageWritable;
String state = Environment.getExternalStorageState();
return Environment.MEDIA_MOUNTED.equals(state);
if (Environment.MEDIA_MOUNTED.equals(state)) {
// we can read and write the media
mExternalStorageAvailable = mExternalStorageWritable = true;
} else if (Environment.MEDIA_MOUNTED_READ_ONLY.equals(state)) {
// we can only read the media
mExternalStorageAvailable = true;
mExternalStorageWritable = false;
} else {
// something else is wrong. It may be one of many other states, but all we need
// to know is we can neither read nor write
mExternalStorageAvailable = mExternalStorageWritable = false;
}
return mExternalStorageAvailable && mExternalStorageWritable;
}
/**
* delete
*
* @param path
* @param promise
* @url https://firebase.google.com/docs/reference/js/firebase.storage.Reference#delete
*/
@ReactMethod
public void delete(final String path,
final Callback callback) {
public void delete(final String path, final Promise promise) {
StorageReference reference = this.getReference(path);
reference.delete().addOnSuccessListener(new OnSuccessListener<Void>() {
@Override
public void onSuccess(Void aVoid) {
WritableMap data = Arguments.createMap();
data.putString("success", "success");
data.putString("path", path);
callback.invoke(null, data);
promise.resolve(null);
}
}).addOnFailureListener(new OnFailureListener() {
@Override
public void onFailure(Exception exception) {
callback.invoke(makeErrorPayload(1, exception));
public void onFailure(@NonNull Exception exception) {
promiseRejectStorageException(promise, exception);
}
});
}
/**
* getDownloadURL
*
* @param path
* @param promise
* @url https://firebase.google.com/docs/reference/js/firebase.storage.Reference#getDownloadURL
*/
@ReactMethod
public void getDownloadURL(final String path,
final Callback callback) {
Log.d(TAG, "Download url for remote path: " + path);
public void getDownloadURL(final String path, final Promise promise) {
Log.d(TAG, "getDownloadURL path " + path);
final StorageReference reference = this.getReference(path);
Task<Uri> downloadTask = reference.getDownloadUrl();
@ -117,70 +141,88 @@ public class RNFirebaseStorage extends ReactContextBaseJavaModule {
.addOnSuccessListener(new OnSuccessListener<Uri>() {
@Override
public void onSuccess(Uri uri) {
callback.invoke(null, uri.toString());
promise.resolve(uri.toString());
}
})
.addOnFailureListener(new OnFailureListener() {
@Override
public void onFailure(@NonNull Exception exception) {
callback.invoke(makeErrorPayload(1, exception));
promiseRejectStorageException(promise, exception);
}
});
}
/**
* getMetadata
*
* @param path
* @param promise
* @url https://firebase.google.com/docs/reference/js/firebase.storage.Reference#getMetadata
*/
@ReactMethod
public void getMetadata(final String path,
final Callback callback) {
public void getMetadata(final String path, final Promise promise) {
StorageReference reference = this.getReference(path);
reference.getMetadata().addOnSuccessListener(new OnSuccessListener<StorageMetadata>() {
@Override
public void onSuccess(StorageMetadata storageMetadata) {
WritableMap data = getMetadataAsMap(storageMetadata);
callback.invoke(null, data);
promise.resolve(getMetadataAsMap(storageMetadata));
}
}).addOnFailureListener(new OnFailureListener() {
@Override
public void onFailure(Exception exception) {
callback.invoke(makeErrorPayload(1, exception));
public void onFailure(@NonNull Exception exception) {
promiseRejectStorageException(promise, exception);
}
});
}
/**
* updateMetadata
*
* @param path
* @param metadata
* @param promise
* @url https://firebase.google.com/docs/reference/js/firebase.storage.Reference#updateMetadata
*/
@ReactMethod
public void updateMetadata(final String path,
final ReadableMap metadata,
final Callback callback) {
public void updateMetadata(final String path, final ReadableMap metadata, final Promise promise) {
StorageReference reference = this.getReference(path);
StorageMetadata md = buildMetadataFromMap(metadata);
reference.updateMetadata(md).addOnSuccessListener(new OnSuccessListener<StorageMetadata>() {
@Override
public void onSuccess(StorageMetadata storageMetadata) {
WritableMap data = getMetadataAsMap(storageMetadata);
callback.invoke(null, data);
promise.resolve(data);
}
}).addOnFailureListener(new OnFailureListener() {
@Override
public void onFailure(Exception exception) {
callback.invoke(makeErrorPayload(1, exception));
public void onFailure(@NonNull Exception exception) {
promiseRejectStorageException(promise, exception);
}
});
}
/**
* downloadFile
*
* @param path
* @param localPath
* @param promise
* @url https://firebase.google.com/docs/reference/js/firebase.storage.Reference#downloadFile
*/
@ReactMethod
public void downloadFile(final String path,
final String localPath,
final Callback callback) {
public void downloadFile(final String path, final String localPath, final Promise promise) {
if (!isExternalStorageWritable()) {
Log.w(TAG, "downloadFile failed: external storage not writable");
WritableMap error = Arguments.createMap();
final int errorCode = 1;
error.putDouble("code", errorCode);
error.putString("description", "downloadFile failed: external storage not writable");
callback.invoke(error);
promise.reject(
"storage/invalid-device-file-path",
"The specified device file path is invalid or is restricted."
);
return;
}
Log.d(TAG, "downloadFile from remote path: " + path);
Log.d(TAG, "downloadFile path: " + path);
StorageReference reference = this.getReference(path);
reference.getStream(new StreamDownloadTask.StreamProcessor() {
@ -190,56 +232,101 @@ public class RNFirebaseStorage extends ReactContextBaseJavaModule {
String pathMinusFileName = indexOfLastSlash > 0 ? localPath.substring(0, indexOfLastSlash) + "/" : "/";
String filename = indexOfLastSlash > 0 ? localPath.substring(indexOfLastSlash + 1) : localPath;
File fileWithJustPath = new File(pathMinusFileName);
fileWithJustPath.mkdirs();
// directoriesCreated assignment for not consumed warning
Boolean directoriesCreated = fileWithJustPath.mkdirs();
File fileWithFullPath = new File(pathMinusFileName, filename);
FileOutputStream output = new FileOutputStream(fileWithFullPath);
int bufferSize = 1024;
byte[] buffer = new byte[bufferSize];
int len = 0;
int len;
while ((len = inputStream.read(buffer)) != -1) {
output.write(buffer, 0, len);
}
output.close();
}
}).addOnProgressListener(new OnProgressListener<StreamDownloadTask.TaskSnapshot>() {
@Override
public void onProgress(StreamDownloadTask.TaskSnapshot taskSnapshot) {
Log.d(TAG, "Got download progress " + taskSnapshot);
Log.d(TAG, "downloadFile progress " + taskSnapshot);
WritableMap event = getDownloadTaskAsMap(taskSnapshot);
handleStorageEvent(STORAGE_STATE_CHANGED, path, event);
sendJSEvent(STORAGE_STATE_CHANGED, path, event);
}
}).addOnPausedListener(new OnPausedListener<StreamDownloadTask.TaskSnapshot>() {
@Override
public void onPaused(StreamDownloadTask.TaskSnapshot taskSnapshot) {
Log.d(TAG, "Download is paused " + taskSnapshot);
Log.d(TAG, "downloadFile paused " + taskSnapshot);
WritableMap event = getDownloadTaskAsMap(taskSnapshot);
handleStorageEvent(STORAGE_STATE_CHANGED, path, event);
sendJSEvent(STORAGE_STATE_CHANGED, path, event);
}
}).addOnSuccessListener(new OnSuccessListener<StreamDownloadTask.TaskSnapshot>() {
@Override
public void onSuccess(StreamDownloadTask.TaskSnapshot taskSnapshot) {
Log.d(TAG, "Successfully downloaded file " + taskSnapshot);
Log.d(TAG, "downloadFile success" + taskSnapshot);
WritableMap resp = getDownloadTaskAsMap(taskSnapshot);
handleStorageEvent(STORAGE_DOWNLOAD_SUCCESS, path, resp);
//TODO: A little hacky, but otherwise throws a not consumed exception
sendJSEvent(STORAGE_DOWNLOAD_SUCCESS, path, resp);
resp = getDownloadTaskAsMap(taskSnapshot);
callback.invoke(null, resp);
promise.resolve(resp);
}
}).addOnFailureListener(new OnFailureListener() {
@Override
public void onFailure(@NonNull Exception exception) {
Log.e(TAG, "Failed to download file " + exception.getMessage());
//TODO: JS Error event
callback.invoke(makeErrorPayload(1, exception));
Log.e(TAG, "downloadFile failure " + exception.getMessage());
// TODO sendJS error event
promiseRejectStorageException(promise, exception);
}
});
}
/**
* setMaxDownloadRetryTime
*
* @param milliseconds
* @url https://firebase.google.com/docs/reference/js/firebase.storage.Storage#setMaxDownloadRetryTime
*/
@ReactMethod
public void putFile(final String path, final String localPath, final ReadableMap metadata, final Callback callback) {
public void setMaxDownloadRetryTime(final double milliseconds) {
FirebaseStorage.getInstance().setMaxDownloadRetryTimeMillis((long) milliseconds);
}
/**
* setMaxOperationRetryTime
*
* @param milliseconds
* @url https://firebase.google.com/docs/reference/js/firebase.storage.Storage#setMaxOperationRetryTime
*/
@ReactMethod
public void setMaxOperationRetryTime(final double milliseconds) {
FirebaseStorage.getInstance().setMaxOperationRetryTimeMillis((long) milliseconds);
}
/**
* setMaxUploadRetryTime
*
* @param milliseconds
* @url https://firebase.google.com/docs/reference/js/firebase.storage.Storage#setMaxUploadRetryTime
*/
@ReactMethod
public void setMaxUploadRetryTime(final double milliseconds) {
FirebaseStorage.getInstance().setMaxUploadRetryTimeMillis((long) milliseconds);
}
/**
* putFile
*
* @param path
* @param localPath
* @param metadata
* @param promise
* @url https://firebase.google.com/docs/reference/js/firebase.storage.Reference#putFile
*/
@ReactMethod
public void putFile(final String path, final String localPath, final ReadableMap metadata, final Promise promise) {
StorageReference reference = this.getReference(path);
Log.i(TAG, "Upload file: " + localPath + " to " + path);
Log.i(TAG, "putFile: " + localPath + " to " + path);
try {
Uri file;
@ -259,60 +346,46 @@ public class RNFirebaseStorage extends ReactContextBaseJavaModule {
@Override
public void onFailure(@NonNull Exception exception) {
// handle unsuccessful uploads
Log.e(TAG, "Failed to upload file " + exception.getMessage());
//TODO: JS Error event
callback.invoke(makeErrorPayload(1, exception));
Log.e(TAG, "putFile failure " + exception.getMessage());
// TODO sendJS error event
promiseRejectStorageException(promise, exception);
}
})
.addOnSuccessListener(new OnSuccessListener<UploadTask.TaskSnapshot>() {
@Override
public void onSuccess(UploadTask.TaskSnapshot taskSnapshot) {
Log.d(TAG, "Successfully uploaded file " + taskSnapshot);
Log.d(TAG, "putFile success " + taskSnapshot);
WritableMap resp = getUploadTaskAsMap(taskSnapshot);
handleStorageEvent(STORAGE_UPLOAD_SUCCESS, path, resp);
//TODO: A little hacky, but otherwise throws a not consumed exception
sendJSEvent(STORAGE_UPLOAD_SUCCESS, path, resp);
resp = getUploadTaskAsMap(taskSnapshot);
callback.invoke(null, resp);
promise.resolve(resp);
}
})
.addOnProgressListener(new OnProgressListener<UploadTask.TaskSnapshot>() {
@Override
public void onProgress(UploadTask.TaskSnapshot taskSnapshot) {
Log.d(TAG, "Got upload progress " + taskSnapshot);
WritableMap event = getUploadTaskAsMap(taskSnapshot);
handleStorageEvent(STORAGE_STATE_CHANGED, path, event);
Log.d(TAG, "putFile progress " + taskSnapshot);
sendJSEvent(STORAGE_STATE_CHANGED, path, getUploadTaskAsMap(taskSnapshot));
}
})
.addOnPausedListener(new OnPausedListener<UploadTask.TaskSnapshot>() {
@Override
public void onPaused(UploadTask.TaskSnapshot taskSnapshot) {
Log.d(TAG, "Upload is paused " + taskSnapshot);
Log.d(TAG, "putFile paused " + taskSnapshot);
WritableMap event = getUploadTaskAsMap(taskSnapshot);
handleStorageEvent(STORAGE_STATE_CHANGED, path, event);
sendJSEvent(STORAGE_STATE_CHANGED, path, event);
}
});
} catch (Exception ex) {
final int errorCode = 2;
callback.invoke(makeErrorPayload(errorCode, ex));
} catch (Exception exception) {
promiseRejectStorageException(promise, exception);
}
}
//Firebase.Storage methods
@ReactMethod
public void setMaxDownloadRetryTime(final double milliseconds) {
FirebaseStorage.getInstance().setMaxDownloadRetryTimeMillis((long) milliseconds);
}
@ReactMethod
public void setMaxOperationRetryTime(final double milliseconds) {
FirebaseStorage.getInstance().setMaxOperationRetryTimeMillis((long) milliseconds);
}
@ReactMethod
public void setMaxUploadRetryTime(final double milliseconds) {
FirebaseStorage.getInstance().setMaxUploadRetryTimeMillis((long) milliseconds);
}
/**
* Internal helper to detect if ref is from url or a path.
* @param path
* @return
*/
private StorageReference getReference(String path) {
if (path.startsWith("url::")) {
String url = path.substring(5);
@ -322,6 +395,32 @@ public class RNFirebaseStorage extends ReactContextBaseJavaModule {
}
}
/**
* Internal helper to convert content:// uri's to a real path
* @param uri
* @return
*/
private String getRealPathFromURI(final String uri) {
Cursor cursor = null;
try {
String[] proj = {MediaStore.Images.Media.DATA};
cursor = getReactApplicationContext().getContentResolver().query(Uri.parse(uri), proj, null, null, null);
int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
cursor.moveToFirst();
return cursor.getString(column_index);
} finally {
if (cursor != null) {
cursor.close();
}
}
}
/**
* Converts a RN ReadableMap into a StorageMetadata instance
*
* @param metadata
* @return
*/
private StorageMetadata buildMetadataFromMap(ReadableMap metadata) {
StorageMetadata.Builder metadataBuilder = new StorageMetadata.Builder();
Map<String, Object> m = Utils.recursivelyDeconstructReadableMap(metadata);
@ -333,6 +432,55 @@ public class RNFirebaseStorage extends ReactContextBaseJavaModule {
return metadataBuilder.build();
}
/**
* Convert an download task snapshot to a RN WritableMAP
*
* @param taskSnapshot
* @return
*/
private WritableMap getDownloadTaskAsMap(final StreamDownloadTask.TaskSnapshot taskSnapshot) {
WritableMap resp = Arguments.createMap();
resp.putDouble("bytesTransferred", taskSnapshot.getBytesTransferred());
resp.putString("ref", taskSnapshot.getStorage().getPath());
resp.putString("state", this.getTaskStatus(taskSnapshot.getTask()));
resp.putDouble("totalBytes", taskSnapshot.getTotalByteCount());
return resp;
}
/**
* Convert an upload task snapshot to a RN WritableMAP
*
* @param taskSnapshot
* @return
*/
private WritableMap getUploadTaskAsMap(UploadTask.TaskSnapshot taskSnapshot) {
WritableMap resp = Arguments.createMap();
if (taskSnapshot != null) {
resp.putDouble("bytesTransferred", taskSnapshot.getBytesTransferred());
resp.putString("downloadUrl", taskSnapshot.getDownloadUrl() != null ? taskSnapshot.getDownloadUrl().toString() : null);
StorageMetadata d = taskSnapshot.getMetadata();
if (d != null) {
WritableMap metadata = getMetadataAsMap(d);
resp.putMap("metadata", metadata);
}
resp.putString("ref", taskSnapshot.getStorage().getPath());
resp.putString("state", this.getTaskStatus(taskSnapshot.getTask()));
resp.putDouble("totalBytes", taskSnapshot.getTotalByteCount());
}
return resp;
}
/**
* Converts storageMetadata into a map
*
* @param storageMetadata
* @return
*/
private WritableMap getMetadataAsMap(StorageMetadata storageMetadata) {
WritableMap metadata = Arguments.createMap();
metadata.putString("bucket", storageMetadata.getBucket());
@ -351,9 +499,14 @@ public class RNFirebaseStorage extends ReactContextBaseJavaModule {
metadata.putString("contentType", storageMetadata.getContentType());
WritableArray downloadURLs = Arguments.createArray();
for (Uri uri : storageMetadata.getDownloadUrls()) {
List<Uri> _downloadURLS = storageMetadata.getDownloadUrls();
if (_downloadURLS != null) {
for (Uri uri : _downloadURLS) {
downloadURLs.pushString(uri.getPath());
}
}
metadata.putArray("downloadURLs", downloadURLs);
WritableMap customMetadata = Arguments.createMap();
@ -365,99 +518,113 @@ public class RNFirebaseStorage extends ReactContextBaseJavaModule {
return metadata;
}
private String getRealPathFromURI(final String uri) {
Cursor cursor = null;
try {
String[] proj = {MediaStore.Images.Media.DATA};
cursor = getReactApplicationContext().getContentResolver().query(Uri.parse(uri), proj, null, null, null);
int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
cursor.moveToFirst();
return cursor.getString(column_index);
} finally {
if (cursor != null) {
cursor.close();
}
}
}
private WritableMap getDownloadTaskAsMap(final StreamDownloadTask.TaskSnapshot taskSnapshot) {
WritableMap resp = Arguments.createMap();
resp.putDouble("bytesTransferred", taskSnapshot.getBytesTransferred());
resp.putString("ref", taskSnapshot.getStorage().getPath());
resp.putString("state", this.getTaskStatus(taskSnapshot.getTask()));
resp.putDouble("totalBytes", taskSnapshot.getTotalByteCount());
return resp;
}
private WritableMap getUploadTaskAsMap(final UploadTask.TaskSnapshot taskSnapshot) {
StorageMetadata d = taskSnapshot.getMetadata();
WritableMap resp = Arguments.createMap();
resp.putDouble("bytesTransferred", taskSnapshot.getBytesTransferred());
resp.putString("downloadUrl", taskSnapshot.getDownloadUrl() != null ? taskSnapshot.getDownloadUrl().toString() : null);
resp.putString("ref", taskSnapshot.getStorage().getPath());
resp.putString("state", this.getTaskStatus(taskSnapshot.getTask()));
resp.putDouble("totalBytes", taskSnapshot.getTotalByteCount());
if (taskSnapshot.getMetadata() != null) {
WritableMap metadata = getMetadataAsMap(taskSnapshot.getMetadata());
resp.putMap("metadata", metadata);
}
return resp;
}
/**
* Returns the task status as string
*
* @param task
* @return
*/
private String getTaskStatus(StorageTask<?> task) {
if (task.isInProgress()) {
return "RUNNING";
return "running";
} else if (task.isPaused()) {
return "PAUSED";
return "paused";
} else if (task.isSuccessful() || task.isComplete()) {
return "SUCCESS";
return "success";
} else if (task.isCanceled()) {
return "CANCELLED";
return "cancelled";
} else if (task.getException() != null) {
return "ERROR";
return "error";
} else {
return "UNKNOWN";
return "unknown";
}
}
private void handleStorageEvent(final String name, final String path, WritableMap body) {
WritableMap evt = Arguments.createMap();
evt.putString("eventName", name);
evt.putString("path", path);
evt.putMap("body", body);
/**
* @param name
* @param path
* @param body
*/
private void sendJSEvent(final String name, final String path, WritableMap body) {
WritableMap event = Arguments.createMap();
Utils.sendEvent(this.getReactApplicationContext(), STORAGE_EVENT, evt);
event.putString("eventName", name);
event.putString("path", path);
event.putMap("body", body);
Utils.sendEvent(this.getReactApplicationContext(), STORAGE_EVENT, event);
}
private void handleStorageError(final String path, final StorageException error) {
WritableMap body = Arguments.createMap();
body.putString("path", path);
body.putString("message", error.getMessage());
/**
* Reject a promise with a web sdk error code
*
* @param promise
* @param exception
*/
private void promiseRejectStorageException(Promise promise, Exception exception) {
String code = "storage/unknown";
String message = exception.getMessage();
WritableMap evt = Arguments.createMap();
evt.putString("eventName", STORAGE_ERROR);
evt.putMap("body", body);
try {
StorageException storageException = (StorageException) exception;
Utils.sendEvent(this.getReactApplicationContext(), STORAGE_ERROR, evt);
switch (storageException.getErrorCode()) {
case StorageException.ERROR_UNKNOWN:
code = "storage/unknown";
message = "An unknown error has occurred.";
break;
case StorageException.ERROR_OBJECT_NOT_FOUND:
code = "storage/object-not-found";
message = "No object exists at the desired reference.";
break;
case StorageException.ERROR_BUCKET_NOT_FOUND:
code = "storage/bucket-not-found";
message = "No bucket is configured for Firebase Storage.";
break;
case StorageException.ERROR_PROJECT_NOT_FOUND:
code = "storage/project-not-found";
message = "No project is configured for Firebase Storage.";
break;
case StorageException.ERROR_QUOTA_EXCEEDED:
code = "storage/quota-exceeded";
message = "Quota on your Firebase Storage bucket has been exceeded.";
break;
case StorageException.ERROR_NOT_AUTHENTICATED:
code = "storage/unauthenticated";
message = "User is unauthenticated. Authenticate and try again.";
break;
case StorageException.ERROR_NOT_AUTHORIZED:
code = "storage/unauthorized";
message = "User is not authorized to perform the desired action.";
break;
case StorageException.ERROR_RETRY_LIMIT_EXCEEDED:
code = "storage/retry-limit-exceeded";
message = "The maximum time limit on an operation (upload, download, delete, etc.) has been exceeded.";
break;
case StorageException.ERROR_INVALID_CHECKSUM:
code = "storage/non-matching-checksum";
message = "File on the client does not match the checksum of the file received by the server.";
break;
case StorageException.ERROR_CANCELED:
code = "storage/cancelled";
message = "User cancelled the operation.";
break;
}
} finally {
promise.reject(code, message, exception);
}
}
private WritableMap makeErrorPayload(double code, Exception ex) {
WritableMap error = Arguments.createMap();
error.putDouble("code", code);
error.putString("message", ex.getMessage());
return error;
}
/**
* Constants bootstrapped on react native app boot
* e.g. firebase.storage.Native.DOCUMENT_DIRECTORY_PATH
*
* @return
*/
@Override
public Map<String, Object> getConstants() {
final Map<String, Object> constants = new HashMap<>();
constants.put(DocumentDirectory, 0);
constants.put(DocumentDirectoryPath, this.getReactApplicationContext().getFilesDir().getAbsolutePath());
constants.put(TemporaryDirectoryPath, null);
constants.put(PicturesDirectoryPath, Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES).getAbsolutePath());