using System.Diagnostics; using Admin.NET.Core; using SqlSugar; namespace Admin.NET.Plugin.AiDOP.Infrastructure; /// /// 一次性迁移:将历史 TenantId=1 的 AiDOP 业务数据更新为框架默认租户。 /// 幂等——若已无 TenantId=1 的行则不做任何事。 /// public static class AidopTenantMigration { private static readonly (string Table, string TenantColumn)[] Tables = { ("ado_smart_ops_kpi_master", "TenantId"), ("ado_smart_ops_layout_item", "TenantId"), ("ado_smart_ops_home_module", "TenantId"), ("ado_s9_kpi_value_l1_day", "tenant_id"), ("ado_s9_kpi_value_l2_day", "tenant_id"), ("ado_s9_kpi_value_l3_day", "tenant_id"), ("ado_s9_kpi_value_l4_day", "tenant_id"), ("ado_smart_ops_kpi_atomic_day", "tenant_id") }; public static void MigrateOldTenantId(ISqlSugarClient db) { EnsureKpiValueL4Table(db); EnsureKpiAtomicTable(db); var target = SqlSugarConst.DefaultTenantId; if (target == 1) return; foreach (var (table, tenantColumn) in Tables) { try { db.Ado.ExecuteCommand( $"UPDATE `{table}` SET `{tenantColumn}` = @target WHERE `{tenantColumn}` = 1", new { target }); } catch (Exception ex) { Trace.TraceWarning($"AidopTenantMigration [{table}]: {ex.Message}"); } } DropOldUniqueIndex(db); BackfillThresholds(db); } public static void EnsureKpiValueL4Table(ISqlSugarClient db) { try { db.Ado.ExecuteCommand( "CREATE TABLE IF NOT EXISTS `ado_s9_kpi_value_l4_day` LIKE `ado_s9_kpi_value_l3_day`"); } catch (Exception ex) { Trace.TraceWarning($"AidopTenantMigration ensure L4 value table: {ex.Message}"); } } public static void EnsureKpiAtomicTable(ISqlSugarClient db) { RepairKpiAtomicTableSchemaIfNeeded(db); try { db.Ado.ExecuteCommand( """ CREATE TABLE IF NOT EXISTS ado_smart_ops_kpi_atomic_day ( id BIGINT NOT NULL AUTO_INCREMENT COMMENT '主键', tenant_id BIGINT NOT NULL COMMENT '租户ID', factory_id BIGINT NOT NULL DEFAULT 1 COMMENT '工厂ID', domain_code VARCHAR(50) NOT NULL COMMENT '业务对象域', metric_code VARCHAR(50) NOT NULL COMMENT '指标编码', metric_level TINYINT NOT NULL DEFAULT 1 COMMENT '指标层级', stat_date DATE NOT NULL COMMENT '统计日期', customer_code VARCHAR(100) NOT NULL DEFAULT '' COMMENT '客户编码', product_code VARCHAR(100) NOT NULL DEFAULT '' COMMENT '产品编码', order_no VARCHAR(100) NOT NULL DEFAULT '' COMMENT '订单号', production_line VARCHAR(100) NOT NULL DEFAULT '' COMMENT '产线', supplier_code VARCHAR(100) NOT NULL DEFAULT '' COMMENT '供应商编码', material_code VARCHAR(100) NOT NULL DEFAULT '' COMMENT '物料编码', po_no VARCHAR(100) NOT NULL DEFAULT '' COMMENT '采购单号', warehouse_code VARCHAR(100) NOT NULL DEFAULT '' COMMENT '仓库编码', work_order_no VARCHAR(100) NOT NULL DEFAULT '' COMMENT '工单号', equipment_code VARCHAR(100) NOT NULL DEFAULT '' COMMENT '设备编码', outbound_no VARCHAR(100) NOT NULL DEFAULT '' COMMENT '出库单号', sum_value DECIMAL(18, 6) NULL COMMENT '可累加值', numerator_value DECIMAL(18, 6) NULL COMMENT '比率分子', denominator_value DECIMAL(18, 6) NULL COMMENT '比率分母', sample_count INT NULL COMMENT '样本数', min_value DECIMAL(18, 6) NULL COMMENT '最小值', max_value DECIMAL(18, 6) NULL COMMENT '最大值', batch_id VARCHAR(100) NULL COMMENT 'MDP批次', calc_time DATETIME(3) NULL COMMENT '计算时间', grain VARCHAR(20) NOT NULL DEFAULT 'day' COMMENT '统计粒度', is_deleted TINYINT NOT NULL DEFAULT 0 COMMENT '删除标识', PRIMARY KEY (id), UNIQUE KEY uk_atomic_grain ( tenant_id, factory_id, domain_code, metric_code, stat_date, customer_code, product_code, order_no, production_line, supplier_code, material_code, po_no, warehouse_code, work_order_no, equipment_code, outbound_no ), KEY idx_tenant_domain_metric_date (tenant_id, domain_code, metric_code, stat_date), KEY idx_tenant_module_lookup (tenant_id, factory_id, metric_code, stat_date) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='KPI原子聚合层(日)' """); EnsureKpiAtomicColumnDefaults(db); } catch (Exception ex) { Trace.TraceWarning($"AidopTenantMigration ensure KPI atomic table: {ex.Message}"); } } /// /// 补齐维度列 DEFAULT '',避免 CodeFirst 误建表后 INSERT 缺列报错。 /// private static void EnsureKpiAtomicColumnDefaults(ISqlSugarClient db) { try { db.Ado.ExecuteCommand( """ ALTER TABLE ado_smart_ops_kpi_atomic_day MODIFY customer_code VARCHAR(100) NOT NULL DEFAULT '' COMMENT '客户编码', MODIFY product_code VARCHAR(100) NOT NULL DEFAULT '' COMMENT '产品编码', MODIFY order_no VARCHAR(100) NOT NULL DEFAULT '' COMMENT '订单号', MODIFY production_line VARCHAR(100) NOT NULL DEFAULT '' COMMENT '产线', MODIFY supplier_code VARCHAR(100) NOT NULL DEFAULT '' COMMENT '供应商编码', MODIFY material_code VARCHAR(100) NOT NULL DEFAULT '' COMMENT '物料编码', MODIFY po_no VARCHAR(100) NOT NULL DEFAULT '' COMMENT '采购单号', MODIFY warehouse_code VARCHAR(100) NOT NULL DEFAULT '' COMMENT '仓库编码', MODIFY work_order_no VARCHAR(100) NOT NULL DEFAULT '' COMMENT '工单号', MODIFY equipment_code VARCHAR(100) NOT NULL DEFAULT '' COMMENT '设备编码', MODIFY outbound_no VARCHAR(100) NOT NULL DEFAULT '' COMMENT '出库单号', MODIFY grain VARCHAR(20) NOT NULL DEFAULT 'day' COMMENT '统计粒度', MODIFY is_deleted TINYINT NOT NULL DEFAULT 0 COMMENT '删除标识' """); } catch (Exception ex) { Trace.TraceWarning($"AidopTenantMigration ensure KPI atomic column defaults: {ex.Message}"); } } /// /// 早期 SqlSugar 误建 PascalCase 列名时,DROP 后由 EnsureKpiAtomicTable 按 snake_case 重建。 /// private static void RepairKpiAtomicTableSchemaIfNeeded(ISqlSugarClient db) { try { var columns = db.Ado.SqlQuery( """ SELECT COLUMN_NAME FROM information_schema.COLUMNS WHERE TABLE_SCHEMA = DATABASE() AND TABLE_NAME = 'ado_smart_ops_kpi_atomic_day' """); if (columns == null || columns.Count == 0) return; var hasSnakeDomain = columns.Any(c => string.Equals(c, "domain_code", StringComparison.OrdinalIgnoreCase)); var hasPascalDomain = columns.Any(c => string.Equals(c, "DomainCode", StringComparison.OrdinalIgnoreCase)); if (hasSnakeDomain || !hasPascalDomain) return; Trace.TraceWarning("AidopTenantMigration: ado_smart_ops_kpi_atomic_day 检测到 PascalCase 列名,将 DROP 后重建为 snake_case。"); db.Ado.ExecuteCommand("DROP TABLE ado_smart_ops_kpi_atomic_day"); } catch (Exception ex) { Trace.TraceWarning($"AidopTenantMigration repair KPI atomic schema: {ex.Message}"); } } private static void BackfillThresholds(ISqlSugarClient db) { try { db.Ado.ExecuteCommand( "UPDATE ado_smart_ops_kpi_master SET YellowThreshold=95, RedThreshold=80 " + "WHERE YellowThreshold IS NULL AND Direction='higher_is_better'"); db.Ado.ExecuteCommand( "UPDATE ado_smart_ops_kpi_master SET YellowThreshold=110, RedThreshold=120 " + "WHERE YellowThreshold IS NULL AND Direction='lower_is_better'"); } catch (Exception ex) { Trace.TraceWarning($"AidopTenantMigration backfill thresholds: {ex.Message}"); } } private static void DropOldUniqueIndex(ISqlSugarClient db) { try { var rows = db.Ado.SqlQuery( "SHOW INDEX FROM ado_smart_ops_kpi_master WHERE Key_name = 'uk_kpi_master_code'"); if (rows != null && rows.Count > 0) db.Ado.ExecuteCommand("ALTER TABLE ado_smart_ops_kpi_master DROP INDEX uk_kpi_master_code"); } catch (Exception ex) { Trace.TraceWarning($"AidopTenantMigration drop old index: {ex.Message}"); } } }