ApplePay.php 3.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  1. <?php
  2. namespace app\api\controller;
  3. use think\Controller;
  4. use think\Db;
  5. /**
  6. * 苹果支付
  7. * Class ApplePay
  8. */
  9. class ApplePay extends Base
  10. {
  11. protected $noNeedLogin = [];
  12. protected $noNeedRight = '*';
  13. protected $orderNum = null;
  14. protected $model = null;
  15. protected $relationSearch = true;
  16. protected $appleCode = [
  17. 21000 => 'App Store无法读取你提供的JSON数据',
  18. 21002 => '收据数据不符合格式',
  19. 21003 => '收据无法被验证',
  20. 21004 => '你提供的共享密钥和账户的共享密钥不一致',
  21. 21005 => '收据服务器当前不可用',
  22. 21006 => '收据是有效的,但订阅服务已经过期。当收到这个信息时,解码后的收据信息也包含在返回内容中',
  23. 21007 => '收据信息是测试用(sandbox),但却被发送到产品环境中验证',
  24. 21008 => '收据信息是产品环境中使用,但却被发送到测试环境中验证'
  25. ];
  26. /**
  27. * @title 验证支付票据 完成订单接口
  28. */
  29. public function verifyReceipt()
  30. {
  31. $receipt = $this->request->param('receipt/s', '');
  32. if (empty($receipt)) $this->error('订单错误');
  33. $this->orderNum = $this->request->param('order_id/s', '');
  34. $order = $this->model->where(array('order_id' => $this->orderNum))->find();
  35. if (empty($order)) {
  36. $this->error('订单错误');
  37. }
  38. if ($order['state'] == 1) $this->error('订单已成功支付,请确认');
  39. $time = time();
  40. file_put_contents("applePay.txt", "\n" . date("Y-m-d H:i:s", $time) . ",支付凭证:" . $receipt, FILE_APPEND);$jsonItem = json_encode(['receipt-data' => $receipt]);
  41. $url = 'https://buy.itunes.apple.com/verifyReceipt'; //正式
  42. //模拟post提交(下面会贴出来),将前端获取到的凭证,去和苹果换取详细的支付信息
  43. $result = $this->http_post_json($jsonItem, $url);
  44. if ($result['status'] == '21007') {
  45. //验证失败 返回app错误状态
  46. $url = 'https://sandbox.itunes.apple.com/verifyReceipt'; //测试
  47. $result = $this->http_post_json($jsonItem, $url);
  48. }
  49. file_put_contents("applePay.txt", json_encode($result) . "\n" . "\n", FILE_APPEND);
  50. //如果检测到 等于 0 就是支付成功,其他的错误码去获取对应错误信息
  51. if ($result['status'] !== 0) {
  52. //验证失败 返回app错误状态
  53. $this->error($this->appleCode[$result['status']]);
  54. }
  55. //接下来就是做自己的业务逻辑
  56. $this->success('充值成功');
  57. }
  58. //模拟post提交
  59. public function http_post_json($json, $url)
  60. {
  61. $ch = curl_init($url);
  62. curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
  63. curl_setopt($ch, CURLOPT_POST, true);
  64. curl_setopt($ch, CURLOPT_POSTFIELDS, $json);
  65. curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, 0); //这两行一定要加,不加会报SSL 错误
  66. curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, 0);
  67. $response = curl_exec($ch);
  68. $errno = curl_errno($ch);
  69. $errmsg = curl_error($ch);
  70. curl_close($ch);
  71. $data = json_decode($response, true);
  72. return $data;
  73. }
  74. }