作者 | 乔克
来源 | 运维开发故事
监控是整个运维以及产品整个生命周期最重要的一环,它旨在事前能够及时预警发现故障,事中能够结合监控数据定位问题,事后能够提供数据用于分析问题。
监控贯穿应用的整个生命周期。即从程序设计、开发、部署、下线。其主要的服务对象有:
技术
业务
技术通过监控系统可以了解技术的环境状态,可以帮助检测、诊断、解决技术环境中的故障和问题。然而监控系统的最终目标是业务,是为了更好的支持业务运行,确保业务的持续开展。
所以监控的目的可以简单归纳如下:1、能够对系统进行7*24小时的实时监控 2、能够及时反馈系统状态 3、保证平台的稳定运行 3、保证服务的安全可靠 4、保证业务的持续运行
监控由上至下可以分为:
业务监控
应用监控
操作系统
其中业务监控主要是研发提供一些业务指标、业务数据,对其增长率、错误率等进行告警或者展示,需要提前定义规范甚至埋点。
应用程序的监控主要有探针和内省。其中探针主要是从外部探测应用程序的特征,比如监听端口是否有响应。内省主要是查看应用程序内部的内容,应用程序通过检测并返回其内部的状态,内部的组件,事务和性能等度量,它可以直接将事件、日志和指标直接发送给监控工具。
操作系统主要是监控主要组件的使用率、饱和度以及错误,比如CPU的使用率、CPU的负载等。
监控的主要方式有:
健康检查。健康检查是对应用本身健康状况的监控,检查服务是否还正常存活。
日志。日志是排查问题的主要方式,日志可以提供丰富的信息用于定位和解决问题。
调用链监控。调用链监控可以完整的呈现出一次请求的全部信息,包括服务调用链路、所耗时间等。
指标监控。指标是一些基于时间序列的离散数据点,通过聚合和计算后能反映出一些重要指标的趋势。
在上述4中监控方式中,健康检查是云平台等基础设施提供的能力,日志则一般有单独的日志中心进行日志的采集、存储、计算和查询,调用链监控一般也有独立的解决方案进行服务调用的埋点、采集、计算和查询,指标监控则是通过一些 exporter 抓取目标暴露的指标,然后对这些指标数据进行清理、聚合,我们通过聚合出来的数据进行展示、告警等。
说明:该方案主要是针对指标监控
云平台提供健康检查能力,直接在云平台中配置。
成熟的开源日志解决方案是ELK。
调用链健康使用第三方的健康软件,常用的有 skywalking、zikpin、pinpoint、elastic APM、Cat。
其中 zikpin 和 cat 对代码有一定的侵入性,而 skywalking、pinpoint、elastic APM 是基于字节码注入技术,对代码没有侵入性,而且改动最小。
pinpoint 的 agent 仅支持 java 和 php,而 skywalking 和 elastic APM 都支持多种语言,比如 Java/Node.js/Go 等。
在云原生环境下,skywalking 和 elastic APM 更适合。
elastic APM 直接使用 es 作为存储,可以在 kibana 上直接看应用信息,但是其关系图是需要产生一定的费用。
skywalking 是国人开源的一款产品,已经毕业于 Apache 基金会,社区非常活跃,版本迭代很快,专为微服务、云原生架构和基于容器(Docker、K8s、Mesos)架构而设计。
pinpoint 和 skywalking 的对比可参考:
http://skywalking.apache.org/zh/blog/2019-02-24-skywalking-pk-pinpoint.html
在云原生环境、传统的监控方式并不适合,在传统环境,zabbix 无疑是首选,但是在云原生环境,Prometheus 却成为了热门,其主要原因有:
成熟的社区支撑。Prometheus 是 CNCF 的毕业项目,有许多大厂做背书,社区庞大,是首推的云原生监控解决方案。
易于部署和运维。Prometheus 核心只有一个二进制文件,没有其他的第三方依赖,部署运维均十分方便。
采用 Pull 模型,通过 HTTP 的 Pull 方式从各个监控目标拉取监控数据。Push 模型一般通过 Agent 方式去采集信息并推送到收集器中,每个服务的 Agent 都需要配置监控数据项与监控服务端的信息,在大量服务时会加大运维难度;另外,采用Push 模型,在流量高峰期间监控服务端会同时接收到大量请求和数据,会给监控服务端造成很大压力,严重时甚至服务不可用。
强大的数据模型。Prometheus 采集到的监控数据均以指标的形式存在于内置的时序数据库中,除了基本的指标名称外,还支持自定义的标签。通过标签可以定义出丰富的维度,方便进行监控数据的聚合和计算。
强大的查询语言 PromQL。通过 PromQL 可以实现对监控数据的查询、聚合、可视化、告警。
完善的生态。常见的操作系统、数据库、中间件、类库、编程语言,Prometheus都提供了接入方案,并且提供了Java/Golang/Ruby/Python 等语言的客户端SDK,能够快速实现自定义的监控逻辑。
高性能。Prometheus 单一实例即可处理数以百计的监控指标,每秒处理数十万的数据,在数据采集和查询方面有着优异的性能表现。
注意:由于采集的数据有可能丢失,Prometheus 并不适合对采集数据要求 100%准确的场景。
监控系统的整体框架如下:
Prometheus Server:用于抓取指标、存储时间序列数据
exporter:暴露指标让任务来抓
pushgateway:push 的方式将指标数据推送到该网关
alertmanager:处理报警的报警组件
adhoc:用于数据查询
其流程很简单,Prometheus server 端可以直接接收或者通过 pushgateway 获取到数据,存储到 TSDB中,然后对数据进行规则整理,通过 Altermanager 进行报警或者通过 Grafana 等工具进行展示。
监控系统一般采用分层的方式划分监控对象。在我们的监控系统中,主要关注以下几种类型的监控对象:
主机监控,主要指主机节点软、硬件资源的一些监控数据。
容器环境监控,主要指服务所处运行环境的一些监控数据。
应用服务监控,主要指服务本身的基础数据指标,提现服务自身的运行状况。
第三方接口监控,主要指调用其他外部服务接口的情况。
对于应用服务和第三方接口监控,我们常用的指标包括:响应时间、请求量QPS、成功率。
6.1.1、为什么需要主机监控
主机是系统的载体,一切系统应用都运行在主机之上,如果某台或者几台主机宕机,会导致上面运行的所以应用都没办法正常提供服务,严重者导致生产事故。所以对主机的监控与预警是非常有必要的,我们可以在其出故障之前对其进行处理,避免严重的事故发生。
6.1.2、如何判断资源情况
主机的监控主要从一下三个方面来综合考虑其状态:
使用率:资源忙于工作的平均时间,通常是随时间变化的百分比
饱和度:资源队列的长度
错误:资源错误事件的计数
6.1.3、哪些资源需要监控
主机的主要资源对象有:
CPU
内存
磁盘
可用性
服务状态
网络
6.1.4、如何进行监控
在 Prometheus 监控方案中,主机的资源指标是通过 node-exporter 来进行采集,然后存储在 Prometheus 时序数据库里,然后可以通过 PromQL 来查询各个指标的具体情况。
1、CPU
CPU主要从使用率和饱和度来进行监控。
(1)、使用率,指标node_cpu_seconds_total
通常会根据CPU使用率超过多少来进行告警,比如当 CPU 使用率大于 80%,则进行告警。
当然 CPU 是一个 Gauge 类型的,它的数据是会上下增减的,所以我们在判断 CPU 使用率的时候通常是一段时间内 CPU 持续高达多少的时候才进行告警,比如下面的表达式就是统计 5 分钟内 CPU 使用率大于 60% 的主机:
100-(avg(irate(node_cpu_seconds_total{mode="idle"}[5m])) by(instance)* 100) > 60
CPU 指标还有用户态、内核态的指标数据,这个根据情况来进行监控。
(2)、饱和度,指标node_load
CPU的饱和度通常指的是CPU的负载情况。正常情况下CPU的整体负载不超过CPU的总数,比如2颗CPU,则负载不超过2。
我们收集到的指标有1分钟、5分钟、15分钟的负载数据,在配置监控的时候选择好统计时间,一般情况下会选择5分钟的负载作为统计,如下表示5分钟的负载大于CPU的总数的2倍:
node_load5 > on (instance) 2 * count by(instance)(node_cpu_seconds_total{mode="idle"})
2、内存
内存主要从使用率和饱和度来进行监控。
(1)、使用率
内存的使用率可以直观的看到整体CPU的使用情况,其计算方式使用(free + buffer + cache)/ total。指标主要有:
node_memory_MemTotal_bytes:主机上的总内存
node_memory_MemFree_bytes:主机上的可用内存
node_memory_Buffers_bytes:缓冲缓存中的内存
node_memory_Cached_bytes:页面缓存中的内存
比如下面的表达式是用于统计内存使用率大于 80%:
100 - sum(node_memory_MemFree_bytes{job="node-exporter"} + node_memory_Buffers_bytes{job="node-exporter"} + node_memory_Cached_bytes{job="node-exporter"})by (instance) / sum(node_memory_MemTotal_bytes{job="node-exporter"})by(instance)*100 > 80
(2)、饱和度
内存的饱和度是指内存和磁盘的读写情况来监控。指标有:
node_vmstat_pswpin:系统每秒从磁盘读到内存的字节数,单位KB
node_vmstat_pswpout:系统每秒从内存写到磁盘的字节数,单位KB
3、磁盘
磁盘的监控有点特殊,我们不按着 USE 的方法去测量。如果单考虑其使用率并没有多大的效果,因为 10G 剩余 20% 和 1T剩余 20% 对我们的影响是不一样的,所以我们可以监控其增长趋势以及方向。
比如:根据前面 1h 的磁盘增长情况来预测在 4h 内是否会把磁盘用完。
predict_linear(node_filesystem_free_bytes{job="node-exporter",mountpoint!=""}[1h], 4*3600)
当然,如果仅仅这样预测也会产生很多垃圾告警,因为在某一小时的增长速度可能很快,这样算下来预测会在接下来的4小时内使用完,但是我们登上主机一看,才用了40%,这时候就算有告警我们也不会处理。
所以我们还可以再加一个条件,比如磁盘使用率大于80%并且在接下来的4小时内会使用完。如下:
(100 - (node_filesystem_avail_bytes{fstype!="",job="node-exporter"} / node_filesystem_size_bytes{fstype!="",job="node-exporter"} * 100)>80) and (predict_linear(node_filesystem_free_bytes{job="node-exporter",mountpoint!="",device!="rootfs"}[1h],4 * 3600) < 0)
除此之外,我们还需要监控磁盘的IO,不论是云主机磁盘还是物理磁盘,每块盘都有其对应的 IOPS,如果某个主机的 IO 很高,也会导致其他的问题,比如系统繁忙、负载很高等情况。
node-exporter 中定义了其指标,仅需要对其进行聚合,然后对聚合后的数据进行页面展示或者告警处理。其聚合公式如下:
100-(avg(irate(node_disk_io_time_seconds_total[1m])) by(instance)* 100)
4、可用性
可用性是指的主机可用性,我们可以通过up
指标来判断主机是否可用,如果其值等于0表示宕机,等于1表示存活。比如下面即可表示主机不可用:
up{job="node-exporter"}==0
5、服务状态
服务状态旨在监控关键服务,比如 docker.service,ssh.service,kubelet.service 等。指标为:
node_systemd_unit_state
比如监听 docker.service 的服务状态为存活:
node_systemd_unit_state{name="docker.service",state="active"} == 1
监控主要服务,以便于我们能在服务出问题的第一时间收到消息进行处理。
6、网络
网络主要是监控其在每台主机上的出入流量,还有 TCP 连接状态。
prometheus 的 node-exporter 会抓取每台主机的网卡以及其出入网卡的流量,还有每台主机的 TCP 状态,我们可以将需要的指标进行聚合,根据聚合后的指标数据再进行页面展示或者告警处理。
比如统计流入的流量:
((sum(rate (node_network_receive_bytes_total{device!~'tap.*|veth.*|br.*|docker.*|virbr*|lo*'}[5m])) by (instance)) / 100)
统计流出的流量:
((sum(rate (node_network_transmit_bytes_total{device!~'tap.*|veth.*|br.*|docker.*|virbr*|lo*'}[5m])) by (instance)) / 100)
以及统计 TCP 状态为 ESTABLISHED 的数量:
node_netstat_Tcp_CurrEstab
我们可以根据每个指标做对应的监控以及告警。
6.2.1、为什么需要容器监控
在云原生时代,容器是我们应用的载体,它相当于应用的基础设施,所以对其的监控是很有必要的。我们在创建一个容器的时候往往会给其cpu和内存一个限制值,特别是内存,如果其使用达到了限制值,就会导致OOM,这时候我们就会做升级配置或查找原因处理。
6.2.2、监控的指标对象主要有哪些
监控对象主要有一下:
cpu
memory
事件
6.2.3、如何进行监控
我们使用cAdvisor来获取容器指标(kubelet已经集成了这个服务)。
1、CPU
在容器中,就简单通过其使用率来监控其状态,我们通过其(使用量/limit)来得到其使用率。如下:
sum(
node_namespace_pod_container:container_cpu_usage_seconds_total:sum_rate
* on(namespace,pod)
group_left(workload, workload_type) mixin_pod_workload
) by (workload, workload_type,namespace,pod)
/sum(
kube_pod_container_resource_limits_cpu_cores
* on(namespace,pod)
group_left(workload, workload_type) mixin_pod_workload
) by (workload, workload_type,namespace,pod) * 100 > 80
如果 CPU 的使用率持续大于我们设定的阈值,则考虑增加 CPU 的 Limit 值。
2、memory
和 CPU 一样,通过其使用率来观察容器的内存是否充足。如下:
sum(
container_memory_working_set_bytes
* on(namespace,pod)
group_left(workload, workload_type) mixin_pod_workload
) by (namespace,pod) / sum(
kube_pod_container_resource_limits_memory_bytes
* on(namespace,pod)
group_left(workload, workload_type) mixin_pod_workload
) by (namespace,pod) * 100 / 2 > 80
如果内存的使用率大于我们设定的阈值,则考虑是否需要增加 Pod 的内存了。
3、事件
这里的事件针对的是 kubernetes 的 pod 事件。在 Kubernetes 中,事件分为两种,一种是 Warning 事件,表示产生这个事件的状态转换是在非预期的状态之间产生的;另外一种是 Normal 事件,表示期望到达的状态,和目前达到的状态是一致的。
我们用一个 Pod 的生命周期进行举例,当创建一个 Pod 的时候,首先 Pod 会进入Pending 的状态,等待镜像的拉取,当镜像录取完毕并通过健康检查的时候,Pod 的状态就变为 Running。此时会生成 Normal 的事件。
而如果在运行中,由于 OOM 或者其他原因造成 Pod 宕掉,进入 Failed 的状态,而这种状态是非预期的,那么此时会在 Kubernetes 中产生 Warning 的事件。
针对这种场景而言,如果我们能够通过监控事件的产生就可以非常及时的查看到一些容易被资源监控忽略的问题。
在 kubernetes 中通过 kube-eventer 来进行事件监控,然后针对不同的事件来进行告警通知。
6.3.1、为什么需要应用服务监控
应用是业务的载体,也是用户最直观的体验,应用的状态与否直接关系到业务的优良以及用户的体验。如果没有对其做好一定的监控措施,可能会出现以下问题:
无法识别或诊断故障
无法衡量应用程序的运行性能
无法衡量应用程序或组件的业务指标以及成功与否,例如跟踪销售数据或交易价值
6.3.2、有哪些监控指标
应用程序监控
HTTP接口:URL存活、请求量、耗时、异常量
JVM :GC次数、GC耗时、各个内存区域的大小、当前线程数、死锁线程数
线程池:活跃线程数、任务队列大小、任务执行耗时、拒绝任务数
连接池:总连接数、活跃连接数
业务指标:视业务来定,比如PV、订单量等
6.3.3、如何进行监控
1、应用程序监控
应用程序指标可以衡量应用程序的性能和状态,包括应用程序最终用户的体验,如延迟和响应时间。在这背后,我们测量了应用程序的吞吐量:请求、请求量、事务和事务时间。
「(1)、HTTP 接口监控」
可以使用 prometheus 的 blackbox_exporter 来进行接口存活的监控,可以用于对 http,https,tcp,dns 以及 ICMP 协议进行探测,从而抓取数据进行监控。
「(2)、JVM 监控」
通过在应用中埋点来暴露 JVM 数据,使用 Prometheus 监控采集JVM 数据,借助 Prometheus Grafana 大盘来展示 JVM 数据,并创建报警,即可实现利用 Prometheus 监控 JVM 的目的。
(1)、在 pom.xml 文件中添加 Maven 依赖。
io.prometheus
simpleclient_hotspot
0.6.0
(2)、在可以执行初始化的位置添加初始化 JVM Exporter 的方法。
@PostConstruct
public void initJvmExporter() {
io.prometheus.client.hotspot.DefaultExports.initialize();
}
(3)、在 /src/main/resources/application.properties 文件中配置用于 Prometheus 监控的端口(Port)和路径(Path)
management.port: 8081
endpoints.prometheus.path: prometheus-metrics
(4)、在 /src/main/java/com/monitise/prometheus_demo/PrometheusDemoApplication.java 文件中打开 HTTP 端口
@SpringBootApplication
// sets up the prometheus endpoint /prometheus-metrics
@EnablePrometheusEndpoint
// exports the data at /metrics at a prometheus endpoint
@EnableSpringBootMetricsCollector
public class PrometheusDemoApplication {
public static void main(String[] args) {
SpringApplication.run(PrometheusDemoApplication.class, args);
}
}
然后在部署应用的时候暴露接口和数据就可以进行采集了。
由于应用比较多就可以通过自动发现的方式来做。我们在 service 中加上下面的 annotations,就可以自动发现了。
prometheus.io/scrape: 'true'
prometheus.io/path: '/prometheus-metrics'
prometheus.io/port: '8081'
2、业务指标监控
业务指标是应用程序指标的更进一层,它们通常与应用程序指标同义。如果你考虑将对特定服务的请求数量作为应用程序指标进行测量,那么业务指标通常会对请求的内容执行某些操作。
一个应用程序指标的示例可能是测量支付交易的延迟,相应的业务指标可能是每个支付交易的价值。业务指标可能包括新用户/客户的数量、销售数量、按价值或位置划分的销售额,或者其他任何有助于衡量业务状况的指标。
6.4.1、为什么需要第三方接口监控
第三方接口的优良直接影响自身业务,所以对第三方接口的异常情况监控是非常重要的。主要是其的响应时间、存活性以及成功率。
6.4.2、有哪些监控指标
响应时间
存活性
成功率
6.4.3、如何进行监控
可以使用 prometheus 的 blackbox_exporter 来进行接口的监控。
通过第三方接口监控的维度,我们可以方便地将自身服务与所使用到的第三方服务关联起来,以统一的视图展示服务用到了哪些第三方服务接口、这些第三方服务接口的响应时间和成功率是多少。
当服务出现异常时,对于定位问题有很大帮助;同时,一些内部的服务可能监控报警并不全面,第三方监控也能帮助他们提升服务质量。
达到什么阈值需要告警?对应的故障等级是多少?不需要处理的告警不是好告警,可见定义合理的阈值有多重要,否则只会降低运维效率或者让监控系统失去它的作用。
Prometheus 允许基于 PromQL 定义报警的触发条件,Prometheus 周期性的对 PromQL 进行计算,当满足条件时就会向 Alertmanager 发送报警信息。
在配置告警规则的时候,我们将按组进行分类,这样就可以对相同组下的告警进行聚合,方便配置以及查看。Alertmanager 在接收到报警后,可以对报警进行分组、抑制、静默等额外处理,然后路由到不同的接收器。
Alertmanager 支持多种报警通知方式,除常用的邮件通知外,还支持钉钉、企业微信等方式,也支持通过 webhook 自定义通知方式。我们可以按轻重缓急定义不同的通知方式,这样就可以根据不同通知方式采取不同的措施。
收到故障告警后,一定要有相应的处理流程和oncall机制,让故障及时被跟进处理。
在处理故障之前,需要先清晰的认识是什么样的故障,然后再采取什么样的措施。所以我们就需要对故障等级做一个划分。例如将系统故障等级按照《信息系统安全等级保护基本要求》具体划分为四个等级,一级和二级故障为重大故障;三级和四级故障为一般性故障。
8.1.1、一级故障
系统发生故障,预计将已经严重影响公司生产业务系统,导致相关生产业务系统中断1小时以上,并预计24小时以内无法恢复的,具备以下一个或几个特征,既定义为一级故障。
公司机房网络与阿里云VPC网络出现故障,导致工作人员和用户无法访问相关业务系统;
WEB网站和APP系统等关键服务器宕机或有其他原因导致拒绝提供服务的;
利用技术手段造成业务数据被修改、假冒、泄漏、窃取的信息系统安全事件;
由病毒造成关键业务系统不能正常提供服务。
8.1.2、二级故障
信息系统发生故障,预计将或已经严重影响公司生产业务系统,导致相关生产业务系统中断1小时以上,并预计24小时以内可以恢复的,具备以下一个或几个特征,即定义为二级故障。
公司机房网络与阿里云VPC出现线路和设备故障;
WEB网站和APP系统等关键服务器宕机或有其他原因导致拒绝提供服务的;
12小时以内无法解决的三级故障。
8.1.3、三级故障
满足以下条件之一,即定义为三级故障。
故障发生后,影响到信息系统的运行效率,速度变慢,但不影响业务系统访问;
故障发生后预计在12小时以内恢复;
24小时以内无法解决的四级故障
8.1.4、四级故障
满足以下条件之一,即定义为四级故障。
故障发生后,可随时应急处理,不会影响系统的全面运行;
生产业务系统设备因病毒攻击等原因,造成网络数据出现偶尔掉包,但不影响系统的正常访问和运行。
8.2.1、故障发现
工作人员在发现故障或接收到故障报告后,首先要记录故障发生时间和发现时间,及发现部门,发现人及联系电话,对故障的等级进行初步判定,并报告相关人员进行处理。
8.2.2、故障处理
发生故障的系统通知到运维人员,运维人员应先询问了解设备和配置近期的变更情况,查清故障的影响范围,从而确定故障的等级和发生故障的可能位置;
对于一般性故障按照规定的故障升级上报要求进行上报,并在处理过程中及时向主管领导通报故障处理情况;
对于重大故障按照规定的故障升级上报要求进行上报,并在处理过程中及时向主管领导通报故障处理情况。
8.2.3、故障上报
根据故障等级和发生的时限,要对故障的情况进行及时的上报,并对报告人,告知人际时间内容进行记录。重大故障由故障处理组领导负责上报,一般性故障由故障处理人员负责上报。故障升级上报时限如下表所示: