OrderPackageReceivedSyncService.php 13 KB

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