Ionic Capacitor Build
Upload your source maps using ionic capacitor and Sentry CLI.
In this guide, you'll learn how to successfully upload source maps for TypeScript using our sentry-cli tool.
This guide assumes the following:
- sentry-cliversion >=- 2.17.0
- Sentry JavaScript SDK version >= 7.47.0
This guide is only applicable if you're using the command ionic capacitor build to compile your project. If you use another command, you'll most likely want to follow that guide instead.
You can use the Sentry Wizard to give you an initial setup, but manual changes will still be required in order to set up source maps for this build setup.
npx @sentry/wizard@latest -i sourcemaps
The wizard will guide you through the following steps:
- Logging into Sentry and selecting a project
- Installing the necessary Sentry packages
- Configuring your build tool to generate and upload source maps
- Configuring your CI to upload source maps
First, configure the TypeScript compiler to output source maps:
tsconfig.json{
  "compilerOptions": {
    "sourceMap": true,
    "inlineSources": true,
    // Set `sourceRoot` to  "/" to strip the build path prefix from
    // generated source code references. This will improve issue grouping in Sentry.
    "sourceRoot": "/"
  }
}
Generating source maps may expose them to the public, potentially causing your source code to be leaked. You can prevent this by configuring your server to deny access to .js.map files, or by deleting the source maps before deploying your application.
See the installation instructions for Sentry CLI.
For more info on sentry-cli configuration visit the Sentry CLI configuration docs.
Make sure sentry-cli is configured for your project. For that you can use environment variables:
.env.localSENTRY_ORG=example-org
SENTRY_PROJECT=example-project
SENTRY_AUTH_TOKEN=sntrys_YOUR_TOKEN_HERE
Debug IDs are used to match the stack frame of an event with its corresponding minified source and source map file. Visit What are Debug IDs to learn more.
To inject Debug IDs, use the following command after you build your project:
// Paths can vary so check where your assets/public and App/public folder are.
sentry-cli sourcemaps inject /path/to/directory/android/app/src/main/assets/public
sentry-cli sourcemaps inject /path/to/directory/ios/App/public
At the end of minified source files, you should include a comment named debugId, like this:
example_minified_file.js...
//# debugId=<debug_id>
//# sourceMappingURL=<sourcemap_url>
Source maps should also contain a field named debug_id:
example_source_map.js.map{
    ...
    "debug_id":"<debug_id>",
    ...
}
After you've injected Debug IDs into your artifacts, upload them using the following command:
// Paths can vary so check where your assets/public and App/public folder are.
sentry-cli sourcemaps upload /path/to/directory/android/app/src/main/assets/public
sentry-cli sourcemaps upload /path/to/directory/ios/App/public
Open up Sentry and navigate to Project Settings > Source Maps. If you choose “Artifact Bundles” in the tabbed navigation, you'll see all the artifact bundles that have been successfully uploaded to Sentry.
If you're following this guide from your local machine, then you've successfully:
- Generated minified source and source map files (artifacts) by running your application's build process
- Injected Debug IDs into the artifacts you've just generated
- Uploaded those artifacts to Sentry with our upload command
The last step is deploying a new version of your application using the generated artifacts you created in step one. We strongly recommend that you integrate sentry-cli into your CI/CD Pipeline, to ensure each subsequent deploy will automatically inject debug IDs into each artifact and upload them directly to Sentry.
Warning
Only follow these optional steps if you have concluded that you absolutely need them. Using release and dist values will make your artifact upload more specific, but will also make the entire process less forgiving, which may lead to your code not being unminified by Sentry.
Provide a release property in your SDK options.
Sentry.init({
  // This value must be identical to the release name specified during upload
  // with the `sentry-cli`.
  release: "<release_name>",
});
Afterwards, run the sourcemaps upload command with the additional --release option. Please ensure that the value specified for <release_name> is the same value specified in your SDK options.
// Paths can vary so check where your assets/public and App/public folder are.
sentry-cli sourcemaps upload --release=<release_name> /path/to/directory/android/app/src/main/assets/public
sentry-cli sourcemaps upload --release=<release_name> /path/to/directory/ios/App/public
Running upload with --release doesn't automatically create a release in Sentry. Either wait until the first event with the new release set in Sentry.init is sent to Sentry, or create a release with the same name in a separate step with the CLI.
In addition to release, you can also add a dist to your uploaded artifacts to set the distribution identifier for uploaded files. To do so, run the sourcemaps upload command with the additional --dist option.
Provide release and dist properties in your SDK options.
Sentry.init({
  // These values must be identical to the release and dist names specified during upload
  // with the `sentry-cli`.
  release: "<release_name>",
  dist: "<dist_name>",
});
The distribution identifier is used to distinguish between multiple files of the same name within a single release. dist can be used to disambiguate build or deployment variants.
// Paths can vary so check where your assets/public and App/public folder are.
sentry-cli sourcemaps upload --release=<release_name> --dist=<dist_name> /path/to/directory/android/app/src/main/assets/public
sentry-cli sourcemaps upload --release=<release_name> --dist=<dist_name> /path/to/directory/ios/App/public
During compilation, TypeScript will inject some of its runtime dependencies into the output files it produces if needed. It can include things like polyfills for function generators or APIs that may not be available in all the environments. However, the fact that there aren't any original sources makes it impossible to map frames from compiled code to the original sources.
As a workaround, you'll need to tell the TypeScript compiler to use tslib, its own 3rd party package, (which is internally the part of a compiler) instead of injecting runtime dependencies. You'll only need to change what's inside the TypeScript config file, not your source code. Here's how:
- Make sure that tslibis listed as a dependency in yourpackage.jsonfile.
- Once that's done, add two entries in the compilerOptionssection of yourtsconfig.json:
- "noEmitHelpers": trueand -- "importHelpers": true.
That's it! Now you should be able to correctly map the source maps for all your stack trace frames, including internal TypeScript compiler code snippets.
Our documentation is open source and available on GitHub. Your contributions are welcome, whether fixing a typo (drat!) or suggesting an update ("yeah, this would be better").