目录
2-3-4 扫描二维码发送回调请求,向官方发送请求 - 获取用户的微信唯一ID并保存数据库
一、微信公众号分类
1-1 公众号 - 不能主动发送信息
- - 认证的公众号:付费,需要营业执照,可以发布多篇文章
- - 未认证的公众号:一天只能发布一篇文章
1-2 服务号 - 用于微信推送
- 需要申请并认证
- 可以主动给用户推送消息
- 必须关注服务号才能给用户推送消息
- 允许沙箱环境的开发测试
1-3 企业号 - 企业内部使用
二、微信推送Demo 官方API文档
2-1 测试微信公众号关注用户列表
2-2 用户表结构设计
import hashlib from django.db import models class UserInfo(models.Model): username = models.CharField("用户名", max_length=64, unique=True) password = models.CharField("密码", max_length=64) # save方法md5生产 uid = models.CharField(verbose_name='个人唯一ID', max_length=64, unique=True) # 微信官方的唯一id wx_id = models.CharField(verbose_name="微信ID", max_length=128, blank=True, null=True, db_index=True) # 重写save方法,创建唯一id - 每创建用户时,为用户自动生成个人唯一ID def save(self, *args, **kwargs): if not self.pk: m = hashlib.md5() m.update(self.username.encode(encoding="utf-8")) self.uid = m.hexdigest() super(UserInfo, self).save(*args, **kwargs)
2-3 实现思路
2-3-1 用户登陆信息
def login(request): """ 用户登录 :param request: :return: """ # models.UserInfo.objects.create(username='luffy',password=123) if request.method == "POST": user = request.POST.get('user') pwd = request.POST.get('pwd') obj = models.UserInfo.objects.filter(username=user, password=pwd).first() # 用户信息存在,将用户信息保存到session中 if obj: request.session['user_info'] = {'id': obj.id, 'name': obj.username, 'uid': obj.uid} return redirect('/bind/') else: return render(request, 'login.html')
2-3-2 登陆成功跳转页
总结:
- 将之前官网上获取的公众号信息图片,另存为保存在项目目录中
- 使用jquery的qrcode插件生成二维码
- 点击按钮获取二维码,并且向后台发送了获取二维码的请求
{% load staticfiles %} <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <div style="width: 600px;margin: 0 auto"> <h1>请关注XXX服务号,并绑定个人用户(用于以后的消息提醒)</h1> <div> <h3>第一步:关注XXX微信服务号</h3> <img style="height: 100px;width: 100px" src="{% static "img/MyWX.jpg" %}"> </div> <input type="button" value="下一步【获取绑定二维码】" onclick="getBindUserQcode()"> <div> <h3>第二步:绑定个人账户</h3> <div id="qrcode" style="width: 250px;height: 250px;background-color: white;margin: 100px auto;"></div> </div> </div> <script src="{% static "js/jquery.min.js" %}"></script> {#qrcode 可以生成二维码 #} <script src="{% static "js/jquery.qrcode.min.js" %}"></script> <script src="{% static "js/qrcode.js" %}"></script> <script> function getBindUserQcode() { $.ajax({ url: '/bind_qcode/', type: 'GET', success: function (result) { console.log(result); //result.data 取出后台生成的一个地址,即二维码请求连接 //通过js生成一个二维码图片放到div中 $('#qrcode').empty().qrcode({text: result.data}); } }); } </script> </body> </html>
2-3-3 获取二维码响应视图函数,发送回调函数请求
# 登陆判断组件 @auth def bind_qcode(request): """ 生成二维码 :param request: :return: """ ret = {'code': 1000} try: access_url = "https://open.weixin.qq.com/connect/oauth2/authorize?appid={appid}&redirect_uri={redirect_uri}&response_type=code&scope=snsapi_userinfo&state={state}#wechat_redirect" access_url = access_url.format( # 商户的appid appid=settings.WECHAT_CONFIG["app_id"], # 'wx6edde7a6a97e4fcd', # 回调地址 redirect_uri=settings.WECHAT_CONFIG["redirect_uri"], # 当前登录用户的唯一id state=request.session['user_info']['uid'] # 为当前用户生成MD5值 ) ret['data'] = access_url except Exception as e: ret['code'] = 1001 ret['msg'] = str(e) return JsonResponse(ret)
# ############# 微信setting配置 ############## WECHAT_CONFIG = { 'app_id': 'wx3e1f0883236623f9', 'appsecret': '508ec4590702c76e6863be6df01ad95a', 'redirect_uri': 'http://42.56.89.12/callback/', }
2-3-4 扫描二维码发送回调请求,向官方发送请求 - 获取用户的微信唯一ID并保存数据库
def callback(request): """ 用户在手机微信上扫码后,微信自动调用该方法。 用于获取扫码用户的唯一ID,以后用于给他推送消息。 :param request: :return: """ # access_url = "https://open.weixin.qq.com/connect/oauth2/" \ # "authorize?appid={appid}&redirect_uri={redirect_uri}" \ # "&response_type=code&scope=snsapi_userinfo&state={state}#wechat_redirect" code = request.GET.get("code") # 用户md5值,用户唯一id state = request.GET.get("state") # 发送请求给官方,获取该用户openId(用户唯一,用于给用户发送消息) # request模块朝https://api.weixin.qq.com/sns/oauth2/access_token地址发get请求 res = requests.get( url="https://api.weixin.qq.com/sns/oauth2/access_token", params={ "appid": 'wx3e1f0883236623f9', "secret": '508ec4590702c76e6863be6df01ad95a', "code": code, # 固定写法 "grant_type": 'authorization_code', } ).json() # res.data-是json格式;res是一个字典 # res=json.loads(res.data) # 获取的到openid表示用户授权成功,openid即用户微信的id openid = res.get("openid") # 若获取openid,则存入对应用户数据库 if openid: models.UserInfo.objects.filter(uid=state).update(wx_id=openid) response = "<h1>授权成功 %s </h1>" % openid else: response = "<h1>用户扫码之后,手机上的提示</h1>" return HttpResponse(response)
2-3-5 给指定微信ID用户推送信息
def sendmsg(request): # 获取token认证 def get_access_token(): """ 获取微信全局接口的凭证(默认有效期俩个小时) 如果不每天请求次数过多, 通过设置缓存即可 """ result = requests.get( # 发送获取token请求 url="https://api.weixin.qq.com/cgi-bin/token", params={ "grant_type": "client_credential", "appid": settings.WECHAT_CONFIG['app_id'], "secret": settings.WECHAT_CONFIG['appsecret'], } ).json() if result.get("access_token"): access_token = result.get('access_token') else: access_token = None return access_token access_token = get_access_token() openid = models.UserInfo.objects.get(id=1).wx_id # 用户发送消息方法 def send_custom_msg(): body = { "touser": openid, # 发送文本类型消息 "msgtype": "text", "text": { "content": '你好!!!' } } response = requests.post( # 给官方发送推送请求 url="https://api.weixin.qq.com/cgi-bin/message/custom/send", # 放到路径?后面的东西,用于通过token认证 params={ 'access_token': access_token }, # 这是post请求body体中的内容,使用bytes类型 data=bytes(json.dumps(body, ensure_ascii=False), encoding='utf-8') ) # 这里可根据回执code进行判定是否发送成功(也可以根据code根据错误信息) # 返回字典信息格式 result = response.json() return result def send_template_msg(): """ 发送模版消息 """ res = requests.post( url="https://api.weixin.qq.com/cgi-bin/message/template/send", params={ 'access_token': access_token }, json={ "touser": openid, "template_id": 'IaSe9s0rukUfKy4ZCbP4p7Hqbgp1L4hG6_EGobO2gMg', "data": { "first": { "value": "lqz", "color": "#173177" }, "keyword1": { "value": "大帅哥", "color": "#173177" }, } } ) result = res.json() return result result = send_custom_msg() if result.get('errcode') == 0: return HttpResponse('发送成功') return HttpResponse('发送失败')
2-3-6 设置后台回调路由
2-3-7 模板消息接口 官方模板消息接口文档
def send_template_msg(): """ 发送模版消息 """ res = requests.post( url="https://api.weixin.qq.com/cgi-bin/message/template/send", params={ 'access_token': access_token }, json={ "touser": openid, "template_id": 'IaSe9s0rukUfKy4ZCbP4p7Hqbgp1L4hG6_EGobO2gMg', "data": { "first": { "value": "hello!", "color": "#173177" }, "keyword1": { "value": "我可爱吗?", "color": "#173177" }, } } ) result = res.json() return result
三、流程测试
3-1 发送消息测试