php开发微信小程序退款(原路返回)laravel

退款接口文档:https://pay.weixin.qq.com/wiki/doc/api/wxa/wxa_api.php?chapter=9_4

条件:laravel框架

1.根据订单号或者支付号,即可进行退款
2.退款金额可以少于支付金额(扣除手续费什么的)
3.退款需要证书(两种方式:windows将apiclient_cert.p12执行一遍即可导入系统;其他系统需要在代码中引入另外2个.pem文件;建议都使用引入文件,比较方便);证书获取说明https://pay.weixin.qq.com/wiki/doc/api/wxa/wxa_api.php?chapter=4_3

注:linux遇上一个bug,正反斜杠写法导致的错误;
原来的引入路径为"app_path().'\Libs\common\wx_cert\apiclient_key.pem';//证书路径",这个写法在linux上会导致postXmlSSLCurl()函数返回400错误;
改为正确写法:"app_path().'/Libs/common/wx_cert/apiclient_cert.pem";//证书路径

以下为退款类

<?php
namespace App\Libs\common;
class WeixinRefund {


    function __construct($openid,$appid,$mch_id,$key,$outTradeNo,$totalFee,$outRefundNo,$refundFee){
        //初始化退款类需要的变量
        $this->openid = $openid;//openid
        $this->APPID = $appid;
        $this->MCHID = $mch_id;
        $this->key = $key;
        $this->outTradeNo = $outTradeNo;//订单号order_sn
        $this->totalFee = $totalFee;//订单总金额
        $this->outRefundNo = $outRefundNo;//退款单号
        $this->refundFee = $refundFee;//需要退款的金额
        $this->SSLCERT_PATH = app_path().'/Libs/common/wx_cert/apiclient_cert.pem';//证书路径
        $this->SSLKEY_PATH = app_path().'/Libs/common/wx_cert/apiclient_key.pem';//证书路径
        
        
    }
    public function refund(){
    
        //对外暴露的退款接口
        $result = $this->wxrefundapi();
        return $result;
    }
    private function wxrefundapi(){
        //通过微信api进行退款流程
        $parma = array(
            'appid'=> $this->APPID,
            'mch_id'=> $this->MCHID,
            'nonce_str'=> $this->createNoncestr(),
            'out_refund_no'=> $this->outRefundNo,
            'out_trade_no'=> $this->outTradeNo,
            'total_fee'=> floatval(($this->totalFee) * 100),
            'refund_fee'=> floatval(($this->refundFee) * 100),
      
        );

        $parma['sign'] = $this->getSign($parma);
        $xmldata = $this->arrayToXml($parma);
        
        $xmlresult = $this->postXmlSSLCurl($xmldata,'https://api.mch.weixin.qq.com/secapi/pay/refund');

        //echo '<prE>';var_dump($xmlresult);die;
        $result = $this->xmlToArray($xmlresult);
        return $result;
    }
//需要使用证书的请求
    function postXmlSSLCurl($xml,$url,$second=30)
    {
        $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);
        //设置证书
        //使用证书:cert 与 key 分别属于两个.pem文件
        //默认格式为PEM,可以注释
        curl_setopt($ch, CURLOPT_SSLCERTTYPE, 'PEM');
        curl_setopt($ch, CURLOPT_SSLCERT, $this->SSLCERT_PATH);
        //默认格式为PEM,可以注释
        curl_setopt($ch, CURLOPT_SSLKEYTYPE, 'PEM');
        curl_setopt($ch, CURLOPT_SSLKEY, $this->SSLKEY_PATH);
        //post提交方式
        curl_setopt($ch, CURLOPT_POST, true);
        curl_setopt($ch, CURLOPT_POSTFIELDS, $xml);
        $data = curl_exec($ch);
        //返回结果
        if ($data) {
            curl_close($ch);
            return $data;
        } else {
            $error = curl_errno($ch);
            echo "curl出错,错误码:$error" . "<br>";
            curl_close($ch);
            return false;
        }
    }
    
    
    

    //作用:生成签名
    private function getSign($Obj) {
        foreach ($Obj as $k => $v) {
            $Parameters[$k] = $v;
        }
        //签名步骤一:按字典序排序参数
        ksort($Parameters);
        $String = $this->formatBizQueryParaMap($Parameters, false);
        //签名步骤二:在 string 后加入 KEY
        $String = $String . "&key=" . $this->key;
        //签名步骤三:MD5 加密
        $String = md5($String);
        //签名步骤四:所有字符转为大写
        $result_ = strtoupper($String);
        return $result_;
    }
    
    
    ///作用:格式化参数,签名过程需要使用
    private function formatBizQueryParaMap($paraMap, $urlencode) {
        $buff = "";
        ksort($paraMap);
        foreach ($paraMap as $k => $v) {
            if ($urlencode) {
                $v = urlencode($v);
            }
            $buff .= $k . "=" . $v . "&";
        }
        $reqPar = '';
        if (strlen($buff) > 0) {
            $reqPar = substr($buff, 0, strlen($buff) - 1);
        }
        return $reqPar;
    }



    //作用:产生随机字符串,不长于 32 位
    private function createNoncestr($length = 32) {
        $chars = "abcdefghijklmnopqrstuvwxyz0123456789";
        $str = "";
        for ($i = 0; $i < $length; $i++) {
            $str .= substr($chars, mt_rand(0, strlen($chars) - 1), 1);
        }
        return $str;
    }


    //数组转换成 xml
    private function arrayToXml($arr) {
    
        $xml = "<xml>
                   <appid>".$arr['appid']."</appid>
                   <mch_id>".$arr['mch_id']."</mch_id>
                   <nonce_str>".$arr['nonce_str']."</nonce_str> 
                   <out_refund_no>".$arr['out_refund_no']."</out_refund_no>
                   <out_trade_no>".$arr['out_trade_no']."</out_trade_no>
                   <refund_fee>".$arr['refund_fee']."</refund_fee>
                   <total_fee>".$arr['total_fee']."</total_fee>
                   <transaction_id></transaction_id>
                   <sign>".$arr['sign']."</sign>
                </xml>";
        
   /*     $xml = "";
        foreach ($arr as $key => $val) {
            if (is_array($val)) {
                $xml .= "<" . $key . ">" . arrayToXml($val) . "</" . $key . ">";
            } else {
                $xml .= "<" . $key . ">" . $val . "</" . $key . ">";
            }
        }
        $xml .= "</xml>";
        */
        //echo($xml);die;
        return $xml;
    }


    //xml 转换成数组
    private function xmlToArray($xml) {
        //禁止引用外部 xml 实体
        libxml_disable_entity_loader(true);
        $xmlstring = simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA);
        $val = json_decode(json_encode($xmlstring), true);
        return $val;
    }
}

调用该类:

public function wx_refund($order_id,$price){

      ////此处省略了获取参数的相关代码

        //调用退款接口(openid/订单号/订单金额/退款号/退款金额 注:这里订单号=退款号,订单实付金额=退款金额
        $wx_refund = new WeixinRefund($openid,$config['c_appid'],$mch['c_appid'],$mch['c_key'], $order['t_order_number'], $totalFee, $order['t_order_number'], $refundFee);
        $return = $wx_refund->refund();//返回值为数组格式

        $return['order_id'] = $order_id;

        $this->log_add('退款返回值',$return);//写入本地日志,使用的是laravel自带的Logger日志记录函数

        if ($return['result_code'] == 'SUCCESS') {//支付成功处理订单信息(更新订单支付状态
            return true;
        }else{
            return false;
        }
    }

以下为退款成功时的返回值,作参考使用

{
    ["return_code"] => string(7)"SUCCESS"
    ["return_msg"] => string(2)"OK" 
    ["appid"] => string(18)"wxafe501b449dcae9a" 
    ["mch_id"] => string(10)"1568688591" 
    ["nonce_str"] => string(16)"QrTSO1nsbUm85lav" 
    ["sign"] => string(32)"0475BBBECB3E0EA44CE2BF0DC87948F3" 
    ["result_code"] => string(7)"SUCCESS" 
    ["transaction_id"] => string(28)"4200000455202001013871227500"
    ["out_trade_no"] => string(32)"20200101175340157787242047007765" 
    ["out_refund_no"] => string(32)"20200101175340157787242047007765"
    ["refund_id"] => string(29)"50000403182020010113932540592" 
    ["refund_channel"] => array(0) {}
   ["refund_fee"] => string(1)"1" 
    ["coupon_refund_fee"] => string(1)"0" 
    ["total_fee"] => string(1)"1" 
    ["cash_fee"] => string(1)"1"
    ["coupon_refund_count"] => string(1)"0" 
    ["cash_refund_fee"] => string(1)"1"
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,670评论 5 460
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,928评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,926评论 0 320
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,238评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,112评论 4 356
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,138评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,545评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,232评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,496评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,596评论 2 310
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,369评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,226评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,600评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,906评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,185评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,516评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,721评论 2 335

推荐阅读更多精彩内容

  • 关于微信支付 1. 生活中的微信支付   目前我们日常生活中接触得比较多的线上电子支付方式主要有两种,一种是支付宝...
    小流歌_阅读 581评论 0 0
  • 最近一直在加班做项目,好久没有更新文章了,这段时间也遇到了不少问题,今天要分享的主要是django开发微信...
    Code人生阅读 2,909评论 2 8
  • 基于PHP的微信支付教程 微信支付作为各大移动支付方式之一,本课程只要向大家介绍并使用微信支付的常用功能,进而集合...
    獨愛凌亂中那抹獨特的華麗阅读 8,538评论 0 7
  • 2019年12月26日 文档: https://pay.weixin.qq.com/wiki/doc/api/wx...
    wangyu2488阅读 1,320评论 0 1
  • 我是黑夜里大雨纷飞的人啊 1 “又到一年六月,有人笑有人哭,有人欢乐有人忧愁,有人惊喜有人失落,有的觉得收获满满有...
    陌忘宇阅读 8,503评论 28 53