微信企业转账【完整DEMO代码】

  1. /** 
  2. *   配置账号信息 
  3. *   配置要和证书在一起!!!! 
  4. */  
  5.   
  6. class WxTransfersConfig  
  7. {  
  8.     //=======【基本信息设置】=====================================  
  9.     //  
  10.     /** 
  11.      * TODO: 修改这里配置为您自己申请的商户信息 
  12.      * 微信公众号信息配置 
  13.      *  
  14.      * APPID:绑定支付的APPID(必须配置,开户邮件中可查看) 
  15.      *  
  16.      * MCHID:商户号(必须配置,开户邮件中可查看) 
  17.      *  
  18.      * KEY:商户支付密钥,参考开户邮件设置(必须配置,登录商户平台自行设置) 
  19.      * 设置地址:https://pay.weixin.qq.com/index.php/account/api_cert 
  20.      *  
  21.      */  
  22.     const APPID = '';  
  23.     const MCHID = '';  
  24.     const KEY = '';  
  25.     //=======【证书路径设置】=====================================  
  26.     /** 
  27.      * TODO:设置商户证书路径 
  28.      * 证书路径,注意应该填写绝对路径,发送红包和查询需要,可登录商户平台下载 
  29.      * API证书下载地址:https://pay.weixin.qq.com/index.php/account/api_cert,下载之前需要安装商户操作证书) 
  30.      * @var path 跟这个文件同一目录下的cert文件夹放置证书!!!! 
  31.      */  
  32.     const SSLCRET12 = 'cert/apiclient_cert.p12';  
  33.     const SSLCERT_PATH = 'cert/apiclient_cert.pem';  
  34.     const SSLKEY_PATH = 'cert/apiclient_key.pem';  
  35.     const SSLROOTCA = 'cert/rootca.pem';  
  36.       
  37.     //=======【证书路径设置】=====================================  
  38.     /** 
  39.      * 获取文件的路径,证书需要完整路径 
  40.      * @return string 
  41.      */  
  42.     public static function getRealPath(){  
  43.         return __DIR__.'/';  
  44.     }  
  45. }  

[php]  view plain  copy
  1. require_once "WxTransfers.Config.php";  
  2.   
  3. /** 
  4.  * 微信企业转账工具类 
  5.  */  
  6. class WxTransfers  
  7. {  
  8.     // 企业转账请求地址  
  9.     const TRANSFERS_URL = 'https://api.mch.weixin.qq.com/mmpaymkttransfers/promotion/transfers';  
  10.       
  11.     //获取转账信息地址  
  12.     const GETINFO_URL='https://api.mch.weixin.qq.com/mmpaymkttransfers/gettransferinfo';  
  13.       
  14.     // 转账需要的配置 'wxappid','mch_id','key'  
  15.     private $_keys;  
  16.       
  17.     // 转账需要的证书文件 'api_cert', 'api_key', 'rootca',请传入绝对路径!!!  
  18.     private $_cert;  
  19.       
  20.     protected $log_file;  
  21.       
  22.     public $error;  
  23.       
  24.     // 相关配置必备参数  
  25.     protected $_parameters = array();  
  26.       
  27.     // 最后一次生产的订单号  
  28.     protected $_lastPartnerTradeNo;  
  29.       
  30.     // 记录最后一次发送请求的结果对象  
  31.     protected $_lastResult;  
  32.       
  33.     // 最后一次随机数  
  34.     protected $_lastRandNum;  
  35.       
  36.     public function __construct($config)  
  37.     {  
  38.         $keys = array(  
  39.             'wxappid',  
  40.             'mch_id',  
  41.             'key'  
  42.         );  
  43.         $files = array(  
  44.             'api_cert',  
  45.             'api_key',  
  46.             'rootca'  
  47.         );  
  48.       
  49.         foreach ($keys as $key) {  
  50.             try {  
  51.                 $this->_keys[$key] = $config[$key];  
  52.             } catch (Exception $e) {  
  53.                 throw new Exception('参数缺失:' . $key);  
  54.             }  
  55.         }  
  56.       
  57.         foreach ($files as $file) {  
  58.             try {  
  59.                 $cret_file = $config[$file];  
  60.                 if (is_file($cret_file)) {  
  61.                     $this->_cert[$file] = $cret_file;  
  62.                 }  
  63.             } catch (Exception $e) {  
  64.                 throw new Exception('证书错误');  
  65.             }  
  66.         }  
  67.     }  
  68.   
  69.     public function transfers($parameters){  
  70.   
  71.         $this->log($parameters'SEND_PARAM');  
  72.           
  73.         $this->setParameter('mchid'$this->_keys['mch_id']);  
  74.         $this->setParameter('mch_appid'$this->_keys['wxappid']);  
  75.           
  76.         $must = array(  
  77.             'openid',  
  78.             'check_name',  
  79.             're_user_name',  
  80.             'amount',  
  81.             'desc',  
  82.             'spbill_create_ip',  
  83.         );  
  84.         foreach ($must as $key) {  
  85.             if (isset($parameters[$key]) && $parameters[$key]) {  
  86.                 $this->setParameter($key$parameters[$key]);  
  87.             } else  
  88.                 if (! isset($this->_parameters[$key]) || ! $this->_parameters[$key]) {  
  89.                     $this->error = '参数缺损:' . $key;  
  90.                     return false;  
  91.                 }  
  92.         }  
  93.         if (! isset($parameters['partner_trade_no'])) {  
  94.             $parameters['partner_trade_no'] = $this->getPartnerTradeNo();  
  95.         }  
  96.           
  97.         $this->setParameter('partner_trade_no'$parameters['partner_trade_no']);  
  98.           
  99.         $this->setParameter('nonce_str'$this->getRand(30, 3));  
  100.           
  101.         $postXml = $this->_createXml();  
  102.           
  103.         if (! $postXml) {  
  104.             return false;  
  105.         }  
  106.         $this->log($postXml'SEND_XML');  
  107.           
  108.         $result = $this->curl_post_ssl(self::TRANSFERS_URL, $postXml);  
  109.           
  110.         $this->log($result'RESULT_XML');  
  111.           
  112.         if (! $result) {  
  113.             return false;  
  114.         }  
  115.         $resultObj = simplexml_load_string($result'SimpleXMLElement', LIBXML_NOCDATA);  
  116.           
  117.         $this->_lastResult = $resultObj;  
  118.           
  119.         if ($resultObj->return_code == 'SUCCESS') { // 成功标识  
  120.                   
  121.             if ($resultObj->result_code == 'SUCCESS') {  
  122.           
  123.                 return $resultObj->send_listid;  
  124.             }  
  125.                   
  126.             if ($resultObj->return_msg) {  
  127.                 $this->error = (string) $resultObj->return_msg;  
  128.                 return false;  
  129.             }  
  130.                   
  131.             $this->error = (string) $resultObj->err_code_des;  
  132.             return false;  
  133.         }  
  134.           
  135.         if ($resultObj->return_code != 'FAIL') {  
  136.             $this->error = '返回信息格式异常';  
  137.             return false;  
  138.         }  
  139.           
  140.         $this->error = (string) $resultObj->return_msg;  
  141.         return false;  
  142.     }  
  143.       
  144.     /** 
  145.      * 获取转账信息 
  146.      * @param unknown $partner_trade_no 
  147.      * @return boolean|SimpleXMLElement 
  148.      */  
  149.     public function getInfo($partner_trade_no){  
  150.         $param = array(  
  151.             'nonce_str' => $this->getRand(30, 3),  
  152.             'partner_trade_no'=> $partner_trade_no ,  
  153.             'mch_id'    => $this->_keys['mch_id'],  
  154.             'appid'     => $this->_keys['wxappid'],  
  155.         );  
  156.           
  157.         ksort($param);  
  158.         $unSignParaString = $this->_formatQueryParaMap($param, false);  
  159.         $param['sign'] = $this->_sign($unSignParaString$this->_keys['key']);  
  160.           
  161.         $xml = $this->arrayToXml($param);  
  162.           
  163.         $this->log($xml'GETINFO_XML');  
  164.           
  165.         $result = $this->curl_post_ssl(self::GETINFO_URL, $xml);  
  166.           
  167.         if(!$result){  
  168.             return false ;  
  169.         }  
  170.           
  171.         $this->log($result'RESULT_XML');  
  172.           
  173.         $resultObj = simplexml_load_string($result'SimpleXMLElement', LIBXML_NOCDATA);  
  174.         $this->_lastResult = $resultObj ;  
  175.         if($resultObj->return_code == 'SUCCESS'){//成功标识  
  176.           
  177.             if($resultObj->result_code == 'SUCCESS'){  
  178.                 return $resultObj ;  
  179.             }  
  180.           
  181.             if($resultObj->return_msg){  
  182.                 $this->error = $resultObj->return_msg ;  
  183.                 return false ;  
  184.             }  
  185.           
  186.             $this->error = $resultObj->err_code_des ;  
  187.             return false ;  
  188.         }  
  189.           
  190.         if($resultObj->return_code != 'FAIL'){  
  191.             $this->error = '返回信息格式异常';  
  192.             return false ;  
  193.         }  
  194.           
  195.         $this->error = $resultObj->return_msg ;  
  196.         return false ;  
  197.     }  
  198.     /** 
  199.      * 设置所需要的参数 
  200.      * @param  $parameter 键值数组/键 
  201.      * @param  $value 值 
  202.      * @return WxBonusApi 
  203.      */  
  204.     public function setParameter($parameter$value = null)  
  205.     {  
  206.         if (! is_array($parameter)) {  
  207.             return $this->setParameter(array(  
  208.                 $parameter => $value  
  209.             ));  
  210.         }  
  211.       
  212.         foreach ($parameter as $key => $value) {  
  213.             $key = trim($key);  
  214.             $value = trim($value);  
  215.             $this->_parameters[$key] = $value;  
  216.         }  
  217.         return $this;  
  218.     }  
  219.       
  220.     /** 
  221.      * 获取参数值 
  222.      * @param  $parameter 键名 
  223.      * @return multitype: 
  224.      */  
  225.     public function getParameter($parameter)  
  226.     {  
  227.         return $this->_parameters[$parameter];  
  228.     }  
  229.       
  230.     /** 
  231.      * 获取随机数 
  232.      * @param number $len 随机数的位数 
  233.      * @param number $type 取值范围 1表示数字 2小写字母 4大写字母 
  234.      * @return string 
  235.      */  
  236.     public function getRand($len = 30, $type = 0)  
  237.     {  
  238.         $str = '';  
  239.         $max = - 1;  
  240.       
  241.         if (! $type) {  
  242.             $type = 3;  
  243.         }  
  244.       
  245.         if ($type & 1) {  
  246.             $str .= '1234567890';  
  247.             $max += 10;  
  248.         }  
  249.       
  250.         if ($type & 2) {  
  251.             $str .= 'abcdefghijklmnopqrstuvwxyz';  
  252.             $max += 26;  
  253.         }  
  254.       
  255.         if ($type & 4) {  
  256.             $str .= 'ABCDEFGHIJKLMNOPQRSTUVWXYZ';  
  257.             $max += 26;  
  258.         }  
  259.       
  260.         $rand = '';  
  261.         for ($i = 0; $i < $len$i ++) {  
  262.             $rand .= $str[rand(0, $max)];  
  263.         }  
  264.       
  265.         return $rand;  
  266.     }  
  267.       
  268.     /** 
  269.      * 生成商户的订单号 
  270.      * @return string 
  271.      */  
  272.     public function getPartnerTradeNo()  
  273.     {  
  274.         $this->_lastPartnerTradeNo = $this->_parameters['mch_id'] . date('YmdHis') . $this->getRand(4, 1); // $this->getRandNum();  
  275.         return $this->_lastPartnerTradeNo;  
  276.     }  
  277.       
  278.       
  279.     /** 
  280.      * 获取最后一次创建生成的订单号 
  281.      * @return string 
  282.      */  
  283.     public function getLastPartnerTradeNo()  
  284.     {  
  285.         return $this->_lastPartnerTradeNo;  
  286.     }  
  287.       
  288.       
  289.       
  290.     /** 
  291.      * 创建XML的方法 
  292.      * @param number $retcode 
  293.      * @param string $reterrmsg 
  294.      * @return boolean|string 
  295.      */  
  296.     private function _createXml()  
  297.     {  
  298.         try {  
  299.             $sign = $this->_getSign();  
  300.             if (! $sign) {  
  301.                 return false;  
  302.             }  
  303.             $this->setParameter('sign'$sign);  
  304.                   
  305.             return $this->arrayToXml($this->_parameters);  
  306.         } catch (Exception $e) {  
  307.             $this->error = $e->getMessage();  
  308.             return false;  
  309.         }  
  310.     }  
  311.       
  312.       
  313.     /** 
  314.      * 参数转换成XML 
  315.      * @param array $arr 参数数组 
  316.      * @return string 
  317.      */  
  318.     public function arrayToXml($arr)  
  319.     {  
  320.         $xml = "";  
  321.         foreach ($arr as $key => $val) {  
  322.             if (is_numeric($val)) {  
  323.                 $xml .= "<" . $key . ">" . $val . " . $key . ">";  
  324.             } else {  
  325.                 $xml .= "<" . $key . "> . $val . "]]> . $key . ">";  
  326.             }  
  327.         }  
  328.         $xml .= "";  
  329.         return $xml;  
  330.     }  
  331.       
  332.     /** 
  333.      * 获得签名结果 
  334.      * @return boolean|Ambigous  
  335.      */  
  336.     protected function _getSign()  
  337.     {  
  338.         try {  
  339.                   
  340.             if ($this->_checkSign() == false) { // 检查生成签名参数  
  341.                 $this->error = '生成签名参数缺失!';  
  342.                 $this->log(json_encode($this->_parameters, JSON_UNESCAPED_UNICODE), 'ERROR_Sign_XML');  
  343.                 return false;  
  344.             }  
  345.                   
  346.             ksort($this->_parameters);  
  347.             $unSignParaString = $this->_formatQueryParaMap($this->_parameters, false);  
  348.                   
  349.             return $this->_sign($unSignParaString$this->_keys['key']);  
  350.         } catch (Exception $e) {  
  351.             $this->error = $e->getMessage();  
  352.             return false;  
  353.         }  
  354.     }  
  355.       
  356.     /** 
  357.      * 检查签名所需参数是否齐全 
  358.      * @return boolean 
  359.      */  
  360.     private function _checkSign()  
  361.     {  
  362.         // return true;       
  363.         if ($this->_parameters["mch_appid"] == null ||  
  364.             $this->_parameters["mchid"] == null ||  
  365.             //$this->_parameters["device_info"] == null || 设备id  
  366.             $this->_parameters["nonce_str"] == null ||  
  367.             $this->_parameters["partner_trade_no"] == null ||  
  368.             $this->_parameters["openid"] == null ||  
  369.             $this->_parameters["check_name"] == null ||  
  370.             $this->_parameters["re_user_name"] == null ||  
  371.             $this->_parameters["desc"] == null ||  
  372.             $this->_parameters["spbill_create_ip"] == null) {  
  373.                 return false;  
  374.             }  
  375.             return true;  
  376.     }  
  377.       
  378.     /** 
  379.      * 
  380.      * @param  $paraMap 
  381.      * @param  $urlencode 
  382.      * @return string 
  383.      */  
  384.     private function _formatQueryParaMap($paraMap,$urlencode)  
  385.     {  
  386.         $buff = "";  
  387.         ksort($paraMap);  
  388.         foreach ($paraMap as $k => $v) {  
  389.             if (null != $v && "null" != $v && "sign" != $k) {  
  390.                 if ($urlencode) {  
  391.                     $v = urlencode($v);  
  392.                 }  
  393.                 $buff .= $k . "=" . $v . "&";  
  394.             }  
  395.         }  
  396.         $reqPar;  
  397.         if (strlen($buff) > 0) {  
  398.             $reqPar = substr($buff, 0, strlen($buff) - 1);  
  399.         }  
  400.         return $reqPar;  
  401.     }  
  402.       
  403.       
  404.     /** 
  405.      * 签名 
  406.      * @param $content 签名的字符串 
  407.      * @param $key 密钥 
  408.      * @throws Exception 
  409.      * @return string|boolean 
  410.      */  
  411.     private function _sign($content$key)  
  412.     {  
  413.         try {  
  414.             if (null == $key) {  
  415.                 $this->error = '签名key不能为空!';  
  416.                 return false;  
  417.             }  
  418.             if (null == $content) {  
  419.                 $this->error = '签名内容不能为空';  
  420.                 return false;  
  421.             }  
  422.             $signStr = $content . "&key=" . $key;  
  423.       
  424.             return strtoupper(md5($signStr));  
  425.                   
  426.         } catch (Exception $e) {  
  427.             $this->error = $e->getMessage();  
  428.             return false;  
  429.         }  
  430.     }  
  431.       
  432.     /** 
  433.      * cURL抓取 
  434.      * 
  435.      * @param $url 链接地址 
  436.      * @param $vars 参数 
  437.      * @param 
  438.      *          $second 
  439.      * @param 
  440.      *          $aHeader 
  441.      * @return mixed|boolean 
  442.      */  
  443.     function curl_post_ssl($url$data$second = 30, $aHeader = array())  
  444.     {  
  445.         $ch = curl_init();  
  446.         // 超时时间  
  447.         curl_setopt($ch, CURLOPT_TIMEOUT, $second);  
  448.         curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);  
  449.         // 这里设置代理,如果有的话  
  450.         curl_setopt($ch, CURLOPT_URL, $url);  
  451.         curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);  
  452.         curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);  
  453.         // cert 与 key 分别属于两个.pem文件  
  454.         curl_setopt($ch, CURLOPT_SSLCERT, $this->_cert['api_cert']);  
  455.         curl_setopt($ch, CURLOPT_SSLKEY, $this->_cert['api_key']);  
  456.         curl_setopt($ch, CURLOPT_CAINFO, $this->_cert['rootca']);  
  457.         if (count($aHeader) >= 1) {  
  458.             curl_setopt($ch, CURLOPT_HTTPHEADER, $aHeader);  
  459.         }  
  460.         curl_setopt($ch, CURLOPT_POST, 1);  
  461.         curl_setopt($ch, CURLOPT_POSTFIELDS, $data);  
  462.         $data = curl_exec($ch);  
  463.         if ($data) {  
  464.             curl_close($ch);  
  465.             return $data;  
  466.         } else {  
  467.             $this->log(json_encode($this->_cert));  
  468.             $this->error = 'aa:'.curl_errno($ch);  
  469.             curl_close($ch);  
  470.             return false;  
  471.         }  
  472.     }  
  473.       
  474.     /** 
  475.      * 获取服务器ip 
  476.      * 
  477.      * @return string 
  478.      */  
  479.     public function getServerIp()  
  480.     {  
  481.         $server_ip = '127.0.0.1';  
  482.         if (isset($_SERVER)) {  
  483.             if (isset($_SERVER['SERVER_ADDR']) && $_SERVER['SERVER_ADDR']) {  
  484.                 $server_ip = $_SERVER['SERVER_ADDR'];  
  485.             } elseif (isset($_SERVER['LOCAL_ADDR']) && $_SERVER['LOCAL_ADDR']) {  
  486.                 $server_ip = $_SERVER['LOCAL_ADDR'];  
  487.             }  
  488.         } else {  
  489.             $server_ip = getenv('SERVER_ADDR');  
  490.         }  
  491.         return $server_ip;  
  492.     }  
  493.       
  494.     /** 
  495.      * 设置日志目录文件 
  496.      * 
  497.      * @param unknown $file 
  498.      */  
  499.     public function setLogFile($file)  
  500.     {  
  501.         $this->log_file = $file;  
  502.     }  
  503.       
  504.     /** 
  505.      * 写日志 
  506.      * 
  507.      * @param $msg 写入的信息 
  508.      * @param $type 日志类型作为查询标示 
  509.      */  
  510.     public function log($msg$type)  
  511.     {  
  512.         if ($this->log_file) {  
  513.             $log = str_replace(array(  
  514.                 "\r\n",  
  515.                 "\r",  
  516.                 "\n"  
  517.             ), array(  
  518.                 "",  
  519.                 "",  
  520.                 ""  
  521.             ), $msg);  
  522.             error_log($type . ' ' . date('Y-m-d H:i:s') . ' ' . json_encode($log,JSON_UNESCAPED_UNICODE) . "\r\n", 3, $this->log_file);  
  523.         }  
  524.     }  
  525.       
  526. }  

[php]  view plain  copy
[php]  view plain  copy
  1. include 'WxTransfers.Api.php';  
  2. class WxTransfers{  
  3.     /**  
[php]  view plain  copy
  1. "white-space:pre">    *调用方法即可测试  
[php]  view plain  copy
  1. "white-space:pre">    */  
  2.     public function index(){  
  3.           
  4.         $path = WxTransfersConfig::getRealPath(); // 证书文件路径  
  5.         $config['wxappid'] = WxTransfersConfig::APPID;  
  6.         $config['mch_id'] = WxTransfersConfig::MCHID;  
  7.         $config['key'] = WxTransfersConfig::KEY;  
  8.         $config['PARTNERKEY'] = WxTransfersConfig::KEY;  
  9.         $config['api_cert'] = $path . WxTransfersConfig::SSLCERT_PATH;  
  10.         $config['api_key'] = $path . WxTransfersConfig::SSLKEY_PATH;  
  11.         $config['rootca'] = $path . WxTransfersConfig::SSLROOTCA;  
  12.           
  13.         $wxtran=new WxTransfers($config);  
  14.           
  15.         $wxtran->setLogFile('D:\\transfers.log');//日志地址  
  16.           
  17.         //转账  
  18.         $data=array(  
  19.             'openid'=>'',//openid  
  20.             'check_name'=>'NO_CHECK',//是否验证真实姓名参数  
  21.             're_user_name'=>'11',//姓名  
  22.             'amount'=>100,//最小1元 也就是100  
  23.             'desc'=>'企业转账测试',//描述  
  24.             'spbill_create_ip'=>$wxtran->getServerIp(),//服务器IP地址  
  25.         );  
  26.         var_dump(json_encode($wxtran->transfers($data),JSON_UNESCAPED_UNICODE));  
  27.         var_dump($wxtran->error);  
  28.   
  29.         //获取转账信息  
  30.         var_dump($wxtran->getInfo('11111111'));  
  31.         var_dump($wxtran->error);  
  32.     }  
  33.       
  34. }  

你可能感兴趣的:(微信企业转账【完整DEMO代码】)