婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > golang日志框架之logrus的使用

golang日志框架之logrus的使用

熱門標簽:廣州呼叫中心外呼系統 高德地圖標注口訣 地圖標注的汽車標 中國地圖標注省會高清 浙江高速公路地圖標注 西部云谷一期地圖標注 江西轉化率高的羿智云外呼系統 學海導航地圖標注 南通如皋申請開通400電話

golang日志庫

golang標準庫的日志框架非常簡單,僅僅提供了print,panic和fatal三個函數對于更精細的日志級別、日志文件分割以及日志分發等方面并沒有提供支持。所以催生了很多第三方的日志庫,但是在golang的世界里,沒有一個日志庫像slf4j那樣在Java中具有絕對統治地位。golang中,流行的日志框架包括logrus、zap、zerolog、seelog等。

logrus是目前Github上star數量最多的日志庫,目前(2018.08,下同)star數量為8119,fork數為1031。logrus功能強大,性能高效,而且具有高度靈活性,提供了自定義插件的功能。很多開源項目,如docker,prometheus等,都是用了logrus來記錄其日志。

zap是Uber推出的一個快速、結構化的分級日志庫。具有強大的ad-hoc分析功能,并且具有靈活的儀表盤。zap目前在GitHub上的star數量約為4.3k。

seelog提供了靈活的異步調度、格式化和過濾功能。目前在GitHub上也有約1.1k。

logrus特性

logrus具有以下特性:

  • 完全兼容golang標準庫日志模塊:logrus擁有六種日志級別:debug、info、warn、error、fatal和panic,這是golang標準庫日志模塊的API的超集。如果您的項目使用標準庫日志模塊,完全可以以最低的代價遷移到logrus上。
  • 可擴展的Hook機制:允許使用者通過hook的方式將日志分發到任意地方,如本地文件系統、標準輸出、logstash、elasticsearch或者mq等,或者通過hook定義日志內容和格式等。
  • 可選的日志輸出格式:logrus內置了兩種日志格式,JSONFormatter和TextFormatter,如果這兩個格式不滿足需求,可以自己動手實現接口Formatter,來定義自己的日志格式。
  • Field機制:logrus鼓勵通過Field機制進行精細化的、結構化的日志記錄,而不是通過冗長的消息來記錄日志。
  • logrus是一個可插拔的、結構化的日志框架。

logrus的使用

第一個示例

最簡單的使用logrus的示例如下:

package main

import (
 log "github.com/sirupsen/logrus"
)

func main() {
 log.WithFields(log.Fields{
  "animal": "walrus",
 }).Info("A walrus appears")
}

上面代碼執行后,標準輸出上輸出如下:

time="2018-08-11T15:42:22+08:00" level=info msg="A walrus appears" animal=walrus

logrus與golang標準庫日志模塊完全兼容,因此您可以使用log“github.com/sirupsen/logrus”替換所有日志導入。
logrus可以通過簡單的配置,來定義輸出、格式或者日志級別等。

package main

import (
  "os"
  log "github.com/sirupsen/logrus"
)

func init() {
  // 設置日志格式為json格式
  log.SetFormatter(log.JSONFormatter{})

  // 設置將日志輸出到標準輸出(默認的輸出為stderr,標準錯誤)
  // 日志消息輸出可以是任意的io.writer類型
  log.SetOutput(os.Stdout)

  // 設置日志級別為warn以上
  log.SetLevel(log.WarnLevel)
}

func main() {
  log.WithFields(log.Fields{
    "animal": "walrus",
    "size":  10,
  }).Info("A group of walrus emerges from the ocean")

  log.WithFields(log.Fields{
    "omg":  true,
    "number": 122,
  }).Warn("The group's number increased tremendously!")

  log.WithFields(log.Fields{
    "omg":  true,
    "number": 100,
  }).Fatal("The ice breaks!")
}

Logger

logger是一種相對高級的用法, 對于一個大型項目, 往往需要一個全局的logrus實例,即logger對象來記錄項目所有的日志。如:

package main

import (
  "github.com/sirupsen/logrus"
  "os"
)

// logrus提供了New()函數來創建一個logrus的實例。
// 項目中,可以創建任意數量的logrus實例。
var log = logrus.New()

func main() {
  // 為當前logrus實例設置消息的輸出,同樣地,
  // 可以設置logrus實例的輸出到任意io.writer
  log.Out = os.Stdout

  // 為當前logrus實例設置消息輸出格式為json格式。
  // 同樣地,也可以單獨為某個logrus實例設置日志級別和hook,這里不詳細敘述。
  log.Formatter = logrus.JSONFormatter{}

  log.WithFields(logrus.Fields{
    "animal": "walrus",
    "size":  10,
  }).Info("A group of walrus emerges from the ocean")
}

Fields

前一章提到過,logrus不推薦使用冗長的消息來記錄運行信息,它推薦使用Fields來進行精細化的、結構化的信息記錄。
例如下面的記錄日志的方式:

log.Fatalf("Failed to send event %s to topic %s with key %d", event, topic, key)
````

在logrus中不太提倡,logrus鼓勵使用以下方式替代之:





div class="se-preview-section-delimiter">/div>

```go
log.WithFields(log.Fields{
 "event": event,
 "topic": topic,
 "key": key,
}).Fatal("Failed to send event")

前面的WithFields API可以規范使用者按照其提倡的方式記錄日志。但是WithFields依然是可選的,因為某些場景下,使用者確實只需要記錄儀一條簡單的消息。

通常,在一個應用中、或者應用的一部分中,都有一些固定的Field。比如在處理用戶http請求時,上下文中,所有的日志都會有request_iduser_ip。為了避免每次記錄日志都要使用log.WithFields(log.Fields{"request_id": request_id, "user_ip": user_ip}),我們可以創建一個logrus.Entry實例,為這個實例設置默認Fields,在上下文中使用這個logrus.Entry實例記錄日志即可。

requestLogger := log.WithFields(log.Fields{"request_id": request_id, "user_ip": user_ip})
requestLogger.Info("something happened on that request") # will log request_id and user_ip
requestLogger.Warn("something not great happened")

Hook

logrus最令人心動的功能就是其可擴展的HOOK機制了,通過在初始化時為logrus添加hook,logrus可以實現各種擴展功能。

Hook接口

logrus的hook接口定義如下,其原理是每此寫入日志時攔截,修改logrus.Entry。

// logrus在記錄Levels()返回的日志級別的消息時會觸發HOOK,
// 按照Fire方法定義的內容修改logrus.Entry。
type Hook interface {
  Levels() []Level
  Fire(*Entry) error
}

一個簡單自定義hook如下,DefaultFieldHook定義會在所有級別的日志消息中加入默認字段appName="myAppName"

type DefaultFieldHook struct {
}

func (hook *DefaultFieldHook) Fire(entry *log.Entry) error {
  entry.Data["appName"] = "MyAppName"
  return nil
}

func (hook *DefaultFieldHook) Levels() []log.Level {
  return log.AllLevels
}

hook的使用也很簡單,在初始化前調用log.AddHook(hook)添加相應的hook即可。

logrus官方僅僅內置了syslog的hook。

此外,但Github也有很多第三方的hook可供使用,文末將提供一些第三方HOOK的連接。

問題與解決方案

盡管logrus有諸多優點,但是為了靈活性和可擴展性,官方也削減了很多實用的功能,例如:

  • 沒有提供行號和文件名的支持
  • 輸出到本地文件系統沒有提供日志分割功能
  • 官方沒有提供輸出到ELK等日志處理中心的功能

但是這些功能都可以通過自定義hook來實現。

記錄文件名和行號

logrus的一個很致命的問題就是沒有提供文件名和行號,這在大型項目中通過日志定位問題時有諸多不便。Github上的logrus的issue#63:Log filename and line number創建于2014年,四年過去了仍是open狀態~~~
網上給出的解決方案分位兩類,一就是自己實現一個hook;二就是通過裝飾器包裝logrus.Entry。兩種方案網上都有很多代碼,但是大多無法正常工作。但總體來說,解決問題的思路都是對的:通過標準庫的runtime模塊獲取運行時信息,并從中提取文件名,行號和調用函數名。

標準庫runtime模塊的Caller(skip int)函數可以返回當前goroutine調用棧中的文件名,行號,函數信息等,參數skip表示表示返回的棧幀的層次,0表示runtime.Caller的調用著。返回值包括響應棧幀層次的pc(程序計數器),文件名和行號信息。為了提高效率,我們先通過跟蹤調用棧發現,從runtime.Caller()的調用者開始,到記錄日志的生成代碼之間,大概有8到11層左右,所有我們在hook中循環第8到11層調用棧應該可以找到日志記錄的生產代碼。

 

此外,runtime.FuncForPC(pc uintptr) *Func可以返回指定pc的函數信息。

所有我們要實現的hook也是基于以上原理,使用runtime.Caller()依次循環調用棧的第7~11層,過濾掉sirupsen包內容,那么第一個非siupsenr包就認為是我們的生產代碼了,并返回pc以便通過runtime.FuncForPC()獲取函數名稱。然后將文件名、行號和函數名組裝為source字段塞到logrus.Entry中即可。

time="2018-08-11T19:10:15+08:00" level=warning msg="postgres_exporter is ready for scraping on 0.0.0.0:9295..." source="postgres_exporter/main.go:60:main()"
time="2018-08-11T19:10:17+08:00" level=error msg="!!!msb info not found" source="postgres/postgres_query.go:63:QueryPostgresInfo()"
time="2018-08-11T19:10:17+08:00" level=error msg="get postgres instances info failed, scrape metrics failed, error:msb env not found" source="collector/exporter.go:71:Scrape()"

日志本地文件分割

logrus本身不帶日志本地文件分割功能,但是我們可以通過file-rotatelogs進行日志本地文件分割。 每次當我們寫入日志的時候,logrus都會調用file-rotatelogs來判斷日志是否要進行切分。關于本地日志文件分割的例子網上很多,這里不再詳細介紹,奉上代碼:

import (
  "github.com/lestrrat-go/file-rotatelogs"
  "github.com/rifflock/lfshook"
  log "github.com/sirupsen/logrus"
  "time"
)

func newLfsHook(logLevel *string, maxRemainCnt uint) log.Hook {
  writer, err := rotatelogs.New(
    logName+".%Y%m%d%H",
    // WithLinkName為最新的日志建立軟連接,以方便隨著找到當前日志文件
    rotatelogs.WithLinkName(logName),

    // WithRotationTime設置日志分割的時間,這里設置為一小時分割一次
    rotatelogs.WithRotationTime(time.Hour),

    // WithMaxAge和WithRotationCount二者只能設置一個,
    // WithMaxAge設置文件清理前的最長保存時間,
    // WithRotationCount設置文件清理前最多保存的個數。
    //rotatelogs.WithMaxAge(time.Hour*24),
    rotatelogs.WithRotationCount(maxRemainCnt),
  )

  if err != nil {
    log.Errorf("config local file system for logger error: %v", err)
  }

  level, ok := logLevels[*logLevel]

  if ok {
    log.SetLevel(level)
  } else {
    log.SetLevel(log.WarnLevel)
  }

  lfsHook := lfshook.NewHook(lfshook.WriterMap{
    log.DebugLevel: writer,
    log.InfoLevel: writer,
    log.WarnLevel: writer,
    log.ErrorLevel: writer,
    log.FatalLevel: writer,
    log.PanicLevel: writer,
  }, log.TextFormatter{DisableColors: true})

  return lfsHook
}

使用上述本地日志文件切割的效果如下:

將日志發送到elasticsearch

將日志發送到elasticsearch是很多日志監控系統的選擇,將logrus日志發送到elasticsearch的原理是在hook的每次fire調用時,使用golang的es客戶端將日志信息寫到elasticsearch。elasticsearch官方沒有提供golang客戶端,但是有很多第三方的go語言客戶端可供使用,我們選擇elastic。elastic提供了豐富的文檔,以及Java中的流式接口,使用起來非常方便。

client, err := elastic.NewClient(elastic.SetURL("http://localhost:9200"))
  if err != nil {
    log.Panic(err)
  }

// Index a tweet (using JSON serialization)
tweet1 := Tweet{User: "olivere", Message: "Take Five", Retweets: 0}
put1, err := client.Index().
  Index("twitter").
  Type("tweet").
  Id("1").
  BodyJson(tweet1).
  Do(context.Background())

考慮到logrus的Fields機制,可以實現如下數據格式:

msg := struct {
  Host   string
  Timestamp string `json:"@timestamp"`
  Message  string
  Data   logrus.Fields
  Level   string
}

其中Host記錄產生日志主機信息,在創建hook是指定。其他數據需要從logrus.Entry中取得。測試過程我們選擇按照此原理實現的第三方HOOK:elogrus。其使用如下:

import (
  "github.com/olivere/elastic"
  "gopkg.in/sohlich/elogrus"
)

func initLog() {
  client, err := elastic.NewClient(elastic.SetURL("http://localhost:9200"))
  if err != nil {
    log.Panic(err)
  }
  hook, err := elogrus.NewElasticHook(client, "localhost", log.DebugLevel, "mylog")
  if err != nil {
    log.Panic(err)
  }
  log.AddHook(hook)
}

從Elasticsearch查詢得到日志存儲,效果如下:

GET http://localhost:9200/mylog/_search

HTTP/1.1 200 OK
content-type: application/json; charset=UTF-8
transfer-encoding: chunked

{
 "took": 1,
 "timed_out": false,
 "_shards": {
  "total": 5,
  "successful": 5,
  "failed": 0
 },
 "hits": {
  "total": 2474,
  "max_score": 1.0,
  "hits": [
   {
    "_index": "mylog",
    "_type": "log",
    "_id": "AWUw13jWnMZReb-jHQup",
    "_score": 1.0,
    "_source": {
     "Host": "localhost",
     "@timestamp": "2018-08-13T01:12:32.212818666Z",
     "Message": "!!!msb info not found",
     "Data": {},
     "Level": "ERROR"
    }
   },
   {
    "_index": "mylog",
    "_type": "log",
    "_id": "AWUw13jgnMZReb-jHQuq",
    "_score": 1.0,
    "_source": {
     "Host": "localhost",
     "@timestamp": "2018-08-13T01:12:32.223103348Z",
     "Message": "get postgres instances info failed, scrape metrics failed, error:msb env not found",
     "Data": {
      "source": "collector/exporter.go:71:Scrape()"
     },
     "Level": "ERROR"
    }
   },
   //...
   {
    "_index": "mylog",
    "_type": "log",
    "_id": "AWUw2f1enMZReb-jHQu_",
    "_score": 1.0,
    "_source": {
     "Host": "localhost",
     "@timestamp": "2018-08-13T01:15:17.212546892Z",
     "Message": "!!!msb info not found",
     "Data": {
      "source": "collector/exporter.go:71:Scrape()"
     },
     "Level": "ERROR"
    }
   },
   {
    "_index": "mylog",
    "_type": "log",
    "_id": "AWUw2NhmnMZReb-jHQu1",
    "_score": 1.0,
    "_source": {
     "Host": "localhost",
     "@timestamp": "2018-08-13T01:14:02.21276903Z",
     "Message": "!!!msb info not found",
     "Data": {},
     "Level": "ERROR"
    }
   }
  ]
 }
}

將日志發送到其他位置

將日志發送到日志中心也是logrus所提倡的,雖然沒有提供官方支持,但是目前Github上有很多第三方hook可供使用:

logrus_amqp:Logrus hook for Activemq。

logrus-logstash-hook:Logstash hook for logrus。

mgorus:Mongodb Hooks for Logrus。

logrus_influxdb:InfluxDB Hook for Logrus。

logrus-redis-hook:Hook for Logrus which enables logging to RELK stack (Redis, Elasticsearch, Logstash and Kibana)。

等等,上述第三方hook我這里沒有具體驗證,大家可以根據需要自行嘗試。

其他注意事項

Fatal處理

和很多日志框架一樣,logrus的Fatal系列函數會執行os.Exit(1)。但是logrus提供可以注冊一個或多個fatal handler函數的接口logrus.RegisterExitHandler(handler func() {} ),讓logrus在執行os.Exit(1)之前進行相應的處理。fatal handler可以在系統異常時調用一些資源釋放api等,讓應用正確的關閉。

線程安全

默認情況下,logrus的api都是線程安全的,其內部通過互斥鎖來保護并發寫。互斥鎖工作于調用hooks或者寫日志的時候,如果不需要鎖,可以調用logger.SetNoLock()來關閉之。可以關閉logrus互斥鎖的情形包括:

  • 沒有設置hook,或者所有的hook都是線程安全的實現。
  • 寫日志到logger.Out已經是線程安全的了,如logger.Out已經被鎖保護,或者寫文件時,文件是以O_APPEND方式打開的,并且每次寫操作都小于4k。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • mac下安裝golang框架iris的方法
  • golang常用庫之操作數據庫的orm框架-gorm基本使用詳解
  • golang 網絡框架之gin的使用方法
  • 詳解Golang Iris框架的基本使用

標簽:德宏 曲靖 東營 常州 許昌 吐魯番 貴州 保定

巨人網絡通訊聲明:本文標題《golang日志框架之logrus的使用》,本文關鍵詞  golang,日志,框架,之,logrus,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《golang日志框架之logrus的使用》相關的同類信息!
  • 本頁收集關于golang日志框架之logrus的使用的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    亚洲五码中文字幕| 国产麻豆午夜三级精品| 在线播放91灌醉迷j高跟美女 | 91国产免费观看| 国产美女av一区二区三区| 一区二区视频免费在线观看| 欧美日韩免费一区二区三区| 国产大片一区二区| 毛片av中文字幕一区二区| 久久久精品免费免费| 欧美三级资源在线| 高潮精品一区videoshd| 久久精品国内一区二区三区| 无码av免费一区二区三区试看| 777午夜精品免费视频| 91小视频在线| 国产一区二区按摩在线观看| 亚洲欧美另类小说视频| 欧美韩国一区二区| 欧美二区在线观看| 国产宾馆实践打屁股91| 一区二区三区免费网站| 一区二区三区.www| 日韩电影在线看| 香蕉久久夜色精品国产使用方法| 国产精品福利在线播放| 337p日本欧洲亚洲大胆色噜噜| 久久亚洲一区二区三区明星换脸 | 免费xxxx性欧美18vr| 久久99精品国产麻豆婷婷| 亚洲小说欧美激情另类| 亚洲色图欧美在线| 国内精品国产成人| 99久久综合99久久综合网站| 不卡的看片网站| 91在线观看高清| 久久99精品久久久久久动态图 | 一本到高清视频免费精品| 91亚洲精品久久久蜜桃| 成人国产精品免费观看视频| www.亚洲免费av| 东方欧美亚洲色图在线| 国产成人av电影在线| 99免费精品视频| 欧美丰满嫩嫩电影| 欧美一区二区福利视频| 久久久久久久综合色一本| 亚洲在线免费播放| 国产激情一区二区三区桃花岛亚洲| 波波电影院一区二区三区| 白白色亚洲国产精品| 欧美老肥妇做.爰bbww| 欧美国产1区2区| 日欧美一区二区| 国产福利精品一区| 欧美一区二区国产| 欧美大片在线观看| 亚洲一二三四在线| 国产一区二区不卡| 色丁香久综合在线久综合在线观看| 91行情网站电视在线观看高清版| 国产午夜精品久久久久久久 | 豆国产96在线|亚洲| 国产欧美日韩激情| 国产呦精品一区二区三区网站| av午夜一区麻豆| 久久精品视频网| 精品一区二区三区久久| 精品美女一区二区三区| 亚洲综合免费观看高清完整版在线| 成人免费视频caoporn| 亚洲色图在线视频| 国产宾馆实践打屁股91| 日韩精品综合一本久道在线视频| 欧美日韩精品一二三区| 三级欧美韩日大片在线看| 色狠狠一区二区三区香蕉| 国产精品视频你懂的| 成人性视频免费网站| 欧美电影精品一区二区| 国产九九视频一区二区三区| 一区二区三区在线视频免费| 欧美婷婷六月丁香综合色| 亚洲欧美日韩人成在线播放| 国产精品456| 日韩福利视频网| 欧美一区二区播放| 国产一区二区看久久| 日韩中文字幕区一区有砖一区| 日韩三级高清在线| 久久超碰97人人做人人爱| 国产精品蜜臀av| www国产亚洲精品久久麻豆| 国产69精品久久99不卡| 国产精品久久一级| 69p69国产精品| 97aⅴ精品视频一二三区| 亚洲成人综合网站| 欧美国产成人在线| 精品福利一二区| 99久久亚洲一区二区三区青草| 亚洲一区二区三区四区不卡| 欧美成人一区二区三区| 欧美在线播放高清精品| 九九精品一区二区| 青娱乐精品视频在线| 国产一区二区三区精品欧美日韩一区二区三区 | 色中色一区二区| 韩国毛片一区二区三区| 青青草国产成人99久久| 精品一区在线看| 成人一级黄色片| 色偷偷88欧美精品久久久| 三级不卡在线观看| 日韩美女啊v在线免费观看| 日本一区二区三区电影| 中文字幕不卡三区| 一区二区免费在线播放| 亚洲第一av色| 国产精一区二区三区| 色婷婷久久久久swag精品| 精品成人一区二区| www..com久久爱| 中文字幕在线观看不卡视频| 欧美精品久久一区| 99精品视频在线免费观看| 亚洲伦理在线精品| 97久久超碰国产精品| 亚洲国产一区二区a毛片| 精品免费视频.| 1000部国产精品成人观看| 56国语精品自产拍在线观看| 欧美sm美女调教| 91麻豆精品国产91| 欧美一区二区三区日韩视频| 国产精品毛片高清在线完整版| 中文字幕一区二区三区蜜月 | 亚洲综合色区另类av| 精品国产精品一区二区夜夜嗨| 久久久91精品国产一区二区三区| 亚洲欧美影音先锋| 久久爱另类一区二区小说| 韩国视频一区二区| 一本一本久久a久久精品综合麻豆| 欧美亚洲一区二区在线| 欧美一区二区三区免费| 亚洲福利一区二区三区| 美女视频免费一区| 欧美亚洲国产一区二区三区| 国产精品视频在线看| 国产69精品久久777的优势| 欧美电影影音先锋| 亚洲欧美欧美一区二区三区| 国产91精品久久久久久久网曝门| 欧美tk—视频vk| 亚洲成a人片在线不卡一二三区| 不卡电影一区二区三区| 中文乱码免费一区二区| 自拍偷拍欧美精品| 99久久精品99国产精品| 国产午夜亚洲精品羞羞网站| 韩国成人精品a∨在线观看| 日韩小视频在线观看专区| 亚洲成a人在线观看| 在线一区二区视频| 国产在线不卡视频| 最新中文字幕一区二区三区| 粉嫩av一区二区三区在线播放| 欧美综合在线视频| 国产在线播放一区| 亚洲欧美日韩在线不卡| 成人性视频免费网站| 亚洲视频中文字幕| 欧美日精品一区视频| 九色综合狠狠综合久久| 久久久美女艺术照精彩视频福利播放| 精品一区二区三区久久| 国产色91在线| 欧美日韩日日骚| 国产一区91精品张津瑜| 中文字幕中文乱码欧美一区二区| 欧美日韩一级二级| 成人免费毛片片v| 免费精品99久久国产综合精品| 久久综合久久99| 欧美色爱综合网| 91视视频在线观看入口直接观看www | 亚洲男人天堂av| 欧美熟乱第一页| 91丨国产丨九色丨pron| 麻豆91精品视频| 亚洲一二三区视频在线观看| 国产精品国产精品国产专区不蜜| 欧美日韩精品一区二区三区蜜桃| 欧美亚洲综合网| 成人动漫一区二区三区| 激情文学综合丁香| 精品一区二区成人精品| 中文字幕一区二区三区蜜月| 精品久久五月天|