Python开发微信公众平台的方法详解【基于weixin-knife】


Posted in Python onJuly 08, 2017

本文实例讲述了Python开发微信公众平台的方法。分享给大家供大家参考,具体如下:

这两天将之前基于微信公众平台的代码重构了下,基础功能以库的方式提供,提供了demo使用的是django,看着之前为赶进度写的代码真的惨不忍睹,所以weixin-knife产生了,正如其名,提供的是必要的功能,而不是完整的应用。weixin-knife可以很方便的处理关注,取关注事件,处理文本消息,回复用户信息,jssdk处理,oauth认证,以及微信支付。

github地址:https://github.com/Skycrab/weixin-knife。

首先看看怎么用

from .weixin import handler as HD
@HD.subscribe
def subscribe(xml):
  return "welcome to brain"
@HD.unsubscribe
def subscribe(xml):
  print "leave"
  return "leave brain"

上面处理了关注和取关事件,通过装饰器处理的还算透明。

处理文本消息,回复图文消息如下:

@HD.text
def text(xml):
  content = xml.Content
  if content == "111":
    return {"Title":"美女", "Description":"比基尼美女", "PicUrl":"http://9smv.com/static/mm/uploads/150411/2-150411115450247.jpg", "Url":"http://9smv.com/beauty/list?category=5"}
  elif content == "222":
    return [
      ["比基尼美女", "比基尼美女", "http://9smv.com/static/mm/uploads/150411/2-150411115450247.jpg", "http://9smv.com/beauty/list?category=5"],
      ["长腿美女", "长腿美女", "http://9smv.com/static/mm/uploads/150506/2-150506111A9648.jpg", "http://9smv.com/beauty/list?category=8"]
    ]
  elif content == "push":
    Helper.send_text_message(xml.FromUserName, "推送消息测试")
    return "push ok"
  return "hello world"

如何文本是111或222,我们回复图文消息,如何使push,我们使用客服接口推送消息,其它返回“hello world"

一般我们会使用oauth网页授权获取用户的openid,如果是多个链接都需要通过oauth处理,代码会很难看,通过装饰器可以很好的处理这个问题。

def sns_userinfo_callback(callback=None):
  """网页授权获取用户信息装饰器
  callback(openid, userinfo):
    return user
  """
  def wrap(func):
    @wraps(func)
    def inner(*args, **kwargs):
      request = args[0] #django第一个参数request
      openid = request.COOKIES.get('openid')
      userinfo = None
      if not openid:
        code = request.GET.get("code")
        if not code:
          current = "http://"+ request.get_host() + request.get_full_path()
          return redirect(WeixinHelper.oauth2(current))
        else:
          data = json.loads(WeixinHelper.getAccessTokenByCode(code))
          access_token, openid, refresh_token = data["access_token"], data["openid"], data["refresh_token"]
          #WeixinHelper.refreshAccessToken(refresh_token)
          userinfo = json.loads(WeixinHelper.getSnsapiUserInfo(access_token, openid))
      else:
        ok, openid = Helper.check_cookie(openid)
        if not ok:
          return redirect("/")
      request.openid = openid
      if callable(callback):
        request.user = callback(openid, userinfo)
      response = func(request)
      return response
    return inner
  return wrap
sns_userinfo = sns_userinfo_callback()

在所有需要用户openid的函数前使用sns_userinfo装饰器就可以了,callback函数接收openid,userinfo,返回用户实例,这样就可以使用request.user获取当前用户

@sns_userinfo
def oauth(request):
  """网页授权获取用户信息"""
  resp = HttpResponse(request.openid)
  resp.set_cookie("openid", Helper.sign_cookie(request.openid))
  return resp

使用oauth需要保存cookie,不然每次用户请求都需要授权,需要走一遍完整的oauth流程,拖慢整体响应。

weixin-knife提供了微信支付支持,稍微修改我之前移植的官方PHP版本,https://github.com/Skycrab/wzhifuSDK

@sns_userinfo
def pay(request):
  response = render_to_response("pay.html")
  response.set_cookie("openid", Helper.sign_cookie(request.openid))
  return response
@sns_userinfo
@catch
def paydetail(request):
  """获取支付信息"""
  openid = request.openid
  money = request.POST.get("money") or "0.01"
  money = int(float(money)*100)
  jsApi = JsApi_pub()
  unifiedOrder = UnifiedOrder_pub()
  unifiedOrder.setParameter("openid",openid) #商品描述
  unifiedOrder.setParameter("body","充值测试") #商品描述
  timeStamp = time.time()
  out_trade_no = "{0}{1}".format(WxPayConf_pub.APPID, int(timeStamp*100))
  unifiedOrder.setParameter("out_trade_no", out_trade_no) #商户订单号
  unifiedOrder.setParameter("total_fee", str(money)) #总金额
  unifiedOrder.setParameter("notify_url", WxPayConf_pub.NOTIFY_URL) #通知地址
  unifiedOrder.setParameter("trade_type", "JSAPI") #交易类型
  unifiedOrder.setParameter("attach", "6666") #附件数据,可分辨不同商家(string(127))
  try:
    prepay_id = unifiedOrder.getPrepayId()
    jsApi.setPrepayId(prepay_id)
    jsApiParameters = jsApi.getParameters()
  except Exception as e:
    print(e)
  else:
    print jsApiParameters
    return HttpResponse(jsApiParameters)
FAIL, SUCCESS = "FAIL", "SUCCESS"
@catch
def payback(request):
  """支付回调"""
  xml = request.raw_post_data
  #使用通用通知接口
  notify = Notify_pub()
  notify.saveData(xml)
  print xml
  #验证签名,并回应微信。
  #对后台通知交互时,如果微信收到商户的应答不是成功或超时,微信认为通知失败,
  #微信会通过一定的策略(如30分钟共8次)定期重新发起通知,
  #尽可能提高通知的成功率,但微信不保证通知最终能成功
  if not notify.checkSign():
    notify.setReturnParameter("return_code", FAIL) #返回状态码
    notify.setReturnParameter("return_msg", "签名失败") #返回信息
  else:
    result = notify.getData()
    if result["return_code"] == FAIL:
      notify.setReturnParameter("return_code", FAIL)
      notify.setReturnParameter("return_msg", "通信错误")
    elif result["result_code"] == FAIL:
      notify.setReturnParameter("return_code", FAIL)
      notify.setReturnParameter("return_msg", result["err_code_des"])
    else:
      notify.setReturnParameter("return_code", SUCCESS)
      out_trade_no = result["out_trade_no"] #商户系统的订单号,与请求一致。
      ###检查订单号是否已存在,以及业务代码
  return HttpResponse(notify.returnXml())

pay.html就是使用WeixinJSBridge.invode调用

$.post("/paydetail",{
  money: $momey
  },function(data){
   if(data){
    var jsonobj = eval('('+data+')');
    WeixinJSBridge.invoke('getBrandWCPayRequest', {
       "appId" : jsonobj.appId, //公众号名称,由商户传入
       "timeStamp" : jsonobj.timeStamp, //时间戳
       "nonceStr" : jsonobj.nonceStr, //随机串
       "package" : jsonobj.package,//扩展包
       "signType" : "MD5", //微信签名方式:1.sha1
       "paySign" : jsonobj.paySign //微信签名
       });
   }
  }
 );

由于access_token, jsapi_ticket需要缓存,而缓存方式又依赖于具体环境,所以提供了一个Helper类,使用了django 的cache缓存。

class Helper(object):
  """微信具体逻辑帮组类"""
  @class_property
  def access_token(cls):
    key = "ACCESS_TOKEN"
    token = cache.get(key)
    if not token:
      data = json.loads(WeixinHelper.getAccessToken())
      token, expire = data["access_token"], data["expires_in"]
      cache.set(key, token, expire-300)
    return token
  @class_property
  def jsapi_ticket(cls):
    key = "JSAPI_TICKET"
    ticket = cache.get(key)
    if not ticket:
      data = json.loads(WeixinHelper.getJsapiTicket(cls.access_token))
      ticket, expire = data["ticket"], data["expires_in"]
      cache.set(key, ticket, expire-300)
    return ticket

class_property提供了类级别的property,当然实例也是可以用的。

class class_property(object):
  """ A property can decorator class or instance
  class Foo(object):
    @class_property
    def foo(cls):
      return 42
  print(Foo.foo)
  print(Foo().foo)
  """
  def __init__(self, func, name=None, doc=None):
    self.__name__ = name or func.__name__
    self.__module__ = func.__module__
    self.__doc__ = doc or func.__doc__
    self.func = func
  def __get__(self, obj, type=None):
    value = self.func(type)
    return value

使用weixin-knife助力公众平台开发,你完全可以稍加修改用于flask等其它web框架。

希望本文所述对大家Python程序设计有所帮助。

Python 相关文章推荐
python实现12306火车票查询器
Apr 20 Python
Python模拟鼠标点击实现方法(将通过实例自动化模拟在360浏览器中自动搜索python)
Aug 23 Python
python中的计时器timeit的使用方法
Oct 20 Python
启动targetcli时遇到错误解决办法
Oct 26 Python
python将一组数分成每3个一组的实例
Nov 14 Python
详解python深浅拷贝区别
Jun 24 Python
Pytorch 实现自定义参数层的例子
Aug 17 Python
基于Django实现日志记录报错信息
Dec 17 Python
解决django无法访问本地static文件(js,css,img)网页里js,cs都加载不了
Apr 07 Python
python PIL模块的基本使用
Sep 29 Python
python删除csv文件的行列
Apr 06 Python
基于python定位棋子位置及识别棋子颜色
Jul 26 Python
python 中random模块的常用方法总结
Jul 08 #Python
Python调用微信公众平台接口操作示例
Jul 08 #Python
HTML中使用python屏蔽一些基本功能的方法
Jul 07 #Python
CentOS 7下安装Python 3.5并与Python2.7兼容并存详解
Jul 07 #Python
浅谈Python中的可变对象和不可变对象
Jul 07 #Python
Python MySQL数据库连接池组件pymysqlpool详解
Jul 07 #Python
Python用Pillow(PIL)进行简单的图像操作方法
Jul 07 #Python
You might like
解析阿里云ubuntu12.04环境下配置Apache+PHP+PHPmyadmin+MYsql
2013/06/26 PHP
使用PHP接受文件并获得其后缀名的方法
2015/08/05 PHP
Javascript实例教程(19) 使用HoTMetal(5)
2006/12/23 Javascript
基于jquery的模态div层弹出效果
2010/08/21 Javascript
arguments对象验证函数的参数是否合法
2015/06/26 Javascript
直接拿来用的15个jQuery代码片段
2015/09/23 Javascript
JavaScript常用数组算法小结
2016/02/13 Javascript
JavaScript实现DOM对象选择器
2016/09/24 Javascript
将JSON字符串转换成Map对象的方法
2016/11/30 Javascript
jQuery扩展+xml实现表单验证功能的方法
2016/12/25 Javascript
ES6概念 ymbol.for()方法
2016/12/25 Javascript
React中上传图片到七牛的示例代码
2017/10/10 Javascript
jQuery中 DOM节点操作方法大全
2017/10/12 jQuery
用POSTMAN发送JSON格式的POST请求示例
2018/09/04 Javascript
vue实现添加与删除图书功能
2018/10/07 Javascript
JavaScript实现好看的跟随彩色气泡效果
2020/02/06 Javascript
解决vue项目router切换太慢问题
2020/07/19 Javascript
Nest.js散列与加密实例详解
2021/02/24 Javascript
[03:54]DOTA2英雄梦之声_第06期_昆卡
2014/06/23 DOTA
[03:42]2018完美盛典-《加冕》
2018/12/16 DOTA
[01:18:31]DOTA2-DPC中国联赛定级赛 LBZS vs Magma BO3第一场 1月10日
2021/03/11 DOTA
寻找网站后台地址的python脚本
2014/09/01 Python
实例讲解Python设计模式编程之工厂方法模式的使用
2016/03/02 Python
TensorFlow搭建神经网络最佳实践
2018/03/09 Python
keras的load_model实现加载含有参数的自定义模型
2020/06/22 Python
python中执行smtplib失败的处理方法
2020/07/01 Python
python selenium 获取接口数据的实现
2020/12/07 Python
HTML5本地存储之Web Storage应用介绍
2013/01/06 HTML / CSS
中国一家专注拼团的社交购物网站:拼多多
2018/06/13 全球购物
weblogic面试题
2016/03/07 面试题
大学毕业生文采飞扬的自我鉴定
2013/12/03 职场文书
消防宣传口号
2014/06/16 职场文书
网吧消防安全责任书
2014/07/29 职场文书
2014年村支部书记四风对照检查材料思想汇报
2014/10/02 职场文书
JavaScript阻止事件冒泡的方法
2021/12/06 Javascript
win11开机发生死循环重启怎么办?win11开机发生死循环重启解决方法
2022/08/05 数码科技