<?php /** * * example目录下为简单的支付样例,仅能用于搭建快速体验微信支付使用 * 样例的作用仅限于指导如何使用sdk,在安全上面仅做了简单处理, 复制使用样例代码时请慎重 * 请勿直接直接使用样例对外提供服务 * **/ require_once "../lib/WxPay.Api.php"; require_once "WxPay.Config.php"; /** * * 刷卡支付实现类 * 该类实现了一个刷卡支付的流程,流程如下: * 1、提交刷卡支付 * 2、根据返回结果决定是否需要查询订单,如果查询之后订单还未变则需要返回查询(一般反复查10次) * 3、如果反复查询10订单依然不变,则发起撤销订单 * 4、撤销订单需要循环撤销,一直撤销成功为止(注意循环次数,建议10次) * * 该类是微信支付提供的样例程序,商户可根据自己的需求修改,或者使用lib中的api自行开发,为了防止 * 查询时hold住后台php进程,商户查询和撤销逻辑可在前端调用 * * @author widy * */ class MicroPay { /** * * 提交刷卡支付,并且确认结果,接口比较慢 * @param WxPayMicroPay $microPayInput * @throws WxpayException * @return 返回查询接口的结果 */ public function pay($microPayInput) { //①、提交被扫支付 $config = new WxPayConfig(); $result = WxPayApi::micropay($config, $microPayInput, 5); //如果返回成功 if(!array_key_exists("return_code", $result) || !array_key_exists("result_code", $result)) { echo "接口调用失败,请确认是否输入是否有误!"; throw new WxPayException("接口调用失败!"); } //取订单号 $out_trade_no = $microPayInput->GetOut_trade_no(); //②、接口调用成功,明确返回调用失败 if($result["return_code"] == "SUCCESS" && $result["result_code"] == "FAIL" && $result["err_code"] != "USERPAYING" && $result["err_code"] != "SYSTEMERROR") { return false; } //③、确认支付是否成功 $queryTimes = 10; while($queryTimes > 0) { $succResult = 0; $queryResult = $this->query($out_trade_no, $succResult); //如果需要等待1s后继续 if($succResult == 2){ sleep(2); continue; } else if($succResult == 1){//查询成功 return $queryResult; } else {//订单交易失败 break; } } //④、次确认失败,则撤销订单 if(!$this->cancel($out_trade_no)) { throw new WxpayException("撤销单失败!"); } return false; } /** * * 查询订单情况 * @param string $out_trade_no 商户订单号 * @param int $succCode 查询订单结果 * @return 0 订单不成功,1表示订单成功,2表示继续等待 */ public function query($out_trade_no, &$succCode) { $queryOrderInput = new WxPayOrderQuery(); $queryOrderInput->SetOut_trade_no($out_trade_no); $config = new WxPayConfig(); try{ $result = WxPayApi::orderQuery($config, $queryOrderInput); } catch(Exception $e) { Log::ERROR(json_encode($e)); } if($result["return_code"] == "SUCCESS" && $result["result_code"] == "SUCCESS") { //支付成功 if($result["trade_state"] == "SUCCESS"){ $succCode = 1; return $result; } //用户支付中 else if($result["trade_state"] == "USERPAYING"){ $succCode = 2; return false; } } //如果返回错误码为“此交易订单号不存在”则直接认定失败 if($result["err_code"] == "ORDERNOTEXIST") { $succCode = 0; } else{ //如果是系统错误,则后续继续 $succCode = 2; } return false; } /** * * 撤销订单,如果失败会重复调用10次 * @param string $out_trade_no * @param 调用深度 $depth */ public function cancel($out_trade_no, $depth = 0) { try { if($depth > 10){ return false; } $clostOrder = new WxPayReverse(); $clostOrder->SetOut_trade_no($out_trade_no); $config = new WxPayConfig(); $result = WxPayApi::reverse($config, $clostOrder); //接口调用失败 if($result["return_code"] != "SUCCESS"){ return false; } //如果结果为success且不需要重新调用撤销,则表示撤销成功 if($result["result_code"] != "SUCCESS" && $result["recall"] == "N"){ return true; } else if($result["recall"] == "Y") { return $this->cancel($out_trade_no, ++$depth); } } catch(Exception $e) { Log::ERROR(json_encode($e)); } return false; } }