From a6b48b0aaa040df6dcf01ef967a871c47e71831e Mon Sep 17 00:00:00 2001 From: Dmitry Patsura Date: Fri, 30 Jun 2017 10:08:20 +0900 Subject: [PATCH 1/9] [Android] Support multipart form upload, refs #33 --- .../main/java/com/vydia/UploaderModule.java | 121 ++++++++++++------ 1 file changed, 84 insertions(+), 37 deletions(-) diff --git a/android/src/main/java/com/vydia/UploaderModule.java b/android/src/main/java/com/vydia/UploaderModule.java index 2c7b9f70..9e4a6858 100644 --- a/android/src/main/java/com/vydia/UploaderModule.java +++ b/android/src/main/java/com/vydia/UploaderModule.java @@ -18,6 +18,8 @@ import com.facebook.react.modules.core.DeviceEventManagerModule; import net.gotev.uploadservice.BinaryUploadRequest; +import net.gotev.uploadservice.HttpUploadRequest; +import net.gotev.uploadservice.MultipartUploadRequest; import net.gotev.uploadservice.ServerResponse; import net.gotev.uploadservice.UploadInfo; import net.gotev.uploadservice.UploadNotificationConfig; @@ -108,17 +110,35 @@ public void startUpload(ReadableMap options, final Promise promise) { return; } } + if (options.hasKey("headers") && options.getType("headers") != ReadableType.Map) { promise.reject(new IllegalArgumentException("headers must be a hash.")); return; } + if (options.hasKey("notification") && options.getType("notification") != ReadableType.Map) { promise.reject(new IllegalArgumentException("notification must be a hash.")); return; } + String requestType = "raw"; + + if (options.hasKey("type")) { + requestType = options.getString("type"); + if (requestType == null) { + promise.reject(new IllegalArgumentException("type must be string.")); + return; + } + + if (!requestType.equals("raw") || !requestType.equals("multipart")) { + promise.reject(new IllegalArgumentException("type should be string: raw or multipart.")); + return; + } + } + WritableMap notification = new WritableNativeMap(); notification.putBoolean("enabled", true); + if (options.hasKey("notification")) { notification.merge(options.getMap("notification")); } @@ -126,48 +146,74 @@ public void startUpload(ReadableMap options, final Promise promise) { String url = options.getString("url"); String filePath = options.getString("path"); String method = options.hasKey("method") && options.getType("method") == ReadableType.String ? options.getString("method") : "POST"; + final String customUploadId = options.hasKey("customUploadId") && options.getType("method") == ReadableType.String ? options.getString("customUploadId") : null; + try { - final BinaryUploadRequest request = (BinaryUploadRequest) new BinaryUploadRequest(this.getReactApplicationContext(), url) - .setMethod(method) - .setFileToUpload(filePath) - .setMaxRetries(2) - .setDelegate(new UploadStatusDelegate() { - @Override - public void onProgress(Context context, UploadInfo uploadInfo) { - WritableMap params = Arguments.createMap(); - params.putString("id", customUploadId != null ? customUploadId : uploadInfo.getUploadId()); - params.putInt("progress", uploadInfo.getProgressPercent()); //0-100 - sendEvent("progress", params); - } - - @Override - public void onError(Context context, UploadInfo uploadInfo, Exception exception) { - WritableMap params = Arguments.createMap(); - params.putString("id", customUploadId != null ? customUploadId : uploadInfo.getUploadId()); - params.putString("error", exception.getMessage()); - sendEvent("error", params); - } - - @Override - public void onCompleted(Context context, UploadInfo uploadInfo, ServerResponse serverResponse) { - WritableMap params = Arguments.createMap(); - params.putString("id", customUploadId != null ? customUploadId : uploadInfo.getUploadId()); - params.putInt("responseCode", serverResponse.getHttpCode()); - params.putString("responseBody", serverResponse.getBodyAsString()); - sendEvent("completed", params); - } - - @Override - public void onCancelled(Context context, UploadInfo uploadInfo) { - WritableMap params = Arguments.createMap(); - params.putString("id", customUploadId != null ? customUploadId : uploadInfo.getUploadId()); - sendEvent("cancelled", params); - } - }); + UploadStatusDelegate statusDelegate = new UploadStatusDelegate() { + @Override + public void onProgress(Context context, UploadInfo uploadInfo) { + WritableMap params = Arguments.createMap(); + params.putString("id", customUploadId != null ? customUploadId : uploadInfo.getUploadId()); + params.putInt("progress", uploadInfo.getProgressPercent()); //0-100 + sendEvent("progress", params); + } + + @Override + public void onError(Context context, UploadInfo uploadInfo, Exception exception) { + WritableMap params = Arguments.createMap(); + params.putString("id", customUploadId != null ? customUploadId : uploadInfo.getUploadId()); + params.putString("error", exception.getMessage()); + sendEvent("error", params); + } + + @Override + public void onCompleted(Context context, UploadInfo uploadInfo, ServerResponse serverResponse) { + WritableMap params = Arguments.createMap(); + params.putString("id", customUploadId != null ? customUploadId : uploadInfo.getUploadId()); + params.putInt("responseCode", serverResponse.getHttpCode()); + params.putString("responseBody", serverResponse.getBodyAsString()); + sendEvent("completed", params); + } + + @Override + public void onCancelled(Context context, UploadInfo uploadInfo) { + WritableMap params = Arguments.createMap(); + params.putString("id", customUploadId != null ? customUploadId : uploadInfo.getUploadId()); + sendEvent("cancelled", params); + } + }; + + HttpUploadRequest request; + + if (requestType.equals("raw")) { + request = new BinaryUploadRequest(this.getReactApplicationContext(), url) + .setMethod(method) + .setFileToUpload(filePath) + .setMaxRetries(2) + .setDelegate(statusDelegate); + } else { + if (!options.hasKey("field")) { + promise.reject(new IllegalArgumentException("field is required field for multipart type.")); + return; + } + + if (options.getType("field") != ReadableType.String) { + promise.reject(new IllegalArgumentException("field must be string.")); + return; + } + + request = new MultipartUploadRequest(this.getReactApplicationContext(), url) + .setMethod(method) + .addFileToUpload(filePath, options.getString("field")) + .setMaxRetries(2) + .setDelegate(statusDelegate); + } + if (notification.getBoolean("enabled")) { request.setNotificationConfig(new UploadNotificationConfig()); } + if (options.hasKey("headers")) { ReadableMap headers = options.getMap("headers"); ReadableMapKeySetIterator keys = headers.keySetIterator(); @@ -180,6 +226,7 @@ public void onCancelled(Context context, UploadInfo uploadInfo) { request.addHeader(key, headers.getString(key)); } } + String uploadId = request.startUpload(); promise.resolve(customUploadId != null ? customUploadId : uploadId); } catch (Exception exc) { From 70cd259955de004e1a7cda30768553d4d9bf6e1b Mon Sep 17 00:00:00 2001 From: Dmitry Patsura Date: Fri, 30 Jun 2017 10:09:59 +0900 Subject: [PATCH 2/9] [JS] Flow: StartUploadArgs - add fields: type & field --- index.js | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/index.js b/index.js index fc96d15e..ce96511e 100644 --- a/index.js +++ b/index.js @@ -13,6 +13,10 @@ export type NotificationArgs = { export type StartUploadArgs = { url: string, path: string, + // Optional, because raw is default + type?: 'raw' | 'multipart', + // This option is needed for multipart type + field?: string, headers?: Object, notification?: NotificationArgs } From 55f140265fbe53504e15044ffe040f5e902561f1 Mon Sep 17 00:00:00 2001 From: Dmitry Patsura Date: Fri, 30 Jun 2017 12:49:42 +0900 Subject: [PATCH 3/9] [iOS] Start working on multipart support --- ios/VydiaRNFileUploader.m | 20 ++++++++++++++++---- 1 file changed, 16 insertions(+), 4 deletions(-) diff --git a/ios/VydiaRNFileUploader.m b/ios/VydiaRNFileUploader.m index a218252a..2117a090 100644 --- a/ios/VydiaRNFileUploader.m +++ b/ios/VydiaRNFileUploader.m @@ -113,15 +113,18 @@ - (NSString *)guessMIMETypeFromFileName: (NSString *)fileName { { thisUploadId = uploadId++; } + NSString *uploadUrl = options[@"url"]; NSString *fileURI = options[@"path"]; - NSString *method = options[@"method"]; - NSString *customUploadId = options[@"customUploadId"]; + NSString *method = options[@"method"] ?: @"POST"; + NSString *uploadType = options[@"type"] ?: @"raw"; + NSString *customUploadId = options[@"customUploadId"]; NSDictionary *headers = options[@"headers"]; - + @try { NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString: uploadUrl]]; - request.HTTPMethod = method ? method : @"POST"; + [request setHTTPMethod: method]; + [headers enumerateKeysAndObjectsUsingBlock:^(id _Nonnull key, id _Nonnull val, BOOL * _Nonnull stop) { if ([val respondsToSelector:@selector(stringValue)]) { val = [val stringValue]; @@ -130,8 +133,17 @@ - (NSString *)guessMIMETypeFromFileName: (NSString *)fileName { [request setValue:val forHTTPHeaderField:key]; } }]; + + if ([uploadType isEqualToString:@"multipart"]) { + NSString *uuidStr = [[NSUUID UUID] UUIDString]; + [request addValue:[NSString stringWithFormat:@"multipart/form-data; boundary=%@", uuidStr] forHTTPHeaderField:@"Content-Type"]; + } else { + + } + NSURLSessionDataTask *uploadTask = [[self urlSession:thisUploadId] uploadTaskWithRequest:request fromFile:[NSURL URLWithString: fileURI]]; uploadTask.taskDescription = customUploadId ? customUploadId : [NSString stringWithFormat:@"%i", thisUploadId]; + [uploadTask resume]; resolve(uploadTask.taskDescription); } From 1206c600c40a8e1b74ae988552912868a72f3543 Mon Sep 17 00:00:00 2001 From: Dmitry Patsura Date: Sun, 2 Jul 2017 16:41:41 +0900 Subject: [PATCH 4/9] [iOS] Implement support for multipart form, refs #33 --- ios/VydiaRNFileUploader.m | 43 ++++++++++++++++++++++++++++++++++----- 1 file changed, 38 insertions(+), 5 deletions(-) diff --git a/ios/VydiaRNFileUploader.m b/ios/VydiaRNFileUploader.m index 2117a090..744bb4fe 100644 --- a/ios/VydiaRNFileUploader.m +++ b/ios/VydiaRNFileUploader.m @@ -94,7 +94,6 @@ - (NSString *)guessMIMETypeFromFileName: (NSString *)fileName { return (__bridge NSString *)(MIMEType); } - /* * Starts a file upload. * Options are passed in as the first argument as a js hash: @@ -113,11 +112,12 @@ - (NSString *)guessMIMETypeFromFileName: (NSString *)fileName { { thisUploadId = uploadId++; } - + NSString *uploadUrl = options[@"url"]; NSString *fileURI = options[@"path"]; NSString *method = options[@"method"] ?: @"POST"; NSString *uploadType = options[@"type"] ?: @"raw"; + NSString *fieldName = options[@"field"]; NSString *customUploadId = options[@"customUploadId"]; NSDictionary *headers = options[@"headers"]; @@ -134,14 +134,21 @@ - (NSString *)guessMIMETypeFromFileName: (NSString *)fileName { } }]; + NSURLSessionDataTask *uploadTask; + if ([uploadType isEqualToString:@"multipart"]) { NSString *uuidStr = [[NSUUID UUID] UUIDString]; - [request addValue:[NSString stringWithFormat:@"multipart/form-data; boundary=%@", uuidStr] forHTTPHeaderField:@"Content-Type"]; - } else { + [request setValue:[NSString stringWithFormat:@"multipart/form-data; boundary=%@", uuidStr] forHTTPHeaderField:@"Content-Type"]; + + NSData *httpBody = [self createBodyWithBoundary:uuidStr path:fileURI fieldName:fieldName]; + [request setHTTPBody: httpBody]; + // I am sorry about warning, but Upload tasks from NSData are not supported in background sessions. + uploadTask = [[self urlSession:thisUploadId] uploadTaskWithRequest:request fromData: nil]; + } else { + uploadTask = [[self urlSession:thisUploadId] uploadTaskWithRequest:request fromFile:[NSURL URLWithString: fileURI]]; } - NSURLSessionDataTask *uploadTask = [[self urlSession:thisUploadId] uploadTaskWithRequest:request fromFile:[NSURL URLWithString: fileURI]]; uploadTask.taskDescription = customUploadId ? customUploadId : [NSString stringWithFormat:@"%i", thisUploadId]; [uploadTask resume]; @@ -152,6 +159,32 @@ - (NSString *)guessMIMETypeFromFileName: (NSString *)fileName { } } +- (NSData *)createBodyWithBoundary:(NSString *)boundary + path:(NSString *)path + fieldName:(NSString *)fieldName { + + NSMutableData *httpBody = [NSMutableData data]; + + // resolve path + NSURL *fileUri = [NSURL URLWithString: path]; + NSString *pathWithoutProtocol = [fileUri path]; + + NSData *data = [[NSFileManager defaultManager] contentsAtPath:pathWithoutProtocol]; + + NSString *filename = [path lastPathComponent]; + NSString *mimetype = [self guessMIMETypeFromFileName:path]; + + [httpBody appendData:[[NSString stringWithFormat:@"--%@\r\n", boundary] dataUsingEncoding:NSUTF8StringEncoding]]; + [httpBody appendData:[[NSString stringWithFormat:@"Content-Disposition: form-data; name=\"%@\"; filename=\"%@\"\r\n", fieldName, filename] dataUsingEncoding:NSUTF8StringEncoding]]; + [httpBody appendData:[[NSString stringWithFormat:@"Content-Type: %@\r\n\r\n", mimetype] dataUsingEncoding:NSUTF8StringEncoding]]; + [httpBody appendData:data]; + [httpBody appendData:[@"\r\n" dataUsingEncoding:NSUTF8StringEncoding]]; + + [httpBody appendData:[[NSString stringWithFormat:@"--%@--\r\n", boundary] dataUsingEncoding:NSUTF8StringEncoding]]; + + return httpBody; +} + - (NSURLSession *)urlSession: (int) thisUploadId{ if(_urlSession == nil) { NSURLSessionConfiguration *sessionConfigurationt = [NSURLSessionConfiguration backgroundSessionConfigurationWithIdentifier:BACKGROUND_SESSION_ID]; From 26fcb1d3b0c0ff177e8c3fcba893cd47130ec240 Mon Sep 17 00:00:00 2001 From: Dmitry Patsura Date: Tue, 4 Jul 2017 13:14:53 +0900 Subject: [PATCH 5/9] [Android] Fix type check, refs #33 --- android/src/main/java/com/vydia/UploaderModule.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/android/src/main/java/com/vydia/UploaderModule.java b/android/src/main/java/com/vydia/UploaderModule.java index 9e4a6858..a0849826 100644 --- a/android/src/main/java/com/vydia/UploaderModule.java +++ b/android/src/main/java/com/vydia/UploaderModule.java @@ -130,7 +130,7 @@ public void startUpload(ReadableMap options, final Promise promise) { return; } - if (!requestType.equals("raw") || !requestType.equals("multipart")) { + if (!requestType.equals("raw") && !requestType.equals("multipart")) { promise.reject(new IllegalArgumentException("type should be string: raw or multipart.")); return; } From a2579f63ae837e0475dd9a493c76c758158ed7c1 Mon Sep 17 00:00:00 2001 From: Dmitry Patsura Date: Tue, 4 Jul 2017 14:52:09 +0900 Subject: [PATCH 6/9] [Android] Remove unneeded comment --- .../src/main/java/com/vydia/UploaderModule.java | 14 ++------------ 1 file changed, 2 insertions(+), 12 deletions(-) diff --git a/android/src/main/java/com/vydia/UploaderModule.java b/android/src/main/java/com/vydia/UploaderModule.java index a0849826..547a93a6 100644 --- a/android/src/main/java/com/vydia/UploaderModule.java +++ b/android/src/main/java/com/vydia/UploaderModule.java @@ -85,18 +85,8 @@ public void getFileInfo(String path, final Promise promise) { } /* - * Starts a file upload. - * Options are passed in as the first argument as a js hash: - * { - * url: string. url to post to. - * path: string. path to the file on the device - * headers: hash of name/value header pairs - * method: HTTP method to use. Default is "POST" - * notification: hash for customizing tray notifiaction - * enabled: boolean to enable/disabled notifications, true by default. - * } - * - * Returns a promise with the string ID of the upload. + * Starts a file upload. + * Returns a promise with the string ID of the upload. */ @ReactMethod public void startUpload(ReadableMap options, final Promise promise) { From d969df2bdf4caea47fc17fa0f9cd378ceac5d079 Mon Sep 17 00:00:00 2001 From: Dmitry Patsura Date: Tue, 4 Jul 2017 14:55:48 +0900 Subject: [PATCH 7/9] [JS] Flow: add customUploadId field to StartUploadArgs --- index.js | 1 + 1 file changed, 1 insertion(+) diff --git a/index.js b/index.js index ce96511e..709c3e84 100644 --- a/index.js +++ b/index.js @@ -17,6 +17,7 @@ export type StartUploadArgs = { type?: 'raw' | 'multipart', // This option is needed for multipart type field?: string, + customUploadId?: string, headers?: Object, notification?: NotificationArgs } From e9ce3bf1696a03f342e2a783fde08fdebb502dc0 Mon Sep 17 00:00:00 2001 From: Dmitry Patsura Date: Tue, 4 Jul 2017 16:24:09 +0900 Subject: [PATCH 8/9] [Android] Setup customUploadId to HttpUploadRequest --- .../main/java/com/vydia/UploaderModule.java | 28 ++++++++++++------- 1 file changed, 18 insertions(+), 10 deletions(-) diff --git a/android/src/main/java/com/vydia/UploaderModule.java b/android/src/main/java/com/vydia/UploaderModule.java index 547a93a6..649708ae 100644 --- a/android/src/main/java/com/vydia/UploaderModule.java +++ b/android/src/main/java/com/vydia/UploaderModule.java @@ -177,11 +177,13 @@ public void onCancelled(Context context, UploadInfo uploadInfo) { HttpUploadRequest request; if (requestType.equals("raw")) { - request = new BinaryUploadRequest(this.getReactApplicationContext(), url) - .setMethod(method) - .setFileToUpload(filePath) - .setMaxRetries(2) - .setDelegate(statusDelegate); + if (customUploadId != null) { + request = new BinaryUploadRequest(this.getReactApplicationContext(), customUploadId, url) + .setFileToUpload(filePath); + } else { + request = new BinaryUploadRequest(this.getReactApplicationContext(), url) + .setFileToUpload(filePath); + } } else { if (!options.hasKey("field")) { promise.reject(new IllegalArgumentException("field is required field for multipart type.")); @@ -193,13 +195,19 @@ public void onCancelled(Context context, UploadInfo uploadInfo) { return; } - request = new MultipartUploadRequest(this.getReactApplicationContext(), url) - .setMethod(method) - .addFileToUpload(filePath, options.getString("field")) - .setMaxRetries(2) - .setDelegate(statusDelegate); + if (customUploadId != null) { + request = new MultipartUploadRequest(this.getReactApplicationContext(), customUploadId, url) + .addFileToUpload(filePath, options.getString("field")); + } else { + request = new MultipartUploadRequest(this.getReactApplicationContext(), url) + .addFileToUpload(filePath, options.getString("field")); + } } + request.setMethod(method) + .setMaxRetries(2) + .setDelegate(statusDelegate); + if (notification.getBoolean("enabled")) { request.setNotificationConfig(new UploadNotificationConfig()); } From dc453ef1d529910cddf5a9a8da4836b70fc32e4f Mon Sep 17 00:00:00 2001 From: Dmitry Patsura Date: Wed, 27 Sep 2017 20:27:10 +0900 Subject: [PATCH 9/9] [Refactoring] Android: remove uneeded checks --- .../main/java/com/vydia/UploaderModule.java | 18 ++++-------------- 1 file changed, 4 insertions(+), 14 deletions(-) diff --git a/android/src/main/java/com/vydia/UploaderModule.java b/android/src/main/java/com/vydia/UploaderModule.java index 649708ae..95a4f159 100644 --- a/android/src/main/java/com/vydia/UploaderModule.java +++ b/android/src/main/java/com/vydia/UploaderModule.java @@ -177,13 +177,8 @@ public void onCancelled(Context context, UploadInfo uploadInfo) { HttpUploadRequest request; if (requestType.equals("raw")) { - if (customUploadId != null) { - request = new BinaryUploadRequest(this.getReactApplicationContext(), customUploadId, url) - .setFileToUpload(filePath); - } else { - request = new BinaryUploadRequest(this.getReactApplicationContext(), url) - .setFileToUpload(filePath); - } + request = new BinaryUploadRequest(this.getReactApplicationContext(), customUploadId, url) + .setFileToUpload(filePath); } else { if (!options.hasKey("field")) { promise.reject(new IllegalArgumentException("field is required field for multipart type.")); @@ -195,13 +190,8 @@ public void onCancelled(Context context, UploadInfo uploadInfo) { return; } - if (customUploadId != null) { - request = new MultipartUploadRequest(this.getReactApplicationContext(), customUploadId, url) - .addFileToUpload(filePath, options.getString("field")); - } else { - request = new MultipartUploadRequest(this.getReactApplicationContext(), url) - .addFileToUpload(filePath, options.getString("field")); - } + request = new MultipartUploadRequest(this.getReactApplicationContext(), customUploadId, url) + .addFileToUpload(filePath, options.getString("field")); } request.setMethod(method)