user-refunds.service.ts 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  1. import { DataSource, Repository } from 'typeorm';
  2. import { OrderRefund } from '../entities/order-refund.entity';
  3. import { Order } from '../entities/order.entity';
  4. import { CreateOrderRefundDto, UpdateOrderRefundDto } from '../schemas/order-refund.schema';
  5. import type { z } from 'zod';
  6. export class UserRefundsService {
  7. private orderRefundRepository: Repository<OrderRefund>;
  8. private orderRepository: Repository<Order>;
  9. constructor(dataSource: DataSource) {
  10. this.orderRefundRepository = dataSource.getRepository(OrderRefund);
  11. this.orderRepository = dataSource.getRepository(Order);
  12. }
  13. /**
  14. * 获取当前用户的退款列表
  15. */
  16. async getUserRefundsList(
  17. page: number,
  18. pageSize: number,
  19. keyword?: string,
  20. userId?: number,
  21. filters?: any
  22. ): Promise<[OrderRefund[], number]> {
  23. if (!userId) {
  24. throw new Error('用户ID不能为空');
  25. }
  26. const queryBuilder = this.orderRefundRepository
  27. .createQueryBuilder('refund')
  28. .leftJoinAndSelect('refund.order', 'order')
  29. .where('order.userId = :userId', { userId });
  30. if (keyword) {
  31. queryBuilder.andWhere(
  32. '(refund.orderNo LIKE :keyword OR refund.refundOrderNo LIKE :keyword)',
  33. { keyword: `%${keyword}%` }
  34. );
  35. }
  36. // 处理状态过滤
  37. if (filters && filters.state !== undefined) {
  38. queryBuilder.andWhere('refund.state = :state', { state: filters.state });
  39. }
  40. const [data, total] = await queryBuilder
  41. .orderBy('refund.createdAt', 'DESC')
  42. .skip((page - 1) * pageSize)
  43. .take(pageSize)
  44. .getManyAndCount();
  45. return [data, total];
  46. }
  47. /**
  48. * 根据ID获取当前用户的退款详情
  49. */
  50. async getUserRefundById(id: number, userId?: number): Promise<OrderRefund | null> {
  51. if (!userId) {
  52. throw new Error('用户ID不能为空');
  53. }
  54. const refund = await this.orderRefundRepository
  55. .createQueryBuilder('refund')
  56. .leftJoinAndSelect('refund.order', 'order')
  57. .where('refund.id = :id', { id })
  58. .getOne();
  59. if (!refund) {
  60. return null;
  61. }
  62. // 检查权限
  63. if (refund.order.userId !== userId) {
  64. throw new Error('无权访问该退款记录');
  65. }
  66. return refund;
  67. }
  68. /**
  69. * 为当前用户创建退款
  70. */
  71. async createUserRefund(data: z.infer<typeof CreateOrderRefundDto>, userId?: number): Promise<OrderRefund> {
  72. if (!userId) {
  73. throw new Error('用户ID不能为空');
  74. }
  75. // 验证订单是否属于当前用户
  76. const order = await this.orderRepository.findOne({
  77. where: { orderNo: data.orderNo || undefined, userId }
  78. });
  79. if (!order) {
  80. throw new Error('订单不存在或无权为该订单创建退款');
  81. }
  82. const refund = this.orderRefundRepository.create({
  83. ...data,
  84. createdBy: userId,
  85. updatedBy: userId
  86. });
  87. return await this.orderRefundRepository.save(refund);
  88. }
  89. /**
  90. * 更新当前用户的退款
  91. */
  92. async updateUserRefund(
  93. id: number,
  94. data: z.infer<typeof UpdateOrderRefundDto>,
  95. userId?: number
  96. ): Promise<OrderRefund | null> {
  97. if (!userId) {
  98. throw new Error('用户ID不能为空');
  99. }
  100. // 验证退款是否属于当前用户
  101. const existingRefund = await this.getUserRefundById(id, userId);
  102. if (!existingRefund) {
  103. return null;
  104. }
  105. const updatedRefund = this.orderRefundRepository.merge(existingRefund, {
  106. ...data,
  107. updatedBy: userId
  108. });
  109. return await this.orderRefundRepository.save(updatedRefund);
  110. }
  111. /**
  112. * 删除当前用户的退款
  113. */
  114. async deleteUserRefund(id: number, userId?: number): Promise<boolean> {
  115. if (!userId) {
  116. throw new Error('用户ID不能为空');
  117. }
  118. // 验证退款是否属于当前用户
  119. const existingRefund = await this.getUserRefundById(id, userId);
  120. if (!existingRefund) {
  121. return false;
  122. }
  123. await this.orderRefundRepository.remove(existingRefund);
  124. return true;
  125. }
  126. }