mirror of
https://github.com/zhigang1992/react-native.git
synced 2026-04-28 12:15:37 +08:00
Merge RCTPackagerClient into RCTPackagerConnection
Reviewed By: mmmulani Differential Revision: D4988204 fbshipit-source-id: 78e0df5268bfc11e4e0edf8e60494e55022cd9f2
This commit is contained in:
committed by
Facebook Github Bot
parent
71e84e6ee8
commit
e7680131d7
@@ -8,10 +8,14 @@
|
||||
*/
|
||||
|
||||
#import <React/RCTDefines.h>
|
||||
#import <React/RCTPackagerClientResponder.h>
|
||||
|
||||
#if RCT_DEV // Only supported in dev mode
|
||||
|
||||
@class RCTPackagerClientResponder;
|
||||
@class RCTSRWebSocket;
|
||||
|
||||
extern const int RCT_PACKAGER_CLIENT_PROTOCOL_VERSION;
|
||||
|
||||
@protocol RCTPackagerClientMethod
|
||||
|
||||
- (void)handleRequest:(id)params withResponder:(RCTPackagerClientResponder *)responder;
|
||||
@@ -19,12 +23,11 @@
|
||||
|
||||
@end
|
||||
|
||||
@interface RCTPackagerClient : NSObject
|
||||
@interface RCTPackagerClientResponder : NSObject
|
||||
|
||||
- (instancetype)initWithURL:(NSURL *)url;
|
||||
- (void)addHandler:(id<RCTPackagerClientMethod>)handler forMethod:(NSString *)name;
|
||||
- (void)start;
|
||||
- (void)stop;
|
||||
- (instancetype)initWithId:(id)msgId socket:(RCTSRWebSocket *)socket;
|
||||
- (void)respondWithResult:(id)result;
|
||||
- (void)respondWithError:(id)error;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@@ -9,93 +9,57 @@
|
||||
|
||||
#import "RCTPackagerClient.h"
|
||||
|
||||
#import <React/RCTConvert.h>
|
||||
#import <React/RCTDefines.h>
|
||||
#import <React/RCTLog.h>
|
||||
#import <React/RCTReconnectingWebSocket.h>
|
||||
#import <React/RCTSRWebSocket.h>
|
||||
#import <React/RCTUtils.h>
|
||||
|
||||
#import "RCTPackagerClientResponder.h"
|
||||
|
||||
#if RCT_DEV // Only supported in dev mode
|
||||
|
||||
@interface RCTPackagerClient () <RCTWebSocketProtocolDelegate>
|
||||
@end
|
||||
const int RCT_PACKAGER_CLIENT_PROTOCOL_VERSION = 2;
|
||||
|
||||
@implementation RCTPackagerClient {
|
||||
RCTReconnectingWebSocket *_socket;
|
||||
NSMutableDictionary<NSString *, id<RCTPackagerClientMethod>> *_handlers;
|
||||
@implementation RCTPackagerClientResponder {
|
||||
id _msgId;
|
||||
__weak RCTSRWebSocket *_socket;
|
||||
}
|
||||
|
||||
- (instancetype)initWithURL:(NSURL *)url
|
||||
- (instancetype)initWithId:(id)msgId socket:(RCTSRWebSocket *)socket
|
||||
{
|
||||
if (self = [super init]) {
|
||||
_socket = [[RCTReconnectingWebSocket alloc] initWithURL:url];
|
||||
_socket.delegate = self;
|
||||
_handlers = [NSMutableDictionary new];
|
||||
_msgId = msgId;
|
||||
_socket = socket;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)addHandler:(id<RCTPackagerClientMethod>)handler forMethod:(NSString *)name
|
||||
- (void)respondWithResult:(id)result
|
||||
{
|
||||
_handlers[name] = handler;
|
||||
}
|
||||
|
||||
- (void)start
|
||||
{
|
||||
_socket.delegate = self;
|
||||
[_socket start];
|
||||
}
|
||||
|
||||
- (void)stop
|
||||
{
|
||||
[_socket stop];
|
||||
}
|
||||
|
||||
- (BOOL)isSupportedVersion:(NSNumber *)version
|
||||
{
|
||||
NSArray<NSNumber *> *const kSupportedVersions = @[ @(RCT_PACKAGER_CLIENT_PROTOCOL_VERSION) ];
|
||||
return [kSupportedVersions containsObject:version];
|
||||
}
|
||||
|
||||
- (void)webSocket:(RCTSRWebSocket *)webSocket didReceiveMessage:(id)message
|
||||
{
|
||||
if (!_handlers) {
|
||||
return;
|
||||
}
|
||||
|
||||
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 (![self isSupportedVersion:msg[@"version"]]) {
|
||||
RCTLogError(@"%@ received message with not supported version %@", [self class], msg[@"version"]);
|
||||
return;
|
||||
}
|
||||
|
||||
id<RCTPackagerClientMethod> methodHandler = _handlers[msg[@"method"]];
|
||||
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
|
||||
}
|
||||
|
||||
if (msg[@"id"]) {
|
||||
[methodHandler handleRequest:msg[@"params"]
|
||||
withResponder:[[RCTPackagerClientResponder alloc] initWithId:msg[@"id"]
|
||||
socket:webSocket]];
|
||||
NSDictionary<NSString *, id> *msg = @{
|
||||
@"version": @(RCT_PACKAGER_CLIENT_PROTOCOL_VERSION),
|
||||
@"id": _msgId,
|
||||
@"result": result,
|
||||
};
|
||||
NSError *jsError = nil;
|
||||
NSString *message = RCTJSONStringify(msg, &jsError);
|
||||
if (jsError) {
|
||||
RCTLogError(@"%@ failed to stringify message with error %@", [self class], jsError);
|
||||
} else {
|
||||
[methodHandler handleNotification:msg[@"params"]];
|
||||
[_socket send:message];
|
||||
}
|
||||
}
|
||||
|
||||
- (void)respondWithError:(id)error
|
||||
{
|
||||
NSDictionary<NSString *, id> *msg = @{
|
||||
@"version": @(RCT_PACKAGER_CLIENT_PROTOCOL_VERSION),
|
||||
@"id": _msgId,
|
||||
@"error": error,
|
||||
};
|
||||
NSError *jsError = nil;
|
||||
NSString *message = RCTJSONStringify(msg, &jsError);
|
||||
if (jsError) {
|
||||
RCTLogError(@"%@ failed to stringify message with error %@", [self class], jsError);
|
||||
} else {
|
||||
[_socket send:message];
|
||||
}
|
||||
}
|
||||
@end
|
||||
|
||||
@@ -1,26 +0,0 @@
|
||||
/**
|
||||
* 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 <React/RCTDefines.h>
|
||||
|
||||
#if RCT_DEV // Only supported in dev mode
|
||||
|
||||
extern const int RCT_PACKAGER_CLIENT_PROTOCOL_VERSION;
|
||||
|
||||
@class RCTSRWebSocket;
|
||||
|
||||
@interface RCTPackagerClientResponder : NSObject
|
||||
|
||||
- (instancetype)initWithId:(id)msgId socket:(RCTSRWebSocket *)socket;
|
||||
- (void)respondWithResult:(id)result;
|
||||
- (void)respondWithError:(id)error;
|
||||
|
||||
@end
|
||||
|
||||
#endif
|
||||
@@ -1,67 +0,0 @@
|
||||
/**
|
||||
* 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 "RCTPackagerClientResponder.h"
|
||||
|
||||
#import <React/RCTLog.h>
|
||||
#import <React/RCTSRWebSocket.h>
|
||||
#import <React/RCTUtils.h>
|
||||
|
||||
#if RCT_DEV // Only supported in dev mode
|
||||
|
||||
const int RCT_PACKAGER_CLIENT_PROTOCOL_VERSION = 2;
|
||||
|
||||
@implementation RCTPackagerClientResponder {
|
||||
id _msgId;
|
||||
__weak RCTSRWebSocket *_socket;
|
||||
}
|
||||
|
||||
- (instancetype)initWithId:(id)msgId socket:(RCTSRWebSocket *)socket
|
||||
{
|
||||
if (self = [super init]) {
|
||||
_msgId = msgId;
|
||||
_socket = socket;
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)respondWithResult:(id)result
|
||||
{
|
||||
NSDictionary<NSString *, id> *msg = @{
|
||||
@"version": @(RCT_PACKAGER_CLIENT_PROTOCOL_VERSION),
|
||||
@"id": _msgId,
|
||||
@"result": result,
|
||||
};
|
||||
NSError *jsError = nil;
|
||||
NSString *message = RCTJSONStringify(msg, &jsError);
|
||||
if (jsError) {
|
||||
RCTLogError(@"%@ failed to stringify message with error %@", [self class], jsError);
|
||||
} else {
|
||||
[_socket send:message];
|
||||
}
|
||||
}
|
||||
|
||||
- (void)respondWithError:(id)error
|
||||
{
|
||||
NSDictionary<NSString *, id> *msg = @{
|
||||
@"version": @(RCT_PACKAGER_CLIENT_PROTOCOL_VERSION),
|
||||
@"id": _msgId,
|
||||
@"error": error,
|
||||
};
|
||||
NSError *jsError = nil;
|
||||
NSString *message = RCTJSONStringify(msg, &jsError);
|
||||
if (jsError) {
|
||||
RCTLogError(@"%@ failed to stringify message with error %@", [self class], jsError);
|
||||
} else {
|
||||
[_socket send:message];
|
||||
}
|
||||
}
|
||||
@end
|
||||
|
||||
#endif
|
||||
@@ -14,6 +14,7 @@
|
||||
#if RCT_DEV
|
||||
|
||||
@class RCTBridge;
|
||||
@protocol RCTPackagerClientMethod;
|
||||
|
||||
/**
|
||||
* Encapsulates connection to React Native packager
|
||||
@@ -21,7 +22,7 @@
|
||||
@interface RCTPackagerConnection : NSObject
|
||||
|
||||
- (instancetype)initWithBridge:(RCTBridge *)bridge;
|
||||
- (void)connect;
|
||||
- (void)addHandler:(id<RCTPackagerClientMethod>)handler forMethod:(NSString *)name;
|
||||
|
||||
@end
|
||||
|
||||
|
||||
@@ -13,22 +13,38 @@
|
||||
|
||||
#import <React/RCTAssert.h>
|
||||
#import <React/RCTBridge.h>
|
||||
#import <React/RCTConvert.h>
|
||||
#import <React/RCTDefines.h>
|
||||
#import <React/RCTLog.h>
|
||||
#import <React/RCTReconnectingWebSocket.h>
|
||||
#import <React/RCTSRWebSocket.h>
|
||||
#import <React/RCTUtils.h>
|
||||
#import <React/RCTWebSocketObserverProtocol.h>
|
||||
|
||||
#import "RCTPackagerClient.h"
|
||||
#import "RCTReloadPackagerMethod.h"
|
||||
#import "RCTSamplingProfilerPackagerMethod.h"
|
||||
|
||||
#if RCT_DEV
|
||||
|
||||
@interface RCTPackagerConnection () <RCTWebSocketProtocolDelegate>
|
||||
@end
|
||||
|
||||
@implementation RCTPackagerConnection {
|
||||
RCTBridge *_bridge;
|
||||
RCTReconnectingWebSocket *_socket;
|
||||
NSMutableDictionary<NSString *, id<RCTPackagerClientMethod>> *_handlers;
|
||||
}
|
||||
|
||||
- (instancetype)initWithBridge:(RCTBridge *)bridge
|
||||
{
|
||||
if (self = [super init]) {
|
||||
_bridge = bridge;
|
||||
|
||||
_handlers = [NSMutableDictionary new];
|
||||
_handlers[@"reload"] = [[RCTReloadPackagerMethod alloc] initWithBridge:_bridge];
|
||||
_handlers[@"pokeSamplingProfiler"] = [[RCTSamplingProfilerPackagerMethod alloc] initWithBridge:_bridge];
|
||||
|
||||
[self connect];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
@@ -45,25 +61,20 @@
|
||||
// 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
|
||||
static NSMutableDictionary<NSString *, RCTPackagerClient *> *jsPackagerClients = nil;
|
||||
if (jsPackagerClients == nil) {
|
||||
jsPackagerClients = [NSMutableDictionary new];
|
||||
static NSMutableDictionary<NSString *, RCTReconnectingWebSocket *> *socketConnections = nil;
|
||||
if (socketConnections == nil) {
|
||||
socketConnections = [NSMutableDictionary new];
|
||||
}
|
||||
|
||||
NSString *key = [url absoluteString];
|
||||
RCTPackagerClient *packagerClient = jsPackagerClients[key];
|
||||
if (!packagerClient) {
|
||||
packagerClient = [[RCTPackagerClient alloc] initWithURL:url];
|
||||
jsPackagerClients[key] = packagerClient;
|
||||
} else {
|
||||
[packagerClient stop];
|
||||
RCTReconnectingWebSocket *webSocket = socketConnections[key];
|
||||
if (!webSocket) {
|
||||
webSocket = [[RCTReconnectingWebSocket alloc] initWithURL:url];
|
||||
[webSocket start];
|
||||
socketConnections[key] = webSocket;
|
||||
}
|
||||
|
||||
[packagerClient addHandler:[[RCTReloadPackagerMethod alloc] initWithBridge:_bridge]
|
||||
forMethod:@"reload"];
|
||||
[packagerClient addHandler:[[RCTSamplingProfilerPackagerMethod alloc] initWithBridge:_bridge]
|
||||
forMethod:@"pokeSamplingProfiler"];
|
||||
[packagerClient start];
|
||||
webSocket.delegate = self;
|
||||
}
|
||||
|
||||
- (NSURL *)packagerURL
|
||||
@@ -82,6 +93,59 @@
|
||||
return [NSURL URLWithString:[NSString stringWithFormat:@"%@://%@:%@/message?role=ios-rn-rctdevmenu", scheme, host, port]];
|
||||
}
|
||||
|
||||
|
||||
- (void)addHandler:(id<RCTPackagerClientMethod>)handler forMethod:(NSString *)name
|
||||
{
|
||||
_handlers[name] = handler;
|
||||
}
|
||||
|
||||
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
|
||||
{
|
||||
if (!_handlers) {
|
||||
return;
|
||||
}
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
id<RCTPackagerClientMethod> methodHandler = _handlers[msg[@"method"]];
|
||||
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
|
||||
}
|
||||
|
||||
if (msg[@"id"]) {
|
||||
[methodHandler handleRequest:msg[@"params"]
|
||||
withResponder:[[RCTPackagerClientResponder alloc] initWithId:msg[@"id"]
|
||||
socket:webSocket]];
|
||||
} else {
|
||||
[methodHandler handleNotification:msg[@"params"]];
|
||||
}
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
#endif
|
||||
|
||||
@@ -9,10 +9,10 @@
|
||||
|
||||
#import <React/RCTPackagerClient.h>
|
||||
|
||||
#if RCT_DEV // Only supported in dev mode
|
||||
|
||||
@class RCTBridge;
|
||||
|
||||
#if RCT_DEV // Only supported in dev mode
|
||||
|
||||
@interface RCTReloadPackagerMethod : NSObject <RCTPackagerClientMethod>
|
||||
|
||||
- (instancetype)initWithBridge:(RCTBridge *)bridge;
|
||||
|
||||
Reference in New Issue
Block a user