using HH.WCS.Mobox3.WeiLi.api;
|
using HH.WCS.Mobox3.WeiLi.core;
|
using HH.WCS.Mobox3.WeiLi.dispatch;
|
using HH.WCS.Mobox3.WeiLi.models;
|
using HH.WCS.Mobox3.WeiLi.util;
|
using HH.WCS.Mobox3.WeiLi.wms;
|
using Newtonsoft.Json;
|
using SqlSugar;
|
using System;
|
using System.Collections.Generic;
|
using System.Linq;
|
using System.Net.Sockets;
|
using System.Security.Cryptography;
|
using System.Text;
|
using static HH.WCS.Mobox3.WeiLi.api.OtherModel;
|
|
namespace HH.WCS.Mobox3.WeiLi.process
|
{
|
internal class TaskProcess
|
{
|
|
private static HttpHelper httpHelper = new HttpHelper();
|
#region 任务相关
|
//--------------------------------------------------任务相关--------------------------------------------------
|
/// <summary>
|
/// 取货卸货完成,缓存位状态更新
|
/// </summary>
|
/// <param name="mst"></param>
|
/// <param name="load"></param>
|
internal static void CacheBitUpdate(WCSTask mst, bool load)
|
{
|
var trayCarryCount = mst.N_CNTR_COUNT > 0 ? mst.N_CNTR_COUNT : 1;
|
if (load)
|
{
|
Console.WriteLine($"任务{mst.S_CODE} 货位{mst.S_START_LOC}取货完成,起点解绑容器{mst.S_CNTR_CODE}");
|
LogHelper.Info($"任务{mst.S_CODE} 货位{mst.S_START_LOC}取货完成,起点解绑容器{mst.S_CNTR_CODE}");
|
LocationHelper.UnBindingLoc(mst.S_START_LOC, mst.S_CNTR_CODE.Split(',').ToList());
|
}
|
else
|
{
|
Console.WriteLine($"任务{mst.S_CODE} 货位{mst.S_END_LOC}卸货完成,终点绑定容器{mst.S_CNTR_CODE}");
|
LogHelper.Info($"任务{mst.S_CODE} 货位{mst.S_END_LOC}卸货完成,终点绑定容器{mst.S_CNTR_CODE}");
|
LocationHelper.BindingLoc(mst.S_END_LOC, mst.S_CNTR_CODE.Split(',').ToList());
|
}
|
|
}
|
|
/// <summary>
|
/// 任务取消,缓存位状态更新
|
/// </summary>
|
/// <param name="mst"></param>
|
internal static void CacheBitCancelUpdate(WCSTask mst)
|
{
|
LogHelper.Info($"任务取消 任务调度类型{mst.S_SCHEDULE_TYPE}");
|
var db = new SqlHelper<object>().GetInstance();
|
//任务取消,取货完成前的,起点的loadingCount和终点unLoadingCount都清除,取货完成的只处理终点
|
if (mst.S_SCHEDULE_TYPE == "WCS")
|
{
|
if (WCSHelper.CheckActionRecordExist(mst.S_CODE, 5))
|
{
|
//根据客户现场要求,如果取货完成任务失败人工拉到终点,我们就当卸货完成处理;如果是人工拉走到其它区域,我们就解锁终点,删除托盘。
|
//终点绑定
|
CacheBitUpdate(mst, false);
|
LocationHelper.UnLockLoc(mst.S_END_LOC);
|
var wmsTask = db.Queryable<WMSTask>().Where(a => a.S_CODE == mst.S_OP_CODE).First();
|
if (wmsTask != null)
|
{
|
LocationHelper.UnLockLoc(wmsTask.S_END_LOC);
|
}
|
}
|
else
|
{
|
//起点终点解锁
|
var wmsTask = db.Queryable<WMSTask>().Where(a => a.S_CODE == mst.S_OP_CODE).First();
|
if (wmsTask != null)
|
{
|
LocationHelper.UnLockLoc(wmsTask.S_END_LOC);
|
LocationHelper.UnLockLoc(mst.S_START_LOC);
|
LocationHelper.UnLockLoc(mst.S_END_LOC);
|
}
|
}
|
}
|
else
|
{
|
if (WCSHelper.CheckActionRecordExist(mst.S_CODE, 4))
|
{
|
//终点数量置0
|
db.Updateable<Location>().SetColumns(it => new Location() { N_CURRENT_NUM = 0 }).Where(a => a.S_CODE == mst.S_END_LOC).ExecuteCommand();
|
}
|
else
|
{
|
db.Updateable<Location>().SetColumns(it => new Location() { N_CURRENT_NUM = 0 }).Where(a => a.S_CODE == mst.S_START_LOC).ExecuteCommand();
|
}
|
|
|
LocationHelper.UnLockLoc(mst.S_START_LOC);
|
LocationHelper.UnLockLoc(mst.S_END_LOC);
|
var wmsTask = db.Queryable<WMSTask>().Where(a => a.S_CODE == mst.S_OP_CODE).First();
|
if (wmsTask != null)
|
{
|
LocationHelper.UnLockLoc(wmsTask.S_END_LOC);
|
Reducequantity(wmsTask);
|
}
|
db.Deleteable<LocCntrRel>().Where(a => a.S_CNTR_CODE == mst.S_CNTR_CODE).ExecuteCommand();
|
db.Deleteable<CntrItemRel>().Where(a => a.S_CNTR_CODE == mst.S_CNTR_CODE).ExecuteCommand();
|
|
}
|
|
|
|
|
}
|
|
public static void Reducequantity(WMSTask wmsTask)
|
{
|
var db = new SqlHelper<object>().GetInstance();
|
if (wmsTask.S_OP_DEF_NAME == "码盘入库")
|
{
|
//降入库单明细的分配量
|
var itemlist = db.Queryable<CntrItemRel>().Where(a => a.S_CNTR_CODE == wmsTask.S_CNTR_CODE).ToList();
|
if (itemlist.Count > 0)
|
{
|
LogHelper.Info($"查找到物料明细{itemlist.Count}条");
|
foreach (var item in itemlist)
|
{
|
var exp = Expressionable.Create<TN_Inbound_Detail>();
|
exp.And(it => it.S_IO_NO == item.S_BS_NO);
|
exp.And(it => it.N_BS_ROW_NO == item.S_WU);
|
exp.And(it => it.S_ITEM_CODE == item.S_ITEM_CODE);
|
exp.AndIF(!string.IsNullOrEmpty(item.S_BATCH_NO), it => it.S_BATCH_NO == item.S_BATCH_NO);
|
exp.AndIF(!string.IsNullOrEmpty(item.S_ERP_WH_CODE), it => it.S_ERP_WH_CODE == item.S_ERP_WH_CODE);
|
exp.AndIF(!string.IsNullOrEmpty(item.S_OWNER), it => it.S_OWNER == item.S_OWNER);
|
exp.AndIF(!string.IsNullOrEmpty(item.S_SUPPLIER_NO), it => it.S_SUPPLIER_NO == item.S_SUPPLIER_NO);
|
var InWorkOrder = db.Queryable<TN_Inbound_Detail>().Where(exp.ToExpression()).First();
|
InWorkOrder.F_ACC_B_QTY = InWorkOrder.F_ACC_B_QTY - item.F_QTY;
|
db.Updateable(InWorkOrder).UpdateColumns(it => new { it.F_ACC_B_QTY }).ExecuteCommand();
|
}
|
}
|
else
|
{
|
LogHelper.Info($"未找到托盘{wmsTask.S_CNTR_CODE}的物料明细");
|
}
|
}
|
else
|
{
|
LogHelper.Info($"作业类型为{wmsTask.S_OP_DEF_NAME} 无须降量");
|
}
|
}
|
|
/// <summary>
|
/// 安全请求
|
/// </summary>
|
/// <param name="mst"></param>
|
/// <param name="state"></param>
|
/// <param name="forkliftNo"></param>
|
/// <param name="extData"></param>
|
internal static void OperateReq(WCSTask mst, int state, string forkliftNo, string extData)
|
{
|
if (state == 1012)
|
{
|
CheckRfid(mst, forkliftNo);
|
}
|
|
//信号未定 后续对接
|
if (state == 1101 || state == 1103)
|
{
|
if (mst != null)
|
{
|
if (state == 1101)
|
{
|
//请求取货
|
Request(mst, true);
|
}
|
else
|
{
|
//请求卸货
|
Request(mst, false);
|
|
}
|
}
|
}
|
//信号未定后续对接
|
if (state == 1102 || state == 1104)
|
{
|
if (mst != null)
|
{
|
if (state == 1102)
|
{
|
//取货完成
|
Request2(mst, true);
|
}
|
else
|
{
|
//卸货完成
|
Request2(mst, false);
|
}
|
}
|
}
|
|
if (state == 1004)
|
{
|
if (mst.N_ERR == 200)
|
{
|
NDCApi.ChangeOrderParam(mst.S_CODE, 6, "1");
|
}
|
|
if (mst.N_ERR == 500)
|
{
|
var db = new SqlHelper<object>().GetInstance();
|
Location startLoc = null;
|
Location endLoc = null;
|
if (!string.IsNullOrEmpty(mst.S_ERR_LOC))
|
{
|
endLoc = db.Queryable<Location>().Where(a => a.S_CODE == mst.S_ERR_LOC).First();
|
startLoc = db.Queryable<Location>().Where(a => a.S_CODE == mst.S_END_LOC).First();
|
if (endLoc != null && startLoc != null)
|
{
|
//替换作业终点 及任务起点终点
|
var wmsTask = db.Queryable<WMSTask>().Where(a => a.S_CODE == mst.S_OP_CODE).First();
|
wmsTask.S_END_LOC = endLoc.S_CODE;
|
wmsTask.S_END_AREA = endLoc.S_AREA_CODE;
|
wmsTask.S_END_WH = endLoc.S_WH_CODE;
|
db.Updateable(wmsTask).UpdateColumns(a => new { a.S_END_LOC, a.S_END_AREA, a.S_END_WH }).ExecuteCommand();
|
mst.S_END_LOC = endLoc.S_CODE;
|
mst.S_END_AREA = endLoc.S_AREA_CODE;
|
mst.S_END_WH = endLoc.S_WH_CODE;
|
mst.S_START_LOC = startLoc.S_CODE;
|
mst.S_START_AREA = startLoc.S_AREA_CODE;
|
mst.S_START_WH = startLoc.S_WH_CODE;
|
mst.S_ERR_LOC = "";
|
db.Updateable(mst).UpdateColumns(a => new { a.S_END_LOC, a.S_END_AREA, a.S_END_WH, a.S_START_LOC, a.S_START_AREA, a.S_START_WH, a.S_ERR_LOC }).ExecuteCommand();
|
|
|
TaskProcess.Reducequantity(wmsTask);
|
//NDCApi.ChangeOrderParam(mst.S_CODE, 1, startLoc.S_AGV_SITE);
|
//NDCApi.ChangeOrderParam(mst.S_CODE, 2, endLoc.S_AGV_SITE);
|
//NDCApi.ChangeOrderParam(mst.S_CODE, 4, "0");
|
//
|
//NDCApi.ChangeOrderParam(mst.S_CODE, 6, "2");
|
NDCApi.ChangeOrderParam(mst.S_CODE, 1, $"{startLoc.S_AGV_SITE};{endLoc.S_AGV_SITE};0;0;0;2");
|
}
|
}
|
}
|
}
|
}
|
|
internal static void AddTaskAction(ApiModel.AgvTaskState model)
|
{
|
var db = new SqlHelper<object>().GetInstance();
|
var task = db.Queryable<Task_Action>().Where(a => a.TASK_NO == model.task_no && a.STATE == model.State).First();
|
if (task == null)
|
{
|
task = new Task_Action
|
{
|
TASK_NO = model.task_no,
|
STATE = model.State,
|
LOCKNO = model.LockNo,
|
EXT_DATA = model.ext_data,
|
EXT1 = model.Ext1,
|
EXT2 = model.Ext2,
|
FORKLIFT_NO = model.forklift_no,
|
ERRCODE = model.ErrCode,
|
N_CNTR_COUNT = model.N_CNTR_COUNT
|
};
|
db.Insertable(task).ExecuteCommand();
|
}
|
}
|
|
/// <summary>
|
/// 取卸货完成退出安全交互
|
/// </summary>
|
/// <param name="mst"></param>
|
/// <param name="v"></param>
|
internal static void Request2(WCSTask mst, bool v)
|
{
|
var db = new SqlHelper<object>().GetInstance();
|
var Url = Settings.HASeverUrl;
|
if (Url != null)
|
{
|
string trkType = "";
|
string stnNo = "";
|
if (v)
|
{
|
//取货请求
|
trkType = "1";
|
var info = db.Queryable<PeiZhi>().Where(a => a.BitCode == mst.S_START_LOC).First();
|
if (info != null)
|
{
|
stnNo = info.PlcLocation;
|
}
|
}
|
else
|
{
|
//卸货请求
|
trkType = "2";
|
var info = db.Queryable<PeiZhi>().Where(a => a.BitCode == mst.S_END_LOC).First();
|
if (info != null)
|
{
|
stnNo = info.PlcLocation;
|
}
|
}
|
|
var req = JsonConvert.SerializeObject(new
|
{
|
requestPk = mst.S_CODE,
|
trkType = trkType,
|
method = "Complete",
|
stnNo = stnNo,
|
carNo = "1001",
|
clientCode = "WMS",
|
reqTime = DateTime.Now.ToString("yyyy-MM-dd hh:mm:ss")
|
});
|
|
var feed = new HttpHelper().WebPost(Url + "agvCallback", req);
|
LogHelper.Info($"安全交互 任务号={mst.S_CODE} url={Url + "agvCallback"} 传输数据={req} 返回数据={feed}");
|
if (!string.IsNullOrEmpty(feed))
|
{
|
var res = JsonConvert.DeserializeObject<PlcSendTaskres>(feed);
|
if (res.code == "0")
|
{
|
if (res.isAllow == "1")
|
{
|
////修改任务表数据为请求成功 S_NOTE :1
|
//mst.S_NOTE = "1";
|
//db.Updateable(mst).UpdateColumns(it => new { it.S_NOTE }).ExecuteCommand();
|
}
|
|
}
|
}
|
}
|
|
}
|
|
/// <summary>
|
/// 请求取卸货安全交互
|
/// </summary>
|
/// <param name="mst"></param>
|
/// <param name="v"></param>
|
internal static void Request(WCSTask mst, bool v)
|
{
|
var db = new SqlHelper<object>().GetInstance();
|
var Url = Settings.HASeverUrl;
|
if (Url != null)
|
{
|
string trkType = "";
|
string stnNo = "";
|
if (v)
|
{
|
//取货请求
|
trkType = "1";
|
var info = db.Queryable<PeiZhi>().Where(a => a.BitCode == mst.S_START_LOC).First();
|
if (info != null)
|
{
|
stnNo = info.PlcLocation;
|
}
|
}
|
else
|
{
|
//卸货请求
|
trkType = "2";
|
var info = db.Queryable<PeiZhi>().Where(a => a.BitCode == mst.S_END_LOC).First();
|
if (info != null)
|
{
|
stnNo = info.PlcLocation;
|
}
|
}
|
|
var req = JsonConvert.SerializeObject(new
|
{
|
requestPk = mst.S_CODE,
|
trkType = trkType,
|
method = "AgvRequest",
|
stnNo = stnNo,
|
carNo = "1001",
|
clientCode = "WMS",
|
reqTime = DateTime.Now.ToString("yyyy-MM-dd hh:mm:ss")
|
});
|
|
var feed = new HttpHelper().WebPost(Url + "agvCallback", req);
|
LogHelper.Info($"安全交互 任务号={mst.S_CODE} url={Url + "agvCallback"} 传输数据={req} 返回数据={feed}");
|
if (!string.IsNullOrEmpty(feed))
|
{
|
var res = JsonConvert.DeserializeObject<PlcSendTaskres>(feed);
|
if (res.code == "0")
|
{
|
if (res.isAllow == "1")
|
{
|
NDCApi.ChangeOrderParam(mst.S_CODE, 6, "1");
|
}
|
}
|
}
|
}
|
}
|
|
|
internal static void UpdateAgvNo(WCSTask mst, string forkliftNo)
|
{
|
var db = new SqlHelper<object>().GetInstance();
|
mst.S_EQ_NO = forkliftNo;
|
db.Updateable(mst).UpdateColumns(it => new { it.S_EQ_NO }).ExecuteCommand();
|
}
|
|
internal static void ThirdReportStatus(WCSTask mst, int state, string forkliftNo)
|
{
|
var db = new SqlHelper<object>().GetInstance();
|
if (WCSHelper.CheckActionRecordExist(mst.S_CODE, 4) && state == 7)
|
{
|
state = 2;
|
}
|
//if (state == 2)
|
//{
|
// if (Settings.DJCodes.Where(a => a.DJAreaCode == mst.S_END_AREA).FirstOrDefault() != null)
|
// {
|
// InspectionAreaStockUpdate(mst.S_CNTR_CODE, mst.S_END_LOC);
|
// }
|
//}
|
if (state == 4 || state == 2 || state == 7)
|
{
|
AgvState(mst, state, forkliftNo);
|
}
|
}
|
|
internal static void CheckRfid(WCSTask mst, string agv)
|
{
|
Console.WriteLine($"扫码请求 task={mst.S_CODE} agv={agv}");
|
LogHelper.Info($"扫码请求 task={mst.S_CODE} agv={agv}");
|
|
var codeReader = Settings.AgvRfids.Where(a => a.agv == agv).FirstOrDefault();
|
if (codeReader != null)
|
{
|
LogHelper.Info($"agv={codeReader.agv},ip={codeReader.ip}");
|
for (int i = 1; i <= 3; i++)
|
{
|
//try
|
//{
|
//var res = OITcpHelper.HexTransit(new OITcpHelper.StrTransitData { data = "4C 4F 4E 0D", host = codeReader.ip, port = 9004 });
|
var res = SendHexOnce(codeReader.ip, codeReader.port, "41");
|
LogHelper.Info($"扫码返回报文{res}");
|
//if (string.IsNullOrEmpty(res))
|
//{
|
// NDCApi.ChangeOrderParam(mst.S_CODE, 6, "3");
|
//}
|
//else
|
//{
|
if (res.Length >= 18)
|
{
|
res = res.Substring(2, 16);
|
}
|
|
|
var code = hexToStr(res);
|
Console.WriteLine(code);
|
LogHelper.Info($"托盘信息为={mst.S_CNTR_CODE},扫码结果为={code},结果:{mst.S_CNTR_CODE.Trim() == code.Trim()}");
|
if (!string.IsNullOrEmpty(code) && res.Trim() != "022503")
|
{
|
if (mst.S_CNTR_CODE.Trim() == code.Trim())
|
{
|
NDCApi.ChangeOrderParam(mst.S_CODE, 6, "1");
|
}
|
else
|
{
|
NDCApi.ChangeOrderParam(mst.S_CODE, 6, "2");
|
}
|
return;
|
}
|
else if (i == 3)
|
{
|
if (!string.IsNullOrEmpty(code))
|
{
|
NDCApi.ChangeOrderParam(mst.S_CODE, 6, "3");
|
}
|
else
|
{
|
NDCApi.ChangeOrderParam(mst.S_CODE, 6, "4");
|
}
|
}
|
//}
|
|
//}
|
//catch (Exception ex)
|
//{
|
// LogHelper.Error($"扫码流程异常 异常信息{ex.Message}", ex);
|
//}
|
|
}
|
}
|
}
|
|
private static string hexToStr(string hexString)
|
{
|
hexString = hexString.Replace(" ", "");
|
if ((hexString.Length % 2) != 0)
|
hexString += " ";
|
byte[] returnBytes = new byte[hexString.Length / 2];
|
for (int i = 0; i < returnBytes.Length; i++)
|
returnBytes[i] = Convert.ToByte(hexString.Substring(i * 2, 2), 16);
|
|
return Encoding.ASCII.GetString(returnBytes);
|
}
|
|
private static string SendHexOnce(string ip, int port, string hex)
|
{
|
var res = string.Empty;
|
try
|
{
|
Socket client = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
|
//client.Connect(ip, port);
|
IAsyncResult connectResult = client.BeginConnect(ip, port, null, null);
|
if (!connectResult.AsyncWaitHandle.WaitOne(5000))
|
{
|
client.Close();
|
return "";
|
}
|
client.ReceiveTimeout = 2000;
|
if (client.Connected)
|
{
|
client.Send(Hex2Bytes(hex));
|
byte[] buffer = new byte[1024];
|
|
var length = client.Receive(buffer, SocketFlags.None);
|
byte[] data = new byte[length];
|
Array.Copy(buffer, data, length);
|
res = BitConverter.ToString(data).Replace("-", "");
|
|
|
client.Disconnect(true);
|
client.Dispose();
|
}
|
client = null;
|
}
|
catch (Exception ex)
|
{
|
LogHelper.Error(ex.Message, ex);
|
}
|
return res;
|
}
|
|
internal static byte[] Hex2Bytes(string hexString)
|
{
|
hexString = hexString.Replace(" ", "");
|
if ((hexString.Length % 2) != 0)
|
hexString += " ";
|
byte[] returnBytes = new byte[hexString.Length / 2];
|
for (int i = 0; i < returnBytes.Length; i++)
|
returnBytes[i] = Convert.ToByte(hexString.Substring(i * 2, 2), 16);
|
return returnBytes;
|
}
|
|
private static void AgvState(WCSTask mst, int state, string forkliftNo)
|
{
|
try
|
{
|
if (!mst.S_OP_CODE.Contains("OP"))
|
{
|
//调用AGV完成接口
|
var url = Settings.ZTSeverUrl2 + "/business/BU_HX01";
|
var req = new { TaskNo = mst.S_OP_CODE, State = state, AgvNo = int.Parse(forkliftNo), cntrNo = mst.S_CNTR_CODE, workNo = mst.S_OP_CODE };
|
LogHelper.Info($"agv完成任务回报 地址{url} 接口参数{JsonConvert.SerializeObject(req)}");
|
var feedback = new HttpHelper().WebPost(url, JsonConvert.SerializeObject(req));
|
if (!string.IsNullOrEmpty(feedback))
|
{
|
var weilires = JsonConvert.DeserializeObject<WeiLiResult>(feedback);
|
if (weilires.code == 200)
|
{
|
LogHelper.Info($"接口反馈成功");
|
}
|
else
|
{
|
LogHelper.Info($"agv完成任务回报 反馈失败 失败原因={weilires.msg}");
|
}
|
}
|
else
|
{
|
LogHelper.Info($"agv完成任务回报 反馈超时");
|
}
|
}
|
else
|
{
|
var db = new SqlHelper<object>().GetInstance();
|
var wmsTask = db.Queryable<WMSTask>().Where(a => a.S_CODE == mst.S_OP_CODE).First();
|
if (wmsTask != null)
|
{
|
if (!string.IsNullOrEmpty(wmsTask.S_BS_NO))
|
{
|
//调用AGV完成接口
|
var url = Settings.ZTSeverUrl2 + "/business/BU_HX01";
|
var req = new { TaskNo = wmsTask.S_BS_NO, State = state, AgvNo = int.Parse(forkliftNo), cntrNo = mst.S_CNTR_CODE, workNo = wmsTask.S_BS_NO };
|
LogHelper.Info($"agv完成任务回报 地址{url} 接口参数{JsonConvert.SerializeObject(req)}");
|
var feedback = new HttpHelper().WebPost(url, JsonConvert.SerializeObject(req));
|
if (!string.IsNullOrEmpty(feedback))
|
{
|
var weilires = JsonConvert.DeserializeObject<WeiLiResult>(feedback);
|
if (weilires.code == 200)
|
{
|
LogHelper.Info($"接口反馈成功");
|
}
|
else
|
{
|
LogHelper.Info($"agv完成任务回报 反馈失败 失败原因={weilires.msg}");
|
}
|
}
|
else
|
{
|
LogHelper.Info($"agv完成任务回报 反馈超时");
|
}
|
}
|
}
|
else LogHelper.Info($"任务{mst.S_CODE} 没有找到作业号为{mst.S_OP_CODE}的作业");
|
}
|
}
|
catch (Exception ex)
|
{
|
LogHelper.Error($"agv完成任务回报异常 异常信息={ex.Message}", ex);
|
}
|
}
|
|
public static void InspectionAreaStockUpdate(string CNTR_CODE, string END_LOC)
|
{
|
try
|
{
|
var db = new SqlHelper<object>().GetInstance();
|
var url = Settings.ZTSeverUrl + "InspectionAreaStockUpdate";
|
var req = new { cntrCode = CNTR_CODE, location = END_LOC, data = new List<object>() };
|
var cn = db.Queryable<CntrItemRel>().Where(a => a.S_CNTR_CODE == CNTR_CODE).ToList();
|
if (cn.Count > 0)
|
{
|
var grouplist = cn.GroupBy(a => new { a.S_ITEM_CODE, a.S_BATCH_NO, a.S_OWNER, a.S_ERP_WH_CODE, a.S_SUPPLIER_NO }).ToList();
|
for (int i = 0; i < grouplist.Count; i++)
|
{
|
var num = 0;
|
var serialNo = new List<string>();
|
var S_ITEM_CODE = grouplist[i].Key.S_ITEM_CODE;
|
var S_BATCH_NO = grouplist[i].Key.S_BATCH_NO;
|
var S_OWNER = grouplist[i].Key.S_OWNER;
|
var S_ERP_WH_CODE = grouplist[i].Key.S_ERP_WH_CODE;
|
var S_SUPPLIER_NO = grouplist[i].Key.S_SUPPLIER_NO;
|
var itemgroup = cn.Where(it => it.S_ITEM_CODE == S_ITEM_CODE && it.S_BATCH_NO == S_BATCH_NO && it.S_OWNER == S_OWNER && it.S_ERP_WH_CODE == S_ERP_WH_CODE && it.S_SUPPLIER_NO == S_SUPPLIER_NO).ToList();
|
if (itemgroup.Count > 0)
|
{
|
itemgroup.ForEach(a =>
|
{
|
//填充数据
|
num = num + Convert.ToInt32(a.F_QTY);
|
if (!string.IsNullOrEmpty(a.S_SERIAL_NO))
|
{
|
serialNo.Add(a.S_SERIAL_NO);
|
}
|
});
|
req.data.Add(new { itemCode = S_ITEM_CODE, factoryCode = S_OWNER, stockCode = S_ERP_WH_CODE, qty = num, vendorCode = S_SUPPLIER_NO, batchNo = S_BATCH_NO, serialNo = serialNo });
|
}
|
}
|
LogHelper.Info($"检验区库存增加接口 地址{url} 接口参数{JsonConvert.SerializeObject(req)}");
|
var feedback = new HttpHelper().WebPost(url, JsonConvert.SerializeObject(req));
|
LogHelper.Info($"检验区库存增加接口 地址{url} 接口参数{JsonConvert.SerializeObject(req)} 返回参数{feedback}");
|
if (!string.IsNullOrEmpty(feedback))
|
{
|
var weilires = JsonConvert.DeserializeObject<WeiLiResult>(feedback);
|
if (weilires.code == 200)
|
{
|
LogHelper.Info($"接口反馈成功");
|
db.Deleteable<CntrItemRel>().Where(a => a.S_CNTR_CODE == CNTR_CODE).ExecuteCommand();
|
}
|
else
|
{
|
LogHelper.Info($"检验区库存增加接口 反馈失败 失败原因={weilires.msg}");
|
}
|
}
|
else
|
{
|
LogHelper.Info($"检验区库存增加接口 反馈超时");
|
}
|
}
|
else LogHelper.Info($"托盘{CNTR_CODE} 未绑定物料");
|
|
}
|
catch (Exception ex)
|
{
|
LogHelper.Error($"检验区库存增加异常 异常信息={ex.Message}", ex);
|
}
|
|
}
|
|
public static void receiveUnloadConfirm(string CNTR_CODE, string END_LOC)
|
{
|
try
|
{
|
var db = new SqlHelper<object>().GetInstance();
|
var url = Settings.ZTSeverUrl + "receiveUnloadConfirm";
|
var req = new { cntrCode = CNTR_CODE, location = END_LOC, data = new List<object>() };
|
var cn = db.Queryable<CntrItemRel>().Where(a => a.S_CNTR_CODE == CNTR_CODE).ToList();
|
if (cn.Count > 0)
|
{
|
var grouplist = cn.GroupBy(a => new { a.S_ITEM_CODE, a.S_BATCH_NO, a.S_OWNER, a.S_ERP_WH_CODE, a.S_SUPPLIER_NO }).ToList();
|
for (int i = 0; i < grouplist.Count; i++)
|
{
|
var num = 0;
|
var serialNo = new List<string>();
|
var S_ITEM_CODE = grouplist[i].Key.S_ITEM_CODE;
|
var S_BATCH_NO = grouplist[i].Key.S_BATCH_NO;
|
var S_OWNER = grouplist[i].Key.S_OWNER;
|
var S_ERP_WH_CODE = grouplist[i].Key.S_ERP_WH_CODE;
|
var S_SUPPLIER_NO = grouplist[i].Key.S_SUPPLIER_NO;
|
var itemgroup = cn.Where(it => it.S_ITEM_CODE == S_ITEM_CODE && it.S_BATCH_NO == S_BATCH_NO && it.S_OWNER == S_OWNER && it.S_ERP_WH_CODE == S_ERP_WH_CODE && it.S_SUPPLIER_NO == S_SUPPLIER_NO).ToList();
|
if (itemgroup.Count > 0)
|
{
|
itemgroup.ForEach(a =>
|
{
|
//填充数据
|
num = num + Convert.ToInt32(a.F_QTY);
|
if (!string.IsNullOrEmpty(a.S_SERIAL_NO))
|
{
|
serialNo.Add(a.S_SERIAL_NO);
|
}
|
});
|
req.data.Add(new { itemCode = S_ITEM_CODE, factoryCode = S_OWNER, stockCode = S_ERP_WH_CODE, qty = num, vendorCode = S_SUPPLIER_NO, batchNo = S_BATCH_NO, serialNo = serialNo });
|
}
|
}
|
LogHelper.Info($"收货装框确认接口 地址{url} 接口参数{JsonConvert.SerializeObject(req)}");
|
var feedback = new HttpHelper().WebPost(url, JsonConvert.SerializeObject(req));
|
LogHelper.Info($"收货装框确认接口 地址{url} 接口参数{JsonConvert.SerializeObject(req)} 返回参数{feedback}");
|
if (!string.IsNullOrEmpty(feedback))
|
{
|
var weilires = JsonConvert.DeserializeObject<WeiLiResult>(feedback);
|
if (weilires.code == 200)
|
{
|
LogHelper.Info($"接口反馈成功");
|
}
|
else
|
{
|
LogHelper.Info($"收货装框确认接口 反馈失败 失败原因={weilires.msg}");
|
}
|
}
|
else
|
{
|
LogHelper.Info($"收货装框确认接口 反馈超时");
|
}
|
}
|
else LogHelper.Info($"托盘{CNTR_CODE} 未绑定物料");
|
|
}
|
catch (Exception ex)
|
{
|
LogHelper.Error($"收货装框确认接口 异常信息={ex.Message}", ex);
|
}
|
|
}
|
|
/// <summary>
|
/// 任务拦截
|
/// </summary>
|
/// <param name="task"></param>
|
/// <returns></returns>
|
internal static bool Intercept(WCSTask mst)
|
{
|
var result = false;
|
//出库任务是批量生成的,初始终点我们先给一个虚拟点,不推送。有单独的现场去判断出库缓存区光电,空了再给出库任务分配终点
|
if (mst.S_END_LOC.Trim() == "出库虚拟点")
|
{
|
result = true;
|
}
|
return result;
|
}
|
|
/// <summary>
|
/// 任务状态更新处理
|
/// </summary>
|
/// <param name="mst"></param>
|
/// <param name="state"></param>
|
internal static void OperateStatus(WCSTask mst, int state)
|
{
|
if (state == 4)
|
{
|
CacheBitUpdate(mst, true);
|
}
|
if (state == 6)//卸货完成
|
{
|
CacheBitUpdate(mst, false);
|
}
|
if (state == 7)
|
{
|
CacheBitCancelUpdate(mst);
|
}
|
}
|
|
private static object locLocker = new object();
|
|
/// <summary>
|
/// 堆叠库区出入库任务申请
|
/// </summary>
|
/// <param name="ls"></param>
|
/// <param name="cntrs"></param>
|
/// <param name="area"></param>
|
/// <param name="itemCode"></param>
|
/// <param name="itemBatch"></param>
|
/// <param name="taskType"></param>
|
/// <param name="insStock"></param>
|
/// <returns></returns>
|
internal static bool ApplyTN_Task(Location ls, ref List<string> cntrs, string area, string itemCode, string itemBatch, string taskType, bool insStock = true)
|
{
|
var result = false;
|
lock (locLocker)
|
{
|
try
|
{
|
if (insStock)
|
{
|
Console.WriteLine($"MoboxHelperCreateTask: {area}-{itemCode}-{itemBatch}-{taskType}");
|
var endTN_Location = GetLocation4In(area, itemCode, itemBatch, 3);
|
if (endTN_Location != null)
|
{
|
var endLayer = endTN_Location.N_CURRENT_NUM == 0 ? 1 : 2;
|
var taskNo = DateTime.Now.Ticks.ToString();
|
result = TaskProcess.CreateTransport(ls.S_CODE, endTN_Location.S_CODE, taskType, cntrs, 1, endLayer, 3, 70);
|
}
|
else
|
{
|
Console.WriteLine($"MoboxHelperCreateTask: 未找到终点货位");
|
}
|
}
|
else
|
{
|
var startTN_Location = GetLocation4Out(area, itemCode, itemBatch, 3);
|
if (startTN_Location != null)
|
{
|
var startLayer = startTN_Location.N_CURRENT_NUM <= 3 ? 1 : 2;
|
var taskNo = DateTime.Now.Ticks.ToString();
|
var carryCount = startTN_Location.N_CURRENT_NUM > 3 ? startTN_Location.N_CURRENT_NUM - 3 : startTN_Location.N_CURRENT_NUM;
|
//出库要从起点获取托盘
|
var cntrList = LocationHelper.GetLocCntr(startTN_Location.S_CODE);
|
if (cntrList.Count == startTN_Location.N_CURRENT_NUM)
|
{
|
cntrs = cntrList.OrderByDescending(a => a.T_CREATE).Take(carryCount).Select(a => a.S_CNTR_CODE.Trim()).ToList();
|
result = TaskProcess.CreateTransport(startTN_Location.S_CODE, ls.S_CODE, taskType, cntrs, startLayer, 1, carryCount, 65);
|
}
|
else
|
{
|
Console.WriteLine($"起点托盘数量和货位容器表不符合,请检查【货位表】和【货位容器表】");
|
}
|
|
}
|
}
|
}
|
catch (Exception ex)
|
{
|
|
Console.WriteLine("MoboxHelperCreateTask:" + ex.Message);
|
LogHelper.Error("MoboxHelperCreateTask:" + ex.Message, ex);
|
}
|
}
|
return result;
|
|
|
}
|
|
private static Location GetLocation4Out(string area, string itemCode, string itemBatch, int v)
|
{
|
throw new NotImplementedException();
|
}
|
|
private static Location GetLocation4In(string area, string itemCode, string itemBatch, int v)
|
{
|
throw new NotImplementedException();
|
}
|
|
/// <summary>
|
/// 普通货架区的出入库申请
|
/// </summary>
|
/// <param name="ls"></param>
|
/// <param name="cntrs"></param>
|
/// <param name="area"></param>
|
/// <param name="taskType"></param>
|
/// <param name="itemCode"></param>
|
/// <param name="insStock"></param>
|
/// <returns></returns>
|
internal static bool ApplyNormalTN_Task(Location ls, ref List<string> cntrs, string area, string taskType, string itemCode, bool insStock = true)
|
{
|
var result = false;
|
lock (locLocker)
|
{
|
try
|
{
|
if (insStock)
|
{
|
Console.WriteLine($"MoboxHelperCreateTask: {area}-{taskType}");
|
var endTN_Location = new Location();
|
if (endTN_Location != null)
|
{
|
var taskNo = DateTime.Now.Ticks.ToString();
|
result = TaskProcess.CreateTransport(ls.S_CODE, endTN_Location.S_CODE, taskType, cntrs, 70);
|
}
|
else
|
{
|
Console.WriteLine($"MoboxHelperCreateTask: 未找到终点货位");
|
}
|
}
|
else
|
{
|
var startTN_Location = new Location();
|
if (startTN_Location != null)
|
{
|
//出库要从起点获取托盘
|
var cntrList = LocationHelper.GetLocCntr(startTN_Location.S_CODE);
|
if (cntrList.Count == startTN_Location.N_CURRENT_NUM)
|
{
|
result = TaskProcess.CreateTransport(startTN_Location.S_CODE, ls.S_CODE, taskType, new List<string> { cntrList[0].S_CNTR_CODE }, 65);
|
}
|
else
|
{
|
Console.WriteLine($"起点托盘数量和货位容器表不符合,请检查【货位表】和【货位容器表】");
|
}
|
|
}
|
}
|
}
|
catch (Exception ex)
|
{
|
|
Console.WriteLine("MoboxHelperCreateTask:" + ex.Message);
|
LogHelper.Error("MoboxHelperCreateTask:" + ex.Message, ex);
|
}
|
}
|
return result;
|
|
|
}
|
/// <summary>
|
/// 推送任务
|
/// </summary>
|
/// <param name="mst"></param>
|
internal static bool SendTask(WCSTask mst)
|
{
|
var db = new SqlHelper<object>().GetInstance();
|
var result = false;
|
var start = "0"; var end = "0";
|
var Trow = "0";
|
var taskType = mst.S_TYPE.Trim();
|
if (mst.N_B_STATE == 0)
|
{
|
start = LocationHelper.GetAgvSite(mst.S_START_LOC);
|
end = LocationHelper.GetAgvSite(mst.S_END_LOC);
|
|
if (Settings.ConnetAreas.Where(a => a.InLocList.Contains(mst.S_END_LOC)).FirstOrDefault() != null)
|
{
|
Trow = "1024";
|
}
|
|
var wmsTask = db.Queryable<WMSTask>().Where(a => a.S_CODE == mst.S_OP_CODE).First();
|
if (wmsTask != null && wmsTask.N_TRAY_WEIGHT > 0)
|
{
|
// 1000 2000 3000 4000 5000
|
Trow = (wmsTask.N_TRAY_WEIGHT * 4096).ToString();
|
}
|
|
|
Console.WriteLine($"SendTask {mst.S_CODE}");
|
Console.WriteLine("start=" + start);
|
Console.WriteLine("end= " + end);
|
var dic = new List<param>();
|
dic.Add(new param { name = "From", value = start });
|
dic.Add(new param { name = "To", value = end });
|
dic.Add(new param { name = "FRow", value = "4" });
|
dic.Add(new param { name = "TRow", value = Trow });
|
dic.Add(new param { name = "AGV", value = "0" });
|
|
var res = NDCApi.AddOrderNew(1, mst.N_PRIORITY, mst.S_CODE, dic);
|
if (res.err_code == 0)
|
{
|
//推送成功,修改任务优先级
|
mst.N_B_STATE = 1;
|
WCSHelper.UpdateStatus(mst, "已推送");
|
result = true;
|
}
|
|
}
|
return result;
|
}
|
|
/// <summary>
|
/// 设备任务取消
|
/// </summary>
|
/// <param name="task"></param>
|
internal static bool Cancel(WCSTask task)
|
{
|
var result = false;
|
var Url = Settings.HASeverUrl;
|
if (Url != null)
|
{
|
var model = new PlcCancelTaskreq
|
{
|
contNo = task.S_CNTR_CODE,
|
requestPk = task.S_CODE
|
};
|
var feed = httpHelper.WebPost(Url + "cancel", JsonConvert.SerializeObject(model));
|
LogHelper.Info($"任务{task.S_CODE} 推送任务参数={JsonConvert.SerializeObject(model)} 下游反馈={feed}");
|
if (!string.IsNullOrEmpty(feed))
|
{
|
var res = JsonConvert.DeserializeObject<PlcSendTaskres>(feed);
|
if (res.code == "0")
|
{
|
LogHelper.Info($"任务{task.S_CODE} 取消成功");
|
result = true;
|
}
|
else
|
{
|
LogHelper.Info($"任务{task.S_CODE} 取消失败");
|
}
|
}
|
}
|
else
|
{
|
LogHelper.Info($"取消下游PLC任务 获取API地址配置失败 请检查配置文件");
|
}
|
return result;
|
}
|
|
|
public class PlcCancelTaskreq
|
{
|
/// <summary>
|
/// 请求编号 唯一标示
|
/// </summary>
|
public string requestPk { get; set; }
|
/// <summary>
|
/// 托盘号
|
/// </summary>
|
public string contNo { get; set; }
|
/// <summary>
|
/// 操作人 默认
|
/// </summary>
|
public string clientCode { get; set; } = "WMS";
|
/// <summary>
|
/// 操作时间 默认
|
/// </summary>
|
public string reqTime { get; set; } = DateTime.Now.ToString("yyyy-MM-dd hh:mm:ss");
|
}
|
|
/// <summary>
|
/// 设备任务推送
|
/// </summary>
|
/// <param name="task"></param>
|
internal static void PlcSendTask(WCSTask task)
|
{
|
var from = task.S_START_LOC;
|
var to = task.S_END_LOC;
|
//var from = LocationHelper.GetAgvSite(task.S_START_LOC).ToString();
|
//var to = LocationHelper.GetAgvSite(task.S_END_LOC).ToString();
|
var db = new SqlHelper<object>().GetInstance();
|
var Url = Settings.HASeverUrl;
|
if (Url != null)
|
{
|
var agvbit = db.Queryable<PeiZhi>().Where(a => a.BitCode == task.S_START_LOC || a.BitCode == task.S_END_LOC).ToList();
|
|
foreach (var item in agvbit)
|
{
|
if (item.BitCode == task.S_START_LOC)
|
{
|
from = item.PlcLocation;
|
}
|
else
|
{
|
to = item.PlcLocation;
|
}
|
}
|
|
|
//var endInfo = Settings.GetInOutLocationList().Where(a => a.Location == task.S_END_LOC).FirstOrDefault();
|
//if (endInfo != null)
|
//{
|
// to = endInfo.Code;
|
//}
|
|
var model = new PlcSendTaskreq
|
{
|
contNo = task.S_CNTR_CODE,
|
//frmPos = LocationHelper.GetAgvSite(task.S_START_LOC).ToString(),
|
//toPos = LocationHelper.GetAgvSite(task.S_END_LOC).ToString(),
|
contType = "",
|
frmPos = from,
|
toPos = to,
|
requestPk = task.S_CODE,
|
trkPrty = task.N_PRIORITY.ToString(),
|
trkType = task.S_OP_NAME == "入库" ? "1" : task.S_OP_NAME == "出库" ? "2" : "3"
|
};
|
var feed = httpHelper.WebPost(Url + "receive", JsonConvert.SerializeObject(model));
|
LogHelper.Info($"任务{task.S_CODE} 推送任务参数={JsonConvert.SerializeObject(model)} 下游反馈={feed}");
|
if (!string.IsNullOrEmpty(feed))
|
{
|
var res = JsonConvert.DeserializeObject<PlcSendTaskres>(feed);
|
if (res.code == "0")
|
{
|
task.N_B_STATE = 1;
|
WCSHelper.UpdateStatus(task, "已推送");
|
}
|
}
|
}
|
else
|
{
|
LogHelper.Info($"推送下游PLC任务 获取API地址配置失败 请检查配置文件");
|
}
|
|
}
|
|
public class PlcSendTaskreq
|
{
|
/// <summary>
|
/// 请求编号 唯一标示
|
/// </summary>
|
public string requestPk { get; set; }
|
/// <summary>
|
/// 托盘号
|
/// </summary>
|
public string contNo { get; set; }
|
/// <summary>
|
/// 托盘类型 默认
|
/// </summary>
|
public string contType { get; set; } = "";
|
/// <summary>
|
/// 任务类型 出库 入库 移动
|
/// </summary>
|
public string trkType { get; set; }
|
/// <summary>
|
/// 任务优先级 1-999
|
/// </summary>
|
public string trkPrty { get; set; }
|
/// <summary>
|
/// 起始位置
|
/// </summary>
|
public string frmPos { get; set; }
|
/// <summary>
|
/// 目的地
|
/// </summary>
|
public string toPos { get; set; }
|
/// <summary>
|
/// 托盘空满状态 0:空 1:满
|
/// </summary>
|
public string isFull { get; set; } = "1";
|
/// <summary>
|
/// 搬运组号 默认
|
/// </summary>
|
public string groupNo { get; set; } = "";
|
/// <summary>
|
/// 操作人 默认
|
/// </summary>
|
public string clientCode { get; set; } = "WMS";
|
/// <summary>
|
/// 操作时间 默认
|
/// </summary>
|
public string reqTime { get; set; } = DateTime.Now.ToString("yyyy-MM-dd hh:mm:ss");
|
}
|
|
|
public class PlcSendTaskres
|
{
|
/// <summary>
|
/// 0:成功
|
/// </summary>
|
public string code { get; set; }
|
public string msg { get; set; }
|
/// <summary>
|
/// 任务编号
|
/// </summary>
|
public string requestPk { get; set; }
|
public string new_locate_no { get; set; }
|
public string isAllow { get; set; }
|
}
|
|
/// <summary>
|
/// 创建搬运任务
|
/// </summary>
|
/// <param name="start"></param>
|
/// <param name="end"></param>
|
/// <param name="taskType"></param>
|
/// <param name="cntrs"></param>
|
/// <param name="startLayer"></param>
|
/// <param name="endLayer"></param>
|
/// <param name="trayCarryCount"></param>
|
/// <param name="priority"></param>
|
/// <returns></returns>
|
public static bool CreateTransport(string start, string end, string taskType, List<string> cntrs, int startLayer, int endLayer, int trayCarryCount = 1, int priority = 1)
|
{
|
var result = false;
|
//批次号存托盘号,1~3个托盘
|
var trayCodes = string.Join(",", cntrs);
|
var taskNo = DateTime.Now.Ticks.ToString();
|
|
|
var res = WCSHelper.CreateTask(taskNo, start.Trim(), end.Trim(), taskType, priority, trayCodes, trayCarryCount, startLayer, endLayer);
|
if (res)
|
{
|
result = true;
|
//任务创建成功,起点货位出库锁定,终点货位入库锁定
|
LocationHelper.LockLoc(start, 1);
|
LocationHelper.LockLoc(end, 2);
|
}
|
return result;
|
}
|
|
|
public static bool CreateTransport(string start, string end, string taskType, List<string> cntrs, int priority = 1)
|
{
|
var result = false;
|
//批次号存托盘号,1~3个托盘
|
var trayCodes = string.Join(",", cntrs);
|
var taskNo = DateTime.Now.Ticks.ToString();
|
|
|
var res = WCSHelper.CreateTask(taskNo, start.Trim(), end.Trim(), taskType, priority, trayCodes, 1, 1, 1);
|
if (res)
|
{
|
result = true;
|
LocationHelper.LockLoc(start, 2);
|
LocationHelper.LockLoc(end, 1);
|
|
}
|
return result;
|
}
|
#endregion
|
|
|
|
}
|
}
|