日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

url能访问但new file()找不到文件_Go Web编程给自己写的服务器添加错误和访问日志...

發(fā)布時(shí)間:2024/8/1 编程问答 33 豆豆
生活随笔 收集整理的這篇文章主要介紹了 url能访问但new file()找不到文件_Go Web编程给自己写的服务器添加错误和访问日志... 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

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

初始化日志記錄器

我們先來做一下初始化工作,在項(xiàng)目里初始化記錄錯(cuò)誤日志和訪問日志的記錄器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)
}
  • 我們新定義一個(gè)package在init函數(shù)中來初始化記錄器,這樣服務(wù)器成功啟動(dòng)前就會初始化好記錄器。

  • /tmp/log這個(gè)目錄要提前創(chuàng)建好,執(zhí)行init函數(shù)時(shí)會自動(dòng)創(chuàng)建好access.log和error.log。

添加錯(cuò)誤日志

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

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ù)器的時(shí)候自己實(shí)現(xiàn)了Server類型的對象,那么現(xiàn)在要做的就是將上面初始化好的錯(cuò)誤日志的記錄器指定給Server的ErrorLog字段。

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),
????}
????...
}

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

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

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

添加訪問日志

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

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

type?ResponseWriter?interface?{
????Header()?Header

????Write([]byte)?(int,?error)

????WriteHeader(statusCode?int)
}

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

利用Go語言結(jié)構(gòu)體類型嵌套匿名類型后,結(jié)構(gòu)體擁有了被嵌套類型的所有導(dǎo)出字段和方法的特性,我們可以很方便地對原來的ResponseWriter做一層包裝,然后只重新實(shí)現(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
}

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

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

func?AccessLogging?(f?http.Handler)?http.Handler?{

????//?創(chuàng)建一個(gè)新的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)用下一個(gè)中間件或者最終的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)
????...
}

不過有兩點(diǎn)需要注意一下

  • 這里為了演示獲取響應(yīng)數(shù)據(jù)記錄了response_body字段,如果是接口響應(yīng)內(nèi)容記錄下還可以,但是如果是HTML還是不記錄的為好。

  • 初始化ResponseWithRecorder時(shí)默認(rèn)設(shè)置了statusCode是因?yàn)?#xff0c;服務(wù)器正確返回響應(yīng)時(shí)不會顯式調(diào)用WriteHeader方法,只有在返回NOT_FOUND之類的錯(cuò)誤的時(shí)候才會調(diào)用WriteHeader方法,針對這種情況需要在初始化的時(shí)候把statusCode的默認(rèn)值設(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的方法無法獲取代理后的真實(shí)IP,請悉知。

推薦閱讀

  • Go Web編程--SecureCookie實(shí)現(xiàn)客戶端Session管理


喜歡本文的朋友,歡迎關(guān)注“Go語言中文網(wǎng)”:

Go語言中文網(wǎng)啟用微信學(xué)習(xí)交流群,歡迎加微信:274768166,投稿亦歡迎

總結(jié)

以上是生活随笔為你收集整理的url能访问但new file()找不到文件_Go Web编程给自己写的服务器添加错误和访问日志...的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。