From bb3bb842bcdd0bb202d65b931bfb9747212e8e47 Mon Sep 17 00:00:00 2001
From: MSuMshk <2039814060@qq.com>
Date: Thu, 29 Jan 2026 13:30:49 +0000
Subject: [PATCH] =?UTF-8?q?feat:=20=E9=85=8D=E9=80=81=E5=8D=95=E6=9F=A5?=
=?UTF-8?q?=E8=AF=A2=E6=94=AF=E6=8C=81tenantId=E5=8F=AF=E9=80=89=E8=BF=87?=
=?UTF-8?q?=E6=BB=A4?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.../Controllers/DeliveriesController.cs | 3 ++
.../CreateDeliveryOrderCommandHandler.cs | 24 +++++++++---
.../DeleteDeliveryOrderCommandHandler.cs | 13 +++----
.../GetDeliveryOrderByIdQueryHandler.cs | 17 +++------
.../SearchDeliveryOrdersQueryHandler.cs | 19 ++++------
.../UpdateDeliveryOrderCommandHandler.cs | 31 +++++++++------
.../Queries/SearchDeliveryOrdersQuery.cs | 5 +++
.../Repositories/IDeliveryRepository.cs | 18 +++++++++
.../App/Repositories/EfDeliveryRepository.cs | 38 +++++++++++++++++++
9 files changed, 120 insertions(+), 48 deletions(-)
diff --git a/src/Api/TakeoutSaaS.AdminApi/Controllers/DeliveriesController.cs b/src/Api/TakeoutSaaS.AdminApi/Controllers/DeliveriesController.cs
index 6f0f953..841cac3 100644
--- a/src/Api/TakeoutSaaS.AdminApi/Controllers/DeliveriesController.cs
+++ b/src/Api/TakeoutSaaS.AdminApi/Controllers/DeliveriesController.cs
@@ -41,6 +41,7 @@ public sealed class DeliveriesController(IMediator mediator) : BaseApiController
///
/// 查询配送单列表。
///
+ /// 租户 ID(可选;为空表示跨租户查询)。
/// 订单 ID。
/// 配送状态。
/// 页码。
@@ -53,6 +54,7 @@ public sealed class DeliveriesController(IMediator mediator) : BaseApiController
[PermissionAuthorize("delivery:read")]
[ProducesResponseType(typeof(ApiResponse>), StatusCodes.Status200OK)]
public async Task>> List(
+ [FromQuery] long? tenantId,
[FromQuery] long? orderId,
[FromQuery] DeliveryStatus? status,
[FromQuery] int page = 1,
@@ -64,6 +66,7 @@ public sealed class DeliveriesController(IMediator mediator) : BaseApiController
// 1. 组装查询参数
var result = await mediator.Send(new SearchDeliveryOrdersQuery
{
+ TenantId = tenantId,
OrderId = orderId,
Status = status,
Page = page,
diff --git a/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/CreateDeliveryOrderCommandHandler.cs b/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/CreateDeliveryOrderCommandHandler.cs
index ad4149f..1934efd 100644
--- a/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/CreateDeliveryOrderCommandHandler.cs
+++ b/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/CreateDeliveryOrderCommandHandler.cs
@@ -4,21 +4,35 @@ using TakeoutSaaS.Application.App.Deliveries.Commands;
using TakeoutSaaS.Application.App.Deliveries.Dto;
using TakeoutSaaS.Domain.Deliveries.Entities;
using TakeoutSaaS.Domain.Deliveries.Repositories;
+using TakeoutSaaS.Domain.Orders.Repositories;
+using TakeoutSaaS.Shared.Abstractions.Constants;
+using TakeoutSaaS.Shared.Abstractions.Exceptions;
namespace TakeoutSaaS.Application.App.Deliveries.Handlers;
///
/// 创建配送单命令处理器。
///
-public sealed class CreateDeliveryOrderCommandHandler(IDeliveryRepository deliveryRepository, ILogger logger)
+public sealed class CreateDeliveryOrderCommandHandler(
+ IDeliveryRepository deliveryRepository,
+ IOrderRepository orderRepository,
+ ILogger logger)
: IRequestHandler
{
///
public async Task Handle(CreateDeliveryOrderCommand request, CancellationToken cancellationToken)
{
- // 1. 构建配送单实体
+ // 1. 查询订单以确定租户
+ var order = await orderRepository.FindByIdAsync(request.OrderId, cancellationToken);
+ if (order is null)
+ {
+ throw new BusinessException(ErrorCodes.NotFound, "订单不存在");
+ }
+
+ // 2. (空行后) 构建配送单实体并写入租户
var deliveryOrder = new DeliveryOrder
{
+ TenantId = order.TenantId,
OrderId = request.OrderId,
Provider = request.Provider,
ProviderOrderId = request.ProviderOrderId?.Trim(),
@@ -32,14 +46,14 @@ public sealed class CreateDeliveryOrderCommandHandler(IDeliveryRepository delive
FailureReason = request.FailureReason?.Trim()
};
- // 2. 持久化配送单
+ // 3. (空行后) 持久化配送单
await deliveryRepository.AddDeliveryOrderAsync(deliveryOrder, cancellationToken);
await deliveryRepository.SaveChangesAsync(cancellationToken);
- // 3. 记录日志
+ // 4. (空行后) 记录日志
logger.LogInformation("创建配送单 {DeliveryOrderId} 对应订单 {OrderId}", deliveryOrder.Id, deliveryOrder.OrderId);
- // 4. 映射 DTO 返回
+ // 5. (空行后) 映射 DTO 返回
return MapToDto(deliveryOrder, []);
}
diff --git a/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/DeleteDeliveryOrderCommandHandler.cs b/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/DeleteDeliveryOrderCommandHandler.cs
index 308c942..93f1c5a 100644
--- a/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/DeleteDeliveryOrderCommandHandler.cs
+++ b/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/DeleteDeliveryOrderCommandHandler.cs
@@ -2,7 +2,6 @@ using MediatR;
using Microsoft.Extensions.Logging;
using TakeoutSaaS.Application.App.Deliveries.Commands;
using TakeoutSaaS.Domain.Deliveries.Repositories;
-using TakeoutSaaS.Shared.Abstractions.Tenancy;
namespace TakeoutSaaS.Application.App.Deliveries.Handlers;
@@ -11,26 +10,24 @@ namespace TakeoutSaaS.Application.App.Deliveries.Handlers;
///
public sealed class DeleteDeliveryOrderCommandHandler(
IDeliveryRepository deliveryRepository,
- ITenantProvider tenantProvider,
ILogger logger)
: IRequestHandler
{
///
public async Task Handle(DeleteDeliveryOrderCommand request, CancellationToken cancellationToken)
{
- // 1. 获取租户并定位配送单
- var tenantId = tenantProvider.GetCurrentTenantId();
- var existing = await deliveryRepository.FindByIdAsync(request.DeliveryOrderId, tenantId, cancellationToken);
+ // 1. 定位配送单(跨租户)
+ var existing = await deliveryRepository.FindByIdAsync(request.DeliveryOrderId, cancellationToken);
if (existing == null)
{
return false;
}
- // 2. 删除并保存
- await deliveryRepository.DeleteDeliveryOrderAsync(request.DeliveryOrderId, tenantId, cancellationToken);
+ // 2. (空行后) 删除并保存
+ await deliveryRepository.DeleteDeliveryOrderAsync(request.DeliveryOrderId, existing.TenantId, cancellationToken);
await deliveryRepository.SaveChangesAsync(cancellationToken);
- // 3. 记录删除日志
+ // 3. (空行后) 记录删除日志
logger.LogInformation("删除配送单 {DeliveryOrderId}", request.DeliveryOrderId);
return true;
diff --git a/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/GetDeliveryOrderByIdQueryHandler.cs b/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/GetDeliveryOrderByIdQueryHandler.cs
index d863d4b..f62fb9d 100644
--- a/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/GetDeliveryOrderByIdQueryHandler.cs
+++ b/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/GetDeliveryOrderByIdQueryHandler.cs
@@ -3,7 +3,6 @@ using TakeoutSaaS.Application.App.Deliveries.Dto;
using TakeoutSaaS.Application.App.Deliveries.Queries;
using TakeoutSaaS.Domain.Deliveries.Entities;
using TakeoutSaaS.Domain.Deliveries.Repositories;
-using TakeoutSaaS.Shared.Abstractions.Tenancy;
namespace TakeoutSaaS.Application.App.Deliveries.Handlers;
@@ -11,27 +10,23 @@ namespace TakeoutSaaS.Application.App.Deliveries.Handlers;
/// 配送单详情查询处理器。
///
public sealed class GetDeliveryOrderByIdQueryHandler(
- IDeliveryRepository deliveryRepository,
- ITenantProvider tenantProvider)
+ IDeliveryRepository deliveryRepository)
: IRequestHandler
{
///
public async Task Handle(GetDeliveryOrderByIdQuery request, CancellationToken cancellationToken)
{
- // 1. 读取当前租户标识
- var tenantId = tenantProvider.GetCurrentTenantId();
-
- // 2. 查询配送单主体
- var order = await deliveryRepository.FindByIdAsync(request.DeliveryOrderId, tenantId, cancellationToken);
+ // 1. 查询配送单主体(跨租户)
+ var order = await deliveryRepository.FindByIdAsync(request.DeliveryOrderId, cancellationToken);
if (order == null)
{
return null;
}
- // 3. 查询配送事件明细
- var events = await deliveryRepository.GetEventsAsync(order.Id, tenantId, cancellationToken);
+ // 2. (空行后) 查询配送事件明细
+ var events = await deliveryRepository.GetEventsAsync(order.Id, order.TenantId, cancellationToken);
- // 4. 映射为 DTO 返回
+ // 3. (空行后) 映射为 DTO 返回
return MapToDto(order, events);
}
diff --git a/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/SearchDeliveryOrdersQueryHandler.cs b/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/SearchDeliveryOrdersQueryHandler.cs
index 7fcdd5a..d58c114 100644
--- a/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/SearchDeliveryOrdersQueryHandler.cs
+++ b/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/SearchDeliveryOrdersQueryHandler.cs
@@ -3,7 +3,6 @@ using TakeoutSaaS.Application.App.Deliveries.Dto;
using TakeoutSaaS.Application.App.Deliveries.Queries;
using TakeoutSaaS.Domain.Deliveries.Repositories;
using TakeoutSaaS.Shared.Abstractions.Results;
-using TakeoutSaaS.Shared.Abstractions.Tenancy;
namespace TakeoutSaaS.Application.App.Deliveries.Handlers;
@@ -11,29 +10,25 @@ namespace TakeoutSaaS.Application.App.Deliveries.Handlers;
/// 配送单列表查询处理器。
///
public sealed class SearchDeliveryOrdersQueryHandler(
- IDeliveryRepository deliveryRepository,
- ITenantProvider tenantProvider)
+ IDeliveryRepository deliveryRepository)
: IRequestHandler>
{
///
public async Task> Handle(SearchDeliveryOrdersQuery request, CancellationToken cancellationToken)
{
- // 1. 获取当前租户标识
- var tenantId = tenantProvider.GetCurrentTenantId();
+ // 1. 查询配送单列表(可选租户过滤)
+ var orders = await deliveryRepository.SearchAsync(request.TenantId, request.Status, request.OrderId, cancellationToken);
- // 2. 查询配送单列表(租户隔离)
- var orders = await deliveryRepository.SearchAsync(tenantId, request.Status, request.OrderId, cancellationToken);
-
- // 3. 本地排序
+ // 2. (空行后) 本地排序
var sorted = ApplySorting(orders, request.SortBy, request.SortDescending);
- // 4. 本地分页
+ // 3. (空行后) 本地分页
var paged = sorted
.Skip((request.Page - 1) * request.PageSize)
.Take(request.PageSize)
.ToList();
- // 5. 映射 DTO
+ // 4. (空行后) 映射 DTO
var items = paged.Select(order => new DeliveryOrderDto
{
Id = order.Id,
@@ -52,7 +47,7 @@ public sealed class SearchDeliveryOrdersQueryHandler(
CreatedAt = order.CreatedAt
}).ToList();
- // 6. 返回分页结果
+ // 5. (空行后) 返回分页结果
return new PagedResult(items, request.Page, request.PageSize, orders.Count);
}
diff --git a/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/UpdateDeliveryOrderCommandHandler.cs b/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/UpdateDeliveryOrderCommandHandler.cs
index 0c3f2c2..d88d2ae 100644
--- a/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/UpdateDeliveryOrderCommandHandler.cs
+++ b/src/Application/TakeoutSaaS.Application/App/Deliveries/Handlers/UpdateDeliveryOrderCommandHandler.cs
@@ -4,7 +4,9 @@ using TakeoutSaaS.Application.App.Deliveries.Commands;
using TakeoutSaaS.Application.App.Deliveries.Dto;
using TakeoutSaaS.Domain.Deliveries.Entities;
using TakeoutSaaS.Domain.Deliveries.Repositories;
-using TakeoutSaaS.Shared.Abstractions.Tenancy;
+using TakeoutSaaS.Domain.Orders.Repositories;
+using TakeoutSaaS.Shared.Abstractions.Constants;
+using TakeoutSaaS.Shared.Abstractions.Exceptions;
namespace TakeoutSaaS.Application.App.Deliveries.Handlers;
@@ -13,24 +15,29 @@ namespace TakeoutSaaS.Application.App.Deliveries.Handlers;
///
public sealed class UpdateDeliveryOrderCommandHandler(
IDeliveryRepository deliveryRepository,
- ITenantProvider tenantProvider,
+ IOrderRepository orderRepository,
ILogger logger)
: IRequestHandler
{
///
public async Task Handle(UpdateDeliveryOrderCommand request, CancellationToken cancellationToken)
{
- // 1. 获取当前租户标识
- var tenantId = tenantProvider.GetCurrentTenantId();
-
- // 2. 查询目标配送单
- var existing = await deliveryRepository.FindByIdAsync(request.DeliveryOrderId, tenantId, cancellationToken);
+ // 1. 查询目标配送单(跨租户)
+ var existing = await deliveryRepository.FindByIdAsync(request.DeliveryOrderId, cancellationToken);
if (existing == null)
{
return null;
}
- // 3. 更新字段
+ // 2. (空行后) 查询订单以确定租户
+ var order = await orderRepository.FindByIdAsync(request.OrderId, cancellationToken);
+ if (order is null)
+ {
+ throw new BusinessException(ErrorCodes.NotFound, "订单不存在");
+ }
+
+ // 3. (空行后) 更新字段并写入租户
+ existing.TenantId = order.TenantId;
existing.OrderId = request.OrderId;
existing.Provider = request.Provider;
existing.ProviderOrderId = request.ProviderOrderId?.Trim();
@@ -43,15 +50,15 @@ public sealed class UpdateDeliveryOrderCommandHandler(
existing.DeliveredAt = request.DeliveredAt;
existing.FailureReason = request.FailureReason?.Trim();
- // 4. 持久化变更
+ // 4. (空行后) 持久化变更
await deliveryRepository.UpdateDeliveryOrderAsync(existing, cancellationToken);
await deliveryRepository.SaveChangesAsync(cancellationToken);
- // 5. 记录更新日志
+ // 5. (空行后) 记录更新日志
logger.LogInformation("更新配送单 {DeliveryOrderId}", existing.Id);
- // 6. 查询事件并返回映射结果
- var events = await deliveryRepository.GetEventsAsync(existing.Id, tenantId, cancellationToken);
+ // 6. (空行后) 查询事件并返回映射结果
+ var events = await deliveryRepository.GetEventsAsync(existing.Id, existing.TenantId, cancellationToken);
return MapToDto(existing, events);
}
diff --git a/src/Application/TakeoutSaaS.Application/App/Deliveries/Queries/SearchDeliveryOrdersQuery.cs b/src/Application/TakeoutSaaS.Application/App/Deliveries/Queries/SearchDeliveryOrdersQuery.cs
index 751d90c..b8528c9 100644
--- a/src/Application/TakeoutSaaS.Application/App/Deliveries/Queries/SearchDeliveryOrdersQuery.cs
+++ b/src/Application/TakeoutSaaS.Application/App/Deliveries/Queries/SearchDeliveryOrdersQuery.cs
@@ -10,6 +10,11 @@ namespace TakeoutSaaS.Application.App.Deliveries.Queries;
///
public sealed class SearchDeliveryOrdersQuery : IRequest>
{
+ ///
+ /// 租户 ID(可选;为空表示跨租户查询)。
+ ///
+ public long? TenantId { get; init; }
+
///
/// 订单 ID(可选)。
///
diff --git a/src/Domain/TakeoutSaaS.Domain/Deliveries/Repositories/IDeliveryRepository.cs b/src/Domain/TakeoutSaaS.Domain/Deliveries/Repositories/IDeliveryRepository.cs
index f97f67e..c2de823 100644
--- a/src/Domain/TakeoutSaaS.Domain/Deliveries/Repositories/IDeliveryRepository.cs
+++ b/src/Domain/TakeoutSaaS.Domain/Deliveries/Repositories/IDeliveryRepository.cs
@@ -17,6 +17,14 @@ public interface IDeliveryRepository
/// 配送单实体或 null。
Task FindByIdAsync(long deliveryOrderId, long tenantId, CancellationToken cancellationToken = default);
+ ///
+ /// 依据标识获取配送单(跨租户)。
+ ///
+ /// 配送单 ID。
+ /// 取消标记。
+ /// 配送单实体或 null。
+ Task FindByIdAsync(long deliveryOrderId, CancellationToken cancellationToken = default);
+
///
/// 依据订单标识获取配送单。
///
@@ -68,6 +76,16 @@ public interface IDeliveryRepository
/// 配送单列表。
Task> SearchAsync(long tenantId, DeliveryStatus? status, long? orderId, CancellationToken cancellationToken = default);
+ ///
+ /// 按状态查询配送单(可选租户过滤)。
+ ///
+ /// 租户 ID(为空则不做租户过滤)。
+ /// 配送状态。
+ /// 订单 ID。
+ /// 取消标记。
+ /// 配送单列表。
+ Task> SearchAsync(long? tenantId, DeliveryStatus? status, long? orderId, CancellationToken cancellationToken = default);
+
///
/// 更新配送单。
///
diff --git a/src/Infrastructure/TakeoutSaaS.Infrastructure/App/Repositories/EfDeliveryRepository.cs b/src/Infrastructure/TakeoutSaaS.Infrastructure/App/Repositories/EfDeliveryRepository.cs
index 9c75fb4..1083ee7 100644
--- a/src/Infrastructure/TakeoutSaaS.Infrastructure/App/Repositories/EfDeliveryRepository.cs
+++ b/src/Infrastructure/TakeoutSaaS.Infrastructure/App/Repositories/EfDeliveryRepository.cs
@@ -23,6 +23,16 @@ public sealed class EfDeliveryRepository(TakeoutAdminDbContext context) : IDeliv
.FirstOrDefaultAsync(cancellationToken);
}
+ ///
+ public Task FindByIdAsync(long deliveryOrderId, CancellationToken cancellationToken = default)
+ {
+ // 1. 按主键查询(跨租户)
+ return context.DeliveryOrders
+ .AsNoTracking()
+ .Where(x => x.Id == deliveryOrderId)
+ .FirstOrDefaultAsync(cancellationToken);
+ }
+
///
public Task FindByOrderIdAsync(long orderId, long tenantId, CancellationToken cancellationToken = default)
{
@@ -84,6 +94,34 @@ public sealed class EfDeliveryRepository(TakeoutAdminDbContext context) : IDeliv
.ToListAsync(cancellationToken);
}
+ ///
+ public async Task> SearchAsync(long? tenantId, DeliveryStatus? status, long? orderId, CancellationToken cancellationToken = default)
+ {
+ // 1. 构建查询(可选租户过滤)
+ var query = context.DeliveryOrders.AsNoTracking();
+ if (tenantId.HasValue)
+ {
+ query = query.Where(x => x.TenantId == tenantId.Value);
+ }
+
+ // 2. (空行后) 可选过滤:配送状态
+ if (status.HasValue)
+ {
+ query = query.Where(x => x.Status == status.Value);
+ }
+
+ // 3. (空行后) 可选过滤:订单标识
+ if (orderId.HasValue)
+ {
+ query = query.Where(x => x.OrderId == orderId.Value);
+ }
+
+ // 4. (空行后) 排序并返回
+ return await query
+ .OrderByDescending(x => x.CreatedAt)
+ .ToListAsync(cancellationToken);
+ }
+
///
public Task UpdateDeliveryOrderAsync(DeliveryOrder deliveryOrder, CancellationToken cancellationToken = default)
{