No Description

RNFetchBlobReqBuilder.m 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  1. //
  2. // RNFetchBlobReqBuilder.m
  3. // RNFetchBlob
  4. //
  5. // Created by Ben Hsieh on 2016/7/9.
  6. // Copyright © 2016年 wkh237. All rights reserved.
  7. //
  8. #import <Foundation/Foundation.h>
  9. #import "RNFetchBlobReqBuilder.h"
  10. #import "RNFetchBlobNetwork.h"
  11. #import "RNFetchBlobConst.h"
  12. #import "RNFetchBlobFS.h"
  13. #import "IOS7Polyfill.h"
  14. #if __has_include(<React/RCTAssert.h>)
  15. #import <React/RCTLog.h>
  16. #else
  17. #import "RCTLog.h"
  18. #endif
  19. @interface RNFetchBlobReqBuilder()
  20. {
  21. }
  22. @end
  23. @implementation RNFetchBlobReqBuilder
  24. // Fetch blob data request
  25. +(void) buildMultipartRequest:(NSDictionary *)options
  26. taskId:(NSString *)taskId
  27. method:(NSString *)method
  28. url:(NSString *)url
  29. headers:(NSDictionary *)headers
  30. form:(NSArray *)form
  31. onComplete:(void(^)(NSURLRequest * req, long bodyLength))onComplete
  32. {
  33. // NSString * encodedUrl = [url stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
  34. NSString * encodedUrl = url;
  35. // send request
  36. __block NSMutableURLRequest *request = [[NSMutableURLRequest alloc] initWithURL:[NSURL URLWithString: encodedUrl]];
  37. __block NSMutableDictionary *mheaders = [[NSMutableDictionary alloc] initWithDictionary:[RNFetchBlobNetwork normalizeHeaders:headers]];
  38. NSTimeInterval timeStamp = [[NSDate date] timeIntervalSince1970];
  39. NSNumber * timeStampObj = [NSNumber numberWithDouble: timeStamp];
  40. // generate boundary
  41. __block NSString * boundary = [NSString stringWithFormat:@"RNFetchBlob%d", timeStampObj];
  42. dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
  43. __block NSMutableData * postData = [[NSMutableData alloc] init];
  44. // combine multipart/form-data body
  45. [[self class] buildFormBody:form boundary:boundary onComplete:^(NSData *formData) {
  46. if(formData != nil) {
  47. [postData appendData:formData];
  48. // close form data
  49. [postData appendData: [[NSString stringWithFormat:@"--%@--\r\n", boundary] dataUsingEncoding:NSUTF8StringEncoding]];
  50. [request setHTTPBody:postData];
  51. }
  52. // set content-length
  53. [mheaders setValue:[NSString stringWithFormat:@"%lu",[postData length]] forKey:@"Content-Length"];
  54. [mheaders setValue:@"100-continue" forKey:@"Expect"];
  55. // appaned boundary to content-type
  56. [mheaders setValue:[NSString stringWithFormat:@"multipart/form-data; charset=utf-8; boundary=%@", boundary] forKey:@"content-type"];
  57. [request setHTTPMethod: method];
  58. [request setAllHTTPHeaderFields:mheaders];
  59. onComplete(request, [formData length]);
  60. }];
  61. });
  62. }
  63. // Fetch blob data request
  64. +(void) buildOctetRequest:(NSDictionary *)options
  65. taskId:(NSString *)taskId
  66. method:(NSString *)method
  67. url:(NSString *)url
  68. headers:(NSDictionary *)headers
  69. body:(NSString *)body
  70. onComplete:(void(^)(__weak NSURLRequest * req, long bodyLength))onComplete
  71. {
  72. // NSString * encodedUrl = [url stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding];
  73. NSString * encodedUrl = url;
  74. // send request
  75. __block NSMutableURLRequest *request = [[NSMutableURLRequest alloc]
  76. initWithURL:[NSURL URLWithString: encodedUrl]];
  77. __block NSMutableDictionary *mheaders = [[NSMutableDictionary alloc] initWithDictionary:[RNFetchBlobNetwork normalizeHeaders:headers]];
  78. // move heavy task to another thread
  79. dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
  80. NSMutableData * blobData;
  81. long size = -1;
  82. // if method is POST or PUT, convert data string format
  83. if([[method lowercaseString] isEqualToString:@"post"] || [[method lowercaseString] isEqualToString:@"put"]) {
  84. // generate octet-stream body
  85. if(body != nil) {
  86. __block NSString * cType = [[self class] getHeaderIgnoreCases:@"content-type" fromHeaders:mheaders];
  87. __block NSString * transferEncoding = [[self class] getHeaderIgnoreCases:@"transfer-encoding" fromHeaders:mheaders];
  88. // when headers does not contain a key named "content-type" (case ignored), use default content type
  89. if(cType == nil)
  90. {
  91. [mheaders setValue:@"application/octet-stream" forKey:@"Content-Type"];
  92. }
  93. // when body is a string contains file path prefix, try load file from the path
  94. if([body hasPrefix:FILE_PREFIX]) {
  95. __block NSString * orgPath = [body substringFromIndex:[FILE_PREFIX length]];
  96. orgPath = [RNFetchBlobFS getPathOfAsset:orgPath];
  97. if([orgPath hasPrefix:AL_PREFIX])
  98. {
  99. [RNFetchBlobFS readFile:orgPath encoding:@"utf8" resolver:nil rejecter:nil onComplete:^(NSData *content) {
  100. [request setHTTPBody:content];
  101. [request setHTTPMethod: method];
  102. [request setAllHTTPHeaderFields:mheaders];
  103. onComplete(request, [content length]);
  104. }];
  105. return;
  106. }
  107. size = [[[NSFileManager defaultManager] attributesOfItemAtPath:orgPath error:nil] fileSize];
  108. if(transferEncoding != nil && [[transferEncoding lowercaseString] isEqualToString:@"chunked"])
  109. {
  110. [request setHTTPBodyStream: [NSInputStream inputStreamWithFileAtPath:orgPath ]];
  111. }
  112. else
  113. {
  114. __block NSData * bodyBytes = [NSData dataWithContentsOfFile:orgPath ];
  115. [request setHTTPBody:bodyBytes];
  116. }
  117. }
  118. // otherwise convert it as BASE64 data string
  119. else {
  120. __block NSString * cType = [[self class]getHeaderIgnoreCases:@"content-type" fromHeaders:mheaders];
  121. // when content-type is application/octet* decode body string using BASE64 decoder
  122. if([[cType lowercaseString] hasPrefix:@"application/octet"] || [[cType lowercaseString] RNFBContainsString:@";base64"])
  123. {
  124. __block NSString * ncType = [[cType stringByReplacingOccurrencesOfString:@";base64" withString:@""]stringByReplacingOccurrencesOfString:@";BASE64" withString:@""];
  125. if([mheaders valueForKey:@"content-type"] != nil)
  126. [mheaders setValue:ncType forKey:@"content-type"];
  127. if([mheaders valueForKey:@"Content-Type"] != nil)
  128. [mheaders setValue:ncType forKey:@"Content-Type"];
  129. blobData = [[NSData alloc] initWithBase64EncodedString:body options:0];
  130. [request setHTTPBody:blobData];
  131. size = [blobData length];
  132. }
  133. // otherwise use the body as-is
  134. else
  135. {
  136. size = [body length];
  137. [request setHTTPBody: [body dataUsingEncoding:NSUTF8StringEncoding]];
  138. }
  139. }
  140. }
  141. }
  142. [request setHTTPMethod: method];
  143. [request setAllHTTPHeaderFields:mheaders];
  144. onComplete(request, size);
  145. });
  146. }
  147. +(void) buildFormBody:(NSArray *)form boundary:(NSString *)boundary onComplete:(void(^)(NSData * formData))onComplete
  148. {
  149. __block NSMutableData * formData = [[NSMutableData alloc] init];
  150. if(form == nil)
  151. onComplete(nil);
  152. else
  153. {
  154. __block int i = 0;
  155. __block int count = [form count];
  156. // a recursive block that builds multipart body asynchornously
  157. void __block (^getFieldData)(id field) = ^(id field)
  158. {
  159. NSString * name = [field valueForKey:@"name"];
  160. __block NSString * content = [field valueForKey:@"data"];
  161. NSString * contentType = [field valueForKey:@"type"];
  162. // skip when the form field `name` or `data` is empty
  163. if(content == nil || name == nil)
  164. {
  165. i++;
  166. getFieldData([form objectAtIndex:i]);
  167. RCTLogWarn(@"RNFetchBlob multipart request builder has found a field without `data` or `name` property, the field will be removed implicitly.");
  168. return;
  169. }
  170. contentType = contentType == nil ? @"application/octet-stream" : contentType;
  171. // field is a text field
  172. if([field valueForKey:@"filename"] == nil || content == nil) {
  173. [formData appendData:[[NSString stringWithFormat:@"--%@\r\n", boundary] dataUsingEncoding:NSUTF8StringEncoding]];
  174. [formData appendData:[[NSString stringWithFormat:@"Content-Disposition: form-data; name=\"%@\"\r\n", name] dataUsingEncoding:NSUTF8StringEncoding]];
  175. [formData appendData:[[NSString stringWithFormat:@"Content-Type: text/plain\r\n\r\n"] dataUsingEncoding:NSUTF8StringEncoding]];
  176. [formData appendData:[[NSString stringWithFormat:@"%@\r\n", content] dataUsingEncoding:NSUTF8StringEncoding]];
  177. }
  178. // field contains a file
  179. else {
  180. NSMutableData * blobData;
  181. if(content != nil)
  182. {
  183. // append data from file asynchronously
  184. if([content hasPrefix:FILE_PREFIX])
  185. {
  186. NSString * orgPath = [content substringFromIndex:[FILE_PREFIX length]];
  187. orgPath = [RNFetchBlobFS getPathOfAsset:orgPath];
  188. [RNFetchBlobFS readFile:orgPath encoding:@"utf8" resolver:nil rejecter:nil onComplete:^(NSData *content) {
  189. NSString * filename = [field valueForKey:@"filename"];
  190. [formData appendData:[[NSString stringWithFormat:@"--%@\r\n", boundary] dataUsingEncoding:NSUTF8StringEncoding]];
  191. [formData appendData:[[NSString stringWithFormat:@"Content-Disposition: form-data; name=\"%@\"; filename=\"%@\"\r\n", name, filename] dataUsingEncoding:NSUTF8StringEncoding]];
  192. [formData appendData:[[NSString stringWithFormat:@"Content-Type: %@\r\n\r\n", contentType] dataUsingEncoding:NSUTF8StringEncoding]];
  193. [formData appendData:content];
  194. [formData appendData:[[NSString stringWithFormat:@"\r\n"] dataUsingEncoding:NSUTF8StringEncoding]];
  195. i++;
  196. if(i < count)
  197. {
  198. __block NSDictionary * nextField = [form objectAtIndex:i];
  199. getFieldData(nextField);
  200. }
  201. else
  202. {
  203. onComplete(formData);
  204. getFieldData = nil;
  205. }
  206. }];
  207. return ;
  208. }
  209. else
  210. blobData = [[NSData alloc] initWithBase64EncodedString:content options:0];
  211. }
  212. NSString * filename = [field valueForKey:@"filename"];
  213. [formData appendData:[[NSString stringWithFormat:@"--%@\r\n", boundary] dataUsingEncoding:NSUTF8StringEncoding]];
  214. [formData appendData:[[NSString stringWithFormat:@"Content-Disposition: form-data; name=\"%@\"; filename=\"%@\"\r\n", name, filename] dataUsingEncoding:NSUTF8StringEncoding]];
  215. [formData appendData:[[NSString stringWithFormat:@"Content-Type: %@\r\n\r\n", contentType] dataUsingEncoding:NSUTF8StringEncoding]];
  216. [formData appendData:blobData];
  217. [formData appendData:[[NSString stringWithFormat:@"\r\n"] dataUsingEncoding:NSUTF8StringEncoding]];
  218. blobData = nil;
  219. }
  220. i++;
  221. if(i < count)
  222. {
  223. __block NSDictionary * nextField = [form objectAtIndex:i];
  224. getFieldData(nextField);
  225. }
  226. else
  227. {
  228. onComplete(formData);
  229. getFieldData = nil;
  230. }
  231. };
  232. __block NSDictionary * nextField = [form objectAtIndex:i];
  233. getFieldData(nextField);
  234. }
  235. }
  236. +(NSString *) getHeaderIgnoreCases:(NSString *)field fromHeaders:(NSMutableDictionary *) headers {
  237. NSString * normalCase = [headers valueForKey:field];
  238. NSString * ignoredCase = [headers valueForKey:[field lowercaseString]];
  239. if( normalCase != nil)
  240. return normalCase;
  241. else
  242. return ignoredCase;
  243. }
  244. @end