vmstorage如何将原始指标转换为有组织的历史
vmstorage是VictoriaMetrics中负责处理长期存储的组件。
读取和解析数据
在vmstorage接收到数据之后,并不会直接读取这些数据。首先会检查读并发限速器(限制为2倍的CPU cores),如果读操作过多,则最终会排队等候处理。
vmstorage每次会读取一个block,block的结构如下,包含一个表示block大小的8字节的size
字段以及一个body字段。一个block不能超过100MB。
有一个边缘场景值得注意:即当vmstorage在速磁盘空间不足时会转变为只读模式。该模式下vmstorage会对接收到的数据响应"read-only ack",并忽略实际内容,vminsert会识别此种确认类型并重发数据(本文后面讨论)。
本阶段中,vmagent仅使用字节流方式读取原始block,并不会对其解析。这些原始字节最终会被拆分为rows进行处理。
如果block过大,vmstorage会以chunks为单位进行处理,每次处理10,000 rows数据并将其写入存储。
查找每条metric的TSID
TSID用于表示不同的时间序列,有如下作用:
数据存储优化:
- TSID 在存储时序数据时起到索引的作用。VictoriaMetrics 使用 TSID 来将时间序列数据映射到磁盘的存储位置,从而避免直接存储复杂的指标名称和标签。
- 通过 TSID,可以快速找到时间序列对应的样本(samples,时间戳与值的组合)并高效地写入或读取数据。
查询加速:
- 当查询某个指标时,VictoriaMetrics 会解析查询中指定的标签或指标名称,通过倒排索引查找与查询条件匹配的 TSID。
每条metric包含如下几个关键部分:
- 一个metrics名称
- 一组metrics labels
- 一个时间戳(毫秒)
- 一个浮点数表示的metric value
在计算metric的TSID之前,首先需要创建"规范的指标名称",即将metric name和labels组合起来,然后按名字的字母顺序排列,目的是防止包含相同labels的metrics,只是因为labels顺序不同而被认为是不同的metric,如metric{instance="host",job="app"}
和 metric{job="app",instance="host"}
是相同的metric,只是label顺序不同。
TSID是一种可以表示时间序列的唯一数字,用于快速定位数据。
上一节中,vmstorage从block中获取到了原始的metrics(无排序),然后通过查询TSID缓存来判断是否已经存在对应的TSID,如果存在,则直接插入该指标:
如果不存在,则需要向IndexDB查询,由于涉及随机磁盘查询,因此这是一个比较慢的过程。在查找成功后缓存结果。
如果不存在,则说明这是一个全新的metric。此时,系统需要生成一个新的TSID,并将其同时注册到内存缓存和IndexDB,包括如下步骤:
- 将"规范指标名称"映射到新的TSID
- 设置反向映射,这样TSID可以指向"规范指标名称"
- 倒排索引包含"规范指标名称"中的每个label,可以帮助系统在使用标签过滤时快速查询时序数据
- 创建以天为单位的索引,用于优化按时间范围查找数据的场景。
注册新的时间序列涉及向磁盘写入与之相关的所有信息,该过程可能会拖慢整个系统,特别是当metric拥有很多labels或非常长的labels时。
这也是为什么要关注churn rate的原因。vmstorage可以按小时 (-storage.maxHourlySeries
)和天(-storage.maxDailySeries
)限制新创建的时序数据总量。
向内存缓冲插入数据
一旦注册好TSID,VictoriaMetrics就可以处理实际的数据样本,包括TSID、时间戳、值等,并将其放入一个内存缓冲(称为"raw-row shards"),且一个partition(表示一个月的数据)的shards数目等于CPU cores的数目。例如,机器有4 cores,则每个月的数据有4个shards,每个shards最多可以有8 MB的数据,约149,796 rows。
如果shard被填满,则这些rows会被推入一个称为"pending series"的地方,等待被处理成“LSM part”,并最终写入磁盘。只有刷新到LSM part的数据才能被查询到,之后便完成了本block的数据处理,可以开始处理下一个block。
数据如何写入磁盘
在如下两种情况中,Shard缓冲会刷新数据:
- 当缓冲达到阈值(约120MB),刷新pending series
- 如果距上一次刷新超过2s,则系统会自动刷新 pending series和raw-row shards
在刷新过程中,数据会转换为一个LSM part,LSM part中的项会根据TSID和时间戳进行排序。
LSM Parts的类型
每个partition(涵盖一个月的数据)会将其数据组织为3种LSM parts类型:
- 内存 part:存放raw-row shards首次刷新后的数据,此时数据可以被搜索和查询
- Small part:比内存part稍大,存储在持久化磁盘上
- big part:最大的parts,存储在磁盘上
vmstorage同一时间最多可以持有60个内存parts,占用约10%的系统内存。例如,vmstorage内存为10GB,则内存parts占1GB,每个part约1MB~17MB。
随着数据的写入,会创建越来越多的parts,当LSM parts过多(无论是内存还是磁盘)时,每个查询(如来自grafana的查询)都需要扫描并合并这些parts,可能会拖慢系统。
为了防止上述问题,vmstorage依赖两个关键处理:刷新和合并。
- 刷新:将所有内存parts刷新到磁盘的small parts。每5s,vmstorage(
-inmemoryDataFlushInterval
)会将内存parts刷新到基于磁盘或文件的parts上。 - 合并:将多个parts合并为更高效的存储。这并不意味着将所有small parts合并为big parts,而是将一部分small parts合并为稍大一些的small parts
合并过程
合并并不是固定调度的。只要有parts累积,系统就会尝试合并这些parts,将内存parts合并为较大的内存part,将small parts合并为较大的small part,将big parts合并为较大的big part。
small parts不能超过10MB,big parts最大可以占用大约剩余磁盘空间/4,但不能超过1TB。注意small parts和big parts只是系统在合并过程中评估出来的需要创建的part类型,并不是说small parts一定小于big parts。
small parts合并的结果最终会被写入磁盘,该过程中会执行去重操作。
去重是确认并移除那些几乎相等,但记录时间略微不同的时间点。通常发生在出于冗余或可靠性目的,而使用两个或多个监控系统将相同指标并发往同一个存储的场景。
默认关闭去重,可以通过-dedup.minScrapeInterval
启用去重功能。
Retention, Free Disk Space Guard和 Downsampling
vmstorage的默认回收周期是1个月。需要注意的是,每个part包含很多样本,只要有一个样本在回收周期内,则必须保留整个part。
Free Disk Space Watcher: Read Only Mode
一开始提到,在磁盘空间不足的情况下,vmstorage会进入read-only模式,该模式下,vminset会接收到数据发送的确认信息,但vmstorage会忽略掉这些数据。此时vmstorage仍然能够提供查询请求,但停止接收任何写数据。一旦释放了磁盘空间(-storage.minFreeDiskSpaceBytes
,默认10MB),vmstorage会退出read-only模式。
Partition的结构
无论是内存parts,small parts还是big parts,其数据都是列模式,即TSID、时间戳和值都不会组合在一个记录中,而是被分散到不同的列,每一列都保存在各自的文件中。
- 所有 TSIDs 都保存在
index.bin
. - 所有时间戳都保存在
timestamps.bin
. - 所有值都保存在
values.bin
.
对于内存parts,这些列结构已经就绪,可以直接刷新到基于文件的parts中。
列模式便于压缩和快速查找。 timestamps.bin
和 values.bin
中的每个block表示单个TSID行,一个block最多可以有8192 行。
index.bin
的每一行包括多个block首部,一个block首部包含:
- block的TSID
- block的行数
- block在
timestamps.bin
和values.bin
中的位置