using AutoMapper; using HslCommunication; using LogLibrary.Log; using Mapster; using Newtonsoft.Json; using OfficeOpenXml.FormulaParsing.Excel.Functions.Math; using SqlSugar; using System.ComponentModel; using System.Diagnostics.CodeAnalysis; using System.Reflection; using System.Text; using System.Threading.Tasks; using WIDESEA_Common.Log; using WIDESEAWCS_BasicInfoRepository; using WIDESEAWCS_BasicInfoService; using WIDESEAWCS_Common; using WIDESEAWCS_Common.APIEnum; using WIDESEAWCS_Common.TaskEnum; using WIDESEAWCS_Core; using WIDESEAWCS_Core.BaseRepository; using WIDESEAWCS_Core.BaseServices; using WIDESEAWCS_Core.Enums; using WIDESEAWCS_Core.Helper; using WIDESEAWCS_Core.LogHelper; using WIDESEAWCS_DTO.MES; using WIDESEAWCS_DTO.TaskInfo; using WIDESEAWCS_IBasicInfoRepository; using WIDESEAWCS_IBasicInfoService; using WIDESEAWCS_ISystemServices; using WIDESEAWCS_ITaskInfoRepository; using WIDESEAWCS_ITaskInfoService; using WIDESEAWCS_Model.Models; using WIDESEAWCS_QuartzJob; using WIDESEAWCS_QuartzJob.Models; using WIDESEAWCS_QuartzJob.Repository; using WIDESEAWCS_QuartzJob.Service; using WIDESEAWCS_TaskInfoRepository; namespace WIDESEAWCS_TaskInfoService { public partial class TaskService : ServiceBase, ITaskService { private readonly IRouterService _routerService; private readonly ITaskExecuteDetailService _taskExecuteDetailService; private readonly ITaskExecuteDetailRepository _taskExecuteDetailRepository; private readonly ISys_ConfigService _sys_ConfigService; private readonly IMapper _mapper; private readonly IDt_StationManagerService _stationManagerService; private readonly IDt_StationManagerRepository _stationManagerRepository; private readonly IRouterRepository _routerRepository; private readonly ITask_HtyRepository _taskHtyRepository; private readonly IApiInfoRepository _apiInfoRepository; private readonly IUnitOfWorkManage _unitOfWorkManage; private Dictionary _taskOrderBy = new() { {nameof(Dt_Task.Grade), OrderByType.Asc }, {nameof(Dt_Task.CreateDate), OrderByType.Asc}, }; public Dictionary TaskOrderBy { get { return _taskOrderBy; } set { _taskOrderBy = value; } } #region 获取所有任务类型、状态 #endregion 获取所有任务类型、状态 public TaskService(ITaskRepository BaseDal, IRouterService routerService, ITaskExecuteDetailService taskExecuteDetailService, ITaskExecuteDetailRepository taskExecuteDetailRepository, IMapper mapper, ISys_ConfigService sys_ConfigService, IDt_StationManagerService stationManagerService, IDt_StationManagerRepository stationManagerRepository, IRouterRepository routerRepository, ITask_HtyRepository htyRepository, IApiInfoRepository apiInfoRepository,IUnitOfWorkManage unitOfWorkManage) : base(BaseDal) { _routerService = routerService; _taskExecuteDetailService = taskExecuteDetailService; _taskExecuteDetailRepository = taskExecuteDetailRepository; _mapper = mapper; _sys_ConfigService = sys_ConfigService; _stationManagerService = stationManagerService; _stationManagerRepository = stationManagerRepository; _routerRepository = routerRepository; _taskHtyRepository = htyRepository; _apiInfoRepository = apiInfoRepository; _unitOfWorkManage = unitOfWorkManage; } /// /// 接收MES任务 /// /// /// public WebResponseContent ReceiveMESTask([NotNull] MESTaskDTO taskDTOs) { WebResponseContent content = new WebResponseContent(); try { if (BaseDal.QueryFirst(x => x.TaskNum == taskDTOs.taskId || x.PalletCode == taskDTOs.containerCode) != null) { return content.Error("该任务已存在!"); } Dt_Task task = _mapper.Map(taskDTOs); task.Creater = "MES"; // 判断任务类型是否为出库任务 if (task.TaskType.GetTaskTypeGroup() == TaskTypeGroup.OutbondGroup) { } // 判断任务类型是否为入库任务 else if (task.TaskType.GetTaskTypeGroup() == TaskTypeGroup.InboundGroup) { } // 判断任务类型是否为移库任务 else if (task.TaskType.GetTaskTypeGroup() == TaskTypeGroup.RelocationGroup) { } BaseDal.AddData(task); _taskExecuteDetailService.AddTaskExecuteDetailS(task.TaskNum, "接收WMS任务"); content = WebResponseContent.Instance.OK("成功"); } catch (Exception ex) { content = WebResponseContent.Instance.Error($"任务接收错误,错误信息:{ex.Message}返回数据行{ex.StackTrace}"); } return content; } /// /// 更新任务异常信息显示 /// /// 任务号 /// 异常信息 public WebResponseContent UpdateTaskExceptionMessage(int taskNum, string message) { WebResponseContent content = new WebResponseContent(); try { var task = BaseDal.QueryFirst(x => x.TaskNum == taskNum); if (task == null) return WebResponseContent.Instance.Error($"未找到该任务信息,任务号:【{taskNum}】"); if (task.TaskType.GetTaskTypeGroup() == TaskTypeGroup.CarryGroup) { task.TaskState = (int)TaskAGVCarryStatusEnum.AGV_CarryException; } task.ExceptionMessage = message; task.ModifyDate = DateTime.Now; BaseDal.UpdateData(task); _taskExecuteDetailService.AddTaskExecuteDetail(task.TaskId, task.ExceptionMessage); content = WebResponseContent.Instance.OK(); } catch (Exception ex) { content = WebResponseContent.Instance.Error(ex.Message); } return content; } /// /// 将任务状态修改为下一个状态 /// /// 任务号 public WebResponseContent UpdateTaskStatusToNext(int taskNum) { WebResponseContent content = new WebResponseContent(); try { var task = BaseDal.QueryFirst(x => x.TaskNum == taskNum); if (task == null) return WebResponseContent.Instance.Error($"未找到该任务信息,任务号:【{taskNum}】"); return UpdateTaskStatusToNext(task); } catch (Exception ex) { content = WebResponseContent.Instance.Error(ex.Message); } return content; } /// /// 将任务状态修改为下一个状态 /// /// 任务实体对象 /// public WebResponseContent UpdateTaskStatusToNext([NotNull] Dt_Task task) { WebResponseContent? content = new WebResponseContent(); try { int oldState = task.TaskState; if (task.TaskState.GetTaskStateGroup() == TaskStateGroup.OutbondGroup) { int nextStatus = task.TaskState.GetNextNotCompletedStatus(); task.TaskState = nextStatus; if (task.TaskState == (int)TaskAGVCarryStatusEnum.AGV_CarryFinish) { BaseDal.DeleteAndMoveIntoHty(task, App.User.UserId > 0 ? OperateTypeEnum.人工完成 : OperateTypeEnum.自动完成); } } else { throw new Exception($"任务类型错误,未找到该任务类型,任务号:【{task.TaskNum}】,任务类型:【{task.TaskType}】"); } if (task.TaskState <= 0) { return content = WebResponseContent.Instance.Error($"该任务状态不可跳转到下一步,任务号:【{task.TaskNum}】,任务状态:【{task.TaskState}】"); } task.ModifyDate = DateTime.Now; task.Modifier = "System"; BaseDal.UpdateData(task); _taskExecuteDetailService.AddTaskExecuteDetail(task.TaskId, App.User.UserId > 0 ? $"人工手动将任务状态从【{oldState}】跳转到【{task.TaskState}】" : $"系统自动流程,任务状态从【{oldState}】转到【{task.TaskState}】"); #region 更新任务状态 // 获取WMSip地址 var config = _sys_ConfigService.GetConfigsByCategory(CateGoryConst.CONFIG_SYS_IPAddress); var wmsBase = config.FirstOrDefault(x => x.ConfigKey == SysConfigKeyConst.WMSIP_BASE)?.ConfigValue; var updateTask = config.FirstOrDefault(x => x.ConfigKey == SysConfigKeyConst.UpdateTaskStatus)?.ConfigValue; if (wmsBase == null || updateTask == null) { throw new InvalidOperationException("WMS IP 未配置"); } var wmsIpAddress = wmsBase + updateTask; var result = HttpHelper.PostAsync(wmsIpAddress, new { TaskNum = task.TaskNum, TaskState = task.TaskState }.ToJsonString()).Result; content = JsonConvert.DeserializeObject(result); #endregion 更新任务状态 content = WebResponseContent.Instance.OK(); } catch (Exception ex) { content = WebResponseContent.Instance.Error(ex.Message); } return content; } /// /// 恢复挂起任务 /// /// 任务号 /// 返回处理结果 public WebResponseContent TaskStatusRecovery(int taskNum) { WebResponseContent content = new WebResponseContent(); try { var task = BaseDal.QueryFirst(x => x.TaskNum == taskNum); if (task == null) return WebResponseContent.Instance.Error($"未找到该任务信息,任务号:【{taskNum}】"); if (task.TaskState != (int)TaskAGVCarryStatusEnum.AGV_CarryException && task.TaskState != (int)TaskAGVCarryStatusEnum.AGV_CarryException) { return content = WebResponseContent.Instance.Error($"该任务状态不可恢复,任务号:【{taskNum}】,任务状态:【{task.TaskState}】"); } var taskExecuteDetail = _taskExecuteDetailRepository.QueryFirst(x => x.TaskId == task.TaskId && x.IsNormal, new Dictionary { { nameof(Dt_TaskExecuteDetail.TaskDetailId), OrderByType.Desc } }); if (taskExecuteDetail != null) { task.TaskState = taskExecuteDetail.TaskState; } else { throw new Exception($"未找到任务号{taskNum}恢复状态"); } task.ExceptionMessage = string.Empty; BaseDal.UpdateData(task); _taskExecuteDetailService.AddTaskExecuteDetail(task.TaskId, $"人工恢复挂起任务,恢复挂起时任务状态【{task.TaskState}】"); content = WebResponseContent.Instance.OK(); } catch (Exception ex) { content = WebResponseContent.Instance.Error(ex.Message); } return content; } /// /// 回滚任务状态 /// /// 任务号 /// 返回处理结果 public WebResponseContent RollbackTaskStatusToLast(int taskNum) { WebResponseContent content = new WebResponseContent(); try { var task = BaseDal.QueryFirst(x => x.TaskNum == taskNum); if (task == null) return WebResponseContent.Instance.Error($"未找到该任务信息,任务号:【{taskNum}】"); int oldState = task.TaskState; var taskExecuteDetail = _taskExecuteDetailRepository.QueryFirst(x => x.TaskId == task.TaskId && x.TaskState < task.TaskState && x.TaskState > 0, new Dictionary { { nameof(Dt_TaskExecuteDetail.TaskDetailId), OrderByType.Desc } }); if (taskExecuteDetail != null) { task.TaskState = taskExecuteDetail.TaskState; task.CurrentAddress = taskExecuteDetail.CurrentAddress; task.NextAddress = taskExecuteDetail.NextAddress; } else { return content = WebResponseContent.Instance.Error($"未找到任务明细信息,该任务状态不可回滚到上一步,任务号:【{taskNum}】,任务状态:【{task.TaskState}】"); } task.ExceptionMessage = string.Empty; BaseDal.UpdateData(task); _taskExecuteDetailService.AddTaskExecuteDetail(task.TaskId, $"人工将任务状态从【{oldState}】回滚到【{task.TaskState}】"); content = WebResponseContent.Instance.OK(); } catch (Exception ex) { content = WebResponseContent.Instance.Error(ex.Message); } return content; } private string GetIpAddress(string baseIp, string name) { var configz = _sys_ConfigService.GetConfigsByCategory(CateGoryConst.CONFIG_SYS_IPAddress); var wcsBasez = configz.Where(x => x.ConfigKey == baseIp).FirstOrDefault()?.ConfigValue; var address = configz.Where(x => x.ConfigKey == name).FirstOrDefault()?.ConfigValue; if (wcsBasez == null || address == null) { throw new InvalidOperationException("WMS IP 未配置"); } return wcsBasez + address; } #region 重写方法 public override WebResponseContent DeleteData(object[] key) { foreach (var item in key) { var task = BaseDal.QueryFirst(x => x.TaskId == Convert.ToInt32(item)); var taskHty = task.Adapt(); taskHty.Creater = App.User.UserName != null ? App.User.UserName : "System"; _taskHtyRepository.AddData(taskHty); } return base.DeleteData(key); } #endregion 重写方法 /// /// 更新任务信息及添加任务明细记录 /// /// 任务原对象实例(未修改的数据对象) /// 修改后的任务状态 /// 修改后的设备编号 /// 修改后的起始地址 /// 修改后的目标地址 /// 修改后的当前地址 /// 修改后的下一地址 public void UpdateTask(Dt_Task task, TaskAGVCarryStatusEnum taskStatus, string deviceCode = "", string sourceAddress = "", string targetAddress = "", string currentAddress = "", string nextAddress = "", string roadwayNo = "") { StringBuilder stringBuilder = new StringBuilder(App.User?.UserId == 0 ? $"系统自动流程" : "人工手动流程"); if (task.SourceAddress != sourceAddress && !string.IsNullOrEmpty(sourceAddress)) { stringBuilder.Append($",起始地址由{task.SourceAddress}变更为{sourceAddress}"); task.SourceAddress = sourceAddress; } if (task.TargetAddress != targetAddress && !string.IsNullOrEmpty(targetAddress)) { stringBuilder.Append($",目标地址由{task.TargetAddress}变更为{targetAddress}"); task.TargetAddress = targetAddress; } if (task.CurrentAddress != currentAddress && !string.IsNullOrEmpty(currentAddress)) { stringBuilder.Append($",当前位置由{task.CurrentAddress}变更为{currentAddress}"); task.CurrentAddress = currentAddress; } if (task.NextAddress != nextAddress && !string.IsNullOrEmpty(nextAddress)) { stringBuilder.Append($",下一位置由{task.NextAddress}变更为{nextAddress}"); task.NextAddress = nextAddress; } if (task.Roadway != roadwayNo && !string.IsNullOrEmpty(roadwayNo)) { stringBuilder.Append($",巷道号由{task.Roadway}变更为{roadwayNo}"); task.Roadway = roadwayNo; } if (task.TaskState != taskStatus.ObjToInt()) { string newStatus = $"{taskStatus}"; try { List enums = Enum.GetValues(typeof(TaskAGVCarryStatusEnum)).Cast().ToList(); FieldInfo? fieldInfo = typeof(TaskAGVCarryStatusEnum).GetField((taskStatus).ToString()); if (fieldInfo != null) { DescriptionAttribute? description = fieldInfo.GetCustomAttribute(); if (description != null) { newStatus = $"{description.Description}({taskStatus})"; } } } catch { } string oldStatus = $"{task.TaskState}"; try { FieldInfo? fieldInfo2 = typeof(RouterInOutType).GetField(((TaskAGVCarryStatusEnum)task.TaskState).ToString()); if (fieldInfo2 != null) { DescriptionAttribute? description2 = fieldInfo2.GetCustomAttribute(); if (description2 != null) { oldStatus = $"{description2.Description}({task.TaskState})"; } } } catch { } stringBuilder.Append($",任务状态由{oldStatus}变更为{newStatus}"); task.TaskState = taskStatus.ObjToInt(); } BaseDal.UpdateData(task); _taskExecuteDetailService.AddTaskExecuteDetail(task.TaskId, stringBuilder.ToString()); } // 在类中添加静态锁对象 private static readonly object _taskCompleteLock = new object(); /// /// 任务完成 /// /// /// public WebResponseContent TaskCompleted(int taskNum) { WebResponseContent content = new WebResponseContent(); // 使用锁确保同一任务不会并发处理 lock (_taskCompleteLock) { try { WriteLog.Write_Log("任务完成", "任务完成接口", "任务号", $"任务:{taskNum}"); // 再次查询任务状态,防止在锁等待期间任务状态已改变 Dt_Task task = BaseDal.QueryFirst(x => x.TaskNum == taskNum); if (task == null) { return content.Error($"任务{taskNum}不存在"); } //任务完成逻辑 _unitOfWorkManage.BeginTran(); _unitOfWorkManage.CommitTran(); content.OK("任务完成"); } catch (Exception ex) { _unitOfWorkManage.RollbackTran(); content.Error(ex.Message); } return content; } } } }