简体   繁体   English

来自NotificationManager的java.lang.SecurityException?

[英]java.lang.SecurityException from NotificationManager?

We use push notifications in our game to notify users about certain in-game things when the game receives a pause/suspend call from the engine. 当游戏收到来自引擎的暂停/挂起电话时,我们会在游戏中使用推送通知来通知用户某些游戏内事物。

I've been looking at crash logs from the Google Developer Console and a large porition of them is caused by a SecurityException from the use of push noticiations. 我一直在从Google Developer Console查看崩溃日志,其中很大一部分是由于使用推送通知导致的SecurityException引起的。

Call stack: 调用堆栈:

 java.lang.RuntimeException: 

  at android.app.ActivityThread.handleReceiver (ActivityThread.java:2697)
  at android.app.ActivityThread.access$1500 (ActivityThread.java:178)
  at android.app.ActivityThread$H.handleMessage (ActivityThread.java:1466)
  at android.os.Handler.dispatchMessage (Handler.java:107)
  at android.os.Looper.loop (Looper.java:194)
  at android.app.ActivityThread.main (ActivityThread.java:5560)
  at java.lang.reflect.Method.invokeNative (Native Method)
  at java.lang.reflect.Method.invoke (Method.java:525)
  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run (ZygoteInit.java:844)
  at com.android.internal.os.ZygoteInit.main (ZygoteInit.java:611)
  at dalvik.system.NativeStart.main (Native Method)

Caused by: java.lang.SecurityException: 
  at android.os.Parcel.readException (Parcel.java:1425)
  at android.os.Parcel.readException (Parcel.java:1379)
  at android.app.INotificationManager$Stub$Proxy.enqueueNotificationWithTag (INotificationManager.java:239)
  at android.app.NotificationManager.notify (NotificationManager.java:132)
  at android.app.NotificationManager.notify (NotificationManager.java:108)
  at tinytitan.tinylib.NotificationPublisher.onReceive (NotificationPublisher.java:21)
  at android.app.ActivityThread.handleReceiver (ActivityThread.java:2690)

That 'tinytitan' line is from our Java push notif plugin. 那行“ tinytitan”来自我们的Java push notif插件。 Source code: 源代码:

package tinytitan.tinylib;

import android.app.Notification;
import android.app.NotificationManager;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;

public class NotificationPublisher extends BroadcastReceiver {

    public static String NOTIFICATION_ID = "notification-id";
    public static String NOTIFICATION = "notification";

    /**
     * Called when the BroadcastReceiver is receiving an Intent
     */
    public void onReceive(Context context, Intent intent) {
        NotificationManager notificationManager = (NotificationManager)context.getSystemService(Context.NOTIFICATION_SERVICE);
        Notification notification = intent.getParcelableExtra(NOTIFICATION);
        int id = intent.getIntExtra(NOTIFICATION_ID, 0);
        notificationManager.notify(id, notification); // <-- LINE# 21
    }
}

Here's out manifest files (there's multiple cause we use different plugins): 这是清单文件(有多种原因,我们使用了不同的插件):

Why are we getting this exception and how to fix it? 为什么我们会收到此异常以及如何解决它?

EDIT I tried posting the Manifest files content but the formatting wouldn't let me do it, so I stripped out the angle brackets. 编辑我试图发布清单文件的内容,但是格式不允许我这样做,所以我去掉了尖括号。

uses-permission android:name="android.permission.INTERNET"
uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"
uses-permission android:name="com.android.vending.BILLING"
uses-permission android:name="com.google.android.c2dm.permission.RECEIVE"
uses-permission android:name="android.permission.WAKE_LOCK"
uses-permission android:name="android.permission.VIBRATE"
uses-permission android:name="android.permission.GET_TASKS"
uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"
uses-permission android:name="com.android.vending.CHECK_LICENSE"
uses-permission android:name="android.permission.android.permission.READ_PHONE_STATE"
uses-permission android:name="android.permission.ACCESS_WIFI_STATE"

EDIT: Rest of the plugin's code: 编辑:其余的插件代码:

/**
 * Shows a Notification with a delay
 */
public static void scheduleNotification(int id, long delayMs, String title, String message, boolean vibrate, boolean lightBlink, boolean sound, String largeImage, String smallImage)
{
    Notification notification = getNotification(title, message, vibrate, lightBlink, sound, smallImage, largeImage);

    Intent notificationIntent = new Intent(UnityPlayer.currentActivity, NotificationPublisher.class);
    notificationIntent.putExtra(NotificationPublisher.NOTIFICATION_ID, id);
    notificationIntent.putExtra(NotificationPublisher.NOTIFICATION, notification);
    PendingIntent pendingIntent = PendingIntent.getBroadcast(UnityPlayer.currentActivity, id, notificationIntent, PendingIntent.FLAG_UPDATE_CURRENT);

    long futureInMillis = SystemClock.elapsedRealtime() + delayMs;
    AlarmManager alarmManager = (AlarmManager)UnityPlayer.currentActivity.getSystemService(Context.ALARM_SERVICE);
    alarmManager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP, futureInMillis, pendingIntent);
}

/**
 * Cancels a notification waiting to be added to the notification query or removes a notification
 * from the notification query.
 */
public static void cancelNotification(int id)
{
    //Remove it from the alarm manager (if present)
    AlarmManager am = (AlarmManager)UnityPlayer.currentActivity.getSystemService(Context.ALARM_SERVICE);
    Intent intent = new Intent(UnityPlayer.currentActivity, NotificationPublisher.class);
    PendingIntent pendingIntent = PendingIntent.getBroadcast(UnityPlayer.currentActivity, id, intent, PendingIntent.FLAG_UPDATE_CURRENT);
    am.cancel(pendingIntent);

    //remove it from the notification query (if present)
    android.app.NotificationManager notificationManager = (android.app.NotificationManager)UnityPlayer.currentActivity.getSystemService(Context.NOTIFICATION_SERVICE);
    notificationManager.cancel(id);
}

/**
 * Removes all notifications that are in the notification query
 */
public static void clearAll(){
    android.app.NotificationManager notificationManager = (android.app.NotificationManager)UnityPlayer.currentActivity.getSystemService(Context.NOTIFICATION_SERVICE);
    notificationManager.cancelAll();
}

The call from C#: 来自C#的调用:

    TinyAndroidServices.ScheduleNotification(id, delay, title, message, true, true, true, "app_icon", androidSmallIcon);

You should use this permission 您应该使用此权限

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

Not this: 不是这个:

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

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM