关于被微信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

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

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

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

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,除了自己

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

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

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

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

第二步: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,请根据你自己的框架来处理这些代码了。

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

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

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

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

请在评论中指出()or发我邮箱[email protected]。

时间: 2024-08-11 07:41:00

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

微信APP支付的坑 - errorcode=-1

关于微信App支付errorcode=-1的原因网上的大多数的说明和微信的官方文档差不多,不外乎app签名包名对应问题,这些原因都很好排除. 但最要命的就是下图标3 其他异常! 这是[调起微信支付接口]要传递的参数以及返回结果说明,其中标1和标2很有意思,有意思在哪呢?就是微信在这里告诉你怎么生成对应的数据,但如果真实环境中你就这么用了,那返回结果肯定是-1. 那这两个参数我们要用什么呢? 答案是[调起微信支付]前的上一步[统一下单]中签名用到的随机字符串和时间戳,但可惜的是文档中并没有明确说明

.net 微信APP支付接口的开发流程以及坑(转)

流程 申请APP的微信支付 申请成功之后得到APPID 商户号 以及自己设置商户号的支付密码 这时就可以开发接口了 微信APP支付API:https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=9_1 微信APP开发SDK:https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=11_1 接口开发 1.就说三个接口吧主要这三个接口是必须用的(如图),使用的方法SDK里面应该

微信app支付android客户端以及.net服务端实现

由于公司运营需要,需要在客户端(android/ios)增加微信以及支付宝支付,在调用微信app支付时遇到一些问题,也算是一些踩过的坑,记录下来 ,希望能对.net开发者服务端网站更快的集成微信app支付. 1.开发所需资料:微信开放平台应用的appid以及appsecert,商户平台的商户号以及api安全里面里面设置的key,详见 微信支付账户相关信息; 2.微信开发者平台完善应用平台的相关信息,android应用签名必须用打包签名过的发布版本apk(这一步很重用),包名必须一致,可以用微信提

微信App支付(JAVA端)

low话不多说,直接上代码! 红色框框是核心jar包! 黑色框框是获取客户端的IP地址工具类! 紫色框框是微信支付的流程代码! 蓝色框框是订单实体类! 由于小黑我技术不咋地,所以以下代码仅供参考,copy过去后是跑不起来的,不过可以经过改动代码使其跑起来,反正思路是这样的! (代码处如有发现错误的自行矫正修改,例如没有对事务控制,逻辑有误等!! 还有,大神看也,不喜勿喷) 此处代码就是用来调用微信支付SDK的,然后SDK会返回几个参数,这几个参数就是给APP端(IOS/安卓)调起支付的参数!! 

.Net后台实现微信APP支付

上一节分享了微信小程序支付的后台,这一节来分享一下微信APP支付的后台.微信APP支付和微信小程序差别不大,微信APP支付后台不需要微信登录凭证.后台下单时交易类型(trade_type)不再是"JSAPI",而是"APP".商户后台传递给支付端的下单参数也有所不同.由于微信小程序支付和APP支付使用的APPID不同,索性直接写了两套支付,不再在代码里区分究竟该使用小程序支付的配置参数还是APP支付的参数. 官方是这样介绍的 具体实现: 新建AppPayConfig

微信app支付返回-1的问题

我也是被坑就当留个纪念 前两天查了各种关于微信app支付返回-1的都是ERR_COMM 问题然后各种 验证最后还是误解 第三天去验证了一下微信开放平台发现了问题 appid 不在同一个开放平台 项目之前要做微信登 录上了一个开放平台,后来要做支付,负责支付申请的又单独开了一个开放平台在里面又重新 添加了应用,重新生成了appid ,项目一直是用的第一个开放平台的appid,所以没对上,更换 到申请了支付的开放平台上的appid 问题解决. 原文地址:https://www.cnblogs.com

微信app支付 ci框架做的

/**     * 组合微信app支付  获得prepayid     * @param int $order_num     */    private function _wxpay_request($order_num = 0)    {                //判断订单编号必须是数组并且不为0        check_order_num($order_num); //引入微信支付类            libraries_include("wxpay/", &qu

微信app支付php开发

前几天做微信app支付,遇到了支付失败的问题,app进行支付的时候,返回-1.查了好多资料,没找到.后来经过排查,是因为签名的问题.不多说,直接上代码.服务器端的demo版本是v3. require(dirname(dirname(__FILE__))."/lib/WxPay.Api.php");            //生成订单            $out_trade_no = trim($_POST['orderid']);            //总金额(1表示1分钱) 

微信app支付(android端+java后台)

本文讲解使用微信支付接口完成在android开发的原生态app中完成微信支付功能, 文章具体讲解了前端android如何集成微信支付功能以及后台如何组装前端需要支付信息, 话不多话, 具体看文章内容吧00:00 / 07:03正常 本实例项目运行条件: 开发环境: [Android Studio] 到微信开放平台注册帐号并且创建移动应用 https://open.weixin.qq.com/cgi-bin/frame?t=home/app_tmpl&lang=zh_CN Column 1 Col