@@ -0,0 +1,179 @@
# 重构方案:将 `app/service/plan_service.go` 的核心逻辑迁移到 `domain/plan/plan_service.go`
## 目标:
* `app/service/plan_service.go` (应用服务层): 仅负责接收 DTO、将 DTO 转换为领域实体、调用 `domain/plan/plan_service` 的领域方法,并将领域方法返回的领域实体转换为 DTO 返回。
* `domain/plan/plan_service.go` (领域层): 封装所有与计划相关的业务逻辑、验证规则、状态管理以及对领域实体的查询操作。
## 详细步骤:
### 第一步:修改 `domain/plan/plan_service.go` (领域层)
1. **引入必要的依赖** :
* `repository.PlanRepository` :用于与计划数据存储交互。
* `repository.DeviceRepository` :如果计划逻辑中需要设备信息。
* `models.Plan` :领域实体。
* `errors` 和 `gorm.ErrRecordNotFound` :用于错误处理。
* `models.PlanTypeSystem` , `models.PlanStatusEnabled` , `models.PlanContentTypeSubPlans` , `models.PlanContentTypeTasks` 等常量。
* `git.huangwc.com/pig/pig-farm-controller/internal/infra/models`
* `git.huangwc.com/pig/pig-farm-controller/internal/infra/repository`
* `errors`
* `gorm.io/gorm`
2. **定义领域层错误** : 将 `app/service/plan_service.go` 中定义的错误(`ErrPlanNotFound` , `ErrPlanCannotBeModified` 等)迁移到 `domain/plan/plan_service.go` ,并根据领域层的语义进行调整。
```go
var (
// ErrPlanNotFound 表示未找到计划
ErrPlanNotFound = errors.New("计划不存在")
// ErrPlanCannotBeModified 表示计划不允许修改
ErrPlanCannotBeModified = errors.New("系统计划不允许修改")
// ErrPlanCannotBeDeleted 表示计划不允许删除
ErrPlanCannotBeDeleted = errors.New("系统计划不允许删除")
// ErrPlanCannotBeStarted 表示计划不允许手动启动
ErrPlanCannotBeStarted = errors.New("系统计划不允许手动启动")
// ErrPlanAlreadyEnabled 表示计划已处于启动状态
ErrPlanAlreadyEnabled = errors.New("计划已处于启动状态,无需重复操作")
// ErrPlanNotEnabled 表示计划未处于启动状态
ErrPlanNotEnabled = errors.New("计划当前不是启用状态")
// ErrPlanCannotBeStopped 表示计划不允许停止
ErrPlanCannotBeStopped = errors.New("系统计划不允许停止")
)
` ``
3. **扩展 ` plan.Service` 接口**:
* 将 ` app/service/plan_service.go` 中 ` PlanService` 接口的所有方法(` CreatePlan`, ` GetPlanByID`, ` ListPlans`, ` UpdatePlan`, ` DeletePlan`, ` StartPlan`, ` StopPlan`)添加到 ` domain/plan/Service` 接口中。
* 这些方法的参数和返回值将直接使用领域实体(` *models.Plan`)或基本类型,而不是 DTO。例如:
* ` CreatePlan(plan *models.Plan) (*models.Plan, error)`
* ` GetPlanByID(id uint) (*models.Plan, error)`
* ` ListPlans(opts repository.ListPlansOptions, page, pageSize int) ([]models.Plan, int64, error)`
* ` UpdatePlan(plan *models.Plan) (*models.Plan, error)`
* ` DeletePlan(id uint) error`
* ` StartPlan(id uint) error`
* ` StopPlan(id uint) error`
4. **修改 ` planServiceImpl` 结构体**:
* 添加 ` planRepo repository.PlanRepository` 字段。
* 添加 ` deviceRepo repository.DeviceRepository` 字段 (如果需要)。
* ` analysisPlanTaskManager plan.AnalysisPlanTaskManager` 字段保持不变。
` ``go
type planServiceImpl struct {
executionManager ExecutionManager
taskManager AnalysisPlanTaskManager
planRepo repository.PlanRepository // 新增
// deviceRepo repository.DeviceRepository // 如果需要,新增
logger *logs.Logger
}
` ``
5. **实现 ` plan.Service` 接口中的新方法**:
* 将 ` app/service/plan_service.go` 中 ` planService` 的所有业务逻辑方法(` CreatePlan`, ` GetPlanByID`, ` ListPlans`, ` UpdatePlan`, ` DeletePlan`, ` StartPlan`, ` StopPlan`)的实现,迁移到 ` domain/plan/planServiceImpl` 中。
* **关键修改点**:
* **参数和返回值**: 确保这些方法现在接收和返回的是 ` *models.Plan` 或其他领域实体,而不是 DTO。
* **业务逻辑**: 保留所有的业务规则、验证和状态管理逻辑。
* **依赖**: 这些方法将直接调用 ` planRepo` 和 ` analysisPlanTaskManager`。
* **日志**: 日志记录保持不变,但可能需要调整日志信息以反映领域层的上下文。
* **错误处理**: 错误处理逻辑保持不变,但现在将返回领域层定义的错误。
* **ContentType 自动判断**: ` CreatePlan` 和 ` UpdatePlan` 中的 ` ContentType` 自动判断逻辑应该保留在领域层。
* **执行计数器重置**: ` UpdatePlan` 和 ` StartPlan` 中的执行计数器重置逻辑应该保留在领域层。
* **系统计划限制**: 对系统计划的修改、删除、启动、停止限制逻辑应该保留在领域层。
* **验证和重排顺序**: ` models.Plan` 的 ` ValidateExecutionOrder()` 和 ` ReorderSteps()` 方法的调用应该在 ` CreatePlan` 和 ` UpdatePlan` 方法的领域层实现中进行,而不是在 DTO 转换函数中。
6. **修改 ` NewPlanService` 函数**: 接收 ` repository.PlanRepository` 和 ` repository.DeviceRepository` (如果需要) 作为参数,并注入到 ` planServiceImpl` 中。
` ``go
func NewPlanService(
executionManager ExecutionManager,
taskManager AnalysisPlanTaskManager,
planRepo repository.PlanRepository, // 新增
// deviceRepo repository.DeviceRepository, // 如果需要,新增
logger *logs.Logger,
) Service {
return &planServiceImpl{
executionManager: executionManager,
taskManager: taskManager,
planRepo: planRepo, // 注入
// deviceRepo: deviceRepo, // 注入
logger: logger,
}
}
` ``
### 第二步:修改 ` app/service/plan_service.go` (应用服务层)
1. **修改 ` planService` 结构体**:
* 移除 ` planRepo repository.PlanRepository` 字段。
* 将 ` analysisPlanTaskManager plan.AnalysisPlanTaskManager` 字段替换为 ` domainPlanService plan.Service`。
` ``go
type planService struct {
logger *logs.Logger
// planRepo repository.PlanRepository // 移除
domainPlanService plan.Service // 替换为领域层的服务接口
// analysisPlanTaskManager plan.AnalysisPlanTaskManager // 移除,由 domainPlanService 内部持有
}
` ``
2. **修改 ` NewPlanService` 函数**:
* 接收 ` domainPlanService plan.Service` 作为参数。
* 将 ` planRepo` 和 ` analysisPlanTaskManager` 的注入替换为 ` domainPlanService`。
` ``go
func NewPlanService(
logger *logs.Logger,
// planRepo repository.PlanRepository, // 移除
domainPlanService plan.Service, // 接收领域层服务
// analysisPlanTaskManager plan.AnalysisPlanTaskManager, // 移除
) PlanService {
return &planService{
logger: logger,
domainPlanService: domainPlanService, // 注入领域层服务
}
}
` ``
3. **修改 ` PlanService` 接口**:
* 接口定义保持不变,仍然接收和返回 DTO。
4. **修改 ` planService` 接口实现**:
* **` CreatePlan`**:
* 接收 ` dto.CreatePlanRequest`。
* 使用 ` dto.NewPlanFromCreateRequest` 将 DTO 转换为 ` *models.Plan`。**注意:此时 ` NewPlanFromCreateRequest` 不再包含 ` ValidateExecutionOrder()` 和 ` ReorderSteps()` 的调用。**
* 调用 ` s.domainPlanService.CreatePlan(*models.Plan)`。
* 将返回的 ` *models.Plan` 转换为 ` dto.PlanResponse`。
* **` GetPlanByID`**:
* 调用 ` s.domainPlanService.GetPlanByID(id)`。
* 将返回的 ` *models.Plan` 转换为 ` dto.PlanResponse`。
* **` ListPlans`**:
* 将 ` dto.ListPlansQuery` 转换为 ` repository.ListPlansOptions`。
* 调用 ` s.domainPlanService.ListPlans(...)`。
* 将返回的 ` []models.Plan` 转换为 ` []dto.PlanResponse`。
* **` UpdatePlan`**:
* 使用 ` dto.NewPlanFromUpdateRequest` 将 ` dto.UpdatePlanRequest` 转换为 ` *models.Plan`。**注意:此时 ` NewPlanFromUpdateRequest` 不再包含 ` ValidateExecutionOrder()` 和 ` ReorderSteps()` 的调用。**
* 设置 ` plan.ID = id`。
* 调用 ` s.domainPlanService.UpdatePlan(*models.Plan)`。
* 将返回的 ` *models.Plan` 转换为 ` dto.PlanResponse`。
* **` DeletePlan`**:
* 调用 ` s.domainPlanService.DeletePlan(id)`。
* **` StartPlan`**:
* 调用 ` s.domainPlanService.StartPlan(id)`。
* **` StopPlan`**:
* 调用 ` s.domainPlanService.StopPlan(id)`。
* **错误处理**: 应用服务层将捕获领域层返回的错误,并可能将其转换为更适合应用服务层或表示层的错误信息(例如,将领域层的 ` ErrPlanNotFound` 转换为 ` app/service` 层定义的 ` ErrPlanNotFound`)。
### 第三步:修改 ` internal/app/dto/plan_converter.go`
1. **移除 ` NewPlanFromCreateRequest` 和 ` NewPlanFromUpdateRequest` 中的领域逻辑**:
* 从 ` NewPlanFromCreateRequest` 和 ` NewPlanFromUpdateRequest` 函数中移除 ` plan.ValidateExecutionOrder()` 和 ` plan.ReorderSteps()` 的调用。这些逻辑应该由领域服务来处理。
### 第四步:更新 ` main.go` 或其他依赖注入点
* 调整 ` NewPlanService` 的调用,确保正确注入 ` domain/plan/Service` 的实现。
## 风险与注意事项:
* **事务管理**: 如果领域层的方法需要事务,确保事务在领域层内部或由应用服务层协调。
* **错误映射**: 仔细处理领域层错误到应用服务层错误的映射,确保对外暴露的错误信息是恰当的。
* **循环依赖**: 确保 ` domain` 层不依赖 ` app` 层,` app` 层可以依赖 ` domain` 层。
* * *测试**: 重构后需要对所有相关功能进行全面的单元测试和集成测试。