当前位置:   article > 正文

微信支付结果通知,回调的策略

微信支付回调完成 服务器通知

支付完成后,微信会把相关支付结果和用户信息发送给商户,商户需要接收处理,并返回应答。

对后台通知交互时,如果微信收到商户的应答不是成功或超时,微信认为通知失败,微信会通过一定的策略定期重新发起通知,尽可能提高通知的成功率,但微信不保证通知最终能成功。 (通知频率为15/15/30/180/1800/1800/1800/1800/3600,单位:秒)

注意:同样的通知可能会多次发送给商户系统。商户系统必须能够正确处理重复的通知。
推荐的做法是,当收到通知进行处理时,首先检查对应业务数据的状态,判断该通知是否已经处理过,如果没有处理过再进行处理,如果处理过直接返回结果成功。在对业务数据进行状态检查和处理之前,要采用数据锁进行并发控制,以避免函数重入造成的数据混乱。

特别提醒:商户系统对于支付结果通知的内容一定要做签名验证,并校验返回的订单金额是否与商户侧的订单金额一致,防止数据泄漏导致出现“假通知”,造成资金损失。

  1. $str='<xml><return_code><![CDATA[SUCCESS]]></return_code><return_msg><![CDATA[OK]]></return_msg></xml>';
  2. exit($str);
  1. //微信支付回调
  2. public function order_notice(){
  3. $xml = $GLOBALS['HTTP_RAW_POST_DATA'];
  4. //将服务器返回的XML数据转化为数组
  5. $data = $this->FromXml($xml);
  6. // 保存微信服务器返回的签名sign
  7. $data_sign = $data['sign'];
  8. // sign不参与签名算法
  9. unset($data['sign']);
  10. //$sign = self::makeSign($data);
  11. $sign = $this->makeSign($data);
  12. Clog::setLog($data,'order_notice');
  13. // 判断签名是否正确 判断支付状态
  14. if ( ($sign===$data_sign) && ($data['return_code']=='SUCCESS') && ($data['result_code']=='SUCCESS') ) {
  15. //获取服务器返回的数据
  16. $order_num = $data['out_trade_no']; //订单单号
  17. $openid = $data['openid']; //付款人openID
  18. $total_fee = $data['total_fee']; //付款金额
  19. $transaction_id = $data['transaction_id']; //微信支付流水号
  20. $res = $this->order_notice_data_deal($order_num,$openid,$total_fee,$transaction_id);
  21. if (!$res) {
  22. $result = -2;
  23. } else {
  24. $result = 0;
  25. }
  26. }else{
  27. $result = -1;
  28. }
  29. // 返回状态给微信服务器
  30. if ($result == 0) { // 成功之后不会再回调
  31. $str='<xml><return_code><![CDATA[SUCCESS]]></return_code><return_msg><![CDATA[OK]]></return_msg></xml>';
  32. } elseif ($result == -1){ // 失败后会继续发送几次回调
  33. $str='<xml><return_code><![CDATA[FAIL]]></return_code><return_msg><![CDATA[签名失败]]></return_msg></xml>';
  34. } elseif ($result == -2) { // 失败后会继续发送几次回调
  35. $str='<xml><return_code><![CDATA[FAIL]]></return_code><return_msg><![CDATA[更改状态失败]]></return_msg></xml>';
  36. }
  37. Clog::setLog($result,'order_notice_result');
  38. exit($str);
  39. }

附上微信支付助手函数

  1. public function ToXml($array){
  2. if(!is_array($array)|| count($array) <= 0){
  3. return ;
  4. }
  5. $xml = '<xml version="1.0">';
  6. foreach ($array as $key=>$val){
  7. if (is_numeric($val)){
  8. $xml.="<".$key.">".$val."</".$key.">";
  9. }else{
  10. $xml.="<".$key."><![CDATA[".$val."]]></".$key.">";
  11. }
  12. }
  13. $xml.="</xml>";
  14. return $xml;
  15. }
  16. public function FromXml($xml){
  17. if(!$xml){
  18. // 人工抛出错误
  19. throw new Exception("xml数据异常!");
  20. }
  21. //将XML转为array
  22. //禁止引用外部xml实体
  23. libxml_disable_entity_loader(true);
  24. $this->values = json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);
  25. return $this->values;
  26. }
  27. public function MakeSign($data)
  28. {
  29. //签名步骤一:按字典序排序参数
  30. ksort($data);
  31. $string = $this->ToUrlParams($data);
  32. //签名步骤二:在string后加入KEY
  33. $string = $string . "&key=".C('WEIXIN_PAY_KEY');
  34. //签名步骤三:MD5加密
  35. $string = md5($string);
  36. //签名步骤四:所有字符转为大写
  37. $result = strtoupper($string);
  38. return $result;
  39. }
  40. public function ToUrlParams($array)
  41. {
  42. $buff = "";
  43. foreach ($array as $k => $v)
  44. {
  45. if($k != "sign" && $v != "" && !is_array($v)){
  46. $buff .= $k . "=" . $v . "&";
  47. }
  48. }
  49. $buff = trim($buff, "&");
  50. return $buff;
  51. }
  52. // createNonceStr
  53. public function createNonceStr($length = 16) {
  54. $chars = "abcdefghijklmnopqrstuvwxyz0123456789";
  55. $str ="";
  56. for ( $i = 0; $i < $length; $i++ ) {
  57. $str .= substr($chars, mt_rand(0, strlen($chars)-1), 1);
  58. }
  59. return $str;
  60. }

转载于:https://www.cnblogs.com/jiqing9006/p/10224600.html

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/很楠不爱3/article/detail/132274
推荐阅读
相关标签
  

闽ICP备14008679号