gin_recovery.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  1. package http_middleware
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "git.haoqitour.com/haoqi/go-common/utils"
  6. "git.haoqitour.com/haoqi/go-common/utils/date"
  7. "git.haoqitour.com/haoqi/go-common/utils/logger"
  8. "github.com/gin-gonic/gin"
  9. "net/http"
  10. "strings"
  11. )
  12. // Recovery中间件(统一错误处理)
  13. func Recovery(hostPrefix string, logger *logger.Logger) gin.HandlerFunc {
  14. return func(ctxt *gin.Context) {
  15. defer func() {
  16. if err := recover(); err != nil && logger != nil {
  17. stack := string(utils.GetStack(4))
  18. if i, ok := ctxt.Get(HTTPRequestStartTime); ok {
  19. if start, ok := i.(date.Datetime); ok {
  20. end := date.Now()
  21. latency := end.Sub(start)
  22. logger = logger.WithField("latency", fmt.Sprintf("%v", latency))
  23. }
  24. }
  25. clientIP := ctxt.ClientIP()
  26. comment := ctxt.Errors.ByType(gin.ErrorTypePrivate).String()
  27. req, fields := splitUri(hostPrefix, ctxt)
  28. logHttpRequest := true
  29. if i, ok := ctxt.Get(IsNoLogHTTPRequest); ok {
  30. if isNoLogHttpRequest, ok := i.(bool); ok && isNoLogHttpRequest {
  31. logHttpRequest = false
  32. }
  33. }
  34. if logHttpRequest {
  35. if reader, ok := ctxt.Request.Body.(*reqBodyLogReader); ok {
  36. logger = logger.WithField("requestBody", reader.buffer.String())
  37. }
  38. }
  39. logger = logger.
  40. WithField("clientIP", clientIP).
  41. WithField("comment", comment).
  42. WithFields(fields)
  43. var res *utils.Res
  44. if r, ok := err.(*utils.Res); ok {
  45. res = r
  46. } else if r, ok := err.(utils.Res); ok {
  47. res = &r
  48. }
  49. if res != nil {
  50. httpStatus := http.StatusBadRequest
  51. var s1, s2 string
  52. if nil != &res.Head {
  53. if res.Head.HttpStatus > 0 {
  54. httpStatus = res.Head.HttpStatus
  55. }
  56. s1 = res.Head.ErrMsg
  57. s2 = res.Head.Detail
  58. if strings.Contains(s2, s1) {
  59. s1 = ""
  60. }
  61. if strings.Contains(s1, s2) {
  62. s2 = ""
  63. }
  64. } else {
  65. res.Head = utils.ResHead{ErrCode: -1, ErrMsg: "未定义"}
  66. }
  67. if i, ok := ctxt.Get(IsLogHTTPResponse); ok {
  68. if isLogHttpResponse, ok := i.(bool); ok && isLogHttpResponse {
  69. respB, _ := json.Marshal(res)
  70. logger = logger.WithField("responseBody", string(respB))
  71. }
  72. }
  73. logger.
  74. WithCaller(4).
  75. WithField("tag", "Custom warn").
  76. WithField("httpStatus", httpStatus).
  77. WithField("errCode", res.Head.ErrCode).
  78. WithField("errMsg", strings.TrimSpace(fmt.Sprintf("%d %s %s", res.Head.ErrCode, s1, s2))).
  79. Warn(req)
  80. ctxt.JSON(httpStatus, res)
  81. ctxt.Abort()
  82. } else {
  83. res := utils.E(9999, "未定义", nil)
  84. if i, ok := ctxt.Get(IsLogHTTPResponse); ok {
  85. if isLogHttpResponse, ok := i.(bool); ok && isLogHttpResponse {
  86. respB, _ := json.Marshal(res)
  87. logger = logger.WithField("responseBody", string(respB))
  88. }
  89. }
  90. logger.
  91. WithCaller(5).
  92. WithField("tag", "Catch Exception").
  93. WithField("httpStatus", http.StatusBadRequest).
  94. WithField("errCode", 9999).
  95. WithField("errMsg", fmt.Sprintf("9999 %s", err)).
  96. Error(req)
  97. println(stack)
  98. ctxt.JSON(http.StatusBadRequest, res)
  99. ctxt.Abort()
  100. }
  101. }
  102. }()
  103. ctxt.Next()
  104. }
  105. }