using Microsoft.AspNetCore.Mvc; using NetTaste; using Newtonsoft.Json; using PlcSiemens.Core.Extension; using ServiceCenter.Attributes; using ServiceCenter.Logs; using ServiceCenter.Redis; using ServiceCenter.SqlSugars; using WCS.Core; using WCS.Entity; using WCS.WorkEngineering.Extensions; using WCS.WorkEngineering.Protocol.Station; using WCS.WorkEngineering.Systems; using WCS.WorkEngineering.WebApi.Models.AGV; using WCS.WorkEngineering.WebApi.Models.AGV.Request; using WCS.WorkEngineering.WebApi.Models.AGV.Response; namespace WCS.WorkEngineering.WebApi.Controllers { /// /// AGV相关接口控制器 /// [ApiController] [Route("api/[controller]/[action]")] public class AgvController : ControllerBase { /// /// AGV任务下发测试 /// /// 任务类型 /// RFID /// 目标位置 /// [HttpPost] //public string AgvDebug(int type, string code, string pos) //{ // try // { // switch (type) // { // case 1: // AgvApi.机台补空(pos, code, "1"); // break; // case 2: // //AgvApi.机台补满(); // break; // //case 3: // // AgvApi.满轮入库(code, pos, Guid.NewGuid().ToString().Replace("-", ""), "1"); // // break; // default: // break; // } // return "成功"; // } // catch (Exception ex) // { // return $"Error-----" + // $"{ex.Message}------" + // $"{ex.StackTrace}"; // } //} /// /// 背负式AGV请求出库任务 /// /// 请求参数 /// [HttpPost] public ApplyEmptySpoolResponse ApplyEmptySpool([FromBody] AgvFillEmptySpaceRequest reqDto) { var key = $"WCS:Lock:AGV:{nameof(ApplyEmptySpool)}"; ApplyEmptySpoolResponse agvFill = new ApplyEmptySpoolResponse(); try { if (RedisHub.Default.Get(key) != null) { agvFill.ResCode = Models.WMS.Response.ResponseStatusCodeEnum.DataSaveErr; agvFill.ResMsg = $"[{nameof(ApplyEmptySpool)}]--触发并发管控"; } else { RedisHub.Default.Set(key, nameof(ApplyEmptySpool)); LogHub.InterfacePublish(nameof(ApplyEmptySpool), $"传入参数--{JsonConvert.SerializeObject(reqDto)}"); try { agvFill.ResCode = Models.WMS.Response.ResponseStatusCodeEnum.DataSaveErr; if (!World.IsStart) { agvFill.ResCode = Models.WMS.Response.ResponseStatusCodeEnum.DataSaveErr; agvFill.ResMsg = "WCS初始化中"; return agvFill; } var obj = World.GetSystemInstance().Invoke("输送机") as List; // 检测三个站台是否有货 obj = obj.Where(v => v.Entity.Code is "1012" or "1014" or "1016").Where(v => v.Data3.Status.HasFlag(StationStatus.PH_Status)).ToList(); if (!obj.Any()) { agvFill.ResCode = Models.WMS.Response.ResponseStatusCodeEnum.DataSaveErr; agvFill.ResMsg = "无空轮"; return agvFill; } SqlSugarHelper.Do(db => { var res = WmsApi.GetTunnelEmptyConCount(); var agvStations = db.Default.Queryable().SplitTable(tabs => tabs.Take(2)) .Where(v => v.Status < AGVTaskStatus.Complete3 && v.TaskType == AGVTaskType.CallForMaterial).Select(v => v.Station).ToList(); obj = obj.Where(v => !agvStations.Contains(v.Entity.Code)).ToList(); if (!obj.Any()) { agvFill.ResCode = Models.WMS.Response.ResponseStatusCodeEnum.DataSaveErr; agvFill.ResMsg = "无可用取货站点"; return; } foreach (var item in res.ResDataList) { var station = Device.All.Where(v => v.Code == "TY" + item.Tunnel.ToString()) .Select(v => v.Targets).SelectMany(v => v) .Where(v => v.HasProtocol(typeof(IStation520))) .Where(v => v.Code is "1012" or "1014" or "1016") .FirstOrDefault(); item.Tunnel = station.ToInt(); } var stationNo = res.ResDataList.OrderBy(v => v.Count).Select(v => v.Tunnel.ToString()).ToList(); var dev = obj.MinBy(v => stationNo.IndexOf(v.Entity.Code)); var task = db.Default.Queryable().First(v => v.ID == dev.Data.TaskNumber) ?? throw new Exception("无有效任务"); var id = db.GetAGVTaskId(); var agv = new WCS_AgvTaskInfo() { ID = id, AgvID = $"HJBK{id}{task.ID}", TaskType = AGVTaskType.CallForMaterial, Status = AGVTaskStatus.NewBuild, Station = dev.Entity.Code, AddWho = "WCS" }; //创建对应的AGV任务 db.Default.Insertable(agv).SplitTable().ExecuteCommand(); task.AgvTaskID = agv.ID; task.Status = Entity.TaskStatus.AGVExecution; db.Default.Updateable(task).ExecuteCommand(); task.AddWCS_TASK_DTL(db, dev.Entity.Code, "AGV", "agv执行中"); agvFill.LocCode = dev.Entity.Code; agvFill.SpoolType = "4"; agvFill.ResMsg = ""; agvFill.ResCode = Models.WMS.Response.ResponseStatusCodeEnum.Sucess; agvFill.TaskCode = agv.AgvID; }); } catch (Exception ex) { agvFill.ResCode = Models.WMS.Response.ResponseStatusCodeEnum.DataSaveErr; agvFill.ResMsg = ex.Message; } LogHub.InterfacePublish(nameof(ApplyEmptySpool), $"返回参数{JsonConvert.SerializeObject(agvFill)}"); } } finally { RedisHub.Default.Del(key); } return agvFill; } /// /// AGV执行回调 /// /// /// [HttpPost] public AgvCallbackResponse AgvCallback([FromBody] AgvCallbackRequest reqDto) { var key = $"WCS:Lock:AGV:{nameof(AgvCallback)}"; var res = new AgvCallbackResponse() { code = AgvResponseCode.Fail, message = "失败" }; try { if (RedisHub.Default.Get(key) != null) { res.code = AgvResponseCode.Error; res.message = $"[{nameof(AgvCallback)}]--触发并发管控"; } else { RedisHub.Default.Set(key, nameof(AgvCallback)); WCS_TaskInfo taskInfo = null; try { SqlSugarHelper.Do(db => { //跟据AGVid找到对应的AGV任务 var agvTask = db.Default.Queryable().SplitTable(tabs => tabs.Take(2)).First(v => v.AgvID == reqDto.taskCode && v.Status < AGVTaskStatus.MissionCompleted); if (agvTask == null) { res.code = AgvResponseCode.Fail; res.message = "未找到对应的AGV任务"; } else { switch (reqDto.method) { //case "start": //表示请求巷道 // agvTask.Status = AGVTaskStatus.RequestOrPermission1; // break; //case "end": //表示请求巷道 // agvTask.Status = AGVTaskStatus.RequestOrPermission1; // break; //入库任务、叫料任务、搬运任务 case "applyContinue": //表示请求巷道 agvTask.AgvStatus = AGVTaskStatus.RequestOrPermission1; break; case "applySecurity": //表示请求放货或取货 agvTask.AgvStatus = AGVTaskStatus.RequestOrPermission2; break; case "hjend_2": //补空任务完成 agvTask.AgvStatus = AGVTaskStatus.MissionCompleted; break; case "endhjBM": //取满任务完成 agvTask.AgvStatus = AGVTaskStatus.MissionCompleted; break; case "end": //二楼出满任务完成 agvTask.AgvStatus = AGVTaskStatus.MissionCompleted; break; case "tcEnd": //机台补空任务完成 agvTask.AgvStatus = AGVTaskStatus.MissionCompleted; break; case "exc_end": //异常信息上抛-值不匹配 agvTask.AgvStatus = AGVTaskStatus.MissionCompleted; break; case "outbin": //小车退出取货位 agvTask.AgvStatus = AGVTaskStatus.Complete3; break; case "cancel": //取消任务 //agvTask.AgvStatus = AGVTaskStatus.Cancel; break; default: break; } db.Default.Updateable(agvTask).SplitTable().ExecuteCommand(); res.code = AgvResponseCode.Success; res.message = "成功"; } }); } catch (Exception ex) { res.code = AgvResponseCode.Error; res.message = ex.Message; } } } finally { RedisHub.Default.Del(key); } return res; } } }