WorkOrderService.php 19 KB

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