#region << 版 本 注 释 >> /*---------------------------------------------------------------- * 命名空间:WIDESEAWCS_TaskInfoService * 创建者:胡童庆 * 创建时间:2024/8/2 16:13:36 * 版本:V1.0.0 * 描述: * * ---------------------------------------------------------------- * 修改人: * 修改时间: * 版本:V1.0.1 * 修改说明: * *----------------------------------------------------------------*/ #endregion << 版 本 注 释 >> using AutoMapper; using Newtonsoft.Json; using SqlSugar; using System.Diagnostics.CodeAnalysis; using System.Net.Http.Headers; using WIDESEA_DTO.Agv; using WIDESEAWCS_Common; using WIDESEAWCS_Common.APIEnum; using WIDESEAWCS_Common.TaskEnum; using WIDESEAWCS_Core; using WIDESEAWCS_Core.BaseServices; using WIDESEAWCS_Core.Enums; using WIDESEAWCS_Core.Helper; using WIDESEAWCS_Core.LogHelper; using WIDESEAWCS_DTO; using WIDESEAWCS_DTO.TaskInfo; using WIDESEAWCS_IBasicInfoRepository; using WIDESEAWCS_ITaskInfoRepository; using WIDESEAWCS_ITaskInfoService; using WIDESEAWCS_Model.Models; using WIDESEAWCS_QuartzJob.Models; using WIDESEAWCS_QuartzJob.Repository; using WIDESEAWCS_QuartzJob.Service; using ICacheService = WIDESEAWCS_Core.Caches.ICacheService; namespace WIDESEAWCS_TaskInfoService { public partial class TaskService : ServiceBase, ITaskService { private readonly IMapper _mapper; private readonly ICacheService _cacheService; private readonly IRouterService _routerService; private readonly ITaskExecuteDetailService _taskExecuteDetailService; private readonly ITaskExecuteDetailRepository _taskExecuteDetailRepository; private readonly IStationMangerRepository _stationMangerRepository; private readonly IRouterRepository _routerRepository; private readonly IApiInfoRepository _apiInfoRepository; private List apiInfos; private Dictionary _taskOrderBy = new() { {nameof(Dt_Task.Grade),OrderByType.Desc }, {nameof(Dt_Task.TaskNum),OrderByType.Asc}, {nameof(Dt_Task.CreateDate),OrderByType.Asc}, }; private List? Warehouses = new List(); public Dictionary TaskOrderBy { get { return _taskOrderBy; } set { _taskOrderBy = value; } } public List TaskInboundTypes => typeof(TaskTypeEnum).GetEnumIndexList().Where(x => x >= 500 && x < 900).ToList(); public List TaskOutboundTypes => typeof(TaskTypeEnum).GetEnumIndexList().Where(x => x >= 100 && x < 500).ToList(); public List TaskRelocationTypes => typeof(TaskTypeEnum).GetEnumIndexList().Where(x => x >= 900 && x < 1000).ToList(); public TaskService(ITaskRepository BaseDal, IMapper mapper, ICacheService cacheService, IRouterService routerService, ITaskExecuteDetailService taskExecuteDetailService, ITaskExecuteDetailRepository taskExecuteDetailRepository, IStationMangerRepository stationMangerRepository, IRouterRepository routerRepository, IApiInfoRepository apiInfoRepository) : base(BaseDal) { _mapper = mapper; _cacheService = cacheService; _routerService = routerService; _taskExecuteDetailService = taskExecuteDetailService; _taskExecuteDetailRepository = taskExecuteDetailRepository; _stationMangerRepository = stationMangerRepository; _routerRepository = routerRepository; _apiInfoRepository = apiInfoRepository; string? cacheStr = _cacheService.Get(nameof(Dt_Warehouse)); if (!string.IsNullOrEmpty(cacheStr)) { Warehouses = JsonConvert.DeserializeObject>(cacheStr); } string? apiInfoStr = _cacheService.Get("apiInfos"); if (!string.IsNullOrEmpty(apiInfoStr)) { List? infos = JsonConvert.DeserializeObject>(apiInfoStr); if (infos == null || infos.Count == 0) { apiInfos = new List(); } else { apiInfos = infos; } } } /// /// 接收WMS任务信息 /// /// WMS任务对象集合 /// 返回处理结果 public WebResponseContent ReceiveWMSTask([NotNull] List taskDTOs) { WebResponseContent content = new WebResponseContent(); try { bool flag = false; List tasks = new List(); foreach (var item in taskDTOs) { if (BaseDal.QueryFirst(x => x.TaskNum == item.TaskNum) != null) { flag = true; continue; } Dt_Task task = _mapper.Map(item); task.Creater = "WMS"; task.TaskState = (int)TaskStatusEnum.New; task.CurrentAddress = item.SourceAddress; Dt_Router? router; } BaseDal.AddData(tasks); _taskExecuteDetailService.AddTaskExecuteDetail(tasks.Select(x => x.TaskNum).ToList(), "接收WMS任务"); content = (flag || tasks.Count > 0) ? WebResponseContent.Instance.OK("成功") : WebResponseContent.Instance.Error("失败"); } catch (Exception ex) { content = WebResponseContent.Instance.Error($"任务接收错误,错误信息:{ex.Message}"); } return content; } /// /// 根据托盘号、起始地址向WMS请求任务 /// /// 托盘号 /// 起始地址 /// public WebResponseContent RequestWMSTask(string palletCode, string sourceAddress, string materielBoxCode = "") { WebResponseContent content = new WebResponseContent(); try { Dt_StationManger stationManger = _stationMangerRepository.QueryFirst(x => x.StationCode == sourceAddress); if (stationManger == null) { return WebResponseContent.Instance.Error($"未找到站台信息"); } string address = AppSettings.Get("WMSApiAddress"); if (string.IsNullOrEmpty(address)) { return WebResponseContent.Instance.Error($"未找到WMSApi地址"); } string responseStr = ""; if (!string.IsNullOrEmpty(materielBoxCode)) { responseStr = HttpHelper.Get($"{address}/api/Task/DeviceRequestInboundTask?stationCode={sourceAddress}&roadwayNo={stationManger.StackerCraneCode}&palletCode={palletCode}&materielBoxCode={materielBoxCode}"); } else { responseStr = HttpHelper.Get($"{address}/api/Task/DeviceRequestInboundTask?stationCode={sourceAddress}&roadwayNo={stationManger.StackerCraneCode}&palletCode={palletCode}"); } WebResponseContent? responseContent = JsonConvert.DeserializeObject(responseStr); if (responseContent != null && responseContent.Status && responseContent.Data != null) { WMSTaskDTO? taskDTO = JsonConvert.DeserializeObject(responseContent.Data.ToString()); if (taskDTO != null) { content = ReceiveWMSTask(new List { taskDTO }); } } } catch (Exception ex) { content = WebResponseContent.Instance.Error(ex.Message); } return content; } // /// 请求入库任务 /// /// 托盘号 /// 起始地址 /// /// 巷道 /// public WebResponseContent RequestYLWMSTaskSimple(string palletCode, string sourceAddress,int taskNum=0) { WebResponseContent content = new WebResponseContent(); try { string address = AppSettings.Get("WMSApiAddress"); if (string.IsNullOrEmpty(address)) { return WebResponseContent.Instance.Error($"未找到WMSApi地址"); } string responseStr = HttpHelper.Get($"{address}/api/Task/RequestYLWMSTaskSimple?stationCode={sourceAddress}&palletCode={palletCode}&taskNum{taskNum}"); WebResponseContent? responseContent = JsonConvert.DeserializeObject(responseStr); if (responseContent != null && responseContent.Status && responseContent.Data != null) { WMSTaskDTO? taskDTO = JsonConvert.DeserializeObject(responseContent.Data.ToString()); if (taskDTO != null) { content = ReceiveWMSTask(new List { taskDTO }); } } else { content = responseContent ?? content.Error("结果错误"); } } catch (Exception ex) { content = WebResponseContent.Instance.Error(ex.Message); } return content; } // /// 请求入库巷道 /// /// public WebResponseContent AssignCPRoadwayNo() { WebResponseContent content = new WebResponseContent(); try { string address = AppSettings.Get("WMSApiAddress"); if (string.IsNullOrEmpty(address)) { return WebResponseContent.Instance.Error($"未找到WMSApi地址"); } string responseStr = HttpHelper.Get($"{address}/api/Task/AssignCPRoadwayNo"); if (!responseStr.IsNullOrEmpty()) { content.OK("成功", responseStr); } } catch (Exception ex) { content = WebResponseContent.Instance.Error(ex.Message); } return content; } // /// 请求原料入库巷道 /// /// public WebResponseContent AssignYLRoadwayNo(string palletCode) { WebResponseContent content = new WebResponseContent(); try { string address = AppSettings.Get("WMSApiAddress"); if (string.IsNullOrEmpty(address)) { return WebResponseContent.Instance.Error($"未找到WMSApi地址"); } string responseStr = HttpHelper.Get($"{address}/api/Task/AssignYLRoadwayNo?palletCode={palletCode}"); WebResponseContent? responseContent = JsonConvert.DeserializeObject(responseStr); if (responseContent != null && responseContent.Status && responseContent.Data != null) { content = responseContent; } } catch (Exception ex) { content = WebResponseContent.Instance.Error(ex.Message); } return content; } // /// 请求入库任务 /// /// 托盘号 /// 起始地址 /// /// 巷道 /// public WebResponseContent RequestWMSTaskSimple(string palletCode, string sourceAddress,string roadWay="", int taskType=630, int taskNum = 0, string targetAddress = "") { WebResponseContent content = new WebResponseContent(); try { //string address = AppSettings.Get("WMSApiAddress"); //if (string.IsNullOrEmpty(address)) //{ // return WebResponseContent.Instance.Error($"未找到WMSApi地址"); //} //string responseStr = HttpHelper.Get($"{address}/api/Task/DeviceRequestInboundTaskSimple?stationCode={sourceAddress}&palletCode={palletCode}"); //WebResponseContent? responseContent = JsonConvert.DeserializeObject(responseStr); //if (responseContent != null && responseContent.Status && responseContent.Data != null) //{ // WMSTaskDTO? taskDTO = JsonConvert.DeserializeObject(responseContent.Data.ToString()); // if (taskDTO != null) // { WMSTaskDTO? taskDTO = new WMSTaskDTO() { TaskNum = taskNum ==0 ? DateTime.Now.ToString("mmss").ObjToInt(): taskNum, PalletCode= palletCode, RoadWay= roadWay, TaskType= taskType, PalletType=1, TaskStatus= TaskStatusEnum.New.ObjToInt(), SourceAddress= sourceAddress, TargetAddress= targetAddress, WarehouseId=2, Grade=0, }; content = ReceiveWMSTask(new List { taskDTO }); // } //} } catch (Exception ex) { content = WebResponseContent.Instance.Error(ex.Message); } return content; } public WebResponseContent MESBoxCodeNotice(string boxCode) { WebResponseContent content = new WebResponseContent(); try { string address = AppSettings.Get("WMSApiAddress"); if (string.IsNullOrEmpty(address)) { return WebResponseContent.Instance.Error($"未找到WMSApi地址"); } string responseStr = HttpHelper.Get($"{address}/api/Mes/MESBoxCodeNotice?boxCode={boxCode}"); return content.OK(responseStr); } catch (Exception ex) { content = WebResponseContent.Instance.Error(ex.Message); } return content; } public static string Post(string serviceAddress, string requestJson = "", string contentType = "application/json", Dictionary? headers = null) { string result = string.Empty; DateTime beginDate = DateTime.Now; try { using (HttpContent httpContent = new StringContent(requestJson)) { httpContent.Headers.ContentType = new MediaTypeHeaderValue("application/json"); using HttpClient httpClient = new HttpClient(); httpClient.Timeout = new TimeSpan(0, 0, 30); string LoginToken = AppSettings.Get("MESLoginToken"); headers = new Dictionary { //正式 { "LoginToken", LoginToken } }; if (headers != null) { foreach (var header in headers) httpClient.DefaultRequestHeaders.Add(header.Key, header.Value); } HttpResponseMessage responseMessage = httpClient.PostAsync(serviceAddress, httpContent).Result; result = responseMessage.Content.ReadAsStringAsync().Result; } return result; } catch (Exception e) { throw new Exception(e.Message); } finally { Logger.Add(serviceAddress, requestJson == null ? "" : requestJson, result, beginDate); } } /// /// 向WMS申请分配货位 /// /// 任务号 /// 巷道号 /// public string? RequestAssignLocation(int taskNum, string roadwayNo) { string address = AppSettings.Get("WMSApiAddress"); if (string.IsNullOrEmpty(address)) throw new Exception("未找到WMSApi地址"); string responseStr = HttpHelper.Get($"{address}/api/Task/AssignInboundTaskLocation?taskNum={taskNum}&roadwayNo={roadwayNo}"); WebResponseContent? responseContent = JsonConvert.DeserializeObject(responseStr); if (responseContent != null && responseContent.Status && responseContent.Data != null) { return responseContent.Data.ToString(); } return ""; } /// /// 向WMS申请出库AGV终点 /// /// 任务号 /// /// public string? RequestTargetAddress(int taskNum) { string address = AppSettings.Get("WMSApiAddress"); if (string.IsNullOrEmpty(address)) throw new Exception("未找到WMSApi地址"); string responseStr = HttpHelper.Get($"{address}/api/Task/AssignOutTargetAddress?taskNum={taskNum}"); WebResponseContent? responseContent = JsonConvert.DeserializeObject(responseStr); if (responseContent != null && responseContent.Status && responseContent.Data != null) { return responseContent.Data.ToString(); } return ""; } /// /// 更新任务异常信息显示 /// /// 任务号 /// 异常信息 public WebResponseContent UpdateTaskExceptionMessage(int taskNum, string message) { WebResponseContent content = new WebResponseContent(); try { Dt_Task task = BaseDal.QueryFirst(x => x.TaskNum == taskNum); if (task == null) return WebResponseContent.Instance.Error($"未找到该任务信息,任务号:【{taskNum}】"); task.TaskState = (int)TaskStatusEnum.Pending; task.ExceptionMessage = message; task.ModifyDate = DateTime.Now; BaseDal.UpdateData(task); _taskExecuteDetailService.AddTaskExecuteDetail(task.TaskNum, task.ExceptionMessage); 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 { Dt_Task task = BaseDal.QueryFirst(x => x.TaskNum == taskNum); if (task == null) return WebResponseContent.Instance.Error($"未找到该任务信息,任务号:【{taskNum}】"); if (task.TaskState != (int)TaskStatusEnum.Pending) { return content = WebResponseContent.Instance.Error($"该任务状态不可恢复,任务号:【{taskNum}】,任务状态:【{task.TaskState}】"); } Dt_TaskExecuteDetail 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 { task.TaskState = (int)TaskStatusEnum.New; } task.ExceptionMessage = string.Empty; BaseDal.UpdateData(task); _taskExecuteDetailService.AddTaskExecuteDetail(task.TaskNum, $"人工恢复挂起任务,恢复挂起时任务状态【{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(); try { Dt_Task task = BaseDal.QueryFirst(x => x.TaskNum == taskNum); if (task == null) return WebResponseContent.Instance.Error($"未找到该任务信息,任务号:【{taskNum}】"); int oldState = task.TaskState; Dt_TaskExecuteDetail 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.TaskNum, $"人工将任务状态从【{oldState}】回滚到【{task.TaskState}】"); content = WebResponseContent.Instance.OK(); } catch (Exception ex) { content = WebResponseContent.Instance.Error(ex.Message); } return content; } /// /// 接受WMS手动完成任务 /// /// /// public WebResponseContent RecWMSTaskCompleted(int taskNum) { try { Dt_Task task = BaseDal.QueryFirst(x => x.TaskNum == taskNum); if (task != null) { task.TaskState = TaskStatusEnum.Finish.ObjToInt(); BaseDal.DeleteAndMoveIntoHty(task, OperateTypeEnum.人工完成); } return WebResponseContent.Instance.OK(); } catch (Exception ex) { return WebResponseContent.Instance.Error(ex.Message); } } public Dt_Task QueryBarCodeAGVFinishTask(int TaskNum, string currentAddress) { return BaseDal.QueryFirst(x => TaskInboundTypes.Contains(x.TaskType) && x.TaskState == (int)TaskStatusEnum.AGV_Finish && x.CurrentAddress == currentAddress && x.TaskNum == TaskNum, TaskOrderBy); } } }