php下应用微信支付功能

2018-12-03  本文已影响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调起微信进行支付

上一篇下一篇

猜你喜欢

热点阅读