mirror of
https://github.com/status-im/react-native.git
synced 2025-01-10 09:35:48 +00:00
8d397b4cbc
Summary: The JavaScript ecosystem doesn't have the notion of a built-in native module loader. Even Node is decoupled from its module loader. The module loader system is just JS that runs on top of the global `process` object which has all the built-in goodies. Additionally there is no such thing as a global require. That is something unique to our providesModule system. In other module systems such as node, every require is contextual. Even registered npm names are localized by version. The only global namespace that is accessible to the host environment is the global object. Normally module systems attaches itself onto the hooks provided by the host environment on the global object. Currently, we have two forms of dispatch that reaches directly into the module system. executeJSCall which reaches directly into require. Everything now calls through the BatchedBridge module (except one RCTLog edge case that I will fix). I propose that the executors calls directly onto `BatchedBridge` through an instance on the global so that everything is guaranteed to go through it. It becomes the main communication hub. I also propose that we drop the dynamic requires inside of MessageQueue/BatchBridge and instead have the modules register themselves with the bridge. executeJSCall was originally modeled after the XHP equivalent. The XHP equivalent was designed that way because the act of doing the call was the thing that defined a dependency on the module from the page. However, that is not how React Native works. The JS side is driving the dependencies by virtue of requiring new modules and frameworks and the existence of dependencies is driven by the JS side, so this design doesn't make as much sense. The main driver for this is to be able to introduce a new module system like Prepack's module system. However, it also unlocks the possibility to do dead module elimination even in our current module system. It is currently not possible because we don't know which module might be called from native. Since the module system now becomes decoupled we could publish all our providesModule modules as npm/CommonJS modules using a rewrite script. That's what React Core does. That way people could use any CommonJS bundler such as Webpack, Closure Compiler, Rollup or some new innovation to create a JS bundle. This diff expands the executeJSCalls to the BatchedBridge's three individual pieces to make them first class instead of being dynamic. This removes one layer of abstraction. Hopefully we can also remove more of the things that register themselves with the BatchedBridge (various EventEmitters) and instead have everything go through the public protocol. ReactMethod/RCT_EXPORT_METHOD. public Reviewed By: vjeux Differential Revision: D2717535 fb-gh-sync-id: 70114f05483124f5ac5c4570422bb91a60a727f6
213 lines
6.6 KiB
Objective-C
213 lines
6.6 KiB
Objective-C
/**
|
|
* The examples provided by Facebook are for non-commercial testing and
|
|
* evaluation purposes only.
|
|
*
|
|
* Facebook reserves all rights not expressly granted.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NON INFRINGEMENT. IN NO EVENT SHALL
|
|
* FACEBOOK BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
|
|
* AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
|
|
#import <mach/mach_time.h>
|
|
|
|
#import <XCTest/XCTest.h>
|
|
|
|
#import "RCTContextExecutor.h"
|
|
#import "RCTUtils.h"
|
|
|
|
#define RUN_PERF_TESTS 0
|
|
|
|
@interface RCTContextExecutorTests : XCTestCase
|
|
|
|
@end
|
|
|
|
@implementation RCTContextExecutorTests
|
|
{
|
|
RCTContextExecutor *_executor;
|
|
}
|
|
|
|
- (void)setUp
|
|
{
|
|
[super setUp];
|
|
_executor = [RCTContextExecutor new];
|
|
[_executor setUp];
|
|
}
|
|
|
|
- (void)testNativeLoggingHookExceptionBehavior
|
|
{
|
|
dispatch_semaphore_t doneSem = dispatch_semaphore_create(0);
|
|
[_executor executeApplicationScript:[@"var x = {toString: function() { throw 1; }}; nativeLoggingHook(x);" dataUsingEncoding:NSUTF8StringEncoding]
|
|
sourceURL:[NSURL URLWithString:@"file://"]
|
|
onComplete:^(__unused id error){
|
|
dispatch_semaphore_signal(doneSem);
|
|
}];
|
|
dispatch_semaphore_wait(doneSem, DISPATCH_TIME_FOREVER);
|
|
[_executor invalidate];
|
|
}
|
|
|
|
#if RUN_PERF_TESTS
|
|
|
|
static uint64_t _get_time_nanoseconds(void)
|
|
{
|
|
static struct mach_timebase_info tb_info = {0, 0};
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^{
|
|
int ret = mach_timebase_info(&tb_info);
|
|
assert(0 == ret);
|
|
});
|
|
|
|
return (mach_absolute_time() * tb_info.numer) / tb_info.denom;
|
|
}
|
|
|
|
- (void)testDeserializationPerf
|
|
{
|
|
// This test case checks the assumption that deserializing objects from JavaScript
|
|
// values one-by-one via ObjC JSC API is slower than using JSON string
|
|
// You might want to switch your tests schema to "Release" build configuration
|
|
|
|
JSContextGroupRef group = JSContextGroupCreate();
|
|
JSGlobalContextRef context = JSGlobalContextCreateInGroup(group, NULL);
|
|
id message = @[@[@1, @2, @3, @4], @[@{@"a": @1}, @{@"b": @2}], (id)kCFNull];
|
|
NSString *code = RCTJSONStringify(message, NULL);
|
|
JSStringRef script = JSStringCreateWithCFString((__bridge CFStringRef)code);
|
|
JSValueRef error = NULL;
|
|
JSValueRef value = JSEvaluateScript(context, script, NULL, NULL, 0, &error);
|
|
XCTAssertTrue(error == NULL);
|
|
|
|
id obj;
|
|
uint64_t start = _get_time_nanoseconds();
|
|
for (int i = 0; i < 10000; i++) {
|
|
JSStringRef jsonJSString = JSValueCreateJSONString(context, value, 0, nil);
|
|
NSString *jsonString = (__bridge NSString *)JSStringCopyCFString(kCFAllocatorDefault, jsonJSString);
|
|
JSStringRelease(jsonJSString);
|
|
|
|
obj = RCTJSONParse(jsonString, NULL);
|
|
}
|
|
NSLog(@"JSON Parse time: %.2fms", (_get_time_nanoseconds() - start) / 1000000.0);
|
|
|
|
JSStringRelease(script);
|
|
JSGlobalContextRelease(context);
|
|
JSContextGroupRelease(group);
|
|
}
|
|
|
|
- (void)testJavaScriptCallSpeed
|
|
{
|
|
/**
|
|
* Since we almost don't change the RCTContextExecutor logic, and this test is
|
|
* very likely to become flaky (specially accross different devices) leave it
|
|
* to be run manually
|
|
*
|
|
* Previous Values: If you change the executor code, you should update this values
|
|
*/
|
|
|
|
int const runs = 4e5;
|
|
int const frequency = 10;
|
|
double const threshold = 0.1;
|
|
static double const expectedTimes[] = {
|
|
0x1.6199943826cf1p+13,
|
|
0x1.a3bc0a81551c3p+13,
|
|
0x1.d49fbb8602fe3p+13,
|
|
0x1.d1f64085ecb7bp+13,
|
|
};
|
|
|
|
dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
|
|
NSString *script = @" \
|
|
var modules = { \
|
|
module: { \
|
|
method: function () { \
|
|
return true; \
|
|
} \
|
|
} \
|
|
}; \
|
|
var Bridge = { \
|
|
callFunctionReturnFlushedQueue: function(module, method, args) { \
|
|
modules[module].apply(modules[module], args); \
|
|
} \
|
|
}; \
|
|
function require(module) { \
|
|
return Bridge; \
|
|
} \
|
|
";
|
|
|
|
[_executor executeApplicationScript:[script dataUsingEncoding:NSUTF8StringEncoding] sourceURL:[NSURL URLWithString:@"http://localhost:8081/"] onComplete:^(__unused NSError *error) {
|
|
NSMutableArray *params = [NSMutableArray new];
|
|
id data = @1;
|
|
for (int i = 0; i < 4; i++) {
|
|
double samples[runs / frequency];
|
|
int size = runs / frequency;
|
|
double total = 0;
|
|
for (int j = 0; j < runs; j++) {
|
|
@autoreleasepool {
|
|
double start = _get_time_nanoseconds();
|
|
[_executor callFunctionOnModule:@"module"
|
|
method:@"method"
|
|
arguments:params
|
|
callback:^(__unused id json, __unused NSError *unused) {
|
|
}];
|
|
double run = _get_time_nanoseconds() - start;
|
|
if ((j % frequency) == frequency - 1) { // Warmup
|
|
total += run;
|
|
samples[j/frequency] = run;
|
|
}
|
|
}
|
|
}
|
|
|
|
double mean = total / size;
|
|
double variance = 0;
|
|
|
|
for (int j = 0; j < size; j++) {
|
|
variance += pow(samples[j] - mean, 2);
|
|
}
|
|
variance /= size;
|
|
|
|
double standardDeviation = sqrt(variance);
|
|
double marginOfError = standardDeviation * 1.645;
|
|
|
|
double lower = mean - marginOfError;
|
|
double upper = mean + marginOfError;
|
|
|
|
int s = 0;
|
|
total = 0;
|
|
for (int j = 0; j < size; j++) {
|
|
double v = samples[j];
|
|
if (v >= lower && v <= upper) {
|
|
samples[s++] = v;
|
|
total += v;
|
|
}
|
|
}
|
|
mean = total / s;
|
|
|
|
lower = mean * (1.0 - threshold);
|
|
upper = mean * (1.0 + threshold);
|
|
|
|
double expected = expectedTimes[i];
|
|
|
|
NSLog(@"Previous: %lf, New: %f -> %a", expected, mean, mean);
|
|
if (upper < expected) {
|
|
NSLog(@"You made JS calls with %d argument(s) %.2lf%% faster :) - Remember to update the tests with the new value: %a",
|
|
i, (1 - (double)mean / expected) * 100, mean);
|
|
}
|
|
|
|
|
|
XCTAssertTrue(lower < expected, @"You made JS calls with %d argument(s) %.2lf%% slower :( - If that's *really* necessary, update the tests with the new value: %a",
|
|
i, ((double)mean / expected - 1) * 100, mean);
|
|
|
|
[params addObject:data];
|
|
}
|
|
dispatch_semaphore_signal(semaphore);
|
|
}];
|
|
|
|
while (dispatch_semaphore_wait(semaphore, DISPATCH_TIME_NOW)) {
|
|
[[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode
|
|
beforeDate:[NSDate dateWithTimeIntervalSinceNow:10]];
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
@end
|