OrderService.php 46 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377
  1. <?php
  2. namespace App\Services\Client;
  3. use App\Enums\OrderGrabRecordStatus;
  4. use App\Enums\OrderRecordStatus;
  5. use App\Enums\OrderSource;
  6. use App\Enums\OrderStatus;
  7. use App\Enums\OrderType;
  8. use App\Enums\PaymentMethod;
  9. use App\Enums\ProjectStatus;
  10. use App\Enums\TechnicianAuthStatus;
  11. use App\Enums\TechnicianStatus;
  12. use App\Enums\UserStatus;
  13. use App\Models\AgentConfig;
  14. use App\Models\AgentInfo;
  15. use App\Models\CoachConfig;
  16. use App\Models\CoachUser;
  17. use App\Models\MemberUser;
  18. use App\Models\Order;
  19. use App\Models\OrderGrabRecord;
  20. use App\Models\OrderRecord;
  21. use App\Models\Project;
  22. use App\Models\SysConfig;
  23. use App\Models\User;
  24. use App\Models\WalletPaymentRecord;
  25. use App\Models\WalletRefundRecord;
  26. use Carbon\Carbon;
  27. use Exception;
  28. use Illuminate\Support\Facades\Auth;
  29. use Illuminate\Support\Facades\DB;
  30. use Illuminate\Support\Facades\Log;
  31. use Illuminate\Support\Facades\Redis;
  32. use SimpleSoftwareIO\QrCode\Facades\QrCode;
  33. class OrderService
  34. {
  35. protected AgentService $agentService;
  36. protected ProjectService $projectService;
  37. public function __construct(
  38. AgentService $agentService,
  39. ProjectService $projectService
  40. ) {
  41. $this->agentService = $agentService;
  42. $this->projectService = $projectService;
  43. }
  44. /**
  45. * 订单初始化
  46. *
  47. * 初始化订单信息,包括用户钱包、技师信息、项目信息、地址信息和订单金额等
  48. *
  49. * @param int $userId 用户ID
  50. * @param array $data 订单数据
  51. * @return array 返回初始化的订单信息
  52. *
  53. * @throws \Exception 初始化失败时抛出异常
  54. */
  55. public function initialize(int $userId, array $data): array
  56. {
  57. try {
  58. // 参数验证
  59. abort_if(empty($data['project_id']), 400, '项目ID不能为空');
  60. abort_if(empty($data['coach_id']), 400, '技师ID不能为空');
  61. return DB::transaction(function () use ($userId, $data) {
  62. $user = MemberUser::find($userId);
  63. abort_if(! $user || $user->state != UserStatus::OPEN->value, 400, '用户状态异常');
  64. // 查询用户钱包
  65. $wallet = $user->wallet;
  66. // 查询默认地址
  67. $address = $user->address;
  68. $areaCode = $address ? $address->area_code : ($data['area_code'] ?? null);
  69. abort_if(empty($areaCode), 400, '区域编码不能为空');
  70. // 查询技师数据
  71. $coach = $this->validateCoach($data['coach_id']);
  72. // 查询技师可服务时间段
  73. $availableTimeSlots = app(CoachService::class)->getSchedule($coach->id);
  74. // 获取项目详情
  75. $project = $this->projectService->getProjectDetail($data['project_id'], $areaCode);
  76. abort_if(! $project, 400, '项目不存在');
  77. // 计算订单金额
  78. $amounts = $this->calculateOrderAmount(
  79. $userId,
  80. $address?->id ?? 0,
  81. $data['coach_id'],
  82. $data['project_id'],
  83. $project->agent_id,
  84. false,
  85. $data['latitude'],
  86. $data['longitude']
  87. );
  88. return [
  89. 'wallet' => $wallet,
  90. 'coach' => $coach,
  91. 'project' => $project,
  92. 'address' => $address,
  93. 'amounts' => $amounts,
  94. 'availableTimeSlots' => $availableTimeSlots,
  95. ];
  96. });
  97. } catch (Exception $e) {
  98. Log::error('订单初始化失败', [
  99. 'userId' => $userId,
  100. 'data' => $data,
  101. 'error' => $e->getMessage(),
  102. 'trace' => $e->getTraceAsString(),
  103. ]);
  104. throw $e;
  105. }
  106. }
  107. /**
  108. * 创建订单
  109. */
  110. public function createOrder(int $userId, array $data): array
  111. {
  112. // TODO:检测技师可服务时间段
  113. return DB::transaction(function () use ($userId, $data) {
  114. // 1. 参数校验
  115. $user = MemberUser::where('id', $userId)
  116. ->where('state', UserStatus::OPEN->value())
  117. ->firstOrFail();
  118. $project = Project::where('id', $data['project_id'])
  119. ->where('state', ProjectStatus::OPEN->value())
  120. ->firstOrFail();
  121. // 2. 订单类型判断
  122. $orderType = $data['order_type'];
  123. // 关键操作:验证必要参数
  124. abort_if(empty($data['project_id']), 400, '项目ID不能为空');
  125. // 上门订单必须指定技师和地址
  126. abort_if($orderType == OrderType::VISIT->value && empty($data['coach_id']), 400, '技师ID不能为空');
  127. abort_if($orderType == OrderType::VISIT->value && empty($data['address_id']), 400, '地址ID不能为空');
  128. // 加钟订单必须指定原订单
  129. abort_if($orderType == OrderType::OVERTIME->value && empty($data['order_id']), 400, '原订单ID不能为空');
  130. // 到店订单必须指定店铺
  131. abort_if($orderType == OrderType::SHOP->value && empty($data['shop_id']), 400, '店铺ID不能为空');
  132. // 应急订单必须指定地址
  133. abort_if($orderType == OrderType::EMERGENCY->value && empty($data['address_id']), 400, '地址ID不能为空');
  134. // 3. 验证技师
  135. // 4. 根据订单类型处理
  136. // 上门订单
  137. if ($orderType == OrderType::VISIT->value) {
  138. $this->validateCoach($data['coach_id']);
  139. }
  140. // 加钟订单
  141. if ($orderType == OrderType::OVERTIME->value) {
  142. $originalOrder = $this->getOriginalOrder($user, $data['order_id']);
  143. $data['address_id'] = $originalOrder->address_id;
  144. $this->validateCoach($originalOrder->coach_id);
  145. abort_if(! in_array($originalOrder->state, [OrderStatus::SERVING->value, OrderStatus::FINISHED->value]), 400, '原订单状态不允许加钟');
  146. $data = $this->prepareAddTimeData($originalOrder, $data);
  147. }
  148. $address = $user->addresses()
  149. ->where('id', $data['address_id'])
  150. ->firstOrFail();
  151. // 5. 计算订单金额
  152. $data['use_balance'] = $data['use_balance'] ?? false;
  153. $amounts = $this->calculateOrderAmount(
  154. $userId,
  155. $address->id,
  156. $data['coach_id'],
  157. $data['project_id'],
  158. $project?->agent_id,
  159. $data['use_balance']
  160. );
  161. // 6. 验证金额和余额
  162. abort_if($amounts['total_amount'] <= 0, 400, '订单金额异常');
  163. if ($data['payment_type'] == PaymentMethod::BALANCE->value) {
  164. $wallet = $user->wallet;
  165. abort_if($wallet->available_balance < $amounts['balance_amount'], 400, '可用余额不足');
  166. }
  167. // 项目服务时长(分钟)
  168. $data['duration'] = $project->duration;
  169. // 7. 创建订单记录
  170. $order = $this->createOrderRecord($userId, $data, $orderType, $address, $data['payment_type'], (object) $amounts);
  171. // 8. 余额支付处理
  172. if ($order->payment_type == PaymentMethod::BALANCE->value && $orderType != OrderType::GRAB->value) {
  173. $this->handleBalancePayment($user, $order, $orderType);
  174. }
  175. return [
  176. 'order_id' => $order->id,
  177. 'payment_type' => $order->payment_type,
  178. ];
  179. });
  180. }
  181. // 提取方法:验证技师
  182. public function validateCoach(int $coachId): CoachUser
  183. {
  184. return CoachUser::where('id', $coachId)
  185. ->where('state', TechnicianStatus::ACTIVE->value)
  186. ->whereHas('info', fn ($q) => $q->where('state', TechnicianAuthStatus::PASSED->value))
  187. ->whereHas('qual', fn ($q) => $q->where('state', TechnicianAuthStatus::PASSED->value))
  188. ->whereHas('real', fn ($q) => $q->where('state', TechnicianAuthStatus::PASSED->value))
  189. ->firstOrFail();
  190. }
  191. // 提取方法:获取原始订单
  192. private function getOriginalOrder($user, $orderId): Order
  193. {
  194. $originalOrder = $user->orders->where('id', $orderId)
  195. ->whereIn('state', [OrderStatus::SERVING->value, OrderStatus::FINISHED->value])
  196. ->firstOrFail();
  197. return $originalOrder;
  198. }
  199. // 提取方法:准备加钟订单数据
  200. private function prepareAddTimeData($originalOrder, $data): array
  201. {
  202. if ($originalOrder->state == OrderStatus::SERVING->value) {
  203. $startTime = now();
  204. } else {
  205. $startTime = now();
  206. }
  207. return [
  208. ...$data,
  209. 'order_id' => $data['order_id'],
  210. 'address_id' => $originalOrder->address_id,
  211. 'service_time' => $startTime,
  212. 'coach_id' => $originalOrder->coach_id,
  213. ];
  214. }
  215. // 提取方法:创建订单记录
  216. private function createOrderRecord($userId, $data, $orderType, $address, $payment_type, object $amounts): Order
  217. {
  218. // 计算服务开始和结束时间
  219. $serviceStartTime = Carbon::parse($data['service_time']);
  220. $serviceEndTime = $serviceStartTime->copy()->addMinutes($data['duration']);
  221. // TODO: 优化结构
  222. $order = new Order;
  223. $order->user_id = $userId;
  224. $order->order_no = 'O'.Carbon::now()->format('YmdHis').rand(1000, 9999);
  225. $order->project_id = $data['project_id'];
  226. $order->coach_id = $data['coach_id'];
  227. $order->type = $orderType;
  228. $order->state = OrderStatus::CREATED->value;
  229. $order->source = OrderSource::PLATFORM->value;
  230. $order->total_amount = $amounts->total_amount;
  231. $order->balance_amount = $amounts->balance_amount;
  232. $order->pay_amount = $amounts->pay_amount;
  233. $order->project_amount = $amounts->project_amount;
  234. $order->traffic_amount = $orderType == OrderType::OVERTIME->value ? 0 : $amounts->delivery_fee;
  235. $order->payment_type = ($data['use_balance'] && $amounts->pay_amount == 0) ? PaymentMethod::BALANCE->value : $payment_type;
  236. $order->service_time = $data['service_time'];
  237. $order->address_id = $data['address_id'];
  238. $order->longitude = $address->longitude;
  239. $order->latitude = $address->latitude;
  240. $order->location = $address->location;
  241. $order->address = $address->address;
  242. $order->area_code = $address->area_code;
  243. $order->service_start_time = $serviceStartTime;
  244. $order->service_end_time = $serviceEndTime;
  245. $order->save();
  246. // 创建订单记录
  247. OrderRecord::create([
  248. 'order_id' => $order->id,
  249. 'object_id' => $userId,
  250. 'object_type' => MemberUser::class,
  251. 'state' => OrderStatus::CREATED->value,
  252. 'remark' => $orderType == OrderType::OVERTIME->value ? '加钟订单' : '创建订单',
  253. ]);
  254. if ($orderType == OrderType::GRAB->value) {
  255. // 将订单地址经纬度存入redis的geo类型
  256. Redis::geoadd(
  257. 'order_locations',
  258. $order->longitude,
  259. $order->latitude,
  260. 'order:'.$order->id
  261. );
  262. }
  263. return $order;
  264. }
  265. // 提取方法:处理余额支付
  266. private function handleBalancePayment($user, $order, $orderType): void
  267. {
  268. $order->state = $orderType == OrderType::VISIT->value || $orderType == OrderType::GRAB->value ? OrderStatus::PAID->value : OrderStatus::SERVING->value;
  269. $order->save();
  270. OrderRecord::create([
  271. 'order_id' => $order->id,
  272. 'object_id' => $user->id,
  273. 'object_type' => MemberUser::class,
  274. 'state' => OrderRecordStatus::PAID->value,
  275. 'remark' => '余额支付',
  276. ]);
  277. $user->wallet->decrement('total_balance', $order->balance_amount);
  278. $user->wallet->decrement('available_balance', $order->balance_amount);
  279. }
  280. /**
  281. * 取消订单
  282. */
  283. public function cancelOrder(int $userId, int $orderId): array
  284. {
  285. return DB::transaction(function () use ($userId, $orderId) {
  286. try {
  287. // 1. 验证用户和订单
  288. $order = $this->validateOrderForCancel($userId, $orderId);
  289. abort_if($order->state == OrderStatus::CANCELLED->value, 400, '订单已取消');
  290. // 2. 处理退款
  291. if (in_array($order->state, [OrderStatus::PAID->value, OrderStatus::ACCEPTED->value, OrderStatus::DEPARTED->value])) {
  292. $this->handleCancelRefund($order);
  293. }
  294. // 3. 完成订单取消
  295. $this->completeCancel($order, $userId);
  296. // 4. 通知技师
  297. if ($order->coach_id) {
  298. // event(new OrderCancelledEvent($order));
  299. }
  300. return ['message' => '订单已取消'];
  301. } catch (Exception $e) {
  302. $this->logCancelOrderError($e, $userId, $orderId);
  303. throw $e;
  304. }
  305. });
  306. }
  307. /**
  308. * 验证订单取消条件
  309. */
  310. private function validateOrderForCancel(int $userId, int $orderId): Order
  311. {
  312. // 复用之前的用户验证逻辑
  313. $user = MemberUser::where('id', $userId)
  314. ->where('state', UserStatus::OPEN->value)
  315. ->firstOrFail();
  316. // 验证订单状态
  317. $order = Order::where('user_id', $userId)
  318. ->where('id', $orderId)
  319. ->whereIn('state', [OrderStatus::CREATED->value, OrderStatus::ASSIGNED->value, OrderStatus::PAID->value, OrderStatus::ACCEPTED->value, OrderStatus::DEPARTED->value])
  320. ->lockForUpdate()
  321. ->firstOrFail();
  322. return $order;
  323. }
  324. /**
  325. * 处理订单取消退款
  326. */
  327. private function handleCancelRefund(Order $order): void
  328. {
  329. $user = $order->user;
  330. switch ($order->state) {
  331. case OrderStatus::ACCEPTED->value: // 已接单
  332. // 扣除20%费用
  333. $deductAmount = ($order->payment_amount + $order->balance_amount - $order->traffic_amount) * 0.2;
  334. $this->handleRefund($user, $order, $deductAmount, false);
  335. break;
  336. case OrderStatus::DEPARTED->value: // 已出发
  337. // 扣除50%费用并扣除路费
  338. $deductAmount = ($order->payment_amount + $order->balance_amount - $order->traffic_amount) * 0.5;
  339. $this->handleRefund($user, $order, $deductAmount, true);
  340. break;
  341. case OrderStatus::PAID->value: // 已支付
  342. // 返还所有金额
  343. $deductAmount = 0;
  344. $this->handleRefund($user, $order, $deductAmount, false);
  345. break;
  346. case OrderStatus::CREATED->value:
  347. // 待支付状态直接取消,无需退款
  348. break;
  349. default:
  350. abort(400, '当前订单状态不允许取消');
  351. }
  352. }
  353. /**
  354. * 完成订单取消
  355. */
  356. private function completeCancel(Order $order, int $userId): void
  357. {
  358. // 添加订单取消记录
  359. OrderRecord::create([
  360. 'order_id' => $order->id,
  361. 'object_id' => $userId,
  362. 'object_type' => MemberUser::class,
  363. 'state' => OrderRecordStatus::CANCELLED->value,
  364. 'remark' => '用户取消订单',
  365. ]);
  366. // 修改订单状态
  367. $order->state = OrderStatus::CANCELLED->value;
  368. $order->save();
  369. // 如果有技师,可能需要通知技师订单已取消
  370. if ($order->coach_id) {
  371. // TODO: 发送通知给技师
  372. // event(new OrderCancelledEvent($order));
  373. }
  374. }
  375. /**
  376. * 处理退款
  377. */
  378. private function handleRefund(MemberUser $user, Order $order, float $deductAmount, bool $deductTrafficFee): void
  379. {
  380. // 关键操作:计算实际退款金额
  381. $refundAmount = $order->payment_amount + $order->balance_amount;
  382. if ($deductTrafficFee) {
  383. $refundAmount -= $order->traffic_amount;
  384. // TODO: 记录技师路费收入
  385. }
  386. $refundAmount -= $deductAmount;
  387. // 优先余额支付金额中扣除
  388. $balanceRefund = min($order->balance_amount, $refundAmount);
  389. if ($balanceRefund > 0) {
  390. $this->createRefundRecords($user, $order, $balanceRefund);
  391. }
  392. // 剩余退款金额从支付金额中退还
  393. $paymentRefund = $refundAmount - $balanceRefund;
  394. if ($paymentRefund > 0) {
  395. $this->createRefundRecords($user, $order, $paymentRefund);
  396. }
  397. // 记录平台收入
  398. if ($deductAmount > 0) {
  399. // TODO: 添加平台收入记录
  400. // PlatformIncome::create([...]);
  401. }
  402. }
  403. /**
  404. * 创建退款记录
  405. */
  406. private function createRefundRecords($user, $order, $amount, $type = 'balance'): void
  407. {
  408. $refundMethod = $type;
  409. $remark = $type == 'balance' ? '订单取消退还余额' : '订单取消退还支付金额';
  410. // 创建退款记录
  411. $refundRecord = $user->wallet->refundRecords()->create([
  412. 'refund_method' => $refundMethod,
  413. 'total_refund_amount' => $order->payment_amount + $order->balance_amount,
  414. 'actual_refund_amount' => '0.00',
  415. 'wallet_balance_refund_amount' => $type == 'balance' ? $amount : '0.00',
  416. 'recharge_balance_refund_amount' => '0.00',
  417. 'remark' => $remark,
  418. 'order_id' => $order->id,
  419. ]);
  420. // 创建交易记录
  421. $user->wallet->transRecords()->create([
  422. 'amount' => $amount,
  423. 'owner_type' => get_class($refundRecord),
  424. 'owner_id' => $refundRecord->id,
  425. 'remark' => $remark,
  426. 'trans_type' => 'income',
  427. 'storage_type' => 'balance',
  428. 'before_balance' => $user->wallet->total_balance,
  429. 'after_balance' => $user->wallet->total_balance + $amount,
  430. 'before_recharge_balance' => '0.00',
  431. 'after_recharge_balance' => '0.00',
  432. 'trans_time' => now(),
  433. 'state' => 'success',
  434. ]);
  435. // 更新钱包余额
  436. $user->wallet->increment('total_balance', $amount);
  437. $user->wallet->increment('available_balance', $amount);
  438. $user->wallet->save();
  439. }
  440. /**
  441. * 记录订单取消错误日志
  442. */
  443. private function logCancelOrderError(Exception $e, int $userId, int $orderId): void
  444. {
  445. // 复用之前的日记录方法
  446. Log::error('取消订单失败:', [
  447. 'message' => $e->getMessage(),
  448. 'user_id' => $userId,
  449. 'order_id' => $orderId,
  450. 'trace' => $e->getTraceAsString(),
  451. ]);
  452. }
  453. /**
  454. * 结束订单
  455. */
  456. public function finishOrder(int $userId, int $orderId): array
  457. {
  458. return DB::transaction(function () use ($userId, $orderId) {
  459. try {
  460. // 1. 验证用户和订单
  461. $order = $this->validateOrderForFinish($userId, $orderId);
  462. // 2. 验证技师状态
  463. $coach = $this->validateCoach($order->coach_id);
  464. // 4. 完成订单
  465. $this->completeOrder($order, $userId);
  466. // 5. 通知技师
  467. // event(new OrderFinishedEvent($order));
  468. return ['message' => '订单已完成'];
  469. } catch (Exception $e) {
  470. $this->logFinishOrderError($e, $userId, $orderId);
  471. throw $e;
  472. }
  473. });
  474. }
  475. /**
  476. * 验证订单完成条件
  477. */
  478. private function validateOrderForFinish(int $userId, int $orderId): Order
  479. {
  480. // 验证用户状态
  481. $user = MemberUser::where('id', $userId)
  482. ->where('state', UserStatus::OPEN->value)
  483. ->firstOrFail();
  484. // 验证订单状态
  485. $order = Order::where('user_id', $userId)
  486. ->where('id', $orderId)
  487. ->where('state', OrderStatus::SERVING->value)
  488. ->lockForUpdate()
  489. ->firstOrFail();
  490. return $order;
  491. }
  492. /**
  493. * 完成订单
  494. */
  495. private function completeOrder(Order $order, int $userId): void
  496. {
  497. // 1. 创建订单记录
  498. OrderRecord::create([
  499. 'order_id' => $order->id,
  500. 'object_id' => $userId,
  501. 'object_type' => MemberUser::class,
  502. 'state' => OrderRecordStatus::COMPLETED->value,
  503. 'remark' => '服务完成',
  504. ]);
  505. // 2. 更新订单状态
  506. $order->state = OrderStatus::FINISHED->value;
  507. $order->save();
  508. }
  509. /**
  510. * 记录订单完成错误日志
  511. */
  512. private function logFinishOrderError(Exception $e, int $userId, int $orderId): void
  513. {
  514. Log::error('结束订单失败:', [
  515. 'message' => $e->getMessage(),
  516. 'user_id' => $userId,
  517. 'order_id' => $orderId,
  518. 'trace' => $e->getTraceAsString(),
  519. ]);
  520. }
  521. /**
  522. * 确认技师离开
  523. */
  524. public function confirmLeave(int $userId, int $orderId): array
  525. {
  526. return DB::transaction(function () use ($userId, $orderId) {
  527. try {
  528. // 1. 参数校验
  529. $order = Order::where('user_id', $userId)
  530. ->where('id', $orderId)
  531. ->where('state', OrderStatus::FINISHED->value) // 订单状态必须是服务结束
  532. ->firstOrFail();
  533. if (! $order) {
  534. throw new Exception('订单不能撤离');
  535. }
  536. // 2. 添加订单撤离记录
  537. OrderRecord::create([
  538. 'order_id' => $orderId,
  539. 'object_id' => $userId,
  540. 'object_type' => MemberUser::class,
  541. 'state' => OrderRecordStatus::LEFT->value,
  542. 'remark' => '技师已离开',
  543. ]);
  544. // 3. 修改订单状态为撤离
  545. $order->state = OrderStatus::LEFT->value;
  546. $order->save();
  547. return ['message' => '已确技师离开'];
  548. } catch (Exception $e) {
  549. Log::error('确认技师离开失败:', [
  550. 'message' => $e->getMessage(),
  551. 'user_id' => $userId,
  552. 'order_id' => $orderId,
  553. ]);
  554. throw $e;
  555. }
  556. });
  557. }
  558. /**
  559. * 获取订单列表
  560. */
  561. public function getOrderList(int $user_id): \Illuminate\Contracts\Pagination\LengthAwarePaginator
  562. {
  563. $user = MemberUser::find($user_id);
  564. return $user->orders()
  565. ->with([
  566. 'coach.info:id,nickname,avatar,gender',
  567. ])
  568. ->orderBy('created_at', 'desc')
  569. ->paginate(10);
  570. }
  571. /**
  572. * 获取订单详情
  573. */
  574. public function getOrderDetail(int $userId, int $orderId): Order
  575. {
  576. $user = MemberUser::find($userId);
  577. return $user->orders()
  578. ->where('id', $orderId) // 需要添加订单ID条件
  579. ->with([
  580. 'coach.info:id,nickname,avatar,gender',
  581. 'records' => function ($query) {
  582. $query->orderBy('created_at', 'asc');
  583. },
  584. ])
  585. ->firstOrFail();
  586. }
  587. /**
  588. * 订单退款
  589. */
  590. public function refundOrder(int $orderId): array
  591. {
  592. // 使用 Auth::user() 获取用户对象
  593. $user = Auth::user();
  594. return DB::transaction(function () use ($orderId, $user) {
  595. // 查询并锁定订单
  596. $order = Order::where('id', $orderId)
  597. ->where('user_id', $user->id)
  598. ->where('state', 'pending')
  599. ->lockForUpdate()
  600. ->firstOrFail();
  601. // 更新订单状态
  602. $order->state = 'refunded';
  603. $order->save();
  604. // 添加订单记录
  605. OrderRecord::create([
  606. 'order_id' => $orderId,
  607. 'object_id' => $user->id,
  608. 'object_type' => 'user',
  609. 'state' => 'refund',
  610. 'remark' => '订单退款',
  611. ]);
  612. // 创建退款记录
  613. WalletRefundRecord::create([
  614. 'order_id' => $orderId,
  615. 'user_id' => $user->id,
  616. 'amount' => $order->total_amount,
  617. 'state' => 'success',
  618. ]);
  619. return ['message' => '退款成功'];
  620. });
  621. }
  622. /**
  623. * 获取代理商配置
  624. */
  625. public function getAgentConfig(int $agentId): array
  626. {
  627. $agent = AgentInfo::where('id', $agentId)
  628. ->where('state', 'enable')
  629. ->firstOrFail();
  630. // $config = AgentConfig::where('agent_id', $agentId)->firstOrFail();
  631. return [
  632. // 'min_distance' => $config->min_distance,
  633. // 'min_fee' => $config->min_fee,
  634. // 'per_km_fee' => $config->per_km_fee
  635. ];
  636. }
  637. /**
  638. * 获取技师配置
  639. */
  640. public function getCoachConfig(int $coachId): array
  641. {
  642. $coach = CoachUser::where('id', $coachId)
  643. ->where('state', 'enable')
  644. ->where('auth_state', 'passed')
  645. ->firstOrFail();
  646. // $config = CoachConfig::where('coach_id', $coachId)->firstOrFail();
  647. return [
  648. // 'delivery_fee_type' => $config->delivery_fee_type,
  649. // 'charge_delivery_fee' => $config->charge_delivery_fee
  650. ];
  651. }
  652. /**
  653. * 计算路费金额
  654. *
  655. * @param int $coachId 技师ID
  656. * @param int $projectId 项目ID
  657. * @param int|null $agentId 代理商ID
  658. * @param float $distance 距离(公里)
  659. * @return float 路费金额
  660. *
  661. * @throws Exception
  662. */
  663. public function calculateDeliveryFee(
  664. int $coachId,
  665. int $projectId,
  666. ?int $agentId,
  667. float $distance
  668. ): float {
  669. try {
  670. // 1. 校验技师
  671. $coach = CoachUser::where('state', TechnicianStatus::ACTIVE->value)
  672. ->whereHas('info', fn ($q) => $q->where('state', TechnicianAuthStatus::PASSED->value))
  673. ->whereHas('real', fn ($q) => $q->where('state', TechnicianAuthStatus::PASSED->value))
  674. ->whereHas('qual', fn ($q) => $q->where('state', TechnicianAuthStatus::PASSED->value))
  675. ->with(['projects' => fn ($q) => $q->where('project_id', $projectId)])
  676. ->find($coachId);
  677. abort_if(! $coach, 404, '技师不存在或状态异常');
  678. // 2. 校验技师项目
  679. $coachProject = $coach->projects->first();
  680. abort_if(! $coachProject, 404, '技师项目不存在');
  681. // 3. 判断是否免收路费
  682. if ($coachProject->traffic_fee_type == 'free') {
  683. return 0;
  684. }
  685. // 4. 获取路费配置
  686. $config = $this->getDeliveryFeeConfig($agentId);
  687. abort_if(! $config, 404, '路费配置不存在');
  688. // 5. 计算路费
  689. $fee = $this->calculateFee($distance, $config);
  690. // 6. 判断是否往返
  691. return $coachProject->delivery_fee_type == 'round_trip'
  692. ? bcmul($fee, '2', 2)
  693. : $fee;
  694. } catch (Exception $e) {
  695. Log::error(__CLASS__.'->'.__FUNCTION__.'计算路费失败:', [
  696. 'message' => $e->getMessage(),
  697. 'coach_id' => $coachId,
  698. 'project_id' => $projectId,
  699. 'agent_id' => $agentId,
  700. 'distance' => $distance,
  701. 'trace' => $e->getTraceAsString(),
  702. ]);
  703. throw $e;
  704. }
  705. }
  706. /**
  707. * 获取路费配置
  708. */
  709. private function getDeliveryFeeConfig(?int $agentId): ?object
  710. {
  711. // 优先获取代理商配置
  712. if ($agentId) {
  713. $agent = AgentInfo::where('state', 'enable')
  714. ->with(['projectConfig'])
  715. ->find($agentId);
  716. if ($agent && $agent->projectConfig) {
  717. return $agent->projectConfig;
  718. }
  719. }
  720. // 获取系统配置
  721. return SysConfig::where('key', 'delivery_fee')->first();
  722. }
  723. /**
  724. * 计算路费
  725. */
  726. private function calculateFee(float $distance, object $config): float
  727. {
  728. // 最小距离内按起步价计算
  729. if ($distance <= $config->min_distance) {
  730. return (float) $config->min_fee;
  731. }
  732. // 超出最小距离部分按每公里费用计算
  733. $extraDistance = bcsub($distance, $config->min_distance, 2);
  734. $extraFee = bcmul($extraDistance, $config->per_km_fee, 2);
  735. return bcadd($config->min_fee, $extraFee, 2);
  736. }
  737. /**
  738. * 计算订单金额
  739. *
  740. * @param int $userId 用户ID
  741. * @param int $addressId 地址ID
  742. * @param int $coachId 技师ID
  743. * @param int $projectId 项目ID
  744. * @param int $agentId 代理商ID
  745. * @param bool $useBalance 是否使用余额
  746. * @param float $distance 距离
  747. * @param int $lat 纬度
  748. * @param int $lng 经度
  749. *
  750. * @throws Exception
  751. */
  752. public function calculateOrderAmount(
  753. int $userId,
  754. int $addressId,
  755. ?int $coachId,
  756. int $projectId,
  757. ?int $agentId = null,
  758. bool $useBalance = false,
  759. float $distance = 0,
  760. int $lat = 0,
  761. int $lng = 0
  762. ): array {
  763. try {
  764. // 1. 参数校验
  765. $user = MemberUser::find($userId);
  766. abort_if(! $user || $user->state != UserStatus::OPEN->value, 404, '用户不存在或状态异常');
  767. // 2. 查询技师项目
  768. $coach = $coachId ? $this->validateCoach($coachId) : null;
  769. $coachProject = $coach ? $coach->projects()
  770. ->where('state', ProjectStatus::OPEN->value)
  771. ->where('project_id', $projectId)
  772. ->first() : null;
  773. // 3. 查询基础项目
  774. $project = Project::where('id', $projectId)
  775. ->where('state', ProjectStatus::OPEN->value())
  776. ->first();
  777. abort_if(! $project, 404, '项目不存在或状态异常');
  778. // 4. 计算距离
  779. if (floatval($distance) <= 0 && $coachId) {
  780. $address = $user->addresses()->find($addressId) ?? ['latitude' => $lat, 'longitude' => $lng];
  781. $coachService = app(CoachService::class);
  782. $coachDetail = $coachService->getCoachDetail($coachId, $address['latitude'], $address['longitude']);
  783. $distance = $coachDetail['distance'] ?? 0;
  784. }
  785. // 5. 获取项目价格
  786. $projectAmount = $this->getProjectPrice($project, $agentId, $projectId);
  787. // 6. 计算路费
  788. $deliveryFee = $coachId ? $this->calculateDeliveryFee($coachId, $projectId, $agentId, $distance) : 0;
  789. // 7. 计算优惠券金额
  790. $couponAmount = $this->calculateCouponAmount();
  791. // 8. 计算总金额
  792. $totalAmount = bcadd($projectAmount, $deliveryFee, 2);
  793. $totalAmount = bcsub($totalAmount, $couponAmount, 2);
  794. $totalAmount = max(0, $totalAmount);
  795. // 9. 计算余额支付金额
  796. $balanceAmount = 0;
  797. $payAmount = $totalAmount;
  798. if ($useBalance && $totalAmount > 0) {
  799. $wallet = $user->wallet;
  800. abort_if(! $wallet, 404, '用户钱包不存在');
  801. if ($wallet->available_balance >= $totalAmount) {
  802. $balanceAmount = $totalAmount;
  803. $payAmount = 0;
  804. } else {
  805. $balanceAmount = $wallet->available_balance;
  806. $payAmount = bcsub($totalAmount, $balanceAmount, 2);
  807. }
  808. }
  809. return [
  810. 'total_amount' => $totalAmount,
  811. 'balance_amount' => $balanceAmount,
  812. 'pay_amount' => $payAmount,
  813. 'coupon_amount' => $couponAmount,
  814. 'project_amount' => $projectAmount,
  815. 'delivery_fee' => $deliveryFee,
  816. ];
  817. } catch (Exception $e) {
  818. Log::error(__CLASS__.'->'.__FUNCTION__.'计算订单金额失败:', [
  819. 'message' => $e->getMessage(),
  820. 'user_id' => $userId,
  821. 'project_id' => $projectId,
  822. 'trace' => $e->getTraceAsString(),
  823. ]);
  824. throw $e;
  825. }
  826. }
  827. /**
  828. * 获取项目价格
  829. */
  830. private function getProjectPrice($project, ?int $agentId, int $projectId): float
  831. {
  832. $price = $project->price;
  833. if ($agentId) {
  834. $agent = AgentInfo::where('state', 'enable')->find($agentId);
  835. if ($agent) {
  836. $agentProject = $agent->projects()
  837. ->where('state', 'enable')
  838. ->where('project_id', $projectId)
  839. ->first();
  840. if ($agentProject) {
  841. $price = $agentProject->price;
  842. }
  843. }
  844. }
  845. return (float) $price;
  846. }
  847. /**
  848. * 计算优惠券金额
  849. */
  850. private function calculateCouponAmount(): float
  851. {
  852. $couponAmount = 0;
  853. if (request()->has('coupon_id')) {
  854. // TODO: 优惠券逻辑
  855. }
  856. return $couponAmount;
  857. }
  858. /**
  859. * 计算支付金额分配
  860. */
  861. private function calculatePaymentAmounts($user, float $totalAmount, bool $useBalance): array
  862. {
  863. $balanceAmount = 0;
  864. $payAmount = $totalAmount;
  865. if ($useBalance) {
  866. $wallet = $user->wallet;
  867. if (! $wallet) {
  868. throw new Exception('用户钱包不存在');
  869. }
  870. if ($wallet->available_balance >= $totalAmount) {
  871. $balanceAmount = $totalAmount;
  872. $payAmount = 0;
  873. } else {
  874. $balanceAmount = $wallet->available_balance;
  875. $payAmount = bcsub($totalAmount, $balanceAmount, 2);
  876. }
  877. }
  878. return [$balanceAmount, $payAmount];
  879. }
  880. /**
  881. * 获取订单抢单池列表
  882. *
  883. * @param int $orderId 订单ID
  884. * @return array 抢单池列表
  885. */
  886. public function getOrderGrabList(int $orderId): array
  887. {
  888. try {
  889. // 查询订单信息
  890. $order = Order::where('id', $orderId)
  891. ->whereIn('state', [OrderStatus::CREATED->value])
  892. ->firstOrFail();
  893. // 查询抢单池列表
  894. $grabList = $order->grabRecords()->with(['coach.info'])->get();
  895. // 格式化返回数据
  896. $result = [];
  897. foreach ($grabList as $grab) {
  898. $coach = $grab->coach;
  899. $result[] = [
  900. 'id' => $grab->id,
  901. 'coach_id' => $coach->id,
  902. 'nickname' => $coach->info->nickname,
  903. 'avatar' => $coach->info->avatar,
  904. 'distance' => $grab->distance,
  905. 'created_at' => $grab->created_at->format('Y-m-d H:i:s'),
  906. ];
  907. }
  908. return $result;
  909. } catch (\Exception $e) {
  910. Log::error('获取订单抢单池列表失败', [
  911. 'error' => $e->getMessage(),
  912. 'order_id' => $orderId,
  913. ]);
  914. throw $e;
  915. }
  916. }
  917. /**
  918. * 指定技师
  919. */
  920. public function assignCoach(int $userId, int $orderId, int $coachId): bool
  921. {
  922. return DB::transaction(function () use ($userId, $orderId, $coachId) {
  923. try {
  924. // 1. 验证参数
  925. $order = $this->validateAssignOrder($userId, $orderId);
  926. // 2. 验证技师
  927. $coach = $this->validateCoach($coachId);
  928. // 3. 检查抢单池是否已有抢单成功记录
  929. $existsGrabSuccess = $order->grabRecords()
  930. ->where('state', OrderGrabRecordStatus::SUCCEEDED->value)
  931. ->exists();
  932. abort_if($existsGrabSuccess, 400, '该订单已抢单完成');
  933. // 4. 更新订单信息
  934. $this->updateOrderForAssign($order, $coachId);
  935. // 5. 创建订单记录(指派)
  936. $this->createAssignRecord($order, $userId);
  937. // 6. 处理支付
  938. if ($order->payment_type === PaymentMethod::BALANCE->value) {
  939. $this->handleBalancePaymentForAssign($order, $userId, $coachId);
  940. }
  941. return true;
  942. } catch (Exception $e) {
  943. $this->logAssignCoachError($e, $userId, $orderId, $coachId);
  944. throw $e;
  945. }
  946. });
  947. }
  948. /**
  949. * 创建指派记录
  950. */
  951. private function createAssignRecord(Order $order, int $userId): void
  952. {
  953. OrderRecord::create([
  954. 'order_id' => $order->id,
  955. 'object_id' => $userId,
  956. 'object_type' => MemberUser::class,
  957. 'state' => OrderRecordStatus::ASSIGNED->value,
  958. 'remark' => '指定技师',
  959. ]);
  960. }
  961. /**
  962. * 处理指派订单的余额支付
  963. */
  964. private function handleBalancePaymentForAssign(Order $order, int $userId, int $coachId): void
  965. {
  966. // 验证余额
  967. $user = MemberUser::find($userId);
  968. $wallet = $user->wallet;
  969. abort_if($wallet->available_balance < $order->balance_amount, 400, '可用余额不足');
  970. // 扣除余额
  971. $wallet->decrement('total_balance', $order->balance_amount);
  972. $wallet->decrement('available_balance', $order->balance_amount);
  973. // 更新订单状态
  974. $order->update(['state' => OrderStatus::PAID->value]);
  975. // 创建钱包支付记录
  976. WalletPaymentRecord::create([
  977. 'order_id' => $order->id,
  978. 'wallet_id' => $wallet->id,
  979. 'payment_no' => 'balance_'.$order->id,
  980. 'payment_method' => 'balance',
  981. 'total_amount' => $order->balance_amount,
  982. 'actual_amount' => 0,
  983. 'used_wallet_balance' => $order->balance_amount,
  984. 'used_recharge_balance' => 0,
  985. 'state' => 'success',
  986. ]);
  987. // 创建接单记录
  988. // OrderRecord::create([
  989. // 'order_id' => $order->id,
  990. // 'object_id' => $userId,
  991. // 'object_type' => MemberUser::class,
  992. // 'state' => OrderRecordStatus::ASSIGNED->value,
  993. // 'remark' => '已分配技师',
  994. // ]);
  995. // 创建支付成功记录
  996. OrderRecord::create([
  997. 'order_id' => $order->id,
  998. 'object_id' => $userId,
  999. 'object_type' => MemberUser::class,
  1000. 'state' => OrderRecordStatus::PAID->value,
  1001. 'remark' => '余额支付成功',
  1002. ]);
  1003. // 更新抢单记录
  1004. $this->updateGrabRecords($order->id, $coachId);
  1005. }
  1006. /**
  1007. * 验证指定技师的订单条件
  1008. */
  1009. private function validateAssignOrder(int $userId, int $orderId): Order
  1010. {
  1011. // 验证用户状态
  1012. $user = MemberUser::where('id', $userId)
  1013. ->where('state', UserStatus::OPEN->value)
  1014. ->firstOrFail();
  1015. // 验证订单状态
  1016. $order = $user->orders()
  1017. ->where('id', $orderId)
  1018. ->whereIn('state', [OrderStatus::CREATED->value, OrderStatus::PAID->value])
  1019. ->lockForUpdate()
  1020. ->firstOrFail();
  1021. return $order;
  1022. }
  1023. /**
  1024. * 更新订单信息
  1025. */
  1026. private function updateOrderForAssign(Order $order, int $coachId): void
  1027. {
  1028. // 修改订单技师
  1029. $order->coach_id = $coachId;
  1030. // 待支付订单需要重新计算金额
  1031. if ($order->state == OrderStatus::CREATED->value) {
  1032. $amounts = $this->calculateOrderAmount(
  1033. $order->user_id,
  1034. $order->address_id,
  1035. $coachId,
  1036. $order->project_id,
  1037. $order->agent_id,
  1038. $order->payment_type === PaymentMethod::BALANCE->value
  1039. );
  1040. // 更新订单金额
  1041. $order->total_amount = $amounts['total_amount'];
  1042. $order->balance_amount = $amounts['balance_amount'];
  1043. $order->pay_amount = $amounts['pay_amount'];
  1044. $order->discount_amount = $amounts['coupon_amount'];
  1045. $order->project_amount = $amounts['project_amount'];
  1046. $order->traffic_amount = $amounts['delivery_fee'];
  1047. }
  1048. $order->save();
  1049. }
  1050. /**
  1051. * 更新抢单记录
  1052. */
  1053. private function updateGrabRecords(int $orderId, int $coachId): void
  1054. {
  1055. OrderGrabRecord::where('order_id', $orderId)
  1056. ->update([
  1057. 'state' => OrderGrabRecordStatus::SUCCEEDED->value,
  1058. 'coach_id' => $coachId,
  1059. ]);
  1060. }
  1061. /**
  1062. * 记录指派技师错误日志
  1063. */
  1064. private function logAssignCoachError(Exception $e, int $userId, int $orderId, int $coachId): void
  1065. {
  1066. Log::error('分配技师失败:', [
  1067. 'message' => $e->getMessage(),
  1068. 'user_id' => $userId,
  1069. 'order_id' => $orderId,
  1070. 'coach_id' => $coachId,
  1071. ]);
  1072. }
  1073. /**
  1074. * 生成订单核销码
  1075. *
  1076. * @param int $userId 用户ID
  1077. * @param int $orderId 订单ID
  1078. * @return array 返回核销码信息
  1079. *
  1080. * @throws Exception
  1081. */
  1082. public function generateVerificationCode(int $userId, int $orderId): array
  1083. {
  1084. try {
  1085. // 1. 验证用户
  1086. $user = MemberUser::where('id', $userId)
  1087. ->where('state', UserStatus::OPEN->value)
  1088. ->firstOrFail();
  1089. // 2. 验证订单状态和归属
  1090. $order = $user->orders()
  1091. ->where('id', $orderId)
  1092. ->whereIn('state', [
  1093. OrderStatus::PAID->value,
  1094. OrderStatus::ACCEPTED->value,
  1095. OrderStatus::DEPARTED->value,
  1096. OrderStatus::ARRIVED->value,
  1097. ])
  1098. ->firstOrFail();
  1099. // 3. 生成时间戳
  1100. $timestamp = time();
  1101. // 4. 生成签名
  1102. $sign = md5("order_{$order->id}_{$timestamp}_".config('app.key'));
  1103. // 5. 组装二维码内容
  1104. $qrCode = "order_{$order->id}_{$timestamp}_{$sign}";
  1105. // 6. 生成二维码图片
  1106. try {
  1107. // 生成JPEG格式的二维码
  1108. $qrCodeImage = QrCode::format('png')
  1109. // 设置二维码大小为200px
  1110. ->size(200)
  1111. // 设置二维码边距为1
  1112. ->margin(1)
  1113. // 设置纠错级别为最高级别H
  1114. ->errorCorrection('H')
  1115. // 生成二维码图片
  1116. ->generate($qrCode);
  1117. // 将JPEG转为base64
  1118. $qrCodeBase64 = 'data:image/png;base64,'.base64_encode($qrCodeImage);
  1119. } catch (Exception $e) {
  1120. Log::error('生成二维码图片失败:', [
  1121. 'order_id' => $orderId,
  1122. 'error' => $e->getMessage(),
  1123. ]);
  1124. // 如果二维码生成失败,仍然返回文本内容
  1125. $qrCodeBase64 = null;
  1126. }
  1127. // 7. 返回结果
  1128. return [
  1129. 'order_id' => $order->id,
  1130. 'qr_code' => $qrCode,
  1131. 'qr_image' => $qrCodeBase64,
  1132. 'expired_at' => date('Y-m-d H:i:s', $timestamp + 300),
  1133. 'state' => $order->state,
  1134. ];
  1135. } catch (Exception $e) {
  1136. Log::error('生成订单核销码失败:', [
  1137. 'user_id' => $userId,
  1138. 'order_id' => $orderId,
  1139. 'error' => $e->getMessage(),
  1140. 'trace' => $e->getTraceAsString(),
  1141. ]);
  1142. throw $e;
  1143. }
  1144. }
  1145. /**
  1146. * 验证核销码
  1147. *
  1148. * @param string $qrCode 核销码
  1149. * @param int $orderId 订单ID
  1150. *
  1151. * @throws Exception
  1152. */
  1153. public function verifyCode(string $qrCode, int $orderId): bool
  1154. {
  1155. try {
  1156. // 1. 验证订单
  1157. $order = Order::where('id', $orderId)
  1158. ->where('state', OrderStatus::PAID->value)
  1159. ->firstOrFail();
  1160. // 2. 解析二维码
  1161. // 二维码格式: order_{order_id}_{timestamp}_{sign}
  1162. $parts = explode('_', $qrCode);
  1163. abort_if(count($parts) !== 4, 400, '二维码格式错误');
  1164. [$prefix, $scanOrderId, $timestamp, $sign] = $parts;
  1165. // 3. 验证前缀
  1166. abort_if($prefix !== 'order', 400, '无效的二维码');
  1167. // 4. 验证订单ID
  1168. abort_if((int) $scanOrderId !== $order->id, 400, '二维码与订单不匹配');
  1169. // 5. 验证时间戳(二维码5分钟内有效)
  1170. $qrTimestamp = (int) $timestamp;
  1171. $now = time();
  1172. abort_if($now - $qrTimestamp > 300, 400, '二维码已过期');
  1173. // 6. 验证签名
  1174. $correctSign = md5("order_{$order->id}_{$timestamp}_".config('app.key'));
  1175. abort_if($sign !== $correctSign, 400, '二维码签名错误');
  1176. return true;
  1177. } catch (Exception $e) {
  1178. Log::error('验证核销码失败:', [
  1179. 'order_id' => $orderId,
  1180. 'qr_code' => $qrCode,
  1181. 'error' => $e->getMessage(),
  1182. 'trace' => $e->getTraceAsString(),
  1183. ]);
  1184. throw $e;
  1185. }
  1186. }
  1187. }