0、写在前面的话

支付系统是一个老生常谈的话题,我也相信每个公司开发的支付系统不尽相同,因为业务形态并不太一样。

在此,我并不想讲一个大而全的支付系统,个人也没有能力去阐述。

在我看来,一个支付系统应提供支付渠道管理,支付网关,基本支付/退款/转账能力,支付记录/明细,及其相关的监控运维系统。

至于所谓的账务清算,对账功能,账户体系,风控体系,现金流量管理,应该纳入到「财务系统」,大概是大佬们谈论的都是广义的「支付系统」吧!

而我今天只谈狭义的「支付系统」。

目前,支付的流程包含了三大部分:发起支付,发起退款,接收回调。

考虑到吞吐量的影响,将原先同步的编程方式改为异步的编程方式,不出意外的话,将会使用到Java8的ExecutorService和CompletableFuture。

此外,还用到了公司其他的现成的东西:RabbitMQ,Redis,MongoDB。

我是打算将这套支付系统设计成与具体业务无关,可以纳入到公司的公共平台系统中。

具体是如何做到的,请接着往下读。

1、发起支付

这一部分讲述的是客户端和服务端如何配合完成一次支付请求。服务端必须要有一个意识,最终发起支付的还是客户端,服务端提供一些必要的参数配置信息。

发起支付的架构图如下所示:

跟着标注的序号,可以跟踪到一个支付请求是如何发起的(Sequence Diagram就免了),流程描述如下:

  1. Submit a pay task,当客户端需要发起支付的时候,起始是向支付任务队列里面加入了一个新的支付任务,这个过程是异步实现的。先根据客户端提交的参数,构造好一个新的支付任务;
  2. Offer a task,开启一个异步任务,做的事情就是向MQ中添加一个新的支付任务,等待被消费;
  3. Pay task description,一旦异步任务被成功创建,将会把第一步构造好的支付任务信息直接return给客户端;
  4. Poll a task,与此同时,支付任务的消费者将新的支付任务poll下来进行执行;
  5. Send a pay request,这一步需要根据实际情况而定。并不是所有的支付请求都要先经过第三方支付平台,比如支付宝;而对于微信,则还需要凭支付参数申请一个prepay_id,再经由客户端发起支付;
  6. Response,没什么好说的,第三方渠道返回的支付必要参数;
  7. Cache result,至此,一个支付任务可以算是完成了,可以将任务的执行结果(无论成功与否)缓存在Redis中,随时等待客户端的回访;
  8. Query result,客户端在提交支付任务后,间隔一定时间后(建议2~3s),发起一个结果查询的请求;
  9. Query,直接进Redis查找结果;
  10. Synchronize,这是一个异步的操作,将支付任务的执行结果“顺便”同步到MongoDB中,并删除Redis中缓存的任务执行结果。持久化到MongoDB主要是为后续的容错,重试,数据分析等提供落地的数据源;
  11. Return,由Redis返回给应用服务器;
  12. Return payment,应用服务器再将最终的支付对象返回给客户端。

让我们更深入一点,我们来看三张Class Diagram:

① 先说说支付任务(PayTask)部分。PayTask和Payment两个都是MongoDB中的Document对象,但在任务执行期间,PayTask是用Redis进行缓存的,方便客户端随时发起Query,任务执行成功后,会生成Payment对象,最终PayTask和Payment都会持久化到MongoDB中。在PayService中,有对支付任务的一些基本操作,包括任务提交,取消,重试,构建等等。

② 再说说任务的执行(runner)。这部分和RabbitMQ紧密相关,一旦一个支付任务形成了,就会放入任务执行队列中,由消费者取出执行。在TaskRunner中,有两个基本的接口方法:run(task)、retry(task),分别是执行任务和重试任务。在AbstractPayTaskRunner中已经封装好了这两个方法,继承AbstractPayTaskRunner需要实现doTask方法,从返回值可以看出,这个过程是异步化的。关于Retry机制,用户可以设置重试与否,一旦设置了TaskInfo.needRetry=true(不出意外,默认就是允许重试),就启用了Retry机制。还可以设置重试的次数(TaskInfo.retryTimes),默认三次,分别间隔1s,2s,3s,间隔时间以公差为1的等差数列组成。当然不会让用户无限重试,系统内置有一个最大重试次数,最大重试次数内置为5次。

为什么是5次?

你感受一下,1s,2s,3s,4s,5s,整个请求链条就被拉长到了15s,这对客户端简直就是灾难了!!

③ 接着说一下支付渠道(PayChannel)。这部分设计与具体的支付渠道对接联系比较紧密了,包括支付参数配置,支付参数处理,签名/验签等等。

④ 最后解释一下支付参数(PayParams)。

大部分还是能看懂的,我解释几个关键的property:

1) appId,这是为了区分不同的产品所设置的。现实中,很有可能一个产品会申请与之对应的支付渠道,然后在支付平台中创建应用,设置好对应的支付参数,系统将会分配一个appId,凭此值就可以直接定位到各个支付参数。如果想再更完善一点,可以再区分一下测试环境和正式环境;

2) amount,这里代表的是支付金额的意思,但是这套支付系统的金额单位统一设置成 人民币【分】;

3) metadata,理论上,元数据这个字段没啥限制,要是非要说有限制,那么就是字段长度了——5000个字符。这个字段的想象空间还是很大的:用于填写丰富的交易相关信息,用于在增长智能系统产品中进行深入商业分析。包括交易行为多维分析、人群分析、产品转化路径、个性化推荐、智能补贴、定向推送等。看产品经理要怎么玩了;

5) credential,这个字段非常非常重要,其中装载的就是客户端最终发起支付请求的凭证,会作为Payment对象的一部分返回给客户端;

MongoDB的document字段设计

解释一下为什么要用MongoDB:

个人觉得,如果这个通用服务要得到较好的推广(甚至是开源),用MySQL等关系型数据库是不二之选,因为一个完整实用的系统,必然是少不了数据库的,如果一旦用了一些非传统的东西,必然会提高一部分人的对接成本。有的人一看不符合团队的技术栈,直接就不考虑了。

为什么我还是要用MongoDB呢?

① 团队的技术栈里面有这么个东西,不用白不用;

② MongoDB普及程度实在是不要太高,还不用上点NoSQL的东西,感觉自己分分钟被OUT掉了;

③ 要存储的数据结构需要支持动态扩展的特性,我就看中MongoDB的灵活性,如下是要存储的数据结构:

document_name = “Payment”

{"payId": "pay_Oyvrf9vP880STm1e9G5CSCm1","method": "yoogurt.taxi.pay","version": "v1.0","timestamp": 1473044885,"created": 1473042835,"paid": false,"appId": "app_KiPGa98abDmLe9ev","channel": "wx","orderNo": "20161899798416","clientIp": "192.168.18.189","amount": 10000,"subject": "用户充值订单(¥100.0)","body": "用户充值订单(¥100.0)","paidTime": null,"transactionNo": "","metadata": {"user_id": "170204469176","phone_number": "13811234567"},"credential": {"appId": "wx4932b5159d18311e","partnerId": "1269774001","prepayId": "wx201609051033574da13955420883291539","nonceStr": "1e99d8ffdde926ed9cbdf4d2e614abad","timeStamp": "1473042837","packageValue": "Sign=WXPay","sign": "1CECCE6B13C956DEBA88800B3DEC4DBE"},"extra": {},"statusCode": "","message": "","description": ""
}
复制代码

其中,metadata,credential,extra这类字段,并没有一个特别固定的规范,用MySQL要冗余一下字段才行,或者针对每个渠道去分表,想想都觉得烦!

MySQL

因为这套支付系统被设计成为支持多应用,多渠道,所以此处用到MySQL存放一些应用配置。 E-R图免了,直接上数据库表结构:

① pay_channel:可供接入的支付渠道

② app_settings:支付应用信息

③ app_channel:应用已接入的支付渠道

④ alipay_settings:支付宝参数设置

⑤ wx_settings:微信app支付参数设置

如果想要增加支付渠道,只需要添加一张对应的支付参数设置表。

2、发起退款

不出意外,客户在平台的每笔订单都可以发起退款,而且还能分批退,也就是同一个订单,可以多次发起退款申请,只要保证退款总额不超出实付总额。 架构图如下所示:

跟发起支付请求的流程有很多相似之处,不再一一解释了,两个关键的地方说明一下:

  1. 客户端发起退款请求的时候,需要携带payId,就是支付对象的id。这就意味着,支付系统的调用方需要维护payId与orderNo的对应关系,务必在客户端发起退款请求之前,获取到正确的payId;
  2. 承接上一步,这才有了图中的第5、6个步骤,从MongoDB中查询之前的支付对象。第三方渠道通常会要求在退款的时候指定一个退款单号,因为一笔订单可以分多次退款,所以不建议将订单号作为退款单号使用。这里的退款单号由支付系统生成并维护。

这部分的执行流程和之前类似,客户端发起退款请求,形成一个退款任务(RefundTask),放入任务队列中,消费者取出并执行各自的业务逻辑,退款成功会生成Refund对象,并持久化到MongoDB中。

MongoDB

document_name = "Refund"

{"payId": "pay_Oyvrf9vP880STm1e9G5CSCm1","method": "yoogurt.taxi.pay","version": "v1.0","timestamp": 1473044885,"created": 1473042835,"refundId": "refund_kmw1vrf9wSrP1e9Gkp05CSCm1","appId": "app_KiPGa98abDmLe9ev","orderNo": "20161899798416","clientIp": "192.168.18.189","amount": 10000,"succeedTime": 1473150835,"transactionNo": "6405996874204000684260056054","refundStatus": "success","message": "","metadata": {"user_id": "170204469176","phone_number": "13811234567"},"description": ""
}
复制代码

3、接收回调

这部分功能被设计成了事件驱动类型,所以webhooks当仁不让。

因为各个渠道的回调内容都不尽相同,所以这部分设计会按支付渠道切分。

架构图如下:

用户在支付完毕后,第三方支付渠道通过发起支付时指定的回调地址对商户进行支付成功的异步通知。

这部分的执行流程和之前类似,在各自的PayChannel中解析好回调参数,形成一个回调事件(Event),并持久化到MongoDB中,然后再生成一个回调任务(EventTask),放入任务队列中,消费者取出并执行各自的业务逻辑,这里的消费者就是上游的业务服务系统。

MongoDB

document_name = “Event”

{"eventId": "evt_la06CoQAiPojSgJKe5gt3nwq","created": 1427555016,"eventType": "pay.succeeded","data": {"payId": "pay_Oyvrf9vP880STm1e9G5CSCm1","method": "yoogurt.taxi.pay","version": "v1.0","timestamp": 1473044885,"created": 1473042835,"paid": false,"appId": "app_KiPGa98abDmLe9ev","channel": "wx","orderNo": "20161899798416","clientIp": "192.168.18.189","amount": 10000,"subject": "用户充值订单(¥100.0)","body": "用户充值订单(¥100.0)","paidTime": null,"transactionNo": "","statusCode": "","message": "","metadata": {"user_id": "170204469176","phone_number": "13811234567"},"credential": {"appId": "wx4932b5159d18311e","partnerId": "1269774001","prepayId": "wx201609051033574da13955420883291539","nonceStr": "1e99d8ffdde926ed9cbdf4d2e614abad","timeStamp": "1473042837","packageValue": "Sign=WXPay","sign": "1CECCE6B13C956DEBA88800B3DEC4DBE"},"extra": {},"description": ""},"retryTimes": 0
}
复制代码

特别说明一下data字段:

如果是支付成功事件,则返回对应的Payment对象;

如果是退款成功时间,则返回对应的Refund对象。

总结

可能有的读者通篇看下来,觉得这并不是什么支付系统,仅仅是对接了一下第三方支付渠道,勉强算是支付渠道网关吧!

如果你有这种感受,我也是非常认同的。

个人认为这篇文章还是比较接地气的,没有太多理论的东西,看到的更多是实现层面的内容,就差贴代码了!

坦白地讲,第三方支付渠道对接了不少次,却并没有像现在这样系统地去设计,去总结。

我用过几次ping++的产品,在企业级聚合支付领域,ping++算是业界领先者了,所以,我的一些数据结构设计还是与其有几分相似的,ping++以后也会是我模仿和比较的对象。

这次也是我的支付系统实现所迈出的第一步,今后也会不断丰富,完善我自己的支付系统。

希望对你有所帮助!

THANKS!

每日干货分享,传递互联网世界有价值的讯息,微信公众号:jishuhui_2015

转载于:https://juejin.im/post/5a47914d6fb9a0451b04e3c4

记一次支付系统的设计体验相关推荐

  1. 支付系统整体设计:整体架构设计以及注意要点(三)

    一般来说,银行会提供两种支付途径:无跳转的快捷支付接口和带跳转的网银接口.前者在绑卡,支付的时候,不需要跳到银行页面上去处理,后者则需要在银行的网银页面上完成.显然前者对用户来说体验要好多了,用户流程 ...

  2. andy学java系列之J2ME的移动支付系统的设计与实现

    andy学java系列 J2ME的移动支付系统的设计与实现 ----三星SDK支付API介绍 移动支付是移动电子商务中的最重要的部分之一.安全性.私密性.易用性是移动支付的最重要的几个问题.目前有许多 ...

  3. java系列之J2ME的移动支付系统的设计与实现

    andy学java系列 J2ME的移动支付系统的设计与实现 ----三星SDK支付API介绍 移动支付是移动电子商务中的最重要的部分之一.安全性.私密性.易用性是移动支付的最重要的几个问题.目前有许多 ...

  4. 基于SET协议的电子支付系统模块设计

    基于Internet的电子商务以其具有传统商务模式不可比拟的优点而在当今世界蓬勃发展.电子商务发展的关键问题就是交易的安全性,也就是网络上的信息安全,即网上电子支付的安全实现.SET安全电子交易协议是 ...

  5. 《支付系统-收银台设计》

    1 序 本文属于支付系统设计系列,原文参见Ping++的<支付系统白皮书> 2 支付方式的选择 收银台的常见支付方式有两种:收单,充值. 收单:通过各种支付方式对业务订单进行付款. 充值: ...

  6. 支付系统整体设计:整体架构设计以及注意要点(一)

    016-11-23 01:43:00 来源: 凤凰牌老熊 导读: 在支付系统中,支付网关和支付渠道的对接是最核心的功能.其中支付网关是对外提供服务的接口,所有需要渠道支持的资金操作都需要通过网关分发到 ...

  7. 支付系统架构设计详解

    内容导读:支付永远是一个公司的核心领域,因为这是一个有交易属性公司的命脉.那么,支付系统到底长什么样,又是怎么运行交互的呢? 抛开带有支付牌照的金融公司的支付架构,下述链路和系统组成基本上符合绝大多数 ...

  8. 支付系统数据库设计思考

    主支付表 字段名 字段类型 备注 id bigint(16) 主键id order_id varchar(24) 订单号 bus_pay_no varchar(24) 支付id Pay_status ...

  9. 支付系统架构设计----整体结构图

最新文章

  1. Kruskal算法 - C语言详解
  2. MFC 蜂鸣声或播放音频
  3. Spring Boot缓存注解@Cacheable、@CacheEvict、@CachePut使用
  4. CentOS远程监控
  5. mysql 集群怎么卸载节点_Greenplum移除节点
  6. 使用nagios监控io,内存
  7. MapReduce on Yarn 的流程和架构图
  8. 专升本计算机专业是理工类吗,理工类专接本有些专业
  9. 机器学习 神经网络 神经元_神经网络如何学习?
  10. 完成中国福利彩票快三的程序设计
  11. openssl SM2签名密钥生成
  12. Google收购YouTube一周年:开挖的视频金矿
  13. 汇川小型PLC-MODBUS(485)通讯模式
  14. 程序员的一百万种变现方式 03,努力多赚零花钱
  15. Python 库 Geopy 的用法,经纬度坐标转换、经纬度距离计算
  16. oracle银行借贷系统,Oracle ERP系統借贷关系表
  17. 用C语言实现简单的一字棋游戏
  18. TopOn广告SDK——聚合广告SDK
  19. Codeforces Round #506 (Div. 3)题解
  20. 窃 听 器--郭德纲相声

热门文章

  1. 前端一HTML:二十二元素显示方式案例
  2. IP地址的分类及各类IP的最大网络数、网络号范围和最大主机数
  3. MySQL使用命令备份和还原数据库
  4. [Shell] 文件名截取的问题:bash .vs. csh
  5. shell脚本 逐行读取文本并且 进行字符串的截取
  6. JAVA帮助文档全系列 JDK1.5 JDK1.6 JDK1.7 官方中英完整版下载
  7. 51CTO推荐博客、博客之星名单【2014年】
  8. oracle emp数据库或数据误删恢复
  9. video processing on Mac and iOS
  10. [笔记] 分频计数(七)