即时通讯协议之Qunar

标签: 术→技巧 研发 即时通讯 | 发表时间:2021-11-09 22:37 | 作者:钱魏Way
出处:https://www.biaodianfu.com

Qunar 由于业务上对 IM 系统的需求,以及对 IM 需要支持的功能和扩展,结合市面上已有的 IM 的实现,实现了自己的一套完善的办公 IM 和客服 IM 系统。具备了以下几个重要特点:实时性,可靠性,一致性,安全性,扩展性,高并发。

Startalk是去哪儿开源的一款通用的,高性能的企业级im套件。Startalk 前身是去哪儿的Qtalk。其内核也在去哪儿旅行和去哪儿网站上扮演着着客服服务工具的角色。也就是说,一套内核同时为去哪儿网提供了内部企业办公和商家tob业务的支撑。

IM 常见实现方案

XMPP 协议

XMPP 是一个开放式的 XML 协议,设计用于准实时消息和出席信息以及请求-响应服务。

XMPP 协议单元包括三个大类:

  • Presence:Presence 决定了 XMPP 实体的状态,用来告诉服务器该实体是在线、离线或者繁忙;
  • Message:用户之间发送和接收的消息;
  • IQ:请求-响应类型的报文;

优点:XMPP 有大量的优秀实现,且社区环境良好,功能和扩展能力丰富。

缺点:报文较大,耗费网络流量和电量。

MQTT 协议

MQTT(Message Queuing Telemetry Transport,消息队列遥测传输协议),是一种基于发布/订阅(publish/subscribe)模式的”轻量级”通讯协议,该协议构建于 TCP/IP 协议上,由 IBM 在 1999 年发布。MQTT 最大优点在于,可以以极少的代码和有限的带宽,为连接远程设备提供实时可靠的消息服务。作为一种低开销、低带宽占用的即时通讯协议,使其在物联网、小型设备、移动应用等方面有较广泛的应用。

MQTT 是一个基于客户端-服务器的消息发布/订阅传输协议。 MQTT 协议是轻量、简单、开放和易于实现的,这些特点使它适用范围非常广泛。正是由于它的简单,也带来了他的缺点:需要自己去实现聊天,好友等 IM 的逻辑。

Qunar IM实现方案

协议的选择

基于上面调研的常用 IM 协议,我们最终选择了 XMPP 协议最为最开始的实现协议:

  • 因为可以用最小的开发工作来实现基本的聊天功能;
  • 可以使用已有的扩展插件,更快实现更多的功能;
  • 针对 XMPP 的缺点,做针对性的修改,将网络传输这一阶段,改成 protocol buffer 协议,弥补网络流量和电量的短板。

开源项目的选择

基于:

  • ejabberd 是基于 Jabber/XMPP 协议的即时通讯服务器;
  • 由 GPLv2 授权(免费和开放源码);
  • 采用 Erlang/OTP 开发,它的特点是,跨平台,容错,集群和模块;
  • Ejabberd 是可扩展性最好的一种 Jabber/XMPP 服务器之一,支持分布多个服务器,并且具有容错处理,单台服务器失效不影响整个 cluster 运作;
  • Erlang 的调度和 GC 策略更适合 IM 的实时性要求(见参考)。

团队最终选择使用 ejabberd 开源实现来快速实现自己的 IM 功能。

架构设计

  • 客户端通过两条连接来和服务器进行通讯
    • TCP 长连接( web 使用的 websocket ):该连接上交互的是和状态相关或者多端同步的报文;
    • HTTP 连接:和状态无关或者不需要多端同步的。
  • 负载均衡
    • TCP 长连接通过 LVS 或者 HA 来做负载均衡;
    • HTTP 通过 nginx 来做负载均衡。
  • 数据
    • 数据或直接放入数据库或者进入 MQ ,供需要方订阅消费;
    • 高频访问数据放到 redis ,供应用频繁查询,减小数据压力。
  • 管理维护
    • 提供内网接口,供其他系统扩展 IM 功能;
    • 提供监控和维护工具,方便系统维护和故障处理。

ejabberd 架构

消息流转过程为:

  • 客户端通过长连接,将消息发送给服务器,进入到负责该连接的ejabberd_c2s进程;
  • ejabberd_c2s 进程处理完之后,调动ejabberd_router:route(From,To,Stanza)来路由该消息;
  • ejabberd_router处理自己的公共逻辑;
  • 然后如果是发给该 IM 系统的消息,会将消息发送给ejabberd_local进程;
  • ejabberd_local判断如果To是具体某个人,则会把消息发送给负责该用户连接的 ejabberd_sm进程;
  • ejabberd_sm进程会查询收消息的用户,有几个在线设备,然后将改消息发送给该用户的多个ejabberd_c2s进程;
  • ejabberd_c2s进程将消息通过自己负责的 TCP 连接,将消息发送给客户端。

这样,一条消息就完整的从用户 A 传输到了用户 B,实现消息的即时沟通。

ejabberd 功能扩展

在上一节 ejabberd 架构图中,我们可以在每个步骤中添加 hook 函数,添加自己的扩展功能。我们添加的扩展功能有:

使用 protocolbuffer 协议

由于 XMPP 协议具有:XML 报文流量大、耗电高等缺点,我们通过使用 protocolbuffer 替换掉 xml 报文,实现客户端到服务器之间的流量传输,减少报文流量和降低手机端的耗电量。在服务端将 protocolbuffer 再转换成 ejabberd 服务器使用的 xml 格式,减少服务端 im 逻辑的修改。

消息可靠性

通过以下渠道来确保 IM 消息的可靠性:

  • 设备在线时,通过消息确认回执来保证消息正确的发送出去和接收到;
  • 设备不在线时,当再次登录的时候,通过 HTTP 接口,拉取从上次退出到这次登录时间段内所有的历史消息;
  • 每条消息具有唯一 id,确保发送消息时候的幂等性,重复发送同一条消息,只会展示和存储一条消息。

消息确认回执

我们在 ejabberd_c2s进程收到消息的时候,添加一个 hook 函数,用来作为服务器对收到消息的确认,同时将时间戳也返回给客户端,作为该消息的时间戳。

消息确认回执,主要解决的是问题是:保证消息至少一次发送成功,只有客户端收到了服务器的确认回执,才会认为消息到达了服务器,得到了响应的处理;否则客户端就会认为消息没有到达服务器,发送失败了。

消息同步

当我们同一个账号同时登录多个设备的时候,需要感知到在其他设备收发的消息,并同步展示给用户。所以我们在收发消息时,需要做必要的处理,以实现多设备同步的功能。

同步发送消息给其他设备:

发布消息到消息队列

为了扩展 IM 功能,我们需要把所有的 IM 的消息和时间发布到消息队列,供其他系统订阅消费,实现消息的统计、分析和存储。所以我们将时间和消息分类放入到 kafka,实现消息的异步处理。

目前发送发送的包括:消息、上下线事件、驼圈事件以及 @事件等

发送消息的 http 接口

为了给其他系统提供发送消息的服务,我们通过提供 http 接口的方式,来模拟来自用户的消息,实现该功能。

下发 IM 认证凭证

我们可以在 IM 里嵌入其它系统,来扩展 IM 的能力,包括但不限于移动 OA 、运维报警等系统。IM 客户端可以在跳转到其它系统的时候,带上 IM 的认证凭证,由其它系统来调用 IM 接口来认证身份。如果认证通过则表明是通过正在登录的 IM 客户端访问的,否则不允许访问该系统。这样就避免了让用户重复的认证身份,提高办公效率和用户体验。

IM 认证凭证的流程是:

  • 当 IM 长连接建立成功且认证通过后,服务器会通过长连接下发 token 给客户端
  • 客户端请求 IM 的 HTTP 接口的时候要带着 token ,用于身份认证
  • 客户端打开其它受信系统的时候,也会带上 token ,用于其它系统做身份认证
  • 当客户端与服务器的长连接断开的时候,服务器会销毁该 token ,使其失效。

增量拉取

在一些客户端和服务器之间需要同步的数据拉取上,我们采用增量更新的逻辑,减少每次服务器的响应数据集,加快客户端的登录和同步流程。

实现方式上,IM 采用以更新时间作为查询的 key ,服务器在每次更新数据的时候,都要更新 updatetime 字段。在客户端再次登录的时候,使用本地最新的更新时间去拉取数据,服务器如果存在比客户端更新的数据,则将增量的数据返回给客户端。

应用的场景有:

  • 组织架构更新
  • 消息历史更新
  • 群列表更新
  • 好友列表更新
  • 个人配置更新

IM 功能扩展

机器人实现

由于我们需要通过 IM 实现一些自助服务或者智能回复,我们需要在 IM 的扩展上实现该功能。

首先我们已经有了所有消息的队列服务功能,然后我们基于消息队列,订阅所有的消息,然后根据系统配置,将发送给特定机器人的消息,转发给对应的机器人服务。机器人服务收到发给自己的消息时,通过自己的系统配置或者自主学习的问题库,进行对应的操作,并调用 IM 的发消息接口,返回给咨询用户特定的消息。我们可以通过该方式,实现各种自助和智能化的服务。节省人力成本和提高办公效率。

客服系统实现

对于客服系统,和普通 IM 有一些不同之处。在用户看来,他是在和一个店铺或者一个官方客服在聊天。实际上,后面可能是多个不同的客服,可能还会用到排队、会话超时等逻辑,所以要在常用的 IM 功能上来做扩展。

客服系统订阅所有的 IM 消息,当用户发送消息给客服的时候,客服系统需要对咨询做排队,客服分配,会话建立,然后将用户发给客服的消息转换成发给具体某一个客服的消息,然后发送给客服。

用户-------------> 店铺     转换成     店铺-----------> 客服
客服-------------> 店铺     转换成     店铺-----------> 用户

数据指标

对于 IM 主要的指标,我们主要关注的有:

  • 同时在线数
  • 建立 TCP 的量
  • 消息量

下面是对应指标的实际数值

  • 同时在线数: 20W 左右
  • 建立 TCP 的量( QPS ):3W 左右
  • 收到消息的量( QPS ):3W 左右
  • 发出消息的量( QPS ):3W 左右

总结

通过实现基本的 IM 功能,以及各种扩展功能,我们总结出一些 IM 核心功能:

  • 提供稳定的 TCP 长连接服务
  • 提供统一的认证服务
  • 提供高性能的消息订阅和发送消息给其他服务的能力

相关链接:

参考链接:

相关 [即时通讯 协议 qunar] 推荐:

即时通讯协议之Qunar

- - 标点符
Qunar 由于业务上对 IM 系统的需求,以及对 IM 需要支持的功能和扩展,结合市面上已有的 IM 的实现,实现了自己的一套完善的办公 IM 和客服 IM 系统. 具备了以下几个重要特点:实时性,可靠性,一致性,安全性,扩展性,高并发. Startalk是去哪儿开源的一款通用的,高性能的企业级im套件.

基于XMPP协议的手机多方多端即时通讯方案

- 山石 - 博客园-首页原创精华区
基于XMPP协议的手机多方多端即时通讯方案. 基于XMPP协议的手机多方多端即时通讯方案. 3、       为什么选择XMPP协议. 5、 Xmpp提供电子名片协议. 1、       什么是Openfire. 2、       为什么使用Openfire. 3、       Windows下搭建服务器.

Python 抓取 qunar 最低机票

- diaoxsh - python.cn(jobs, news)
# 感谢 [email protected] 投递. 小白我今天为了拿刚入门的python练手,写了个比较有实用价值的脚本,可以帮大家省钱了. 主要是通过(http://woodpecker.org.cn/diveintopython3/xml.html )清楚了做法.. context = urllib.urlopen('http://ws.qunar.com/holidayService.jcp?lane=上海-长沙').

PgVector在Qunar&途家的运维实践

- -
一、  为什么需要向量数据库. 随着AI相关技术的发展尤其是大语言模型(LLM)的广泛应用,海量的非结构化数据随之而来,如何存储以及高效检索这些数据成为热点问题,在此背景下AI时代的DB基座——向量数据库便应运而生了. 向量数据库支持存储AI算法经过Embedding后产生的向量类型数据,通过索引技术和向量相似度距离查询方法来支持向量数据的高效检索,解决了AI领域对于向量数据存储和高效检索的问题.

企业即时通讯市场研究

- chris - 月光博客
  企业即时通信,简称EIM(Enterprise Instant Messaging),它是一种面向企业终端使用者的网络沟通工具服务,使用者可以通过安装了即时通信的终端机进行两人或多人之间的实时沟通. 交流内容包括文字、界面、语音、视频及文件互发等.   根据国外调研机构eMarketer数据显示,截止到2010年底,企业即时通讯市场规模将达到6.88亿美元.

即时通讯解决方案参考

- - CSDN博客Web前端推荐文章
方案1、使用GCM服务(Google Cloud Messaging). 简介:Google推出的云消息服务,即第二代的C2DM. 优点:Google提供的服务、原生、简单,无需实现和部署服务端. 缺点:Android版本限制(必须大于2.2版本),该服务在国内不够稳定、需要用户绑定Google帐号,受限于Google.

如何保障即时通讯的安全性

- 老杜 - apple4us
接着上次谈电邮安全的问答,讲讲即时通讯(IM)的安全问题. 这会是一个系列,本篇的主答题者是 Rio,并综合了读者在留言中的意见. 一、QQ, MSN, GTalk, Skype,哪个最安全,哪个最不安全. 答:因为众所周知的原因,最不靠谱的当然是 QQ,所以如果你有秘密或者艳照之类,千万不要通过 QQ 传递.

联通即时通讯“沃友”今日上线 首推Android版

- franklee - cnBeta.COM
8月5日消息,中国联通“沃友”Android校园版和PC客户端于今日正式发布,下载地址为(im.wo.com.cn). “沃友”由中国联通子公司联通宽带在线有限公司负责运营,定位为一款面向互联网和手机用户,跨运营商、跨平台的即时通信软件. 用户安装 “沃友”后,客户端将扫描分析用户的手机通信录,帮助用户寻找手机通信录中已经使用“沃友”的朋友.

三星在Android Market发布移动即时通讯服务ChatOn

- Elic - cnBeta.COM
三星今天向Android Market投放了免费移动即时通讯服务ChatOn的客户端. 除三星Bada系统外,ChatOn还兼容Android、iOS等主要手机操作系统. ChatOn适用于所有主要智能机和功能手机平台,可以让用户在不受系统平台限制的情况下与朋友、家人进行互动,进行一对一对话、群聊、群发信息,共享图 片、视频、语音信息和联系人.

文本即时通讯动了移动运营商的蛋糕

- - Solidot
移动平台上的文本即时通讯软件WhatsApp、微信、Kakao Talk和Line,正日益取代短信成为数亿人不可或缺的通讯工具. 文本即时通讯的费用主要来自使用的流量,相比短信要低得多,因此严重冲击了移动运营商. 用户收发一条短信可能需要支付高达20美分的费用,而运营商的成本仅为0.01美分. 根据市场研究公司Ovum的数据,截止2012年底,文本即时通讯应用程序的崛起已导致运营商损失了230亿美元的收入.