[android][storage] storage implementation using react native promises
This commit is contained in:
parent
b78ebc7de2
commit
e16b788ad7
@ -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());
|
||||
|
Loading…
x
Reference in New Issue
Block a user