You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

254 lines
14 KiB
C#

using AngleSharp.Dom;
using Khd.Core.Domain.Dto.wcs;
using Khd.Core.Domain.Models;
using Khd.Core.EntityFramework;
using Khd.Core.Wcs.Global;
using Masuit.Tools.Logging;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using Khd.Core.Domain.Dto.webapi;
using Khd.Core.Plc.S7;
using Khd.Core.Wcs.Wcs;
using System.Data;
using Microsoft.EntityFrameworkCore;
using Z.EntityFramework.Plus;
using Masuit.Tools;
using Microsoft.IdentityModel.Tokens;
namespace Khd.Core.Wcs.Wcs
{
/// <summary>
/// 出库点线程
/// </summary>
public class OutWarePoint
{
private readonly IHost _host;
private readonly Plc.S7.Plc _plc;
List<NodeSetting> ScanPoint { get; set; }//点位信息
BaseSitenode? sitenode { get; set; }//站台信息
NodeSetting? NodeSettingCarNo { get; set; }
NodeSetting? NodeSettingCarState { get; set; }
NodeSetting? NodeSettingCarRun { get; set; }
NodeSetting? NodeSettingWcsState { get; set; }
NodeSetting? NodeSettingWcsSend { get; set; }
NodeSetting? NodeSettingWcsSendMaterial { get; set; }
NodeSetting? NodeSettingPLCSendSendMaterialstate { get; set; }
Thread FlowPointThread;
public OutWarePoint(IHost host, Plc.S7.Plc plc, string siteNo)
{
this._host = host;
this._plc = plc;
//this.ScanPoint = StaticData.NodeSettingList.Where(t => t.siteNo == siteNo).ToList();//加载当前站点所对应的点位
this.sitenode = StaticData.SiteNodeList.FirstOrDefault(t => t.siteNo == siteNo);//获取当前站台信息
this.NodeSettingCarNo = this.ScanPoint.FirstOrDefault(t => t.plcpointNo.Contains("carno"));
this.NodeSettingCarState = this.ScanPoint.FirstOrDefault(t => t.plcpointNo.Contains("carstate"));
this.NodeSettingCarRun = this.ScanPoint.FirstOrDefault(t => t.plcpointNo.Contains("carrun"));
this.NodeSettingWcsState = this.ScanPoint.FirstOrDefault(t => t.plcpointNo.Contains("plcsendK"));
this.NodeSettingWcsSend = this.ScanPoint.FirstOrDefault(t => t.plcpointNo.Contains("wcsend"));
this.NodeSettingWcsSendMaterial = this.ScanPoint.FirstOrDefault(t => t.plcpointNo.Contains("wcssendmessage"));
this.NodeSettingPLCSendSendMaterialstate = this.ScanPoint.FirstOrDefault(t => t.plcpointNo.Contains("plcsendmessage"));
try
{
//默认启动,清理plc的上位机写入点位值
this._plc.Write(NodeSettingCarRun.plcpointAddress, MainCentralControl.QingKongDianWei);
this._plc.Write(NodeSettingWcsSend.plcpointAddress, MainCentralControl.QingKongDianWei);
this._plc.Write(NodeSettingWcsSendMaterial.plcpointAddress, MainCentralControl.QingKongDianWei);
}
catch (Exception ex)
{
Console.WriteLine("站点" + siteNo + " 初始化数据异常" + ex.Message);
LogManager.Error(ex);
}
}
/// <summary>
/// 启动上件扫描监听
/// </summary>
public void StartPoint()
{
FlowPointThread = new Thread(MonitorInLocatorPoint);
FlowPointThread.Start();
}
public void MonitorInLocatorPoint()
{
while (true)
{
try
{
//通用逻辑下,根据上件记录表,可区分业务,入库,出库,回库
//- 入库 - 根据小车任务表,可区分流入哪个站点
//- 出库 - 根据小车任务表,可区分流入哪个站点
//- 回库 - 根据小车任务表,可区分流入哪个站点
//- 特殊逻辑:下一站点多个站点,按多条线缓存数优先进少的;
var carno = this._plc.Read(NodeSettingCarNo.plcpointAddress);
var carstate = this._plc.Read(NodeSettingCarState.plcpointAddress);
var carrun = this._plc.Read(NodeSettingCarRun.plcpointAddress);
var wcsstate = this._plc.Read(NodeSettingWcsState.plcpointAddress);
var wcsend = this._plc.Read(NodeSettingWcsSend.plcpointAddress);
if (carno != null && carstate != null && carrun != null && wcsstate != null && wcsend != null)
{
//清除点位信息
if (Convert.ToInt32(carno) == 0 && Convert.ToInt32(wcsend) == 1 && Convert.ToInt32(wcsstate) == 1)
{
this._plc.Write(NodeSettingCarRun.plcpointAddress, MainCentralControl.QingKongDianWei); //清空小车去向点位
this._plc.Write(NodeSettingWcsSendMaterial.plcpointAddress, MainCentralControl.QingKongDianWei); //清空车身长度点位
this._plc.Write(NodeSettingWcsSend.plcpointAddress, MainCentralControl.QingKongDianWei); //清空wcs处理完成点位
}
//正常读到小车信息
if (Convert.ToInt32(carno) > 0 && Convert.ToInt32(wcsend) == 0 && Convert.ToInt32(wcsstate) == 0 && Convert.ToInt32(carstate) == 1 && Convert.ToInt32(carrun) == 0)
{
var downLength = GetTargetTo(Convert.ToInt32(carno));
if (string.IsNullOrWhiteSpace(downLength))
{
//Console.WriteLine("OutWarePoint类 WriteMaterialMessage方法未确认返回值,查看报错日志!");
Thread.Sleep(1000);
continue;
}
object ToInt16downLength = MainCentralControl.getValue("2", downLength);
this._plc.Write(this.NodeSettingCarRun.plcpointAddress, MainCentralControl.WcsMoRenQuXiang);//写入小车去向
this._plc.Write(this.NodeSettingWcsSendMaterial.plcpointAddress, ToInt16downLength); //写入车身长度
this._plc.Write(this.NodeSettingWcsSend.plcpointAddress, MainCentralControl.WcsChuLiWanCheng);//写入wcs处理完成
string zfc = downLength == "2900" ? "空挂具" : "挂具";
LogManager.Info($"当前时间{DateTime.Now} >>> {zfc}:{carno}经过下件站点:{this.sitenode.siteNo},WCS写入去向:1,写入车身长度:{ToInt16downLength};");
}
}
}
catch (Exception ex)
{
LogManager.Error(ex);
}
finally
{
Thread.Sleep(2000);
}
}
}
/// <summary>
/// 根据上件点,物料号,获取下一点位信息
/// </summary>
/// <param name="siteNo">交互所属站点</param>
/// <param name="carno">到位挂具号码</param>
/// <param name="isOver">是否是订单结束处理的判断传参</param>
/// <returns></returns>
private string GetTargetTo(int carno)
{
try
{
using var scope = _host.Services.CreateScope();
using var dbContext = scope.ServiceProvider.GetRequiredService<DefaultDbContext>();
var basecarModel = dbContext.BaseCar.Where(t => t.carNo == carno).FirstOrDefault();
var waitdownlineModel = dbContext.BaseWaitdownline.Where(t => t.carNo == carno).FirstOrDefault();
if (waitdownlineModel == null && basecarModel != null)
{
//添加车辆在Waitdownline表中的缓存信息
BaseWaitdownline waitdownModel = new()
{
id = Guid.NewGuid(),
carId = basecarModel.id,
carNo = basecarModel.carNo,
carName = basecarModel.carName,
downline = 2,
materielNo = "",
materielNum = 0,
isDelete = 0,
createTime = DateTime.Now,
createBy = "DownLine"
};
dbContext.Add(waitdownModel);
dbContext.SaveChanges();
return "2900";//调试无信息挂具写入数值
}
else if (waitdownlineModel != null)
{
if (!string.IsNullOrWhiteSpace(waitdownlineModel.materielNo))//挂具绑定物料表中,物料不为空时
{
//查找车型长度后返回写入Plc
string sqlQuery = @$"
SELECT
b1.material_no AS materialNo,
b1.k46up_length AS k46upLength,
b1.k48up_length AS k48upLength,
b1.down_length AS downLength,
b2.order_code AS definefield1,
b2.line_code AS definefield2,
b2.prod_code AS definefield3
FROM
base_materialinfo b1
JOIN base_production_order_split b2
ON b2.prod_code LIKE CONCAT('%', b1.material_no, '%')
WHERE
b2.order_code = '{waitdownlineModel.materielNo}'";
var QueryOrder = dbContext.ExecuteSqlQuery<BaseMaterialinfo>(sqlQuery).ToList();
if (QueryOrder?.Count > 0 && !string.IsNullOrWhiteSpace(QueryOrder[0].downLength))
{
RelieveCarAndMaterial(carno, waitdownlineModel.materielNo);//下件点下件解除挂具与物料信息绑定关系
return QueryOrder[0].downLength;
}
}
else if (string.IsNullOrWhiteSpace(waitdownlineModel.materielNo))//挂具绑定物料表中,物料为空时
{
return "2900";//与电气沟通是空时写入长度最小值
}
}
//添加下线记录后修正任务和等待下线记录表
void RelieveCarAndMaterial(int carno, string vin)
{
//挂具在下件点交互完成后记录下线数据
var orderMessage = dbContext.BaseProductionOrderSplit.Where(t => t.orderCode == vin).ToList();
if (orderMessage?.Count > 0 && waitdownlineModel != null)
{
string XianBie = orderMessage[0].lineCode;
char lastChar = XianBie[XianBie.Length - 1];
var XianBieNum = Char.GetNumericValue(lastChar).ToString();
BaseFollowmessage baseFollowmessage = new()
{
id = Guid.NewGuid().ToString(),
sid = orderMessage[0].id,
vinCode = orderMessage[0].orderCode,
upSite = waitdownlineModel.createBy,
downSite = XianBieNum == "1" ? "K02" : "K07",
carNo = carno,
materialName = orderMessage[0].prodDesc,
lineCode = orderMessage[0].lineCode,
lineName = orderMessage[0].lineCode == "一线" ? "一线" : "二线",
isover = 1,
est = orderMessage[0].est,
quantity = orderMessage[0].quantity,
actualquantity = 1,
productionSequence = orderMessage[0].productionSequence,
createBy = "OrderOverInsert",
createDate = DateTime.Now
};
dbContext.Add(baseFollowmessage);
dbContext.SaveChanges();
//解除base_waitdownline表中挂具与任务VIN码绑定关系,修改车辆线上状态为已下线
dbContext.BaseWaitdownline.Where(t => t.id == waitdownlineModel.id).Update(a => new BaseWaitdownline() { materielNo = string.Empty, materielNum = 0, createTime = DateTime.Now, downline = 2, createBy = string.Empty, definefield1 = string.Empty});
//修改base_production_order_split表订单为结束状态
dbContext.BaseProductionOrderSplit.Where(t => t.id == orderMessage[0].id).Update(a => new BaseProductionOrderSplit() { isover = 1 });
}
}
//LogManager.Info($"OutWarePoint类 WriteMaterialMessage方法未确认返回值,从底部返回NULL!");
return "";
}
catch (Exception ex)
{
//LogManager.Info($"错误日志输出 >>> OutWarePoint类 WriteMaterialMessage 方法报错 {ex}");
LogManager.Error(ex);
return null;
}
}
}
}