GFramework/GFramework.Core/Extensions/StoreEventBusExtensions.cs
gewuyou ff553977e3 chore(license): 补齐 Apache-2.0 文件头治理
- 新增许可证文件头检查与修复脚本

- 补充维护者手动修复 PR 工作流和 CI 校验

- 更新贡献指南中的文件头说明

- 补齐仓库维护源码和配置文件的许可证声明
2026-05-03 19:39:49 +08:00

143 lines
5.3 KiB
C#

// Copyright (c) 2025-2026 GeWuYou
// SPDX-License-Identifier: Apache-2.0
using GFramework.Core.Abstractions.Events;
using GFramework.Core.Abstractions.StateManagement;
using GFramework.Core.Events;
using GFramework.Core.StateManagement;
namespace GFramework.Core.Extensions;
/// <summary>
/// 为 Store 提供到 EventBus 的兼容桥接扩展。
/// 该扩展面向旧模块渐进迁移场景,使现有事件消费者可以继续观察 Store 的 action 分发和状态变化。
/// </summary>
public static class StoreEventBusExtensions
{
/// <summary>
/// 将 Store 的 dispatch 和状态变化同时桥接到 EventBus。
/// dispatch 事件会逐次发布;状态变化事件会复用 Store 自身的通知折叠语义,因此批处理中只发布最终状态。
/// </summary>
/// <typeparam name="TState">状态树的根状态类型。</typeparam>
/// <param name="store">源 Store。</param>
/// <param name="eventBus">目标事件总线。</param>
/// <param name="publishDispatches">是否发布每次 action 分发事件。</param>
/// <param name="publishStateChanges">是否发布状态变化事件。</param>
/// <returns>用于拆除桥接的句柄。</returns>
public static IUnRegister BridgeToEventBus<TState>(
this Store<TState> store,
IEventBus eventBus,
bool publishDispatches = true,
bool publishStateChanges = true)
{
if (store is null)
{
throw new ArgumentNullException(nameof(store));
}
if (eventBus is null)
{
throw new ArgumentNullException(nameof(eventBus));
}
IUnRegister? dispatchBridge = null;
IUnRegister? stateBridge = null;
if (publishDispatches)
{
dispatchBridge = store.BridgeDispatchesToEventBus(eventBus);
}
if (publishStateChanges)
{
stateBridge = store.BridgeStateChangesToEventBus(eventBus);
}
return new DefaultUnRegister(() =>
{
dispatchBridge?.UnRegister();
stateBridge?.UnRegister();
});
}
/// <summary>
/// 将 Store 的每次 dispatch 结果桥接到 EventBus。
/// 该桥接通过中间件实现,因此即使某次分发未改变状态,也会发布对应的 dispatch 事件。
/// </summary>
/// <typeparam name="TState">状态树的根状态类型。</typeparam>
/// <param name="store">源 Store。</param>
/// <param name="eventBus">目标事件总线。</param>
/// <returns>用于移除 dispatch 桥接中间件的句柄。</returns>
public static IUnRegister BridgeDispatchesToEventBus<TState>(this Store<TState> store, IEventBus eventBus)
{
if (store is null)
{
throw new ArgumentNullException(nameof(store));
}
if (eventBus is null)
{
throw new ArgumentNullException(nameof(eventBus));
}
return store.RegisterMiddleware(new DispatchEventBusMiddleware<TState>(eventBus));
}
/// <summary>
/// 将 Store 的状态变化桥接到 EventBus。
/// 该桥接复用 Store 的订阅通知语义,因此只会在状态真正变化时发布事件。
/// </summary>
/// <typeparam name="TState">状态树的根状态类型。</typeparam>
/// <param name="store">源 Store。</param>
/// <param name="eventBus">目标事件总线。</param>
/// <returns>用于移除状态变化桥接的句柄。</returns>
public static IUnRegister BridgeStateChangesToEventBus<TState>(this IReadonlyStore<TState> store,
IEventBus eventBus)
{
if (store is null)
{
throw new ArgumentNullException(nameof(store));
}
if (eventBus is null)
{
throw new ArgumentNullException(nameof(eventBus));
}
return store.Subscribe(state =>
eventBus.Send(new StoreStateChangedEvent<TState>(state, DateTimeOffset.UtcNow)));
}
/// <summary>
/// 用于把 dispatch 结果桥接到 EventBus 的内部中间件。
/// 选择中间件而不是改写 Store 核心提交流程,是为了把兼容层成本保持在可选扩展中。
/// </summary>
/// <typeparam name="TState">状态树的根状态类型。</typeparam>
private sealed class DispatchEventBusMiddleware<TState>(IEventBus eventBus) : IStoreMiddleware<TState>
{
/// <summary>
/// 目标事件总线。
/// </summary>
private readonly IEventBus _eventBus = eventBus ?? throw new ArgumentNullException(nameof(eventBus));
/// <summary>
/// 执行后续 dispatch 管线,并在结束后把分发结果发送到 EventBus。
/// </summary>
/// <param name="context">当前分发上下文。</param>
/// <param name="next">后续管线。</param>
public void Invoke(StoreDispatchContext<TState> context, Action next)
{
next();
var dispatchRecord = new StoreDispatchRecord<TState>(
context.Action,
context.PreviousState,
context.NextState,
context.HasStateChanged,
context.DispatchedAt);
_eventBus.Send(new StoreDispatchedEvent<TState>(dispatchRecord));
}
}
}