需求背景:微信提現(xiàn)到用戶的零錢
官方文檔:https://pay.weixin.qq.com/wiki/doc/api/tools/mch_pay.php?chapter=14_1
<?php
/**
* 功能說明:微信退款(企業(yè)付款到零錢)
*/
class WxRefund
{
//應(yīng)用appid
public $pay_appid = '';
//商戶id
public $pay_mchid = '';
//商戶密鑰
public $pay_mchkey = '';
public function __construct()
{
$this->pay_appid = config('WEIXINPAY.APPID');
$this->pay_mchid = config('WEIXINPAY.PAYMCHID');
$this->pay_mchkey = config('WEIXINPAY.PAYMCHKEY');
}
/**
* 作用:生成簽名
*/
private function getSign($Parameters)
{
//簽名步驟一:按字典序排序參數(shù)
ksort($Parameters);
$buff = "";
foreach ($Parameters as $k => $v) {
$buff .= $k . "=" . $v . "&";
}
$String = '';
if (strlen($buff) > 0) {
$String = substr($buff, 0, strlen($buff) - 1);
}
//簽名步驟二:在string后加入KEY
$String = $String . "&key=" . $this->pay_mchkey;
//簽名步驟三:MD5加密
$String = md5($String);
//簽名步驟四:所有字符轉(zhuǎn)為大寫
$result_ = strtoupper($String);
return $result_;
}
/**
* 作用:array轉(zhuǎn)xml
*/
private function arrayToXml($arr)
{
$xml = "<xml>";
foreach ($arr as $key => $val) {
$xml .= "<" . $key . ">" . $val . "</" . $key . ">";
}
$xml .= "</xml>";
return $xml;
}
/**
* xml轉(zhuǎn)換為數(shù)組
*/
public function xmlToArray($xml)
{
//禁止引用外部xml實(shí)體
libxml_disable_entity_loader(true);
$xmlstring = simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA);
$val = json_decode(json_encode($xmlstring), true);
return $val;
}
/**
* 作用:以post方式提交xml到對(duì)應(yīng)的接口url
*/
private function postXmlCurl($xml, $url)
{
$ch = curl_init(); //初始化curl
curl_setopt($ch, CURLOPT_URL, $url); //抓取指定網(wǎng)頁
curl_setopt($ch, CURLOPT_HEADER, 0); //設(shè)置header
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); //要求結(jié)果為字符串且輸出到屏幕上
curl_setopt($ch, CURLOPT_POST, 1); //post提交方式
curl_setopt($ch, CURLOPT_POSTFIELDS, $xml); // 增加 HTTP Header(頭)里的字段
curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE); // 終止從服務(wù)端進(jìn)行驗(yàn)證
curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE);
//(3)API證書安全
// 1.證書文件不能放在web服務(wù)器虛擬目錄吓著,應(yīng)放在有訪問權(quán)限控制的目錄中鲤嫡,防止被他人下載;
// 2.建議將證書文件名改為復(fù)雜且不容易猜測(cè)的文件名绑莺;
// 3.商戶服務(wù)器要做好病毒和木馬防護(hù)工作暖眼,不被非法侵入者竊取證書文件。
$zs1 = "證書文件夾/weixin/********.pem"; // 證書地址
$zs2 = "證書文件夾/weixin/******.pem"; // 證書地址
curl_setopt($ch, CURLOPT_SSLCERT, $zs1);
curl_setopt($ch, CURLOPT_SSLKEY, $zs2);
$data = curl_exec($ch); //運(yùn)行curl
if (curl_errno($ch)) {
echo 'Errno:' . curl_error($ch);
}
curl_close($ch);
return $data;
}
/**
* 產(chǎn)生隨機(jī)字符串纺裁,不長(zhǎng)于32位
*
* @param int $length
* @return 產(chǎn)生的隨機(jī)字符串
*/
public static function getNonceStr($length = 32)
{
$chars = "abcdefghijklmnopqrstuvwxyz0123456789";
$str = "";
for ($i = 0; $i < $length; $i++) {
$str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
}
return $str;
}
/**
* 企業(yè)付款到零錢
* @param string $partner_trade_no 商戶訂單號(hào)
* @param float $amount 發(fā)送的金額(分)
* @param string $re_openid 收款人的 openid
* @param string $desc 企業(yè)付款描述信息 (必填)
* @param string $check_name 收款用戶姓名 (選填)
* @return [type] [description] 這里沒有使用姓名強(qiáng)校驗(yàn)
*/
public function sendMoney($partner_trade_no, $amount, $re_openid, $desc = '測(cè)試', $check_name = '')
{
$data = array(
'mch_appid' => $this->pay_appid,// 商戶賬號(hào)appid
'mchid' => $this->pay_mchid,// 商戶號(hào)
'nonce_str' => $this->getNonceStr(),// 隨機(jī)字符串
'partner_trade_no' => $partner_trade_no,// 商戶訂單號(hào)
'openid' => $re_openid,// 用戶openid
'check_name' => 'NO_CHECK',// 校驗(yàn)用戶姓名選項(xiàng), NO_CHECK:不校驗(yàn)真實(shí)姓名 FORCE_CHECK:強(qiáng)校驗(yàn)真實(shí)姓名
// 're_user_name' => $check_name,// 收款用戶姓名
'amount' => $amount * 100,// 金額
'desc' => $desc,// 企業(yè)付款描述信息
// 'spbill_create_ip'=> IP,// Ip地址
);
//生成簽名算法
$data['sign'] = $this->getSign($data);
$xml = $this->arrayToXml($data);
$url = 'https://api.mch.weixin.qq.com/mmpaymkttransfers/promotion/transfers'; // 調(diào)用接口
$res = $this->postXmlCurl($xml, $url);
$return = $this->xmlToArray($res);
if($return['return_code'] == 'SUCCESS' && $return['result_code'] == 'SUCCESS' && $return['err_code'] == 'SUCCESS'){
return array('status'=>1,'payment_no'=>$return['payment_no']);
}else{
return array('status'=>0,'message'=>$return['err_code_des']);
}
}
}