123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133 |
- package middleware
- import (
- "bytes"
- "encoding/json"
- "io"
- "net/http"
- "net/url"
- "server/dao"
- "strconv"
- "strings"
- "sync"
- "time"
- "server/utils"
- "github.com/gin-gonic/gin"
- "go.uber.org/zap"
- "server/global"
- "server/service"
- )
- var operationRecordService = service.ServiceGroupApp.SystemServiceGroup.OperationRecordService
- var respPool sync.Pool
- var bufferSize = 1024
- func init() {
- respPool.New = func() interface{} {
- return make([]byte, bufferSize)
- }
- }
- 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 {
- global.GVA_LOG.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 != nil && 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 := dao.SysOperationRecord{
- Ip: c.ClientIP(),
- Method: c.Request.Method,
- Path: c.Request.URL.Path,
- Agent: c.Request.UserAgent(),
- Body: "",
- UserID: userId,
- }
- // 上传文件时候 中间件日志进行裁断操作
- if strings.Contains(c.GetHeader("Content-Type"), "multipart/form-data") {
- record.Body = "[文件]"
- } else {
- if len(body) > bufferSize {
- record.Body = "[超出记录长度]"
- } else {
- record.Body = string(body)
- }
- }
- 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) > bufferSize {
- // 截断
- record.Body = "超出记录长度"
- }
- }
- if err := operationRecordService.CreateSysOperationRecord(record); err != nil {
- global.GVA_LOG.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)
- }
|