#docs #articles #setup #beginner #insights #presets
Setting Up the 'Errors' Preset to Reduce Churn
TelemetryDeck ships with a set of insights that can be useful to learn what kind of issues your users encounter most in your apps. Here's how to set them up.
Why Track Errors?
Success is often defined in numbers such as Monthly Active Users (MAU) or Monthly Recurring Revenue (MRR). Churn is a metric that tracks the percentage of active/paying users who stopped using/paying for your app.
A smooth experience without bugs & issues is one of the key factors contributing to a lower churn rate. If you do a good job in continuuously fixing the most common issues with your product, you’re on the right track for success. That’s why it’s important that you detect any common issues your users get stuck at while using your app.
Sending the Signal
To report unexpected events to TelemetryDeck, send the event name TelemetryDeck.Error.occurred
with the parameter TelemetryDeck.Error.id
set to something that can be used to group errors of the same kind. For example, using the Swift SDK you could simply pass error.localizedDescription
whenever an exception is thrown that you don’t expect to happen:
do {
let object = try JSONDecoder().decode(Object.self, from: data)
} catch {
// your error handling code
TelemetryDeck.signal(
"TelemetryDeck.Error.occurred",
parameters: ["TelemetryDeck.Error.id": error.localizedDescription]
)
}
The Swift SDK ships with a convenience method for so you don’t have to remember the event & parameter keys:
TelemetryDeck.errorOccurred(id: error.localizedDescription)
The errorOccurred
function also accepts the same arguments as the signal
function (namely parameters
, floatValue
, customUserID
) in case you want to provide additional context info.
Separation of ID & Message
While the above code is a good starting point, the localized nature of the localizedDescription
message attached to all thrown exceptions in Swift isn’t optimal. The same issue will be reported with different messages simply because the text will differ based on the users language settings. And you might have even created your own error types that provide dynamic content such as the file path in the error message, which makes things even worse. To see which errors affect most users, it’s best to give the same kind of errror the same ID.
So, whenever possible, it’s recommended that you instead pass a made-up value to the TelemetryDeck.Error.id
parameter that rather represents the context of the error. The full message can be provided with the optional parameter TelemetryDeck.Error.message
like so:
do {
let object = try JSONDecoder().decode(Object.self, from: data)
} catch {
// your error handling code
TelemetryDeck.signal(
"TelemetryDeck.Error.occurred",
parameters: [
"TelemetryDeck.Error.id": "ImportObject.jsonDecode",
"TelemetryDeck.Error.message": error.localizedDescription
]
)
}
Again, the Swift SDK makes this call shorter:
TelemetryDeck.errorOccurred(id: "ImportObject.jsonDecode", message: error.localizedDescription)
For your own Error
types, you could introduce an IdentifiableError
protocol and conform to that to make this process easier (the Swift SDK has this protocol built-in):
protocol IdentifiableError: Error {
var id: String { get }
}
enum MyError: String, IdentifiableError {
case fileMissing
case invalidFormat
var id: String { self.rawValue }
}
Now you can pass error.id
for the TelemetryDeck.Error.id
parameter whenever you encounter an error that can be casted to IdentifiableError
. For system errors, you could fall back to something like String(describing: type(of: error))
. Ths Swift SDK can handle types conforming to IdentifiableError
directly:
do {
let object = try JSONDecoder().decode(Object.self, from: data)
} catch {
// your error handling code
if let myError = error as? MyError {
TelemetryDeck.errorOccurred(identifiableError: myError)
}
}
Note that error.localizedDescription
will be sent as the message
by default, but you can override it.
Built-In Error Categories
Reporting exceptions that you didn’t expect to happen isn’t enough to cover all “unexpected behaviors” that you will encounter in your app. We found that unexpected behavior generally falls into one of the following 3 categories:
- Unexpected Thrown Exceptions (e.g. parsing errors, I/O errors, permission errors)
- Unexpected User Input (e.g. invalid text format, invalid number format, invalid date range)
- Unexpected App State (e.g. inconsistent navigation request, invalid combination of form options)
Each of these has a dedicated chart in the “Errors” tab, you just need to report one of thrown-exception
, user-input
, or app-state
to the parameter TelemetryDeck.Error.category
.
Here’s some guidance on when to use which category in Swift:
- A clear sign to report a
thrown-exception
error is ado-catch
clause or uses oftry?
in Swift where you can send the error signal when it returnsnil
. - Whenever you you make use of the nil-coalescing operator
??
or unwrap an Optional withif-let
orguard-let
, potentially some kind of conversion of user input into another type might happen with a fallback behavior – this is a typicaluser-input
error. - Search for any uses of
assert
orassertionFailure
in your code and additionally report these detected unexpected states of your app during runtime asapp-state
errors. If you weren’t aware, theassert
/assertionFailure
functions are similar tofatalError
/precondition
/preconditionFailure
with the difference that they only stop program execution during DEBUG builds, not in production builds.
A full signal that reports a user-input
category error could end up looking something like this in Swift:
var hourlyRate: Int {
if let hourlyRate = Int(self.textFieldInput) {
return hourlyRate
} else {
TelemetryDeck.signal(
"TelemetryDeck.Error.occurred",
parameters: [
"TelemetryDeck.Error.id": "ProjectForm.hourlyRateConversionFailed",
"TelemetryDeck.Error.category": "user-input",
"TelemetryDeck.Error.message": "Text '\(self.textFieldInput)' could not be converted to type 'Int'."
]
)
return 0 // fallback value
}
}
In the Swift SDK, you can instead call this shorter function with a built-in ErrorCategory
enum:
TelemetryDeck.errorOccurred(
id: "ProjectForm.hourlyRateConversionFailed",
category: .userInput,
message: "Text '\(self.textFieldInput)' could not be converted to type 'Int'."
)
Please note that when calling the TelemetryDeck.errorOccurred(identifiableError:)
function, the category is implicitly set to .thrownException
, but you can override it if needed.
Effect on Privacy & App Tracking Transparency
If you are sending dynamic values such as error.localizedDescription
or if any of the parameter fields contain user-dynamic data such as file paths or input data, some user data might be sent to TelemetryDeck. It really depends on the nature of this data and how you plan to use it that influences what fields in App Tracking Transparency you need to add. You might need to adjust your privacy report accordingly.
But TelemetryDeck does not attempt to link collected data to the users identity, nor do we use data for tracking purposes. To protect your users privacy, we urge you to not send any data that might identify your users.