YitIdHelperExtension.cs 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. //// 麻省理工学院许可证
  2. ////
  3. //// 版权所有 (c) 2021-2023 zuohuaijun,大名科技(天津)有限公司 联系电话/微信:18020030720 QQ:515096995
  4. ////
  5. //// 特此免费授予获得本软件的任何人以处理本软件的权利,但须遵守以下条件:在所有副本或重要部分的软件中必须包括上述版权声明和本许可声明。
  6. ////
  7. //// 软件按“原样”提供,不提供任何形式的明示或暗示的保证,包括但不限于对适销性、适用性和非侵权的保证。
  8. //// 在任何情况下,作者或版权持有人均不对任何索赔、损害或其他责任负责,无论是因合同、侵权或其他方式引起的,与软件或其使用或其他交易有关。
  9. namespace Admin.NET.Core;
  10. /// <summary>
  11. /// YitIdHelper 自动获取WorkId拓展
  12. /// </summary>
  13. public static class YitIdHelperExtension
  14. {
  15. private const string MainLockName = "IdGen:WorkerId:Lock";
  16. private const string MainValueKey = "IdGen:WorkerId:Value";
  17. private static readonly List<string> _workIds = new();
  18. private static SnowIdOptions _options;
  19. public static void AddYitIdHelper(this IServiceCollection services, SnowIdOptions options)
  20. {
  21. _options = options;
  22. // 排除开发环境和Windows服务器
  23. if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows) || App.WebHostEnvironment.IsDevelopment())
  24. {
  25. YitIdHelper.SetIdGenerator(_options);
  26. return;
  27. }
  28. var maxLength = Math.Pow(2, _options.WorkerIdBitLength.ParseToDouble());
  29. for (int i = 0; i < maxLength; i++)
  30. {
  31. _workIds.Add(i.ToString());
  32. }
  33. Random ran = new();
  34. int milliseconds = ran.Next(10, 1000);
  35. Thread.Sleep(milliseconds);
  36. SetWorkId();
  37. }
  38. private static void SetWorkId()
  39. {
  40. var lockName = $"{_options.WorkerPrefix}{MainLockName}";
  41. var valueKey = $"{_options.WorkerPrefix}{MainValueKey}";
  42. var minWorkId = 0;
  43. var maxWorkId = Math.Pow(2, _options.WorkerIdBitLength.ParseToDouble());
  44. var client = App.GetService<ICache>();
  45. var redisLock = client.AcquireLock(lockName, 10000, 15000, true);
  46. var keys = client.Keys.Where(o => o.Contains($"{_options.WorkerPrefix}{valueKey}:*"));
  47. var tempWorkIds = _workIds;
  48. foreach (var key in keys)
  49. {
  50. var tempWorkId = key[key.LastIndexOf(":", StringComparison.Ordinal)..];
  51. tempWorkIds.Remove(tempWorkId);
  52. }
  53. try
  54. {
  55. string workIdKey = "";
  56. foreach (var item in tempWorkIds)
  57. {
  58. string workIdStr = "";
  59. workIdStr = item;
  60. workIdKey = $"{valueKey}:{workIdStr}";
  61. var exist = client.Get<bool>(workIdKey);
  62. if (exist)
  63. {
  64. workIdKey = "";
  65. continue;
  66. }
  67. Console.WriteLine($"###########当前应用WorkId:【{workIdStr}】###########");
  68. long workId = workIdStr.ParseToLong();
  69. if (workId < minWorkId || workId > maxWorkId)
  70. continue;
  71. // 设置雪花Id算法机器码
  72. YitIdHelper.SetIdGenerator(new IdGeneratorOptions
  73. {
  74. WorkerId = (ushort)workId,
  75. WorkerIdBitLength = _options.WorkerIdBitLength,
  76. SeqBitLength = _options.SeqBitLength
  77. });
  78. client.Set(workIdKey, true, TimeSpan.FromSeconds(15));
  79. break;
  80. }
  81. if (string.IsNullOrWhiteSpace(workIdKey)) throw Oops.Oh("未设置有效的机器码,启动失败");
  82. // 开一个任务设置当前workId过期时间
  83. Task.Run(() =>
  84. {
  85. while (true)
  86. {
  87. client.SetExpire(workIdKey, TimeSpan.FromSeconds(15));
  88. //Task.Delay(5000);
  89. Thread.Sleep(10000);
  90. }
  91. });
  92. }
  93. catch (Exception e)
  94. {
  95. throw Oops.Oh($"{e.Message};{e.StackTrace};{e.StackTrace}");
  96. }
  97. finally
  98. {
  99. redisLock?.Dispose();
  100. }
  101. }
  102. }