GCM – новый сервис Push-уведомлений от Google

в 9:30, , рубрики: android, gcm, Google API, google cloud messaging, push, Песочница, Разработка под android, метки: , , ,

image
Ранее в Android использовался C2DM (Cloud to Device Messaging), как сервис доставки Push-уведомлений на устройство. Но 26 июня он был официально отменен гуглом. На его место пришел новый GCM (Google Cloud Messaging).

Похожие названия. Одинаковая роль. В чем же разница ?

  • Для использования GCM, необходимо получить Simple API Key в консоли Google APIs.
  • Для GCM нужно получить Sender ID. Он является эквивалентом электронной почты в C2DM. Получить его можно опять-таки из консоли Google APIs, а точнее из URL:
    https://code.google.com/apis/console/#project:{SENDER_ID}
  • Уведомления в GCM имеют формат JSON вместе с простым текстом.
  • GCM может отправлять уведомления сразу на несколько устройств.
  • Теперь одно устройство с одним идентификатором регистрации может получать уведомления сразу с нескольких серверов.
  • Теперь уведомления могут иметь время жизни до 4-х недель. GCM будет хранить их до истечения срока.
  • Теперь можно отправлять уведомления до 4Кб с полезной нагрузкой. Это будет очень выгодно для реал-тайма различных чатов. Однако данный метод будет сильней кушать батарейку устройства.
  • Теперь нет необходимости передавать идентификатор устройства на сервер, чтобы избежать повторных регистраций одного устройства. Канонический идентификатор регистрации определяется GCM по последней регистрации устройства. И если сервер отправит уведомление со старым идентификатором, то GCM вернет канонический (последний) идентификатор, на который надо будет заменить старый.

Настройка GCM

Начнем с Android Manifest

Сначала нужно прописать разрешения:

<uses-permission android:name="android.permission.WAKE_LOCK"/>
<permission android:name="<i>{package}</i>.permission.C2D_MESSAGE" android:protectionLevel="signature" />
<uses-permission android:name="<i>{package}</i>.permission.C2D_MESSAGE" /> 
<uses-permission android:name="com.google.android.c2dm.permission.RECEIVE" /> 

Затем ресивер и сервис:

		
<receiver android:name="com.google.android.gcm.GCMBroadcastReceiver" android:permission="com.google.android.c2dm.permission.SEND" >
			<intent-filter>
				<action android:name="com.google.android.c2dm.intent.RECEIVE" />
				<action android:name="com.google.android.c2dm.intent.REGISTRATION" />
				<category android:name="<i>{package}</i>" />
			</intent-filter>
		</receiver>
		
		<service android:name=".GCMIntentService" />

* {package} заменить на ваш пакет (у меня com.habrahabr.gcm)

Затем в корневом каталоге пакета создаем класс GCMIntentService, наследуемый от GCMBaseIntentService:

package {package};

import android.app.Activity;
import android.app.IntentService;
import android.app.Notification;
import android.app.NotificationManager;
import android.app.PendingIntent;
import android.content.Context;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log;

import com.google.android.gcm.GCMBaseIntentService;

public class GCMIntentService extends GCMBaseIntentService {
	
    private static final String TAG = "GCMIntentService";

    public GCMIntentService() {
        super(GCMConfig.SENDER_ID);
    }

    @Override
    protected void onRegistered(Context context, String registrationId) {
        Log.i(TAG, "Device registered");
        // Здесь мы должны отправить registrationId на наш сервер, чтобы он смог на него отправлять уведомления
    }

    @Override
    protected void onUnregistered(Context context, String registrationId) {
        Log.i(TAG, "Device unregistered");
        
    }

    @Override
    protected void onMessage(Context context, Intent intent) {
        Log.i(TAG, "Received new message");
    }

    @Override
    protected void onDeletedMessages(Context context, int total) {
        Log.i(TAG, "Received deleted messages notification");
    }

    @Override
    public void onError(Context context, String errorId) {
        Log.i(TAG, "Received error: " + errorId);
    }

    @Override
    protected boolean onRecoverableError(Context context, String errorId) {
        Log.i(TAG, "Received recoverable error: " + errorId);
        return super.onRecoverableError(context, errorId);
    }
}
И уже после этого в главном активити прописываем:

        // Делаем проверки
        GCMRegistrar.checkDevice(this);
        GCMRegistrar.checkManifest(this);
        
        // Достаем идентификатор регистрации
        final String regId = GCMRegistrar.getRegistrationId(this);
        
        if (regId.equals("")) { // Если отсутствует, то регистрируемся
          GCMRegistrar.register(this, GCMConfig.SENDER_ID);
        } else {
          Log.v("GCM", "Already registered: " + regId);
        }

Теперь все готово, за исключением отправки самих уведомлений с сервера, но думаю, что для одной статьи этого пока достаточно.

Исходные коды получившегося приложения
GCM Architectural Overview
GCM Advanced Topics

Автор: questman

* - обязательные к заполнению поля