当前位置: 首页 > 编程日记 > 正文

用node实现websocket协议

协议

WebSocket是一种基于TCP之上的客户端与服务器全双工通讯的协议,它在HTML5中被定义,也是新一代webapp的基础规范之一。

它突破了早先的AJAX的限制,关键在于实时性,服务器可以主动推送内容 到客户端!可能的应用有:多人在线游戏,即时聊天,实时监控,远程桌面,新闻服务器等等。

对于我自己,当前最想尝试的是canvas+websocket组合起来能做什么。

实现

由于握手的过程是一个标准的HTTP请求,因此 websocket 的实现有两种选择:1)TCP上实现; 2) 现有HTTP软件上实现。后者的优势在于可以共用现有的HTTP服务器端口,并且不用重新实现认证功能和解析HTTP请求的功能。

这个示例中使用的 node 的HTTP模块。(TCP版及所有文件见 附件)

node服务器端代码:


var http = require('http');
var url = require('url');
// var mime = require('mime');
var crypto = require('crypto');var port = 4400;
var server = http.createServer();server.listen(port,function() {console.log('server is running on localhost:',port);server.on('connection',function(s) {console.log('on connection ',s);}).on('request',onrequest).on('upgrade',onupgrade);});var onrequest = function(req,res) {console.log( Object.keys(req) ,req.url,req['upgrade']);if( !req.upgrade ){// 非upgrade请求选择:中断或提供普通网页res.writeHead(200, { 'content-type': 'text/plain' });res.write( 'WebSocket server works!' );}res.end();return;
};var onupgrade = function (req,sock,head) {// console.log('方法:',Object.keys(sock));if(req.headers.upgrade !== 'WebSocket'){console.warn('非法连接');sock.end();return;}bind_sock_event(sock);try{handshake(req,sock,head);}catch(e){console.error(e);sock.end();}
};// 包装将要发送的帧
var wrap = function(data) {var fa = 0x00, fe = 0xff, data = data.toString()len = 2+Buffer.byteLength(data),buff = new Buffer(len);buff[0] = fa;buff.write(data,1);buff[len-1] = fe;return buff;
}
// 解开接收到的帧
var unwrap = function(data) {return data.slice(1,data.length-1);
}var bind_sock_event = function(sock) {sock.on('data',function(buffer) {var data = unwrap(buffer);console.log('socket receive data : ',buffer,data,'\n>>> '+data);// send('hello html5,'+Date.now())sock.emit('send',data);}).on('close',function() {console.log('socket close');}).on('end',function() {console.log('socket end');}).on('send',function(data) { //自定义事件sock.write(wrap(data),'binary');})
};var get_part = function(key) {var empty   = '',spaces  = key.replace(/\S/g,empty).length,part    = key.replace(/\D/g,empty);if(!spaces) throw {message:'Wrong key: '+key,name:'HandshakeError'}return get_big_endian(part / spaces);
}var get_big_endian = function(n) {  return String.fromCharCode.apply(null,[3,2,1,0].map(function(i) { return n >> 8*i & 0xff }))
}var challenge = function(key1,key2,head) {var sum = get_part(key1) + get_part(key2) + head.toString('binary');return crypto.createHash('md5').update(sum).digest('binary');
}var handshake = function(req,sock,head) {var output = [],h = req.headers, br = '\r\n';// headeroutput.push('HTTP/1.1 101 WebSocket Protocol Handshake','Upgrade: WebSocket','Connection: Upgrade','Sec-WebSocket-Origin: ' + h.origin,'Sec-WebSocket-Location: ws://' + h.host + req.url,'Sec-WebSocket-Protocol: my-custom-chat-protocol'+br);// bodyvar c = challenge(h['sec-websocket-key1'],h['sec-websocket-key2'],head);output.push(c);sock.write(output.join(br),'binary');
}

浏览器客户端代码:


<html>
<head><title>WebSocket Demo</title>
</head>
<style type="text/css">textarea{width:400px;height:150px;display:block;overflow-y:scroll;}#output{width:600px;height:400px;background:whiteSmoke;padding:1em .5em;color:#000;border:none;}button{padding:.2em 1em;}
</style>
<link href="layout.css" rel="stylesheet" type="text/css" /> 
<body><textarea id="output" readonly="readonly"></textarea>
<br>
<textarea id="input"></textarea>
<button id="send">send</button><script type="text/javascript">
// localhost
var socket = new WebSocket('ws://192.168.144.131:4400/')
socket.onopen = function(e) {log(e.type);socket.send('hello node');
}
socket.onclose = function(e) {log(e.type);
}
socket.onmessage = function(e) {log('receive @ '+ new Date().toLocaleTimeString() +'\n'+e.data);output.scrollTop = output.scrollHeight
}
socket.onclose = function(e) {log(e.type);
}
socket.addEventListener('close',function() {log('a another close event handler..');
},false);// dom
var id = function(id) {return document.getElementById(id);
}
var output = id('output'), input = id('input'), send = id('send');
var log = function(msg) {output.textContent += '> '+msg + '\n'
}
send.addEventListener('click',function() {socket.send(input.value);
},false);</script>
</body>
</html>

细节

在 http 协议之上的 websocket 协议实现只有两步:握手,发送数据。

握手

握手的过程被称为 challenge-response。首先客户端发起一个名为Upgrade的HTTP GET请求,服务器验证此请求,给出101响应以表示接受此次协议升级,握手即完成了。

chrome inspector美化过的握手信息:


Request URL:ws://192.168.144.131:4400/pub/chat?q=me
Request Method:GET
Status Code:101 WebSocket Protocol HandshakeRequest Headers
Connection:Upgrade
Host:192.168.144.131:4400
Origin:http://localhost:800
Sec-WebSocket-Key1:p2    G 947T 80  661 jAf2
Sec-WebSocket-Key2:z Z Q ^326 5 9= 7s1  1 7H4
Sec-WebSocket-Protocol::my-custom-chat-protocol
Upgrade:WebSocket
(Key3):7C:44:56:CA:1F:19:D2:0AResponse Headers
Connection:Upgrade
Sec-WebSocket-Location:ws://192.168.144.131:4400/pub/chat?q=me
Sec-WebSocket-Origin:http://localhost:800
Sec-WebSocket-Protocol:my-custom-chat-protocol
Upgrade:WebSocket
(Challenge Response):52:DF:2C:F4:50:C2:8E:98:14:B7:7D:09:CF:C8:33:40

请求头部分

  • Host: websocket服务器主机
  • Connection: 连接类型
  • Upgrade: 协议升级类型
  • Origin: 访问来源
  • Sec-WebSocket-Protocol: 可选,子协议名称,由应用自己定义,多个协议用空格分割。(*另外一个仅剩的可选项是cookie)
  • Sec-WebSocket-Key1: 安全认证key,xhr请求不能伪造'sec-'开头的请求头。
  • Sec-WebSocket-Key2: 同上
  • Key3: 响应体内容,8字节随机。

响应头部分

  • Sec-WebSocket-Protocol: 必须包含请求的子协议名
  • Sec-WebSocket-Origin: 必须等于请求的来源
  • Sec-WebSocket-Location: 必须等于请求的地址
  • Challenge Response: 响应体内容,根据请求中三个key计算得来,16字节。

应答字符串计算过程伪代码:


part_1 = key1中所有数字 / key1中空格数量
part_2 同上
sum = big_endian(part_1)+big_endian(part_2)+key3
challenge_response = md5_digest(sum);

32位整数的big_endian计算策略:


# 很类似于rgba颜色计算,从下面的函数可以看出计算过程
var big_endian = function(n) {return [3,2,1,0].map(function(i) { return n >> 8*i & 0xff });
}
big_endian(0xcc77aaff);
// -> [204, 119, 170, 255]

发送数据

WebSocket API的被设计成用事件处理数据,客户端只要得到事件通知就可以获取到完整的数据,而不需要手动处理缓冲器。

这种情况下,每一笔数据被称为一帧。在规范的定义中,它的头部必须以0x00开始,尾部属性以0xff结束,这样每一次数据发送至少有两个字节。

服务器实现中,收到数据时要截掉头尾;而发送数据是要包装头尾。格式如下:


# '你好'的原始二进制表示,请求头和这里都是utf8编码
<Buffer e4 bd a0 e5 a5 bd>
# 包装后的二进制表示。
<Buffer 00 e4 bd a0 e5 a5 bd ff>

参考链接

websocket draft: http://www.whatwg.org/specs/web-socket-protocol/

w3 websocket api:http://dev.w3.org/html5/websockets/

whatwg webapp规范:http://www.whatwg.org/specs/web-apps/current-work/

有趣的交互证明:Zero-knowledge proofs

附件

查看或下载:https://gist.github.com/1066475

转载于:https://www.cnblogs.com/ambar/archive/2011/06/30/node-websocket-server.html

相关文章:

反向春运成为新趋势 客流年增9%

资料图&#xff1a;春运。殷立勤 摄 中新社北京1月18日电 (记者 周音)近年来&#xff0c;反向春运成为新趋势。中国铁路总公司18日披露&#xff0c;反向春运客流以年增9%左右的速度增长。 传统春运是大城市返乡回家过年。反向春运是年轻人选择将老家的父母和孩子接来自己工作的…

把.sql文件上传到服务器上

使用xftp工具&#xff0c;在root文件夹下新建myProject文件夹&#xff0c;然后把.sql文件拖拽过去即可。 进入xshell&#xff0c; 进入到mysql&#xff1a; mysql -u root -p输入密码&#xff1a; create database 数据库名&#xff1b;use 数据库名&#xff1b;source ~/新…

最先进的开源游戏引擎KlayGE 3.12.0发布

转载请注明出处为KlayGE游戏引擎&#xff0c;本文地址为http://www.klayge.org/2011/06/30/%e6%9c%80%e5%85%88%e8%bf%9b%e7%9a%84%e5%bc%80%e6%ba%90%e6%b8%b8%e6%88%8f%e5%bc%95%e6%93%8eklayge-3-12-0%e5%8f%91%e5%b8%83/ KlayGE 3.12.0在上半年的最后一天发布了&#xff01…

如何防止博客文章被窃取

写文章的优点&#xff1a; 1.整理自己的学习过程&#xff0c;思想心得。 2.看到自己的文章被别人转载了&#xff0c;心理愉悦&#xff0c;说明作者写的文章有价值&#xff0c;可以帮助他人。 缺点&#xff1a; 1.花费时间和精力。 2.说是抄袭别人。明明自己是原创&#xff0c;别…

js中的装饰器执行顺序

/*** 执行顺讯* [(property)...]->[(parameter->method)...]->constructor->class* [属性...]->[((方法参数...)->方法)...]->[constructor...]->class* 声明周期 property|parameter|method|constructor|class* 声明周期 [始化完毕]init->[属性添加…

关于springboot vue前后端分离项目部署到阿里云轻量服务器(前后端分开部署)

0.购买阿里云服务器 1.安装jdk 使用yml安装 2.安装mysql 3.安装nginx 4.打包后端项目 后端项目更改&#xff1a; 在pom.xml文件中&#xff0c;增加打包成jar包的配置文件 application.properties配置文件中更改数据库信息&#xff0c;端口号&#xff1a;&#xff08;所使…

.NET泛型解析(下)

上一篇对.NET中的泛型进行了详细的介绍以及使用泛型的好处是什么,这篇将更加深入的去了解泛型的其他的知识点,重头戏. 【1】泛型方法 上一篇我们也说过了,泛型可以是类,结构,接口,在这些泛型类型中定义的方法都可以叫做泛型方法,都可以引用由泛型类型本身指定的一个类型参数例如…

spark集群使用hanlp进行分布式分词操作说明

本篇分享一个使用hanlp分词的操作小案例&#xff0c;即在spark集群中使用hanlp完成分布式分词的操作&#xff0c;文章整理自【qq_33872191】的博客&#xff0c;感谢分享&#xff01;以下为全文&#xff1a;分两步&#xff1a;第一步&#xff1a;实现hankcs.hanlp/corpus.io.IIO…

让VirtualBox的虚拟机器在电脑开机时自动启动

当你安装很多套Virtualbox的虚拟机器系统后&#xff0c;希望能在开机后自动启动虚拟机器的系统。 Linux (Host OS):在你的/etc/rc.local中加入下列几行VBoxVRDP -startvm WinXP & VBoxVRDP -startvm Win2003 & VBoxVRDP -startvm LinuxFC6 & Windows (Host OS): 开…

L1-025 正整数A+B

不确定的点&#xff1a; 1.数据用什么类型输入&#xff0c;如果用字符串类型输入&#xff0c;怎么判断它是不是正整数 2.怎么判断哪部分是A&#xff0c;哪部分是B 解析 c语言’\0’ 意思&#xff1a; 字符常量占一个字节的内存空间。字符串常量占的内存字节数等于字符串中字节…

制作显示欢迎信息的脚本程序

终端程序恐怕是Linux用户使用最为频繁的了。我的Debian系统启动后便是直接进入的终端界面。为了在每次登录时或者是在X视窗环境下打开终端程序时显示一些欢迎信息&#xff0c;比如当前的日期、名人警句等&#xff0c;从而可以增加一些生活情趣&#xff0c;就可以创建一个脚本程…

文章分页浏览(二)

分页的方法: View Code publicstringOutputBySize(stringp_strContent, stringbType) { stringm_strRet ""; intm_intPageSize 500;//文章每页大小 intm_intCurrentPage 1;//设置第一页为初始页 intm_intTotalPage 0; intm_intArticlelengt…

云计算时代,如何选择适合自己的云服务器厂商?

据百科定义&#xff0c;云服务器是一种处理能力可弹性伸缩的计算服务&#xff0c;帮助您快速构建更稳定、安全的应用&#xff0c;降低开发运维的难度和整体IT成本&#xff0c;使您能够更专注于核心业务的创新。云服务器相对传统服务器有些优势&#xff1a;按国内市场上云服务器…

解决mysql建立的数据库名字不能带大写字母

1、在安装目录下&#xff0c;找到 my.ini文件 2、找到 [mysqId]节点 3、在它下面添加 lower_case_table_names2如果设置为0的话&#xff0c;我的Mysql服务不能重启 4、重启Mysql 直接手动重启即可&#xff0c;在服务中&#xff0c;找到Mysql服务&#xff0c;先停止&#xff0c…

CSS position财产

CSS在position位置信息要素用于表示属性。 有三个起飞值&#xff1a;static, absolute, relative。假设元件不显式配置position财产&#xff0c;该元素默认position 值至static。 1、static&#xff1a;这是表示该元素依照排列和嵌套的顺序和规则应该在的位置&#xff0c;此时设…

通过产品ID得到collection!!!

You can do as following for filtering products id 10 and 12$products->addAttributeToFilter(entity_id, array(in>array(10,12)));当然这种方式的上面&#xff0c;必须用 ->addAttributeToSelect();下面这种方式比较实用的&#xff0c;如果想按照名字排序&#…

jupyter notebook出现cannot import name 'create_prompt_application'问题(Died Kernel)

应该是在安装其它python第三方库时更新了prompt-toolkit版本&#xff0c;降级到下面的版本即可&#xff1a; sudo pip install prompt-toolkit1.0.15 转载于:https://www.cnblogs.com/darklights/p/10302706.html

导入sql时出现Invalid default value for ‘create_time‘报错处理方法

&#xff08;上图是初始的sql文件的内容&#xff09; 在开发微信小程序时&#xff0c;需要导入.sql文件&#xff0c;但是最一开始导入的时候没有任何改动进行了导入&#xff0c;报错如下 PS E:\weichatApp\my-project\server> node tools/initdb.js 开始初始化数据库... 准…

Python相关机器学习

Python机器学习库 Python的机器学习库汇总与梳理 机器学习之开源库大总结 转载于:https://www.cnblogs.com/SFMing/p/4590261.html

Django 图片上传upload_to路径指定失效的问题记录

为什么80%的码农都做不了架构师&#xff1f;>>> 初始方法一&#xff1a; 疑虑&#xff1a;model使用upload_to自定义路径方法失效&#xff0c;指定路径也失效。最后以Views中指定MEDIA_URL和MEDIA_ROOT做拼接&#xff0c;并且自行判断并建立文件夹&#xff0c;手动…

javascript tab切换类LixTabs最新版

javascript Tab切换类LixTabs&#xff0c;更新至0.5版: 受snandy的“读jquery”系列的启发&#xff0c;改进了代码&#xff0c;现在调用LixTabs时不用加new了。即可以这样写&#xff1a;var tab Tabs();把原来的参数evt&#xff0c;改成了易理解的event(我的疏忽)总代码量&…

linux虚拟机文件挂载

把U盘中的文件上传到linux虚拟机中&#xff0c;可以采用挂载的方式。 &#xff08;1&#xff09;启动虚拟机 &#xff08;2&#xff09;把U盘插入电脑中 &#xff08;3&#xff09;输入命令 fdisk -l可以查看新的分区 &#xff08;4&#xff09; cd /mnt mkdir usb mount /d…

HDU 1757 A Simple Math Problem

Problem Description Lele now is thinking about a simple function f(x).If x < 10 f(x) x.If x > 10 f(x) a0 * f(x-1) a1 * f(x-2) a2 * f(x-3) …… a9 * f(x-10);And ai(0<i<9) can only be 0 or 1 .Now, I will give a0 ~ a9 and two positive intege…

mariadb 内存占用优化

本文由云社区发表 作者&#xff1a;工程师小熊 摘要&#xff1a;我们在使用mariadb的时候发现有时候不能启动起来&#xff0c;在使用过程中mariadb占用的内存很大&#xff0c;在这里学习下mariadb与内存相关的配置项&#xff0c;对mariadb进行调优。 查询最高内存占用 使用以下…

windows程序设计之对话框简介1

这里先介绍下wParam和lParam&#xff0c;对于鼠标而言&#xff0c;LOWORD(wParam)和HIWORD(wParam)代表鼠标位置x,y坐标&#xff0c;对于菜单和控件而言&#xff0c;两者wParam的低字节都是各自的ID&#xff0c;即LOWORD(wParam)都是ID。两者的高字节对菜单而言是0&#xff0c;…

linux虚拟机下安装Tomcat

&#xff08;1&#xff09;首先通过挂载的方式把 tomcat的安装包从U盘上传到虚拟机中 我上传的路径是 &#xff1a;usr/tomcat &#xff08;2&#xff09; cd /usr/tomcat tar xzvf 压缩包的名字 ##进行解压&#xff08;3&#xff09;进到tomcat安装目录下的bin文件夹 ./…

unity中使用自定义shader进行光照贴图烘培无法出现透明度的坑爹问题

最近开发中在对场景进行光照贴图烘焙时发现一个坑爹问题&#xff0c;在使用自定义shader的时候&#xff0c;shader命名中必须包含Transparent路径&#xff0c;否则烘焙的时候不对alpha通道进行计算&#xff0c;烘焙出来都是狗皮膏药 比如一个shader叫 Shader "xx/UnlitAlp…

动软代码生成器教程——懒人有福了

很多时候项目必须是三层架构模式&#xff0c;但是很多繁琐的代码让多数程序员闹心……那有没有一个省时省力的工具快速的帮我们搞定三层架构呢&#xff1f;回答是肯定的&#xff0c;很早之前技术牛人李天平就开发出了这么一款工具&#xff0c;目前该工具还在不断的更新&#xf…

unity3d做简单小游戏可以吗?

可以吗&#xff1f;当然。如果是独立开发&#xff0c;主要在美工&#xff0c;这类的游戏程序简单&#xff0c;有些基础就行&#xff0c;美工要做得好可不容易&#xff0c;要是效果要求不高&#xff0c;随便在max拉几个模型吧。unity方面&#xff0c;熟悉一下&#xff0c;如果有…

逻辑覆盖测试(一)语句覆盖

语句覆盖&#xff1a; 设计测试用例时保证程序的每条语句至少执行一次。 简单来说&#xff0c;就是每个语句都覆盖一遍。 例子&#xff1a; 流程图如下&#xff1a; 测试用例如下&#xff1a; x4,z9&#xff0c;第一个if语句执行到了&#xff1b; x4,y7,第二个if语句为true…