1、简介
LaravelCashier 为通过 Stripe实现订阅支付服务提供了一个优雅平滑的接口。它封装了几乎所有你恐惧编写的样板化的订阅支付代码。除了基本的订阅管理外,Cashier还支持处理优惠券、订阅升级/替换、订阅“数量”、取消宽限期,甚至生成PDF 发票。
1.1 安装&配置
Composer
首先,添加 Cashier 包到 composer.json文件并运行 composer update命令:
"laravel/cashier": "~6.0"
服务提供者
接下来,在 config/app.php配置文件中注册服务提供者: Laravel\Cashier\CashierServiceProvider。
迁移
使用 Cashier 之前,我们需要准备好数据库。我们需要添加一个字段到 users表,还要创建新的 subscriptions表来处理所有用户订阅:
Schema::table('users', function ($table) { $table->string('stripe_id')->nullable(); $table->string('card_brand')->nullable(); $table->string('card_last_four')->nullable();});Schema::create('subscriptions', function ($table) { $table->increments('id'); $table->integer('user_id'); $table->string('name'); $table->string('stripe_id'); $table->string('stripe_plan'); $table->integer('quantity'); $table->timestamp('trial_ends_at')->nullable(); $table->timestamp('ends_at')->nullable(); $table->timestamps();});
创建好迁移后,只需简单运行 migrate命令,相应修改就会更新到数据库。
设置模型
接下来,添加 Billabletrait 到 User模型类:
use Laravel\Cashier\Billable;class User extends Authenticatable{ use Billable;}
Stripe键
最后,在配置文件 config/services.php中设置 Stripe 键:
'stripe' => [ 'model' => 'User', 'secret' => env('STRIPE_API_SECRET'),],
2、订阅实现
2.1 创建订阅
要创建一个订阅,首先要获取一个账单模型的实例,通常是 App\User的实例。获取到该模型实例之后,你可以使用 newSubscription方法来创建该模型的订阅:
$user = User::find(1);$user->newSubscription('main', 'monthly')->create($creditCardToken);
第一个传递给 newSubscription方法的参数是该订阅的名字,如果应用只有一个订阅,可以将其称作 main或 primary,第二个参数用于指定用户订阅的 Stripe计划,该值对应 Stripe 中相应计划的 id。
create方法会自动创建这个 Stripe 订阅,同时更新数据库中 Stripe 的客户 ID(即 users表中的 stripe_id)和其它相关的账单信息。如果你的订阅计划有试用期,试用期结束时间也会自动被设置到数据库相应字段。
额外的用户信息
如果你想要指定额外的客户信息,你可以将其作为第二个参数传递给 create方法:
$user->newSubscription('main', 'monthly')->create($creditCardToken, [ 'email' => $email, 'description' => 'Our First Customer']);
要了解更多 Stripe 支持的字段,可以查看 Stripe 关于 创建消费者的文档。
优惠券
如果你想要在创建订阅的时候使用优惠券,可以使用 withCoupon方法:
$user->newSubscription('main', 'monthly') ->withCoupon('code') ->create($creditCardToken);
2.2 检查订阅状态
用户订阅你的应用后,你可以使用各种便利的方法来简单检查订阅状态。首先,如果用户有一个有效的订阅,则 subscribed方法返回 true,即使订阅现在出于试用期:
if ($user->subscribed('main')) { //}
subscribed方法还可以用于路由中间件,基于用户订阅状态允许你对路由和控制器的访问进行过滤:
public function handle($request, Closure $next){ if ($request->user() && ! $request->user()->subscribed('main')) { // This user is not a paying customer... return redirect('billing'); } return $next($request);}
如果你想要判断一个用户是否还在试用期,可以使用 onTrial方法,该方法在为还处于试用期的用户显示警告信息很有用:
if ($user->->subscription('main')->onTrial()) { //}
onPlan方法可用于判断用户是否基于 Stripe ID 订阅了给定的计划:
if ($user->onPlan('monthly')) { //}
已取消的订阅状态
要判断用户是否曾经是有效的订阅者,但现在取消了订阅,可以使用 cancelled方法:
if ($user->subscription('main')->cancelled()) { //}
你还可以判断用户是否曾经取消过订阅,但现在仍然在“宽限期”直到完全失效。例如,如果一个用户在3月5号取消了一个实际有效期到3月10号的订阅,该用户处于“宽限期”直到3月10号。注意 subscribed方法在此期间仍然返回 true。
if ($user->subscription('main')->onGracePeriod()) { //}
2.3 修改订阅
用户订阅应用后,偶尔想要改变到新的订阅计划,要将用户切换到新的订阅,使用 swap方法。例如,我们可以轻松切换用户到 premium订阅:
$user = App\User::find(1);$user->subscription('main')->swap('stripe-plan-id');
如果用户在试用,试用期将会被维护。还有,如果订阅存在数量,数量也可以被维护。切换订阅计划后,
可以使用 invoice方法立即给用户开发票:
$user->subscription('main')->swap('stripe-plan-id');$user->invoice();
2.4 订阅数量
有时候订阅也会被数量影响,例如,应用中每个账户每月需要付费$10,要简单增加或减少订阅数量,使用 incrementQuantity和 decrementQuantity方法:
$user = User::find(1);$user->subscription('main')->incrementQuantity();// Add five to the subscription's current quantity...$user->subscription('main')->incrementQuantity(5);$user->subscription('main')->decrementQuantity();// Subtract five to the subscription's current quantity...$user->subscription('main')->decrementQuantity(5);
你也可以使用 updateQuantity方法指定数量:
$user->subscription('main')->updateQuantity(10);
想要了解更多订阅数量信息,查阅相关 Stripe文档。
2.5 订阅税金
在 Cashier 中,提供 tax_percent值发送给 Stripe 很简单。要指定用户支付订阅的税率,实现账单模型的 getTaxPercent方法,并返回一个在0到100之间的数值,不要超过两位小数:
public function getTaxPercent() { return 20;}
这将使你可以在模型基础上使用税率,对跨越不同国家的用户很有用。
2.6 取消订阅
要取消订阅,可以调用用户订阅上的 cancel方法:
$user->subscription('main')->cancel();
当订阅被取消时,Cashier 将会自动设置数据库中的 subscription_ends_at字段。该字段用于了解 subscribed方法什么时候开始返回 false。例如,如果客户3月1号份取消订阅,但订阅直到3月5号才会结束,那么 subscribed方法继续返回 true直到3月5号。
你可以使用 onGracePeriod方法判断用户是否已经取消订阅但仍然在“宽限期”:
if ($user->subscription('main')->onGracePeriod()) { //}
2.7 恢复订阅
如果用户已经取消订阅但想要恢复该订阅,可以使用 resume方法,前提是该用户必须在宽限期内:
$user->subscription('main')->resume();
如果该用户取消了一个订阅然后在订阅失效之前恢复了这个订阅,则不会立即支付该账单,取而代之的,他们的订阅只是被重新激活,并回到正常的支付周期。
3、处理 StripeWebhook
3.1 订阅失败处理
如果客户的信用卡失效怎么办?不用担心—— Cashier 自带了 Webhook 控制器,该控制器可以很方便地为你取消客户订阅。只需要定义如下控制器路由:
Route::post('stripe/webhook', 'Laravel\Cashier\WebhookController@handleWebhook');
就是这样!失败的支付将会被该控制器捕获和处理。当 Stripe 判断订阅失败(正常情况下尝试支付失败三次后)时该控制器将会取消客户的订阅。不要忘了:你需要在 Stripe 控制面板设置中配置相应的 webhook URI,否则不能正常工作。
由于 Stripe webhooks 需要通过 Laravel 的CSRF验证,所以我们将该 URI 置于 VerifyCsrfToken中间件排除列表中:
protected $except = [ 'stripe/*',];
3.2 其它Webhooks
如果你有额外想要处理的 Stripe webhook 事件,只需简单继承 Webhook 控制器, 你的方法名应该和 Cashier 期望的约定一致,尤其是方法应该以“handle”开头并以驼峰命名法命名。例如,如果你想要处理 invoice.payment_succeededwebhook,你应该添加 handleInvoicePaymentSucceeded方法到控制器:
<?phpnamespace App\Http\Controller;use Laravel\Cashier\WebhookController as BaseController;class WebhookController extends BaseController{ /** * 处理 stripe webhook. * * @param array $payload * @return Response */ public function handleInvoicePaymentSucceeded($payload) { // 处理该事件 }}
4、一次性付款
如果你想要使用订阅客户的信用卡一次性结清账单,可以使用账单模型实例上的 charge方法,该方法接收付款金额(应用使用的货币的最小单位对应的金额数值)作为参数,例如,下面的例子使用信用卡支付100美分,或1美元:
$user->charge(100);
charge方法接收一个数组作为第二个参数,允许你传递任何你想要传递的底层 Stripe 账单创建参数:
$user->charge(100, [ 'source' => $token, 'receipt_email' => $user->email,]);
如果支付失败 charge方法将返回 false,这通常表明付款被拒绝:
if ( ! $user->charge(100)) { // The charge was denied...}
如果支付成功,该方法将会返回一个完整的 Stripe 响应。
5、发票
你可以使用 invoices方法轻松获取账单模型的发票数组:
$invoices = $user->invoices();
当列出客户发票时,你可以使用发票的辅助函数来显示相关的发票信息。例如,你可能想要在表格中列出每张发票,从而方便用户下载它们:
<table> @foreach ($invoices as $invoice) <tr> <td>{{ $invoice->dateString() }}</td> <td>{{ $invoice->dollars() }}</td> <td><a href="/user/invoice/{{ $invoice->id }}">Download</a></td> </tr> @endforeach</table>
生成PDF发票
在生成PDF分票之前,需要安装 PHP 库 dompdf:
composer require dompdf/dompdf
在路由或控制器中,使用 downloadInvoice方法生成发票的 PDF 下载,该方法将会自动生成相应的 HTTP 响应发送下载到浏览器:
Route::get('user/invoice/{invoice}', function ($invoiceId) { return Auth::user()->downloadInvoice($invoiceId, [ 'vendor' => 'Your Company', 'product' => 'Your Product', ]);});

PHP仍然流行的原因是其易用性、灵活性和强大的生态系统。1)易用性和简单语法使其成为初学者的首选。2)与web开发紧密结合,处理HTTP请求和数据库交互出色。3)庞大的生态系统提供了丰富的工具和库。4)活跃的社区和开源性质使其适应新需求和技术趋势。

PHP和Python都是高层次的编程语言,广泛应用于Web开发、数据处理和自动化任务。1.PHP常用于构建动态网站和内容管理系统,而Python常用于构建Web框架和数据科学。2.PHP使用echo输出内容,Python使用print。3.两者都支持面向对象编程,但语法和关键字不同。4.PHP支持弱类型转换,Python则更严格。5.PHP性能优化包括使用OPcache和异步编程,Python则使用cProfile和异步编程。

PHP主要是过程式编程,但也支持面向对象编程(OOP);Python支持多种范式,包括OOP、函数式和过程式编程。PHP适合web开发,Python适用于多种应用,如数据分析和机器学习。

PHP起源于1994年,由RasmusLerdorf开发,最初用于跟踪网站访问者,逐渐演变为服务器端脚本语言,广泛应用于网页开发。Python由GuidovanRossum于1980年代末开发,1991年首次发布,强调代码可读性和简洁性,适用于科学计算、数据分析等领域。

PHP适合网页开发和快速原型开发,Python适用于数据科学和机器学习。1.PHP用于动态网页开发,语法简单,适合快速开发。2.Python语法简洁,适用于多领域,库生态系统强大。

PHP在现代化进程中仍然重要,因为它支持大量网站和应用,并通过框架适应开发需求。1.PHP7提升了性能并引入了新功能。2.现代框架如Laravel、Symfony和CodeIgniter简化开发,提高代码质量。3.性能优化和最佳实践进一步提升应用效率。

PHPhassignificantlyimpactedwebdevelopmentandextendsbeyondit.1)ItpowersmajorplatformslikeWordPressandexcelsindatabaseinteractions.2)PHP'sadaptabilityallowsittoscaleforlargeapplicationsusingframeworkslikeLaravel.3)Beyondweb,PHPisusedincommand-linescrip

PHP类型提示提升代码质量和可读性。1)标量类型提示:自PHP7.0起,允许在函数参数中指定基本数据类型,如int、float等。2)返回类型提示:确保函数返回值类型的一致性。3)联合类型提示:自PHP8.0起,允许在函数参数或返回值中指定多个类型。4)可空类型提示:允许包含null值,处理可能返回空值的函数。


热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

MinGW - 适用于 Windows 的极简 GNU
这个项目正在迁移到osdn.net/projects/mingw的过程中,你可以继续在那里关注我们。MinGW:GNU编译器集合(GCC)的本地Windows移植版本,可自由分发的导入库和用于构建本地Windows应用程序的头文件;包括对MSVC运行时的扩展,以支持C99功能。MinGW的所有软件都可以在64位Windows平台上运行。

适用于 Eclipse 的 SAP NetWeaver 服务器适配器
将Eclipse与SAP NetWeaver应用服务器集成。

Dreamweaver Mac版
视觉化网页开发工具

EditPlus 中文破解版
体积小,语法高亮,不支持代码提示功能

安全考试浏览器
Safe Exam Browser是一个安全的浏览器环境,用于安全地进行在线考试。该软件将任何计算机变成一个安全的工作站。它控制对任何实用工具的访问,并防止学生使用未经授权的资源。