- Describe:
- the challenges and benefits of unit testing Asynchronous Operations
- Implement basic examples of:
- unit tests of an Asynchronous Operation (a background dataTask) using the
XCTestExpectation
class,
- Let's play a Quizlet game.... Link to study flash cards
Instruments is a powerful performance analysis and testing tool. It comes with Xcode's toolset.
You can use it during:
- development
- testing
- debugging
It provides tools called Instruments that are used to profile errors in you app.
An Instrument collects data over a period of time.
Uses data provided by the operating system to collect call stacks of relevant threads at a fixed time interval.
Why?
- We can analyze an app's responsiveness and performance.
A track shows time series trace data corresponding to one source of events (such as a process, thread or CPU core)
Explore trace data for a selected track. See the functions called on each thread during the trace.
Provides richer info from the Instruments in use. (For example, for Time Profiler, we see the heaviest call stack)
A selector placed at a specific time in the trace.Selects all traced events or intervals occurring at that time.
- Time Profiler shows how your app is spending time
- Check main thread when responsiveness issues occur
- Profile with difficult workloads or older devices
- Instruments supports all platforms and the simulator. But watch out when using the simulator. 👀
- Gives you trends, are things getting faster or slower? Observes all the threads in your app at a fixed interval of time and correlates the information.
10 min - Take out an Xcode project (maybe your final project 😀) and run the Time Profiler. See if everything looks good so far.
Unit testing is most commonly applied to synchronous operations because their outputs can be observed and validated immediately after invoking the function under test.
Whether the output is a function return value, a state change, or the result of methods invoked on a dependency, all of these results occur right away and in the same thread.
And, with synchronous operations, when you write the assertions in the Then
phase of your unit test, you are reasonably guaranteed that the outputs have already been set, so you can safely compare actual results against your expected ones.
Simple. Only one thread. And no worries that your outputs might not be set prior to test completion.
However, modern iOS development requires a great deal of asynchronous operations in which results might not come immediately after a function is invoked.
Asynchronous operations are operations that do not execute directly within the current flow of code. This might be because they run:
- on a different thread
- in a delegate method
- in a callback
Key examples:
- Networking (most common)
- Core Data
- Resource-expensive screen drawing code or events
Asynchronous operations are typically developed in one of two ways:
- Completion handler block
- Delegate method
Networking is the most common asynchronous operation. Fetching data from remote web services has latency — it takes time to for signals to travel across the globe — and has many variables that can result in errors or delays.
Testing asynchronous code comes with the benefit of uncovering poor design decisions and facilitating clean implementations.
The core challenge is that, in typical unit tests, a test is considered over as soon as its function returns.
With asynchronous operations:
- Functions do not return their result to the caller immediately but deliver it later via callback functions, blocks, notifications, or similar mechanisms, which makes testing more difficult.
- When the function under test returns, any asynchronous code will be ignored because it will run after the test has already finished.
This makes unit testing difficult because results can be unpredictable: In the Then
phase of your unit test, the results may or may not have been set to the outputs for you to observe and verify. When you write your assertions, the test may pass this time (if the outputs have been set), but fail at another time (if the outputs haven’t been set).
It can also result in false positives.
As a result, asynchronous testing requires special handling.
Fortunately, Xcode has built-in support to help with unit testing of asynchronous operations.
To test that asynchronous operations behave as expected, you create one or more Expectations within your test, and then fulfill those expectations when the asynchronous operation completes successfully.
Apple describes the XCTestExpectation
class as "An expected outcome in an asynchronous test."
class XCTestExpectation : NSObject
...and the XCTestExpectation
class's fulfill()
function is simply described as "Marks the expectation as having been met."
func fulfill()
Note that it is an error to call the
fulfill()
method on an expectation that has already been fulfilled, or when the test case that vended the expectation has already completed.
Steps required to unit test asynchronous operations will vary depending on the function(s) under test.
In general, when using the XCTestExpectation
class, your test method waits until all expectations are fulfilled or a specified timeout expires.
To illustrate a common scenario, here are the general steps you would execute to create a unit test for a background download task using an instance of the the XCTestExpectation
class: 1
-
Create a new instance of
XCTestExpectation
. -
Use URLSession's
dataTask(with:)
method to create a background data task that executes your download work on a background thread. -
After starting the data task, use the
wait(for expectations:_)
function of theXCtest
class — with the timeout parameter that you specify — to set how long the main thread will wait for the expectation to be fulfilled.
/*!
* @method -waitForExpectations:timeout:
* Wait on a group of expectations for up to the specified timeout. May return early based on fulfillment
* of the waited on expectations.
*/
open func wait(for expectations: [XCTestExpectation], timeout seconds: TimeInterval)
- When the data task completes, its
completion handler
verifies that the downloaded data is non-nil, and fulfills the expectation by calling itsfulfill()
method to indicate that the background task completed successfully.
The fulfillment of the expectation on the background thread provides a point of synchronization to indicate that the background task is complete. As long as the background task fulfills the expectation within the duration specified in the timeout parameter, this test method will pass.
There are two ways for the test to fail:
- The data returned to the completion handler is
nil
, causingXCTAssertNotNil(_:_:file:line:)
to trigger a test failure. - The data task does not call its completion handler before the timeout expires, perhaps because of a slow network connection or other data retrieval problem. As a result, the expectation is not fulfilled before the wait timeout expires, triggering a test failure.
In the starter app, a previous developer created a testDownloadWebData()
function to unit test a background data task. However, that function is missing critical functionality.
Your task is to complete the testDownloadWebData()
function so that it successfully validates its targeted background task.
- Complete the Code
- find the
//TODO:
s left in the function and add or change whatever is needed (see Example Scenario above for clues)
- Run the code and examine results
- What happened? Why?
- Apply any additional required fixes
- Run it again...
- Make it fail
Review the Successful and Unsuccessful Completion notes in the Example Scenario above.
- reduce timeout to 0.1