2017-04-20 15:20:03 +00: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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#import "RCTPackagerConnection.h"
|
|
|
|
|
|
|
|
#import <objc/runtime.h>
|
|
|
|
|
|
|
|
#import <React/RCTAssert.h>
|
|
|
|
#import <React/RCTBridge.h>
|
2017-10-05 02:06:22 +00:00
|
|
|
#import <React/RCTBundleURLProvider.h>
|
2017-05-05 15:32:15 +00:00
|
|
|
#import <React/RCTConvert.h>
|
|
|
|
#import <React/RCTDefines.h>
|
|
|
|
#import <React/RCTLog.h>
|
|
|
|
#import <React/RCTReconnectingWebSocket.h>
|
|
|
|
#import <React/RCTSRWebSocket.h>
|
|
|
|
#import <React/RCTUtils.h>
|
2017-04-20 15:20:03 +00:00
|
|
|
|
2017-06-15 18:52:05 +00:00
|
|
|
#import "RCTPackagerConnectionBridgeConfig.h"
|
2017-04-20 15:20:03 +00:00
|
|
|
#import "RCTReloadPackagerMethod.h"
|
|
|
|
#import "RCTSamplingProfilerPackagerMethod.h"
|
|
|
|
|
|
|
|
#if RCT_DEV
|
|
|
|
|
2017-10-05 02:06:22 +00:00
|
|
|
static dispatch_queue_t RCTPackagerConnectionQueue()
|
|
|
|
{
|
|
|
|
static dispatch_queue_t queue;
|
|
|
|
static dispatch_once_t onceToken;
|
|
|
|
dispatch_once(&onceToken, ^{
|
|
|
|
queue = dispatch_queue_create("com.facebook.RCTPackagerConnectionQueue", DISPATCH_QUEUE_SERIAL);
|
|
|
|
});
|
|
|
|
return queue;
|
|
|
|
};
|
|
|
|
|
2017-05-05 15:32:15 +00:00
|
|
|
@interface RCTPackagerConnection () <RCTWebSocketProtocolDelegate>
|
|
|
|
@end
|
|
|
|
|
2017-04-20 15:20:03 +00:00
|
|
|
@implementation RCTPackagerConnection {
|
2017-06-15 18:52:05 +00:00
|
|
|
NSURL *_packagerURL;
|
2017-05-05 15:32:15 +00:00
|
|
|
RCTReconnectingWebSocket *_socket;
|
|
|
|
NSMutableDictionary<NSString *, id<RCTPackagerClientMethod>> *_handlers;
|
2017-04-20 15:20:03 +00:00
|
|
|
}
|
|
|
|
|
2017-10-05 02:06:22 +00:00
|
|
|
+ (void)checkDefaultConnectionWithCallback:(void (^)(BOOL isRunning))callback
|
|
|
|
queue:(dispatch_queue_t)queue
|
|
|
|
{
|
|
|
|
RCTBundleURLProvider *const settings = [RCTBundleURLProvider sharedSettings];
|
|
|
|
NSURLComponents *components = [NSURLComponents new];
|
|
|
|
components.scheme = @"http";
|
|
|
|
components.host = settings.jsLocation ?: @"localhost";
|
|
|
|
components.port = @(kRCTBundleURLProviderDefaultPort);
|
|
|
|
components.path = @"/status";
|
|
|
|
[NSURLConnection sendAsynchronousRequest:[NSURLRequest requestWithURL:components.URL]
|
|
|
|
queue:[NSOperationQueue mainQueue]
|
|
|
|
completionHandler:^(NSURLResponse *response, NSData *data, NSError *connectionError) {
|
|
|
|
NSString *const status = data != nil
|
|
|
|
? [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]
|
|
|
|
: nil;
|
|
|
|
BOOL isRunning = [status isEqualToString:@"packager-status:running"];
|
|
|
|
|
|
|
|
dispatch_async(queue, ^{
|
|
|
|
callback(isRunning);
|
|
|
|
});
|
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2017-06-15 18:52:05 +00:00
|
|
|
+ (instancetype)connectionForBridge:(RCTBridge *)bridge
|
2017-04-20 15:20:03 +00:00
|
|
|
{
|
2017-06-15 18:52:05 +00:00
|
|
|
RCTPackagerConnectionBridgeConfig *config = [[RCTPackagerConnectionBridgeConfig alloc] initWithBridge:bridge];
|
|
|
|
return [[[self class] alloc] initWithConfig:config];
|
|
|
|
}
|
2017-05-05 15:32:15 +00:00
|
|
|
|
2017-06-15 18:52:05 +00:00
|
|
|
- (instancetype)initWithConfig:(id<RCTPackagerConnectionConfig>)config
|
|
|
|
{
|
|
|
|
if (self = [super init]) {
|
|
|
|
_packagerURL = [config packagerURL];
|
|
|
|
_handlers = [[config defaultPackagerMethods] mutableCopy];
|
2017-05-05 15:32:15 +00:00
|
|
|
[self connect];
|
2017-04-20 15:20:03 +00:00
|
|
|
}
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)connect
|
|
|
|
{
|
|
|
|
RCTAssertMainQueue();
|
|
|
|
|
2017-06-15 18:52:05 +00:00
|
|
|
NSURL *url = _packagerURL;
|
2017-04-20 15:20:03 +00:00
|
|
|
if (!url) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The jsPackagerClient is a static map that holds different packager clients per the packagerURL
|
|
|
|
// In case many instances of DevMenu are created, the latest instance that use the same URL as
|
|
|
|
// previous instances will override given packager client's method handlers
|
2017-05-05 15:32:15 +00:00
|
|
|
static NSMutableDictionary<NSString *, RCTReconnectingWebSocket *> *socketConnections = nil;
|
|
|
|
if (socketConnections == nil) {
|
|
|
|
socketConnections = [NSMutableDictionary new];
|
2017-04-20 15:20:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NSString *key = [url absoluteString];
|
2017-10-05 02:06:22 +00:00
|
|
|
_socket = socketConnections[key];
|
|
|
|
if (!_socket) {
|
|
|
|
_socket = [[RCTReconnectingWebSocket alloc] initWithURL:url];
|
|
|
|
_socket.delegateDispatchQueue = RCTPackagerConnectionQueue();
|
|
|
|
[_socket start];
|
|
|
|
socketConnections[key] = _socket;
|
2017-04-20 15:20:03 +00:00
|
|
|
}
|
|
|
|
|
2017-10-05 02:06:22 +00:00
|
|
|
_socket.delegate = self;
|
2017-04-20 15:20:03 +00:00
|
|
|
}
|
|
|
|
|
2017-10-05 02:06:22 +00:00
|
|
|
- (void)stop
|
|
|
|
{
|
|
|
|
[_socket stop];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-05-05 15:32:15 +00:00
|
|
|
- (void)addHandler:(id<RCTPackagerClientMethod>)handler forMethod:(NSString *)name
|
|
|
|
{
|
2017-10-05 02:06:22 +00:00
|
|
|
@synchronized(self) {
|
|
|
|
_handlers[name] = handler;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (id<RCTPackagerClientMethod>)handlerForMethod:(NSString *)name
|
|
|
|
{
|
|
|
|
@synchronized(self) {
|
|
|
|
return _handlers[name];
|
|
|
|
}
|
2017-05-05 15:32:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static BOOL isSupportedVersion(NSNumber *version)
|
|
|
|
{
|
|
|
|
NSArray<NSNumber *> *const kSupportedVersions = @[ @(RCT_PACKAGER_CLIENT_PROTOCOL_VERSION) ];
|
|
|
|
return [kSupportedVersions containsObject:version];
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma mark - RCTWebSocketProtocolDelegate
|
|
|
|
|
|
|
|
- (void)webSocket:(RCTSRWebSocket *)webSocket didReceiveMessage:(id)message
|
|
|
|
{
|
|
|
|
NSError *error = nil;
|
|
|
|
NSDictionary<NSString *, id> *msg = RCTJSONParse(message, &error);
|
|
|
|
|
|
|
|
if (error) {
|
|
|
|
RCTLogError(@"%@ failed to parse message with error %@\n<message>\n%@\n</message>", [self class], error, msg);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isSupportedVersion(msg[@"version"])) {
|
|
|
|
RCTLogError(@"%@ received message with not supported version %@", [self class], msg[@"version"]);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-10-05 02:06:22 +00:00
|
|
|
id<RCTPackagerClientMethod> methodHandler = [self handlerForMethod:msg[@"method"]];
|
2017-05-05 15:32:15 +00:00
|
|
|
if (!methodHandler) {
|
|
|
|
if (msg[@"id"]) {
|
|
|
|
NSString *errorMsg = [NSString stringWithFormat:@"%@ no handler found for method %@", [self class], msg[@"method"]];
|
|
|
|
RCTLogError(errorMsg, msg[@"method"]);
|
|
|
|
[[[RCTPackagerClientResponder alloc] initWithId:msg[@"id"]
|
|
|
|
socket:webSocket] respondWithError:errorMsg];
|
|
|
|
}
|
|
|
|
return; // If it was a broadcast then we ignore it gracefully
|
|
|
|
}
|
|
|
|
|
2017-10-05 02:06:22 +00:00
|
|
|
dispatch_queue_t methodQueue = [methodHandler respondsToSelector:@selector(methodQueue)]
|
|
|
|
? [methodHandler methodQueue]
|
|
|
|
: dispatch_get_main_queue();
|
|
|
|
|
|
|
|
dispatch_async(methodQueue, ^{
|
|
|
|
if (msg[@"id"]) {
|
|
|
|
[methodHandler handleRequest:msg[@"params"]
|
|
|
|
withResponder:[[RCTPackagerClientResponder alloc] initWithId:msg[@"id"]
|
|
|
|
socket:webSocket]];
|
|
|
|
} else {
|
|
|
|
[methodHandler handleNotification:msg[@"params"]];
|
|
|
|
}
|
|
|
|
});
|
2017-05-05 15:32:15 +00:00
|
|
|
}
|
|
|
|
|
2017-08-29 14:59:40 +00:00
|
|
|
- (void)webSocketDidOpen:(RCTSRWebSocket *)webSocket
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-08-07 13:57:56 +00:00
|
|
|
- (void)webSocket:(RCTSRWebSocket *)webSocket didCloseWithCode:(NSInteger)code reason:(NSString *)reason wasClean:(BOOL)wasClean
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-04-20 15:20:03 +00:00
|
|
|
@end
|
|
|
|
|
|
|
|
#endif
|