WorkOrderService.php 19 KB

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