Paypal如何实现循环扣款(订阅)?
起因
业务需求要集成Paypal,实现循环扣款功能,然而百度
和GOOGLE
了一圈,除官网外,没找到相关开发教程,只好在Paypal
上看,花了两天后集成成功,这里对如何使用Paypal的支付接口做下总结。
Paypal现在有多套接口:
- 通过
Braintree
(后面会谈Braintree)实现Express Checkout; - 创建App,通过REST Api的接口方式(现在的主流接口方式);
- NVP/SOAP API apps的接口(旧接口);
Braintree的接口
Braintree
是Paypal
收购的一家公司,它除了支持Paypal
的支付外,还提供了升级计划,信用卡,客户信息等一系列全套的管理,使用上更方便;这些功能Paypal
第二套REST
接口其实也集成了大部分,但是Paypal
的Dashboard
不能直接管理这些信息而Braintree
可以,所以我其实我更愿意用Braintree
。关键是我使用的后端框架是Laravel
,它的cashier
解决方案默认可以支持Braintee
,所以这套接口是我的首选。但是当我把它的功能都实现后发现一个蛋疼的问题:Braintree
在国内不支持。。。。。。卒。。。
REST API
这是顺应时代发展的产物,如果你之前用过OAuth 2.0
与REST API
,那看这些接口应该不会有什么困惑。
旧接口
除非REST API
接口有不能满足的,比如政策限制,否则不推荐使用。全世界都在往OAuth 2.0
的认证方式和REST API
的API使用方式迁移,干嘛逆势而行呢。因此在REST API
能解决问题情况下,我也没对这套接口做深入比较。
REST API的介绍
官方的API参考文档https://developer.paypal.com/webapps/developer/docs/api/对于其API和使用方式有较详细的介绍,但是如果自己直接调这些API还是很繁琐的,同时我们只想尽快完成业务要求而不是陷入对API的深入了解。
那么如何开始呢,建议直接安装官方提供的PayPal-PHP-SDK,通过其Wiki
作为起点。
在完成首个例子之前,请确保你有Sandbox
帐号,并正确配置了:
Client ID
Client Secret
Webhook API
(必须是https
开头且是443
端口,本地调试建议结合ngrok
反向代理生成地址)Returnurl
(注意项同上)
在完成Wiki
的首个例子后,理解下接口的分类有助于完成你的业务需求,下面我对接口分类做个介绍,请结合例子理解http://paypal.github.io/PayPal-PHP-SDK/sample/#payments。
Payments
一次性支付接口,不支持循环捐款。主要支付内容有支持Paypal
支付,信用卡支付,通过已保存的信用卡支持(需要使用Vault
接口,会有这样的接口主要是PCI
的要求,不允许一般的网站采集信用卡的敏感信息),支持付给第三方收款人。Payouts
没用到,忽略;Authorization and Capture
支持直接通过Paypal
的帐号登陆你的网站,并获取相关信息;Sale
跟商城有关,没用到,忽略;Order
跟商城有关,没用到,忽略;Billing Plan & Agreements
升级计划和签约,也就是订阅功能,实现循环扣款必须使用这里的功能,这是本文的重点;Vault
存储信用卡信息Payment Experience
没用到,忽略;Notifications
处理Webhook
的信息,重要,但不是本文关注内容;Invoice
票据处理;Identity
认证处理,实现OAuth 2.0
的登陆,获取对应token
以便请求其他API
,这块Paypal-PHP-SDK
已经做进去,本文也不谈。
如何实现循环扣款
分四个步骤:
- 创建升级计划,并激活;
- 创建订阅(创建Agreement),然后将跳转到
Paypal
的网站等待用户同意; - 用户同意后,执行订阅
- 获取扣款帐单
1.创建升级计划
升级计划对应Plan
这个类。这一步有几个注意点:
升级计划
创建后,处于CREATED
状态,必须将状态修改为ACTIVE
才能正常使用。Plan
有PaymentDefinition
和MerchantPreferences
两个对象,这两个对象都不能为空;- 如果想创建
TRIAL
类型的计划,该计划还必须有配套的REGULAR
的支付定义,否则会报错; - 看代码有调用一个
setSetupFee
(非常,非常,非常重要)方法,该方法设置了完成订阅后首次扣款的费用,而Agreement
对象的循环扣款方法设置的是第2次开始时的费用。
以创建一个Standard
的计划为例,其参数如下:
$param = [
"name" => "standard_monthly",
"display_name" => "Standard Plan",
"desc" => "standard Plan for one month",
"type" => "REGULAR",
"frequency" => "MONTH",
"frequency_interval" => 1,
"cycles" => 0,
"amount" => 20,
"currency" => "USD"
];
创建并激活计划代码如下:
//上面的$param例子是个数组,我的实际应用传入的实际是个对象,用户理解下就好。
public function createPlan($param)
{
$apiContext = $this->getApiContext();
$plan = new Plan();
// # Basic Information
// Fill up the basic information that is required for the plan
$plan->setName($param->name)
->setDescription($param->desc)
->setType('INFINITE');//例子总是设置为无限循环
// # Payment definitions for this billing plan.
$paymentDefinition = new PaymentDefinition();
// The possible values for such setters are mentioned in the setter method documentation.
// Just open the class file. e.g. lib/PayPal/Api/PaymentDefinition.php and look for setFrequency method.
// You should be able to see the acceptable values in the comments.
$paymentDefinition->setName($param->name)
->setType($param->type)
->setFrequency($param->frequency)
->setFrequencyInterval((string)$param->frequency_interval)
->setCycles((string)$param->cycles)
->setAmount(new Currency(array('value' => $param->amount, 'currency' => $param->currency)));
// Charge Models
$chargeModel = new ChargeModel();
$chargeModel->setType('TAX')
->setAmount(new Currency(array('value' => 0, 'currency' => $param->currency)));
$returnUrl = config('payment.returnurl');
$merchantPreferences = new MerchantPreferences();
$merchantPreferences->setReturnUrl("$returnUrl?success=true")
->setCancelUrl("$returnUrl?success=false")
->setAutoBillAmount("yes")
->setInitialFailAmountAction("CONTINUE")
->setMaxFailAttempts("0")
->setSetupFee(new Currency(array('value' => $param->amount, 'currency' => 'USD')));
$plan->setPaymentDefinitions(array($paymentDefinition));
$plan->setMerchantPreferences($merchantPreferences);
// For Sample Purposes Only.
$request = clone $plan;
// ### Create Plan
try {
$output = $plan->create($apiContext);
} catch (Exception $ex) {
return false;
}
$patch = new Patch();
$value = new PayPalModel('{"state":"ACTIVE"}');
$patch->setOp('replace')
->setPath('/')
->setValue($value);
$patchRequest = new PatchRequest();
$patchRequest->addPatch($patch);
$output->update($patchRequest, $apiContext);
return $output;
}
2.创建订阅(创建Agreement),然后将跳转到Paypal
的网站等待用户同意
Plan
创建后,要怎么让用户订阅呢,其实就是创建Agreement
,关于Agreement
,有以下注意点:
- 正如前面所述,
Plan
对象的setSetupFee
方法,设置了完成订阅后首次扣款的费用,而Agreement
对象的循环扣款方法设置的是第2次开始时的费用。 setStartDate
方法设置的是第2次扣款时的时间,因此如果你按月循环,应该是当前时间加一个月,同时该方法要求时间格式是ISO8601
格式,使用Carbon
库可轻松解决;- 在创建
Agreement
的时候,此时还没有生成唯一ID,于是我碰到了一点小困难:那就是当用户完成订阅的时候,我怎么知道这个订阅是哪个用户的?通过Agreement
的getApprovalLink
方法得到的URL,里面的token
是唯一的,我通过提取该token
作为识别方式,在用户完成订阅后替换成真正的ID。
例子参数如下:
$param = [
'id' => 'P-26T36113JT475352643KGIHY',//上一步创建Plan时生成的ID
'name' => 'Standard',
'desc' => 'Standard Plan for one month'
];
代码如下:
public function createPayment($param)
{
$apiContext = $this->getApiContext();
$agreement = new Agreement();
$agreement->setName($param['name'])
->setDescription($param['desc'])
->setStartDate(Carbon::now()->addMonths(1)->toIso8601String());
// Add Plan ID
// Please note that the plan Id should be only set in this case.
$plan = new Plan();
$plan->setId($param['id']);
$agreement->setPlan($plan);
// Add Payer
$payer = new Payer();
$payer->setPaymentMethod('paypal');
$agreement->setPayer($payer);
// For Sample Purposes Only.
$request = clone $agreement;
// ### Create Agreement
try {
// Please note that as the agreement has not yet activated, we wont be receiving the ID just yet.
$agreement = $agreement->create($apiContext);
// ### Get redirect url
// The API response provides the url that you must redirect
// the buyer to. Retrieve the url from the $agreement->getApprovalLink()
// method
$approvalUrl = $agreement->getApprovalLink();
} catch (Exception $ex) {
return "create payment failed, please retry or contact the merchant.";
}
return $approvalUrl;//跳转到$approvalUrl,等待用户同意
}
函数执行后返回$approvalUrl
,记得通过redirect($approvalUrl)
跳转到Paypal
的网站等待用户支付。
用户同意后,执行订阅
用户同意后,订阅还未完成,必须执行Agreement
的execute
方法才算完成真正的订阅。这一步的注意点在于
- 完成订阅后,并不等于扣款,可能会延迟几分钟;
- 如果第一步的
setSetupFee
费用设置为0,则必须等到循环扣款的时间到了才会产生订单;
代码片段如下:
public function onPay($request)
{
$apiContext = $this->getApiContext();
if ($request->has('success') && $request->success == 'true') {
$token = $request->token;
$agreement = new \PayPal\Api\Agreement();
try {
$agreement->execute($token, $apiContext);
} catch(\Exception $e) {
return ull;
return $agreement;
}
return null;
}
获取交易记录
订阅后,可能不会立刻产生交易扣费的交易记录,如果为空则过几分钟再次尝试。本步骤注意点:
start_date
与end_date
不能为空- 实际测试时,该函数返回的对象不能总是返回空的
JSON
对象,因此如果有需要输出JSON
,请根据AgreementTransactions
的API说明,手动取出对应参数。
/** 获取交易记录
* @param $id subscription payment_id
* @warning 总是获取该subscription的所有记录
*/
public function transactions($id)
{
$apiContext = $this->getApiContext();
$params = ['start_date' => date('Y-m-d', strtotime('-15 years')), 'end_date' => date('Y-m-d', strtotime('+5 days'))];
try {
$result = Agreement::searchTransactions($id, $params, $apiContext);
} catch(\Exception $e) {
Log::error("get transactions failed" . $e->getMessage());
return null;
}
return $result->getAgreementTransactionList() ;
}
最后,Paypal官方当然也有对应的教程,不过是调用原生接口的,跟我上面流程不一样点在于只说了前3步,供有兴趣的参考:https://developer.paypal.com/docs/integration/direct/billing-plans-and-agreements/。
需要考虑的问题
功能是实现了,但是也发现不少注意点:
- 国内使用
Sandbox
测试时连接特别慢,经常提示超时或出错,因此需要特别考虑执行中途用户关闭页面的情况; - 一定要实现
webhook
,否则当用户进Paypal
取消订阅时,你的网站将得不到通知; 订阅
(Agreement
)一旦产生,除非主动取消,否则将一直生效。因此如果你的网站设计了多个升级计划(比如Basic
,Standard
,Advanced
),当用户已经订阅某个计划后,去切换升级计划时,开发上必须取消前一个升级计划;- 用户同意订阅-(取消旧订阅-完成新订阅的签约-修改用户信息为新的订阅),括号整个过程 应该是原子操作,同时耗时又长,因此应该将其放到队列中执行直到成功体验会更好。
Paypal如何实现循环扣款(订阅)?的更多相关文章
- Seata 动态配置订阅与降级实现原理
Seata 的动态降级需要结合配置中心的动态配置订阅功能.动态配置订阅,即通过配置中心监听订阅,根据需要读取已更新的缓存值,ZK.Apollo.Nacos 等第三方配置中心都有现成的监听器可实现动态刷 ...
- Angular数据双向绑定
Angular数据双向绑定 AngularJS诞生于2009年,由Misko Hevery 等人创建,后为Google所收购.是一款优秀的前端JS框架,已经被用于Google的多款产品当中.Angul ...
- java Future 模式
考慮這樣一個情況,使用者可能快速翻頁瀏覽文件中,而圖片檔案很大,如此在瀏覽到有圖片的頁數時,就會導致圖片的載入,因而造成使用者瀏覽文件時會有停頓 的現象,所以我們希望在文件開啟之後,仍有一個背景作業持 ...
- Cartographer源码阅读(1):程序入口
带着几个思考问题: (1)IMU数据的使用,如何融合,Kalman滤波? (2)图优化的具体实现,闭环检测的策略? (3)3D激光的接入和闭环策略? 1. 安装Kdevelop工具: http://b ...
- 由浅入深了解EventBus:(四)
事件注册 在EventBus3.0框架中订阅者对事件进行注册/订阅是通过EventBus类中的register方法来实现的,register的方法参数就是我们的订阅者的实例; public void ...
- Eventbus 使用方法和原理分析
对于 Eventbus ,相信很多 Android 小伙伴都用到过. 1.创建事件实体类 所谓的事件实体类,就是传递的事件,一个组件向另一个组件发送的信息可以储存在一个类中,该类就是一个事件,会被 E ...
- vue系列---响应式原理实现及Observer源码解析(一)
_ 阅读目录 一. 什么是响应式? 二:如何侦测数据的变化? 2.1 Object.defineProperty() 侦测对象属性值变化 2.2 如何侦测数组的索引值的变化 2.3 如何监听数组内容的 ...
- Redis(九)高可用专栏之Sentinel模式
本文讲述Redis高可用方案中的哨兵模式--Sentinel,RedisClient中的Jedis如何使用以及使用原理. Redis主从复制 Redis Sentinel模式 Jedis中的Senti ...
- Dubbo(七):redis注册中心的应用
上篇我们讲了Dubbo中有一个非常本质和重要的功能,那就是服务的自动注册与发现,而这个功能是通过注册中心来实现的.上篇中使用zookeeper实现了注册中心的功能,同时了提了dubbo中有其他许多的注 ...
随机推荐
- bootstrap table 的searchParam参数传递
bootstrap table 的searchParam自定义参数传递 Bootstrap Table返回的数据为value 和 rows Long total代表的是多少条(总数) List< ...
- Android 推送
安卓推送方案及比较 http://www.eoe.cn/news/11955.html ******************************************************** ...
- python网络编程--线程(锁,GIL锁,守护线程)
1.线程 1.进程与线程 进程有很多优点,它提供了多道编程,让我们感觉我们每个人都拥有自己的CPU和其他资源,可以提高计算机的利用率.很多人就不理解了,既然进程这么优秀,为什么还要线程呢?其实,仔细观 ...
- WPF 背景网格图
利用DrawingBrush来画出背景网格图 <DrawingBrush Viewport="0,0,80,80" ViewportUnits="Absolute& ...
- 内置函数enumerate()使用
描述 enumerate() 函数用于将一个可遍历的数据对象(如列表.元组或字符串)组合为一个索引序列,同时列出数据和数据下标,一般用在 for 循环当中. Python 2.3. 以上版本可用,2. ...
- 个人常用的win7 快捷键
1.Win + D – 显示桌面 2.Win+L 锁定系统 3.Win + R – 打开运行窗口 4.Win+M 最小化所有窗口 当按下后当前所有窗口全都最小化.再次按下这个组 ...
- kali linux之无线渗透
无线技术变化大,难度大,既新鲜刺激,又压力山大.一半协议 一半理论 无线技术特点: 行业发展迅猛 互联网的重要入口 边界模糊 安全实施缺失而且困难 对技术不了解造成配置不当 企业网络私自接入ap破坏 ...
- Jmeter分布式测试实战
一.Jmeter分布式测试基础 1.Jmeter分布式测试原因: 在使用Jmeter进行接口的性能测试时,由于Jmeter 是JAVA应用,对负载机的CPU和内存消耗比较大.所以当需要模拟数以万计的并 ...
- CTF常见加密方式汇总
1.栅栏密码 在IDF训练营里做过一道关于栅栏密码的问题. 栅栏密码的解法很简单,也有点复杂,字符长度因数多得会有很多个密码.对,栅栏密码的解法就是:计算该字符串是否为合数,若为合数,则求出该合数除本 ...
- 为解决Samba windows 无法访问 尝试过的方法
1, 通过 vi /etc/sysconfig/selinux 把 SELINUX=enforcing 修改为SELINUX= disable 退出保存,并且重启.(设置了) 2, 把wind ...