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{}
|
}
|
|
// GetTask 获取任务,未完成的开始时间小于等于当前时间,结束时间大于当前时间的任务
|
func (slf TaskService) GetTask(page, pageSize int, mode constvar.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 == constvar.TaskModeUnStarted {
|
search.SetStatus(model.ProcedureStatusWaitProcess).
|
SetStartTimeMax(nowTs).
|
SetEndTimeMin(nowTs).
|
SetOrder("start_time asc")
|
} else if mode == constvar.TaskModeCurrent {
|
search.SetStatusNot(model.ProcedureStatusFinished).
|
SetStartTimeMax(nowTs).
|
SetEndTimeMin(nowTs).
|
SetOrder("status desc, start_time asc")
|
} else if mode == constvar.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
|
}
|
|
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...)
|
}
|
|
count, _ = search.SetStatus(model.ProcedureStatusWaitProcess).SetStartTimeMax(nowTs).SetEndTimeMin(nowTs).Count()
|
|
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()
|
}
|