crashpad/test/gtest_runner_ios.mm
Rohit Rao 54bbd7d0d5 Adds support for running GTests on iOS.
iOS needs to run tests from within the context of a UIApplication, and
it needs to periodically spin the runloop to ensure that the watchdog
does not kill the app for being unresponsive.

BUG=crashpad:31

Change-Id: Ia1d881e478d4f83c236b475a21529760c06100c7
Reviewed-on: https://chromium-review.googlesource.com/c/crashpad/crashpad/+/1904226
Commit-Queue: Rohit Rao <rohitrao@chromium.org>
Reviewed-by: Mark Mentovai <mark@chromium.org>
2020-01-07 17:08:53 +00:00

115 lines
3.6 KiB
Plaintext

// Copyright 2019 The Crashpad Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#include "test/gtest_runner_ios.h"
#import <UIKit/UIKit.h>
#include "gtest/gtest.h"
@interface UIApplication (Testing)
- (void)_terminateWithStatus:(int)status;
@end
namespace {
// The iOS watchdog timer will kill an app that doesn't spin the main event
// loop often enough. This uses a Gtest TestEventListener to spin the current
// loop after each test finishes. However, if any individual test takes too
// long, it is still possible that the app will get killed.
class IOSRunLoopListener : public testing::EmptyTestEventListener {
public:
virtual void OnTestEnd(const testing::TestInfo& test_info) {
@autoreleasepool {
// At the end of the test, spin the default loop for a moment.
NSDate* stop_date = [NSDate dateWithTimeIntervalSinceNow:0.001];
[[NSRunLoop currentRunLoop] runUntilDate:stop_date];
}
}
};
void RegisterTestEndListener() {
testing::TestEventListeners& listeners =
testing::UnitTest::GetInstance()->listeners();
listeners.Append(new IOSRunLoopListener);
}
} // namespace
@interface CrashpadUnitTestDelegate : NSObject
@property(nonatomic, readwrite, strong) UIWindow* window;
- (void)runTests;
@end
@implementation CrashpadUnitTestDelegate
- (BOOL)application:(UIApplication*)application
didFinishLaunchingWithOptions:(NSDictionary*)launchOptions {
self.window = [[UIWindow alloc] initWithFrame:UIScreen.mainScreen.bounds];
self.window.backgroundColor = UIColor.whiteColor;
[self.window makeKeyAndVisible];
UIViewController* controller = [[UIViewController alloc] init];
[self.window setRootViewController:controller];
// Add a label with the app name.
UILabel* label = [[UILabel alloc] initWithFrame:controller.view.bounds];
label.text = [[NSProcessInfo processInfo] processName];
label.textAlignment = NSTextAlignmentCenter;
label.textColor = UIColor.blackColor;
[controller.view addSubview:label];
// Queue up the test run.
[self performSelector:@selector(runTests) withObject:nil afterDelay:0.1];
return YES;
}
- (void)runTests {
RegisterTestEndListener();
int exitStatus = RUN_ALL_TESTS();
// If a test app is too fast, it will exit before Instruments has has a
// a chance to initialize and no test results will be seen.
// TODO(crbug.com/137010): Figure out how much time is actually needed, and
// sleep only to make sure that much time has elapsed since launch.
[NSThread sleepUntilDate:[NSDate dateWithTimeIntervalSinceNow:2.0]];
self.window = nil;
// Use the hidden selector to try and cleanly take down the app (otherwise
// things can think the app crashed even on a zero exit status).
UIApplication* application = [UIApplication sharedApplication];
[application _terminateWithStatus:exitStatus];
exit(exitStatus);
}
@end
namespace crashpad {
namespace test {
void IOSLaunchApplicationAndRunTests(int argc, char* argv[]) {
@autoreleasepool {
int exit_status =
UIApplicationMain(argc, argv, nil, @"CrashpadUnitTestDelegate");
exit(exit_status);
}
}
} // namespace crashpad
} // namespace test