美文网首页
php下应用微信支付功能

php下应用微信支付功能

作者: 磐璃 | 来源:发表于2018-12-03 16:20 被阅读0次

    一、微信支付所包含的类别

    1. 付款码付款
    2. JSAPI支付
    3. Native支付
    4. APP支付
    5. H5支付
    6. 小程序支付

    日常网页开发中付款码付款不常用;APP支付还没做研究;今天主要将剩下的几种方式
    任何支付首先都需要预下单在微信平台做同一下单处理

    github源码地址

    二、统一下单

    统一下单根据支付类型不同需要设置不同的 trade_type

    1.以扫码支付统一下单为例

    public function unifiedOrder($order_no,$total_fee)
    {
            // 当前时间
            $time = time();
            // 生成随机字符串
            $nonceStr = md5($time.'#_pay@sign');
            // API参数
            $params = [
                'appid' => config('wxpay.appid'),//公众账户ID
                'mch_id' => config('wxpay.mch_id'),//商户号
                'attach' => '某某分店等信息',//附加数据
                'body' => '腾讯充值中心-QQ会员充值',//商品描述
                'nonce_str' => $nonceStr,//随机字符串
                'sign' => 'MD5',//加密方式默认MD5可以省略
                'notify_url' => base_url() . 'notice.php',  // 异步通知地址,支付成功后微信支付异步回调地址
                'out_trade_no' => $order_no,//商户订单号;如果有修改价格的需求建议表中单独设立微信支付编号;注意保证唯一性
                'spbill_create_ip' => \request()->ip(),//终端IP
                'total_fee' => $total_fee * 100, // 价格:单位分,默认币种人民币
                'trade_type' => 'NATIVE',
            ];
    
            // 生成签名
            $params['sign'] = $this->makeSign($params);
    
            // 请求API
            $url = 'https://api.mch.weixin.qq.com/pay/unifiedorder';
            $result = $this->postXmlCurl($this->toXml($params), $url);//发出请求
            $prepay = $this->fromXml($result);//格式转换
    
            // 请求失败;只有返回数据return_code和result_code都为SUCCESS统一下单才成功
            if ($prepay['return_code'] !== 'SUCCESS' || $prepay['result_code'] !== 'SUCCESS') {
                return false;
            }
    
            return $prepay;
    }
    

    生成签名函数

    private function makeSign($values)
    {
            //签名步骤一:按字典序排序参数
            ksort($values);
            $string = $this->toUrlParams($values);
            //签名步骤二:在string后加入KEY
            $string = $string . '&key=' .config('wxpay.key');//支付商户账户获取key
            //签名步骤三:MD5加密
            $string = md5($string);
            //签名步骤四:所有字符转为大写
            $result = strtoupper($string);
            return $result;
    }
    

    格式化参数为url格式

    private function toUrlParams($values)
    {
            $buff = '';
            foreach ($values as $k => $v) {
                if ($k != 'sign' && $v != '' && !is_array($v)) {
                    $buff .= $k . '=' . $v . '&';
                }
            }
            return trim($buff, '&');
    }
    

    进行POST请求

     private function postXmlCurl($xml, $url,$is_cert = false, $second = 30)
    {
            $ch = curl_init();
            // 设置超时
            curl_setopt($ch, CURLOPT_TIMEOUT, $second);
            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);
    
            if($is_cert){
                curl_setopt($ch, CURLOPT_SSLCERTTYPE, 'PEM');
                curl_setopt($ch, CURLOPT_SSLCERT, 'apiclient_cert.pem');//证书的物理绝对路径
                //默认格式为PEM,可以注释
                curl_setopt($ch, CURLOPT_SSLKEYTYPE, 'PEM');
                curl_setopt($ch, CURLOPT_SSLKEY, 'apiclient_key.pem');//证书的物理绝对路径
            }
    
            // 运行curl
            $data = curl_exec($ch);
            curl_close($ch);
            return $data;
    
    }
    

    将数据格式转换为xml

    private function toXml($values)
    {
            if (!is_array($values)
                || count($values) <= 0
            ) {
                return false;
            }
    
            $xml = "<xml>";
            foreach ($values as $key => $val) {
                if (is_numeric($val)) {
                    $xml .= "<" . $key . ">" . $val . "</" . $key . ">";
                } else {
                    $xml .= "<" . $key . "><![CDATA[" . $val . "]]></" . $key . ">";
                }
            }
            $xml .= "</xml>";
            return $xml;
    }
    

    将xml格式转换为数组

    private function fromXml($xml)
    {
            // 禁止引用外部xml实体
            libxml_disable_entity_loader(true);
            return json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);
    }
    

    2.集成多项支付的统一下单

    默认支付模式为NATIVE,再使用小程序支付时必须传入openid
    如果下单失败返回false,成功返回微信支付后台返回值

    public function unifiedOrder($wxpay_no,$total_fee,$trade_type='NATIVE',$openid='')
    {
            // 当前时间
            $time = time();
            // 生成随机字符串
            $nonceStr = md5($time.'#_pay@sign');
            // API参数
            $params = [
                'appid' => config('wxpay.appid'),//公众账户ID
                'mch_id' => config('wxpay.mch_id'),//商户号
                'attach' => '某某分店等信息',//附加数据
                'body' => '腾讯充值中心-QQ会员充值',//商品描述
                'nonce_str' => $nonceStr,//随机字符串
                'sign' => 'MD5',//加密方式默认MD5可以省略
                'notify_url' => base_url() . 'notice.php',  // 异步通知地址,支付成功后微信支付异步回调地址
                'out_trade_no' => $wxpay_no,//商户订单号;如果有修改价格的需求建议表中单独设立微信支付编号;注意保证唯一性
                'spbill_create_ip' => \request()->ip(),//终端IP
                'total_fee' => $total_fee * 100, // 价格:单位分,默认币种人民币
                'trade_type' => $trade_type,
            ];
    
            //如果为JSAPI支付
            if($trade_type == 'JSAPI'){
                $params['openid'] = $openid;
            }
    
            // 生成签名
            $params['sign'] = $this->makeSign($params);
    
            // 请求API
            $url = 'https://api.mch.weixin.qq.com/pay/unifiedorder';
            $result = $this->postXmlCurl($this->toXml($params), $url);//发出请求
            $prepay = $this->fromXml($result);//格式转换
    
            // 请求失败;只有返回数据return_code和result_code都为SUCCESS统一下单才成功
            if ($prepay['return_code'] !== 'SUCCESS' || $prepay['result_code'] !== 'SUCCESS') {
                return false;
            }
    
            return $prepay;
    }
    

    三、订单查询

    传入微信支付单号

    public function orderQuery($wspay_no)
    {
            // 当前时间
            $time = time();
            // 生成随机字符串
            $nonceStr = md5($time.'#_pay@sign');
            // API参数
            $params = [
                'appid' => config('wxpay.appid'),//公众账户ID
                'mch_id' => config('wxpay.mch_id'),//商户号
                'nonce_str' => $nonceStr,
                'out_trade_no' => $wspay_no,
            ];
    
            // 生成签名
            $params['sign'] = $this->makeSign($params);
            // 请求API
            $url = 'https://api.mch.weixin.qq.com/pay/orderquery';
            $result = $this->postXmlCurl($this->toXml($params), $url);
            $prepay = $this->fromXml($result);
    
            return $prepay;
    }
    

    四、关闭订单

    订单生成后不能马上关闭,最短时间间隔为5分钟

    public function closeOrder($wspay_no)
    {
            // 当前时间
            $time = time();
            // 生成随机字符串
            $nonceStr = md5($time.'#_pay@sign');
            // API参数
            $params = [
                'appid' => config('wxpay.appid'),//公众账户ID
                'mch_id' => config('wxpay.mch_id'),//商户号
                'nonce_str' => $nonceStr,
                'out_trade_no' => $wspay_no,
            ];
    
            // 生成签名
            $params['sign'] = $this->makeSign($params);
            // 请求API
            $url = 'https://api.mch.weixin.qq.com/pay/closeorder';
            $result = $this->postXmlCurl($this->toXml($params), $url);
            $prepay = $this->fromXml($result);
    
            return $prepay;
     }
    

    五、申请退款

    1、交易时间超过一年的订单无法提交退款
    2、微信支付退款支持单笔交易分多次退款,多次退款需要提交原支付订单的商户订单号和设置不同的退款单号。申请退款总金额不能超过订单金额。 一笔退款失败后重新提交,请不要更换退款单号,请使用原商户退款单号
    3、请求频率限制:150qps,即每秒钟正常的申请退款请求次数不超过150次
    错误或无效请求频率限制:6qps,即每秒钟异常或错误的退款申请请求不超过6次
    4、每个支付订单的部分退款次数不能超过50次
    5、此请求需要安全证书的认证,路径地址在postXmlCur()里配置

    public function refund($wxpay_no,$out_refund_no,$total_fee,$refund_fee)
    {
            // 当前时间
            $time = time();
            // 生成随机字符串
            $nonceStr = md5($time.'#_pay@sign');
            // API参数
            $params = [
                'appid' => 'wx6bf5eec027a0fe45',//公众账户ID
                'mch_id' => '1511613151',//商户号
                'nonce_str' => $nonceStr,//随机字符串
                'out_trade_no' => $wxpay_no,//微信支付号
                'out_refund_no' => $out_refund_no,//退款编号
                'total_fee' => $total_fee * 100,//单号总金额
                'refund_fee' => $refund_fee * 100,//退款金额,不能超过总金额
                'notify_url' =>  base_url() . 'refundNoticy',//退款成功通知地址
            ];
    
            // 生成签名
            $params['sign'] = $this->makeSign($params);
    
            // 请求API
            $url = 'https://api.mch.weixin.qq.com/secapi/pay/refund';
            $result = $this->postXmlCurl($this->toXml($params), $url,true);
            $prepay = $this->fromXml($result);
    
            return $prepay;
    }
    

    六、请求支付的调用

    1. NATIVE模式(扫码支付)

    根据返回数据code_url链接生成二维码供客户扫码识别

    2.小程序支付(JSAPI)

    需要再次签名
    小程序调用支付窗口需要参数$nonceStr,$prepayId,$timeStamp以及加密签名后返回的paySign

    private function makePaySign($nonceStr, $prepayId, $timeStamp)
    {
            $data = [
                'appId' => config('wxpay.app_id'),//公众账户ID
                'nonceStr' => $nonceStr,//随机字符串,与调起支付时相同
                'package' => 'prepay_id=' . $prepayId,//统一下单返回的prepay_id
                'signType' => 'MD5',
                'timeStamp' => $timeStamp,//时间戳,与调起支付时相同
            ];
    
            //签名步骤一:按字典序排序参数
            ksort($data);
    
            $string = $this->toUrlParams($data);
    
            //签名步骤二:在string后加入KEY
            $string = $string . '&key=' . config('wxpay.key');
    
            //签名步骤三:MD5加密
            $string = md5($string);
    
            //签名步骤四:所有字符转为大写
            $result = strtoupper($string);
    
            return $result;
    }
    

    3.H5支付(MWEB)

    根据返回的mweb_url调起微信进行支付

    相关文章

      网友评论

          本文标题:php下应用微信支付功能

          本文链接:https://www.haomeiwen.com/subject/lzrycqtx.html