2024-12-26 16:42:05 +08:00
|
|
|
|
using Serein.Library.Api;
|
2024-10-20 12:10:57 +08:00
|
|
|
|
using System;
|
|
|
|
|
|
using System.Collections.Concurrent;
|
|
|
|
|
|
using System.Threading;
|
|
|
|
|
|
using System.Threading.Channels;
|
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
namespace Serein.Library.Utils
|
|
|
|
|
|
{
|
|
|
|
|
|
|
2025-07-30 21:15:07 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 基于 Channel 的触发器实现
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <typeparam name="TSignal"></typeparam>
|
2024-12-26 16:42:05 +08:00
|
|
|
|
public class ChannelFlowTrigger<TSignal> : IFlowTrigger<TSignal>
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
// 使用并发字典管理每个枚举信号对应的 Channel
|
2024-12-23 23:19:10 +08:00
|
|
|
|
private readonly ConcurrentDictionary<TSignal, Channel<TriggerResult<object>>> _channels = new ConcurrentDictionary<TSignal, Channel<TriggerResult<object>>>();
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2024-12-23 23:19:10 +08:00
|
|
|
|
/// 获取或创建指定信号的 Channel
|
2024-10-20 12:10:57 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="signal">枚举信号标识符</param>
|
2024-12-23 23:19:10 +08:00
|
|
|
|
/// <returns>对应的 Channel</returns>
|
|
|
|
|
|
private Channel<TriggerResult<object>> GetOrCreateChannel(TSignal signal)
|
|
|
|
|
|
{
|
2025-08-23 14:48:19 +08:00
|
|
|
|
return _channels.GetOrAdd(signal, _ => Channel.CreateUnbounded<TriggerResult<object>>());
|
2024-12-23 23:19:10 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-30 21:15:07 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 等待信号触发并指定超时时间
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <typeparam name="TResult"></typeparam>
|
|
|
|
|
|
/// <param name="signal"></param>
|
|
|
|
|
|
/// <param name="outTime"></param>
|
|
|
|
|
|
/// <returns></returns>
|
2024-12-23 23:19:10 +08:00
|
|
|
|
public async Task<TriggerResult<TResult>> WaitTriggerWithTimeoutAsync<TResult>(TSignal signal, TimeSpan outTime)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
var channel = GetOrCreateChannel(signal);
|
|
|
|
|
|
var cts = new CancellationTokenSource();
|
|
|
|
|
|
|
|
|
|
|
|
// 异步任务:超时后自动触发信号
|
|
|
|
|
|
_ = Task.Run(async () =>
|
|
|
|
|
|
{
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
await Task.Delay(outTime, cts.Token);
|
2024-12-23 23:19:10 +08:00
|
|
|
|
var outResult = new TriggerResult<object>()
|
|
|
|
|
|
{
|
|
|
|
|
|
Type = TriggerDescription.Overtime
|
|
|
|
|
|
};
|
|
|
|
|
|
await channel.Writer.WriteAsync(outResult);
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
catch (OperationCanceledException)
|
|
|
|
|
|
{
|
|
|
|
|
|
// 超时任务被取消
|
|
|
|
|
|
}
|
|
|
|
|
|
}, cts.Token);
|
|
|
|
|
|
|
|
|
|
|
|
// 等待信号传入(超时或手动触发)
|
2024-12-23 23:19:10 +08:00
|
|
|
|
var result = await WaitTriggerAsync<TResult>(signal); // 返回一个可以超时触发的等待任务
|
|
|
|
|
|
return result;
|
|
|
|
|
|
|
2024-10-20 12:10:57 +08:00
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-30 21:15:07 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 等待信号触发
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <typeparam name="TResult"></typeparam>
|
|
|
|
|
|
/// <param name="signal"></param>
|
|
|
|
|
|
/// <returns></returns>
|
2024-12-23 23:19:10 +08:00
|
|
|
|
public async Task<TriggerResult<TResult>> WaitTriggerAsync<TResult>(TSignal signal)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
var channel = GetOrCreateChannel(signal);
|
|
|
|
|
|
// 等待信号传入(超时或手动触发)
|
2024-12-23 23:19:10 +08:00
|
|
|
|
var result = await channel.Reader.ReadAsync();
|
|
|
|
|
|
if (result.Value is TResult data)
|
|
|
|
|
|
{
|
|
|
|
|
|
return new TriggerResult<TResult>()
|
|
|
|
|
|
{
|
|
|
|
|
|
Value = data,
|
|
|
|
|
|
Type = TriggerDescription.External,
|
|
|
|
|
|
};
|
|
|
|
|
|
}
|
|
|
|
|
|
else
|
|
|
|
|
|
{
|
|
|
|
|
|
return new TriggerResult<TResult>()
|
|
|
|
|
|
{
|
|
|
|
|
|
Type = TriggerDescription.TypeInconsistency,
|
|
|
|
|
|
};
|
|
|
|
|
|
}
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-30 21:15:07 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 调用触发器
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <typeparam name="TResult"></typeparam>
|
|
|
|
|
|
/// <param name="signal"></param>
|
|
|
|
|
|
/// <param name="value"></param>
|
|
|
|
|
|
/// <returns></returns>
|
2024-12-23 23:19:10 +08:00
|
|
|
|
public async Task<bool> InvokeTriggerAsync<TResult>(TSignal signal, TResult value)
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
if (_channels.TryGetValue(signal, out var channel))
|
|
|
|
|
|
{
|
|
|
|
|
|
// 手动触发信号
|
2024-12-23 23:19:10 +08:00
|
|
|
|
var result = new TriggerResult<object>()
|
|
|
|
|
|
{
|
|
|
|
|
|
Type = TriggerDescription.External,
|
|
|
|
|
|
Value = value
|
|
|
|
|
|
};
|
|
|
|
|
|
await channel.Writer.WriteAsync(result);
|
2024-10-20 12:10:57 +08:00
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-07-30 21:15:07 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 取消所有触发器
|
|
|
|
|
|
/// </summary>
|
2024-12-23 23:19:10 +08:00
|
|
|
|
public void CancelAllTrigger()
|
2024-10-20 12:10:57 +08:00
|
|
|
|
{
|
|
|
|
|
|
foreach (var channel in _channels.Values)
|
|
|
|
|
|
{
|
|
|
|
|
|
channel.Writer.Complete();
|
|
|
|
|
|
}
|
|
|
|
|
|
_channels.Clear();
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-10-27 00:54:10 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-10-20 12:10:57 +08:00
|
|
|
|
}
|