一、开篇
自从SkyWalking开始在公司推广,时不时会在排查问题的人群中听到这样的话:“你咋还没接SkyWalking?接入后,一眼就看出是哪儿的问题了…",正如同事所说的,在许多情况下,SkyWalking就是这么秀。作为实践者,我非常感谢SkyWalking,因为这款国产全链路监控产品给公司的伙伴们带来了实实在在的帮助;也特别感谢公司的领导和同事们,正因为他们的支持和帮助,才让这套SkyWalking(V8.5.0)系统从起初的有用进化到现在的好用;从几十亿的Segment储能上限、几十秒的查询耗时,优化到千亿级的Segment储能、毫秒级的查询耗时。
小提示:
SkyWalking迭代速度很快,公司使用的是8.5.0版本,其新版本的性能肯定有改善。
Segment是SkyWalking中提出的概念,表示一次请求在某个服务内的执行链路片段的合集,一个请求在多个服务中先后产生的Segment串起来构成一个完整的Trace,如下图所示:
SkyWalking的这次实践,截止到现在有一年多的时间,回顾总结一下这段历程中的些许积累和收获,愿能反哺社区,给有需求的道友提供个案例借鉴;也希望能收获到专家们的指导建议,把项目做得更好。因为安全约束,要把有些内容和谐掉,但也努力把这段历程中那些靓丽的风景,尽可能完整地呈现给大家。
二、为什么需要全链路监控
随着微服务架构的演进,单体应用按照服务维度进行拆分,组织架构也随之演进以横向、纵向维度拆分;一个业务请求的执行轨迹,也从单体应用时期一个应用实例内一个接口,变成多个服务实例的多个接口;对应到组织架构,可能跨越多个BU、多个Owner。虽然微服务架构高内聚低耦合的优势是不言而喻的,但是低耦合也有明显的副作用,它在现实中给跨部门沟通、协作带来额外的不可控的开销;因此开发者尤其是终端业务侧的架构师、管理者,特别需要一些可以帮助理解系统拓扑和用于分析性能问题的工具,便于在架构调整、性能检测和发生故障时,缩减沟通协作方面的精力和时间耗费,快速定位并解决问题。
我所在的平安健康互联网股份有限公司(文中简称公司),是微服务架构的深度实践者。公司用互联网技术搭建医疗服务平台,致力于构筑专业的医患桥梁,提供专业、全面、高品质、一站式企业健康管理服务。为了进一步提高系统服务质量、提升问题响应效率,部门在21年结合自身的一些情况,决定对现行的全链路监控系统进行升级,目的与以下网络中常见的描述基本一致:
快速发现问题
判断故障影响范围
梳理服务依赖并判断依赖的合理性
分析链路性能并实施容量规划
三、为什么选择SkyWalking
在做技术选型时,网络中搜集的资料显示,谷歌的Dapper系统,算是链路追踪领域的始祖。受其公开论文中提出的概念和理念的影响,一些优秀的企业、个人先后做出不少非常nice的产品,有些还在社区开源共建,如:韩国的Pinpoint,Twitter的Zipkin,Uber的Jaeger及中国的SkyWalking等,我司选型立项的过程中综合考虑的因素较多,这里只归纳一下SkyWalking吸引我们的2个优势:
1)产品的完善度高:
java生态,功能丰富
社区活跃,迭代迅速
2)链路追踪、拓扑分析的能力强:
插件丰富,探针无侵入
采用先进的流式拓扑分析设计
“好东西不需要多说,实际行动告诉你“,这句话我个人非常喜欢,关于SkyWalking的众多的优点,网络上可以找到很多,此处先不逐一比较、赘述了。
四、预研阶段
当时最新版本8.5.0,梳理分析8.x的发布记录后,评估此版本的核心功能是蛮稳定的,于是基于此版本开始了SkyWalking的探索之旅。当时的认知是有限的,串行思维模型驱使我将关注的问题聚焦在架构原理是怎样、有什么副作用这2个方面:
1)架构和原理:
agent端 主要关注 Java Agent的机制、SkyWalking Agent端的配置、插件的工作机制、数据采集及上报的机制。
服务端 主要关注 角色和职责、模块和配置、数据接收的机制、指标构建的机制、指标聚合的机制及指标存储的机制。
存储端 主要关注 数据量,存储架构要求以及资源评估。
2)副作用:
功能干扰
性能损耗
1、架构和原理
SkyWalking社区很棒,官网文档和官方出版的书籍有较系统化的讲解,因为自己在APM系统以及Java Agent方面有一些相关的经验沉淀,通过在这两个渠道的学习,对Agent端和OAP(服务端)很快便有了较系统化的认知。在做系统架构选型时,评估数据量会比较大(成千上万的JVM实例数,每天采集的Segment数量可能是50-100亿的级别),所以传输通道选择Kafka、存储选择Elasticsearch,如此简易版的架构以及数据流转如下图所示:
这里有几处要解释一下:
Agent上报数据给OAP端,有grpc通道和kafka通道,当时就盲猜grpc通道可能撑不住,所以选择kafka通道来削峰;kafka通道是在8.x里加入的。
千亿级的数据用ES来做存储肯定是可以的。
图中L1聚合的意思是:SkyWalking OAP服务端 接收数据后,构建metric并完成metric 的Level-1聚合,这里简称L1聚合。
图中L2聚合的意思是:服务端 基于metric的Level-1聚合结果,再做一次聚合,即Level-2聚合,这里简称L2聚合。后续把纯Mixed角色的集群拆成了两个集群。
2、副作用
对于质量团队和接入方来说,他们最关注的问题是,接入SkyWalking后:
是否对应用有功能性干扰
在运行期能带来哪些性能损耗
这两个问题从3个维度来得到答案:
1)网络资料显示:
Agent带来的性能损耗在5%以内
未搜到功能性干扰相关的资料(盲猜没有这方面问题)
2)实现机制评估:
字节码增强机制是JVM提供的机制,SkyWalking使用的字节码操控框架ByteBuddy也是成熟稳定的;通过自定义ClassLoader来加载管理插件类,不会产生冲突和污染。
Agent内插件开发所使用的AOP机制是基于模板方法模式实现的,风控很到位,即使插件的实现逻辑有异常也不影响用户逻辑的执行。
插件采集数据跟上报逻辑之间用了一个轻量级的无锁环形队列进行解耦,算是一种保护机制;这个队列在MPSC场景下性能还不错;队列采用满时丢弃的策略,不会有积压阻塞和OOM。
3)性能测试验证:
测试的老师针对dubbo、http这两种常规RPC通信场景,进行压力测试和稳定性测试,结果与网络资料描述一致,符合预期。
五、POC阶段
在POC阶段,接入几十个种子应用,在非生产环境试点观察,同时完善插件补全链路,对接公司的配置中心,对接发布系统,完善自监控,全面准备达到推广就绪状态。
1、对接发布系统
为了对接公司的发布系统,方便系统的发布,将SkyWalking应用拆分为4个子应用:
这里有个考虑,暂定先使用纯Mixed角色的单集群,有性能问题时就试试 Receiver+Aggregator双角色集群模式,最终选哪种视效果而定。
SkyWalking Agent端是基于Java Agent机制实现的,采用的是启动挂载模式;启动挂载需在启动脚本里加入挂载Java Agent的逻辑,发布系统实现这个功能需要注意2点:
启动脚本挂载SkyWalking Agent的环节,尽量让用户无感知。
发布系统在挂载Agent的时候,给Agent指定应用名称和所属分组信息。
SkyWalking Agent的发布和升级也由发布系统来负责;Agent的升级采用了灰度管控的方案,控制的粒度是应用级和实例级两种:
按照应用灰度,可给应用指定使用什么版本的Agent
按照应用的实例灰度,可给应用指定其若干实例使用什么版本的Agent
2、完善插件补全链路
针对公司OLTP技术栈,量身定制了插件套,其中大部分在开源社区的插件库中有,缺失的部分通过自研快速补齐。
这些插件给各组件的核心环节埋点,采集数据上报给SkyWalking后,Web端的【追踪】页面就能勾勒出丰满完美的请求执行链路;这对架构师理解真实架构,测试同学验证逻辑变更和分析性能损耗,开发同学精准定位问题都非常的有帮助。这里借官方在线Demo的截图一用(抱歉后端程序员,*特效都没做出来,丰满画面还请自行脑补)
友情小提示:移除不用的插件对程序编译打包和减少应用启动耗时很有帮助。
3、压测稳测
测试的老师,针对SkyWalking Agent端的插件套,设计了丰富的用例,压力测试和稳定性测试的结果都符合预期;每家公司的标准不尽一致,此处不再赘述。
4、对接自研的配置中心
把应用中繁杂的配置交给配置中心来管理是非常必要的,配置中心既能提供启动时的静态配置,又能管理运行期的动态配置,而且外部化配置的机制特别容易满足容器场景下应用的无状态化要求。啰嗦一下,举2个例子:
调优时,修改参数的值不用来一遍开发到测试再到生产的发布。
观测系统状态,修改日志配置后不需要来一遍开发到测试再到生产的发布。
Skywaling在外接配置中心这块儿,适配了市面中主流的配置中心产品。而公司的配置中心是自研的,需要对接一下,得益于SkyWalking提供的模块化管理机制,只用扩展一个模块即可。
在POC阶段,梳理服务端各模块的功能,能感受到其配置化做的不错,配置项很丰富,管控的粒度也很细;在POC阶段几乎没有变动,除了对Webapp模块的外部化配置稍作改造,与配置中心打通以便在配置中心管理Webapp模块中Ribbon和Hystrix的相关配置。
5、完善自监控
自监控是说监控SkyWalking系统内各模块的运转情况:
完善自监控后的架构如下图所示:
6、自研Native端SDK
公司移动端的应用很核心,也要使用链路追踪的功能,社区缺了这块,于是基于SkyWalking的协议,移动端的伙伴们自研了一套SDK,弥补了Native端链路数据的缺失,也在后来的秒开页面指标统计中发挥了作用。随着口口相传,不断有团队提出需求、加入建设,所以也在持续迭代中;内容很多,这里先不展开。
7、小结
POC阶段数据量不大,主要是发现系统的各种功能性问题,查缺补漏。
六、优化阶段
SkyWalking的正式推广采用的是城市包围农村的策略;公司的核心应用作为第一批次接入,这个策略有几个好处:
核心应用的监管是重中之重,优先级默认最高。
核心应用的上下游应用,会随着大家对SkyWalking依赖的加深,而逐步自主接入。
当然安全是第一位的,无论新系统多好、多厉害,其引入都需遵守安全稳定的前提要求。既要安全又要快速还要方便,于是基于之前Agent灰度接入的能力,在发布系统中增加应用Owner自助式灰度接入和快速卸载SkyWalking Agent的能力,即应用负责人可自主选择哪个应用接入,接入几个实例,倘若遇到问题仅通过重启即可完成快速卸载;这个能力在推广的前期发挥了巨大的作用;毕竟安全第一,信任也需逐步建立。
随着应用的接入、使用,我们也逐渐遇到了一些问题,这里按照时间递增的顺序将问题和优化效果快速的介绍给大家,更多技术原理的内容计划在公众号『架构染色』专辑【SkyWalking调优系列】补充。开始之前有几个事项要说明:
下文中提到的数字仅代表我司的情况,标注的Segment数量是处理这个问题的那段时间的情况,并不是说达到这个数量才开始出现这个现象。
这些数值以及当时的现象,受到宿主机配置、Segment数据的大小、存储处理能力等多种因素的影响;请关注调整的过程和效果,不必把数字和现象对号入座哈。
1、启动耗时
1)问题
有同事反馈应用启动变慢,排查发现容器中多数应用启动的总耗时,在接入SkyWalking前是2秒,接入后变成了16秒以上,公司很多核心应用的实例数很多,这样的启动损耗对它们的发布影响太大。
2)优化
记录启动耗时并随着其他启动数据上报到服务端,方便查看对比。
优化Kafka Reporter的启动过程,将启动耗时减少了3-4秒。
优化类匹配和增强环节(重点)后,容器中的应用启动总耗时从之前16秒以上降低到了3秒内。
梳理Kafka 启动和上报的过程中,顺带调整了Agent端的数据上报到kafka的分区选择策略,将一个JVM实例中的数据全部发送到同一个的分区中,如此在L1层的聚合就完成了JVM实例级的Metric聚合,需注意调整Kafka分片数来保证负载均衡。
2、kafka积压-6亿segment/天
1)问题
SkyWalking OAP端消费慢,导致Kafka中Segment积压。未能达到能用的目标。
2)优化
从SkyWalking OAP端的监控指标中没有定位出哪个环节的问题,把服务端单集群拆为双集群,即把 Mixed角色的集群 ,修改为 Receiver 角色(接收和L1聚合)的集群 ,并加入 Aggregation角色(L2聚合)的集群,调整成了双集群模式,数据流传如下图所示:
3、kafka积压-8亿segment/天
1)问题
SkyWalking OAP端消费慢,导致Kafka中Segment积压,监控指标能看出是在ES存储环节慢,未能达到能用的目标。
2)优化
优化segment保存到ES的批处理过程,调整BulkProcessor的线程数和批处理大小。
优化metrics保存到ES的批处理过程,调整批处理的时间间隔、线程数、批处理大小以及刷盘时间。
4、kafka积压-20亿segment/天
1)问题
Aggregation集群的实例持续Full GC,Receiver集群通过grpc给Aggregation集群发送metric失败。未能达到能用的目标。
2)优化
增加ES节点、分片,效果不明显。
ES集群有压力,但无法精准定位出是什么数据的什么操作引发的。采用分治策略,尝试将数据拆分,从OAP服务端读写逻辑调整,将ES单集群拆分为 trace集群 和 metric集群;之后对比ES的监控指标明确看出是metric集群读写压力太大。
优化Receiver集群metric的L1聚合,完成1分钟的数据聚合后,再提交给Aggregation集群做L2聚合。
Aggregation集群metric的L2 聚合是基于db实现的,会有 空读-写-再读-累加-更新写 这样的逻辑,每次写都会有读,调整逻辑是:提升读的性能,优化缓存机制减少读的触发;调整间隔,避免触发累加和更新。
将metric批量写ES操作调整成BulkProcessor。
ES的metric集群使用SSD存储,增加节点数和分片数。
这一次的持续优化具有里程碑式的意义,Kafka消费很快,OAP各机器的Full GC没了,ES的各方面指标也很稳定;接下来开始优化查询,提升易用性。
5、trace查询慢-25亿segment/天
1)问题
Web端【追踪】页中的查询都很慢,仅保存了15天的数据,按照traceId查询耗时要20多秒,按照条件查询trace列表的耗时更糟糕;这给人的感受就是“一肚子墨水倒不出来”,未能达到好用的目标。
2)优化
ES查询优化方面的信息挺多,但通过百度筛选出解决此问题的有效方案,就要看咱家爱犬的品类了;当时搜集整理了并尝试了N多优化条款,可惜没有跟好运偶遇,结论是颜值不可靠。言归正传,影响读写性能的基本要素有3个:读写频率,数据规模,硬件性能;trace的情况从这三个维度来套一套模板:
这个分析没有得出具有指导意义的结论,读写频率这里粒度太粗,用户的使用情况跟时间也有紧密的关系,情况大概是:
当天的数据是读多写多(当天不断有新数据写入,基于紧急响应的需求,问题出现时可能是近实时的排查处理)。
前一天的数据是读多写少(一般也会有问题隔天密集上报的情况,0点后会有前一天数据延迟到达的情况)。
再早的话无新数据写入,数据越早被读的概率也越小。
基于以上分析,增加时间维度并细化更多的参考因素后,分析模型变成了这样:
从上表可以看出,整体呈现出hot-warm数据架构的需求之势,近1-2天为hot数据,之前的为warm数据;恰好ES7提供了hot-warm架构支持,按照hot-warm改造后架构如下图所示:
恰逢公司ES中台调优版的ES发布,其内置的ZSTD压缩算法空间压缩效果非常显著。
对 trace集群进行hot-warm架构调整,查询耗时从20多秒变成了2-3秒,效果是非常明显的。
从查询逻辑进一步调整,充分利用ES的数据分片、路由机制,把全量检索调整为精准检索,即降低检索时需要扫描的数据量,把2-3秒优化到毫秒。
这里要炫一个5毛特效,这套机制下,Segment数据即使是保留半年的,按照TraceId查询的耗时也是毫秒。
至此完成了查询千亿级Trace数据只要毫秒级耗时的阶段性优化。
6、仪表盘和拓扑查询慢
1)问题
Web端的【拓扑】页,在开始只有几十个应用的时候,虽然很慢,但还是能看到数据,随着应用增多后,【拓扑】页面数据请求一直是超时(配置的60s超时)的,精力有限,先通过功能降级把这个页面隐藏了;【仪表盘】的指标查询也非常的慢,未能达到好用的目标。
2)优化
Web端的【仪表盘】页和【拓扑】页是对SkyWalking里metric数据的展现,metric数据同trace数据一样满足hot-warm的特征。
①metric集群采用hot-warm架构调整,之后仪表盘中的查询耗时也都减小为毫秒级。
②【拓扑】页接口依然是超时(60s),对拓扑这里做了几个针对性的调整:
把内部的循环调用合并,压缩调用次数。
去除非必要的查询。
拆分隔离通用索引中的数据,避免互相干扰。
全量检索调整为精准检索,即降低检索时需要扫描的数据量。
至此完成了拓扑页数据查询毫秒级耗时的阶段性优化。
7、小结
SkyWalking调优这个阶段,恰逢上海疫情封城,既要为生存抢菜,又要翻阅学习着各种ES原理、调优的文档资料,一行一行反复的品味思考SkyWalking相关的源码,尝试各种方案去优化它,梦中都在努力提升它的性能。疫情让很多人变得焦虑烦躁,但以我的感受来看在系统的性能压力下疫情不值一提。凡事贵在坚持,时间搞定了诸多困难,调优的效果是很显著的。
可能在业务价值驱动的价值观中这些技术优化不产生直接业务价值,顶多是*特效,但从其他维度来看它价值显著:
对个人来说,技术有提升。
对团队来说,实战练兵提升战力,团队协作加深友情;特别感谢ES中台这段时间的鼎力支持!
对公司来说,易用性的提升将充分发挥SkyWalking的价值,在问题发生时,给到同事们切实、高效的帮助,使得问题可以被快速响应;须知战争拼的是保障。
这期间其实也是有考虑过其他的2个方案的:
使用降低采样率的兜底方案;但为了得到更准确的指标数据,以及后续其他的规划而坚持了全采样。
采用ClickHouse优化存储;因为公司有定制优化的ES版本,所以就继续在ES上做存储优化,刚好借此机会验证一下。后续【全链路结构化日志】的存储会使用ClickHouse。
这个章节将内容聚焦在落地推广时期技术层面的准备和调优,未描述团队协调、推广等方面的情况;因每个公司情况不同,所以并未提及;但其实对多数公司来说,有些项目的推广比技术本身可能难度更大,这个项目也遇到过一些困难,PM去推广是既靠能力又靠颜值, 以后有机会再与大家探讨。
七、未来规划
H5、Native以及后端应用都在持续接入中,相应的SDK也在不断地迭代;目前正在基于已建立的链路通道,完善【全链路业务状态追踪】和【全链路结构化日志追踪】,旨在给运营、客服、运维、开发等服务在一线的同事们提供多视角一站式的观测平台,全方位提升系统服务质量、提高问题响应速度。