2019独角兽企业重金招聘Python工程师标准>>>

服务器端要做得事很多,虽然逻辑不是很复杂,但是我们必须要分析清楚我们要做哪些事,请看下图:

通过这张图,我们看出,服务器端的接口一共有6个,分别处理:

  1. 手机客户端登录
  2. 首页
  3. 二维码图片流
  4. long polling维持
  5. 接收手机客户端已扫描的通知
  6. 接收手机客户端已确认登录的通知
那么一个一个解决
首先是手机客户端登录,在上一篇我们介绍的手机客户端登录我们仅仅模拟一下,因此用户只需要提交一个用户名,服务器则通过SHA1对用户名加密,将密文返回作为token。为了将来验证这个密文是否OK,我们将用户名和密文保存在redis内供将来验证使。
需要引用的包:
[javascript]  view plain copy
  1. var http = require('http'), url = require('url'), fs = require('fs'), querystring = require('querystring'),qrcode = require('qrcode'), UUID = require('uuid-js'), sha1 = require('sha1'), redis = require('redis'), redisClient = redis
  2. .createClient('10087', '192.168.111.122'), redisKey = 'QRCODE_LOGIN_TOKEN';

redis 的客户端也一并创建了,并设置了key

web服务的基础结构如下:
[javascript]  view plain copy
  1. http.createServer(function(req, res) {
  2. // parse URL
  3. var url_parts = url.parse(req.url);
  4. var path = url_parts.pathname;
  5. var uuid4 = UUID.create();
  6. var _sessionID = uuid4.toString();
  7. if (path == '/') {
  8. //...
  9. } else if (path == '/poll') {
  10. // console.log('polling');
  11. } else if (path == '/qrcodeimage') {
  12. // 二维码的请求,参数为sessionID
  13. } else if (path == '/moblogin') {
  14. // 返回用户名对应的token,简单采用sha1加密
  15. } else if (path == '/scanned') {
  16. console.log('scanned');
  17. } else if (path == '/confirmed') {
  18. console.log('confirmed');
  19. } else {
  20. res.writeHead(200, {
  21. 'Content-Type' : 'text/html; charset=UTF-8'
  22. });
  23. res.end();
  24. }
  25. }).listen(9999, '192.168.111.109');
  26. console.log('服务器已运行在端口9999.');

通过分析,我们无非就是为这6个分支添加逻辑。

这次案例是一个试验,因此我们代码编写的也比较简单,如果使用类似express等框架的话,会更加方便一些。
先看看第一个接口,登录,返回sha1的token
[javascript]  view plain copy
  1. if (path == '/moblogin') {
  2. <span style="white-space:pre">      </span>// 返回用户名对应的token,简单采用sha1加密
  3. var userName = urlDecode(url_parts.query);
  4. var token = sha1(userName);
  5. // userHash.set(token, userName);
  6. // 保存token到redis
  7. redisClient.hset(redisKey, token, userName);
  8. res.writeHead(200, {
  9. 'Content-Type' : 'text/html; charset=UTF-8'
  10. });
  11. res.end(token);
  12. <span style="white-space:pre">  </span>}

下面是首页,如果用户敲击的url是一个不带参数的地址,事实上,用户初次访问肯定不带任何参数,而我们这个页面的目的是必须要有sessionID,因为首页内包含的2个子请求是必须具备sessionID参数的。因此我们要做url做一个分析和强制跳转:

[javascript]  view plain copy
  1. if (path == '/') {
  2. var sessionID = url_parts.query;
  3. if (typeof (sessionID) == "undefined" || sessionID == "") {
  4. // 访问首页没有参数,自动跳转
  5. res.writeHead(200, {
  6. 'Refresh' : '0; url=/?' + _sessionID,
  7. 'Content-Type' : 'text/html; charset=UTF-8'
  8. });
  9. res.end();
  10. } else {
  11. // 处理首页,刷新一条sessionID和二维码
  12. generateIndex(sessionID, req, res);
  13. }
  14. }

也就是说当直接访问/的时候,服务器强制将请求重定向并包含sessionID信息

[javascript]  view plain copy
  1. function generateIndex(sessionID, req, res) {
  2. fs.readFile('./index.html', 'UTF-8', function(err, data) {
  3. data = data.replace(/SESSIONID/g, sessionID);
  4. res.writeHead(200, {
  5. 'Content-Type' : 'text/html; charset=UTF-8'
  6. });
  7. res.end(data);
  8. });
  9. }

当访问的地址符合/?sessionID的时候,服务器读取一个html页面,并将其中的二维码和long polling需要的参数替换为sessionID

[html]  view plain copy
  1. <html>
  2. <head>
  3. <script src="http://code.jquery.com/jquery-1.6.4.min.js"></script>
  4. <script>
  5. var poll = function() {
  6. $.getJSON('/poll?SESSIONID', function(response) {
  7. var cmd = response.cmd;
  8. if (cmd == 'scanned') {
  9. scanned();
  10. } else if (cmd == 'pclogin') {
  11. var username=response.username;
  12. pclogin(username);
  13. }
  14. poll();
  15. });
  16. }
  17. var pclogin = function(username) {
  18. $('#output').text('欢迎您:' + username + ',您已成功登录');
  19. }
  20. var scanned = function() {
  21. $('#output').text('已成功扫描,等待手机确认登录');
  22. }
  23. poll();
  24. </script>
  25. </head>
  26. <body>
  27. <p align="center"><img src="/qrcodeimage?SESSIONID">
  28. </p>
  29. </body>
  30. </html>

二维码的请求在我们第二篇就已经介绍过,这里不再重复。

那么维持long polling的接口
[javascript]  view plain copy
  1. if (path == '/poll') {
  2. // console.log('polling');
  3. var sessionID = url_parts.query;
  4. var sessionObj = {
  5. 'sessionID' : sessionID,
  6. 'res' : res
  7. };
  8. clients.push(sessionObj);
  9. console.log('client added' + sessionObj);
  10. }

在处理接收客户端完成扫描和确认登录的时候,逻辑比较类似,都是先验证用户的token是否存在,商用的话可能还要有些更安全的考虑

然后根据sessionID找到维持long polling的客户端对象,并且返回相关的操作指令
[javascript]  view plain copy
  1. function handleScanned(res, token, sessionID) {
  2. // console.log(">>>" + token + "," + sessionID);
  3. var success = false;
  4. if (typeof (token) != "undefined") {
  5. // 验证是否包含用户信息已确认是登录的用户
  6. var userName;
  7. redisClient.hget(redisKey, token, function(err, reply) {
  8. userName = reply;
  9. // console.log("username=" + userName);
  10. if (typeof (userName) != "undefined") {
  11. // 用户存在
  12. for ( var int = 0; int < clients.length; int++) {
  13. var clientobj = clients[int];
  14. var savedSession = clientobj.sessionID;
  15. var client = clientobj.res;
  16. if (savedSession == sessionID) {
  17. // 页面存在
  18. client.end(JSON.stringify({
  19. cmd : 'scanned'
  20. }));
  21. clients.splice(int, 1);
  22. success = true;
  23. break;
  24. }
  25. }
  26. }
  27. res.writeHead(200, {
  28. 'Content-Type' : 'text/html; charset=UTF-8'
  29. });
  30. if (success) {
  31. res.end("scanned");
  32. } else {
  33. res.end("error");
  34. }
  35. });
  36. }
  37. }

至此,我们的完整的二维码扫描登录的流程就已经走完了。

放在服务器上运行一下,完全OK,如果想作为daemon的话可以使用forever包。
经过这几篇的介绍,我们不难发现其实这个效果的实现并不是很复杂,关键在于你要把整个逻辑理顺和想清楚。
同时由于这个案例涉及的技术也较多,技术不全面的话也很难形成完成的解决方案。
思考:
这个案例中还存在哪些问题
  1. 微信27秒是事出有因的,考虑到http请求有可能在客户端因为长时间无响应而被终止,因此27秒自动刷新long polling可以有效的防治连接断掉,而在我们这个案例里,并没有去实现这个功能。首先我觉得实现起来没有问题,不难,另外,这些点应该由你们自己去实现,我更加关注的是分析业务。
  2. 关于页面session的内涵,应该可以附加一些加密的信息,对于客户端只是传递这些信息,因此不涉及解密操作,而服务器端就可以验证sessionID的合法性,目前如果你访问/?的时候自己宿便敲sessionID也是可以的,服务器没有做任何验证和限制。
  3. 关于long polling客户端的response对象的维持和清理,在本例中我们直接采用了js的数组进行存储,因此每次都是遍历。如果商用,必然用采用哈希的方式来存储,同时可能还必须存储在数据库内。
  4. 本例只是在客户端确认登录之后在页面上显示确认登录,并没有跳转到某页面,但是实际应用的时候,可能会携带某个服务器生成的钥匙去redirect到某个url,只有目标地址确认这个钥匙是登录确认信息之后才会以某用户方式登录,这个还是希望大家能实现,逻辑很简单,只是本例略掉。
原文连接:http://blog.csdn.net/otangba/article/details/8273952

转载于:https://my.oschina.net/jeffzhao/blog/107934

微信扫描二维码登入实现,网页端相关推荐

  1. 【转】扫描二维码登入安全吗?

    转载自 https://abcdabcd987.com/qrcode-login/ 昨天在知乎上看到了一个问题微信淘宝设计扫码登录的理由是什么,牺牲人性化来加强安全性?,本以为这是一个送分题,可是点开 ...

  2. 扫描二维码登入PC的工作原理

    不知道从哪里看的了,有侵权找我删,我只是想记录一下

  3. Day212.OAuth2、微信二维码登入注册功能、用户登录信息前后端供、讲师列表前后端 -谷粒学院

    谷粒学院 OAuth2的使用场景 一.OAuth2解决什么问题 1.OAuth2提出的背景 照片拥有者想要在云冲印服务上打印照片,云冲印服务需要访问云存储服务上的资源 2.图例 资源拥有者:照片拥有者 ...

  4. 微信扫描二维码快速登录网站

    在近期的一个项目中用到了微信扫描注册.登录网站功能所以整理了下希望对读者有帮助. 首先,你需要有一个没有绑定微信.微信公众平台的邮箱注册成为微信开放平台开发者,在管理中心创建移动应用.或者网站应用获得 ...

  5. 微信扫描二维码和浏览器扫描二维码 ios和Android 分别进入不用的提示页面

    实现微信扫描二维码和浏览器扫描二维码 ios和Android 分别进入不用的提示页面 而进入商城下载该项目 详情地址:gitee.com/DuJiaHui123- 1.创建完之后 替换文件里面的ios ...

  6. 微信扫描二维码在内置浏览器打不开文件的下载链接怎么办?哪些api接口可以解决...

    有哪些api接口可以实现微信扫描二维码在内置浏览器打开文件的下载链接? 经常看到贴吧上有人吐槽微信的检测系统太严格了,动不动就拦截第三方链接.怎么才能解决,怎么才能避免等等一系列的问题.因为平时我也会 ...

  7. 微信扫描二维码-电脑上网

    展视网北京科技有限公司--cuidc 由于 wifi 成为人们生活中不可或缺的一部分,店家 wifi 免费上网的招数也就日益增多. 今天我们介绍一种破解<微信扫描二维码上网>的招数,此招数 ...

  8. 使用电脑微信扫描二维码

    使用电脑微信扫描二维码 将二维码图片发送至文件传输助手 在与文件传输助手的聊天界面点开图片,右键,点击识别图中二维码即可 扫描完成.

  9. 微信扫描二维码跳转至浏览器打开 jsp

    微信扫描二维码总是用默认浏览器打开,无法下载任何东西.怎么办呢? 微信识别到打开为pdf 时,会自动给跳转至浏览器,所以解决方法就是服务器判断请求端为微信时,返回头部添加 Content-type:a ...

最新文章

  1. handlebars.js {{#if}}中的逻辑运算符是有条件的
  2. Spring Boot怎么样处理静态资源(静态资源映射规则)_Web开发
  3. cmd写java程序_用cmd写一个最简单的Java程序
  4. Windows Socket五种I/O模型详细介绍(精)
  5. 22行满分代码:L1-054 福到了 (15分)
  6. html清除矩形边区域,canvas清除矩形指定颜色
  7. [Ruby on Rails]Rails 3使用ActionMailer通过163发送邮件
  8. C++变量初始化问题
  9. Tushare实战分析美国国债收益率与利率的关系
  10. 我们通常所说的利率是指_我们通常所说的利率是指()。 A.市场利率B.名义利率C.实际利率D.固定利率...
  11. 胡巴动态表情包 捉妖记胡巴QQ表情无水印下载
  12. IT通过什么途径去美国工作?
  13. CMD 常用命令总结
  14. 骨传导也有动铁单元般的音质,绝对品质,真的是诚意满满
  15. DDNS动态域名解析IPv6地址
  16. ajax仿百度搜索效果,利用autocomplete.js实现仿百度搜索效果(ajax动态获取后端[C#]数据)...
  17. fullcalendar应用(一)
  18. Vue传递对象数据,后台解析并使用
  19. sp_WhoIsActive
  20. C# 添加、修改、删除PDF书签

热门文章

  1. 基于tcp和udp的socket实现
  2. 编写运行最简单的java程序——使用记事本编写java程序
  3. 《Python数据可视化编程实战》——5.5 用OpenGL制作动画
  4. UIPickerView 修改必须滚动才修改值的bug
  5. 保护SNMP协议服务安全的三个步骤
  6. Visual studio 2005如何实现源码管理
  7. s3 aws_您需要了解的有关AWS S3的所有信息
  8. leetcode dfs_深度优先搜索:具有6个Leetcode示例的DFS图遍历指南
  9. Android开发中应避免的重大错误
  10. Python 用户的三次登录机会