fix: 公告模块第一次修复

This commit is contained in:
2025-12-27 18:22:30 +08:00
parent bc09d9ca2e
commit 57f4c2d394
7 changed files with 63 additions and 11 deletions

View File

@@ -94,8 +94,8 @@ public sealed class PlatformAnnouncementsController(IMediator mediator, ITenantC
/// </code> /// </code>
/// </remarks> /// </remarks>
[HttpGet] [HttpGet]
[PermissionAuthorize("platform-announcement:create")] [PermissionAuthorize("platform-announcement:read", "platform-announcement:create")]
[SwaggerOperation(Summary = "查询平台公告列表", Description = "需要权限platform-announcement:create")] [SwaggerOperation(Summary = "查询平台公告列表", Description = "需要权限:platform-announcement:read 或 platform-announcement:create")]
[ProducesResponseType(typeof(ApiResponse<PagedResult<TenantAnnouncementDto>>), StatusCodes.Status200OK)] [ProducesResponseType(typeof(ApiResponse<PagedResult<TenantAnnouncementDto>>), StatusCodes.Status200OK)]
[ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status403Forbidden)] [ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status403Forbidden)]
public async Task<ApiResponse<PagedResult<TenantAnnouncementDto>>> List([FromQuery] GetTenantsAnnouncementsQuery query, CancellationToken cancellationToken) public async Task<ApiResponse<PagedResult<TenantAnnouncementDto>>> List([FromQuery] GetTenantsAnnouncementsQuery query, CancellationToken cancellationToken)
@@ -126,8 +126,8 @@ public sealed class PlatformAnnouncementsController(IMediator mediator, ITenantC
/// </code> /// </code>
/// </remarks> /// </remarks>
[HttpGet("{announcementId:long}")] [HttpGet("{announcementId:long}")]
[PermissionAuthorize("platform-announcement:create")] [PermissionAuthorize("platform-announcement:read", "platform-announcement:create")]
[SwaggerOperation(Summary = "获取平台公告详情", Description = "需要权限platform-announcement:create")] [SwaggerOperation(Summary = "获取平台公告详情", Description = "需要权限:platform-announcement:read 或 platform-announcement:create")]
[ProducesResponseType(typeof(ApiResponse<TenantAnnouncementDto>), StatusCodes.Status200OK)] [ProducesResponseType(typeof(ApiResponse<TenantAnnouncementDto>), StatusCodes.Status200OK)]
[ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status404NotFound)] [ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status404NotFound)]
[ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status403Forbidden)] [ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status403Forbidden)]

View File

@@ -8,6 +8,7 @@ using TakeoutSaaS.Application.App.Tenants.Dto;
using TakeoutSaaS.Application.App.Tenants.Queries; using TakeoutSaaS.Application.App.Tenants.Queries;
using TakeoutSaaS.Module.Authorization.Attributes; using TakeoutSaaS.Module.Authorization.Attributes;
using TakeoutSaaS.Shared.Abstractions.Results; using TakeoutSaaS.Shared.Abstractions.Results;
using TakeoutSaaS.Shared.Abstractions.Tenancy;
using TakeoutSaaS.Shared.Web.Api; using TakeoutSaaS.Shared.Web.Api;
namespace TakeoutSaaS.AdminApi.Controllers; namespace TakeoutSaaS.AdminApi.Controllers;
@@ -18,7 +19,7 @@ namespace TakeoutSaaS.AdminApi.Controllers;
[ApiVersion("1.0")] [ApiVersion("1.0")]
[Authorize] [Authorize]
[Route("api/admin/v{version:apiVersion}/tenants/{tenantId:long}/announcements")] [Route("api/admin/v{version:apiVersion}/tenants/{tenantId:long}/announcements")]
public sealed class TenantAnnouncementsController(IMediator mediator) : BaseApiController public sealed class TenantAnnouncementsController(IMediator mediator, ITenantContextAccessor tenantContextAccessor) : BaseApiController
{ {
private const string TenantIdHeaderName = "X-Tenant-Id"; private const string TenantIdHeaderName = "X-Tenant-Id";
@@ -49,6 +50,13 @@ public sealed class TenantAnnouncementsController(IMediator mediator) : BaseApiC
[ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status403Forbidden)] [ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status403Forbidden)]
public async Task<ApiResponse<PagedResult<TenantAnnouncementDto>>> Search(long tenantId, [FromQuery] GetTenantsAnnouncementsQuery query, CancellationToken cancellationToken) public async Task<ApiResponse<PagedResult<TenantAnnouncementDto>>> Search(long tenantId, [FromQuery] GetTenantsAnnouncementsQuery query, CancellationToken cancellationToken)
{ {
if (!Request.Headers.TryGetValue(TenantIdHeaderName, out var tenantHeader) || string.IsNullOrWhiteSpace(tenantHeader))
{
var request = query with { TenantId = 0 };
var platformResult = await ExecuteAsPlatformAsync(() => mediator.Send(request, cancellationToken));
return ApiResponse<PagedResult<TenantAnnouncementDto>>.Ok(platformResult);
}
var headerError = EnsureTenantHeader<PagedResult<TenantAnnouncementDto>>(); var headerError = EnsureTenantHeader<PagedResult<TenantAnnouncementDto>>();
if (headerError != null) if (headerError != null)
{ {
@@ -369,4 +377,18 @@ public sealed class TenantAnnouncementsController(IMediator mediator) : BaseApiC
return null; return null;
} }
private async Task<T> ExecuteAsPlatformAsync<T>(Func<Task<T>> action)
{
var original = tenantContextAccessor.Current;
tenantContextAccessor.Current = new TenantContext(0, null, "platform");
try
{
return await action();
}
finally
{
tenantContextAccessor.Current = original;
}
}
} }

View File

@@ -37,6 +37,11 @@ public sealed class CreateTenantAnnouncementCommandHandler(
throw new BusinessException(ErrorCodes.ValidationFailed, "目标受众类型不能为空"); throw new BusinessException(ErrorCodes.ValidationFailed, "目标受众类型不能为空");
} }
if (request.EffectiveTo.HasValue && request.EffectiveFrom >= request.EffectiveTo.Value)
{
throw new BusinessException(ErrorCodes.ValidationFailed, "生效开始时间必须早于结束时间");
}
if (request.TenantId == 0 && request.PublisherScope != PublisherScope.Platform) if (request.TenantId == 0 && request.PublisherScope != PublisherScope.Platform)
{ {
throw new BusinessException(ErrorCodes.ValidationFailed, "TenantId=0 仅允许平台公告"); throw new BusinessException(ErrorCodes.ValidationFailed, "TenantId=0 仅允许平台公告");

View File

@@ -58,8 +58,15 @@ public sealed class PublishAnnouncementCommandHandler(
announcement.RevokedAt = null; announcement.RevokedAt = null;
announcement.RowVersion = request.RowVersion; announcement.RowVersion = request.RowVersion;
await announcementRepository.UpdateAsync(announcement, cancellationToken); try
await announcementRepository.SaveChangesAsync(cancellationToken); {
await announcementRepository.UpdateAsync(announcement, cancellationToken);
await announcementRepository.SaveChangesAsync(cancellationToken);
}
catch (Exception exception) when (exception.GetType().Name == "DbUpdateConcurrencyException")
{
throw new BusinessException(ErrorCodes.Conflict, "公告已被修改,请刷新后重试");
}
// 4. 发布领域事件 // 4. 发布领域事件
await eventPublisher.PublishAsync( await eventPublisher.PublishAsync(

View File

@@ -52,8 +52,15 @@ public sealed class RevokeAnnouncementCommandHandler(
announcement.RevokedAt = DateTime.UtcNow; announcement.RevokedAt = DateTime.UtcNow;
announcement.RowVersion = request.RowVersion; announcement.RowVersion = request.RowVersion;
await announcementRepository.UpdateAsync(announcement, cancellationToken); try
await announcementRepository.SaveChangesAsync(cancellationToken); {
await announcementRepository.UpdateAsync(announcement, cancellationToken);
await announcementRepository.SaveChangesAsync(cancellationToken);
}
catch (Exception exception) when (exception.GetType().Name == "DbUpdateConcurrencyException")
{
throw new BusinessException(ErrorCodes.Conflict, "公告已被修改,请刷新后重试");
}
// 4. 发布领域事件 // 4. 发布领域事件
await eventPublisher.PublishAsync( await eventPublisher.PublishAsync(

View File

@@ -52,8 +52,15 @@ public sealed class UpdateTenantAnnouncementCommandHandler(ITenantAnnouncementRe
announcement.RowVersion = request.RowVersion; announcement.RowVersion = request.RowVersion;
// 4. 持久化 // 4. 持久化
await announcementRepository.UpdateAsync(announcement, cancellationToken); try
await announcementRepository.SaveChangesAsync(cancellationToken); {
await announcementRepository.UpdateAsync(announcement, cancellationToken);
await announcementRepository.SaveChangesAsync(cancellationToken);
}
catch (Exception exception) when (exception.GetType().Name == "DbUpdateConcurrencyException")
{
throw new BusinessException(ErrorCodes.Conflict, "公告已被修改,请刷新后重试");
}
// 5. 返回 DTO // 5. 返回 DTO
return announcement.ToDto(false, null); return announcement.ToDto(false, null);

View File

@@ -1,10 +1,14 @@
using Microsoft.EntityFrameworkCore.Infrastructure;
using Microsoft.EntityFrameworkCore.Migrations; using Microsoft.EntityFrameworkCore.Migrations;
using TakeoutSaaS.Infrastructure.App.Persistence;
#nullable disable #nullable disable
namespace TakeoutSaaS.Infrastructure.Migrations namespace TakeoutSaaS.Infrastructure.Migrations
{ {
/// <inheritdoc /> /// <inheritdoc />
[DbContext(typeof(TakeoutAppDbContext))]
[Migration("20251225090000_AddTenantAnnouncementRowVersionTrigger")]
public partial class AddTenantAnnouncementRowVersionTrigger : Migration public partial class AddTenantAnnouncementRowVersionTrigger : Migration
{ {
/// <inheritdoc /> /// <inheritdoc />