python flask 微信支付下单及支付结果回调处理代码笔记

2024-06-21 22:36

本文主要是介绍python flask 微信支付下单及支付结果回调处理代码笔记,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

因为微信支付下单的时候不允许同一个订单重复下单,所以在下单之前 对订单号加了三位随机字符串 在回调时 进行截取操作

@order_api.route('/user/wx_xcx_pay/notify', methods=['POST'])
def new_notify():headers = request.headers# print('headers', headers)wxpay = wx_pay_func()result = wxpay.callback(headers, request.data)if result and result.get('event_type') == 'TRANSACTION.SUCCESS':resp = result.get('resource')out_trade_no = resp.get('out_trade_no')  # 本系统订单号transaction_id = resp.get('transaction_id')  # 微信支付订单号first_character = out_trade_no[0]  # 获取开头字母 确定支付类型order_number = out_trade_no[:-3]  # 获取实际订单号success_time = resp.get('success_time')  # 支付完成时间payer = resp.get('payer')  # 付款人信息amount = resp.get('amount').get('total')  # 金额 单位分payment_amount = Decimal(amount) / Decimal(100)  # 实际支付金额 单位元# 上门取 或者 上门洗if first_character in ['X', 'Q']:return order_pay_callback(order_number, amount, success_time, transaction_id, resp)# 充值订单elif first_character == 'C':return recharge_pay_callback(order_number, transaction_id, payment_amount)# 补差价订单elif first_character == 'B':return price_difference_payment_callback(order_number, transaction_id, payment_amount)else:return reg_func(200, None, '支付失败')def load_private_key(file_name):# 获取当前文件的目录current_dir = os.path.dirname(__file__)# 构建完整的文件路径file_path = os.path.join(current_dir, file_name)try:# 安全地打开文件with open(file_path, 'r') as file:private_key = file.read()return private_keyexcept FileNotFoundError:print(f"Error: The file {file_name} was not found in the directory {current_dir}.")return Noneexcept Exception as e:print(f"An error occurred: {e}")return None# 初始化
def wx_pay_func():mch_id = get_config('wx_xcx_pay_mid')appid = get_config('wx_xcx_appid')APIV3_KEY = get_config('wx_xcx_pay_secret')# 日志记录器,记录web请求和回调细节logging.basicConfig(filename=os.path.join(os.getcwd(), 'demo.log'), level=logging.DEBUG, filemode='a',format='%(asctime)s - %(process)s - %(levelname)s: %(message)s')LOGGER = logging.getLogger("demo")# 代理设置,None或者{"https": "http://10.10.1.10:1080"},详细格式参见https://requests.readthedocs.io/en/latest/user/advanced/#proxiesPROXY = None# 接入模式:False=直连商户模式,True=服务商模式PARTNER_MODE = FalseCERT_DIR = None# 请求超时时间配置TIMEOUT = (10, 30)  # 建立连接最大超时时间是10s,读取响应的最大超时时间是30sPRIVATE_KEY = load_private_key('apiclient_key.pem')CERT_SERIAL_NO = get_config('CERT_SERIAL_NO')api_url = get_config('api_domain')NOTIFY_URL = f'https://{api_url}/user/wx_xcx_pay/notify'print('NOTIFY_URL', NOTIFY_URL)wxpay = WeChatPay(wechatpay_type=WeChatPayType.NATIVE,mchid=mch_id,private_key=PRIVATE_KEY,cert_serial_no=CERT_SERIAL_NO,apiv3_key=APIV3_KEY,appid=appid,notify_url=NOTIFY_URL,cert_dir=CERT_DIR,logger=LOGGER,partner_mode=PARTNER_MODE,proxy=PROXY,timeout=TIMEOUT)return wxpaydef generate_random_string():# 字符集合,包括大写字母、小写字母和数字characters = string.ascii_letters + string.digits# 从字符集合中随机选择3个字符random_string = ''.join(random.choices(characters, k=3))return random_stringdef xcx_pay(openid, amount, order_number):wxpay = wx_pay_func()str1 = generate_random_string()out_trade_no = order_number + str1description = out_trade_nopayer = {'openid': openid}code, message = wxpay.pay(description=description,out_trade_no=out_trade_no,amount={'total': amount},pay_type=WeChatPayType.MINIPROG,payer=payer)result = json.loads(message)if code in range(200, 300):APPID = get_config('wx_xcx_appid')prepay_id = result.get('prepay_id')timestamp = str(int(time.time()))noncestr = str(uuid.uuid4()).replace('-', '')package = 'prepay_id=' + prepay_idsign = wxpay.sign(data=[APPID, timestamp, noncestr, package])signtype = 'RSA'result = {'appId': APPID,'timeStamp': timestamp,'nonceStr': noncestr,'package': 'prepay_id=%s' % prepay_id,'signType': signtype,'paySign': sign}return reg_func(200, result, '微信支付下单成功')else:return reg_func(500, result.get('code'), '微信支付下单失败')

代码解释

这段Python代码实现了一个处理微信支付通知的API端点。以下是详细解释:

路由定义与请求处理函数

@order_api.route('/user/wx_xcx_pay/notify', methods=['POST'])
def new_notify():

这段代码定义了一个Flask路由,当客户端以POST请求访问/user/wx_xcx_pay/notify时,会调用new_notify函数。

获取请求头和初始化微信支付对象

    headers = request.headerswxpay = wx_pay_func()

首先获取请求的头信息,然后调用wx_pay_func函数初始化微信支付对象。

处理微信支付回调

    result = wxpay.callback(headers, request.data)

使用微信支付对象的callback方法处理回调数据,返回结果存储在result变量中。

判断支付是否成功并提取相关信息

    if result and result.get('event_type') == 'TRANSACTION.SUCCESS':resp = result.get('resource')out_trade_no = resp.get('out_trade_no')transaction_id = resp.get('transaction_id')first_character = out_trade_no[0]order_number = out_trade_no[:-3]success_time = resp.get('success_time')payer = resp.get('payer')amount = resp.get('amount').get('total')payment_amount = Decimal(amount) / Decimal(100)

检查支付结果是否成功,若成功,则从回调结果中提取相关信息,包括系统订单号、微信支付订单号、支付类型标识、实际订单号、支付完成时间、付款人信息以及支付金额。

根据支付类型调用不同的处理函数

        if first_character in ['X', 'Q']:return order_pay_callback(order_number, amount, success_time, transaction_id, resp)elif first_character == 'C':return recharge_pay_callback(order_number, transaction_id, payment_amount)elif first_character == 'B':return price_difference_payment_callback(order_number, transaction_id, payment_amount)else:return reg_func(200, None, '支付失败')

根据订单号的第一个字符判断支付类型,并调用相应的处理函数。

初始化微信支付对象函数

def wx_pay_func():mch_id = get_config('wx_xcx_pay_mid')appid = get_config('wx_xcx_appid')APIV3_KEY = get_config('wx_xcx_pay_secret')logging.basicConfig(filename=os.path.join(os.getcwd(), 'demo.log'), level=logging.DEBUG, filemode='a', format='%(asctime)s - %(process)s - %(levelname)s: %(message)s')LOGGER = logging.getLogger("demo")PROXY = NonePARTNER_MODE = FalseCERT_DIR = NoneTIMEOUT = (10, 30)PRIVATE_KEY = load_private_key('apiclient_key.pem')CERT_SERIAL_NO = get_config('CERT_SERIAL_NO')api_url = get_config('api_domain')NOTIFY_URL = f'https://{api_url}/user/wx_xcx_pay/notify'wxpay = WeChatPay(wechatpay_type=WeChatPayType.NATIVE,mchid=mch_id,private_key=PRIVATE_KEY,cert_serial_no=CERT_SERIAL_NO,apiv3_key=APIV3_KEY,appid=appid,notify_url=NOTIFY_URL,cert_dir=CERT_DIR,logger=LOGGER,partner_mode=PARTNER_MODE,proxy=PROXY,timeout=TIMEOUT)return wxpay

该函数初始化并返回一个微信支付对象,包含商户号、应用ID、API密钥等必要的配置,以及日志记录器和请求超时设置等。

函数解释

生成随机字符串函数
def generate_random_string():# 字符集合,包括大写字母、小写字母和数字characters = string.ascii_letters + string.digits# 从字符集合中随机选择3个字符random_string = ''.join(random.choices(characters, k=3))return random_string
  • 功能:生成一个包含大写字母、小写字母和数字的随机字符串,长度为3。
  • 步骤
    • 定义字符集合,包括所有字母和数字。
    • 随机选择3个字符,并组合成一个字符串返回。
小程序支付函数
def xcx_pay(openid, amount, order_number):wxpay = wx_pay_func()str1 = generate_random_string()out_trade_no = order_number + str1description = out_trade_nopayer = {'openid': openid}code, message = wxpay.pay(description=description,out_trade_no=out_trade_no,amount={'total': amount},pay_type=WeChatPayType.MINIPROG,payer=payer)result = json.loads(message)if code in range(200, 300):APPID = get_config('wx_xcx_appid')prepay_id = result.get('prepay_id')timestamp = str(int(time.time()))noncestr = str(uuid.uuid4()).replace('-', '')package = 'prepay_id=' + prepay_idsign = wxpay.sign(data=[APPID, timestamp, noncestr, package])signtype = 'RSA'result = {'appId': APPID,'timeStamp': timestamp,'nonceStr': noncestr,'package': 'prepay_id=%s' % prepay_id,'signType': signtype,'paySign': sign}return reg_func(200, result, '微信支付下单成功')else:return reg_func(500, result.get('code'), '微信支付下单失败')
  • 功能:为微信小程序生成支付订单,并返回支付参数给前端。

  • 参数

    • openid:用户的微信OpenID。
    • amount:支付金额(单位为分)。
    • order_number:订单号。
  • 步骤

    1. 初始化微信支付对象。
    2. 生成随机字符串并附加到订单号后面,形成新的商户订单号(out_trade_no)。
    3. 调用微信支付对象的pay方法,传入支付描述、商户订单号、支付金额、支付类型和付款人信息。
    4. 解析支付请求的返回结果。
    5. 如果请求成功(状态码在200到299之间):
      • 获取应用ID和预支付交易会话标识(prepay_id)。
      • 生成当前时间戳、随机字符串和支付签名。
      • 构建支付参数并返回成功信息。
    6. 如果请求失败,返回错误信息。

总结

  1. generate_random_string 函数用于生成一个长度为3的随机字符串,以增强订单号的唯一性。
  2. xcx_pay 函数用于处理微信小程序支付请求,生成支付订单并返回支付参数给前端。

中文解释由chatgpt协助整理

这篇关于python flask 微信支付下单及支付结果回调处理代码笔记的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

python管理工具之conda安装部署及使用详解

《python管理工具之conda安装部署及使用详解》这篇文章详细介绍了如何安装和使用conda来管理Python环境,它涵盖了从安装部署、镜像源配置到具体的conda使用方法,包括创建、激活、安装包... 目录pytpshheraerUhon管理工具:conda部署+使用一、安装部署1、 下载2、 安装3

Python进阶之Excel基本操作介绍

《Python进阶之Excel基本操作介绍》在现实中,很多工作都需要与数据打交道,Excel作为常用的数据处理工具,一直备受人们的青睐,本文主要为大家介绍了一些Python中Excel的基本操作,希望... 目录概述写入使用 xlwt使用 XlsxWriter读取修改概述在现实中,很多工作都需要与数据打交

使用Python实现在Word中添加或删除超链接

《使用Python实现在Word中添加或删除超链接》在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能,本文将为大家介绍一下Python如何实现在Word中添加或... 在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能。通过添加超

Python MySQL如何通过Binlog获取变更记录恢复数据

《PythonMySQL如何通过Binlog获取变更记录恢复数据》本文介绍了如何使用Python和pymysqlreplication库通过MySQL的二进制日志(Binlog)获取数据库的变更记录... 目录python mysql通过Binlog获取变更记录恢复数据1.安装pymysqlreplicat

利用Python编写一个简单的聊天机器人

《利用Python编写一个简单的聊天机器人》这篇文章主要为大家详细介绍了如何利用Python编写一个简单的聊天机器人,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 使用 python 编写一个简单的聊天机器人可以从最基础的逻辑开始,然后逐步加入更复杂的功能。这里我们将先实现一个简单的

基于Python开发电脑定时关机工具

《基于Python开发电脑定时关机工具》这篇文章主要为大家详细介绍了如何基于Python开发一个电脑定时关机工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 简介2. 运行效果3. 相关源码1. 简介这个程序就像一个“忠实的管家”,帮你按时关掉电脑,而且全程不需要你多做

Python实现高效地读写大型文件

《Python实现高效地读写大型文件》Python如何读写的是大型文件,有没有什么方法来提高效率呢,这篇文章就来和大家聊聊如何在Python中高效地读写大型文件,需要的可以了解下... 目录一、逐行读取大型文件二、分块读取大型文件三、使用 mmap 模块进行内存映射文件操作(适用于大文件)四、使用 pand

python实现pdf转word和excel的示例代码

《python实现pdf转word和excel的示例代码》本文主要介绍了python实现pdf转word和excel的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、引言二、python编程1,PDF转Word2,PDF转Excel三、前端页面效果展示总结一

在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码

《在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码》在MyBatis的XML映射文件中,trim元素用于动态添加SQL语句的一部分,处理前缀、后缀及多余的逗号或连接符,示... 在MyBATis的XML映射文件中,<trim>元素用于动态地添加SQL语句的一部分,例如SET或W

Python xmltodict实现简化XML数据处理

《Pythonxmltodict实现简化XML数据处理》Python社区为提供了xmltodict库,它专为简化XML与Python数据结构的转换而设计,本文主要来为大家介绍一下如何使用xmltod... 目录一、引言二、XMLtodict介绍设计理念适用场景三、功能参数与属性1、parse函数2、unpa