using Admin.NET.Plugin.AiDOP.Infrastructure; using Admin.NET.Plugin.AiDOP.Order; using Admin.NET.Plugin.AiDOP.ProcurementExecution; using Admin.NET.Plugin.AiDOP.Production; using Admin.NET.Plugin.AiDOP.Supply; using Furion.Schedule; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; using SqlSugar; using System.Text.Json; namespace Admin.NET.Plugin.AiDOP.Job; /// /// 应用启动后按 S1→S2→S3→S4 顺序执行 MDP 全量同步,构建模块 KPI 汇总与 KPI 原子聚合层。 /// 新环境发布后无需等待 hourly 任务即可在看板筛选中看到数据。 /// [JobDetail("job_smart_ops_kpi_mdp_bootstrap", Description = "智慧运营KPI:启动时按序执行S1→S2→S3→S4 MDP同步(含原子聚合层)", GroupName = "default", Concurrent = false)] [PeriodSeconds(1, TriggerId = "trigger_smart_ops_kpi_mdp_bootstrap", Description = "应用启动后立即执行一次", MaxNumberOfRuns = 1, RunOnStart = true)] public class SmartOpsKpiMdpBootstrapJob : IJob { private readonly IServiceScopeFactory _scopeFactory; private readonly ILogger _logger; public SmartOpsKpiMdpBootstrapJob(IServiceScopeFactory scopeFactory, ILoggerFactory loggerFactory) { _scopeFactory = scopeFactory; _logger = loggerFactory.CreateLogger(nameof(SmartOpsKpiMdpBootstrapJob)); } public async Task ExecuteAsync(JobExecutingContext context, CancellationToken stoppingToken) { using var scope = _scopeFactory.CreateScope(); var db = scope.ServiceProvider.GetRequiredService(); try { AidopTenantMigration.EnsureKpiAtomicTable(db); _logger.LogInformation("SmartOpsKpiMdpBootstrapJob: ado_smart_ops_kpi_atomic_day 已就绪"); } catch (Exception ex) { _logger.LogError(ex, "SmartOpsKpiMdpBootstrapJob: 原子聚合表初始化失败"); } await RunStepAsync("S1", stoppingToken, async sp => { var result = await sp.GetRequiredService().RunFullAsync(stoppingToken, "BOOTSTRAP"); return new { result.StageRows, result.StandardRows, result.DwdRows, result.KpiRows, result.AtomicRows }; }); await RunStepAsync("S2", stoppingToken, async sp => { var result = await sp.GetRequiredService().RunFullAsync(stoppingToken, "BOOTSTRAP"); return new { result.StageRows, result.StandardRows, result.DwdRows, result.KpiRows, result.AtomicRows }; }); await RunStepAsync("S3", stoppingToken, async sp => { var result = await sp.GetRequiredService().RunFullAsync(stoppingToken, "BOOTSTRAP"); return new { result.StageRows, result.StandardRows, result.DwdRows, result.KpiRows, result.AtomicRows }; }); await RunStepAsync("S4", stoppingToken, async sp => { var result = await sp.GetRequiredService().RunFullAsync(stoppingToken, "BOOTSTRAP"); return new { result.StageRows, result.StandardRows, result.DwdRows, result.KpiRows }; }); _logger.LogInformation("SmartOpsKpiMdpBootstrapJob: 全部 MDP 引导同步已完成"); } private async Task RunStepAsync(string step, CancellationToken stoppingToken, Func> action) { using var scope = _scopeFactory.CreateScope(); try { stoppingToken.ThrowIfCancellationRequested(); var payload = await action(scope.ServiceProvider); _logger.LogInformation("SmartOpsKpiMdpBootstrapJob {Step} 完成 {Payload}", step, JsonSerializer.Serialize(payload)); } catch (OperationCanceledException) when (stoppingToken.IsCancellationRequested) { _logger.LogInformation("SmartOpsKpiMdpBootstrapJob {Step} 收到停止信号", step); throw; } catch (Exception ex) { _logger.LogError(ex, "SmartOpsKpiMdpBootstrapJob {Step} 执行失败,继续后续步骤", step); } } }