WorkOrderService.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443
  1. <?php
  2. namespace App\Services;
  3. use App\Jobs\SyncRejectedBillRejectingStatusJob;
  4. use App\OrderIssue;
  5. use App\OrderIssueProcessLog;
  6. use App\OrderIssueRejectedBill;
  7. use App\OrderIssueType;
  8. use App\OrderPackage;
  9. use App\Traits\ServiceAppAop;
  10. use App\WorkOrder;
  11. use App\WorkOrderDetail;
  12. use Illuminate\Support\Facades\Auth;
  13. class WorkOrderService
  14. {
  15. use ServiceAppAop;
  16. protected $modelClass = WorkOrder::class;
  17. public $logService;
  18. public $imageService;
  19. public $commoditiesService;
  20. public $detailService;
  21. public $issueTypeService;
  22. public $orderService;
  23. public $orderIssueService;
  24. public $orderIssueProcessLogService;
  25. public function __construct(
  26. WorkOrderLogService $logService,
  27. WorkOrderImageService $imageService,
  28. WorkOrderCommoditiesService $commoditiesService,
  29. WorkOrderDetailService $detailService,
  30. OrderIssueTypeService $issueTypeService,
  31. OrderService $orderService,
  32. OrderIssueService $orderIssueService,
  33. OrderIssueProcessLogService $orderIssueProcessLogService
  34. )
  35. {
  36. $this->logService = $logService;
  37. $this->imageService = $imageService;
  38. $this->commoditiesService = $commoditiesService;
  39. $this->detailService = $detailService;
  40. $this->issueTypeService = $issueTypeService;
  41. $this->orderService = $orderService;
  42. $this->orderIssueService = $orderIssueService;
  43. $this->orderIssueProcessLogService = $orderIssueProcessLogService;
  44. }
  45. public function createOrResetWorkOrder($order, $issueType, $remark, $process_progress = '商家创建')
  46. {
  47. $workOrder = WorkOrder::query()->where('order_id', $order->id)->orderByDesc('created_at')->first();
  48. if ($workOrder) {
  49. $this->detailService->undoneTagsByWorkOrder($workOrder);
  50. $workOrder->update([
  51. 'remark' => $remark,
  52. 'order_issue_type_id' => $issueType->id,
  53. 'creator_id' => Auth::id() ?? '',
  54. 'status' => WorkOrder::$DEFAULT_STATUS,
  55. 'process_progress' => $process_progress,
  56. 'type' => $process_progress,
  57. 'last_handler_id' => Auth::id() ?? '',
  58. 'created_at' => now(),
  59. ]);
  60. return $workOrder;
  61. }
  62. return WorkOrder::query()->create([
  63. 'order_id' => $order->id,
  64. 'logistic_id' => $order->logistic_id ?? '',
  65. 'owner_id' => $order->owner_id ?? '',
  66. 'creator_id' => Auth::id() ?? '',
  67. 'remark' => $remark,
  68. 'outer_table_name' => 'orders',
  69. 'order_issue_type_id' => $issueType->id,
  70. 'uniquely_tag' => $order->code,
  71. 'status' => WorkOrder::$DEFAULT_STATUS,
  72. 'process_progress' => $process_progress,
  73. 'type' => $process_progress,
  74. 'last_handler_id' => Auth::id() ?? '',
  75. ]);
  76. }
  77. public function createAndNotification($order, $orderIssueType, $remark, $process_progress = '商家创建'): WorkOrder
  78. {
  79. /** @var WorkOrder $workOrder */
  80. $workOrder = $this->createOrResetWorkOrder($order, $orderIssueType, $remark, $process_progress);
  81. $workOrder->notification();
  82. SyncRejectedBillRejectingStatusJob::dispatch($workOrder->order);
  83. return $workOrder;
  84. }
  85. /**
  86. * @param WorkOrder $workOrder
  87. * @param WorkOrderDetail $detail
  88. */
  89. public function end(WorkOrder $workOrder, WorkOrderDetail $detail)
  90. {
  91. $this->detailService->endDetail($detail);
  92. $workOrder->update([
  93. 'status' => WorkOrder::$END_STATUS,
  94. 'owner_tag' => WorkOrder::$DEFAULT_TAG,
  95. 'logistic_tag' => WorkOrder::$DEFAULT_TAG,
  96. 'bao_shi_tag' => WorkOrder::$DEFAULT_TAG,
  97. 'work_order_status' => 0,
  98. ]);
  99. $this->logService->createLog($detail, '完结', '完结工单');
  100. }
  101. /**
  102. * 商家完结工单
  103. * @param WorkOrderDetail $detail
  104. */
  105. public function ownerEndWorkOrderDetail(WorkOrderDetail $detail)
  106. {
  107. $detail->update([
  108. 'status' => WorkOrder::$END_STATUS,
  109. 'process_progress' => '完成',
  110. 'last_status' => WorkOrder::$TO_BO_OWNER_END_STATUS,
  111. ]);
  112. $detail->workOrder()->update([
  113. 'status' => WorkOrder::$END_STATUS,
  114. 'process_progress' => '完成',
  115. 'last_status' => WorkOrder::$TO_BO_OWNER_END_STATUS,
  116. 'owner_tag' => WorkOrder::$DEFAULT_TAG,
  117. 'logistic_tag' => WorkOrder::$DEFAULT_TAG,
  118. 'bao_shi_tag' => WorkOrder::$DEFAULT_TAG,
  119. 'work_order_status' => 0,
  120. ]);
  121. $this->detailService->endDetail($detail); // 标记为处理过
  122. $this->logService->createLog($detail, '完结', '货主完结');
  123. }
  124. /**
  125. * 商家批量完结工单
  126. * @param $details
  127. */
  128. public function ownerBatchEndWorkOrderDetails($details)
  129. {
  130. foreach ($details as $detail) {
  131. $this->ownerEndWorkOrderDetail($detail);
  132. }
  133. }
  134. public function find($id): WorkOrder
  135. {
  136. /** @var WorkOrder $item */
  137. $item = WorkOrder::query()->where('id', $id)->first();
  138. return $item;
  139. }
  140. public function getDefaultWith($id)
  141. {
  142. return WorkOrder::query()->defaultWith()->find($id);
  143. }
  144. /**
  145. * 生成问题件
  146. * @param $work_orders
  147. * @return array
  148. */
  149. public function buildOrderIssue($work_orders): array
  150. {
  151. foreach ($work_orders as $work_order) {
  152. $inner_params[] = [
  153. 'order_id' => $work_order->order_id, 'order_issue_type_id' => $work_order->order_issue_type_id, 'result_explain' => $work_order->remark,
  154. ];
  155. }
  156. if (!isset($inner_params)) return ['success' => false, 'message' => '创建问题件失败'];
  157. try {
  158. return app('OrderIssueService')->buildOrderIssue($inner_params);
  159. } catch (\Exception $e) {
  160. return ['success' => false, 'message' => '刷新页面后重试'];
  161. }
  162. }
  163. /**
  164. * 按 指定工单类型 和 唯一标识 确认工单是否存在
  165. * @param $work_order_type_id
  166. * @param $uniquely_tag
  167. * @return array|false[]
  168. */
  169. public function exists($work_order_type_id, $uniquely_tag): array
  170. {
  171. $query = WorkOrder::query()->where('work_order_type_id', $work_order_type_id);
  172. if (is_array($uniquely_tag)) $query->whereIn('uniquely_tag', $uniquely_tag);
  173. if (is_string($uniquely_tag)) $query->where('uniquely_tag', $uniquely_tag);
  174. $items = $query->get();
  175. if (count($items) > 0) {
  176. $exists_nos = $items->map(function ($item) {
  177. return $item->uniquely_tag;
  178. });
  179. return ['success' => true, 'data' => $exists_nos];
  180. }
  181. return ['success' => false];
  182. }
  183. /**
  184. * 标记已有 问题件 的工单
  185. * @param $workOrders
  186. */
  187. public function tags(&$workOrders)
  188. {
  189. $order_ids = $workOrders->map(function ($item) {
  190. return $item->order_id;
  191. });
  192. $order_issues = OrderIssue::query()->whereIn('order_id', $order_ids)->get();
  193. $codes = [];
  194. foreach ($order_issues as $order_issue) {
  195. $codes[$order_issue->order_id] = true;
  196. }
  197. foreach ($workOrders as &$workOrder) {
  198. if (array_key_exists($workOrder->order_id, $codes)) $workOrder->is_issue_order = true;
  199. else $workOrder->is_issue_order = false;
  200. }
  201. }
  202. /**
  203. * 判断是否拦截工单 称重
  204. * @param $logistic_number
  205. * @return bool
  206. */
  207. public function isIntercept($logistic_number): bool
  208. {
  209. $package_query = OrderPackage::query()->select('order_id')->where('logistic_number', $logistic_number);
  210. $order_issue_query = OrderIssueType::query()->select('id')->where('name', '拦截');
  211. return WorkOrder::query()->whereIn('order_id', $package_query)->whereIn('order_issue_type_id', $order_issue_query)->exists();
  212. }
  213. /**
  214. * 标记工单
  215. * @param $orders
  216. */
  217. public function tagWorkOrder(&$orders)
  218. {
  219. $orderNos = data_get($orders, '*.orderno');
  220. $workOrders = WorkOrder::query()->with('order')->whereIn('order_id', function ($query) use ($orderNos) {
  221. $query->from('orders')->select('id')->whereIn('code', $orderNos);
  222. })->get();
  223. $tags = [];
  224. $workOrders->each(function ($workOrder) use (&$tags) {
  225. $order_code = $workOrder->order->code ?? null;
  226. if ($order_code) $tags[$order_code] = true;
  227. });
  228. foreach ($orders as &$order) {
  229. if (array_key_exists($order->orderno, $tags)) $order->is_work_order = true;
  230. else $order->is_work_order = false;
  231. }
  232. }
  233. /**
  234. * 校验工单是否存在
  235. * @param $nos
  236. * @return mixed
  237. */
  238. public function checkWorkOrder($nos)
  239. {
  240. return WorkOrder::query()->defaultWith()->whereIn('order_id', function ($query) use ($nos) {
  241. $query->from('orders')->selectRaw('id');
  242. if (is_array($nos))
  243. $query->whereIn('code', $nos);
  244. else {
  245. $query->where('code', $nos);
  246. }
  247. })->get();
  248. }
  249. /**
  250. * 同步订单
  251. * @param $order_no
  252. * @return mixed
  253. */
  254. public function syncOrder($order_no)
  255. {
  256. return $this->orderService->first(['code' => $order_no]);
  257. }
  258. /**
  259. * 承运商标记在处理
  260. * @param WorkOrderDetail $detail
  261. */
  262. public function logisticHandlerTag(WorkOrderDetail $detail)
  263. {
  264. $detail->logisticTagHandle();
  265. $last_status = WorkOrder::$enums['status'][$detail->status];
  266. $detail->update([
  267. 'status' => WorkOrder::$LOGISTIC_HANDLER_STATUS,
  268. 'process_progress' => '承运商处理中',
  269. 'last_status' => $last_status,
  270. 'logistic_handle_tag' => 1,
  271. ]);
  272. $detail->workOrder()->update([
  273. 'status' => WorkOrder::$LOGISTIC_HANDLER_STATUS,
  274. 'process_progress' => '承运商处理中',
  275. 'last_status' => $last_status,
  276. 'logistic_tag' => WorkOrder::$NO_STATE_TAG,
  277. 'owner_tag' => WorkOrder::$DEFAULT_TAG,
  278. 'bao_shi_tag' => WorkOrder::$DEFAULT_TAG,
  279. 'work_order_status' => WorkOrder::$DEFAULT_TAG,
  280. ]);
  281. $detail->processLogs()->create([
  282. 'work_order_id' => $detail->work_order_id,
  283. 'work_order_detail_id' => $detail->id,
  284. 'user_id' => Auth::id(),
  285. 'content' => '承运商标记在处理',
  286. 'status' => '未同步',
  287. 'type' => '处理',
  288. ]);
  289. app(WorkOrderLogService::class)->createLog($detail, '处理', '标记处理中');
  290. }
  291. public function syncWorkOrder($rejectedBill)
  292. {
  293. /** @var OrderIssue $order_issue */
  294. $orderIssueRejectedBill = OrderIssueRejectedBill::query()->where('logistic_number_return', $rejectedBill['logistic_number_return'])->first();
  295. $order_issue = OrderIssue::query()->find($orderIssueRejectedBill->order_issue_id);
  296. if (!$order_issue || $order_issue->rejecting_status !== '全部退回') return;
  297. /** @var WorkOrderInterceptService $workOrderInterceptService */
  298. $workOrderInterceptService = app(WorkOrderInterceptService::class);
  299. /** @var WorkOrder $workOrder */
  300. $workOrder = WorkOrder::query()->where('order_id', $order_issue->order_id)->orderByDesc('created_at')->first();
  301. if (!$workOrder) return;
  302. $workOrderInterceptService->autoReviewIntercept($workOrder);
  303. }
  304. /**
  305. * 每日工单定时任务timingTask
  306. */
  307. public function timingTask()
  308. {
  309. $this->updateBaoShiHandlerTask();
  310. $this->updateLogisticHandlerTask();
  311. }
  312. private function updateLogisticHandlerTask()
  313. {
  314. // 将当日承运商处理过 且 状态为`无`的 标记为滞留状态 `滞`
  315. WorkOrder::query()->where('status', 3)
  316. ->where('logistic_tag', WorkOrder::$NO_STATE_TAG)
  317. ->update(['logistic_tag' => WorkOrder::$STRAND_TAG]);
  318. }
  319. private function updateBaoShiHandlerTask()
  320. {
  321. // 将当日宝时处理过 且 状态为`无`的 标记为滞留状态 `滞`
  322. WorkOrder::query()->whereIn('status', [1, 4])
  323. ->where('bao_shi_tag', WorkOrder::$NO_STATE_TAG)
  324. ->update(['bao_shi_tag' => WorkOrder::$STRAND_TAG]);
  325. }
  326. /**
  327. * 同步处理日志到问题件
  328. * @param WorkOrderDetail $detail
  329. */
  330. public function endOrderIssueAndSyncProcessLogs(WorkOrderDetail $detail)
  331. {
  332. /**
  333. * @var OrderIssue $order_issue
  334. * @var WorkOrder $work_order
  335. */
  336. $work_order = $detail->workOrder;
  337. $order_issue = OrderIssue::query()->withTrashed()->where('order_id', $work_order->order_id)->first();
  338. if($order_issue->deleted_at ?? false){
  339. $order_issue->restore();
  340. }
  341. $orderIssueType = app(OrderIssueTypeService::class)->firstOrCreate(['name' => '错漏发']);
  342. if (!$order_issue) {
  343. $order_issue = OrderIssue::query()->create([
  344. 'order_id' => $work_order->order_id,
  345. 'result_explain' => $work_order->remark,
  346. 'imported_status' => '正常',
  347. 'order_issue_type_id' => $detail->order_issue_type_id,
  348. ]);
  349. $order_issue->update(['created_at' => $detail->created_at]);
  350. }
  351. $order_issue->update([
  352. 'logistic_indemnity_money' => $work_order->logistic_indemnity_money,
  353. 'logistic_express_remission' => $work_order->logistic_express_remission,
  354. 'baoshi_indemnity_money' => $work_order->bao_shi_indemnity_money,
  355. 'baoshi_express_remission' => $work_order->bao_shi_express_remission,
  356. 'user_owner_group_id' => $work_order->user_owner_group_id,
  357. ]);
  358. $work_order_user_workgroup_ids = $work_order->userWorkGroups()->get()->map(function ($item) {
  359. return $item->id;
  360. })->toArray();
  361. $order_issue_work_groups = $order_issue->userWorkgroups()->get()->map(function ($item) {
  362. return $item->id;
  363. })->toArray();
  364. $order_issue->userWorkgroups()->attach(array_diff($work_order_user_workgroup_ids, $order_issue_work_groups));
  365. $process_logs = $detail->processLogs()->where('status', '未同步')->get();
  366. $processLogs = $process_logs->map((function ($process_log) use ($order_issue) {
  367. return [
  368. 'order_issue_id' => $order_issue->id,
  369. 'user_id' => $process_log->user_id,
  370. 'content' => $process_log->content,
  371. 'created_at' => $process_log->created_at,
  372. 'updated_at' => $process_log->updated_at,
  373. 'type' => $process_log->type,
  374. 'tag' => $process_log->tag,
  375. ];
  376. }))->toArray();
  377. if ($detail->order_issue_type_id == $orderIssueType->id) {
  378. $commodityLog = $detail->commodities()->where('check_result', '!=', '核实未错漏发')->get()->map(function ($item) use ($order_issue) {
  379. $count = 0;
  380. $abnormalAmount = intval($item->abnormal_amount);
  381. $baoShiCheckAmount = intval($item->bao_shi_check_amount);
  382. if ($item->abnormal_type === '少发') {
  383. $count = $baoShiCheckAmount - $abnormalAmount;
  384. } else if ($item->abnormal_type === '多发') {
  385. $count = $abnormalAmount- $baoShiCheckAmount;
  386. }
  387. return [
  388. 'order_issue_id' => $order_issue->id,
  389. 'user_id' => Auth::id(),
  390. 'content' => $item->sku . ' ' . $item->commodity->name . ' ' . $item->abnormal_type . '(' . $count.')',
  391. 'created_at' => $item->updated_at,
  392. 'updated_at' => $item->updated_at,
  393. 'type' => '处理',
  394. 'tag' => 0
  395. ];
  396. })->toArray();
  397. $processLogs = array_merge($commodityLog, $processLogs);
  398. }
  399. $order_issue->logs()->insert($processLogs);
  400. $this->orderIssueService->endOrderIssues([$order_issue->id]);
  401. $detail->processLogs()->where('status', '未同步')->update(['status' => '同步']);
  402. }
  403. public function customRejectedStatus($ids, $customRejectedStatus): int
  404. {
  405. return WorkOrder::query()->whereIn('id', $ids)->update(['custom_rejected_status' => $customRejectedStatus]);
  406. }
  407. }