Thinkphp6接入PayPal支付

沙盒环境示例

创建扩展封装类

<?php
namespace lib;

class PayPalApi
{
    //clientId
    private $clientId;
    //clientSecret
    private $clientSecret;
    //服务器地址
    private $host = 'https://api-m.sandbox.paypal.com/';
    //主机头
    private $headers = [];
    //api凭证
    private $token = '';
    //报文主体
    private $bodys = '';

    /**
     * 初始化类
     */
    public function __construct(array $config = [])
    {
        $this->clientId = isset($config['clientId']) ? $config['clientId'] : '';
        $this->clientSecret = isset($config['clientSecret']) ? $config['clientSecret'] : '';
        $this->getAccessToken();
    }
    /**
     * 创建订单
     */
    public function create(array $data)
    {
        if (!$this->token) {
            return [
                'errcode' => 5001,
                'errmsg' => '获取token失败',
            ];
        }
        if (!(is_array($data) && $data)) {
            return [
                'errcode' => 5002,
                'errmsg' => '数据格式有误',
            ];
        }
        $trade_no = isset($data['trade_no']) ? filterTags($data['trade_no']) : '';
        $trade_amount = isset($data['trade_amount']) ? sprfloat($data['trade_amount']) : 0;
        $funds_item = isset($data['funds_item']) ? filterTags($data['funds_item']) : '';
        $currency_code = isset($data['currency_code']) ? filterTags($data['currency_code']) : '';
        if (!$trade_no) {
            return [
                'errcode' => 5003,
                'errmsg' => '交易编号不能为空',
            ];
        }
        if ($trade_amount <= 0) {
            return [
                'errcode' => 5004,
                'errmsg' => '交易金额无效',
            ];
        }
        $this->headers = [
            'Content-Type: application/json',
            'Authorization: Bearer ' . $this->token,
            'PayPal-Request-Id: Trade-' . $trade_no,
        ];
        $bodys = [
            'intent' => 'CAPTURE',
            'application_context' => [
                'brand_name' => 'BKB',
                'landing_page' => 'LOGIN',
                'user_action' => 'PAY_NOW',
                'return_url' => request()->domain() . '/pc/paypal/callback',
                'cancel_url' => request()->domain() . '/pc/paypal/fail',
            ],
            'purchase_units' => [
                [
                    'reference_id' => $trade_no,
                    'description' => $funds_item,
                    'amount' => [
                        'currency_code' => $currency_code,
                        'value' => $trade_amount,
                        'breakdown' => [
                            'item_total' => [
                                'currency_code' => $currency_code,
                                'value' => $trade_amount,
                            ],
                            'shipping' => [
                                'currency_code' => $currency_code,
                                'value' => 0,
                            ],
                            'handling' => [
                                'currency_code' => $currency_code,
                                'value' => 0,
                            ],
                            'tax_total' => [
                                'currency_code' => $currency_code,
                                'value' => 0,
                            ],
                            'shipping_discount' => [
                                'currency_code' => $currency_code,
                                'value' => 0,
                            ],
                        ],
                    ],
                ],
            ],
        ];
        $this->bodys = json_encode($bodys, JSON_UNESCAPED_UNICODE);
        $url = $this->host . 'v2/checkout/orders';
        $result = json_decode($this->post($url), true);
        if (!$result) {
            return [
                'errcode' => 5005,
                'errmsg' => '远程服务器无应答',
            ];
        }
        file_put_contents(ROOT_PATH . 'paypal.log', print_r($result, true), FILE_APPEND);
        if (isset($result['name']) && isset($result['message'])) {
            return [
                'errcode' => 5006,
                'errmsg' => '[' . $result['name'] . '] - ' . $result['message'],
            ];
        }
        if (!isset($result['status'])) {
            return [
                'errcode' => 5007,
                'errmsg' => '远程服务器响应异常',
            ];
        }
        if ('CREATED' !== $result['status']) {
            $desc = isset($result['description']) ? filterTags($result['description'], 1) : $result['status'];
            return [
                'errcode' => 5008,
                'errmsg' => $desc,
            ];
        }
        $trade_token = isset($result['id']) ? filterTags($result['id']) : '';
        if (!$trade_token) {
            return [
                'errcode' => 5009,
                'errmsg' => '远程服务器响应异常',
            ];
        }
        return [
            'errcode' => 0,
            'errmsg' => 'success',
            'data' => [
                'trade_token' => $trade_token,
            ],
        ];
    }
    /**
     * 捕获订单
     */
    public function capture(array $data)
    {
        if (!$this->token) {
            return [
                'errcode' => 5001,
                'errmsg' => '获取token失败',
            ];
        }
        if (!(is_array($data) && $data)) {
            return [
                'errcode' => 5002,
                'errmsg' => '数据格式有误',
            ];
        }
        $trade_no = isset($data['trade_no']) ? filterTags($data['trade_no']) : '';
        $trade_amount = isset($data['trade_amount']) ? sprfloat($data['trade_amount']) : 0;
        $trade_token = isset($data['trade_token']) ? filterTags($data['trade_token']) : '';
        $funds_item = isset($data['funds_item']) ? filterTags($data['funds_item']) : '';
        $currency_code = isset($data['currency_code']) ? filterTags($data['currency_code']) : '';
        if (!$trade_no) {
            return [
                'errcode' => 5003,
                'errmsg' => '交易编号不能为空',
            ];
        }
        if ($trade_amount <= 0) {
            return [
                'errcode' => 5004,
                'errmsg' => '交易金额无效',
            ];
        }
        if (!$trade_token) {
            return [
                'errcode' => 5005,
                'errmsg' => '交易令牌不能为空',
            ];
        }
        $this->headers = [
            'Content-Type: application/json',
            'Authorization: Bearer ' . $this->token,
            'PayPal-Request-Id: Trade-' . $trade_token,
        ];
        $bodys = [
            'intent' => 'AUTHORIZE',
            'application_context' => [
                'brand_name' => 'BKB',
                'landing_page' => 'LOGIN',
                'user_action' => 'PAY_NOW',
                'return_url' => request()->domain() . '/pc/paypal/callback',
                'cancel_url' => request()->domain() . '/pc/paypal/fail',
            ],
            'purchase_units' => [
                [
                    'reference_id' => $trade_no,
                    'description' => $funds_item,
                    'amount' => [
                        'currency_code' => $currency_code,
                        'value' => $trade_amount,
                        'breakdown' => [
                            'item_total' => [
                                'currency_code' => $currency_code,
                                'value' => $trade_amount,
                            ],
                            'shipping' => [
                                'currency_code' => $currency_code,
                                'value' => 0,
                            ],
                            'handling' => [
                                'currency_code' => $currency_code,
                                'value' => 0,
                            ],
                            'tax_total' => [
                                'currency_code' => $currency_code,
                                'value' => 0,
                            ],
                            'shipping_discount' => [
                                'currency_code' => $currency_code,
                                'value' => 0,
                            ],
                        ],
                    ],
                ],
            ],
        ];
        $this->bodys = json_encode($bodys, JSON_UNESCAPED_UNICODE);
        $url = $this->host . 'v2/checkout/orders/' . $trade_token . '/capture';
        $result = json_decode($this->post($url), true);
        if (!$result) {
            return [
                'errcode' => 5006,
                'errmsg' => '远程服务器无应答',
            ];
        }
        file_put_contents(ROOT_PATH . 'paypal.log', print_r($result, true), FILE_APPEND);
        if (!isset($result['status'])) {
            return [
                'errcode' => 5007,
                'errmsg' => '远程服务器响应异常',
            ];
        }
        if ('COMPLETED' !== $result['status']) {
            $desc = isset($result['description']) ? filterTags($result['description'], 1) : $result['status'];
            return [
                'errcode' => 5008,
                'errmsg' => $desc,
            ];
        }
        if (!isset($result['purchase_units'][0]['payments']['captures'][0]['id'])) {
            return [
                'errcode' => 5009,
                'errmsg' => '远程服务器响应异常',
            ];
        }
        $payment_no = filterTags($result['purchase_units'][0]['payments']['captures'][0]['id']);
        if (!$payment_no) {
            return [
                'errcode' => 5010,
                'errmsg' => '远程服务器响应异常',
            ];
        }
        $payment_time = isset($resource['update_time']) ? strtotime($resource['update_time']) : 0;
        if ($payment_time <= 0) {
            $payment_time = time();
        }
        if ($payment_time > time()) {
            $payment_time = time();
        }
        return [
            'errcode' => 0,
            'errmsg' => 'success',
            'data' => [
                'payment_no' => $payment_no,
                'payment_time' => $payment_time,
            ],
        ];
    }
    /**
     * 退款
     */
    public function refund(array $data)
    {
        if (!$this->token) {
            return [
                'errcode' => 5001,
                'errmsg' => '获取token失败',
            ];
        }
        if (!(is_array($data) && $data)) {
            return [
                'errcode' => 5002,
                'errmsg' => '数据格式有误',
            ];
        }
        $capture_id = isset($data['capture_id']) ? filterTags($data['capture_id']) : '';
        $amount = isset($data['amount']) ? sprfloat($data['amount']) : 0;
        $currency_code = isset($data['currency_code']) ? filterTags($data['currency_code']) : '';
        if (!$capture_id) {
            return [
                'errcode' => 5003,
                'errmsg' => '交易流水号不能为空',
            ];
        }
        if ($amount <= 0) {
            return [
                'errcode' => 5004,
                'errmsg' => '退款金额无效',
            ];
        }
        if (!$currency_code) {
            return [
                'errcode' => 5005,
                'errmsg' => '货币代号不能为空',
            ];
        }
        $this->headers = [
            'Content-Type: application/json',
            'Authorization: Bearer ' . $this->token,
            'PayPal-Request-Id: Refund-' . $capture_id,
        ];
        $bodys = [
            'amount' => [
                'value' => $amount,
                'currency_code' => $currency_code,
            ],
        ];
        $this->bodys = json_encode($bodys, JSON_UNESCAPED_UNICODE);
        $url = $this->host . 'v2/payments/captures/' . $capture_id . '/refund';
        $result = json_decode($this->post($url), true);
        if (!$result) {
            return [
                'errcode' => 5006,
                'errmsg' => '远程服务器无应答',
            ];
        }
        file_put_contents(ROOT_PATH . 'paypal.log', print_r($result, true), FILE_APPEND);
        if (!isset($result['status'])) {
            return [
                'errcode' => 5007,
                'errmsg' => '远程服务器响应异常',
            ];
        }
        if ('COMPLETED' !== $result['status']) {
            $desc = isset($result['description']) ? filterTags($result['description'], 1) : $result['status'];
            return [
                'errcode' => 5008,
                'errmsg' => $desc,
            ];
        }
        if (!isset($result['id'])) {
            return [
                'errcode' => 5009,
                'errmsg' => '远程服务器响应异常',
            ];
        }
        $payment_no = filterTags($result['id']);
        if (!$payment_no) {
            return [
                'errcode' => 5010,
                'errmsg' => '远程服务器响应异常',
            ];
        }
        $payment_time = isset($resource['update_time']) ? strtotime($resource['update_time']) : 0;
        if ($payment_time <= 0) {
            $payment_time = time();
        }
        if ($payment_time > time()) {
            $payment_time = time();
        }
        return [
            'errcode' => 0,
            'errmsg' => 'success',
            'data' => [
                'payment_no' => $payment_no,
                'payment_time' => $payment_time,
            ],
        ];
    }
    /**
     * 获取token
     */
    private function getAccessToken()
    {
        $token = '';
        $file = ASSETS_PATH . 'paypal/access_token.json';
        if (is_file($file)) {
            $data = json_decode(file_get_contents($file), true);
            $time = isset($data['expired_time']) ? fnumber($data['expired_time']) : 0;
            if ($time > time()) {
                $token = isset($data['access_token']) ? filterTags($data['access_token']) : '';
            }
        }
        if (!$token) {
            $this->headers = [
                'Content-Type: application/json',
                'Accept-Language: en_US',
            ];
            $this->bodys = 'grant_type=client_credentials';
            $url = $this->host . 'v1/oauth2/token';
            $result = json_decode($this->post($url, true), true);
            if ($result && isset($result['access_token'])) {
                $token = $result['access_token'];
                $time = $result['expires_in'];
                $path = dirname($file);
                if (!is_dir($path)) {
                    mkdir($path, 0755, true);
                }
                file_put_contents($file, json_encode([
                    'access_token' => $token,
                    'expired_time' => time() + $time,
                ]));
            }
        }
        $this->token = $token;
    }
    /*
     * http协议发送get请求
     */
    private function get(string $url)
    {
        $ch = curl_init();
        curl_setopt($ch, CURLOPT_URL, $url);
        curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
        curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 2);
        curl_setopt($ch, CURLOPT_HEADER, 0);
        curl_setopt($ch, CURLOPT_HTTPHEADER, $this->headers);
        $out = curl_exec($ch);
        curl_close($ch);
        return $out;
    }
    /*
     * http协议发送post请求
     */
    private function post(string $url, bool $credentials = false)
    {
        $ch = curl_init();
        curl_setopt($ch, CURLOPT_URL, $url);
        curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'POST');
        curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false);
        curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false);
        curl_setopt($ch, CURLOPT_FOLLOWLOCATION, 1);
        curl_setopt($ch, CURLOPT_AUTOREFERER, 1);
        curl_setopt($ch, CURLOPT_HEADER, false);
        curl_setopt($ch, CURLOPT_HTTPHEADER, $this->headers);
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
        if ($credentials) {
            curl_setopt($ch, CURLOPT_USERPWD, $this->clientId . ':' . $this->clientSecret);
        }
        curl_setopt($ch, CURLOPT_POSTFIELDS, $this->bodys);
        $result = curl_exec($ch);
        curl_close($ch);
        return $result;
    }
}

未完待续……

欢迎加入IT技术交流接单微信群

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/542524.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

YOLO系列简记

本文主要参考了论文 A Comprehensive Review of YOLO Architectures in Computer Vision: From YOLOv1 to YOLOv8 and YOLO-NAS&#xff0c;以及其中提到的各 YOLO 原论文。 NMS 对所有检测框&#xff0c;按置信度降序排序。选择最高置信度的检测框&#xff0c;添加到最终结果…

java学习之路-继承

文章目录 前言 目录 1.1继承的概念 1.2继承有什么好处&#xff0c;为何要继承 1.3继承的语句 1.4父类成员的访问 1.4.1 子类中访问父类的成员变量 1.4.2 子类中访问父类的成员方法 1.5 super关键字 2.子类构造方法 2.1如何创建构造方法 2.2创建构造方法 3.super和this 【相同点…

C++(2) —— 通讯录管理系统

目录 1、系统需求 2、创建项目 3、菜单功能 4、退出功能 5、添加联系人 6、显示联系人 7、删除联系人 8、查找联系人 9、修改联系人 10、清空联系人 1、系统需求 2、创建项目 3、菜单功能 // 1、菜单界面 void showMenu() {cout << "--------------------…

团结引擎+OpenHarmony 1配置篇

团结引擎OpenHarmony 1 配置篇 app团结鸿蒙化第一课一 DevEco Studio 下载安装二 团结引擎三 出包 app团结鸿蒙化第一课 1 团结引擎配置2 DevEco Studio 配置 一 DevEco Studio 下载安装 申请开发者套件 1 注册华为账号 签署协议 官网 2 认真填写 DevEco Studio 开发套件申请…

etcd相关知识整理归纳 —— 筑梦之路

什么是etcd? Etcd 是 CoreOS 团队于2013年6月发起的开源项目&#xff0c;它的目标是构建一个高可用的分布式键值(key-value)数据库。etcd内部采用raft协议作为一致性算法&#xff0c;Etcd基于 Go 语言实现。 名字由来&#xff0c;它源于两个方面&#xff0c;unix的“/etc”文件…

JAVA实现人工智能,采用框架SpringAI

Spring AI介绍 Spring AI是AI工程师的一个应用框架&#xff0c;它提供了一个友好的API和开发AI应用的抽象&#xff0c;旨在简化AI应用的开发工序&#xff0c;例如开发一款基于ChatGPT的对话应用程序。 项目地址&#xff1a;https://github.com/spring-projects-experimental/sp…

FFmpeg: 自实现ijkplayer播放器--09音频重采样输出

文章目录 流程图音视设备输出回调函数重采样写入音频流因SDL输出音频采样格式为S16(一个采样点2个字节),而音频解码后采样格式通常为float planar(一个采样点4个字节),故需要重采样 重采样的条件:音频解码后的任意一个参数和需要的参数不同时,进行重采样,参数为: 采样格…

格式化D盘后C盘内的文件会受影响吗?深度解析

在计算机的日常使用中&#xff0c;磁盘格式化是一个常见的操作&#xff0c;它能帮助我们清除磁盘上的数据&#xff0c;为新的数据腾出空间。然而&#xff0c;当涉及到系统盘和其他存储盘时&#xff0c;许多用户会担心一个问题&#xff1a;如果我格式化了非系统盘&#xff0c;比…

【Kafka】Kafka 架构深入

Kafka 工作流程及文件存储机制 Kafka 中消息是以 topic 进行分类的&#xff0c;生产者生产消息&#xff0c;消费者消费消息&#xff0c;都是面向 topic 的。 topic 是逻辑上的概念&#xff0c;而 partition 是物理上的概念&#xff0c;每个 partition 对应于一个 log 文件&am…

认识异常(2)

❤️❤️前言~&#x1f973;&#x1f389;&#x1f389;&#x1f389; hellohello~&#xff0c;大家好&#x1f495;&#x1f495;&#xff0c;这里是E绵绵呀✋✋ &#xff0c;如果觉得这篇文章还不错的话还请点赞❤️❤️收藏&#x1f49e; &#x1f49e; 关注&#x1f4a5;&a…

gemini1.5 API调用

https://ai.google.dev/pricing?hlzh-cn 查询可用的model https://generativelanguage.googleapis.com/v1beta/models?keyxxx 使用postman调用 https://generativelanguage.googleapis.com/v1beta/models/gemini-1.5-pro-latest:generateContent?keyxxx https://ai.google…

JavaSE——常用API进阶二(3/8)-Date、SimpleDateFormat(构造器、常用的方法、用法示例、时间格式的常见符号)

目录 Date 构造器、常用的方法 用法示例 SimpleDateFormat 构造器、格式化时间的方法 时间格式的常见符号 用法示例 解析字符串时间成为日期对象 接下来会学习JDK8以前传统的日期、时间&#xff0c;以及JDK8开始新增的日期、时间&#xff1b;有部分项目还是有在使用JDK…

【C++学习】深入理解C++异常处理机制:异常类型,捕获和处理策略

文章目录 ♫一.异常的提出♫二.异常的概念♫三.异常的使用♫3.1 异常的抛出和捕获♫3.2.异常的重新抛出♫3.3异常安全♫3.4 异常规范 ♫4.自定义异常体系♫5.C标准库的异常体系♫6.异常的优缺点 ♫一.异常的提出 之前&#xff1a; C语言传统的处理错误的方式与带来的弊端&…

【位运算】Leetcode 两整数之和

题目解析 371. 两整数之和 算法讲解 异或的本质就是无进位相加&#xff0c;但是我们需要处理进位&#xff0c;就需要知道哪一位上有进位&#xff0c;再让无进位相加的结果 进位即可&#xff0c;在重复这个过程&#xff0c;当进位等于0的时候&#xff0c;说明相加的过程已经结…

Windows环境下删除MySQL

文章目录 一、关闭MySQL服务1、winR打开运行&#xff0c;输入services.msc回车2、服务里找到MySQL并停止 二、卸载MySQL软件1、打开控制模板--卸载程序--卸载MySQL相关的所有组件 三、删除MySQL在物理硬盘上的所有文件1、删除MySQL的安装目录&#xff08;默认在C盘下的Program …

CSS盒模型(详讲)

目录 概述&#xff1a; 内容区&#xff08;content&#xff09;&#xff1a; 内边距&#xff08;paddingj&#xff09;&#xff1a; 前言&#xff1a; 设置内边距&#xff1a; 边框&#xff08;border&#xff09;&#xff1a; 前言&#xff1a; 示例&#xff1a; 外边…

飞驰云联入选金融信创生态实验室「金融信创优秀解决方案」

近日&#xff0c;由中国人民银行领导、中国金融电子化集团有限公司牵头组建的金融信创生态实验室发布了第三期金融信创优秀解决方案&#xff0c;Ftrans飞驰云联“文件数据传输解决方案”成功入选&#xff01; 本次金融信创优秀解决方案遴选经方案征集、方案初审、专家评审等多环…

unity android 打包

现在使用的unity版本hub不支持导入support&#xff0c;只能自己下载对应的支持 找到对应的sdk&#xff0c;ndk

计算机组成原理【CO】Ch2 数据的表示和应用

文章目录 大纲2.1 数制与编码2.2 运算方法和运算电路2.3 浮点数的表示和运算 【※】带标志加法器OFSFZFCF计算机怎么区分有符号数无符号数? 【※】存储排列和数据类型转换数据类型大小数据类型转换 进位计数制进制转换2的次幂 各种码的基本特性无符号整数的表示和运算带符号整…

牛客研究生复试刷题(1)

KY30进制转换 1.最开始没有考虑到大数问题,可以说是没考虑完全,输入类型使用的是int64_t,只ac了一半测试用例。所以在数很大找不到合适的数据类型存储时,要考虑使用string来存放。 2.使用string存放数字的时候就要考虑:字符和数字之间的转换。字符转换成数字:str[i]-0,…