Due to a couple issues, I want the XCTest target in a project to run a separate app delegate. Using ObjC, this was a relatively straightforward process: manipulate main.m
(see: https://stackoverflow.com/a/15725328/1299041).
Since it seems that a Swift application is initialized with @UIApplicationMain
in the AppDelegate, is it possible to initialize with a separate AppDelegate for the test target?
It's strongly unrecommended to add conditions to normal code checking if its being tested. Instead you should mock your AppDelegate
in tests to do whatever you want.
Then you could replace delegate of UIApplication is setUp
in super class of your each XCTestCase
'es.
class MockAppDelegate:NSObject, UIApplicationDelegate {
}
class BaseTest: XCTestCase {
override func setUp() {
super.setUp()
UIApplication.shared.delegate = MockAppDelegate()
}
}
class Test1: BaseTest {
override func setUp() {
super.setUp()
// normal testing
}
}
If you still want to stop code execution for tests this is my method that works well:
You can add startup parameter to app which indicates that this is test run
These parameters are accessible from NSUserDefaults
#define IS_TESTS [[NSUserDefaults standardUserDefaults] boolForKey:@"TESTING"]
To achieve this is Swift you need to take a couple of steps:
If you are using Storyboards, create your view stack programmatically on your AppDelegate.
Remove Main.storyboard from your project configuration
Delete @UIApplicationMain from the beginning of your AppDelegate and add this code.
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
let storyboard = UIStoryboard(name: "Main", bundle: nil)
let vc = storyboard.instantiateInitialViewController()
let window = UIWindow(frame: UIScreen.main.bounds)
window.rootViewController = vc
window.makeKeyAndVisible()
self.window = window
return true
}
Create a new file at the root your target and call it main.swift.
Add this code if you don't need to do any setup for your tests
import UIKit
let kIsRunningTests = NSClassFromString("XCTestCase") != nil
let kAppDelegateClass = kIsRunningTests ? nil : NSStringFromClass(AppDelegate.self)
UIApplicationMain(CommandLine.argc, CommandLine.unsafeArgv, nil, kAppDelegateClass)
If you need to make some configuration before you run the tests, create a new class FakeAppDelegate as a subclass from NSObject and add your setup code there.
Put this code in main.swift
import UIKit
let kIsRunningTests = NSClassFromString("XCTestCase") != nil
let kAppDelegateClass = kIsRunningTests ? NSStringFromClass(FakeAppDelegate.self) : NSStringFromClass(AppDelegate.self)
UIApplicationMain(CommandLine.argc, CommandLine.unsafeArgv, nil, kAppDelegateClass)
This solution to this is as follows:
Duplicate your existing application's Target and rename it to something appropriate. In your case maybe 'TestingHarness' or some such. Note that you'll also want to change the bundle identifier and rename the corresponding Info.plist file. Renaming the Info.plist file means you'll need to change the Info.plist filename setting in your new target's Build Settings tab to match the new name.
Create another *AppDelegate.swift
file. In your case I'd call it TestAppDelegate.swift
.
Copy over your existing AppDelegate.swift
file's contents into TestAppDelegate.swift
and edit as desired. Make sure to leave the @UIApplicationMain
annotation and implement the needed UIApplicationDelegate
callbacks.
Change the target membership of each of your *AppDelegate.swift
files so that AppDelegate.swift
is not included in your new 'TestHarness' target and TestAppDelegate.swift
is not included in your main app's target. (You edit a file's Target Membership by selecting it in the File Browser and opening the File Inspector which you can access in the right-sidebar by default, or by choosing it in the menu under View -> Utilities.)
Now you have two separate targets with separate App Delegates that you can build and run independently. The final step is to select your new 'TestHarness' target as the Host Application for your test targets. (Click the top-level project entry in the File Browser, then click your desired test Target in the sub-listing. On the General tab you'll see Host Application as the only available drop down.)
Note: these instructions are for Xcode 7.2.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With