2024-11-08 10:10:59 +08:00
|
|
|
|
using System;
|
2025-01-24 11:44:27 +08:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
using System.Linq;
|
|
|
|
|
|
using System.Text.Json;
|
2024-11-08 10:10:59 +08:00
|
|
|
|
using System.Threading;
|
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
using DOAN.Infrastructure.PLC;
|
|
|
|
|
|
using DOAN.Model.PBL;
|
2025-01-24 11:44:27 +08:00
|
|
|
|
using DOAN.ServiceCore.Signalr;
|
|
|
|
|
|
using Microsoft.AspNetCore.SignalR;
|
2024-11-08 10:10:59 +08:00
|
|
|
|
using Microsoft.Extensions.Hosting;
|
|
|
|
|
|
using SqlSugar.IOC;
|
2025-01-24 11:44:27 +08:00
|
|
|
|
|
2024-11-08 10:10:59 +08:00
|
|
|
|
namespace DOAN.ServiceCore
|
|
|
|
|
|
{
|
2025-01-16 10:13:50 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 永驻线程
|
|
|
|
|
|
/// 功能:检测传感器信号,判断箱子数
|
|
|
|
|
|
/// </summary>
|
2024-11-08 11:31:08 +08:00
|
|
|
|
public class DoanBackgroundService : IHostedService, IDisposable
|
2024-11-08 10:10:59 +08:00
|
|
|
|
{
|
2025-01-24 11:44:27 +08:00
|
|
|
|
private readonly IHubContext<PBLhub> notificationHubContext;
|
2024-11-08 10:10:59 +08:00
|
|
|
|
private readonly CancellationTokenSource _cancellationTokenSource = new CancellationTokenSource();
|
|
|
|
|
|
private Task _executingTask;
|
2024-11-13 15:53:15 +08:00
|
|
|
|
private PLCTool pLCTool;
|
2025-01-24 11:44:27 +08:00
|
|
|
|
private List<Storagelocation> storagelocationList = new List<Storagelocation>();
|
|
|
|
|
|
private List<PlcAddressTable> pointPositionList = new List<PlcAddressTable>();
|
|
|
|
|
|
private Timer refreshTimer;
|
|
|
|
|
|
|
|
|
|
|
|
public DoanBackgroundService(IHubContext<PBLhub> hubContext)
|
|
|
|
|
|
{
|
|
|
|
|
|
notificationHubContext = hubContext;
|
|
|
|
|
|
}
|
2024-11-08 10:10:59 +08:00
|
|
|
|
|
|
|
|
|
|
public Task StartAsync(CancellationToken cancellationToken)
|
|
|
|
|
|
{
|
2024-11-13 15:53:15 +08:00
|
|
|
|
pLCTool = new PLCTool();
|
|
|
|
|
|
pLCTool.ConnectPLC();
|
2024-11-08 10:10:59 +08:00
|
|
|
|
|
2025-01-24 11:44:27 +08:00
|
|
|
|
// 初始化料架层和点位表数据
|
|
|
|
|
|
RefreshData(null);
|
|
|
|
|
|
|
|
|
|
|
|
// 启动后台任务
|
|
|
|
|
|
_executingTask = ExecuteAsync(_cancellationTokenSource.Token);
|
2024-11-08 10:10:59 +08:00
|
|
|
|
|
2025-01-24 11:44:27 +08:00
|
|
|
|
// 设置定时器每分钟刷新一次料架层和点位表
|
|
|
|
|
|
refreshTimer = new Timer(RefreshData, null, TimeSpan.Zero, TimeSpan.FromMinutes(1));
|
2024-11-08 10:10:59 +08:00
|
|
|
|
|
|
|
|
|
|
return _executingTask.IsCompleted ? _executingTask : Task.CompletedTask;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public async Task StopAsync(CancellationToken cancellationToken)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 请求取消后台任务
|
|
|
|
|
|
_cancellationTokenSource.Cancel();
|
|
|
|
|
|
|
2025-01-24 11:44:27 +08:00
|
|
|
|
// 停止定时器
|
|
|
|
|
|
refreshTimer?.Change(Timeout.Infinite, Timeout.Infinite);
|
|
|
|
|
|
refreshTimer?.Dispose();
|
|
|
|
|
|
|
2024-11-08 10:10:59 +08:00
|
|
|
|
// 等待后台任务完成
|
|
|
|
|
|
await Task.WhenAny(_executingTask, Task.Delay(Timeout.Infinite, cancellationToken));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-01-24 11:44:27 +08:00
|
|
|
|
private static bool GetInvertedBit(byte b, int position)
|
2025-01-16 10:13:50 +08:00
|
|
|
|
{
|
|
|
|
|
|
if (position < 0 || position > 7)
|
|
|
|
|
|
{
|
|
|
|
|
|
throw new ArgumentOutOfRangeException(nameof(position), "Position must be between 0 and 7.");
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
byte mask = (byte)(1 << position);
|
|
|
|
|
|
bool isSet = (b & mask) != 0;
|
2025-01-24 11:44:27 +08:00
|
|
|
|
return !isSet; // 返回取反后的值
|
2025-01-16 10:13:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2025-01-20 15:55:34 +08:00
|
|
|
|
/// 功能:检测传感器信号,判断箱子数并更新库存及日志
|
2025-01-16 10:13:50 +08:00
|
|
|
|
/// </summary>
|
2025-01-20 15:55:34 +08:00
|
|
|
|
/// <param name="stoppingToken">取消令牌</param>
|
2025-01-16 10:13:50 +08:00
|
|
|
|
/// <returns></returns>
|
2024-11-08 10:10:59 +08:00
|
|
|
|
private async Task ExecuteAsync(CancellationToken stoppingToken)
|
|
|
|
|
|
{
|
2025-01-20 15:55:34 +08:00
|
|
|
|
try
|
|
|
|
|
|
{
|
2024-11-13 15:53:15 +08:00
|
|
|
|
while (!stoppingToken.IsCancellationRequested)
|
|
|
|
|
|
{
|
2025-01-24 11:44:27 +08:00
|
|
|
|
// 读取PLC I/O状态(12个字节)
|
|
|
|
|
|
byte[] plcSensorValues;
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
plcSensorValues = pLCTool.ReadAllValue("VB100", 12);
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
Console.WriteLine($"读取PLC数据异常: {ex.Message}\n堆栈跟踪: {ex.StackTrace}");
|
|
|
|
|
|
continue;
|
|
|
|
|
|
}
|
2025-01-16 10:13:50 +08:00
|
|
|
|
|
2025-01-24 11:44:27 +08:00
|
|
|
|
var updateStoragelocationList = new List<Storagelocation>();
|
|
|
|
|
|
var inventoryLogs = new List<Inventorylog>();
|
2025-01-16 10:13:50 +08:00
|
|
|
|
|
2025-01-24 11:44:27 +08:00
|
|
|
|
foreach (var layerItem in storagelocationList)
|
2025-01-20 15:55:34 +08:00
|
|
|
|
{
|
|
|
|
|
|
// 获取这个料架层的点位表
|
2025-01-24 11:44:27 +08:00
|
|
|
|
var layerPoints = pointPositionList.Where(it => it.FkStorageId == layerItem.Id).ToList();
|
2025-01-16 10:13:50 +08:00
|
|
|
|
|
2025-01-24 11:44:27 +08:00
|
|
|
|
if (layerPoints.Any())
|
2024-11-13 15:53:15 +08:00
|
|
|
|
{
|
2025-01-20 16:15:40 +08:00
|
|
|
|
int currentPackageCount = 0; // 默认最小值为0
|
2024-11-08 10:10:59 +08:00
|
|
|
|
|
2025-01-24 11:44:27 +08:00
|
|
|
|
foreach (var point in layerPoints)
|
2024-11-13 15:53:15 +08:00
|
|
|
|
{
|
2025-01-20 15:55:34 +08:00
|
|
|
|
int row = point.ByteNum - 100;
|
|
|
|
|
|
int col = point.BitNum - 1;
|
2025-01-16 10:13:50 +08:00
|
|
|
|
|
2025-01-24 11:44:27 +08:00
|
|
|
|
if (plcSensorValues != null && row >= 0 && row < plcSensorValues.Length)
|
2025-01-16 10:13:50 +08:00
|
|
|
|
{
|
2025-01-20 15:55:34 +08:00
|
|
|
|
currentPackageCount += GetInvertedBit(plcSensorValues[row], col) ? 1 : 0;
|
2025-01-16 10:13:50 +08:00
|
|
|
|
}
|
2024-11-13 15:53:15 +08:00
|
|
|
|
}
|
2024-11-08 11:31:08 +08:00
|
|
|
|
|
2025-01-20 15:55:34 +08:00
|
|
|
|
// 检查箱子数量变化并记录日志
|
2025-01-24 11:44:27 +08:00
|
|
|
|
int previousPackageCount = layerItem.PackageNum.GetValueOrDefault();
|
|
|
|
|
|
if (currentPackageCount > previousPackageCount)
|
2025-01-20 15:55:34 +08:00
|
|
|
|
{
|
2025-01-24 11:44:27 +08:00
|
|
|
|
UpdateAndLog(layerItem, currentPackageCount, 2, "出库", inventoryLogs);
|
2025-01-20 15:55:34 +08:00
|
|
|
|
updateStoragelocationList.Add(layerItem);
|
|
|
|
|
|
}
|
2025-01-24 11:44:27 +08:00
|
|
|
|
else if (currentPackageCount < previousPackageCount)
|
2025-01-20 15:55:34 +08:00
|
|
|
|
{
|
2025-01-24 11:44:27 +08:00
|
|
|
|
UpdateAndLog(layerItem, currentPackageCount, 1, "入库", inventoryLogs);
|
2025-01-20 15:55:34 +08:00
|
|
|
|
updateStoragelocationList.Add(layerItem);
|
|
|
|
|
|
}
|
2025-01-24 11:44:27 +08:00
|
|
|
|
|
|
|
|
|
|
// 补料报警触发
|
|
|
|
|
|
if (layerItem.IsLackAlarm == 1 && currentPackageCount <= layerItem.AlarmNum.GetValueOrDefault())
|
|
|
|
|
|
{
|
|
|
|
|
|
var alarmData = new
|
|
|
|
|
|
{
|
|
|
|
|
|
RackCode = layerItem.RackCode,
|
|
|
|
|
|
LayerNum = layerItem.LayerNum,
|
|
|
|
|
|
CurrentPackageCount = currentPackageCount,
|
|
|
|
|
|
AlarmThreshold = layerItem.AlarmNum
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
string alarmMessage = System.Text.Json.JsonSerializer.Serialize(alarmData);
|
|
|
|
|
|
await notificationHubContext.Clients.All.SendAsync("PBL_lack_alarm", alarmMessage);
|
|
|
|
|
|
}
|
2025-01-16 10:13:50 +08:00
|
|
|
|
}
|
2024-11-08 11:31:08 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-01-20 15:55:34 +08:00
|
|
|
|
// 更新库存
|
2025-01-24 11:44:27 +08:00
|
|
|
|
if (updateStoragelocationList.Any())
|
2025-01-20 15:55:34 +08:00
|
|
|
|
{
|
|
|
|
|
|
await DbScoped.SugarScope.CopyNew().Updateable(updateStoragelocationList).ExecuteCommandAsync();
|
2025-01-24 13:09:04 +08:00
|
|
|
|
// 发送库存变更Socket通知
|
|
|
|
|
|
string changeMessage = "库存变动";
|
|
|
|
|
|
await notificationHubContext.Clients.All.SendAsync("PBL_storagelocation_change", changeMessage);
|
2025-01-20 15:55:34 +08:00
|
|
|
|
}
|
2024-11-08 11:31:08 +08:00
|
|
|
|
|
2025-01-20 15:55:34 +08:00
|
|
|
|
// 插入库存变更日志
|
2025-01-24 11:44:27 +08:00
|
|
|
|
if (inventoryLogs.Any())
|
2025-01-20 15:55:34 +08:00
|
|
|
|
{
|
|
|
|
|
|
await DbScoped.SugarScope.CopyNew().Insertable(inventoryLogs).ExecuteCommandAsync();
|
|
|
|
|
|
}
|
|
|
|
|
|
// 添加延迟以避免频繁查询
|
|
|
|
|
|
await Task.Delay(200, stoppingToken);
|
2024-11-08 10:10:59 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-01-20 15:55:34 +08:00
|
|
|
|
catch (OperationCanceledException)
|
|
|
|
|
|
{
|
|
|
|
|
|
Console.WriteLine("任务已取消");
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
Console.WriteLine($"DoanBackGround线程ExecuteAsync异常: {ex.Message}\n堆栈跟踪: {ex.StackTrace}");
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-01-24 11:44:27 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 刷新料架层和点位表数据
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="state"></param>
|
|
|
|
|
|
private void RefreshData(object state)
|
|
|
|
|
|
{
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
using (var scope = DbScoped.SugarScope.CopyNew())
|
|
|
|
|
|
{
|
|
|
|
|
|
storagelocationList = scope.Queryable<Storagelocation>().ToListAsync().Result;
|
|
|
|
|
|
pointPositionList = scope.Queryable<PlcAddressTable>().ToListAsync().Result;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
Console.WriteLine($"刷新数据异常: {ex.Message}\n堆栈跟踪: {ex.StackTrace}");
|
|
|
|
|
|
// 如果刷新失败,保持现有数据不变
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-01-20 15:55:34 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 更新料架位置的箱子数量并记录库存日志
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="storageLocation">料架位置对象</param>
|
|
|
|
|
|
/// <param name="newPackageCount">新的箱子数量</param>
|
|
|
|
|
|
/// <param name="operation">操作类型 (1-出库, 2-入库)</param>
|
|
|
|
|
|
/// <param name="operatorName">操作员名称</param>
|
|
|
|
|
|
/// <param name="inventoryLogs">库存日志列表</param>
|
2025-01-24 11:44:27 +08:00
|
|
|
|
private void UpdateAndLog(
|
|
|
|
|
|
Storagelocation storageLocation,
|
|
|
|
|
|
int newPackageCount,
|
|
|
|
|
|
int operation,
|
|
|
|
|
|
string operatorName,
|
|
|
|
|
|
List<Inventorylog> inventoryLogs)
|
2025-01-20 15:55:34 +08:00
|
|
|
|
{
|
|
|
|
|
|
storageLocation.PackageNum = newPackageCount;
|
|
|
|
|
|
storageLocation.UpdatedBy = operatorName;
|
|
|
|
|
|
storageLocation.UpdatedTime = DateTime.Now;
|
|
|
|
|
|
|
2025-01-24 11:44:27 +08:00
|
|
|
|
var inventoryLog = new Inventorylog
|
2025-01-20 15:55:34 +08:00
|
|
|
|
{
|
|
|
|
|
|
Id = SnowFlakeSingle.Instance.NextId().ToString(),
|
|
|
|
|
|
RackCode = storageLocation.RackCode,
|
|
|
|
|
|
Partnumber = storageLocation.Partnumber,
|
|
|
|
|
|
Operation = operation,
|
2025-01-24 11:44:27 +08:00
|
|
|
|
PackageNum = Math.Abs(newPackageCount - storageLocation.PackageNum.GetValueOrDefault()),
|
2025-01-20 15:55:34 +08:00
|
|
|
|
CreatedBy = operatorName,
|
|
|
|
|
|
CreatedTime = DateTime.Now.ToLocalTime()
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
inventoryLogs.Add(inventoryLog);
|
2024-11-08 10:10:59 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public void Dispose()
|
|
|
|
|
|
{
|
2024-11-13 15:53:15 +08:00
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
pLCTool.ConnectClose();
|
|
|
|
|
|
_cancellationTokenSource.Cancel();
|
2025-01-24 11:44:27 +08:00
|
|
|
|
_executingTask?.Wait();
|
2024-11-13 15:53:15 +08:00
|
|
|
|
_cancellationTokenSource.Dispose();
|
2025-01-24 11:44:27 +08:00
|
|
|
|
refreshTimer?.Change(Timeout.Infinite, Timeout.Infinite);
|
|
|
|
|
|
refreshTimer?.Dispose();
|
2024-11-13 15:53:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
Console.WriteLine("DoanBackGround线程Dispose异常:" + ex.Message);
|
|
|
|
|
|
}
|
2024-11-08 10:10:59 +08:00
|
|
|
|
}
|
2025-01-24 11:44:27 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-11-13 15:53:15 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|