1、微信提現(xiàn)到指定的銀行卡
2、官方文檔:https://pay.weixin.qq.com/wiki/doc/api/tools/mch_pay.php?chapter=24_1&index=1
<?php
namespace data\service\Pay;
/**
* 功能說(shuō)明:微信退款(企業(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)為大寫(xiě)
$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)頁(yè)
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);
// API證書(shū)安全
// 1.證書(shū)文件不能放在web服務(wù)器虛擬目錄,應(yīng)放在有訪問(wèn)權(quán)限控制的目錄中,防止被他人下載;
// 2.建議將證書(shū)文件名改為復(fù)雜且不容易猜測(cè)的文件名;
// 3.商戶服務(wù)器要做好病毒和木馬防護(hù)工作,不被非法侵入者竊取證書(shū)文件。
$zs1 = "證書(shū)文件夾/weixin/********.pem"; // 證書(shū)地址
$zs2 = "證書(shū)文件夾/weixin/******.pem"; // 證書(shū)地址
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;
}
/**
* 公鑰加密,銀行卡號(hào)和姓名需要RSA算法加密
* @param string $data 需要加密的字符串,銀行卡/姓名
* @return null|string 加密后的字符串
*/
private function publicEncrypt($data)
{
// 進(jìn)行RSA公鑰加密
$pubkey = openssl_pkey_get_public(file_get_contents('證書(shū)地址/weixin/*****.pem')); // 證書(shū)地址和文件名
$encrypt_data = '';
$encrypted = '';
$r = openssl_public_encrypt($data, $encrypt_data, $pubkey, OPENSSL_PKCS1_OAEP_PADDING);
if ($r) {//加密成功,返回base64編碼的字符串
return base64_encode($encrypted . $encrypt_data);
} else {
return false;
}
}
/**
* 產(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è)付款到銀行卡接口
* @params string $out_trade_no : 商戶訂單號(hào)
* @params int $amount : 付款金額,單位分
* @params string $enc_bank_no : 收款方銀行卡號(hào)
* @params string $enc_true_name : 收款方用戶名
* @params string $bank_code : 收款方開(kāi)戶行銀行編號(hào)bank_code
* @params string $desc : 付款備注
* return string $payment_no :支付成功的訂單號(hào)
*/
public function payForBank($out_trade_no,$amount,$enc_bank_no,$enc_true_name,$bank_code,$desc='企業(yè)付款到銀行卡')
{
$data = [
'amount' => $amount * 100,
'bank_code' => $bank_code,
'desc' => $desc,
'enc_bank_no' => $this->publicEncrypt($enc_bank_no),
'enc_true_name' => $this->publicEncrypt($enc_true_name),
'mch_id' => $this->pay_mchid,
'nonce_str' => $this->getNonceStr(),
'partner_trade_no' => $out_trade_no
];
$data['sign'] = $this->getSign($data);
$xml = $this->arrayToXml($data);
$url = 'https://api.mch.weixin.qq.com/mmpaysptrans/pay_bank';
$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']);
}
}
}