扫微信小程序码实现网站登陆实现解析


Posted in Javascript onAugust 20, 2019

使用扫小程序码登陆网站

网络上关于实现本本功能的文章很多,但是给出案列的几乎没有,今天笔者实现用小程序码实现网站登陆,体验地址如下

https://idea.techidea8.com/open/idea.shtml?id=5

思路

核心流程

扫微信小程序码实现网站登陆实现解析

关键流程

建立场景sceneid和websocket的绑定关系

获得sceneid

场景ID可以前端生成,也可以后端生成,只需要保证sceneid的同一时间唯一性即可。

前端生成可以采用随机数加时间戳的形式,也可以用uuid算法

//时间戳
var sceneid ="scend-" + new Data().getTime() + Math.ceil(Math.random()*888888+1000000);

建立websocket

var ws = new WebSocket("ws://192.168.0.106/websocket?clientid="+sceneid )
ws.onopen=function(env){
  console.log(env)
} 
ws.onmessage=function(env){
  var data = env.data;
  //这个data 就是后端发来的用户数据
}

后端建立websocket

后端采用go语言github.com/gorilla/websocket包建立websocket.因为golang 非常适合高并发场景。

func (ctrl *PushCtrl) websocket(w http.ResponseWriter, req *http.Request) {
  //fmt.Printf("%+v",request.Header)
  //todo 检验接入是否合法
  //checkToken(userId int64,token string)
  query := req.URL.Query()
  clientid := query.Get("clientid")
  conn, err := (&websocket.Upgrader{
    CheckOrigin: func(r *http.Request) bool {
      return true
    },
  }).Upgrade(w, req, nil)

  if err != nil {
    log.Println(err.Error())
    return
  }
  clientMap.Store(clientid, conn)
  go func(clientId string, conn *websocket.Conn) {
    //处理出错信息
    defer func() {
      conn.Close()
      clientMap.Delete(clientid)
    }()
    for {
      _, _, err := conn.ReadMessage()
      if err != nil {
        log.Println(err.Error())
        return
      }
    }
  }(clientid, conn)
}

我们采用sync.map建立sceneid和websocket的对应关系

clientMap.Save(sceneid,conn)

对于因为异常断开的con我们需要移除

clientMap.Delete(sceneid)

获得小程序二维码

获得小程序的accesstoken

小程序access 请求接口如下

``

该接口日使用频率有限制,因此我们需要将这些数据缓存起来,缓存方案很多,有redis,也有内存,我们这里直接用一个变量即可存储

accesstoken :=""
func GetAccessToken() string{
  return accesstoken 
}
func RefreshAccessToken()string{
  url = ""
  resp := httpget(url)
  //resp 是一个包含accesstoken的json字符串,我们解析这个json即可
  accesstoken = decodeaccesstokenfromjson(resp)
  return accesstoken 
}

我们还需要一个滴答计数器用来刷新accesstoken,accesstoken 的有效期是7200秒,我们4000秒刷新一次accesstoken 即可

func refreshAccessToken(){
  ticker := time.NewTicker(time.Second *4000)
  for{
    select {
      case <-ticker.C:
        RefreshAccessToken()
    }
  }
}

在init.go 中的init方法中启动协程

go refreshAccessToken()

小程序二维码编程技巧

小程序二维码请求接口如下

https://api.weixin.qq.com/wxa/getwxacodeunlimit?access_token=ACCESSTOKEN

由于该链接会返回俩种结果

错误返回Json

{
"errcode":400001,
"errmsg":"什么什么原因"
}

正确返回j图片buffer

这是一个二进制码流,因此我们需要对返回结果进行标准化处理

我们建议一种标准化结果

{
code:0,
data:"",
msg:"结果说明"
}

|参数|说明|

|-|-|-|

|code|标识成功或者失败,200为成功,400为失败|

|data|图片的base64格式编码|

|msg|结果说明,或者出错提示|

代码示例如下

url := fmt.Sprintf("https://api.weixin.qq.com/wxa/getwxacodeunlimit?access_token=%s", token)
ret, err := util.PostJSON(url, arg)
if err != nil {
    util.FailMsg(w, err.Error())
    return
}
jsonstr := string(ret)
if strings.Contains(jsonstr, "errmsg") {
  util.FailMsg(w, jsonstr)
  return
} else {
  base64data := base64.StdEncoding.EncodeToString(ret)
  util.RespOk(w, "data:image/png;base64,"+base64data)
}

其中util是笔者封装的常用工具包

常用工具包

扫微信小程序码实现网站登陆实现解析

前端请求成功后获得data,可以利用$("#qrcode").attr("src",res.data)刷新图片二维码

function refreshqrcode(){
  clientId = "scene-"+new Date().getTime();
  var api = restgo.buildapi("miniapp/getwxacodeunlimit")
  restgo.post(api,{"scene":clientId}).then(res=>{
    if(res.code==200){
      $("#qrcode").attr("src",res.data)
    }else{
      alert(res.msg)  
    }
    intiwebsocket()
  },res=>{
    alert(res.msg)
  })
}

小程序处理关键点

采用好的框架

我们编程过程才用了uniapp,这个框架不错,完全是vue的语法,

一套代码可以生成H5/小程序/android/ios

我们采用了vue的watch特性,通过监听userid来确定是否发送信息

watch:{
   userid:function(a,b){      
      if(a==0){
        return 
      }      
      this.loaddata()
      //如果clientid是空的说明不是扫码进来的  
      if(!this.clientid){
        return 
      }
//如果是扫码进来的,那么我们还需要推送消息到服务器后端
server.PublishMsg(this.clientid,this.userid,this.role,this.avatarUrl,this.nickName).then(res=>{
        tip.error(res.msg)
      },res=>{
        tip.error(res.msg)
      })      
   }
 }

获取sceneid

我们通过onload方法获得sceneid

onLoad(arg) {
    if(!!arg && !!arg.scene){
          this.clientid = decodeURIComponent(arg.scene)
    }        
  },

核心代码如下

通过gotUserInfo获得用户头像、昵称等,通过uni.login获得code

gotUserInfo :function(e) {
       //获得用户头像、昵称、
       uni.login(
       {
         success:(res) =>{
              //通过wx.login获得code
              userInfo.code = res.code
                        //统一传递到后端   
            this.authwithcode(userInfo)
         }
       }
       )
      }
     },
    authwithcode:function(userInfo){
            //code获得openid在后端做,做了后继续去查User表
      server.AuthWithCode(userInfo).then(res=>{
          //如果这个openid绑定了用户                                  
          if(res.data.id>0){
            //这里res.data 就是user对象,可以继续操作
            return ;
          }
          //如果没有那么注册
          server.RegisterWithOpenId(res.data.mini_openid,userInfo.avatarUrl,userInfo.nickName)#.then(res=>{
            //注册成功则返回
            //这里res.data 就是user对象可以继续操作            
          },res=>{
            tip.error(res.msg)
          })
        },res=>{
          tip.error(res.msg||"");
        })
      }
    },

体验地址

本文所有应用体验地址如下

互联网行业解决方案吧https://idea.techidea8.com/open/idea.shtml?id=5

关于代码

代码获得地址

小程序码登陆

代码配置

请认真阅读readme.md

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
JavaScipt基本教程之前言
Jan 16 Javascript
Javascript条件判断使用小技巧总结
Sep 08 Javascript
ie focus bug 解决方法
Sep 03 Javascript
js实现window.open不被拦截的解决方法汇总
Oct 30 Javascript
javascript实现tab切换特效
Nov 12 Javascript
实例解析jQuery中proxy()函数的用法
May 24 Javascript
JS实现的系统调色板完整实例
Dec 21 Javascript
JavaScript中关于class的调用方法
Nov 28 Javascript
Javascript原生ajax请求代码实例
Feb 20 Javascript
通过实例解析chrome如何在mac环境中安装vue-devtools插件
Jul 10 Javascript
node.js使用express-fileupload中间件实现文件上传
Jul 16 Javascript
JavaScript设计模式之原型模式详情
Jun 21 Javascript
vue+element-ui+axios实现图片上传
Aug 20 #Javascript
vue element upload实现图片本地预览
Aug 20 #Javascript
JS中的算法与数据结构之集合(Set)实例详解
Aug 20 #Javascript
Vue + Element UI图片上传控件使用详解
Aug 20 #Javascript
微信小程序项目总结之记账小程序功能的实现(包括后端)
Aug 20 #Javascript
ES6中Symbol、Set和Map用法详解
Aug 20 #Javascript
Vue+Element UI+vue-quill-editor富文本编辑器及插入图片自定义
Aug 20 #Javascript
You might like
PHP is_dir() 判断给定文件名是否是一个目录
2010/05/10 PHP
优化PHP程序的方法小结
2012/02/23 PHP
Yii实现MySQL多数据库和读写分离实例分析
2014/12/03 PHP
php实现的简单检验登陆类
2015/06/18 PHP
一个级联菜单代码学习及removeClass与addClass的应用
2013/01/24 Javascript
javascript 三种方法实现获得和设置以及移除元素属性
2013/03/20 Javascript
javascript实现的左右无缝滚动效果
2016/09/19 Javascript
Bootstrap下拉菜单更改为悬停(hover)触发的方法
2017/05/24 Javascript
jquery中有哪些api jQuery主要API
2017/11/20 jQuery
Nodejs中crypto模块的安全知识讲解
2018/01/03 NodeJs
JS实现的input选择图片本地预览功能示例
2018/08/29 Javascript
jquery UI实现autocomplete在获取焦点时得到显示列表功能示例
2019/06/04 jQuery
在Vue项目中使用Typescript的实现
2019/12/19 Javascript
JavaScript中变量提升机制示例详解
2019/12/27 Javascript
Python中文编码那些事
2014/06/25 Python
Python 进程之间共享数据(全局变量)的方法
2019/07/16 Python
Django+zTree构建组织架构树的方法
2019/08/21 Python
python实现静态服务器
2019/09/05 Python
将python2.7添加进64位系统的注册表方式
2019/11/20 Python
Python如何使用字符打印照片
2020/01/03 Python
使用tensorflow根据输入更改tensor shape
2020/06/23 Python
Python wordcloud库安装方法总结
2020/12/31 Python
纯CSS3绘制打火机动画火焰效果
2016/07/18 HTML / CSS
世界著名的顶级牛排:Omaha Steak(奥马哈牛排)
2016/09/20 全球购物
SKECHERS斯凯奇中国官网:来自美国的运动休闲品牌
2018/11/14 全球购物
Richards网上商店:当代时尚,遍布巴西
2019/11/03 全球购物
函数只定义了一次, 调用了一次, 但编译器提示非法重定义了-什么问题?
2014/10/03 面试题
诉讼授权委托书
2014/10/15 职场文书
幼儿园中班教师个人工作总结
2015/02/06 职场文书
2015年敬老月活动总结
2015/03/27 职场文书
2015年医院工作总结范文
2015/04/09 职场文书
导游词之丽江普济寺
2019/10/22 职场文书
python之json文件转xml文件案例讲解
2021/08/07 Python
PyTorch中permute的使用方法
2022/04/26 Python
python三子棋游戏
2022/05/04 Python
JS前端轻量fabric.js系列物体基类
2022/08/05 Javascript