写点什么

揭秘字节跳动基于 Hudi 的实时数据湖平台

  • 2021 年 12 月 30 日
  • 本文字数:3531 字

    阅读完需:约 12 分钟

揭秘字节跳动基于Hudi的实时数据湖平台

本文整理自 Apache Hadoop Meetup 2021 北京站字节跳动数据平台数据湖团队研发工程师耿筱喻《字节跳动基于 Hudi 的实时数据湖平台介绍》的分享实录。内容主要包含四部分内容:首先是 Hudi 和字节跳动实时数据湖平台简介;其次字节跳动实时数据湖平台的应用场景介绍;第三针字节跳动的实时数据湖平台针对应用场景的优化与特性;最后介绍我们的未来规划。

Hudi 和字节跳动实时数据湖平台简介

Hudi 是一个流式数据湖平台,提供 ACID 功能,支持实时消费增量数据、离线批量更新数据,并且可以通过 Spark、Flink、Presto 等计算引擎进行查询。



Hudi 表由 timeline 和 file group 两大项构成。Timeline 由一个个 commit 构成,一次写入过程对应时间线中的一个 commit,记录本次写入修改的文件。


相较于传统数仓,Hudi 要求每条记录必须有唯一的主键,并且同分区内,相同主键只存在在一个 file group 中。底层存储由多个 file group 构成,有其特定的 file ID。File group 内的文件分为 base file 和 log file, log file 记录对 base file 的修改,通过 compaction 合并成新的 base file,多个版本的 base file 会同时存在。


在表的更新方面,Hudi 表分为 COW 和 MOR 两种类型:

  • COW 表:适用于离线批量更新场景,对于更新数据,会先读取旧的 base file,然后合并更新数据,生成新的 base file。

  • MOR 表:适用于实时高频更新场景,更新数据会直接写入 log file 中,读时再进行合并。为了减少读放大的问题,会定期合并 log file 到 base file 中。


对于更新数据,Hudi 通过索引快速定位数据所属的 file group。目前 Hudi 已支持 Bloom Filter Index、Hbase index 以及 Bucket Index,其中 Bucket Index 尚未合并到主分支

字节跳动基于 Hudi 的实时数据湖平台

字节跳动基于 Hudi 的实时数据湖平台,通过秒级数据可见支持实时数仓。除了提供 Hudi 社区的所有功能外,还支持基于数据湖的元数据管理系统、行列级别的并发更新、Bucket Index 和 Append 模式等特性。

字节跳动实时数据湖平台应用场景

01-典型 Hudi Pipeline 场景

一个典型的 pipeline 是,MySQL 侧的 binlog 生产到 Kafka:

  • 实时场景直接通过 Spark Streaming 或 Flink 消费这部分更新数据,写入数据湖,供下游业务使用。

  • 批量场景会先将 binlog 通过 dump service 存储到 HDFS 上,再按照小时/天级粒度更新到数据。

02-推荐场景

在字节的推荐场景中,为服务离线数据分析挖掘需求,需要将数据从类 Hbase 的存储导出到离线存储中,并且可以提供高效的 OLAP 访问。因此我们基于数据湖构建 BigTable 的 CDC。



此外,在特征工程和模型训练场景中,需要将推荐系统 Serving 时获得的数据和端上埋点数据这两类实时数据流通过主键合并到一起,作为机器学习样本。因此我们希望可以借助数据湖的能力,低成本的批量添加特征列。

03-数仓场景

数仓 backfill 场景中,需要对历史全量数据进行部分行、列的更新,在 Hive 模式下,需要将增量数据和历史全量进行 join,重新生成全量数据。其中,部分表的存量数据到达百 PB 级别。我们通过数据湖极大的减少了计算资源消耗,提升了端到端的性能。



04-近实时数仓场景

数仓场景中,对于一张底层分析表,往往是通过多个数据源的数据组合拼接而成,每个数据源都包含相同的主键列,和其他不同的属性列。在传统数仓场景中,需要先将每个数据源数据 dump 成 Hive 表,然后再将多张 Hive 表按主键 join 后生成最终的完整 schema 的大表,延迟可到达天级别。我们通过数据湖使实时成为可能,并且提供列拼接能力,使下游数据分析性能大幅提升。


字节跳动针对 Hudi 的优化和新特性

01-湖仓一体元数据服务

Hive Metastore 是元数据的事实标准,但是基于目录的元数据管理方式太粗,没有办法满足数据湖以 commit 的形式管理元数据的需求。我们提出了适用于数据湖场景下的元数据管理系统 Hudi Metastore,并基于此设计了湖仓统一的元数据管理系统。


整个架构分为三部分引擎层、元数据层、存储层。元数据层对外提供统一的元数据视图,与 HMS 完全兼容,可无缝对接多个计算引擎。元数据层的 Catalog Service 接收来自引擎层的访问请求,按规则路由到不同的 Metastore 上。元数据层通过 Catalog Service 屏蔽底层多 Metastore 的异构性。



Hudi Metastore 作为数据湖元数据管理系统,支持 commit 形式的元数据管理,基于乐观锁和 CAS 支持并发更新;持久化元数据的 Snapshot,通过缓存常被访问的元数据、索引信息,提供高效查询;提供分区裁剪功能。整体设计包括:

  • 底层存储可插拔,不依赖某个特定的存储系统,可以是 HDFS、KV、MySQL。

  • 轻量且易于扩展,服务无状态,支持水平扩展;存储可通过拆库/表的方式纵向扩展。

  • 与 Hive Metastore 兼容。

02-行级并发写入



我们基于 Hudi Metastore 和乐观锁的假设,实现了并发写入,并且支持灵活的行列冲突检查策略。冲突检查会在 instant 状态变换的两个节点进行,一个是 requested 转 inflight 状态,一个是 inflight 转 completed 状态。其中,后者状态变换时,会进行加锁操作,以实现版本隔离。



冲突检查即是对 instant 创建到状态变化的过程中其他已经完成/正在执行的 instant 之间的进行冲突检查,检查策略分为行列两种:


  • 行级别的冲突检查即是不能同时有两个 instant 往同一个 file group 写。

  • 列级别的冲突检查即是可以有两个 instant 往同一个 file group 写,但是两个 instant 写入的 schema 不可以存在交集:

1、每个 instant 只写入 schema 中的部分列,log 文件中的数据只包含 schema 中的部分。

2、Compaction 按主键拼接不同列下的数据,Parquet 文件中存储的数据拥有完整的 schema。



03-Bucket Index


Hudi 目前的两种索引方式,Bloom Filter Index 在大数据场景下,假阳性的问题会导致查询效率变差,而 Hbase Index 会引入额外的外部系统,从而提升运维代价。因此,我们希望能有一个轻量且高效的索引方式。


Bucket Index 是一种基于哈希的索引。每个分区被分成 N 个桶,每个桶对应一个 file group。对于更新数据,对更新数据的主键计算哈希,再对分桶数取模快速定位到 file group,提升导入实时性。




现有的计算引擎大都会利用表的 Bucket 分布做查询优化,提升查询性能。优化规则包含两种:

  • Bucket Pruning:利用表的 Bucket 分布对读取数据进行剪枝。

  • Bucket Join:利用表的 Bucket 分布减少 Join/Aggregate 带来的 shuffle 操作。

04-Append 模式支持



Hudi 要求每条数据都有唯一主键和比较列,用于数据更新时定位 file group 和新旧数据比较。数据定位 file group 过程需要先根据索引构建主键到 file group 的映射关系,然后与更新数据按照主键进行 join,从而找到每条更新数据对应的 file group。


对于日志场景,无确定的主键,并且用户查询也仅仅是对某些列进行 count 操作,因此更新数据只需要直接追加到任一文件末尾即可,也就是 Append 模式。为此,我们提出了 NonIndex 方案,无需指定主键和比较列,更新过程也无需构建主键到 file group 的映射关系,避免了 join,提升了导入的实时性。


Hudi 要求每条数据都有唯一主键和比较列,用于数据更新时定位 file group 和新旧数据比较。数据定位 file group 过程需要先根据索引构建主键到 file group 的映射关系,然后与更新数据按照主键进行 join,从而找到每条更新数据对应的 file group。


对于日志场景,无确定的主键,并且用户查询也仅仅是对某些列进行 count 操作,因此更新数据只需要直接追加到任一文件末尾即可,也就是 Append 模式。为此,我们提出了 NonIndex 方案,无需指定主键和比较列,更新过程也无需构建主键到 file group 的映射关系,避免了 join,提升了导入的实时性。

未来规划

字节跳动未来会针对 Hudi 开展更多方向的优化,包括:支持部分列更新下的完成 Bin Log 消费;可扩展哈希索引;存储服务数据秒级可见;基于 Merge Tree 的文件分布等,后续这些新特性也将逐步贡献到社区。


本文中介绍的功能优化,目前已通过火山引擎产品“湖仓一体分析服务”向外部企业输出湖仓一体分析服务 LAS(Lakehouse Analytics Service)是面向湖仓一体架构的 Serverless 数据处理分析服务,提供一站式的海量数据存储计算和交互分析能力,完全兼容 Spark、Presto、Flink 生态,帮助企业轻松完成数据价值洞察。

其他关联产品

火山引擎大数据研发治理套件 DataLeap

一站式数据中台套件,帮助用户快速完成数据集成、开发、运维、治理、资产、安全等全套数据中台建设,帮助数据团队有效的降低工作成本和数据维护成本、挖掘数据价值、为企业决策提供数据支撑。


火山引擎 E-MapReduce

支持构建开源 Hadoop 生态的企业级大数据分析系统,完全兼容开源,提供 Hadoop、Spark、Hive、Flink 集成和管理,帮助用户轻松完成企业大数据平台的构建,降低运维门槛,快速形成大数据分析能力。


欢迎关注字节跳动数据平台同名公众号

发布于: 刚刚
用户头像

公众号byte-dataplatform 2021.12.29 加入

字节跳动数据平台团队,赋能字节跳动各业务线,对内支持字节绝大多数业务线,对外发布了火山引擎品牌下的数据智能产品,服务行业企业客户。同名公众号欢迎了解。

评论

发布
暂无评论
揭秘字节跳动基于Hudi的实时数据湖平台