欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Go?Web編程添加服務(wù)器錯誤和訪問日志

 更新時間:2022年06月25日 11:24:56   作者:KevinYan11  
這篇文章主要為大家介紹了Go?Web編程添加服務(wù)器錯誤日志和訪問日志的示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

前言

錯誤日志和訪問日志是一個服務(wù)器必須支持的功能,我們教程里使用的服務(wù)器到目前為止還沒有這兩個功能。正好前兩天也寫了篇介紹logrus日志庫的文章,那么今天的文章里就給我們自己寫的服務(wù)器加上錯誤日志和訪問日志的功能。在介紹添加訪問日志的時候會介紹一種通過編寫中間件獲取HTTP響應(yīng)的StausCodeBody的方法。

Go Web 編程系列的每篇文章的源代碼都打了對應(yīng)版本的軟件包,供大家參考。公眾號中回復(fù)gohttp11獲取本文源代碼

初始化日志記錄器

我們先來做一下初始化工作,在項目里初始化記錄錯誤日志和訪問日志的記錄器Logger

//?./utils/vlog
package?vlog
import?(
????"github.com/sirupsen/logrus"
????"os"
)
var?ErrorLog?*logrus.Logger
var?AccessLog?*logrus.Logger
var?errorLogFile?=?"./tmp/log/error.log"
var?accessLogFile?=?"./tmp/log/access.log"
func?init?()?{
????initErrorLog()
????initAccessLog()
}
func?initErrorLog()?{
????ErrorLog?=?logrus.New()
????ErrorLog.SetFormatter(&logrus.JSONFormatter{})
????file?,?err?:=?os.OpenFile(errorLogFile,?os.O_RDWR?|?os.O_CREATE?|?os.O_APPEND,?0755)
????if?err?!=?nil?{
????????panic(err)
????}
????ErrorLog.SetOutput(file)
}
func?initAccessLog()?{
????AccessLog?=?logrus.New()
????AccessLog.SetFormatter(&logrus.JSONFormatter{})
????file?,?err?:=?os.OpenFile(accessLogFile,?os.O_RDWR?|?os.O_CREATE?|?os.O_APPEND,?0755)
????if?err?!=?nil?{
????????panic(err)
????}
????AccessLog.SetOutput(file)
}
  • 我們新定義一個packageinit函數(shù)中來初始化記錄器,這樣服務(wù)器成功啟動前就會初始化好記錄器。
  • /tmp/log這個目錄要提前創(chuàng)建好,執(zhí)行init函數(shù)時會自動創(chuàng)建好access.logerror.log。

添加錯誤日志

我們創(chuàng)建服務(wù)器使用的net/http包的Server類型中,有一個ErrorLog字段供開發(fā)者設(shè)置記錄錯誤日志用的記錄器Logger,默認使用的是log包默認的記錄器(應(yīng)該是系統(tǒng)的標準錯誤):

type?Server?struct?{
???Addr????string??//?TCP?address?to?listen?on,?":http"?if?empty
???Handler?Handler?//?handler?to?invoke,?http.DefaultServeMux?if?nil
???...
???//?ErrorLog?specifies?an?optional?logger?for?errors?accepting
???//?connections,?unexpected?behavior?from?handlers,?and
???//?underlying?FileSystem?errors.
???//?If?nil,?logging?is?done?via?the?log?package's?standard?logger.
???ErrorLog?*log.Logger
?????...
}

我們之前在創(chuàng)建服務(wù)器的時候自己實現(xiàn)了Server類型的對象,那么現(xiàn)在要做的就是將上面初始化好的錯誤日志的記錄器指定給ServerErrorLog字段。

func?main()?{
??...
? ??//?將logrus的Logger轉(zhuǎn)換為io.Writer
????errorWriter?:=?vlog.ErrorLog.Writer()
? ??//?記得關(guān)閉io.Writer
????defer?errorWriter.Close()
????server?:=?&http.Server{
????????Addr:????":8080",
????????Handler:?muxRouter,
? ? ? ??//?用記錄器轉(zhuǎn)換成的io.Writer創(chuàng)建log.Logger
????????ErrorLog:?log.New(vlog.ErrorLog.Writer(),?"",?0),
????}
????...
}

添加好錯誤日志的記錄器后,我們找個路由處理函數(shù),在里面故意制造運行時錯誤驗證一下是否能記錄到錯誤。

func?(*HelloHandler)?ServeHTTP(w?http.ResponseWriter,?r?*http.Request)?{
???ints?:=?[]int{0,?1,?2}
???fmt.Fprintf(w,?"%v",?ints[0:5])
}

在上面處理函數(shù)中,通過切片表達式越界故意制造了一個運行時錯誤,打開error.log后能看到文件里已經(jīng)記錄到這個運行時錯誤及其Stack trace

添加訪問日志

Server對象可以設(shè)置錯誤日志的記錄器不一樣,訪問日志只能是我們通過自己編寫中間件的方式來實現(xiàn)了。在記錄訪問日志的中間件里我們會記錄ipmethod,pathquery,request_body,statusresponse_body這些個字段的內(nèi)容。

statusresponse_body兩個字段來自請求對應(yīng)的響應(yīng)。響應(yīng)在net/http包里是用http.ResponseWriter接口表示的

type?ResponseWriter?interface?{
????Header()?Header
????Write([]byte)?(int,?error)
????WriteHeader(statusCode?int)
}

接口本身以及net/http提供的實現(xiàn)都沒有讓我們進行讀取的方法,所以在編寫的用于記錄訪問日志的中間件里需要對net/http庫本身實現(xiàn)的ResponseWriter做一層包裝。

利用Go語言結(jié)構(gòu)體類型嵌套匿名類型后,結(jié)構(gòu)體擁有了被嵌套類型的所有導(dǎo)出字段和方法的特性,我們可以很方便地對原來的ResponseWriter做一層包裝,然后只重新實現(xiàn)需要更改的方法即可:

type?ResponseWithRecorder?struct?{
???http.ResponseWriter
???statusCode?int
???body?bytes.Buffer
}
func?(rec?*ResponseWithRecorder)?WriteHeader(statusCode?int)?{
???rec.ResponseWriter.WriteHeader(statusCode)
???rec.statusCode?=?statusCode
}
func?(rec?*ResponseWithRecorder)?Write(d?[]byte)?(n?int,?err?error)?{
???n,?err?=?rec.ResponseWriter.Write(d)
???if?err?!=?nil?{
??????return
???}
???rec.body.Write(d)
???return
}

定義好新的類型后我們重新實現(xiàn)了WriteHeaderWrite方法,在向原來的ReponseWriter中寫入后也會向ResponseWriteRecoder.statusCode和ResponseWriteRecoder.body寫入對應(yīng)的數(shù)據(jù)。這樣我們就可以在中間件里通過這兩個字段訪問響應(yīng)碼和響應(yīng)數(shù)據(jù)了。

記錄訪問日志的中間件定義如下:

func?AccessLogging?(f?http.Handler)?http.Handler?{
????//?創(chuàng)建一個新的handler包裝http.HandlerFunc
????return?http.HandlerFunc(func(w?http.ResponseWriter,?r?*http.Request)?{
????????buf?:=?new(bytes.Buffer)
????????buf.ReadFrom(r.Body)
????????logEntry?:=?vlog.AccessLog.WithFields(logrus.Fields{
????????????"ip":?r.RemoteAddr,
????????????"method":?r.Method,
????????????"path":?r.RequestURI,
????????????"query":?r.URL.RawQuery,
????????????"request_body":?buf.String(),
????????})
????????wc?:=?&ResponseWithRecorder{
????????????ResponseWriter:?w,
????????????statusCode:?http.StatusOK,
????????????body:?bytes.Buffer{},
????????}
????????//?調(diào)用下一個中間件或者最終的handler處理程序
????????f.ServeHTTP(wc,?r)
????????defer?logEntry.WithFields(logrus.Fields{
????????????"status":?wc.statusCode,
????????????"response_body":?wc.body.String(),
????????}).Info()
????})
}

Router上應(yīng)用創(chuàng)建好的AccessLogging中間件后,就可以正常的記錄服務(wù)器的訪問日志了。

//?router/router.go
func?RegisterRoutes(r?*mux.Router)?{
????...
????//?apply?Logging?middleware
????r.Use(middleware.Logging(),?middleware.AccessLogging)
????...
}

不過有兩點需要注意一下

  • 這里為了演示獲取響應(yīng)數(shù)據(jù)記錄了response_body字段,如果是接口響應(yīng)內(nèi)容記錄下還可以,但是如果是HTML還是不記錄的為好。
  • 初始化ResponseWithRecorder時默認設(shè)置了statusCode是因為,服務(wù)器正確返回響應(yīng)時不會顯式調(diào)用WriteHeader方法,只有在返回NOT_FOUND之類的錯誤的時候才會調(diào)用WriteHeader方法,針對這種情況需要在初始化的時候把statusCode的默認值設(shè)置為200。

現(xiàn)在再訪問服務(wù)器后打開access.log會看到剛剛的訪問日志,就能看到剛剛請求的url,method,客戶端IP等信息了。

{"ip":"......","level":"info","method":"GET","msg":"","path":"/index/","query":"","request_body":"","response_body":"Hello World1","status":200,"time":"2020-03-26T04:21:46Z"}

注意:文章只為說明演示方便,獲取IP的方法無法獲取代理后的真實IP,請悉知。

以上就是Go Web編程添加服務(wù)器錯誤和訪問日志的詳細內(nèi)容,更多關(guān)于Go Web服務(wù)器錯誤訪問日志的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 在Go語言項目中使用Zap日志庫的操作過程

    在Go語言項目中使用Zap日志庫的操作過程

    本文將先介紹Go語言原生的日志庫的使用,然后詳細介紹非常流行的Uber開源的zap日志庫,同時會介紹如何搭配·Lumberjack·實現(xiàn)日志的切割和歸檔,對Go使用Zap日志庫相關(guān)知識感興趣的朋友一起看看吧
    2024-03-03
  • 詳解Go語言如何判斷兩個對象是否相等

    詳解Go語言如何判斷兩個對象是否相等

    在編程中,判斷兩個對象是否相等是一項常見的任務(wù),同時判斷對象是否相等在很多情況下都非常重要,所以在接下來的內(nèi)容中,我們將詳細介紹在?Go?語言中如何判斷對象是否相等的方法和技巧,需要的可以參考一下
    2023-06-06
  • Go?實現(xiàn)?WebSockets之創(chuàng)建?WebSockets

    Go?實現(xiàn)?WebSockets之創(chuàng)建?WebSockets

    這篇文章主要介紹了Go?實現(xiàn)?WebSockets之創(chuàng)建?WebSockets,文章主要探索?WebSockets,并簡要介紹了它們的工作原理,并仔細研究了全雙工通信,想了解更多相關(guān)內(nèi)容的小伙伴可以參考一下
    2022-04-04
  • golang實現(xiàn)openssl自簽名雙向認證的詳細步驟

    golang實現(xiàn)openssl自簽名雙向認證的詳細步驟

    這篇文章主要介紹了golang實現(xiàn)openssl自簽名雙向認證的詳細步驟,本文分步驟給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧
    2024-03-03
  • Go語言學(xué)習(xí)之數(shù)組的用法詳解

    Go語言學(xué)習(xí)之數(shù)組的用法詳解

    數(shù)組是相同數(shù)據(jù)類型的一組數(shù)據(jù)的集合,數(shù)組一旦定義長度不能修改,數(shù)組可以通過下標(或者叫索引)來訪問元素。本文將通過示例詳細講解Go語言中數(shù)組的使用,需要的可以參考一下
    2022-04-04
  • Golang將Map的鍵值對調(diào)的實現(xiàn)示例

    Golang將Map的鍵值對調(diào)的實現(xiàn)示例

    本文主要介紹了Golang將Map的鍵值對調(diào)的實現(xiàn)示例,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-02-02
  • golang中日期操作之日期格式化及日期轉(zhuǎn)換

    golang中日期操作之日期格式化及日期轉(zhuǎn)換

    在編程中,程序員會經(jīng)常使用到日期相關(guān)操作,下面這篇文章主要給大家介紹了關(guān)于golang中日期操作之日期格式化及日期轉(zhuǎn)換的相關(guān)資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下
    2022-11-11
  • Go到底能不能實現(xiàn)安全的雙檢鎖(推薦)

    Go到底能不能實現(xiàn)安全的雙檢鎖(推薦)

    這篇文章主要介紹了Go到底能不能實現(xiàn)安全的雙檢鎖,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-05-05
  • 詳解Go語言如何實現(xiàn)二叉樹遍歷

    詳解Go語言如何實現(xiàn)二叉樹遍歷

    這篇文章主要為大家詳解介紹了Go語言中如何實現(xiàn)二叉樹遍歷,文中的示例代碼講解詳細,對我們學(xué)習(xí)Go語言有一定幫助,需要的可以參考一下
    2022-04-04
  • go浮點數(shù)轉(zhuǎn)字符串保留小數(shù)點后N位的完美解決方法

    go浮點數(shù)轉(zhuǎn)字符串保留小數(shù)點后N位的完美解決方法

    這篇文章主要介紹了go浮點數(shù)轉(zhuǎn)字符串保留小數(shù)點后N位解決辦法,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-05-05

最新評論