Kafka存储系统在Twitter的应用分析

当开发者通过API消费Twitter的公共数据时,他们需要获得可靠性、速度和稳定性方面的保证。因此,在不久前,我们推出了Account Activity Replay API帮助开发者们提升他们系统的稳定性。这个API是一个数据恢复工具,开发者可以用它来检索最早发生在5天前的事件,恢复由于各种原因(包括在实时传递时突然发生的服务器中断)没有被传递的事件。

除了构建API来提升开发者体验,我们还做了一些优化:

  • 提高Twitter内部工程师的生产力

  • 保持系统的可维护性。具体来说,就是尽量减少开发人员、站点可靠性工程师和其他与系统交互的人员的上下文切换

基于这些原因,在构建这个API所依赖的回放系统时,我们利用了Account Activity API现有的实时系统设计。这有助于我们重用现有的工作,并最小化上下文切换负担和培训工作。

实时系统采用了发布和订阅架构。为了保持架构的一致性,构建一个可以读取数据的存储层,我们想到了传统的流式技术——Kafka。

背景

两个数据中心产生实时事件,事件被写入到跨数据中心的主题上,实现数据冗余。

但并不是所有的事件都需要被传递,所以会有一个内部应用程序负责对事件进行筛选。这个应用程序消费来自这些主题的事件,根据保存在键值存储中的一组规则来检查每一个事件,并决定是否应该通过我们的公共API 将消息传递给特定的开发者。事件是通过Webhook传递的,每个Webhook URL都有一个开发人员负责维护,并有唯一的ID标识。

存储和分区

通常,在构建一个需要存储层的回放系统时,人们可能会使用基于Hadoop和HDFS的架构。但我们选择了Kafka,主要基于以下两个原因:

  • 已有的实时系统采用了发布和订阅架构

  • 回放系统存储的事件量不是PB级的,我们存储的数据不会超过几天。此外,执行Hadoop的MapReduce作业比在Kafka上消费数据成本更高、速度更慢,达不到开发者的期望

要利用实时管道来构建回放管道,首先要确保事件被存储在Kafka中。我们把Kafka主题叫作delivery_log,每个数据中心都有一个这样的主题。然后,这些主题被交叉复制,实现数据冗余,以便支持来自数据中心之外的重放请求。事件在被传递之前经过去重操作。

在这个Kafka主题上,我们使用默认的分区机制创建了多个分区,分区与WebhookId的散列值(事件记录的键)一一对应。

我们考虑过使用静态分区,但最终决定不使用它,因为如果其中一个开发人员生成的事件多于其他开发人员,那么这个分区包含的数据将多于其他分区,造成了分区的不均衡。

相反,我们选择固定数量的分区,然后使用默认分区策略来分布数据,这样就降低了分区不均衡的风险,并且不需要读取Kafka主题的所有分区。重放服务基于请求的WebhookId来确定要读取哪个分区,并为该分区启动一个新的Kafka消费者。主题的分区数量不会发生变化,因为这会影响键的散列和事件的分布。

我们使用了固态磁盘,根据每个时间段读取的事件数量来分配空间。我们选择这种磁盘而不是传统的硬盘驱动器,以此来获得更快的处理速度,并减少与查找和访问操作相关的开销。因为我们需要访问低频数据,无法获得页面缓存优化的好处,所以最好是使用固态磁盘。

为了最小化存储空间,我们使用了snappy压缩算法。我们知道大部分处理工作都在消费端,之所以选择snappy,是因为它在解压时比其他Kafka所支持的压缩算法 (如gzip和lz4) 更快。

请求和处理

在我们设计的这个系统中,通过API调用来发送重放请求。我们从请求消息体中获取WebhookId和要重放的事件的日期范围。这些请求被持久化到MySQL中,相当于进入了队列,直到它们被重放服务读取。请求中的日期范围用于确定要读取的分区的偏移量。消费者对象的offsetForTimes函数用于获取偏移量。

重放服务处理每个重放请求,它们通过MySQL相互协调,处理数据库中的下一个需要重放的记录。重放进程定期轮询MySQL,获取需要被处理的挂起作业。一个请求会在各种状态之间转换。等待被处理的请求处于开放状态(OPEN STATE),刚退出队列的请求处于启动状态(STARTED STATE),正在被处理的请求处于进行中状态(ONGOING STATE),已处理完成的请求将转换到已完成状态(COMPLETED STATE)。一个重放进程只会选择一个尚未启动的请求 (即处于打开状态的请求)。

每隔一段时间,当一个工作进程将一个请求退出队列后,它会在MySQL表中写入时间戳,表示正在处理当前的重放作业。如果重放进程在处理请求时死掉了,相应的作业将被重新启动。因此,除了将处于打开状态的请求退出队列之外,重放操作还将读取处于已开始或正在进行中的、在预定义的分钟数内没有心跳的作业。

在读取事件时会进行去重操作,然后事件被发布到消费者端的Webhook URL上。去重是通过维护被读取事件的散列值缓存来实现的。如果遇到具有相同散列值的事件,就不传递这个事件。

总的来说,我们的解决方案与传统的将Kafka作为实时、流式系统的用法不一样。我们成功地将Kafka作为存储系统,构建了一个API,在进行事件恢复时提升了用户体验和数据访问能力。我们利用已有的实时系统设计让系统的维护变得更加容易。此外,客户数据的恢复速度达到了我们的预期。


相关大数据培训开发技术知识,关注我,有更多精彩内容与您分享!

(0)

相关推荐

  • 消息中间件

    摘要:消息存储对于每一款消息队列都非常重要,那么Kafka在这方面是如何来设计做到高效的呢?Kafka这款分布式消息队列使用文件系统和操作系统的页缓存(page cache)分别存储和缓存消息,摒弃了 ...

  • CTO问我Pulsar替代Kafka的实际意义是什么

    最近一年,Pulsar可谓出尽了风头.社区中一直在鼓吹Pulsar就是用来取代Apache Kafka的主宰地位的.Pulsar提供了一套兼容Kafka的API,让大数据工程师很丝滑不费力的从Kafk ...

  • Kafka系列1:Kafka概况

    Kafka是当前分布式系统中最流行的消息中间件之一,凭借着其高吞吐量的设计,在日志收集系统和消息系统的应用场景中深得开发者喜爱.本篇就聊聊Kafka相关的一些知识点.主要包括以下内容: Kafka简介 ...

  • 30分钟带你了解「消息中间件」Kafka、RocketMQ

    消息中间件的应用场景 主流 MQ 框架及对比 说明 Kafka 优点 Kafka 缺点 RocketMQ Pulsar 发展趋势 各公司发展 Kafka Kafka 是什么? Kafka 术语 Kaf ...

  • spark的存储系统--BlockManager源码分析

    根据之前的一系列分析,我们对spark作业从创建到调度分发,到执行,最后结果回传driver的过程有了一个大概的了解.但是在分析源码的过程中也留下了大量的问题,最主要的就是涉及到的spark中重要的几 ...

  • R语言推特twitter网络转发可视化分析

    原文链接:http://tecdat.cn/?p=5124 包含关键词"生物信息学"的推文示例 第1步:加载所需的软件包 # 加载所需的软件包library(igraph) 第2步 ...

  • R语言对推特twitter数据进行文本情感分析

    tweets <-trump_tweets_df>%select(id, statusSource, text, created) %>%extract(statusSource, ...

  • Kafka丢失数据问题优化及重复消费原因分析

    数据丢失是一件非常严重的事情事,针对数据丢失的问题我们需要有明确的思路来确定问题所在,针对这段时间的总结,我个人面对kafka数据丢失问题的解决思路如下: 1. 是否真正的存在数据丢失问题 比如有很多 ...

  • 面相分析:如何从面相去判断一个人的穷富

    注:本文转载自网络,如有侵权,请联系删除谢谢. 虽说每个人都渴望自己的物质生活条件可以更富裕一些,但赚钱的多少并不是单纯的只是看自己的能力或者付出的多少,同时也要看自身的气运.有些人似乎天生就是富贵命 ...

  • 你买的公司值不值这个钱?杜邦分析法ROE,就看这三点

    你买的公司值不值这个钱?杜邦分析法ROE,就看这三点

  • 解决不良?看这个图解8D分析法。直观明了!

    8D(8 Disciplines)即问题解决8步法,最早是福特公司使用的经典质量问题分析手法,对于解决工厂中存在的问题是一个很有用的工具,尤其在面对重大不良时,它能建立一个体系,让整个团队共享信息,并 ...

  • 面相分析:有唇珠的人面相分析 唇珠在面相中代表什么

    注:本文转载自网络,如有侵权,请联系删除谢谢. 有唇珠的人有福气吗?今天,小编为大家带来有唇珠的人面相分析,告诉你唇珠在面相中代表什么.有唇珠的嘴唇,微闭微张时,唇珠非常的明显,两唇之间的弓形弧度非常 ...

  • 桂枝汤及其加味方方证分析

    临证加减再变通,是中医临床遣方用药的原则之一.一些配伍合理.用药简约.疗效确切的经典名方(可称为"母方")在临床运用时,往往会根据患者个体病证特征进行化裁,从而变化产生许多新的有效 ...