Error Analysis
Prerequisites
Section titled PrerequisitesContentsquare SDK is included
Section titled Contentsquare SDK is includedYou must include the Contentsquare SDK first using the same integration method and understand how the SDK works.
Update to the latest SDK version
Section titled Update to the latest SDK versionIn order to enable Error Analysis in your app and get the most stable version, it is required to upgrade the SDK to its latest version.
Screen tracking implemented
Section titled Screen tracking implementedTracking will start at the 1st screenview event, it is required to have screen tracking implemented. Make sure to follow the iOS Track screens sections.
Reminder about User consent
Section titled Reminder about User consentIf you are in the process of implementing the SDK for the 1st time (or choose to take this update as an opportunity to review your Privacy related implementation), make sure to follow the iOS Privacy section and use the Opt-in API to get the user consent, otherwise no data will be collected.
Get Started
Section titled Get StartedAdd Contentsquare Error Analysis to your app
Section titled Add Contentsquare Error Analysis to your appHow to include it
Section titled How to include itYou should include the Contentsquare SDK first and use the same integration method.
Using Swift Package Manager
Section titled Using Swift Package Manager-
In Xcode, Same as Contentsquare SDK, Add the following link via
File > Add Packages…
: -
In your targets’ General settings tab, in the Frameworks, Libraries, and Embedded Content section, add
ContentsquareErrorAnalysisModule
.
Using Carthage
Section titled Using CarthageAdd the following line to your Cartfile:
Then run:
Drag and drop ContentsquareModuleErrorAnalysis.xcframework
from the Carthage/Build folder to your targets’ General settings tab, in the Frameworks, Libraries, and Embedded Content section.
Using CocoaPods
Section titled Using CocoaPodsOur SDK can be linked dynamically or statically:
Dynamic linking
Section titled Dynamic linkingDynamic linking is the default behavior with use_frameworks!
,
Add the following line to your Podfile:
Static linking
Section titled Static linkingIf you specify static linking in your Podfile with use_frameworks! :linkage => :static
,
Add the following line to your Podfile:
Manual Integration
Section titled Manual IntegrationOur SDK can be linked dynamically or statically:
Dynamic linking
Section titled Dynamic linkingGet the manual integration framework
Section titled Get the manual integration framework- Go to the iOS SDK GitHub repository ↗.
- Find the newest version available (unless instructed otherwise by your CS contact).
- Under
Assets
you should be able to findContentsquareErrorAnalysisModuleDynamicManually.xcframework.zip
, download the file.
Include the framework
Section titled Include the framework- Unzip
ContentsquareErrorAnalysisModuleDynamicManually.xcframework.zip
and you should seeContentsquareErrorAnalysisModule.xcframework
: - Copy
ContentsquareErrorAnalysisModule.xcframework
to any folder in your project. - In your
target
->General
->Frameworks, Libraries and Embedded Content
, addContentsquareErrorAnalysisModule.xcframework
by clicking ”+” -> “Add Other…” -> “Add Files…”. - Clean build folder and run.
Static linking
Section titled Static linkingGet the manual integration framework
Section titled Get the manual integration framework- Go to the iOS SDK GitHub repository ↗.
- Find the newest version available (unless instructed otherwise by your CS contact).
- Under
Assets
you should be able to findContentsquareErrorAnalysisModuleStaticManually.xcframework.zip
, download the file.
Include the framework
Section titled Include the framework- Unzip
ContentsquareErrorAnalysisModuleStaticManually.xcframework.zip
and you should seeContentsquareErrorAnalysisModule.xcframework
: - Copy
ContentsquareErrorAnalysisModule.xcframework
to any folder in your project. - In your
target
->General
->Frameworks, Libraries and Embedded Content
, addContentsquareErrorAnalysisModule.xcframework
by clicking ”+” -> “Add Other…” -> “Add Files…”. - Clean build folder and run.
Start the SDK
Section titled Start the SDKYou do not need to do anything to start the Error Analysis SDK, it will start itself with Contentsquare SDK.
Validate SDK integration
Section titled Validate SDK integrationWhen the SDK starts, you should see a log like this one:
Sample App
Section titled Sample AppFor best implementation practices of our library, explore the Contentsquare for iOS sample app ↗.
API Errors
Section titled API ErrorsAutomatic network inspection
Section titled Automatic network inspectionAPI Errors automatically collects failed network requests that use URLSession
.
Add custom monitoring for specific network requests
Section titled Add custom monitoring for specific network requestsThe API Errors automatically collects most network requests for your app. However, some requests might not be collected or you might use a different library to make network requests. In these cases, you can use the following API HTTPMetric
to manually collect data.
Removing Personal Data in request URL path
Section titled Removing Personal Data in request URL pathBy default, the API Errors feature collects the URL path of the failed API requests. To prevent the collection of Personal Data in the URL path, you can rewrite the request URL path with the setURLMaskingPatterns
SDK API.
Replace a step of the path - meaning between two slashes (/) - containing Personal Data with a variable:
:user_id
becomes CS_ANONYMIZED_USER_ID:address
becomes CS_ANONYMIZED_ADDRESS
Example
Section titled ExampleURL before masking | URL after masking |
---|---|
https://www.contentsquare.com/users/123/address/castle+black | https://www.contentsquare.com/users/CS_ANONYMIZED_USER_ID/address/CS_ANONYMIZED_ADDRESS |
Debugging and Logging
Section titled Debugging and LoggingIf in-app features are enabled, a log should appear indicating if API Errors is enabled or disabled:
When an API Error event occurs, a log should appear with the details of the event:
See iOS Debugging and Logging section for more details about debugging and logging.
How API Errors works
Section titled How API Errors worksInitialization
Section titled InitializationThe way our SDK works is by auto-starting with the application launch and automatically collects failed network requests that use URLSession
.
Configuration
Section titled ConfigurationOnce started, our SDK fetches its config from our servers. It will start collecting data from network events if the API Errors setting is enabled in the config (this is handled by the Contentsquare team).
Tracking
Section titled TrackingThe SDK monitors only the API Errors with response code above 400, and generates analytics data. These events are then locally stored, and eventually sent to our servers in batches.
Sending data
Section titled Sending dataFor each network error, a new event will be sent in analytics and Session Replay data. Check the following sections to learn more about how data is processed and sent:
API Troubleshooting Details
Section titled API Troubleshooting DetailsAPI Errors troubleshooting details enables you to collect more information about API errors so you can troubleshoot errors faster.
With this feature you will be able to see three types of additional API error details in the Event Stream of Session Replay.
- The HTTP headers of the request and the response.
- The body (the data sent by the request or received in the response).
- The query parameters of the request endpoint (of the URL of the information you request for).
See API Troubleshooting Details ↗ for more details.
Collected data points
Section titled Collected data pointsOnly network calls with error (response code above 400) will be collected. Here is the list of data collected:
- URL (without query strings)
- HTTP method
- Response code
- Timestamp of the request
- Timestamp of the response
- HTTP headers of the request
- HTTP headers of the response
- HTTP body of the request
- HTTP body of the response
- Query parameters of the request endpoint
Known limitations and recommendations
Section titled Known limitations and recommendationsAutomatic collection limitations
Section titled Automatic collection limitationsThe auto collection doesn’t work for the following methods:
Workaround: For the async
methods, use the corresponding Contentsquare methods:
For other methods, use the custom monitoring APIs.
Conflict with Firebase Performance SDK on auto-collection
Section titled Conflict with Firebase Performance SDK on auto-collectionAPI Errors is compatible with Firebase Performance auto-collection, but the HTTP body from the response won’t be collected by the Error Analysis SDK.
Workaround: Remove FirebasePerformance from your target’s dependencies.
It may also not be compatible with other network auto collection tools.
Crash Reporter
Section titled Crash ReporterSetup Firebase Crashlytics compatibility mode
Section titled Setup Firebase Crashlytics compatibility modeBased on our test, the Contentsquare Crash Reporter is compatible with Firebase Crashlytics as long as the Contentsquare Crash Reporter is initialized first, otherwise some crashes can be missing in Crashlytics.
You can use this API to launch Firebase Crashlytics after Contentsquare Crash Reporter
Upload dSYMs
Section titled Upload dSYMs“When Xcode compiles your source code into machine code, it generates a list of symbols in your app — class names, global variables, and method and function names. These symbols correspond to the file and line numbers where they’re defined; this association creates a debug symbol” See Apple’s documentation for more details ↗.
When an app crashes, the operating system collects diagnostic information about what the app was doing at the time of crash. Some of the most important parts of the crash report are presented as hexadecimal addresses. To translate those addresses into readable function names and line numbers from your source code, a process called symbolication is used. See Apple’s documentation for more details ↗.
In order to symbolicate the crashes reported by a specific version of your app and make them readable, you need to provide the symbol files (dSYMs) that were generated when your app was built.
Build your app with dSYMs
Section titled Build your app with dSYMsMake sure Xcode is generating dSYMs for your app:
- Open your project in Xcode.
- Select the project file in the Xcode Navigator.
- Select your main target.
- Select the
Build Settings
tab and clickAll
to show all the build settings. - Filter by
Debug Information Format
. - Set
Debug Information Format
toDWARF with dSYM File
for all the build types you want to generate dSYMs. - Rebuild your main target.
Locate dSYMs
Section titled Locate dSYMsDownload dSYMs from the App Store
Section titled Download dSYMs from the App StoreIf you build your main target using bitcode (Bitcode Enabled
turned on in build settings) and you uploaded your app’s symbols to the App Store, you need to download the dSYMs from the App Store:
- Log in to App Store Connect ↗, then select
My Apps
. - Select your app from the grid.
- Select the build you want to download a dSYM for.
- Click
Build Details
>Download dSYM
.
In case you use bitcode to build your main target but you didn’t upload your dSYMs to the App Store, you need to find your dSYMs on your local machine and restore hidden symbols using the BCSymbolMaps Xcode generates.
Find dSYMs on your local machine
Section titled Find dSYMs on your local machineYou can find the dSYMs in the .xcarchive
directory on disk:
- In Xcode, open the Organizer window (Window -> Organizer).
- Select your app from the list.
- Control-click an archive and select
Show in Finder
. - A Finder window should appear. Control-click the
xcarchive
package and selectShow Package Contents
. - There should be a dSYMs directory that contains the dSYMs generated in Xcode’s archiving process. If you use the
Download Debug Symbols
option from Xcode’s organizer, the recompiled bitcode dSYMs are also downloaded to this directory.
Restore hidden symbols using BCSymbolMaps
Section titled Restore hidden symbols using BCSymbolMapsRestoring hidden symbols using BCSymbolMaps is only required if you built your main target using bitcode (Bitcode Enabled
turned on in build settings). If you used bitcode but downloaded the dSYMs from the App Store, the dSYMs will be recompiled and you can skip this step.
To restore hidden symbols you can run the following command:
Get project ID
Section titled Get project ID- Login to the Contentsquare platform on https://app.contentsquare.com ↗
- Make sure to be on the right project
- The project ID can be found in the URL query parameter
project
:https://app.contentsquare.com/#/{MODULE_NAME}?project={PROJECT_ID}&hash={HASH}]
Get project credentials
Section titled Get project credentialsFollow the dedicated documentation from the Help Center to get the client ID and client Secret: How to create API credentials ↗.
Use script to upload dSYMs
Section titled Use script to upload dSYMsTo upload dSYMs to Error Analysis you need to use the provided upload-symbols
script. You need to provide the path where the dSYM files are located as a positional argument, and the Project ID, Client ID and Client Secret either as keyword arguments or set them as environment variables.
Key | Environment Key |
---|---|
—project-id | ERROR_ANALYSIS_PROJECT_ID |
—client-id | ERROR_ANALYSIS_CLIENT_ID |
—client-secret | ERROR_ANALYSIS_CLIENT_SECRET |
If both are provided, the keyword arguments will prevail over the environment variables.
Using Swift Package Manager
Section titled Using Swift Package ManagerIf you are using SPM to integrate the Error Analysis SDK, the upload-symbols
script will be located in the SPM checkout path for the SDK.
To execute the script using keyword arguments, you can use the following command:
To use environment variables instead, you can use the following commands:
Using other dependency managers
Section titled Using other dependency managersIf you are using Carthage, CocoaPods or you are manually integrating the Error Analysis SDK, you will need to download the upload-symbols
script from the latest SDK release.
- Go to the CS_iOS_SDK releases page ↗.
- Click on
Show all assets
located at the bottom of the Assets list of the latest release. - Click on
upload-symbols-script.zip
. - Unzip the script.
- Move the script to your desired location.
At this point, to execute the script using keyword arguments, you can use the following command:
To use environment variables instead, you can use the following commands:
Integrate with a CI/CD pipeline
Section titled Integrate with a CI/CD pipelineWe strongly recommend to run this as a task in a CI/CD pipeline for the obfuscated builds.
Debugging and Logging
Section titled Debugging and LoggingEnabling crash reporting will conflict with any attached debuggers, so make sure a debugger isn’t attached when you crash the app. In case a debugger is attached to the device, Crash Reporter won’t be initialized and a log will be printed in the console:
To test if Crash Reporter is working properly, you first need to disconnect your test device or simulator from Xcode debugger:
- Tap on the Edit scheme… option of your current scheme
- Disable the Debug executable option in your scheme configuration.
- Build and run your current scheme.
If Crash Reporter is successfully initialized when you launch your application, you should see a log printed in the console:
Alternatively, you can detach your application from Xcode debugger by stopping it after is launched. This way Crash Reporter will be initialized and will start reporting crashes, but you won’t be able to see the logs printed in the console:
- Build and run your current scheme.
- Wait until the app is up and running.
- Click on Stop running the scheme.
- Open the app directly from your test device or simulator.
Run your application and force a crash (you can do it by adding a fatalError()
in your code). Launch the application again using Xcode and you should see a log printed in the console that indicates that a crash has been detected and sent:
How Crash Reporter works
Section titled How Crash Reporter worksInitialization
Section titled InitializationCrash Reporter is started automatically and begins to report crashes when your application is launched.
Configuration
Section titled ConfigurationOnce started, our SDK fetches its configuration from our servers. It will start reporting crashes if the Crash Reporter setting is enabled (this is handled by the Contentsquare team).
Reporting
Section titled ReportingWhen a crash occurs, a report is created and stored locally in the device. Once the app is launched again, the report is sent to our servers and then removed from the local storage.
Sending data
Section titled Sending dataFor each crash, a new event will be sent to analytics and Session Replay data when the app is launched again after the crash occurred and the network conditions allow for the server to be reached.
Our requests use lowPriority
↗.
Collected data points
Section titled Collected data pointsCrash Reporter handles multiple types of crashes, including:
- Mach kernel exceptions
- Fatal signals
- C++ exceptions
- Objective-C exceptions
For each crash we collect information of the application, system, process, threads, stack traces and some other events metadata you can find in collected data points.
Known limitations and recommendations
Section titled Known limitations and recommendationsCompatibility with other crash reporters
Section titled Compatibility with other crash reportersBeyond Firebase Crashlytics (see Setup Firebase Crashlytics compatibility mode), the Contentsquare Crash Reporter is not compatible with other crash reporters. For more details, reach out to your Contentsquare contact.
Impact on Performance
Section titled Impact on PerformanceWe always strive to be non-intrusive, and transparent to the developers of the client app. We apply this rule on the performance as well. These are the technical specifics we can share on performance, if you have any questions feel free to reach out to us.
The following performance results were obtained under the following conditions:
Condition | Value |
---|---|
Device model | iPhone 12 Pro Max 256GB |
iOS version | 15.6.1 |
Test App built using Xcode version | 14.3.1 |
Test App built with Swift version | 5.8 |
Property | Value |
---|---|
SDK size (installed size, no Bitcode) | 1.5 MB |
Max RAM usage | <1MB |
Max SDK CPU peak on event | <6% |