Files
TakeoutSaaS.TenantApi/src/Api/TakeoutSaaS.AdminApi/Controllers/BillingsController.cs
2026-01-04 21:22:26 +08:00

304 lines
12 KiB
C#
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
using MediatR;
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Mvc;
using Microsoft.Net.Http.Headers;
using System.ComponentModel.DataAnnotations;
using TakeoutSaaS.Application.App.Billings.Commands;
using TakeoutSaaS.Application.App.Billings.Dto;
using TakeoutSaaS.Application.App.Billings.Queries;
using TakeoutSaaS.Module.Authorization.Attributes;
using TakeoutSaaS.Shared.Abstractions.Results;
using TakeoutSaaS.Shared.Web.Api;
namespace TakeoutSaaS.AdminApi.Controllers;
/// <summary>
/// 账单管理。
/// </summary>
[ApiVersion("1.0")]
[Authorize]
[Route("api/admin/v{version:apiVersion}/billings")]
public sealed class BillingsController(IMediator mediator) : BaseApiController
{
/// <summary>
/// 分页查询账单列表。
/// </summary>
/// <returns>账单分页结果。</returns>
[HttpGet]
[PermissionAuthorize("bill:read")]
[ProducesResponseType(typeof(ApiResponse<PagedResult<BillingListDto>>), StatusCodes.Status200OK)]
public async Task<ApiResponse<PagedResult<BillingListDto>>> GetList([FromQuery] GetBillingListQuery query, CancellationToken cancellationToken)
{
// 1. 查询账单列表
var result = await mediator.Send(query, cancellationToken);
// 2. 返回分页结果
return ApiResponse<PagedResult<BillingListDto>>.Ok(result);
}
/// <summary>
/// 获取账单详情。
/// </summary>
/// <param name="id">账单 ID。</param>
/// <param name="cancellationToken">取消标记。</param>
/// <returns>账单详情。</returns>
[HttpGet("{id:long}")]
[PermissionAuthorize("bill:read")]
[ProducesResponseType(typeof(ApiResponse<BillingDetailDto>), StatusCodes.Status200OK)]
[ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status404NotFound)]
public async Task<ApiResponse<BillingDetailDto>> GetDetail(long id, CancellationToken cancellationToken)
{
// 1. 查询账单详情(若不存在则抛出业务异常,由全局异常处理转换为 404
var result = await mediator.Send(new GetBillingDetailQuery { BillingId = id }, cancellationToken);
// 2. 返回详情
return ApiResponse<BillingDetailDto>.Ok(result);
}
/// <summary>
/// 手动创建账单。
/// </summary>
/// <param name="command">创建账单命令。</param>
/// <param name="cancellationToken">取消标记。</param>
/// <returns>创建的账单信息。</returns>
[HttpPost]
[PermissionAuthorize("bill:create")]
[ProducesResponseType(typeof(ApiResponse<BillingDetailDto>), StatusCodes.Status200OK)]
public async Task<ApiResponse<BillingDetailDto>> Create([FromBody, Required] CreateBillingCommand command, CancellationToken cancellationToken)
{
// 1. 创建账单
var result = await mediator.Send(command, cancellationToken);
// 2. 返回创建结果
return ApiResponse<BillingDetailDto>.Ok(result);
}
/// <summary>
/// 更新账单状态。
/// </summary>
/// <param name="id">账单 ID。</param>
/// <param name="command">更新状态命令。</param>
/// <param name="cancellationToken">取消标记。</param>
/// <returns>更新结果。</returns>
[HttpPut("{id:long}/status")]
[PermissionAuthorize("bill:update")]
[ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status200OK)]
[ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status404NotFound)]
public async Task<ApiResponse<object>> UpdateStatus(long id, [FromBody, Required] UpdateBillingStatusCommand command, CancellationToken cancellationToken)
{
// 1. 绑定账单标识
command = command with { BillingId = id };
// 2. 更新账单状态(若不存在则抛出业务异常,由全局异常处理转换为 404
await mediator.Send(command, cancellationToken);
// 3. 返回成功结果
return ApiResponse<object>.Ok(null);
}
/// <summary>
/// 取消账单。
/// </summary>
/// <param name="id">账单 ID。</param>
/// <param name="reason">取消原因(可选)。</param>
/// <param name="cancellationToken">取消标记。</param>
/// <returns>取消结果。</returns>
[HttpDelete("{id:long}")]
[PermissionAuthorize("bill:delete")]
[ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status200OK)]
[ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status404NotFound)]
public async Task<ApiResponse<object>> Cancel(long id, [FromQuery] string? reason, CancellationToken cancellationToken)
{
// 1. 取消账单(取消原因支持可选)
await mediator.Send(new CancelBillingCommand { BillingId = id, Reason = reason ?? string.Empty }, cancellationToken);
// 2. 返回成功结果
return ApiResponse<object>.Ok(null);
}
/// <summary>
/// 获取账单支付记录。
/// </summary>
/// <param name="id">账单 ID。</param>
/// <param name="cancellationToken">取消标记。</param>
/// <returns>支付记录列表。</returns>
[HttpGet("{id:long}/payments")]
[PermissionAuthorize("bill:read")]
[ProducesResponseType(typeof(ApiResponse<List<PaymentRecordDto>>), StatusCodes.Status200OK)]
public async Task<ApiResponse<List<PaymentRecordDto>>> GetPayments(long id, CancellationToken cancellationToken)
{
// 1. 查询支付记录
var result = await mediator.Send(new GetBillingPaymentsQuery { BillingId = id }, cancellationToken);
// 2. 返回列表
return ApiResponse<List<PaymentRecordDto>>.Ok(result);
}
/// <summary>
/// 记录支付(线下支付确认)。
/// </summary>
/// <param name="id">账单 ID。</param>
/// <param name="command">记录支付命令。</param>
/// <param name="cancellationToken">取消标记。</param>
/// <returns>支付记录信息。</returns>
[HttpPost("{id:long}/payments")]
[PermissionAuthorize("bill:pay")]
[ProducesResponseType(typeof(ApiResponse<PaymentRecordDto>), StatusCodes.Status200OK)]
[ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status404NotFound)]
public async Task<ApiResponse<PaymentRecordDto>> RecordPayment(long id, [FromBody, Required] RecordPaymentCommand command, CancellationToken cancellationToken)
{
// 1. 绑定账单标识
command = command with { BillingId = id };
// 2. 记录支付
var result = await mediator.Send(command, cancellationToken);
// 3. 返回支付记录
return ApiResponse<PaymentRecordDto>.Ok(result);
}
/// <summary>
/// 一键确认收款(记录支付 + 立即审核通过)。
/// </summary>
/// <param name="id">账单 ID。</param>
/// <param name="command">确认收款命令。</param>
/// <param name="cancellationToken">取消标记。</param>
/// <returns>确认后的支付记录。</returns>
[HttpPost("{id:long}/payments/confirm")]
[PermissionAuthorize("bill:pay")]
[ProducesResponseType(typeof(ApiResponse<PaymentRecordDto>), StatusCodes.Status200OK)]
[ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status404NotFound)]
public async Task<ApiResponse<PaymentRecordDto>> ConfirmPayment(long id, [FromBody, Required] ConfirmPaymentCommand command, CancellationToken cancellationToken)
{
// 1. 绑定账单标识
command = command with { BillingId = id };
// 2. 一键确认收款(含:写入 VerifiedBy/VerifiedAt并同步更新账单已收金额/状态)
var result = await mediator.Send(command, cancellationToken);
// 3. 返回结果
return ApiResponse<PaymentRecordDto>.Ok(result);
}
/// <summary>
/// 审核支付记录。
/// </summary>
/// <param name="paymentId">支付记录 ID。</param>
/// <param name="command">审核参数。</param>
/// <param name="cancellationToken">取消标记。</param>
/// <returns>审核后的支付记录。</returns>
[HttpPut("payments/{paymentId:long}/verify")]
[PermissionAuthorize("bill:update")]
[ProducesResponseType(typeof(ApiResponse<PaymentRecordDto>), StatusCodes.Status200OK)]
[ProducesResponseType(typeof(ApiResponse<object>), StatusCodes.Status404NotFound)]
public async Task<ApiResponse<PaymentRecordDto>> VerifyPayment(long paymentId, [FromBody, Required] VerifyPaymentCommand command, CancellationToken cancellationToken)
{
// 1. 绑定支付记录标识
command = command with { PaymentId = paymentId };
// 2. 审核支付记录
var result = await mediator.Send(command, cancellationToken);
// 3. 返回审核结果
return ApiResponse<PaymentRecordDto>.Ok(result);
}
/// <summary>
/// 批量更新账单状态。
/// </summary>
/// <param name="command">批量更新命令。</param>
/// <param name="cancellationToken">取消标记。</param>
/// <returns>更新条数。</returns>
[HttpPost("batch/status")]
[PermissionAuthorize("bill:update")]
[ProducesResponseType(typeof(ApiResponse<int>), StatusCodes.Status200OK)]
public async Task<ApiResponse<int>> BatchUpdateStatus([FromBody, Required] BatchUpdateStatusCommand command, CancellationToken cancellationToken)
{
// 1. 执行批量更新
var affected = await mediator.Send(command, cancellationToken);
// 2. 返回更新条数
return ApiResponse<int>.Ok(affected);
}
/// <summary>
/// 导出账单Excel/PDF/CSV
/// </summary>
/// <param name="query">导出请求。</param>
/// <param name="cancellationToken">取消标记。</param>
/// <returns>导出文件。</returns>
[HttpPost("export")]
[PermissionAuthorize("bill:read")]
[Produces("application/octet-stream")]
[ProducesResponseType(typeof(FileContentResult), StatusCodes.Status200OK)]
public async Task<IActionResult> Export([FromBody, Required] ExportBillingsQuery query, CancellationToken cancellationToken)
{
// 1. 执行导出
var bytes = await mediator.Send(query, cancellationToken);
// 2. 解析格式并生成文件名
var extension = ResolveExportFileExtension(query.Format);
var fileName = $"billings_{DateTime.UtcNow:yyyyMMdd_HHmmss}.{extension}";
// 3. 显式写入 Content-Disposition确保浏览器以附件形式下载
Response.Headers[HeaderNames.ContentDisposition] = new ContentDispositionHeaderValue("attachment")
{
FileName = fileName,
FileNameStar = fileName
}.ToString();
// 4. 返回二进制流(统一 octet-stream避免被默认 JSON Produces 影响)
return File(bytes, "application/octet-stream");
}
/// <summary>
/// 获取账单统计数据。
/// </summary>
/// <param name="query">统计查询参数。</param>
/// <param name="cancellationToken">取消标记。</param>
/// <returns>统计结果。</returns>
[HttpGet("statistics")]
[PermissionAuthorize("bill:read")]
[ProducesResponseType(typeof(ApiResponse<BillingStatisticsDto>), StatusCodes.Status200OK)]
public async Task<ApiResponse<BillingStatisticsDto>> Statistics([FromQuery] GetBillingStatisticsQuery query, CancellationToken cancellationToken)
{
// 1. 查询统计数据
var result = await mediator.Send(query, cancellationToken);
// 2. 返回统计结果
return ApiResponse<BillingStatisticsDto>.Ok(result);
}
/// <summary>
/// 获取逾期账单列表。
/// </summary>
/// <param name="query">逾期列表查询参数。</param>
/// <param name="cancellationToken">取消标记。</param>
/// <returns>逾期账单分页结果。</returns>
[HttpGet("overdue")]
[PermissionAuthorize("bill:read")]
[ProducesResponseType(typeof(ApiResponse<PagedResult<BillingListDto>>), StatusCodes.Status200OK)]
public async Task<ApiResponse<PagedResult<BillingListDto>>> Overdue([FromQuery] GetOverdueBillingsQuery query, CancellationToken cancellationToken)
{
// 1. 查询逾期账单分页列表
var result = await mediator.Send(query, cancellationToken);
// 2. 返回分页结果
return ApiResponse<PagedResult<BillingListDto>>.Ok(result);
}
private static string ResolveExportFileExtension(string? format)
{
// 1. 归一化导出格式
var normalized = (format ?? string.Empty).Trim();
// 2. 映射扩展名
return normalized.ToUpperInvariant() switch
{
"PDF" => "pdf",
"CSV" => "csv",
_ => "xlsx"
};
}
}