PHP 小程序支付类

<?php

namespace app\api\controller;
use think\Controller;
use think\Db;
class Payment extends Controller {
    
    static public $treeList = array();
    public function initialize(){
        parent::initialize();


        $this->config['appid'] ='小程序的APPID';//$this->config['appid'];
        $this->config['mch_id'] = '商户号';
        $this->config['api_key'] = '小程序密钥';
        
    }
    
    public function notify()
    {

        $postXml = $GLOBALS["HTTP_RAW_POST_DATA"];
        //禁止引用外部xml实体   
        libxml_disable_entity_loader(true);
        $xmlstring = simplexml_load_string($postXml, 'SimpleXMLElement', LIBXML_NOCDATA);
        $attr = json_decode(json_encode($xmlstring), true);
        file_put_contents('jiesi.txt', var_export($attr,true));
        if ($attr['result_code'] == 'SUCCESS' && $attr['return_code'] == 'SUCCESS') {
            $order_sn = $attr['out_trade_no'];
            
            db("costlog")->where(['order'=>$order_sn])->update(['status'=>1]);
            
            
            //db("zls")->insert()
            
            if ($res) {
                $return = ['return_code' => 'SUCCESS', 'return_msg' => 'OK'];
                $xml = '<xml>';
                foreach ($return as $k => $v) {
                    $xml .= '<' . $k . '><![CDATA[' . $v . ']]></' . $k . '>';
                }
                $xml .= '</xml>';
                return '<xml><return_code>SUCCESS</return_code><return_msg>OK</return_msg></xml>';
            }
        }
    
//     public function pay(){
//         $rrs= $this->getWXPayInfo('ddd',time(),1,'oLYxK5LaSbJWxwqtzhLLY6KJDHuw');
//         exit(json_encode($rrs));
//     }

    /*
    $body 内容 
    $out_trade_no 订单号
    $total_fee 价格
    $open_id openid
     */
    public function getWXPayInfo($body, $out_trade_no, $total_fee, $open_id){
        $response = $this->getPrePayOrder($body, $out_trade_no, $total_fee, $open_id);
        // return $response;
        if ($response['return_code'] == 'FAIL') {
            return ajaxreturn($response['return_msg'],2);
            // return $response;
        }
        $x = $this->getOrder($response['prepay_id']);
        $data1['wdata'] = $x;
        // $data1['pay_money'] = $order['order_amount'];
        return $data1;
    }


    public function getPrePayOrder($body, $out_trade_no, $total_fee, $open_id)
    {
    
            
        $url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
        //$notify_url = $this->config["notify_url"];

        $onoce_str = $this->getRandChar(32);
        $data["appid"] = $this->config['appid'];
        $data["body"] = $body;
        $data["mch_id"] = $this->config['mch_id'];//商户ID
        $data["nonce_str"] = $onoce_str;//32位随机数
        $data["notify_url"] =  'https://'.$_SERVER['HTTP_HOST'].'/api/payment/notify';//回调地址
        $data["out_trade_no"] = $out_trade_no;//订单编号
        $data["spbill_create_ip"] = $this->get_client_ip();//IP地址
        $data["total_fee"] = $total_fee;//订单价格
        $data["trade_type"] = "JSAPI";
        $data["openid"] = $open_id;
        $s = $this->getSign($data, false);
        $data["sign"] = $s;
        // return $data;
        $xml = $this->arrayToXml($data);
        $response = $this->postXmlCurl($xml, $url);
    
        // 将微信返回的结果xml转成数组
        return $this->xmlstr_to_array($response);
    }

    // 执行第二次签名,才能返回给客户端使用
    public function getOrder($prepayId)
    {
    
        $data["appId"] = $this->config['appid'];
        $data["nonceStr"] = $this->config['mch_id'];
        $data["package"] = "prepay_id=" . $prepayId;
        $data['signType'] = "MD5";
        $data["timeStamp"] = time();
    

        $s = $this->getSign1($data, false);
        $data["sign"] = $s;
        return $data;
    }
  
// function payment()
//  {
//      $postXml = $GLOBALS["HTTP_RAW_POST_DATA"];
//      //禁止引用外部xml实体   
//      libxml_disable_entity_loader(true);
//      $xmlstring = simplexml_load_string($postXml, 'SimpleXMLElement', LIBXML_NOCDATA);
//      $attr = json_decode(json_encode($xmlstring), true);
//      if ($attr['result_code'] == 'SUCCESS' && $attr['return_code'] == 'SUCCESS') {
//        
//          //回调
//          
//          if ($res) {
//              $return = ['return_code' => 'SUCCESS', 'return_msg' => 'OK'];
//              $xml = '<xml>';
//              foreach ($return as $k => $v) {
//                  $xml .= '<' . $k . '><![CDATA[' . $v . ']]></' . $k . '>';
//              }
//              $xml .= '</xml>';
//              return '<xml><return_code>SUCCESS</return_code><return_msg>OK</return_msg></xml>';
//          }
//      }
//
//  }
//


    // 获取指定长度的随机字符串
    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;
    }

    /*
     * 生成签名
     */
    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['api_key'];
        // echo "<textarea style='width: 50%; height: 150px;'>$String</textarea> <br />";
        // 签名步骤三:MD5加密
        $result_ = strtoupper(md5($String));
        return $result_;
    }
   
    /*
     * 生成签名
     */
    function getSign1($Obj)
    {
        foreach ($Obj as $k => $v) {
            $Parameters[strtolower($k)] = $v;
        }
        // 签名步骤一:按字典序排序参数
        ksort($Parameters);
        //$String = $this->formatBizQueryParaMap($Parameters, false);
        $String = "appId=" . $Obj['appId'] . "&nonceStr=" . $Obj['nonceStr'] . "&package=" . $Obj['package'] . "&signType=MD5&timeStamp=" . $Obj['timeStamp'];

        // echo "【string】 =".$String."</br>";
        // 签名步骤二:在string后加入KEY
        $String = $String . "&key=" . $this->config['api_key'];
        // echo "<textarea style='width: 50%; height: 150px;'>$String</textarea> <br />";
        // 签名步骤三:MD5加密
        //echo $String;
        $result_ = strtoupper(md5($String));
        return $result_;
    }

    /*
     * 获取当前服务器的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;
    }

    // 数组转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
        $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;
        }
    }

    /**
     * xml转成数组
     */
    function xmlstr_to_array($xmlstr)
    {
        //$doc = new \DOMDocument();
        //$doc->loadXML($xmlstr);
        //return $this->domnode_to_array($doc->documentElement);

        //禁止引用外部xml实体

        libxml_disable_entity_loader(true);

        $xmlstring = simplexml_load_string($xmlstr, 'SimpleXMLElement', LIBXML_NOCDATA);

        $val = json_decode(json_encode($xmlstring), true);

        return $val;


    }

    // 将数组转成uri字符串
    function formatBizQueryParaMap($paraMap, $urlencode)
    {
        $buff = "";
        ksort($paraMap);
        foreach ($paraMap as $k => $v) {
            if ($urlencode) {
                $v = urlencode($v);
            }
            $buff .= strtolower($k) . "=" . $v . "&";
        }
        $reqPar;
        if (strlen($buff) > 0) {
            $reqPar = substr($buff, 0, strlen($buff) - 1);
        }
        return $reqPar;
    }

    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;
    }
    
    

 
}

?>

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值