Spring Boot集成Stripe快速入门demo

2024-09-01 02:36

本文主要是介绍Spring Boot集成Stripe快速入门demo,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.什么是Stripe?

一体化全球支付平台,开启收入增长引擎,针对不同规模业务打造的支付解决方案,满足从初创公司到跨国企业的多维度需求,助力全球范围内线上线下付款。

  • 转化更多客户: 通过内置的优化功能、100 多种支付方式及一键结账来提高转化率。实现线上和线下付款一体化,提供无缝的客户体验。
  • 全球覆盖,本地体验: 引入多样化支付方式,采用当地货币呈现价格,以此快速拓展新市场,实现向 195+ 个国家/地区的跨境销售,有效降低管理多币种的成本。
  • 减少欺诈,增加收入: Stripe 的机器学习优化功能基于数十亿数据点进行深度训练,为您自动降低欺诈风险,同时提升交易授权率。
  • 降本增效,快速拓展: 通过提高开发人员的工作效率,节省时间和工程资源。凭借领先的可靠性避免宕机,并通过替代支付方式和路径降低成本。

2.代码工程

实验目标

1.实现一次支付功能
2.实现订阅支付功能

pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"><parent><artifactId>springboot-demo</artifactId><groupId>com.et</groupId><version>1.0-SNAPSHOT</version></parent><modelVersion>4.0.0</modelVersion><artifactId>stripe</artifactId><properties><maven.compiler.source>8</maven.compiler.source><maven.compiler.target>8</maven.compiler.target></properties><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-autoconfigure</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency><dependency><groupId>com.sparkjava</groupId><artifactId>spark-core</artifactId><version>2.9.4</version></dependency><dependency><groupId>com.google.code.gson</groupId><artifactId>gson</artifactId><version>2.9.1</version></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><version>1.18.20</version><scope>provided</scope></dependency><dependency><groupId>com.stripe</groupId><artifactId>stripe-java</artifactId><version>25.7.0</version></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-thymeleaf</artifactId></dependency></dependencies>
</project>

controller

package com.et.stripe.controller;import com.et.stripe.common.Response;
import com.et.stripe.service.StripeService;import com.stripe.model.Coupon;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.ResponseBody;@Controller
public class PaymentController {@Value("${stripe.keys.public}")private String API_PUBLIC_KEY;private StripeService stripeService;public PaymentController(StripeService stripeService) {this.stripeService = stripeService;}@GetMapping("/")public String homepage() {return "homepage";}@GetMapping("/subscription")public String subscriptionPage(Model model) {model.addAttribute("stripePublicKey", API_PUBLIC_KEY);return "subscription";}@GetMapping("/charge")public String chargePage(Model model) {model.addAttribute("stripePublicKey", API_PUBLIC_KEY);return "charge";}/*========== REST APIs for Handling Payments ===================*/@PostMapping("/create-subscription")public @ResponseBodyResponse createSubscription(String email, String token, String plan, String coupon) {//validate dataif (token == null || plan.isEmpty()) {return new Response(false, "Stripe payment token is missing. Please, try again later.");}//create customer firstString customerId = stripeService.createCustomer(email, token);if (customerId == null) {return new Response(false, "An error occurred while trying to create a customer.");}//create subscriptionString subscriptionId = stripeService.createSubscription(customerId, plan, coupon);if (subscriptionId == null) {return new Response(false, "An error occurred while trying to create a subscription.");}// Ideally you should store customerId and subscriptionId along with customer object here.// These values are required to update or cancel the subscription at later stage.return new Response(true, "Success! Your subscription id is " + subscriptionId);}@PostMapping("/cancel-subscription")public @ResponseBodyResponse cancelSubscription(String subscriptionId) {boolean status = stripeService.cancelSubscription(subscriptionId);if (!status) {return new Response(false, "Failed to cancel the subscription. Please, try later.");}return new Response(true, "Subscription cancelled successfully.");}@PostMapping("/coupon-validator")public @ResponseBodyResponse couponValidator(String code) {Coupon coupon = stripeService.retrieveCoupon(code);if (coupon != null && coupon.getValid()) {String details = (coupon.getPercentOff() == null ? "$" + (coupon.getAmountOff() / 100) : coupon.getPercentOff() + "%") +" OFF " + coupon.getDuration();return new Response(true, details);} else {return new Response(false, "This coupon code is not available. This may be because it has expired or has " +"already been applied to your account.");}}@PostMapping("/create-charge")public @ResponseBodyResponse createCharge(String email, String token) {//validate dataif (token == null) {return new Response(false, "Stripe payment token is missing. Please, try again later.");}//create chargeString chargeId = stripeService.createCharge(email, token, 999); //$9.99 USDif (chargeId == null) {return new Response(false, "An error occurred while trying to create a charge.");}// You may want to store charge id along with order informationreturn new Response(true, "Success! Your charge id is " + chargeId);}
}

service

package com.et.stripe.service;import com.stripe.Stripe;
import com.stripe.model.Charge;
import com.stripe.model.Coupon;
import com.stripe.model.Customer;
import com.stripe.model.Subscription;
import com.stripe.param.SubscriptionCancelParams;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;import java.util.HashMap;
import java.util.Map;@Service
public class StripeService {@Value("${stripe.keys.secret}")private String API_SECRET_KEY;public StripeService() {}public String createCustomer(String email, String token) {String id = null;try {Stripe.apiKey = API_SECRET_KEY;Map<String, Object> customerParams = new HashMap<>();// add customer unique id here to track them in your web applicationcustomerParams.put("description", "Customer for " + email);customerParams.put("email", email);customerParams.put("source", token); // ^ obtained with Stripe.js//create a new customerCustomer customer = Customer.create(customerParams);id = customer.getId();} catch (Exception ex) {ex.printStackTrace();}return id;}public String createSubscription(String customerId, String plan, String coupon) {String id = null;try {Stripe.apiKey = API_SECRET_KEY;Map<String, Object> item = new HashMap<>();item.put("price", plan);Map<String, Object> items = new HashMap<>();items.put("0", item);Map<String, Object> params = new HashMap<>();params.put("customer", customerId);params.put("items", items);//add coupon if availableif (!coupon.isEmpty()) {params.put("coupon", coupon);}Subscription sub = Subscription.create(params);id = sub.getId();} catch (Exception ex) {ex.printStackTrace();}return id;}public boolean cancelSubscription(String subscriptionId) {boolean status;try {Stripe.apiKey = API_SECRET_KEY;Subscription sub = Subscription.retrieve(subscriptionId);sub.cancel((SubscriptionCancelParams) null);status = true;} catch (Exception ex) {ex.printStackTrace();status = false;}return status;}public Coupon retrieveCoupon(String code) {try {Stripe.apiKey = API_SECRET_KEY;return Coupon.retrieve(code);} catch (Exception ex) {ex.printStackTrace();}return null;}public String createCharge(String email, String token, int amount) {String id = null;try {Stripe.apiKey = API_SECRET_KEY;Map<String, Object> chargeParams = new HashMap<>();chargeParams.put("amount", amount);chargeParams.put("currency", "usd");chargeParams.put("description", "Charge for " + email);chargeParams.put("source", token); // ^ obtained with Stripe.js//create a chargeCharge charge = Charge.create(chargeParams);id = charge.getId();} catch (Exception ex) {ex.printStackTrace();}return id;}}

templates

homepage

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head><title>Homepage</title><!--Bootstrap 4 CSS--><link href="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/css/bootstrap.min.css" rel="stylesheet"><!--Bootstrap 4 JavaScript--><script src="https://code.jquery.com/jquery-3.3.1.min.js"></script><script src="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/js/bootstrap.min.js"></script>
</head>
<body class="bg-light pt-5"><!--hero section-->
<section class="py-5"><div class="container"><div class="row"><div class="col-lg-7 col-md-10 col-12 my-auto mx-auto text-center"><h1>Stripe Payment Examples</h1><p class="lead mb-4">What would you like to do?</p><a class="btn btn-primary" th:href="@{/subscription}">Create Recurring Subscription</a><a class="btn btn-success" th:href="@{/charge}">Create One-Time Charge</a><p class="mt-5 text-muted"><small>An example project by <a target="_blank"th:href="@{https://attacomsian.com}">Atta</a>.</small></p></div></div></div>
</section></body>
</html>

charge

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head><title>Charge</title><!--Bootstrap 4 CSS--><link href="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/css/bootstrap.min.css" rel="stylesheet"><!--Bootstrap 4 JavaScript--><script src="https://code.jquery.com/jquery-3.3.1.min.js"></script><script src="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/js/bootstrap.min.js"></script><!--Stripe JavaScript Library--><script src="https://js.stripe.com/v3/"></script>
</head>
<body class="bg-light pt-5"><!--hero section-->
<section class="py-5"><div class="container"><div class="row"><div class="col-lg-6 col-md-8 col-12 my-auto mx-auto"><h1>Stripe One-Time Charge</h1><p class="lead mb-4">Please fill the form below to complete the order payment</p><div class="card mb-4"><div class="card-body"><h5>Leather Bag</h5><p class="lead">USD 9.99</p></div></div><form action="#" id="payment-form" method="post"><input id="api-key" th:value="${stripePublicKey}" type="hidden"><div class="form-group"><label class="font-weight-medium" for="card-element">Enter credit or debit card below</label><div class="w-100" id="card-element"><!-- A Stripe Element will be inserted here. --></div></div><div class="form-group"><input class="form-control" id="email" name="email"placeholder="Email Address" required type="email"></div><!-- Used to display Element errors. --><div class="text-danger w-100" id="card-errors" role="alert"></div><div class="form-group pt-2"><button class="btn btn-primary btn-block" id="submitButton" type="submit">Pay With Your Card</button><div class="small text-muted mt-2">Pay securely with Stripe. By clicking the button above, you agreeto our <a href="#" target="_blank">Terms of Service</a>,<a href="#" target="_blank">Privacy</a> and<a href="#" target="_blank">Refund</a> policies.</div></div></form><p class="mt-5 text-muted"><small>An example project by <a target="_blank" th:href="@{https://attacomsian.com}">Atta</a>.</small></p></div></div></div>
</section><!--custom javascript for handling subscription-->
<script>$(function () {var API_KEY = $('#api-key').val();// Create a Stripe client.var stripe = Stripe(API_KEY);// Create an instance of Elements.var elements = stripe.elements();// Create an instance of the card Element.var card = elements.create('card');// Add an instance of the card Element into the `card-element` <div>.card.mount('#card-element');// Handle real-time validation errors from the card Element.card.addEventListener('change', function (event) {var displayError = document.getElementById('card-errors');if (event.error) {displayError.textContent = event.error.message;} else {displayError.textContent = '';}});// Handle form submission.var form = document.getElementById('payment-form');form.addEventListener('submit', function (event) {event.preventDefault();// handle paymenthandlePayments();});//handle card submissionfunction handlePayments() {stripe.createToken(card).then(function (result) {if (result.error) {// Inform the user if there was an error.var errorElement = document.getElementById('card-errors');errorElement.textContent = result.error.message;} else {// Send the token to your server.var token = result.token.id;var email = $('#email').val();$.post("/create-charge",{email: email, token: token},function (data) {alert(data.details);}, 'json');}});}});
</script></body>
</html>

subscription

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head><title>Subscription</title><!--Bootstrap 4 CSS--><link href="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/css/bootstrap.min.css" rel="stylesheet"><!--Bootstrap 4 JavaScript--><script src="https://code.jquery.com/jquery-3.3.1.min.js"></script><script src="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/js/bootstrap.min.js"></script><!--Stripe JavaScript Library--><script src="https://js.stripe.com/v3/"></script>
</head>
<body class="bg-light pt-5"><!--hero section-->
<section class="py-5"><div class="container"><div class="row"><div class="col-lg-6 col-md-8 col-12 my-auto mx-auto"><h1>Stripe Recurring Subscription</h1><p class="lead mb-4">Please fill the form below to complete the payment</p><h5 class="mb-2">Choose your payment plan</h5><p class="text-muted">60% OFF when you upgrade to annual plan.</p><div class="py-2"><div class="custom-control custom-radio"><input class="custom-control-input" id="monthly-plan" name="premium-plan" type="radio"value="price_1PtRC5KYHXnPEBSj85nrvuKC"/><label class="custom-control-label" for="monthly-plan"><strong>Monthly $9.99</strong><br/><small class="text-muted">Pay $9.99 every month and get access to all premium features.</small></label></div><div class="custom-control custom-radio mt-3"><input checked="" class="custom-control-input" id="annually-plan" name="premium-plan"type="radio" value="annual-subscription"/><label class="custom-control-label" for="annually-plan"><strong>Yearly $49.99</strong><span class="badge badge-primary ml-1">60% OFF</span><br/><small class="text-muted">Pay $49.99 every year and get access to all premium features.</small></label></div></div><form action="#" id="payment-form" method="post"><input id="api-key" th:value="${stripePublicKey}" type="hidden"><div class="form-group"><label class="font-weight-medium" for="card-element">Enter credit or debit card below</label><div class="w-100" id="card-element"><!-- A Stripe Element will be inserted here. --></div></div><div class="form-group"><input class="form-control" id="email" name="email"placeholder="Email Address" required type="email"></div><div class="form-group"><input class="form-control" id="coupon" name="coupon"placeholder="Coupon code (optional)" type="text"></div><!-- Used to display Element errors. --><div class="text-danger w-100" id="card-errors" role="alert"></div><div class="form-group pt-2"><button class="btn btn-primary btn-block" id="submitButton" type="submit">Pay With Your Card</button><div class="small text-muted mt-2">Pay securely with Stripe. By clicking the button above, you agreeto our <a href="#" target="_blank">Terms of Service</a>,<a href="#" target="_blank">Privacy</a> and<a href="#" target="_blank">Refund</a> policies.</div></div></form><p class="mt-5 text-muted"><small>An example project by <a target="_blank" th:href="@{https://attacomsian.com}">Atta</a>.</small></p></div></div></div>
</section><!--custom javascript for handling subscription-->
<script>$(function () {var API_KEY = $('#api-key').val();// Create a Stripe client.var stripe = Stripe(API_KEY);// Create an instance of Elements.var elements = stripe.elements();// Create an instance of the card Element.var card = elements.create('card');// Add an instance of the card Element into the `card-element` <div>.card.mount('#card-element');// Handle real-time validation errors from the card Element.card.addEventListener('change', function (event) {var displayError = document.getElementById('card-errors');if (event.error) {displayError.textContent = event.error.message;} else {displayError.textContent = '';}});// Handle form submission.var form = document.getElementById('payment-form');form.addEventListener('submit', function (event) {event.preventDefault();//validate coupon if anyvar code = $('#coupon').val().trim();if (code.length > 0) {$.post("/coupon-validator",{code: code},function (data) {if (data.status) {handlePayments();} else {alert(data.details);}}, 'json');} else {handlePayments();}});//handle card submissionfunction handlePayments() {stripe.createToken(card).then(function (result) {if (result.error) {// Inform the user if there was an error.var errorElement = document.getElementById('card-errors');errorElement.textContent = result.error.message;} else {// Send the token to your server.var token = result.token.id;var plan = $('input[name="premium-plan"]:checked').val();var email = $('#email').val();var coupon = $('#coupon').val();$.post("/create-subscription",{email: email, token: token, plan: plan, coupon: coupon},function (data) {alert(data.details);}, 'json');}});}});
</script></body>
</html>

application.yaml

#Stripe keys - REPLACE WITH ACTUAL KEYS
stripe.keys.public=pk_test_xxxxx
stripe.keys.secret=sk_test_xxxxx
#Don't cache thymeleaf files - FOR TEST PURPOSE ONLY
spring.thymeleaf.cache=false

以上只是一些关键代码,所有代码请参见下面代码仓库

代码仓库

  • GitHub - Harries/springboot-demo: a simple springboot demo with some components for example: redis,solr,rockmq and so on.(stripe)

3.测试

启动Spring Boot应用

一次支付测试

http://127.0.0.1:8080/charge

charge

可以在这个地址 Test card numbers | Stripe Documentation,找到测试卡号, 输入测试卡号,显示支付成功

charge-success

订阅支付测试

http://127.0.0.1:8080/subscription

subscription

输入测试卡号,显示订阅成功(注意:是订阅产品上创建多个价格,而不是创建多个产品)

subscription-success

stripe控制台交易信息

transactions

4.引用

  • Test card numbers | Stripe Documentation
  • Stripe | Financial Infrastructure to Grow Your Revenue
  • Spring Boot集成Stripe快速入门demo | Harries Blog™

这篇关于Spring Boot集成Stripe快速入门demo的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1125697

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Java中的.close()举例详解

《Java中的.close()举例详解》.close()方法只适用于通过window.open()打开的弹出窗口,对于浏览器的主窗口,如果没有得到用户允许是不能关闭的,:本文主要介绍Java中的.... 目录当你遇到以下三种情况时,一定要记得使用 .close():用法作用举例如何判断代码中的 input

Spring Gateway动态路由实现方案

《SpringGateway动态路由实现方案》本文主要介绍了SpringGateway动态路由实现方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随... 目录前沿何为路由RouteDefinitionRouteLocator工作流程动态路由实现尾巴前沿S