寫(xiě)的接口給測(cè)試測(cè)試,現(xiàn)在還沒(méi)有頁(yè)面,直接測(cè)試接口。使用
總是報(bào)錯(cuò),大致錯(cuò)誤信息如下:
err="invalid character '-' in numeric literal"
這是由于我的接口要求將參數(shù)按照json格式傳遞到后臺(tái),結(jié)果測(cè)試同事使用了form-data格式,所以才會(huì)有上面這個(gè)錯(cuò)誤。
=============補(bǔ)充2018-11-09 18:20:00=============
剛剛又出現(xiàn)了這個(gè)EOF的問(wèn)題,前端確定已經(jīng)按照json格式傳參,但是還是有這個(gè)問(wèn)題。
通過(guò)wireshark抓包發(fā)現(xiàn),前端給的Content-Length為0,說(shuō)明沒(méi)有將參數(shù)傳入后臺(tái)。
后來(lái)前端核查代碼發(fā)現(xiàn),確實(shí)是沒(méi)有將參數(shù)傳入,只是定義了
補(bǔ)充:gin json 獲取_Gin框架系列 自定義錯(cuò)誤處理
概述
很多讀者在后臺(tái)向我要 Gin 框架實(shí)戰(zhàn)系列的 Demo 源碼,在這里再說(shuō)明一下,源碼我都更新到 GitHub 上,地址:https://github.com/xinliangnote/Go
開(kāi)始今天的文章,為什么要自定義錯(cuò)誤處理?默認(rèn)的錯(cuò)誤處理方式是什么?
那好,咱們就先說(shuō)下默認(rèn)的錯(cuò)誤處理。
默認(rèn)的錯(cuò)誤處理是 errors.New("錯(cuò)誤信息"),這個(gè)信息通過(guò) error 類型的返回值進(jìn)行返回。
舉個(gè)簡(jiǎn)單的例子:
func hello(name string) (str string, err error) {
if name == "" {
err = errors.New("name 不能為空")
return
}
str = fmt.Sprintf("hello: %s", name)
return
}
當(dāng)調(diào)用這個(gè)方法時(shí):
var name = ""
str, err := hello(name)
if err != nil {
fmt.Println(err.Error())
return
}
這就是默認(rèn)的錯(cuò)誤處理,下面還會(huì)用這個(gè)例子進(jìn)行說(shuō)。
這個(gè)默認(rèn)的錯(cuò)誤處理,只是得到了一個(gè)錯(cuò)誤信息的字符串。
然而...
我還想得到發(fā)生錯(cuò)誤時(shí)的 時(shí)間、 文件名、 方法名、 行號(hào) 等信息。
我還想得到錯(cuò)誤時(shí)進(jìn)行告警,比如 短信告警、 郵件告警、 微信告警 等。
我還想調(diào)用的時(shí)候,不那么復(fù)雜,就和默認(rèn)錯(cuò)誤處理類似,比如:
alarm.WeChat("錯(cuò)誤信息")
return
這樣,我們就得到了我們想要的信息( 時(shí)間、 文件名、 方法名、 行號(hào)),并通過(guò) 微信 的方式進(jìn)行告警通知我們。
同理, alarm.Email("錯(cuò)誤信息")、 alarm.Sms("錯(cuò)誤信息") 我們得到的信息是一樣的,只是告警方式不同而已。
還要保證,我們業(yè)務(wù)邏輯中,獲取錯(cuò)誤的時(shí)候,只獲取錯(cuò)誤信息即可。
上面這些想出來(lái)的,就是今天要實(shí)現(xiàn)的,自定義錯(cuò)誤處理,我們就實(shí)現(xiàn)之前,先說(shuō)下 Go 的錯(cuò)誤處理。
錯(cuò)誤處理
package main
import (
"errors"
"fmt"
)
func hello(name string) (str string, err error) {
if name == "" {
err = errors.New("name 不能為空")
return
}
str = fmt.Sprintf("hello: %s", name)
return
}
func main() {
var name = ""
fmt.Println("param:", name)
str, err := hello(name)
if err != nil {
fmt.Println(err.Error())
return
}
fmt.Println(str)
}
輸出:
param: Tom
hello: Tom
當(dāng) name = "" 時(shí),輸出:
param:
name 不能為空
建議每個(gè)函數(shù)都要有錯(cuò)誤處理,error 應(yīng)該為最后一個(gè)返回值。
咱們一起看下官方 errors.go
// Copyright 2011 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// Package errors implements functions to manipulate errors.
package errors
// New returns an error that formats as the given text.
func New(text string) error {
return errorString{text}
}
// errorString is a trivial implementation of error.
type errorString struct {
s string
}
func (e *errorString) Error() string {
return e.s
}
上面的代碼,并不復(fù)雜,參照上面的,咱們進(jìn)行寫(xiě)一個(gè)自定義錯(cuò)誤處理。
自定義錯(cuò)誤處理
咱們定義一個(gè) alarm.go,用于處理告警。
廢話不多說(shuō),直接看代碼。
package alarm
import (
"encoding/json"
"fmt"
"ginDemo/common/function"
"path/filepath"
"runtime"
"strings"
)
type errorString struct {
s string
}
type errorInfo struct {
Time string `json:"time"`
Alarm string `json:"alarm"`
Message string `json:"message"`
Filename string `json:"filename"`
Line int `json:"line"`
Funcname string `json:"funcname"`
}
func (e *errorString) Error() string {
return e.s
}
func New (text string) error {
alarm("INFO", text)
return errorString{text}
}
// 發(fā)郵件
func Email (text string) error {
alarm("EMAIL", text)
return errorString{text}
}
// 發(fā)短信
func Sms (text string) error {
alarm("SMS", text)
return errorString{text}
}
// 發(fā)微信
func WeChat (text string) error {
alarm("WX", text)
return errorString{text}
}
// 告警方法
func alarm(level string, str string) {
// 當(dāng)前時(shí)間
currentTime := function.GetTimeStr()
// 定義 文件名、行號(hào)、方法名
fileName, line, functionName := "?", 0 , "?"
pc, fileName, line, ok := runtime.Caller(2)
if ok {
functionName = runtime.FuncForPC(pc).Name()
functionName = filepath.Ext(functionName)
functionName = strings.TrimPrefix(functionName, ".")
}
var msg = errorInfo {
Time : currentTime,
Alarm : level,
Message : str,
Filename : fileName,
Line : line,
Funcname : functionName,
}
jsons, errs := json.Marshal(msg)
if errs != nil {
fmt.Println("json marshal error:", errs)
}
errorJsonInfo := string(jsons)
fmt.Println(errorJsonInfo)
if level == "EMAIL" {
// 執(zhí)行發(fā)郵件
} else if level == "SMS" {
// 執(zhí)行發(fā)短信
} else if level == "WX" {
// 執(zhí)行發(fā)微信
} else if level == "INFO" {
// 執(zhí)行記日志
}
}
看下如何調(diào)用:
package v1
import (
"fmt"
"ginDemo/common/alarm"
"ginDemo/entity"
"github.com/gin-gonic/gin"
"net/http"
)
func AddProduct(c *gin.Context) {
// 獲取 Get 參數(shù)
name := c.Query("name")
var res = entity.Result{}
str, err := hello(name)
if err != nil {
res.SetCode(entity.CODE_ERROR)
res.SetMessage(err.Error())
c.JSON(http.StatusOK, res)
c.Abort()
return
}
res.SetCode(entity.CODE_SUCCESS)
res.SetMessage(str)
c.JSON(http.StatusOK, res)
}
func hello(name string) (str string, err error) {
if name == "" {
err = alarm.WeChat("name 不能為空")
return
}
str = fmt.Sprintf("hello: %s", name)
return
}
訪問(wèn):http://localhost:8080/v1/product/add?name=a
{
"code": 1,
"msg": "hello: a",
"data": null
}
未拋出錯(cuò)誤,不會(huì)輸出信息。
訪問(wèn):http://localhost:8080/v1/product/add
{
"code": -1,
"msg": "name 不能為空",
"data": null
}
拋出了錯(cuò)誤,輸出信息如下:
{"time":"2019-07-23 22:19:17","alarm":"WX","message":"name 不能為空","filename":"絕對(duì)路徑/ginDemo/router/v1/product.go","line":33,"funcname":"hello"}
可能這會(huì)有同學(xué)說(shuō):“用上一篇分享的數(shù)據(jù)綁定和驗(yàn)證,將傳入的參數(shù)進(jìn)行 binding:"required" 也可以實(shí)現(xiàn)呀”。
我只能說(shuō):“同學(xué)呀,你不理解我的良苦用心,這只是個(gè)例子,大家可以在一些復(fù)雜的業(yè)務(wù)邏輯判斷場(chǎng)景中使用自定義錯(cuò)誤處理”。
到這里,報(bào)錯(cuò)時(shí)我們收到了 時(shí)間、 錯(cuò)誤信息、 文件名、 行號(hào)、 方法名 了。
調(diào)用起來(lái),也比較簡(jiǎn)單。
雖然標(biāo)記了告警方式,還是沒(méi)有進(jìn)行告警通知呀。
我想說(shuō),在這里存儲(chǔ)數(shù)據(jù)到隊(duì)列中,再執(zhí)行異步任務(wù)具體去消耗,這塊就不實(shí)現(xiàn)了,大家可以去完善。
讀取 文件名、 方法名、 行號(hào) 使用的是 runtime.Caller()。
我們還知道,Go 有 panic 和 recover,它們是干什么的呢,接下來(lái)咱們就說(shuō)說(shuō)。
panic 和 recover
當(dāng)程序不能繼續(xù)運(yùn)行的時(shí)候,才應(yīng)該使用 panic 拋出錯(cuò)誤。
當(dāng)程序發(fā)生 panic 后,在 defer(延遲函數(shù)) 內(nèi)部可以調(diào)用 recover 進(jìn)行控制,不過(guò)有個(gè)前提條件,只有在相同的 Go 協(xié)程中才可以。
panic 分兩個(gè),一種是有意拋出的,一種是無(wú)意的寫(xiě)程序馬虎造成的,咱們一個(gè)個(gè)說(shuō)。
有意拋出的 panic:
package main
import (
"fmt"
)
func main() {
fmt.Println("-- 1 --")
defer func() {
if r := recover(); r != nil {
fmt.Printf("panic: %s\n", r)
}
fmt.Println("-- 2 --")
}()
panic("i am panic")
}
輸出:
-- 1 --
panic: i am panic
-- 2 --
無(wú)意拋出的 panic:
package main
import (
"fmt"
)
func main() {
fmt.Println("-- 1 --")
defer func() {
if r := recover(); r != nil {
fmt.Printf("panic: %s\n", r)
}
fmt.Println("-- 2 --")
}()
var slice = [] int {1, 2, 3, 4, 5}
slice[6] = 6
}
輸出:
-- 1 --
panic: runtime error: index out of range
-- 2 --
上面的兩個(gè)我們都通過(guò) recover 捕獲到了,那我們?nèi)绾卧?Gin 框架中使用呢?如果收到 panic 時(shí),也想進(jìn)行告警怎么實(shí)現(xiàn)呢?
既然想實(shí)現(xiàn)告警,先在 ararm.go 中定義一個(gè) Panic() 方法,當(dāng)項(xiàng)目發(fā)生 panic 異常時(shí),調(diào)用這個(gè)方法,這樣就實(shí)現(xiàn)告警了。
// Panic 異常
func Panic (text string) error {
alarm("PANIC", text)
return errorString{text}
}
那我們?cè)趺床东@到呢?
使用中間件進(jìn)行捕獲,寫(xiě)一個(gè) recover 中間件。
package recover
import (
"fmt"
"ginDemo/common/alarm"
"github.com/gin-gonic/gin"
)
func Recover() gin.HandlerFunc {
return func(c *gin.Context) {
defer func() {
if r := recover(); r != nil {
alarm.Panic(fmt.Sprintf("%s", r))
}
}()
c.Next()
}
}
路由調(diào)用中間件:
r.Use(logger.LoggerToFile(), recover.Recover())
//Use 可以傳遞多個(gè)中間件。
驗(yàn)證下吧,咱們先拋出兩個(gè)異常,看看能否捕獲到?
還是修改 product.go 這個(gè)文件吧。
有意拋出 panic:
package v1
import (
"fmt"
"ginDemo/entity"
"github.com/gin-gonic/gin"
"net/http"
)
func AddProduct(c *gin.Context) {
// 獲取 Get 參數(shù)
name := c.Query("name")
var res = entity.Result{}
str, err := hello(name)
if err != nil {
res.SetCode(entity.CODE_ERROR)
res.SetMessage(err.Error())
c.JSON(http.StatusOK, res)
c.Abort()
return
}
res.SetCode(entity.CODE_SUCCESS)
res.SetMessage(str)
c.JSON(http.StatusOK, res)
}
func hello(name string) (str string, err error) {
if name == "" {
// 有意拋出 panic
panic("i am panic")
return
}
str = fmt.Sprintf("hello: %s", name)
return
}
訪問(wèn):http://localhost:8080/v1/product/add
界面是空白的。
拋出了異常,輸出信息如下:
{"time":"2019-07-23 22:42:37","alarm":"PANIC","message":"i am panic","filename":"絕對(duì)路徑/ginDemo/middleware/recover/recover.go","line":13,"funcname":"1"}
很顯然,定位的文件名、方法名、行號(hào)不是我們想要的。
需要調(diào)整 runtime.Caller(2),這個(gè)代碼在 alarm.go的alarm 方法中。
將 2 調(diào)整成 4 ,看下輸出信息:
{"time":"2019-07-23 22:45:24","alarm":"PANIC","message":"i am panic","filename":"絕對(duì)路徑/ginDemo/router/v1/product.go","line":33,"funcname":"hello"}
這就對(duì)了。
無(wú)意拋出 panic:
// 上面代碼不變
func hello(name string) (str string, err error) {
if name == "" {
// 無(wú)意拋出 panic
var slice = [] int {1, 2, 3, 4, 5}
slice[6] = 6
return
}
str = fmt.Sprintf("hello: %s", name)
return
}
訪問(wèn):http://localhost:8080/v1/product/add
界面是空白的。
拋出了異常,輸出信息如下:
{"time":"2019-07-23 22:50:06","alarm":"PANIC","message":"runtime error: index out of range","filename":"絕對(duì)路徑/runtime/panic.go","line":44,"funcname":"panicindex"}
很顯然,定位的文件名、方法名、行號(hào)也不是我們想要的。
將 4 調(diào)整成 5 ,看下輸出信息:
{"time":"2019-07-23 22:55:27","alarm":"PANIC","message":"runtime error: index out of range","filename":"絕對(duì)路徑/ginDemo/router/v1/product.go","line":34,"funcname":"hello"}
這就對(duì)了。
奇怪了,這是為什么?
在這里,有必要說(shuō)下 runtime.Caller(skip) 了。
skip 指的調(diào)用的深度。
為 0 時(shí),打印當(dāng)前調(diào)用文件及行數(shù)。
為 1 時(shí),打印上級(jí)調(diào)用的文件及行數(shù)。
依次類推...
在這塊,調(diào)用的時(shí)候需要注意下,我現(xiàn)在還沒(méi)有好的解決方案。
我是將 skip(調(diào)用深度),當(dāng)一個(gè)參數(shù)傳遞進(jìn)去。
比如:
// 發(fā)微信
func WeChat (text string) error {
alarm("WX", text, 2)
return errorString{text}
}
// Panic 異常
func Panic (text string) error {
alarm("PANIC", text, 5)
return errorString{text}
}
具體的代碼就不貼了。
但是,有意拋出 Panic 和 無(wú)意拋出 Panic 的調(diào)用深度又不同,怎么辦?
1、盡量將有意拋出的 Panic 改成拋出錯(cuò)誤的方式。
2、想其他辦法搞定它。
就到這吧。
里面涉及到的代碼,我會(huì)更新到 GitHub。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
您可能感興趣的文章:- 解決golang gin框架跨域及注解的問(wèn)題
- golang創(chuàng)建文件目錄os.Mkdir,os.MkdirAll的區(qū)別說(shuō)明
- golang 實(shí)現(xiàn)json類型不確定時(shí)的轉(zhuǎn)換
- Golang 如何判斷數(shù)組某個(gè)元素是否存在(isset)
- go 判斷兩個(gè) slice/struct/map 是否相等的實(shí)例