|
|
@@ -0,0 +1,119 @@
|
|
|
+using Microsoft.Extensions.Hosting;
|
|
|
+
|
|
|
+namespace Admin.NET.Core;
|
|
|
+
|
|
|
+/// <summary>
|
|
|
+/// YitIdHelper 自动获取WorkId拓展
|
|
|
+/// </summary>
|
|
|
+public static class YitIdHelperExtension
|
|
|
+{
|
|
|
+ private const int MinWorkId = 0;
|
|
|
+ private const int MaxWorkId = 63;
|
|
|
+
|
|
|
+ private const string MainLockName = "IdGen:WorkerId:Lock";
|
|
|
+ private const string MainValueKey = "IdGen:WorkerId:Value";
|
|
|
+
|
|
|
+ private static readonly List<string> _workIds = new();
|
|
|
+ private static SnowIdOptions _options;
|
|
|
+
|
|
|
+ public static void AddYitIdHelper(this IServiceCollection services, SnowIdOptions options)
|
|
|
+ {
|
|
|
+ _options = options;
|
|
|
+
|
|
|
+ // 排除开发环境和Windows服务器
|
|
|
+ if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows) || App.WebHostEnvironment.IsDevelopment())
|
|
|
+ {
|
|
|
+ YitIdHelper.SetIdGenerator(_options);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ var maxLength = Math.Pow(2, _options.WorkerIdBitLength.ParseToDouble());
|
|
|
+
|
|
|
+ for (int i = 0; i < maxLength; i++)
|
|
|
+ {
|
|
|
+ _workIds.Add(i.ToString());
|
|
|
+ }
|
|
|
+
|
|
|
+ Random ran = new();
|
|
|
+ int milliseconds = ran.Next(10, 1000);
|
|
|
+ Thread.Sleep(milliseconds);
|
|
|
+
|
|
|
+ SetWorkId();
|
|
|
+ }
|
|
|
+
|
|
|
+ private static void SetWorkId()
|
|
|
+ {
|
|
|
+ var lockName = $"{_options.CachePrefix}{MainLockName}";
|
|
|
+ var valueKey = $"{_options.CachePrefix}{MainValueKey}"; ;
|
|
|
+
|
|
|
+ var client = App.GetService<ICache>();
|
|
|
+ var redisLock = client.AcquireLock(lockName, 10000, 15000, true);
|
|
|
+ var keys = client.Keys.Where(o => o.Contains($"{_options.CachePrefix}{valueKey}:*"));
|
|
|
+
|
|
|
+ var tempWorkIds = _workIds;
|
|
|
+ foreach (var key in keys)
|
|
|
+ {
|
|
|
+ var tempWorkId = key[key.LastIndexOf(":", StringComparison.Ordinal)..];
|
|
|
+ tempWorkIds.Remove(tempWorkId);
|
|
|
+ }
|
|
|
+
|
|
|
+ try
|
|
|
+ {
|
|
|
+ string workIdKey = "";
|
|
|
+ foreach (var item in tempWorkIds)
|
|
|
+ {
|
|
|
+ string workIdStr = "";
|
|
|
+
|
|
|
+ workIdStr = item;
|
|
|
+ workIdKey = $"{valueKey}:{workIdStr}";
|
|
|
+ var exist = client.Get<bool>(workIdKey);
|
|
|
+
|
|
|
+ if (exist)
|
|
|
+ {
|
|
|
+ workIdKey = "";
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+
|
|
|
+ Console.WriteLine($"###########当前应用WorkId:【{workIdStr}】###########");
|
|
|
+
|
|
|
+ long workId = workIdStr.ParseToLong();
|
|
|
+
|
|
|
+ if (workId is < MinWorkId or > MaxWorkId)
|
|
|
+ continue;
|
|
|
+
|
|
|
+ // 设置雪花Id算法机器码
|
|
|
+ YitIdHelper.SetIdGenerator(new IdGeneratorOptions
|
|
|
+ {
|
|
|
+ WorkerId = (ushort)workId,
|
|
|
+ WorkerIdBitLength = _options.WorkerIdBitLength,
|
|
|
+ SeqBitLength = _options.SeqBitLength
|
|
|
+ });
|
|
|
+
|
|
|
+ client.Set(workIdKey, true, TimeSpan.FromSeconds(15));
|
|
|
+
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (string.IsNullOrWhiteSpace(workIdKey)) throw Oops.Oh("未设置有效的机器码,启动失败");
|
|
|
+
|
|
|
+ // 开一个任务设置当前workId过期时间
|
|
|
+ Task.Run(() =>
|
|
|
+ {
|
|
|
+ while (true)
|
|
|
+ {
|
|
|
+ client.SetExpire(workIdKey, TimeSpan.FromSeconds(15));
|
|
|
+ //Task.Delay(5000);
|
|
|
+ Thread.Sleep(10000);
|
|
|
+ }
|
|
|
+ });
|
|
|
+ }
|
|
|
+ catch (Exception e)
|
|
|
+ {
|
|
|
+ throw Oops.Oh($"{e.Message};{e.StackTrace};{e.StackTrace}");
|
|
|
+ }
|
|
|
+ finally
|
|
|
+ {
|
|
|
+ redisLock?.Dispose();
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|