using Admin.NET.Core.Service; using Elastic.Clients.Elasticsearch; using WCS.Application.Entity; using WCS.Application.Service.WcsDevice.Dto; namespace WCS.Application; /// /// 设备信息服务 /// [ApiDescriptionSettings(ApplicationConst.GroupName, Order = 100)] public class WcsDeviceService : IDynamicApiController, ITransient { private readonly SqlSugarRepository _wcsDeviceRep; private readonly SysCacheService _sysCacheService; public WcsDeviceService(SqlSugarRepository wcsDeviceRep, SysCacheService sysCacheService) { _wcsDeviceRep = wcsDeviceRep; _sysCacheService = sysCacheService; } /// /// 分页查询设备信息 /// /// /// [HttpPost] [ApiDescriptionSettings(Name = "Page")] [DisplayName("分页查询设备信息")] public async Task> Page(PageWcsDeviceInput input) { if (input.Field.IsNullOrEmpty()) { input.Field = "u.Id"; input.Order = "desc"; } input.SearchKey = input.SearchKey?.Trim(); var query = _wcsDeviceRep.AsQueryable() .WhereIF(!string.IsNullOrEmpty(input.SearchKey), u => u.Text.Contains(input.SearchKey) ) .WhereIF(input.PlcId > 0, u => u.PlcId == input.PlcId) .WhereIF(!string.IsNullOrWhiteSpace(input.Text), u => u.Text.Contains(input.Text.Trim())) //处理外键和TreeSelector相关字段的连接 .LeftJoin((u, plcid) => u.PlcId == plcid.Id) .Select((u, plcid) => new WcsDeviceOutput { Id = u.Id, PlcId = u.PlcId, DeviceType = (DeviceTypeEnum)u.DeviceType, PlcIdIP = plcid.IP, Level = (DeviceLevelEnum)u.Level, DbNumber = u.DbNumber, StationNum = u.StationNum, PlcPos = u.PlcPos, WcsPos = u.WcsPos, PosType = u.PosType, LedIP = u.LedIP, Text = u.Text, CreateTime = u.CreateTime, UpdateTime = u.UpdateTime, CreateUserId = u.CreateUserId, CreateUserName = u.CreateUserName, UpdateUserId = u.UpdateUserId, UpdateUserName = u.UpdateUserName, CreateOrgId = u.CreateOrgId, CreateOrgName = u.CreateOrgName, IsDelete = u.IsDelete, }); return await query.OrderBuilder(input).ToPagedListAsync(input.Page, input.PageSize); } /// /// 增加设备信息 /// /// /// [HttpPost] [ApiDescriptionSettings(Name = "Add")] [DisplayName("增加设备信息")] public async Task Add(AddWcsDeviceInput input) { var entity = input.Adapt(); await _wcsDeviceRep.InsertAsync(entity); return entity.Id; } /// /// 删除设备信息 /// /// /// [HttpPost] [ApiDescriptionSettings(Name = "Delete")] [DisplayName("删除设备信息")] public async Task Delete(DeleteWcsDeviceInput input) { var entity = await _wcsDeviceRep.GetFirstAsync(u => u.Id == input.Id) ?? throw Oops.Oh(ErrorCodeEnum.D1002); await _wcsDeviceRep.FakeDeleteAsync(entity); //假删除 //await _wcsDeviceRep.DeleteAsync(entity); //真删除 } /// /// 更新设备信息 /// /// /// [HttpPost] [ApiDescriptionSettings(Name = "Update")] [DisplayName("更新设备信息")] public async Task Update(UpdateWcsDeviceInput input) { var entity = input.Adapt(); await _wcsDeviceRep.AsUpdateable(entity).IgnoreColumns(ignoreAllNullColumns: true).ExecuteCommandAsync(); } /// /// 获取设备信息 /// /// /// [HttpGet] [ApiDescriptionSettings(Name = "Detail")] [DisplayName("获取设备信息")] public async Task Detail([FromQuery] QueryByIdWcsDeviceInput input) { return await _wcsDeviceRep.GetFirstAsync(u => u.Id == input.Id); } /// /// 获取PlcId列表 /// /// [ApiDescriptionSettings(Name = "WcsPlcPlcIdDropdown"), HttpGet] [DisplayName("获取PlcId列表")] public async Task WcsPlcPlcIdDropdown() { return await _wcsDeviceRep.Context.Queryable() .Select(u => new { Label = u.Text, Value = u.Id } ).ToListAsync(); } /// /// 生成点位 /// /// /// [HttpPost] [ApiDescriptionSettings(Name = "GeneratePos")] [DisplayName("生成点位")] public async Task GeneratePos(GeneratePosInput input) { var modDevice = await _wcsDeviceRep.GetByIdAsync(input.Id); var listPosition = new List(); listPosition.Add(new WcsPosition() { DeviceId = modDevice.Id, StationNum = modDevice.StationNum, PlcPos = input.Pos.ToString(), PosType = PLCDataTypeEnum.String, Text = "TaskNo" }); listPosition.Add(new WcsPosition() { DeviceId = modDevice.Id, StationNum = modDevice.StationNum, PlcPos = (input.Pos + 4).ToString(), PosType = PLCDataTypeEnum.UShort, Text = "TaskType" }); listPosition.Add(new WcsPosition() { DeviceId = modDevice.Id, StationNum = modDevice.StationNum, PlcPos = (input.Pos + 6).ToString(), PosType = PLCDataTypeEnum.UShort, Text = "StartLocatNo" }); listPosition.Add(new WcsPosition() { DeviceId = modDevice.Id, StationNum = modDevice.StationNum, PlcPos = (input.Pos + 8).ToString(), PosType = PLCDataTypeEnum.UShort, Text = "EndLocatNo" }); await _wcsDeviceRep.Context.Insertable(listPosition).ExecuteCommandAsync(); } /// /// 获取设备信息列表 /// /// /// [HttpGet] [ApiDescriptionSettings(Name = "List")] [DisplayName("获取设备信息列表")] public async Task> List([FromQuery] PageWcsDeviceInput input) { var list = await _wcsDeviceRep.AsQueryable() .LeftJoin((a, b) => a.PlcId == b.Id) .Where((a, b) => a.DeviceType == DeviceTypeEnum.Business) .Select((a, b) => new WcsDeviceOutput() { Type = b.Type }, true) .ToListAsync(); //获取跺机的状态 foreach (var modDevice in list) { if (_sysCacheService.ExistKey("PLCCONN" + modDevice.PlcId)) { var cachePlc = _sysCacheService.Get("PLCCONN" + modDevice.PlcId); modDevice.Status = cachePlc.IsConn; if (modDevice.Status) { //读取plc的值 var modConn = PLCTaskAction.listPlcConn.FirstOrDefault(s => s != null && s.PlcId == modDevice.PlcId); if (modConn == null) { continue; } try { var listPosition = await _wcsDeviceRep.Context.Queryable().Where(s => s.DeviceId == modDevice.Id).ToListAsync(); (var result, var plc) = modConn.GetPlcDBValue(modDevice.PosType, modDevice.DbNumber, modDevice.PlcPos); modDevice.Plc = Convert.ToString(plc); (result, var wcs) = modConn.GetPlcDBValue(modDevice.PosType, modDevice.DbNumber, modDevice.WcsPos); modDevice.Wcs = Convert.ToString(wcs); //任务号 var modPositionTask = listPosition.FirstOrDefault(s => s.Text == "任务号"); (result, var taskNo) = modConn.GetPlcDBValue(modPositionTask.PosType, modDevice.DbNumber, modPositionTask.PlcPos, modPositionTask.StringLength); modDevice.TaskNo = Convert.ToString(taskNo); //任务类型 var modPositionTaskType = listPosition.FirstOrDefault(s => s.Text == "任务类型"); (result, var taskType) = modConn.GetPlcDBValue(modPositionTaskType.PosType, modDevice.DbNumber, modPositionTaskType.PlcPos); modDevice.TaskType = (TaskTypeEnum)Convert.ToInt32(taskType); //起始工位 var modPositionStartLocatNo = listPosition.FirstOrDefault(s => s.Text == "起始工位"); (result, var startLocatNo) = modConn.GetPlcDBValue(modPositionStartLocatNo.PosType, modDevice.DbNumber, modPositionStartLocatNo.PlcPos); modDevice.StartLocatNo = Convert.ToString(startLocatNo); //目的工位 var modPositionEndLocatNo = listPosition.FirstOrDefault(s => s.Text == "目的工位"); (result, var endLocatNo) = modConn.GetPlcDBValue(modPositionEndLocatNo.PosType, modDevice.DbNumber, modPositionEndLocatNo.PlcPos); modDevice.EndLocatNo = Convert.ToString(endLocatNo); //托盘码 var modPositionPalletNo = listPosition.FirstOrDefault(s => s.Text == "托盘码"); (result, var palletNo) = modConn.GetPlcDBValue(modPositionPalletNo.PosType, modDevice.DbNumber, modPositionPalletNo.PlcPos, modPositionPalletNo.StringLength); modDevice.PalletNo = Convert.ToString(palletNo); if (cachePlc.Type == PLCTypeEnum.ConveyorLine) { //放货排 var modPositionReleaseRow = listPosition.FirstOrDefault(s => s.Text == "放货排"); (result, var releaseRow) = modConn.GetPlcDBValue(modPositionReleaseRow.PosType, modDevice.DbNumber, modPositionReleaseRow.PlcPos); modDevice.ReleaseRow = Convert.ToInt32(releaseRow); //放货列 var modPositionReleaseCol = listPosition.FirstOrDefault(s => s.Text == "放货列"); (result, var releaseCol) = modConn.GetPlcDBValue(modPositionReleaseCol.PosType, modDevice.DbNumber, modPositionReleaseCol.PlcPos); modDevice.ReleaseCol = Convert.ToInt32(releaseCol); //放货层 var modPositionReleaseStorey = listPosition.FirstOrDefault(s => s.Text == "放货层"); (result, var releaseStorey) = modConn.GetPlcDBValue(modPositionReleaseStorey.PosType, modDevice.DbNumber, modPositionReleaseStorey.PlcPos); modDevice.ReleaseStorey = Convert.ToInt32(releaseStorey); //取货排 var modPositionPickRow = listPosition.FirstOrDefault(s => s.Text == "取货排"); (result, var pickRow) = modConn.GetPlcDBValue(modPositionPickRow.PosType, modDevice.DbNumber, modPositionPickRow.PlcPos); modDevice.PickRow = Convert.ToInt32(pickRow); //取货列 var modPositionPickCol = listPosition.FirstOrDefault(s => s.Text == "取货列"); (result, var pickCol) = modConn.GetPlcDBValue(modPositionPickCol.PosType, modDevice.DbNumber, modPositionPickCol.PlcPos); modDevice.PickCol = Convert.ToInt32(pickCol); //取货层 var modPositionPickStorey = listPosition.FirstOrDefault(s => s.Text == "取货层"); (result, var pickStorey) = modConn.GetPlcDBValue(modPositionPickStorey.PosType, modDevice.DbNumber, modPositionPickStorey.PlcPos); modDevice.PickStorey = Convert.ToInt32(pickStorey); } } catch (Exception) { } } } else { modDevice.Status = false; } //modDevice.TaskNo = "TK00001"; //modDevice.TaskType = TaskTypeEnum.In; //modDevice.PalletNo = "2024209032"; //modDevice.StartLocatNo = "010101"; //modDevice.EndLocatNo = "020202"; //modDevice.Wcs = new Random().Next(100).ToString(); //modDevice.Plc = new Random().Next(100).ToString(); //modDevice.Status = true; } return list; } [HttpPost] [ApiDescriptionSettings(Name = "WriteValue")] [DisplayName("写入值")] public async Task WriteValue(WriteWcsDeviceInput modDevice) { var modPlc = await _wcsDeviceRep.Context.Queryable().FirstAsync(s => s.Id == modDevice.PlcId); if (modPlc == null) throw Oops.Bah("找不到PLC信息"); PLCUtil modUtil = new PLCUtil(modPlc); var listPosition = await _wcsDeviceRep.Context.Queryable().Where(s => s.DeviceId == modDevice.Id).ToListAsync(); //(var result, var plc) = modConn.GetPlcDBValue(modDevice.PosType, modDevice.DbNumber, modDevice.PlcPos); //modDevice.Plc = Convert.ToString(plc); //任务号 var modPositionTask = listPosition.FirstOrDefault(s => s.Text == "任务号"); if (modPositionTask != null) modUtil.SetPlcDBValue(modPositionTask.PosType, modPositionTask.PlcPos, modDevice.TaskNo); //任务类型 var modPositionTaskType = listPosition.FirstOrDefault(s => s.Text == "任务类型"); if (modPositionTaskType != null) modUtil.SetPlcDBValue(modPositionTaskType.PosType, modPositionTaskType.PlcPos, modDevice.TaskType.ToString()); //起始工位 var modPositionStartLocatNo = listPosition.FirstOrDefault(s => s.Text == "起始工位"); if (modPositionStartLocatNo != null) modUtil.SetPlcDBValue(modPositionStartLocatNo.PosType, modPositionStartLocatNo.PlcPos, modDevice.StartLocatNo.ToString()); //目的工位 var modPositionEndLocatNo = listPosition.FirstOrDefault(s => s.Text == "目的工位"); if (modPositionEndLocatNo != null) modUtil.SetPlcDBValue(modPositionEndLocatNo.PosType, modPositionEndLocatNo.PlcPos, modDevice.EndLocatNo.ToString()); //托盘码 var modPositionPalletNo = listPosition.FirstOrDefault(s => s.Text == "托盘码"); if (modPositionPalletNo != null) modUtil.SetPlcDBValue(modPositionPalletNo.PosType, modPositionPalletNo.PlcPos, modDevice.PalletNo.ToString()); if (modPlc.Type == PLCTypeEnum.ConveyorLine) { //放货排 var modPositionReleaseRow = listPosition.FirstOrDefault(s => s.Text == "放货排"); if (modPositionReleaseRow != null) modUtil.SetPlcDBValue(modPositionReleaseRow.PosType, modPositionReleaseRow.PlcPos, modDevice.ReleaseRow.ToString()); //放货列 var modPositionReleaseCol = listPosition.FirstOrDefault(s => s.Text == "放货列"); if (modPositionReleaseRow != null) modUtil.SetPlcDBValue(modPositionReleaseCol.PosType, modPositionReleaseCol.PlcPos, modDevice.ReleaseCol.ToString()); //放货层 var modPositionReleaseStorey = listPosition.FirstOrDefault(s => s.Text == "放货层"); if (modPositionReleaseStorey != null) modUtil.SetPlcDBValue(modPositionReleaseStorey.PosType, modPositionReleaseStorey.PlcPos, modDevice.ReleaseStorey.ToString()); //取货排 var modPositionPickRow = listPosition.FirstOrDefault(s => s.Text == "取货排"); if (modPositionPickRow != null) modUtil.SetPlcDBValue(modPositionPickRow.PosType, modPositionPickRow.PlcPos, modDevice.PickRow.ToString()); //取货列 var modPositionPickCol = listPosition.FirstOrDefault(s => s.Text == "取货列"); if (modPositionPickCol != null) modUtil.SetPlcDBValue(modPositionPickCol.PosType, modPositionPickCol.PlcPos, modDevice.PickCol.ToString()); //取货层 var modPositionPickStorey = listPosition.FirstOrDefault(s => s.Text == "取货层"); if (modPositionPickStorey != null) modUtil.SetPlcDBValue(modPositionPickStorey.PosType, modPositionPickStorey.PlcPos, modDevice.PickStorey.ToString()); } modUtil.Close(); } #region 分拣码垛 /// /// 获取码垛机器人和拆垛机器人列表 /// /// [HttpGet] [ApiDescriptionSettings(Name = "WcsPackPlcList")] [DisplayName("获取码垛机器人和拆垛机器人列表")] public async Task WcsPackPlcList() { return await _wcsDeviceRep.Context.Queryable() .Where(w => w.Type == PLCTypeEnum.RobotPalletizer || w.Type == PLCTypeEnum.StackingRobot) .OrderByDescending(o => o.Type) .Select(u => new { id = u.Id, name = u.Text } ).ToListAsync(); } /// /// 获取设备对应工位列表 /// /// /// [HttpGet] [ApiDescriptionSettings(Name = "WcsPackStationPlcList")] [DisplayName("获取设备对应工位列表")] public async Task> WcsPackStationPlcList([FromQuery] WcsDeviceBaseInput entry) { return await _wcsDeviceRep.Context.Queryable() .LeftJoin((device, task) => device.StationNum == task.Port) .Where((device, task) => device.PlcId == entry.PlcId) .OrderBy((device, task) => device.CreateTime) .Select((device, task) => new WcsDeviceTaskOrderDto() { Id=device.Id, Text=device.Text, TaskNo=task.TaskNo, OrderNo=task.OrderNo, LotNo=task.LotNo, SkuNo=task.SkuNo, SkuName=task.SkuName, LineNo=task.LineNo, Status =task.Status, PZNo=task.PZNo, Qty=task.Qty }) .ToListAsync(); } #endregion }