WxPay.MicroPay.php 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. <?php
  2. /**
  3. *
  4. * example目录下为简单的支付样例,仅能用于搭建快速体验微信支付使用
  5. * 样例的作用仅限于指导如何使用sdk,在安全上面仅做了简单处理, 复制使用样例代码时请慎重
  6. * 请勿直接直接使用样例对外提供服务
  7. *
  8. **/
  9. require_once "WxPay.Api.php";
  10. /**
  11. *
  12. * 刷卡支付实现类
  13. * 该类实现了一个刷卡支付的流程,流程如下:
  14. * 1、提交刷卡支付
  15. * 2、根据返回结果决定是否需要查询订单,如果查询之后订单还未变则需要返回查询(一般反复查10次)
  16. * 3、如果反复查询10订单依然不变,则发起撤销订单
  17. * 4、撤销订单需要循环撤销,一直撤销成功为止(注意循环次数,建议10次)
  18. *
  19. * 该类是微信支付提供的样例程序,商户可根据自己的需求修改,或者使用lib中的api自行开发,为了防止
  20. * 查询时hold住后台php进程,商户查询和撤销逻辑可在前端调用
  21. *
  22. * @author widy
  23. *
  24. */
  25. class MicroPay
  26. {
  27. public function __construct($config)
  28. {
  29. $this->config = $config;
  30. }
  31. /**
  32. *
  33. * 提交刷卡支付,并且确认结果,接口比较慢
  34. * @param WxPayMicroPay $microPayInput
  35. * @throws WxpayException
  36. * @return 返回查询接口的结果
  37. */
  38. public function pay($microPayInput)
  39. {
  40. //①、提交被扫支付
  41. $result = WxPayApi::micropay($this->config, $microPayInput, 5);
  42. //如果返回成功
  43. if(!array_key_exists("return_code", $result)
  44. || !array_key_exists("result_code", $result))
  45. {
  46. echo "接口调用失败,请确认是否输入是否有误!";
  47. throw new WxPayException("接口调用失败!");
  48. }
  49. //取订单号
  50. $out_trade_no = $microPayInput->GetOut_trade_no();
  51. //②、接口调用成功,明确返回调用失败
  52. if($result["return_code"] == "SUCCESS" &&
  53. $result["result_code"] == "FAIL" &&
  54. $result["err_code"] != "USERPAYING" &&
  55. $result["err_code"] != "SYSTEMERROR")
  56. {
  57. return false;
  58. }
  59. //③、确认支付是否成功
  60. $queryTimes = 10;
  61. while($queryTimes > 0)
  62. {
  63. $succResult = 0;
  64. $queryResult = $this->query($out_trade_no, $succResult);
  65. //如果需要等待1s后继续
  66. if($succResult == 2){
  67. sleep(2);
  68. continue;
  69. } else if($succResult == 1){//查询成功
  70. return $queryResult;
  71. } else {//订单交易失败
  72. break;
  73. }
  74. }
  75. //④、次确认失败,则撤销订单
  76. if(!$this->cancel($out_trade_no))
  77. {
  78. throw new WxpayException("撤销单失败!");
  79. }
  80. return false;
  81. }
  82. /**
  83. *
  84. * 查询订单情况
  85. * @param string $out_trade_no 商户订单号
  86. * @param int $succCode 查询订单结果
  87. * @return 0 订单不成功,1表示订单成功,2表示继续等待
  88. */
  89. public function query($out_trade_no, &$succCode)
  90. {
  91. $queryOrderInput = new WxPayOrderQuery();
  92. $queryOrderInput->SetOut_trade_no($out_trade_no);
  93. try{
  94. $result = WxPayApi::orderQuery($this->config, $queryOrderInput);
  95. } catch(Exception $e) {
  96. Log::ERROR(json_encode($e));
  97. }
  98. if($result["return_code"] == "SUCCESS"
  99. && $result["result_code"] == "SUCCESS")
  100. {
  101. //支付成功
  102. if($result["trade_state"] == "SUCCESS"){
  103. $succCode = 1;
  104. return $result;
  105. }
  106. //用户支付中
  107. else if($result["trade_state"] == "USERPAYING"){
  108. $succCode = 2;
  109. return false;
  110. }
  111. }
  112. //如果返回错误码为“此交易订单号不存在”则直接认定失败
  113. if($result["err_code"] == "ORDERNOTEXIST")
  114. {
  115. $succCode = 0;
  116. } else{
  117. //如果是系统错误,则后续继续
  118. $succCode = 2;
  119. }
  120. return false;
  121. }
  122. /**
  123. *
  124. * 撤销订单,如果失败会重复调用10次
  125. * @param string $out_trade_no
  126. * @param 调用深度 $depth
  127. */
  128. public function cancel($out_trade_no, $depth = 0)
  129. {
  130. try {
  131. if($depth > 10){
  132. return false;
  133. }
  134. $clostOrder = new WxPayReverse();
  135. $clostOrder->SetOut_trade_no($out_trade_no);
  136. $result = WxPayApi::reverse($this->config, $clostOrder);
  137. //接口调用失败
  138. if($result["return_code"] != "SUCCESS"){
  139. return false;
  140. }
  141. //如果结果为success且不需要重新调用撤销,则表示撤销成功
  142. if($result["result_code"] != "SUCCESS"
  143. && $result["recall"] == "N"){
  144. return true;
  145. } else if($result["recall"] == "Y") {
  146. return $this->cancel($out_trade_no, ++$depth);
  147. }
  148. } catch(Exception $e) {
  149. Log::ERROR(json_encode($e));
  150. }
  151. return false;
  152. }
  153. }