微信小程序支付全問題解決

這幾天在作小程序的支付,沒有用官方的SDK,這裏就純用官方的文檔搞一發。php

* 注做者使用的PHP,不過支付流程都是這樣 html

開發前必讀

主要流程

  • 小程序前端發送求參請求
  • 接受請求封裝 「統一下單」 獲取package
  • 小程序接受 「統一下單」 獲取的package值帶入wx.requestPayment發起支付請求

準備工具

  • 申請小程序微信支付
  • 拿到小程序微信支付的商戶號及設置祕鑰

注意:小程序就只須要這兩步,若是是web的話還須要設置支付目錄受權域名,文檔裏面也有寫的:https://pay.weixin.qq.com/wik... 前端


統一下單

官方文檔: https://pay.weixin.qq.com/wik...
/**
 * 統一訂單
 */
public function unifiedorder(){
    // 如下配置是必填項,若有其它需求請自行配置
    $config = array(
        'appid'         =>    'xxxxxxx',//這裏是小程序appid
        'mch_id'        =>    'xxxxxxx',//商戶ID
        'nonce_str'     =>    $this->getNonceStr(),//隨機字符串
        'body'          =>    '這裏是測試 - 測試',//請按照文檔要求填寫合格名稱
        'out_trade_no'  =>    time().$this->getNonceStr(2),//流水單號
        'total_fee'     =>    '20',//金額,分爲單位,這裏是0.2元
        'spbill_create_ip' => '123.123.123.123',//當前IP
        'notify_url'    =>    'http://xxxx.com',//請恕我愚昧,我沒搞懂他有什麼用
        'trade_type'    =>    'JSAPI',//必須填寫JSAPI
        'openid'        =>    'xxxxxxxx'//當前用戶的openid,在trade_type=JSAPI的時候,此項就變成必填項了
    );
    $config['sign'] = $this->getSignPay($config);
    $xmlData = $this->ToXml($config);//轉成xml數據
    $postData = $this->http_post($xmlData);
    $arrayData = $this->FromXml($postData);
    if($arrayData['return_code'] == 'SUCCESS' || $arrayData['result_code'] == 'SUCCESS'){
        return $arrayData['prepay_id'];//重點來了:走了這麼多路,就爲了這個值。到這一步就證實成功一多半了。
    }else{
        return $arrayData;//返回錯誤
    }
}

/**
 * 獲取簽名
 */
public function getSignPay($config){
    $key = 'xxxxxxx';//商戶祕鑰,就是本身生成的32位密碼
    $strA = 'appid='.$config['appid'].'&body='.$config['body'].'&mch_id='.$config['mch_id'].'&nonce_str='.$config['nonce_str'].'&notify_url='.$config['notify_url'].'&spbill_create_ip'.$config['spbill_create_ip'].'&total_fee='.$config['total_fee'].'&trade_type='.$config['trade_type'];//ASCII 字典序
    $strB = $strA.'&key='.$key;
    $sign = strtoupper(md5($strB));//大寫MD5
    return $sign;
}

/**
 * 隨機字符串 32位
 */
public function getNonceStr($length = 32){
    $chars = "abcdefghijklmnopqrstuvwxyz0123456789";  
    $str ="";
    for ( $i = 0; $i < $length; $i++ )  {  
        $str .= substr($chars, mt_rand(0, strlen($chars)-1), 1);  
    } 
    return $str;
}

/**
 * array轉XML
 */
public function ToXml($data){
    if(!is_array($data) || count($data) <= 0){
        throw new WxPayException("數組數據異常!");
    }
    $xml = "<xml>";
    foreach ($data as $key=>$val){
        $xml.="<".$key.">".$val."</".$key.">";
    }
    $xml.="</xml>";
    return $xml; 
}

/**
 * xml轉array
 */
public function FromXml($xml){    
    if(!$xml){
        throw new WxPayException("xml數據異常!");
    }
    libxml_disable_entity_loader(true);
    $this->values = json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);
    return $this->values;
}


/**
 * post 請求
 */
public function http_post($url,$param,$post_file=false){
    $oCurl = curl_init();
    if(stripos($url,"https://")!==FALSE){
        curl_setopt($oCurl, CURLOPT_SSL_VERIFYPEER, FALSE);
        curl_setopt($oCurl, CURLOPT_SSL_VERIFYHOST, false);
        curl_setopt($oCurl, CURLOPT_SSLVERSION, 1); //CURL_SSLVERSION_TLSv1
    }
    if (PHP_VERSION_ID >= 50500 && class_exists('\CURLFile')) {
        $is_curlFile = true;
    } else {
        $is_curlFile = false;
        if (defined('CURLOPT_SAFE_UPLOAD')) {
            curl_setopt($oCurl, CURLOPT_SAFE_UPLOAD, false);
        }
    }
    if (is_string($param)) {
        $strPOST = $param;
    }elseif($post_file) {
        if($is_curlFile) {
            foreach ($param as $key => $val) {
                if (substr($val, 0, 1) == '@') {

                }
            }
        }
        $strPOST = $param;
    } else {
        $aPOST = array();
        foreach($param as $key=>$val){
            $aPOST[] = $key."=".urlencode($val);
        }
        $strPOST =  join("&", $aPOST);
    }
    curl_setopt($oCurl, CURLOPT_URL, $url);
    curl_setopt($oCurl, CURLOPT_RETURNTRANSFER, 1 );
    curl_setopt($oCurl, CURLOPT_POST,true);
    curl_setopt($oCurl, CURLOPT_POSTFIELDS,$strPOST);
    $sContent = curl_exec($oCurl);
    $aStatus = curl_getinfo($oCurl);
    curl_close($oCurl);
    if(intval($aStatus["http_code"])==200){
        return $sContent;
    }else{
        return false;
    }
}

好了如今已經獲取到了 prepay_id 的值,咱們的統一下單就算完成了,其實我更樂意叫他數據封裝web


小程序微信支付

官方文檔: https://developers.weixin.qq....

先來一個插曲,首先咱們小程序的前端須要去觸發pay,實現的功能確定是要點擊小程序的一個觸發,而後才能支付對吧,json

pay:function(e){
    //這裏面使用post去請求。而後經過我接下來要寫的API支付代碼獲取小程序支付參數
    success:function(res){
         wx.requestPayment({
             'timeStamp':toString(res.timeStamp),//這裏轉字符串,這裏被坑過,不轉的話可能會出現total_fee爲空
             'nonceStr':toString(res.nonceStr),
             'package':toString(res.package),
             'signType':'MD5',
             'paySign':toString(res.paySign),
             success:function(res){
                 console.log(res);//這裏能夠跳轉到帶參地址
             },
             fail:function(res){
                 console.info('支付失敗',res);
             },
             complete:function(){
                 console.info('支付觸發回調',res);
             }
         })  
    }
       
}

api支付

也就是上面小程序代碼的後端請求地址小程序

/**
 * api組裝數據
 */
public function payApiBlack(){
    $appid = 'xxxxxx';//小程序appid,上面有重複,不過這樣比較直觀
    $timeStamp = time();
    $nonceStr = $this->getNonceStr();//這是調用統一下單裏面的方法,爲了直觀,我把這些代碼都寫在了一個類裏
    $package = 'prepay_id='.$this->unifiedorder();
    $signType = 'MD5';
    $key = 'xxxxxx';//這裏是商戶祕鑰,32位,同上面也有
    $strA = 'appId='.$appid.'&nonceStr='.$nonceStr.'package='.$package.'&= signType='.$signType.'&timeStamp='.$timeStamp.'&key='.$key;
    $paySign = strtoupper(md5($strA));
    $data = array(
        'appid'=>$appid,
        'timeStamp'=>$timeStamp,
        'nonceStr'=>$nonceStr,
        'package'=>$package,
        'signType'=>$signType
    );
    return $data;//返回給小程序
}

以上就是所有代碼,還有小程序的支付回調沒有什麼信息,因此,個人思路判斷success後進行跳轉帶參後端

//此代碼爲wx.requestPayment success,部分代碼省略
//res 回調參數包括用戶uid及其餘重要傳遞
success:function(res){
    wx.redirect({
        url:'pages/pay/done?uid='+res.uid
    })
}

固然那個統一下單的 notify_url 好像與回調有關,至於怎麼用,試了幾回回調的CURD都沒反應,因此有空再研究啦。api

以上代碼僅做爲支付流程解釋,因此真正要用到項目上,仍是去套官方的SDK吧,畢竟涉及到錢嘛數組

相關文章
相關標籤/搜索