|
|
using Furion;
|
|
|
using Myshipping.Application.Entity;
|
|
|
using Myshipping.Core;
|
|
|
using Myshipping.Core.Entity;
|
|
|
using Myshipping.Core.Service;
|
|
|
using Newtonsoft.Json.Linq;
|
|
|
using System;
|
|
|
using System.Collections.Generic;
|
|
|
using System.Linq;
|
|
|
using System.Text;
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
namespace Myshipping.Application.EDI
|
|
|
{
|
|
|
/// <summary>
|
|
|
/// 中远API订舱
|
|
|
/// </summary>
|
|
|
public static class ZhongYuanSoApiHelper
|
|
|
{
|
|
|
public async static Task<KeyValuePair<bool, string>> DoPost(long bookingId)
|
|
|
{
|
|
|
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 order = await repOrder.AsQueryable().Filter(null, true).FirstAsync(o => o.Id == bookingId);
|
|
|
if (order == null)
|
|
|
{
|
|
|
return new KeyValuePair<bool, string>(false, "订舱信息未找到");
|
|
|
}
|
|
|
|
|
|
var custOrder = await repCustOrder.AsQueryable().Filter(null, true).FirstAsync(x => x.BookingId == order.Id);
|
|
|
if (custOrder == null)
|
|
|
{
|
|
|
return new KeyValuePair<bool, string>(false, "客户订舱信息未找到");
|
|
|
}
|
|
|
|
|
|
var sysConfigList = await cache.GetAllSysConfig();
|
|
|
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<bool, string>(false, "中远订舱API的KEY和密钥为配置,请联系管理员");
|
|
|
}
|
|
|
|
|
|
|
|
|
BookingSoTemplate template = null;
|
|
|
DjyCustomerContact custContact = null;
|
|
|
//查找模板:
|
|
|
//1.根据客户订舱信息中的BookingUserId和BookingTenantId,去客户信息中根据CustSysId查找客户(公司)及联系人(员工)信息
|
|
|
//2.根据找到的客户及联系人信息,查找中远订舱模板
|
|
|
if (custOrder.BookingUserId > 0 && custOrder.BookingTenantId > 0)
|
|
|
{
|
|
|
custContact = await repCustomer.AsQueryable()
|
|
|
.InnerJoin<DjyCustomerContact>((cust, contact) => cust.Id == contact.CustomerId)
|
|
|
.Where((cust, contact) => cust.CustSysId == custOrder.TenantId && contact.CustSysId == custOrder.BookingUserId)
|
|
|
.Select((cust, contact) => contact)
|
|
|
.SingleAsync();
|
|
|
if (custContact == null)
|
|
|
{
|
|
|
return new KeyValuePair<bool, string>(false, "未找到客户及联系人信息");
|
|
|
}
|
|
|
|
|
|
template = await repTemplate.AsQueryable().FirstAsync(x => x.CarrierId == order.CARRIERID && x.UserId == custContact.Id && x.IsEnable);
|
|
|
if (template == null)
|
|
|
{
|
|
|
return new KeyValuePair<bool, string>(false, "未找到订舱模板");
|
|
|
}
|
|
|
}
|
|
|
|
|
|
var mappingCtn = await cache.GetAllMappingCtn();
|
|
|
var mappingFrt = await cache.GetAllMappingFrt();
|
|
|
var mappingPortLoad = await cache.GetAllMappingPortLoad();
|
|
|
var mappingPort = await cache.GetAllMappingPort();
|
|
|
|
|
|
var postModel = new ZhongYuanSoApiModel();
|
|
|
postModel.userKey = sCfgUserKey.Value;
|
|
|
postModel.userSecret = sCfgUserSecret.Value;
|
|
|
|
|
|
if (!string.IsNullOrEmpty(custOrder.ExtendData))
|
|
|
{
|
|
|
var extObj = JObject.Parse(custOrder.ExtendData);
|
|
|
postModel.webAccount = extObj.GetStringValue("Account");
|
|
|
postModel.webPassword = extObj.GetStringValue("Password");
|
|
|
}
|
|
|
else
|
|
|
{
|
|
|
return new KeyValuePair<bool, string>(false, "未找到订舱账号信息");
|
|
|
}
|
|
|
|
|
|
postModel.uploadType = "DRAFT"; //DRAFT, TEMPLATE, BOOKING分别对应:草稿, 模板, 订舱
|
|
|
postModel.saveName = template.TemplateName;
|
|
|
|
|
|
var mapPortLoad = mappingPortLoad.FirstOrDefault(x => x.Module == "DjyCustBooking" && x.CarrierCode == "COSCO" && x.Code == custOrder.PORTLOADCODE);
|
|
|
var mapPort = mappingPort.FirstOrDefault(x => x.Module == "DjyCustBooking" && x.CarrierCode == "COSCO" && x.Code == custOrder.PORTDISCHARGECODE);
|
|
|
if (mapPortLoad == null)
|
|
|
{
|
|
|
return new KeyValuePair<bool, string>(false, $"未找到起运港映射信息:{custOrder.PORTLOADCODE}");
|
|
|
}
|
|
|
|
|
|
if (mapPort == null)
|
|
|
{
|
|
|
return new KeyValuePair<bool, string>(false, $"未找到目的港映射信息:{custOrder.PORTDISCHARGECODE}");
|
|
|
}
|
|
|
|
|
|
postModel.routes = new ZhongYuanSoApiRoute()
|
|
|
{
|
|
|
originCity = mapPortLoad.MapCode,
|
|
|
destinationCity = mapPort.MapCode,
|
|
|
vesselName = custOrder.VESSEL,
|
|
|
voyageNumber = custOrder.VOYNO,
|
|
|
serviceCode = custOrder.LANECODE,
|
|
|
sailSchedulePriority = template.Priority.Split(',').ToList(),
|
|
|
};
|
|
|
|
|
|
#region 收发通及货代
|
|
|
postModel.shipperInfo = new ZhongYuanSoApiSFT()
|
|
|
{
|
|
|
firstName = template.ShipperFirstName,
|
|
|
lastName = template.ShipperLastName,
|
|
|
country = template.ShipperCountry,
|
|
|
state = template.ShipperProvince,
|
|
|
city = template.ShipperCity,
|
|
|
partyName = template.ShipperName,
|
|
|
addressDes = template.ShipperAddress,
|
|
|
phone = new ZhongYuanSoApiPhone() { countryCode = template.ShipperPhoneCountryCode, areaCode = template.ShipperPhoneCode, number = template.ShipperPhone },
|
|
|
postalCode = template.ShipperPostCode
|
|
|
};
|
|
|
|
|
|
postModel.consigneeInfo = new ZhongYuanSoApiSFT()
|
|
|
{
|
|
|
firstName = template.ConsigneeFirstName,
|
|
|
lastName = template.ConsigneeLastName,
|
|
|
country = template.ConsigneeCountry,
|
|
|
state = template.ConsigneeProvince,
|
|
|
city = template.ConsigneeCity,
|
|
|
partyName = template.ConsigneeName,
|
|
|
addressDes = template.ConsigneeAddress,
|
|
|
phone = new ZhongYuanSoApiPhone() { countryCode = template.ConsigneePhoneCountryCode, areaCode = template.ConsigneePhoneCode, number = template.ConsigneePhone },
|
|
|
postalCode = template.ConsigneePostCode
|
|
|
};
|
|
|
|
|
|
postModel.notifyInfo = new ZhongYuanSoApiSFT()
|
|
|
{
|
|
|
firstName = template.NotifypartFirstName,
|
|
|
lastName = template.NotifypartLastName,
|
|
|
country = template.NotifypartCountry,
|
|
|
state = template.NotifypartProvince,
|
|
|
city = template.NotifypartCity,
|
|
|
partyName = template.NotifypartName,
|
|
|
addressDes = template.NotifypartAddress,
|
|
|
phone = new ZhongYuanSoApiPhone() { countryCode = template.NotifypartPhoneCountryCode, areaCode = template.NotifypartPhoneCode, number = template.NotifypartPhone },
|
|
|
postalCode = template.NotifypartPostCode
|
|
|
};
|
|
|
|
|
|
postModel.forwarderInfo = new ZhongYuanSoApiSFT()
|
|
|
{
|
|
|
firstName = template.BookingFirstName,
|
|
|
lastName = template.BookingLastName,
|
|
|
country = template.BookingCountry,
|
|
|
state = template.BookingProvince,
|
|
|
city = template.BookingCity,
|
|
|
partyName = template.BookingName,
|
|
|
addressDes = template.BookingAddress,
|
|
|
phone = new ZhongYuanSoApiPhone() { countryCode = template.BookingPhoneCountryCode, areaCode = template.BookingPhoneCode, number = template.BookingPhone },
|
|
|
postalCode = template.BookingPostCode
|
|
|
};
|
|
|
#endregion
|
|
|
|
|
|
postModel.cargoInfo = new ZhongYuanSoApiCargoInfo()
|
|
|
{
|
|
|
cargoName = CargoIdZhongyuan(custOrder.CARGOID),
|
|
|
cargoDes = custOrder.DESCRIPTION,
|
|
|
hsCode = custOrder.HSCODE,
|
|
|
};
|
|
|
|
|
|
var ctns = await repCtn.AsQueryable().Where(x => x.BILLID == custOrder.Id).ToListAsync();
|
|
|
postModel.boxInfos = new List<ZhongYuanSoApiBoxInfo>();
|
|
|
foreach (var ctn in ctns)
|
|
|
{
|
|
|
if (!ctn.CTNNUM.HasValue || !ctn.KGS.HasValue)
|
|
|
{
|
|
|
return new KeyValuePair<bool, string>(false, $"所有箱的箱量和毛重都不能为空");
|
|
|
}
|
|
|
|
|
|
var mapCtn = mappingCtn.FirstOrDefault(x => x.Module == "DjyCustBooking" && x.CarrierCode == "COSCO" && x.Code == ctn.CTNCODE);
|
|
|
if (mapCtn == null)
|
|
|
{
|
|
|
return new KeyValuePair<bool, string>(false, $"未找箱型映射信息:{ctn.CTNCODE}");
|
|
|
}
|
|
|
|
|
|
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 == "COSCO" && x.Code == custOrder.FRTCODE);
|
|
|
if (mapFrt == null)
|
|
|
{
|
|
|
return new KeyValuePair<bool, string>(false, $"未找到付款方式映射信息:{custOrder.FRTCODE}");
|
|
|
}
|
|
|
|
|
|
postModel.rateInfos = new ZhongYuanSoApiRateInfo()
|
|
|
{
|
|
|
paymentMethod = mapFrt.MapCode
|
|
|
};
|
|
|
|
|
|
postModel.special = new ZhongYuanSoApiSpecial()
|
|
|
{
|
|
|
emailAddresses = custContact.Email,
|
|
|
remarksForEntireBooking = custOrder.SOREMARK
|
|
|
};
|
|
|
|
|
|
return new KeyValuePair<bool, string>();
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
/// 获取中远货物类型
|
|
|
/// </summary>
|
|
|
/// <param name="cargoId"></param>
|
|
|
/// <returns></returns>
|
|
|
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;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
/// 中远API订舱传输对象
|
|
|
/// </summary>
|
|
|
public class ZhongYuanSoApiModel
|
|
|
{
|
|
|
/// <summary>
|
|
|
/// 用户key
|
|
|
/// </summary>
|
|
|
public string userKey { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 用户secret
|
|
|
/// </summary>
|
|
|
public string userSecret { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 网站账户
|
|
|
/// </summary>
|
|
|
public string webAccount { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 网站密码
|
|
|
/// </summary>
|
|
|
public string webPassword { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 上传类型
|
|
|
/// </summary>
|
|
|
public string uploadType { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 在选择草稿及模板时, 保存时填写的名称
|
|
|
/// </summary>
|
|
|
public string saveName { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 在选择草稿及模板时的描述
|
|
|
/// </summary>
|
|
|
public string saveDes { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 路线信息
|
|
|
/// </summary>
|
|
|
public ZhongYuanSoApiRoute routes { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 发货人信息
|
|
|
/// </summary>
|
|
|
public ZhongYuanSoApiSFT shipperInfo { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 收货人信息
|
|
|
/// </summary>
|
|
|
public ZhongYuanSoApiSFT consigneeInfo { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 通知人信息
|
|
|
/// </summary>
|
|
|
public ZhongYuanSoApiSFT notifyInfo { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 货代信息
|
|
|
/// </summary>
|
|
|
public ZhongYuanSoApiSFT forwarderInfo { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 货物信息
|
|
|
/// </summary>
|
|
|
public ZhongYuanSoApiCargoInfo cargoInfo { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 箱信息
|
|
|
/// </summary>
|
|
|
public List<ZhongYuanSoApiBoxInfo> boxInfos { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 费率相关信息
|
|
|
/// </summary>
|
|
|
public ZhongYuanSoApiRateInfo rateInfos { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 特殊要求及备注
|
|
|
/// </summary>
|
|
|
public ZhongYuanSoApiSpecial special { get; set; }
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
/// 路线信息
|
|
|
/// </summary>
|
|
|
public class ZhongYuanSoApiRoute
|
|
|
{
|
|
|
/// <summary>
|
|
|
/// 出发城市
|
|
|
/// </summary>
|
|
|
public string originCity { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 目的城市
|
|
|
/// </summary>
|
|
|
public string destinationCity { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 船名
|
|
|
/// </summary>
|
|
|
public string vesselName { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 航次
|
|
|
/// </summary>
|
|
|
public string voyageNumber { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 航线代码
|
|
|
/// </summary>
|
|
|
public string serviceCode { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 船期选择优先级
|
|
|
/// </summary>
|
|
|
public List<string> sailSchedulePriority { get; set; }
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
/// 收发通信息
|
|
|
/// </summary>
|
|
|
public class ZhongYuanSoApiSFT
|
|
|
{
|
|
|
/// <summary>
|
|
|
/// 名字
|
|
|
/// </summary>
|
|
|
public string partyName { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 国家
|
|
|
/// </summary>
|
|
|
public string country { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 省
|
|
|
/// </summary>
|
|
|
public string state { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 城市名
|
|
|
/// </summary>
|
|
|
public string city { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 区
|
|
|
/// </summary>
|
|
|
public string county { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 详细地址
|
|
|
/// </summary>
|
|
|
public string addressDes { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 邮编
|
|
|
/// </summary>
|
|
|
public string postalCode { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 姓
|
|
|
/// </summary>
|
|
|
public string firstName { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 名
|
|
|
/// </summary>
|
|
|
public string lastName { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 电话
|
|
|
/// </summary>
|
|
|
public ZhongYuanSoApiPhone phone { get; set; }
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
/// 收发通电话
|
|
|
/// </summary>
|
|
|
public class ZhongYuanSoApiPhone
|
|
|
{
|
|
|
/// <summary>
|
|
|
/// 国家代码
|
|
|
/// </summary>
|
|
|
public string countryCode { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 区号
|
|
|
/// </summary>
|
|
|
public string areaCode { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 电话号码
|
|
|
/// </summary>
|
|
|
public string number { get; set; }
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
/// 货物信息
|
|
|
/// </summary>
|
|
|
public class ZhongYuanSoApiCargoInfo
|
|
|
{
|
|
|
/// <summary>
|
|
|
/// 货物类型
|
|
|
/// </summary>
|
|
|
public string cargoName { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 货物描述
|
|
|
/// </summary>
|
|
|
public string cargoDes { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// HS代码
|
|
|
/// </summary>
|
|
|
public string hsCode { get; set; }
|
|
|
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
/// 箱信息
|
|
|
/// </summary>
|
|
|
public class ZhongYuanSoApiBoxInfo
|
|
|
{
|
|
|
/// <summary>
|
|
|
/// 箱量
|
|
|
/// </summary>
|
|
|
public int boxNum { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 箱型
|
|
|
/// </summary>
|
|
|
public string boxType { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 毛重
|
|
|
/// </summary>
|
|
|
public string weight { get; set; }
|
|
|
|
|
|
/// <summary>
|
|
|
/// 重量单位
|
|
|
/// </summary>
|
|
|
public string weightUnit { get; set; }
|
|
|
|
|
|
}
|
|
|
|
|
|
public class ZhongYuanSoApiRateInfo
|
|
|
{
|
|
|
/// <summary>
|
|
|
/// 付款方式
|
|
|
/// </summary>
|
|
|
public string paymentMethod { get; set; }
|
|
|
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
/// 特殊要求及备注
|
|
|
/// </summary>
|
|
|
public class ZhongYuanSoApiSpecial
|
|
|
{
|
|
|
/// <summary>
|
|
|
/// 备注信息
|
|
|
/// </summary>
|
|
|
public string remarksForEntireBooking { get; set; }
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
/// 邮箱地址
|
|
|
/// </summary>
|
|
|
public string emailAddresses { get; set; }
|
|
|
}
|
|
|
}
|