Python支付寶支付(沙箱)
1.安裝python-alipay-sdk
# cmd中輸入
pip install python-alipay-sdk
2.生成並使用密鑰
進入支付寶開放平台-->登陸-->導航欄開發中心-->研發服務-->沙箱應用-->應用公鑰(根據提示下載支付寶提供的windows版本的密鑰生成工具)-->將生成的應用公鑰復制到應用公鑰
3.新建Django項目
新建Django項目,然后將剛才生成的應用私鑰和支付寶公鑰放到項目根目錄下新建的keys文件中。
第一個是你應用的私鑰,第二個是支付寶公鑰(就在沙箱應用應用公鑰的旁邊,我忘記改名字了0 0)。由於我當時把這個存成應用公鑰,所有在最后的支付頁面后端會報錯:alipay.exceptions.AliPayValidationError(無法驗證 0 0)。存儲的時候需要在密鑰的頭尾加上-----BEGIN PUBLIC KEY-----:
# app_private_2048.txt
-----BEGIN PUBLIC KEY-----
應用私鑰
-----BEGIN PUBLIC KEY-----
# app_public_2048
-----BEGIN PUBLIC KEY-----
支付寶公鑰
-----BEGIN PUBLIC KEY-----
然后配置settings,在末尾加上下面內容:
# settings.py
# 支付寶參數配置
# 在支付寶開放平台的沙箱應用中獲取APPID和支付寶網關
ALIPAY_APPID = '這里是你的APPID'
ALIPAY_URL = 'https://openapi.alipaydev.com/gateway.do' # 支付寶網關
4、代碼
templates-->index.html:
# index.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>支付</title>
<script src="/static/jquery-3.3.1.js"></script>
<script>
$(function () {
$('#btn').click(function () {
var order_id = "2016101400682827";
var req_data = {
order_id: order_id,
csrfmiddlewaretoken: "{{ csrf_token }}"
};
$.post("/pay/", req_data, function (data) {
window.open(data.url)
});
$.get("/check_pay/?order_id=" + order_id, function (data) {
if (0 == data.code) {
// 支付成功
alert("支付成功");
location.reload();
} else {
alert(data.message)
}
})
})
})
</script>
</head>
<body>
<input type="button" id="btn" value="支付">
</body>
</html>
views.py:
# views.py
from django.shortcuts import render
from django.http import JsonResponse
from alipay import AliPay
import os
import time
from django.conf import settings
def index(request):
return render(request, "index.html", locals())
def pay(request):
order_id = request.POST.get("order_id")
# 創建用於進行支付寶支付的工具對象
alipay = AliPay(
appid=settings.ALIPAY_APPID,
app_notify_url=None, # 默認回調url
app_private_key_path=os.path.join(settings.BASE_DIR, r"keys\app_private_2048.txt"),
alipay_public_key_path=os.path.join(settings.BASE_DIR, r"keys\app_public_2048.txt"),
# 支付寶的公鑰,驗證支付寶回傳消息使用,不是你自己的公鑰,
sign_type="RSA2", # RSA 或者 RSA2
debug=True # 默認False 配合沙箱模式使用
)
# 電腦網站支付,需要跳轉到https://openapi.alipaydev.com/gateway.do? + order_string
order_string = alipay.api_alipay_trade_page_pay(
out_trade_no=f'xx{time.time()}', # 生成隨時間變動而變動的唯一訂單號
total_amount=str(0.01), # 將Decimal類型轉換為字符串交給支付寶
subject="測試訂單",
body="Python入門視頻",
return_url="https://example.com",
notify_url="https://example.com/notify" # 可選, 不填則使用默認notify url
)
# 讓用戶進行支付的支付寶頁面網址
url = settings.ALIPAY_URL + "?" + order_string
return JsonResponse({"code": 0, "message": "請求支付成功", "url": url})
def check_pay(request):
# 創建用於進行支付寶支付的工具對象
order_id = request.GET.get("order_id")
alipay = AliPay(
appid=settings.ALIPAY_APPID,
app_notify_url=None, # 默認回調url
app_private_key_path=os.path.join(settings.BASE_DIR, r"keys\app_private_2048.txt"),
alipay_public_key_path=os.path.join(settings.BASE_DIR, r"keys\app_public_2048.txt"),
# 支付寶的公鑰,驗證支付寶回傳消息使用,不是你自己的公鑰,
sign_type="RSA2", # RSA2,官方推薦,配置公鑰的時候能看到
debug=True # 默認False 配合沙箱模式使用
)
while True:
# 調用alipay工具查詢支付結果
response = alipay.api_alipay_trade_query(order_id) # response是一個字典
# 判斷支付結果
code = response.get("code") # 支付寶接口調用成功或者錯誤的標志
trade_status = response.get("trade_status") # 用戶支付的情況
if code == "10000" and trade_status == "TRADE_SUCCESS":
# 表示用戶支付成功
# 返回前端json,通知支付成功
return JsonResponse({"code": 0, "message": "支付成功"})
elif code == "40004" or (code == "10000" and trade_status == "WAIT_BUYER_PAY"):
# 表示支付寶接口調用暫時失敗,(支付寶的支付訂單還未生成) 后者 等待用戶支付
# 繼續查詢
print(code)
print(trade_status)
continue
else:
# 支付失敗
# 返回支付失敗的通知
return JsonResponse({"code": 1, "message": "支付失敗"})
urls.py:
# urls.py
from django.conf.urls import url
from django.contrib import admin
from pay import views
urlpatterns = [
url(r'^admin/', admin.site.urls),
url(r'^$', views.index, name='index'),
url(r"^pay/$", views.pay, name='pay'),
url(r"^check_pay/$", views.check_pay, name='check_pay'),
]