博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
支付宝支付
阅读量:7097 次
发布时间:2019-06-28

本文共 6772 字,大约阅读时间需要 22 分钟。

utils/pay.py

from datetime import datetimefrom Crypto.PublicKey import RSAfrom Crypto.Signature import PKCS1_v1_5from Crypto.Hash import SHA256from urllib.parse import quote_plusfrom urllib.parse import urlparse, parse_qsfrom base64 import decodebytes, encodebytesimport 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 = appid        self.app_notify_url = app_notify_url        self.app_private_key_path = app_private_key_path        self.app_private_key = None        self.return_url = return_url        with open(self.app_private_key_path) as fp:            self.app_private_key = RSA.importKey(fp.read())        self.alipay_public_key_path = alipay_public_key_path        with 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_url            data["return_url"] = self.return_url        return data    def 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_string    def 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_key        signer = PKCS1_v1_5.new(key)        signature = signer.sign(SHA256.new(unsigned_string))        # base64 编码,转换为unicode表示并移除回车        sign = encodebytes(signature).decode("utf8").replace("\n", "")        return sign    def _verify(self, raw_content, signature):        # 开始计算签名        key = self.alipay_public_key        signer = PKCS1_v1_5.new(key)        digest = SHA256.new()        digest.update(raw_content.encode("utf8"))        if signer.verify(digest, decodebytes(signature.encode("utf8"))):            return True        return False    def 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)
View Code

 

views.py
from django.shortcuts import render,redirect,HttpResponsefrom django.views.decorators.csrf import csrf_exemptfrom utils.pay import AliPayimport timefrom django.conf import settingsdef aliPay():    obj = AliPay(        appid=settings.APPID,        app_notify_url=settings.NOTIFY_URL,  # 如果支付成功,支付宝会向这个地址发送POST请求(校验是否支付已经完成)        return_url=settings.RETURN_URL,  # 如果支付成功,重定向回到你的网站的地址。        alipay_public_key_path=settings.PUB_KEY_PATH,  # 支付宝公钥        app_private_key_path=settings.PRI_KEY_PATH,  # 应用私钥        debug=True,  # 默认False,    )    return objdef index(request):    if request.method == 'GET':        return render(request,'index.html')    alipay = aliPay()    # 对购买的数据进行加密    money = float(request.POST.get('price'))    out_trade_no = "x2" + str(time.time())    # 1. 在数据库创建一条数据:状态(待支付)    query_params = alipay.direct_pay(        subject="充气式韩红",  # 商品简单描述        out_trade_no= out_trade_no,  # 商户订单号        total_amount=money,  # 交易金额(单位: 元 保留俩位小数)    )    pay_url = "https://openapi.alipaydev.com/gateway.do?{}".format(query_params)    return redirect(pay_url)def pay_result(request):    """    支付完成后,跳转回的地址    :param request:    :return:    """    params = request.GET.dict()    sign = params.pop('sign', None)    alipay = aliPay()    status = alipay.verify(params, sign)    if status:        return HttpResponse('支付成功')    return HttpResponse('支付失败')@csrf_exemptdef update_order(request):    """    支付成功后,支付宝向该地址发送的POST请求(用于修改订单状态)    :param request:    :return:    """    if request.method == 'POST':        from urllib.parse import parse_qs        body_str = request.body.decode('utf-8')        post_data = parse_qs(body_str)        post_dict = {}        for k, v in post_data.items():            post_dict[k] = v[0]        alipay = aliPay()        sign = post_dict.pop('sign', None)        status = alipay.verify(post_dict, sign)        if status:            # 修改订单状态            out_trade_no = post_dict.get('out_trade_no')            print(out_trade_no)            # 2. 根据订单号将数据库中的数据进行更新            return HttpResponse('支付成功')        else:            return HttpResponse('支付失败')    return HttpResponse('')
View Code

urls.py

from django.conf.urls import urlfrom django.contrib import adminfrom app01 import viewsurlpatterns = [    url(r'^admin/', admin.site.urls),    url(r'^index/', views.index),    url(r'^pay_result/', views.pay_result),    url(r'^update_order/', views.update_order),]
View Code
settings.py
# 支付相关配置APPID = "20108250039412"NOTIFY_URL = "http://127.0.0.1:80/update_order/"RETURN_URL = "http://127.0.0.1:80/pay_result/"PRI_KEY_PATH = "keys/app_private_2048.txt"PUB_KEY_PATH = "keys/alipay_public_2048.txt"
View Code
 
pay.html
    
Title
{% csrf_token %}
View Code

 

 
 

转载于:https://www.cnblogs.com/Guishuzhe/p/9756472.html

你可能感兴趣的文章
String.xml的特殊用法(国际化、字体)和转换字符集
查看>>
打素数表(筛法)
查看>>
使用iscroll插件实现下拉刷新功能
查看>>
单源最短路:Bellman-Ford算法 及 证明
查看>>
线程2
查看>>
[Shiro] - shiro之SSM中的使用
查看>>
(Windows)Scala学习2--控制结构
查看>>
固定布局、流动布局、弹性布局哪一种适合你(上)
查看>>
Vue设置反向代理实现跨域
查看>>
[Web 前端] mobx教程(二)-mobx主要概念
查看>>
解决pycharm远程连接mysql错误1130代码的方法
查看>>
给font awesome中加入自定义图片
查看>>
JS数组控制台排序
查看>>
创业泡沫真他妈的大出血来了 --个人需要创业以谋生,行业需要创业以求变,国家需要创业以维稳...
查看>>
【转】安卓Java的虚拟机区别
查看>>
atitit.提升开发效率---MDA 软件开发方式的革命(5)----列表查询建模
查看>>
Codeforces 91C Ski Base 加边求欧拉回路数量
查看>>
深度学习项目实战——“年龄预测”
查看>>
金融安全资讯精选 2017年第二期:金融网络安全和反欺诈方法论_金融新兴技术成熟度几何?...
查看>>
预编译指令包括:宏定义;条件编译;文件包含(就是include)
查看>>