tp6端
<?php
declare (strict_types = 1);
namespace app\controller\Index;
use app\BaseController;
use app\model\Set;
use think\Request;
class Order_Payment_Controller extends BaseController
{
/**
* wxPay
*/
public function wxPays(Request $request)
{
$data = $request->get();//这里是前端传的值
//订单编号
$data['goods_number'] = 'wx'.date('Ymd').substr(implode(NULL, array_map('ord', str_split(substr(uniqid(), 7, 13), 1))), 0, 8);
if($data['present'] >0.03){ //这里做判断,官方要求微信支付金额必须大于等于0.03
$ss = $this->weixinapp($data);
return json(['code'=>200,'goods_number'=>$data['goods_number'],'data'=>$ss]);
}else{
return json(['code'=>200,'goods_number'=>$data['goods_number'],'data'=>'支付成功']);
}
}
/**
* 这里访问小程序ID,商户号,商户密匙,支付商证书信息
*/
public function set()
{
$set = Set::find(1);
return $set;
}
//统一下单接口
private function unifiedorder($data) {
$set = $this->set();
$url = 'https://api.mch.weixin.qq.com/pay/unifiedorder';
$parameters = array(
'appid' => $set['set_appid'], //小程序ID
'mch_id' => $set['set_mch_id'], //商户号
'nonce_str' => $this->createNoncestr(), //随机字符串
'body' => $data['specificationse_name'], //商品描述
'out_trade_no' => $data['goods_number'], //商户订单号
'total_fee' => floatval($data['present'] * 100), //总金额 单位 分
'spbill_create_ip' => $_SERVER['REMOTE_ADDR'], //终端IP
'notify_url' => 'https://***.*****.com', //通知地址 确保外网能正常访问
'openid' => $data['openid'], //用户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(){
$set = $this->set();
$data['sub_appid'] = $set['set_appid'];//商户appid
$data['sub_mch_id'] = $set['set_mch_id'];//商户(商户号)
$data['sub_api'] = $set['set_api'];//商户密匙
$data['sub_cert'] = $set['sub_cert'];//证书路径.cert
$data['sub_key'] = $set['sub_key'];//证书路径.key
return $data;
}
//签名(调起数据签名)
private function weixinapp($data) {
$wxClass = $this->wxClass();
//统一下单接口
$unifiedorder = $this->unifiedorder($data);
$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 = "ABCDEFGHIJKIMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
$str = "";
for ($i = 0; $i < $length; $i++) {
$str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
}
return $str;
}
//生成签名
private function getSign($Obj) {
foreach ($Obj as $k => $v) {
$Parameters[$k] = $v;
}
//签名步骤一:按字典序排序参数
ksort($Parameters);
$String = $this->formatBizQueryParaMap($Parameters, false);
//签名步骤二:在string后加入KEY
// $String = $String . "&key=" . $this->key;
$set = $this->set();
$String = $String . "&key=".$set['set_api'];
//签名步骤三:MD5加密
$String = md5($String);
//签名步骤四:所有字符转为大写
$result_ = strtoupper($String);
return $result_;
}
//格式化参数,签名过程需要使用
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);
return $data;
} else {
$error = curl_errno($ch);
curl_close($ch);
throw new WxPayException("curl出错,错误码:$error");
}
}
/**
* 退款配置信息
* */
public function wxClass_retreat(){
$system = $this->set();
$data['sub_appid'] = $system['set_appid'];//商户appid(小程序)
$data['sub_mch_id'] = $system['set_mch_id'];//商户(商户号)
$data['sub_api'] = $system['set_api'];//商户密匙
$data['sub_cert'] = "*****/****/apiclient_cert.pem";//商证书cert
$data['sub_key'] = "*****/****/apiclient_key.pem";//商证书key
return $data;
}
/**
* 同意退款
*/
public function order_retreat_yes($data){
$order = Order::where('user_openid', $data['user_openid'])->where('goods_number', $data['goods_number'])->find();//查订单是否存在
if(!empty($order)){
$time = time();
$hounian = strtotime("+12 month",$order['goods_order_time']);//1年之后的时间戳
if ($time > $hounian){//微信支付规则,订单超过一年不允许退款
return json(['code' => 200, 'message' => '超期订单不允许退款,退款失败']);
}else{
$wxClass = $this->wxClass_retreat();
$url = 'https://api.mch.weixin.qq.com/secapi/pay/refund';
$time = time();
$out_refund_no = 'wxtk' . $time;//退款单
$parameters = array(
'appid' => "{$wxClass['sub_appid']}", //应用ID,固定
'mch_id' => "{$wxClass['sub_mch_id']}", //商户号,固定
'nonce_str' => $this->createNoncestr(), //随机字符串
'out_refund_no' => $out_refund_no, //商户内部唯一退款单号
'out_trade_no' => $data['goods_number'], //商户订单号,pay_sn码 1.1二选一,微信生成的订单号,在支付通知中有返回
// 'transaction_id'=>'1',//微信订单号 1.2二选一,商户侧传给微信的订单号
'refund_fee' => floatval($data['goods_discount_price'] * 100), //退款金额
'total_fee' => floatval($data['goods_discount_price'] * 100), //总金额
);
$parameters['sign'] = $this->getSign($parameters);
$xmlData = $this->arrayToXml($parameters);
$path_cert = 'wxClass/wxPay/apiclient_cert.pem';
$path_key = 'wxClass/wxPay/apiclient_key.pem';
$useCert = true;
$return = $this->xmlToArray($this->postXmlCurls($xmlData, $url, $useCert, 60, $path_cert, $path_key));
if ($return['result_code'] == "SUCCESS" && $return['return_code'] == "SUCCESS" && $return['return_msg'] == "OK") {
$order -> order_states = 5;
$order -> goods_refund_complete_time = $time;
$order -> save();
return json(['code' => 200, 'message' => '退款成功']);
} else {
if ($return['result_code'] == "FAIL" && $return['return_code'] == "SUCCESS" && $return['return_msg'] == "OK") {
$order -> order_states = 5;
$order -> goods_refund_complete_time = $time;
$order -> save();
return json(['code' => 200, 'message' => '已退款']);
} else {
return json(['code' => 200, 'data' => $return, 'message' => '退款失败']);
}
}
}
}else{
return json(['code' => 200, 'message' => '订单不存在,退款失败']);
}
}
}
uniapp 端 微信支付
uni.requestPayment({
timeStamp: req.data.data.timeStamp,
nonceStr: req.data.data.nonceStr,
package: req.data.data.package,
signType: 'MD5',
paySign: req.data.data.paySign,
success: (reqs) => {
if (reqs.errMsg == "requestPayment:ok") {
uni.showToast({
icon: 'none',
title: '支付成功',
duration: 1500
});
})
} else {
uni.showToast({
icon: 'error',
title: '支付异常',
duration: 2000
});
}
},
fail(reqs) {
uni.showToast({
icon: 'error',
title: '支付失败',
duration: 2000
});
}
})