AutoFinishOrder.php 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. <?php
  2. namespace App\Jobs;
  3. use App\Enums\OrderRecordStatus;
  4. use App\Enums\OrderStatus;
  5. use App\Models\CoachUser;
  6. use App\Models\Order;
  7. use App\Models\OrderRecord;
  8. use Illuminate\Bus\Queueable;
  9. use Illuminate\Contracts\Queue\ShouldQueue;
  10. use Illuminate\Foundation\Bus\Dispatchable;
  11. use Illuminate\Queue\InteractsWithQueue;
  12. use Illuminate\Queue\SerializesModels;
  13. use Illuminate\Support\Facades\DB;
  14. use Illuminate\Support\Facades\Log;
  15. use App\Enums\TechnicianWorkStatus;
  16. class AutoFinishOrder implements ShouldQueue
  17. {
  18. use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
  19. /**
  20. * 任务最大尝试次数
  21. *
  22. * @var int
  23. */
  24. public $tries = 3;
  25. /**
  26. * 任务运行的超时时间
  27. *
  28. * @var int
  29. */
  30. public $timeout = 120;
  31. /**
  32. * Create a new job instance.
  33. */
  34. public function __construct(
  35. public Order $order
  36. ) {}
  37. /**
  38. * Execute the job.
  39. */
  40. public function handle(): void
  41. {
  42. try {
  43. DB::transaction(function () {
  44. // 获取订单信息并加锁
  45. $order = Order::query()
  46. ->with(['project', 'coach'])
  47. ->where('id', $this->order->id)
  48. ->lockForUpdate()
  49. ->first();
  50. // 检查订单是否存在
  51. if (! $order) {
  52. Log::error('自动结束订单失败 - 订单不存在', [
  53. 'order_id' => $this->order->id,
  54. ]);
  55. return;
  56. }
  57. // 检查订单状态是否为服务中
  58. if ($order->state !== OrderStatus::SERVICING->value) {
  59. Log::info('自动结束订单跳过 - 订单状态不是服务中', [
  60. 'order_id' => $order->id,
  61. 'current_state' => $order->state,
  62. ]);
  63. return;
  64. }
  65. // 更新订单状态为已完成
  66. $order->state = OrderStatus::SERVICE_END->value;
  67. $order->updated_at = now();
  68. $order->save();
  69. // 检测并更新技师状态
  70. if ($order->coach && $order->coach->work_status === TechnicianWorkStatus::BUSY->value) {
  71. $order->coach->update([
  72. 'work_status' => TechnicianWorkStatus::FREE->value
  73. ]);
  74. Log::info('自动结束订单 - 技师状态已更新为空闲', [
  75. 'order_id' => $order->id,
  76. 'coach_id' => $order->coach_id
  77. ]);
  78. }
  79. // 记录订单状态变更日志
  80. OrderRecord::create([
  81. 'order_id' => $order->id,
  82. 'state' => OrderRecordStatus::COMPLETED->value,
  83. 'object_id' => $order->coach_id,
  84. 'object_type' => CoachUser::class,
  85. 'remark' => '系统自动完成服务',
  86. ]);
  87. // TODO: 发送通知给用户和技师
  88. // event(new OrderFinishedEvent($order));
  89. Log::info('自动结束订单成功', [
  90. 'order_id' => $order->id,
  91. 'updated_at' => $order->updated_at,
  92. ]);
  93. });
  94. } catch (\Exception $e) {
  95. Log::error('自动结束订单失败', [
  96. 'order_id' => $this->order->id,
  97. 'error' => $e->getMessage(),
  98. 'trace' => $e->getTraceAsString(),
  99. ]);
  100. throw $e;
  101. }
  102. }
  103. /**
  104. * 处理失败的任务
  105. */
  106. public function failed(\Throwable $exception): void
  107. {
  108. Log::error('自动结束订单任务执行失败', [
  109. 'order_id' => $this->order->id,
  110. 'error' => $exception->getMessage(),
  111. 'trace' => $exception->getTraceAsString(),
  112. ]);
  113. }
  114. }