2015-07-13 08:42:32 -07:00
/ * *
* 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 .
* /
2015-06-06 13:37:36 -07:00
# import < mach / mach_time . h >
# import < XCTest / XCTest . h >
2015-12-16 02:49:27 -08:00
# import "RCTJSCExecutor.h"
2015-06-06 13:37:36 -07:00
# import "RCTUtils.h"
2015-07-16 09:23:23 -07:00
# define RUN_PERF _TESTS 0
2015-06-06 13:37:36 -07:00
2015-12-16 02:49:27 -08:00
@ interface RCTJSCExecutorTests : XCTestCase
2015-06-06 13:37:36 -07:00
@ end
2015-12-16 02:49:27 -08:00
@ implementation RCTJSCExecutorTests
2015-06-09 15:42:10 -07:00
{
2015-12-16 02:49:27 -08:00
RCTJSCExecutor * _executor ;
2015-06-09 15:42:10 -07:00
}
- ( void ) setUp
{
[ super setUp ] ;
2015-12-16 02:49:27 -08:00
_executor = [ RCTJSCExecutor new ] ;
2015-06-09 15:42:10 -07:00
[ _executor setUp ] ;
}
2015-06-06 13:37:36 -07:00
- ( void ) testNativeLoggingHookExceptionBehavior
{
dispatch_semaphore _t doneSem = dispatch_semaphore _create ( 0 ) ;
2015-10-16 08:10:25 -07:00
[ _executor executeApplicationScript : [ @ "var x = {toString: function() { throw 1; }}; nativeLoggingHook(x);" dataUsingEncoding : NSUTF8StringEncoding ]
2015-06-06 13:37:36 -07:00
sourceURL : [ NSURL URLWithString : @ "file://" ]
2015-06-15 07:53:45 -07:00
onComplete : ^ ( __unused id error ) {
2015-06-06 13:37:36 -07:00
dispatch_semaphore _signal ( doneSem ) ;
} ] ;
dispatch_semaphore _wait ( doneSem , DISPATCH_TIME _FOREVER ) ;
2015-06-09 15:42:10 -07:00
[ _executor invalidate ] ;
2015-06-06 13:37:36 -07:00
}
2015-07-16 09:23:23 -07:00
# if RUN_PERF _TESTS
2015-06-06 13:37:36 -07:00
static uint64_t _get _time _nanoseconds ( void )
{
2015-06-15 07:53:45 -07:00
static struct mach_timebase _info tb_info = { 0 , 0 } ;
2015-06-06 13:37:36 -07:00
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 ) ;
2015-06-12 11:05:01 -07:00
id message = @ [ @ [ @ 1 , @ 2 , @ 3 , @ 4 ] , @ [ @ { @ "a" : @ 1 } , @ { @ "b" : @ 2 } ] , ( id ) kCFNull ] ;
2015-06-06 13:37:36 -07:00
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 ) ;
}
2015-07-16 09:23:23 -07:00
- ( void ) testJavaScriptCallSpeed
2015-06-06 13:37:36 -07:00
{
/ * *
2015-12-16 02:49:27 -08:00
* Since we almost don ' t change the RCTJSCExecutor logic , and this test is
2015-12-15 09:08:39 -08:00
* very likely to become flaky ( specially across different devices ) leave it
2015-06-06 13:37:36 -07:00
* to be run manually
*
* Previous Values : If you change the executor code , you should update this values
* /
int const runs = 4 e5 ;
int const frequency = 10 ;
double const threshold = 0.1 ;
static double const expectedTimes [ ] = {
0 x1 .6199943826 cf1p + 13 ,
0 x1 . a3bc0a81551c3p + 13 ,
0 x1 . d49fbb8602fe3p + 13 ,
0 x1 . d1f64085ecb7bp + 13 ,
} ;
dispatch_semaphore _t semaphore = dispatch_semaphore _create ( 0 ) ;
NSString * script = @ " \
var modules = { \
module : { \
method : function ( ) { \
return true ; \
} \
} \
} ; \
2015-12-08 15:57:34 -08:00
var Bridge = { \
callFunctionReturnFlushedQueue : function ( module , method , args ) { \
modules [ module ] . apply ( modules [ module ] , args ) ; \
} \
} ; \
2015-06-06 13:37:36 -07:00
function require ( module ) { \
2015-12-08 15:57:34 -08:00
return Bridge ; \
2015-06-06 13:37:36 -07:00
} \
" ;
2015-10-16 08:10:25 -07:00
[ _executor executeApplicationScript : [ script dataUsingEncoding : NSUTF8StringEncoding ] sourceURL : [ NSURL URLWithString : @ "http://localhost:8081/" ] onComplete : ^ ( __unused NSError * error ) {
2015-08-17 07:35:34 -07:00
NSMutableArray * params = [ NSMutableArray new ] ;
2015-06-06 13:37:36 -07:00
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 ( ) ;
2015-12-08 15:57:34 -08:00
[ _executor callFunctionOnModule : @ "module"
method : @ "method"
arguments : params
callback : ^ ( __unused id json , __unused NSError * unused ) {
} ] ;
2015-06-06 13:37:36 -07:00
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 ] ] ;
}
}
2015-07-16 09:23:23 -07:00
# endif
2015-06-06 13:37:36 -07:00
@ end