using Furion; using Furion.Logging; using Furion.RemoteRequest.Extensions; using Myshipping.Application.EDI.Dtos; using Myshipping.Application.Entity; using Myshipping.Core; using Myshipping.Core.Entity; using Myshipping.Core.Service; using Newtonsoft.Json.Linq; using System.Collections.Generic; using System.Linq; using System.Text.RegularExpressions; using System.Threading.Tasks; namespace Myshipping.Application.EDI { /// /// 中远API订舱 /// public static class ZhongYuanSoApiHelper { public async static Task> DoPost(long custOrderId) { var repCustOrder = App.GetService>(); //var repOrder = App.GetService>(); var repCtn = App.GetService>(); var repCustomer = App.GetService>(); var repContact = App.GetService>(); var repTemplate = App.GetService>(); var cache = App.GetService(); var cacheService = App.GetService(); var custOrder = await repCustOrder.AsQueryable().Filter(null, true).FirstAsync(x => x.Id == custOrderId); if (custOrder == null) { return new KeyValuePair(false, "客户订舱信息未找到"); } var sysConfigList = await cache.GetAllSysConfig(); var sCfgSpiderUrl = sysConfigList.FirstOrDefault(x => x.Code == "ZhongYuanApiSpiderUrl" && x.GroupCode == "DJY_CONST"); if (sCfgSpiderUrl == null) { return new KeyValuePair(false, "中远订舱API的URL地址未配置,请联系管理员"); } var sCfgUserKey = sysConfigList.FirstOrDefault(x => x.Code == "ZhongYuanApiSpiderKey" && x.GroupCode == "DJY_CONST"); var sCfgUserSecret = sysConfigList.FirstOrDefault(x => x.Code == "ZhongYuanApiSpiderSecret" && x.GroupCode == "DJY_CONST"); if (sCfgUserKey == null || sCfgUserSecret == null) { return new KeyValuePair(false, "中远订舱API的KEY和密钥未配置,请联系管理员"); } BookingSoTemplate template = null; DjyCustomerContact custContact = null; var postModel = new ZhongYuanSoApiModel { webAccount = custOrder.BookingAccount, webPassword = custOrder.BookingPassword }; //查找模板: //1.根据客户订舱信息中的BookingUserId和BookingTenantId,去客户信息中根据CustSysId查找客户(公司)及联系人(员工)信息 //2.根据找到的客户及联系人信息,查找中远订舱模板 if (custOrder.BookingUserId > 0 && custOrder.BookingTenantId > 0) { custContact = await repCustomer.AsQueryable().Filter(null, true) .InnerJoin((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(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(false, "未找到订舱模板"); } } else { return new KeyValuePair(false, "未找到客户端公司和用户ID"); } var mappingCtn = await cache.GetAllMappingCtn(); var mappingFrt = await cache.GetAllMappingFrt(); var mappingPortLoad = await cache.GetAllMappingPortLoad(); var mappingPort = await cache.GetAllMappingPort(); postModel.userKey = sCfgUserKey.Value; postModel.userSecret = sCfgUserSecret.Value; postModel.uploadType = template.Category; //DRAFT, TEMPLATE, BOOKING分别对应:草稿, 模板, 订舱 postModel.saveName = template.TemplateName; //起运港 var mapPortLoad = mappingPortLoad.FirstOrDefault(x => x.Module == "DjyCustBooking" && x.CarrierCode == CarrierCodeConst.COSCO && x.Code == custOrder.PORTLOADCODE); if (mapPortLoad == null) { return new KeyValuePair(false, $"未找到起运港映射信息:{custOrder.PORTLOADCODE}"); } //目的地,2024-5-9,衣国豪需求,由目的港改为目的地 var mapDestination = mappingPort.FirstOrDefault(x => x.Module == "DjyCustBooking" && x.CarrierCode == CarrierCodeConst.COSCO && x.Code == custOrder.DESTINATIONCODE); if (mapDestination == null) { return new KeyValuePair(false, $"未找到目的地映射信息:{custOrder.DESTINATIONCODE}"); } //运输条款 var mappingService = await cacheService.GetAllMappingService(); var mappService = mappingService.FirstOrDefault(x => x.Module == "DjyCustBooking" && x.CarrierCode == CarrierCodeConst.COSCO && x.Code == custOrder.SERVICE); if (mappService == null) { return new KeyValuePair(false, $"未找到运输条款映射信息:{custOrder.SERVICE}"); } if (!Regex.IsMatch(mappService.MapCode, "^[A-Za-z]+-[A-Za-z]+$")) { return new KeyValuePair(false, $"映射配置不正确:{mappService.MapCode}"); } string[] arrService = null; if (mappService == null) { arrService = custOrder.SERVICE.Split('-'); } else { arrService = mappService.MapCode.Split('-'); } //var startDay = template.StartDay.HasValue ? template.StartDay.Value : 1; //var endWeek = template.EndWeek.HasValue ? template.EndWeek.Value : 2; // 路线数据 //postModel.routes = new ZhongYuanSoApiRoute() //{ // originCity = mapPortLoad.MapCode, // destinationCity = mapPort.MapCode, // polPortName = custOrder.PolPortName, // podPortName = custOrder.PodPortName, // vesselName = custOrder.VESSEL, // voyageNumber = custOrder.VOYNO, // serviceCode = custOrder.LANECODE, // //sailSchedulePriority = template.Priority.Split(',').ToList(), // outboundHaulage = mapServArr[0], // inboundHaulage = mapServArr[1], // searchConditionDate = custOrder.ETD.Value.AddDays(-2).ToString("yyyy-MM-dd"), // numberOfWeeks = 2, // etd = custOrder.ETD.Value.ToString("yyyy-MM-dd"), //}; #region 收发通及货代 //发货人 ZhongYuanSoApiPhone shipperPhone = null; //if (string.IsNullOrEmpty(template.ShipperName)) //{ shipperPhone = new ZhongYuanSoApiPhone() { countryCode = custOrder.ShipperPhoneCountryCode, areaCode = custOrder.ShipperPhoneCode, number = custOrder.ShipperPhone }; //} //else //{ // shipperPhone = new ZhongYuanSoApiPhone() // { // countryCode = template.ShipperPhoneCountryCode, // areaCode = template.ShipperPhoneCode, // number = template.ShipperPhone // }; //} shipperPhone.countryCode = shipperPhone.countryCode == null ? "" : shipperPhone.countryCode; shipperPhone.areaCode = shipperPhone.areaCode == null ? "" : shipperPhone.areaCode; shipperPhone.number = shipperPhone.number == null ? "" : shipperPhone.number; postModel.shipperInfo = new ZhongYuanSoApiSFT() { firstName = custOrder.ShipperFirstName, lastName = custOrder.ShipperLastName, country = custOrder.ShipperCountry, state = custOrder.ShipperProvince, city = custOrder.ShipperCity, partyName = custOrder.ShipperName, addressDes = custOrder.ShipperAddress, phone = shipperPhone, postalCode = custOrder.ShipperPostCode }; //收货人 ZhongYuanSoApiPhone consigneePhone = null; //if (string.IsNullOrEmpty(template.ConsigneeName)) //{ consigneePhone = new ZhongYuanSoApiPhone() { countryCode = custOrder.ConsigneePhoneCountryCode, areaCode = custOrder.ConsigneePhoneCode, number = custOrder.ConsigneePhone }; //} //else //{ // consigneePhone = new ZhongYuanSoApiPhone() // { // countryCode = template.ConsigneePhoneCountryCode, // areaCode = template.ConsigneePhoneCode, // number = template.ConsigneePhone // }; //} consigneePhone.countryCode = consigneePhone.countryCode == null ? "" : consigneePhone.countryCode; consigneePhone.areaCode = consigneePhone.areaCode == null ? "" : consigneePhone.areaCode; consigneePhone.number = consigneePhone.number == null ? "" : consigneePhone.number; postModel.consigneeInfo = new ZhongYuanSoApiSFT() { firstName = custOrder.ConsigneeFirstName, lastName = custOrder.ConsigneeLastName, country = custOrder.ConsigneeCountry, state = custOrder.ConsigneeProvince, city = custOrder.ConsigneeCity, partyName = custOrder.ConsigneeName, addressDes = custOrder.ConsigneeAddress, phone = consigneePhone, postalCode = custOrder.ConsigneePostCode }; //通知人 ZhongYuanSoApiPhone notifyPhone = null; //if (string.IsNullOrEmpty(template.NotifypartName)) //{ notifyPhone = new ZhongYuanSoApiPhone() { countryCode = custOrder.NotifypartPhoneCountryCode, areaCode = custOrder.NotifypartPhoneCode, number = custOrder.NotifypartPhone }; //} //else //{ // notifyPhone = new ZhongYuanSoApiPhone() // { // countryCode = template.NotifypartPhoneCountryCode, // areaCode = template.NotifypartPhoneCode, // number = template.NotifypartPhone // }; //} notifyPhone.countryCode = notifyPhone.countryCode == null ? "" : notifyPhone.countryCode; notifyPhone.areaCode = notifyPhone.areaCode == null ? "" : notifyPhone.areaCode; notifyPhone.number = notifyPhone.number == null ? "" : notifyPhone.number; postModel.notifyInfo = new ZhongYuanSoApiSFT() { firstName = custOrder.NotifypartFirstName, lastName = custOrder.NotifypartLastName, country = custOrder.NotifypartCountry, state = custOrder.NotifypartProvince, city = custOrder.NotifypartCity, partyName = custOrder.NotifypartName, addressDes = custOrder.NotifypartAddress, phone = notifyPhone, postalCode = custOrder.NotifypartPostCode }; //货代 ZhongYuanSoApiPhone forwarderPhone = null; //if (string.IsNullOrEmpty(template.BookingName)) //{ forwarderPhone = new ZhongYuanSoApiPhone() { countryCode = custOrder.BookingPhoneCountryCode, areaCode = custOrder.BookingPhoneCode, number = custOrder.BookingPhone }; //} //else //{ // forwarderPhone = new ZhongYuanSoApiPhone() // { // countryCode = template.BookingPhoneCountryCode, // areaCode = template.BookingPhoneCode, // number = template.BookingPhone // }; //} forwarderPhone.countryCode = forwarderPhone.countryCode == null ? "" : forwarderPhone.countryCode; forwarderPhone.areaCode = forwarderPhone.areaCode == null ? "" : forwarderPhone.areaCode; forwarderPhone.number = forwarderPhone.number == null ? "" : forwarderPhone.number; postModel.forwarderInfo = new ZhongYuanSoApiSFT() { firstName = custOrder.BookingFirstName, lastName = custOrder.BookingLastName, country = custOrder.BookingCountry, state = custOrder.BookingProvince, city = custOrder.BookingCity, partyName = custOrder.BookingName, addressDes = custOrder.BookingAddress, phone = forwarderPhone, postalCode = custOrder.BookingPostCode, }; #endregion postModel.cargoInfo = new ZhongYuanSoApiCargoInfo() { cargoName = CargoIdZhongyuan(custOrder.CARGOID), cargoDes = custOrder.DESCRIPTION, hsCode = custOrder.HSCODE, }; var ctns = await repCtn.AsQueryable().Filter(null, true).Where(x => x.BILLID == custOrder.Id).ToListAsync(); postModel.boxInfos = new List(); // 箱型转换:用于订舱参数“路线信息”里的箱型数据 var mapCtnCodeList = new List(); foreach (var ctn in ctns) { if (!ctn.CTNNUM.HasValue || !ctn.KGS.HasValue) { return new KeyValuePair(false, $"所有箱的箱量和毛重都不能为空"); } var mapCtn = mappingCtn.FirstOrDefault(x => x.Module == "DjyCustBooking" && x.CarrierCode == CarrierCodeConst.COSCO && x.Code == ctn.CTNCODE); if (mapCtn == null) { return new KeyValuePair(false, $"未找箱型映射信息:{ctn.CTNCODE}"); } mapCtnCodeList.Add(mapCtn.MapName); var apiBox = new ZhongYuanSoApiBoxInfo() { boxType = mapCtn.MapCode, boxNum = ctn.CTNNUM.Value, weight = ctn.KGS.Value.ToString(), weightUnit = "KGS" }; postModel.boxInfos.Add(apiBox); } var mapFrt = mappingFrt.FirstOrDefault(x => x.Module == "DjyCustBooking" && x.CarrierCode == CarrierCodeConst.COSCO && x.Code == custOrder.FRTCODE); if (mapFrt == null) { return new KeyValuePair(false, $"未找到付款方式映射信息:{custOrder.FRTCODE}"); } postModel.truckingContacts = new ZhongYuanSoApiTruckingContacts() { isDoorAdvised = true }; postModel.rateInfos = new ZhongYuanSoApiRateInfo() { paymentMethod = mapFrt.MapCode, serviceContractNum = custOrder.CONTRACTNO }; //大简云客户订舱接收BC邮箱 var djyBookMail = sysConfigList.FirstOrDefault(x => x.Code == "DjyCustomerBookReceiveBcMail"); var bcMail = ""; if (!string.IsNullOrEmpty(custOrder.OpMail)) //优先使用东胜上传的邮箱 { bcMail = custOrder.OpMail.Split(',')[0]; } else if (!string.IsNullOrEmpty(custContact.Email)) { bcMail = custContact.Email; } //else //{ // bcMail = template.BcReceiveEmail; //} if (djyBookMail != null) { bcMail += ";" + djyBookMail.Value; } postModel.special = new ZhongYuanSoApiSpecial() { emailAddresses = bcMail, remarksForEntireBooking = custOrder.SOREMARK }; // 船期数据 if (string.IsNullOrEmpty(custOrder.ExtendData)) { return new KeyValuePair(false, $"船期数据信息不存在,请选择船期"); } var extData = JObject.Parse(custOrder.ExtendData); postModel.shipInfo = extData.GetJObjectValue("shipInfo"); // 路线数据(中远-新格式) postModel.routes = new { numberOfWeeks = 2, searchConditionDate = custOrder.ETD.Value.AddDays(-2).ToString("yyyy-MM-dd"), originName = mapPortLoad.MapName, destinationName = mapDestination.MapName, outboundHaulage = arrService[0], inboundHaulage = arrService[1], contractNo = custOrder.CONTRACTNO, cargoNature = custOrder.CARGOID switch { "S" => "General", "R" => "Reefer", "D" => "Dangerous", _ => "" }, containerTypeList = mapCtnCodeList }; var postModelJson = postModel.ToJsonString(); Log.Information($"发送API数据给爬虫,custOrder.BookingId:{custOrder.BookingId},custOrder.BOOKINGNO:{custOrder.BOOKINGNO},custOrder.BookingId:{custOrder.BookingId}({sCfgSpiderUrl.Value}):{postModelJson}"); var rtn = await sCfgSpiderUrl.Value.SetBody(postModelJson).PostAsStringAsync(); //var rtn = await "http://www.baidu323.com".SetBody(postModel).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 bookingNo = jdata.GetStringValue("bookingNo"); if (!string.IsNullOrEmpty(bookingNo)) { extObj["CustNO"] = bookingNo; custOrder.ExtendData = extObj.ToJsonString(); await repCustOrder.AsUpdateable(custOrder).UpdateColumns(x => new { x.ExtendData }).ExecuteCommandAsync(); Log.Information($"回写CC号:{bookingNo} {custOrder.Id} {custOrder.BOOKINGNO}"); } } return new KeyValuePair(true, "发送成功"); } else { return new KeyValuePair(false, jobjRtn.GetStringValue("msg")); } } /// /// 获取中远货物类型 /// /// /// private static string CargoIdZhongyuan(string cargoId) { if (cargoId is "S") { return "General"; } else if (cargoId is "R") { return "Reefer"; } else if (cargoId is "D") { return "Dangerous"; } return null; } } /// /// 中远API订舱传输对象 /// public class ZhongYuanSoApiModel: BookingHelperBaseModel { /// /// 在选择草稿及模板时, 保存时填写的名称 /// public string saveName { get; set; } /// /// 在选择草稿及模板时的描述 /// public string saveDes { get; set; } /// /// 路线信息 /// public object routes { get; set; } //public ZhongYuanSoApiRoute routes { get; set; } /// /// 发货人信息 /// public ZhongYuanSoApiSFT shipperInfo { get; set; } /// /// 收货人信息 /// public ZhongYuanSoApiSFT consigneeInfo { get; set; } /// /// 通知人信息 /// public ZhongYuanSoApiSFT notifyInfo { get; set; } /// /// 货代信息 /// public ZhongYuanSoApiSFT forwarderInfo { get; set; } /// /// 货物信息 /// public ZhongYuanSoApiCargoInfo cargoInfo { get; set; } /// /// 箱信息 /// public List boxInfos { get; set; } /// /// 拖运方式:到门卸货预约 /// public ZhongYuanSoApiTruckingContacts truckingContacts { get; set; } /// /// 费率相关信息 /// public ZhongYuanSoApiRateInfo rateInfos { get; set; } /// /// 特殊要求及备注 /// public ZhongYuanSoApiSpecial special { get; set; } /// /// 船期数据 /// public object shipInfo { get; set; } /// /// 关联订舱信息 /// public object mark {get;set;} } /// /// 路线信息 /// public class ZhongYuanSoApiRoute { /// /// 出发城市 /// public string originCity { get; set; } /// /// 目的城市 /// public string destinationCity { get; set; } /// /// 起始港名 /// public string polPortName { get; set; } /// /// 目的港名 /// public string podPortName { get; set; } /// /// 船名 /// public string vesselName { get; set; } /// /// 航次 /// public string voyageNumber { get; set; } /// /// 航线代码 /// public string serviceCode { get; set; } ///// ///// 船期选择优先级 ///// //public List sailSchedulePriority { get; set; } /// /// 出发地运输条款 /// public string outboundHaulage { get; set; } /// /// 目的地运输条款 /// public string inboundHaulage { get; set; } /// /// 起运时间 /// public string searchConditionDate { get; set; } /// /// 时间范围,按周计算, 默认为2 /// public int numberOfWeeks { get; set; } = 2; /// /// etd /// public string etd { get; set; } } /// /// 收发通信息 /// public class ZhongYuanSoApiSFT { /// /// 名字 /// public string partyName { get; set; } /// /// 国家 /// public string country { get; set; } /// /// 省 /// public string state { get; set; } /// /// 城市名 /// public string city { get; set; } /// /// 区 /// public string county { get; set; } /// /// 详细地址 /// public string addressDes { get; set; } /// /// 邮编 /// public string postalCode { get; set; } /// /// 姓 /// public string firstName { get; set; } /// /// 名 /// public string lastName { get; set; } /// /// 电话 /// public ZhongYuanSoApiPhone phone { get; set; } } /// /// 收发通电话 /// public class ZhongYuanSoApiPhone { /// /// 国家代码 /// public string countryCode { get; set; } /// /// 区号 /// public string areaCode { get; set; } /// /// 电话号码 /// public string number { get; set; } } /// /// 货物信息 /// public class ZhongYuanSoApiCargoInfo { /// /// 货物类型 /// public string cargoName { get; set; } /// /// 货物描述 /// public string cargoDes { get; set; } /// /// HS代码 /// public string hsCode { get; set; } } /// /// 箱信息 /// public class ZhongYuanSoApiBoxInfo { /// /// 箱量 /// public int boxNum { get; set; } /// /// 箱型 /// public string boxType { get; set; } /// /// 毛重 /// public string weight { get; set; } /// /// 重量单位 /// public string weightUnit { get; set; } } /// /// 费率 /// public class ZhongYuanSoApiRateInfo { /// /// 付款方式 /// public string paymentMethod { get; set; } /// /// 服务合同号 /// public string serviceContractNum { get; set; } } /// /// 托运方式 /// public class ZhongYuanSoApiTruckingContacts { /// /// 是否进一步通知 /// public bool isDoorAdvised { get; set; } } /// /// 特殊要求及备注 /// public class ZhongYuanSoApiSpecial { /// /// 备注信息 /// public string remarksForEntireBooking { get; set; } /// /// 邮箱地址 /// public string emailAddresses { get; set; } } }