未从后端收到GCM推送通知

时间:2014-07-16 22:25:41

标签: android google-cloud-messaging

我正在开发一款包含推送通知的新应用。以下是我到目前为止所做的事情:

public class C2DMMessageReceiver extends IntentService {
public static final int NOTIFICATION_ID = 1;
private NotificationManager mNotificationManager;
private final String TAG = "GCM Notification receiver..";
NotificationCompat.Builder builder;

public C2DMMessageReceiver() {
    super("GcmIntentService");
}

@Override
protected void onHandleIntent(Intent intent) {
    Bundle extras = intent.getExtras();
    GoogleCloudMessaging gcm = GoogleCloudMessaging.getInstance(this);
    // The getMessageType() intent parameter must be the intent you received
    // in your BroadcastReceiver.
    String messageType = gcm.getMessageType(intent);

    if (!extras.isEmpty()) {  // has effect of unparcelling Bundle
        /*
         * Filter messages based on message type. Since it is likely that GCM
         * will be extended in the future with new message types, just ignore
         * any message types you're not interested in, or that you don't
         * recognize.
         */
        if (GoogleCloudMessaging.
                MESSAGE_TYPE_SEND_ERROR.equals(messageType)) {
            sendNotification("Send error: " + extras.toString());
        } else if (GoogleCloudMessaging.
                MESSAGE_TYPE_DELETED.equals(messageType)) {
            sendNotification("Deleted messages on server: " +
                    extras.toString());
            // If it's a regular GCM message, do some work.
        } else if (GoogleCloudMessaging.
                MESSAGE_TYPE_MESSAGE.equals(messageType)) {
            // This loop represents the service doing some work.
            for (int i=0; i<5; i++) {
                Log.i(TAG, "Working... " + (i+1)
                        + "/5 @ " + SystemClock.elapsedRealtime());
                try {
                    Thread.sleep(5000);
                } catch (InterruptedException e) {
                }
            }
            Log.i(TAG, "Completed work @ " + SystemClock.elapsedRealtime());
            // Post notification of received message.
            sendNotification("Received: " + extras.toString());
            Log.i(TAG, "Received: " + extras.toString());
        }
    }
    // Release the wake lock provided by the WakefulBroadcastReceiver.
    C2DMRegistrationReceiver.completeWakefulIntent(intent);
}

// Put the message into a notification and post it.
// This is just one simple example of what you might choose to do with
// a GCM message.
private void sendNotification(String msg) {
    mNotificationManager = (NotificationManager)
            this.getSystemService(Context.NOTIFICATION_SERVICE);

    PendingIntent contentIntent = PendingIntent.getActivity(this, 0,
            new Intent(this, MainActivity.class), 0);

    NotificationCompat.Builder mBuilder =
            new NotificationCompat.Builder(this)
                    .setSmallIcon(R.drawable.ic_launcher)
                    .setContentTitle("עדכוני TGSpot")
                    .setStyle(new NotificationCompat.BigTextStyle()
                            .bigText(msg))
                    .setContentText(msg);

    mBuilder.setContentIntent(contentIntent);
    mNotificationManager.notify(NOTIFICATION_ID, mBuilder.build());
}
}

这是处理从我的第三方服务器后端代码接收来自GCM的数据的代码。

设备的令牌已成功保存在服务器上,并通过常规HTTP Post请求推送到服务器。

这是Android清单:

    <permission android:name="com.tgspot.android.permission.C2D_MESSAGE"
    android:protectionLevel="signature"/>

<uses-permission android:name="com.tgspot.android.permission.C2D_MESSAGE"/>
<uses-permission android:name="com.google.android.c2dm.permission.RECEIVE"/>

<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

<uses-permission android:name="android.permission.WAKE_LOCK" />

<uses-permission android:name="android.permission.GET_ACCOUNTS" />
<uses-permission android:name="android.permission.USE_CREDENTIALS" />

<application
    android:allowBackup="true"
    android:icon="@drawable/ic_launcher"
    android:theme="@style/Theme.Base.AppCompat.Light.DarkActionBar"
    android:label="@string/app_name">
    <receiver
        android:name=".C2DMRegistrationReceiver"
        android:permission="com.google.android.c2dm.permission.SEND"
        android:enabled="true">
        <intent-filter>
            <action android:name="com.google.android.c2dm.intent.REGISTRATION" >
            </action>

            <category android:name="com.tgspot.android" />
        </intent-filter>
    </receiver>
    <service android:name=".C2DMMessageReceiver">
        <intent-filter>
         <action android:name="com.google.android.c2dm.intent.RECEIVE" />
        </intent-filter>
    </service>
    <activity
        android:name="com.tgspot.android.MainActivity"
        android:label="@string/app_name">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
</application>

C2DMRegistrationReceiver.class

public class C2DMRegistrationReceiver extends WakefulBroadcastReceiver {

@Override
public void onReceive(Context context, Intent intent) {
    String action = intent.getAction();
    Log.w("C2DM", "Registration Receiver called");

    if ("com.google.android.c2dm.intent.REGISTRATION".equals(action)) {
        Log.w("C2DM", "Received registration ID");
        final String registrationId = intent
                .getStringExtra("registration_id");
        String error = intent.getStringExtra("error");

        Log.d("C2DM", "dmControl: registrationId = " + registrationId
                + ", error = " + error);
        String deviceId = Settings.Secure.getString(context.getContentResolver(),
                Settings.Secure.ANDROID_ID);
        sendRegistrationIdToServer(deviceId, registrationId);
        // Also save it in the preference to be able to show it later
        saveRegistrationId(context, registrationId);
    } else if ("com.google.android.c2dm.intent.RECEIVE".equals(action)) {
        ComponentName comp = new ComponentName(context.getPackageName(),
                C2DMMessageReceiver.class.getName());
        startWakefulService(context, (intent.setComponent(comp)));
        setResultCode(Activity.RESULT_OK);
    }
}

private void saveRegistrationId(Context context, String registrationId) {
    SharedPreferences prefs = PreferenceManager
            .getDefaultSharedPreferences(context);
    SharedPreferences.Editor edit = prefs.edit();
    edit.putString("C2DMM_AUTH", registrationId);
    edit.commit();
}

public void sendRegistrationIdToServer(String deviceId,
                                       String registrationId) {


    Log.d("C2DM", "Sending registration ID to my application server");
    final HttpClient client = new DefaultHttpClient();
    final HttpPost post = new HttpPost("http://www.tgspot.co.il/register/");

    // set up the POST data
    List<NameValuePair> nameValuePairs = new ArrayList<NameValuePair>(1);
    // Get the deviceID
    nameValuePairs.add(new BasicNameValuePair("os", "Android"));
    nameValuePairs.add(new BasicNameValuePair("token",
            registrationId));

    // encode the POST data
    try {
        post.setEntity(new UrlEncodedFormEntity(nameValuePairs));
    } catch (UnsupportedEncodingException e1) {
        //e1.printStackTrace();
    }

    // execute the HTML POST call to hit the server
    new Thread(new Runnable() {
        public void run() {
            try {
                HttpResponse response = client.execute(post);
            } catch (ClientProtocolException e) {
                //e.printStackTrace();
            } catch (IOException e) {
                //e.printStackTrace();
            }
        }
    }).start();
    }
  }

1 个答案:

答案 0 :(得分:1)

在您的清单中,您遗失<action android:name="com.google.android.c2dm.intent.RECEIVE" /> C2DMRegistrationReceiver的意图过滤器。没有它,GCM消息广播就无法到达应用程序。

您还没有在问题中包含清单的权限部分,因此我无法说出您是否遗漏了其他内容。

您的C2DMRegistrationReceiver仅处理注册意图。它还应该处理com.google.android.c2dm.intent.RECEIVE。这些是接收GCM消息的意图。

实际上,如果您使用新的同步注册方法与GoogleCloudMessaging.register(我发现您已经在使用该类),C2DMRegistrationReceiver根本不必处理REGISTRATION意图。

您的onReceive方法应更改为:

public void onReceive(Context context, Intent intent) {
    String action = intent.getAction();
    Log.w("C2DM", "Registration Receiver called");

    if ("com.google.android.c2dm.intent.REGISTRATION".equals(action)) {
        Log.w("C2DM", "Received registration ID");
        final String registrationId = intent
                .getStringExtra("registration_id");
        String error = intent.getStringExtra("error");

        Log.d("C2DM", "dmControl: registrationId = " + registrationId
                + ", error = " + error);
        String deviceId = Settings.Secure.getString(context.getContentResolver(),
                Settings.Secure.ANDROID_ID);
        sendRegistrationIdToServer(deviceId, registrationId);
        // Also save it in the preference to be able to show it later
        saveRegistrationId(context, registrationId);
    } else if ("com.google.android.c2dm.intent.RECEIVE".equals(action)) {
        ComponentName comp = new ComponentName(context.getPackageName(),
                C2DMMessageReceiver.class.getName());
        startWakefulService(context, (intent.setComponent(comp)));
        setResultCode(Activity.RESULT_OK);
    }
}

这样com.google.android.c2dm.intent.RECEIVE广播就会触发显示通知的C2DMMessageReceiver服务。

如果您使用新的注册方法,则根本不需要第一个if条件。