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.

535 lines
20 KiB
C#

1 month ago
using Furion;
using Furion.FriendlyException;
1 month ago
using Furion.Logging;
using Furion.RemoteRequest.Extensions;
1 month ago
using Myshipping.Application.EDI.Dtos;
using Myshipping.Application.Entity;
using Myshipping.Core;
using Myshipping.Core.Entity;
using Myshipping.Core.Service;
1 month ago
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;
1 month ago
using System.Collections.Generic;
using System.Linq;
1 month ago
using System.Text.RegularExpressions;
1 month ago
using System.Threading.Tasks;
namespace Myshipping.Application.EDI
{
/// <summary>
/// ONE API订舱
/// </summary>
public static class ONESoApiHelper
{
1 month ago
public async static Task<KeyValuePair<bool, string>> DoPost(long custOrderId)
{
var repCustOrder = App.GetService<SqlSugarRepository<BookingCustomerOrder>>();
var repOrder = App.GetService<SqlSugarRepository<BookingOrder>>();
var repCtn = App.GetService<SqlSugarRepository<BookingCtn>>();
var repCustomer = App.GetService<SqlSugarRepository<DjyCustomer>>();
var repContact = App.GetService<SqlSugarRepository<DjyCustomerContact>>();
var repTemplate = App.GetService<SqlSugarRepository<BookingSoTemplate>>();
var cache = App.GetService<ISysCacheService>();
var cacheService = App.GetService<ISysCacheService>();
var custOrder = await repCustOrder.AsQueryable().Filter(null, true).FirstAsync(x => x.Id == custOrderId);
if (custOrder == null)
{
return new KeyValuePair<bool, string>(false, "客户订舱信息未找到");
}
var sysConfigList = await cache.GetAllSysConfig();
var sCfgSpiderUrl = sysConfigList.FirstOrDefault(x => x.Code == "BookingPostApiServerAddr" && x.GroupCode == "DJY_CONST");
if (sCfgSpiderUrl == null)
{
return new KeyValuePair<bool, string>(false, "订舱API的URL地址未配置请联系管理员");
}
var sCfgUserKey = sysConfigList.FirstOrDefault(x => x.Code == "BookingPostApiKey" && x.GroupCode == "DJY_CONST");
var sCfgUserSecret = sysConfigList.FirstOrDefault(x => x.Code == "BookingPostApiSecret" && x.GroupCode == "DJY_CONST");
if (sCfgUserKey == null || sCfgUserSecret == null)
{
return new KeyValuePair<bool, string>(false, "订舱API的KEY和密钥未配置请联系管理员");
}
1 month ago
if (string.IsNullOrEmpty(custOrder.CARGOID))
{
return new KeyValuePair<bool, string>(false, "订单货物类型为空");
}
1 month ago
1 month ago
if (custOrder.CARGOID is not ("S" or "R"))
{
return new KeyValuePair<bool, string>(false, "订单货物类型需为“普通货”或“冻柜”");
}
1 month ago
1 month ago
BookingSoTemplate template = null;
DjyCustomerContact custContact = null;
var postModel = new ONESoApiModel
{
webAccount = custOrder.BookingAccount,
webPassword = custOrder.BookingPassword,
1 month ago
1 month ago
userKey = sCfgUserKey.Value,
userSecret = sCfgUserSecret.Value,
1 month ago
1 month ago
mark = new
{
BookingCustomerOrderId = custOrder.Id,
BookingNo = custOrder.BOOKINGNO,
BookingId = custOrder.BookingId,
}
1 month ago
};
//查找模板:
//1.根据客户订舱信息中的BookingUserId和BookingTenantId去客户信息中根据CustSysId查找客户公司及联系人员工信息
//2.根据找到的客户及联系人信息,查找订舱模板
if (custOrder.BookingUserId > 0 && custOrder.BookingTenantId > 0)
{
custContact = await repCustomer.AsQueryable().Filter(null, true)
.InnerJoin<DjyCustomerContact>((cust, contact) => cust.Id == contact.CustomerId)
.Where((cust, contact) => cust.CustSysId == custOrder.BookingTenantId && contact.CustSysId == custOrder.BookingUserId)
.Select((cust, contact) => contact)
.SingleAsync();
if (custContact == null)
{
return new KeyValuePair<bool, string>(false, "未找到客户及联系人信息");
}
//根据:用户+船司+船司账号+约号,找到启用的模板
template = await repTemplate.AsQueryable().Filter(null, true).FirstAsync(x => x.IsDeleted == false && x.CarrierId == custOrder.CARRIERID && x.UserId == custContact.Id && x.IsEnable);
if (template == null)
{
return new KeyValuePair<bool, string>(false, "未找到订舱模板");
}
}
else
{
return new KeyValuePair<bool, string>(false, "未找到客户端公司和用户ID");
}
postModel.uploadType = template.Category; //DRAFT, TEMPLATE, BOOKING分别对应草稿, 模板, 订舱
postModel.saveName = template.TemplateName;
var mappingCtn = await cache.GetAllMappingCtn();
var mappingFrt = await cache.GetAllMappingFrt();
var mappingPortLoad = await cache.GetAllMappingPortLoad();
var mappingPort = await cache.GetAllMappingPort();
//收货地
var mapPlaceReceipt = mappingPortLoad.FirstOrDefault(x => x.Module == "DjyCustBooking" && x.CarrierCode == CarrierCodeConst.ONE && x.Code == custOrder.PLACERECEIPTCODE);
if (mapPlaceReceipt == null)
{
1 month ago
return new KeyValuePair<bool, string>(false, $"未找到收货地映射信息:{custOrder.PLACERECEIPTCODE}");
1 month ago
}
1 month ago
//目的地
var mapDestination = mappingPort.FirstOrDefault(x => x.Module == "DjyCustBooking" && x.CarrierCode == CarrierCodeConst.ONE && x.Code == custOrder.DESTINATIONCODE);
1 month ago
if (mapDestination == null)
1 month ago
{
1 month ago
return new KeyValuePair<bool, string>(false, $"未找到目的地映射信息:{custOrder.DESTINATIONCODE}");
1 month ago
}
1 month ago
//运输条款
var mappingService = await cacheService.GetAllMappingService();
var mappService = mappingService.FirstOrDefault(x => x.Module == "DjyCustBooking" && x.CarrierCode == CarrierCodeConst.ONE && x.Code == custOrder.SERVICE);
if (mappService == null)
1 month ago
{
1 month ago
return new KeyValuePair<bool, string>(false, $"未找到运输条款映射信息:{custOrder.SERVICE}");
1 month ago
}
1 month ago
if (!Regex.IsMatch(mappService.MapCode, "^[A-Za-z]+-[A-Za-z]+$"))
1 month ago
{
1 month ago
return new KeyValuePair<bool, string>(false, $"映射配置不正确:{mappService.MapCode}");
}
string[] arrService = null;
if (mappService == null)
{
arrService = custOrder.SERVICE.Split('-');
}
else
{
arrService = mappService.MapCode.Split('-');
1 month ago
}
1 month ago
//routes
1 month ago
postModel.routes = new ONESoApiRoute()
{
searchConditionDate = custOrder.ETD.Value.ToString("yyyy-MM-dd"),
originName = mapPlaceReceipt.MapCode,
destinationName = mapDestination.MapCode,
1 month ago
numberOfWeeks = "2",
outboundHaulage = arrService[0],
inboundHaulage = arrService[1],
};
1 month ago
1 month ago
// shipInfo
var extData = JObject.Parse(custOrder.ExtendData);
postModel.shipInfo = extData.GetJObjectValue("shipInfo");
1 month ago
1 month ago
//合约信息
postModel.contractInfo = new ONESoApiContractInfo()
{
contractNO = custOrder.CONTRACTNO,
1 month ago
namedAccount = custOrder.NamedAccount
1 month ago
};
1 month ago
decimal ctnAllKgs = 0;
// boxInfos
var ctns = await repCtn.AsQueryable().Filter(null, true).Where(x => x.BILLID == custOrder.Id).ToListAsync();
postModel.boxInfos = new List<ONESoApiBoxInfo>();
foreach (var ctn in ctns)
{
if (!ctn.KGS.HasValue)
{
return new KeyValuePair<bool, string>(false, $"所有箱的毛重都不能为空");
}
ctnAllKgs += (decimal)ctn.KGS;
var mapCtn = mappingCtn.FirstOrDefault(x => x.Module == "DjyCustBooking" && x.CarrierCode == CarrierCodeConst.ONE && x.Code == ctn.CTNCODE);
if (mapCtn == null)
{
return new KeyValuePair<bool, string>(false, $"未找箱型映射信息:{ctn.CTNCODE}");
}
var apiBox = new ONESoApiBoxInfo()
{
boxType = custOrder.CARGOID switch
{
"S" => "DRY",
"R" => "REEFER",
_ => throw Oops.Oh("暂不支持的货物类型")
},
boxSize = mapCtn.MapCode,
quantity = ctn.PKGS.Value,
1 month ago
};
postModel.boxInfos.Add(apiBox);
}
postModel.cargoInfo = new ONESoApiCargoInfo()
{
commodity = custOrder.DESCRIPTION,
weight = ctnAllKgs
};
postModel.contactInfo = new ONESoApiContactInfo()
{
shipperName = custOrder.ShipperName,
forwarderName = custOrder.BookingName,
consigneeName = custOrder.ConsigneeName,
1 month ago
};
postModel.remarkInfo = new ONESoApiRemarkInfo()
{
4 weeks ago
officeName = custOrder.BookingAddr,
specialInstruction = custOrder.SOREMARK
1 month ago
};
postModel.copyNum = custOrder.CopyNum;
// 邮箱
postModel.emailList = new List<string>();
1 month ago
if (!string.IsNullOrWhiteSpace(custOrder.OpMail))
{
var mailStr = custOrder.OpMail.Replace("", ",").Replace(";", ",").Replace("", ",");
4 weeks ago
var mailList = mailStr.Split(',').Where(x => !string.IsNullOrWhiteSpace(x)).Select(x => x.Trim()).ToList();
postModel.emailList.AddRange(mailList);
}
if (!string.IsNullOrEmpty(template.BcReceiveEmail))
{
var mailStr = template.BcReceiveEmail.Replace("", ",").Replace(";", ",").Replace("", ",");
var mailList = template.BcReceiveEmail.Split(',').Where(x => !string.IsNullOrWhiteSpace(x)).Select(x => x.Trim()).ToList();
postModel.emailList.AddRange(mailList);
}
postModel.emailList = postModel.emailList.Distinct().ToList();
//else if (!string.IsNullOrEmpty(custContact.Email))
//{
// postModel.emailList = new List<string>() { custContact.Email };
//}
//else if (!string.IsNullOrEmpty(template.BcReceiveEmail))
//{
// postModel.emailList = new List<string>() { template.BcReceiveEmail };
//}
var apiUrl = sCfgSpiderUrl.Value;
1 month ago
if (!apiUrl.EndsWith("/"))
{
apiUrl += "/";
}
apiUrl += "v1/one/booking/auto";
var jsonStr = JsonConvert.SerializeObject(postModel, new JsonSerializerSettings() { NullValueHandling = NullValueHandling.Ignore });
Log.Information($"发送API数据给爬虫custOrder.BookingId:{custOrder.BookingId}custOrder.BOOKINGNO:{custOrder.BOOKINGNO}custOrder.BookingId{custOrder.BookingId}{apiUrl}{jsonStr}");
var rtn = await apiUrl.SetBody(jsonStr, "application/json")
.PostAsStringAsync();
Log.Information($"爬虫返回:{rtn}");
var jobjRtn = JObject.Parse(rtn);
if (jobjRtn.GetIntValue("code") == 200)
{
var jdata = jobjRtn.GetJObjectValue("data");
if (jdata != null)
{
JObject extObj = null;
if (!string.IsNullOrEmpty(custOrder.ExtendData))
{
extObj = JObject.Parse(custOrder.ExtendData);
}
else
{
extObj = new JObject();
}
var bookingNoArr = jdata.GetJArrayValue("bookingNoList");
extObj["CustNO"] = bookingNoArr;
custOrder.ExtendData = extObj.ToJsonString();
await repCustOrder.AsUpdateable(custOrder).UpdateColumns(x => new { x.ExtendData }).ExecuteCommandAsync();
Log.Information($"回写CC号 {custOrder.Id} {custOrder.BOOKINGNO}");
}
return new KeyValuePair<bool, string>(true, "发送成功");
}
else
{
return new KeyValuePair<bool, string>(false, jobjRtn.GetStringValue("msg"));
}
1 month ago
}
1 month ago
}
/// <summary>
/// ONE订舱传输对象
/// </summary>
public class ONESoApiModel : BookingHelperBaseModel
{
/// <summary>
/// 箱信息数组,箱型箱量等信息
/// </summary>
1 month ago
public List<ONESoApiBoxInfo> boxInfos { get; set; }
1 month ago
/// <summary>
/// 货物信息数据,品名, 重量等信息
/// </summary>
1 month ago
public ONESoApiCargoInfo cargoInfo { get; set; }
1 month ago
/// <summary>
/// 收发通等联系信息,按照网站规则, 只提供对应的名字即可, 但是需要与网站上一致
/// </summary>
1 month ago
public ONESoApiContactInfo contactInfo { get; set; }
1 month ago
/// <summary>
/// 约号相关信息,约号必填, namedAccount可不传, 但是必须与实际对应
/// </summary>
1 month ago
public ONESoApiContractInfo contractInfo { get; set; }
1 month ago
/// <summary>
/// 备注信息,数据字典, 部分参数目前不涉及, 只传备注文本即可
/// </summary>
1 month ago
public ONESoApiRemarkInfo remarkInfo { get; set; }
1 month ago
/// <summary>
/// 路线信息,港口,船期等信息
/// </summary>
1 month ago
public ONESoApiRoute routes { get; set; }
1 month ago
/// <summary>
/// 模板名字,保存为模板时这个字段必填
/// </summary>
public string saveName { get; set; }
/// <summary>
/// 船期数据,船期查询的结果中选择的数据集, 原样上传
/// </summary>
public object shipInfo { get; set; }
1 month ago
/// <summary>
/// 复制的单数
/// </summary>
public int copyNum { get; set; }
/// <summary>
/// 邮件联系人列表
/// </summary>
public List<string> emailList { get; set; }
1 month ago
}
/// <summary>
/// ONEBOxInfo箱相关数据
/// </summary>
1 month ago
public class ONESoApiBoxInfo
1 month ago
{
/// <summary>
/// 箱型,每个箱类型对应自己可选的箱型, 需要做映射
/// </summary>
public string boxSize { get; set; }
/// <summary>
/// 箱类型,每个箱类型对应自己可选的箱型, 需要做映射
/// </summary>
public string boxType { get; set; }
/// <summary>
/// 箱量,当前箱型的总箱量
/// </summary>
1 month ago
public decimal quantity { get; set; }
1 month ago
/// <summary>
/// SOC箱量不能大于总箱量
/// </summary>
public long? socQuantity { get; set; }
}
/// <summary>
/// 货物信息数据,品名, 重量等信息
///
/// ONECargoInfo货物相关信息
/// </summary>
1 month ago
public class ONESoApiCargoInfo
1 month ago
{
/// <summary>
/// 品名,需和网站上一致
/// </summary>
public string commodity { get; set; }
/// <summary>
/// 返回的日期,精确到天, 可以不传
/// </summary>
public string returnDate { get; set; }
/// <summary>
/// 重量,重量
/// </summary>
1 month ago
public decimal weight { get; set; }
1 month ago
/// <summary>
/// 重量单位,重量单位
/// </summary>
1 month ago
public string weightUnit { get; set; }
1 month ago
}
/// <summary>
/// 收发通等联系信息,按照网站规则, 只提供对应的名字即可, 但是需要与网站上一致
/// ONEContactInfo联系人相关数据
/// </summary>
1 month ago
public class ONESoApiContactInfo
1 month ago
{
/// <summary>
/// 收货人,可以不传
/// </summary>
public string consigneeName { get; set; }
/// <summary>
/// 货代名,和网站上保值一致, 且已经在账户的通讯录中
/// </summary>
public string forwarderName { get; set; }
/// <summary>
/// 发货人名字,和网站上保值一致, 且已经在账户的通讯录中
/// </summary>
public string shipperName { get; set; }
}
/// <summary>
/// 约号相关信息,约号必填, namedAccount可不传, 但是必须与实际对应
/// ONEContractInfo客户约号信息
/// </summary>
1 month ago
public class ONESoApiContractInfo
1 month ago
{
/// <summary>
/// 客户约号,必填,且应与网站数据一致
/// </summary>
public string contractNO { get; set; }
1 month ago
/// <summary>
/// 约号对应的账户名,如果传则必须与网站一直, 不传则默认选择 Unable to find Named Account or Not Applicable
/// </summary>
public string namedAccount { get; set; }
}
/// <summary>
/// 备注信息,数据字典, 部分参数目前不涉及, 只传备注文本即可
/// ONERemarkInfoONE订舱时备注信息
/// </summary>
1 month ago
public class ONESoApiRemarkInfo
1 month ago
{
/// <summary>
/// Booking Freight Forwarder Ref. No.
/// </summary>
public string bkgFFRefNo { get; set; }
/// <summary>
/// Booking Shipper Ref. No.
/// </summary>
public string bkgShRefNo { get; set; }
/// <summary>
/// Invoice Ref. No.
/// </summary>
public string invoiceRefNo { get; set; }
/// <summary>
/// Manual Booking Number
/// </summary>
public string manualBookingNo { get; set; }
/// <summary>
/// 订舱网点名
/// </summary>
public string officeName { get; set; }
/// <summary>
/// S/I Freight Forwarder No.
/// </summary>
public string siFFNo { get; set; }
/// <summary>
/// S/I Shipper Ref. No.
/// </summary>
public string siSHRefNo { get; set; }
/// <summary>
/// 备注信息,备注信息, 默认为空
/// </summary>
public string specialInstruction { get; set; }
}
/// <summary>
/// 路线信息,港口,船期等信息
/// OneAutoRoutes路线信息
/// </summary>
1 month ago
public class ONESoApiRoute
1 month ago
{
/// <summary>
/// 交货地,交货地, 与网站上应一致, 注意不要CY及DOOR信息, 运输方式有相关参数
/// </summary>
public string destinationName { get; set; }
/// <summary>
/// 运送形态
/// </summary>
1 month ago
public string inboundHaulage { get; set; }
1 month ago
/// <summary>
/// 时间范围,按周计算, 取值范围2, 4, 6, 8
/// </summary>
1 month ago
public string numberOfWeeks { get; set; }
1 month ago
/// <summary>
/// 收货地,收货地, 与网站上应一致, 注意不要CY及DOOR信息, 运输方式有相关参数
/// </summary>
public string originName { get; set; }
/// <summary>
/// 运送方式
/// </summary>
1 month ago
public string outboundHaulage { get; set; }
1 month ago
/// <summary>
/// 查询日期,默认为两天后
/// </summary>
public string searchConditionDate { get; set; }
}
}