OrderPackageReceivedSyncService.php 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406
  1. <?php
  2. namespace App\Services;
  3. use App\Jobs\LogisticAliJiSuSync;
  4. use App\Jobs\LogisticSFSync;
  5. use App\Jobs\LogisticYDSync;
  6. use App\Jobs\LogisticYTOSync;
  7. use App\Jobs\LogisticZopSync;
  8. use App\OrderPackage;
  9. use Carbon\Carbon;
  10. use Exception;
  11. use Illuminate\Database\Eloquent\Collection;
  12. class OrderPackageReceivedSyncService
  13. {
  14. protected $logisticSFService;
  15. protected $logisticZopService;
  16. /**
  17. * 同步快递信息
  18. * 1 如果当前时间大于初始化时间 每日执行一次,更新order_packages中创建时间大于初始化时间,没有异常,用户未收货的全部订单的快递路由状态
  19. * 2 如果当前时间小于等于初始化时间,执行初始化脚本,将数据库中全部小于等于初始化时间的数据更新
  20. * @throws Exception
  21. */
  22. public function syncLogisticRoute($is_to_init = false)
  23. {
  24. ini_set('max_execution_time', 2 * 60 * 60);
  25. ini_set('memory_limit', '1024M');
  26. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法", '');
  27. $query = OrderPackage::query()
  28. ->select(['logistic_number', 'order_id', 'id'])
  29. ->with(['order' => function ($query) {
  30. return $query->select(['id', 'logistic_id'])->with('logistic:id,name,code');
  31. }]);
  32. if ($is_to_init) {//当前时间小于等于初始化时间
  33. $initDate = Carbon::parse(config('api_logistic.init_date'));
  34. //初始化查询一个月的数据,exception为否
  35. $query = $query->where('sent_at', '>=', $initDate)
  36. ->whereNull('received_at');
  37. } else {//查询20天以内的数据
  38. $query = $query->where('sent_at', '>=', now()->subDays(config('api_logistic.querying_days'))->startOfDay())
  39. ->whereNull('received_at');
  40. }
  41. $query->chunkById(1000, function ($orderPackages) {
  42. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法", json_encode(data_get($orderPackages,'*.logistic_number')));
  43. $logisticNumbers = $this->buildData($orderPackages);
  44. //sf
  45. if (array_key_exists('SF', $logisticNumbers)) {
  46. $SFLogisticNumbers = $logisticNumbers['SF'];
  47. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法-SF", json_encode($SFLogisticNumbers));
  48. foreach ($SFLogisticNumbers as $logisticNumber) {
  49. // LogService::log(OrderPackageReceivedSyncService::class, "同步SF快递单号", $logisticNumber);
  50. LogisticSFSync::dispatch($logisticNumber);
  51. }
  52. }
  53. //更新中通
  54. if (array_key_exists('ZTO', $logisticNumbers)) {
  55. $ZTOLogisticNumbers = $logisticNumbers['ZTO'];
  56. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法-ZTO", json_encode($ZTOLogisticNumbers));
  57. foreach ($ZTOLogisticNumbers as $logisticNumber) {
  58. // LogService::log(OrderPackageReceivedSyncService::class, "同步ZTO快递单号", $logisticNumber);
  59. LogisticZopSync::dispatch($logisticNumber);
  60. }
  61. }
  62. //更新韵达
  63. if (array_key_exists('YUNDA', $logisticNumbers)) {
  64. $YDLogisticNumbers = $logisticNumbers['YUNDA'];
  65. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法-YUNDA", json_encode($YDLogisticNumbers));
  66. foreach ($YDLogisticNumbers as $logistic_number) {
  67. // LogService::log(OrderPackageReceivedSyncService::class, "同步YUNDA快递单号", $logisticNumber);
  68. LogisticYDSync::dispatch($logistic_number);
  69. }
  70. }
  71. //更新圆通
  72. if (array_key_exists('YTO', $logisticNumbers)) {
  73. $YTOLogisticNumbers = $logisticNumbers['YTO'];
  74. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法-YTO", json_encode($YTOLogisticNumbers));
  75. foreach ($YTOLogisticNumbers as $logistic_number) {
  76. if ($logistic_number) LogisticYTOSync::dispatch($logistic_number);
  77. }
  78. }
  79. });
  80. }
  81. public function syncLogisticRouteByAliJiSu()
  82. {
  83. ini_set('max_execution_time', 2 * 60 * 60);
  84. $query = OrderPackage::query()
  85. ->select(['logistic_number', 'order_id', 'id'])
  86. ->whereIn('order_id', function ($query) {
  87. $query->from('orders')->selectRaw('id')->whereIn('logistic_id', function ($builder) {
  88. $builder->from('logistics')->selectRaw('id')->where('type', '=', '快递')->whereNotIn('belong_company', ['顺丰', '中通', '韵达', '圆通', '京东']);
  89. });
  90. });
  91. $query = $query->where('sent_at', '>=', now()->subDays(config('api_logistic.querying_days')))
  92. ->whereNull('received_at');
  93. $query->chunkById(200, function ($orderPackages) {
  94. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法-阿里公用接口", json_encode($orderPackages));
  95. foreach ($orderPackages as $orderPackage) {
  96. if ($orderPackage && $orderPackage->logistic_number) LogisticAliJiSuSync::dispatch($orderPackage->logistic_number);
  97. }
  98. });
  99. //TODO 暂时不同步京东
  100. // $this->syncLogisticRouteJD();
  101. }
  102. public function syncLogisticRouteJD()
  103. {
  104. ini_set('max_execution_time', 60);
  105. $query = OrderPackage::query()
  106. ->select(['logistic_number', 'order_id', 'id'])
  107. ->whereIn('order_id', function ($query) {
  108. $query->from('orders')->selectRaw('id')->whereIn('logistic_id', function ($builder) {
  109. $builder->from('logistics')->selectRaw('id')->where('type', '!=', '物流')->where('belong_company', '京东');
  110. });
  111. });
  112. $query = $query->where('created_at', '>=', now()->subDays(config('api_logistic.querying_days')))
  113. ->whereNull('received_at')->where('logistic_number', 'like', 'JD%');
  114. $query->chunkById(200, function ($orderPackages) {
  115. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法-JD", json_encode($orderPackages));
  116. foreach ($orderPackages as $orderPackage) {
  117. if ($orderPackage && $orderPackage->logistic_number) LogisticAliJiSuSync::dispatch($orderPackage->logistic_number);
  118. }
  119. });
  120. }
  121. /**
  122. * 根据传递的承运商与快递单号更新快递信息
  123. * @param array $logisticNumbers 快递单号
  124. * example: ['SF' => ['SF1038651915891', 'SF1038651413847', 'SF1038611050071'],'ZT'=>['75424148714142','548464120822', '75424147834290']....]
  125. * @throws Exception 快递接口调用或者返回的信息有误,无法更新指定的快递路由信息
  126. */
  127. public function syncLogisticRouteApi(array $logisticNumbers)
  128. {
  129. $this->update($this->getLogisticRoutes($logisticNumbers));
  130. }
  131. /**
  132. * 获取快件揽收信息
  133. * @param array $request [
  134. * 'SF' => ['SF1038651915891', 'SF1038651413847', 'SF1038611050071'],
  135. * 'ZT'=>['75424148714142','548464120822', '75424147834290']
  136. * ]
  137. * @return array
  138. * @throws Exception
  139. */
  140. public function getLogisticRoutes(array $request): array
  141. {
  142. $this->logisticSFService = new LogisticSFService();
  143. $resultSF = [];
  144. $resultYD = [];
  145. $resultYT = [];
  146. $resultOther = [];
  147. foreach ($request as $key => $logisticNums) {
  148. switch ($key) {
  149. case "SF":
  150. $resultSF = $this->logisticSFService->get($logisticNums);
  151. break;
  152. case "YD":
  153. $resultYD = [];
  154. break;
  155. case "YT":
  156. $resultYT = [];
  157. break;
  158. default:
  159. $resultOther = [];
  160. break;
  161. }
  162. }
  163. return array_merge($resultSF, $resultYD, $resultYT, $resultOther);
  164. }
  165. /**
  166. * 根据快递单号更新状态
  167. * @param array $logisticResponses
  168. */
  169. public function update(array $logisticResponses)
  170. {
  171. foreach ($logisticResponses as $logisticResponse) {
  172. if (empty($logisticResponse)) continue;
  173. $orderPackage = OrderPackage::query()->where('logistic_number', $logisticResponse['logistic_number'])->first();
  174. //如果已经收货,状态改为已签收
  175. if ($logisticResponse['received_at'] ?? false) {
  176. $logisticResponse['status'] = '已签收';
  177. }
  178. //未查询到路由信息改为 揽件异常
  179. if (empty($logisticResponse['transfer_status'])) {
  180. $logisticResponse['status'] = '揽件异常';
  181. //未查询到路由信息且未称重的 疑似库内丢件
  182. if (empty($orderPackage->weighed_at)) {
  183. $logisticResponse['status'] = '疑似库内丢件';
  184. }
  185. } else if (count($logisticResponse['transfer_status'])<=3) {//快递记录不为空但是小于3条
  186. $logisticResponse['status'] = '揽件异常';
  187. }
  188. //标记为手动更新的 status不更新
  189. if ($orderPackage->is_manual_update) {
  190. unset($logisticResponse['status']);
  191. }
  192. if (isset($logisticResponse['status'])) $orderPackage->status = $logisticResponse['status'];
  193. if (isset($logisticResponse['received_at'])) $orderPackage->received_at = $logisticResponse['received_at'];
  194. if (isset($logisticResponse['transfer_status']) && !empty($logisticResponse['transfer_status'])) $orderPackage->transfer_status = $logisticResponse['transfer_status'];
  195. $orderPackage->save();
  196. }
  197. }
  198. /**
  199. * 将orderPackage集合分类并摘取指定数据
  200. * @param Collection $orderPackages
  201. * @return array
  202. */
  203. private function buildData(Collection $orderPackages): array
  204. {
  205. $data = [];
  206. foreach ($orderPackages as $orderPackage) {
  207. try {
  208. $logisticCode = $orderPackage->order->logistic->code;
  209. } catch (Exception $e) {
  210. // LogService::log(OrderPackageReceivedSyncService::class, "快递同步按照承运商分组异常", json_encode($orderPackage??[]));
  211. continue;
  212. }
  213. $key = config('api_logistic.logistic.' . $logisticCode);
  214. if (!isset($data[$key])) {
  215. $data[$key] = [];
  216. }
  217. $data[$key][] = $orderPackage->logistic_number;
  218. }
  219. return $data;
  220. }
  221. /**
  222. * @param array $data
  223. * @param $lastRouteDate
  224. * @return array
  225. */
  226. public function setExceptionType(array $data, $lastRouteDate): array
  227. {
  228. //设置默认异常为否
  229. $data['exception_type'] = '无';
  230. $data['exception'] = '否';
  231. $logistic_number = $data['logistic_number'];
  232. /** @var OrderPackage $orderPackage */
  233. $orderPackage = OrderPackage::query()->with('order')->where('logistic_number', $logistic_number)->first();
  234. $delivered_duration = now()->diffInHours(Carbon::parse($orderPackage['sent_at']));
  235. $last_routed_duration = now()->diffInHours(Carbon::parse($lastRouteDate));
  236. $VALID_HOURS = 4;
  237. $SHORT_RESPONSE_HOURS = (function ($province) {
  238. switch ($province) {
  239. case '浙江省':
  240. case '江苏省':
  241. case '上海':
  242. case '安徽省':
  243. return 24;
  244. case '北京':
  245. case '天津':
  246. case '江西省':
  247. case '湖北省':
  248. case '湖南省':
  249. case '广东省':
  250. case '福建省':
  251. case '山东省':
  252. case '河北省':
  253. case '河南省':
  254. case '山西省':
  255. case '四川省':
  256. case '陕西省':
  257. case '重庆':
  258. case '广西壮族自治区':
  259. case '贵州省':
  260. case '云南省':
  261. case '海南省':
  262. case '吉林省':
  263. case '黑龙江省':
  264. case '辽宁省':
  265. return 72;
  266. case '青海省':
  267. case '宁夏回族自治区':
  268. case '甘肃省':
  269. case '内蒙古自治区':
  270. case '新疆维吾尔自治区':
  271. case '西藏自治区':
  272. return 120;
  273. default:
  274. return 24;
  275. }
  276. })($orderPackage->order->province);
  277. $LONG_RESPONSE_HOURS = (function ($province) {
  278. switch ($province) {
  279. case '浙江省':
  280. case '江苏省':
  281. case '上海':
  282. case '安徽省':
  283. return 72;
  284. case '北京':
  285. case '天津':
  286. case '江西省':
  287. case '湖北省':
  288. case '湖南省':
  289. case '广东省':
  290. case '福建省':
  291. case '山东省':
  292. case '河北省':
  293. case '河南省':
  294. case '山西省':
  295. case '四川省':
  296. case '陕西省':
  297. case '重庆':
  298. case '广西壮族自治区':
  299. case '贵州省':
  300. case '云南省':
  301. case '海南省':
  302. case '吉林省':
  303. case '黑龙江省':
  304. case '辽宁省':
  305. return 120;
  306. case '青海省':
  307. case '宁夏回族自治区':
  308. case '甘肃省':
  309. case '内蒙古自治区':
  310. case '新疆维吾尔自治区':
  311. case '西藏自治区':
  312. return 168;
  313. default:
  314. return 72;
  315. }
  316. })($orderPackage->order->province);
  317. $SENDING_RESPONSE_HOURS = 48;
  318. $HAVEN_SECOND_GOT_HOURS = 24;
  319. $IS_ROUTED = 1; //0000 0001 有路由信息
  320. $IS_IN_VALID_TIME = 2; //0000 0010 大于4小时
  321. $IS_WEIGHED = 4; //0000 0100 称重过
  322. $IS_RECEIVED = 8; //0000 1000 已经收货
  323. $IS_SENDING = 16; //0001 0000 正在派送
  324. $IS_SHORT_NO_RESPONSE = 32; //0010 0000 中转异常
  325. $IS_LONG_NO_RESPONSE = 64; //0010 0000 疑似丢件
  326. $IS_SENDING_NO_RESPONSE = 128; //0010 0000 派送异常
  327. $IS_SECOND_ROUTE_HAVE = 256; //0100 0000 揽件异常
  328. $conclusion = (function () use (
  329. $data, $delivered_duration, $last_routed_duration,
  330. $VALID_HOURS, $IS_ROUTED, $IS_IN_VALID_TIME, $IS_WEIGHED, $IS_RECEIVED, $IS_SENDING, $IS_SHORT_NO_RESPONSE, $IS_LONG_NO_RESPONSE, $IS_SENDING_NO_RESPONSE,
  331. $SHORT_RESPONSE_HOURS, $LONG_RESPONSE_HOURS, $SENDING_RESPONSE_HOURS, $HAVEN_SECOND_GOT_HOURS, $IS_SECOND_ROUTE_HAVE,
  332. $orderPackage
  333. ) {
  334. $conclusion = 0;
  335. $conclusion |= !empty($data['transfer_status']) ? $IS_ROUTED : 0;
  336. $conclusion |= ($delivered_duration > $VALID_HOURS) ? $IS_IN_VALID_TIME : 0;
  337. $conclusion |= ($orderPackage->weighed_at) ? $IS_WEIGHED : 0;
  338. $conclusion |= ($data['status'] == '已签收') ? $IS_RECEIVED : 0;
  339. $conclusion |= ($data['status'] == '派送中') ? $IS_SENDING : 0;//
  340. $conclusion |= ($last_routed_duration > $SHORT_RESPONSE_HOURS && $last_routed_duration < $LONG_RESPONSE_HOURS) ? $IS_SHORT_NO_RESPONSE : 0;
  341. $conclusion |= ($last_routed_duration > $LONG_RESPONSE_HOURS) ? $IS_LONG_NO_RESPONSE : 0;
  342. $conclusion |= ($last_routed_duration > $SENDING_RESPONSE_HOURS && $data['status'] == '派送中') ? $IS_SENDING_NO_RESPONSE : 0;
  343. $conclusion |= ($delivered_duration > $HAVEN_SECOND_GOT_HOURS && $data['routes_length'] < 3) ? $IS_SECOND_ROUTE_HAVE : 0;//和出库时间比较 超过指定时间,路由信息小于三条
  344. return $conclusion;
  345. })();
  346. switch ($conclusion) {
  347. case $IS_IN_VALID_TIME:
  348. $data['exception_type'] = '疑似库内丢件';
  349. break;
  350. case $IS_IN_VALID_TIME | $IS_WEIGHED:
  351. $data['exception_type'] = '揽件异常';
  352. break;
  353. case $IS_ROUTED | $IS_IN_VALID_TIME | $IS_SHORT_NO_RESPONSE:
  354. case $IS_ROUTED | $IS_IN_VALID_TIME | $IS_SHORT_NO_RESPONSE | $IS_WEIGHED:
  355. $data['exception_type'] = '中转异常';
  356. break;
  357. case $IS_ROUTED | $IS_IN_VALID_TIME | $IS_LONG_NO_RESPONSE:
  358. case $IS_ROUTED | $IS_IN_VALID_TIME | $IS_LONG_NO_RESPONSE | $IS_WEIGHED:
  359. $data['exception_type'] = '疑似丢件';
  360. break;
  361. default:
  362. break;
  363. }
  364. if ($conclusion
  365. == ($conclusion | $IS_ROUTED | $IS_IN_VALID_TIME | $IS_SENDING | $IS_SENDING_NO_RESPONSE)) {
  366. $data['exception_type'] = '派件异常';
  367. }
  368. if ($conclusion
  369. == ($conclusion | $IS_SECOND_ROUTE_HAVE)) {
  370. $data['exception_type'] = '揽件异常';
  371. $data['exception'] = '是';
  372. }
  373. switch ($conclusion) {
  374. case $IS_IN_VALID_TIME:
  375. case $IS_IN_VALID_TIME | $IS_WEIGHED:
  376. case $IS_ROUTED | $IS_SHORT_NO_RESPONSE:
  377. case $IS_LONG_NO_RESPONSE:
  378. $data['exception'] = '是';
  379. break;
  380. default:
  381. break;
  382. }
  383. return [
  384. 'exception_type' => array_key_exists('exception_type', $data) ? $data['exception_type'] : null,
  385. 'exception' => array_key_exists('exception', $data) ? $data['exception'] : null,
  386. ];
  387. }
  388. }