调整目录结构
This commit is contained in:
		
							
								
								
									
										69
									
								
								internal/domain/audit/service.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								internal/domain/audit/service.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| // Package audit 提供了用户操作审计相关的功能 | ||||
| package audit | ||||
|  | ||||
| import ( | ||||
| 	"time" | ||||
|  | ||||
| 	"git.huangwc.com/pig/pig-farm-controller/internal/infra/logs" | ||||
| 	"git.huangwc.com/pig/pig-farm-controller/internal/infra/models" | ||||
| 	"git.huangwc.com/pig/pig-farm-controller/internal/infra/repository" | ||||
| 	// 移除对 "github.com/gin-gonic/gin" 的直接依赖 | ||||
| ) | ||||
|  | ||||
| // RequestContext 封装了审计日志所需的请求上下文信息 | ||||
| type RequestContext struct { | ||||
| 	ClientIP   string | ||||
| 	HTTPPath   string | ||||
| 	HTTPMethod string | ||||
| } | ||||
|  | ||||
| // Service 定义了审计服务的接口 | ||||
| type Service interface { | ||||
| 	LogAction(user *models.User, reqCtx RequestContext, actionType, description string, targetResource interface{}, status models.AuditStatus, resultDetails string) | ||||
| } | ||||
|  | ||||
| // service 是 Service 接口的实现 | ||||
| type service struct { | ||||
| 	userActionLogRepository repository.UserActionLogRepository | ||||
| 	logger                  *logs.Logger | ||||
| } | ||||
|  | ||||
| // NewService 创建一个新的审计服务实例 | ||||
| func NewService(repo repository.UserActionLogRepository, logger *logs.Logger) Service { | ||||
| 	return &service{userActionLogRepository: repo, logger: logger} | ||||
| } | ||||
|  | ||||
| // LogAction 记录一个用户操作。它在一个新的 goroutine 中异步执行,以避免阻塞主请求。 | ||||
| func (s *service) LogAction(user *models.User, reqCtx RequestContext, actionType, description string, targetResource interface{}, status models.AuditStatus, resultDetails string) { | ||||
| 	// 不再从 context 中获取用户信息,直接使用传入的 user 对象 | ||||
| 	if user == nil { | ||||
| 		s.logger.Warnw("无法记录审计日志:传入的用户对象为 nil") | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	log := &models.UserActionLog{ | ||||
| 		Time:          time.Now(), | ||||
| 		UserID:        user.ID, | ||||
| 		Username:      user.Username, // 用户名快照 | ||||
| 		SourceIP:      reqCtx.ClientIP, | ||||
| 		ActionType:    actionType, | ||||
| 		Description:   description, | ||||
| 		Status:        status, | ||||
| 		HTTPPath:      reqCtx.HTTPPath, | ||||
| 		HTTPMethod:    reqCtx.HTTPMethod, | ||||
| 		ResultDetails: resultDetails, | ||||
| 	} | ||||
|  | ||||
| 	// 使用模型提供的方法来设置 TargetResource | ||||
| 	if err := log.SetTargetResource(targetResource); err != nil { | ||||
| 		s.logger.Errorw("无法记录审计日志:序列化 targetResource 失败", "error", err) | ||||
| 		// 即使序列化失败,我们可能仍然希望记录操作本身,所以不在此处 return | ||||
| 	} | ||||
|  | ||||
| 	// 异步写入数据库,不阻塞当前请求 | ||||
| 	go func() { | ||||
| 		if err := s.userActionLogRepository.Create(log); err != nil { | ||||
| 			s.logger.Errorw("异步保存审计日志失败", "error", err) | ||||
| 		} | ||||
| 	}() | ||||
| } | ||||
		Reference in New Issue
	
	Block a user