Skip to content

This is a template for setting up a push notification server and then handling them on iOS and Android devices.

License

Notifications You must be signed in to change notification settings

uacaps/PushNotificationsTemplate

Repository files navigation

Push Notifications Template

Android Screenshot iPhone Screenshot

Laziness is Good

Let's face it, setting up push notifications for the first time is a bit daunting... There's the SenderId the RegistrationId, the AppId. What does it all mean??? When do I use what? Why isn't there a working sample project that I can drop in my app information and it just works?

Well, we at CAPS wondered the same thing, so we went ahead and set one up for everyone!

Table of Contents

The iOS App

To start receiving push notifications on your iOS devices, there is a little bit of set up work that must be done before writing any code in Xcode. Here's what you need to do:

Create An App ID

  • Go to the iOS Developer Center
  • Click "Certificates, Identifiers and Profiles"
  • Create an AppId
  • Make sure you enable Push Notifications at the very bottom of the page

Provisioning Profile

  • Create a new Development Provisioning Profile for the AppId you created

Certificates

  • Create a new certificate of type "Apple Push Notification service SSL (Sandbox)"
  • Select the same AppID you've been using so far.
  • Open "Keychain Access" on your Mac
  • Click "Keychain Access->Certificate Assistant->Request A Certificate from a Certificate Authority" in the menu
  • Enter in your credentials, and make sure it's "Saved to Disk", not "Email to Authority"
  • Save it to Desktop
  • Go back to your web browser, and continue where you left off in the dev portal
  • Upload the certificate request and it will give you a certificate to download
  • Add the downloaded certificate to Keychain Access
  • Select the certificate, and then click "File->Export Items" in the menu
  • Export it as a .p12 file - this is what you'll add to your Push Server. We will be using them in The Server App section

Coding The App

After adding the .p12 file to your Push Server, and setting that up, you're ready to begin adding Push Notification functionality to your iOS app. There's really not much you need to do here - and we've created a special class for you that handles a lot of the functionality, PusherMan.{h,m}. Add those 2 files to your project, and then:

#import "PusherMan.h" into

  • AppDelegate.h
  • Any class/controller where you will be registering notifications with your server

PusherMan is a singleton class that just holds onto your device token during the duration of the app, and handles a couple auxiliary methods for registering for notifications with Apple, and retrieving the types of notification that a current device is registered for. Once you've imported PusherMan to your classes, there are only three more methods to add to your AppDelegate.m file, and you are ready to roll. Add these three methods below.

#pragma mark - Push Notification Methods
- (void)application:(UIApplication*)application didRegisterForRemoteNotificationsWithDeviceToken:(NSData*)deviceToken {
    // Successfully registered the current device with Apple's Push Notification Service
	[PusherMan setDeviceToken:deviceToken];
}

- (void)application:(UIApplication*)application didFailToRegisterForRemoteNotificationsWithError:(NSError*)error {
    // The App was unsuccessful in registering this device for APNS
	NSLog(@"Failed to get token, error: %@", error);
}

- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo {
    // This method is called whenever the App receives a Push Notification from Apple,
    // and the app is open - or they tap on the actual push notification on screen that
    // then launches this app and calls this method.
    NSLog(@"Notification: %@", userInfo);
    UIAlertView *alertView = [[UIAlertView alloc] initWithTitle:@"New Noticiation" message:userInfo[@"alert"] delegate:nil cancelButtonTitle:@"Ok!" otherButtonTitles:nil];
    [alertView show];
}

All of these methods are delegate callbacks for either successfully registering for Push Notifications, failing to register, or actually receiving the push notification on your device. These three have to be implemented to receive push notification functionality. Also, at the very top of your AppDelegate should be the method, - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions. Add this line to that method:

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
    [PusherMan registerAppForPushNotifications];
    return YES;
}

All that line does is begin the process of registering for notifications, and then the delegate methods you implemented a second ago handle the rest.

Coding Style for Notifications

Once you've set up your app to receive/handle push notifications, you still need to talk to your server to let it know how/when to actually send the notifications. How your app does this is entirely up to you, and changes on a project by project basis.

The Android App

First let us take a look at the sample Android app provided in the Android App Template folder of the root directory. Inside you will find an app with 3 files.

  • MainActivity.java

    • The startup activity for the app. Nothing special here, but all of our registration setup with Google will go in this class.
  • GcmIntentService.java

    • This class will be what handles the push notification when it arrives. The onHandleIntent() method handles the raw intent from the push notification services of the operating system, while sendNotification(String msg) will display the text of the notification in the Notifcation Drawer.
  • GcmBroadcastReceiver.java

    • This class allows you to receive intents, in our case GCM Intents. It's basically like a switchboard that routes the notification to the GcmIntentService.java class.

The Sender Id

Take a look in the MainActivity.java and you will find this variable near the top: String SENDER_ID = "971352002353";. So, you may be wondering where this seemingly random number comes from. It is actually the Project Number for your app. You can also find it in the goole developer console. For our sample app, we have provided you with this number

Sender Id

Retrieve the Device Registration Id

Every Android device will have a Device Registration Id for push notifications for a specific app. It is a long string of characters that your device will have assigned to it when the app registers for push notifications. Each time you have a new app that you want to send push notifications to, you will need to retreive this identifier from google.

Good news: Our sample Android app already does this! Take a look in the doInBackground() method in PushAsyncTask. Here you are fetching the Device Registration Id from Google, and having it print to the LogCat. In a true system, once you receive the id, you would upload it to your server for safe keeping, but for the sake of example, run the sample app on your droid device and then grab the Device Registration Id from the LogCat. It should look something like this

APA71bGDFHk6HCWxskob04URTmd-MDV3FdKJarba0CcMgkxRtpQdSTqg9zoWDioimi0L-fNiTcgepiRsdGyMbv2gW1FM4FZFV9xlikaSiKrY8s-b3BH2T-bii6kEojdXoM9FR0I6vj2E8WDWLbApaHHYgoBU6wuwWA

Go ahead and copy/paste this somewhere, as we will need it when it comes time to send a push notification to your device.

The Server App

We have provided a sample push server using PushSharp, a great C# library for sending Android and iOS push notifications (and blackberry and windows phone). You can install it through the nuget package manager in visual studio, but for brevity, we have already set that up for you in the sample app.

The push server app is written in MVC 4 and uses Web API to send a test push notification in an event-driven manner, perfect for getting things up and running. Open up the project solution in the Push Server Template folder and head to the Controllers folder to find PushController.cs Here you will find the following Web API method:

[HttpGet]
public void testPush()
{
	var push = new PushBroker();
            
            //**** iOS Notification ******
            //Establish the connections to your certificates. Here we make one for dev and another for production
            byte[] appleCertificate = null;
            //appleCertificate = Properties.Resources.DEV_CERT_NAME;
            //appleCertificate = Properties.Resources.PROD_CERT_NAME;

            //If the file exists, go ahead and use it to send an apple push notification
            if (appleCertificate != null)
            {

                //Give the apple certificate and its password to the push broker for processing
                push.RegisterAppleService(new ApplePushChannelSettings(appleCertificate, "password"));

                //Queue the iOS push notification
                push.QueueNotification(new AppleNotification()
                               .ForDeviceToken("DEVICE_TOKEN_HERE")
                               .WithAlert("Hello World!")
                               .WithBadge(7)
                               .WithSound("sound.caf"));
            }
            //*********************************


	//**** Android Notification ******
	//Register the GCM Service and sending an Android Notification with your browser API key found in your google API Console for your app. Here, we use ours.
	push.RegisterGcmService(new GcmPushChannelSettings("AIzaSyD3J2zRHVMR1BPPnbCVaB1D_qWBYGC4-uU"));

	//Queue the Android notification. Unfortunately, we have to build this packet manually. 
	push.QueueNotification(new GcmNotification().ForDeviceRegistrationId("DEVICE_REGISTRATION_ID")
                      .WithJson("{\"alert\":\"Hello World!\",\"badge\":7,\"sound\":\"sound.caf\"}"));
	//*********************************
}

iOS Notifications

iOS notifications are authenticated through .p12 certificates, particularly the certificates we set up in The iOS App section. A certificate will be bundled with your notification when you send it to Apple for processing, allowing them to know you are approved to send notifications for a specific app. Typically, you have two certificatets, one for development and one for production, both of which we have made a place for in the sample app.

Once you have generated these two certificates, we are going to add them as a resource. Here are the steps to make that happen:

  • Go to the Solution explorer and right click on your project file (not solution file). Select "Properties".
  • From the left-hand menu of the properties pane, select Resources.
  • Click the "Add Resource" dropdown and select "Add Existing File". Navigate to your certifcates and select it. Repeat for the other certificate, if you have two (for dev and production).
  • Once your certificates are added to the resources, you can access them by the following line in the example code.
appleCertificate = Properties.Resources.DEV_CERT_NAME;
  • Now that your certificates are in place, you need only provide a device token and to queue your message!

Android (Google Cloud Messaging) Notifications

AIzaSyD3J2zRHVMR1BPPnbCVaB1D_qWBYGC4-uU in the GcmPushChannelSettings constructor is the test app's api key retreived from the Google API Console. You can find this key here.

API Console

Making It Work

Android Device Credentials

In the testPush() method in PushController.cs, you should see a string called "DEVICE_REGISTARATION_ID". Replace it with a string containing the Device Registration Id we retrieved in The Android App section. We are now ready to test!

iPhone Device Credentials

Also, in the testPush() method in PushController.cs, you shoould see a string called "DEVICE_TOKEN_HERE". Replace it with the iOS device token you retrieved in The iOS App section. This tells PushSharp and, by proxy, Apple which device to send the push notification to.

Testing 1,2,3

Find the play button up top and run the visual studio project. You should be met with a simple page representing the MVC template running on your localhost. Now, navigate to the url

http://localhost:PORT_NUMBER/api/push/testPush

This should trigger the Web API method testPush() to be hit and send a push notification to your phone!

Scaling

So this is great for testing and all, but what about building a system? Here are a few tips and tricks that might help.

  • If you are using PushSharp, you would simply queue additional notifictions for additional devices.
  • Store device tokens in your database and then trigger them on events in your system for registered users.
  • Consider moving the push broker to be a global variable so it is not instantiated every time you want to send a message. If your push load is extra heavy, definitely consider moving the notification triggering to a background thread or other server.

Server Options

In case C#/Microsoft platform isn't your cup of tea for the server stack, there are some other options you can use. Here's some we recommend:

Credits

License

Copyright (c) 2012 The Board of Trustees of The University of Alabama All rights reserved.

Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:

  1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
  2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
  3. Neither the name of the University nor the names of the contributors may be used to endorse or promote products derived from this software without specific prior written permission.

THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

About

This is a template for setting up a push notification server and then handling them on iOS and Android devices.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published