1--- 2title: Push notifications setup 3sidebar_title: Setup 4description: Learn how to setup push notifications, get credentials for development and production, and test sending push notifications. 5--- 6 7import { Terminal } from '~/ui/components/Snippet'; 8import { Step } from '~/ui/components/Step'; 9import ImageSpotlight from '~/components/plugins/ImageSpotlight'; 10import { BoxLink } from '~/ui/components/BoxLink'; 11 12To utilize Expo's push notification service, you must configure your app by installing a set of libraries, implementing functions to handle notifications, and setting up credentials for Android and iOS. Once you have completed the steps mentioned in this guide, you'll be able to test sending and receiving notifications on a device. 13 14To get the client-side ready for push notifications, the following things are required: 15 16- The user's permission to send them push notifications. 17- The user's [`ExpoPushToken`](/versions/latest/sdk/notifications/#expopushtoken). 18 19## Prerequisites 20 21The following steps described in this guide use [EAS Build](/build/introduction/). However, you can use the `expo-notifications` library without EAS Build by building [your project locally](/workflow/customizing/). 22 23<Step label="1"> 24 25## Install libraries 26 27Run the following command to install the `expo-notifications`, `expo-device` and `expo-constants` libraries: 28 29<Terminal cmd={['$ npx expo install expo-notifications expo-device expo-constants']} /> 30 31- [`expo-notifications`](/versions/latest/sdk/notifications) library is used to request a user's permission and to fetch the `ExpoPushToken`. It is not supported on an Android Emulator or an iOS Simulator. 32- [`expo-device`](/versions/latest/sdk/device) is used to check whether the app is running on a physical device. 33- [`expo-constants`](/versions/latest/sdk/constants) is used to get the `projectId` value from the app config. 34 35</Step> 36 37<Step label="2"> 38 39## Add a minimal working example 40 41The code below shows a working example of how to register for, send, and receive push notifications in a React Native app. Copy and paste it into your project: 42 43```jsx App.js 44import { useState, useEffect, useRef } from 'react'; 45import { Text, View, Button, Platform } from 'react-native'; 46import * as Device from 'expo-device'; 47import * as Notifications from 'expo-notifications'; 48import Constants from 'expo-constants'; 49 50/* @info This handler determines how your app handles notifications that come in while the app is foregrounded. */ 51Notifications.setNotificationHandler({ 52 handleNotification: async () => ({ 53 shouldShowAlert: true, 54 shouldPlaySound: false, 55 shouldSetBadge: false, 56 }), 57}); 58/* @end */ 59 60// Can use this function below or use Expo's Push Notification Tool from: https://expo.dev/notifications 61async function sendPushNotification(expoPushToken) { 62 const message = { 63 to: expoPushToken, 64 sound: 'default', 65 title: 'Original Title', 66 body: 'And here is the body!', 67 data: { someData: 'goes here' }, 68 }; 69 70 await fetch('https://exp.host/--/api/v2/push/send', { 71 method: 'POST', 72 headers: { 73 Accept: 'application/json', 74 'Accept-encoding': 'gzip, deflate', 75 'Content-Type': 'application/json', 76 }, 77 body: JSON.stringify(message), 78 }); 79} 80 81async function registerForPushNotificationsAsync() { 82 let token; 83 /* @info You should make sure the app is running on a physical device since push notifications don't work on an emulator/simulator. */ 84 if (Device.isDevice) { 85 /* @end */ 86 const { status: existingStatus } = await Notifications.getPermissionsAsync(); 87 let finalStatus = existingStatus; 88 if (existingStatus !== 'granted') { 89 const { status } = await Notifications.requestPermissionsAsync(); 90 finalStatus = status; 91 } 92 if (finalStatus !== 'granted') { 93 alert('Failed to get push token for push notification!'); 94 return; 95 } 96 /* @info This provides the ExpoPushToken which is attributed based on the ID of the project. */ 97 token = await Notifications.getExpoPushTokenAsync({ 98 projectId: Constants.expoConfig.extra.eas.projectId, 99 }); 100 /* @end */ 101 console.log(token); 102 } else { 103 alert('Must use physical device for Push Notifications'); 104 } 105 106 /* @info On Android, you need to specify a channel. */ 107 if (Platform.OS === 'android') { 108 Notifications.setNotificationChannelAsync('default', { 109 name: 'default', 110 importance: Notifications.AndroidImportance.MAX, 111 vibrationPattern: [0, 250, 250, 250], 112 lightColor: '#FF231F7C', 113 }); 114 } 115 /* @end */ 116 117 return token; 118} 119 120export default function App() { 121 const [expoPushToken, setExpoPushToken] = useState(''); 122 const [notification, setNotification] = useState(false); 123 const notificationListener = useRef(); 124 const responseListener = useRef(); 125 126 useEffect(() => { 127 registerForPushNotificationsAsync().then(token => setExpoPushToken(token)); 128 129 /* @info This listener is fired whenever a notification is received while the app is foregrounded. */ 130 notificationListener.current = Notifications.addNotificationReceivedListener(notification => { 131 setNotification(notification); 132 }); 133 /* @end */ 134 135 /* @info This listener is fired whenever a user taps on or interacts with a notification (works when an app is foregrounded, backgrounded, or killed). */ 136 responseListener.current = Notifications.addNotificationResponseReceivedListener(response => { 137 console.log(response); 138 }); 139 /* @end */ 140 141 return () => { 142 Notifications.removeNotificationSubscription(notificationListener.current); 143 Notifications.removeNotificationSubscription(responseListener.current); 144 }; 145 }, []); 146 147 return ( 148 <View style={{ flex: 1, alignItems: 'center', justifyContent: 'space-around' }}> 149 <Text>Your expo push token: {expoPushToken}</Text> 150 <View style={{ alignItems: 'center', justifyContent: 'center' }}> 151 <Text>Title: {notification && notification.request.content.title} </Text> 152 <Text>Body: {notification && notification.request.content.body}</Text> 153 <Text>Data: {notification && JSON.stringify(notification.request.content.data)}</Text> 154 </View> 155 <Button 156 title="Press to Send Notification" 157 onPress={async () => { 158 await sendPushNotification(expoPushToken); 159 }} 160 /> 161 </View> 162 ); 163} 164``` 165 166### Configure `projectId` 167 168Using the previous example, when you are registering for push notifications, you need to use [`projectId`](/versions/latest/sdk/constants/#easconfig). This property is used to attribute Expo push token to the specific project. For projects using EAS, the `projectId` property represents the Universally Unique Identifier (UUID) of that project. 169 170`projectId` is automatically set when you create a development build. However, **we recommend setting it manually in your project's code**. To do so, you can use [`expo-constants`](/versions/latest/sdk/constants/) to get the `projectId` value from the app config. 171 172```js 173token = await Notifications.getExpoPushTokenAsync({ 174 projectId: Constants.expoConfig.extra.eas.projectId, 175}); 176``` 177 178One advantage of attributing the Expo push token to your project's ID is that it doesn't change when a project is transferred between different accounts or the existing account gets renamed. 179 180</Step> 181 182<Step label="3"> 183 184## Get Credentials for development builds 185 186For Android and iOS, there are different requirements to set up your credentials. 187 188### Android 189 190For Android, you need to configure **Firebase Cloud Messaging (FCM)** to get your credentials and set up your Expo project. It is required for all Android apps using Expo SDK. 191 192> **warning** FCM is not currently available for `expo-notifications` on iOS. 193 194#### Setting up FCM 195 1961. To create a Firebase project, go to the [Firebase console](https://console.firebase.google.com/) and click on **Add project**. 197 1982. In the console, click the setting icon next to **Project overview** and open **Project settings**. Then, under **Your apps**, click the Android icon to open **Add Firebase to your Android app** and follow the steps. **Make sure that the Android package name you enter is the same as the value of `android.package` from your app.json.** 199 2003. After registering the app, download the **google-services.json** file and place it in your project's root directory. 201 202 > The **google-services.json** file contains unique and non-secret identifiers of your Firebase project. For more information, see [Understand Firebase Projects](https://firebase.google.com/docs/projects/learn-more#config-files-objects). 203 2044. In **app.json**, add an `android.googleServicesFile` field with the relative path to the downloaded **google-services.json** file. If you placed it in the root directory, the path is: 205 206 ```json app.json 207 { 208 "android": { 209 "googleServicesFile": "./google-services.json" 210 } 211 } 212 ``` 213 2145. For push notifications to work correctly, Firebase requires the API key to either be unrestricted (the key can call any API) or have access to both **Firebase Cloud Messaging API** and **Firebase Installations API**. The API key is found under the `client.api_key.current_key` field in **google-services.json** file: 215 216 ```json google-services.json 217 { 218 "client": [ 219 { 220 "api_key": [ 221 { 222 "current_key": "<your Google Cloud Platform API key>" 223 } 224 ] 225 } 226 ] 227 } 228 ``` 229 2306. Firebase also creates an API key in the Google Cloud Platform Credentials console with a name like **Android key (auto-created by Firebase)**. This could be a different key than the one found in **google-services.json**. 231 2327. To be sure that both the `current_key` and the **Android key** in the Credentials console are the same, go to the [Google Cloud API Credentials console](https://console.cloud.google.com/apis/credentials) and click on **Show key** to verify their value. It will be marked as **unrestricted**. 233 234 > Firebase projects with multiple Android apps might contain duplicated data under the `client` array in the **google-services.json**. This can cause issues when the app is fetching the push notification token. **Make sure to only have one client object with the correct keys and metadata in google-services.json**. 235 236Now you can re-build the development build using the `eas build` command. At this point, if you need to create a development build, see [create a development build for a device](/develop/development-builds/create-a-build/#create-a-development-build-for-the-device). 237 238#### Upload server credentials 239 240For Expo to send push notifications from our servers and use your credentials, you'll have to upload your secret server key to your project's Expo dashboard. 241 2421. In the Firebase console, next to **Project overview**, click gear icon to open **Project settings**. 243 2442. Click on the **Cloud Messaging** tab in the Settings pane. 245 2463. Copy the token listed next to the **Server key**. 247 248 > Server Key is only available in **Cloud Messaging API (Legacy)**, which is disabled by default. <br/> Enable it by clicking the three-dot menu > **Manage API in Google Cloud Console** and following the steps in the console. Once the legacy messaging API is enabled, you should see Server Key in that section. 249 250 <ImageSpotlight 251 alt="Getting the server key from Firebase console's Cloud messaging tab." 252 src="/static/images/notifications/server-key-from-fcm.jpg" 253 style={{ maxWidth: 760 }} 254 /> 255 2564. In your [Expo account's](https://expo.dev/) dashboard, select your project, and click on **Credentials** in the navigation menu. Then, click on your **Application Identifier** that follows the pattern: `com.company.app`. 257 2585. Under **Service Credentials** > **FCM Server Key**, click **Add a FCM Server Key** > **Google Cloud Messaging Token** and add the **Server key** from **step 3**. 259 260> Expo Notifications only supports the **Cloud Messaging API (Legacy)** key at this time. This key is deprecated by Firebase. However, it will continue to work until June 30, 2024. We will provide information on migrating to the new v1 key in the future. 261 262### iOS 263 264> **warning** A paid Apple Developer Account is required to generate credentials. 265 266For iOS, make sure you have [registered your iOS device](/develop/development-builds/create-a-build/#create-a-development-build-for-the-device) on which you want to test before running the `eas build` command for the first time. 267 268If you create a development build for the first time, you'll be asked to enable push notifications. Answer yes to the following questions when prompted by the EAS CLI: 269 270- Setup Push Notifications for your project 271- Generating a new Apple Push Notifications service key 272 273<br /> 274 275> If you are not using EAS Build, run `eas credentials` manually. 276 277</Step> 278 279<Step label="4"> 280 281## Test using the push notifications tool 282 283After creating and installing the development build, you can use [Expo's push notifications tool](https://expo.dev/notifications) to quickly send a test notification to your device. 284 2851. Start the development server for your project: 286 287 <Terminal cmd={['$ npx expo start --dev-client']} /> 288 2892. Open the development build on your device. 290 2913. After the `ExpoPushToken` is generated, enter the value in the Expo push notifications tool with other details (for example, a message title and body). 292 2934. Click on the **Send a Notification** button. 294 295<ImageSpotlight 296 alt="Expo push notifications tool overview." 297 src="/static/images/notifications/push-notifications-tool-overview.png" 298 style={{ maxWidth: 1200 }} 299/> 300 301After sending the notification from the tool, you should see the notification on your device. Below is an example of an Android device receiving a push notification. 302 303<ImageSpotlight 304 alt="An Android device receiving a push notification." 305 src="/static/images/notifications/notification-on-android.png" 306 style={{ maxWidth: 360 }} 307/> 308 309</Step> 310 311## Next step 312 313<BoxLink 314 title="Send notifications using Expo's Push API" 315 description="Learn how to set your back-end using Expo's Push API, implementation practices, common errors and security best practices." 316 href="/push-notifications/sending-notifications" 317/> 318