推送通知已成功发送,但设备未收到(偶尔)

ash*_*ina 9 php iphone apple-push-notifications

从昨天开始,我遇到了一些设备无法接收推送通知的问题.证书/设备令牌似乎是正确的,因为该设备用于成功接收推送通知直到昨天.

在服务器端,没有错误或连接拒绝,并且推送通知似乎每次都成功发送.

但是,在很多情况下,设备无法正确接收推送.

一些周边信息:

  • 我在生产环境中这样做.
  • 服务器端没有错误/连接拒绝
  • 我每次都发送完全相同的JSON.
  • 自昨天起,我们的2台设备未收到AT ALL的推送通知
  • 我们的设备中有一个接收推送通知的成功率(约70%)低于昨天
  • 即使是现在,我们仍有1到2台设备成功接收推送通知.
  • 所有上述设备都能够在生产环境中正确接收推送通知,直到昨天.

当推送成功时,以及当设备没有收到时,服务器端结果没有区别......因此几乎不可能识别问题.

这是我使用的服务器端PHP代码:

        $ctx = stream_context_create();
        stream_context_set_option($ctx, 'ssl', 'local_cert', $this->apnsData[$development]['certificate']);
        $fp = stream_socket_client($this->apnsData[$development]['ssl'], $error, $errorString, 100, (STREAM_CLIENT_C ONNECT|STREAM_CLIENT_PERSISTENT), $ctx);

        if(!$fp){
                $this->_pushFailed($pid);
                $this->_triggerError("Failed to connect to APNS: {$error} {$errorString}.");
        }
        else {
                $msg = chr(0).pack("n",32).pack('H*',$token).pack("n",strlen($message)).$message;
                $fwrite = fwrite($fp, $msg);
                if(!$fwrite) {
                        error_log("[APNS] push failed...");
                        $this->_pushFailed($pid);
                        $this->_triggerError("Failed writing to stream.", E_USER_ERROR);
                }
                else {
                        error_log("[APNS] push successful! ::: $token -> $message ($fwrite bytes)");
                }
        }
        fclose($fp);
Run Code Online (Sandbox Code Playgroud)

日志告诉我推送成功(删除令牌以保护隐私):

[Wed Dec 12 11:42:00 2012] [error] [client 10.161.6.177] [APNS] push successful! ::: aa4f******44 -> {"aps":{"alert":{"body":"\\u300casdfasdf\\u300d","action-loc-key":"OK"},"badge":4,"sound":"chime"}} (134 bytes)
Run Code Online (Sandbox Code Playgroud)

我该如何解决这个问题?

ash*_*ina 4

我自己解决了这个问题,所以我会发布答案。

我收到建议,不建议为每条消息打开和关闭套接字,正如苹果官方文档中所述:

“您还应该跨多个通知保留与 APN 的连接。APN 可能会将快速、重复建立和拆除的连接视为拒绝服务攻击。出现错误时,APN 会关闭发生错误的连接。”

我修复了我的架构,以便在多个 APNS 调用中保留连接,现在它可以正常工作。我创建了一个基于 apns-php 的排队系统(https://code.google.com/p/apns-php/)。

这是我的代码,供需要的人使用:

https://github.com/ashiina/APNS-QueueServer