EnumToDictJob.cs 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. // Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
  2. //
  3. // 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
  6. namespace Admin.NET.Core;
  7. /// <summary>
  8. /// 枚举转字典
  9. /// </summary>
  10. [JobDetail("job_EnumToDictJob", Description = "枚举转字典", GroupName = "default", Concurrent = false)]
  11. [PeriodSeconds(1, TriggerId = "trigger_EnumToDictJob", Description = "枚举转字典", MaxNumberOfRuns = 1, RunOnStart = true)]
  12. public class EnumToDictJob : IJob
  13. {
  14. private readonly IServiceScopeFactory _scopeFactory;
  15. private const string DefaultTagType = null;
  16. private const int OrderOffset = 10;
  17. public EnumToDictJob(IServiceScopeFactory scopeFactory)
  18. {
  19. _scopeFactory = scopeFactory;
  20. }
  21. public async Task ExecuteAsync(JobExecutingContext context, CancellationToken stoppingToken)
  22. {
  23. var originColor = Console.ForegroundColor;
  24. Console.ForegroundColor = ConsoleColor.Yellow;
  25. Console.WriteLine($"【{DateTime.Now}】系统枚举转换字典");
  26. using var serviceScope = _scopeFactory.CreateScope();
  27. var db = serviceScope.ServiceProvider.GetRequiredService<ISqlSugarClient>().CopyNew();
  28. var sysEnumService = serviceScope.ServiceProvider.GetRequiredService<SysEnumService>();
  29. var sysDictTypeList = GetDictByEnumType(sysEnumService.GetEnumTypeList());
  30. // 校验枚举类命名规范,字典相关功能中需要通过后缀判断是否为枚举类型
  31. Console.ForegroundColor = ConsoleColor.Red;
  32. foreach (var dictType in sysDictTypeList.Where(x => !x.Code.EndsWith("Enum")))
  33. Console.WriteLine($"【{DateTime.Now}】系统枚举转换字典的枚举类名称必须以Enum结尾: {dictType.Code} ({dictType.Name})");
  34. sysDictTypeList = sysDictTypeList.Where(x => x.Code.EndsWith("Enum")).ToList();
  35. await SyncEnumToDictInfoAsync(db, sysDictTypeList);
  36. Console.ForegroundColor = ConsoleColor.Yellow;
  37. try
  38. {
  39. await db.BeginTranAsync();
  40. var storageable1 = await db.Storageable(sysDictTypeList)
  41. .SplitUpdate(it => it.Any())
  42. .SplitInsert(_ => true)
  43. .ToStorageAsync();
  44. await storageable1.AsInsertable.ExecuteCommandAsync(stoppingToken);
  45. await storageable1.AsUpdateable.ExecuteCommandAsync(stoppingToken);
  46. Console.WriteLine($"【{DateTime.Now}】系统枚举类转字典类型数据: 插入{storageable1.InsertList.Count}条, 更新{storageable1.UpdateList.Count}条, 共{storageable1.TotalList.Count}条。");
  47. var storageable2 = await db.Storageable(sysDictTypeList.SelectMany(x => x.Children).ToList())
  48. .WhereColumns(it => new { it.DictTypeId, it.Value })
  49. .SplitUpdate(it => it.Any())
  50. .SplitInsert(_ => true)
  51. .ToStorageAsync();
  52. await storageable2.AsInsertable.ExecuteCommandAsync(stoppingToken);
  53. await storageable2.AsUpdateable.UpdateColumns(u => new
  54. {
  55. u.Label,
  56. u.Code,
  57. u.Value
  58. }).ExecuteCommandAsync(stoppingToken);
  59. Console.WriteLine($"【{DateTime.Now}】系统枚举项转字典值数据: 插入{storageable2.InsertList.Count}条, 更新{storageable2.UpdateList.Count}条, 共{storageable2.TotalList.Count}条。");
  60. await db.CommitTranAsync();
  61. }
  62. catch (Exception error)
  63. {
  64. await db.RollbackTranAsync();
  65. Log.Error($"系统枚举转换字典操作错误:{error.Message}\n堆栈跟踪:{error.StackTrace}", error);
  66. throw;
  67. }
  68. finally
  69. {
  70. Console.ForegroundColor = originColor;
  71. }
  72. }
  73. /// <summary>
  74. /// 用于同步枚举转字典数据
  75. /// </summary>
  76. /// <param name="db"></param>
  77. /// <param name="list"></param>
  78. private async Task SyncEnumToDictInfoAsync(SqlSugarClient db, List<SysDictType> list)
  79. {
  80. var codeList = list.Select(x => x.Code).ToList();
  81. foreach (var dbDictType in await db.Queryable<SysDictType>().ClearFilter().Where(x => codeList.Contains(x.Code)).ToListAsync() ?? new())
  82. {
  83. var enumDictType = list.First(x => x.Code == dbDictType.Code);
  84. if (enumDictType.Id == dbDictType.Id)
  85. {
  86. // 字典值表字段改变后每条字典值记录会多出一条,此处用于删除多余的字典值数据
  87. var dataValueList = enumDictType.Children.Select(e => e.Value).ToList();
  88. await db.Deleteable<SysDictData>().Where(x => x.DictTypeId == dbDictType.Id && !dataValueList.Contains(x.Value)).ExecuteCommandAsync();
  89. continue;
  90. }
  91. // 数据不一致则删除
  92. await db.Deleteable<SysDictData>().Where(x => x.DictTypeId == dbDictType.Id).ExecuteCommandAsync();
  93. await db.Deleteable<SysDictType>().Where(x => x.Id == dbDictType.Id).ExecuteCommandAsync();
  94. Console.WriteLine($"【{DateTime.Now}】删除字典数据: {dbDictType.Name}-{dbDictType.Code}");
  95. }
  96. }
  97. /// <summary>
  98. /// 枚举信息转字典
  99. /// </summary>
  100. /// <param name="enumTypeList"></param>
  101. /// <returns></returns>
  102. private List<SysDictType> GetDictByEnumType(List<EnumTypeOutput> enumTypeList)
  103. {
  104. var orderNo = 1;
  105. var list = new List<SysDictType>();
  106. foreach (var type in enumTypeList)
  107. {
  108. var dictType = new SysDictType
  109. {
  110. Id = 900000000000 + CommonUtil.GetFixedHashCode(type.TypeFullName),
  111. SysFlag = YesNoEnum.Y,
  112. Code = type.TypeName,
  113. Name = type.TypeDescribe,
  114. Remark = type.TypeFullName
  115. };
  116. dictType.Children = type.EnumEntities.Select(x => new SysDictData
  117. {
  118. Id = dictType.Id + orderNo++,
  119. DictTypeId = dictType.Id,
  120. Code = x.Name,
  121. Label = x.Describe,
  122. Value = x.Value.ToString(),
  123. OrderNo = x.Value + OrderOffset,
  124. TagType = x.Theme != "" ? x.Theme : DefaultTagType,
  125. TenantId = SqlSugarConst.DefaultTenantId
  126. }).ToList();
  127. list.Add(dictType);
  128. }
  129. return list;
  130. }
  131. }