揭秘字节跳动云原生Spark History 服务 UIService
本文是字节跳动数据平台数据引擎SparkSQL团队针对 Spark History Server (SHS) 的优化实践分享。

文 | 字节跳动数据平台—数据引擎—SparkSQL团队
在字节跳动内部,我们实现了一套全新的云原生 Spark History 服务—— UIService,相比开源的 SHS,UIService 存储占用和访问延迟均降低 90% 以上,目前 UIService 服务已经在字节跳动内部广泛使用,并且作为火山引擎湖仓一体分析服务 LAS(LakeHouse Analytics Service)的默认服务。
LAS
业务背景
开源 Spark History Server 架构
为了能够更好理解本次重构的背景和意义,首先对原生 Spark History Server 原理做个简单的介绍。

开源 Spark History Server 流程图
Spark History 建立在 Spark 事件(Spark Event)体系之上。在 Spark 任务运行期间会产生大量包含运行信息的SparkListenerEvent,例如 ApplicationStart / StageCompleted / MetricsUpdate 等等,都有对应的 SparkListenerEvent 实现。所有的 event 会发送到ListenerBus中,被注册在ListenerBus中的所有listener监听。其中EventLoggingListener是专门用于生成 event log 的监听器。它会将 event 序列化为 Json 格式的 event log 文件,写到文件系统中(如 HDFS)。通常一个机房的任务的文件都存储在一个路径下。
在 History Server 侧,核心逻辑在 FsHistoryProvider中。FsHistoryProvider 会维持一个线程间歇扫描配置好的 event log 存储路径,遍历其中的 event log 文件,提取其中概要信息(主要是 appliaction_id, user, status, start_time, end_time, event_log_path),维护一个列表。当用户访问 UI,会从列表中查找请求所需的任务,如果存在,就完整读取对应的 event log 文件,进行解析。解析的过程就是一个回放过程(replay)。Event log 文件中的每一行是一个序列化的 event,将它们逐行反序列化,并使用 ReplayListener将其中信息反馈到 KVStore 中,还原任务的状态。
无论运行时还是 History Server,任务状态都存储在有限几个类的实例中,而它们则存储在 KVStore中,KVStore是 Spark 中基于内存的KV存储,可以存储任意的类实例。前端会从KVStore查询所需的对象,实现页面的渲染。
痛点
存储空间开销大
Spark 的事件体系非常详细,导致 event log 记录的事件数量非常大,对于UI显示来说,大部分 event 是无用的。并且 event log 一般使用 json 明文存储,空间占用较大。对于比较复杂或时间长的任务,event log 可以达到几十GB。字节内部7天的 event log 占用约 3.2 PB 的 HDFS 存储空间。
回放效率差,延迟高
History Server 采用回放解析 event log 的方式还原 Spark UI,有大量的计算开销,当任务较大就会有明显的响应延迟,响应延迟是指从用户发起前端访问到页面 UI 完全渲染出来的等待时长。作业结束之后,用户可能要等十几分钟甚至半小时才能通过 History Server 看到作业历史。而大型作业结束后,用户往往希望尽快看到作业历史从而根据作业历史进行问题诊断和作业优化,用户等待 UI 完成渲染时间过长,非常影响用户体验。
扩展性差
如上所述,History Server 的FsHistoryProvider在回放解析文件之前,需要先扫描配置的 event log 路径,遍历其中的 event log,将所有文件的元信息加载到内存中,这使得原生服务成为了有状态的服务。因此每次服务重启,都需要重新加载整个路径,才能对外服务。每个任务在完成后,也需要等待下一轮扫描才能被访问到。
当集群任务数量增多,每一轮扫描文件的耗时以及元信息内存占用都会增加,这也要求服务有越来越高的资源配置。如果通过拆分 event log 路径来缩小单实例的压力,需要对路由规则进行改造,运维难度增大。目前,字节跳动内部通过增加 UIService 实例就可以方便的进行水平扩展。
非云原生
Spark History Server 并非是云原生的服务,在公有云场景下改造和维护成本高。首先公有云场景需要进行租户资源隔离,其次公有云场景下不同用户的 workload 差异很大,不同用户任务量有数量级的差别,会出现大量长尾作业。为每个用户单独部署 History Server 计算和存储成本过大且不均衡,而部署统一的 History Server 无法做到资源隔离,一旦出现问题影响较多用户,两种方式运维成本都会很高。火山引擎湖仓一体分析引擎 LAS(Lakehouse Analytics Service),提供了云原生的 UIService,可以有效解决上述问题。
UIService
方案
为了解决前面的三个问题,我们尝试对 History Server 进行改造。
如上所述,无论运行中的 Spark Driver 还是 History Server,都是通过监听 event,将其中包含的任务变化信息反映到几种 UI 相关的类的实例中,然后存入KVStore供 UI 渲染。也就是说,KVStore中存储着 UI 显示所需的完备信息。对于 History Server 的用户来说,绝大多数情况下我们只关心任务的最终状态,而无需关心引起状态变化的具体 event。
因此,我们可以只将 KVStore 持久化下来,而不需要存储大量冗余的 event 信息。此外,KVStore原生支持了 Kryo 序列化,性能明显于 Json 序列化。我们基于此思想重写了一套新的 History Server 系统,命名为 UIService。

UIService框架图
实现
UIMetaStore
KVStore中和 UI 相关的所有类实例,我们将这些类统称为 UIMeta 类。具体包括 AppStatusStore和SQLAppStatusStore中的信息(如下所列)。我们定义一个类 UIMetaStore来抽象,一个UIMetaStore即一个任务所有 UI 信息的集合。
UIMetaStore所包含信息:
#AppStatusStore
org.apache.spark.status.JobDataWrapper
org.apache.spark.status.ExecutorStageSummaryWrapper
org.apache.spark.status.ApplicationInfoWrapper
org.apache.spark.status.PoolData
org.apache.spark.status.ExecutorSummaryWrapper
org.apache.spark.status.StageDataWrapper
org.apache.spark.status.AppSummary
org.apache.spark.status.RDDOperationGraphWrapper
org.apache.spark.status.TaskDataWrapper
org.apache.spark.status.ApplicationEnvironmentInfoWrapper
#SQLAppStatusStore
org.apache.spark.sql.execution.ui.SQLExecutionUIData
org.apache.spark.sql.execution.ui.SparkPlanGraphWrapper
UIMetaStore 还定义了持久化文件的数据结构,结构如下:
4-Byte Magic Number: "UI_S"
----------- Body ---------------
4_byte_length_of_class_name | class_name_str1 | 4_byte_length | serialized_of_class1_instance1
4_byte_length_of_class_name | class_name_str1 | 4_byte_length | serialized_of_class1_instance2
4_byte_length_of_class_name | class_name_str2 | 4_byte_length | serialized_of_class2_instance1
4_byte_length_of_class_name | class_name_str2 | 4_byte_length | serialized_of_class2_instance2
- Magic Number用于文件类型标识校验。
- Body 是 UIMetaStore 的主体数据,使用连续存储。每一个 UI 相关的类实例,会序列化成四个片段:类名长度(4 byte long 类型)+ 类名(string 类型)+ 数据长度(4 byte long 类型)+ 序列化的数据(二进制类型)。在读取时顺序读取,每个元素先读取长度信息,再根据长度读取后续相应数据进行反序列化。
- 使用 Spark 原生的KVStoreSerializer序列化,可以保证前后兼容性。
UIMetaLoggingListener
类似于EventLoggingListener,为 UIMeta 开发了专用的 Listener —— UIMetaLoggingListener,用于监听事件,写 UIMeta 文件。
和EventLoggingListener进行对比:EventLoggingListener每接受一个 event 都会触发写,写的是序列化的 event;而UIMetaLoggingListener只会被特定的 event 触发,目前是只会被stageEnd,JobEnd 事件触发,但每次写操作是批量的写,将上一阶段的UIMetaStore的信息完整地持久化。
做一个类比,EventLoggingListener好比流式,不断地追加写,而 UIMetaLoggingListener类似于批式,定期将任务状态快照下来。
UIMetaProvider
替换原先的FsHistoryProvider,主要区别在于:
将读取 event log 文件和回放生成KVStore的流程改为读取UIMetaFile,反序列化出UIMetaStore。
去掉了FsHistoryProvider的路径扫描逻辑;每次 UI 访问,根据 appid 和路径规则,直接去读取 UIMetaFile 解析。这使得 UIService 无需预加载所有文件元信息,不需要随着任务数量增加提高服务器配置,方便了水平扩展。
优化
避免重复写
由于每个 stage 完成都会触发写 UIMeta 文件,这样对于 UIMeta 的很多元素,可能会出现重复持久化的情况,增加写入耗时和文件的大小。因此我们在UIMetaLoggingListener内部维护了一个 map,记录已经被序列化的实例。在写 UIMeta 文件时进行过滤,只写没有写过或者数据发生改变的元素。这样可以杜绝大部分的写冗余。
此外,开发期间发现,占用空间最大的是task级别信息TaskDataWrapper。在一个 stage 完成触发写时。可能会将仍处于 RUNNING 状态的 stage 的 task 序列化下来,这样当 RUNNING 的 stage 完成时,task 信息会再被写一次,也会造成数据冗余,因此我们对序列化TaskDataWrapper信息进行过滤,在 stage 结束时只持久化状态是 Completed 的 task 信息。
支持回退到 event log
鉴于 UIService 在初期有存在问题的风险,我们还支持了回退机制,即访问一个任务的 UI,优先尝试走 UIService 的路径:解析 UIMeta 文件,如果 UIMeta 文件不存在或者解析报错,会回退到读 event log 文件的路径,避免 UI 访问失败。同时还支持将 event log 文件转换成 UIMeta 文件,这样下一次调用时就可以使用 UIService。这个功能保证我们迁移过程的平滑。
收益
存储收益
线上测试显示存储平均减少85%,总量减少92.4%。
下图显示了某机房 event log 和 UIMeta 存储占用监控,可以看到 UIMeta 较 event log 在存储量上有数量级的减少。目前字节内部7天的 event log 占用存储空间 3.2 PB,改用 UIMeta 后,空间占用只有350TB。
凭借 UIService 的存储优势,我们可以保留更长时间的日志信息,有助于历史分析,问题复盘。目前我们已从保留7天日志提高到了保留30天,并可以根据需求增大保留时间。

某机房 event log/UIMeta HDFS存储监控对比
访问延迟收益
访问延迟:平均缩短 35%,PCT90/95/99 分别减少 84.6%/90.8%/93.7%

访问延迟百分位分布
如下图所示,UIService 的 UI 访问延迟整体较比 event log 向左移,长尾任务明显减少。

访问延迟分布图
架构收益
去掉了原生 History Server 遍历路径,预加载的耗时环节,消除从任务完成到 History Server 可访问的时间间隔,从原本的平均 10min 左右降低到秒级,任务完成即可立即对外提供服务。同时使 History Server 可以水平扩展,能更好应对未来任务量增长带来的挑战。
目前,字节跳动内部我们通过增加 UIService 实例就可以方便的进行横向扩展。在火山引擎湖仓一体分析服务 LAS 中,我们也基于 UIService 实现了支持租户访问隔离,云原生的,可按需伸缩的 Spark History Server。
欢迎关注字节跳动数据平台同名公众号
揭秘字节跳动云原生Spark History 服务 UIService的更多相关文章
- Java云原生崛起微服务框架Quarkus入门实践
@ 目录 概述 定义 GraalVM简介 为何使用 特性 官方性能 实战 入门示例 步骤 安装GraalVM 创建quarkus工程 Idea导入项目 Idea运行和调试 打包成普通的Jar 打包成依 ...
- 火山引擎 DataLeap:揭秘字节跳动数据血缘架构演进之路
更多技术交流.求职机会,欢迎关注字节跳动数据平台微信公众号,回复[1]进入官方交流群 DataLeap 是火山引擎数智平台 VeDI 旗下的大数据研发治理套件产品,帮助用户快速完成数据集成.开发.运维 ...
- 云原生 go-zero 微服务框架
0. go-zero介绍 go-zero是一个集成了各种工程实践的web和rpc框架.通过弹性设计保障了大并发服务端的稳定性,经受了充分的实战检验. go-zero包含极简的API定义和生成工具goc ...
- Dubbo 迈出云原生重要一步 - 应用级服务发现解析
作者 | 刘军(陆龟) Apache Dubbo PMC 概述 社区版本 Dubbo 从 2.7.5 版本开始,新引入了一种基于实例(应用)粒度的服务发现机制,这是我们为 Dubbo 适配云原生基础 ...
- 阿里巴巴 Kubernetes 能力再获 CNCF 认可 | 云原生生态周报 Vol. 32
作者 | 丁海洋 陈有坤 李鹏 孙健波 业界要闻 阿里巴巴 Kubernetes 技术能力再获 CNCF 认可 CNCF 官网发布博文<Demystifying Kubernetes as ...
- 网易云通过KCSP认证,云原生技术实力再获认可
近日,网易云通过KCSP认证,正式成为CNCF官方认可的Kubernetes服务提供商,也标志着网易云在云原生领域的技术实力得到了业界认可. Kubernetes是第一个从CNCF毕业的开源项目,凭借 ...
- 云原生应用基金会CNCF
2006 年 8 月 9 日,埃里克·施密特(EricSchmidt)在搜索引擎大会上首次提出了“云计算”(Cloud Computing)的概念.一转眼十年过去了,它的发展势如破竹,不断渗透当代的 ...
- 开放下载 | 《Knative 云原生应用开发指南》开启云原生时代 Serverless 之门
点击下载<Knative 云原生应用开发指南> 自 2018 年 Knative 项目开源后,就得到了广大开发者的密切关注.Knative 在 Kubernetes 之上提供了一套完整的应 ...
- Kubernetes v1.17 版本解读 | 云原生生态周报 Vol. 31
作者 | 徐迪.李传云.黄珂.汪萌海.张晓宇.何淋波 .陈有坤.李鹏审核 | 陈俊 上游重要进展 1. Kubernetes v1.17 版本发布 功能稳定性是第一要务.v1.17 包含 22 个增强 ...
- 阿里云如何基于标准 K8s 打造边缘计算云原生基础设施
作者 | 黄玉奇(徙远) 阿里巴巴高级技术专家 关注"阿里巴巴云原生"公众号,回复关键词 1219 即可下载本文 PPT 及实操演示视频. 导读:伴随 5G.IoT 的发展,边缘 ...
随机推荐
- JUC并发编程学习笔记(十四)异步回调
异步回调 Future设计的初衷:对将来的某个事件的结果进行建模 在Future类的子类中可以找到CompletableFuture,在介绍中可以看到这是为非异步的请求使用一些异步的方法来处理 点进具 ...
- 单个Nginx发布多个react静态页面
在有些网络环境中,端口是一种比较稀缺的资源,而我们又恰好有多个前端项目需要发布,我们可以采取将多个项目映射到同一个端口上面的方案加以解决. 本教程前端项目主要以react为主,部署在linux服务器上 ...
- AtCoder Beginner Contest 328 (ABC328)
A. Not Too Hard 模拟. Code B. 11/11 模拟. Code C. Consecutive Description 给你一个字符串 \(S\),有 \(Q\) 次询问,每次输入 ...
- Java并发(十七)----变量的线程安全分析
1.成员变量和静态变量是否线程安全 如果它们没有共享,则线程安全 如果它们被共享了,根据它们的状态是否能够改变,又分两种情况 如果只有读操作,则线程安全 如果有读写操作,则这段代码是临界区,需要考虑线 ...
- LambdaQueryWrapper常用方法
/***附加条件构造器LambdaQueryWrapper常用方法 ---这几个肯定够用了*/wrapper.eq("实体类::查询字段", "条件值"); / ...
- Mybatis|MybatisPlus批量插入
创建一个SpringBoot工程 <?xml version="1.0" encoding="UTF-8"?> <project xmlns= ...
- KNN算法实战——海伦约会(KDtree优化)
本文通过海伦约会的例子来测试之前写的KDTree的效果,并且探讨了特征是否进行归一化对整个模型的表现的影响.最后发现在机器学习中,特征归一化确实对模型能提供非常大的帮助. 1 from KDTree ...
- ceph集群搭建详细教程(ceph-deploy)
ceph-deploy比较适合生产环境,不是用cephadm搭建.相对麻烦一些,但是并不难,细节把握好就行,只是命令多一些而已. 实验环境 服务器主机 public网段IP(对外服务) cluster ...
- python获取已安装程序列表
python获取已安装程序列表 本文主要讲述通过python脚本获取android 设备已安装列表. 首先,Python本身无法直接获取Android设备上已安装的程序列表,所以这里主要借助adb命令 ...
- 解压.tar.gz文件的命令
要解压以 .tar.gz 或 .tgz 扩展名结尾的文件,可以使用 tar 命令.通常,这些文件是使用 tar 和 gzip 压缩的.以下是解压 .tar.gz 文件的命令: tar -xzvf 文件 ...