php实现微信企业转账功能
更新时间:2018年10月02日 09:28:16 作者:Mr_Yanghao
这篇文章主要为大家详细介绍了php实现微信企业转账功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
本文实例为大家分享了php实现微信企业转账的具体代码,供大家参考,具体内容如下
<?php /** * 配置账号信息 * 配置要和证书在一起!!!! */ class WxTransfersConfig { //=======【基本信息设置】============== // /** * TODO: 修改这里配置为您自己申请的商户信息 * 微信公众号信息配置 * * APPID:绑定支付的APPID(必须配置,开户邮件中可查看) * * MCHID:商户号(必须配置,开户邮件中可查看) * * KEY:商户支付密钥,参考开户邮件设置(必须配置,登录商户平台自行设置) * 设置地址:https://pay.weixin.qq.com/index.php/account/api_cert * */ const APPID = ''; const MCHID = ''; const KEY = ''; //=======【证书路径设置】===================================== /** * TODO:设置商户证书路径 * 证书路径,注意应该填写绝对路径,发送红包和查询需要,可登录商户平台下载 * API证书下载地址:https://pay.weixin.qq.com/index.php/account/api_cert,下载之前需要安装商户操作证书) * @var path 跟这个文件同一目录下的cert文件夹放置证书!!!! */ const SSLCRET12 = 'cert/apiclient_cert.p12'; const SSLCERT_PATH = 'cert/apiclient_cert.pem'; const SSLKEY_PATH = 'cert/apiclient_key.pem'; const SSLROOTCA = 'cert/rootca.pem'; //=======【证书路径设置】===================================== /** * 获取文件的路径,证书需要完整路径 * @return string */ public static function getRealPath(){ return __DIR__.'/'; } }
微信企业转账工具类:
<?php require_once "WxTransfers.Config.php"; /** * 微信企业转账工具类 */ class WxTransfers { // 企业转账请求地址 const TRANSFERS_URL = 'https://api.mch.weixin.qq.com/mmpaymkttransfers/promotion/transfers'; //获取转账信息地址 const GETINFO_URL='https://api.mch.weixin.qq.com/mmpaymkttransfers/gettransferinfo'; // 转账需要的配置 'wxappid','mch_id','key' private $_keys; // 转账需要的证书文件 'api_cert', 'api_key', 'rootca',请传入绝对路径!!! private $_cert; protected $log_file; public $error; // 相关配置必备参数 protected $_parameters = array(); // 最后一次生产的订单号 protected $_lastPartnerTradeNo; // 记录最后一次发送请求的结果对象 protected $_lastResult; // 最后一次随机数 protected $_lastRandNum; public function __construct($config) { $keys = array( 'wxappid', 'mch_id', 'key' ); $files = array( 'api_cert', 'api_key', 'rootca' ); foreach ($keys as $key) { try { $this->_keys[$key] = $config[$key]; } catch (Exception $e) { throw new Exception('参数缺失:' . $key); } } foreach ($files as $file) { try { $cret_file = $config[$file]; if (is_file($cret_file)) { $this->_cert[$file] = $cret_file; } } catch (Exception $e) { throw new Exception('证书错误'); } } } public function transfers($parameters){ $this->log($parameters, 'SEND_PARAM'); $this->setParameter('mchid', $this->_keys['mch_id']); $this->setParameter('mch_appid', $this->_keys['wxappid']); $must = array( 'openid', 'check_name', 're_user_name', 'amount', 'desc', 'spbill_create_ip', ); foreach ($must as $key) { if (isset($parameters[$key]) && $parameters[$key]) { $this->setParameter($key, $parameters[$key]); } else if (! isset($this->_parameters[$key]) || ! $this->_parameters[$key]) { $this->error = '参数缺损:' . $key; return false; } } if (! isset($parameters['partner_trade_no'])) { $parameters['partner_trade_no'] = $this->getPartnerTradeNo(); } $this->setParameter('partner_trade_no', $parameters['partner_trade_no']); $this->setParameter('nonce_str', $this->getRand(30, 3)); $postXml = $this->_createXml(); if (! $postXml) { return false; } $this->log($postXml, 'SEND_XML'); $result = $this->curl_post_ssl(self::TRANSFERS_URL, $postXml); $this->log($result, 'RESULT_XML'); if (! $result) { return false; } $resultObj = simplexml_load_string($result, 'SimpleXMLElement', LIBXML_NOCDATA); $this->_lastResult = $resultObj; if ($resultObj->return_code == 'SUCCESS') { // 成功标识 if ($resultObj->result_code == 'SUCCESS') { return $resultObj->send_listid; } if ($resultObj->return_msg) { $this->error = (string) $resultObj->return_msg; return false; } $this->error = (string) $resultObj->err_code_des; return false; } if ($resultObj->return_code != 'FAIL') { $this->error = '返回信息格式异常'; return false; } $this->error = (string) $resultObj->return_msg; return false; } /** * 获取转账信息 * @param unknown $partner_trade_no * @return boolean|SimpleXMLElement */ public function getInfo($partner_trade_no){ $param = array( 'nonce_str' => $this->getRand(30, 3), 'partner_trade_no'=> $partner_trade_no , 'mch_id' => $this->_keys['mch_id'], 'appid' => $this->_keys['wxappid'], ); ksort($param); $unSignParaString = $this->_formatQueryParaMap($param, false); $param['sign'] = $this->_sign($unSignParaString, $this->_keys['key']); $xml = $this->arrayToXml($param); $this->log($xml, 'GETINFO_XML'); $result = $this->curl_post_ssl(self::GETINFO_URL, $xml); if(!$result){ return false ; } $this->log($result, 'RESULT_XML'); $resultObj = simplexml_load_string($result, 'SimpleXMLElement', LIBXML_NOCDATA); $this->_lastResult = $resultObj ; if($resultObj->return_code == 'SUCCESS'){//成功标识 if($resultObj->result_code == 'SUCCESS'){ return $resultObj ; } if($resultObj->return_msg){ $this->error = $resultObj->return_msg ; return false ; } $this->error = $resultObj->err_code_des ; return false ; } if($resultObj->return_code != 'FAIL'){ $this->error = '返回信息格式异常'; return false ; } $this->error = $resultObj->return_msg ; return false ; } /** * 设置所需要的参数 * @param $parameter 键值数组/键 * @param $value 值 * @return WxBonusApi */ public function setParameter($parameter, $value = null) { if (! is_array($parameter)) { return $this->setParameter(array( $parameter => $value )); } foreach ($parameter as $key => $value) { $key = trim($key); $value = trim($value); $this->_parameters[$key] = $value; } return $this; } /** * 获取参数值 * @param $parameter 键名 * @return multitype: */ public function getParameter($parameter) { return $this->_parameters[$parameter]; } /** * 获取随机数 * @param number $len 随机数的位数 * @param number $type 取值范围 1表示数字 2小写字母 4大写字母 * @return string */ public function getRand($len = 30, $type = 0) { $str = ''; $max = - 1; if (! $type) { $type = 3; } if ($type & 1) { $str .= '1234567890'; $max += 10; } if ($type & 2) { $str .= 'abcdefghijklmnopqrstuvwxyz'; $max += 26; } if ($type & 4) { $str .= 'ABCDEFGHIJKLMNOPQRSTUVWXYZ'; $max += 26; } $rand = ''; for ($i = 0; $i < $len; $i ++) { $rand .= $str[rand(0, $max)]; } return $rand; } /** * 生成商户的订单号 * @return string */ public function getPartnerTradeNo() { $this->_lastPartnerTradeNo = $this->_parameters['mch_id'] . date('YmdHis') . $this->getRand(4, 1); // $this->getRandNum(); return $this->_lastPartnerTradeNo; } /** * 获取最后一次创建生成的订单号 * @return string */ public function getLastPartnerTradeNo() { return $this->_lastPartnerTradeNo; } /** * 创建XML的方法 * @param number $retcode * @param string $reterrmsg * @return boolean|string */ private function _createXml() { try { $sign = $this->_getSign(); if (! $sign) { return false; } $this->setParameter('sign', $sign); return $this->arrayToXml($this->_parameters); } catch (Exception $e) { $this->error = $e->getMessage(); return false; } } /** * 参数转换成XML * @param array $arr 参数数组 * @return string */ public function arrayToXml($arr) { $xml = "<xml>"; foreach ($arr as $key => $val) { if (is_numeric($val)) { $xml .= "<" . $key . ">" . $val . "</" . $key . ">"; } else { $xml .= "<" . $key . "><![CDATA[" . $val . "]]></" . $key . ">"; } } $xml .= "</xml>"; return $xml; } /** * 获得签名结果 * @return boolean|Ambigous <string, boolean> */ protected function _getSign() { try { if ($this->_checkSign() == false) { // 检查生成签名参数 $this->error = '生成签名参数缺失!'; $this->log(json_encode($this->_parameters, JSON_UNESCAPED_UNICODE), 'ERROR_Sign_XML'); return false; } ksort($this->_parameters); $unSignParaString = $this->_formatQueryParaMap($this->_parameters, false); return $this->_sign($unSignParaString, $this->_keys['key']); } catch (Exception $e) { $this->error = $e->getMessage(); return false; } } /** * 检查签名所需参数是否齐全 * @return boolean */ private function _checkSign() { // return true; if ($this->_parameters["mch_appid"] == null || $this->_parameters["mchid"] == null || //$this->_parameters["device_info"] == null || 设备id $this->_parameters["nonce_str"] == null || $this->_parameters["partner_trade_no"] == null || $this->_parameters["openid"] == null || $this->_parameters["check_name"] == null || $this->_parameters["re_user_name"] == null || $this->_parameters["desc"] == null || $this->_parameters["spbill_create_ip"] == null) { return false; } return true; } /** * * @param $paraMap * @param $urlencode * @return string */ private function _formatQueryParaMap($paraMap,$urlencode) { $buff = ""; ksort($paraMap); foreach ($paraMap as $k => $v) { if (null != $v && "null" != $v && "sign" != $k) { if ($urlencode) { $v = urlencode($v); } $buff .= $k . "=" . $v . "&"; } } $reqPar; if (strlen($buff) > 0) { $reqPar = substr($buff, 0, strlen($buff) - 1); } return $reqPar; } /** * 签名 * @param $content 签名的字符串 * @param $key 密钥 * @throws Exception * @return string|boolean */ private function _sign($content, $key) { try { if (null == $key) { $this->error = '签名key不能为空!'; return false; } if (null == $content) { $this->error = '签名内容不能为空'; return false; } $signStr = $content . "&key=" . $key; return strtoupper(md5($signStr)); } catch (Exception $e) { $this->error = $e->getMessage(); return false; } } /** * cURL抓取 * * @param $url 链接地址 * @param $vars 参数 * @param * $second * @param * $aHeader * @return mixed|boolean */ function curl_post_ssl($url, $data, $second = 30, $aHeader = array()) { $ch = curl_init(); // 超时时间 curl_setopt($ch, CURLOPT_TIMEOUT, $second); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); // 这里设置代理,如果有的话 curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false); curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false); // cert 与 key 分别属于两个.pem文件 curl_setopt($ch, CURLOPT_SSLCERT, $this->_cert['api_cert']); curl_setopt($ch, CURLOPT_SSLKEY, $this->_cert['api_key']); curl_setopt($ch, CURLOPT_CAINFO, $this->_cert['rootca']); if (count($aHeader) >= 1) { curl_setopt($ch, CURLOPT_HTTPHEADER, $aHeader); } curl_setopt($ch, CURLOPT_POST, 1); curl_setopt($ch, CURLOPT_POSTFIELDS, $data); $data = curl_exec($ch); if ($data) { curl_close($ch); return $data; } else { $this->log(json_encode($this->_cert)); $this->error = 'aa:'.curl_errno($ch); curl_close($ch); return false; } } /** * 获取服务器ip * * @return string */ public function getServerIp() { $server_ip = '127.0.0.1'; if (isset($_SERVER)) { if (isset($_SERVER['SERVER_ADDR']) && $_SERVER['SERVER_ADDR']) { $server_ip = $_SERVER['SERVER_ADDR']; } elseif (isset($_SERVER['LOCAL_ADDR']) && $_SERVER['LOCAL_ADDR']) { $server_ip = $_SERVER['LOCAL_ADDR']; } } else { $server_ip = getenv('SERVER_ADDR'); } return $server_ip; } /** * 设置日志目录文件 * * @param unknown $file */ public function setLogFile($file) { $this->log_file = $file; } /** * 写日志 * * @param $msg 写入的信息 * @param $type 日志类型作为查询标示 */ public function log($msg, $type) { if ($this->log_file) { $log = str_replace(array( "\r\n", "\r", "\n" ), array( "", "", "" ), $msg); error_log($type . ' ' . date('Y-m-d H:i:s') . ' ' . json_encode($log,JSON_UNESCAPED_UNICODE) . "\r\n", 3, $this->log_file); } } }
<?php include 'WxTransfers.Api.php'; class WxTransfers{ /** *调用方法即可测试 */ public function index(){ $path = WxTransfersConfig::getRealPath(); // 证书文件路径 $config['wxappid'] = WxTransfersConfig::APPID; $config['mch_id'] = WxTransfersConfig::MCHID; $config['key'] = WxTransfersConfig::KEY; $config['PARTNERKEY'] = WxTransfersConfig::KEY; $config['api_cert'] = $path . WxTransfersConfig::SSLCERT_PATH; $config['api_key'] = $path . WxTransfersConfig::SSLKEY_PATH; $config['rootca'] = $path . WxTransfersConfig::SSLROOTCA; $wxtran=new WxTransfers($config); $wxtran->setLogFile('D:\\transfers.log');//日志地址 //转账 $data=array( 'openid'=>'',//openid 'check_name'=>'NO_CHECK',//是否验证真实姓名参数 're_user_name'=>'11',//姓名 'amount'=>100,//最小1元 也就是100 'desc'=>'企业转账测试',//描述 'spbill_create_ip'=>$wxtran->getServerIp(),//服务器IP地址 ); var_dump(json_encode($wxtran->transfers($data),JSON_UNESCAPED_UNICODE)); var_dump($wxtran->error); //获取转账信息 var_dump($wxtran->getInfo('11111111')); var_dump($wxtran->error); } }
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。
相关文章
php+redis实现多台服务器内网存储session并读取示例
这篇文章主要介绍了php+redis实现多台服务器内网存储session并读取示例,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧2017-01-01PHP中把stdClass Object转array的几个方法
PHP和JS通讯通常都用json,但用 json 传过来的数组并不是标准的array,而是 stdClass 类型。那么我们可以参考下面的几个方法进行转换。2014-05-05如何使用PHP Embed SAPI实现Opcodes查看器
这篇文章主要介绍了如何使用PHP Embed SAPI实现Opcodes查看器的相关资料,需要的朋友可以参考下2015-11-11WordPress上传图片错误:不是合法的JSON响应解决办法
这篇文章主要给大家介绍了关于WordPress上传图片错误:不是合法的JSON响应的解决办法,WordPress提示JSON错误通常是由于服务器配置或插件冲突引起的,文中通过代码介绍的非常详细,需要的朋友可以参考下2024-08-08
最新评论