123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511 |
- //
- // RNFetchBlob.m
- //
- // Created by wkh237 on 2016/4/28.
- //
-
- #import "RNFetchBlob.h"
- #import "RCTConvert.h"
- #import "RCTLog.h"
- #import <Foundation/Foundation.h>
- #import "RCTBridge.h"
- #import "RCTEventDispatcher.h"
-
- // lib event
- NSString *const MSG_EVENT = @"RNFetchBlobMessage";
- NSString *const MSG_EVENT_LOG = @"log";
- NSString *const MSG_EVENT_WARN = @"warn";
- NSString *const MSG_EVENT_ERROR = @"error";
- NSString *const CONFIG_USE_TEMP = @"fileCache";
- NSString *const CONFIG_FILE_PATH = @"path";
- NSString *const FS_EVENT_DATA = @"data";
- NSString *const FS_EVENT_END = @"end";
- NSString *const FS_EVENT_WARN = @"warn";
- NSString *const FS_EVENT_ERROR = @"error";
-
- ////////////////////////////////////////
- //
- // File system access methods
- //
- ////////////////////////////////////////
-
- @implementation FetchBlobFS
-
- @synthesize outStream;
- @synthesize inStream;
- @synthesize encoding;
- @synthesize callback;
- @synthesize taskId;
- @synthesize path;
-
-
-
- + (NSString *) getCacheDir {
-
- return [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) firstObject];
- }
-
- + (NSString *) getDocumentDir {
-
- return NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
- }
-
- + (NSString *) getMusicDir {
- return NSSearchPathForDirectoriesInDomains(NSMusicDirectory, NSUserDomainMask, YES);
- }
-
- + (NSString *) getMovieDir {
- return NSSearchPathForDirectoriesInDomains(NSMoviesDirectory, NSUserDomainMask, YES);
- }
-
- + (NSString *) getPictureDir {
- return NSSearchPathForDirectoriesInDomains(NSPicturesDirectory, NSUserDomainMask, YES);
- }
-
- + (NSString *) getTempPath:(NSString*)taskId {
-
- NSString * documentDir = [NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES) firstObject];
- NSString * filename = [NSString stringWithFormat:@"RNFetchBlobTmp_%s", taskId];
- NSString * tempPath = [documentDir stringByAppendingString: filename];
- return tempPath;
- }
-
- - (id)initWithCallback:(RCTResponseSenderBlock)callback {
- self = [super init];
- self.callback = callback;
- return self;
- }
-
- - (id)initWithBridgeRef:(RCTBridge *)bridgeRef {
- self = [super init];
- self.callback = callback;
- return self;
- }
-
- - (void)openWithPath:(NSString *)destPath {
- self.outStream = [[NSOutputStream alloc]init];
- [self.outStream initToFileAtPath:destPath append:NO];
- }
-
-
- - (void)openWithId:(NSString *)taskId {
-
- NSString * tmpPath = [[self class ]getTempPath: taskId];
- // create a file stream
- [self openWithPath:tmpPath];
-
- }
-
- // Write file chunk into an opened stream
- - (void)write:(NSString *) chunk {
- [self.outStream write:[chunk cStringUsingEncoding:NSASCIIStringEncoding] maxLength:chunk.length];
- }
-
- - (void)readWithPath:(NSString *)path useEncoding:(NSString *)encoding {
- self.inStream = [[NSInputStream alloc]init];
- self.encoding = encoding;
- [self.inStream setDelegate:self];
-
- }
-
- - (void)readWithTaskId:(NSString *)taskId withPath:(NSString *)path useEncoding:(NSString *)encoding {
- self.taskId = taskId;
- self.path = path;
- if(path == nil)
- [self readWithPath:[[self class]getTempPath:taskId] useEncoding:encoding];
- else
- [self readWithPath:path useEncoding:encoding];
- }
-
- // close file stream
- - (void)closeOutStream {
- if(self.outStream != nil) {
- [self.outStream close];
- self.outStream = nil;
- }
-
- }
-
- - (void)closeInStream {
- if(self.inStream != nil) {
- [self.inStream close];
- [self.inStream removeFromRunLoop:[NSRunLoop currentRunLoop] forMode:NSRunLoopCommonModes];
- }
-
- }
-
- #pragma mark RNFetchBlobFS read stream delegate
-
- - (void)stream:(NSStream *)stream handleEvent:(NSStreamEvent)eventCode {
-
- switch(eventCode) {
-
- case NSStreamEventHasBytesAvailable:
- {
-
-
- NSMutableData * chunkData = [[NSMutableData data] init];
-
- uint8_t buf[1024];
- unsigned int len = 0;
- len = [(NSInputStream *)stream read:buf maxLength:1024];
- // still have data in stream
- if(len) {
- [chunkData appendBytes:(const void *)buf length:len];
- // TODO : file read progress ?
- // [bytesRead setIntValue:[bytesRead intValue]+len];
-
- // dispatch data event
- NSString * encodedChunk = [NSString alloc];
- if( [self.encoding caseInsensitiveCompare:@"utf8"] ) {
- encodedChunk = [encodedChunk initWithData:chunkData encoding:NSUTF8StringEncoding];
- }
- else if ( [self.encoding caseInsensitiveCompare:@"ascii"] ) {
- encodedChunk = [encodedChunk initWithData:chunkData encoding:NSASCIIStringEncoding];
- }
- else if ( [self.encoding caseInsensitiveCompare:@"base64"] ) {
- encodedChunk = [chunkData base64EncodedStringWithOptions:0];
- }
- else {
- [self.bridge.eventDispatcher
- sendAppEventWithName: [NSString stringWithFormat:@"RNFetchBlobStream%s", self.taskId]
- body:@{
- @"event": FS_EVENT_ERROR,
- @"detail": @"unrecognized encoding"
- }
- ];
- return;
- }
- [self.bridge.eventDispatcher
- sendAppEventWithName: [NSString stringWithFormat:@"RNFetchBlobStream%s", self.taskId]
- body:@{
- @"event": FS_EVENT_DATA,
- @"detail": encodedChunk
- }
- ];
- }
- // end of stream
- else {
- [self.bridge.eventDispatcher
- sendAppEventWithName: [NSString stringWithFormat:@"RNFetchBlobStream%s", self.taskId]
- body:@{
- @"event": FS_EVENT_END,
- @"detail": @""
- }
- ];
- }
- break;
- }
- case NSStreamEventErrorOccurred:
- {
- [self.bridge.eventDispatcher
- sendAppEventWithName: [NSString stringWithFormat:@"RNFetchBlobStream%s", self.taskId]
- body:@{
- @"event": FS_EVENT_ERROR,
- @"detail": @"error when read file with stream"
- }
- ];
- break;
- }
-
- }
-
- }
-
- @end
-
- ////////////////////////////////////////
- //
- // HTTP request handler
- //
- ////////////////////////////////////////
-
- @implementation FetchBlobUtils
-
-
- @synthesize taskId;
- @synthesize expectedBytes;
- @synthesize receivedBytes;
- @synthesize respData;
- @synthesize callback;
- @synthesize bridge;
- @synthesize options;
-
-
- // removing case from headers
- + (NSMutableDictionary *) normalizeHeaders:(NSDictionary *)headers {
-
- NSMutableDictionary * mheaders = [[NSMutableDictionary alloc]init];
- for(NSString * key in headers) {
- [mheaders setValue:[headers valueForKey:key] forKey:[key lowercaseString]];
- }
-
- return mheaders;
- }
-
- - (id)init {
- self = [super init];
- return self;
- }
-
-
- - (void) sendRequest:(NSDictionary *)options bridge:(RCTBridge *)bridgeRef taskId:(NSString *)taskId withRequest:(NSURLRequest *)req callback:(RCTResponseSenderBlock) callback {
- self.taskId = taskId;
- self.respData = [[NSMutableData alloc] initWithLength:0];
- self.callback = callback;
- self.bridge = bridgeRef;
- self.expectedBytes = 0;
- self.receivedBytes = 0;
- self.options = options;
-
- NSString * path = [self.options valueForKey:CONFIG_FILE_PATH];
-
- // open file stream for write
- if( path != nil) {
- self.fileStream = [[FetchBlobFS alloc]initWithCallback:self.callback];
- [self.fileStream openWithPath:path];
- }
- else if ( [self.options valueForKey:CONFIG_USE_TEMP] == YES ) {
- self.fileStream = [[FetchBlobFS alloc]initWithCallback:self.callback];
- [self.fileStream openWithId:taskId];
- }
-
- NSURLConnection *conn = [[NSURLConnection alloc] initWithRequest:req delegate:self startImmediately:NO];
- [conn scheduleInRunLoop:[NSRunLoop mainRunLoop] forMode:NSRunLoopCommonModes];
- [conn start];
-
- if(!conn) {
- callback(@[[NSString stringWithFormat:@"RNFetchBlob could not initialize connection"], [NSNull null]]);
- }
- }
-
-
- #pragma mark NSURLConnection delegate methods
-
-
- - (void) connection:(NSURLConnection *)connection didReceiveResponse:(nonnull NSURLResponse *)response {
- [UIApplication sharedApplication].networkActivityIndicatorVisible = YES;
- expectedBytes = [response expectedContentLength];
- }
-
-
- - (void) connection:(NSURLConnection *)connection didReceiveData:(nonnull NSData *)data {
- receivedBytes += data.length;
-
- Boolean fileCache = [self.options valueForKey:CONFIG_USE_TEMP];
- NSString * path = [self.options valueForKey:CONFIG_FILE_PATH];
-
- // write to tmp file
- if( fileCache == YES || path != nil ) {
- [self.fileStream write:data];
- }
- // cache data in memory
- else {
- [respData appendData:data];
- }
-
- [self.bridge.eventDispatcher
- sendAppEventWithName:@"RNFetchBlobProgress"
- body:@{
- @"taskId": taskId,
- @"written": [NSString stringWithFormat:@"%d", receivedBytes],
- @"total": [NSString stringWithFormat:@"%d", expectedBytes]
- }
- ];
- }
-
- - (void) connection:(NSURLConnection *)connection didSendBodyData:(NSInteger)bytesWritten totalBytesWritten:(NSInteger)totalBytesWritten totalBytesExpectedToWrite:(NSInteger)totalBytesExpectedToWrite {
-
- expectedBytes = totalBytesExpectedToWrite;
- receivedBytes += totalBytesWritten;
- [self.bridge.eventDispatcher
- sendAppEventWithName:@"RNFetchBlobProgress"
- body:@{
- @"taskId": taskId,
- @"written": [NSString stringWithFormat:@"%d", receivedBytes],
- @"total": [NSString stringWithFormat:@"%d", expectedBytes]
- }
- ];
-
- }
-
- - (void) connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {
-
- [UIApplication sharedApplication].networkActivityIndicatorVisible = NO;
-
- [self.fileStream closeInStream];
- [self.fileStream closeOutStream];
-
- callback(@[[error localizedDescription], [NSNull null]]);
- }
-
- - (NSCachedURLResponse *) connection:(NSURLConnection *)connection willCacheResponse: (NSCachedURLResponse *)cachedResponse {
- return nil;
- }
-
-
- // handle 301 and 302 responses
- - (NSURLRequest *)connection:(NSURLConnection *)connection willSendRequest:(NSURLRequest *)request redirectResponse:response {
- return request;
- }
-
- // request complete
- - (void) connectionDidFinishLoading:(NSURLConnection *)connection {
-
- NSData * data;
- if(respData != nil)
- data = [NSData dataWithData:respData];
- else
- data = [[NSData alloc] init];
-
- NSString * path = [NSString stringWithString:[self.options valueForKey:CONFIG_FILE_PATH]];
-
- [self.fileStream closeInStream];
-
- // if fileCache is true or file path is given, return a path
- if( path != nil ) {
- callback(@[[NSNull null], path]);
- }
- // when fileCache option is set but no path specified, save to tmp path
- else if( [self.options valueForKey:CONFIG_USE_TEMP] == YES || path != nil ) {
- NSString * tmpPath = [FetchBlobFS getTempPath:taskId];
- callback(@[[NSNull null], tmpPath]);
- }
- // otherwise return base64 string
- else {
- callback(@[[NSNull null], [data base64EncodedStringWithOptions:0]]);
- }
- }
-
- @end
-
-
- ////////////////////////////////////////
- //
- // Exported native methods
- //
- ////////////////////////////////////////
-
- #pragma mark RNFetchBlob exported methods
-
- @implementation RNFetchBlob
-
- @synthesize bridge = _bridge;
-
- RCT_EXPORT_MODULE();
-
- // Fetch blob data request
- RCT_EXPORT_METHOD(fetchBlobForm:(NSDictionary *)options taskId:(NSString *)taskId method:(NSString *)method url:(NSString *)url headers:(NSDictionary *)headers form:(NSArray *)form callback:(RCTResponseSenderBlock)callback)
- {
-
- // send request
- NSMutableURLRequest *request = [[NSMutableURLRequest alloc]
- initWithURL:[NSURL
- URLWithString: url]];
-
- NSMutableDictionary *mheaders = [[NSMutableDictionary alloc] initWithDictionary:[ FetchBlobUtils normalizeHeaders:headers]];
-
-
- NSTimeInterval timeStamp = [[NSDate date] timeIntervalSince1970];
- NSNumber * timeStampObj = [NSNumber numberWithDouble: timeStamp];
-
- // generate boundary
- NSString * boundary = [NSString stringWithFormat:@"RNFetchBlob%d", timeStampObj];
-
- // if method is POST or PUT, convert data string format
- if([[method lowercaseString] isEqualToString:@"post"] || [[method lowercaseString] isEqualToString:@"put"]) {
- NSMutableData * postData = [[NSMutableData alloc] init];
-
- // combine multipart/form-data body
- for(id field in form) {
- NSString * name = [field valueForKey:@"name"];
- NSString * content = [field valueForKey:@"data"];
- // field is a text field
- if([field valueForKey:@"filename"] == nil || content == [NSNull null]) {
- [postData appendData:[[NSString stringWithFormat:@"--%@\r\n", boundary] dataUsingEncoding:NSUTF8StringEncoding]];
- [postData appendData:[[NSString stringWithFormat:@"Content-Disposition: form-data; name=\"%@\"\r\n", name] dataUsingEncoding:NSUTF8StringEncoding]];
- [postData appendData:[[NSString stringWithFormat:@"Content-Type: text/plain\r\n\r\n"] dataUsingEncoding:NSUTF8StringEncoding]];
- [postData appendData:[[NSString stringWithFormat:@"%@\r\n", content] dataUsingEncoding:NSUTF8StringEncoding]];
- }
- // field contains a file
- else {
- NSData* blobData = [[NSData alloc] initWithBase64EncodedString:content options:0];
- NSString * filename = [field valueForKey:@"filename"];
- [postData appendData:[[NSString stringWithFormat:@"--%@\r\n", boundary] dataUsingEncoding:NSUTF8StringEncoding]];
- [postData appendData:[[NSString stringWithFormat:@"Content-Disposition: form-data; name=\"%@\"; filename=\"%@\"\r\n", name, filename] dataUsingEncoding:NSUTF8StringEncoding]];
- [postData appendData:[[NSString stringWithFormat:@"Content-Type: application/octet-stream\r\n\r\n"] dataUsingEncoding:NSUTF8StringEncoding]];
- [postData appendData:blobData];
- [postData appendData:[[NSString stringWithFormat:@"\r\n"] dataUsingEncoding:NSUTF8StringEncoding]];
- }
-
- }
- // close form data
- [postData appendData: [[NSString stringWithFormat:@"--%@--\r\n", boundary] dataUsingEncoding:NSUTF8StringEncoding]];
- [request setHTTPBody:postData];
- // set content-length
- [mheaders setValue:[NSString stringWithFormat:@"%d",[postData length]] forKey:@"Content-Length"];
- [mheaders setValue:[NSString stringWithFormat:@"100-continue",[postData length]] forKey:@"Expect"];
- // appaned boundary to content-type
- [mheaders setValue:[NSString stringWithFormat:@"multipart/form-data; charset=utf-8; boundary=%@", boundary] forKey:@"content-type"];
-
- }
-
- [request setHTTPMethod: method];
- [request setAllHTTPHeaderFields:mheaders];
-
-
- // send HTTP request
- FetchBlobUtils * utils = [[FetchBlobUtils alloc] init];
- [utils sendRequest:options bridge:self.bridge taskId:taskId withRequest:request callback:callback];
-
- }
-
- // Fetch blob data request
- RCT_EXPORT_METHOD(fetchBlob:(NSDictionary *)options taskId:(NSString *)taskId method:(NSString *)method url:(NSString *)url headers:(NSDictionary *)headers body:(NSString *)body callback:(RCTResponseSenderBlock)callback)
- {
- // send request
- NSMutableURLRequest *request = [[NSMutableURLRequest alloc]
- initWithURL:[NSURL
- URLWithString: url]];
-
- NSMutableDictionary *mheaders = [[NSMutableDictionary alloc] initWithDictionary:[FetchBlobUtils normalizeHeaders:headers]];
-
- // if method is POST or PUT, convert data string format
- if([[method lowercaseString] isEqualToString:@"post"] || [[method lowercaseString] isEqualToString:@"put"]) {
-
- if(body != nil) {
- // generate octet-stream body
- NSData* blobData = [[NSData alloc] initWithBase64EncodedString:body options:0];
- NSMutableData* postBody = [[NSMutableData alloc] init];
- [postBody appendData:[NSData dataWithData:blobData]];
- [request setHTTPBody:postBody];
- [mheaders setValue:@"application/octet-stream" forKey:@"content-type"];
- }
- }
-
- [request setHTTPMethod: method];
- [request setAllHTTPHeaderFields:mheaders];
-
- // send HTTP request
- FetchBlobUtils * utils = [[FetchBlobUtils alloc] init];
- [utils sendRequest:options bridge:self.bridge taskId:taskId withRequest:request callback:callback];
-
- }
-
- RCT_EXPORT_METHOD(readStream:(NSString *)taskId withPath:(NSString *)path withEncoding:(NSString *)encoding) {
- FetchBlobFS *fileStream = [[FetchBlobFS alloc] initWithBridgeRef:self.bridge];
- [fileStream readWithTaskId:taskId withPath:path useEncoding:encoding];
- }
-
- RCT_EXPORT_METHOD(flush:(NSString *)taskId withPath:(NSString *)path) {
- NSError * error = nil;
- NSString * tmpPath = nil;
- if(path != nil)
- tmpPath = path;
- else
- tmpPath = [FetchBlobFS getTempPath:taskId];
- [[NSFileManager defaultManager] removeItemAtPath:path error:&error];
- }
-
- @end
|