首先引入 PayPal的sdk 这里我引入的是1.0.4版本的

            <!-- 贝宝支付  SDK   --><dependency><groupId>com.paypal.sdk</groupId><artifactId>checkout-sdk</artifactId><paypal-sdk.version>1.0.4</paypal-sdk.version></dependency>

yml文件引入公司在官网的相关配置
这里我做了yml的文件环境隔离,限免的配置做了示例

  #  贝宝支付的测试服
pay:  paypal:clientId: AeEX1PNMNaP3RuV8JTBMznAhs_gOfFwloG6SG3TiQh1_MBj0clientSecret: EMDda7g_Q7KmOiH08qJfg-dAb8d2THkYtzRR#测试的modemode: sandbox#正式的modemode: live
/*** @since 2.0.0*/
@Data
@AllArgsConstructor
@NoArgsConstructor
@ApiModel
public class PayPalVo extends RequestModel {/**  本站订单号 */@ApiModelProperty(value = "本站订单号")private String orderNo;/**  三方订单号 */@ApiModelProperty(value = "本站订单号")private String outOrderNo;/** 1 书币充值 ; 2 plus会员充值;3 premium会员充值 */@ApiModelProperty(value = "1 书币充值 ; 2 plus会员充值;3 premium会员充值")private Integer orderType;
}
PayPal支付
```java/*** 贝宝支付** @param payPalVo 请求对象* @return 返回结果*/@PostMapping("/payPalNotify")@ApiOperation(value = "贝宝支付二次验证")@Deprecatedpublic ResponseModel<PayOrderDto> payPalNotify(@RequestBody PayPalVo payPalVo){if (ObjectUtils.isEmpty(payPalVo.getOrderNo()) || ObjectUtils.isEmpty(payPalVo.getOutOrderNo()) || ObjectUtils.isEmpty(payPalVo.getOrderType())){return ResponseModel.error(appConfigStatusCodeService.getStatusCode(payPalVo.getLanguage(), ResponseCnSate.PARAM_EXCEPTION));}// 根据paypal的订单id,捕获订单付款String outOrderNo = payPalVo.getOutOrderNo();String orderNo = payPalVo.getOrderNo();Integer orderType = payPalVo.getOrderType();// 校验该订单是否处理if (payOrderService.isDispose(orderNo) == PayConstants.ORDER_IS_DISPOSE) {return ResponseModel.error(appConfigStatusCodeService.getStatusCode(payPalVo.getLanguage(), ResponseCnSate.ORDER_PROCESSED));}//进行二次验证Boolean pay = payPalUtils.verifyOrderInfo(outOrderNo);//如果成功则执行后面的逻辑if (pay){boolean b = orderService.finishOrder(orderNo, orderType,payPalVo.getLanguage());if (!b){return ResponseModel.error(appConfigStatusCodeService.getStatusCode(payPalVo.getLanguage(), ResponseCnSate.ORDER_HANDLER_FAIL));}//保存订单号返回订单信息PayOrderDto payOrderDto =orderService.saveAndGetOrderInfo(payPalVo.getUserId(),outOrderNo,orderNo);if (ObjectUtils.isNotEmpty(payOrderDto)){return ResponseModel.success(payOrderDto);}return ResponseModel.error(appConfigStatusCodeService.getStatusCode(payPalVo.getLanguage(), ResponseCnSate.RETURN_ORDER_INFO_FAIL));}//失败返回提示return ResponseModel.error(appConfigStatusCodeService.getStatusCode(payPalVo.getLanguage(), ResponseCnSate.AUTH_FAIL_TWO));}

后面为了让大家都看的明白 我单独抽取了一个工具,简单的做了描述

package com.wyzz.global.untils.paypal;import com.baomidou.mybatisplus.core.toolkit.ObjectUtils;import com.paypal.core.PayPalEnvironment;
import com.paypal.core.PayPalHttpClient;
import com.paypal.http.HttpResponse;
import com.paypal.http.serializer.Json;
import com.paypal.orders.*;
import com.paypal.payments.CapturesGetRequest;
import lombok.extern.slf4j.Slf4j;
import java.io.IOException;
import java.util.HashMap;
import java.util.List;
import java.util.Map;import org.json.JSONObject;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;/*** PayPal工具类** @since 2.0.0*/
@Service
@Slf4j
public class PayPalUtils {/*** paypal配置应用的客户端ID*/@Value("${pay.paypal.clientId}")private String clientId;/*** paypal配置应用的客户端密钥*/@Value("${pay.paypal.clientSecret}")private String clientSecret;/*** paypal连接环境:live表示生产,sandbox表示沙盒*/@Value("${pay.paypal.mode}")private String mode;/*** paypal的http客户端* @param mode 环境信息* @param clientId 客户端ID* @param clientSecret 密钥* @return*/public PayPalHttpClient client(String mode, String clientId, String clientSecret) {log.info("mode={}, clientId={}, clientSecret={}", mode, clientId, clientSecret);PayPalEnvironment environment = mode.equals("live") ? new PayPalEnvironment.Live(clientId, clientSecret) : new PayPalEnvironment.Sandbox(clientId, clientSecret);return new PayPalHttpClient(environment);}/*** 构建订单请求体* @return*/public OrderRequest buildRequestBody() {return new OrderRequest();}/*** 校验订单信息:先扣款,后查询订单详情* @param orderId 前端提供的palpay的订单id* @return 订单存在并已扣款成功返回true*/public Boolean verifyOrderInfo(String orderId) {Boolean f = false;try {// 1.用户授权支付成功,进行扣款操作this.captureOrder(orderId);} catch (Exception e) {e.printStackTrace();} finally {// 2.扣款操作失败时也要查询订单,确认是否已扣款try {OrdersGetRequest request = new OrdersGetRequest(orderId);HttpResponse<Order> response = client(mode, clientId, clientSecret).execute(request);System.out.println("Status Code: " + response.statusCode());System.out.println("Status: " + response.result().status());System.out.println("Order id: " + response.result().id());if(response.result().purchaseUnits().get(0).payments() != null) {List<Capture> captures = response.result().purchaseUnits().get(0).payments().captures();if(captures != null) {for (Capture capture : captures) {if (capture.id() != null && capture.id().length() > 0) {f = true;}}}}} catch (Exception e2) {e2.printStackTrace();}}return f;}/*** 用户授权支付成功,进行扣款操作:* 用户通过CreateOrder生成 approveUrl 跳转paypal支付成功后,只是授权,并没有将用户的钱打入我们的paypal账户,我们需要通过 CaptureOrder接口,将钱打入我的PayPal账户* @param orderId 前端提供的palpay的订单id*/public void captureOrder(String orderId){OrdersCaptureRequest request = new OrdersCaptureRequest(orderId);request.requestBody(this.buildRequestBody());HttpResponse<Order> response = null;try {response = client(mode, clientId, clientSecret).execute(request);} catch (IOException e1) {try {log.error("第1次调用paypal扣款失败");response = client(mode, clientId, clientSecret).execute(request);} catch (Exception e) {try {log.error("第2次调用paypal扣款失败");response = client(mode, clientId, clientSecret).execute(request);} catch (Exception e2) {log.error("第3次调用paypal扣款失败,失败原因 {}", e2.getMessage() );}}}if (ObjectUtils.isNotEmpty(response)) {log.info("Status Code = {}, Status = {}, OrderID = {}", response.statusCode(), response.result().status(), response.result().id());for (LinkDescription link : response.result().links()) {log.info("Links-{}: {}    \tCall Type: {}", link.rel(), link.href(), link.method());}for (PurchaseUnit purchaseUnit : response.result().purchaseUnits()) {for (Capture capture : purchaseUnit.payments().captures()) {log.info("Capture id: {}", capture.id());log.info("status: {}", capture.status());log.info("invoice_id: {}", capture.invoiceId());if("COMPLETED".equals(capture.status())) {//进行数据库操作,修改订单状态为已支付成功,尽快发货(配合回调和CapturesGet查询确定成功)log.info("支付成功,状态为=COMPLETED");}if("PENDING".equals(capture.status())) {log.info("status_details: {}", capture.captureStatusDetails().reason());String reason = "PENDING";if(capture.captureStatusDetails() != null && capture.captureStatusDetails().reason() != null) {reason = capture.captureStatusDetails().reason();}//进行数据库操作,修改订单状态为已支付成功,但触发了人工审核,请审核通过后再发货(配合回调和CapturesGet查询确定成功)log.info("支付成功,状态为=PENDING : {}", reason);}}}Payer buyer = response.result().payer();log.info("Buyer Email Address: {}", buyer.email());log.info("Buyer Name: {} {}", buyer.name().givenName(), buyer.name().surname());}}/*** 查询订单详情* @param orderId 前端提供的palpay的订单id* @throws IOException*/public void ordersGetRequest(String orderId) throws IOException {OrdersGetRequest request = new OrdersGetRequest(orderId);HttpResponse<Order> response = null;try {response = client(mode, clientId, clientSecret).execute(request);} catch (Exception e) {try {System.out.println("调用paypal订单查询失败,链接异常1");response = client(mode, clientId, clientSecret).execute(request);} catch (Exception e2) {try {System.out.println("调用paypal订单查询失败,链接异常2");response = client(mode, clientId, clientSecret).execute(request);} catch (Exception e3) {System.out.println("调用paypal订单查询失败,链接异常3");System.out.println(e3.getMessage());}}}System.out.println("Status Code: " + response.statusCode());System.out.println("Status: " + response.result().status());System.out.println("Order id: " + response.result().id());if(response.result().purchaseUnits().get(0).payments() != null) {List<Capture> captures = response.result().purchaseUnits().get(0).payments().captures();if(captures != null) {for (Capture capture : captures) {System.out.println("\t订单编号= " + capture.invoiceId() + "\tCapture Id= " + capture.id() + "\tCapture status= " + capture.status() + "\tCapture amount= " + capture.amount().currencyCode() + ":" + capture.amount().value());}}List<Refund> refunds = response.result().purchaseUnits().get(0).payments().refunds();if(refunds != null) {for (Refund refund : refunds) {System.out.println("\t售后编号= " + refund.invoiceId() + "\tRefund Id= " + refund.id() + "\tRefund status= " + refund.status() + "\tRefund amount= " + refund.amount().currencyCode() + ":" + refund.amount().value());}}}System.out.println("Links: ");for (LinkDescription link : response.result().links()) {System.out.println("\t" + link.rel() + ": " + link.href() + "\tCall Type: " + link.method());}}/*** 查询扣款详情* @param orderId 前端提供的palpay的订单id* @throws IOException*/public void capturesGetRequest(String orderId) throws IOException {CapturesGetRequest request = new CapturesGetRequest(orderId);HttpResponse<com.paypal.payments.Capture> response = client(mode, clientId, clientSecret).execute(request);System.out.println("Status Code: " + response.statusCode());System.out.println("Status: " + response.result().status());System.out.println("Capture ids: " + response.result().id());System.out.println("Links: ");for (com.paypal.payments.LinkDescription link : response.result().links()) {System.out.println("\t" + link.rel() + ": " + link.href() + "\tCall Type: " + link.method());}}public Map<String,Object> createOrder(OrderRequest orderRequest) {OrdersCreateRequest request = new OrdersCreateRequest();request.header("prefer","return=representation");request.requestBody(orderRequest);
//        PayPalClient payPalClient = new PayPalClient();HttpResponse<Order> response = null;try {response = client(mode, clientId, clientSecret).execute(request);} catch (IOException e1) {try {log.error("第1次调用paypal订单创建失败");response = client(mode, clientId, clientSecret).execute(request);} catch (Exception e) {try {log.error("第2次调用paypal订单创建失败");response = client(mode, clientId, clientSecret).execute(request);} catch (Exception e2) {log.error("第3次调用paypal订单创建失败,失败原因:{}", e2.getMessage());}}}
//        String approve = "";Map<String,Object> map = new HashMap();if (response.statusCode() == 201) {log.info("Status Code = {}, Status = {}, OrderID = {}, Intent = {}", response.statusCode(), response.result().status(), response.result().id(), response.result().checkoutPaymentIntent());map.put("Id",response.result().id());for (LinkDescription link : response.result().links()) {log.info("Links-{}: {}    \tCall Type: {}", link.rel(), link.href(), link.method());if(link.rel().equals("approve")) {//                    approve = link.href();map.put("outOrderNo",link.href());}}String totalAmount = response.result().purchaseUnits().get(0).amountWithBreakdown().currencyCode() + ":" + response.result().purchaseUnits().get(0).amountWithBreakdown().value();log.info("Total Amount: {}", totalAmount);
//            String json= new JSONObject(new Json().serialize(response.result())).toString(4);
//            log.info("createOrder response body: {}", json);}return map;}//    public static void main(String[] args) throws Exception {//        PayPalUtils payPalUtils = new PayPalUtils();
//        payPalUtils.verifyOrderInfo("45S82276S8854414K");
//        payPalUtils.ordersGetRequest("45S82276S8854414K");
//        payPalUtils.captureOrder("18K07174PX6483500");
//        payPalUtils.capturesGetRequest("31V65486WC667442G");
//        // issue : ORDER_ALREADY_CAPTURED 订单已被捕获
//        // intent=CAPTURE 每个订单只允许捕获一次
//    }
}

不敢说描述的有多么到位 只希望给java将要对接paypal的同学一点儿参考

完整版PayPal支付(java后端教程)相关推荐

  1. 微信小程序:独家微信社群人脉小程序源码带后端控制源码完整版端控带简单教程

    这是一款小编自营的一款人脉小程序系统 小编自营大概有三个多月了吧一直没有给大家公开 本款小程序群二维码自动采集推送的,所以大家不用担心没有群难运营 小编运营几个月不靠一丝一毫的推广每天自然流量都是一千 ...

  2. 视频教程-大型Java项目视频教程_王勇老师DRP项目教程完整版292集-Java

    大型Java项目视频教程_王勇老师DRP项目教程完整版292集 动力节点王勇老师,CCTV<影响力对话>栏目特约嘉宾,Java培训知名讲师,中国Java培训领军人物,北京动力节点创始人,董 ...

  3. php电销源码部署,【独家分享】最新价值4800接单运营版电销语音机器人完整版源码+文字安装教程...

    [独家分享]最新价值4800接单运营版[电销语音机器人]完整版源码+文字安装教程 完完整整,接单运营级的东西,含非常完整的教程.搭建过程中如果有不明白的地方,可以直接联系.,或者群内喊社区大佬协助. ...

  4. 微信小程序支付 Java后端代码详解

    微信小程序发起支付 Java后台处理代码---- 直接上代码吧! 我把自己的业务逻辑代码删了,但是都有注释的 莫慌! package com.mvc.controller;import java.io ...

  5. oracle11g客户端完整版和精简版安装教程

    完整版 一.创建用户和组 [root@zhjhapp6 ~]# groupadd oinstall [root@zhjhapp6 ~]# groupadd dba [root@zhjhapp6 ~]# ...

  6. 蓝桥杯2020年上半场省赛完整版题解(Java 大学B组)

    A 解密 本题总分:5 分 问题描述 小明设计了一种文章加密的方法:对于每个字母 c cc,将它变成某个另外的字符 Tc .下表给出了字符变换的规则: 例如,将字符串 YeRi 加密可得字符串 EaF ...

  7. python教程5小时完整版_Python零基础入门教程5小时完整版(北京理工大2020年版)

    这是面向零基础人群的Python教程,通过5个小时的学习,你能够了解Python的基础语法,并且编写100行左右的代码来解决日常生活中遇到的问题. 每个章节学练结合.先讲解Python的基础知识,然后 ...

  8. 解决Aucc2021在win10/11系统上安装失败问题Audition 2021中文完整版_永久使用安装教程

    aucc2021是一款专业的混音.修整及精确编辑音频的软件,提供了创建.混合.编辑和复原音频内容的多轨.波形和光谱显示功能,无论你是要录制音乐.无线电广播,还是为录像配音,它都可以很好的满足你的使用需 ...

  9. 达内第二个月考java核心题目,(完整版)达内科技JAVA项目经理入职考试--I(不含答案)--新...

    达内科技项目经理入职考试 (JAVA方向 A卷) 考试说明: 本考试为达内科技JAVA产品线项目经理入职考试,考试使用闭卷.笔试的形式.满分100分,考试时间120分钟.考试期间不允许使用电脑,不可打 ...

  10. 三方 app微信支付 java后端实现

    废话不多说,直接上代码 生成prepay_id给前段返回 (如果是web端的就把 return 改成 (WebUtil.response(),如果是APP就直接用不用变) 这个写的一个Dome连接地址 ...

最新文章

  1. Visual Studio视觉编程工具(推荐四个)
  2. JS数字金额转为大写金额
  3. RecycleView中使用Glide加载图片防止加载错乱
  4. Python计算机视觉:第六章 图像聚类
  5. ReportViewer教程(9)-给报表增加页打印日期编号
  6. java冒泡法优化_数据结构java版之冒泡排序及优化
  7. IE10-浏览器实现placeholder效果
  8. 如何在C ++中实现内联函数?
  9. 【干货】关于机器学习的知识点,全在这篇文章里了
  10. SQL:PostgreSQL+PostGIS的安装以及C# GDAL开发环境配置
  11. vscode还用装git_如何给VScode配置git
  12. 嵌入式mysql数据库文件读取_使用嵌入式关系型SQLite数据库存储数据
  13. 常见的十几种编程语言介绍
  14. 【专栏必读】王道考研408数据结构万字笔记、题目题型总结、注意事项、目录导航和思维导图
  15. 创新设计思维总结报告
  16. 27 | 风控系统:如何从海量业务数据中,挖掘黑灰产?
  17. SpringCloud 单Eureka简单例子consumer-provider
  18. hadoop reducer不执行问题及解决
  19. 整理苹果官网上iOS的各种辅助功能
  20. java+ElementUI前后端分离旅游项目第七天 权限管理和图形报表

热门文章

  1. (QT)大华(海康)网络摄像头人脸采集和人脸识别SDK的二次开发
  2. 回归的误差服从正态分布吗_线性回归中的正态分布
  3. C++程序加速的12个方法
  4. Android app接入微信人脸支付详解
  5. Jupyter notebook系列(2):使用技巧(快捷键,多行输出,多光标操作,Unix系统命令的使用,查看输入输出历史,保存记录点与分享,抑制文末输出,图片输出,多环境kernel切换)
  6. Jmeter性能测试之命令行执行和生成测试报告
  7. mysql 两表关联查询 group by having
  8. openstack 分布式路由器dvr
  9. Android接入google地图
  10. 计算机复制功能快捷键,电脑复制快捷键是什么(全部复制粘贴的快捷键是什么)...