支付宝接入

支付宝接入

python-alipay-sdk
pycryptodome

一、电脑网站支付

1.1 获取支付宝密钥

沙箱网址

1.APPID
2.应用私钥
3.支付宝公钥

在这里插入图片描述

在这里插入图片描述

1.2 存放密钥

  1. 在与 settings.py 的同级目录下创建 pem 文件夹
  2. pem 文件夹下创建 app_private_key.pemalipay_public_key.pem
    • app_private_key.pem :存放应用私钥(选择非Java语言)
    • alipay_public_key.pem :存放支付宝公钥
  3. 注意存放格式!!!

在这里插入图片描述

支付宝公钥

-----BEGIN PUBLIC KEY-----
MIIBIjANBgkxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
-----END PUBLIC KEY-----

应用私钥

-----BEGIN RSA PRIVATE KEY-----
MIIEogIBAAKCAQEAgZXESCF0wkPXCKjLxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
-----END RSA PRIVATE KEY-----

1.3 配置 settings

settings.py

import os
# 应用私钥(选择非Java语言)pem 是文件夹名称
APP_PRIVATE_KEY_PATH = os.path.join(os.path.dirname(os.path.abspath(__file__)), 'pem', 'app_private_key.pem')
# 支付宝公钥
ALIPAY_PUBLIC_KEY_PATH = os.path.join(os.path.dirname(os.path.abspath(__file__)), 'pem', 'alipay_public_key.pem')# 沙箱应用的 APPID --> https://open.alipay.com/develop/sandbox/app
APP_ID = '20xxxx9'# 加密方式
SIGN = 'RSA2'# 是否是支付宝测试环境(沙箱环境),如果采用真是支付宝环境,配置False
DEBUG = True# 支付网关
GATEWAY = 'https://openapi.alipaydev.com/gateway.do' if DEBUG else 'https://openapi.alipay.com/gateway.do'BASE_URL = 'http://127.0.0.1:8000'  # dajngo 服务器
LUFFY_URL = 'http://127.0.0.1:3000'  # react 前端地址NOTIFY_URL = BASE_URL + "/order/success/"  # 付款后支付宝向 http://127.0.0.1:8000/order/success/ 发送post请求
RETURN_URL = LUFFY_URL + "/#/pay/success"  # 付款后支付宝向 http://127.0.0.1:3000/#/pay/success  发送get请求,请求前端的支付成功页面

在这里插入图片描述

理论知识(必看)

1.用户选中订单点击支付,向Dajngo后端发送请求,Django后端接口根据商品信息生成能够唤醒支付界面的 pay_url,前端接收并打开pay_url
2.用户登录支付宝,并成功支付后
3.第三方服务器【支付宝】向我们配置的return_url发送一个GET请求
4.同时向notify_url发送一个异步回调POST请求

注意

  • RETURN_URL 和 NOTIFY_URL 必须能够使用公网访问
  • 公网访问:其它电脑能访问你设置的 url
  • 借助内网穿透工具即可natapp内网穿透应用

auth.py

# 1、放行的网址
if request.path_info in ['/login/', '/register/', '/order/success/']:return

注意

如果你的Django项目设置了拦截的话,请放行notify_url地址,因为支付宝发送过来的请求不带认证令牌!

aliPayConfig.py

from datetime import datetime
from Crypto.PublicKey import RSA
from Crypto.Signature import PKCS1_v1_5
from Crypto.Hash import SHA256
from urllib.parse import quote_plus
from urllib.parse import urlparse, parse_qs
from base64 import decodebytes, encodebytes
import jsonclass AliPay(object):"""支付宝支付接口(PC端支付接口)"""def __init__(self, appid, app_notify_url, app_private_key_path,alipay_public_key_path, return_url, debug=False):self.appid = appidself.app_notify_url = app_notify_urlself.app_private_key_path = app_private_key_pathself.app_private_key = Noneself.return_url = return_urlwith open(self.app_private_key_path) as fp:self.app_private_key = RSA.importKey(fp.read())self.alipay_public_key_path = alipay_public_key_pathwith open(self.alipay_public_key_path) as fp:self.alipay_public_key = RSA.importKey(fp.read())if debug is True:self.__gateway = "https://openapi.alipaydev.com/gateway.do"else:self.__gateway = "https://openapi.alipay.com/gateway.do"def direct_pay(self, subject, out_trade_no, total_amount, return_url=None, **kwargs):biz_content = {"subject": subject,"out_trade_no": out_trade_no,"total_amount": total_amount,"product_code": "FAST_INSTANT_TRADE_PAY",  # 默认不需要修改# "qr_pay_mode":4}biz_content.update(kwargs)data = self.build_body("alipay.trade.page.pay", biz_content, self.return_url)return self.sign_data(data)def build_body(self, method, biz_content, return_url=None):data = {"app_id": self.appid,"method": method,"charset": "utf-8","sign_type": "RSA2","timestamp": datetime.now().strftime("%Y-%m-%d %H:%M:%S"),"version": "1.0","biz_content": biz_content}if return_url is not None:data["notify_url"] = self.app_notify_urldata["return_url"] = self.return_urlreturn datadef sign_data(self, data):data.pop("sign", None)# 排序后的字符串unsigned_items = self.ordered_data(data)unsigned_string = "&".join("{0}={1}".format(k, v) for k, v in unsigned_items)sign = self.sign(unsigned_string.encode("utf-8"))# ordered_items = self.ordered_data(data)quoted_string = "&".join("{0}={1}".format(k, quote_plus(v)) for k, v in unsigned_items)# 获得最终的订单信息字符串signed_string = quoted_string + "&sign=" + quote_plus(sign)return signed_stringdef ordered_data(self, data):complex_keys = []for key, value in data.items():if isinstance(value, dict):complex_keys.append(key)# 将字典类型的数据dump出来for key in complex_keys:data[key] = json.dumps(data[key], separators=(',', ':'))return sorted([(k, v) for k, v in data.items()])def sign(self, unsigned_string):# 开始计算签名key = self.app_private_keysigner = PKCS1_v1_5.new(key)signature = signer.sign(SHA256.new(unsigned_string))# base64 编码,转换为unicode表示并移除回车sign = encodebytes(signature).decode("utf8").replace("\n", "")return signdef _verify(self, raw_content, signature):# 开始计算签名key = self.alipay_public_keysigner = PKCS1_v1_5.new(key)digest = SHA256.new()digest.update(raw_content.encode("utf8"))if signer.verify(digest, decodebytes(signature.encode("utf8"))):return Truereturn Falsedef verify(self, data, signature):if "sign_type" in data:sign_type = data.pop("sign_type")# 排序后的字符串unsigned_items = self.ordered_data(data)message = "&".join(u"{}={}".format(k, v) for k, v in unsigned_items)return self._verify(message, signature)

alipayTools.py

from AIChatProject import settings
from chatApp.aliPayConfig import AliPay  # 注意是从aliPayConfig.py中引用的,而不是python-alipay-sdk的基础AliPay包def ali():curr_alipay = AliPay(appid=settings.APP_ID,  # 应用IDapp_notify_url=settings.NOTIFY_URL,  # 异步POST回调地址return_url=settings.RETURN_URL,  # 同步GET回调地址app_private_key_path=settings.APP_PRIVATE_KEY_PATH,  # 应用私钥存放路径alipay_public_key_path=settings.ALIPAY_PUBLIC_KEY_PATH,  # 支付宝公钥存放路径debug=settings.DEBUG  # true是沙箱, false是真实环境)return curr_alipay

models.py

class UserInfo(models.Model):id = models.AutoField(primary_key=True, verbose_name='用户编号')user_account = models.CharField(verbose_name='账号', max_length=32, unique=True)user_password = models.CharField(verbose_name='密码', max_length=255)user_account_balance = models.DecimalField(verbose_name='账户余额', max_digits=10, decimal_places=2, default=10000)auto_type = ((1, '本地用户'),(2, '第三方用户'),)user_type = models.SmallIntegerField(verbose_name='用户类型', choices=auto_type, default=1)class 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="支付时间")# DO_NOTHING 表示不采取任何操作,当关联的对象被删除时,不执行任何处理动作。user = models.ForeignKey(UserInfo, related_name='chatapp_userinfo', on_delete=models.DO_NOTHING, db_constraint=False, verbose_name="下单用户")created_time = models.DateTimeField(auto_now_add=True, verbose_name='创建时间')

views.py

class OrderSerializer(serializers.ModelSerializer):class Meta:model = Orderfields = '__all__'# 一、理论知识第1步
class AliPayView(APIView):def post(self, request):# 1.实例化支付宝curr_alipay = ali()# 2.获取用户购买的订单data = request.datauser = request.userInfo# 3.建立支付宝唤醒pay_url# 3.1 必要参数(subject, out_trade_no, total_amount)subject = data.get('subject', '')out_trade_no = str(uuid.uuid4())total_amount = data.get('total_amount', '')# 3.2 调用实例的方法生成请求参数query_params = curr_alipay.direct_pay(subject, out_trade_no, total_amount)# 3.3 参数拼接pay_url = settings.GATEWAY + '?' + query_params# 4.创建数据库订单user_order = {'subject': subject,'total_amount': total_amount,'out_trade_no': out_trade_no,'user': user.id}serializer = OrderSerializer(data=user_order)if serializer.is_valid():# 5.数据库存储订单serializer.save()return Response(R(data=pay_url).to_dict())else:return Response(R(error='订单存档失败!').to_dict())# 二、理论知识第4步:支付宝服务器发送POST请求
# notify_url 异步请求异步通知接口
class PayNotifyView(APIView):def post(self, request):curr_alipay = ali()try:# 1.获取支付传递多来的参数result_data = request.data.dict()# 2.将 sign 取出signature = result_data.pop('sign')out_trade_no = result_data.get('out_trade_no')trade_no = result_data.get('trade_no')# 3.验签result = curr_alipay.verify(result_data, signature)# 确认是否支付成功if result and result_data["trade_status"] in ("TRADE_SUCCESS", "TRADE_FINISHED"):# 4.完成订单更新:订单状态、流水号、支付时间result_order = Order.objects.filter(out_trade_no=out_trade_no).first()# 4.1 获取订单对应的用户u_id = result_order.user.idorder_user = UserInfo.objects.filter(id=u_id).first()print('验签成功,当前用户是:', u_id, order_user.user_account)# 4.2 校验订单是否重复提交!if result_order.order_status == 1:return Response(R(data={'message': '该订单已经完成','free_account': order_user.user_account_balance}).to_dict())# 4.3 修改订单状态Order.objects.filter(out_trade_no=out_trade_no, order_status=0).update(order_status=1, trade_no=trade_no)# 5.更新用户的余额new_free_tokens = order_user.user_account_balance + result_order.total_amount * Constant.PRICE_ONEprint(f'old token {order_user.user_account_balance} new token {new_free_tokens}')UserInfo.objects.filter(id=u_id).update(user_account_balance=new_free_tokens)# 完成日志记录return Response(R(data={'message': '充值成功', 'free_account': new_free_tokens}).to_dict())else:# logger.error('%s订单支付失败' % out_trade_no)print('%s订单支付失败,校验未通过或trade_status不对' % out_trade_no)except Exception as e:print('报错--->', e)passprint('%s订单支付失败' % out_trade_no)return Response(R(error='%s订单支付失败' % out_trade_no).to_dict())

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/20746.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python pytest脚本执行工具

pytest脚本执行工具 支持获取当前路径下所有.py脚本 添加多个脚本,一起执行 import tkinter as tk from tkinter import filedialog import subprocess import os from datetime import datetimedef select_script():script_path filedialog.askopenfilename(fil…

11. 利用Tomcat服务器配置HTTPS双向认定

文章目录 Tomcat配置HTTPS1.为服务器生成证书2.为客户端生成证书3.让服务器信任客户端证书4.将该文件导入到服务器的证书库,添加为一个信任证书使用命令如下:5.查看证书库6.让客户端信任服务器证书7.配置tomcat8.验证 Tomcat配置HTTPS 1.启动cmd控制台&…

使用Pytorch加载预训练模型及修改网络结构

Pytorch有自带的训练好的AlexNet、VGG、ResNet等网络架构。详见官网 1.加载预训练模型 import torch import torchvision import torch.nn as nn import torch.optim as optim import torch.nn.functional as F import torchvision.transforms as transforms import torchvis…

逆波兰式是什么?

1. 逆波兰式是什么? 逆波兰式,也称逆波兰记法(Reverse Polish Notation,缩写为RPN),是一种在数学和计算机科学中用于表示算术表达式的方法。它的特点是操作符在操作数的后面,不需要括号来改变运…

DevOps基础服务2——Jenkins

文章目录 一、基本了解1.1 CI/CD介绍1.2 基于Docker的CI/CD 二、安装jenkins三、页面管理3.1 当前系统用户配置3.2 系统配置3.3 全局工具配置3.4 插件管理3.4.1 安装插件3.4.2 上传插件 3.5 用户设置3.6 查看日志3.7 汉化设置 一、基本了解 DEVOPS概念: DevOps是一种…

网络ping丢包什么原因(附解决方案)

​  数据包丢失是一种网络问题,当通过网络(或设备之间或通过 Internet)传输的数据包在传输过程中丢失或丢弃并且无法到达目的地时,就会发生这种情况。简单来说,数据包丢失是指数据包无法通过互联网从发送者成功传输到接收者。 如何检测数据…

LiveGBS流媒体平台GB/T28181功能-支持UDP、TCP被动、TCP主动模式流传输模式之间有什么区别?

LiveGBS流媒体平台-国标流媒体服务几种流传输模式UDP、TCP被动、TCP主动模式区别是什么 1、背景1、GB28181流传输模式1.1、UDP1.2、TCP 被动1.3、TCP 主动 2、切换流传输模式2.1、编辑2.2、下拉切换 3、搭建GB28181视频直播平台 1、背景 国标GB28181协议中,视频流到…

eNSP-VRRP虚拟路由器冗余技术

VRRP-虚拟路由器冗余技术 文章目录 VRRP-虚拟路由器冗余技术一、拓扑结构二、基本配置三、测试验证四、知识点详解1.VRRP路由器2.报文格式3.工作过程 一、拓扑结构 二、基本配置 R1: #配置ip <Huawei>sys [Huawei]sys r1 [r1]int g0/0/0 [r1-GigabitEthernet0/0/0]ip a…

Perl 7 - 使用 Perlbrew 管理perl 版本

文章目录 关于 Perlbrew安装 Perlbrew使用 perlbrew 安装/管理 perl 版本 关于 Perlbrew 官网&#xff1a;https://perlbrew.pl 相关文档&#xff1a; App::perlbrew https://metacpan.org/pod/App::perlbrew Perlbrew 是一个工具&#xff0c;用于管理您$HOME 目录(或您指定的…

嵌入式LinuxLED驱动开发实验

目录&#xff1a; 1. Linux下LED灯的驱动原理1.1. 地址映射1.1.1. 实际物理地址映射到虚拟地址的函数1.1.2. 内存访问函数 2.硬件原理图分析3. 实验程序编写3.1. 驱动程序编写3.2. 应用程序编写 4. 运行测试 1. Linux下LED灯的驱动原理 我们在裸机实验的时候&#xff0c;都是通…

【数据结构】红黑树

文章目录 红黑树1. 红黑树的概念2. 红黑树的性质3. 红黑树节点的定义4. 红黑树的结构5. 红黑树的插入操作 红黑树 1. 红黑树的概念 红黑树&#xff0c;是一种二叉搜索树&#xff0c;但在每个节点上增加一个存储位表示节点的颜色&#xff0c;可以是Red或者是Black。通过任何一…

【Docker】Docker的部署含服务和应用、多租环境、Linux内核的详细介绍

前言 Docker 是一个开源的应用容器引擎&#xff0c;让开发者可以打包他们的应用以及依赖包到一个可移植的容器中,然后发布到任何流行的Linux或Windows操作系统的机器上,也可以实现虚拟化,容器是完全使用沙箱机制,相互之间不会有任何接口。 &#x1f4d5;作者简介&#xff1a;热…