支付宝

简介

支付宝是什么不用多说了,本次教程适合初学者

 

SRE实战 互联网时代守护先锋,助力企业售后服务体系运筹帷幄!一键直达领取阿里云限量特价优惠。

前提准备

话不多说,干就完了

 

1.注册开发者账号,设置公钥私钥

首先进入支付宝开发者平台:传送门 ,有账号直接登录,没账号用你平时用来付款收钱的账号登录,然后用这个账号激活注册成开发者账号就行了

登录之后点开发中心:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第1张

先用沙箱测试一下,点研发服务

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第2张

 

 跳转到此页面,设置一个公钥

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第3张

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第4张

 

 

先点查看公钥生成:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第5张

 

 进入开发文档:传送门  根据文档步骤下载秘钥生成工具,根据说明文档运行程序

 

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第6张

 

本次教程使用的是Python,所以一定要选非java,然后点击生成秘钥

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第7张

 

 再点打开秘钥路径就可以看到已经生成了两个文件:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第8张

 

用这里的公钥和私钥,填入刚才那个设置页面:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第9张

如果在保存时提示什么公钥签名啥啥的,反正不成功的提示,利用秘钥生成工具重新生成一次填入即可

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第10张

接着按这里的文档走就行:传送门  需要一个安卓手机下载沙箱版的支付宝作为测试

 

2.安装sdk 

 

之前使用支付宝,都要去github上download大神写的sdk,现在pyi社区已经有官方的sdk了:传送门  所以,直接pip install alipay-sdk-python 安装:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第11张

 但是到最后会报错:

 前后端分离djangorestframework—— 接入支付宝支付平台 Python 第12张

 

然后看到说是因为安装支付宝的sdk时,由于需要安装这个库,而这个库需要依赖微软的visual c++才行,这就尴尬了,据查,要安装visual C++最好的办法是,下载visual Studio集成开发工具,注意不是visual studio code,两个是不同的软件,虽然都可以当开发工具,我的理解就是visual studio比visual studio code多了那些需要的运行库

 visual studio官方下载链接最新版   另外据网查,可以只安装visual studio 2015版就行,visual studio 2015下载链接 ,总共有几个G,解压并安装,只安装这个c++ 2015

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第13张

 

等好长一会儿:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第14张

 

 然后再安装支付宝的sdk看看,还是不行

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第15张

 按报错提示,那个c1.exe不被认识,所以还要设置环境变量:变量名:VCINSTALLDIR,变量值就是vc的路径

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第16张

 

设置了环境变量后关闭cmd,重新打开cmd, 先使用命令设置:set CL=/FI"%VCINSTALLDIR%\\INCLUDE\\stdint.h" 这条命令是为了让刚才我们设置的环境变量生效

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第17张

 

然后pip 安装,终于成功了

 前后端分离djangorestframework—— 接入支付宝支付平台 Python 第18张

注:如果报错:UnicodeDecodeError: 'utf-8' codec can't decode byte... 将CMD的终端编码用“CHCP 65001”命令改为“UTF-8”后再安装即可

 

进入支付宝支付开发

 

好接着进入真正的开发阶段了,本次选用【电脑页面支付】:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第19张

 

然后进到支付宝的api文档:传送门

选用  统一收单下单支付接口  

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第20张

进入的目录就不展示了,里面的参数配置就是正式的支付接口,这个就根据自己实际情况配置了,本次我们只用沙箱账号测试一下就行了,和正式的配置是一样的,因为正式的配置要认证商家才行,所以为了方便且精简的演示,就用沙箱账号了

 

在前面设置公钥私钥那个页面,把安卓版的沙箱支付宝下载安装好,使用这里的账号密码登录:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第21张

 

 准备好之后,现在进入真正的开发了。

 

1.创建一个django项目,一个简单的支付宝充值话费的

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第22张

 

配置文件里导入rest_framework  app,设置url,其中pay则是一会儿要用到的测试url

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第23张

html文件:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第24张

 

 

view,就按照pypi社区上支付宝给的案例调整代码就行:

 

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第25张
#!/usr/bin/env python # -*- coding: utf-8 -*-
import logging import traceback from alipay.aop.api.AlipayClientConfig import AlipayClientConfig from alipay.aop.api.DefaultAlipayClient import DefaultAlipayClient from alipay.aop.api.FileItem import FileItem from alipay.aop.api.domain.AlipayTradeAppPayModel import AlipayTradeAppPayModel from alipay.aop.api.domain.AlipayTradePagePayModel import AlipayTradePagePayModel from alipay.aop.api.domain.AlipayTradePayModel import AlipayTradePayModel from alipay.aop.api.domain.GoodsDetail import GoodsDetail from alipay.aop.api.domain.SettleDetailInfo import SettleDetailInfo from alipay.aop.api.domain.SettleInfo import SettleInfo from alipay.aop.api.domain.SubMerchant import SubMerchant from alipay.aop.api.request.AlipayOfflineMaterialImageUploadRequest import AlipayOfflineMaterialImageUploadRequest from alipay.aop.api.request.AlipayTradeAppPayRequest import AlipayTradeAppPayRequest from alipay.aop.api.request.AlipayTradePagePayRequest import AlipayTradePagePayRequest from alipay.aop.api.request.AlipayTradePayRequest import AlipayTradePayRequest from alipay.aop.api.response.AlipayOfflineMaterialImageUploadResponse import AlipayOfflineMaterialImageUploadResponse from alipay.aop.api.response.AlipayTradePayResponse import AlipayTradePayResponse logging.basicConfig( level=logging.INFO, format='%(asctime)s %(levelname)s %(message)s', filemode='a',) logger = logging.getLogger('') if __name__ == '__main__': """ 设置配置,包括支付宝网关地址、app_id、应用私钥、支付宝公钥等,其他配置值可以查看AlipayClientConfig的定义。 """ alipay_client_config = AlipayClientConfig() alipay_client_config.server_url = 'https://openapi.alipay.com/gateway.do' alipay_client_config.app_id = '[your app_id]' alipay_client_config.app_private_key = '[your app private key]' alipay_client_config.alipay_public_key = '[alipay public key]'

    """ 得到客户端对象。 注意,一个alipay_client_config对象对应一个DefaultAlipayClient,定义DefaultAlipayClient对象后,alipay_client_config不得修改,如果想使用不同的配置,请定义不同的DefaultAlipayClient。 logger参数用于打印日志,不传则不打印,建议传递。 """ client = DefaultAlipayClient(alipay_client_config=alipay_client_config, logger=logger) """ 系统接口示例:alipay.trade.pay """
    # 对照接口文档,构造请求对象
    model = AlipayTradePayModel() model.auth_code = "282877775259787048" model.body = "Iphone6 16G" goods_list = list() goods1 = GoodsDetail() goods1.goods_id = "apple-01" goods1.goods_name = "ipad" goods1.price = 10 goods1.quantity = 1 goods_list.append(goods1) model.goods_detail = goods_list model.operator_id = "yx_001" model.out_trade_no = "20180510AB014" model.product_code = "FACE_TO_FACE_PAYMENT" model.scene = "bar_code" model.store_id = "" model.subject = "huabeitest" model.timeout_express = "90m" model.total_amount = 1 request = AlipayTradePayRequest(biz_model=model) # 如果有auth_token、app_auth_token等其他公共参数,放在udf_params中
    # udf_params = dict()
    # from alipay.aop.api.constant.ParamConstants import *
    # udf_params[P_APP_AUTH_TOKEN] = "xxxxxxx"
    # request.udf_params = udf_params
    # 执行请求,执行过程中如果发生异常,会抛出,请打印异常栈
    response_content = None try: response_content = client.execute(request) except Exception as e: print(traceback.format_exc()) if not response_content: print("failed execute") else: response = AlipayTradePayResponse() # 解析响应结果
 response.parse_response_content(response_content) print(response.body) if response.is_success(): # 如果业务成功,则通过respnse属性获取需要的值
            print("get response trade_no:" + response.trade_no) else: # 如果业务失败,则从错误码中可以得知错误情况,具体错误码信息可以查看接口文档
            print(response.code + "," + response.msg + "," + response.sub_code + "," + response.sub_msg) """ 带文件的系统接口示例:alipay.offline.material.image.upload """
    # 如果没有找到对应Model类,则直接使用Request类,属性在Request类中
    request = AlipayOfflineMaterialImageUploadRequest() request.image_name = "我的店" request.image_type = "jpg"
    # 设置文件参数
    f = open("/Users/foo/Downloads/IMG.jpg", "rb") request.image_content = FileItem(file_name="IMG.jpg", file_content=f.read()) f.close() response_content = None try: response_content = client.execute(request) except Exception as e: print(traceback.format_exc()) if not response_content: print("failed execute") else: response = AlipayOfflineMaterialImageUploadResponse() response.parse_response_content(response_content) if response.is_success(): print("get response image_url:" + response.image_url) else: print(response.code + "," + response.msg + "," + response.sub_code + "," + response.sub_msg) """ 页面接口示例:alipay.trade.page.pay """
    # 对照接口文档,构造请求对象
    model = AlipayTradePagePayModel() model.out_trade_no = "pay201805020000226" model.total_amount = 50 model.subject = "测试" model.body = "支付宝测试" model.product_code = "FAST_INSTANT_TRADE_PAY" settle_detail_info = SettleDetailInfo() settle_detail_info.amount = 50 settle_detail_info.trans_in_type = "userId" settle_detail_info.trans_in = "2088302300165604" settle_detail_infos = list() settle_detail_infos.append(settle_detail_info) settle_info = SettleInfo() settle_info.settle_detail_infos = settle_detail_infos model.settle_info = settle_info sub_merchant = SubMerchant() sub_merchant.merchant_id = "2088301300153242" model.sub_merchant = sub_merchant request = AlipayTradePagePayRequest(biz_model=model) # 得到构造的请求,如果http_method是GET,则是一个带完成请求参数的url,如果http_method是POST,则是一段HTML表单片段
    response = client.page_execute(request, http_method="GET") print("alipay.trade.page.pay response:" + response) """ 构造唤起支付宝客户端支付时传递的请求串示例:alipay.trade.app.pay """ model = AlipayTradeAppPayModel() model.timeout_express = "90m" model.total_amount = "9.00" model.seller_id = "2088301194649043" model.product_code = "QUICK_MSECURITY_PAY" model.body = "Iphone6 16G" model.subject = "iphone" model.out_trade_no = "201800000001201" request = AlipayTradeAppPayRequest(biz_model=model) response = client.sdk_execute(request) print("alipay.trade.app.pay response:" + response)
pypi社区上支付宝官方案例 前后端分离djangorestframework—— 接入支付宝支付平台 Python 第27张
#!/usr/bin/env python # -*- coding: utf-8 -*-
from alipay.aop.api.AlipayClientConfig import AlipayClientConfig from alipay.aop.api.DefaultAlipayClient import DefaultAlipayClient from alipay.aop.api.domain.AlipayTradePagePayModel import AlipayTradePagePayModel from alipay.aop.api.request.AlipayTradePagePayRequest import AlipayTradePagePayRequest from rest_framework.views import APIView from rest_framework.response import Response from django.shortcuts import render, redirect, HttpResponse def alipayclient(): """ 设置配置,包括支付宝网关地址、app_id、应用私钥、支付宝公钥等,其他配置值可以查看AlipayClientConfig的定义。 """ alipay_client_config = AlipayClientConfig() alipay_client_config.server_url = 'https://openapi.alipay.com/gateway.do' alipay_client_config.app_id = '您的app_Id' with open("keys/private_key.txt") as f: alipay_client_config.app_private_key = f.read() # 阿里的公钥
    with open("keys/public_key.txt") as f: alipay_client_config.alipay_public_key = f.read() """ 得到客户端对象。 注意,一个alipay_client_config对象对应一个DefaultAlipayClient,定义DefaultAlipayClient对象后,alipay_client_config不得修改,如果想使用不同的配置,请定义不同的DefaultAlipayClient。 logger参数用于打印日志,不传则不打印,建议传递。 """ client = DefaultAlipayClient(alipay_client_config=alipay_client_config) return client """ 系统接口示例:alipay.trade.pay """


class AlipayView(APIView): def get(self, request): return render(request, 'pay.html') def post(self, request): money = request.data.get('money') if not money.isdigit(): return Response('错误,只能是数字') money = int(money) client = alipayclient() """ 页面接口示例:alipay.trade.page.pay """
        # 对照接口文档,构造请求对象
        model = AlipayTradePagePayModel() import time model.out_trade_no = 'pay201805020000226' model.total_amount = money model.subject = "测试" model.body = "支付宝测试" model.product_code = "FAST_INSTANT_TRADE_PAY" request = AlipayTradePagePayRequest(biz_model=model) # 得到构造的请求,如果http_method是GET,则是一个带完成请求参数的url,如果http_method是POST,则是一段HTML表单片段
        response = client.page_execute(request, http_method="GET") print("alipay.trade.page.pay response:" + response) return redirect(response)
views

 

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第29张
from django.contrib import admin from django.urls import path from app.views import AlipayView urlpatterns = [ path('admin/', admin.site.urls), path('pay/',AlipayView.as_view()), ]
urls   前后端分离djangorestframework—— 接入支付宝支付平台 Python 第31张
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>支付宝电脑端支付-话费充值</title>
</head>
<body>

<form method="post">
    {% csrf_token %}
    <label for="money">金额:</label>
    <input type="text" placeholder="请输入充值金额" id="money" name="money">
    <label for="phone">号码:</label>
    <input type="text" value="183XXXXX" id="phone" name="phone">
    <input type="submit" value="立即支付宝充值">
</form>
</body>
</html>
pay.html

 

 

 

 

 

 

然后重启项目: 

注意,有坑,真的有坑,照这么设置启动,点击支付的时候会跳到这个页面:

 

第一个坑:

 

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第33张

 

 

反正就这两个页面,无效的appID,其实我用的那个ID就是支付宝沙箱给的ID号,所以不可能有错,但是注意了,我们用的是沙箱账号,而我们直接拷贝的支付宝给的案例,用的url是【openapi.alipay.com】,这个url是正式的url,并不是沙箱测试的url,所以沙箱测试的url是【https://openapi.alipaydev.com/gateway.do】

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第34张

 

 

第二个坑: 

修改url之后,再次提交,还是说订单信息有错误,

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第35张

 

 

到底是哪里的问题呢?再检查打码,还是刚才的逻辑,沙箱账号啊,那么在初始化支付宝配置时,读源码得,需要添加这个参数 【sandbox_debug = True】:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第36张

 

第三个坑:

按上面的修改重启,还是报错: 

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第37张

 

 

 

查看支付宝给的这个错误代码解析,交易信息被篡改,好像也没有说清楚啥问题

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第38张

 

我试着改了下这个流水号,把它改成了用事件戳随机生成的,【model.out_trade_no = "pay" + str(time.time())】

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第39张

 

重启项目,输入金额23

 前后端分离djangorestframework—— 接入支付宝支付平台 Python 第40张

 

 

点击立即支付宝充值,这页面终于出来了

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第41张

 

 

那么那个流水号有多大作用呢?我直接删除看看:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第42张

 

 

 

那么就得必须带上它了

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第43张

 

但是支付宝官方sdk里给的案例也貌似没有说清楚啊,只是说看着像时间戳:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第44张

但是也没说这玩意得按时间戳来实时接收,还不能写死了,其实呢,仔细一想,这个流水号是不是得按当时付款时间来啊?难道以后每个时间段支付的都是同一个流水号?这不乱套了吗?支付宝官方没有提的原因,我猜啊,很大可能,是因为之前不是有大神写过sdk了嘛,那么那些参数或许在哪个大神的sdk里有解释了,而官方的这个sdk是之后发布的,所以可能开发者以为之前这老哥都开发过了,所以这些没必要再提了。好的,反正这里有个坑,注意就行了

 

接着后面的操作,打开手机,用那个沙箱支付宝app登录,扫码支付:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第45张

 

2.完善代码 

可以支付了对吧,然后按个订单流水号就是刚才设置的那个咯,但是电脑页面还是这样,没有任何反馈啊,作为商家我们根本不知道买家那边什么情况,到底支付没有也不知道对吧,不可能一个一个的去自己账单里查账吧?如果人多呢,几个人同时支付怎么办呢?

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第46张

 

 

所以,这里还得配置一下,再写一个url来接收一下数据,并返回支付结果:

url:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第47张

 

view:

主视图类添加两个url属性

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第48张

 

 新的url对应的识图类:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第49张

 

重启项目,支付查看结果:

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第50张

 

 

 

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第51张

 

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第52张

 

 

 

 终端打印的结果:

 

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第53张

 

OK,终于完事儿了

 

 

相关代码:

 

前后端分离djangorestframework—— 接入支付宝支付平台 Python 第54张
from django.contrib import admin from django.urls import path from app.views import AlipayView from app.views import PayHandlerView urlpatterns = [ path('admin/', admin.site.urls), path('pay/',AlipayView.as_view()), path('alipay_handler',PayHandlerView.as_view()) ]
url 前后端分离djangorestframework—— 接入支付宝支付平台 Python 第56张
#!/usr/bin/env python # -*- coding: utf-8 -*-
from alipay.aop.api.AlipayClientConfig import AlipayClientConfig from alipay.aop.api.DefaultAlipayClient import DefaultAlipayClient from alipay.aop.api.domain.AlipayTradePagePayModel import AlipayTradePagePayModel from alipay.aop.api.request.AlipayTradePagePayRequest import AlipayTradePagePayRequest from rest_framework.views import APIView from rest_framework.response import Response from django.shortcuts import render, redirect, HttpResponse def alipayclient(): """ 设置配置,包括支付宝网关地址、app_id、应用私钥、支付宝公钥等,其他配置值可以查看AlipayClientConfig的定义。 """ alipay_client_config = AlipayClientConfig(sandbox_debug=True) alipay_client_config.server_url = 'https://openapi.alipaydev.com/gateway.do' alipay_client_config.app_id = '您的app_id' with open("keys/private_key.txt") as f: alipay_client_config.app_private_key = f.read() # 阿里的公钥
    with open("keys/public_key.txt") as f: alipay_client_config.alipay_public_key = f.read() """ 得到客户端对象。 注意,一个alipay_client_config对象对应一个DefaultAlipayClient,定义DefaultAlipayClient对象后,alipay_client_config不得修改,如果想使用不同的配置,请定义不同的DefaultAlipayClient。 logger参数用于打印日志,不传则不打印,建议传递。 """ client = DefaultAlipayClient(alipay_client_config=alipay_client_config) return client """ 系统接口示例:alipay.trade.pay """


class AlipayView(APIView): def get(self, request): return render(request, 'pay.html') def post(self, request): money = request.data.get('money') if not money.isdigit(): return Response('错误,只能是数字') money = int(money) client = alipayclient() """ 页面接口示例:alipay.trade.page.pay """
        # 对照接口文档,构造请求对象
        model = AlipayTradePagePayModel() import time model.out_trade_no = 'pay' + str(time.time()) model.total_amount = money model.subject = "测试" model.body = "支付宝测试" model.product_code = "FAST_INSTANT_TRADE_PAY" request = AlipayTradePagePayRequest(biz_model=model) # 得到构造的请求,如果http_method是GET,则是一个带完成请求参数的url,如果http_method是POST,则是一段HTML表单片段
        # get请求 用户支付成功后返回的页面请求地址,这个可以是公网地址
        request.return_url = "http://127.0.0.1:8002/alipay_handler"
        # post请求 用户支付成功通知商户的请求地址
        request.notify_url = "http://127.0.0.1:8002/alipay_handler" response = client.page_execute(request, http_method="GET") print("alipay.trade.page.pay response:" + response) return redirect(response) class PayHandlerView(APIView): def get(self, request): # return_url的回调地址
        print(request.data) # 用户支付成功之后回到哪
        return HttpResponse("用户支付成功") def post(self, request): print(request.data) return HttpResponse("notify_url")
views

html文件没变,同上

 

总结:

  • 就是安装sdk的时候很烦人,如果你觉得不习惯,可以用github上那些大神写的sdk。我个人觉得,官方的只是安装有点繁琐,后面的配置是挺简单的
  • 在使用沙箱测试支付的时候的时候注意那三个坑就行了,因为沙箱测试和正式的支付还是与差距的

 

扫码关注我们
微信号:SRE实战
拒绝背锅 运筹帷幄