PHP微信V3支付完整版(包括微信退款回调大集合)

<?php

use think\facade\Db; //引入Db
#*注使用微信V3支付需要PHP版本7.2以上并且安装sodium扩展才能进行解密参数
#*注使用微信V3支付需要PHP版本7.2以上并且安装sodium扩展才能进行解密参数
#*注使用微信V3支付需要PHP版本7.2以上并且安装sodium扩展才能进行解密参数

class Wxpay
{
     /*
     * 微信支付统一下单
     * @$name 订单名称
     * @$ordernumber 订单单号(自己生成的)
     * @$money  金额
     * @$openid  用户微信的openid
    */
    public function wechartAddOrder($name, $ordernumber, $money, $openid)
    {
        $appid = env('WECHAT.appid'); // 微信APPID
        $mchid = env('WECHAT.mch_id'); // 商户号
        $xlid = env('WECHAT.serial_number');//API序列号

        $out_trade_no = order_id();//生成支付单号(和订单号不一样)这样做是保证同一个订单可以被多次进行支付,因为每次支付单号不一样
        $url = "https://api.mch.weixin.qq.com/v3/pay/transactions/jsapi";
        $urlarr = parse_url($url);
        $data = array();
        $randstr = getrandstr(16);//随机字符串长度不超过32(加密使用的)
        $time = time();
        $data['appid'] = $appid;
        $data['mchid'] = $mchid;
        $data['description'] = $name;//商品描述
        $data['attach'] = $ordernumber;//订单编号
        $data['out_trade_no'] = $out_trade_no;//支付订单编号
        $data['notify_url'] = "";//回调接口
        $data['amount']['total'] = $money;
        $data['payer']['openid'] = $openid;//用户付款的openID
        $data = json_encode($data); //转json串
        $key = $this->getSign($data, $urlarr['path'], $randstr, $time);//微信支付签名加密
        $token = sprintf('mchid="%s",serial_no="%s",nonce_str="%s",timestamp="%d",signature="%s"', $mchid, $xlid, $randstr, $time, $key);//头部信息
        $header = array(
            'Content-Type:' . 'application/json; charset=UTF-8',
            'Accept:application/json',
            'User-Agent:*/*',
            'Authorization: WECHATPAY2-SHA256-RSA2048 ' . $token
        );
        $ret = $this->curl_post_https($url, $data, $header); //发送请求进行预支付
        $ret = json_decode($ret, true);
        if (isset($ret['prepay_id'])) { //拉起预支付成功
            $res['timeStamp'] = $time;//时间戳
            $res['nonceStr'] = $randstr;//随机字符串
            $res['signType'] = 'RSA';//签名算法,暂支持 RSA
            $res['package'] = 'prepay_id=' . $ret['prepay_id'];//统一下单接口返回的 prepay_id 参数值,提交格式如:prepay_id=*
            $res['paySign'] = $this->getWechartSign($appid,$time,$res['nonceStr'],$res['package']);//签名
            return $res; //返回给前端需要的串
        }
        return $ret;
    }

    /**
     * 申请退款API
     * @$name   订单商品名称
     * @$out_trade_no   订单号
     * @$money   订单退款金额
     * @$transaction_id  微信支付成功的流水单号
     */
    public function refundOrders($name, $out_trade_no, $transaction_id,$money)
    {
        $mchid = env('WECHAT.mch_id');  // 商户号
        $xlid = env('WECHAT.serial_number'); //API序列号
        $url = "https://api.mch.weixin.qq.com/v3/refund/domestic/refunds";
        $urlarr = parse_url($url);
        $data['transaction_id'] = $transaction_id; //原支付交易对应的微信订单号。
        $data['out_refund_no'] = $out_trade_no; //退款单号
        $data['reason'] = "订单". $name ."的退款"; //退款单号
        $data['notify_url'] = ""; //退款回调接口
        $data['amount']['refund'] = $money; //退款金额(只能小于等于支付金额)
        $data['amount']['total'] = $money; //退款金额(只能小于等于支付金额)
        $data['amount']['currency'] = "CNY"; //金额类型
        $randstr = getrandstr(16); //随机字符串长度不超过32
        $time = time();
        $data = json_encode($data);
        $key = $this->getSign($data, $urlarr['path'], $randstr, $time); //签名
        $token = sprintf('mchid="%s",serial_no="%s",nonce_str="%s",timestamp="%d",signature="%s"', $mchid, $xlid, $randstr, $time, $key);//头部信息
        $header = array(
            'Content-Type:' . 'application/json; charset=UTF-8',
            'Accept:application/json',
            'User-Agent:*/*',
            'Authorization: WECHATPAY2-SHA256-RSA2048 ' . $token
        );
        $ret = $this->curl_post_https($url, $data, $header);
        $ret = json_decode($ret, true);
        return $ret;
    }

    /*
     * 统一下单之后前端需要的微信支付所需签名
     * 微信支付签名
     * */
    public function getSign($data = array(), $url, $randstr, $time)
    {
        $str = "POST" . "\n" . $url . "\n" . $time . "\n" . $randstr . "\n" . $data . "\n";
        $key = file_get_contents(root_path() . '/vendor/path/apiclient_key.pem');//在商户平台下载的秘钥
        $str = $this->getSha256WithRSA($str, $key);
        return $str;
    }

    /*
     * 统一下单调起支付的签名
     * */
    public function getWechartSign($appid, $timeStamp, $noncestr, $prepay_id)
    {
        $str = $appid . "\n" . $timeStamp . "\n" . $noncestr . "\n" . $prepay_id . "\n";
        $key = file_get_contents(root_path() . '/vendor/path/apiclient_key.pem');//在商户平台下载的秘钥
        $str = $this->getSha256WithRSA($str, $key);
        return $str;
    }
    /*
     * 签名方法
     * */
    public function getSha256WithRSA($content, $privateKey)
    {
        $binary_signature = "";
        $algo = "SHA256";
        openssl_sign($content, $binary_signature, $privateKey, $algo);
        $sign = base64_encode($binary_signature);
        return $sign;
    }
    /*
     * POST请求方法
     * PHP CURL HTTPS POST
     * */
    public function curl_post_https($url, $data, $header)
    { // 模拟提交数据函数
        $curl = curl_init(); // 启动一个CURL会话
        curl_setopt($curl, CURLOPT_URL, $url); // 要访问的地址
        curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, 0); // 对认证证书来源的检查
        curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 2); // 从证书中检查SSL加密算法是否存在
        curl_setopt($curl, CURLOPT_USERAGENT, $_SERVER['HTTP_USER_AGENT']); // 模拟用户使用的浏览器
        curl_setopt($curl, CURLOPT_FOLLOWLOCATION, 1); // 使用自动跳转
        curl_setopt($curl, CURLOPT_AUTOREFERER, 1); // 自动设置Referer
        curl_setopt($curl, CURLOPT_POST, 1); // 发送一个常规的Post请求
        curl_setopt($curl, CURLOPT_POSTFIELDS, $data); // Post提交的数据包
        curl_setopt($curl, CURLOPT_TIMEOUT, 30); // 设置超时限制防止死循环
        curl_setopt($curl, CURLOPT_HEADER, 0); // 显示返回的Header区域内容
        curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1); // 获取的信息以文件流的形式返回

        curl_setopt($curl, CURLOPT_HTTPHEADER, $header);
        $tmpInfo = curl_exec($curl); // 执行操作
        if (curl_errno($curl)) {
            echo 'Errno' . curl_error($curl);//捕抓异常
        }
        curl_close($curl); // 关闭CURL会话
        return $tmpInfo; // 返回数据,json格式
    }
    /*
    * 微信支付回调
    * */
    public function payment_notify($request)
    {
        $input_data = $request->param();
        $key = env('WECHAT.key');//商户平台设置的api v3 密码
        $text = base64_decode($input_data['resource']['ciphertext']); //解密
        /* =========== 使用V3支付需要PHP7.2.6安装sodium扩展才能进行解密参数  ================ */
        $str = sodium_crypto_aead_aes256gcm_decrypt($text, $input_data['resource']['associated_data'], $input_data['resource']['nonce'], $key);
        $res = json_decode($str, true);
        if ($res['trade_state'] == 'SUCCESS') {
            Db::startTrans();
            try {
                /* 成功操作 */
                Db::commit();
                $a = array(
                    "code" => "SUCCESS",
                    "message" => "成功"
                );
                return json_encode($a);
            } catch (\Exception $e) {
                Db::rollback();
                $a = array(
                    "code" => "ERROR",
                    "message" => "失败"
                );
                return json_encode($a);
            }
        }
        $a = array(
            "code" => "ERROR",
            "message" => "失败"
        );
        return json_encode($a);
    }

    /*
     * 微信退款回调
     * */
    public function refund_notify($request)
    {
        $input_data = $request->param();
        $key = env('WECHAT.key');//商户平台设置的api v3 密码
        $text = base64_decode($input_data['resource']['ciphertext']);
        /* =========== 使用V3支付需要PHP7.2.6安装sodium扩展才能进行解密参数  ================ */
        $str = sodium_crypto_aead_aes256gcm_decrypt($text, $input_data['resource']['associated_data'], $input_data['resource']['nonce'], $key);
        $res = json_decode($str, true);
        if ($res['refund_status'] == 'SUCCESS') {
            Db::startTrans();
            try {
                /*回调处理逻辑内容*/
                Db::commit();
                $a = array(
                    "code" => "SUCCESS",
                    "message" => "成功"
                );
                return json_encode($a);
            } catch (\Exception $e) {
                Db::rollback();
                $a = array(
                    "code" => "ERROR",
                    "message" => "失败"
                );
                return json_encode($a);
            }
        }
        $a = array(
            "code" => "ERROR",
            "message" => "失败"
        );
        return json_encode($a);
    }
}
  • 3
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,关于微信支付信息的解密,你可以按照以下步骤进行: 1. 从微信支付平台获取到的原始加密信息,包括加密的文本信息、加密的随机串、加密的签名信息等。 2. 使用商户平台的私钥对加密的随机串进行解密,得到对称加密的密钥。 3. 使用对称加密的密钥对加密的文本信息进行解密,得到明文的信息。 4. 对明文的信息进行验签,确保信息的整性和可信度。 下面是一个示例代码,可以帮助你微信支付信息的解密和验签过程: ```php /** * 微信支付信息解密和验签示例代码 */ // 从微信支付平台获取的原始加密信息 $xml = file_get_contents('php://input'); // 从商户平台获取私钥 $private_key = file_get_contents('/path/to/your/private/key.pem'); // 解析信息中的加密文本、随机串和签名信息 $data = simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA); $encrypt_data = base64_decode($data->Encrypt); $nonce = $data->Nonce; $signature = $data->Signature; // 使用商户平台的私钥解密随机串,得到对称加密的密钥 $rand_str = openssl_decrypt($nonce, 'AES-256-ECB', $private_key, OPENSSL_RAW_DATA); $key = substr($rand_str, 0, 32); // 使用对称加密的密钥解密加密文本,得到明文的信息 $iv = substr($rand_str, 32); $decrypted = openssl_decrypt($encrypt_data, 'AES-256-CBC', $key, OPENSSL_RAW_DATA, $iv); // 验证明文的信息的签名是否正确 $merchant_id = 'YOUR_MERCHANT_ID'; $timestamp = $data->TimeStamp; $msg_signature = $data->MsgSignature; $signature_params = [$merchant_id, $timestamp, $nonce, $decrypted]; sort($signature_params, SORT_STRING); $signature_str = implode($signature_params); $signature = sha1($signature_str); if ($signature === $msg_signature) { // 验签通过,对明文的信息进行处理 $decoded = json_decode($decrypted); // TODO: 处理信息 } else { // 验签失败,信息可能被篡改,不予处理 header('HTTP/1.1 400 Bad Request'); echo 'Invalid signature'; } ``` 以上代码仅供参考,具体实现方式可能会有所不同,需要根据自己的实际情况进行整。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值