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

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

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

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

222 lines
8.2 KiB
C#

// Copyright (c) 2025-2026 GeWuYou
// SPDX-License-Identifier: Apache-2.0
using System;
using System.Collections.Generic;
using System.Threading.Tasks;
using GFramework.Core.Abstractions.Architectures;
using GFramework.Core.Abstractions.Command;
using GFramework.Core.Abstractions.Enums;
using GFramework.Core.Abstractions.Query;
namespace GFramework.Core.Tests.Architectures;
/// <summary>
/// 覆盖测试架构上下文替身的共享事件与显式失败契约。
/// </summary>
[TestFixture]
public class TestArchitectureContextBehaviorTests
{
private static IEnumerable<TestCaseData> EventContextFactories()
{
yield return new TestCaseData(
new Func<IArchitectureContext>(() => new TestArchitectureContext()))
.SetArgDisplayNames(nameof(TestArchitectureContext));
yield return new TestCaseData(
new Func<IArchitectureContext>(() => new TestArchitectureContextV3()))
.SetArgDisplayNames(nameof(TestArchitectureContextV3));
}
/// <summary>
/// 验证测试上下文会把事件注册与发送委托到同一个事件总线实例。
/// </summary>
[Test]
public void RegisterEvent_And_SendEvent_Should_Use_Shared_EventBus()
{
var context = new TestArchitectureContext();
var eventReceived = false;
context.RegisterEvent<TestEventV2>(_ => eventReceived = true);
context.SendEvent<TestEventV2>();
Assert.That(eventReceived, Is.True);
}
/// <summary>
/// 验证用于 ArchitectureServices 的上下文替身也会把事件注册与发送委托到同一个事件总线实例。
/// </summary>
[Test]
public void RegisterEvent_And_SendEvent_On_TestArchitectureContextV3_Should_Use_Shared_EventBus()
{
var context = new TestArchitectureContextV3();
var eventReceived = false;
context.RegisterEvent<TestEventV2>(_ => eventReceived = true);
context.SendEvent<TestEventV2>();
Assert.That(eventReceived, Is.True);
}
/// <summary>
/// 验证两类架构测试替身都会对事件 API 的空参数执行显式校验。
/// </summary>
/// <param name="createContext">创建待验证测试上下文的工厂。</param>
[TestCaseSource(nameof(EventContextFactories))]
public void Event_APIs_Should_Validate_Null_Arguments(Func<IArchitectureContext> createContext)
{
var context = createContext();
Assert.That(() => context.SendEvent<TestEventV2>(null!), Throws.TypeOf<ArgumentNullException>());
Assert.That(() => context.RegisterEvent<TestEventV2>(null!), Throws.TypeOf<ArgumentNullException>());
Assert.That(() => context.UnRegisterEvent<TestEventV2>(null!), Throws.TypeOf<ArgumentNullException>());
}
/// <summary>
/// 验证两类架构测试替身在注销事件处理器后都不会继续分发同一回调。
/// </summary>
/// <param name="createContext">创建待验证测试上下文的工厂。</param>
[TestCaseSource(nameof(EventContextFactories))]
public void UnRegisterEvent_Should_Stop_Dispatch(Func<IArchitectureContext> createContext)
{
var context = createContext();
var eventReceived = false;
void Handler(TestEventV2 _)
{
eventReceived = true;
}
context.RegisterEvent<TestEventV2>(Handler);
context.UnRegisterEvent<TestEventV2>(Handler);
context.SendEvent<TestEventV2>();
Assert.That(eventReceived, Is.False);
}
/// <summary>
/// 验证测试上下文的旧版命令与查询入口会显式抛出未支持异常。
/// </summary>
[Test]
public async Task Legacy_Entries_Should_Throw_Or_Return_Faulted_Tasks()
{
var context = new TestArchitectureContext();
Assert.That(() => context.SendCommand(new TestCommandV2()), Throws.TypeOf<NotSupportedException>());
Assert.That(
() => context.SendCommand(new TestCommandWithResultV2 { Result = 1 }),
Throws.TypeOf<NotSupportedException>());
Assert.That(() => context.SendQuery(new TestQueryV2 { Result = 1 }), Throws.TypeOf<NotSupportedException>());
Assert.That(
async () => await context.SendCommandAsync(new TestAsyncCommand()).ConfigureAwait(false),
Throws.TypeOf<NotSupportedException>());
Assert.That(
async () => await context.SendCommandAsync(new TestAsyncCommandWithResult()).ConfigureAwait(false),
Throws.TypeOf<NotSupportedException>());
Assert.That(
async () => await context.SendQueryAsync(new TestAsyncQuery()).ConfigureAwait(false),
Throws.TypeOf<NotSupportedException>());
}
/// <summary>
/// 验证用于 ArchitectureServices 的上下文替身也会把旧版入口显式标记为不支持。
/// </summary>
[Test]
public async Task Legacy_Entries_On_TestArchitectureContextV3_Should_Throw_Or_Return_Faulted_Tasks()
{
var context = new TestArchitectureContextV3();
Assert.That(() => context.SendCommand(new TestCommandV2()), Throws.TypeOf<NotSupportedException>());
Assert.That(
() => context.SendCommand(new TestCommandWithResultV2 { Result = 1 }),
Throws.TypeOf<NotSupportedException>());
Assert.That(() => context.SendQuery(new TestQueryV2 { Result = 1 }), Throws.TypeOf<NotSupportedException>());
Assert.That(
async () => await context.SendCommandAsync(new TestAsyncCommand()).ConfigureAwait(false),
Throws.TypeOf<NotSupportedException>());
Assert.That(
async () => await context.SendCommandAsync(new TestAsyncCommandWithResult()).ConfigureAwait(false),
Throws.TypeOf<NotSupportedException>());
Assert.That(
async () => await context.SendQueryAsync(new TestAsyncQuery()).ConfigureAwait(false),
Throws.TypeOf<NotSupportedException>());
}
/// <summary>
/// 验证两类架构测试替身在接口视角下都会以 no-op 方式接受生命周期钩子。
/// </summary>
[Test]
public void RegisterLifecycleHook_Via_Interface_Should_Return_Original_Hook()
{
IArchitecture withRegistry = new TestArchitectureWithRegistry(new TestRegistry());
IArchitecture withoutRegistry = new TestArchitectureWithoutRegistry();
var hook = new NoOpLifecycleHook();
Assert.That(withRegistry.RegisterLifecycleHook(hook), Is.SameAs(hook));
Assert.That(withoutRegistry.RegisterLifecycleHook(hook), Is.SameAs(hook));
}
/// <summary>
/// 为旧版异步命令入口提供最小实现的测试命令。
/// </summary>
private sealed class TestAsyncCommand : IAsyncCommand
{
public Task ExecuteAsync()
{
return Task.CompletedTask;
}
public IArchitectureContext GetContext()
{
throw new NotSupportedException();
}
public void SetContext(IArchitectureContext context)
{
ArgumentNullException.ThrowIfNull(context);
}
}
/// <summary>
/// 为旧版异步命令入口提供最小实现的带结果测试命令。
/// </summary>
private sealed class TestAsyncCommandWithResult : IAsyncCommand<int>
{
public Task<int> ExecuteAsync()
{
return Task.FromResult(1);
}
public IArchitectureContext GetContext()
{
throw new NotSupportedException();
}
public void SetContext(IArchitectureContext context)
{
ArgumentNullException.ThrowIfNull(context);
}
}
/// <summary>
/// 为旧版异步查询入口提供最小实现的测试查询。
/// </summary>
private sealed class TestAsyncQuery : IAsyncQuery<int>
{
public Task<int> DoAsync()
{
return Task.FromResult(1);
}
}
/// <summary>
/// 为生命周期钩子接口提供空实现的测试替身。
/// </summary>
private sealed class NoOpLifecycleHook : IArchitectureLifecycleHook
{
public void OnPhase(ArchitecturePhase phase, IArchitecture architecture)
{
ArgumentNullException.ThrowIfNull(architecture);
}
}
}