/* * Copyright (c) Facebook, Inc. and its affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. */ #import #if RCT_DEV #import #import #import #import static NSString *const kDebuggerMsgDisable = @"{ \"id\":1,\"method\":\"Debugger.disable\" }"; static NSString *getServerHost(NSURL *bundleURL, NSNumber *port) { NSString *host = [bundleURL host]; if (!host) { host = @"localhost"; } // this is consistent with the Android implementation, where http:// is the // hardcoded implicit scheme for the debug server. Note, packagerURL // technically looks like it could handle schemes/protocols other than HTTP, // so rather than force HTTP, leave it be for now, in case someone is relying // on that ability when developing against iOS. return [NSString stringWithFormat:@"%@:%@", host, port]; } static NSURL *getInspectorDeviceUrl(NSURL *bundleURL) { NSNumber *inspectorProxyPort = @8081; NSString *inspectorProxyPortStr = [[[NSProcessInfo processInfo] environment] objectForKey:@"RCT_METRO_PORT"]; if (inspectorProxyPortStr && [inspectorProxyPortStr length] > 0) { inspectorProxyPort = [NSNumber numberWithInt:[inspectorProxyPortStr intValue]]; } NSString *escapedDeviceName = [[[UIDevice currentDevice] name] stringByAddingPercentEncodingWithAllowedCharacters:NSCharacterSet.URLQueryAllowedCharacterSet]; NSString *escapedAppName = [[[NSBundle mainBundle] bundleIdentifier] stringByAddingPercentEncodingWithAllowedCharacters:NSCharacterSet.URLQueryAllowedCharacterSet]; return [NSURL URLWithString:[NSString stringWithFormat:@"http://%@/inspector/device?name=%@&app=%@", getServerHost(bundleURL, inspectorProxyPort), escapedDeviceName, escapedAppName]]; } static NSURL *getAttachDeviceUrl(NSURL *bundleURL, NSString *title) { NSNumber *metroBundlerPort = @8081; NSString *metroBundlerPortStr = [[[NSProcessInfo processInfo] environment] objectForKey:@"RCT_METRO_PORT"]; if (metroBundlerPortStr && [metroBundlerPortStr length] > 0) { metroBundlerPort = [NSNumber numberWithInt:[metroBundlerPortStr intValue]]; } NSString *escapedDeviceName = [[[UIDevice currentDevice] name] stringByAddingPercentEncodingWithAllowedCharacters:NSCharacterSet.URLHostAllowedCharacterSet]; NSString *escapedAppName = [[[NSBundle mainBundle] bundleIdentifier] stringByAddingPercentEncodingWithAllowedCharacters:NSCharacterSet.URLHostAllowedCharacterSet]; return [NSURL URLWithString:[NSString stringWithFormat:@"http://%@/attach-debugger-nuclide?title=%@&device=%@&app=%@", getServerHost(bundleURL, metroBundlerPort), title, escapedDeviceName, escapedAppName]]; } @implementation RCTInspectorDevServerHelper RCT_NOT_IMPLEMENTED(-(instancetype)init) static NSMutableDictionary *socketConnections = nil; static void sendEventToAllConnections(NSString *event) { for (NSString *socketId in socketConnections) { [socketConnections[socketId] sendEventToAllConnections:event]; } } static void displayErrorAlert(UIViewController *view, NSString *message) { UIAlertController *alert = [UIAlertController alertControllerWithTitle:nil message:message preferredStyle:UIAlertControllerStyleAlert]; [view presentViewController:alert animated:YES completion:nil]; dispatch_after(dispatch_time(DISPATCH_TIME_NOW, NSEC_PER_SEC * 2.5), dispatch_get_main_queue(), ^{ [alert dismissViewControllerAnimated:YES completion:nil]; }); } + (void)attachDebugger:(NSString *)owner withBundleURL:(NSURL *)bundleURL withView:(UIViewController *)view { NSURL *url = getAttachDeviceUrl(bundleURL, owner); NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url]; [request setHTTPMethod:@"GET"]; __weak UIViewController *viewCapture = view; [[[NSURLSession sharedSession] dataTaskWithRequest:request completionHandler:^( __unused NSData *_Nullable data, __unused NSURLResponse *_Nullable response, NSError *_Nullable error) { UIViewController *viewCaptureStrong = viewCapture; if (error != nullptr && viewCaptureStrong != nullptr) { displayErrorAlert(viewCaptureStrong, @"The request to attach Nuclide couldn't reach Metro!"); } }] resume]; } + (void)disableDebugger { sendEventToAllConnections(kDebuggerMsgDisable); } + (RCTInspectorPackagerConnection *)connectWithBundleURL:(NSURL *)bundleURL { NSURL *inspectorURL = getInspectorDeviceUrl(bundleURL); // Note, using a static dictionary isn't really the greatest design, but // the packager connection does the same thing, so it's at least consistent. // This is a static map that holds different inspector clients per the inspectorURL if (socketConnections == nil) { socketConnections = [NSMutableDictionary new]; } NSString *key = [inspectorURL absoluteString]; RCTInspectorPackagerConnection *connection = socketConnections[key]; if (!connection || !connection.isConnected) { connection = [[RCTInspectorPackagerConnection alloc] initWithURL:inspectorURL]; socketConnections[key] = connection; [connection connect]; } return connection; } @end #endif