Issues Adopting Swift Testing
I’m in the process of migrating from XCTest to Swift Testing. The basic stuff is pretty easy to translate, and in many cases I don’t even need to change the code inside my test methods. I’ve long been writing what Swift Testing calls expectations in terms of my own helper functions such as eq()
, unwrap()
, and fail()
instead of XCTAssertEqualObjects()
and friends, XCTUnwrap()
, and XCTFail
(). This is more concise, and I have many overloads for eq()
that provide more precise reporting. I can simply reimplement these helpers atop #expect()
, #require()
, and Issue.record()
.
Although Swift Testing supports structs, you can also use classes as suites of tests. So I can still use a common base class that provides standard functionality for all my tests. Each project has a subclass with its own test data and project-specific functionality, and sometimes I have more layers for groups of tests that have lots of common setup. It seems like this architecture will continue to work, and it’s much nicer with Swift Testing because I can do the setup in init()
instead of having to override func setUpWithError()
. With XCTest, I also always had to override func tearDownWithError()
to clear out property values because otherwise they would still exist while the next test ran. Swift Testing, as far as I know, does not have this problem, so I can rely on deinit
to clear out the properties automatically and only override it when I need to do more than that.
All this said, I have run into some issues where the migration is not as straightforward, and I discuss some of these below. I’m new at this, and of course I’m hoping that some of this stuff I’m just wrong about and that there’s actually a better way.
Floating Point Equality
Swift Testing doesn’t seem to have the equivalent of XCTAssertEqual(_:_:accuracy:)
. This is easy enough to work around but a curious omission. Maybe it’s waiting on language support for approximate equality.
Implicitly Unwrapped Optionals
The vision document criticizes XCTest’s “frequent need for implicitly-unwrapped optional (IUO) properties in test subclasses.” Swift Testing makes this a bit better in that you are initializing your properties in init()
instead of in setUpWithError()
, so you don’t have to worry about them all having values before your code is run. But this doesn’t really fix the problem when you are using a hierarchy of test classes to set up common state. I generally call super.setUpWithError()
in the first line because setup of the subclass will depend on properties that were set up by the base class. But this is backwards from Swift’s initializer rules, where the subclass needs to initialize its properties before calling super.init()
—and without using any helper member functions. So it seems like I still need IUOs or perhaps lazy properties, though those don’t work well with error handling.
Confirmations Don’t Work Like Expectations
With XCTest, I could create multiple expectations in the same test, tell it to wait for all of them, and it would run the run loop and make sure they were all fulfilled, even though the order might be undefined. For example, I have tests to prevent memory leaks where I add a bunch of objects that I expect to be deallocated by the end of the test. Each gets a separate expectation, and when it goes away it calls the corresponding fulfill()
method. I also have tests that make sure a particular block of code results in a notification being posted. Swift Testing’s confirmations seem designed to handle the latter type of test but not the former. As far as I can tell, you can only await
one confirmation at a time, so you have to knew the order in which they will be confirmed. It also doesn’t run the run loop while waiting. I can probably rewrite these tests without using confirmations, just spinning the run loop until all the objects are deallocated.
Parallel Tests in the Same Process
With XCTest, parallel tests run in separate processes, which means that each test has its own copy of global state and doesn’t really need to know that it’s running in parallel, unless you’re writing to UserDefaults
or something. With Swift Testing, everything runs in one process, and you can opt out certain tests or suites using the .serialized
trait. However, if you’re doing this because of global state you would need to apply it everywhere, which would be easy to forget (in that case, I might as well use --no-parallel
) and would make things much slower. I don’t want that.
My tests use several different kinds of global state. Some classes have flags to make them behave differently for testing purposes, e.g. to refresh right away instead of asynchronously. Some classes have failure handlers that will log errors in production but should report them as test failures (using the appropriate framework) when testing—and sometimes they get overridden in testing when we are actually testing the failure reporting. And I have some functionality like getting the current Date
that I swap out so that I can feed the tests certain known values.
With XCTest, these could all be just regular static properties. It’s not worth using dependency injection and passing stuff down many levels when these things only need to change during testing. With Swift Testing, the answer seems to be to use task-local properties so that each test gets its own independent copy. But this is easier said than done.
The first issue is that I’m deploying my apps on versions of macOS that don’t support Swift Concurrency. It seems like I should be able to do something like:
#if TEST @TaskLocal @available(macOS 10.15, *) #endif static var flag = true
to use Swift Concurrency only during testing, but I still ran into some compiler errors. Xcode seems to not figure out that the test target has a higher deployment target and that when compiling for testing it’s set to use the Test scheme, which sets TEST
. I’ve long had problems like this with Swift and conditional compilation. Maybe there’s a way to work around it, but I haven’t looked into it much yet because there’s a larger issue.
I don’t know how to concisely set the task-local values when setting up my tests. With XCTest, I would just assign to the properties in setUpWithError()
. This is not possible with Swift Testing because @TaskLocal
doesn’t work with assignment statements. It’s structured, so you are supposed to give it a closure:
Class.$flag.withValue(false) { // Do test stuff }
I want to do this once per test class, usually just inheriting the value from the base class. I don’t want to have to do it in every test method. With XCTest, there is a hook so I could do something like:
override func invokeTest() { Class.$flag.withValue(false) { super.invokeTest() } }
but as far as I know there is no way to do this with Swift Testing. Maybe it will eventually be possible through custom execution traits, though that seems a bit awkward for this purpose.
Previously:
- Swift Testing in Xcode 16
- How to Control the World
- How to Automate Memory Leak Detection With XCTest
- @autoreleasepool Uses in 2019 Swift
- Using Lazy Variables to Work Around Swift Initialization Rules
- Let Your Swift XCTest Methods Throw
- How to Do XCTestCase tearDown Wrong (and Right)
- Swift init()
Update (2025-01-08): Swift Testing does not yet have the equivalent to XCTest’s class-level +setUp
and +tearDown
APIs.
10 Comments RSS · Twitter · Mastodon
Confirmations also didn't work how I hoped/expected. I found https://github.com/dfed/swift-testing-expectation to behave more like how I'd hoped they would.
Luckily, all our Dependencies were in swift-dependencies already (which uses TaskLocals under the hood), so all good there.
@brzz Thanks for the link. From what I can see, that expectation package also only lets you wait for one expectation at a time.
Dependencies uses task-locals, but as far as I’m aware it doesn’t solve the core issue of needing to put each individual test’s code into a closure in order to access the variables. You can create the objects within a single Dependencies closure in init()
, but then the values go out of scope and revert to the defaults within the test method unless you use a closure there, too. Right?
@Michael Tsai I think you could use multiple expectations and then at the end of the test async let
or a TaskGroup
.
swift-dependences has support for TestTraits
through their DependenciesTestSupport
target. https://github.com/pointfreeco/swift-dependencies/blob/main/Sources/DependenciesTestSupport/TestTrait.swift
@brzz TaskGroup
looks promising, thanks.
Thanks for the pointer to the _DependenciesTrait
stuff! That looks like it will do the job by allowing the setup to be done outside of init()
. It’s a bit more awkward than the Test Scoping Traits proposal that I mentioned but works with the current Swift Testing. I wasn’t aware of prepare(for test: Test)
, which can essentially do what invokeTest()
did. I can probably just make my own trait and do what I need to do directly, without needing Dependencies (though that looks good for those who need more).
I was able to figure out the conditional compilation issue. It was a bug in my code where I was missing an availability check in a dependent target. Somehow this made Xcode freak out and report errors about implicitly importing Core Audio and SwiftUI (neither of which I’m using) in relation to a totally different target.
Looking at this some more, prepare(for test: Test)
is harder to use than I thought because, unlike invokeTest()
, it doesn’t actually run the test. So I can’t easily use withValue()
. It looks like Dependencies works around that by populating a global lookup table so that later it can find the proper task-local value based on the current test’s ID. This is rather involved and seems not worth doing myself without using the Dependencies framework itself, which I don’t want to do at this time and can’t easily do without upping my deployment target to include Swift Concurrency. Also, reading more about TaskLocal
, I’m not sure it will even work for me because the values wouldn’t be propagated to GCD threads. So this whole approach seems like a non-starter. I’m going to try to restructure my code to eliminate places where I don’t really need to override values and to do old-fashioned injection where necessary.
Great point that the swift-testing-expectation
library didn't have an easy way to handle multiple expectations. I wrote up a little helper here: https://github.com/dfed/swift-testing-expectation/pull/8
Would love to know if that solves your use case!
@dfed Thanks. It certainly looks like the kind of API I would want. I will have to see after I port my tests whether it works or whether some of them need to have the run loop spin while waiting. The other thing that jumped out at me is that it looks like it will always wait for the duration
even if the expectations are fulfilled earlier? That seems bad, though maybe it isn’t that big of a deal since the tests are running in parallel.
> It certainly looks like the kind of API I would want. I will have to see after I port my tests whether it works or whether some of them need to have the run loop spin while waiting.
Great! I expect it'll work just fine, since the `await` suspends and allows the runloop to continue while we wait.
> The other thing that jumped out at me is that it looks like it will always wait for the `duration` even if the expectations are fulfilled earlier?
It looks that way because I got a little tricky with the `try await Task.sleep(…)`. When an expectation is fulfilled I `cancel()` the task, which causes the `Task.sleep(…)` call to `throw CancellationError()`, which immediately exits the `wait` task. We're very much aligned that neither of us want to leave tasks around consuming resources after they are no longer necessary.