123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135 |
- package middleware
- import (
- "bytes"
- "encoding/json"
- "github.com/sirupsen/logrus"
- "io"
- "net/http"
- "net/url"
- "strconv"
- "strings"
- "sync"
- "time"
- "lc-base-frame/utils"
- "github.com/gin-gonic/gin"
- "go.uber.org/zap"
- "lc-base-frame/model/system"
- "lc-base-frame/service"
- )
- var operationRecordService = service.ServiceGroupApp.SystemServiceGroup.OperationRecordService
- var respPool sync.Pool
- func init() {
- respPool.New = func() interface{} {
- return make([]byte, 1024)
- }
- }
- func OperationRecord() gin.HandlerFunc {
- return func(c *gin.Context) {
- var body []byte
- var userId int
- if c.Request.Method != http.MethodGet {
- var err error
- body, err = io.ReadAll(c.Request.Body)
- if err != nil {
- logrus.Error("read body from request error:", zap.Error(err))
- } else {
- c.Request.Body = io.NopCloser(bytes.NewBuffer(body))
- }
- } else {
- query := c.Request.URL.RawQuery
- query, _ = url.QueryUnescape(query)
- split := strings.Split(query, "&")
- m := make(map[string]string)
- for _, v := range split {
- kv := strings.Split(v, "=")
- if len(kv) == 2 {
- m[kv[0]] = kv[1]
- }
- }
- body, _ = json.Marshal(&m)
- }
- claims, _ := utils.GetClaims(c)
- if claims.BaseClaims.ID != 0 {
- userId = int(claims.BaseClaims.ID)
- } else {
- id, err := strconv.Atoi(c.Request.Header.Get("x-user-id"))
- if err != nil {
- userId = 0
- }
- userId = id
- }
- record := system.SysOperationRecord{
- Ip: c.ClientIP(),
- Method: c.Request.Method,
- Path: c.Request.URL.Path,
- Agent: c.Request.UserAgent(),
- Body: string(body),
- UserID: userId,
- }
- // 上传文件时候 中间件日志进行裁断操作
- if strings.Contains(c.GetHeader("Content-Type"), "multipart/form-data") {
- if len(record.Body) > 1024 {
- // 截断
- newBody := respPool.Get().([]byte)
- copy(newBody, record.Body)
- record.Body = string(newBody)
- defer respPool.Put(newBody[:0])
- }
- }
- writer := responseBodyWriter{
- ResponseWriter: c.Writer,
- body: &bytes.Buffer{},
- }
- c.Writer = writer
- now := time.Now()
- c.Next()
- latency := time.Since(now)
- record.ErrorMessage = c.Errors.ByType(gin.ErrorTypePrivate).String()
- record.Status = c.Writer.Status()
- record.Latency = latency
- record.Resp = writer.body.String()
- if strings.Contains(c.Writer.Header().Get("Pragma"), "public") ||
- strings.Contains(c.Writer.Header().Get("Expires"), "0") ||
- strings.Contains(c.Writer.Header().Get("Cache-Control"), "must-revalidate, post-check=0, pre-check=0") ||
- strings.Contains(c.Writer.Header().Get("Content-Type"), "application/force-download") ||
- strings.Contains(c.Writer.Header().Get("Content-Type"), "application/octet-stream") ||
- strings.Contains(c.Writer.Header().Get("Content-Type"), "application/vnd.ms-excel") ||
- strings.Contains(c.Writer.Header().Get("Content-Type"), "application/download") ||
- strings.Contains(c.Writer.Header().Get("Content-Disposition"), "attachment") ||
- strings.Contains(c.Writer.Header().Get("Content-Transfer-Encoding"), "binary") {
- if len(record.Resp) > 1024 {
- // 截断
- newBody := respPool.Get().([]byte)
- copy(newBody, record.Resp)
- record.Resp = string(newBody)
- defer respPool.Put(newBody[:0])
- }
- }
- if err := operationRecordService.CreateSysOperationRecord(record); err != nil {
- logrus.Error("create operation record error:", zap.Error(err))
- }
- }
- }
- type responseBodyWriter struct {
- gin.ResponseWriter
- body *bytes.Buffer
- }
- func (r responseBodyWriter) Write(b []byte) (int, error) {
- r.body.Write(b)
- return r.ResponseWriter.Write(b)
- }
|