WorkOrderService.php 13 KB

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