android - 三星 S6 上的打瞌睡模式、GCM 通知和 http 请求

标签 android cordova google-cloud-messaging samsung-mobile phonegap-pushplugin

我的问题与 stackoverflow 的“doze-mode-and-gcm-notifications”和“android-doze-mode-gcm-priority”问题有关,但由于两者的解决方案是使用 Google play Service 8.3.0,所以对我的案子没有帮助。

上下文

我正在通过 GCM 向我的应用程序发送具有高优先级的静默(不直接向用户显示)推送通知。其中一些通知随后会触发各种 http 请求,以向我的 Web 服务发送信息/文件或从我的 Web 服务获取信息/文件。 由于我们需要能够随时执行这些操作,因此我们必须处理手机进入休眠模式的情况。

问题

GCM 通知正确唤醒应用程序,但 Samsung S6 上的 http 请求失败(它们总是返回超时错误)。 cordova-plugin-network-information 不提供插件离线的信息。

在 Nexus 5 设备上进行的相同测试工作得很好。

奖金问题

具有正常优先级的 GCM 通知由处于空闲状态的三星 S6 设备处理。 (根据 android doze documentation ,当处于 IDLE 状态时,设备不会收到任何具有正常优先级的 GCM 通知,这些通知会在设备切换到 IDLE_MAINTENANCE 状态时进行处理。)

设备信息

  • 三星 galaxy S6
  • 连结 5

  • 两者都启用了开发者选项和 usb 调试

  • 两台设备都使用 wifi 连接
  • 两台设备都以相同的方式安装应用程序:使用 USB 数据线连接到我的电脑,然后从 Android Studio 运行应用程序
  • 三星设备是全新的,只安装了基本应用程序。

申请信息

  • Cordova 6.1.1
  • Cordova phonegap-plugin-push 1.7.0
  • Angular 1.5.5/ngCordova 0.1.26-alpha
  • Android Sdk Tools 最后更新(除了 Android 支持库 32,不想马上更新到 33,因为它昨天才发布)

测试

我使用这些命令来激活打瞌睡:

adb shell dumpsys battery unplug
adb shell dumpsys deviceidle force-idle

检查当前状态的命令:

adb shell dumpsys deviceidle | grep mState

状态切换命令:

adb shell dumpsys deviceidle step

从那里我只使用 Postman 请求具有以下数据的 gcm API:

POST https://android.googleapis.com/gcm/send

标题:

Authorization : key=<MYAPIKEY>
Content-Type : application/json

正文(RAW):

{
  "data": { 
    "key": {"key":"value"},
    "content-available": "1"
  },
  "registration_ids": ["<MYAPPTOKEN>"]  
}

最后,如果我需要发送文件,我的应用程序会使用角度 $http 方法或 cordova-plugin-file-transfer 请求我的网络服务

问题

有人遇到过同样的问题吗,你找到解决办法了吗?

有人知道为什么在这种特殊情况下网络不可用(或处于任何状态)吗? (同样,documentation 说:“GCM 高优先级消息让您可靠地唤醒您的应用程序以访问网络”)

如果是这样,您在其他设备型号上是否也遇到了问题?

额外的问题:有人知道为什么三星设备在空闲状态下处理正常的优先级通知吗? (我不知道我是否必须为此创建一个单独的问题,因为它对我来说只是蛋糕上的樱桃,但它可能以某种方式与主要问题相关联)

最佳答案

我也遇到了网络 GCM 问题。我有一个 GCM 服务,可以进行 IM 之类的 Activity 。最近我发现在 Android 7.0(Samsung Galaxy S6s 和 S8)上 sleep 10 分钟后,我的 GCM 服务无法再连接到服务器。这很好奇,因为从文档中听起来我们必须选择加入才能让应用程序受到打瞌睡和应用程序待机的影响。

不过我确实找到了变通办法。如果您使用完全唤醒锁,它会以某种方式允许网络发生。我意识到这是一种蛮力方法。但目前这是我能找到的唯一解决方案。这基本上就是我正在做的......

public class GcmIntentService extends IntentService {
….
    protected void onMessage(Context context, Intent intent) {
    
     PowerManager powerManager = (PowerManager) getApplicationContext().getSystemService(Context.POWER_SERVICE);
        PowerManager.WakeLock wakeLock = powerManager.newWakeLock((PowerManager.FULL_WAKE_LOCK | PowerManager.ACQUIRE_CAUSES_WAKEUP), "GCMOnMessage");

	try {
       logger.d("Processing incoming GCM message. Connecting test. ");
      
       URL url = new URL("http://www.google.com/";;); 
       HttpURLConnection urlConnection = (HttpURLConnection) url.openConnection();
       int status = urlConnection.getResponseCode();
       logger.d("status: " + status);
    }   
    catch (MalformedURLException e) {
				e.printStackTrace();
			} 
    finally {
                logger.d("Releasing wake lock.");
                wakeLock.release();
     }
}

关于android - 三星 S6 上的打瞌睡模式、GCM 通知和 http 请求,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37885789/

相关文章:

Android 使用 MediaStore

java - 谁有资料如何使用人脸检测相机 2 API?

android - 在 android 7.0 中使用 cordova 发送短信时出现安全异常

android - GCMRegistrar.getRegistrationId(context) 无法获取设备 ID

android - 用户是否必须登录到他们的谷歌帐户才能使 Google Cloud Messenger 在 Android 上运行

java - 比较 Java 中的几个复杂列表

android - 应用内结算 : Adding billing on an existing published app

android - 在 Ionic 框架中使用 Genymotion 模拟器

jquery - Android 2.2.2 上的浏览​​器未使用 jQuery 获取 JSONP

android - 如何更改 GCM 推送通知的样式