一、微信支付所包含的类别
- 付款码付款
- JSAPI支付
- Native支付
- APP支付
- H5支付
- 小程序支付
日常网页开发中付款码付款不常用;APP支付还没做研究;今天主要将剩下的几种方式
任何支付首先都需要预下单在微信平台做同一下单处理
二、统一下单
统一下单根据支付类型不同需要设置不同的 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;
}
六、请求支付的调用
- 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调起微信进行支付
网友评论