Make your testers' lives easy and handle errors
Let me know if this sounds familiar: your QA engineer just installed a fresh version of your application. 2 minutes in, you already got your first issue reported: they’re stuck because nothing happens when they press the login button. They’re literally stuck at the entrance of your app and they can’t do their work until you send them another build fixing this issue. That also means stopping whatever you were doing.
One of the main advantages of continuous integration and automated deployment is the quick feedback you get from your testers, your boss, your QA engineer… Although, even with the best continuous integration suite, one simple thing to always think about is error handling.
Error handling is not optional
This may seem like something obvious to you, but a quick search on Github will show you that’s it not the case for everyone.
It’s quite easy to be focused on the result when you’re working on an application. You got that list of tweets showing up in your app, you can move on to the next feature. You’ll keep a day or two in your sprint to tackle all this boring error handling. When Swift 2.0 introduced guard
and started advocating for the “exit early” approach, we’ve probably all written the following code.
func userDidTapOnReload(sender: UIButton) {
guard case let .authenticated(token) = currentState else {
// TODO: Present error because the user shouldn't be able to
// see this screen if they're not authenticated
return
}
delegate?.userDidReload()
}
Problem is, we often forget that even if something is not supposed to happen, it probably will. QA engineers usually have this effect on apps.
It’s ok to crash
The problem with a simple return
is that nothing will happen and there is nothing more frustrating than that for the user. As an alternative, why not just crash the app? First, the tester will know that something definitely went wrong and won’t keep pressing that button hoping for something to happen. Even better, when using a crash reporting tool, you’ll know everything about the crash and won’t have to squeeze the details out of your tester.
Most of the Crash reporting tools, such as Buddybuild have a crash
method you can use, but a fatalError
will work just as well.
func userDidTapOnReload(sender: UIButton) {
guard case let .authenticated(token) = currentState else {
BuddyBuildSDK.crash()
return
}
delegate?.userDidReload()
}
Make sure you handle all the cases
There is one more pattern I want to talk about. It was common in Objective-C but even then it wasn’t a great solution.
func fetch(completion: ([Tweet]?, Error?) -> Void) {
// Fetch the tweets and call the completion handler
}
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
fetch { tweets, error in
if let tweets = tweets {
// Display tweets
} else if let error = error {
// Present error
} else {
// This shouldn't happen soooo
// let's back away slowly
}
}
}
The problem with this kind of code is, once again, that a situation that should never happen probably will. When in the hands of the users, nothing will happen when the data and the error are both nil. Rob Nappier gave one of my favorite talks covering this pattern, why you should avoid it and what to use instead (TL;DW: use a Result enum).
Worst case scenario: log the error
If crashing is not a option for you, at least make sure to log has much information as you can. It will help you when you need to investigate. Castro, my favorite podcast application, is a great example. When you want to send them some feedback, logs can be automatically attached to your message. That way, if you come to them with a bug, they should be able to retrace your steps in the apps quite easily and understand what happened.
Help testers help you: make sure they are able to send you valuable feedback.