192 lines
5.7 KiB
Objective-C
192 lines
5.7 KiB
Objective-C
/* Copyright 2015 Realm Inc - All Rights Reserved
|
|
* Proprietary and Confidential
|
|
*/
|
|
|
|
#import "RealmJSTests.h"
|
|
#import "RCTJavaScriptExecutor.h"
|
|
#import "RCTBridge.h"
|
|
#import "RCTDevMenu.h"
|
|
|
|
@import RealmReact;
|
|
|
|
extern void JSGlobalContextSetIncludesNativeCallStackWhenReportingExceptions(JSGlobalContextRef ctx, bool includesNativeCallStack);
|
|
extern NSMutableArray *RCTGetModuleClasses(void);
|
|
|
|
@interface RealmReactTests : RealmJSTests
|
|
@end
|
|
|
|
@interface RealmReactChromeTests : RealmReactTests
|
|
@end
|
|
|
|
@implementation RealmReactTests
|
|
|
|
+ (void)load {
|
|
NSMutableArray *moduleClasses = RCTGetModuleClasses();
|
|
[moduleClasses removeObject:[RCTDevMenu class]];
|
|
}
|
|
|
|
+ (Class)executorClass {
|
|
return NSClassFromString(@"RCTContextExecutor");
|
|
}
|
|
|
|
+ (NSString *)classNameSuffix {
|
|
return @"";
|
|
}
|
|
|
|
+ (id<RCTJavaScriptExecutor>)currentExecutor {
|
|
Class executorClass = [self executorClass];
|
|
if (!executorClass) {
|
|
return nil;
|
|
}
|
|
|
|
static RCTBridge *s_bridge;
|
|
if (!s_bridge) {
|
|
[[NSNotificationCenter defaultCenter] postNotificationName:RCTReloadNotification object:nil];
|
|
}
|
|
|
|
if (!s_bridge.valid) {
|
|
NSNotification *notification = [self waitForNotification:RCTJavaScriptDidLoadNotification];
|
|
s_bridge = notification.userInfo[@"bridge"];
|
|
assert(s_bridge);
|
|
}
|
|
|
|
if (s_bridge.executorClass != executorClass) {
|
|
s_bridge.executorClass = executorClass;
|
|
[s_bridge reload];
|
|
|
|
// The [RCTBridge reload] method does a dispatch_async that we must run before trying again.
|
|
[[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate dateWithTimeIntervalSinceNow:0.1]];
|
|
|
|
return [self currentExecutor];
|
|
}
|
|
|
|
return [s_bridge valueForKey:@"javaScriptExecutor"];
|
|
}
|
|
|
|
+ (XCTestSuite *)defaultTestSuite {
|
|
XCTestSuite *suite = [super defaultTestSuite];
|
|
id<RCTJavaScriptExecutor> executor = [self currentExecutor];
|
|
|
|
// The executor may be nil if the executorClass was not found (i.e. release build).
|
|
if (!executor) {
|
|
return suite;
|
|
}
|
|
|
|
// FIXME: Remove this nonsense once the crashes go away when a test fails!
|
|
JSGlobalContextRef ctx = RealmReactGetJSGlobalContextForExecutor(executor, false);
|
|
if (ctx) {
|
|
JSGlobalContextSetIncludesNativeCallStackWhenReportingExceptions(ctx, false);
|
|
}
|
|
|
|
NSError *error;
|
|
NSDictionary *testCaseNames = [self invokeMethod:@"getTestNames" arguments:nil error:&error];
|
|
|
|
if (error || !testCaseNames.count) {
|
|
NSLog(@"Error from calling getTestNames() - %@", error ?: @"None returned");
|
|
exit(1);
|
|
}
|
|
|
|
NSString *nameSuffix = [self classNameSuffix];
|
|
if (nameSuffix.length) {
|
|
NSMutableDictionary *renamedTestCaseNames = [[NSMutableDictionary alloc] init];
|
|
for (NSString *name in testCaseNames) {
|
|
renamedTestCaseNames[[name stringByAppendingString:nameSuffix]] = testCaseNames[name];
|
|
}
|
|
testCaseNames = renamedTestCaseNames;
|
|
}
|
|
|
|
for (XCTestSuite *testSuite in [self testSuitesFromDictionary:testCaseNames]) {
|
|
[suite addTest:testSuite];
|
|
}
|
|
|
|
return suite;
|
|
}
|
|
|
|
+ (NSNotification *)waitForNotification:(NSString *)notificationName {
|
|
NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
|
|
__block BOOL condition = NO;
|
|
__block NSNotification *notification;
|
|
|
|
id token = [nc addObserverForName:notificationName object:nil queue:[NSOperationQueue mainQueue] usingBlock:^(NSNotification *note) {
|
|
condition = YES;
|
|
notification = note;
|
|
}];
|
|
|
|
[self waitForCondition:&condition];
|
|
[nc removeObserver:token];
|
|
|
|
return notification;
|
|
}
|
|
|
|
+ (void)waitForCondition:(BOOL *)condition {
|
|
NSRunLoop *runLoop = [NSRunLoop currentRunLoop];
|
|
|
|
while (!*condition) {
|
|
@autoreleasepool {
|
|
[runLoop runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
|
|
}
|
|
}
|
|
}
|
|
|
|
+ (id)invokeMethod:(NSString *)method arguments:(NSArray *)arguments error:(NSError * __strong *)outError {
|
|
id<RCTJavaScriptExecutor> executor = [self currentExecutor];
|
|
|
|
__block BOOL condition = NO;
|
|
__block id result;
|
|
|
|
[executor executeJSCall:@"realm-tests/index.js" method:method arguments:(arguments ?: @[]) callback:^(id json, NSError *error) {
|
|
// The React Native debuggerWorker.js very bizarrely returns an array five empty arrays to signify an error.
|
|
if ([json isKindOfClass:[NSArray class]] && [json isEqualToArray:@[@[], @[], @[], @[], @[]]]) {
|
|
json = nil;
|
|
|
|
if (!error) {
|
|
error = [NSError errorWithDomain:@"JS" code:1 userInfo:@{NSLocalizedDescriptionKey: @"unknown JS error"}];
|
|
}
|
|
}
|
|
|
|
dispatch_async(dispatch_get_main_queue(), ^{
|
|
condition = YES;
|
|
result = json;
|
|
|
|
if (error && outError) {
|
|
*outError = error;
|
|
}
|
|
});
|
|
}];
|
|
|
|
[self waitForCondition:&condition];
|
|
|
|
return result;
|
|
}
|
|
|
|
- (void)invokeMethod:(NSString *)method {
|
|
NSString *module = NSStringFromClass(self.class);
|
|
NSString *suffix = [self.class classNameSuffix];
|
|
|
|
if (suffix.length && [module hasSuffix:suffix]) {
|
|
module = [module substringToIndex:(module.length - suffix.length)];
|
|
}
|
|
|
|
NSError *error;
|
|
[self.class invokeMethod:@"runTest" arguments:@[module, method] error:&error];
|
|
|
|
if (error) {
|
|
// TODO: Parse and use localizedFailureReason info once we can source map the failure location in JS.
|
|
[self recordFailureWithDescription:error.localizedDescription inFile:@(__FILE__) atLine:__LINE__ expected:YES];
|
|
}
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation RealmReactChromeTests
|
|
|
|
+ (Class)executorClass {
|
|
return NSClassFromString(@"RCTWebSocketExecutor");
|
|
}
|
|
|
|
+ (NSString *)classNameSuffix {
|
|
return @"_Chrome";
|
|
}
|
|
|
|
@end
|