关于ios:PHP Apple增强型推送通知读取错误响应

PHP Apple Enhanced Push Notification read error response

在PHP中,如何在发送增强的推送通知时使用fread()检查是否存在错误响应?

我已经阅读了Apple文档,在Google上发表了一些模糊的帖子,并且在此处有关于SO的一些问题/答案,但这仍然很令人困惑。

这是我所看的:
http://developer.apple.com/library/mac/#documentation/NetworkingInternet/Conceptual/RemoteNotificationsPG/IPhoneOSClientImp/IPhoneOSClientImp.html
使用PHP从Apple增强的推送通知中读取错误
iPhone推送通知-错误响应问题

基于以下事实,我将在下面回答我自己的问题:(1)我发现这是一个非常令人困惑的主题,并且(2)我不得不将信息拼凑起来,并进行大量的反复试验才能得出结论。工作,以及(3)这篇鼓励鼓励的博客文章:https://blog.stackoverflow.com/2011/07/its-ok-to-ask-and-answer-your-own-questions/


发送推送通知时,存在几个问题:

  • 如果有问题,Apple会断开与您的连接,但您对此一无所知。使用基本通知时,无法知道是否已全部发送。解决方案:这是使用增强的通知,然后检查错误响应的全部内容。请注意,我们将在数据库查询中使用" ORDER BY id ",然后使用ID作为通知中发送的标识符。这样,如果出现问题,我们可以确切地知道数据库中的哪一行导致了问题(因此我们知道苹果何时断开我们的连接并停止发送通知)。然后,我们可以继续向导致问题的行之后的所有行发送推式通知,而不必重新发送到已经发送给该行的通知。

  • 如果一切正常,Apple不会发送任何响应,因此这可能导致脚本暂停并在fread()等待即将到来的数据时永远等待。解决方案:需要将stream_set_blocking设置为0,以便fread总是立即返回。请注意,这会导致另一个小问题,即fread可以在收到错误响应之前返回,但请参见代码中的解决方法,该方法只是在完成所有发送之后暂停1/2秒,然后再次检查fread 。

  • 您发送多个推送通知的速度要比错误回复返回给您的速度快得多。解决方案:同样,这是上述相同的解决方法...在完成所有发送之后,暂停1/2秒,然后再检查一次fread。

这是我使用PHP的解决方案,它解决了我遇到的所有问题。它很基本,但是可以完成工作。我已经测试过一次发送一些通知以及一次发送120,000。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
<?php
/*
 * Read Error Response when sending Apple Enhanced Push Notification
 *
 * This assumes your iOS devices have the proper code to add their device tokens
 * to the db and also the proper code to receive push notifications when sent.
 *
 */


//database
$host ="localhost";
$user ="my_db_username";
$pass ="my_db_password";
$dbname ="my_db_name";
$con = mysql_connect($host, $user, $pass);
if (!$con) {
    die('Could not connect to database: ' . mysql_error());
} else {
    mysql_select_db($dbname, $con);
}

// IMPORTANT: make sure you ORDER BY id column
$result = mysql_query("SELECT id,token FROM `device_tokens` ORDER BY id");

//Setup notification message
$body = array();
$body['aps'] = array('alert' => 'My push notification message!');
$body['aps']['notifurl'] = 'http://www.myexampledomain.com';
$body['aps']['badge'] = 1;

//Setup stream (connect to Apple Push Server)
$ctx = stream_context_create();
stream_context_set_option($ctx, 'ssl', 'passphrase', 'password_for_apns.pem_file');
stream_context_set_option($ctx, 'ssl', 'local_cert', 'apns.pem');
$fp = stream_socket_client('ssl://gateway.push.apple.com:2195', $err, $errstr, 60, STREAM_CLIENT_CONNECT, $ctx);
stream_set_blocking ($fp, 0); //This allows fread() to return right away when there are no errors. But it can also miss errors during last seconds of sending, as there is a delay before error is returned. Workaround is to pause briefly AFTER sending last notification, and then do one more fread() to see if anything else is there.

if (!$fp) {
    //ERROR
    echo"Failed to connect (stream_socket_client): $err $errstrn";
} else {
    $apple_expiry = time() + (90 * 24 * 60 * 60); //Keep push alive (waiting for delivery) for 90 days

    //Loop thru tokens from database
    while($row = mysql_fetch_array($result)) {
        $apple_identifier = $row["id"];
        $deviceToken = $row["token"];
        $payload = json_encode($body);
        //Enhanced Notification
        $msg = pack("C", 1) . pack("N", $apple_identifier) . pack("N", $apple_expiry) . pack("n", 32) . pack('H*', str_replace(' ', '', $deviceToken)) . pack("n", strlen($payload)) . $payload;
        //SEND PUSH
        fwrite($fp, $msg);
        //We can check if an error has been returned while we are sending, but we also need to check once more after we are done sending in case there was a delay with error response.
        checkAppleErrorResponse($fp);
    }

    //Workaround to check if there were any errors during the last seconds of sending.
    usleep(500000); //Pause for half a second. Note I tested this with up to a 5 minute pause, and the error message was still available to be retrieved

    checkAppleErrorResponse($fp);

    echo 'DONE!';

    mysql_close($con);
    fclose($fp);
}

//FUNCTION to check if there is an error response from Apple
//         Returns TRUE if there was and FALSE if there was not
function checkAppleErrorResponse($fp) {

   //byte1=always 8, byte2=StatusCode, bytes3,4,5,6=identifier(rowID). Should return nothing if OK.
   $apple_error_response = fread($fp, 6);
   //NOTE: Make sure you set stream_set_blocking($fp, 0) or else fread will pause your script and wait forever when there is no response to be sent.

   if ($apple_error_response) {
        //unpack the error response (first byte 'command" should always be 8)
        $error_response = unpack('Ccommand/Cstatus_code/Nidentifier', $apple_error_response);

        if ($error_response['status_code'] == '0') {
            $error_response['status_code'] = '0-No errors encountered';
        } else if ($error_response['status_code'] == '1') {
            $error_response['status_code'] = '1-Processing error';
        } else if ($error_response['status_code'] == '2') {
            $error_response['status_code'] = '2-Missing device token';
        } else if ($error_response['status_code'] == '3') {
            $error_response['status_code'] = '3-Missing topic';
        } else if ($error_response['status_code'] == '4') {
            $error_response['status_code'] = '4-Missing payload';
        } else if ($error_response['status_code'] == '5') {
            $error_response['status_code'] = '5-Invalid token size';
        } else if ($error_response['status_code'] == '6') {
            $error_response['status_code'] = '6-Invalid topic size';
        } else if ($error_response['status_code'] == '7') {
            $error_response['status_code'] = '7-Invalid payload size';
        } else if ($error_response['status_code'] == '8') {
            $error_response['status_code'] = '8-Invalid token';
        } else if ($error_response['status_code'] == '255') {
            $error_response['status_code'] = '255-None (unknown)';
        } else {
            $error_response['status_code'] = $error_response['status_code'] . '-Not listed';
        }

        echo '+ + + + + + ERROR Response Command:' . $error_response['command'] . 'Identifier:' . $error_response['identifier'] . 'Status:' . $error_response['status_code'] . '';
        echo 'Identifier is the rowID (index) in the database that caused the problem, and Apple will disconnect you from server. To continue sending Push Notifications, just start at the next rowID after this Identifier.';

        return true;
   }
   return false;
}
?>


不确定代码的内容,但您应该尝试ApnsPHP,它经过良好测试,可以很好地工作,并且能够为您处理所有可能的异常和错误。

其他替代品

https://github.com/sebastianborggrewe/PHP-Apple-Push-Notification-Server
https://github.com/bortuzar/PHP-Mysql---Apple-Push-Notification-Server

已经测试了3个i例子中的2个,并且没有实现和错误管理的问题。

谢谢

:)