2015-03-23 13:28:42 -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-03-19 09:55:40 -07:00
|
|
|
|
2015-04-21 09:48:29 -07:00
|
|
|
#import "RCTDefines.h"
|
|
|
|
|
|
|
|
#if RCT_DEV // Debug executors are only supported in dev mode
|
|
|
|
|
2015-03-19 09:55:40 -07:00
|
|
|
#import "RCTWebSocketExecutor.h"
|
|
|
|
|
|
|
|
#import "RCTLog.h"
|
2015-04-11 15:08:00 -07:00
|
|
|
#import "RCTSparseArray.h"
|
2015-03-19 09:55:40 -07:00
|
|
|
#import "RCTUtils.h"
|
|
|
|
#import "SRWebSocket.h"
|
|
|
|
|
|
|
|
typedef void (^WSMessageCallback)(NSError *error, NSDictionary *reply);
|
|
|
|
|
|
|
|
@interface RCTWebSocketExecutor () <SRWebSocketDelegate>
|
|
|
|
@end
|
|
|
|
|
2015-04-11 15:08:00 -07:00
|
|
|
@implementation RCTWebSocketExecutor
|
|
|
|
{
|
2015-03-19 09:55:40 -07:00
|
|
|
SRWebSocket *_socket;
|
2015-04-11 15:08:00 -07:00
|
|
|
dispatch_queue_t _jsQueue;
|
|
|
|
RCTSparseArray *_callbacks;
|
2015-03-19 09:55:40 -07:00
|
|
|
dispatch_semaphore_t _socketOpenSemaphore;
|
|
|
|
NSMutableDictionary *_injectedObjects;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (instancetype)init
|
|
|
|
{
|
2015-03-19 12:10:41 -07:00
|
|
|
return [self initWithURL:[NSURL URLWithString:@"http://localhost:8081/debugger-proxy"]];
|
2015-03-19 09:55:40 -07:00
|
|
|
}
|
|
|
|
|
2015-04-11 15:08:00 -07:00
|
|
|
- (instancetype)initWithURL:(NSURL *)URL
|
2015-03-19 09:55:40 -07:00
|
|
|
{
|
|
|
|
if (self = [super init]) {
|
|
|
|
|
2015-04-11 15:08:00 -07:00
|
|
|
_jsQueue = dispatch_queue_create("com.facebook.React.WebSocketExecutor", DISPATCH_QUEUE_SERIAL);
|
|
|
|
_socket = [[SRWebSocket alloc] initWithURL:URL];
|
|
|
|
_socket.delegate = self;
|
|
|
|
_callbacks = [[RCTSparseArray alloc] init];
|
|
|
|
_injectedObjects = [[NSMutableDictionary alloc] init];
|
|
|
|
[_socket setDelegateDispatchQueue:_jsQueue];
|
2015-03-19 12:10:41 -07:00
|
|
|
|
2015-04-11 15:08:00 -07:00
|
|
|
NSURL *startDevToolsURL = [NSURL URLWithString:@"/launch-chrome-devtools" relativeToURL:URL];
|
2015-03-19 12:10:41 -07:00
|
|
|
[NSURLConnection connectionWithRequest:[NSURLRequest requestWithURL:startDevToolsURL] delegate:nil];
|
|
|
|
|
2015-03-19 09:55:40 -07:00
|
|
|
if (![self connectToProxy]) {
|
2015-04-11 15:08:00 -07:00
|
|
|
RCTLogError(@"Connection to %@ timed out. Are you running node proxy? If \
|
|
|
|
you are running on the device, check if you have the right IP \
|
|
|
|
address in `RCTWebSocketExecutor.m`.", URL);
|
2015-03-19 09:55:40 -07:00
|
|
|
[self invalidate];
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
NSInteger retries = 3;
|
|
|
|
BOOL runtimeIsReady = [self prepareJSRuntime];
|
|
|
|
while (!runtimeIsReady && retries > 0) {
|
|
|
|
runtimeIsReady = [self prepareJSRuntime];
|
|
|
|
retries--;
|
|
|
|
}
|
|
|
|
if (!runtimeIsReady) {
|
|
|
|
RCTLogError(@"Runtime is not ready. Do you have Chrome open?");
|
|
|
|
[self invalidate];
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)connectToProxy
|
|
|
|
{
|
|
|
|
_socketOpenSemaphore = dispatch_semaphore_create(0);
|
|
|
|
[_socket open];
|
|
|
|
long connected = dispatch_semaphore_wait(_socketOpenSemaphore, dispatch_time(DISPATCH_TIME_NOW, NSEC_PER_SEC * 2));
|
|
|
|
return connected == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)prepareJSRuntime
|
|
|
|
{
|
|
|
|
__block NSError *initError;
|
|
|
|
dispatch_semaphore_t s = dispatch_semaphore_create(0);
|
2015-04-20 02:09:11 -07:00
|
|
|
[self sendMessage:@{@"method": @"prepareJSRuntime"} context:nil waitForReply:^(NSError *error, NSDictionary *reply) {
|
2015-03-19 09:55:40 -07:00
|
|
|
initError = error;
|
|
|
|
dispatch_semaphore_signal(s);
|
|
|
|
}];
|
|
|
|
long runtimeIsReady = dispatch_semaphore_wait(s, dispatch_time(DISPATCH_TIME_NOW, NSEC_PER_SEC));
|
|
|
|
return runtimeIsReady == 0 && initError == nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)webSocket:(SRWebSocket *)webSocket didReceiveMessage:(id)message
|
|
|
|
{
|
|
|
|
NSError *error = nil;
|
|
|
|
NSDictionary *reply = RCTJSONParse(message, &error);
|
2015-04-11 15:08:00 -07:00
|
|
|
NSNumber *messageID = reply[@"replyID"];
|
|
|
|
WSMessageCallback callback = _callbacks[messageID];
|
2015-03-19 09:55:40 -07:00
|
|
|
if (callback) {
|
|
|
|
callback(error, reply);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)webSocketDidOpen:(SRWebSocket *)webSocket
|
|
|
|
{
|
|
|
|
dispatch_semaphore_signal(_socketOpenSemaphore);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)webSocket:(SRWebSocket *)webSocket didFailWithError:(NSError *)error
|
|
|
|
{
|
|
|
|
RCTLogError(@"WebSocket connection failed with error %@", error);
|
|
|
|
}
|
|
|
|
|
2015-04-20 02:09:11 -07:00
|
|
|
- (void)sendMessage:(NSDictionary *)message context:(NSNumber *)executorID waitForReply:(WSMessageCallback)callback
|
2015-03-19 09:55:40 -07:00
|
|
|
{
|
|
|
|
static NSUInteger lastID = 10000;
|
|
|
|
|
2015-04-11 15:08:00 -07:00
|
|
|
dispatch_async(_jsQueue, ^{
|
2015-03-19 09:55:40 -07:00
|
|
|
if (!self.valid) {
|
2015-04-07 07:36:26 -07:00
|
|
|
NSError *error = [NSError errorWithDomain:@"WS" code:1 userInfo:@{
|
|
|
|
NSLocalizedDescriptionKey: @"socket closed"
|
|
|
|
}];
|
2015-03-19 09:55:40 -07:00
|
|
|
callback(error, nil);
|
|
|
|
return;
|
2015-04-20 02:09:11 -07:00
|
|
|
} else if (executorID && ![RCTGetExecutorID(self) isEqualToNumber:executorID]) {
|
|
|
|
return;
|
2015-03-19 09:55:40 -07:00
|
|
|
}
|
|
|
|
|
2015-04-11 15:08:00 -07:00
|
|
|
NSNumber *expectedID = @(lastID++);
|
|
|
|
_callbacks[expectedID] = [callback copy];
|
2015-03-19 09:55:40 -07:00
|
|
|
NSMutableDictionary *messageWithID = [message mutableCopy];
|
2015-04-11 15:08:00 -07:00
|
|
|
messageWithID[@"id"] = expectedID;
|
2015-03-19 09:55:40 -07:00
|
|
|
[_socket send:RCTJSONStringify(messageWithID, NULL)];
|
2015-04-11 15:08:00 -07:00
|
|
|
});
|
2015-03-19 09:55:40 -07:00
|
|
|
}
|
|
|
|
|
2015-04-11 15:08:00 -07:00
|
|
|
- (void)executeApplicationScript:(NSString *)script sourceURL:(NSURL *)URL onComplete:(RCTJavaScriptCompleteBlock)onComplete
|
2015-03-19 09:55:40 -07:00
|
|
|
{
|
2015-04-20 14:35:45 -07:00
|
|
|
NSDictionary *message = @{@"method": @"executeApplicationScript", @"url": [URL absoluteString], @"inject": _injectedObjects};
|
2015-04-20 02:09:11 -07:00
|
|
|
[self sendMessage:message context:nil waitForReply:^(NSError *error, NSDictionary *reply) {
|
2015-03-19 09:55:40 -07:00
|
|
|
onComplete(error);
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2015-04-20 02:09:11 -07:00
|
|
|
- (void)executeJSCall:(NSString *)name method:(NSString *)method arguments:(NSArray *)arguments context:(NSNumber *)executorID callback:(RCTJavaScriptCallback)onComplete
|
2015-03-19 09:55:40 -07:00
|
|
|
{
|
|
|
|
RCTAssert(onComplete != nil, @"callback was missing for exec JS call");
|
2015-04-11 15:08:00 -07:00
|
|
|
NSDictionary *message = @{
|
2015-04-20 14:35:45 -07:00
|
|
|
@"method": @"executeJSCall",
|
2015-04-11 15:08:00 -07:00
|
|
|
@"moduleName": name,
|
|
|
|
@"moduleMethod": method,
|
|
|
|
@"arguments": arguments
|
|
|
|
};
|
2015-04-20 02:09:11 -07:00
|
|
|
[self sendMessage:message context:executorID waitForReply:^(NSError *socketError, NSDictionary *reply) {
|
2015-03-19 09:55:40 -07:00
|
|
|
if (socketError) {
|
|
|
|
onComplete(nil, socketError);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
NSString *result = reply[@"result"];
|
|
|
|
id objcValue = RCTJSONParse(result, NULL);
|
|
|
|
onComplete(objcValue, nil);
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)injectJSONText:(NSString *)script asGlobalObjectNamed:(NSString *)objectName callback:(RCTJavaScriptCompleteBlock)onComplete
|
|
|
|
{
|
2015-04-11 15:08:00 -07:00
|
|
|
dispatch_async(_jsQueue, ^{
|
|
|
|
_injectedObjects[objectName] = script;
|
2015-03-19 09:55:40 -07:00
|
|
|
onComplete(nil);
|
2015-04-11 15:08:00 -07:00
|
|
|
});
|
2015-03-19 09:55:40 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)invalidate
|
|
|
|
{
|
|
|
|
_socket.delegate = nil;
|
|
|
|
[_socket closeWithCode:1000 reason:@"Invalidated"];
|
|
|
|
_socket = nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isValid
|
|
|
|
{
|
|
|
|
return _socket != nil && _socket.readyState == SR_OPEN;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)dealloc
|
|
|
|
{
|
|
|
|
RCTAssert(!self.valid, @"-invalidate must be called before -dealloc");
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|
2015-04-21 09:48:29 -07:00
|
|
|
|
|
|
|
#endif
|