关于被微信App支付坑的过程

最近因为项目要接入微信App支付(没做过,不了解),然后就开网上一番狂搜+看官方文档,那是一个乱七八糟。

微信在2014年9月10号更新出了v3的版本,结果我竟然拿着v2版本在那里调试=》被坑。

回归到正题:希望接下来的朋友少走一些弯路。

微信总共有三个平台:

公众号平台:https://mp.weixin.qq.com/

商户平台:https://pay.weixin.qq.com

开放平台:https://open.weixin.qq.com/

这里所提到的微信App支付则是用到开放平台。

支付帐号申请下来后,收到财付通的一封邮件

效果如下: 
https://pay.weixin.qq.com/wiki/doc/api/app.php?chapter=3_1

wKiom1ZWZJ7wRk5bAADTW2_jvd4188.jpg650) this.width=650;" src="https://images.winkp.com/imgs/winkp/_winkp/2023/08/2449947648103825545.jpg" title="wechat-pay.jpg" class="aligncenter">

基本上,app支付的流程就是 
1、统一下单(由自己的服务器处理) =》得到预支付订单
2、发起支付(客户端) 
3、支付成功回调(服务器端) 
https://pay.weixin.qq.com/wiki/doc/api/app.php?chapter=9_1

wKiom1ZWZZSiR4aLAAEoPETKgxw320.png650) this.width=650;" src="https://images.winkp.com/imgs/winkp/_winkp/2023/08/3060620666690732843.png" title="we.png" class="aligncenter">

这里就说一下统一下单和支付回调两个后台需要做的操作:(这里只说如何接入,中间碰到的问题解决,会在此文章一并截图或文字提到)

调试了好几天,总算调通让前端能调起支付界面了,整理了一下代码如下:

WechatAppPay.php:

<?php/** * convert xml string to php array - useful to get a serializable value * * @param string $xmlstr * @return array * @author Adrien aka Gaarf */class WxPayHelper{    /*    配置参数    */    var $config = array(        ‘appid‘ => "wxe7380efe00000000",    /*微信开放平台上的应用id*/        ‘mch_id‘ => "1200000000",   /*微信申请成功之后邮件中的商户id*/        ‘api_key‘ => "3a9ce8d8a17cc85bcbaf32544e9cc781",    /*在微信商户平台上自己设定的api密钥 32位*/        ‘notify_url‘ => ‘http://www.wechat.cn/app/commerce_wechat/notify.json‘ /*自定义的回调程序地址id*/    );    public function  __construct() {    }    //获取预支付订单    public function getPrePayOrder($body, $out_trade_no, $total_fee){        $url = "https://api.mch.weixin.qq.com/pay/unifiedorder";        $notify_url = $this->config["notify_url"];        $onoce_str = $this->getRandChar(32);        $data["appid"] = $this->config["appid"];        $data["body"] = $body;        $data["mch_id"] = $this->config[‘mch_id‘];        $data["nonce_str"] = $onoce_str;        $data["notify_url"] = $notify_url;        $data["out_trade_no"] = $out_trade_no;        $data["spbill_create_ip"] = $this->get_client_ip();        $data["total_fee"] = $total_fee;        $data["trade_type"] = "APP";        $s = $this->getSign($data, false);        $data["sign"] = $s;        $xml = $this->arrayToXml($data);        $response = $this->postXmlCurl($xml, $url);        //将微信返回的结果xml转成数组        return $this->xmlstr_to_array($response);    }    //执行第二次签名,才能返回给客户端使用    public function getOrder($prepayId){        $data["appid"] = $this->config["appid"];        $data["noncestr"] = $this->getRandChar(32);;        $data["package"] = "Sign=WXPay";        $data["partnerid"] = $this->config[‘mch_id‘];        $data["prepayid"] = $prepayId;        $data["timestamp"] = time();        $s = $this->getSign($data, false);        $data["sign"] = $s;        return $data;    }    /*        生成签名    */    function getSign($Obj)    {        foreach ($Obj as $k => $v)        {            $Parameters[strtolower($k)] = $v;        }        //签名步骤一:按字典序排序参数        ksort($Parameters);        $String = $this->formatBizQueryParaMap($Parameters, false);        //echo "【string】 =".$String."</br>";        //签名步骤二:在string后加入KEY        $String = $String."&key=".$this->config[‘api_key‘];        //echo "<textarea style=‘width: 50%; height: 150px;‘>$String</textarea> <br />";        //签名步骤三:MD5加密        $result_ = strtoupper(md5($String));        return $result_;    }    //获取指定长度的随机字符串    function getRandChar($length){       $str = null;       $strPol = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyz";       $max = strlen($strPol)-1;       for($i=0;$i<$length;$i++){        $str.=$strPol[rand(0,$max)];//rand($min,$max)生成介于min和max两个数之间的一个随机整数       }       return $str;    }    //数组转xml    function arrayToXml($arr)    {        $xml = "<xml>";        foreach ($arr as $key=>$val)        {             if (is_numeric($val))             {                $xml.="<".$key.">".$val."</".$key.">";              }             else                $xml.="<".$key."><![CDATA[".$val."]]></".$key.">";          }        $xml.="</xml>";        return $xml;     }    //post https请求,CURLOPT_POSTFIELDS xml格式    function postXmlCurl($xml,$url,$second=30)    {               //初始化curl                $ch = curl_init();        //超时时间        curl_setopt($ch,CURLOPT_TIMEOUT,$second);        //这里设置代理,如果有的话        //curl_setopt($ch,CURLOPT_PROXY, ‘8.8.8.8‘);        //curl_setopt($ch,CURLOPT_PROXYPORT, 8080);        curl_setopt($ch,CURLOPT_URL, $url);        curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,FALSE);        curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,FALSE);        //设置header        curl_setopt($ch, CURLOPT_HEADER, FALSE);        //要求结果为字符串且输出到屏幕上        curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);        //post提交方式        curl_setopt($ch, CURLOPT_POST, TRUE);        curl_setopt($ch, CURLOPT_POSTFIELDS, $xml);        //运行curl        $data = curl_exec($ch);        //返回结果        if($data)        {            curl_close($ch);            return $data;        }        else         {             $error = curl_errno($ch);            echo "curl出错,错误码:$error"."<br>";            echo "<a href=‘http://curl.haxx.se/libcurl/c/libcurl-errors.html‘>错误原因查询</a></br>";            curl_close($ch);            return false;        }    }    /*        获取当前服务器的IP    */    function get_client_ip()    {        if ($_SERVER[‘REMOTE_ADDR‘]) {        $cip = $_SERVER[‘REMOTE_ADDR‘];        } elseif (getenv("REMOTE_ADDR")) {        $cip = getenv("REMOTE_ADDR");        } elseif (getenv("HTTP_CLIENT_IP")) {        $cip = getenv("HTTP_CLIENT_IP");        } else {        $cip = "unknown";        }        return $cip;    }    //将数组转成uri字符串    function formatBizQueryParaMap($paraMap, $urlencode)    {        $buff = "";        ksort($paraMap);        foreach ($paraMap as $k => $v)        {            if($urlencode)            {               $v = urlencode($v);            }            $buff .= strtolower($k) . "=" . $v . "&";        }        $reqPar;        if (strlen($buff) > 0)         {            $reqPar = substr($buff, 0, strlen($buff)-1);        }        return $reqPar;    }    /**    xml转成数组    */    function xmlstr_to_array($xmlstr) {      $doc = new DOMDocument();      $doc->loadXML($xmlstr);      return $this->domnode_to_array($doc->documentElement);    }    function domnode_to_array($node) {      $output = array();      switch ($node->nodeType) {       case XML_CDATA_SECTION_NODE:       case XML_TEXT_NODE:        $output = trim($node->textContent);       break;       case XML_ELEMENT_NODE:        for ($i=0, $m=$node->childNodes->length; $i<$m; $i++) {         $child = $node->childNodes->item($i);         $v = $this->domnode_to_array($child);         if(isset($child->tagName)) {           $t = $child->tagName;           if(!isset($output[$t])) {            $output[$t] = array();           }           $output[$t][] = $v;         }         elseif($v) {          $output = (string) $v;         }        }        if(is_array($output)) {         if($node->attributes->length) {          $a = array();          foreach($node->attributes as $attrName => $attrNode) {           $a[$attrName] = (string) $attrNode->value;          }          $output[‘@attributes‘] = $a;         }         foreach ($output as $t => $v) {          if(is_array($v) && count($v)==1 && $t!=‘@attributes‘) {           $output[$t] = $v[0];          }         }        }       break;      }      return $output;    }}?>

建议参考的朋友直接把WechatAppPay.php拷贝放到你的项目中做调用即可。

 注意的地方:

post必须支持https,且参数格式必须是xml 

sign签名的参数包括所有$data,除了自己 

wKiom1ZWaTrSEQKQAATAdekEb6s502.jpg650) this.width=650;" src="https://images.winkp.com/imgs/winkp/_winkp/2023/08/4520796567607982199.jpg" title="weichat.jpg" class="aligncenter">

$data[“spbill_create_ip”]不能随便设定一个ip地址,不要以为调试方便随便设定,结果返回签名错误坑你没商量。一定要是程序执行时所在的服务器ip地址,所以使用get_client_ip()获取就好。

api_key是需要自己进入商户平台设定的,邮件不会发给你哦 

wKioL1ZWamSBw3unAAEKesraZhw096.jpg650) this.width=650;" src="https://images.winkp.com/imgs/winkp/_winkp/2023/08/5955853300003464547.jpg" title="wechat-appkey.jpg" class="aligncenter">

使用在线随机程序产生32个字符就好了

相当重要的是:返回给各户端发起支付时,还要进行二次签名 $WxPayHelper->getOrder

wKioL1ZWa6vgWQKIAAFFbFbB4eo600.png650) this.width=650;" src="https://images.winkp.com/imgs/winkp/_winkp/2023/08/6290321651074001070.png" title="prepay.png" class="aligncenter">第二步:App端支付完成后该调用第一步请求预支付订单时填写的notify_url了。

因为不知道以哪种方式验证回调结果是否为微信的,所以我就没做签名验证的处理

直接拿到结果去做订单的处理了《这里对回调不做过多的解释只说下大体,有疑问的朋友可以参考下http://wyong.blog.51cto.com/1115465/1692322这位朋友的博客,分析的很透彻》

分享一下简单的处理代码:

function services_commerce_wechat_wx_notify(){ require_once ‘WechatAppPay.php‘; $postStr = $GLOBALS[‘HTTP_RAW_POST_DATA‘];//这里拿到微信返回的数据结果 $WechatAppPay = new WxPayHelper(); $getData = $WechatAppPay->xmlstr_to_array($postStr);//为了方便我就直接把结果转成数组,看个人爱好了     //回调通知服务器数据正常$getData[‘result_code‘] == ‘SUCCESS‘    if(!empty($getData[‘return_code‘]) && $getData[‘return_code‘] == ‘SUCCESS‘){       $order_no = $getData[‘out_trade_no‘];     $arr_str=explode("Z",$order_no);       if(!empty($arr_str)){           $order_no1 = $arr_str[1];   $order = commerce_order_load($order_no1); global $user; if ($user->uid == 0) {   $user = user_load($order->uid);   } commerce_quick_wechat_notify_submit1($order, $getData);     }   else {   //error, no order id process   } }}
function commerce_quick_wechat_notify_submit1($order, $notify) {  $flag=‘‘;   if($order->status==‘checkout_complete‘||$order->status==‘completed‘){     echo ‘The bill is finished.‘;  }    // Handle trade types of cases.    switch ($notify[‘return_code‘]) {      // Transaction successful.      case ‘SUCCESS‘:          $flag=commerce_order_status_update($order, ‘checkout_complete‘);        break;    }    if($flag){ commerce_checkout_complete($order); //header("Content-type: text/xml; charset=utf-8"); require_once ‘WechatAppPay.php‘; $WechatAppPay = new WxPayHelper(); $return = array( ‘return_code‘ => ‘SUCCESS‘, ‘return_ok‘ => ‘OK‘, ); $xml = $WechatAppPay->arrayToXml($return); return $xml;    } else{    echo ‘fail‘;     die();  }     }

因为我们用的框架是drupal,请根据你自己的框架来处理这些代码了。

碰到一个巨坑的问题(至今不知什么原因)网上也几乎没搜到结果。

就是回调结果转换成数组之后是这样的:

wKioL1ZWcZSBxolPAACOD-BGmyA001.png650) this.width=650;" src="https://images.winkp.com/imgs/winkp/_winkp/2023/08/2069777478791198949.png" title="return_code.png" class="aligncenter">

但我单独打印print_r($getData[‘result_code‘])这两个字段的时候总是给我返回1,请教朋友也说微信官方也没这个结果的,无法解释。折腾了两天找原因。最后手贱似的清理了一下缓存。就莫名其妙的得到了success了。

文章会有不足的地方,请多多见谅指教,只为分享出来,让后续做的朋友少绕一些圈子。

请在评论中指出()or发我邮箱754634469@qq.com。

本文出自 “为了以后” 博客,转载请与作者联系!

相关文章