76 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			76 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package models
 | ||
| 
 | ||
| import (
 | ||
| 	"time"
 | ||
| 
 | ||
| 	"gorm.io/gorm"
 | ||
| )
 | ||
| 
 | ||
| // 定义系统任务的特殊ID
 | ||
| const (
 | ||
| 	SystemTaskIDResolvePlan int = -1 // 代表“解析计划”的系统任务
 | ||
| )
 | ||
| 
 | ||
| type ExecutionStatus string
 | ||
| 
 | ||
| const (
 | ||
| 	ExecutionStatusStarted   ExecutionStatus = "started"   // 开始执行
 | ||
| 	ExecutionStatusCompleted ExecutionStatus = "completed" // 执行完成
 | ||
| 	ExecutionStatusFailed    ExecutionStatus = "failed"    // 执行失败
 | ||
| 	ExecutionStatusCancelled ExecutionStatus = "cancelled" // 执行取消
 | ||
| 	ExecutionStatusWaiting   ExecutionStatus = "waiting"   // 等待执行 (用于预写日志)
 | ||
| )
 | ||
| 
 | ||
| // PlanExecutionLog 记录整个计划的一次执行历史
 | ||
| type PlanExecutionLog struct {
 | ||
| 	gorm.Model
 | ||
| 	PlanID    uint `gorm:"index"`
 | ||
| 	Status    ExecutionStatus
 | ||
| 	StartedAt time.Time
 | ||
| 	EndedAt   time.Time
 | ||
| 	Error     string
 | ||
| }
 | ||
| 
 | ||
| // TableName 自定义 GORM 使用的数据库表名
 | ||
| func (PlanExecutionLog) TableName() string {
 | ||
| 	return "plan_execution_logs"
 | ||
| }
 | ||
| 
 | ||
| // TaskExecutionLog 记录单个任务的一次执行历史
 | ||
| type TaskExecutionLog struct {
 | ||
| 	gorm.Model
 | ||
| 	PlanExecutionLogID uint `gorm:"index"` // 关联到某次计划执行
 | ||
| 
 | ||
| 	// TaskID 使用 int 类型以容纳特殊的负数ID,代表系统任务
 | ||
| 	TaskID int `gorm:"index"`
 | ||
| 
 | ||
| 	// 关键改动:移除了 OnDelete 约束。
 | ||
| 	Task Task `gorm:"foreignKey:TaskID;constraint:OnUpdate:CASCADE;"`
 | ||
| 
 | ||
| 	Status    ExecutionStatus
 | ||
| 	Output    string // 任务执行的输出或错误信息
 | ||
| 	StartedAt time.Time
 | ||
| 	EndedAt   time.Time
 | ||
| }
 | ||
| 
 | ||
| // TableName 自定义 GORM 使用的数据库表名
 | ||
| func (TaskExecutionLog) TableName() string {
 | ||
| 	return "task_execution_logs"
 | ||
| }
 | ||
| 
 | ||
| // AfterFind 是 GORM 的一个钩子,在查询数据后自动执行
 | ||
| // 我们用它来优雅地处理系统任务的“虚拟”Task定义
 | ||
| func (log *TaskExecutionLog) AfterFind(tx *gorm.DB) (err error) {
 | ||
| 	// 检查是否是我们的“解析计划”系统任务
 | ||
| 	if log.TaskID == SystemTaskIDResolvePlan {
 | ||
| 		// 如果是,手动创建一个写死的 Task 定义并绑定上去
 | ||
| 		// 这使得上层服务在处理日志时,无需关心TaskID是否为负数
 | ||
| 		log.Task = Task{
 | ||
| 			// 注意:这里不能设置 ID,否则 GORM 可能会混淆
 | ||
| 			Name:        "系统:解析并启动计划",
 | ||
| 			Description: "这是一个由系统自动触发的内部任务,用于准备计划的执行。",
 | ||
| 		}
 | ||
| 	}
 | ||
| 	return
 | ||
| }
 |