Django对接支付宝实现支付宝充值金币功能示例


Posted in Python onDecember 17, 2019

很多网站里都有金币、积分之类的虚拟货币,获取这些往往需要充值。那么问题来了,如何在Django中对接支付宝实现支付宝充值金币的功能呢?网上很多资料都是电商的,那些都会带有订单系统之类比较复杂,而充值金币功能不需要实现那么多功能。

效果图如下:

Django对接支付宝实现支付宝充值金币功能示例

Django对接支付宝实现支付宝充值金币功能示例

Django对接支付宝实现支付宝充值金币功能示例

现在就来实现Django对接支付宝支付功能吧!

登录支付宝开放平台

点击进入蚂蚁金服开放平台https://open.alipay.com/platform/home.htm

进入支付宝沙箱环境https://openhome.alipay.com/platform/appDaily.htm?tab=info

Django对接支付宝实现支付宝充值金币功能示例

如图,这里是你沙箱环境的配置,左侧沙箱工具有沙箱支付宝安卓版下载,沙箱账号是你的测试账号。

下载支付宝开放平台开发助手

点击打开下载链接

Django对接支付宝实现支付宝充值金币功能示例

打开工具,生成密钥,然后妥善保管好!!

Django对接支付宝实现支付宝充值金币功能示例

Django对接支付宝实现支付宝充值金币功能示例

保存密钥

Django对接支付宝实现支付宝充值金币功能示例

把刚刚那个应用公钥2048重命名为:pub_2048.txt,把应用私钥2048重命名为:private_2048.txt,把这两个文件放在项目目录下。

注意:密钥的开始和结束一定要加上如下的字符串!!!

-----BEGIN PRIVATE KEY-----
这里粘贴里面的密钥
-----END PRIVATE KEY-----

Django对接支付宝实现支付宝充值金币功能示例

复制支付宝的公钥并保存

Django对接支付宝实现支付宝充值金币功能示例

把这段支付宝公钥复制,重命名为alipay_key_2048.txt,保存到项目目录中,注意加上如下字符串

-----BEGIN PRIVATE KEY-----
这里粘贴里面的密钥
-----END PRIVATE KEY-----

设置公钥

把刚刚的pub_2048.txt里面的字符串复制到如下:

Django对接支付宝实现支付宝充值金币功能示例

调试支付宝支付接口

from datetime import datetime


from Crypto.PublicKey import RSA
from Crypto.Signature import PKCS1_v1_5
from Crypto.Hash import SHA256
from urllib.parse import quote_plus
from base64 import decodebytes, encodebytes

import json


class AliPay(object):
 """
 支付宝支付接口
 """

 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.import_key(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)


def get_alipay_url(app_id, order_sn, order_mount):
 alipay = AliPay(
  appid=app_id,
  app_notify_url="http://127.0.0.1:8000/alipay/return/",
  app_private_key_path="../trade/keys/private_2048.txt",
  alipay_public_key_path="../trade/keys/alipay_key_2048.txt", # 支付宝的公钥,验证支付宝回传消息使用,不是你自己的公钥,
  debug=True, # 默认False,
  return_url="http://127.0.0.1:8000/alipay/return/"
 )

 url = alipay.direct_pay(
  subject=order_sn,
  out_trade_no=order_sn,
  total_amount=order_mount,
 )
 re_url = "https://openapi.alipaydev.com/gateway.do?{data}".format(data=url)

 return re_url


if __name__ == "__main__":
 url = get_alipay_url(
  '2016092600597838',
  '201902923423436',
  1.00
 )
 print(url)

如果输出的url能够打开并且使用沙箱账号支付,说明前面的配置成功了,可以进行视图的编写。

设计数据库订单模型

from django.db import models
from django.conf import settings
from django.utils.encoding import python_2_unicode_compatible

from questioning.utils.models import CreatedUpdatedMixin

@python_2_unicode_compatible
class OrderInfo(CreatedUpdatedMixin, models.Model):
 '''
 充值订单详情
 '''
 ORDER_STATUS = (
 ('TRADE_SUCCESS', '交易支付成功'),
 ('TRADE_CLOSED', '未付款交易超时关闭'),
 ('WAIT_BUYER_PAY', '交易创建'),
 ('TRADE_FINISHED', '交易结束'),
 ('paying', '待支付'),
)
 user = models.ForeignKey(settings.AUTH_USER_MODEL, on_delete=models.CASCADE, verbose_name='用户')
 order_sn = models.CharField(max_length=30, null=True, blank=True, unique=True, verbose_name='订单号')
 trade_no = models.CharField(max_length=100, unique=True, null=True, blank=True, verbose_name='支付订单号')

 pay_status = models.CharField(choices=ORDER_STATUS, max_length=40, verbose_name='订单状态', default='paying')
 order_mount = models.DecimalField(verbose_name="充值金额", max_digits=10,
        decimal_places=2, default=0.00)


 class Meta:
  verbose_name = "充值订单"
  verbose_name_plural = verbose_name

 def __str__(self):
  return self.order_sn

注意充值的金额要设置为models.DecimalField货币类型

然后导入到数据库中

视图函数编写

class AlipayView(LoginRequiredMixin, AuthorRequiredMixin, View):
 """
 支付宝支付
 get方法实现支付宝return_url,如果没有实现也无所谓,post同样可以更新状态
 post方法实现支付宝notify_url,异步更新

 支付宝返回的url如下:
 #http://127.0.0.1:8000/alipay/return/?
 # charset=utf-8&
 # out_trade_no=201902923423436&
 # method=alipay.trade.page.pay.return&
 # total_amount=1.00&
 # sign=CDBMY9NBsp4KICdQoBEVxGWobd0N8y4%2BU09stzUWwlNtLr7ZpELJdM5js20wXv%2FCPp0FGPbRW1YS9DRx0CnKJULZZMqysBUMH2FL39sS0Fgstgy1ydTs7ySXdHziJV0inI%2BDWAsebQqtjk5gQEweUstc%2B%2BnzjdgAulpvWzfJsbknS%2BqUfktSdF2ZOWGhr1CFlfsMFEDS2nzQv4K3E%2BNaeylkzUnRe9M1sjIL%2FYR0wVZ5A3OfHLPf9HzC2B8%2FLu4g7N5Vctkqp2aerDvIkN5SNmDnRGyjOt2b%2BOsLMqG4X06JSsrZT6Ln8PimsrkSOIGbj0gCqscx7BwZfmCQePlCw%3D%3D&
 # trade_no=2019082622001426981000041778&
 # auth_app_id=2016092600597838&
 # version=1.0&app_id=2016092600597838&
 # sign_type=RSA2&
 # seller_id=2088102177296610&
 # timestamp=2019-08-26+13%3A51%3A01
 """
 def dispatch(self, request, *args, **kwargs):
  self.alipay = AliPay(
   appid=settings.ALIPAY_APPID,
   app_notify_url=settings.APP_NOTIFY_URL,
   app_private_key_path=settings.APP_PRIVATE_KEY_PATH,
   alipay_public_key_path=settings.ALIPAY_PUBLIC_KEY_PATH,
   debug=settings.ALIPAY_DEBUG,
   return_url=settings.RETURN_URL
  )
  #处理返回的url参数
  callback_data = {}
  for key, value in request.GET.items():
   callback_data[key] = value
  sign = callback_data.pop('sign', None)
  self.order_sn = callback_data.get('out_trade_no', None) #订单号
  self.trade_no = callback_data.get('trade_no', None) #支付宝订单号

  # 验证签名
  self.verify = self.alipay.verify(callback_data, sign)
  return super(AlipayView, self).dispatch(request, *args, **kwargs)


 def get(self, request):
  """处理支付宝return_url返回"""

  if self.verify:
   self.deposit()
   #返回个人中心页面
   return redirect(reverse('users:detail', kwargs={
    'username': request.user.username
   }))

 def post(self, request):
  """
  处理notify_url
  """
  if self.verify:
   self.deposit()

  return Response('success')

 def deposit(self):
  """充值操作

  1.更新用户的金币信息
  2.更新订单状态为交易成功
  """

  # 数据库中查询订单记录
  order = OrderInfo.objects.get(order_sn=self.order_sn)
  order.trade_no = self.trade_no # 支付宝订单号

  # 把人民币转换成对应的金币
  rmb = order.order_mount
  money = convert_rmb_to_money(rmb)

  # 更新用户的金币
  order.user.money += Decimal(money)
  order.user.save()
  # 订单状态置为交易成功
  order.pay_status = 'TRADE_SUCCESS'
  order.save()

当用户付款成功后,会跳转回来,那个就是return_url,处理这个的是get方法,用来更新用户的金币和订单。

以上这篇Django对接支付宝实现支付宝充值金币功能示例就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python 获取网页编码方式实现代码
Mar 11 Python
在Django的View中使用asyncio的方法
Jul 12 Python
python中下标和切片的使用方法解析
Aug 27 Python
Python GUI编程学习笔记之tkinter事件绑定操作详解
Mar 30 Python
Django模板之基本的 for 循环 和 List内容的显示方式
Mar 31 Python
python模拟实现分发扑克牌
Apr 22 Python
Python用类实现扑克牌发牌的示例代码
Jun 01 Python
使用Keras实现Tensor的相乘和相加代码
Jun 18 Python
python suds访问webservice服务实现
Jun 26 Python
python实现双人五子棋(终端版)
Dec 30 Python
python爬虫之爬取笔趣阁小说
Apr 22 Python
Python基于Opencv识别两张相似图片
Apr 25 Python
Django后端发送小程序微信模板消息示例(服务通知)
Dec 17 #Python
Django项目使用ckeditor详解(不使用admin)
Dec 17 #Python
python主线程与子线程的结束顺序实例解析
Dec 17 #Python
Django通用类视图实现忘记密码重置密码功能示例
Dec 17 #Python
Django集成celery发送异步邮件实例
Dec 17 #Python
python学生信息管理系统实现代码
Dec 17 #Python
pymysql模块的操作实例
Dec 17 #Python
You might like
世界第一个无线广播电台 KDKA
2021/03/01 无线电
上海永华YH-R296(华普R-96)12波段立体声收音机的分析和打理
2021/03/02 无线电
基于mysql的bbs设计(三)
2006/10/09 PHP
基于mysql的论坛(2)
2006/10/09 PHP
PHP中去除换行解决办法小结(PHP_EOL)
2011/11/27 PHP
php导出csv格式数据并将数字转换成文本的思路以及代码分享
2014/06/05 PHP
php中final关键字用法分析
2016/12/07 PHP
php使用parse_str实现查询字符串解析到变量中的方法
2017/02/17 PHP
用Javascript读取中文COOKIE的解决办法
2007/02/15 Javascript
jQuery 学习 几种常用方法
2009/06/11 Javascript
jquery实现带缩略图的全屏图片画廊效果实例
2015/06/25 Javascript
JavaScript资源预加载组件和滑屏组件的使用推荐
2016/03/10 Javascript
Bootstrap实现水平排列的表单
2016/07/04 Javascript
COM组件中调用JavaScript函数详解及实例
2017/02/23 Javascript
详解key在Vue列表渲染时究竟起到了什么作用
2019/04/20 Javascript
layui扩展上传组件模拟进度条的方法
2019/09/23 Javascript
vue实现购物车案例
2020/05/30 Javascript
js 获取扫码枪输入数据的方法
2020/06/10 Javascript
[06:45]DOTA2-DPC中国联赛 正赛 Magma vs LBZS 选手采访
2021/03/11 DOTA
在Python中操作字典之fromkeys()方法的使用
2015/05/21 Python
目前最全的python的就业方向
2018/06/05 Python
解决python中无法自动补全代码的问题
2018/12/04 Python
Python如何把多个PDF文件合并代码实例
2020/02/13 Python
python实现引用其他路径包里面的模块
2020/03/09 Python
Python基于pandas爬取网页表格数据
2020/05/11 Python
利用OpenCV中对图像数据进行64F和8U转换的方式
2020/06/03 Python
Django vue前后端分离整合过程解析
2020/11/20 Python
北京泡泡网网络有限公司.net面试题
2012/07/17 面试题
房地产出纳岗位职责
2013/12/01 职场文书
大学生简历的个人自我评价
2013/12/04 职场文书
颁奖晚会主持词
2014/03/25 职场文书
2014班子成员自我剖析材料思想汇报
2014/10/01 职场文书
个人整改措施书面材料
2014/10/24 职场文书
驾驶员管理制度范本
2015/08/06 职场文书
JVM钩子函数的使用场景详解
2021/08/23 Java/Android
Python使用pandas导入xlsx格式的excel文件内容操作代码
2022/12/24 Python