package service import ( "errors" "git.huangwc.com/pig/pig-farm-controller/internal/app/dto" "git.huangwc.com/pig/pig-farm-controller/internal/domain/plan" "git.huangwc.com/pig/pig-farm-controller/internal/infra/logs" "git.huangwc.com/pig/pig-farm-controller/internal/infra/repository" ) // PlanService 定义了计划相关的应用服务接口 type PlanService interface { // CreatePlan 创建一个新的计划 CreatePlan(req *dto.CreatePlanRequest) (*dto.PlanResponse, error) // GetPlanByID 根据ID获取计划详情 GetPlanByID(id uint) (*dto.PlanResponse, error) // ListPlans 获取计划列表,支持过滤和分页 ListPlans(query *dto.ListPlansQuery) (*dto.ListPlansResponse, error) // UpdatePlan 更新计划 UpdatePlan(id uint, req *dto.UpdatePlanRequest) (*dto.PlanResponse, error) // DeletePlan 删除计划(软删除) DeletePlan(id uint) error // StartPlan 启动计划 StartPlan(id uint) error // StopPlan 停止计划 StopPlan(id uint) error } // planService 是 PlanService 接口的实现 type planService struct { logger *logs.Logger domainPlanService plan.Service // 替换为领域层的服务接口 } // NewPlanService 创建一个新的 PlanService 实例 func NewPlanService( logger *logs.Logger, domainPlanService plan.Service, // 接收领域层服务 ) PlanService { return &planService{ logger: logger, domainPlanService: domainPlanService, // 注入领域层服务 } } // CreatePlan 创建一个新的计划 func (s *planService) CreatePlan(req *dto.CreatePlanRequest) (*dto.PlanResponse, error) { const actionType = "应用服务层:创建计划" // 使用 DTO 转换函数将请求转换为领域实体 planToCreate, err := dto.NewPlanFromCreateRequest(req) if err != nil { s.logger.Errorf("%s: 计划数据校验失败: %v", actionType, err) return nil, err } // 调用领域服务创建计划 createdPlan, err := s.domainPlanService.CreatePlan(planToCreate) if err != nil { s.logger.Errorf("%s: 领域服务创建计划失败: %v", actionType, err) return nil, err // 直接返回领域层错误 } // 将领域实体转换为响应 DTO resp, err := dto.NewPlanToResponse(createdPlan) if err != nil { s.logger.Errorf("%s: 序列化响应失败: %v, Plan: %+v", actionType, err, createdPlan) return nil, errors.New("计划创建成功,但响应生成失败") } s.logger.Infof("%s: 计划创建成功, ID: %d", actionType, createdPlan.ID) return resp, nil } // GetPlanByID 根据ID获取计划详情 func (s *planService) GetPlanByID(id uint) (*dto.PlanResponse, error) { const actionType = "应用服务层:获取计划详情" // 调用领域服务获取计划 plan, err := s.domainPlanService.GetPlanByID(id) if err != nil { s.logger.Errorf("%s: 领域服务获取计划详情失败: %v, ID: %d", actionType, err, id) return nil, err // 直接返回领域层错误 } // 将领域实体转换为响应 DTO resp, err := dto.NewPlanToResponse(plan) if err != nil { s.logger.Errorf("%s: 序列化响应失败: %v, Plan: %+v", actionType, err, plan) return nil, errors.New("获取计划详情失败: 内部数据格式错误") } s.logger.Infof("%s: 获取计划详情成功, ID: %d", actionType, id) return resp, nil } // ListPlans 获取计划列表,支持过滤和分页 func (s *planService) ListPlans(query *dto.ListPlansQuery) (*dto.ListPlansResponse, error) { const actionType = "应用服务层:获取计划列表" // 将 DTO 查询参数转换为领域层可接受的选项 opts := repository.ListPlansOptions{PlanType: query.PlanType} // 调用领域服务获取计划列表 plans, total, err := s.domainPlanService.ListPlans(opts, query.Page, query.PageSize) if err != nil { s.logger.Errorf("%s: 领域服务获取计划列表失败: %v", actionType, err) return nil, err // 直接返回领域层错误 } // 将领域实体列表转换为响应 DTO 列表 planResponses := make([]dto.PlanResponse, 0, len(plans)) for _, p := range plans { resp, err := dto.NewPlanToResponse(&p) if err != nil { s.logger.Errorf("%s: 序列化单个计划响应失败: %v, Plan: %+v", actionType, err, p) // 这里选择跳过有问题的计划,并记录错误,而不是中断整个列表的返回 continue } planResponses = append(planResponses, *resp) } resp := &dto.ListPlansResponse{ Plans: planResponses, Total: total, } s.logger.Infof("%s: 获取计划列表成功, 数量: %d", actionType, len(planResponses)) return resp, nil } // UpdatePlan 更新计划 func (s *planService) UpdatePlan(id uint, req *dto.UpdatePlanRequest) (*dto.PlanResponse, error) { const actionType = "应用服务层:更新计划" // 使用 DTO 转换函数将请求转换为领域实体 planToUpdate, err := dto.NewPlanFromUpdateRequest(req) if err != nil { s.logger.Errorf("%s: 计划数据校验失败: %v", actionType, err) return nil, err } planToUpdate.ID = id // 确保ID被设置 // 调用领域服务更新计划 updatedPlan, err := s.domainPlanService.UpdatePlan(planToUpdate) if err != nil { s.logger.Errorf("%s: 领域服务更新计划失败: %v, ID: %d", actionType, err, id) return nil, err // 直接返回领域层错误 } // 将领域实体转换为响应 DTO resp, err := dto.NewPlanToResponse(updatedPlan) if err != nil { s.logger.Errorf("%s: 序列化响应失败: %v, Updated Plan: %+v", actionType, err, updatedPlan) return nil, errors.New("计划更新成功,但响应生成失败") } s.logger.Infof("%s: 计划更新成功, ID: %d", actionType, updatedPlan.ID) return resp, nil } // DeletePlan 删除计划(软删除) func (s *planService) DeletePlan(id uint) error { const actionType = "应用服务层:删除计划" // 调用领域服务删除计划 err := s.domainPlanService.DeletePlan(id) if err != nil { s.logger.Errorf("%s: 领域服务删除计划失败: %v, ID: %d", actionType, err, id) return err // 直接返回领域层错误 } s.logger.Infof("%s: 计划删除成功, ID: %d", actionType, id) return nil } // StartPlan 启动计划 func (s *planService) StartPlan(id uint) error { const actionType = "应用服务层:启动计划" // 调用领域服务启动计划 err := s.domainPlanService.StartPlan(id) if err != nil { s.logger.Errorf("%s: 领域服务启动计划失败: %v, ID: %d", actionType, err, id) return err // 直接返回领域层错误 } s.logger.Infof("%s: 计划已成功启动, ID: %d", actionType, id) return nil } // StopPlan 停止计划 func (s *planService) StopPlan(id uint) error { const actionType = "应用服务层:停止计划" // 调用领域服务停止计划 err := s.domainPlanService.StopPlan(id) if err != nil { s.logger.Errorf("%s: 领域服务停止计划失败: %v, ID: %d", actionType, err, id) return err // 直接返回领域层错误 } s.logger.Infof("%s: 计划已成功停止, ID: %d", actionType, id) return nil }