thinkphp3.2整和微信公众号支付详细流程 demo

公众号支付是指在微信app中访问的页面通过js直接调起微信支付;

首先第一个步骤登录微信公众平台然后

1.设置域名(设置授权域名和支付域名)

①设置网页授权域名并且设置白名单(添加你自己服务器公网ip)如下图所示

         thinkphp3.2整和微信公众号支付详细流程 demo_第1张图片



②设置支付域名,去微信商户后台,产品中心的 开发配置中设置支付授权目录;把域名改为自己的;如下图所示

thinkphp3.2整和微信公众号支付详细流程 demo_第2张图片

二:导入sdk

/ThinkPHP/Library/Vendor/Weixinpay

该sdk是PHP白均瑶博客码云中的开源项目  点击下载sdk

Weixinpay.php



error_reporting(E_ALL);
ini_set('display_errors', '1');


// 定义时区
ini_set('date.timezone','Asia/Shanghai');


class Weixinpay {
    // 定义配置项
    private $config=array(
        'APPID'              => '', // 微信支付APPID
        'MCHID'              => '', // 微信支付MCHID 商户收款账号
        'KEY'                => '', // 微信支付KEY
        'APPSECRET'          => '',  //公众帐号secert
        'NOTIFY_URL'         => 'http://baijunyao.com/Api/WeixPay/notify/order_number/', // 接收支付状态的连接  改成自己的域名
        );


    // 构造函数
    public function __construct(){
        // 如果是在thinkphp中 那么需要补全/Application/Common/Conf/config.php中的配置
        // 如果不是在thinkphp框架中使用;那么注释掉下面一行代码;直接补全 private $config 即可
        $this->config=C('WEIXINPAY_CONFIG');
    }


    /**
     * 统一下单
     * @param  array $order 订单 必须包含支付所需要的参数 body(产品描述)、total_fee(订单金额)、out_trade_no(订单号)、product_id(产品id)、trade_type(类型:JSAPI,NATIVE,APP)
     */
    public function unifiedOrder($order){
        // 获取配置项
        $weixinpay_config=$this->config;
        $config=array(
            'appid'=>$weixinpay_config['APPID'],
            'mch_id'=>$weixinpay_config['MCHID'],
            'nonce_str'=>'test',
            'spbill_create_ip'=>'192.168.0.1',
            'notify_url'=>$weixinpay_config['NOTIFY_URL']
            );
        // 合并配置数据和订单数据
        $data=array_merge($order,$config);
        // 生成签名
        $sign=$this->makeSign($data);
        $data['sign']=$sign;
        $xml=$this->toXml($data);
        $url = 'https://api.mch.weixin.qq.com/pay/unifiedorder';//接收xml数据的文件
        $header[] = "Content-type: text/xml";//定义content-type为xml,注意是数组
        $ch = curl_init ($url);
        curl_setopt($ch, CURLOPT_URL, $url);
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
        curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false); // 兼容本地没有指定curl.cainfo路径的错误
        curl_setopt($ch, CURLOPT_HTTPHEADER, $header);
        curl_setopt($ch, CURLOPT_POST, 1);
        curl_setopt($ch, CURLOPT_POSTFIELDS, $xml);
        $response = curl_exec($ch);
        if(curl_errno($ch)){
            // 显示报错信息;终止继续执行
            die(curl_error($ch));
        }
        curl_close($ch);
        $result=$this->toArray($response);
        // 显示错误信息
        if ($result['return_code']=='FAIL') {
            die($result['return_msg']);
        }
        $result['sign']=$sign;
        $result['nonce_str']='test';
        return $result;
    }




    /**
     * 验证
     * @return array 返回数组格式的notify数据
     */
    public function notify(){
        // 获取xml
        $xml=file_get_contents('php://input', 'r'); 
        // 转成php数组
        $data=$this->toArray($xml);
        // 保存原sign
        $data_sign=$data['sign'];
        // sign不参与签名
        unset($data['sign']);
        $sign=$this->makeSign($data);
        // 判断签名是否正确  判断支付状态
        if ($sign===$data_sign && $data['return_code']=='SUCCESS' && $data['result_code']=='SUCCESS') {
            $result=$data;
        }else{
            $result=false;
        }
        // 返回状态给微信服务器
        if ($result) {
            $str='';
        }else{
            $str='';
        }
        echo $str;
        return $result;
    }


    /**
     * 输出xml字符
     * @throws WxPayException
    **/
    public function toXml($data){
        if(!is_array($data) || count($data) <= 0){
            throw new WxPayException("数组数据异常!");
        }
        $xml = "";
        foreach ($data as $key=>$val){
            if (is_numeric($val)){
                $xml.="<".$key.">".$val."";
            }else{
                $xml.="<".$key.">";
            }
        }
        $xml.="
";
        return $xml; 
    }


    /**
     * 生成签名
     * @return 签名,本函数不覆盖sign成员变量,如要设置签名需要调用SetSign方法赋值
     */
    public function makeSign($data){
        // 去空
        $data=array_filter($data);
        //签名步骤一:按字典序排序参数
        ksort($data);
        $string_a=http_build_query($data);
        $string_a=urldecode($string_a);
        //签名步骤二:在string后加入KEY
        $config=$this->config;
        $string_sign_temp=$string_a."&key=".$config['KEY'];
        //签名步骤三:MD5加密
        $sign = md5($string_sign_temp);
        // 签名步骤四:所有字符转为大写
        $result=strtoupper($sign);
        return $result;
    }


    /**
     * 将xml转为array
     * @param  string $xml xml字符串
     * @return array       转换得到的数组
     */
    public function toArray($xml){   
        //禁止引用外部xml实体
        libxml_disable_entity_loader(true);
        $result= json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);        
        return $result;
    }


    /**
     * 获取jssdk需要用到的数据
     * @return array jssdk需要用到的数据
     */
    public function getParameters(){
        // 获取配置项
        $config=$this->config;
        // 如果没有get参数没有code;则重定向去获取openid;
        if (!isset($_GET['code'])) {
            // 获取订单号
            $out_trade_no=I('get.out_trade_no',1,'intval');
            // 返回的url
            $redirect_uri=U('Api/Weixinpay/pay','','',true);
            $redirect_uri=urlencode($redirect_uri);
            $url='https://open.weixin.qq.com/connect/oauth2/authorize?appid='.$config['APPID'].'&redirect_uri='.$redirect_uri.'&response_type=code&scope=snsapi_base&state='.$out_trade_no.'#wechat_redirect';
            redirect($url);
        }else{
            // 如果有code参数;则表示获取到openid
            $code=I('get.code');
            // 取出订单号
            $out_trade_no=I('get.state',0,'intval');
            // 组合获取prepay_id的url
            $url='https://api.weixin.qq.com/sns/oauth2/access_token?appid='.$config['APPID'].'&secret='.$config['APPSECRET'].'&code='.$code.'&grant_type=authorization_code';
            // curl获取prepay_id
            $result=$this->https_request($url);
            $result=json_decode($result,true);
            $openid=$result['openid'];
            // 订单数据  请根据订单号out_trade_no 从数据库中查出实际的body、total_fee、out_trade_no、product_id
            $order=array(
                'body'=>'test',// 商品描述(需要根据自己的业务修改)
                'total_fee'=>1,// 订单金额  以(分)为单位(需要根据自己的业务修改)
                'out_trade_no'=>$out_trade_no,// 订单号(需要根据自己的业务修改)
                'product_id'=>'1',// 商品id(需要根据自己的业务修改)
                'trade_type'=>'JSAPI',// JSAPI公众号支付
                'openid'=>$openid// 获取到的openid
            );
            // 统一下单 获取prepay_id
            $unified_order=$this->unifiedOrder($order);
            // 获取当前时间戳
            $time=time();
            // 组合jssdk需要用到的数据
            $data=array(
                'appId'=>$config['APPID'], //appid
                'timeStamp'=>strval($time), //时间戳
                'nonceStr'=>$unified_order['nonce_str'],// 随机字符串
                'package'=>'prepay_id='.$unified_order['prepay_id'],// 预支付交易会话标识
                'signType'=>'MD5'//加密方式
            );
            // 生成签名
            $data['paySign']=$this->makeSign($data);
            return $data;
        }
    }




  

需要注意函数中getParameters中的商品数据需要根据业务实际情况从数据库中获取;

三:配置项

/Application/Common/Conf/config.php
  1.     'WEIXINPAY_CONFIG'       => array(
  2.         'APPID'              => '', // 微信支付APPID
  3.         'MCHID'              => '', // 微信支付MCHID 商户收款账号
  4.         'KEY'                => '', // 微信支付KEY
  5.         'APPSECRET'          => '', // 公众帐号secert (公众号支付专用)
  6.         'NOTIFY_URL'         => 'http://baidexuan.com/Api/Weixinpay/notify', // 接收支付状态的连接
  7.         ),
复制代码
在微信公众平台和微信支付平台凑齐上面这些参数;

则需要在自己的公共函数中增加https_request
/Application/Common/Common/function.php

/**
 * 使用curl获取远程数据
 * @param  string $url url连接
 * @return string      获取到的数据
 */
function https_request($url, $data = null)
{
    $curl = curl_init();
    curl_setopt($curl, CURLOPT_URL, $url);
    curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, FALSE);
    curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, FALSE);
    if (!empty($data)){
        curl_setopt($curl, CURLOPT_POST, 1);
        curl_setopt($curl, CURLOPT_POSTFIELDS, $data);
    }
    curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
    $output = curl_exec($curl);
    curl_close($curl);
    return $output;
}

四:支付方法

/Application/Api/Controller/WeixinpayController.class.php

  1. public function pay(){
  2.     // 导入微信支付sdk
  3.     Vendor('Weixinpay.Weixinpay');
  4.     $wxpay=new \Weixinpay();
  5.     // 获取jssdk需要用到的数据
  6.     $data=$wxpay->getParameters();
  7.     // 将数据分配到前台页面
  8.     $assign=array(
  9.         'data'=>json_encode($data)
  10.         );
  11.     $this->assign($assign);
  12.     $this->display();
  13. }
复制代码
需要html的配合:/home/order/pay.html





支付





   
























复制代码
调用示例:/Application/Home/Controller/orderController.class.php 中的makeorder方法
  1. /**
  2.  * 生成预支付订单,拿着这个订单,发起微信jsapi 支付
  3.  */
  4. public function makeorder(){
  5.         此处生成预支付订单,拿着这个订单,发起微信jsapi 支付
  6.     // 组合url
  7.     $url=U('home/order/pay',array('out_trade_no'=>$out_trade_no));
  8.     // 前往支付
  9.     redirect($url);
  10. }
五:异步接收通知

/Application/Api/Controller/WeixinpayController.class.php
 * notify_url接收页面
 */
public function notify(){
    // ↓↓↓下面的file_put_contents是用来简单查看异步发过来的数据 测试完可以删除;↓↓↓
    // 获取xml
    $xml=file_get_contents('php://input', 'r');
    //转成php数组 禁止引用外部xml实体
    libxml_disable_entity_loader(true);
    $data= json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA));
    file_put_contents('./notify.text', $data);
    // ↑↑↑上面的file_put_contents是用来简单查看异步发过来的数据 测试完可以删除;↑↑↑
    // 导入微信支付sdk
    Vendor('Weixinpay.Weixinpay');
    $wxpay=new \Weixinpay();
    $result=$wxpay->notify();
    if ($result) {
        // 验证成功 修改数据库的订单状态等 $result['out_trade_no']为订单id


    }
}
如果出现签名错误;

可以使用官方的 微信公众平台支付接口调试工具

跟自己生产的签名对比;

然后对比配置;查找不一致的地方;

如果遇到问题,请在下方留言




你可能感兴趣的:(PHP,微信公众平台,PHP使用总结)