Druid.io大查询分析思路

Druid.io 是CPU和IO双密集型的大数据组件,因为Druid架构中无论是处理实时数据摄入的peon进程还是存储历史数据的历史节点,在负责数据存储的同时还需要处理其节点上数据的查询。因为这样的架构,导致Druid的服务节点对磁盘,内存和CPU都有着比较高的要求。而Druid架构本身是无法对查询进行隔离的,所以单个节点上可能并行处理着多条查询。而处理查询的线程资源和内存资源是有限的,所以查询之间可能存在互相影响。在这样的背景下,当查询耗时比较久需要定位原因时,面对纷繁的指标,如果没有正确的思路,那么问题定位可能会浪费开发或运维人员较多的时间。为此,将自己日常工作中学习到的查询问题定位流程总结,方便日后不断优化问题分析流程。此次总结主要以历史数据查询为主。

1. 预备备

1.1 查询链路

Druid查询链路

拿出一张精简的查询流程图,我们先简单的看一下Druid查询的大概流程,不考虑太多的外部依赖。
①首先客户端发送http请求,请求Broker节点(假设集群中没有设置Router节点);
②接着Broker就会根据zookeeper上信息,把客户端查询拆分成不同的subquery,分发给不同的查询节点;
③实时数据查询路由给middleManager上的peon进程,历史数据查询路由给相应的historical节点;
④各个查询节点处理完查询之后把查询节点返回给Broker;
⑤Broker节点把各个节点返回的结果进行合并,返回给客户端。至此,一个完整的查询链路结束。

1.2 Druid监控指标

在之前的文章中也有提过,Druid组件会将自身的指标信息通过emitter发送出来。这一部分信息我们是可以采集到的,Druid原生也支持httpEmitter。通过这个emitter我们可以将指标信息发送给本地或者其他的http服务,通过http服务我们可以将指标信息发送到kafka的topic中,这样我们就可以像摄入一般实时数据那样保存实时的Druid指标信息了。
目前我们生产上是这样采集指标信息的,流程架构如下:

image.png

社区还有一些其他的emitter,没有尝试过,感兴趣的可以去官网了解一下。
Druid的指标还是相当丰富的,这里不做一一介绍,主要说几个跟查询分析有关的指标。

Broker端

metric Description 关联维度
query/time 查询耗时,这个耗时是相对于整个查询而言 Common: dataSource, type, interval, hasFilters, duration, context, remoteAddress, id. Aggregation Queries: numMetrics, numComplexMetrics. GroupBy: numDimensions. TopN: threshold, dimension.
query/node/bytes 各个节点返回的数据 id, status, server.
query/node/ttfb 子查询从节点上返回第一个结果的耗时 id, status, server

Historical端

metric Description 关联维度
query/time 查询耗时,这个耗时是指某条查询在这个单个节点上的耗时 Common: dataSource, type, interval, hasFilters, duration, context, remoteAddress, id. Aggregation Queries: numMetrics, numComplexMetrics. GroupBy: numDimensions. TopN: threshold, dimension
query/segment/time 单个segment的查询耗时 id, status, segment.
query/wait/time 某条查询或者某个segment等待给执行的时间 id, segment

Jetty

metric Description
jetty/numOpenConnections jetty的连接数量

以上,分析大查询的主要指标就是这些了。

2. 淡定分析

一般情况,当大查询出现之后,我们能获取到的信息一般有两个:

  1. queryID
  2. 查询时间

那我们的分析也就从这两个已知信息入手。首先我们要知道的是,一个查询耗时很长,可能并不是它本身涉及到的数据量过大或查询语句过于复杂,也可能是因为其他的大查询占用了太多的资源,导致它的耗时增加。所以,分析这类查询问题的时候我们不能只关注于这个点的问题,更要在纵向上分析其他的的查询是否有问题。
以下为了方便描述,我就使用plyql来查询Druid的指标信息。这个是一款可以将SQL语句转换成Druid查询的工具。没有的话,也可以人工把下面的SQL语句转换成json查询。

2.1 首先确认查询在哪些节点上耗时比较长

./plyql -h druid-broker-host:8082 -q "select __time, host, value_max   from druid_metrics where metric in ('query/time') and __time>='2019-12-09T13:50:00+08:00' and __time<='2019-12-09T14:05:00+08:00' and id = ''test-query-id" '

返回结果如下

image.png

注意:返回的结果中有个8082端口,这个是broker节点的端口号,后面我们可以通过这个确认查询路由到的broker,然后从这个broker日志中捞出queryID对应的查询语句。
我们可以看到这个查询被分发到了不同的历史节点上,而且每个节点的查询耗时都比较长。这里我们可以先从一个历史节点分析,例如:historical01。我们要分析这个查询在这个节点上是否是因为其他的查询才导致的耗时较长。这里我们就要用到两个指标,一个query/wait/time,quer/sgment/time。如果前者过大,则可能是因为节点要处理的查询过多。如果后者过大,则大概率是因为查询本身过于复杂导致,但是这种情况下,一般通过判断查询语句就能看出了,这里就不考虑这种情况了。其实,还有第三种情况就是query/wait/time,quer/sgment/time这两个指标值都不大,但是查询的耗时很长。目前我们生产环境中,第三种情况是比较常见的。这个和第一种情况一样,一般都是因为其他查询的影响导致的。所以我们要查看,这个query所在节点对应时间点的前一段时间是否出现了大查询,即耗时比较长的查询。我们先查看query/wait/timequer/sgment/time这两个指标。

./plyql -h druid-broker-host:8082 -q "select segment,metric, value_max from druid_metrics where metric in ('query/segment/time','query/wait/time') and __time>='2019-12-09T13:50:00+08:00' and __time<='2019-12-09T14:05:00+08:00' and id = 'test-query-id' and host='historical01:8083' "
image.png

从返回的结果来看,query和wait的time都不算很长,但是我们知道这个查询在这个节点上耗时很长。所以肯定有很多的时间是耗费在等待资源上了。那么我们现在要做的就是找出这个占资源的查询,分析它为什么占了资源。

2.2 确认资源使用较多的查询

还是在这个历史节点上,我们查看这个查询之前的一段时间,看看是否有耗时比较大的查询。这里需要使用子查询,因为我们要选出查询耗时较长的查询,而且需要根据时间排序。因为一般都是最早的那个耗时较多的查询导致后面的查询耗时增加。

 ./plyql -h druid-broker-host:8082 -q "select * from (select __time,id, value_max   from druid_metrics where metric in ('query/time') and __time>='2019-12-09T13:50:00+08:00' and __time<='2019-12-09T14:05:00+08:00' and host='historical01:8083' order by value_max desc limit 30) order by __time""
image.png

从结果来看每个查询的耗时都比较大,这样我们是没法确认,我们需要找的是一个分界的值。这里的查询时间还没有到达我们过滤的边界,所以我们增加返回的数据量。

 ./plyql -h druid-broker-host:8082 -q "select * from (select __time,id, value_max   from druid_metrics where metric in ('query/time') and __time>='2019-12-09T13:50:00+08:00' and __time<='2019-12-09T14:05:00+08:00' and host='historical01:8083' order by value_max desc limit 100) order by __time""

image.png

到这里我们就找到了最开始耗时比较长的查询。那么接下来就是分析一波这几个查询。可以用之前类似的query/wait/timequery/segment/time来分析。如果query/segment/time比较大,那么基本上就是这个查询的问题了。接着就可以把这个查询的查询语句从日志里捞出来看了。查询语句记录在broker的日志里,怎么确定broker我们在之前的查询中已经提到过了。
但是,有一种情况也有可能发生,就是这个最开始的几个查询的query/wait/timequery/segment/time这两个指标也不大,但是查询耗时长。这时候就要进行第三步分析了。

3. 确认处理查询的并发量

这个我们可以从两个方面考虑,一个是这段时间的查询访问量,一个是jetty的连接数。当节点处理的查询并发比较大时,也可能会对资源造成较大的压力,导致查询耗时增加。这个可以通过zabbix查看到是节点的性能情况,这里只从Druid层面分析。

3.1 查询访问量

每个查询都会对应一个queryID,我们只需要对这段时间的queryID做个简单的count就可以算出这个节点的查询访问量了。

./plyql -h druid-broker-host:8082 -q "select count(id)   from druid_metrics where metric in ('query/time') and __time>='2019-12-09T13:50:00+08:00' and __time<='2019-12-09T14:05:00+08:00' and host='historical01:8083'"
image.png

如果需要查看dataSource访问量的话,只需要再做个group by:

./plyql -h druid-broker-host:8082 -q "select count(id)   from druid_metrics where metric in ('query/time') and __time>='2019-12-09T13:50:00+08:00' and __time<='2019-12-09T14:05:00+08:00' and host='historical01:8083' group by dataSource"
image.png

3.1 jetty连接数

./plyql -h druid-broker-host:8082 -q "select __time, value_max from druid_metrics where metric in ('jetty/numOpenConnections') and __time>='2019-12-09T13:50:00+08:00' and __time<='2019-12-09T14:05:00+08:00' and host='historical01:8083'"
image.png

这两个指标,可以跟历史数据比较,就可以知道访问量是否有波动了。

4. 小结

其实Druid查询涉及到的内容比较多,这里只讨论如何分析耗时比较长的情况。但是为什么会导致这种情况,如何去避免这种情况的产生?这又会涉及到Druid本身的数据分配和查询路由策略。这一部分若有机会,再做总结。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,383评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,522评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,852评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,621评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,741评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,929评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,076评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,803评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,265评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,582评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,716评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,395评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,039评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,027评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,488评论 2 361
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,612评论 2 350

推荐阅读更多精彩内容

  • 作者: 康凯森 日期: 2017-11-02 分类:OLAP What is Druid Why Druid Dr...
    茂盛哥哥阅读 946评论 0 0
  • 我们知道Druid能够同时提供对大数据集的实时摄入和高效复杂查询的性能,主要原因就是它独到的架构设计和基于Data...
    allin8116阅读 478评论 0 2
  • Druid io总体设计 1.Druid模块架构 1.1 Druid简介 最新版本的Druid采用了位图索引、字典...
    小武大讲堂阅读 1,820评论 0 2
  • 我们知道Druid能够同时提供对大数据集的实时摄入和高效复杂查询的性能,主要原因就是它独到的架构设计和基于Data...
    零度沸腾_yjz阅读 21,508评论 3 17
  • #refer1:http://www.cnblogs.com/xd502djj/p/6408979.html#re...
    liuzx32阅读 1,903评论 0 1