2015-03-23 20:28:42 +00:00
|
|
|
/**
|
|
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This source code is licensed under the BSD-style license found in the
|
|
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
|
|
*/
|
2015-03-14 01:32:38 +00:00
|
|
|
|
|
|
|
#import "RCTTestRunner.h"
|
|
|
|
|
2015-03-24 17:20:13 +00:00
|
|
|
#import "FBSnapshotTestController.h"
|
2015-06-15 14:53:45 +00:00
|
|
|
#import "RCTAssert.h"
|
2015-08-19 10:18:04 +00:00
|
|
|
#import "RCTLog.h"
|
2015-03-14 01:32:38 +00:00
|
|
|
#import "RCTRootView.h"
|
|
|
|
#import "RCTTestModule.h"
|
|
|
|
#import "RCTUtils.h"
|
2015-10-01 11:19:46 +00:00
|
|
|
#import "RCTContextExecutor.h"
|
2015-03-14 01:32:38 +00:00
|
|
|
|
2015-09-04 10:21:57 +00:00
|
|
|
static const NSTimeInterval kTestTimeoutSeconds = 60;
|
|
|
|
static const NSTimeInterval kTestTeardownTimeoutSeconds = 30;
|
2015-05-04 17:35:49 +00:00
|
|
|
|
2015-03-14 01:32:38 +00:00
|
|
|
@implementation RCTTestRunner
|
2015-03-24 17:20:13 +00:00
|
|
|
{
|
2015-04-18 17:43:20 +00:00
|
|
|
FBSnapshotTestController *_testController;
|
2015-07-07 23:19:40 +00:00
|
|
|
RCTBridgeModuleProviderBlock _moduleProvider;
|
2015-03-24 17:20:13 +00:00
|
|
|
}
|
2015-03-14 01:32:38 +00:00
|
|
|
|
2015-07-07 23:19:40 +00:00
|
|
|
- (instancetype)initWithApp:(NSString *)app
|
|
|
|
referenceDirectory:(NSString *)referenceDirectory
|
|
|
|
moduleProvider:(RCTBridgeModuleProviderBlock)block
|
2015-03-14 01:32:38 +00:00
|
|
|
{
|
2015-06-15 14:53:45 +00:00
|
|
|
RCTAssertParam(app);
|
2015-07-07 23:19:40 +00:00
|
|
|
RCTAssertParam(referenceDirectory);
|
2015-06-15 14:53:45 +00:00
|
|
|
|
2015-03-24 17:20:13 +00:00
|
|
|
if ((self = [super init])) {
|
2015-11-02 18:58:55 +00:00
|
|
|
if (!referenceDirectory.length) {
|
|
|
|
referenceDirectory = [[NSBundle bundleForClass:self.class].resourcePath stringByAppendingPathComponent:@"ReferenceImages"];
|
|
|
|
}
|
2015-07-07 23:19:40 +00:00
|
|
|
|
2015-03-24 17:20:13 +00:00
|
|
|
NSString *sanitizedAppName = [app stringByReplacingOccurrencesOfString:@"/" withString:@"-"];
|
|
|
|
sanitizedAppName = [sanitizedAppName stringByReplacingOccurrencesOfString:@"\\" withString:@"-"];
|
2015-04-18 17:43:20 +00:00
|
|
|
_testController = [[FBSnapshotTestController alloc] initWithTestName:sanitizedAppName];
|
2015-07-07 23:19:40 +00:00
|
|
|
_testController.referenceImagesDirectory = referenceDirectory;
|
|
|
|
_moduleProvider = [block copy];
|
|
|
|
|
2015-06-04 10:47:19 +00:00
|
|
|
#if RUNNING_ON_CI
|
|
|
|
_scriptURL = [[NSBundle bundleForClass:[RCTBridge class]] URLForResource:@"main" withExtension:@"jsbundle"];
|
2015-07-07 23:19:40 +00:00
|
|
|
RCTAssert(_scriptURL != nil, @"Could not locate main.jsBundle");
|
2015-06-04 10:47:19 +00:00
|
|
|
#else
|
2015-09-04 10:21:57 +00:00
|
|
|
_scriptURL = [NSURL URLWithString:[NSString stringWithFormat:@"http://localhost:8081/%@.bundle?platform=ios&dev=true", app]];
|
2015-06-04 10:47:19 +00:00
|
|
|
#endif
|
2015-03-14 01:32:38 +00:00
|
|
|
}
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
2015-08-24 10:14:33 +00:00
|
|
|
RCT_NOT_IMPLEMENTED(- (instancetype)init)
|
2015-06-15 14:53:45 +00:00
|
|
|
|
2015-03-24 17:20:13 +00:00
|
|
|
- (void)setRecordMode:(BOOL)recordMode
|
|
|
|
{
|
2015-04-18 17:43:20 +00:00
|
|
|
_testController.recordMode = recordMode;
|
2015-03-24 17:20:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)recordMode
|
2015-03-14 01:32:38 +00:00
|
|
|
{
|
2015-04-18 17:43:20 +00:00
|
|
|
return _testController.recordMode;
|
2015-03-14 01:32:38 +00:00
|
|
|
}
|
|
|
|
|
2015-03-24 17:20:13 +00:00
|
|
|
- (void)runTest:(SEL)test module:(NSString *)moduleName
|
2015-03-14 01:32:38 +00:00
|
|
|
{
|
2015-03-24 17:20:13 +00:00
|
|
|
[self runTest:test module:moduleName initialProps:nil expectErrorBlock:nil];
|
|
|
|
}
|
|
|
|
|
2015-04-15 00:51:28 +00:00
|
|
|
- (void)runTest:(SEL)test module:(NSString *)moduleName
|
|
|
|
initialProps:(NSDictionary *)initialProps expectErrorRegex:(NSString *)errorRegex
|
2015-03-24 17:20:13 +00:00
|
|
|
{
|
|
|
|
[self runTest:test module:moduleName initialProps:initialProps expectErrorBlock:^BOOL(NSString *error){
|
2015-04-11 22:08:00 +00:00
|
|
|
return [error rangeOfString:errorRegex options:NSRegularExpressionSearch].location != NSNotFound;
|
2015-03-14 01:32:38 +00:00
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2015-07-07 23:19:40 +00:00
|
|
|
- (void)runTest:(SEL)test module:(NSString *)moduleName
|
|
|
|
initialProps:(NSDictionary *)initialProps expectErrorBlock:(BOOL(^)(NSString *error))expectErrorBlock
|
2015-03-14 01:32:38 +00:00
|
|
|
{
|
2015-09-04 10:21:57 +00:00
|
|
|
__weak id weakJSContext;
|
|
|
|
|
|
|
|
@autoreleasepool {
|
|
|
|
__block NSString *error = nil;
|
|
|
|
RCTSetLogFunction(^(RCTLogLevel level, NSString *fileName, NSNumber *lineNumber, NSString *message) {
|
|
|
|
if (level >= RCTLogLevelError) {
|
|
|
|
error = message;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
RCTBridge *bridge = [[RCTBridge alloc] initWithBundleURL:_scriptURL
|
|
|
|
moduleProvider:_moduleProvider
|
|
|
|
launchOptions:nil];
|
|
|
|
|
|
|
|
RCTRootView *rootView = [[RCTRootView alloc] initWithBridge:bridge moduleName:moduleName initialProperties:initialProps];
|
|
|
|
rootView.frame = CGRectMake(0, 0, 320, 2000); // Constant size for testing on multiple devices
|
|
|
|
|
|
|
|
NSString *testModuleName = RCTBridgeModuleNameForClass([RCTTestModule class]);
|
|
|
|
RCTTestModule *testModule = rootView.bridge.modules[testModuleName];
|
|
|
|
RCTAssert(_testController != nil, @"_testController should not be nil");
|
|
|
|
testModule.controller = _testController;
|
|
|
|
testModule.testSelector = test;
|
|
|
|
testModule.view = rootView;
|
|
|
|
|
|
|
|
UIViewController *vc = [UIApplication sharedApplication].delegate.window.rootViewController;
|
|
|
|
vc.view = [UIView new];
|
|
|
|
[vc.view addSubview:rootView]; // Add as subview so it doesn't get resized
|
|
|
|
|
|
|
|
NSDate *date = [NSDate dateWithTimeIntervalSinceNow:kTestTimeoutSeconds];
|
|
|
|
while (date.timeIntervalSinceNow > 0 && testModule.status == RCTTestStatusPending && error == nil) {
|
|
|
|
[[NSRunLoop mainRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate dateWithTimeIntervalSinceNow:0.1]];
|
|
|
|
[[NSRunLoop mainRunLoop] runMode:NSRunLoopCommonModes beforeDate:[NSDate dateWithTimeIntervalSinceNow:0.1]];
|
2015-08-19 10:18:04 +00:00
|
|
|
}
|
|
|
|
|
2015-09-04 10:21:57 +00:00
|
|
|
// Take a weak reference to the JS context, so we track its deallocation later
|
|
|
|
// (we can only do this now, since it's been lazily initialized)
|
2015-10-01 11:19:46 +00:00
|
|
|
id jsExecutor = [bridge valueForKeyPath:@"batchedBridge.javaScriptExecutor"];
|
|
|
|
if ([jsExecutor isKindOfClass:[RCTContextExecutor class]]) {
|
|
|
|
weakJSContext = [jsExecutor valueForKey:@"context"];
|
|
|
|
}
|
2015-09-04 10:21:57 +00:00
|
|
|
[rootView removeFromSuperview];
|
2015-07-07 23:19:40 +00:00
|
|
|
|
2015-09-04 10:21:57 +00:00
|
|
|
RCTSetLogFunction(RCTDefaultLogFunction);
|
2015-03-26 01:59:42 +00:00
|
|
|
|
2015-09-04 10:21:57 +00:00
|
|
|
NSArray *nonLayoutSubviews = [vc.view.subviews filteredArrayUsingPredicate:[NSPredicate predicateWithBlock:^BOOL(id subview, NSDictionary *bindings) {
|
|
|
|
return ![NSStringFromClass([subview class]) isEqualToString:@"_UILayoutGuide"];
|
|
|
|
}]];
|
|
|
|
RCTAssert(nonLayoutSubviews.count == 0, @"There shouldn't be any other views: %@", nonLayoutSubviews);
|
2015-04-18 17:43:20 +00:00
|
|
|
|
2015-09-04 10:21:57 +00:00
|
|
|
if (expectErrorBlock) {
|
|
|
|
RCTAssert(expectErrorBlock(error), @"Expected an error but nothing matched.");
|
|
|
|
} else {
|
|
|
|
RCTAssert(error == nil, @"RedBox error: %@", error);
|
|
|
|
RCTAssert(testModule.status != RCTTestStatusPending, @"Test didn't finish within %0.f seconds", kTestTimeoutSeconds);
|
|
|
|
RCTAssert(testModule.status == RCTTestStatusPassed, @"Test failed");
|
|
|
|
}
|
|
|
|
[bridge invalidate];
|
|
|
|
}
|
2015-03-14 01:32:38 +00:00
|
|
|
|
2015-09-04 10:21:57 +00:00
|
|
|
// Wait for the executor to have shut down completely before returning
|
|
|
|
NSDate *teardownTimeout = [NSDate dateWithTimeIntervalSinceNow:kTestTeardownTimeoutSeconds];
|
|
|
|
while (teardownTimeout.timeIntervalSinceNow > 0 && weakJSContext) {
|
2015-05-06 16:33:20 +00:00
|
|
|
[[NSRunLoop mainRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate dateWithTimeIntervalSinceNow:0.1]];
|
|
|
|
[[NSRunLoop mainRunLoop] runMode:NSRunLoopCommonModes beforeDate:[NSDate dateWithTimeIntervalSinceNow:0.1]];
|
2015-03-14 01:32:38 +00:00
|
|
|
}
|
2015-09-04 10:21:57 +00:00
|
|
|
RCTAssert(!weakJSContext, @"JS context was not deallocated after being invalidated");
|
2015-03-14 01:32:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@end
|