Fixes current CI failures and allows tree shaking of native dev support code.

Summary:
* The dev support code moved into a `DevSupport` subspec, meaning that only if the subspec is specified in the user’s Podfile will the packager client, dev menu, etc be included. This is mainly done through checks for header availability.

  It also improves the weird situation where you had to specify the `RCTWebSocket` subspec if you wanted to be able to use the packager client during development.

* I removed hardcoding the release version in the podspec on release, because the podspec still relies on `package.json` when evaluating, so there’s no real point in not also getting the version number from there. This should remove any requirement to perform maintenance of the OSS release script regarding the podspec.
Closes https://github.com/facebook/react-native/pull/12602

Differential Revision: D4621021

Pulled By: ericvicenti

fbshipit-source-id: 6c208371fc40ea607809a6ab05dd3714ed9980cf
This commit is contained in:
Eloy Durán 2017-02-27 13:25:47 -08:00 committed by Facebook Github Bot
parent 24a7665df5
commit 19caaba1d5
6 changed files with 61 additions and 25 deletions

View File

@ -1,10 +1,19 @@
require "json" require "json"
package = JSON.parse(File.read(File.join(__dir__, "package.json"))) package = JSON.parse(File.read(File.join(__dir__, "package.json")))
version = package['version']
source = { :git => 'https://github.com/facebook/react-native.git' }
if version == '1000.0.0'
# This is an unpublished version, use the latest commit hash of the react-native repo, which were presumably in.
source[:commit] = `git rev-parse HEAD`.strip
else
source[:tag] = "v#{version}"
end
Pod::Spec.new do |s| Pod::Spec.new do |s|
s.name = "React" s.name = "React"
s.version = package["version"] s.version = version
s.summary = package["description"] s.summary = package["description"]
s.description = <<-DESC s.description = <<-DESC
React Native apps are built using the React JS React Native apps are built using the React JS
@ -22,7 +31,7 @@ Pod::Spec.new do |s|
s.homepage = "http://facebook.github.io/react-native/" s.homepage = "http://facebook.github.io/react-native/"
s.license = package["license"] s.license = package["license"]
s.author = "Facebook" s.author = "Facebook"
s.source = { :git => "https://github.com/facebook/react-native.git", :tag => "v#{s.version}" } s.source = source
s.default_subspec = "Core" s.default_subspec = "Core"
s.requires_arc = true s.requires_arc = true
s.platform = :ios, "8.0" s.platform = :ios, "8.0"
@ -34,11 +43,17 @@ Pod::Spec.new do |s|
ss.dependency "Yoga", "#{package["version"]}.React" ss.dependency "Yoga", "#{package["version"]}.React"
ss.dependency "React/cxxreact" ss.dependency "React/cxxreact"
ss.source_files = "React/**/*.{c,h,m,mm,S}" ss.source_files = "React/**/*.{c,h,m,mm,S}"
ss.exclude_files = "**/__tests__/*", "IntegrationTests/*", "React/**/RCTTVView.*", "ReactCommon/yoga/*", "React/Cxx*/*" ss.exclude_files = "**/__tests__/*", "IntegrationTests/*", "React/DevSupport/*", "React/Modules/RCTDev{LoadingView,Menu}.*", "React/**/RCTTVView.*", "ReactCommon/yoga/*", "React/Cxx*/*"
ss.framework = "JavaScriptCore" ss.framework = "JavaScriptCore"
ss.libraries = "stdc++" ss.libraries = "stdc++"
end end
s.subspec "DevSupport" do |ss|
ss.dependency "React/Core"
ss.dependency "React/RCTWebSocket"
ss.source_files = "React/DevSupport/*", "React/Modules/RCTDev{LoadingView,Menu}.*"
end
s.subspec "tvOS" do |ss| s.subspec "tvOS" do |ss|
ss.dependency "React/Core" ss.dependency "React/Core"
ss.source_files = "React/**/RCTTVView.{h, m}" ss.source_files = "React/**/RCTTVView.{h, m}"

View File

@ -14,7 +14,6 @@
#import "RCTBridge.h" #import "RCTBridge.h"
#import "RCTBridgeMethod.h" #import "RCTBridgeMethod.h"
#import "RCTConvert.h" #import "RCTConvert.h"
#import "RCTDevLoadingView.h"
#import "RCTDisplayLink.h" #import "RCTDisplayLink.h"
#import "RCTJSCExecutor.h" #import "RCTJSCExecutor.h"
#import "RCTJavaScriptLoader.h" #import "RCTJavaScriptLoader.h"
@ -25,6 +24,10 @@
#import "RCTRedBox.h" #import "RCTRedBox.h"
#import "RCTUtils.h" #import "RCTUtils.h"
#if RCT_DEV && __has_include("RCTDevLoadingView.h")
#import "RCTDevLoadingView.h"
#endif
#define RCTAssertJSThread() \ #define RCTAssertJSThread() \
RCTAssert(![NSStringFromClass([self->_javaScriptExecutor class]) isEqualToString:@"RCTJSCExecutor"] || \ RCTAssert(![NSStringFromClass([self->_javaScriptExecutor class]) isEqualToString:@"RCTJSCExecutor"] || \
[[[NSThread currentThread] name] isEqualToString:RCTJSCThreadName], \ [[[NSThread currentThread] name] isEqualToString:RCTJSCThreadName], \
@ -116,7 +119,7 @@ RCT_NOT_IMPLEMENTED(- (instancetype)initWithDelegate:(id<RCTBridgeDelegate>)dele
sourceCode = source; sourceCode = source;
dispatch_group_leave(initModulesAndLoadSource); dispatch_group_leave(initModulesAndLoadSource);
} onProgress:^(RCTLoadingProgress *progressData) { } onProgress:^(RCTLoadingProgress *progressData) {
#ifdef RCT_DEV #if RCT_DEV && __has_include("RCTDevLoadingView.h")
RCTDevLoadingView *loadingView = [weakSelf moduleForClass:[RCTDevLoadingView class]]; RCTDevLoadingView *loadingView = [weakSelf moduleForClass:[RCTDevLoadingView class]];
[loadingView updateProgress:progressData]; [loadingView updateProgress:progressData];
#endif #endif

View File

@ -24,7 +24,6 @@
#import "RCTAssert.h" #import "RCTAssert.h"
#import "RCTBridge+Private.h" #import "RCTBridge+Private.h"
#import "RCTDefines.h" #import "RCTDefines.h"
#import "RCTDevMenu.h"
#import "RCTDevSettings.h" #import "RCTDevSettings.h"
#import "RCTJSCErrorHandling.h" #import "RCTJSCErrorHandling.h"
#import "RCTJSCProfiler.h" #import "RCTJSCProfiler.h"
@ -34,6 +33,10 @@
#import "RCTProfile.h" #import "RCTProfile.h"
#import "RCTUtils.h" #import "RCTUtils.h"
#if (RCT_PROFILE || RCT_DEV) && __has_include("RCTDevMenu.h")
#import "RCTDevMenu.h"
#endif
NSString *const RCTJSCThreadName = @"com.facebook.react.JavaScript"; NSString *const RCTJSCThreadName = @"com.facebook.react.JavaScript";
NSString *const RCTJavaScriptContextCreatedNotification = @"RCTJavaScriptContextCreatedNotification"; NSString *const RCTJavaScriptContextCreatedNotification = @"RCTJavaScriptContextCreatedNotification";
RCT_EXTERN NSString *const RCTFBJSContextClassKey = @"_RCTFBJSContextClassKey"; RCT_EXTERN NSString *const RCTFBJSContextClassKey = @"_RCTFBJSContextClassKey";
@ -167,6 +170,7 @@ RCT_EXPORT_MODULE()
#if RCT_DEV #if RCT_DEV
static void RCTInstallJSCProfiler(RCTBridge *bridge, JSContextRef context) static void RCTInstallJSCProfiler(RCTBridge *bridge, JSContextRef context)
{ {
#if __has_include("RCTDevMenu.h")
__weak RCTBridge *weakBridge = bridge; __weak RCTBridge *weakBridge = bridge;
__weak RCTDevSettings *devSettings = bridge.devSettings; __weak RCTDevSettings *devSettings = bridge.devSettings;
if (RCTJSCProfilerIsSupported()) { if (RCTJSCProfilerIsSupported()) {
@ -186,6 +190,7 @@ static void RCTInstallJSCProfiler(RCTBridge *bridge, JSContextRef context)
} }
}]]; }]];
} }
#endif
} }
#endif #endif
@ -396,11 +401,13 @@ static NSThread *newJavaScriptThread(void)
// Add toggles for JSC's sampling profiler, if the profiler is enabled // Add toggles for JSC's sampling profiler, if the profiler is enabled
if (JSC_JSSamplingProfilerEnabled(context.JSGlobalContextRef)) { if (JSC_JSSamplingProfilerEnabled(context.JSGlobalContextRef)) {
// Mark this thread as the main JS thread before starting profiling. // Mark this thread as the main JS thread before starting profiling.
JSC_JSStartSamplingProfilingOnMainJSCThread(context.JSGlobalContextRef); JSC_JSStartSamplingProfilingOnMainJSCThread(context.JSGlobalContextRef);
// Allow to toggle the sampling profiler through RN's dev menu
__weak JSContext *weakContext = self->_context.context; __weak JSContext *weakContext = self->_context.context;
#if __has_include("RCTDevMenu.h")
// Allow to toggle the sampling profiler through RN's dev menu
[self->_bridge.devMenu addItem:[RCTDevMenuItem buttonItemWithTitle:@"Start / Stop JS Sampling Profiler" handler:^{ [self->_bridge.devMenu addItem:[RCTDevMenuItem buttonItemWithTitle:@"Start / Stop JS Sampling Profiler" handler:^{
RCTJSCExecutor *strongSelf = weakSelf; RCTJSCExecutor *strongSelf = weakSelf;
if (!strongSelf.valid || !weakContext) { if (!strongSelf.valid || !weakContext) {
@ -408,6 +415,7 @@ static NSThread *newJavaScriptThread(void)
} }
[weakSelf.bridge.devSettings toggleJSCSamplingProfiler]; [weakSelf.bridge.devSettings toggleJSCSamplingProfiler];
}]]; }]];
#endif
// Allow for the profiler to be poked from JS code as well // Allow for the profiler to be poked from JS code as well
// (see SamplingProfiler.js for an example of how it could be used with the JSCSamplingProfiler module). // (see SamplingProfiler.js for an example of how it could be used with the JSCSamplingProfiler module).

View File

@ -18,13 +18,9 @@
#import "JSCSamplingProfiler.h" #import "JSCSamplingProfiler.h"
#import "RCTBridge+Private.h" #import "RCTBridge+Private.h"
#import "RCTBridgeModule.h" #import "RCTBridgeModule.h"
#import "RCTDevMenu.h"
#import "RCTEventDispatcher.h" #import "RCTEventDispatcher.h"
#import "RCTLog.h" #import "RCTLog.h"
#import "RCTPackagerClient.h"
#import "RCTProfile.h" #import "RCTProfile.h"
#import "RCTReloadPackagerMethod.h"
#import "RCTSamplingProfilerPackagerMethod.h"
#import "RCTUtils.h" #import "RCTUtils.h"
NSString *const kRCTDevSettingProfilingEnabled = @"profilingEnabled"; NSString *const kRCTDevSettingProfilingEnabled = @"profilingEnabled";
@ -42,6 +38,11 @@ NSString *const kRCTDevSettingStartSamplingProfilerOnLaunch = @"startSamplingPro
NSString *const kRCTDevSettingsUserDefaultsKey = @"RCTDevMenu"; NSString *const kRCTDevSettingsUserDefaultsKey = @"RCTDevMenu";
#if RCT_DEV #if RCT_DEV
#if __has_include("RCTPackagerClient.h")
#import "RCTPackagerClient.h"
#import "RCTReloadPackagerMethod.h"
#import "RCTSamplingProfilerPackagerMethod.h"
#endif
@interface RCTDevSettingsUserDefaultsDataSource : NSObject <RCTDevSettingsDataSource> @interface RCTDevSettingsUserDefaultsDataSource : NSObject <RCTDevSettingsDataSource>
@ -459,6 +460,9 @@ RCT_EXPORT_METHOD(toggleElementInspector)
- (NSURL *)packagerURL - (NSURL *)packagerURL
{ {
#if !__has_include("RCTWebSocketObserver.h")
return nil;
#else
NSString *host = [_bridge.bundleURL host]; NSString *host = [_bridge.bundleURL host];
NSString *scheme = [_bridge.bundleURL scheme]; NSString *scheme = [_bridge.bundleURL scheme];
if (!host) { if (!host) {
@ -471,6 +475,7 @@ RCT_EXPORT_METHOD(toggleElementInspector)
port = @8081; // Packager default port port = @8081; // Packager default port
} }
return [NSURL URLWithString:[NSString stringWithFormat:@"%@://%@:%@/message?role=ios-rn-rctdevmenu", scheme, host, port]]; return [NSURL URLWithString:[NSString stringWithFormat:@"%@://%@:%@/message?role=ios-rn-rctdevmenu", scheme, host, port]];
#endif
} }
// TODO: Move non-UI logic into separate RCTDevSettings module // TODO: Move non-UI logic into separate RCTDevSettings module
@ -483,6 +488,7 @@ RCT_EXPORT_METHOD(toggleElementInspector)
return; return;
} }
#if __has_include("RCTPackagerClient.h")
// The jsPackagerClient is a static map that holds different packager clients per the packagerURL // 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 // 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 // previous instances will override given packager client's method handlers
@ -505,6 +511,7 @@ RCT_EXPORT_METHOD(toggleElementInspector)
[packagerClient addHandler:[[RCTSamplingProfilerPackagerMethod alloc] initWithBridge:_bridge] [packagerClient addHandler:[[RCTSamplingProfilerPackagerMethod alloc] initWithBridge:_bridge]
forMethod:@"pokeSamplingProfiler"]; forMethod:@"pokeSamplingProfiler"];
[packagerClient start]; [packagerClient start];
#endif
} }
@end @end

View File

@ -16,7 +16,6 @@
#import <mach/mach.h> #import <mach/mach.h>
#import "RCTBridge.h" #import "RCTBridge.h"
#import "RCTDevMenu.h"
#import "RCTDevSettings.h" #import "RCTDevSettings.h"
#import "RCTFPSGraph.h" #import "RCTFPSGraph.h"
#import "RCTInvalidating.h" #import "RCTInvalidating.h"
@ -27,6 +26,10 @@
#import "RCTUIManager.h" #import "RCTUIManager.h"
#import "RCTBridge+Private.h" #import "RCTBridge+Private.h"
#if __has_include("RCTDevMenu.h")
#import "RCTDevMenu.h"
#endif
static NSString *const RCTPerfMonitorCellIdentifier = @"RCTPerfMonitorCellIdentifier"; static NSString *const RCTPerfMonitorCellIdentifier = @"RCTPerfMonitorCellIdentifier";
static CGFloat const RCTPerfMonitorBarHeight = 50; static CGFloat const RCTPerfMonitorBarHeight = 50;
@ -74,11 +77,11 @@ static vm_size_t RCTGetResidentMemorySize(void)
return info.resident_size; return info.resident_size;
} }
@class RCTDevMenuItem;
@interface RCTPerfMonitor : NSObject <RCTBridgeModule, RCTInvalidating, UITableViewDataSource, UITableViewDelegate> @interface RCTPerfMonitor : NSObject <RCTBridgeModule, RCTInvalidating, UITableViewDataSource, UITableViewDelegate>
#if __has_include("RCTDevMenu.h")
@property (nonatomic, strong, readonly) RCTDevMenuItem *devMenuItem; @property (nonatomic, strong, readonly) RCTDevMenuItem *devMenuItem;
#endif
@property (nonatomic, strong, readonly) UIPanGestureRecognizer *gestureRecognizer; @property (nonatomic, strong, readonly) UIPanGestureRecognizer *gestureRecognizer;
@property (nonatomic, strong, readonly) UIView *container; @property (nonatomic, strong, readonly) UIView *container;
@property (nonatomic, strong, readonly) UILabel *memory; @property (nonatomic, strong, readonly) UILabel *memory;
@ -93,7 +96,9 @@ static vm_size_t RCTGetResidentMemorySize(void)
@end @end
@implementation RCTPerfMonitor { @implementation RCTPerfMonitor {
#if __has_include("RCTDevMenu.h")
RCTDevMenuItem *_devMenuItem; RCTDevMenuItem *_devMenuItem;
#endif
UIPanGestureRecognizer *_gestureRecognizer; UIPanGestureRecognizer *_gestureRecognizer;
UIView *_container; UIView *_container;
UILabel *_memory; UILabel *_memory;
@ -142,7 +147,9 @@ RCT_EXPORT_MODULE()
{ {
_bridge = bridge; _bridge = bridge;
#if __has_include("RCTDevMenu.h")
[_bridge.devMenu addItem:self.devMenuItem]; [_bridge.devMenu addItem:self.devMenuItem];
#endif
} }
- (void)invalidate - (void)invalidate
@ -150,6 +157,7 @@ RCT_EXPORT_MODULE()
[self hide]; [self hide];
} }
#if __has_include("RCTDevMenu.h")
- (RCTDevMenuItem *)devMenuItem - (RCTDevMenuItem *)devMenuItem
{ {
if (!_devMenuItem) { if (!_devMenuItem) {
@ -173,6 +181,7 @@ RCT_EXPORT_MODULE()
return _devMenuItem; return _devMenuItem;
} }
#endif
- (UIPanGestureRecognizer *)gestureRecognizer - (UIPanGestureRecognizer *)gestureRecognizer
{ {

View File

@ -11,7 +11,7 @@
/** /**
* This script bumps a new version for open source releases. * This script bumps a new version for open source releases.
* It updates the version in podspec/json/gradle files and makes sure they are consistent between each other * It updates the version in json/gradle files and makes sure they are consistent between each other
* After changing the files it makes a commit and tags it. * After changing the files it makes a commit and tags it.
* All you have to do is push changes to remote and CI will make a new build. * All you have to do is push changes to remote and CI will make a new build.
*/ */
@ -54,17 +54,11 @@ if (sed(`-i`, /^VERSION_NAME=.*/, `VERSION_NAME=${version}`, `ReactAndroid/gradl
exit(1); exit(1);
} }
// - change React.podspec
if (sed(`-i`, /s\.version\s*=.*/, `s.version = \"${version}\"`, `React.podspec`).code) {
echo(`Couldn't update version for React.podspec`);
exit(1);
}
// verify that files changed, we just do a git diff and check how many times version is added across files // verify that files changed, we just do a git diff and check how many times version is added across files
let numberOfChangedLinesWithNewVersion = exec(`git diff -U0 | grep '^[+]' | grep -c ${version} `, {silent: true}) let numberOfChangedLinesWithNewVersion = exec(`git diff -U0 | grep '^[+]' | grep -c ${version} `, {silent: true})
.stdout.trim(); .stdout.trim();
if (+numberOfChangedLinesWithNewVersion !== 3) { if (+numberOfChangedLinesWithNewVersion !== 2) {
echo(`Failed to update all the files. React.podspec, package.json and gradle.properties must have versions in them`); echo(`Failed to update all the files. package.json and gradle.properties must have versions in them`);
echo(`Fix the issue, revert and try again`); echo(`Fix the issue, revert and try again`);
exec(`git diff`); exec(`git diff`);
exit(1); exit(1);