解析:推送通知 "deviceToken" 未定义

PARSE: Push Notifications "deviceToken" undefined

这是我在 Parse Console 上的安装页面中的情况:

如您所见,有些设备有 "deviceToken",有些没有 "deviceToken"。 这不好,因为每个设备都应该有 deviceToken 才能工作。

这怎么可能?这是一个严重的问题,因为那些没有 deviceToken 的 收不到推送通知!

我已经按照 Parse.com 上的所有说明进行操作,并按照他们所说的进行了所有操作,即使是在他们的空白项目的帮助下以及在网络上的各种问题的帮助下。但是没有解决我的问题,现在我什么也做不了了。

我唯一能想到的是,在我的应用程序(它已经在商店中)之前使用 Google Cloud Messaging 然后使用这个新的更新 我决定更改系统并使用 Parse 完全删除 GCM 系统。是不是两个系统有冲突?

我需要解决这个问题,因为你可以想象这是一个严重的错误,现在我的用户中有 3/4 收不到推送通知。

如果您下载应用程序并安装它:一切都很好,deviceToken 没问题。如果因为设备上已经有 GCM 版本而更新应用程序:

对于没有 deviceToken 的设备,我尝试了很多方法:手动插入 deviceToken、卸载并重新安装应用程序、删除 Parse 上的特定行等。但没有什么好...仍然遇到这个问题。

我的代码

Application.java

public class Application extends android.app.Application {

public Application() {
}

@Override
public void onCreate() {
    super.onCreate();

    // Initialize the Parse SDK.
    Parse.initialize(this, "x", "x");

    // Specify an Activity to handle all pushes by default.
    PushService.setDefaultPushCallback(this, MainActivity.class);

}

}

MainActivity.java 在我的 mainActivity 中,我只是将 deviceToken(在我的例子中是 installationId)连接到我的 userAgent: 中,这个工作正常:我总是有 installationId 没有任何问题!这是我在 MainActivity 中唯一要做的事情(我还需要做其他事情吗?saveInBackground、回调等等?)

deviceToken = ParseInstallation.getCurrentInstallation().getInstallationId();
webSettings.setUserAgentString(userAgent + " ||" + deviceToken);

AndroidManifest.xml

<!-- Gestione del guasto-->
<uses-permission android:name="android.permission.CAMERA" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />

<!-- Utilizzo di internet  -->
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

<!-- Permesso di vibrare per le push notifications -->
<uses-permission android:name="android.permission.VIBRATE" />


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

<!-- Mantiene attivo il processore così da poter ricevere in qualsiasi momento le notifiche -->
<uses-permission android:name="android.permission.WAKE_LOCK" />

<!-- Consente di impostare l'allarme per l'aggiornamento automatico -->
<uses-permission android:name="com.android.alarm.permission.SET_ALARM"/>


<permission android:name="com.hoxell.hoxellbrowser.permission.C2D_MESSAGE"
    android:protectionLevel="signature" />
<uses-permission android:name="com.hoxell.hoxellbrowser.permission.C2D_MESSAGE" />
<uses-permission android:name="com.google.android.c2dm.permission.RECEIVE" />

<application
    android:name="com.hoxell.hoxellbrowser.Application"
    android:allowBackup="true"
    android:icon="@drawable/ic_launcher_hoxell"
    android:label="@string/app_name"
    android:theme="@style/AppTheme"
    >

    <!-- Richiesto per le applicazioni che usano Google Play Services -->
    <meta-data android:name="com.google.android.gms.version" android:value="@integer/google_play_services_version" />

    <activity
        android:name=".MainActivity"
        android:label="@string/app_name"
        android:configChanges="keyboardHidden|orientation|screenSize"
        android:windowSoftInputMode="adjustResize"
        android:launchMode="singleTask">

        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

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

    <activity
        android:name=".SettingsActivity"></activity>

    <receiver android:name=".AlarmReceiver"/>


    <service android:name="com.parse.PushService" />
    <receiver android:name="com.parse.ParseBroadcastReceiver">
        <intent-filter>
            <action android:name="android.intent.action.BOOT_COMPLETED" />
            <action android:name="android.intent.action.USER_PRESENT" />
        </intent-filter>
    </receiver>
    <receiver android:name="com.hoxell.hoxellbrowser.Receiver"
        android:exported="false">
        <intent-filter>
            <action android:name="com.parse.push.intent.RECEIVE" />
            <action android:name="com.parse.push.intent.DELETE" />
            <action android:name="com.parse.push.intent.OPEN" />
        </intent-filter>
    </receiver>
    <receiver android:name="com.parse.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="com.hoxell.hoxellbrowser" />
        </intent-filter>
    </receiver>

</application>

Build.gradle

dependencies {
compile fileTree(dir: 'libs', include: ['*.jar'])
// You must install or update the Support Repository through the SDK manager to use this dependency.
compile 'com.android.support:support-v4:20.+' //support
compile "com.google.android.gms:play-services:5.0.+" //play services
compile 'com.parse.bolts:bolts-android:1.+'
compile fileTree(dir: 'libs', include: 'Parse-*.jar')

}

我真的不知道还能做什么,这就是为什么我希望有人能帮助我。

谢谢。

您可以创建 class 并扩展 Application 来初始化您的应用程序。这样初始化 ParsePushupdateParseInstallation.

示例代码:

public class YourappApplication extends Application {

    @Override
    public void onCreate() {
        super.onCreate();
        Parse.initialize(this,YOU_PARSE_applicationId, YOU_PARSE_clientKey);

        ParseInstallation.getCurrentInstallation().saveInBackground();

        ParsePush.subscribeInBackground("", new SaveCallback() {
            @Override
            public void done(ParseException e) {
                if (e == null) {
                    Log.d("com.parse.push", "successfully subscribed to the broadcast channel.");
                } else {
                    Log.e("com.parse.push", "failed to subscribe for push", e);
                }
            }
        });
    }

    public static void updateParseInstallation(ParseUser user) {
        ParseInstallation installation = ParseInstallation.getCurrentInstallation();
        installation.put("userId", user.getObjectId());
        installation.saveInBackground();
    }
}

然后在您的 LoginActivity 中:updateParseInstallation

ParseUser.logInInBackground(username, password, new LogInCallback() {
                        @Override
                        public void done(ParseUser parseUser, ParseException e) {
                            setProgressBarIndeterminateVisibility(false);
                            if(e == null){
                                YourappApplication.updateParseInstallation(parseUser);
                                Intent intent = new Intent(LoginActivity.this,
                                        MainActivity.class);
                                intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                                intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK);
                                startActivity(intent);
                            }
                       }
          });

有些安装确实在其 deviceToken 列中设置了 GCM 注册 ID,这很好地表明您的设置是正确的。安装是否绝对可能缺少 GCM 注册 ID:

  • 设备可能无法访问 Google 服务框架(例如 Amazon Kindle Fire)。如果您设置了非 GCM ParsePush 接收器,对这些设备的推送应该仍然可以传送。
  • 该应用已从设备中删除,因此已从 GCM 中注销。在这种情况下,我们不会自动删除安装,因为这可能会导致意外的数据丢失,前提是您可以在安装中存储任意数据 class.
  • Google 设备尝试自行注册时,服务框架可能不可用。

在前两种情况下,这是按预期工作的,通常不需要担心。

那些是模拟器用户吗?根据我的经验,如果您 运行 在模拟器上安装它,它不会自动设置设备令牌。

您是否在平板电脑上同时使用 2 个用户配置文件?因为我遇到了完全相同的问题,而且根据我的经验,配置文件之间存在冲突。事实上,在我的情况下,一台设备正在将 2 行注册到 Parse 数据库中。一个带有 deviceToken,一个带有 "undefined" deviceToken。

不幸的是我从来没有解决过这个问题,我猜这是 Parse 的一个错误。

我在重命名模块后遇到了类似的问题。

尝试 'Clean Project' 然后 'Rebuild Project'

我遇到了这个问题,但是在使用 phonegap-parse-plugin 的 cordova 应用程序中。我不得不编辑插件中的 android java 代码来修复,所以效果是一样的。

尝试在初始化和 setDefaultPushCallback 之间添加此保存:

ParseInstallation.getCurrentInstallation().save();

必须在他们之间才能工作。

我已经为这个问题苦苦挣扎了很长时间,直到最近我才意识到发生了什么。不仅包名需要匹配,还需要在声明中包含构建风格。

如果您遵循解析教程并将 com.parse.starter 替换为您的包名称,那么如果您没有构建风格,您就完成了,但如果您有,则需要确保将其包含在内:

com.packagename.build_flavor{调试、发布等}

我的问题是清单中缺少此权限:

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

我还必须在之前添加元数据 <service android:name="com.parse.PushService" />:

<meta-data
android:name="com.parse.APPLICATION_ID"
android:value="YourParseApplicationID" />

<meta-data
android:name="com.parse.CLIENT_KEY"
android:value="YourParseClientKey" />

现在检查您是否可以获得deviceToken。如果一切正常,还要在 Parse 中检查您的应用仪表板。

将解析日志记录设置为详细有助于进一步调试:

Parse.setLogLevel(Parse.LOG_LEVEL_VERBOSE);

我注意到我在清单中使用了错误的 "sender_id":

<meta-data android:name="com.parse.push.gcm_sender_id"
    android:value="id:123456789"/>

这可能不是确切的解决方案,但 LOG_LEVEL_VERBOSE 可能有助于调试。

我不知道 tghis 是否对任何人有帮助,但在从我的包含解析安装的应用程序 Class 中删除此行后,我能够正确获取设备令牌和通知。这是代码。注释行(删除的行)使一切正常。任何人都可以解释为什么这样做吗?

package com.test.android.push;

import android.app.Application;

import com.parse.Parse;
import com.parse.ParseInstallation;
import com.parse.ParsePush;

public class FarrApplication extends Application {
@Override
public void onCreate() {
    super.onCreate();
    Parse.initialize(this, "xxxx", "xxxx");
    ParseInstallation.getCurrentInstallation().saveInBackground();
    ParsePush.subscribeInBackground("---"); // REMOVED AND WORKS