提问人:ScottyB 提问时间:2/18/2015 最后编辑:ScottyB 更新时间:4/11/2015 访问量:841
使用 PHP 处理 Apple 推送通知服务的设备令牌的方法
Method for processing device tokens for Apple Push Notification Service with PHP
问:
这个网站上有很多关于如何处理从 PHP 中的 Apple 推送通知服务异步返回的错误的信息。我在PHP中想出了一个似乎效果很好的方法,但我想要一些反馈。
- fflush() 的使用是否正确?我在一些例子中看到过它,但不是全部。
- 我无法让它为故意错误的设备令牌提供错误。为什么?
- 此解决方案是否可以扩展到数千台设备(假设 PHP 最大内存已充分增加)?
- 其他问题?
注意:
- 通知的设备令牌在开始时存储在数组中。
- 它不是异步的,但它会在发送每个通知后检查(过去的)错误,并在最后一次通知后整整一秒钟内再检查一次。
- 它使用较新的“现代”通知格式,而不是原始或扩展格式。
- 它将令牌数组的索引作为标识符发送到 APNS。
- 它使用 checkAppleErrorResponse() 函数读取前 6 个字节并返回 false 或失败的标识符(索引),以便它可以备份并继续下一个令牌。(失败后发送的所有令牌都将失效。
$ctx = stream_context_create();
stream_context_set_option($ctx, 'ssl', 'passphrase', $iosCertPassphrase);
stream_context_set_option($ctx, "ssl", "local_cert", $iosCertKey);
$fp = NULL;
$errno = NULL;
$errstr = NULL;
// same payload for all
$item2 = chr(2) . pack("n", strlen($payload)) . $payload; // payload item has id 2, a 2-byte length ("n") containing length of payload, then payload
$errorID = -1;
while ($errorID !== false) {
$fp = stream_socket_client($iosHost . ':' . $iosPort, $errno, $errstr, 60, STREAM_CLIENT_CONNECT|STREAM_CLIENT_PERSISTENT, $ctx);
if ($fp === FALSE) {
echo('Failed to create socket');
break;
}
stream_set_blocking($fp, 0);
for ($id = $errorID + 1 ; $id < sizeof($iosTokens); $id++) {
$errorID = false;
$item1 = chr(1) . pack('n', 32) . pack("H*", $iosTokens[$id]['device_token']); // device token item has 1-byte id 1, 2-byte length ("n") containing 32, then 32-byte device token
$item3 = chr(3) . pack('n', 4) . pack('N', $id); // notification identifier has 1-byte id 3, 2-byte length ("n") containing 4, then 4-byte identifier
$frame = $item1 . $item2 . $item3;
$msg = chr(2) . pack("N", strlen($frame)) . $frame; // for "modern" push notification format, msg has 1-byte id 2, 4-byte length ("N") containing the length of the frame, then frame
fwrite($fp, $msg);
$errorID = checkAppleErrorResponse($fp);
fflush($fp);
if ($errorID !== false) // if there's an error, stop now
break;
}
// if done with for loop and no errors, pause for a sec and check one last time
if ($errorID === false) {
$read = array($fp);
$null = null;
$changedStreams = stream_select($read, $null, $null, 0, 1000000);
//check if it is actually false
if ($changedStreams === false)
{
//close stream when done.
socket_close($fp);
fclose($fp);
}
elseif ($changedStreams > 0)
{
// set the error and redo starting after errorID index
$errorID = checkAppleErrorResponse($fp);
}
}
}
}
答:
0赞
Nev_Yakamazi
3/17/2015
#1
希望相关。我用过这个。它是基本但纯金的。通过一点点洒水或安全性以及一些更激烈的错误处理,不久之后我就将其升级为一个可以处理我需要的一切的系统。
http://www.raywenderlich.com/2941/how-to-write-a-simple-phpmysql-web-service-for-an-ios-app
评论
0赞
ScottyB
3/19/2015
该链接指向如何为 iOS 应用程序编写 PHP Web 服务的示例,而不是如何执行 iOS 推送通知。该网站上还有另一个关于 iOS 推送通知的教程,但它有点过时且可扩展性不强。
评论