python 微信v3支付接口 签名 Native支付

微信支付需要用到微信公众平台账号、微信商户账号。
注册完成后,我们需要在平台找到 appid、mchid、证书等
具体操作:微信支付接入前准备(Native支付)
这里仅演示 签名的生成方法、Authorization的传递以及出现401 Unauthorized的解决方案。

# _*_ coding: utf-8 _*_
"""
Time:     2022/2/28 14:33
Author:   Jyun
Version:  V 0.1
File:     wxpay.py
Blog:     https://ctrlcv.blog.csdn.net
"""
import time
import json
import random
import string
import requests

from base64 import b64encode
from urllib.parse import urlparse
from Cryptodome.PublicKey import RSA
from Cryptodome.Signature import pkcs1_15
from Cryptodome.Hash import SHA256


# 生成二维码
def to_qr(text):
    import qrcode
    img = qrcode.make(text)
    with open('pay.png', 'wb') as f:
        img.save(f)


class WXPay:
    """ 微信 Native支付
    """

    def __init__(self):
        self.appid = "wx1fxxxxxc0eeexxx" # APPID
        self.mchid = "100000001"  # 商户号
        self.payment_url = 'https://api.mch.weixin.qq.com/v3/pay/transactions/native'  # Native支付下单接口
        self.refund_url = 'https://api.mch.weixin.qq.com/v3/refund/domestic/refunds'  # 退款接口
        self.notify_url = "https://weixin.qq.com/"  # 通知url
        self.serial_no = '48757XXXX9D9CD835841B45969B0XXXXXXXXXXXX'  # 商户证书序列号
        self.apiclient_key = './apiclient_key.pem'  # 本地证书路径

    # 生成签名
    def get_sign(self, sign_str):
        rsa_key = RSA.importKey(open(self.apiclient_key).read())
        signer = pkcs1_15.new(rsa_key)
        digest = SHA256.new(sign_str.encode('utf8'))
        sign = b64encode(signer.sign(digest)).decode('utf-8')
        return sign

    def request(self, url: str, method: str, data: dict = None):
        data = json.dumps(data) if data else ''
        random_str = ''.join(random.choice(string.ascii_uppercase + string.digits) for _ in range(32))
        timestamp = str(int(time.time()))
        sign_str = '\n'.join([
            method.upper(),  # HTTP请求方法
            url.split(urlparse(url).netloc)[-1],  # path+args
            timestamp,  # 时间戳
            random_str,  # 请求随机串
            data, ''  # 请求报文主体
        ])  # 结尾空窜仅用于让后面多一个\n
        sign = self.get_sign(sign_str)

        headers = {
            'Content-Type': 'application/json; charset=UTF-8',
            'Authorization': f'WECHATPAY2-SHA256-RSA2048 mchid="{self.mchid}",nonce_str="{random_str}",signature="{sign}",timestamp="{timestamp}",serial_no="{self.serial_no}"'
        }
        response = requests.request(url=url, method=method, data=data, headers=headers)
        return response

    # 支付
    def payment(self, order_no, total, description):
        data = {
            "mchid": self.mchid,
            "out_trade_no": order_no,  # 订单号
            "appid": self.appid,
            "description": description,  # 商品描述
            "notify_url": self.notify_url,
            "amount": {
                "total": total,  # 总金额(分)
                "currency": "CNY"
            }
        }
        return self.request(self.payment_url, 'POST', data)

    # 退款
    def refund(self, transaction_id, out_refund_no, refund, reason):
        data = {
            "transaction_id": transaction_id,  # 微信支付订单号(交易单号)
            "out_refund_no": out_refund_no,  # 商户退款单号(商户单号)
            "reason": reason,  # 退款原因
            "notify_url": self.notify_url,  # 通知Url
            "amount": {
                "total": refund,  # 订单金额
                "refund": refund,  # 退款金额(分)
                "currency": "CNY"
            }
        }
        return self.request(self.refund_url, 'POST', data)

    def application_bill(self):
        url = 'https://api.mch.weixin.qq.com/v3/bill/tradebill?bill_date=2022-02-28&bill_type=ALL'
        return self.request(url, 'GET')


if __name__ == "__main__":
    wx = WXPay()
    resp = wx.payment("80000000018", 1, "订单支付-测试")
    print(resp.text)
    to_qr(resp.json()['code_url'])

    # resp = wx.refund('4200002222222222000000000000', 'T80000000018', 1, '商品已售完')
    # print(resp.text)

微信支付 V3 接口的开发指南和技术实现主要包括以下几个核心环节: 1. **前期准备**:首先需要注册微信支付商户号,并获取商户号和商户私钥。在微信商户平台生成 API v3 密钥,并下载证书用于后续的签名和验证过程。 2. **配置证书**:将下载的证书配置到项目中,并设置商户号和商户私钥。这些信息通常会用于生成请求签名,确保请求的完整性和安全性。 3. **构建请求参数**:对于每一个微信支付接口,需要构建完整的请求参数,包括接口地址、请求方法(GET/POST)、请求头(Header)以及请求体(Body)。请求头中需要包含 `Authorization` 字段,该字段的值是通过签名算法生成的签名串。 4. **生成签名**:签名微信支付 V3 接口安全机制的重要组成部分。签名算法通常基于商户私钥和请求内容生成。签名生成后需要添加到请求头中,以便微信服务器进行验证。 5. **发送请求**:使用 HTTP 客户端(如 Apache HttpClient 或 Guzzle)发送构建好的请求。在发送请求时,需要确保所有参数正确无误,并且签名已正确附加。 6. **解析响应**:接收到微信支付服务器的响应后,需要对响应内容进行解析。响应内容通常为 JSON 格式,包含支付结果、交易状态等信息。同时,还需要验证响应的签名,确保响应确实来自微信服务器,防止中间人攻击。 7. **处理支付结果**:根据解析后的响应内容,处理支付成功或失败的逻辑。例如,在支付成功后更新订单状态,在支付失败时提示用户重新支付。 在开发过程中,可以借助官方提供的 SDK 简化开发工作。例如,微信支付提供了适用于同语言的 SDK,如 Java SDK 和 Python SDK,内部封装了部分常用方法,开发者只需按照官方 Demo 调用相关接口即可[^2]。 此外,微信支付 V3 接口还支持多种支付场景,包括小程序支付、H5 支付Native 支付等。每种支付场景的接口调用方式略有同,开发者需要根据实际需求选择合适的支付方式。 以下是一个简单的 Java 示例代码,展示如何使用 Apache HttpClient 发送带有签名的请求: ```java import org.apache.http.client.methods.CloseableHttpResponse; import org.apache.http.client.methods.HttpPost; import org.apache.http.entity.StringEntity; import org.apache.http.impl.client.CloseableHttpClient; import org.apache.http.util.EntityUtils; import java.util.*; public class WeChatPayV3 { private static final String MCH_ID = "your_merchant_id"; private static final String PRIVATE_KEY = "your_private_key"; private static final String API_V3_KEY = "your_api_v3_key"; private static final String CERT_SERIAL_NO = "your_certificate_serial_number"; public static String generateSignature(String method, String url, String body) { // 实现签名生成逻辑 return "generated_signature"; } public static void main(String[] args) throws Exception { String url = "https://api.mch.weixin.qq.com/v3/pay/transactions/jsapi"; String body = "{" + "\"mchid\":\"" + MCH_ID + "\"," + "\"out_trade_no\":\"20231001123456\"," + "\"appid\":\"your_appid\"," + "\"description\":\"Test Payment\"," + "\"amount\":{" + "\"total\":1," + "\"currency\":\"CNY\"" + "}," + "\"payer\":{" + "\"openid\":\"user_openid\"" + "}" + "}"; String method = "POST"; String signature = generateSignature(method, url, body); try (CloseableHttpClient httpClient = HttpClients.createDefault()) { HttpPost httpPost = new HttpPost(url); httpPost.setHeader("Content-Type", "application/json"); httpPost.setHeader("Authorization", "WECHATPAY2-SHA256-RSA2048 " + signature); httpPost.setHeader("Wechatpay-Serial", CERT_SERIAL_NO); httpPost.setEntity(new StringEntity(body)); try (CloseableHttpResponse response = httpClient.execute(httpPost)) { String responseBody = EntityUtils.toString(response.getEntity()); System.out.println("Response: " + responseBody); } } } } ``` 在上述代码中,`generateSignature` 方法用于生成签名,`main` 方法中构建了请求并发送至微信支付服务器。收到响应后,程序会打印出响应内容。 为了确保接口的安全性,建议开发者在开发过程中严格遵循微信支付官方文档的要求,尤其是在签名生成和证书管理方面。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

CtrlCV工程师

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值