1. 程式人生 > 程式設計 >golang 輸出重定向:fmt Log,子程序Log,第三方庫logrus的詳解

golang 輸出重定向:fmt Log,子程序Log,第三方庫logrus的詳解

獨立 fmt Log輸出重定向

golang的fmt包的輸出函式 Println、Printf、PrintStack等,預設將列印輸出到os.Stdout、錯誤列印輸出到os.Stderr,os.Stdout 和 os.Stderr 預設值 /dev/stdout /dev/stderr 裝置。

//程式碼摘自:golang封裝包 -> /lib/golang/src/os
var (
 Stdin = NewFile(uintptr(syscall.Stdin),"/dev/stdin")
 Stdout = NewFile(uintptr(syscall.Stdout),"/dev/stdout")
 Stderr = NewFile(uintptr(syscall.Stderr),"/dev/stderr")
)

改變 os.Stdout 和 os.Stderr 值將輸出重定向。

下面程式碼將fmt輸出重定向到/home/fmt.log檔案:

 f,_ := os.OpenFile("/home/fmt.log",os.O_WRONLY|os.O_CREATE|os.O_SYNC|os.O_APPEND,0755)
 os.Stdout = f
 os.Stderr = f

golang 第三方包:logrusLog輸出重定向

logrus Log 預設輸出到 os.Stderr

 func New() *Logger {
 return &Logger{
 Out: os.Stderr,Formatter: new(TextFormatter),Hooks: make(LevelHooks),Level: InfoLevel,}
}

logrus提供封裝方法重定向輸出流:SetOutput;

下面程式碼將fmt輸出重定向到/home/fmt.log檔案:

import (
 "github.com/Sirupsen/logrus"
)
 f,0755)
 logrus.SetOutput(f)

子程序輸出從定向

上面兩種方法都是在程式內部加入程式碼,改變程序輸出重定向輸出流。在linux shell 中我們可以使用 > 符號或者 >> 重定向標準輸出。

在golang語言中,可使用 os/exec 封裝函式啟動子程序,並可使用相關設定重定向子程序的標準輸出(影響 fmt 和 logrus 的預設設定)。

下面程式碼將fmt輸出重定向到/home/exec.log檔案;

 cmd := exec.Command(binary_name,args...)
 f,_ := os.OpenFile("/home/exec.log",os.O_WRONLY|os.O_CREATE|os.O_SYNC,0755)
 cmd.Stdout = f
 cmd.Stderr = f

補充:Go之第三方日誌庫logrus使用

第三方日誌庫logrus使用

日誌是程式中必不可少的一個環節,由於Go語言內建的日誌庫功能比較簡潔,我們在實際開發中通常會選擇使用第三方的日誌庫來進行開發。本文介紹了logrus這個日誌庫的基本使用。

logrus介紹

Logrus是Go(golang)的結構化logger,與標準庫logger完全API相容。

它有以下特點:

完全相容標準日誌庫,擁有七種日誌級別:Trace,Debug,Info,Warning,Error,Fataland Panic。

可擴充套件的Hook機制,允許使用者通過Hook的方式將日誌分發到任意地方,如本地檔案系統,logstash,elasticsearch或者mq等,或者通過Hook定義日誌內容和格式等

可選的日誌輸出格式,內建了兩種日誌格式JSONFormater和TextFormatter,還可以自定義日誌格式

Field機制,通過Filed機制進行結構化的日誌記錄

執行緒安全

安裝

$ go get github.com/sirupsen/logrus

基本示例

使用Logrus最簡單的方法是簡單的包級匯出日誌程式:

package main 
import (
 log "github.com/sirupsen/logrus"
) 
func main() {
 log.WithFields(log.Fields{
  "animal": "dog",}).Info("一條舔狗出現了。")
}

進階示例

對於更高階的用法,例如在同一應用程式記錄到多個位置,你還可以建立logrus Logger的例項:

package main 
import (
 "os"
 "github.com/sirupsen/logrus"
) 
// 建立一個新的logger例項。可以建立任意多個。
var log = logrus.New() 
func main() {
 // 設定日誌輸出為os.Stdout
 log.Out = os.Stdout
 
 // 可以設定像檔案等任意`io.Writer`型別作為日誌輸出
 // file,err := os.OpenFile("logrus.log",os.O_CREATE|os.O_WRONLY|os.O_APPEND,0666)
 // if err == nil {
 // log.Out = file
 // } else {
 // log.Info("Failed to log to file,using default stderr")
 // }
 
 log.WithFields(logrus.Fields{
  "animal": "dog","size":  10,}).Info("一群舔狗出現了。")
}

日誌級別

Logrus有七個日誌級別:Trace,Fataland Panic。

log.Trace("Something very low level.")
log.Debug("Useful debugging information.")
log.Info("Something noteworthy happened!")
log.Warn("You should probably take a look at this.")
log.Error("Something failed but I'm not quitting.")
// 記完日誌後會呼叫os.Exit(1)
log.Fatal("Bye.")
// 記完日誌後會呼叫 panic()
log.Panic("I'm bailing.")

設定日誌級別

你可以在Logger上設定日誌記錄級別,然後它只會記錄具有該級別或以上級別任何內容的條目:

// 會記錄info及以上級別 (warn,error,fatal,panic)

log.SetLevel(log.InfoLevel)   

如果你的程式支援debug或環境變數模式,設定log.Level = logrus.DebugLevel會很有幫助。

欄位

Logrus鼓勵通過日誌欄位進行謹慎的結構化日誌記錄,而不是冗長的、不可解析的錯誤訊息。

例如,區別於使用log.Fatalf("Failed to send event %s to topic %s with key %d"),你應該使用如下方式記錄更容易發現的內容:

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

WithFields的呼叫是可選的。

預設欄位

通常,將一些欄位始終附加到應用程式的全部或部分的日誌語句中會很有幫助。例如,你可能希望始終在請求的上下文中記錄request_id和user_ip。

區別於在每一行日誌中寫上log.WithFields(log.Fields{"request_id": request_id,"user_ip": user_ip}),你可以向下面的示例程式碼一樣建立一個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")

日誌條目

除了使用WithField或WithFields新增的欄位外,一些欄位會自動新增到所有日誌記錄事中:

time:記錄日誌時的時間戳

msg:記錄的日誌資訊

level:記錄的日誌級別

Hooks

你可以新增日誌級別的鉤子(Hook)。例如,向異常跟蹤服務傳送Error、Fatal和Panic、資訊到StatsD或同時將日誌傳送到多個位置,例如syslog。

Logrus配有內建鉤子。在init中新增這些內建鉤子或你自定義的鉤子:

import (
 log "github.com/sirupsen/logrus"
 "gopkg.in/gemnasium/logrus-airbrake-hook.v2" // the package is named "airbrake"
 logrus_syslog "github.com/sirupsen/logrus/hooks/syslog"
 "log/syslog"
) 
func init() {
 
 // Use the Airbrake hook to report errors that have Error severity or above to
 // an exception tracker. You can create custom hooks,see the Hooks section.
 log.AddHook(airbrake.NewHook(123,"xyz","production"))
 
 hook,err := logrus_syslog.NewSyslogHook("udp","localhost:514",syslog.LOG_INFO,"")
 if err != nil {
  log.Error("Unable to connect to local syslog daemon")
 } else {
  log.AddHook(hook)
 }
}
  

意:Syslog鉤子還支援連線到本地syslog(例如. “/dev/log” or “/var/run/syslog” or “/var/run/log”)。有關詳細資訊,請檢視syslog hook README。

格式化

logrus內建以下兩種日誌格式化程式:

logrus.TextFormatter

logrus.JSONFormatter

還支援一些第三方的格式化程式,詳見專案首頁。

記錄函式名

如果你希望將呼叫的函式名新增為欄位,請通過以下方式設定:

log.SetReportCaller(true)   

這會將呼叫者新增為”method”,如下所示:

{"animal":"penguin","level":"fatal","method":"github.com/sirupsen/arcticcreatures.migrate","msg":"a penguin swims by","time":"2014-03-10 19:57:38.562543129 -0400 EDT"}   

注意:,開啟這個模式會增加效能開銷。

執行緒安全

預設的logger在併發寫的時候是被mutex保護的,比如當同時呼叫hook和寫log時mutex就會被請求,有另外一種情況,檔案是以appending mode開啟的, 此時的併發操作就是安全的,可以用logger.SetNoLock()來關閉它。

gin框架使用logrus

// a gin with logrus demo 
var log = logrus.New() 
func init() {
  // Log as JSON instead of the default ASCII formatter.
  log.Formatter = &logrus.JSONFormatter{}
  // Output to stdout instead of the default stderr
  // Can be any io.Writer,see below for File example
  f,_ := os.Create("./gin.log")
  log.Out = f
  gin.SetMode(gin.ReleaseMode) // 設定gin框架模式 線上模式 不會在終端列印多餘的日誌資訊
  gin.DefaultWriter = log.Out
  // Only log the warning severity or above.
  log.Level = logrus.InfoLevel
} 
func main() {
  // 建立一個預設的路由引擎
  r := gin.Default()
  // GET:請求方式;/hello:請求的路徑
  // 當客戶端以GET方法請求/hello路徑時,會執行後面的匿名函式
  r.GET("/hello",func(c *gin.Context) {
    log.WithFields(logrus.Fields{
      "animal": "walrus",}).Warn("A group of walrus emerges from the ocean")
    // c.JSON:返回JSON格式的資料
    c.JSON(200,gin.H{
      "message": "Hello world!",})
  })
  // 啟動HTTP服務,預設在0.0.0.0:8080啟動服務
  r.Run()
}

以上為個人經驗,希望能給大家一個參考,也希望大家多多支援我們。如有錯誤或未考慮完全的地方,望不吝賜教。