Python項(xiàng)目實(shí)戰(zhàn)——手把手教你使用Django框架實(shí)現(xiàn)支付寶付款
一、前言
大家好,我是Python進(jìn)階者。春節(jié)即將過去,大家過年期間肯定各種掏腰包花花花,小編相信大家在支付時(shí)候,微信、支付寶支付肯定是優(yōu)先選擇。今天小編心血來潮,為大家?guī)硪粋€(gè)很有趣的項(xiàng)目,那就是使用Python web框架Django來實(shí)現(xiàn)支付寶支付,廢話不多說,一起來看看如何實(shí)現(xiàn)吧。
二、建立django應(yīng)用
我們來建立一個(gè)Django項(xiàng)目然后在里面創(chuàng)建一個(gè)應(yīng)用,如圖:

三、配置并啟動(dòng)


然后我們?cè)O(shè)置urls文件的內(nèi)容,如圖:

然后再在子應(yīng)用中創(chuàng)建一個(gè)urls.py文件,當(dāng)然你也可以直接將一些視圖函數(shù)寫在項(xiàng)目中的urls.py文件中。最后我們編寫視圖函數(shù)并把視圖函數(shù)添加到urls.py文件中,如圖:


最后我們需要提交更改,打開該項(xiàng)目manage.py文件所在的目錄并打開cmd,輸入如下命令:
python manage.py migrate現(xiàn)在讓我們來本地跑跑這個(gè)項(xiàng)目,還是在該目錄中,如下:
python manage.py runserver
看到輸出的結(jié)果表明這個(gè)子應(yīng)用已經(jīng)啟動(dòng)并返回了結(jié)果。我們也可以不用經(jīng)過子應(yīng)用直接在創(chuàng)建的項(xiàng)目根目錄下運(yùn)行啟動(dòng)Django應(yīng)用,首先在pay目錄下新建一個(gè)view.py文件,然后將其添加到該目錄下的urls.py文件中,如下:


運(yùn)行下看圖:

四、登錄支付寶并生成rsa密鑰
首先登錄咱們要收款的支付寶,地址:
https://auth.alipay.com/login/ant_sso_index.htm?goto=https%3A%2F%2Fopenhome.alipay.com%2Fplatform%2FappDaily.htm%3Ftab%3Dinfo然后進(jìn)行登錄,如圖:

然后點(diǎn)擊RSA2(SHA256)后面的設(shè)置,點(diǎn)擊公鑰并下載支付寶密鑰生成器或者openssl來生成密鑰,這里我選擇支付寶密鑰生成器,如圖:

然后點(diǎn)擊它之后跳轉(zhuǎn)到下載界面下載,如圖:

下載好后打開該工具,選擇好密鑰長度和密鑰格式并生成密鑰,如圖:

然后進(jìn)入公私鑰的目錄,將這個(gè)復(fù)制到我們的Django項(xiàng)目的子應(yīng)用目錄中,并重命名,等下用的著,如圖:


緊接著我們進(jìn)入自己的開發(fā)者中心控制臺(tái),地址:
https://open.alipay.com/platform/developerIndex.htm然后我們?nèi)?chuàng)建一個(gè)應(yīng)用,如圖:

按照要求如實(shí)填寫即可。然后我們來設(shè)置它的接口加密方式,如圖:


驗(yàn)證好了之后填寫剛剛生成的應(yīng)用公鑰,如圖:

此時(shí)會(huì)出現(xiàn)應(yīng)用公鑰和支付寶公鑰,將支付寶公鑰保存起來,如圖:

然后我們將產(chǎn)生的額應(yīng)用公私鑰和支付寶公鑰保存為下列內(nèi)容形式的文件,如圖:


將這三個(gè)文件都保存在rsakey這個(gè)文件夾中。現(xiàn)在準(zhǔn)備工作都做好了,下面開始編寫支付寶支付接口。
注:項(xiàng)目審核通過后才可以使用密鑰調(diào)用支付寶接口噢!
四、PC端支付寶支付接口
這里我們使用一個(gè)類將它封裝起來,如下:
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 = 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)# 將字典類型的數(shù)據(jù)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):# 開始計(jì)算簽名key = self.app_private_keysigner = PKCS1_v1_5.new(key)signature = signer.sign(SHA256.new(unsigned_string))# base64 編碼,轉(zhuǎn)換為unicode表示并移除回車sign = encodebytes(signature).decode("utf8").replace("\n", "")return signdef _verify(self, raw_content, signature):# 開始計(jì)算簽名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)
為了便于調(diào)用,我們將這個(gè)Python文件放在子應(yīng)用的目錄中,命名為pay.py。
五、編寫前端頁面
我們通過前端的商品的名稱和價(jià)格來生成對(duì)應(yīng)的商品信息并發(fā)起付款請(qǐng)求,如下:
index.html(商品主頁)
<html lang="en"><head><meta charset="UTF-8"><title>Documenttitle><style>table,table tr th, table tr td { border:1px solid #0094ff; }table { width:300px; min-height: 25px; line-height: 25px; text-align: center; border-collapse: collapse; padding:2px;}a{text-decoration: none;}style>head><body><h1>歡迎來到購物商場h1><table border="1"><thead>商品目錄thead><tr><td>商品名td><td>商品單價(jià)td><td>商品數(shù)量td><td>是否購買td>tr><tr><td>梨子td><td>0.1td><td>1td><td><a href="{% url 'dingdan' %}">購買a>td>table>body>html>
show.html(支付結(jié)果顯示頁)
<html lang="en"><head><meta charset="UTF-8"><title>Documenttitle>head><body><h1>支付結(jié)果:{{msg}}h1>body>html>
六、編寫視圖函數(shù)處理渲染
from django.shortcuts import render,redirectfrom django.http import HttpResponse,JsonResponsefrom .pay import AliPayimport uuidfrom urllib.parse import parse_qs# Create your views here.def index(request):return render(request,'index.html')def dingdan(request):# 實(shí)例化AliPayalipay = AliPay(appid="自己的APPID",app_notify_url='http://127.0.0.1:8000/paypay/check/',#支付寶會(huì)向這個(gè)地址發(fā)送post請(qǐng)求return_url='http://127.0.0.1:8000/paypay/show/',#支付寶會(huì)向這個(gè)地址發(fā)送get請(qǐng)求app_private_key_path=r'C:\Users\Administrator\Desktop\pay\paypay\rsakey\private2048.txt', # 應(yīng)用私鑰alipay_public_key_path=r'C:\Users\Administrator\Desktop\pay\paypay\rsakey\paypublic.txt', # 支付寶公鑰debug=True, # 默認(rèn)是False)# 定義請(qǐng)求地址傳入的參數(shù)res=alipay.direct_pay(subject='梨子', # 商品描述out_trade_no=str(uuid.uuid4()), # 訂單號(hào)total_amount='0.1', # 交易金額(單位是元,保留兩位小數(shù)))#生成跳轉(zhuǎn)到支付寶支付頁面的urlurl='https://openapi.alipaydev.com/gateway.do?{0}'.format(res)return redirect(url)def show(request):if request.method == 'GET':alipay = AliPay(appid="自己的APPID",app_notify_url='http://127.0.0.1:8000/paypay/check/',return_url='http://127.0.0.1:8000/paypay/show/',app_private_key_path=r'C:\Users\Administrator\Desktop\pay\paypay\rsakey\private2048.txt', # 應(yīng)用私鑰alipay_public_key_path=r'C:\Users\Administrator\Desktop\pay\paypay\rsakey\paypublic.txt', # 支付寶公鑰debug=True, # 默認(rèn)是False)param=request.GET.dict() # 獲取請(qǐng)求攜帶的參數(shù)并轉(zhuǎn)換成字典類型sign=param.pop('sign', None) # 獲取sign的值# 對(duì)sign參數(shù)進(jìn)行驗(yàn)證statu = alipay.verify(param,sign)if statu:return render(request, 'show.html', {'msg': '支付成功'})else:return render(request, 'show.html', {'msg': '支付失敗'})else:return render(request, 'show.html', {'msg': '只支持GET請(qǐng)求,不支持其它請(qǐng)求'})def check(request):if request.method=='POST':alipay=AliPay(appid="自己的APPID",app_notify_url='http://127.0.0.1:8000/paypay/check/', # 支付寶會(huì)向這個(gè)地址發(fā)送post請(qǐng)求return_url='http://127.0.0.1:8000/show_msg/', # 支付寶會(huì)向這個(gè)地址發(fā)送get請(qǐng)求app_private_key_path=r'C:\Users\Administrator\Desktop\pay\paypay\rsakey\private2048.txt', # 應(yīng)用私鑰alipay_public_key_path=r'C:\Users\Administrator\Desktop\pay\paypay\rsakey\paypublic.txt', # 支付寶公鑰debug=True,)body=request.body.decode('utf-8') # 轉(zhuǎn)成字符串post_data = parse_qs(body) # 根據(jù)&符號(hào)分割post_dict = {}for k, v in post_data.items():post_dict[k] = v[0]sign = post_dict.pop('sign', None)status = alipay.verify(post_dict, sign)if status: # 支付成功return HttpResponse('支付成功')else:return HttpResponse('支付失敗')else:return HttpResponse('只支持POST請(qǐng)求')
????????
七、添加路由函數(shù)到url規(guī)則中
from django.urls import pathfrom . import viewsurlpatterns=[path('',views.index,name='index'),path('dingdan/',views.dingdan,name='dingdan'),path('show/',views.show,name='show'),path('check/',views.check,name='check'),]
八、運(yùn)行項(xiàng)目
所有準(zhǔn)備工作都做好了,我們趕緊來試著運(yùn)行下項(xiàng)目吧,如下:

可以看到我們購買商品后鏈接成功跳轉(zhuǎn)到支付界面。
九、總結(jié)
該支付寶支付環(huán)境在沙箱中實(shí)現(xiàn),因此安全性毋庸置疑,代碼小編已經(jīng)打包好了,不過里面的appid還有公私鑰需要大家自行填寫噢。
最后需要本文項(xiàng)目代碼的小伙伴,請(qǐng)?jiān)诠娞?hào)后臺(tái)回復(fù)“支付寶”關(guān)鍵字進(jìn)行獲取,如果在運(yùn)行過程中有遇到任何問題,請(qǐng)隨時(shí)留言或者加小編好友,小編看到會(huì)幫助大家解決bug噢!
往期推薦
