// Copyright (c) 2025-2026 GeWuYou
// SPDX-License-Identifier: Apache-2.0
using GFramework.Core.Abstractions.Query;
using GFramework.Core.Abstractions.Rule;
using GFramework.Core.Cqrs;
using GFramework.Cqrs.Abstractions.Cqrs;
namespace GFramework.Core.Query;
///
/// QueryExecutor 类负责执行查询操作,实现 IQueryExecutor 接口。
/// 该类是密封的,防止被继承。
///
public sealed class QueryExecutor(ICqrsRuntime? runtime = null) : IQueryExecutor
{
private readonly ICqrsRuntime? _runtime = runtime;
///
/// 获取当前执行器是否已接入统一 CQRS runtime。
///
public bool UsesCqrsRuntime => _runtime is not null;
///
/// 执行指定的查询并返回结果。
/// 当查询对象携带可用的架构上下文且执行器已接入统一 runtime 时,
/// 该方法会先把 legacy 查询包装成内部 request 并交给 ,
/// 以复用统一的 dispatch / pipeline 入口;否则回退到 legacy 直接执行。
///
/// 查询结果的类型。
/// 要执行的查询对象,必须实现 IQuery<TResult> 接口。
/// 查询执行的结果,类型为 TResult。
public TResult Send(IQuery query)
{
ArgumentNullException.ThrowIfNull(query);
if (TryResolveDispatchContext(query, out var context) && _runtime is not null)
{
var boxedResult = _runtime.SendAsync(
context,
new LegacyQueryDispatchRequest(
query,
() => query.Do()))
.AsTask()
.GetAwaiter()
.GetResult();
return (TResult)boxedResult!;
}
return query.Do();
}
///
/// 解析当前 legacy 查询应该绑定到哪个架构上下文。
///
/// 即将执行的 legacy 查询对象。
/// 命中时返回可用于 CQRS runtime 的架构上下文。
/// 如果既接入了 runtime 且查询对象提供了上下文,则返回 。
private bool TryResolveDispatchContext(object query, out GFramework.Core.Abstractions.Architectures.IArchitectureContext context)
{
context = null!;
if (_runtime is null || query is not IContextAware contextAware)
{
return false;
}
try
{
context = contextAware.GetContext();
return true;
}
catch (InvalidOperationException)
{
return false;
}
}
}