2
3 - 添加 CreateTriggerWithMenuDto 数据传输对象,用于同时创建触发器及关联菜单
4 - 在 TriggerDataService 中新增 AddTriggerWithMenu 方法,实现触发器与菜单的同时创建
5 - 更新 TriggersViewModel 以使用新的触发器和菜单创建流程
6 - 在 MenuType 枚举中添加 TriggerMenu 类型
7 - 调整 InitializeRepository 中触发器菜单的图标
8 - 更新相关服务中的注释,将 Trigger 替换为 TriggerMenu 以保持一致
9 - 修改时间记录方式,使用 DateTime.Now 替代 DateTime.UtcNow
10 - 优化 TriggerManagementService 中的触发器创建与存储流程
11 - 更新触发器评估和管理服务中的日志文本,统一使用 TriggerMenu 术语
149 lines
6.9 KiB
C#
149 lines
6.9 KiB
C#
using System;
|
||
using System.Collections.Concurrent;
|
||
using System.Collections.Generic;
|
||
using System.Linq;
|
||
using System.Threading.Tasks;
|
||
using DMS.Application.DTOs;
|
||
using DMS.Application.Interfaces.Management;
|
||
using DMS.Application.Services.Management;
|
||
// 明确指定 Timer 类型,避免歧义
|
||
using ThreadingTimer = System.Threading.Timer;
|
||
using TimersTimer = System.Timers.Timer;
|
||
using DMS.Application.Services.Triggers;
|
||
using DMS.Core.Models.Triggers;
|
||
using Microsoft.Extensions.Logging; // 使用 Microsoft.Extensions.Logging.ILogger
|
||
|
||
namespace DMS.Application.Services.Triggers.Impl
|
||
{
|
||
/// <summary>
|
||
/// 触发器评估服务实现
|
||
/// </summary>
|
||
public class TriggerEvaluationService : ITriggerEvaluationService, IDisposable
|
||
{
|
||
private readonly ITriggerManagementService _triggerManagementService;
|
||
// 移除了 IVariableAppService 依赖
|
||
private readonly ITriggerActionExecutor _actionExecutor;
|
||
private readonly ILogger<TriggerEvaluationService> _logger; // 使用标准日志接口
|
||
// 为每个触发器存储抑制定时器
|
||
private readonly ConcurrentDictionary<int, ThreadingTimer> _suppressionTimers = new();
|
||
|
||
public TriggerEvaluationService(
|
||
ITriggerManagementService triggerManagementService,
|
||
// IVariableAppService variableAppService, // 移除此参数
|
||
ITriggerActionExecutor actionExecutor,
|
||
ILogger<TriggerEvaluationService> logger) // 使用标准日志接口
|
||
{
|
||
_triggerManagementService = triggerManagementService ?? throw new ArgumentNullException(nameof(triggerManagementService));
|
||
// _variableAppService = variableAppService ?? throw new ArgumentNullException(nameof(variableAppService));
|
||
_actionExecutor = actionExecutor ?? throw new ArgumentNullException(nameof(actionExecutor));
|
||
_logger = logger ?? throw new ArgumentNullException(nameof(logger));
|
||
}
|
||
|
||
/// <summary>
|
||
/// 评估与指定变量关联的所有激活状态的触发器
|
||
/// </summary>
|
||
public async Task EvaluateTriggersAsync(int variableId, object currentValue)
|
||
{
|
||
try
|
||
{
|
||
var triggers = await _triggerManagementService.GetTriggersForVariableAsync(variableId);
|
||
// 注意:这里不再通过 _variableAppService 获取 Variable,
|
||
// 如果需要 Variable 信息,可以在 ExecuteActionAsync 的 TriggerContext 中携带。
|
||
// 创建一个临时的上下文对象,其中 Variable 可以为 null,
|
||
// 在实际应用中,你可能需要通过某种方式获取 Variable。
|
||
|
||
_logger.LogDebug($"Evaluating {triggers.Count(t => t.IsActive)} active triggers for variable ID: {variableId}");
|
||
|
||
foreach (var trigger in triggers.Where(t => t.IsActive))
|
||
{
|
||
if (!IsWithinSuppressionWindow(trigger)) // Check suppression first
|
||
{
|
||
if (EvaluateCondition(trigger, currentValue))
|
||
{
|
||
|
||
var context = new TriggerContext(trigger, currentValue, null);
|
||
|
||
await _actionExecutor.ExecuteActionAsync(context);
|
||
|
||
// Update last triggered time and start suppression timer if needed
|
||
trigger.LastTriggeredAt = DateTime.UtcNow;
|
||
// For simplicity, we'll assume it's updated periodically or on next load.
|
||
// In a production scenario, you'd likely want to persist this back to the database.
|
||
|
||
// Start suppression timer if duration is set (in-memory suppression)
|
||
if (trigger.SuppressionDuration.HasValue)
|
||
{
|
||
// 使用 ThreadingTimer 避免歧义
|
||
var timer = new ThreadingTimer(_ =>
|
||
{
|
||
trigger.LastTriggeredAt = null; // Reset suppression flag after delay
|
||
_logger.LogInformation($"Suppression lifted for trigger {trigger.Id}");
|
||
// Note: Modifying 'trigger' directly affects the object in the list returned by GetTriggersForVariableAsync().
|
||
// This works for in-memory state but won't persist changes. Consider updating DB explicitly if needed.
|
||
}, null, trigger.SuppressionDuration.Value, Timeout.InfiniteTimeSpan); // Single shot timer
|
||
|
||
// Replace any existing timer for this trigger ID
|
||
_suppressionTimers.AddOrUpdate(trigger.Id, timer, (key, oldTimer) => {
|
||
oldTimer?.Dispose();
|
||
return timer;
|
||
});
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
catch (Exception ex)
|
||
{
|
||
_logger.LogError(ex, "An error occurred while evaluating triggers for variable ID: {VariableId}", variableId);
|
||
}
|
||
}
|
||
|
||
/// <summary>
|
||
/// 内部方法:评估单个触发器的条件
|
||
/// </summary>
|
||
private bool EvaluateCondition(Trigger trigger, object currentValueObj)
|
||
{
|
||
if (currentValueObj == null)
|
||
{
|
||
_logger.LogWarning("Cannot evaluate trigger condition: Current value is null for trigger ID: {TriggerId}", trigger.Id);
|
||
return false; // Cannot evaluate null
|
||
}
|
||
|
||
// 由于移除了条件,所有激活的触发器都会被触发
|
||
_logger.LogInformation("TriggerMenu activated for trigger ID: {TriggerId}", trigger.Id);
|
||
|
||
return true;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 内部方法:检查触发器是否处于抑制窗口期内
|
||
/// </summary>
|
||
private bool IsWithinSuppressionWindow(Trigger trigger)
|
||
{
|
||
if (!trigger.SuppressionDuration.HasValue || !trigger.LastTriggeredAt.HasValue)
|
||
return false;
|
||
|
||
var suppressionEndTime = trigger.LastTriggeredAt.Value.Add(trigger.SuppressionDuration.Value);
|
||
bool isSuppressed = DateTime.UtcNow < suppressionEndTime;
|
||
|
||
if(isSuppressed)
|
||
{
|
||
_logger.LogTrace("TriggerMenu is suppressed (until {SuppressionEnd}) for trigger ID: {TriggerId}", suppressionEndTime, trigger.Id);
|
||
}
|
||
|
||
return isSuppressed;
|
||
}
|
||
|
||
/// <summary>
|
||
/// 实现 IDisposable 以清理计时器资源
|
||
/// </summary>
|
||
public void Dispose()
|
||
{
|
||
foreach (var kvp in _suppressionTimers)
|
||
{
|
||
kvp.Value?.Dispose();
|
||
}
|
||
_suppressionTimers.Clear();
|
||
}
|
||
}
|
||
} |