327 lines
12 KiB
C#
327 lines
12 KiB
C#
|
|
using Microsoft.Extensions.Configuration;
|
|||
|
|
using Prism.Events;
|
|||
|
|
using System.IO;
|
|||
|
|
using System.Net.Http;
|
|||
|
|
using System.Text.Json;
|
|||
|
|
using System.Text.Json.Serialization;
|
|||
|
|
using System.Web;
|
|||
|
|
using YY.Admin.Core;
|
|||
|
|
using YY.Admin.Core.Entity;
|
|||
|
|
using YY.Admin.Core.Events;
|
|||
|
|
using YY.Admin.Core.Helper;
|
|||
|
|
using YY.Admin.Core.Services;
|
|||
|
|
namespace YY.Admin.Services.Service.MixingProductionPlan;
|
|||
|
|
|
|||
|
|
/// <summary>密炼生产计划:MES 只读拉取 + 本地缓存,断网读缓存,联网刷新</summary>
|
|||
|
|
public class MixingProductionPlanService : IMixingProductionPlanService, ISingletonDependency
|
|||
|
|
{
|
|||
|
|
private readonly IHttpClientFactory _httpClientFactory;
|
|||
|
|
private readonly IConfiguration _configuration;
|
|||
|
|
private readonly INetworkMonitor _networkMonitor;
|
|||
|
|
private readonly IEventAggregator _eventAggregator;
|
|||
|
|
private readonly ILoggerService _logger;
|
|||
|
|
private readonly SemaphoreSlim _syncLock = new(1, 1);
|
|||
|
|
private readonly object _cacheLock = new();
|
|||
|
|
private readonly string _cacheFilePath;
|
|||
|
|
private List<MesXslMixingProductionPlan> _localCache = new();
|
|||
|
|
|
|||
|
|
private static readonly JsonSerializerOptions _jsonOpts = new()
|
|||
|
|
{
|
|||
|
|
PropertyNameCaseInsensitive = true,
|
|||
|
|
PropertyNamingPolicy = JsonNamingPolicy.CamelCase,
|
|||
|
|
Converters = { new NullableDateTimeJsonConverter() }
|
|||
|
|
};
|
|||
|
|
|
|||
|
|
public MixingProductionPlanService(
|
|||
|
|
IHttpClientFactory httpClientFactory,
|
|||
|
|
IConfiguration configuration,
|
|||
|
|
INetworkMonitor networkMonitor,
|
|||
|
|
IEventAggregator eventAggregator,
|
|||
|
|
ILoggerService logger)
|
|||
|
|
{
|
|||
|
|
_httpClientFactory = httpClientFactory;
|
|||
|
|
_configuration = configuration;
|
|||
|
|
_networkMonitor = networkMonitor;
|
|||
|
|
_eventAggregator = eventAggregator;
|
|||
|
|
_logger = logger;
|
|||
|
|
|
|||
|
|
var appDataDir = Path.Combine(
|
|||
|
|
Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData),
|
|||
|
|
"YY.Admin", "sync-cache");
|
|||
|
|
Directory.CreateDirectory(appDataDir);
|
|||
|
|
_cacheFilePath = Path.Combine(appDataDir, "mes-xsl-mixing-production-plan-cache.json");
|
|||
|
|
|
|||
|
|
LoadCacheFromDisk();
|
|||
|
|
_logger.Information($"[密炼计划] 服务初始化,缓存={_localCache.Count},在线={_networkMonitor.IsOnline}");
|
|||
|
|
|
|||
|
|
_networkMonitor.StatusChanged += OnNetworkStatusChanged;
|
|||
|
|
if (_networkMonitor.IsOnline)
|
|||
|
|
_ = Task.Run(() => SyncFromRemoteAsync(CancellationToken.None));
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
private string BaseUrl => (_configuration.GetValue<string>("JeecgIntegration:BaseUrl") ?? "http://localhost:8080/jeecg-boot").TrimEnd('/');
|
|||
|
|
private int DefaultTenantId => (int?)_configuration.GetValue<long?>("JeecgIntegration:DefaultTenantId") ?? 1002;
|
|||
|
|
private HttpClient CreateClient() => _httpClientFactory.CreateClient("JeecgApi");
|
|||
|
|
|
|||
|
|
public async Task<MixingProductionPlanPageResult> PageAsync(
|
|||
|
|
int pageNo, int pageSize,
|
|||
|
|
DateTime? planDateFrom = null,
|
|||
|
|
DateTime? planDateTo = null,
|
|||
|
|
string? machineName = null,
|
|||
|
|
int? shiftFlag = null,
|
|||
|
|
string? planNo = null,
|
|||
|
|
string? materialName = null,
|
|||
|
|
CancellationToken ct = default)
|
|||
|
|
{
|
|||
|
|
if (_networkMonitor.IsOnline)
|
|||
|
|
{
|
|||
|
|
try
|
|||
|
|
{
|
|||
|
|
await SyncFromRemoteAsync(ct).ConfigureAwait(false);
|
|||
|
|
}
|
|||
|
|
catch (Exception ex)
|
|||
|
|
{
|
|||
|
|
_logger.Warning($"[密炼计划] 列表拉取失败,使用本地缓存:{ex.Message}");
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
List<MesXslMixingProductionPlan> source;
|
|||
|
|
lock (_cacheLock)
|
|||
|
|
source = _localCache.Select(Clone).ToList();
|
|||
|
|
|
|||
|
|
var filtered = ApplyFilters(source, planDateFrom, planDateTo, machineName, shiftFlag, planNo, materialName);
|
|||
|
|
var total = filtered.Count;
|
|||
|
|
var records = filtered
|
|||
|
|
.Skip(Math.Max(0, (pageNo - 1) * pageSize))
|
|||
|
|
.Take(pageSize)
|
|||
|
|
.ToList();
|
|||
|
|
return new MixingProductionPlanPageResult(records, total, pageNo, pageSize);
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
public async Task<List<MesXslMixingProductionPlan>> GetAllCachedAsync(CancellationToken ct = default)
|
|||
|
|
{
|
|||
|
|
if (_networkMonitor.IsOnline)
|
|||
|
|
{
|
|||
|
|
try
|
|||
|
|
{
|
|||
|
|
await SyncFromRemoteAsync(ct).ConfigureAwait(false);
|
|||
|
|
}
|
|||
|
|
catch (Exception ex)
|
|||
|
|
{
|
|||
|
|
_logger.Warning($"[密炼计划] 全量拉取失败,使用本地缓存:{ex.Message}");
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
lock (_cacheLock)
|
|||
|
|
return _localCache.Select(Clone).ToList();
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
public async Task<bool> SyncFromRemoteAsync(CancellationToken ct = default)
|
|||
|
|
{
|
|||
|
|
await _syncLock.WaitAsync(ct).ConfigureAwait(false);
|
|||
|
|
try
|
|||
|
|
{
|
|||
|
|
if (!_networkMonitor.IsOnline)
|
|||
|
|
return false;
|
|||
|
|
|
|||
|
|
var all = new List<MesXslMixingProductionPlan>();
|
|||
|
|
int pageNo = 1;
|
|||
|
|
const int pageSize = 500;
|
|||
|
|
while (true)
|
|||
|
|
{
|
|||
|
|
var query = HttpUtility.ParseQueryString(string.Empty);
|
|||
|
|
query["pageNo"] = pageNo.ToString();
|
|||
|
|
query["pageSize"] = pageSize.ToString();
|
|||
|
|
query["tenantId"] = DefaultTenantId.ToString();
|
|||
|
|
var url = $"{BaseUrl}/xslmes/mesXslMixingProductionPlan/anon/list?{query}";
|
|||
|
|
|
|||
|
|
using var client = CreateClient();
|
|||
|
|
var resp = await client.GetAsync(url, ct).ConfigureAwait(false);
|
|||
|
|
resp.EnsureSuccessStatusCode();
|
|||
|
|
|
|||
|
|
var json = await resp.Content.ReadAsStringAsync(ct).ConfigureAwait(false);
|
|||
|
|
using var doc = JsonDocument.Parse(json);
|
|||
|
|
if (!doc.RootElement.TryGetProperty("result", out var resultEl)) break;
|
|||
|
|
|
|||
|
|
var page = resultEl.GetProperty("records")
|
|||
|
|
.Deserialize<List<MesXslMixingProductionPlan>>(_jsonOpts) ?? new();
|
|||
|
|
all.AddRange(page);
|
|||
|
|
|
|||
|
|
long total = 0;
|
|||
|
|
if (resultEl.TryGetProperty("total", out var totalEl)) total = totalEl.GetInt64();
|
|||
|
|
if (all.Count >= total || page.Count < pageSize) break;
|
|||
|
|
pageNo++;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
List<MesXslMixingProductionPlan> localSnapshot;
|
|||
|
|
lock (_cacheLock)
|
|||
|
|
localSnapshot = _localCache.Select(Clone).ToList();
|
|||
|
|
|
|||
|
|
var (merged, stats) = MesReadOnlyCacheMergeHelper.Merge(
|
|||
|
|
localSnapshot,
|
|||
|
|
all,
|
|||
|
|
x => x.Id,
|
|||
|
|
IsPlanContentEqual,
|
|||
|
|
Clone);
|
|||
|
|
|
|||
|
|
if (!stats.HasChanges)
|
|||
|
|
{
|
|||
|
|
_logger.Information($"[密炼计划] 与 MES 对比无差异,跳过更新 count={merged.Count}");
|
|||
|
|
return false;
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
lock (_cacheLock)
|
|||
|
|
{
|
|||
|
|
_localCache = merged;
|
|||
|
|
SaveCacheToDiskUnsafe();
|
|||
|
|
}
|
|||
|
|
_logger.Information(
|
|||
|
|
$"[密炼计划] 差异同步完成 total={merged.Count} 新增={stats.Added} 变更={stats.Updated} 删除={stats.Removed}");
|
|||
|
|
return true;
|
|||
|
|
}
|
|||
|
|
catch (Exception ex)
|
|||
|
|
{
|
|||
|
|
_logger.Warning($"[密炼计划] 远程同步失败:{ex.Message}");
|
|||
|
|
throw;
|
|||
|
|
}
|
|||
|
|
finally
|
|||
|
|
{
|
|||
|
|
_syncLock.Release();
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
private void OnNetworkStatusChanged(bool isOnline)
|
|||
|
|
{
|
|||
|
|
if (!isOnline) return;
|
|||
|
|
_ = Task.Run(async () =>
|
|||
|
|
{
|
|||
|
|
try
|
|||
|
|
{
|
|||
|
|
if (!await SyncFromRemoteAsync(CancellationToken.None).ConfigureAwait(false))
|
|||
|
|
return;
|
|||
|
|
_eventAggregator.GetEvent<MixingProductionPlanChangedEvent>()
|
|||
|
|
.Publish(new MixingProductionPlanChangedPayload { Action = "reconnect" });
|
|||
|
|
}
|
|||
|
|
catch (Exception ex)
|
|||
|
|
{
|
|||
|
|
_logger.Warning($"[密炼计划] 重连同步失败:{ex.Message}");
|
|||
|
|
}
|
|||
|
|
});
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
private static bool IsPlanContentEqual(MesXslMixingProductionPlan a, MesXslMixingProductionPlan b) =>
|
|||
|
|
string.Equals(GetPlanFingerprint(a), GetPlanFingerprint(b), StringComparison.Ordinal);
|
|||
|
|
|
|||
|
|
private static string GetPlanFingerprint(MesXslMixingProductionPlan x) =>
|
|||
|
|
JsonSerializer.Serialize(Clone(x), _jsonOpts);
|
|||
|
|
private static List<MesXslMixingProductionPlan> ApplyFilters(
|
|||
|
|
List<MesXslMixingProductionPlan> source,
|
|||
|
|
DateTime? planDateFrom,
|
|||
|
|
DateTime? planDateTo,
|
|||
|
|
string? machineName,
|
|||
|
|
int? shiftFlag,
|
|||
|
|
string? planNo,
|
|||
|
|
string? materialName)
|
|||
|
|
{
|
|||
|
|
IEnumerable<MesXslMixingProductionPlan> q = source;
|
|||
|
|
if (planDateFrom.HasValue)
|
|||
|
|
q = q.Where(x => x.PlanDate?.Date >= planDateFrom.Value.Date);
|
|||
|
|
if (planDateTo.HasValue)
|
|||
|
|
q = q.Where(x => x.PlanDate?.Date <= planDateTo.Value.Date);
|
|||
|
|
if (!string.IsNullOrWhiteSpace(machineName))
|
|||
|
|
q = q.Where(x => (x.MachineName ?? "").Contains(machineName.Trim(), StringComparison.OrdinalIgnoreCase));
|
|||
|
|
if (shiftFlag.HasValue && shiftFlag.Value > 0)
|
|||
|
|
q = q.Where(x => x.ShiftFlag == shiftFlag.Value);
|
|||
|
|
if (!string.IsNullOrWhiteSpace(planNo))
|
|||
|
|
q = q.Where(x => (x.PlanNo ?? "").Contains(planNo.Trim(), StringComparison.OrdinalIgnoreCase));
|
|||
|
|
if (!string.IsNullOrWhiteSpace(materialName))
|
|||
|
|
q = q.Where(x => (x.MaterialName ?? "").Contains(materialName.Trim(), StringComparison.OrdinalIgnoreCase));
|
|||
|
|
|
|||
|
|
return q
|
|||
|
|
.OrderByDescending(x => x.PlanDate ?? DateTime.MinValue)
|
|||
|
|
.ThenBy(x => x.SortNo ?? int.MaxValue)
|
|||
|
|
.ThenBy(x => x.MachineName)
|
|||
|
|
.ToList();
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
private void LoadCacheFromDisk()
|
|||
|
|
{
|
|||
|
|
try
|
|||
|
|
{
|
|||
|
|
if (!File.Exists(_cacheFilePath)) return;
|
|||
|
|
_localCache = JsonSerializer.Deserialize<List<MesXslMixingProductionPlan>>(
|
|||
|
|
File.ReadAllText(_cacheFilePath), _jsonOpts) ?? new();
|
|||
|
|
}
|
|||
|
|
catch
|
|||
|
|
{
|
|||
|
|
_localCache = new();
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
private void SaveCacheToDiskUnsafe() =>
|
|||
|
|
File.WriteAllText(_cacheFilePath, JsonSerializer.Serialize(_localCache, _jsonOpts));
|
|||
|
|
|
|||
|
|
private static MesXslMixingProductionPlan Clone(MesXslMixingProductionPlan x) => new()
|
|||
|
|
{
|
|||
|
|
Id = x.Id,
|
|||
|
|
SortNo = x.SortNo,
|
|||
|
|
MachineId = x.MachineId,
|
|||
|
|
MachineName = x.MachineName,
|
|||
|
|
ShiftFlag = x.ShiftFlag,
|
|||
|
|
PlanDate = x.PlanDate,
|
|||
|
|
PlanNo = x.PlanNo,
|
|||
|
|
PlanId = x.PlanId,
|
|||
|
|
PlanType = x.PlanType,
|
|||
|
|
SourceOrderId = x.SourceOrderId,
|
|||
|
|
MaterialId = x.MaterialId,
|
|||
|
|
MaterialName = x.MaterialName,
|
|||
|
|
OrderNo = x.OrderNo,
|
|||
|
|
OrderDate = x.OrderDate,
|
|||
|
|
FormulaName = x.FormulaName,
|
|||
|
|
PlanWeight = x.PlanWeight,
|
|||
|
|
PlannedCarCount = x.PlannedCarCount,
|
|||
|
|
ScheduledCarCount = x.ScheduledCarCount,
|
|||
|
|
FinishedCarCount = x.FinishedCarCount,
|
|||
|
|
PlanCount = x.PlanCount,
|
|||
|
|
Remark = x.Remark,
|
|||
|
|
TenantId = x.TenantId,
|
|||
|
|
SysOrgCode = x.SysOrgCode,
|
|||
|
|
CreateBy = x.CreateBy,
|
|||
|
|
CreateTime = x.CreateTime,
|
|||
|
|
UpdateBy = x.UpdateBy,
|
|||
|
|
UpdateTime = x.UpdateTime,
|
|||
|
|
DelFlag = x.DelFlag
|
|||
|
|
};
|
|||
|
|
|
|||
|
|
private sealed class NullableDateTimeJsonConverter : JsonConverter<DateTime?>
|
|||
|
|
{
|
|||
|
|
private static readonly string[] Formats =
|
|||
|
|
[
|
|||
|
|
"yyyy-MM-dd HH:mm:ss", "yyyy-MM-dd HH:mm:ss.fff",
|
|||
|
|
"yyyy-MM-ddTHH:mm:ss", "yyyy-MM-ddTHH:mm:ss.fff",
|
|||
|
|
"yyyy-MM-ddTHH:mm:ssZ", "yyyy-MM-ddTHH:mm:ss.fffZ",
|
|||
|
|
"yyyy-MM-dd"
|
|||
|
|
];
|
|||
|
|
|
|||
|
|
public override DateTime? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
|
|||
|
|
{
|
|||
|
|
if (reader.TokenType == JsonTokenType.Null) return null;
|
|||
|
|
if (reader.TokenType == JsonTokenType.String)
|
|||
|
|
{
|
|||
|
|
var raw = reader.GetString();
|
|||
|
|
if (string.IsNullOrWhiteSpace(raw)) return null;
|
|||
|
|
if (DateTime.TryParseExact(raw, Formats, System.Globalization.CultureInfo.InvariantCulture,
|
|||
|
|
System.Globalization.DateTimeStyles.AssumeLocal, out var dt)) return dt;
|
|||
|
|
if (DateTime.TryParse(raw, out var fb)) return fb;
|
|||
|
|
}
|
|||
|
|
throw new JsonException($"无法转换为 DateTime?,token={reader.TokenType}");
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
public override void Write(Utf8JsonWriter writer, DateTime? value, JsonSerializerOptions options)
|
|||
|
|
{
|
|||
|
|
if (value.HasValue) writer.WriteStringValue(value.Value.ToString("yyyy-MM-dd HH:mm:ss"));
|
|||
|
|
else writer.WriteNullValue();
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
}
|