// Copyright (c) 2025-2026 GeWuYou // SPDX-License-Identifier: Apache-2.0 using BenchmarkDotNet.Attributes; using BenchmarkDotNet.Columns; using BenchmarkDotNet.Configs; using BenchmarkDotNet.Diagnosers; using BenchmarkDotNet.Jobs; using BenchmarkDotNet.Order; using System; using System.Collections.Generic; using System.Threading; using System.Threading.Tasks; using GFramework.Core.Abstractions.Logging; using GFramework.Core.Ioc; using GFramework.Core.Logging; using GFramework.Cqrs.Abstractions.Cqrs; using MediatR; using Microsoft.Extensions.DependencyInjection; [assembly: GFramework.Cqrs.CqrsHandlerRegistryAttribute( typeof(GFramework.Cqrs.Benchmarks.Messaging.GeneratedRequestInvokerBenchmarkRegistry))] namespace GFramework.Cqrs.Benchmarks.Messaging; /// /// 对比 request steady-state dispatch 在 direct handler、GFramework 反射路径、GFramework generated invoker 路径与 MediatR 之间的开销差异。 /// [Config(typeof(Config))] public class RequestInvokerBenchmarks { private MicrosoftDiContainer _reflectionContainer = null!; private ICqrsRuntime _reflectionRuntime = null!; private MicrosoftDiContainer _generatedContainer = null!; private ICqrsRuntime _generatedRuntime = null!; private ServiceProvider _serviceProvider = null!; private IMediator _mediatr = null!; private ReflectionBenchmarkRequestHandler _baselineHandler = null!; private ReflectionBenchmarkRequest _reflectionRequest = null!; private GeneratedBenchmarkRequest _generatedRequest = null!; private MediatRBenchmarkRequest _mediatrRequest = null!; /// /// 配置 request invoker benchmark 的公共输出格式。 /// private sealed class Config : ManualConfig { public Config() { AddJob(Job.Default); AddColumnProvider(DefaultColumnProviders.Instance); AddColumn(new CustomColumn("Scenario", static (_, _) => "RequestInvoker")); AddDiagnoser(MemoryDiagnoser.Default); WithOrderer(new DefaultOrderer(SummaryOrderPolicy.FastestToSlowest, MethodOrderPolicy.Declared)); } } /// /// 构建 reflection / generated / MediatR 三组 request dispatch 对照宿主。 /// [GlobalSetup] public void Setup() { LoggerFactoryResolver.Provider = new ConsoleLoggerFactoryProvider { MinLevel = LogLevel.Fatal }; Fixture.Setup("RequestInvoker", handlerCount: 1, pipelineCount: 0); BenchmarkDispatcherCacheHelper.ClearDispatcherCaches(); _baselineHandler = new ReflectionBenchmarkRequestHandler(); _reflectionRequest = new ReflectionBenchmarkRequest(Guid.NewGuid()); _generatedRequest = new GeneratedBenchmarkRequest(Guid.NewGuid()); _mediatrRequest = new MediatRBenchmarkRequest(Guid.NewGuid()); _reflectionContainer = new MicrosoftDiContainer(); _reflectionContainer.RegisterTransient, ReflectionBenchmarkRequestHandler>(); _reflectionRuntime = GFramework.Cqrs.CqrsRuntimeFactory.CreateRuntime( _reflectionContainer, LoggerFactoryResolver.Provider.CreateLogger(nameof(RequestInvokerBenchmarks) + ".Reflection")); _generatedContainer = new MicrosoftDiContainer(); _generatedContainer.RegisterCqrsHandlersFromAssembly(typeof(RequestInvokerBenchmarks).Assembly); _generatedRuntime = GFramework.Cqrs.CqrsRuntimeFactory.CreateRuntime( _generatedContainer, LoggerFactoryResolver.Provider.CreateLogger(nameof(RequestInvokerBenchmarks) + ".Generated")); var services = new ServiceCollection(); services.AddLogging(static builder => Microsoft.Extensions.Logging.FilterLoggingBuilderExtensions.AddFilter( builder, "LuckyPennySoftware.MediatR.License", Microsoft.Extensions.Logging.LogLevel.None)); services.AddSingleton, MediatRBenchmarkRequestHandler>(); services.AddMediatR(static options => options.RegisterServicesFromAssembly(typeof(RequestInvokerBenchmarks).Assembly)); _serviceProvider = services.BuildServiceProvider(); _mediatr = _serviceProvider.GetRequiredService(); } /// /// 释放 MediatR 对照组使用的 DI 宿主,并清理静态 dispatcher 缓存。 /// [GlobalCleanup] public void Cleanup() { _serviceProvider.Dispose(); BenchmarkDispatcherCacheHelper.ClearDispatcherCaches(); } /// /// 直接调用最小 request handler,作为 dispatch 额外开销 baseline。 /// [Benchmark(Baseline = true)] public ValueTask SendRequest_Baseline() { return _baselineHandler.Handle(_reflectionRequest, CancellationToken.None); } /// /// 通过 GFramework.CQRS 反射 request binding 路径发送 request。 /// [Benchmark] public ValueTask SendRequest_GFrameworkReflection() { return _reflectionRuntime.SendAsync(BenchmarkContext.Instance, _reflectionRequest, CancellationToken.None); } /// /// 通过 generated request invoker provider 预热后的 GFramework.CQRS runtime 发送 request。 /// [Benchmark] public ValueTask SendRequest_GFrameworkGenerated() { return _generatedRuntime.SendAsync(BenchmarkContext.Instance, _generatedRequest, CancellationToken.None); } /// /// 通过 MediatR 发送 request,作为外部对照。 /// [Benchmark] public Task SendRequest_MediatR() { return _mediatr.Send(_mediatrRequest, CancellationToken.None); } /// /// Reflection runtime request。 /// /// 请求标识。 public sealed record ReflectionBenchmarkRequest(Guid Id) : GFramework.Cqrs.Abstractions.Cqrs.IRequest; /// /// Reflection runtime response。 /// /// 响应标识。 public sealed record ReflectionBenchmarkResponse(Guid Id); /// /// Generated runtime request。 /// /// 请求标识。 public sealed record GeneratedBenchmarkRequest(Guid Id) : GFramework.Cqrs.Abstractions.Cqrs.IRequest; /// /// Generated runtime response。 /// /// 响应标识。 public sealed record GeneratedBenchmarkResponse(Guid Id); /// /// MediatR request。 /// /// 请求标识。 public sealed record MediatRBenchmarkRequest(Guid Id) : MediatR.IRequest; /// /// MediatR response。 /// /// 响应标识。 public sealed record MediatRBenchmarkResponse(Guid Id); /// /// Reflection runtime 的最小 request handler。 /// public sealed class ReflectionBenchmarkRequestHandler : GFramework.Cqrs.Abstractions.Cqrs.IRequestHandler { /// /// 处理 reflection benchmark request。 /// public ValueTask Handle( ReflectionBenchmarkRequest request, CancellationToken cancellationToken) { return ValueTask.FromResult(new ReflectionBenchmarkResponse(request.Id)); } } /// /// Generated runtime 的最小 request handler。 /// public sealed class GeneratedBenchmarkRequestHandler : GFramework.Cqrs.Abstractions.Cqrs.IRequestHandler { /// /// 处理 generated benchmark request。 /// public ValueTask Handle( GeneratedBenchmarkRequest request, CancellationToken cancellationToken) { return ValueTask.FromResult(new GeneratedBenchmarkResponse(request.Id)); } } /// /// MediatR 对照组的最小 request handler。 /// public sealed class MediatRBenchmarkRequestHandler : MediatR.IRequestHandler { /// /// 处理 MediatR benchmark request。 /// public Task Handle( MediatRBenchmarkRequest request, CancellationToken cancellationToken) { return Task.FromResult(new MediatRBenchmarkResponse(request.Id)); } } }