支付宝支付
入门
"""
1)支付宝API:六大接口
https://docs.open.alipay.com/270/105900/2)支付宝工作流程(见下图):
https://docs.open.alipay.com/270/105898/3)支付宝8次异步通知机制(支付宝对我们服务器发送POST请求,索要 success 7个字符)
https://docs.open.alipay.com/270/105902/
"""
"""
# alipay_public_key.pem
-----BEGIN PUBLIC KEY-----
支付宝公钥
-----END PUBLIC KEY-----# app_private_key.pem
-----BEGIN RSA PRIVATE KEY-----
用户私钥
-----END RSA PRIVATE KEY-----
"""
"""
开发:https://openapi.alipay.com/gateway.do
沙箱:https://openapi.alipaydev.com/gateway.do
"""
支付流程
aliapy二次封装包
GitHub开源框架
https://github.com/fzlee/alipay
依赖
>: pip install python-alipay-sdk --upgrade
>: pip install pyopenssl
结构
libs├── iPay │ ├── __init__.py │ ├── pem │ │ ├── alipay_public_key.pem │ │ ├── app_private_key.pem │ ├── pay.py └── └── settings.py
alipay_public_key.pem
-----BEGIN PUBLIC KEY-----
拿应用公钥跟支付宝换来的支付宝公钥
-----END PUBLIC KEY-----
app_private_key.pem
-----BEGIN RSA PRIVATE KEY-----
通过支付宝公钥私钥签发软件签发的应用私钥
-----END RSA PRIVATE KEY-----
setting.py
import os
APP_PRIVATE_KEY_STRING = open(os.path.join(os.path.dirname(os.path.abspath(__file__)), 'pem', 'app_private_key.pem')).read()
ALIPAY_PUBLIC_KEY_STRING = open(os.path.join(os.path.dirname(os.path.abspath(__file__)), 'pem', 'alipay_public_key.pem')).read()
APP_ID = '2016093000631831'
SIGN = 'RSA2'
DEBUG = True
GATEWAY = 'https://openapi.alipaydev.com/gateway.do' if DEBUG else 'https://openapi.alipay.com/gateway.do'
pay.py
from alipay import AliPay
from . import settings
alipay = AliPay(appid=settings.APP_ID,app_notify_url=None,app_private_key_string=settings.APP_PRIVATE_KEY_STRING,alipay_public_key_string=settings.ALIPAY_PUBLIC_KEY_STRING,sign_type=settings.SIGN,debug=settings.DEBUG
)
gateway = settings.GATEWAY
*init*.py
from .pay import gateway, alipay
补充:在自己项目的配置文件中配置支付宝回调接口:settings.py | dev.py
BASE_URL = 'http://127.0.0.1:8000'
LUFFY_URL = 'http://127.0.0.1:8080'
NOTIFY_URL = BASE_URL + "/order/success/"
RETURN_URL = LUFFY_URL + "/pay/success"
后台 - 支付接口
路由:order/urls.py
from django.urls import path, include
from utils.router import router
from . import views"""
1)支付接口(需要登录认证:是谁):前台提交商品等信息,得到支付链接post方法分析:支付宝回调同步:get给前台 => 前台可以在收到支付宝同步get回调时,ajax异步在给消息同步给后台,也采用get,后台处理前台的get请求异步:post给后台 => 后台直接处理支付宝的post请求
2)支付回调接口(不需要登录认证:哪个订单(订单信息中有非对称加密)、支付宝压根不可能有你的token):get方法:处理前台来的同步回调(不一定能收得到,所有不能在该方法完成后台订单状态等信息操作)post方法:处理支付宝来的异步回调3)订单状态确认接口:随你前台任何时候来校验订单状态的接口
"""
router.register('pay', views.PayViewSet, 'pay')urlpatterns = [path('', include(router.urls)),path('success/', views.SuccessViewSet.as_view({'get': 'get', 'post': 'post'}))
]
模型表:order/models.py
"""
class Order(models.Model):# 主键、总金额、订单名、订单号、订单状态、创建时间、支付时间、流水号、支付方式、支付人(外键) - 优惠劵(外键,可为空)passclass OrderDetail(models.Model):# 订单号(外键)、商品(外键)、实价、成交价 - 商品数量pass
"""
from django.db import models
from user.models import User
from course.models import Courseclass Order(models.Model):"""订单模型"""status_choices = ((0, '未支付'),(1, '已支付'),(2, '已取消'),(3, '超时取消'),)pay_choices = ((1, '支付宝'),(2, '微信支付'),)subject = models.CharField(max_length=150, verbose_name="订单标题")total_amount = models.DecimalField(max_digits=10, decimal_places=2, verbose_name="订单总价", default=0)out_trade_no = models.CharField(max_length=64, verbose_name="订单号", unique=True)trade_no = models.CharField(max_length=64, null=True, verbose_name="流水号")order_status = models.SmallIntegerField(choices=status_choices, default=0, verbose_name="订单状态")pay_type = models.SmallIntegerField(choices=pay_choices, default=1, verbose_name="支付方式")pay_time = models.DateTimeField(null=True, verbose_name="支付时间")user = models.ForeignKey(User, related_name='order_user', on_delete=models.DO_NOTHING, db_constraint=False, verbose_name="下单用户")created_time = models.DateTimeField(auto_now_add=True, verbose_name='创建时间')class Meta:db_table = "luffy_order"verbose_name = "订单记录"verbose_name_plural = "订单记录"def __str__(self):return "%s - ¥%s" % (self.subject, self.total_amount)@propertydef courses(self):data_list = []for item in self.order_courses.all():data_list.append({"id": item.id,"course_name": item.course.name,"real_price": item.real_price,})return data_listclass OrderDetail(models.Model):"""订单详情"""order = models.ForeignKey(Order, related_name='order_courses', on_delete=models.CASCADE, db_constraint=False, verbose_name="订单")course = models.ForeignKey(Course, related_name='course_orders', on_delete=models.CASCADE, db_constraint=False, verbose_name="课程")price = models.DecimalField(max_digits=6, decimal_places=2, verbose_name="课程原价")real_price = models.DecimalField(max_digits=6, decimal_places=2, verbose_name="课程实价")class Meta:db_table = "luffy_order_detail"verbose_name = "订单详情"verbose_name_plural = "订单详情"def __str__(self):try:return "%s的订单:%s" % (self.course.name, self.order.out_trade_no)except:return super().__str__()
支付接口类:order/views.py
from rest_framework.viewsets import GenericViewSet, ViewSet
from rest_framework.mixins import CreateModelMixin
from rest_framework.permissions import IsAuthenticated
from rest_framework.response import Response
from . import models, serializers
class PayViewSet(GenericViewSet, CreateModelMixin):permission_classes = [IsAuthenticated]queryset = models.Order.objects.all()serializer_class = serializers.PaySerializerdef create(self, request, *args, **kwargs):serializer = self.get_serializer(data=request.data, context={'request': request})serializer.is_valid(raise_exception=True)self.perform_create(serializer)return Response(serializer.context['pay_url'])
支付接口序列化类:model/serializers
from rest_framework import serializers
from . import models
from course.models import Course
from rest_framework.exceptions import ValidationError
from django.conf import settingsclass PaySerializer(serializers.ModelSerializer):courses = serializers.PrimaryKeyRelatedField(queryset=Course.objects.all(), write_only=True, many=True)class Meta:model = models.Orderfields = ('subject', 'total_amount', 'pay_type', 'courses')extra_kwargs = {'total_amount': {'required': True},'pay_type': {'required': True},}def _check_total_amount(self, attrs):courses = attrs.get('courses')total_amount = attrs.get('total_amount')total_price = 0for course in courses:total_price += course.priceif total_price != total_amount:raise ValidationError('total_amount error')return total_amountdef _get_out_trade_no(self):import uuidcode = '%s' % uuid.uuid4()return code.replace('-', '')def _get_user(self):return self.context.get('request').userdef _get_pay_url(self, out_trade_no, total_amount, subject):from libs import iPayorder_string = iPay.alipay.api_alipay_trade_page_pay(out_trade_no=out_trade_no,total_amount=float(total_amount), subject=subject,return_url=settings.RETURN_URL,notify_url=settings.NOTIFY_URL,)pay_url = iPay.gateway + '?' + order_stringself.context['pay_url'] = pay_urldef _before_create(self, attrs, user, out_trade_no):attrs['user'] = userattrs['out_trade_no'] = out_trade_nodef validate(self, attrs):total_amount = self._check_total_amount(attrs)out_trade_no = self._get_out_trade_no()user = self._get_user()self._get_pay_url(out_trade_no, total_amount, attrs.get('subject'))self._before_create(attrs, user, out_trade_no)return attrsdef create(self, validated_data):courses = validated_data.pop('courses')order = models.Order.objects.create(**validated_data)for course in courses:models.OrderDetail.objects.create(order=order, course=course, price=course.price, real_price=course.price)return order
前台 - 支付生成页面
课程主页或是详情页或者搜索页
<template>...<span class="buy-now" @click="buy_course(course)">立即购买</span>
</template>
<script>export default {methods: {// 购买课程buy_course(course) {// es6语法下,变量必须定义才能使用let token = this.$cookies.get('token');if (!token) {this.$message({message: '请先登录',type: 'warning'});return false}this.$axios({url: this.$settings.base_url + '/order/pay/',method: 'post',headers: {authorization: 'luffy ' + token,},data: {subject: course.name,total_amount: course.price,pay_type: 1, // 现在只能默认1,为支付宝courses: [course.id] // 后台完成的是基于购物车情况下的接口,所以单购物为群购1个}}).then(response => {let pay_url = response.data;// console.log(pay_url)// 本页面标签调整:可以选择 _self 或 _blankopen(pay_url, '_self');}).catch(error => {console.log(error.response.data)})},}}
</script>
前台 - 支付成功的回调页面
router/index.js
import PaySuccess from '../views/PaySuccess.vue'
// ...
const routes = [// ...{path: '/pay/success',name: 'pay-success',component: PaySuccess},
];
同步回调的参数
`
charset=utf-8&out_trade_no=7f7c7d12d57d45b693e1b49a6b01e1dd&method=alipay.trade.page.pay.return&total_amount=39.00&sign=FUmceqiNMWvxcD%2BUPCHiOTaEwlJ%2FXIXL5UwZWOSI1TwRjPIZVzjRLB4j2G5CQpn472JO8X%2BwMx04dHqjLxqLcY3TRu0XurQ%2FwKTNpyfDrtNuNv0rfGPuVHw52y3blbS7%2FKFVsWryw4%2BBuF2fCrJ4qWH8Zg14Rct7qoMbu73N74WkQtDyzXefiKDbkMMRMfLbelE9TFyeIeygeMId8%2B58mcJMUOh6aQqwpr9bzuBbfJ17fkqU%2F0ys9zGr%2FlDtLL7aAh6BPViqZN%2F9T7byCoferD1BhcSzJNR6V6VuhOdTq8iEaH2XgJT9aIiyHgg3GT1taBBvZX2gK41FSmkguk%2BfsA%3D%3D&trade_no=2020030722001464020500585462&auth_app_id=2016093000631831&version=1.0&app_id=2016093000631831&sign_type=RSA2&seller_id=2088102177958114×tamp=2020-03-07%2014%3A47%3A48
`
// 同步回调没与订单状态
views/PaySuccess.vue
<template><div class="pay-success"><!--如果是单独的页面,就没必要展示导航栏(带有登录的用户)--><Header/><div class="main"><div class="title"><div class="success-tips"><p class="tips">您已成功购买 1 门课程!</p></div></div><div class="order-info"><p class="info"><b>订单号:</b><span>{{ result.out_trade_no }}</span></p><p class="info"><b>交易号:</b><span>{{ result.trade_no }}</span></p><p class="info"><b>付款时间:</b><span><span>{{ result.timestamp }}</span></span></p></div><div class="study"><span>立即学习</span></div></div></div>
</template><script>import Header from "@/components/Header"export default {name: "Success",data() {return {result: {},};},created() {// url后拼接的参数:?及后面的所有参数 => ?a=1&b=2// console.log(location.search);// 解析支付宝回调的url参数let params = location.search.substring(1); // 去除? => a=1&b=2let items = params.length ? params.split('&') : []; // ['a=1', 'b=2']//逐个将每一项添加到args对象中for (let i = 0; i < items.length; i++) { // 第一次循环a=1,第二次b=2let k_v = items[i].split('='); // ['a', '1']//解码操作,因为查询字符串经过编码的if (k_v.length >= 2) {// url编码反解let k = decodeURIComponent(k_v[0]);this.result[k] = decodeURIComponent(k_v[1]);// 没有url编码反解// this.result[k_v[0]] = k_v[1];}}// 解析后的结果// console.log(this.result);// 把地址栏上面的支付结果,再get请求转发给后端this.$axios({url: this.$settings.base_url + '/order/success/' + location.search,method: 'get',}).then(response => {console.log(response.data);}).catch(() => {console.log('支付结果同步失败');})},components: {Header,}}
</script><style scoped>.main {padding: 60px 0;margin: 0 auto;width: 1200px;background: }.main .title {display: flex;-ms-flex-align: center;align-items: center;padding: 25px 40px;border-bottom: 1px solid }.main .title .success-tips {box-sizing: border-box;}.title img {vertical-align: middle;width: 60px;height: 60px;margin-right: 40px;}.title .success-tips {box-sizing: border-box;}.title .tips {font-size: 26px;color: }.info span {color: }.order-info {padding: 25px 48px;padding-bottom: 15px;border-bottom: 1px solid }.order-info p {display: -ms-flexbox;display: flex;margin-bottom: 10px;font-size: 16px;}.order-info p b {font-weight: 400;color: white-space: nowrap;}.study {padding: 25px 40px;}.study span {display: block;width: 140px;height: 42px;text-align: center;line-height: 42px;cursor: pointer;background: border-radius: 6px;font-size: 16px;color: }
</style>
后台 - 支付成功的回调接口
from utils.logging import logger
class SuccessViewSet(ViewSet):authentication_classes = ()permission_classes = ()def get(self, request, *args, **kwargs):out_trade_no = request.query_params.get('out_trade_no')try:models.Order.objects.get(out_trade_no=out_trade_no, order_status=1)return APIResponse(result=True)except:return APIResponse(1, 'error', result=False)def post(self, request, *args, **kwargs):try:result_data = request.data.dict()out_trade_no = result_data.get('out_trade_no')signature = result_data.pop('sign')from libs import iPayresult = iPay.alipay.verify(result_data, signature)if result and result_data["trade_status"] in ("TRADE_SUCCESS", "TRADE_FINISHED"):models.Order.objects.filter(out_trade_no=out_trade_no).update(order_status=1)logger.warning('%s订单支付成功' % out_trade_no)return Response('success')else:logger.error('%s订单支付失败' % out_trade_no)except:passreturn Response('failed')