支付宝开放平台入口
https://open.alipay.com/platform/home.htm1.创建应用
2.沙箱环境
支付宝提供给开发者的模拟支付的环境。跟真实环境是分开的。
沙箱应用:https://openhome.alipay.com/platform/appDaily.htm?tab=info沙箱账号:https://openhome.alipay.com/platform/appDaily.htm?tab=account1、用户下单并调用alipay.trade.page.pay,发起支付请求
2、用户输入用户名,支付密码,并登陆
3、用户选择支付渠道,输入支付密码
4、用户确认支付
5、支付宝get请求returnUrl,返回同步返回参数
6、支付宝post请求notifyUrl,返回异步通知参数
7、商城系统调用alipay.trade.query查看交易状态
提示:
美多商城私钥加密数据,美多商城公钥解密数据。支付宝私钥加密数据,支付宝公钥解密数据。-----BEGIN PUBLIC KEY-----
支付宝公钥内容
-----END PUBLIC KEY-----
选项
方案
请求方法
GET
请求地址
/payment/(?P<order_id>\d+)/
2.请求参数:路径参数
参数名
类型
是否必传
说明
order_id
int
是
订单编号
3.响应结果:JSON
字段
说明
code
状态码
errmsg
错误信息
alipay_url
支付宝登录链接
4.后端接口定义和实现
# 测试账号:pqcanx4910@sandbox.com class PaymentView(LoginRequiredJSONMixin, View): """订单支付功能""" def get(self,request, order_id): # 查询要支付的订单 user = request.user try: order = OrderInfo.objects.get(order_id=order_id, user=user, status=OrderInfo.ORDER_STATUS_ENUM['UNPAID']) except OrderInfo.DoesNotExist: return http.HttpResponseForbidden('订单信息错误') # 创建支付宝支付对象 alipay = AliPay( appid=settings.ALIPAY_APPID, app_notify_url=None, # 默认回调url app_private_key_path=os.path.join(os.path.dirname(os.path.abspath(__file__)), "keys/app_private_key.pem"), alipay_public_key_path=os.path.join(os.path.dirname(os.path.abspath(__file__)), "keys/alipay_public_key.pem"), sign_type="RSA2", debug=settings.ALIPAY_DEBUG ) # 生成登录支付宝连接 order_string = alipay.api_alipay_trade_page_pay( out_trade_no=order_id, total_amount=str(order.total_amount), subject="美多商城%s" % order_id, return_url=settings.ALIPAY_RETURN_URL, ) # 响应登录支付宝连接 # 真实环境电脑网站支付网关:https://openapi.alipay.com/gateway.do? + order_string # 沙箱环境电脑网站支付网关:https://openapi.alipaydev.com/gateway.do? + order_string alipay_url = settings.ALIPAY_URL + "?" + order_string return http.JsonResponse({'code': RETCODE.OK, 'errmsg': 'OK', 'alipay_url': alipay_url})5.支付宝SDK配置参数
# 支付宝 ALIPAY_APPID = '2016091900551154' ALIPAY_DEBUG = True # 表示是沙箱环境还是真实支付环境 ALIPAY_URL = 'https://openapi.alipaydev.com/gateway.do' ALIPAY_RETURN_URL = 'http://www.meiduo.site:8000/payment/status/'我们需要将订单编号和交易流水号进行关联存储,方便用户和商家后续使用。
1.请求方式
选项
方案
请求方法
GET
请求地址
/payment/status/
2.请求参数:路径参数
参考统一收单下单并支付页面接口中的《页面回跳参数》
3.响应结果:HTML
pay_success.html
4.后端接口定义和实现
注意:保存订单支付结果的同时,还需要修改订单的状态为待评价
# 测试账号:pqcanx4910@sandbox.com class PaymentStatusView(View): """保存订单支付结果""" def get(self, request): # 获取前端传入的请求参数 query_dict = request.GET data = query_dict.dict() # 获取并从请求参数中剔除signature signature = data.pop('sign') # 创建支付宝支付对象 alipay = AliPay( appid=settings.ALIPAY_APPID, app_notify_url=None, app_private_key_path=os.path.join(os.path.dirname(os.path.abspath(__file__)), "keys/app_private_key.pem"), alipay_public_key_path=os.path.join(os.path.dirname(os.path.abspath(__file__)), "keys/alipay_public_key.pem"), sign_type="RSA2", debug=settings.ALIPAY_DEBUG ) # 校验这个重定向是否是alipay重定向过来的 success = alipay.verify(data, signature) if success: # 读取order_id order_id = data.get('out_trade_no') # 读取支付宝流水号 trade_id = data.get('trade_no') # 保存Payment模型类数据 Payment.objects.create( order_id=order_id, trade_id=trade_id ) # 修改订单状态为待评价 OrderInfo.objects.filter(order_id=order_id, status=OrderInfo.ORDER_STATUS_ENUM['UNPAID']).update( status=OrderInfo.ORDER_STATUS_ENUM["UNCOMMENT"]) # 响应trade_id context = { 'trade_id':trade_id } return render(request, 'pay_success.html', context) else: # 订单支付失败,重定向到我的订单 return http.HttpResponseForbidden('非法请求')5.渲染支付成功页面信息
<div class="common_list_con clearfix"> <div class="order_success"> <p><b>订单支付成功</b></p> <p>您的订单已成功支付,支付交易号:{{ trade_id }}</p> <p><a href="{{ url('orders:info', args=(1, )) }}">您可以在【用户中心】->【我的订单】查看该订单</a></p> </div> </div>