1--- 2title: Use Sentry 3maxHeadingDepth: 4 4description: A guide on installing and configuring Sentry for crash reporting. 5--- 6 7import { ConfigReactNative } from '~/components/plugins/ConfigSection'; 8import PlatformsSection from '~/components/plugins/PlatformsSection'; 9import { Collapsible } from '~/ui/components/Collapsible'; 10import { Terminal } from '~/ui/components/Snippet'; 11import { Step } from '~/ui/components/Step'; 12import { Tab, Tabs } from '~/ui/components/Tabs'; 13import { CODE } from '~/ui/components/Text'; 14 15[Sentry](http://getsentry.com/) is a crash reporting platform that provides you with **real-time insight into production deployments with info to reproduce and fix crashes**. 16 17It notifies you of exceptions or errors that your users run into while using your app, and organizes them for you on a web dashboard. Reported exceptions include stacktraces, device info, version, and other relevant context automatically; you can also provide additional context that is specific to your application, like the current route and user id. 18 19## Why sentry-expo? 20 21- Sentry treats React Native as a first-class citizen and we have collaborated with Sentry to make sure Expo is, too. 22- It's very easy to set up and use 23- It scales to meet the demands of even the largest projects. 24- We trust it for our projects at Expo. 25- It is free for up to 5,000 events per month. 26- It streamlines your error-reporting code across Android, iOS, and web 27 28> Native crash reporting is not available in Expo Go, it is only available in standalone builds or development builds. 29 30<PlatformsSection title="Platform compatibility" android emulator ios simulator web /> 31 32## Install and configure Sentry 33 34<Step label="1"> 35 36### Sign up for a Sentry account and create a project 37 38Before getting real-time updates on errors and making your app generally incredible, you'll need to make sure you've created a Sentry project. Here's how to do that: 39 40<Step label="1.1"> 41 42[Sign up for Sentry](https://sentry.io/signup/) (it's free), and create a project in your 43Dashboard. Take note of your **organization slug**, **project name**, and **`DSN`** as you'll need 44them later: 45 46- **organization slug** is available in your **Organization settings** tab 47- **project name** is available in your project's **Settings** > **Projects** tab (find it in the list) 48- **`DSN`** is available in your project's **Settings** > **Projects** > **Project name** > **Client Keys 49 (DSN)** tab. 50 51</Step> 52 53<Step label="1.2"> 54 55Go to the [Sentry API section](https://sentry.io/settings/account/api/auth-tokens/), and create an 56**auth token**. The token requires the scopes: `org:read`, `project:releases`, and 57`project:write`. Save them. 58 59</Step> 60 61Once you have each of these: organization slug, project name, DSN, and auth token, you're all set! 62 63</Step> 64 65<Step label="2"> 66 67### Installation 68 69In your project directory, run: 70 71<Terminal cmd={['$ npx expo install sentry-expo']} /> 72 73`sentry-expo` also requires some additional Expo module packages. To install them, run: 74 75<Terminal 76 cmd={['$ npx expo install expo-application expo-constants expo-device @sentry/react-native']} 77/> 78 79</Step> 80 81<Step label="3"> 82 83### Code 84 85#### Initialization 86 87Add the following to your app's main file such as **App.js**: 88 89```js 90import * as Sentry from 'sentry-expo'; 91 92Sentry.init({ 93 dsn: 'YOUR DSN HERE', 94 enableInExpoDevelopment: true, 95 debug: true, // If `true`, Sentry will try to print out useful debugging information if something goes wrong with sending the event. Set it to `false` in production 96}); 97``` 98 99#### Usage 100 101Depending on which platform you are on (mobile or web), use the following methods to access any `@sentry/*` methods for instrumentation, performance, capturing exceptions and so on: 102 103- For React Native, access any `@sentry/react-native` exports with `Sentry.Native.*` 104- For web, access any `@sentry/browser` exports with `Sentry.Browser.*` 105 106```js 107// Access any @sentry/react-native exports via: 108// Sentry.Native.* 109 110// Access any @sentry/browser exports via: 111// Sentry.Browser.* 112 113// The following example uses `captureException()` from Sentry.Native.* to capture errors: 114try { 115 // your code 116} catch (error) { 117 Sentry.Native.captureException(error); 118} 119``` 120 121</Step> 122 123<Step label="4"> 124 125### App configuration 126 127Configuring `sentry-expo` is done through the config plugin in your [app config](/workflow/configuration/). 128 129<ConfigReactNative> 130 131If you use bare workflow, **you should not use the `plugins` property in app config**. Instead, run `yarn sentry-wizard -i reactNative -p ios android` to configure your native projects. This `sentry-wizard` command will add an extra import statement and `Sentry.init` to your project's root file (usually **App.js**) as shown below. Make sure you remove it, however, keep the `sentry-expo` import and original `Sentry.init` call. 132 133```js 134import * as Sentry from '@sentry/react-native'; 135 136Sentry.init({ 137 dsn: 'YOUR DSN', 138}); 139``` 140 141</ConfigReactNative> 142 143#### Configure a `postPublish` hook 144 145Add `expo.hooks` property to your project's **app.json** or **app.config.js** file: 146 147```json app.json 148{ 149 "expo": { 150 /* @hide ... your existing configuration */ /* @end */ 151 "hooks": { 152 "postPublish": [ 153 { 154 "file": "sentry-expo/upload-sourcemaps", 155 "config": { 156 "organization": "sentry org slug, or use the `SENTRY_ORG` environment variable", 157 "project": "sentry project name, or use the `SENTRY_PROJECT` environment variable" 158 } 159 } 160 ] 161 } 162 } 163} 164``` 165 166To upload the source map to Sentry, you must create a [Sentry auth token](https://docs.sentry.io/product/cli/configuration/). After creating your Sentry auth token [here](https://sentry.io/settings/account/api/), you can configure this token through the `SENTRY_AUTH_TOKEN` environment variable in [EAS Build](/build-reference/variables/#using-secrets-in-environment-variables). 167 168> **warning** Never commit secrets to your repository. Keep the Sentry auth token in a safe place, outside your repository and use the environment variable instead. 169 170Besides the auth token, you can also configure the following options through environment variables: 171 172- organization → `SENTRY_ORG` 173- project → `SENTRY_PROJECT` 174 175<Collapsible summary="Additional configuration options"> 176 177In addition to the required config fields above, you can also provide these **optional** fields: 178 179- `setCommits`: boolean value indicating whether or not to tell Sentry about which commits are associated with a new release. This allows Sentry to pinpoint which commits likely caused an issue. 180- `deployEnv`: string indicating the deploy environment. This will automatically send an email to Sentry users who have committed to the release that is being deployed. 181- `distribution`: The name/value to give your distribution (you can think of this as a sub-release). Expo defaults to using your `version` from app.json. **If you provide a custom `distribution`, you must pass the same value to `dist` in your call to `Sentry.init()`, otherwise you will not see stacktraces in your error reports.** 182- `release`: The name you'd like to give your release (for example, `release-feature-ABC`). This defaults to a unique `revisionId` of your JS bundle. **If you provide a custom `release`, you must pass in the same `release` value to `Sentry.init()`, otherwise you will not see stacktraces in your error reports.** 183- `url`: your Sentry URL, only necessary when self-hosting Sentry. 184 185> You can also use environment variables for your config, if you prefer: 186> 187> - setCommits → `SENTRY_SET_COMMITS` 188> - deployEnv → `SENTRY_DEPLOY_ENV` 189> - distribution → `SENTRY_DIST` 190> - release → `SENTRY_RELEASE` 191> - url → `SENTRY_URL` 192 193</Collapsible> 194 195#### Add the Config Plugin 196 197Add `expo.plugins` to your project's **app.json** or **app.config.js** file: 198 199{/* prettier-ignore */} 200```json app.json 201{ 202 "expo": { 203 "plugins": ["sentry-expo"] 204 /* @hide ... your existing configuration */ /* @end */ 205 } 206} 207``` 208 209</Step> 210 211## Source maps 212 213{/* TODO: Drop `expo publish` mention */} 214 215With the `postPublish` hook in place, now all you need to do is run `expo publish` and the source maps will be uploaded automatically. We automatically assign a unique release version for Sentry each time you hit publish, based on the version you specify in **app.json** and a release id on our backend -- this means that if you forget to update the version but hit publish, you will still get a unique Sentry release. 216 217> This hook can also be used as a `postExport` hook if you're [self-hosting your updates](../distribution/custom-updates-server.mdx). 218 219### Uploading source maps at build time 220 221With `expo-updates`, release builds of both Android and iOS apps will create and embed a new update from your JavaScript source at build-time. **This new update will not be published automatically** and will exist only in the binary with which it was bundled. Since it isn't published, the source maps aren't uploaded in the usual way like they are when you run `expo publish` (actually, we are relying on Sentry's native scripts to handle that). Because of this you have some extra things to be aware of: 222 223- Your `release` will automatically be set to Sentry's expected value- `${bundleIdentifier}@${version}+${buildNumber}` (iOS) or `${androidPackage}@${version}+${versionCode}` (Android). 224- Your `dist` will automatically be set to Sentry's expected value: `${buildNumber}` (iOS) or `${versionCode}` (Android). 225- The configuration for build time source maps comes from the **android/sentry.properties** and **ios/sentry.properties** files. For more information, see [Sentry's documentation](https://docs.sentry.io/clients/java/config/#configuration-via-properties-file). Manual configuration is only required for bare projects, the [sentry-expo config plugin handles it otherwise](#add-the-config-plugin). 226- Configuration for `expo publish` and `npx expo export` for projects is done via **app.json**, whether using bare workflow or not. 227 228Skipping or misconfiguring either of these can lead to invalid source maps, and you won't see human-readable stacktraces in your errors. 229 230### Uploading source maps for updates 231 232> This requires SDK 47, with `[email protected]` or above and `[email protected]` or above. 233 234If you're using EAS Update, or if you're self-hosting your updates (this means you run `npx expo export` manually), you need to take the following steps to upload the source maps for your update to Sentry: 235 236- Run `eas update`. This will generate a **dist** folder in your project root, which contains your JavaScript bundles and source maps. This command will also output the 'Android update ID' and 'iOS update ID' that we'll need in the next step. 237- Copy or rename the bundle names in the **dist/bundles** folder to match **index.android.bundle** (Android) or **main.jsbundle** (iOS). 238- Next, you can use the Sentry CLI to upload your bundles and source maps: 239 - `release name` should be set to `${bundleIdentifier}@${version}+${buildNumber}` (iOS) or `${androidPackage}@${version}+${versionCode}` (Android), so for example `[email protected]+1`. 240 - `dist` should be set to the Update ID that `eas update` generated. 241 242<Tabs> 243 <Tab label="Android"> 244 <Terminal 245 cmd={[ 246 '$ node_modules/@sentry/cli/bin/sentry-cli releases \\', 247 ' files <release name> \\', 248 ' upload-sourcemaps \\', 249 ' --dist <Android Update ID> \\', 250 ' --rewrite \\', 251 ' dist/bundles/index.android.bundle dist/bundles/android-<hash>.map', 252 ]} 253 /> 254 </Tab> 255 <Tab label="iOS"> 256 <Terminal 257 cmd={[ 258 '$ node_modules/@sentry/cli/bin/sentry-cli releases \\', 259 ' files <release name> \\', 260 ' upload-sourcemaps \\', 261 ' --dist <iOS Update ID> \\', 262 ' --rewrite \\', 263 ' dist/bundles/main.jsbundle dist/bundles/ios-<hash>.map', 264 ]} 265 /> 266 </Tab> 267</Tabs> 268 269For more information, see Sentry's [instructions for uploading the bundle and source maps](https://docs.sentry.io/platforms/react-native/sourcemaps/#3-upload-the-bundle-and-source-maps). 270 271The steps above define a new 'dist' on Sentry, under the same release as your full app build, and associate the source maps with this new dist. If you want to customize this behavior, you can pass in your own values for `release` and `dist` when you initialize Sentry in your code. For example: 272 273```js 274import * as Sentry from 'sentry-expo'; 275import * as Updates from 'expo-updates'; 276 277Sentry.init({ 278 dsn: 'YOUR DSN', 279 release: 'my release name', 280 dist: 'my dist', 281}); 282``` 283 284These values should match the values you pass to the `sentry-cli` when uploading your source maps. 285 286### Testing Sentry 287 288When building tests for your application, you want to assert that the right flow-tracking or error is being sent to Sentry, but without really sending it to Sentry servers. This way you won't swamp Sentry with false reports during test running and other CI operations. 289 290[`sentry-testkit`](https://zivl.github.io/sentry-testkit) enables Sentry to work natively in your application, and by overriding the default Sentry transport mechanism, the report is not really sent but rather logged locally into memory. In this way, the logged reports can be fetched later for your own usage, verification, or any other use you may have in your local developing/testing environment. 291 292For more information on how to get started, see [`sentry-testkit` documentation](https://zivl.github.io/sentry-testkit/). 293 294> If you're using `jest`, make sure to add `@sentry/.*` and `sentry-expo` to your `transformIgnorePatterns`. 295 296## Error reporting semantics 297 298To ensure that errors are reported reliably, Sentry defers reporting the data to their backend until the next time you load the app after a fatal error rather than trying to report it upon catching the exception. It saves the stacktrace and other metadata to `AsyncStorage` and sends it immediately when the app starts. 299 300## Disabled by default in dev 301 302Unless `enableInExpoDevelopment: true` is set, all your dev/local errors will be ignored and only app releases will report errors to Sentry. You can call methods like `Sentry.Native.captureException(new Error('Oops!'))` but these methods will be no-op. 303 304## Troubleshooting 305 306<Collapsible summary={<>I'm seeing <CODE>error: project not found</CODE> in my build logs</>}> 307 308This error is caused by the script that tries to upload source maps during build. 309 310Make sure your `organization`, `project` and `authToken` properties are set correctly. 311 312</Collapsible> 313 314<Collapsible summary={<><CODE>expo-dev-client</CODE> transactions never finish</>}> 315 316This error is caused by the HTTP request tracking, which creates spans for log requests (`Starting 'http.client' span on transaction...`) to the development server. To fix this, stop creating spans by adding the following code snippet: 317 318```js 319import * as Sentry from 'sentry-expo'; 320import Constants from 'expo-constants'; 321 322Sentry.init({ 323 tracesSampleRate: 1.0, 324 integrations: [ 325 new Sentry.Native.ReactNativeTracing({ 326 shouldCreateSpanForRequest: url => { 327 return !__DEV__ || !url.startsWith(`http://${Constants.expoConfig.hostUri}/logs`); 328 }, 329 }), 330 ], 331}); 332``` 333 334</Collapsible> 335 336## Learn more about Sentry 337 338Sentry does more than just catch fatal errors, learn more about how to use Sentry from their [JavaScript usage](https://docs.sentry.io/platforms/javascript/) documentation. 339