123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454 |
- <?php
- namespace wxpay;
- /**
- * 微信插件 使用需 require_once("./Plugins/WxPay/WxPay.php");
- * @author Jack_YanTC <627495692@qq.com>
- */
- class WxPay {
- /**
- * 初始化参数
- *
- * @param array $options
- * @param $options ['app_id'] APPID:绑定支付的APPID(必须配置,开户邮件中可查看)
- * @param $options ['mch_id'] MCHID:商户号(必须配置,开户邮件中可查看)
- * @param $options ['key'] KEY:商户支付密钥,参考开户邮件设置(必须配置,登录商户平台自行设置)
- * @param $options ['appsecret'] 公众帐号secert(仅JSAPI支付的时候需要配置),
- * @param $options ['notify_url'] 支付宝回调地址
- */
- public function __construct($options = array()) {
- $this->config = !empty($options) ? $options : config('wxchatpay');
- }
- /**
- * 微信支付App
- * @param string $data 业务参数 body out_trade_no total_fee
- * @param string $data ['out_trade_no'] 订单号 必填
- * @param string $data ['total_fee'] 订单金额 必填
- * @param string $data ['body'] 订单详情 必填
- * @return $response 返回app所需字符串
- */
- public function WxPayApp($d) {
- $wxConfig = $this->config;
- $out_trade_no = $d['out_trade_no'];
- $total_fee = abs(floatval($d['total_fee'])) * 100;// 微信支付 单位为分
- $nonce_str = $this->getRandChar(32);
- $ip = $this->get_client_ip();
- if ($ip == '::1')
- $ip = '1.1.1.1';
- $data ["appid"] = $wxConfig["app_id"];
- $data ["body"] = $d['body'];
- $data ["mch_id"] = $wxConfig['mch_id'];
- $data ["nonce_str"] = $nonce_str;
- $data ["notify_url"] = $wxConfig["notify_url"];
- $data ["out_trade_no"] = $out_trade_no;
- $data ["spbill_create_ip"] = $ip;
- $data ["total_fee"] = $total_fee;
- $data ["trade_type"] = "APP";
- $data['time_expire'] = date('YmdHis', time() + 120);
- $s = $this->getSign($data);
- $data ["sign"] = $s;
- $xml = $this->arrayToXml($data);
- $url = 'https://api.mch.weixin.qq.com/pay/unifiedorder';
- $response = $this->postXmlCurl($xml, $url);
- $re = $this->xmlstr_to_array($response);
- if ($re ['return_code'] == 'FAIL') {
- return $re['return_msg'];
- }
- // 二次签名
- $reapp = $this->getOrder($re['prepay_id']);
- return $reapp;
- }
- public function WxPayJs($d) {
- $wxConfig = $this->config;
- $out_trade_no = $d['out_trade_no'];
- $total_fee = abs(floatval($d['total_fee'])) * 100;// 微信支付 单位为分
- // $total_fee = 1;// 微信支付 单位为分
- $nonce_str = $this->getRandChar(32);
- $ip = $this->get_client_ip();
- if ($ip == '::1')
- $ip = '1.1.1.1';
- $trade_type = 'JSAPI';
- $post = [
- 'appid' =>$wxConfig["app_id"] ,
- 'body' =>$d['body'],
- 'mch_id' =>$wxConfig['mch_id'],
- 'nonce_str' =>$nonce_str,
- 'notify_url' =>$wxConfig["notify_url"],
- 'openid' =>$d['openid'],
- 'out_trade_no' =>$out_trade_no,
- 'spbill_create_ip' =>$ip,
- 'time_expire' => date('YmdHis', time() + 120),
- 'total_fee' => $total_fee,
- 'trade_type' =>$trade_type,
- ];
- $sign = $this->sign($post);
- $post_xml = '<xml>
- <appid>'.$wxConfig["app_id"].'</appid>
- <body>'.$d['body'].'</body>
- <mch_id>'.$wxConfig['mch_id'].'</mch_id>
- <nonce_str>'.$nonce_str.'</nonce_str>
- <notify_url>'.$wxConfig["notify_url"].'</notify_url>
- <openid>'.$d['openid'].'</openid>
- <out_trade_no>'.$out_trade_no.'</out_trade_no>
- <spbill_create_ip>'.$ip.'</spbill_create_ip>
- <time_expire>'.$post["time_expire"].'</time_expire>
- <total_fee>'.$post["total_fee"].'</total_fee>
- <trade_type>'.$trade_type.'</trade_type>
- <sign>'.$sign.'</sign>
- </xml>';
- $url = 'https://api.mch.weixin.qq.com/pay/unifiedorder';
- $xml = httpRequest($url,'POST',$post_xml);
- $array = $this->xml($xml);//全要大写1
- if($array['RETURN_CODE'] == 'SUCCESS' && $array['RESULT_CODE'] == 'SUCCESS')
- {
- $time = (string)time();
- $tmp = [];
- $tmp['appId'] = $wxConfig["app_id"];
- $tmp['nonceStr'] = $nonce_str;
- $tmp['package'] = 'prepay_id='.$array['PREPAY_ID'];
- $tmp['signType'] = 'MD5';
- $tmp['timeStamp'] = $time;
- //return ajaxReturn(callback(1,'签名成功',$tmp));
- $data['state'] = 200;
- $data['appid'] = $wxConfig["app_id"];
- $data['timeStamp'] = $time;
- $data['nonceStr'] = $nonce_str;
- $data['signType'] = 'MD5';
- $data['package'] = 'prepay_id='.$array['PREPAY_ID']; //同意下单接口返回的值 prepay_id
- $data['paySign'] = $this->sign($tmp);
- $data['out_trade_no'] = $out_trade_no;
- }
- else
- {
- $data['state'] = 0;
- $data['text'] = 'error';
- $data['RETURN_CODE'] = $array['RETURN_CODE'];
- $data['RESULT_CODE'] = $array['RESULT_CODE'];
- }
- return $data;
- }
- /**
- * 微信App支付退款申请
- * @param array $d 业务参数 body out_trade_no total_fee
- * @param string $data ['out_trade_no'] 订单号 必填
- * @param string $data ['total_fee'] 订单金额 必填
- * @return $response 返回app所需字符串
- */
- public function WxPayRefund($d) {
- $wxConfig = $this->config;
- // $out_trade_no = $d['out_trade_no'];
- // $transaction_id = $d['trade_no'];
- // $total_fee = $d['total_amount'];// 微信支付 单位为分
- // $refund_reason = $d['refund_reason'];
- $nonce_str = $this->getRandChar(32);
- $data ["appid"] = $wxConfig["app_id"];
- $data ["mch_id"] = $wxConfig['mch_id'];
- $data ["nonce_str"] = $nonce_str;
- if ($wxConfig["refund_notify"]) {
- $data ["notify_url"] = $wxConfig["refund_notify"];
- }
- // $data ["out_trade_no"] = $out_trade_no;
- $data ["transaction_id"] = $d['transaction_id'];
- $data ["out_refund_no"] = $d['out_refund_no'];
- $data ["total_fee"] = $d['total_fee']; //1 微信总金额 单位为分
- $data ["refund_fee"] = $d['refund_fee']; //1 微信退款金额 单位为分
- $data ['refund_desc'] = $d['refund_desc'];
- $s = $this->getSign($data);
- $data ["sign"] = $s;
- $xml = $this->arrayToXml($data);
- $url = 'https://api.mch.weixin.qq.com/secapi/pay/refund';
- $response = $this->postXmlCurl($xml, $url);
- $re = $this->xmlstr_to_array($response);
- if ($re ['return_code'] == 'FAIL') {
- return $re['return_msg'];
- }
- return $re;
- }
- /**
- * 微信退款通知
- * @return array 验证正确返回状态及订单数据
- */
- public function WxPayRefundNotifyCheck() {
- $postStr = $GLOBALS['HTTP_RAW_POST_DATA'];
- if (!$postStr) {
- $postStr = file_get_contents("php://input");
- }
- $postObj = simplexml_load_string($postStr, 'SimpleXMLElement', LIBXML_NOCDATA);
- if ($postObj === false) {
- error_log('parse xml error', 3, './wechat_errorlog.txt');
- }
- if ($postObj->return_code != 'SUCCESS') {
- error_log($postObj->return_msg, 3, './wechat_errorlog.txt');
- }
- $arr = (array)$postObj;
- file_put_contents('wx_refund_error_logs.txt', date('Y-m-d H:i:s').'退款数据1!'.json_encode($arr), FILE_APPEND);
- if($arr['return_code'] == 'SUCCESS') {
- $wxConfig = $this->config;
- //解密信息
- require_once("Plugins/WxPay/OpenSSLAES.php");
- $aes = new \OpenSSLAES(md5($wxConfig['key']));
- $decrypted = $aes->decrypt($arr['req_info']);
- $reqObj = simplexml_load_string($decrypted, 'SimpleXMLElement', LIBXML_NOCDATA);
- $arr['req_info'] = (array)$reqObj;
- file_put_contents('wx_refund_error_logs.txt', date('Y-m-d H:i:s').'退款数据2!'.json_encode($arr), FILE_APPEND);
- return array('status' => true, 'data' => $arr);
- }
- return array('status' => false);
- }
- /**
- * 微信签名验证
- * @param string $data 业务参数
- * @return array
- */
- public function WxPayNotifyCheck() {
- // $postStr = $GLOBALS['HTTP_RAW_POST_DATA'];
- // if(!$postStr){
- $postStr = file_get_contents("php://input");
- // }
- $postObj = simplexml_load_string($postStr, 'SimpleXMLElement', LIBXML_NOCDATA);
- if ($postObj === false) {
- error_log('parse xml error', 3, './wechat_errorlog.txt');
- }
- if ($postObj->return_code != 'SUCCESS') {
- error_log($postObj->return_msg, 3, './wechat_errorlog.txt');
- }
- $arr = (array)$postObj;
- unset($arr['sign']);
- if ($this->getSign($arr) == $postObj->sign) {
- return array('status' => true, 'data' => $arr);
- } else
- return array('status' => false);
- }
- /**
- * 以下为微信所需相关方法,请勿修改
- */
- // 执行第二次签名,才能返回给客户端使用
- public function getOrder($prepayId) {
- $data ["appid"] = $this->config ["app_id"];
- $data ["noncestr"] = $this->getRandChar(32);;
- $data ["package"] = "Sign=WXPay";
- $data ["partnerid"] = $this->config ['mch_id'];
- $data ["prepayid"] = $prepayId;
- $data ["timestamp"] = time();
- $s = $this->getSign($data);
- $data ["sign"] = $s;
- return $data;
- }
- //生成签名
- function getSign($Obj) {
- foreach ($Obj as $k => $v) {
- $Parameters [strtolower($k)] = $v;
- }
- // 签名步骤一:按字典序排序参数
- ksort($Parameters);
- $String = $this->formatBizQueryParaMap($Parameters, false);
- // echo "【string】 =".$String."</br>";
- // 签名步骤二:在string后加入KEY
- $String = $String . "&key=" . $this->config ['key'];
- // echo "<textarea style='width: 50%; height: 150px;'>$String</textarea> <br />";
- // 签名步骤三:MD5加密
- $result_ = strtoupper(md5($String));
- return $result_;
- }
- // 获取指定长度的随机字符串
- function getRandChar($length) {
- $str = null;
- $strPol = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyz";
- $max = strlen($strPol) - 1;
- for ($i = 0; $i < $length; $i++) {
- $str .= $strPol [rand(0, $max)]; // rand($min,$max)生成介于min和max两个数之间的一个随机整数
- }
- return $str;
- }
- // 数组转xml
- 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;
- }
- // post https请求,CURLOPT_POSTFIELDS xml格式
- function postXmlCurl($xml, $url, $second = 30) {
- // 初始化curl
- $ch = curl_init();
- // 超时时间
- curl_setopt($ch, CURLOPT_TIMEOUT, $second);
- // 这里设置代理,如果有的话
- // curl_setopt($ch,CURLOPT_PROXY, '8.8.8.8');
- // curl_setopt($ch,CURLOPT_PROXYPORT, 8080);
- 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_SSLKEY, dirname(__FILE__) . '/cert/apiclient_key.pem');
- curl_setopt($ch, CURLOPT_SSLCERT, dirname(__FILE__) . '/cert/apiclient_cert.pem');
- // 运行curl
- $data = curl_exec($ch);
- // 返回结果
- if ($data) {
- curl_close($ch);
- return $data;
- } else {
- $error = curl_errno($ch);
- echo "curl出错,错误码:$error" . "<br>";
- echo "<a href='http://curl.haxx.se/libcurl/c/libcurl-errors.html'>错误原因查询</a></br>";
- curl_close($ch);
- return false;
- }
- }
- //获取当前服务器的IP
- function get_client_ip() {
- if ($_SERVER ['REMOTE_ADDR']) {
- $cip = $_SERVER ['REMOTE_ADDR'];
- } elseif (getenv("REMOTE_ADDR")) {
- $cip = getenv("REMOTE_ADDR");
- } elseif (getenv("HTTP_CLIENT_IP")) {
- $cip = getenv("HTTP_CLIENT_IP");
- } else {
- $cip = "unknown";
- }
- return $cip;
- }
- // 将数组转成uri字符串
- function formatBizQueryParaMap($paraMap, $urlencode) {
- $buff = "";
- ksort($paraMap);
- foreach ($paraMap as $k => $v) {
- if ($urlencode) {
- $v = urlencode($v);
- }
- $buff .= strtolower($k) . "=" . $v . "&";
- }
- if (strlen($buff) > 0) {
- $reqPar = substr($buff, 0, strlen($buff) - 1);
- }
- return $reqPar;
- }
- //xml转成数组
- function xmlstr_to_array($xmlstr) {
- $doc = new \DOMDocument ();
- $doc->loadXML($xmlstr);
- return $this->domnode_to_array($doc->documentElement);
- }
- //dom转成数组
- function domnode_to_array($node) {
- $output = array();
- switch ($node->nodeType) {
- case XML_CDATA_SECTION_NODE :
- case XML_TEXT_NODE :
- $output = trim($node->textContent);
- break;
- case XML_ELEMENT_NODE :
- for ($i = 0, $m = $node->childNodes->length; $i < $m; $i++) {
- $child = $node->childNodes->item($i);
- $v = $this->domnode_to_array($child);
- if (isset ($child->tagName)) {
- $t = $child->tagName;
- if (!isset ($output [$t])) {
- $output [$t] = array();
- }
- $output [$t] [] = $v;
- } elseif ($v) {
- $output = ( string )$v;
- }
- }
- if (is_array($output)) {
- if ($node->attributes->length) {
- $a = array();
- foreach ($node->attributes as $attrName => $attrNode) {
- $a [$attrName] = ( string )$attrNode->value;
- }
- $output ['@attributes'] = $a;
- }
- foreach ($output as $t => $v) {
- if (is_array($v) && count($v) == 1 && $t != '@attributes') {
- $output [$t] = $v [0];
- }
- }
- }
- break;
- }
- return $output;
- }
- //签名排序
- private function sign($data)
- {
- $stringA = '';
- foreach ($data as $key => $value) {
- if(!$value) continue;
- if($stringA) $stringA .= '&'.$key."=".$value;
- else $stringA = $key."=".$value;
- }
- $wx_key = $this->config['key'];
- $stringSignTemp = $stringA.'&key='.$wx_key;
- return strtoupper(md5($stringSignTemp));
- }
- //xml转换
- private function xml($xml)
- {
- $p = xml_parser_create();
- xml_parse_into_struct($p ,$xml ,$vals , $index);
- xml_parser_free($p);
- $data = [];
- foreach ($index as $key => $value) {
- if($key =='xml' || $key=='XML') continue;
- $tag = $vals[$value[0]]['tag'];
- $value = $vals[$value[0]]['value'];
- $data[$tag] = $value;
- }
- return $data;
- }
- }
|