位置: 编程技术 - 正文

socket.io与pm2(cluster)集群搭配的解决方案

编辑:rootadmin

推荐整理分享socket.io与pm2(cluster)集群搭配的解决方案,希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:,内容如对您有帮助,希望把文章链接给更多的朋友!

socket.io与cluster

在线上系统中,需要使用node的多进程模型,我们可以自己实现简易的基于cluster模式的socket分发模型,也可以使用比较稳定的pm2这样进程管理工具。在常规的http服务中,这套模式一切正常,可是一旦server中集成了socket.io服务就会导致ws通道建立失败,即使通过backup的polling方式仍会出现时断时连的现象,因此我们需要解决这种问题,让socket.io充分利用多核。

在这里之所以提到socket.io而未说websocket服务,是因为socket.io在封装websocket基础上又保证了可用性。在客户端未提供websocket功能的基础上使用xhr polling、jsonp或forever iframe的方式进行兼容,同时在建立ws连接前往往通过几次http轮训确保ws服务可用,因此socket.io并不等于websocket。再往底层深入研究,socket.io其实并没有做真正的websocket兼容,而是提供了上层的接口以及namespace服务,真正的逻辑则是在“engine.io”模块。该模块实现握手的http代理、连接升级、心跳、传输方式等,因此研究engine.io模块才能清楚的了解socket.io实现机制。

场景重现

服务端采用express+socket.io的组合方案,搭配pm2的cluster模式,实现一个简易的b/s通信demo:

app.js

index.html

pm2.json

这样,执行命令pm2 start pm2.json即可开启服务,访问.0.0.1:/page,点击按钮发起ws连接,观察控制台即可。

下图清晰显示了socket.io握手的错误:

可见在websocket连接建立之前多出了3个xhr请求,而websocket连接建立失败后又多出了几个xhr请求,同时最后两个xhr请求失败了。

socket.io没有采用直接建立websocket连接的粗暴方式,而是首先通过http请求(xhr)访问服务端的相关轮训配置信息以及sid。此处sid类似sessionID,但是它唯一标识连接,可理解为socketId,以后每次http请求cookie中都必须携带sid(httponly);

第二、三个请求用于确认连接,在socket.io中,post请求是客户端发送消息给服务端的唯一形式,而且post响应一定是“ok”,它的“content-length”一定为2;而get请求主要用于轮训,同时获取服务端的相关消息,这会在下文中有体现;

第四个websocket连接请求失败,这主要是由于与后端http握手失败造成的;

第五个请求为xhr方式的post请求,它是作为websocket通道建立失败后的一种兼容性处理,上文讲述了socket.io的post请求只在客户端需要发送消息给服务端时才会使用,因此,为了证实我们查看消息体:

可见,它携带了客户端发出的消息类型b:message,同时包含消息体{}空对象。对应的,服务端返回“OK”;

第六个请求为xhr方式的get请求,用来获取服务端对第五个请求的响应。

至此,大致分析了socket.io建立连接的大致过程以及连接建立失败后如何兜底的方案,下面分析为何出现握手失败的问题。

socket.io与pm2(cluster)集群搭配的解决方案

原因何在

实例中pm2主进程开启了4个工作进程,由主进程侦听端口并分发请求给工作进程。pm2进程在分发请求的阶段采用了某种算法的均衡,如round-robin或者其他hash方式(但不是iphash),因此在socket.io客户端连接建立阶段发送的多个xhr请求,会被pm2定位到不同的worker进程中。前文中提到每个xhr请求都会携带sid字段标识当前连接,因此当一个携带sid字段的请求被pm2定位到另一个与该连接无关的worker时,就会造成请求失败,返回{"code":1,"message":"Session ID unknown"}错误;即使前三次xhr握手成功,进入websocket连接升级阶段,负责侦听update事件的worker也往往不是之前的那个worder,因此导致websocket连接建立失败。

一言以蔽之,客户端多次请求的服务端进程不是同一个进程才导致的ws连接无法成功建立。那么如何才能解决呢?最简单的方案就是确保客户端的每次请求都可以定位到同一个服务进程即可。当然,分布式session同样可以解决问题,依托第三方缓存类似redis并配合一致性hash算法,确保所有服务进程都可以获取到连接信息,相互配合完成连接建立。但这也仅仅是作者在理论上分析的一种实现方式,并没有测试通过,因为这种分布式架构不仅实现繁杂而且引入了相关依赖redis,不太可取。

那么下文主要针对确保客户端的每次请求都可以定位到同一个服务进程这一点实现解决方案。

多种实现

官方实现

官方提供了一种比较轻便的架构:nginx反向代理+iphash

我们的示例demo中的http服务器只侦听端口,因此必须由pm2分发请求,否则会出现端口占用的错误发生。但是,官方的解决方案是每个进程的socket.io服务器创建不同端口的http服务器,专注用于http握手和升级,由nginx做握手请求的代理。而且针对nginx必须设置iphash,保证同一个客户端的多次请求定位到后端同一个服务进程。

这样,示例demo中会占用5个端口,其中端口为公用的http服务器使用,其他四个端口则只用于ws连接握手。但是这四个端口却如何选取呢?为了保证扩展性以及顺序性,采用与pm2相兼容的方案。pm2会为每个worker进程分配一个id,并且将该id绑定到进程的环境变量中,那么我们就可以利用该worker id生成4个不同的端口号。

app.js

index.html

nginx.conf

在本机绑定hosts地址后开启nginx服务,同时开启服务器,点击按钮建立ws连接成功。

服务端路由

服务端路由,意义在于“服务端做worker的负载均衡,并将选择的worker ip和端口渲染在页面,之后浏览器的所有ws连接默认连接到对应 ip:port的服务器中”。这样只要是服务端渲染的页面都可以采用这种方式实现。

如果页面采用前端异步渲染,仍可以采用这种方式,不过首先通过xhr请求向服务端获取需要握手的http服务器的ip和端口,然后在进行ws连接。

服务端路由的前提仍然是需要针对每个ws服务器分配一个端口,只不过去掉nginx由服务端做ip hash。采用服务端路由架构清晰,而且实现容易,兼容性好。

上帝进程路由

此处的上帝进程即为主进程,类似pm2进程。上帝进程路由则是在上帝进程层面上做请求的定向分发,保证请求主机和进程的一致性。在上帝进程中,针对每个请求的ip做hash,并对每一个ws服务器创建单独的http服务器用于握手升级。

简易代码:

总结

本文实现了三种解决方案,归根到底就是“ip hash”,不同点在于在请求处理的不同阶段做ip hash。

可以在请求处理最前端做iphash,即nginx方式,这也就是第一种方案;

可以在请求处理的第二层分发处做iphash,即上帝进程路由的方式,即第三种;

也可以在请求处理的终端做iphash,即服务端路由的方式,也就是第二种;

同时共享session也同样可以实现,借助socket.io-redis模块也可以实现。

好了,

标签: socket.io与pm2(cluster)集群搭配的解决方案

本文链接地址:https://www.jiuchutong.com/biancheng/380825.html 转载请保留说明!

上一篇:Node.js v8.0.0正式发布!看看带来了哪些主要新特性(node js官网)

下一篇:nodejs socket实现的服务端和客户端功能示例(nodejs-websocket)

  • 评估报告是什么
  • 购进货物支付的运输费用如何计算增值税
  • 证券交易计税依据
  • 企业建厂房购进材料可以抵扣吗
  • 办公费税前扣除标准2023
  • 存货跌价准备在财务报表哪里看
  • 全额计提坏账准备后,多久核销应收账款
  • 年终奖报表怎么做
  • 发票认证了申报系统没有数据
  • 增值税销项抵扣报税后有效期是多长时间
  • 已经抵扣的发票还能红冲吗?怎么操作
  • 小微企业增值税优惠政策最新2023
  • 原材料盘亏会计分录怎么做
  • 辅导期一般纳税人和一般纳税人的区别
  • 汇算清缴期间费用社保填哪里
  • 政府奖励金怎么用
  • 付给其他公司的利息怎么做账
  • 公司罚款作为一种对过错方式的处罚
  • 手工开具发票
  • 物业公司营改增税务筹划方案
  • 股东借款可以转为认缴出资
  • 小微企业附加税优惠政策2023
  • 辅导期纳税人预缴增值税
  • 公司没有进出口权 如何进行业务
  • 利润表里的营业成本包括哪些
  • 房地产开发企业的土地使用权计入哪里
  • 鸿蒙系统通知栏和控制
  • 主板类型大全
  • 如何清除上网记录?
  • 苹果电脑开机声音怎么关
  • 六月雪的养殖方法和注意事项
  • 会计的职责概述
  • 企业所得税申报错误怎么更改
  • 收到苗木发票怎么做账
  • 增值税发票的进项和出项要一致吗
  • 上市公司发行债券股价会涨吗
  • 对公账户发放工资要固定几号打吗
  • PHP:stream_context_get_options()的用法_Stream函数
  • 资产处置损益和待处理财产损溢区别
  • 老生常谈PHP 文件写入和读取(必看篇)
  • vue页面嵌套iframe vue页面
  • vuex用法和原理
  • 暂估入库少了
  • 购货方享受现金流量吗
  • 不动产租赁专票对房产有无影响
  • 万元版和十万元版发票图片
  • sql2005sp4
  • 承兑汇票大回头是什么意思
  • 小规模纳税人企业所得税优惠政策最新2023
  • 存货的入账价值等于
  • 收到的加盟费要交税吗
  • 免征增值税政策的政策有哪些?
  • 减免的企业所得税是否需要征税 税屋
  • 出口退税一般风险商品
  • 公司购买承兑需要交税吗
  • 企业过桥贷款违法吗
  • 事业单位利息收入
  • 押金是否可以抵扣租金
  • 税控维护费减免税款
  • 车保险说给返钱是真的吗
  • 扶贫入股分红政策
  • 接受投资者投入的资产
  • 发票作废了还能恢复吗?
  • win7激活失败提示错误代码0x80072F8F
  • win8有几种版本
  • pptd40nt.exe是什么进程 有什么用 pptd40nt进程查询
  • linuxwho
  • Windows8 PrintScreen屏幕截图如何保存到桌面上
  • excle纵坐标
  • linux中shell脚本实验总结
  • python如何将矩阵转为图像
  • linux更换用户登录命令
  • javascript教程完整版
  • python怎样升级
  • jquery简单例子
  • 使用jquery实现的项目
  • js 截取
  • jquery控制css样式
  • 厦门怎么查询自己名下电动自行车
  • 北京大兴开发区房价
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设