// 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.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; namespace GFramework.Cqrs.Benchmarks.Messaging; /// /// 对比单处理器 notification 在 GFramework.CQRS 与 MediatR 之间的 publish 开销。 /// [Config(typeof(Config))] public class NotificationBenchmarks { private MicrosoftDiContainer _container = null!; private ICqrsRuntime _runtime = null!; private ServiceProvider _serviceProvider = null!; private IPublisher _publisher = null!; private BenchmarkNotification _notification = null!; /// /// 配置 notification benchmark 的公共输出格式。 /// private sealed class Config : ManualConfig { public Config() { AddJob(Job.Default); AddColumnProvider(DefaultColumnProviders.Instance); AddColumn(new CustomColumn("Scenario", static (_, _) => "Notification")); AddDiagnoser(MemoryDiagnoser.Default); WithOrderer(new DefaultOrderer(SummaryOrderPolicy.FastestToSlowest, MethodOrderPolicy.Declared)); } } /// /// 构建 notification publish 所需的最小 runtime 宿主和对照对象。 /// [GlobalSetup] public void Setup() { LoggerFactoryResolver.Provider = new ConsoleLoggerFactoryProvider { MinLevel = LogLevel.Fatal }; Fixture.Setup("Notification", handlerCount: 1, pipelineCount: 0); _container = new MicrosoftDiContainer(); _container.RegisterTransient, BenchmarkNotificationHandler>(); _runtime = GFramework.Cqrs.CqrsRuntimeFactory.CreateRuntime( _container, LoggerFactoryResolver.Provider.CreateLogger(nameof(NotificationBenchmarks))); var services = new ServiceCollection(); services.AddSingleton, BenchmarkNotificationHandler>(); services.AddMediatR(static options => options.RegisterServicesFromAssembly(typeof(NotificationBenchmarks).Assembly)); _serviceProvider = services.BuildServiceProvider(); _publisher = _serviceProvider.GetRequiredService(); _notification = new BenchmarkNotification(Guid.NewGuid()); } /// /// 释放 MediatR 对照组使用的 DI 宿主。 /// [GlobalCleanup] public void Cleanup() { _serviceProvider.Dispose(); } /// /// 通过 GFramework.CQRS runtime 发布 notification。 /// [Benchmark(Baseline = true)] public ValueTask PublishNotification_GFrameworkCqrs() { return _runtime.PublishAsync(BenchmarkContext.Instance, _notification, CancellationToken.None); } /// /// 通过 MediatR 发布 notification,作为外部设计对照。 /// [Benchmark] public Task PublishNotification_MediatR() { return _publisher.Publish(_notification, CancellationToken.None); } /// /// Benchmark notification。 /// /// 通知标识。 public sealed record BenchmarkNotification(Guid Id) : GFramework.Cqrs.Abstractions.Cqrs.INotification, MediatR.INotification; /// /// 同时实现 GFramework.CQRS 与 MediatR 契约的最小 notification handler。 /// public sealed class BenchmarkNotificationHandler : GFramework.Cqrs.Abstractions.Cqrs.INotificationHandler, MediatR.INotificationHandler { /// /// 处理 GFramework.CQRS notification。 /// public ValueTask Handle(BenchmarkNotification notification, CancellationToken cancellationToken) { return ValueTask.CompletedTask; } /// /// 处理 MediatR notification。 /// Task MediatR.INotificationHandler.Handle( BenchmarkNotification notification, CancellationToken cancellationToken) { return Task.CompletedTask; } } }