得物自研客服 IM 中收发聊天消息背后的技术逻辑和思考实现
本文由得物技术 WWQ 分享,原题“客服发送一条消息背后的技术和思”,本文有修订和改动。
1、引言
在企业 IM 客服场景中,客服发送一条消息的背后,需要考虑网络通信、前端展示、后端存储以及安全性等多个方面的技术支持。单从前端层面来说,就需要考虑到消息的显示、状态更新、稳定传输以及极限操作消息不卡顿等场景。随着 IM 系统的不断更新迭代,已经实现了从外采到自研再到一站式全场景工作台的搭建,我们能够很明显地感知到客服对于 IM 的体验要求越来越高了,因此客服发送一条消息背后所涉及的技术和思考也越来越重要。
本文将探秘得物自研客服 IM 中收发聊天消息背后的技术逻辑和思考实现,帮助大家了解如何在 IM 聊天场景中提供高效、安全、可靠和良好的用户体验。
技术交流:
- 移动端 IM 开发入门文章:《新手入门一篇就够:从零开发移动端IM》
- 开源 IM 框架源码:https://github.com/JackJiang2011/MobileIMSDK(备用地址点此)
(本文已同步发布于:http://www.52im.net/thread-4483-1-1.html)
2、相关文章
3、IM 聊天消息的重要性
IM 聊天消息是客服和用户之间最快速、最直观、最高效的双向沟通方式之一。
IM 聊天的重要性体现在以下几个方面:
1)即时响应:及时地解答用户咨询的问题,更快捷的服务用户,提高用户满意度;
2)个性化互动:可以根据用户的需求快速做出个性化回应,从而更好地满足用户需求;
3)数据处理和分析:通过对 IM 聊天消息的处理分析,可以洞察用户需求、用户行为,帮助改进服务质量。
综上:IM 聊天消息的重要性在于提高用户满意度、提高客服作业效率,这也意味着 IM 消息的可靠、高效、安全尤为重要,接下来本文就从前端视角对客服发送一条消息背后的技术和思考进行详细的讲述。
4、客服 IM 消息发展历程
以下是得物客服 IM 消息发展的历程,列举的都是核心技术专项的里程碑节点。
在这个过程中,我们积累了一定的经验和技能,同时也遇到了各种各样的问题和挑战。比如:消息丢失、消息发送失败、消息重复、消息乱序等等方面的问题。
针对这些问题我们也都通过技术专项的方式去逐个解决并达到了预期效果,我们相信,随着技术的不断发展和创新,我们可以更好地提供更加高效便捷的服务。
5、技术逻辑和思考
站在用户/客服角度,发送消息不就是输入消息后点击回车键或点击发送按钮就完成了吗。
看似非常简单,但是从开始输入消息到对方收到消息这个过程实际上有非常强大的技术在高效、稳定支撑。
我们客服 IM 消息链路会涉及到三个核心端口:
1)发出方
2)IM 网关;
3)接收方。
以下将以客服发送一条消息到 IM 网关这个过程简单描述一下涉及到的技术点,反之用户侧发送消息也是类似的。
从上述流程图中可以看到:一条消息的旅程还是非常丰富的,当然其中有一些细节点还没有完全列举出来。
例如:IM 网关的超时重推机制、前端的异常处理(网络异常、超时异常、重试无果等)。我们可以很清晰地看到当客服开始输入消息的时候就开始进行通知对方正常输入,触发消息发送后需要进行消息体的创建、排序、去重检测、网络检测、聊天列表渲染、推入超时重试队列、放入消息拦截器中统一进行消息格式转化并发送。
到这里只仅仅是完成了前端层面的发送工作而已,此时消息是否发送成功还是未知的,还需要监听消息的发送结果。如果在一定时间未收到响应结果会进行第二次消息的重发,直到发送成功或到达最大重试次数就表示该消息的生命周期结束。
一旦收到消息的响应结果就会对消息的状态进行更新(此时消息已完成了排序,不需要进行二次排序),至此第一个环节就完成了处理,IM 网关到客户端也会有类似的处理过程。
纵观整个消息发送以及接收链路,任何一个环节出现问题都会导致消息发送出现问题,就需要非常稳定可靠的技术手段进行保障,主要从以下几个方面讲解一下。
6、消息的可靠性传递
6.1 概述
消息的可靠性传递确保了消息收发双方信息的一致性。这也是我们为什么把消息可靠性传递放在第一个进行讲解。
我们试想一下这样一个场景:经常有消息丢失,客服频繁反馈,每次都要投入研发资源去排查问题。这还是次要的,有可能因为消息的丢失导致用户体验的急剧下降,这就得不偿失了。所以消息的可靠性传递是非常有必要的,而且也是必须的。
那么何为可靠性传递?至少要满足 3 个方面,我们进行详细分享。
6.2 消息的实时性
我们使用 IM 最重要的一方面就是希望对方能够实时接收到我们发送的消息并能够给予回复,这对于提升用户体验尤为重要。如果不在乎实时性我们完全可以使用其他方式,例如邮件、写信甚至飞鸽传书…
一条消息发送给 IM 网关,网关大致需要经历以下 5 个环节的处理:
1)验证消息:敏感词验证、风控送审(同步审核);
2)消息的存储:排序、去重验证等;
3)给发送消息方回复一个 ACK 响应(成功、失败);
4)把消息发送给接收方,如果存在多端登录的场景,还需要保障消息多端同步;
5)超时重试、处理接收方返回的 ACK 等。
从消息的实时性的来说,没有绝对的实时,只能尽量优化。核心的处理逻辑都在 IM 网关,无论是前端还是客户端,处理过程都是非常快的,都在毫秒级别。
我们 IM 网关是 Go 语言开发的,并发处理的能力也是非常高的,所以整个闭合链路的耗时还是非常低的。
6.3 消息的可靠性
众所周知,TCP 本身就是具有可靠性的,但是它只能保障传输层可靠,而应用层之间的可靠性并不能保证(可详读《为何基于TCP协议的移动端IM仍然需要心跳保活机制?》、《从客户端的角度来谈谈移动端IM的消息可靠性和送达机制》、《不为人知的网络编程(十二):彻底搞懂TCP协议层的KeepAlive保活机制》)。我们后续会有针对性的专项文章进行发表,本次就不再赘述。
那我们该如何保障应用之间的可靠性呢?
可靠性的保障就是让发送方知道接收方接收到了消息,这样就表示消息成功传递了。
我们再回头看一下上面讲述消息丢失的场景,消息丢失的问题也是我们在 IM 消息研发过程中遇到的一个让人头疼的问题,排查一个问题需要投入的技术资源是非常巨大的,需要涉及到 H5、IM 网关、服务端以及客户端,对于用户以及客服的使用体验是非常差的。很简单的一个场景,用户发了消息,客服没有收到,没有回复用户,用户以为客服故意不回复,会影响到用户的满意度。
那这个问题该如何解决呢?
大家可以看下《得物从0到1自研客服IM系统的技术实践之路》,其中有讲解过,核心是参考 TCP 协议的 ACK 机制,实现一套基于业务层的 ACK 协议。
这里特别的要注意的是针对批量消息(客服刷新会话、新会话进线等场景),我们采用的是批量 ACK 机制,如果每一个消息都回复 ACK,成本会比较高。我们当初是通过一个 IM 架构升级技术专项协同各端完成了 IM 整体消息触达实现 0 丢失,保证触达,满足 At least once(通过数据埋点验证后得到 100%的触达率)。上线后该场景符合预期效果,相应的问题排查投入也减少了至少 70%+。
6.4 消息的有序性
在开发 IM 过程中有这样一个非常常见的场景,用户问 A 问题后又问题了 B 问题,在客服侧 B 问题排到 A 问题的前面,导致客服的回复也出现了错乱。
当然这只是 IM 消息乱序的一种场景而已。诸如此类的还有很多。
消息乱序产生的原因有很多,例如发送文件后再立即发送消息,文件需要前端先上传到 OSS 获取到 URL 后再发送给用户,上传文件这个过程,用户以及客服都是可以发送消息的,这种场景处理不好就极易出现消息乱序。
不做 IM 是真不会想到客服操作的效率会有多高,之前在处理消息乱序问题的时候有遇到客服连续发送了 2 条消息,间隔只有 300 毫秒,这种高频密集的操作场景在客服的工作场景下是持续性的。
看似一个乱序问题,不考虑清楚用户群体、极限场景、临界值等都不会彻底解决掉这个问题。
再说回我们客服 IM,我们是如何处理消息排序的呢?
在整个开发过程也是比较曲折的,最终是以 IM 网关维护的 Seq 为准,然后返回到发送方,发送再根据消息序号进行排序,确保发送方和接收方消息的排序是一致的。
前端处理的流程如下:
6.5 消息的幂等性
说到消息的幂等性,我们要思考一个问题,为什么会收到多条(>1)相同的消息呢?
肯定是发送方重复发送导致的,那在什么场景下会重复发送?
前面刚讲过应用层的 ACK 机制,如果没有收到对方的 ACK,会在超时时间到达后继续重复发送直到最大重试次数。参考下面的截图会更容易理解,只是模拟消息重试,真实场景中执行频次肯定要比这个时间更久一些。
既然要保证消息的可靠性,消息的重复就是无法避免的。就有可能出现消息幂等性问题。
那怎么解决呢?
我们是利用消息的 Message ID 做去重的,这里会涉及到一个性能问题,排序、去重以及风控信息验证等都需要一定的计算成本,如何保证处理过程系统不卡顿是一个核心问题。
想要了解我们客服 IM 是如何做的,请继续向下看。
7、消息处理的卡顿优化策略
7.1 概述
我们来想一下为什么会出现卡顿?
什么样的场景才能够被视为卡顿呢?
我们一般都会说是因为在 16ms 内无法完成渲染导致的。那么为什么需要在 16ms 内完成呢?
这里我们就要了解一下刷新率(RefreshRate)与帧率(FrameRate):
1)刷新率:指的是屏幕每秒刷新的次数,是针对硬件而言的。浏览器刷新率都在 60Hz(屏幕每秒钟刷新 60 次);
2)帧率:是每秒绘制的帧数,是针对软件而言的。通常只要帧率与刷新率保持一致,我们看到的画面就是流畅的。所以帧率在 60FPS 时我们就不会感觉到卡。
如果:帧率为每秒钟 60 帧,而屏幕刷新率为 30Hz,那么就会出现屏幕上半部分还停留在上一帧的画面,屏幕的下半部分渲染出来的就是下一帧的画面,这种情况被称为画面撕裂。
相反:如果帧率为每秒钟 30 帧,屏幕刷新率为 60Hz,那么就会出现相连两帧显示的是同一画面,这就出现了卡顿。所以单方面的提升帧率或者刷新率是没有意义的,需要两者同时进行提升。浏览器都采用的 60Hz 的刷新率,为了使帧率也能达到 60FPS,那么就要求在 16.67ms 内要完成一帧的绘制(1000ms/60Frame = 16.666ms / Frame)。
IM 消息处理中出现卡顿的情况非常常见,到一定的量级都是一个很难避免的问题。对比我们经常使用电脑,打开多个浏览器页签,稍微时间长点不关机重启,也会感觉到卡顿。但对于 IM 消息处理还是有很多方式进行优化的。
主要涉及以下几方面的优化策略,我来展开讨论。
7.2 异步处理
众所周知 JS 是单线程的,所以采用异步处理机制可以将优先级低的任务推入异步任务队列,让出主线程给优先级高的任务。
比如:客服在输入完消息后需要立即显示的聊天页面,如果存在短暂的不显示,会被认为是系统卡顿了,所以发送消息的优先级是高于接收消息的。
我们对各场景任务优先级做了区分,低优先级的任务都通过异步的方式进行处理。
7.3 分段加载
这里主要针对聊天消息列表,对于大量消息的会话处理,只渲染可视区域的消息降低浏览器的负担,提升响应速度。
列表优化的方案有很多,如下。
1)方案 1:
使用定时器 setTimeout 来实现分批渲染。
这种方式我们一般不推荐,因为在 setTimeout 中对 DOM 进行操作,必须要等到屏幕下次绘制时才能更新到屏幕上,如果两者步调不一致,就可能导致中间某一帧的操作被跨越过去,而直接更新下一帧的元素,从而导致丢帧现象。
2)方案 2:
采用 requestAnimationFrame。相比之下,requestAnimationFrame 的优势还是非常明显的。
主要体现在以下几个方面:
1)requestAnimationFrame 会把每一帧中的所有 DOM 操作集中起来,再一次重绘或回流中就完成,并且重绘或回流的时间间隔紧紧跟随浏览器的刷新频率;
2)在隐藏或不可见的元素中,requestAnimationFrame 将不会进行重绘或回流,这当然就意味着更少的 CPU、GPU 和内存使用量;
3)requestAnimationFrame 是由浏览器专门为动画提供的 API,在运行时浏览器会自动优化方法的调用,并且如果页面不是激活状态下的话,动画会自动暂停,有效节省了 CPU 开销;
4)与 setTimeout 相比,requestAnimationFrame 最大的优势是由系统来决定回调函数的执行时机;
5)requestAnimationFrame 的步伐跟着系统的刷新步伐走。它能保证回调函数在屏幕每一次的刷新间隔中只被执行一次,这样就不会引起丢帧现象。
3)方案 3:
采用 IntersectionObserver。
IntersectionObserver 接口(从属于 Intersection Observer API)为开发者提供了一种可以异步监听目标元素与其祖先或视窗(viewport)交叉状态的手段。祖先元素与视窗(viewport)被称为根(root)。
可以看到,交叉了就是说明当前元素在视窗里,当前就是可见的了。是代替监听滚动加载的不错方案。
当然还有其他方案,还是要根据实际的业务场景选择合适的方案,IM 消息分段加载的难点在于消息的不定高(多种不同类型的消息),计算成本还是有一些昂贵的。所以优化还是要验证一下临界值的,有时候优化不一定会有效。
7.4 消息遍历
上面我们讲到消息排序、去重以及消息状态更新等等,多个会话大量的聊天消息,如果处理不当,卡顿是必现的。
可以先看一下我们优化之前的处理流程,采用的是第三方的 SDK,一堆 for 循环,消息量大一些基本卡住没反应了。
那我们是如何处理这个问题的呢?
基于现有的业务场景重写三方 SDK,将会话维护成独立的实例,核心算法就是采用二分法。
感兴趣的同学可以看之前的这篇文章《得物从0到1自研客服IM系统的技术实践之路》,讲述得比较详细。重写了 IM SDK 之后,客服再也没有反馈过聊天相关的卡顿,聊天首响提升了 20%,成果还是比较显著的。
8、消息安全方面的考虑
在 IM 系统中,消息的安全性是非常重要,开发同学需要具备较强的安全意识,将安全融入到开发流程中,增强系统的安全性和健壮性。
消息安全性方面的事情我们做了很多,这里也不再详细讲解了,有兴趣可以读读下面的文章:
9、消息发送和接收的延迟
消息发送和接收的延迟直接影响用户的使用体验和沟通效率,在上面我们已经分析过一条消息的旅程,出现延迟的原因也比较好分析。
主要有以下 4 点:
1)网络延迟:IM 消息的发送和接收是以长链接的方式进行网络传输的,而网络传输过程中会产生一定的延迟。如果网络延迟高,就会导致消息发送和接收较慢;
2)系统负载:客服在一对多的情况下,多个用户同时在线,系统需要处理大量的消息和请求,导致系统响应速度较慢,这会对客服的体验造成影响;
3)前端延迟:需要经过本地消息队列、缓存等处理,可能导致消息的延迟;
4)消息编码和解码:部分消息需要对数据进行编码和解码,也会消耗一定的时间,从而导致延迟。
既然能分析出原因,我们就能对症下药,可以通过一些优化策略来降低发送和接收的延迟。
目前规划从以下 2 个方面来进行优化。
1)前端方面:
延迟主要在消息的处理和编解码方面,目前我们 IM 消息的数据格式是 JSON,存在序列化和反序列化的过程,这里我们会采用 ProtoBuf 替换 JSON,目前已完成了相关技术调研和测试验证。我们简单来看一下 ProtoBuf(Protocol Buffers)和 JSON 处理耗时的对比。
编码时间:ProtoBuf 的编码时间比 JSON 快得多,因为 ProtoBuf 的编码是二进制的,不需要进行编码转换以及无需进行冗余类型的转换。相对而言,JSON 的编码时间较慢。
解码时间:相比编码,ProtoBuf 的解码效率要稍微低一些。但是,由于 ProtoBuf 的优势在数据量大、结构复杂的情况下更为明显,对于小型数据解码时,两者的效率差异可能不太明显。
有关 Profobuf 的更多资料,可以详读以下文章:
2)网络延迟:
网络延迟我们很难控制,但是可以通过降低消息传输体积进行相关优化。
刚讲了 Protobuf 替换 JSON,Protobuf 是二进制格式,比 JSON 格式更加紧凑,能够使数据包大小大幅度减小,在网络传输中能够减少带宽占用和流量费用。
在 IM 系统中,由于用户数量庞大,消息发送频繁,在数据占用和网络带宽方面是一个巨大的问题,使用 ProtoBuf 能够显著地减少网络带宽消耗,提高系统的性能。
还有一方面就是消息压缩,但是压缩的深度和压缩算法需要慎重选择、验证。
所以使用 ProtoBuf 格式代替 JSON 格式基本可以解掉一大半延迟问题,也是接下来 IM 优化的一个方向。
10、坐席体验和交互的考虑
说到坐席体验和交互方面,我们还是积累了不少经验的。不仅仅是 IM,体验和交互是所有产品都无法绕开的一个话题。
自从做 IM 以来,体验可谓是鞭策我们不断前进的动力,卡顿是一直环绕在我耳边的一个话题。
客服理解的卡顿和我们正常理解的卡顿还是有点不一样的,前期我们也以为是系统卡住导致无法使用了,类似掉帧的场景。
实际却不是:
1)接口请求慢了;
2)有错误的 Tip 提示;
3)页面切换有短暂空白显示;
4)输入消息回车后消息未立刻显示到聊天页面;
5)图片上传的 Loading 提示等等。
以上都会被归为卡顿。
针对这些方面:我们也是不断的进行职场调研、数据分析、优化,客服的满意度提升到了 18%。可能在大家看来做了这么久提升 18%并不是一个比较好的数据,但是针对客服域,提升 18%也是一个相对比较难逾越的数据了。
主要的原因在 2 个方面:
1)第一个方面是很多客服都是 3 个月以内入职的,对于我们做的一些功能优化对比体验是无法感知或缺少功能使用对比的;
2)第二个方面是很多一线客服都来自一线大厂的客服服务团队。
其实反过来想一下,这也是一种正向的驱动,至少我们每次调研都能收集到新的反馈,同更加成熟、优秀产品的体验差距。
体验不是一蹴而就的,不要想着一下子就做到位,一个优秀的用户体验和交互设计需要始终与用户需求和反馈相结合,并不断改进和完善。在实际设计和开发过程中,需要进行不断的测试和优化,以确保系统的质量和可接受性。同时,需要与用户进行积极的沟通和反馈,以便更好地理解用户需求和意见。这一点我们之前是做的不够好的,尤其是新版本的推广,系统的易用性并未达到客服的期望,也是我们后期需要持续改进的一个方面。
体验是以绝大数用户需求为核心的,不能仅仅为了一小部分用户而去牺牲其他用户的使用体验,尤其不能因为某一个用户的反馈意见而做出过多的改变或者牺牲其他用户的利益。体验优化过程的不妥协也是非常重要的策略,在体验优化过程中,必须保持理性和客观,根据用户调研和数据分析进行合理的权衡和决策,以实现最佳的用户体验。
一些小细节的优化也可以起到事半功倍的效果,在 IM 系统中,一些细节的优化包括:及时的消息提示、清晰的消息展示、精确的消息发送时间等等。这些小细节的优化可以直接提高客服的使用效率和体验,从而提高客服满意度。IM 的体验优化我们会一直做下去,有志者事竟成。
11、后续规划
上述技术和思考的细节中有讲到消息的可靠性传递、卡顿优化处理、安全性、效率以及体验等。
接下来的一段时间我们还是以这几个方面为主线进行,持续优化、完善 IM 相关能力。
主要考虑以下几个方面的规划:
1)体验优化:体验是我们一如既往要做的事情,会持续挖掘视觉、交互等层面的优化点,从细节入手,比如:颜色搭配,按键选择等,提供良好的坐席体验;
2)ProtoBuf 替换 JSON:降低消息编码时间、提升解码效率、减少数据包体积、减少网络带宽消耗,提高系统的性能;
3)消息压缩:尤其是针对历史消息、批量消息,使用压缩技术,可以有效的减少数据包的体积;
4)功能扩展:持续完善机器人消息类型,尤其是针对售前导购、坐席辅助。逐步支持消息引用、标记等功能;
5)多语言能力支持:虽然目前还没有接入国际化业务,但在设计层面还是要具备快速扩展的能力。
上述几个方面我们会优先去做重要且紧急的技术改造,并不会一味的创新、优化,还是会以业务为主,紧紧围绕业务和坐席体验展开。
12、本文小结
客服发送一条消息在 IM 应用中看似简单,背后需要考虑的技术细节点是很多的。首先,这需要考虑到消息的发送机制和可靠性。即使是一条简单的消息,也需要经过一系列的加密、编码、传输、安全合规等等处理才能被成功接收。
最重要的是要考虑到数据实时性的问题,各种极限场景下的操作,客服发送的消息需要被及时展示到聊天页并传输给用户,客服同学在一对多的场景下工作,需要确保各会话消息不会出现不一致(丢失、重复),还有消息拦截和异常情况等问题。
因此,客服发送一条消息不仅需要技术能力和数据处理能力,还需要思考坐席体验和数据实时性等方面的问题。开发过程中需要细致入微地处理各种问题并持续优化,从而为客服提供一个稳定、流畅、安全、友好的 IM 应用。
13、参考文章
[1] 得物基于Electron开发客服IM桌面端的技术实践
[3] 为何基于TCP协议的移动端IM仍然需要心跳保活机制?
[4] 从客户端的角度来谈谈移动端IM的消息可靠性和送达机制
[5] 不为人知的网络编程(十二):彻底搞懂TCP协议层的KeepAlive保活机制
[6] 微信新一代通信安全解决方案:基于TLS1.3的MMTLS详解
[7] 基于Netty的IM聊天加密技术学习:一文理清常见的加密概念、术语等
[8] IM通讯协议专题学习(一):Protobuf从入门到精通,一篇就够!
[9] IM通讯协议专题学习(二):快速理解Protobuf的背景、原理、使用、优缺点
[10] IM通讯协议专题学习(三):由浅入深,从根上理解Protobuf的编解码原理
[12] IM消息送达保证机制实现(一):保证在线实时消息的可靠投递
[13] IM消息送达保证机制实现(二):保证离线消息的可靠投递
[15] 一套亿级用户的IM架构技术干货(下篇):可靠性、有序性、弱网优化等
(本文已同步发布于:http://www.52im.net/thread-4483-1-1.html)
评论