核心功能:
- 账单CRUD操作(创建、查询、详情、更新状态、删除)
- 支付记录管理(创建支付、审核支付)
- 批量操作支持(批量更新账单状态)
- 统计分析功能(账单统计、逾期账单查询)
- 导出功能(Excel/PDF/CSV)
API端点 (16个):
- GET /api/admin/v1/billings - 账单列表(分页、筛选、排序)
- POST /api/admin/v1/billings - 创建账单
- GET /api/admin/v1/billings/{id} - 账单详情
- DELETE /api/admin/v1/billings/{id} - 删除账单
- PUT /api/admin/v1/billings/{id}/status - 更新状态
- POST /api/admin/v1/billings/batch/status - 批量更新
- GET /api/admin/v1/billings/{id}/payments - 支付记录
- POST /api/admin/v1/billings/{id}/payments - 创建支付
- PUT /api/admin/v1/billings/payments/{paymentId}/verify - 审核支付
- GET /api/admin/v1/billings/statistics - 统计数据
- GET /api/admin/v1/billings/overdue - 逾期账单
- POST /api/admin/v1/billings/export - 导出账单
架构优化:
- 采用CQRS模式分离读写(MediatR + Dapper + EF Core)
- 完整的领域模型设计(TenantBillingStatement, TenantPayment等)
- FluentValidation请求验证
- 状态机管理账单和支付状态流转
API设计优化 (三项改进):
1. 导出API响应Content-Type改为application/octet-stream
2. 支付审核API添加Approved和Notes可选参数,支持通过/拒绝
3. 移除TenantBillings API中重复的TenantId参数
数据库变更:
- 新增账单相关表及关系
- 支持Snowflake ID主键
- 完整的审计字段支持
🤖 Generated with Claude Code
Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
56 lines
2.0 KiB
C#
56 lines
2.0 KiB
C#
using MediatR;
|
|
using TakeoutSaaS.Application.App.Billings.Dto;
|
|
using TakeoutSaaS.Application.App.Billings.Queries;
|
|
using TakeoutSaaS.Domain.Tenants.Repositories;
|
|
using TakeoutSaaS.Shared.Abstractions.Results;
|
|
|
|
namespace TakeoutSaaS.Application.App.Billings.Handlers;
|
|
|
|
/// <summary>
|
|
/// 获取账单列表查询处理器。
|
|
/// </summary>
|
|
public sealed class GetBillListQueryHandler(
|
|
ITenantBillingRepository billingRepository,
|
|
ITenantRepository tenantRepository)
|
|
: IRequestHandler<GetBillListQuery, PagedResult<BillDto>>
|
|
{
|
|
/// <summary>
|
|
/// 处理获取账单列表请求。
|
|
/// </summary>
|
|
/// <param name="request">查询请求。</param>
|
|
/// <param name="cancellationToken">取消标记。</param>
|
|
/// <returns>分页账单列表。</returns>
|
|
public async Task<PagedResult<BillDto>> Handle(GetBillListQuery request, CancellationToken cancellationToken)
|
|
{
|
|
// 1. 分页查询账单
|
|
var (bills, total) = await billingRepository.SearchPagedAsync(
|
|
request.TenantId,
|
|
request.Status,
|
|
request.StartDate,
|
|
request.EndDate,
|
|
null,
|
|
null,
|
|
request.Keyword,
|
|
request.PageNumber,
|
|
request.PageSize,
|
|
cancellationToken);
|
|
|
|
// 2. 无数据直接返回
|
|
if (bills.Count == 0)
|
|
{
|
|
return new PagedResult<BillDto>([], request.PageNumber, request.PageSize, total);
|
|
}
|
|
|
|
// 3. 批量查询租户信息
|
|
var tenantIds = bills.Select(b => b.TenantId).Distinct().ToArray();
|
|
var tenants = await tenantRepository.FindByIdsAsync(tenantIds, cancellationToken);
|
|
var tenantDict = tenants.ToDictionary(t => t.Id, t => t.Name);
|
|
|
|
// 4. 映射 DTO
|
|
var result = bills.Select(b => b.ToDto(tenantDict.GetValueOrDefault(b.TenantId))).ToList();
|
|
|
|
// 5. 返回分页结果
|
|
return new PagedResult<BillDto>(result, request.PageNumber, request.PageSize, total);
|
|
}
|
|
}
|