| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333 | <?phpnamespace app\common\library;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 : get_addon_config('epay')['wechat'];    }    /**     * 微信支付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;    }    /**     * 微信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($order_data) {        $wxConfig = get_addon_config('epay')['wechat'];        $data['appid'] = $wxConfig['app_id'];        $data['mch_id'] = $wxConfig['mch_id'];        $data['nonce_str'] = $this->getRandChar(32);;        if (isset($wxConfig['refund_notify'])) {            $data['notify_url'] = $wxConfig['refund_notify'];        }        $data['out_trade_no']  = $order_data['out_trade_no'];        $data['out_refund_no'] = $order_data['out_refund_no'];        $data['total_fee']     = $order_data['total_fee'];        $data['refund_fee']    = $order_data['refund_fee'];        $data['refund_desc']   = $order_data['refund_desc'];        $data['sign'] = $this->getSign($data);        $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, APP_PATH . '../addons/epay/certs/apiclient_key.pem');        curl_setopt($ch, CURLOPT_SSLCERT, APP_PATH . '../addons/epay/certs/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;    }}
 |