前情提要

「 话说上回说到!那WebSocket大侠,巧借http之内力,破了敌阵的双工鸳鸯锁,终于突出重围。

然而玄难未了,此时web森林中飞出一只银头红缨枪,划破夜

"莫非!?" websocket大侠喃喃念道,"恐怖如斯,你莫不是就是那个手使单向追魂枪的。。。"

"正是在下!",那人厉声喝道。只见那胸前的纹章铭刻着几个洋文——

读作"EventSource"!」

上一篇文章请看这里:论一个低配版Web实时通信库是如何实现的( WebSocket篇)

引论

simple-socket是我写的一个"低配版"的Web实时通信工具(相对于Socket.io),在参考了相关源码和资料的基础上,实现了前后端实时互通的基本功能,选用了WebSocket ->server-sent-event -> AJAX轮询这三种方式做降级兼容,分为simple-socket-client和simple-socket-server两套代码。

我的上一篇文章讲了如何进行websocket的前后端编码,所以今天来聊一聊event-source这块的

论一个低配版Web实时通信库是如何实现的( WebSocket篇)

github仓库地址

https://github.com/penghuwan/simple-socket

npm命令

npm i simple-socket-serve   (服务端npm包)
npm i simple-socket-client (客户端npm包)

EventSource的前端代码

EventSource的前端API主要有这么四个

  1. 创建es对象:var es = new EventSource(url)

  2. es两端连接事件打开的回调:es.onopen = function () { }

  3. 监听服务端发送事件: es.addEventListener("XXX", function (e) { // e.data }

  4. 监听服务端的message事件es.onmessage = function; 相当于es.addEventListener("message",function);

业务代码如下

(1)前端从服务端接收消息

前端通过监听服务端message事件,接收消息,并解析event和data,然后通过emitter.emit(event, data)触发事件,从而调用socket.on设置的监听回调

function Client() {
this.ws = null
this.es = null; // EventSource对象
init.call(this); // 设置this.type并初始化相关对象例如es或ws
listen.call(this);
// ...
} function listen() {
// 保存this
var self = this;
switch (this.type) {
// 当type为eventsource时,执行以下代码,this.type根据能力检测设置
case 'eventsource':
// 监听触发connect事件,把client对象自身传入当作socket
this.es.onopen = function () {
emitter.emit('connect', self);
};
// 监听服务端传来的message事件
this.es.addEventListener("message", function (e) {
var payload = JSON.parse(e.data);;
var event = payload.event;
var data = payload.data;
emitter.emit(event, data);
}, false);
break;
// ...
}
}

(2)前端发送消息给服务端

由于event-source是单向的,只能从服务端从前端发送消息,而不能从前端发送消息给服务端。这和websocket显著不同

不过别担心,因为我们不是还有AJAX嘛!

对于前端发送消息的情况 我们可以发一个post请求过去,同时借助/eventsource这个路径,告诉服务端这是一个SSE请求

$.ajax({,
type: 'POST',
url: `http://${url}/eventsource`,
data: { event, data },
success: function () {
}
});

EventSource的服务端代码

好像这波就没了吧,OK,我们接下来走下路。

server-sent-event的服务端握手流程

server-sent-event(或event-source),需要借助流(stream)的方式去实现通信。

Stream 是一个抽象接口,Node 中有很多对象实现了这个接口。例如,对http 服务器的request/response 对象就是一个 Stream。

它可以分为四种类型:

  • Readable - 可读操作。

  • Writable - 可写操作。

  • Duplex - 可读可写操作.

  • Transform - 操作被写入数据,然后读出结果。

服务器每次接收的Response是一个Writable,它可以被写入数据,将一个流写入另一个流可以通过调用pipe方法。

所以我们需要创建一个stream的实例,然后通过调用stream.pipe(Response)将流写入响应中,这样就可以被前端es.addEventListener添加的回调给接收到了。

但问题在于 。。。Stream是个抽象接口,Node.js没有给Stream提供构造函数

不过没关系,我们可以这样做:

    • 使用call方法继承stream父函数

    • 使用util.inherits继承stream的原型

    • 重写_read和_write方法(否则会报错)

// 因为我们的流需要写和读,所以使用双工的stream.Duplex构造
function EventStream() {
stream.Duplex.call(this); // 构造函数继承
}
util.inherits(EventStream, stream.Duplex); // 原型继承
// 重写_read和_write方法
EventStream.prototype._read = function () { }
EventStream.prototype._write = function () { }

握手代码逻辑

  1. 创建stream实例,调用pipe方法输送给Response, 同时stream我们保存在socket对象中,在向前端发送数据时候会使用

  2. 将Content-Type字段设置为'text/event-stream',同时Connection设置为'keep-alive'

  3. 将状态码设为200(否则前端onopen方法不会触发)

_handleEShandShake(ctx, socket) {
// 前面定义好的类似stream的类
const eventStream = new EventStream();
// 设置eventStream
socket.setEventStream(eventStream);
// 握手成功后触发onConnection方法,TODO
// 设置符合Event-Source要求的首部
ctx.set({
'Content-Type': 'text/event-stream',
'Cache-Control': 'no-cache',
'Connection': 'keep-alive',
});
// 将Stream赋给body,Koa底层会判断Stream类型并调用pipe方法流入response
ctx.body = eventStream;
// 设置表示请求成功,否则前端onopen方法不会触发
ctx.status = 200;
// 触发connect方法,传递socket对象
this.emit('connect', socket);
}

Event-Source服务端向前端发送消息。

这里要先说下event-source的报文结构了,由四种字段组成

  • event:事件名,对应前端es.addEventLisener设置的事件名

  • data:数据,为字符串

  • id: 消息标识符,可以缺省

  • retry:表示重新连接的时间间隔

这四个字段两两之间用\n分开,而最后一个字段值需要用\n\n做结尾

例如:`event:message\n data: XXX \n\n`

话不多说,看代码

class Socket extends events.EventEmitter {
constructor(socketId) {
super();
}
// 设置
setEventStream(eventStream) {
this.eventStream = eventStream;
} // 自定义的emit,触发的是前端的on
emit(event, data) {
const dataStr = JSON.stringify({event,data})
if (this.transport === 'eventsource') {
if (!this.eventStream) { throw new Error('eventStream不存在,无法emit') };
// 向stream中写入数据,只要stream尚未关闭
// 数据就会传给前端的onmessage方法或addEventListener('message',fuc)方法
this.eventStream.push(`event:message\ndata:${dataStr}\n\n`);
}
}
}

Event-Source服务端接收前端消息

之前说了,event-source是单向的,所以前端到服务端的传送是通过Ajax请求过来的,所以解析下body,触发事件就OK了

故事到这里就结束了。

有诗为证

江河湖泊浪滔滔,WebSocket多逍遥

EventSource先来却后到,Ajax轮询热血逞英豪!

欲知后事如何,且听下回分解!

知乎专栏

最近也在知乎上写文章,感觉破乎的体验很差!没有博客园好!感觉博客园的各位才个个都是人才,说话又好听!我超喜欢在里面的。

所以说。。。大家好,给大家介绍一下这是我的知乎专栏

https://zhuanlan.zhihu.com/c_135367198

这位路过的大哥你有灵气从键盘喷出,看来是百年一遇的代码奇才,就施舍善心关注一下吧,以解小弟拖家带口之忧,养儿奉母之愁(大雾)

【JavaScript】论一个低配版Web实时通信库是如何实现的之二( EventSource篇)的更多相关文章

  1. 【Node/JavaScript】论一个低配版Web实时通信库是如何实现的( WebSocket篇)

    引论 simple-socket是我写的一个"低配版"的Web实时通信工具(相对于Socket.io),在参考了相关源码和资料的基础上,实现了前后端实时互通的基本功能 选用了Web ...

  2. 【Java】利用注解和反射实现一个"低配版"的依赖注入

    在Spring中,我们可以通过 @Autowired注解的方式为一个方法中注入参数,那么这种方法背后到底发生了什么呢,这篇文章将讲述如何用Java的注解和反射实现一个“低配版”的依赖注入. 下面是我们 ...

  3. 搭建一个低配版的Mock Server

    mock翻译过来是模仿的意思,Server是服务器.粗暴点直译就是模仿服务器. 写在前面 通过阅读本文,你将对Mock的使用有一定的了解,对前后端分离的概念有了更深一步的认识,对Koa的使用有一定的了 ...

  4. 基于canvas和web audio实现低配版MikuTap

    导言 最近发掘了一个特别happy的网页小游戏--MikuTap.打开之后沉迷了一下午,导致开发工作没做完差点就要删库跑路了,还好boss瞥了我一眼就没下文了.于是第二天我就继续沉迷,随着一阵抽搐,这 ...

  5. Jenkins 结合 Docker 为 .NET Core 项目实现低配版的 CI&CD

    随着项目的不断增多,最开始单体项目手动执行 docker build 命令,手动发布项目就不再适用了.一两个项目可能还吃得消,10 多个项目每天让你构建一次还是够呛.即便你的项目少,每次花费在发布上面 ...

  6. 搭建react项目(低配版)

    react项目低配版,可作为react相关测试的基础环境,方便快速进行测试. git clone git@github.com:whosMeya/simple-react-app.git git ch ...

  7. java线程学习第一天__低配版的卖面包机

    package Thread;import javax.xml.bind.ValidationEvent;class snacks{    private int  SaledSnacks=0;   ...

  8. unittest框架(惨不忍睹低配版)

    根据我上个随笔的unittest框架优化得来,虽然对于smtp模块还是有点迷糊,不过还是勉强搭建运行成功了,还是先上代码: #login_test.py import requests class L ...

  9. 分享一个低配VPS下运行的mysql配置文件

    在各种内存CPU核心只有1/2核,内存只有512M/1G的vps下,内存.CPU.硬盘都不是太充裕.因此主要思路是,禁止吃内存大户innodb引擎,默认使用MyISAM.禁止吃硬盘大户log-bin, ...

随机推荐

  1. [记录]HAproxy负载均衡配置教程

    HAproxy负载均衡配置教程 一.简介 haproxy是一个开源的高性能负载均衡软件:支持双机热备.虚拟主机和图形化的管理界面,自带强大的对RS健康检查功能:支持TCP(四层).HTTP(七层)应用 ...

  2. TF项目实战(基于SSD目标检测)——人脸检测1

    SSD实战——人脸检测 Tensorflow 一 .人脸检测的困难: 1. 姿态问题 2.不同种族人, 3.光照 遮挡 带眼睛 4.视角不同 5. 不同尺度 二. 数据集介绍以及转化VOC: 1. F ...

  3. HashSet源码分析:JDK源码系列

    1.简介 继续分析源码,上一篇文章把HashMap的分析完毕.本文开始分析HashSet简单的介绍一下. HashSet是一个无重复元素集合,内部使用HashMap实现,所以HashMap的特征耶继承 ...

  4. 个人永久性免费-Excel催化剂功能第92波-地理地址与经纬度互转功能

    GPS设备和手机LBS的兴起,在地理信息存储过程中,在程序.应用级别是需要用经纬度去定位,而在数据分析的级别,特别是省市区镇街的分析,用到的是人可识别的文本类型存储,从设备中采集下来的数据和人工维护的 ...

  5. python面向过程编程小程序 -ATM(里面用了终端打印)

    06.09自我总结 1.文件摆放 ├── xxxx │ ├── run.py │ └── fil_mode.py │ └── data_time.py │ └── loading.py │ └── d ...

  6. c++课程设计:行政区划管理系统

    大一的课程设计基本上除了计算器,就是各种管理系统.(大概吧) 感觉看到题目整个一年的c++好像没学明白似的.基础知识掌握还算可以,真刀真枪的打代码,而且是实现这么些功能,做成一个管理系统,就真正感觉到 ...

  7. Kafka集群部署以及使用

    Kafka集群部署 部署步骤 hadoop102 hadoop103 hadoop104 zk zk zk kafka kafka kafka http://kafka.apache.org/down ...

  8. [LeetCode] 6. ZigZag Conversion (Medium)

    原题链接 把字符串按照 ↓↗↓……的顺序,排列成一个 Z 形,返回 从左到右,按行读得的字符串. 思路: 建立一个二维数组来按行保存字符串. 按照 ↓↗↓……的方向进行对每一行加入字符. 太慢了这个解 ...

  9. Spring Cloud 之 Config与动态路由.

    一.简介  Spring Cloud Confg 是用来为分布式系统中的基础设施和微服务应用提供集中化的外部配置支持,它分为服务端与客户端两个部分.其中服务端也称为分布式配置中心,它是一个独立的微服务 ...

  10. 7kyu (难度系数kyu阶段数值越大难度越低) 数组分组及求和

    几个人排成一排,分成两队.第一个人进入一队,第二个人进入第二队,第三个人进入第一队,以此类推. 给定一个正整数的数组(人的权重),返回两个整数的新数组/元组,其中第一个是第1组的总重量,第二个是第2组 ...