/* * Copyright (c) Meta Platforms, Inc. and affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. */ #import #import #import #import #import #import "RCTNetworkPlugins.h" @interface RCTFileRequestHandler () @end @implementation RCTFileRequestHandler { NSOperationQueue *_fileQueue; std::mutex _operationHandlerMutexLock; } RCT_EXPORT_MODULE() - (void)invalidate { std::lock_guard lock(_operationHandlerMutexLock); if (_fileQueue) { for (NSOperation *operation in _fileQueue.operations) { if (!operation.isCancelled && !operation.isFinished) { [operation cancel]; } } _fileQueue = nil; } } - (BOOL)canHandleRequest:(NSURLRequest *)request { return [request.URL.scheme caseInsensitiveCompare:@"file"] == NSOrderedSame && !RCTIsBundleAssetURL(request.URL); } - (NSOperation *)sendRequest:(NSURLRequest *)request withDelegate:(id)delegate { std::lock_guard lock(_operationHandlerMutexLock); // Lazy setup if (!_fileQueue) { _fileQueue = [NSOperationQueue new]; _fileQueue.maxConcurrentOperationCount = 4; } NSBlockOperation *op = [NSBlockOperation new]; __weak NSBlockOperation *weakOp = op; [op addExecutionBlock:^{ NSBlockOperation *strongOp = weakOp; // Strong reference to avoid deallocation during execution if (strongOp == nil || [strongOp isCancelled]) { return; } // Get content length NSError *error = nil; NSFileManager *fileManager = [NSFileManager new]; NSDictionary *fileAttributes = [fileManager attributesOfItemAtPath:request.URL.path error:&error]; if (!fileAttributes) { [delegate URLRequest:strongOp didCompleteWithError:error]; return; } // Get mime type NSString *fileExtension = [request.URL pathExtension]; NSString *UTI = (__bridge_transfer NSString *)UTTypeCreatePreferredIdentifierForTag( kUTTagClassFilenameExtension, (__bridge CFStringRef)fileExtension, NULL); NSString *contentType = (__bridge_transfer NSString *)UTTypeCopyPreferredTagWithClass((__bridge CFStringRef)UTI, kUTTagClassMIMEType); // Send response NSURLResponse *response = [[NSURLResponse alloc] initWithURL:request.URL MIMEType:contentType expectedContentLength:[fileAttributes[NSFileSize] ?: @-1 integerValue] textEncodingName:nil]; [delegate URLRequest:strongOp didReceiveResponse:response]; // Load data NSData *data = [NSData dataWithContentsOfURL:request.URL options:NSDataReadingMappedIfSafe error:&error]; if (data) { [delegate URLRequest:strongOp didReceiveData:data]; } [delegate URLRequest:strongOp didCompleteWithError:error]; }]; [_fileQueue addOperation:op]; return op; } - (void)cancelRequest:(NSOperation *)op { std::lock_guard lock(_operationHandlerMutexLock); if (!op.isCancelled && !op.isFinished) { [op cancel]; } } - (std::shared_ptr)getTurboModule: (const facebook::react::ObjCTurboModule::InitParams &)params { return nullptr; } @end Class RCTFileRequestHandlerCls(void) { return RCTFileRequestHandler.class; }