<?php
declare (strict_types = 1);
namespace app\index\controller;
use think\Request;
class WxPay
{
/**
* wxPay
*/
public function wxPays(Request $request)
{
$ss = $this->weixinapp();
return json(['code'=>200,'data'=>$ss]);
}
//统一下单接口
private function unifiedorder() {
$wxClass = $this->wxClass();
$url = 'https://api.mch.weixin.qq.com/pay/unifiedorder';
$parameters = array(
'appid' => "{$wxClass['wx_appid']}", //服务商appid
'mch_id' => "{$wxClass['wx_mch_id']}", //服务商-商户号
'sub_appid' => "{$wxClass['sub_appid']}", //特约商户appid
'sub_mch_id' => "{$wxClass['sub_mch_id']}", //特约商户-商户号
'nonce_str' => $this->createNoncestr(), //随机字符串
'body' => '商品支付', //商品描述
'out_trade_no' => time().'3564', //商户订单号
'total_fee' => floatval(10 * 100), //总金额 单位 分
'spbill_create_ip' => $_SERVER['REMOTE_ADDR'], //终端IP
'notify_url' => 'http://api.xxxxxx.com', //通知地址 确保外网能正常访问
'sub_openid' => "okiww5XIrgqVNHJBEahMQN9-PYjk", //用户openid,小程序openid,使用小程序openid,openid要这样写sub_openid
// 'openid' => "odIWu6lhlXmKgQ06P9zHY9LwDGcQ", //公众号openid
'trade_type' => 'JSAPI',//交易类型
);
//统一下单签名
$parameters['sign'] = $this->getSign($parameters);
$xmlData = $this->arrayToXml($parameters);
$return = $this->xmlToArray($this->postXmlCurl($xmlData, $url, 60));
// print_r($return);die;
return $return;
}
/**
* 配置信息
* */
public function wxClass(){
$data['wx_appid'] = "";//微信公众号appid
$data['wx_mch_id'] = "";//微信支付商户号(服务商)
$data['wx_api'] = "";//服务商密匙
$data['wx_cert'] = "/wxClass/wxPay/wx/apiclient_cert.pem";//服务商证书cert
$data['wx_key'] = "/wxClass/wxPay/wx/apiclient_key.pem";//服务商证书key
$data['sub_appid'] = "";//特约商户appid
$data['sub_mch_id'] = "";//特约商户(商户号)
$data['sub_api'] = "";//特约商户密匙
$data['sub_cert'] = "/wxClass/wxPay/sub/apiclient_cert.pem";//特约商证书cert
$data['sub_key'] = "/wxClass/wxPay/sub/apiclient_key.pem";//特约商证书key
return $data;
}
//生成签名
private function getSign($Obj) {
foreach ($Obj as $k => $v) {
$Parameters[$k] = $v;
}
//签名步骤一:按字典序排序参数
ksort($Parameters);
$String = $this->formatBizQueryParaMap($Parameters, false);
//签名步骤二:在string后加入KEY
$String = $String . "&key=密匙是服务商api或特约商户api都可以,如果多个特约商户使用服务商api";
//签名步骤三:MD5加密
$String = md5($String);
//签名步骤四:所有字符转为大写
$result_ = strtoupper($String);
return $result_;
}
//签名(调起数据签名)
private function weixinapp() {
$wxClass = $this->wxClass();
//统一下单接口
$unifiedorder = $this->unifiedorder();
$parameters = array(
'appId' => $wxClass['sub_appid'], //小程序ID
'timeStamp' => '' . time() . '', //时间戳
'nonceStr' => $this->createNoncestr(), //随机串
'package' => 'prepay_id=' . $unifiedorder['prepay_id'], //数据包
'signType' => 'MD5'//签名方式
);
//签名
$parameters['paySign'] = $this->getSign($parameters);
return $parameters;
}
//产生随机字符串,不长于32位
private function createNoncestr($length = 32) {
$chars = "abcdefghijklmnopqrstuvwxyz0123456789";
$str = "";
for ($i = 0; $i < $length; $i++) {
$str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
}
return $str;
}
//格式化参数,签名过程需要使用
private function formatBizQueryParaMap($paraMap, $urlencode) {
$buff = "";
ksort($paraMap);
foreach ($paraMap as $k => $v) {
if ($urlencode) {
$v = urlencode($v);
}
$buff .= $k . "=" . $v . "&";
}
$reqPar = '';
if (strlen($buff) > 0) {
$reqPar = substr($buff, 0, strlen($buff) - 1);
}
return $reqPar;
}
//数组转换成xml
private function arrayToXml($arr) {
$xml = "<root>";
foreach ($arr as $key => $val) {
if (is_array($val)) {
$xml .= "<" . $key . ">" . arrayToXml($val) . "</" . $key . ">";
} else {
$xml .= "<" . $key . ">" . $val . "</" . $key . ">";
}
}
$xml .= "</root>";
return $xml;
}
//xml转换成数组
private function xmlToArray($xml) {
//禁止引用外部xml实体
libxml_disable_entity_loader(true);
$xmlstring = simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA);
$val = json_decode(json_encode($xmlstring), true);
return $val;
}
private static function postXmlCurl($xml, $url, $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);
curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 20);
curl_setopt($ch, CURLOPT_TIMEOUT, 40);
set_time_limit(0);
//运行curl
$data = curl_exec($ch);
//返回结果
if ($data) {
curl_close($ch);
// print_r($data);die;
return $data;
} else {
$error = curl_errno($ch);
curl_close($ch);
throw new WxPayException("curl出错,错误码:$error");
}
}
}
小程序写个接口访问wxPays
// 测试支付接口
dadePay(){
var url = app.globalData.apis+'/index/wxPays';
http.post(url,{},req=>{
var data = req.data.data;
console.log(data);
wx.requestPayment({
timeStamp: data.timeStamp,
nonceStr: data.nonceStr,
package: data.package,
signType: 'MD5',
paySign: data.paySign,
success (res) {
console.log(res)
},
fail (res) { }
})
});
},
服务商模式下和普通支付模式下是一样的写法,就几个参数不一样,下面这几个,其它全部一样写法
2、查账(下面这个查账方法,结合上面的支付签名)直接查账,参考统一下单写法
//微信支付查账调这个
public function getOrder($number){
$this->appid = "wx8142e3d013f70c68";
$wxClass = $this->wxClass();
$url = 'https://api.mch.weixin.qq.com/pay/orderquery';
$parameters = array(
'appid' => "", //服务商appid
'mch_id' => "", //服务商-商户号
'sub_appid' => "", //特约商户appid
'sub_mch_id' => "", //特约商户-商户号
'out_trade_no' => "", //商户订单号
'nonce_str' => $this->createNoncestr(), //随机字符串
);
$parameters['sign'] = $this->getSign($parameters);
$xmlData = $this->arrayToXml($parameters);
$return = $this->xmlToArray($this->postXmlCurl($xmlData, $url, 60));
// print_r($return);die;
return $return;
}
支付和查账不需要证书
3、服务商模式退款
//微信支付退款
public function wxRefunds($number){
$order = Db::query("select * from cdj_order_goods where number='{$number}'");
$this->appid = $order[0]['appid'];//appid
$wxClass = $this->wxClass();
$url = 'https://api.mch.weixin.qq.com/secapi/pay/refund';
$out_refund_no = 'wxtk'.time();//退款单到
$total_fee = $order[0]['charge'];
$parameters = array(
'appid' => "{$wxClass['wx_appid']}", //服务商appid
'mch_id' => "{$wxClass['wx_mch_id']}", //服务商-商户号
'sub_appid' => "{$wxClass['sub_appid']}", //特约商户appid
'sub_mch_id' => "{$wxClass['sub_mch_id']}", //特约商户-商户号
'out_trade_no' => $number, //商户订单号
'nonce_str' => $this->createNoncestr(), //随机字符串
'out_refund_no'=>$out_refund_no,
'total_fee'=>floatval($total_fee*100),
'refund_fee'=>floatval($total_fee*100),
);
$parameters['sign'] = $this->getSign($parameters);
$xmlData = $this->arrayToXml($parameters);
$path_cert = 'wxClass/wxPay/sub/apiclient_cert.pem';
$path_key = 'wxClass/wxPay/sub/apiclient_key.pem';
// $subPay = Db::query("select * from cdj_wxapp_pay_wx where sub_appid='{$this->appid}'");//特约商户证书
$subPay = Db::query("select * from cdj_wxapp_pay_wx where id=1");//服务商证书(现在使用服务商代退款,使用使用服务商证书)
//写入证书
file_put_contents($path_cert, $subPay[0]['sub_cert']);
file_put_contents($path_key, $subPay[0]['sub_key']);
$useCert = true;
$return = $this->xmlToArray($this->postXmlCurls($xmlData, $url,$useCert, 60,$path_cert,$path_key));
return $return;
}
private static function postXmlCurls($xml, $url, $useCert, $second = 30,$path_cert,$path_key)
{
$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);
if($useCert == true){
//设置证书
//使用证书:cert 与 key 分别属于两个.pem文件
curl_setopt($ch,CURLOPT_SSLCERTTYPE,'PEM');
curl_setopt($ch,CURLOPT_SSLCERT, $path_cert);
curl_setopt($ch,CURLOPT_SSLKEYTYPE,'PEM');
curl_setopt($ch,CURLOPT_SSLKEY, $path_key);
}
//post提交方式
curl_setopt($ch, CURLOPT_POST, TRUE);
curl_setopt($ch, CURLOPT_POSTFIELDS, $xml);
//运行curl
$data = curl_exec($ch);
//返回结果
if($data){
curl_close($ch);
return $data;
} else {
$error = curl_errno($ch);
curl_close($ch);
message("退款失败,CURL出错,错误码:".$error);exit;
}
}