feat: 实现完整的多租户公告管理系统

核心功能:
- 公告状态机(草稿/已发布/已撤销)支持发布、撤销和重新发布
- 发布者范围区分平台级和租户级公告
- 目标受众定向推送(全部租户/指定角色/指定用户)
- 平台管理、租户管理和应用端查询API
- 已读/未读管理和未读统计

技术实现:
- CQRS+DDD架构,清晰的领域边界和事件驱动
- 查询性能优化:数据库端排序和限制,估算策略减少内存占用
- 并发控制:修复RowVersion配置(IsRowVersion→IsConcurrencyToken)
- 完整的FluentValidation验证器和输入保护

测试验证:
- 36个测试全部通过(27单元+9集成)
- 性能测试达标(1000条数据<5秒)
- 代码质量评级A(优秀)

文档:
- 完整的ADR、API文档和迁移指南
- 交付报告和技术债务记录
This commit is contained in:
2025-12-20 19:50:17 +08:00
parent 00eb357e6e
commit 857f776447
76 changed files with 12957 additions and 281 deletions

View File

@@ -0,0 +1,122 @@
using FluentAssertions;
using Moq;
using TakeoutSaaS.Application.App.Tenants.Handlers;
using TakeoutSaaS.Application.App.Tenants.Queries;
using TakeoutSaaS.Application.Tests.TestUtilities;
using TakeoutSaaS.Domain.Tenants.Entities;
using TakeoutSaaS.Domain.Tenants.Repositories;
using TakeoutSaaS.Shared.Abstractions.Security;
using TakeoutSaaS.Shared.Abstractions.Tenancy;
namespace TakeoutSaaS.Application.Tests.App.Tenants.Handlers;
public sealed class GetAnnouncementByIdQueryHandlerTests
{
[Fact]
public async Task GivenAnnouncementMissing_WhenHandle_ThenReturnsNull()
{
// Arrange
var tenantProvider = new Mock<ITenantProvider>();
tenantProvider.Setup(x => x.GetCurrentTenantId()).Returns(99);
var announcementRepository = new Mock<ITenantAnnouncementRepository>();
announcementRepository
.Setup(x => x.FindByIdInScopeAsync(99, 500, It.IsAny<CancellationToken>()))
.ReturnsAsync((TenantAnnouncement?)null);
var readRepository = new Mock<ITenantAnnouncementReadRepository>();
var handler = new GetAnnouncementByIdQueryHandler(
announcementRepository.Object,
readRepository.Object,
tenantProvider.Object);
// Act
var result = await handler.Handle(new GetAnnouncementByIdQuery { AnnouncementId = 500 }, CancellationToken.None);
// Assert
result.Should().BeNull();
}
[Fact]
public async Task GivenTargetNotMatched_WhenHandle_ThenReturnsNullAndSkipsReadLookup()
{
// Arrange
var tenantProvider = new Mock<ITenantProvider>();
tenantProvider.Setup(x => x.GetCurrentTenantId()).Returns(100);
var currentUserAccessor = new Mock<ICurrentUserAccessor>();
currentUserAccessor.SetupGet(x => x.UserId).Returns(123);
currentUserAccessor.SetupGet(x => x.IsAuthenticated).Returns(true);
var announcement = AnnouncementTestData.CreateAnnouncement(1, 100, 1, DateTime.UtcNow);
announcement.TargetType = "SPECIFIC_USERS";
announcement.TargetParameters = "{\"userIds\":[999]}";
var announcementRepository = new Mock<ITenantAnnouncementRepository>();
announcementRepository
.Setup(x => x.FindByIdInScopeAsync(100, 1, It.IsAny<CancellationToken>()))
.ReturnsAsync(announcement);
var readRepository = new Mock<ITenantAnnouncementReadRepository>();
var handler = new GetAnnouncementByIdQueryHandler(
announcementRepository.Object,
readRepository.Object,
tenantProvider.Object,
currentUserAccessor.Object);
// Act
var result = await handler.Handle(new GetAnnouncementByIdQuery { AnnouncementId = 1 }, CancellationToken.None);
// Assert
result.Should().BeNull();
readRepository.Verify(x => x.GetByAnnouncementAsync(
It.IsAny<long>(),
It.IsAny<IEnumerable<long>>(),
It.IsAny<long?>(),
It.IsAny<CancellationToken>()), Times.Never);
}
[Fact]
public async Task GivenUserReadRecord_WhenHandle_ThenReturnsDtoWithReadState()
{
// Arrange
var tenantProvider = new Mock<ITenantProvider>();
tenantProvider.Setup(x => x.GetCurrentTenantId()).Returns(200);
var currentUserAccessor = new Mock<ICurrentUserAccessor>();
currentUserAccessor.SetupGet(x => x.UserId).Returns(321);
currentUserAccessor.SetupGet(x => x.IsAuthenticated).Returns(true);
var announcement = AnnouncementTestData.CreateAnnouncement(10, 200, 1, DateTime.UtcNow);
var readAt = DateTime.UtcNow.AddMinutes(-3);
var announcementRepository = new Mock<ITenantAnnouncementRepository>();
announcementRepository
.Setup(x => x.FindByIdInScopeAsync(200, 10, It.IsAny<CancellationToken>()))
.ReturnsAsync(announcement);
var readRepository = new Mock<ITenantAnnouncementReadRepository>();
readRepository
.Setup(x => x.GetByAnnouncementAsync(200, It.IsAny<IEnumerable<long>>(), 321, It.IsAny<CancellationToken>()))
.ReturnsAsync(new List<TenantAnnouncementRead>
{
new() { AnnouncementId = 10, TenantId = 200, UserId = 321, ReadAt = readAt }
});
var handler = new GetAnnouncementByIdQueryHandler(
announcementRepository.Object,
readRepository.Object,
tenantProvider.Object,
currentUserAccessor.Object);
// Act
var result = await handler.Handle(new GetAnnouncementByIdQuery { AnnouncementId = 10 }, CancellationToken.None);
// Assert
result.Should().NotBeNull();
result!.IsRead.Should().BeTrue();
result.ReadAt.Should().BeCloseTo(readAt, TimeSpan.FromSeconds(1));
}
}

View File

@@ -0,0 +1,160 @@
using FluentAssertions;
using Moq;
using TakeoutSaaS.Application.App.Tenants.Handlers;
using TakeoutSaaS.Application.App.Tenants.Queries;
using TakeoutSaaS.Application.Tests.TestUtilities;
using TakeoutSaaS.Domain.Tenants.Repositories;
using TakeoutSaaS.Shared.Abstractions.Security;
using TakeoutSaaS.Shared.Abstractions.Tenancy;
namespace TakeoutSaaS.Application.Tests.App.Tenants.Handlers;
public sealed class GetTenantsAnnouncementsQueryHandlerTests
{
[Fact]
public async Task GivenQuery_WhenHandle_ThenUsesTenantProviderAndOrdersAndPaginates()
{
// Arrange
var tenantProvider = new Mock<ITenantProvider>();
tenantProvider.Setup(x => x.GetCurrentTenantId()).Returns(42);
var currentUserAccessor = new Mock<ICurrentUserAccessor>();
currentUserAccessor.SetupGet(x => x.UserId).Returns(0);
currentUserAccessor.SetupGet(x => x.IsAuthenticated).Returns(false);
var announcements = new List<TakeoutSaaS.Domain.Tenants.Entities.TenantAnnouncement>
{
AnnouncementTestData.CreateAnnouncement(1, 42, priority: 1, effectiveFrom: DateTime.UtcNow.AddDays(-1)),
AnnouncementTestData.CreateAnnouncement(2, 42, priority: 2, effectiveFrom: DateTime.UtcNow.AddDays(-3)),
AnnouncementTestData.CreateAnnouncement(3, 42, priority: 2, effectiveFrom: DateTime.UtcNow.AddDays(-1)),
AnnouncementTestData.CreateAnnouncement(4, 42, priority: 0, effectiveFrom: DateTime.UtcNow)
};
// 模拟数据库端排序:按 priority DESC, effectiveFrom DESC
var sortedAnnouncements = announcements
.OrderByDescending(x => x.Priority)
.ThenByDescending(x => x.EffectiveFrom)
.ToList();
var announcementRepository = new Mock<ITenantAnnouncementRepository>();
announcementRepository
.Setup(x => x.SearchAsync(
It.IsAny<long>(),
It.IsAny<TakeoutSaaS.Domain.Tenants.Enums.AnnouncementStatus?>(),
It.IsAny<TakeoutSaaS.Domain.Tenants.Enums.TenantAnnouncementType?>(),
It.IsAny<bool?>(),
It.IsAny<DateTime?>(),
It.IsAny<DateTime?>(),
It.IsAny<DateTime?>(),
It.IsAny<bool>(),
It.IsAny<int?>(),
It.IsAny<CancellationToken>()))
.ReturnsAsync(sortedAnnouncements);
var announcementReadRepository = new Mock<ITenantAnnouncementReadRepository>();
announcementReadRepository
.Setup(x => x.GetByAnnouncementAsync(
It.IsAny<long>(),
It.IsAny<IEnumerable<long>>(),
It.IsAny<long?>(),
It.IsAny<CancellationToken>()))
.ReturnsAsync(Array.Empty<TakeoutSaaS.Domain.Tenants.Entities.TenantAnnouncementRead>());
var handler = new GetTenantsAnnouncementsQueryHandler(
announcementRepository.Object,
announcementReadRepository.Object,
tenantProvider.Object,
currentUserAccessor.Object);
var query = new GetTenantsAnnouncementsQuery
{
TenantId = 999,
Page = 2,
PageSize = 2
};
// Act
var result = await handler.Handle(query, CancellationToken.None);
// Assert
announcementRepository.Verify(x => x.SearchAsync(
42,
query.Status,
query.AnnouncementType,
query.IsActive,
query.EffectiveFrom,
query.EffectiveTo,
null,
true,
12, // estimatedLimit = page * size * 3 = 2 * 2 * 3 = 12
It.IsAny<CancellationToken>()), Times.Once);
result.TotalCount.Should().Be(4);
result.Items.Select(x => x.Id).Should().Equal(1, 4);
result.Page.Should().Be(2);
result.PageSize.Should().Be(2);
}
[Fact]
public async Task GivenOnlyEffective_WhenHandle_ThenFiltersScheduledPublish()
{
// Arrange
var tenantProvider = new Mock<ITenantProvider>();
tenantProvider.Setup(x => x.GetCurrentTenantId()).Returns(42);
var announcement1 = AnnouncementTestData.CreateAnnouncement(1, 42, priority: 1, effectiveFrom: DateTime.UtcNow.AddDays(-1));
announcement1.ScheduledPublishAt = DateTime.UtcNow.AddMinutes(-10);
var announcement2 = AnnouncementTestData.CreateAnnouncement(2, 42, priority: 1, effectiveFrom: DateTime.UtcNow.AddDays(-1));
announcement2.ScheduledPublishAt = DateTime.UtcNow.AddMinutes(30);
var announcements = new List<TakeoutSaaS.Domain.Tenants.Entities.TenantAnnouncement>
{
announcement1,
announcement2
};
var announcementRepository = new Mock<ITenantAnnouncementRepository>();
announcementRepository
.Setup(x => x.SearchAsync(
It.IsAny<long>(),
It.IsAny<TakeoutSaaS.Domain.Tenants.Enums.AnnouncementStatus?>(),
It.IsAny<TakeoutSaaS.Domain.Tenants.Enums.TenantAnnouncementType?>(),
It.IsAny<bool?>(),
It.IsAny<DateTime?>(),
It.IsAny<DateTime?>(),
It.IsAny<DateTime?>(),
It.IsAny<bool>(),
It.IsAny<int?>(),
It.IsAny<CancellationToken>()))
.ReturnsAsync(announcements);
var announcementReadRepository = new Mock<ITenantAnnouncementReadRepository>();
announcementReadRepository
.Setup(x => x.GetByAnnouncementAsync(
It.IsAny<long>(),
It.IsAny<IEnumerable<long>>(),
It.IsAny<long?>(),
It.IsAny<CancellationToken>()))
.ReturnsAsync(Array.Empty<TakeoutSaaS.Domain.Tenants.Entities.TenantAnnouncementRead>());
var handler = new GetTenantsAnnouncementsQueryHandler(
announcementRepository.Object,
announcementReadRepository.Object,
tenantProvider.Object);
var query = new GetTenantsAnnouncementsQuery
{
OnlyEffective = true,
Page = 1,
PageSize = 10
};
// Act
var result = await handler.Handle(query, CancellationToken.None);
// Assert
result.Items.Should().ContainSingle();
result.Items[0].Id.Should().Be(1);
}
}

View File

@@ -0,0 +1,72 @@
using FluentAssertions;
using Moq;
using TakeoutSaaS.Application.App.Tenants.Handlers;
using TakeoutSaaS.Application.App.Tenants.Queries;
using TakeoutSaaS.Application.Tests.TestUtilities;
using TakeoutSaaS.Domain.Tenants.Enums;
using TakeoutSaaS.Domain.Tenants.Repositories;
using TakeoutSaaS.Shared.Abstractions.Security;
using TakeoutSaaS.Shared.Abstractions.Tenancy;
namespace TakeoutSaaS.Application.Tests.App.Tenants.Handlers;
public sealed class GetUnreadAnnouncementsQueryHandlerTests
{
[Fact]
public async Task GivenUnreadAnnouncements_WhenHandle_ThenUsesTenantProviderAndPaginates()
{
// Arrange
var tenantProvider = new Mock<ITenantProvider>();
tenantProvider.Setup(x => x.GetCurrentTenantId()).Returns(55);
var currentUserAccessor = new Mock<ICurrentUserAccessor>();
currentUserAccessor.SetupGet(x => x.UserId).Returns(0);
currentUserAccessor.SetupGet(x => x.IsAuthenticated).Returns(false);
var announcements = new List<TakeoutSaaS.Domain.Tenants.Entities.TenantAnnouncement>
{
AnnouncementTestData.CreateAnnouncement(1, 55, priority: 1, effectiveFrom: DateTime.UtcNow.AddDays(-1), status: AnnouncementStatus.Published, isActive: true),
AnnouncementTestData.CreateAnnouncement(2, 55, priority: 3, effectiveFrom: DateTime.UtcNow.AddDays(-2), status: AnnouncementStatus.Published, isActive: true),
AnnouncementTestData.CreateAnnouncement(3, 55, priority: 2, effectiveFrom: DateTime.UtcNow, status: AnnouncementStatus.Published, isActive: true)
};
var announcementRepository = new Mock<ITenantAnnouncementRepository>();
announcementRepository
.Setup(x => x.SearchUnreadAsync(
It.IsAny<long>(),
It.IsAny<long?>(),
It.IsAny<AnnouncementStatus?>(),
It.IsAny<bool?>(),
It.IsAny<DateTime?>(),
It.IsAny<CancellationToken>()))
.ReturnsAsync(announcements);
var handler = new GetUnreadAnnouncementsQueryHandler(
announcementRepository.Object,
tenantProvider.Object,
currentUserAccessor.Object);
var query = new GetUnreadAnnouncementsQuery
{
Page = 1,
PageSize = 2
};
// Act
var result = await handler.Handle(query, CancellationToken.None);
// Assert
announcementRepository.Verify(x => x.SearchUnreadAsync(
55,
null,
AnnouncementStatus.Published,
true,
It.IsAny<DateTime?>(),
It.IsAny<CancellationToken>()), Times.Once);
result.Items.Select(x => x.Id).Should().Equal(2, 3);
result.TotalCount.Should().Be(3);
result.Page.Should().Be(1);
result.PageSize.Should().Be(2);
}
}