2015-04-19 12:55:46 -07:00
/ * *
* Copyright ( c ) 2015 - present , Facebook , Inc .
* All rights reserved .
*
* This source code is licensed under the BSD - style license found in the
* LICENSE file in the root directory of this source tree . An additional grant
* of patent rights can be found in the PATENTS file in the same directory .
* /
2015-04-02 07:33:21 -07:00
# import "RCTJavaScriptLoader.h"
# import "RCTBridge.h"
2015-04-25 14:52:18 -07:00
# import "RCTConvert.h"
2015-04-02 07:33:21 -07:00
# import "RCTSourceCode.h"
# import "RCTUtils.h"
2015-10-21 10:53:35 -07:00
# import "RCTPerformanceLogger.h"
2015-04-02 07:33:21 -07:00
2016-03-17 10:34:46 -07:00
# include < sys / stat . h >
uint32_t const RCTRAMBundleMagicNumber = 0 xFB0BD1E5 ;
2015-04-02 07:33:21 -07:00
@ implementation RCTJavaScriptLoader
2015-08-24 09:14:33 -01:00
RCT_NOT _IMPLEMENTED ( - ( instancetype ) init )
2015-06-15 07:53:45 -07:00
2015-07-28 15:48:46 -07:00
+ ( void ) loadBundleAtURL : ( NSURL * ) scriptURL onComplete : ( RCTSourceLoadBlock ) onComplete
2015-04-02 07:33:21 -07:00
{
2015-04-25 14:52:18 -07:00
// Sanitize the script URL
scriptURL = [ RCTConvert NSURL : scriptURL . absoluteString ] ;
2015-10-08 12:28:38 -07:00
if ( ! scriptURL ) {
2015-04-11 15:08:00 -07:00
NSError * error = [ NSError errorWithDomain : @ "JavaScriptLoader" code : 1 userInfo : @ {
2015-10-08 12:28:38 -07:00
NSLocalizedDescriptionKey : @ "No script URL provided."
2015-04-11 15:08:00 -07:00
} ] ;
2015-05-04 10:35:49 -07:00
onComplete ( error , nil ) ;
2015-04-02 07:33:21 -07:00
return ;
2015-04-11 15:08:00 -07:00
}
2015-10-08 12:28:38 -07:00
// Load local script file
if ( scriptURL . fileURL ) {
dispatch_async ( dispatch_get _global _queue ( DISPATCH_QUEUE _PRIORITY _DEFAULT , 0 ) , ^ {
NSError * error = nil ;
2016-03-17 10:34:46 -07:00
NSData * source = nil ;
// Load the first 4 bytes to check if the bundle is regular or RAM ( "Random Access Modules" bundle ) .
// The RAM bundle has a magic number in the 4 first bytes ` ( 0 xFB0BD1E5 ) ` .
// The benefit of RAM bundle over a regular bundle is that we can lazily inject
// modules into JSC as they ' re required .
FILE * bundle = fopen ( scriptURL . path . UTF8String , "r" ) ;
if ( ! bundle ) {
onComplete ( RCTErrorWithMessage ( [ NSString stringWithFormat : @ "Error opening bundle %@" , scriptURL . path ] ) , source ) ;
return ;
}
uint32_t magicNumber ;
if ( fread ( & magicNumber , sizeof ( magicNumber ) , 1 , bundle ) ! = 1 ) {
fclose ( bundle ) ;
onComplete ( RCTErrorWithMessage ( @ "Error reading bundle" ) , source ) ;
return ;
}
magicNumber = NSSwapLittleIntToHost ( magicNumber ) ;
2016-03-21 03:20:49 -07:00
int64_t sourceLength = 0 ;
2016-03-17 10:34:46 -07:00
if ( magicNumber = = RCTRAMBundleMagicNumber ) {
source = [ NSData dataWithBytes : & magicNumber length : sizeof ( magicNumber ) ] ;
struct stat statInfo ;
if ( stat ( scriptURL . path . UTF8String , & statInfo ) ! = 0 ) {
error = RCTErrorWithMessage ( @ "Error reading bundle" ) ;
} else {
sourceLength = statInfo . st_size ;
}
} else {
source = [ NSData dataWithContentsOfFile : scriptURL . path
options : NSDataReadingMappedIfSafe
error : & error ] ;
sourceLength = source . length ;
}
RCTPerformanceLoggerSet ( RCTPLBundleSize , sourceLength ) ;
fclose ( bundle ) ;
2015-10-16 08:10:25 -07:00
onComplete ( error , source ) ;
2015-10-08 12:28:38 -07:00
} ) ;
return ;
}
// Load remote script file
2015-04-02 07:33:21 -07:00
NSURLSessionDataTask * task = [ [ NSURLSession sharedSession ] dataTaskWithURL : scriptURL completionHandler :
^ ( NSData * data , NSURLResponse * response , NSError * error ) {
2015-04-25 14:52:18 -07:00
// Handle general request errors
if ( error ) {
2015-08-24 09:14:33 -01:00
if ( [ error . domain isEqualToString : NSURLErrorDomain ] ) {
2015-09-01 05:12:48 -07:00
NSString * desc = [ @ "Could not connect to development server.\n\nEnsure the following:\n- Node server is running and available on the same network - run 'npm start' from react-native root\n- Node server URL is correctly set in AppDelegate\n\nURL: " stringByAppendingString : scriptURL . absoluteString ] ;
2015-04-25 14:52:18 -07:00
NSDictionary * userInfo = @ {
NSLocalizedDescriptionKey : desc ,
2015-08-24 09:14:33 -01:00
NSLocalizedFailureReasonErrorKey : error . localizedDescription ,
2015-04-25 14:52:18 -07:00
NSUnderlyingErrorKey : error ,
} ;
error = [ NSError errorWithDomain : @ "JSServer"
code : error . code
userInfo : userInfo ] ;
}
2015-05-04 10:35:49 -07:00
onComplete ( error , nil ) ;
2015-04-25 14:52:18 -07:00
return ;
}
2015-04-23 14:39:51 -07:00
2015-04-25 14:52:18 -07:00
// Parse response as text
NSStringEncoding encoding = NSUTF8StringEncoding ;
if ( response . textEncodingName ! = nil ) {
CFStringEncoding cfEncoding = CFStringConvertIANACharSetNameToEncoding ( ( CFStringRef ) response . textEncodingName ) ;
if ( cfEncoding ! = kCFStringEncodingInvalidId ) {
encoding = CFStringConvertEncodingToNSStringEncoding ( cfEncoding ) ;
}
}
// Handle HTTP errors
2015-08-24 09:14:33 -01:00
if ( [ response isKindOfClass : [ NSHTTPURLResponse class ] ] && ( ( NSHTTPURLResponse * ) response ) . statusCode ! = 200 ) {
2015-10-16 08:10:25 -07:00
NSString * rawText = [ [ NSString alloc ] initWithData : data encoding : encoding ] ;
2015-04-25 14:52:18 -07:00
NSDictionary * userInfo ;
NSDictionary * errorDetails = RCTJSONParse ( rawText , nil ) ;
if ( [ errorDetails isKindOfClass : [ NSDictionary class ] ] &&
[ errorDetails [ @ "errors" ] isKindOfClass : [ NSArray class ] ] ) {
2015-11-03 14:45:46 -08:00
NSMutableArray < NSDictionary * > * fakeStack = [ NSMutableArray new ] ;
2015-04-25 14:52:18 -07:00
for ( NSDictionary * err in errorDetails [ @ "errors" ] ) {
[ fakeStack addObject : @ {
@ "methodName" : err [ @ "description" ] ? : @ "" ,
@ "file" : err [ @ "filename" ] ? : @ "" ,
@ "lineNumber" : err [ @ "lineNumber" ] ? : @ 0
} ] ;
}
userInfo = @ {
NSLocalizedDescriptionKey : errorDetails [ @ "message" ] ? : @ "No message provided" ,
@ "stack" : fakeStack ,
} ;
} else {
userInfo = @ { NSLocalizedDescriptionKey : rawText } ;
}
error = [ NSError errorWithDomain : @ "JSServer"
2015-08-24 09:14:33 -01:00
code : ( ( NSHTTPURLResponse * ) response ) . statusCode
2015-04-25 14:52:18 -07:00
userInfo : userInfo ] ;
2015-05-04 10:35:49 -07:00
onComplete ( error , nil ) ;
2015-04-25 14:52:18 -07:00
return ;
}
2015-10-21 10:53:35 -07:00
RCTPerformanceLoggerSet ( RCTPLBundleSize , data . length ) ;
2015-10-16 08:10:25 -07:00
onComplete ( nil , data ) ;
2015-04-25 14:52:18 -07:00
} ] ;
2015-04-02 07:33:21 -07:00
[ task resume ] ;
}
@ end