Prometheus AlertManager 兼容飞书

本文通过修改alermanager的webhook方式调用飞书,发送飞书卡片

发送机制

webhook的实现位于alertmanager/notify/webhook.go
总的来说就是包装msg并发送http请求,重点在于需要修改的是msg对象。

// Notify implements the Notifier interface.
func (n *Notifier) Notify(ctx context.Context, alerts ...*types.Alert) (bool, error) {
    alerts, numTruncated := truncateAlerts(n.conf.MaxAlerts, alerts)
    data := notify.GetTemplateData(ctx, n.tmpl, alerts, n.logger)

    groupKey, err := notify.ExtractGroupKey(ctx)
    if err != nil {
        level.Error(n.logger).Log("err", err)
    }

    msg := &Message{
        Version:         "4",
        Data:            data,
        GroupKey:        groupKey.String(),
        TruncatedAlerts: numTruncated,
    }

    var buf bytes.Buffer
    if err := json.NewEncoder(&buf).Encode(msg); err != nil {
        return false, err
    }

    req, err := http.NewRequest("POST", n.conf.URL.String(), &buf)
    if err != nil {
        return true, err
    }
    req.Header.Set("Content-Type", "application/json")
    req.Header.Set("User-Agent", userAgentHeader)

    resp, err := n.client.Do(req.WithContext(ctx))
    if err != nil {
        return true, err
    }
    notify.Drain(resp)

    return n.retrier.Check(resp.StatusCode, nil)
}

Data对象

type Data struct {
    Receiver string `json:"receiver"`
    Status   string `json:"status"`
    Alerts   Alerts `json:"alerts"`

    GroupLabels       KV `json:"groupLabels"`
    CommonLabels      KV `json:"commonLabels"`
    CommonAnnotations KV `json:"commonAnnotations"`

    ExternalURL string `json:"externalURL"`
}

对应序列化之后的json(取自官方文档)

{
  "version": "4",
  "groupKey": <string>,              // key identifying the group of alerts (e.g. to deduplicate)
  "truncatedAlerts": <int>,          // how many alerts have been truncated due to "max_alerts"
  "status": "<resolved|firing>",
  "receiver": <string>,
  "groupLabels": <object>,
  "commonLabels": <object>,
  "commonAnnotations": <object>,
  "externalURL": <string>,           // backlink to the Alertmanager.
  "alerts": [
    {
      "status": "<resolved|firing>",
      "labels": <object>,
      "annotations": <object>,
      "startsAt": "<rfc3339>",
      "endsAt": "<rfc3339>",
      "generatorURL": <string>       // identifies the entity that caused the alert
    },
  ]
}

飞书所需格式


飞书格式.png

开搞

把msg转换一下就完事了
这是转换类,需要注意的一点是我转换的Annotations里面的行对应的是rule.yml里配置的annotation,因为我配置的三行信息是项目名,环境,ip所以将三行信息写了进去。

package webhook

import (
    "strings"
    "time"
)

type FSMessagev2 struct {
    MsgType string `json:"msg_type"`
    Card    Cards  `json:"card"`
}
type Te struct {
    Content string `json:"content,omitempty"`
    Tag     string `json:"tag,omitempty"`
}

type Element struct {
    Tag      string `json:"tag,omitempty"`
    Text     Te     `json:"text,omitempty"`
    Elements []Te   `json:"elements,omitempty"`
}

type Cards struct {
    Config   Conf      `json:"config"`
    Elements []Element `json:"elements"`
}
type Conf struct {
    WideScreenMode bool `json:"wide_screen_mode"`
}

func transformMsgToFs(message *Message) *FSMessagev2 {
    var NEXT_LINE = "\n"
    list := make([]Element, 0)
    for index, msg := range message.Data.Alerts {
        var text = ""
        if index == 0 {
            text = "**" + msg.Labels.Values()[0] + "**" + NEXT_LINE
        }
        app := strings.Trim(msg.Annotations.Values()[0],"_")
        env := strings.Trim(msg.Annotations.Values()[1],"_")
        ip := strings.Trim(msg.Annotations.Values()[2],"_")
        text += "["+app+"("+env+")]("+"http://gitlab.hio.cn/hio/"+app+"/tree/"+env+")    " + ip + NEXT_LINE
        if msg.Status == "firing" {
            msg.Status = "🔥"
            text += "状态:" + msg.Status
            list = append(list, Element{Tag: "div", Text: Te{Tag: "lark_md", Content: text}})
            list = append(list, Element{Tag: "note", Elements: []Te{{Tag: "plain_text", Content: msg.StartsAt.In(time.Local).Format("2006-01-02 15:04:05")}}})
        } else {
            msg.Status = "✅"
            text += "状态:" + msg.Status
            list = append(list, Element{Tag: "div", Text: Te{Tag: "lark_md", Content: text}})
            list = append(list, Element{Tag: "note", Elements: []Te{{Tag: "plain_text", Content: msg.StartsAt.In(time.Local).Format("2006-01-02 15:04:05") + " ~ " + msg.EndsAt.In(time.Local).Format("2006-01-02 15:04:05")}}})
        }
        if index < len(message.Data.Alerts)-1{
            list = append(list, Element{Tag: "hr"})
        }
    }
    feishu := &FSMessagev2{
        MsgType: "interactive",
        Card: Cards{
            Config: Conf{
                WideScreenMode: false,
            },
            Elements: list,
        },
    }
    return feishu
}


将转换类在webhook中调用

func (n *Notifier) Notify(ctx context.Context, alerts ...*types.Alert) (bool, error) {
    alerts, numTruncated := truncateAlerts(n.conf.MaxAlerts, alerts)
    data := notify.GetTemplateData(ctx, n.tmpl, alerts, n.logger)

    groupKey, err := notify.ExtractGroupKey(ctx)
    if err != nil {
        level.Error(n.logger).Log("err", err)
    }

    msg := &Message{
        Version:         "4",
        Data:            data,
        GroupKey:        groupKey.String(),
        TruncatedAlerts: numTruncated,
    }
        //变化在这里
    fs := transformMsgToFs(msg)
    //b,_ := json.Marshal(fs)
    //fmt.Println(string(b))
     buf := new(bytes.Buffer)
        //变化在这里
    json.NewEncoder(buf).Encode(fs)
    var tr *http.Transport
    tr = &http.Transport{
        TLSClientConfig: &tls.Config{InsecureSkipVerify: true},
    }
    client := &http.Client{Transport: tr}
    res, err := client.Post(n.conf.URL.String(), "application/json", buf)
    if err != nil {
        return true, err
    }
    defer res.Body.Close()
    notify.Drain(res)
    return n.retrier.Check(res.StatusCode, nil)
}

我的告警格式


image.png

对应的label设置

  # 从eureka拉取微服务信息,通过relable替换为采集path,需配合eureka.instantce.metaMap的配置
  - job_name: production
    eureka_sd_configs:
     - server: 'http://XXXX/eureka'
     - server: 'http://XXXXX/eureka'
    relabel_configs:
      - source_labels: [ __meta_eureka_app_instance_metadata_prometheus_path ]
        action: replace
        target_label: __metrics_path__
        regex: (.+)
      - source_labels: [ __meta_eureka_app_instance_port ]
        action: keep
        regex: (8086|8081)
      - source_labels: [ __meta_eureka_app_instance_ip_addr ]
        action: replace
        target_label: instance
      - source_labels: [ job ]
        target_label: env
      - action: labeldrop
        regex: (job)
#      - source_labels: [ __meta_eureka_app_instance_metadata_prometheus_env ]
#        target_label: env
      - source_labels: [ __meta_eureka_app_instance_vip_address ]
        target_label: application

飞书结果


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

推荐阅读更多精彩内容