EAGLEYE监控预警平台

概述

监控预警平台, eagle + eye (鹰眼)的合体词, 寓意可以快速发现问题, 并及时作出响应,Eagleye整个分为:

  • eagleye-java-client: 针对java程序进行日志收集
  • eagleye-alerter: 根据控制台设置的预警规则, 对匹配日志进行实时预警.
  • eagleye-admin: eagleye管理界面,可以看到各个app的接入情况,设置预警规则,进行日志检索等.
  • eagleye-collector: eagleye日志收集器, 该收集器主要是从kafka中收集日志病存储到Elasticsearch中, 也可以用logstash替代.
  • eagleye-task: eagleye的定时监听组建, 如果下游其他系统需要某个app日志,可以直接通过该组建发送.

核心功能说明

首页客户端感知

首页

所有接入eagleye-java-client的应用每分钟都会通过redis发送心跳数据, 这里面包括应用所在机器的IP, PID, 每分钟日志的生产力, 应用最近一次启动到现在日志总量.如果应用进程被kill,则首页上相应的节点会消失.

应用管理

应用管理

在这里进行app管理, 应用在发送日志之前, 需要在log4j配置文件中配置应用的名称, eagleye最终会根据这个配置名称来匹配预警规则. 所以这里的名称需要在应用程序中的log4j配置中的名称一致.

规则管理

规则管理

我们可以通过预警管理设置具体的预警规则, 可以通过正则表达式来匹配日志中的内容, 提示中给出来比较常规的预警规则设置.

预警设置

预警设置
  1. 针对某一个应用里面的某一个预警规则,我们可以设置给谁发送短信预警, 给谁发送邮件预警.

  2. 我们通过预警设置的管理将具体的应用和预警规则进行挂钩.

Zookeeper监控

ZK监控

这里我们借鉴了阿里巴巴的Taokeeper监控软件, 开发了部署更加方便的zookeeper监控, 可以针对每一个集群中的每一个zk实例进行每分钟自检, 如果超出我们设定的预警阈值, 则会进行实时报警.

日志概览

日志概览

我们前期通过Kibana的简单统计页面可以了解各个应用日志的大致情况. 只需要简单的配置即可实现.

日志检索

日志检索

全量日志收集进来之后, 我们需要对它们进行精确检索. 这里我们提供了针对时间区间, IP, PID,线程ID, 应用名称, 日志关键字等的联合检索功能

日志上下文

日志上下文

有些时候我们只检索出满足条件的日志还不能帮助我们快速定位问题, 我们还需要查看日志的上下文, 那么我们可以双击某一条日志, 则可以查看该条日志的前200条和后200条日志.

ES监控

HEAD

可以查看ES集群中详细的Sharding信息.

BIGDESK

可以查看某一个ES实例所在机器的JVM, CPU, 内存等详情.

Zipkin监控

ZIPKIN

可以根据服务名称查看具体的RPC调用链详情.

详细设计

Eagleye整体架构图

eagleye,整体架构图

eagleye作为所有服务的日志监控预警平台, 需要收集所有后台服务的全量日志,主要分成如下几个部分:

  • 各个java服务需要接入的基于log4j appender的日志收集客户端(eagleye-java-client).
  • 我们统一通过Kafka作为日志传输通道, 因为要收集的是所有服务的全量日志,日志量庞大,Kafka非常适合这种分布式收集的场景, 并且可以很容易的进行水平拓展.
  • 我们通过eagleye-alert 从kafka中获取日志, 并根据eagleye-admin中设置的预警规则进行实时预警, eagleye-alert支持水平拓展,并可以做到高可用.
  • 通过logstash进行全量日志的收集并存储到elasticsearch中进行存储, 最终用kibana进行快速的检索和统计.
  • eagleye-admin是eagleye架构的管理控制台, 从这可以进行各种预警规则的配置, 监控各个client, alerter的心跳, es的存储状态, 提供日志检索,统计界面.eagleye-admin作为一个管理控制台,并不需要高可用,所以只需要部署一个实例, 即使admin不可用也不会影响所有日志的收集预警功能.
  • 所有服务器的心跳, 预警规则, 都是通过redis进行通信.

日志收集,存储架构图

日志收集,存储架构图

日志可靠性说明

因为将来会涉及到交易流程的日志跟踪, 所以日志的可靠性需要保证.鉴于之前推动zookeeper的dns设置过程(运维人员需要每台服务器去修改,推动时,不是这台服务器没有设置,就是那台服务器没有设置), 日志收集任务前期在服务端进行日志采集的过程, 并没有引入第三方的收集工具(鉴于logstash的部署问题,我们在后期引入了自己写的eagleye-collector来进行日志中继).在服务端进行日志采集方式, 采用自定义log4j的appdender方式接入采集点.通过kafka作为日志传输通道.当然, 采用这种方式有一个风险点是,假如kafka不可用时,日志将采集不到. 但是会记录到本地log日志中,并不会丢失. 服务端进行日志采集,也有高可用方案,可以每台部署服务的机器都部署logstash进行收集,即使logstash不可用,日志也不会丢失,并且,logstash启动时会将丢失收集的日志重新采集.但是logstash部署在服务端所在的机器,对前期日志采集工作量比较大,并不适合我们前期小步迭代推动eagleye项目.所以我们前期约定,在极端情况下,我们允许小概率收集日志失败的可能.因为采用的是Kafka的队列方式进行传输,所以并不影响我们将来更替高可用的日志采集方案.

日志中继说明

日志中继器的存在,有利于我们水平拓展日志收集,并灵活指定存储媒介.Logstash可以灵活的接入es, hdfs, mysql, cvs, local file 等存储媒介, 并可以灵活的对日志内容进行过滤.Logstash相比flume会更轻量一些,并且是elasticsearch的同源产品.
如果觉得logstash部署麻烦, 可以直接采用后续开发的eagleye-collector进行日志中继.

日志存储

我们选择Elasticsearch进行存储, 当然也可以随时切换成hdfs进行存储, elasticsearch提供了丰富的RESTFul风格的api,和语言无关.ES的集群搭建成本非常低,每一个ES服务会自动发现集群并加入进去.

日志检索, 统计

因为有Kibana的存在, 我们可以很轻易并且快速的从es中检索出我们想要的日志. 可以从多维度进行日志的统计.关键是这些功能的提供, 我们不需要写任何一行代码,后续我们又开发了自定义检索,可以查看某一条日志的上下文.

备注

上图中的channel, relayer, es 可以部署在相同的机器中,也可以进行拆分. 每一层都可以选用其他方案进行替代,并不影响整个系统的运转.

测试

功能测试

性能测试

  • logstash 性能测试:
    单台服务器的单个logstash实例
    内存:500m
    input consumers : 1
    output workers : 2
    机器内存: 4G, 其中2个G被es消耗, 1个多G被日志mock程序占用
    机器8核,普通pc机

    在output配置中
    workers => 2 # 如果这个配置开太大容易内存溢出, 需要根据机器配置进行调整, 在测试环境logstash 内存500m时,2个线程效率最高
    flush_size => 1000 #该配置配合workers进行调整, 如果太大, 也容易内存溢出, 或者增加gc次数.目前这个配置可以达到单台logstash可以20000/s的日志写入

    gc观察
    在bin/logstash.lib.sh 文件中 logstash 采用了 如下gc方式

    JAVA_OPTS="$JAVA_OPTS -XX:+UseParNewGC"
    JAVA_OPTS="$JAVA_OPTS -XX:+UseConcMarkSweepGC"
    

    观察曲线如下图:

    logstash采用自己默认gc

    老年代gc会比较频繁,增加了stop the world的时间,会影响性能.

    注释掉之后的曲线老年代很少出现回收现象, 不会造成stop the world 现象

    logstash采用jvm默认gc

    但是采用jvm默认gc方式后运行了1个小时之后出现了频繁的老年代gc, 如下图:


    logstash采用jvm默认gc

    简单的测试观察之后, 还是建议使用logstash默认的gc

    JAVA_OPTS="$JAVA_OPTS -XX:+UseParNewGC"
    JAVA_OPTS="$JAVA_OPTS -XX:+UseConcMarkSweepGC"
    
  • eagleye-java-client 性能测试

    client往kafka中发送日志可以80000/s, 这取决与网络速度和硬盘速度

    运行时通过jmx观察内存,如下图: 没有一次老年代回收


    eagleye-java-client gc图
  • eagleye-alert 性能测试

    • 单服务, 单机器(244)压力测试:
      cpu使用如下图:


      cpu
      cpu

      内存使用如下图:


      内存
      内存

      吞吐量:3.15w条message/s


      吞吐量
      吞吐量
    • 双服务, 双机器(244, 177)压力测试:
      177的cpu使用如下图:


      cpu
      cpu
    • 177内存使用如下图:


      内存
      内存
    • 177吞吐量:2.32w条message/s

    • 244的cpu使用如下图:


      cpu
      cpu
    • 244内存使用如下图:


      内存
      内存
    • 244吞吐量:3.22w条message/s
      发现单机处理时最大吞吐量 与双机处理时最大吞吐量相同 , 但是通过下面单机双实例测试可以确认服务器网络io不是瓶颈 ,
      初步认为是单个kafka topic 分区的生产力是有限的 , 双机处理时吞吐量的差距是由于kafka topic 分区分配不均衡导致(244得到3个分区 , 177得到两个)
      修改kafka分区数到10个后, 发现单个程序实例即使分配到更多(5个或10个)分区, 吞吐量上限仍然是3w/s ,
      进一步测试得到的结果是程序的性能瓶颈在kafka消费者迭代获取数据部分, 且一个程序中运行多个针对单个topic的ConsumerConnector并不会提高程序吞吐量

    • 单机(244)双实例
      程序1(吞吐量 1.68w条message/s , 分配到2个分区)
      程序2(吞吐量 2.9w条message/s , 分配到3个分区)

    • 总结:
      alerter程序启动时可以通过配置文件loomContext.xml配置alerterConsumerThreadNum(alerter程序kafka消费者线程数).一个kafka消费者线程可以消费多个到0个分区, 但一个分区最多只能被一个消费者线程消费,新的消费程序加入或旧的消费程序失去连接会引发所有分片和所有消费者线程间的再平衡, 但是如果已经是每个分片都被不同的线程消费, 即分片和消费线程为1对1的关系时, 再加入消费程序, 该消费程序将会分配不到分区 导致该消费程序吞吐量为0.单个程序实例的吞吐量在3w/s, 可以通过部署多个实例的方式实现水平扩容, 但需要确保新部署的实例会得到topic分区,如线上服务topic有24个分区, 最多可部署alerter程序实例数为 24/alerterConsumerThreadNum, 即如alerterConsumerThreadNum配置为3, 最多可以部署24/3=8个alerter程序实例.

部署

kafka topic创建

./kafka-topics.sh -create --topic 'EAGLEYE_LOG_CHANNEL' --zookeeper 'spark:2181,witgo:2181' --partitions 10 --replication-factor 1
./kafka-topics.sh -create --topic 'EAGLEYE_ALERT_MAIL' --zookeeper 'spark:2181,witgo:2181' --partitions 10 --replication-factor 1
./kafka-topics.sh -create --topic 'EAGLEYE_ALERT_SMS' --zookeeper 'spark:2181,witgo:2181' --partitions 10 --replication-factor 1

logstash 部署步骤

  • 通过rbenv 对ruby环境安装(logstash 需要有jruby1.7.11 版本支持):

    git clone git://github.com/sstephenson/rbenv.git ~/.rbenv
    
  • 用来编译安装 ruby

    git clone git://github.com/sstephenson/ruby-build.git ~/.rbenv/plugins/ruby-build
    
  • 用来管理 gemset, 可选, 因为有 bundler 也没什么必要

    git clone git://github.com/jamis/rbenv-gemset.git  ~/.rbenv/plugins/rbenv-gemset
    
  • 通过 gem 命令安装完 gem 后无需手动输入 rbenv rehash 命令, 推荐

    git clone git://github.com/sstephenson/rbenv-gem-rehash.git ~/.rbenv/plugins/rbenv-gem-rehash
    
  • 通过 rbenv update 命令来更新 rbenv 以及所有插件, 推荐

    git clone https://github.com/rkh/rbenv-update.git ~/.rbenv/plugins/rbenv-update
    
  • 然后把下面的代码放到 ~/.bash_profile 里

    export PATH="$HOME/.rbenv/bin:$PATH"
    
  • 然后重开一个终端就可以执行 rbenv 了.

    rbenv install 1.9.3-p551 # 安装 1.9.3-p551
    rbenv install jruby-1.7.11 # 安装 jruby-1.7.11  
    rbenv global jruby-1.7.3 # 默认使用 jruby-1.7.3
    
  • 下面从咱们的gitlab上获取已经安装好插件的logstash-1.4.2

    git clone git@git.xxx.com:trade_architect/logstash.git
    
  • 因为logstash需要使用kafka, 所以需要在本机配置, 生产环境的kafka的hosts表

    192.168.1.43 xx-xx-1
    192.168.1.46 xx-xx-2
    192.168.1.60 xx-xx-3
    192.168.1.67 xx-xx-4
    192.168.1.73 xx-xx-5
    
  • 按照logstash进行负载均衡存储的es实例

    (1). git clone git@git.xxx.com:trade_architect/elasticsearch.git

    (2). 在elasticsearch-1.4.4/config/elasticsearch.yml 中设置将需要输出的数据, 日志路径.

     node.name: "eagleye_es_logstash_XXX"   #其中XXX可以标识出该机器
    
     path.data: /home/zhejava/esdata/data   #因为该实例不存储数据使用, 只是用来进行检索和存储时当做集群负载均衡器使用,
     path.work: /home/zhejava/esdata/work  #存放临时文件的位置
     path.logs: /home/zhejava/esdata/logs  #存放日志的位置
    
     transport.tcp.port: 9300  #配置开启, 并修改为9301
    
     http.port: 9200 #配置开启 , 并修改为9201
    
     node.master: false  #开启配置
     node.data: false     #开启配置
    

    (3). 在 elasticsearch-1.4.4/bin/elasticsearch 文件中的 ES_HEAP_SIZE=32768m 修改为 8192m 即可

    (4). 执行

     elasticsearch-1.4.4/bin/elasticsearch -d
    
  • 进入到 logstash-1.4.2 中执行

    sh start.sh 
    

    即可, 在该文件中可以配置logstash具体的日志输出路径.

  • 手动搭建logstash服务
    如果不想使用已经安装好插件的logstash请参照下面的方式安装.如果按照官方文档的部署可以git clone 最新版的logstash-kafka, 并且在该目录下执行 make tarball, 然后生成 logstash-1.4.2.tar.gz,但是这需要有一个畅通的网络, 因为有一个无形的墙,阻挡我们完成这件事, 我个人尝试vpn出去, 也出现了失败, 网络并不稳定.所以我还是阐述一下通过手动搭建一个logstash 服务, 并可以从kafka中消费队列, 并存储消息到elasticsearch中.

    (1) 下载官方logstash-1.4.2 版本, 这是最新的release版本.

    wget https://download.elasticsearch.org/logstash/logstash/logstash-1.4.2.tar.gz
    

    (2) 下载 logstash 针对kafka的插件 logstash-kafka v0.6.2 (或者可以通过git clone获取最新的版本)

    wget https://github.com/joekiller/logstash-kafka/archive/v0.6.2.tar.gz  
    

    (3) 因为logstash-kafka是ruby项目, 所以需要依赖ruby环境(具体见本文上半部分)

    (4) 解压logstash包,得到 $logstash_home, 并创建 $logstash_home/vendor/jar/kafka_2.9.2-0.8.1.1/libs/

    (5) 下载kafka程序包(如果没有, 请去官网下最新0.8.1.1版本.

    wget https://www.apache.org/dyn/closer.cgi?path=/kafka/0.8.1.1/kafka_2.9.2-0.8.1.1.tgz)
    

    (6) 从$kafka_home/lib 目录复制所有jar文件到$logstash_home/vendor/jar/kafka_2.9.2-0.8.1.1/libs/下

    (7) 复制logstash-kafka/lib/logstash里的 inputs 和 outputs 下的 kafka.rb,拷贝到对应的 $logstash_home/lib/logstash 里的 inputs 和 outputs 对应目录下

    (8) 安装logstash-input-kafka

    git clone https://github.com/logstash-plugins/logstash-input-kafka.git
    

    然后进入该目录下运行bundle install, 这时会下载logstash运行时依赖的各种gem (这又是一个大坑, 之前忽略了这一步)

    (9) 在$logstash_home目录下运行:

    GEM_HOME=vendor/bundle/jruby/1.9 GEM_PATH= java -jar vendor/jar/jruby-complete-1.7.11.jar --1.9 ../logstash-kafka/gembag.rb ../logstash-kafka/logstash-kafka.gemspec
    

    (10) 在$logstash_home目录下创建 logstash.yml 配置文件,文件内容为如下:

    input {
        kafka {
            zk_connect => "xxx1:2181,xxx2:2181"
            group_id => "eagleye_logstash"
            topic_id => "EAGLEYE_LOG_CHANNEL"
            consumer_threads => 5  
        }
    }
    
    output {
      elasticsearch {
        cluster => "eagleye_es"
        # 我们可以版本搭建一个es服务, 该服务不存储数据和检索, 只是发现es集群,并转发操作请求
        host => "localhost"
        port => "9301"
        protocol => "transport"
        index => "eagleye-log-%{+YYYY.MM.dd}"
        index_type => "log"
        workers => 2 # 如果这个配置开太大容易内存溢出, 需要根据机器配置进行调整, 在测试环境logstash 内存500m时,2个线程效率最高
        flush_size => 1000 #该配置配合workers进行调整, 如果太大, 也容易内存溢出, 或者增加gc次数.目前这个配置可以达到单台logstash可以20000/s的日志写入
        template_overwrite => true
      }
    }
    

    (11) 启动logstash(进入bin目录)

    nohub logstash agent -f logstash.yml > /home/zhejava/logstash_log/logstash.log &
    

    (12) logstash-1.4.2 默认的elasticsearch-logstash.json 模板并没有生效, 里面只是一个通用的样板,我将 logstash-1.4.2/lib/logstash/outputs/elasticsearch/elasticsearch-template.json 修改为如下内容:

    {
      "template" : "eagleye-*", //这个需要和要操作的index名称匹配,很重要,默认是logstash*,而我们的是eagleye*,所以该模板一直不好使,没注意,大坑啊.
      "settings" : {
        "index.refresh_interval" : "5s"
      },
      "mappings" : {
         "log" : {
                "properties" : {
                    "@timestamp" : {
                        "type" : "date",
                        "format" : "dateOptionalTime"
                    },
                    "timestamp" : {
                        "type" : "date", //日期类型, 在生成json的对象中需要是date类型
                        "format" : "dateOptionalTime" 
                    },
                    "head" : {//涉及到map嵌套,可以这样写
                        "properties" : {
                            "app" : {
                                "type" : "string",
                                "index" : "not_analyzed" //该字段不需要分词索引,统计时不会因为"-"而将一个分开统计
                            }
                        }
                    }
                }
            }
      }
    }
    

    (13) logstash时差8小时问题
    logstash在按每天输出到elasticsearch时,因为时区使用utc,造成每天8:00才创建当天索引,而8:00以前数据则输出到昨天的索引
    修改logstash/lib/logstash/event.rb 可以解决这个问题
    第226行

    .withZone(org.joda.time.DateTimeZone::UTC)
    

    修改为

    .withZone(org.joda.time.DateTimeZone.getDefault())
    

    (14) 碰到的坑
    logstash到此已经部署完毕, 这里碰到一个坑: 是logstash从kafka中消费消息是始终不能转换DefaultDecoder对象为json, 最后发现 v0.6.2版本的logstash-kafka中的inputs插件有问题,这时, 你需要去github中获取logstash-kafka 项目下的logstash-kafka/lib/logstash/inputs/kafka.rb 作为最新的input插件覆盖之前的.

elasticsearch 部署

  • 依赖 jdk 1.7

  • 获取Elasticsearch项目

    git clone git@git.xxx.com:trade_architect/elasticsearch.git
    
  • 在elasticsearch-1.4.4/config/elasticsearch.yml 中设置将需要输出的数据, 日志路径.

    node.name: "eagleye_es_XXX"   #其中XXX可以标识出该机器
    
    path.data: /home/zhejava/esdata/data   #es数据存储位置, 需要是独立的磁盘, 空间越大越好, 到时,公司的全量日志可能会存储在这, 会是高IO操作
    path.work: /home/zhejava/esdata/work  #存放临时文件的位置, 建议有独立的磁盘
    path.logs: /home/zhejava/esdata/logs  #存放日志的位置, 建议独立的磁盘
    
  • 执行 elasticsearch-1.4.4/bin/elasticsearch -d

eagleye-java-client 部署

因为我们的日志收集是对kafka强依赖, 所以需要有一些配置来进行相关的配置才能接入, 具体的配置主要集中在log4j.xml 和spring配置文件中.

  • pom.xml中添加如下内容:

    <!--基于scala-2.9.2-->
    <dependency>
        <groupId>com.xxx</groupId>
        <artifactId>eagleye-java-client</artifactId>
        <version>1.1.0-SNAPSHOT</version>
    </dependency>
    
  • log4j.xml中需要添加一个自定义的appender节点, 具体如下:

    <appender name="eagleye" class="com.xxx.eagleye.client.log4jappender.EagleyeAppender">
        <param name="app" value="eagleye-test" /> <!-- 这里的名称就是eagleye-admin中提到的app , eagleye会按照这个进行约定预警规则的绑定,可以和application name(loom中的)一致 -->
        <layout class="org.apache.log4j.PatternLayout" />
    </appender>
    
    <root>
        <level value="INFO"/>
        <appender-ref ref="console"/>
        <!-- 不要忘记在这里将自定义的eagleye appender 添加到root节点下 -->
        <appender-ref ref="eagleye"/> 
    </root>
    

    如果你用的是log4j.properties文件, 如下配置:

    log4j.appender.eagleye=com.xxx.eagleye.client.log4jappender.EagleyeAppender
    # 这里的名称就是eagleye-admin中提到的app , eagleye会按照这个进行约定预警规则的绑定,可以和application name(loom中的)一致
    log4j.appender.eagleye.app=eagleye-test 
    log4j.appender.eagleye.layout=org.apache.log4j.PatternLayout
    
    #不用忘记在rootLogger中添加eagleye的appender
    log4j.rootLogger=INFO,console,file,eagleye
    
  • 在spring配置文件中进行如下配置:

    <!-- 以下配置可以引入imago进行集中配置管理 需要显式设置lazy-init="false" 因为整个程序没有通过spring去获取eagleyeChannel对象 -->
    <bean id="eagleyeChannel" class="com.xxx.eagleye.client.Configuration" lazy-init="false">
        <!-- 该redis地址主要是记录client的心跳, 日志生产力,日志总量 -->
        <constructor-arg name="redisIp" value="#{configManager.getRedisConfig('trade_public_redis','14.1').ip}"/> 
        <constructor-arg name="redisPort" value="#{configManager.getRedisConfig('trade_public_redis','14.1').port}"/>
        <constructor-arg name="redisPwd" value="#{configManager.getRedisConfig('trade_public_redis','14.1').pwd}"/>
        <constructor-arg name="kafkaBroker" value="#{configManager.getKafkaConfig('trade_public_kafka','1.1').broker}"/> <!-- 需要用到进行日志收集的kafka配置 -->
        <!--<constructor-arg name="kafkaTopic" value="EAGLEYE_LOG_CHANNEL" />--> <!-- 收集日志的topic名称, 该配置默认不需要配置 -->
        <!--<constructor-arg name="sendFreq" value="3000" />--> <!-- 批量操作时间间隔, 单位 ms, 这些时间间隔往kafka批量推送一次 -->
    </bean>
    
  • 因为使用的kafka需要用到域名, 需要配置以下kafka的host表,类似如下:

    192.168.1.158 xxx1
    192.168.1.159 xxx2
    
  • 如果要发送监控数据, 在代码中可以这样写:

    logger.info(MonitorLog.build().type(LogType.EAGLEYE_MONITOR_LOG_REFUND_SHMJCSWGBSL).value(105D));
    

    注: 如果你想看到client发送的具体日志内容, 请在log4j.xml配置做如下配置:

    <logger name="com.xxx.eagleye.client.log4jappender.EagleyeAppender">
        <level value="debug"/>
    </logger>
    

kibana部署和配置

概述:Kibana 是为 Elasticsearch 设计的开源分析和可视化平台。你可以使用 Kibana 来搜索,查看存储在 Elasticsearch 索引中的数据并与之交互。你可以很容易实现高级的数据分析和可视化,以图标的形式展现出来。

部署:

  • 部署一个用于做负载的Elasticsearch,Elasticsearch的部署详见(elasticsearch 部署),部署完成后修改

  • 下载Kibana 4

  • 解压 .zip 或 tar.gz 压缩文件

  • 配置Elasticsearch地址,在config/kibana.yml文件,配置第一步安装的elasticsearch地址.

    elasticsearch_url: "http://localhost:9200"
    
  • bin/kibana,启动kibana

  • 访问5601端口.

配置

  • 配置索引(相当于数据库名),es存入数据是每天创建一个索引,所以kibana要读取数据必须先配置索引,索引可以按天模糊匹配.配置方法如下图.
    配置完索引后,选择时间戳,要选择带"@"符号的timestamp用日志进入es的时间字段.


    kibana
  • 配置discover:配置各种查询数据,进行条件筛选

    kibana

    kibana

    (1). 点击discover菜单,点击搜索口后面的添加按钮(加号),在左侧Field项下会列出所有的字段信息,鼠标移到字段上面,会出现add按钮,点击添加,该字段会出现在上方的select field项下.
    (2). 配置完成后,右边会出现选择的列和数据.
    (3). 点击apply应用所选的配值
    (4). 点击保存,在页面的右上方,有个保存的图标.填写保存的名字

  • 配置visualize:visualize是对discover数据表的各种汇总信息,并以各种个图表显示.

    kibana

    kibana

    (1). 点击visualize菜单,点击搜索口后面的添加按钮(加号).
    (2). 选择样式
    (3). 选择基础数据(discover保存的基础数据)
    (4). 在左侧选择自己需要的统计数据,Metric选项卡中选择统计结果如(count,avg,sum等).buckets下面的选项可以选择如何分组,相当于group by 后面的字段
    (5). 点击apply应用所选的配值,点击右上角保存按钮,然后输入名字保存.

  • 配置dashboard:将discover和visualize保存的各种表格和报表集中在一起显示

    kibana

    kibana

    (1). 点击上图添加按钮,跳转到选择discover和visualize页面.
    (2). 选择需要的discover和visualize页面.此时会在下放显示.
    (3). 拖动改变位置和大小.
    (4). 保存设置好的面板.

  • 搜索框简易使用说明
    (1). 在特定的字段里搜索,例如:line_id:86169
    (2). 可以用 AND/OR 来组合复杂的搜索,AND/OR必须大写,例如:food AND love
    (3). 括号的使用:("played upon" OR "every man") AND stage
    (4). 数值类型的数据可以直接搜索范围:line_id:[30000 TO 80000] AND havoc
    (5). 搜索所有:*
    (6). 如果输入 where1 where2 两个单词,实际搜索的是where1 OR where2 如果要搜索where1 where2,请用双引号引起来"where1 where2"

alerter 部署和配置

  • 部署:
    (1). 获取代码

    git clone git@git.xxx.com:trade_architect/eagleye.git
    

    (2). 进入在eagleye-alerter程序根目录

    cd eagleye/eagleye-alerter 
    

    (3). 在eagleye-alerter程序根目录执行

    git -pull && mvn clean package -Dmaven.test.skip=true -U 
    

    完成后进入文件夹得到 项目压缩包 eagleye-alerter-1.0.0-all.tar.gz

    (4). 解压项目包

    (5). 执行启动

    nohup ./start.sh  > /data/applogs/eagleye/service.log  2>&1  &
    
  • 关闭:
    需要手动kill掉程序进程,可以执行以下命令:

    ps -ef | grep 'com.xxx.eagleye.alerter.App' | grep -v 'grep' |  awk '{print $2}' | xargs  kill
    
  • 配置说明:
    由于alerter程序会和kafka,zookeeper集群进行通信, 所以程序运行前需要配置相关host, 配置如下:

    #kakfa cluster
    192.168.1.43 xx-xx-1
    192.168.1.46 xx-xx-2
    192.168.1.60 xx-xx-3
    192.168.1.67 xx-xx-4
    192.168.1.73 xx-xx-5
    

    loomContext.xml 中可以配置日志toipc kafka消费者的消费线程数(alerterConsumerThreadNum),alerter程序可以部署的最大实例数为 kafka日志topic分区数/alerterConsumerThreadNum.如果部署的程序实例数超过了最大实例数, 新的程序实例将会得消费不到得到topic分区, 不产生吞吐量.

  • 需要注意:

    (1). 需要替换conf/log4j.xml 中的日志文件目录 , 和命令输出日志目录

    (2). 需要修改run.sh 中的变量IP 为服务器IP , HEAP_DUMP_PATH 为日志目录的文件

    (3). 由于alerter即是kafka消费者, 又是kafka生产者, 为了避免死循环, 程序在打印描述性日志前都都加了'~', 创建alerter程序的预警规则时, 应当在规则中过滤日志内容第一个字符为'~'的情况

eagleye-admin部署

  • 执行附件脚本,初始化表.

  • 获取代码

    git clone git@git.xxx.com:trade_architect/eagleye.git
    
  • cd eagleye/

  • 编译

    mvn clean; mvn -DskipTests package
    
  • 在tomcat中部署eagleye-admin/target/eagleye-admin.war包。

  • 初始用户名密码:admin/test

非java项目对接

如果非java服务想对接到该预警平台, 请按照如下json格式:

   {
       "body": "日志内容",
       "head": {
           "app": "eagleye-test",
           "ip": "192.168.1.100",
           "level": "WARN",
           "pid": "3817"
       },
       "timestamp": 1426216784971
   }
   发送到kafka的 名为:EAGLEYE_LOG_CHANNEL 的topic中
   
   如果是直接接入其他非java的日志, 并且还不想格式化上述json,可以直接开启一个logstash实例, 配置文件如下:
   input {
     file {
        path => ["/home/zhejava/logstash/logstash-1.4.2/temp/*.log"]
       # type => "system"
       # start_position => "beginning"
     }
   }
   
   
   filter {
     grok {
       match => { "message" => "" }
       add_field => {
           "head.app" => "logstash-test"
           "head.ip" => "192.168.5.177"
           "head.level" => "WARN"
           #"timestamp" => 1426237376377
           #"head" => "{'app' : 'eagleye-test-b','ip' : '192.168.5.244','level' : 'ERROR','pid' : '9744'}"
       }
   
       remove_field => [ "message" ]
     }
   }
   
   
   # 如果是入kafka请这么写
   output {
     #stdout {
     #   codec => rubydebug
     #   #codec => plain
     #}
   
     kafka {
        #codec => plain {
        #   format => "%{body}"
        #}
        broker_list => "witgo:9092,spark:9092"
        topic_id => "EAGLEYE_LOG_CHANNEL"
   
     }
   }
   
   
   # 如果是入es 请这么写,不推荐直接入es, 这样就不能设置实时预警了
   output {
         elasticsearch {
         cluster => "eagleye_es"
         host => "localhost"
         port => "9301"
         protocol => "transport"
         index => "eagleye-log-%{+YYYY.MM.dd}"
         index_type => "log"
         workers => 2
         flush_size => 1000
         template_overwrite => true
         }
   }

其他

ES自定义检索功能说明

  • elastic search建立索引时, 会使用现有词典进行分词, 如果用户输入的关键词不在当前词库中, 将不会搜索到结果, 遇到这种情况, 建议用户缩小关键词粒度, 并使用多个关键词进行查询
  • 多个关键词可以用空格分开

ES index template功能简介

elasticsearch template 官方文档

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

推荐阅读更多精彩内容