package service import ( "apsClient/conf" "apsClient/constvar" "apsClient/model" "apsClient/model/common" "apsClient/model/response" "apsClient/nsq" "apsClient/pkg/ecode" "apsClient/pkg/logx" "apsClient/pkg/structx" "encoding/json" "fmt" "gorm.io/gorm" "time" ) type TaskService struct { } func NewTaskService() *TaskService { return &TaskService{} } type TaskMode int const ( TaskModeUnStarted TaskMode = 1 //未开始的 TaskModeCurrent TaskMode = 2 TaskModeLastFinished TaskMode = 3 //上一个结束的 ) // GetTask 获取任务,未完成的开始时间小于等于当前时间,结束时间大于当前时间的任务 func (slf TaskService) GetTask(page, pageSize int, mode TaskMode, channels []int32) (taskResp *response.TaskResponse, code int) { var taskList []*response.TaskData var count int64 var workers []*common.ProcedureWorker taskResp = &response.TaskResponse{ Tasks: taskList, TaskCount: count, Workers: slf.WorkerDistinct(workers), } nowTs := time.Now().Unix() var ( err error orders []*model.Order procedures []*model.Procedures workOrderIds []string ) search := model.NewProceduresSearch(nil). SetDeviceId(conf.Conf.System.DeviceId). SetPage(page, pageSize) if mode == TaskModeUnStarted { search.SetStatus(model.ProcedureStatusWaitProcess). SetStartTimeMax(nowTs). SetEndTimeMin(nowTs). SetOrder("start_time asc") } else if mode == TaskModeCurrent { search.SetStatusNot(model.ProcedureStatusFinished). SetStartTimeMax(nowTs). SetEndTimeMin(nowTs). SetOrder("status desc, start_time asc") } else if mode == TaskModeLastFinished { search.SetStatus(model.ProcedureStatusFinished).SetOrder("updated_at desc") if len(channels) > 0 { search.SetChannels(channels) } } procedures, err = search.FindNotTotal() if err != nil { return taskResp, ecode.DBErr } count, err = search.Count() if err != nil { return taskResp, ecode.DBErr } if len(procedures) == 0 { return taskResp, ecode.OK } for _, procedure := range procedures { workOrderIds = append(workOrderIds, procedure.WorkOrderID) } orders, err = model.NewOrderSearch(nil).SetWorkOrderIds(workOrderIds).FindNotTotal() //由工序反推工单 if err != nil { return taskResp, ecode.DBErr } orderMap := make(map[string]*model.Order, len(workOrderIds)) for _, order := range orders { orderMap[order.WorkOrderID] = order } for _, procedure := range procedures { taskData := response.TaskData{ Order: orderMap[procedure.WorkOrderID], Procedure: procedure, AllProcedures: procedure.ProceduresInfo.AllProcedureNames, CurrentProcedureIndex: 0, Channel: procedure.Channel, } index := 0 for _, name := range procedure.ProceduresInfo.AllProcedureNames { if name == procedure.ProceduresInfo.ProcedureName { break } index++ } taskData.CurrentProcedureIndex = index taskList = append(taskList, &taskData) workers = append(workers, procedure.ProceduresInfo.Workers...) } taskResp = &response.TaskResponse{ Tasks: taskList, TaskCount: count, Workers: slf.WorkerDistinct(workers), } return taskResp, ecode.OK } func (slf TaskService) WorkerDistinct(workers []*common.ProcedureWorker) (NoRepeatedWorkers []*common.ProcedureWorker) { exists := make(map[string]bool, 0) for _, worker := range workers { key := fmt.Sprintf("%v%v", worker.WorkerID, worker.StartTime) if !exists[key] { NoRepeatedWorkers = append(NoRepeatedWorkers, worker) exists[key] = true } } return NoRepeatedWorkers } func (slf TaskService) GetProcedureById(id int) (*model.Procedures, int) { procedure, err := model.NewProceduresSearch(nil).SetId(id).First() if err != nil { return nil, ecode.DBErr } return procedure, ecode.OK } func (slf TaskService) UpdateProcedureStatusAndChannel(db *gorm.DB, id int, status model.ProcedureStatus, channel int32) error { if status == model.ProcedureStatusFinished { ProgressCacheUnset(channel) } return model.NewProceduresSearch(db).SetId(id).UpdateByMap(map[string]interface{}{ "status": status, "channel": channel, }) } func (slf TaskService) UpdateProcedureStatus(db *gorm.DB, id int, status model.ProcedureStatus, channel int32) error { if status == model.ProcedureStatusFinished { ProgressCacheUnset(channel) } return model.NewProceduresSearch(db).SetId(id).UpdateByMap(map[string]interface{}{ "status": status, }) } func (slf TaskService) UpdateOrderStatus(db *gorm.DB, id uint, status model.OrderStatus) error { return model.NewOrderSearch(db).SetId(id).UpdateByMap(map[string]interface{}{ "status": status, }) } func (slf TaskService) GetOrderByWorkOrderId(workOrderId string) (order *model.Order, err error) { return model.NewOrderSearch(nil).SetWorkOrderId(workOrderId).First() } func (slf TaskService) GetProcessParams(procedure *model.Procedures, order *model.Order) (processModel *model.ProcessModel, err error) { data, err := model.NewProcessModelSearch(). SetProcedure(procedure.ProceduresInfo.ProcedureName). SetProduct(order.ProductName). SetOrder("id desc").First() if err != nil && err != gorm.ErrRecordNotFound { logx.Errorf("TaskStart GetProcessModel err:%v", err) return nil, err } if err == nil { err = json.Unmarshal([]byte(data.Params), &data.ParamsMap) if err != nil { logx.Errorf("process model json.Unmarshal:%v", err) return } return data, nil } if err == gorm.ErrRecordNotFound { //如果数据库没有从云端获取 caller := nsq.NewCaller(fmt.Sprintf(constvar.NsqTopicProcessParamsRequest, conf.Conf.NsqConf.NodeId), fmt.Sprintf(constvar.NsqTopicProcessParamsResponse, conf.Conf.NsqConf.NodeId)) var result common.ResponseProcessParams err = caller.Call(common.RequestProcessParams{ WorkOrder: procedure.WorkOrderID, OrderId: procedure.OrderID, Product: order.ProductName, Procedure: procedure.ProceduresInfo.ProcedureName, Device: procedure.ProceduresInfo.DeviceName, DeviceId: conf.Conf.System.DeviceId, }, &result, time.Second*3) if err != nil { logx.Errorf("TaskStart GetProcessModel error:%v", err.Error()) return } if result.ParamsMap == nil { logx.Errorf("TaskStart GetProcessModel response miss process params:%v", result) return } processModel = new(model.ProcessModel) err = structx.AssignTo(result, &processModel) return processModel, err } return } // GetNextTask 获取未开始的任务 func (slf TaskService) GetNextTask() (workOrder *model.Order, err error) { nowTs := time.Now().Unix() orderSearch := model.NewOrderSearch(nil) return orderSearch.SetOrder("start_time asc"). SetStartTimeMin(nowTs). SetStatus(model.OrderStatusWaitProcess).First() }