Setting up the certifications for enabling Push Notifications in React Native can be a bit tricky! And now that Parse is being shut down, you're probably looking for alternative solutions for setting up the backend for push notifications.
In enabling push notifications for our React Native app we decided against Parse and set up our own backend using the 'node-apn' module. This tutorial will walk you through the process of setting up PushNotificationIOS in your React Native Project, obtaining the necessary certificates and using the 'node-apn' module.
To get started, I highly recommend reading an overview of the process of sending a Push Notification on the iOS Developer Library Guide: The Path of a Push Notification.
NB: Push notifications are not available in the iOS Simulator in Xcode. To test push notifications, you need an iOS device, as well as an Apple Developer license.
Push notifications are sent from Apple servers to the application on a users phone. This requires your app to have certain certificates to prove it is authorised to send push notifications. We recommend following the step-by-step guide on the Parse website to obtain the necessary certificates. This is a quick overview of the process:
- Create an App ID and the associated SSL certificate on the Apple Developer website associated with your Developer account. This certificate will allow the push notification server we will create to connect to the Apple Push Notification Service (APNs) send push notifications to the application identified by the App ID on a user's phone (identified by their specific device token). Each App ID is required to have its own client SSL certificate.
- 1.1 Create a certificate signing request file ('CertificateSigningRequest.certSigningRequest'). This will authenticate the creation of the SSL certificate.
- 1.2 Create an AppID on the Apple Developer Member Center - every IOS application needs a unique AppID
- 1.3 Configure your App ID for development push notifications (NB a separate certificate is needed for production). This involves uploading the '.certSigningRequest' file, generating an SSL certificate of the form 'cert.cer', downloading it and installing it (if you have a Mac, the certificate will be installed in your keychain when you double click on the download)
- 1.4 Export the certificate from the keychain and name it 'key.p12'.
- Create a new Provisioning Profile for push notifications in the Apple Developer Member Centre and install it.
- A Provisioning Profile authenticates your device to run the app
- In Xcode > Preferences > Accounts click 'Download all' to download the Push notification provisioning profile
NB: For Ad Hoc Production (beta testing) or Production (submission to the App store), new certificates and keys need to be generated so the whole of Step 3 needs to be repeated.
Let's assume the backend for your app has an endpoint e.g. '/push' with a handler that sends push notifications to APNs which then sends it to the user (identified by their device token). Using the node-apn module we can establish a connection with APNs to send push notifications. To set up the connection you need a Certificate and Key files. These correspond to the 'cert.cer' and 'key.p12' files created in Step 3. These files need to be converted into '.pem' files. To do this, follow the tutorial on the node-apn wiki. The key commands are:
$ openssl x509 -in cert.cer -inform DER -outform PEM -out cert.pem
$ openssl pkcs12 -in key.p12 -out key.pem -nodes
Run these commands in your terminal from the folder in which your cert.cer and key.p12 files are saved. These '.pem' files need to be save in the same folder as the APN helper module we will now create. We'll now set up the basic boilerplate for the APN connection. Create a file called apn.js
and add the following code.
var apn = require('apn');
var path = require('path');
var apnConnection = new apn.Connection({
production: false,
cert: path.join(__dirname, 'cert.pem'), //path to cert.pem file
key: path.join(__dirname, 'key.pem'), // path to key.pem file
});
/**
* Create a apn instance.
* @param {String} - device_token
* @return {Object} - new instance object
**/
function getNewApnInstance (device_token) {
return {
device: new apn.Device(device_token),
note: new apn.Notification()
};
}
/**
* Send notification to a specific device identified by the device token
* @param (Function) - callback
*
**/
function sendNotification (device_token) {
var instance = getNewApnInstance(device_token);
instance.note.alert = 'You have a new push notification!';
instance.note.payload = {text: 'Message from Joe Bloggs'};
apnConnection.pushNotification(instance.note, instance.device);
apnConnection.on('transmitted', function (notification, device){
console.log('on.transmitted', arguments);
});
apnConnection.on('transmissionError', function (errorCode, notification, device) {
console.log('on.transmissionError',arguments);
});
};
module.exports = {
sendNotification: sendNotification
}
You can then use these functions in any of your server endpoint handlers. e.g the handler for the '/push' endpoint might look something like this:
var Apn = require('./apn.js');
module.exports = (req, reply) => {
const { device_token } = req.payload;
Apn.sendNotification(device_token);
return reply({
status: 'success',
});
}
The device token is sent in the request payload and the sendNotification
method is used to send the notification to the specific user.
### Step 1: Add PushNotificationIOS Library and Link
Copy the PushNotificationIOS.xcodeproj file (from the node_modules/react-native/Libraries/PushNotifications folder in your project) into the Libraries folder in Xcode. Follow the step-by-step guide to linking libraries in the React Native docs on the website.
Add the following lines of code to the AppDelegate.m file in your project.
At the top of the file add:
#import "RCTPushNotificationManager.h"
At the bottom of the file before '@end':
// Required for the register event.
- (void)application:(UIApplication *)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData *)deviceToken
{
[RCTPushNotificationManager application:application didRegisterForRemoteNotificationsWithDeviceToken:deviceToken];
}
// Required for the notification event.
- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)notification
{
[RCTPushNotificationManager application:application didReceiveRemoteNotification:notification];
}
Require in PushNotificationIOS
at the top of your component. Event listeners can then be attached to the 'register' and 'notification' events. It is best to add these in the top level of your app (index.ios.js
) inside the componentWillMount
lifecycle method.
componentWillMount(){
PushNotificationIOS.addEventListener('register', (token) => console.log('TOKEN', token))
PushNotificationIOS.addEventListener('notification', (notification) => console.log('Notification', notification, "APP state", AppStateIOS.currentState))
// you could check the app state to respond differently to push notifications depending on if the app is running in the background or is currently active.
PushNotificationIOS.requestPermissions();
}
The callbacks for the event listeners could be replaced with Redux actions.
The device token is a hexadecimal string and is like a phone number. It enables the APNs to locate the device on which the app is installed. The callback for the 'register' event can save the device token associated with the particular user.
In this example the currentState of the app is also being checked so that the app can respond to the push notification depending on if the app is running in the background or currently active.
When the app is opened with a push notification use the PushNotificationIOS.popInitialNotification()
method to retrieve the data in the push notification payload.
If you've followed the tutorial through you should be able to run the app and test out sending a push notification from the command line. Run the app on your phone and console.log
your device token. Then send yourself a test push notification e.g. by sending a request to the '/push' endpoint on your server with your device token:
curl localhost:9009/push --data 'device_token=your_token_here'
This tutorial was written while working on a React Native project with @besarthoxhaj, @izaakrogan and @jrans and @foundersandcoders.
If you like what we've done, and want to get in touch tweet me @nikhilaravi.
You might also be interested in some of the React native modules we've created