2015-01-29 17:10:49 -08:00
// Copyright 2004 - present Facebook . All Rights Reserved .
# import "RCTBridge.h"
2015-02-06 15:43:59 -08:00
# import < dlfcn . h >
# import < mach - o / getsect . h >
# import < mach - o / dyld . h >
2015-01-29 17:10:49 -08:00
# import < objc / message . h >
2015-02-06 15:43:59 -08:00
# import < objc / runtime . h >
2015-01-29 17:10:49 -08:00
2015-02-06 15:43:59 -08:00
# import "RCTConvert.h"
2015-01-29 17:10:49 -08:00
# import "RCTInvalidating.h"
2015-02-03 16:02:36 -08:00
# import "RCTEventDispatcher.h"
2015-01-29 17:10:49 -08:00
# import "RCTLog.h"
# import "RCTUtils.h"
2015-02-03 16:12:07 -08:00
/ * *
* Must be kept in sync with ` MessageQueue . js` .
* /
typedef NS_ENUM ( NSUInteger , RCTBridgeFields ) {
RCTBridgeFieldRequestModuleIDs = 0 ,
RCTBridgeFieldMethodIDs ,
RCTBridgeFieldParamss ,
RCTBridgeFieldResponseCBIDs ,
RCTBridgeFieldResponseReturnValues ,
RCTBridgeFieldFlushDateMillis
} ;
2015-02-06 15:43:59 -08:00
/ * *
* This private class is used as a container for exported method info
* /
@ interface RCTModuleMethod : NSObject
@ property ( readonly , nonatomic , assign ) SEL selector ;
@ property ( readonly , nonatomic , copy ) NSString * JSMethodName ;
@ property ( readonly , nonatomic , assign ) NSUInteger arity ;
@ property ( readonly , nonatomic , copy ) NSIndexSet * blockArgumentIndexes ;
@ end
@ implementation RCTModuleMethod
- ( instancetype ) initWithSelector : ( SEL ) selector
JSMethodName : ( NSString * ) JSMethodName
arity : ( NSUInteger ) arity
blockArgumentIndexes : ( NSIndexSet * ) blockArgumentIndexes
{
if ( ( self = [ super init ] ) ) {
_selector = selector ;
_JSMethodName = [ JSMethodName copy ] ;
_arity = arity ;
_blockArgumentIndexes = [ blockArgumentIndexes copy ] ;
}
return self ;
}
- ( NSString * ) description
{
NSString * blocks = @ "no block args" ;
if ( self . blockArgumentIndexes . count > 0 ) {
NSMutableString * indexString = [ NSMutableString string ] ;
[ self . blockArgumentIndexes enumerateIndexesUsingBlock : ^ ( NSUInteger idx , BOOL * stop ) {
[ indexString appendFormat : @ ", %tu" , idx ] ;
} ] ;
blocks = [ NSString stringWithFormat : @ "block args at %@" , [ indexString substringFromIndex : 2 ] ] ;
}
return [ NSString stringWithFormat : @ "<%@: %p; exports -%@ as %@; %@>" , NSStringFromClass ( self . class ) , self , NSStringFromSelector ( self . selector ) , self . JSMethodName , blocks ] ;
}
@ end
# ifdef __LP64 __
typedef uint64_t RCTExportValue ;
typedef struct section_64 RCTExportSection ;
# define RCTGetSectByNameFromHeader getsectbynamefromheader_64
# else
typedef uint32_t RCTExportValue ;
typedef struct section RCTExportSection ;
# define RCTGetSectByNameFromHeader getsectbynamefromheader
# endif
/ * *
* This function parses the exported methods inside RCTBridgeModules and
* generates a dictionary of arrays of RCTModuleMethod objects , keyed
* by module name .
* /
static NSDictionary * RCTExportedMethodsByModule ( void )
{
static NSMutableDictionary * methodsByModule ;
static dispatch_once _t onceToken ;
dispatch_once ( & onceToken , ^ {
Dl_info info ;
dladdr ( & RCTExportedMethodsByModule , & info ) ;
const RCTExportValue mach_header = ( RCTExportValue ) info . dli_fbase ;
const RCTExportSection * section = RCTGetSectByNameFromHeader ( ( void * ) mach_header , "__DATA" , "RCTExport" ) ;
if ( section = = NULL ) {
return ;
}
methodsByModule = [ NSMutableDictionary dictionary ] ;
NSCharacterSet * plusMinusCharacterSet = [ NSCharacterSet characterSetWithCharactersInString : @ "+-" ] ;
for ( RCTExportValue addr = section -> offset ;
addr < section -> offset + section -> size ;
addr + = sizeof ( id ) * 2 ) {
const char * * entry = ( const char * * ) ( mach_header + addr ) ;
NSScanner * scanner = [ NSScanner scannerWithString : @ ( entry [ 0 ] ) ] ;
NSString * plusMinus ;
if ( ! [ scanner scanCharactersFromSet : plusMinusCharacterSet intoString : & plusMinus ] ) continue ;
if ( ! [ scanner scanString : @ "[" intoString : NULL ] ) continue ;
NSString * className ;
if ( ! [ scanner scanUpToString : @ " " intoString : & className ] ) continue ;
[ scanner scanString : @ " " intoString : NULL ] ;
NSString * selectorName ;
if ( ! [ scanner scanUpToString : @ "]" intoString : & selectorName ] ) continue ;
Class class = NSClassFromString ( className ) ;
if ( class = = Nil ) continue ;
SEL selector = NSSelectorFromString ( selectorName ) ;
Method method = ( [ plusMinus characterAtIndex : 0 ] = = ' + ' ? class_getClassMethod : class_getInstanceMethod ) ( class , selector ) ;
if ( method = = nil ) continue ;
unsigned int argumentCount = method_getNumberOfArguments ( method ) ;
NSMutableIndexSet * blockArgumentIndexes = [ NSMutableIndexSet indexSet ] ;
static const char * blockType = @ encode ( typeof ( ^ { } ) ) ;
for ( unsigned int i = 2 ; i < argumentCount ; i + + ) {
char * type = method_copyArgumentType ( method , i ) ;
if ( ! strcmp ( type , blockType ) ) {
[ blockArgumentIndexes addIndex : i - 2 ] ;
}
free ( type ) ;
}
NSString * JSMethodName = strlen ( entry [ 1 ] ) ? @ ( entry [ 1 ] ) : [ NSStringFromSelector ( selector ) componentsSeparatedByString : @ ":" ] [ 0 ] ;
RCTModuleMethod * moduleMethod =
[ [ RCTModuleMethod alloc ] initWithSelector : selector
JSMethodName : JSMethodName
arity : method_getNumberOfArguments ( method ) - 2
blockArgumentIndexes : blockArgumentIndexes ] ;
NSString * moduleName = [ class respondsToSelector : @ selector ( moduleName ) ] ? [ class moduleName ] : className ;
NSArray * moduleMap = methodsByModule [ moduleName ] ;
methodsByModule [ moduleName ] = ( moduleMap ! = nil ) ? [ moduleMap arrayByAddingObject : moduleMethod ] : @ [ moduleMethod ] ;
}
} ) ;
return methodsByModule ;
}
/ * *
* This function scans all classes available at runtime and returns a dictionary
* of classes that implement the RTCBridgeModule protocol , keyed by module name .
* /
static NSDictionary * RCTBridgeModuleClasses ( void )
2015-01-29 17:10:49 -08:00
{
static NSMutableDictionary * modules ;
static dispatch_once _t onceToken ;
dispatch_once ( & onceToken , ^ {
modules = [ NSMutableDictionary dictionary ] ;
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
unsigned int classCount ;
Class * classes = objc_copyClassList ( & classCount ) ;
for ( unsigned int i = 0 ; i < classCount ; i + + ) {
Class cls = classes [ i ] ;
if ( ! class_getSuperclass ( cls ) ) {
// Class has no superclass - it ' s probably something weird
continue ;
}
2015-02-06 15:43:59 -08:00
if ( ! [ cls conformsToProtocol : @ protocol ( RCTBridgeModule ) ] ) {
// Not an RCTBridgeModule
2015-01-29 17:10:49 -08:00
continue ;
}
// Get module name
2015-02-03 16:15:20 -08:00
NSString * moduleName = [ cls respondsToSelector : @ selector ( moduleName ) ] ? [ cls moduleName ] : NSStringFromClass ( cls ) ;
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
// Check module name is unique
id existingClass = modules [ moduleName ] ;
2015-02-06 15:43:59 -08:00
RCTCAssert ( existingClass = = Nil , @ "Attempted to register RCTBridgeModule class %@ for the name '%@', but name was already registered by class %@" , cls , moduleName , existingClass ) ;
2015-01-29 17:10:49 -08:00
modules [ moduleName ] = cls ;
}
free ( classes ) ;
} ) ;
return modules ;
}
2015-02-06 15:43:59 -08:00
/ * *
* This constructs the remote modules configuration data structure ,
* which represents the native modules and methods that will be called
* by JS . A numeric ID is assigned to each module and method , which will
* be used to communicate via the bridge . The structure of each
* module is as follows :
*
* "ModuleName1" : {
* "moduleID" : 0 ,
* "methods" : {
* "methodName1" : {
* "methodID" : 0 ,
* "type" : "remote"
* } ,
* "methodName2" : {
* "methodID" : 1 ,
* "type" : "remote"
* } ,
* etc . . .
* } ,
* "constants" : {
* . . .
* }
* } ,
* etc . . .
* /
static NSMutableDictionary * RCTRemoteModulesByID ;
static NSDictionary * RCTRemoteModulesConfig ( )
{
static NSMutableDictionary * remoteModules ;
static dispatch_once _t onceToken ;
dispatch_once ( & onceToken , ^ {
RCTRemoteModulesByID = [ [ NSMutableDictionary alloc ] init ] ;
remoteModules = [ [ NSMutableDictionary alloc ] init ] ;
[ RCTExportedMethodsByModule ( ) enumerateKeysAndObjectsUsingBlock : ^ ( NSString * moduleName , NSArray * rawMethods , BOOL * stop ) {
NSMutableDictionary * methods = [ NSMutableDictionary dictionaryWithCapacity : rawMethods . count ] ;
[ rawMethods enumerateObjectsUsingBlock : ^ ( RCTModuleMethod * method , NSUInteger methodID , BOOL * stop ) {
methods [ method . JSMethodName ] = @ {
@ "methodID" : @ ( methodID ) ,
@ "type" : @ "remote" ,
} ;
} ] ;
NSDictionary * module = @ {
@ "moduleID" : @ ( remoteModules . count ) ,
@ "methods" : methods
} ;
Class cls = RCTBridgeModuleClasses ( ) [ moduleName ] ;
if ( RCTClassOverridesClassMethod ( cls , @ selector ( constantsToExport ) ) ) {
module = [ module mutableCopy ] ;
( ( NSMutableDictionary * ) module ) [ @ "constants" ] = [ cls constantsToExport ] ;
}
remoteModules [ moduleName ] = module ;
// Add module lookup
RCTRemoteModulesByID [ module [ @ "moduleID" ] ] = moduleName ;
} ] ;
} ) ;
return remoteModules ;
}
/ * *
* As above , but for local modules / methods , which represent JS classes
* and methods that will be called by the native code via the bridge .
* Structure is essentially the same as for remote modules :
*
* "ModuleName1" : {
* "moduleID" : 0 ,
* "methods" : {
* "methodName1" : {
* "methodID" : 0 ,
* "type" : "local"
* } ,
* "methodName2" : {
* "methodID" : 1 ,
* "type" : "local"
* } ,
* etc . . .
* }
* } ,
* etc . . .
* /
static NSMutableDictionary * RCTLocalModuleIDs ;
static NSMutableDictionary * RCTLocalMethodIDs ;
static NSDictionary * RCTLocalModulesConfig ( )
{
static NSMutableDictionary * localModules ;
static dispatch_once _t onceToken ;
dispatch_once ( & onceToken , ^ {
RCTLocalModuleIDs = [ [ NSMutableDictionary alloc ] init ] ;
RCTLocalMethodIDs = [ [ NSMutableDictionary alloc ] init ] ;
NSMutableArray * JSMethods = [ [ NSMutableArray alloc ] init ] ;
// Add globally used methods
[ JSMethods addObjectsFromArray : @ [
@ "Bundler.runApplication" ,
@ "RCTEventEmitter.receiveEvent" ,
@ "RCTEventEmitter.receiveTouches" ,
] ] ;
// NOTE : these methods are currently unused in the OSS project
// @ "Dimensions.set" ,
// @ "RCTDeviceEventEmitter.emit" ,
// @ "RCTNativeAppEventEmitter.emit" ,
// @ "ReactIOS.unmountComponentAtNodeAndRemoveContainer" ,
// Register individual methods from modules
for ( Class cls in RCTBridgeModuleClasses ( ) . allValues ) {
if ( RCTClassOverridesClassMethod ( cls , @ selector ( JSMethods ) ) ) {
[ JSMethods addObjectsFromArray : [ cls JSMethods ] ] ;
}
}
localModules = [ [ NSMutableDictionary alloc ] init ] ;
for ( NSString * moduleDotMethod in JSMethods ) {
NSArray * parts = [ moduleDotMethod componentsSeparatedByString : @ "." ] ;
RCTCAssert ( parts . count = = 2 , @ "'%@' is not a valid JS method definition - expected 'Module.method' format." , moduleDotMethod ) ;
// Add module if it doesn ' t already exist
NSString * moduleName = parts [ 0 ] ;
NSDictionary * module = localModules [ moduleName ] ;
if ( ! module ) {
module = @ {
@ "moduleID" : @ ( localModules . count ) ,
@ "methods" : [ [ NSMutableDictionary alloc ] init ]
} ;
localModules [ moduleName ] = module ;
}
// Add method if it doesn ' t already exist
NSString * methodName = parts [ 1 ] ;
NSMutableDictionary * methods = module [ @ "methods" ] ;
if ( ! methods [ methodName ] ) {
methods [ methodName ] = @ {
@ "methodID" : @ ( methods . count ) ,
@ "type" : @ "local"
} ;
}
// Add module and method lookup
RCTLocalModuleIDs [ moduleDotMethod ] = module [ @ "moduleID" ] ;
RCTLocalMethodIDs [ moduleDotMethod ] = methods [ methodName ] [ @ "methodID" ] ;
}
} ) ;
return localModules ;
}
2015-01-29 17:10:49 -08:00
@ implementation RCTBridge
{
NSMutableDictionary * _moduleInstances ;
id < RCTJavaScriptExecutor > _javaScriptExecutor ;
}
static id < RCTJavaScriptExecutor > _latestJSExecutor ;
- ( instancetype ) initWithJavaScriptExecutor : ( id < RCTJavaScriptExecutor > ) javaScriptExecutor
{
if ( ( self = [ super init ] ) ) {
_javaScriptExecutor = javaScriptExecutor ;
_latestJSExecutor = _javaScriptExecutor ;
2015-02-03 16:02:36 -08:00
_eventDispatcher = [ [ RCTEventDispatcher alloc ] initWithBridge : self ] ;
2015-02-03 16:12:07 -08:00
_shadowQueue = dispatch_queue _create ( "com.facebook.ReactKit.ShadowQueue" , DISPATCH_QUEUE _SERIAL ) ;
2015-02-06 15:43:59 -08:00
2015-02-03 16:15:20 -08:00
// Instantiate modules
2015-02-03 16:12:07 -08:00
_moduleInstances = [ [ NSMutableDictionary alloc ] init ] ;
2015-02-06 15:43:59 -08:00
[ RCTBridgeModuleClasses ( ) enumerateKeysAndObjectsUsingBlock : ^ ( NSString * moduleName , Class moduleClass , BOOL * stop ) {
2015-01-29 17:10:49 -08:00
if ( _moduleInstances [ moduleName ] = = nil ) {
2015-02-03 16:12:07 -08:00
if ( [ moduleClass instancesRespondToSelector : @ selector ( initWithBridge : ) ] ) {
_moduleInstances [ moduleName ] = [ [ moduleClass alloc ] initWithBridge : self ] ;
} else {
_moduleInstances [ moduleName ] = [ [ moduleClass alloc ] init ] ;
}
2015-01-29 17:10:49 -08:00
}
} ] ;
2015-02-06 15:43:59 -08:00
// Inject module data into JS context
NSString * configJSON = RCTJSONStringify ( @ {
@ "remoteModuleConfig" : RCTRemoteModulesConfig ( ) ,
@ "localModulesConfig" : RCTLocalModulesConfig ( )
} , NULL ) ;
dispatch_semaphore _t semaphore = dispatch_semaphore _create ( 0 ) ;
[ _javaScriptExecutor injectJSONText : configJSON asGlobalObjectNamed : @ "__fbBatchedBridgeConfig" callback : ^ ( id err ) {
dispatch_semaphore _signal ( semaphore ) ;
} ] ;
if ( dispatch_semaphore _wait ( semaphore , dispatch_time ( DISPATCH_TIME _NOW , NSEC_PER _SEC ) ) ! = 0 ) {
RCTLogMustFix ( @ "JavaScriptExecutor took too long to inject JSON object" ) ;
}
2015-01-29 17:10:49 -08:00
}
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
return self ;
}
- ( void ) dealloc
{
2015-02-06 15:43:59 -08:00
RCTAssert ( ! self . valid , @ "must call -invalidate before -dealloc" ) ;
2015-01-29 17:10:49 -08:00
}
# pragma mark - RCTInvalidating
- ( BOOL ) isValid
{
return _javaScriptExecutor ! = nil ;
}
- ( void ) invalidate
{
if ( _latestJSExecutor = = _javaScriptExecutor ) {
_latestJSExecutor = nil ;
}
_javaScriptExecutor = nil ;
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
dispatch_sync ( _shadowQueue , ^ {
2015-02-03 16:12:07 -08:00
// Make sure all dispatchers have been executed before continuing
// TODO : is this still needed ?
2015-01-29 17:10:49 -08:00
} ) ;
2015-02-03 16:12:07 -08:00
2015-01-29 17:10:49 -08:00
for ( id target in _moduleInstances . objectEnumerator ) {
if ( [ target respondsToSelector : @ selector ( invalidate ) ] ) {
[ ( id < RCTInvalidating > ) target invalidate ] ;
}
}
[ _moduleInstances removeAllObjects ] ;
}
/ * *
* - TODO ( #5906496 ) : When we build a ` MessageQueue . m` , handling all the requests could
* cause both a queue of "responses" . We would flush them here . However , we
* currently just expect each objc block to handle its own response sending
* using a ` RCTResponseSenderBlock` .
* /
# pragma mark - RCTBridge methods
/ * *
* Like JS : : call , for objective - c .
* /
2015-02-03 16:15:20 -08:00
- ( void ) enqueueJSCall : ( NSString * ) moduleDotMethod args : ( NSArray * ) args
2015-01-29 17:10:49 -08:00
{
2015-02-06 15:43:59 -08:00
NSNumber * moduleID = RCTLocalModuleIDs [ moduleDotMethod ] ;
2015-02-03 16:15:20 -08:00
RCTAssert ( moduleID , @ "Module '%@' not registered." ,
[ [ moduleDotMethod componentsSeparatedByString : @ "." ] firstObject ] ) ;
2015-02-06 15:43:59 -08:00
NSNumber * methodID = RCTLocalMethodIDs [ moduleDotMethod ] ;
2015-02-03 16:15:20 -08:00
RCTAssert ( methodID , @ "Method '%@' not registered." , moduleDotMethod ) ;
[ self _invokeAndProcessModule : @ "BatchedBridge"
method : @ "callFunctionReturnFlushedQueue"
arguments : @ [ moduleID , methodID , args ] ] ;
2015-01-29 17:10:49 -08:00
}
- ( void ) enqueueApplicationScript : ( NSString * ) script url : ( NSURL * ) url onComplete : ( RCTJavaScriptCompleteBlock ) onComplete
{
RCTAssert ( onComplete ! = nil , @ "onComplete block passed in should be non-nil" ) ;
[ _javaScriptExecutor executeApplicationScript : script sourceURL : url onComplete : ^ ( NSError * scriptLoadError ) {
if ( scriptLoadError ) {
onComplete ( scriptLoadError ) ;
return ;
}
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
[ _javaScriptExecutor executeJSCall : @ "BatchedBridge"
method : @ "flushedQueue"
arguments : @ [ ]
callback : ^ ( id objcValue , NSError * error ) {
[ self _handleBuffer : objcValue ] ;
onComplete ( error ) ;
} ] ;
} ] ;
}
# pragma mark - Payload Generation
- ( void ) _invokeAndProcessModule : ( NSString * ) module method : ( NSString * ) method arguments : ( NSArray * ) args
{
NSTimeInterval startJS = RCTTGetAbsoluteTime ( ) ;
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
RCTJavaScriptCallback processResponse = ^ ( id objcValue , NSError * error ) {
NSTimeInterval startNative = RCTTGetAbsoluteTime ( ) ;
[ self _handleBuffer : objcValue ] ;
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
NSTimeInterval end = RCTTGetAbsoluteTime ( ) ;
NSTimeInterval timeJS = startNative - startJS ;
NSTimeInterval timeNative = end - startNative ;
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
// TODO : surface this performance information somewhere
[ [ NSNotificationCenter defaultCenter ] postNotificationName : @ "PERF" object : nil userInfo : @ { @ "JS" : @ ( timeJS * 1000000 ) , @ "Native" : @ ( timeNative * 1000000 ) } ] ;
} ;
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
[ _javaScriptExecutor executeJSCall : module
method : method
arguments : args
callback : processResponse ] ;
}
/ * *
* TODO ( #5906496 ) : Have responses piggy backed on a round trip with ObjC -> JS requests .
* /
- ( void ) _sendResponseToJavaScriptCallbackID : ( NSInteger ) cbID args : ( NSArray * ) args
{
[ self _invokeAndProcessModule : @ "BatchedBridge"
method : @ "invokeCallbackAndReturnFlushedQueue"
arguments : @ [ @ ( cbID ) , args ] ] ;
}
# pragma mark - Payload Processing
- ( void ) _handleBuffer : ( id ) buffer
{
if ( buffer = = nil || buffer = = ( id ) kCFNull ) {
return ;
}
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
if ( ! [ buffer isKindOfClass : [ NSArray class ] ] ) {
RCTLogMustFix ( @ "Buffer must be an instance of NSArray, got %@" , NSStringFromClass ( [ buffer class ] ) ) ;
return ;
}
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
NSArray * requestsArray = ( NSArray * ) buffer ;
NSUInteger bufferRowCount = [ requestsArray count ] ;
NSUInteger expectedFieldsCount = RCTBridgeFieldResponseReturnValues + 1 ;
if ( bufferRowCount ! = expectedFieldsCount ) {
RCTLogMustFix ( @ "Must pass all fields to buffer - expected %zd, saw %zd" , expectedFieldsCount , bufferRowCount ) ;
return ;
}
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
for ( NSUInteger fieldIndex = RCTBridgeFieldRequestModuleIDs ; fieldIndex <= RCTBridgeFieldParamss ; fieldIndex + + ) {
id field = [ requestsArray objectAtIndex : fieldIndex ] ;
if ( ! [ field isKindOfClass : [ NSArray class ] ] ) {
RCTLogMustFix ( @ "Field at index %zd in buffer must be an instance of NSArray, got %@" , fieldIndex , NSStringFromClass ( [ field class ] ) ) ;
return ;
}
}
2015-02-06 15:43:59 -08:00
2015-02-03 16:12:07 -08:00
NSArray * moduleIDs = requestsArray [ RCTBridgeFieldRequestModuleIDs ] ;
NSArray * methodIDs = requestsArray [ RCTBridgeFieldMethodIDs ] ;
NSArray * paramsArrays = requestsArray [ RCTBridgeFieldParamss ] ;
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
NSUInteger numRequests = [ moduleIDs count ] ;
2015-02-03 16:12:07 -08:00
BOOL allSame = numRequests = = [ methodIDs count ] && numRequests = = [ paramsArrays count ] ;
2015-01-29 17:10:49 -08:00
if ( ! allSame ) {
RCTLogMustFix ( @ "Invalid data message - all must be length: %zd" , numRequests ) ;
return ;
}
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
for ( NSUInteger i = 0 ; i < numRequests ; i + + ) {
@ autoreleasepool {
[ self _handleRequestNumber : i
2015-02-06 15:43:59 -08:00
moduleID : moduleIDs [ i ]
2015-02-03 16:12:07 -08:00
methodID : [ methodIDs [ i ] integerValue ]
params : paramsArrays [ i ] ] ;
2015-01-29 17:10:49 -08:00
}
}
2015-02-06 15:43:59 -08:00
2015-02-03 16:12:07 -08:00
// TODO : only used by RCTUIManager - can we eliminate this special case ?
dispatch_async ( _shadowQueue , ^ {
for ( id target in _moduleInstances . objectEnumerator ) {
if ( [ target respondsToSelector : @ selector ( batchDidComplete ) ] ) {
2015-01-29 17:10:49 -08:00
[ target batchDidComplete ] ;
2015-02-03 16:12:07 -08:00
}
2015-01-29 17:10:49 -08:00
}
2015-02-03 16:12:07 -08:00
} ) ;
2015-01-29 17:10:49 -08:00
}
2015-02-03 16:12:07 -08:00
- ( BOOL ) _handleRequestNumber : ( NSUInteger ) i
2015-02-06 15:43:59 -08:00
moduleID : ( NSNumber * ) moduleID
2015-02-03 16:12:07 -08:00
methodID : ( NSInteger ) methodID
params : ( NSArray * ) params
2015-01-29 17:10:49 -08:00
{
2015-02-03 16:12:07 -08:00
if ( ! [ params isKindOfClass : [ NSArray class ] ] ) {
2015-01-29 17:10:49 -08:00
RCTLogMustFix ( @ "Invalid module/method/params tuple for request #%zd" , i ) ;
2015-02-03 16:12:07 -08:00
return NO ;
2015-01-29 17:10:49 -08:00
}
2015-02-03 16:12:07 -08:00
2015-02-06 15:43:59 -08:00
NSString * moduleName = RCTRemoteModulesByID [ moduleID ] ;
if ( ! moduleName ) {
RCTLogMustFix ( @ "Unknown moduleID: %@" , moduleID ) ;
2015-01-29 17:10:49 -08:00
return NO ;
}
2015-02-03 16:12:07 -08:00
2015-01-29 17:10:49 -08:00
NSArray * methods = RCTExportedMethodsByModule ( ) [ moduleName ] ;
2015-02-06 15:43:59 -08:00
if ( methodID >= methods . count ) {
RCTLogMustFix ( @ "Unknown methodID: %zd for module: %@" , methodID , moduleName ) ;
2015-01-29 17:10:49 -08:00
return NO ;
}
2015-02-03 16:12:07 -08:00
2015-01-29 17:10:49 -08:00
RCTModuleMethod * method = methods [ methodID ] ;
NSUInteger methodArity = method . arity ;
if ( params . count ! = methodArity ) {
2015-02-03 16:12:07 -08:00
RCTLogMustFix ( @ "Expected %tu arguments but got %tu invoking %@.%@" ,
methodArity ,
params . count ,
moduleName ,
method . JSMethodName ) ;
2015-01-29 17:10:49 -08:00
return NO ;
}
2015-02-03 16:12:07 -08:00
2015-01-29 17:10:49 -08:00
__weak RCTBridge * weakSelf = self ;
dispatch_async ( _shadowQueue , ^ {
__strong RCTBridge * strongSelf = weakSelf ;
2015-02-03 16:12:07 -08:00
2015-01-29 17:10:49 -08:00
if ( ! strongSelf . isValid ) {
// strongSelf has been invalidated since the dispatch_async call and this
// invocation should not continue .
return ;
}
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
// TODO : we should just store module instances by index , since that ' s how we look them up anyway
id target = strongSelf -> _moduleInstances [ moduleName ] ;
RCTAssert ( target ! = nil , @ "No module found for name '%@'" , moduleName ) ;
SEL selector = method . selector ;
2015-02-03 16:15:20 -08:00
NSMethodSignature * methodSignature = [ target methodSignatureForSelector : selector ] ;
NSInvocation * invocation = [ NSInvocation invocationWithMethodSignature : methodSignature ] ;
[ invocation setArgument : & target atIndex : 0 ] ;
2015-01-29 17:10:49 -08:00
[ invocation setArgument : & selector atIndex : 1 ] ;
2015-02-03 16:12:07 -08:00
2015-01-29 17:10:49 -08:00
// Retain used blocks until after invocation completes .
2015-02-03 16:15:20 -08:00
NS_VALID _UNTIL _END _OF _SCOPE NSMutableArray * blocks = [ NSMutableArray array ] ;
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
[ params enumerateObjectsUsingBlock : ^ ( id param , NSUInteger idx , BOOL * stop ) {
if ( [ param isEqual : [ NSNull null ] ] ) {
param = nil ;
} else if ( [ method . blockArgumentIndexes containsIndex : idx ] ) {
id block = [ strongSelf createResponseSenderBlock : [ param integerValue ] ] ;
[ blocks addObject : block ] ;
param = block ;
}
2015-02-06 15:43:59 -08:00
2015-02-03 16:15:20 -08:00
NSUInteger argIdx = idx + 2 ;
2015-02-06 15:43:59 -08:00
// TODO : can we do this lookup in advance and cache the logic instead of
// recalculating it every time for every parameter ?
2015-02-03 16:15:20 -08:00
BOOL shouldSet = YES ;
const char * argumentType = [ methodSignature getArgumentTypeAtIndex : argIdx ] ;
switch ( argumentType [ 0 ] ) {
case ' : ' :
if ( [ param isKindOfClass : [ NSString class ] ] ) {
SEL selector = NSSelectorFromString ( param ) ;
[ invocation setArgument : & selector atIndex : argIdx ] ;
shouldSet = NO ;
}
break ;
2015-02-06 15:43:59 -08:00
2015-02-03 16:15:20 -08:00
case ' * ' :
if ( [ param isKindOfClass : [ NSString class ] ] ) {
const char * string = [ param UTF8String ] ;
[ invocation setArgument : & string atIndex : argIdx ] ;
shouldSet = NO ;
}
break ;
2015-02-06 15:43:59 -08:00
// TODO : it seems like an error if the param doesn ' t respond
// so we should probably surface that error rather than failing silently
2015-02-03 16:15:20 -08:00
# define CASE ( _value , _type , _selector ) \
2015-02-06 15:43:59 -08:00
case _value : \
if ( [ param respondsToSelector : @ selector ( _selector ) ] ) { \
_type value = [ param _selector ] ; \
[ invocation setArgument : & value atIndex : argIdx ] ; \
shouldSet = NO ; \
} \
break ;
CASE ( ' c ' , char , charValue )
CASE ( ' C ' , unsigned char , unsignedCharValue )
CASE ( ' s ' , short , shortValue )
CASE ( ' S ' , unsigned short , unsignedShortValue )
CASE ( ' i ' , int , intValue )
CASE ( ' I ' , unsigned int , unsignedIntValue )
CASE ( ' l ' , long , longValue )
CASE ( ' L ' , unsigned long , unsignedLongValue )
CASE ( ' q ' , long long , longLongValue )
CASE ( ' Q ' , unsigned long long , unsignedLongLongValue )
CASE ( ' f ' , float , floatValue )
CASE ( ' d ' , double , doubleValue )
CASE ( ' B ' , BOOL , boolValue )
2015-02-03 16:15:20 -08:00
default :
break ;
}
2015-02-06 15:43:59 -08:00
2015-02-03 16:15:20 -08:00
if ( shouldSet ) {
2015-02-06 15:43:59 -08:00
[ invocation setArgument : & param atIndex : argIdx ] ;
2015-02-03 16:15:20 -08:00
}
2015-01-29 17:10:49 -08:00
} ] ;
2015-02-03 16:12:07 -08:00
2015-01-29 17:10:49 -08:00
@ try {
[ invocation invoke ] ;
}
@ catch ( NSException * exception ) {
RCTLogMustFix ( @ "Exception thrown while invoking %@ on target %@ with params %@: %@" , method . JSMethodName , target , params , exception ) ;
}
} ) ;
2015-02-03 16:12:07 -08:00
2015-01-29 17:10:49 -08:00
return YES ;
}
2015-02-03 16:12:07 -08:00
/ * *
* Returns a callback that reports values back to the JS thread .
* TODO ( #5906496 ) : These responses should go into their own queue ` MessageQueue . m` that
* mirrors the JS queue and protocol . For now , we speak the "language" of the JS
* queue by packing it into an array that matches the wire protocol .
* /
- ( RCTResponseSenderBlock ) createResponseSenderBlock : ( NSInteger ) cbID
{
if ( ! cbID ) {
return nil ;
}
2015-02-06 15:43:59 -08:00
2015-02-03 16:12:07 -08:00
return ^ ( NSArray * args ) {
[ self _sendResponseToJavaScriptCallbackID : cbID args : args ] ;
} ;
}
+ ( NSInvocation * ) invocationForAdditionalArguments : ( NSUInteger ) argCount
{
static NSMutableDictionary * invocations ;
static dispatch_once _t onceToken ;
dispatch_once ( & onceToken , ^ {
invocations = [ NSMutableDictionary dictionary ] ;
} ) ;
2015-02-06 15:43:59 -08:00
2015-02-03 16:12:07 -08:00
id key = @ ( argCount ) ;
NSInvocation * invocation = invocations [ key ] ;
if ( invocation = = nil ) {
NSString * objCTypes = [ @ "v@:" stringByPaddingToLength : 3 + argCount withString : @ "@" startingAtIndex : 0 ] ;
NSMethodSignature * methodSignature = [ NSMethodSignature signatureWithObjCTypes : objCTypes . UTF8String ] ;
invocation = [ NSInvocation invocationWithMethodSignature : methodSignature ] ;
invocations [ key ] = invocation ;
}
2015-02-03 16:15:20 -08:00
2015-02-06 15:43:59 -08:00
return invocation ;
2015-01-29 17:10:49 -08:00
}
2015-02-03 16:12:07 -08:00
- ( void ) registerRootView : ( RCTRootView * ) rootView
{
// TODO : only used by RCTUIManager - can we eliminate this special case ?
for ( id target in _moduleInstances . objectEnumerator ) {
if ( [ target respondsToSelector : @ selector ( registerRootView : ) ] ) {
[ target registerRootView : rootView ] ;
}
}
}
2015-01-29 17:10:49 -08:00
+ ( BOOL ) hasValidJSExecutor
{
return ( _latestJSExecutor ! = nil && [ _latestJSExecutor isValid ] ) ;
}
+ ( void ) log : ( NSArray * ) objects level : ( NSString * ) level
{
if ( ! _latestJSExecutor || ! [ _latestJSExecutor isValid ] ) {
RCTLogError ( @ "%@" , RCTLogFormatString ( @ "ERROR: No valid JS executor to log %@." , objects ) ) ;
return ;
}
NSMutableArray * args = [ NSMutableArray arrayWithObject : level ] ;
2015-02-06 15:43:59 -08:00
2015-01-29 17:10:49 -08:00
// TODO ( #5906496 ) : Find out and document why we skip the first object
for ( id ob in [ objects subarrayWithRange : ( NSRange ) { 1 , [ objects count ] - 1 } ] ) {
if ( [ NSJSONSerialization isValidJSONObject : @ [ ob ] ] ) {
[ args addObject : ob ] ;
} else {
[ args addObject : [ ob description ] ] ;
}
}
// Note the js executor could get invalidated while we ' re trying to call this . . . need to watch out for that .
[ _latestJSExecutor executeJSCall : @ "RCTLog"
method : @ "logIfNoNativeHook"
arguments : args
callback : ^ ( id objcValue , NSError * error ) { } ] ;
}
@ end