OrderPackageReceivedSyncService.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262
  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. use Illuminate\Support\Str;
  13. class OrderPackageReceivedSyncService
  14. {
  15. protected $logisticSFService;
  16. protected $logisticZopService;
  17. use \App\Traits\LogisticSyncTrait;
  18. /**
  19. * 同步快递信息
  20. * 1 如果当前时间大于初始化时间 每日执行一次,更新order_packages中创建时间大于初始化时间,没有异常,用户未收货的全部订单的快递路由状态
  21. * 2 如果当前时间小于等于初始化时间,执行初始化脚本,将数据库中全部小于等于初始化时间的数据更新
  22. * @throws Exception
  23. */
  24. public function syncLogisticRoute($is_to_init = false, $logistic_numbers = [])
  25. {
  26. ini_set('max_execution_time', 2 * 60 * 60);
  27. ini_set('memory_limit', '1024M');
  28. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法", '');
  29. if (empty($logistic_numbers)) {
  30. $query = OrderPackage::query()
  31. ->select(['logistic_number', 'order_id', 'id'])
  32. ->with(['order' => function ($query) {
  33. return $query->select(['id', 'logistic_id'])->with('logistic:id,name,code');
  34. }]);
  35. if ($is_to_init) {//当前时间小于等于初始化时间
  36. $initDate = Carbon::parse(config('api_logistic.init_date'));
  37. //初始化查询一个月的数据,exception为否
  38. $query = $query->where('created_at', '>=', $initDate)
  39. ->whereNull('received_at');
  40. } else {//查询20天以内的数据
  41. $query = $query->where('created_at', '>=', now()->subDays(config('api_logistic.querying_days'))->startOfDay())
  42. ->whereNull('received_at');
  43. }
  44. } else {
  45. $query = OrderPackage::query()
  46. ->select(['logistic_number', 'order_id', 'id'])
  47. ->with(['order' => function ($query) {
  48. return $query->select(['id', 'logistic_id'])->with('logistic:id,name,code');
  49. }])
  50. ->whereIn('logistic_number', $logistic_numbers);
  51. }
  52. $query->chunkById(1000, function ($orderPackages) {
  53. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法", json_encode(data_get($orderPackages,'*.logistic_number')));
  54. $logisticNumbers = $this->buildData($orderPackages);
  55. //sf
  56. if (array_key_exists('SF', $logisticNumbers)) {
  57. $SFLogisticNumbers = $logisticNumbers['SF'];
  58. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法-SF", json_encode($SFLogisticNumbers));
  59. foreach ($SFLogisticNumbers as $logisticNumber) {
  60. // LogService::log(OrderPackageReceivedSyncService::class, "同步SF快递单号", $logisticNumber);
  61. LogisticSFSync::dispatch($logisticNumber);
  62. }
  63. }
  64. //更新中通
  65. if (array_key_exists('ZTO', $logisticNumbers)) {
  66. $ZTOLogisticNumbers = $logisticNumbers['ZTO'];
  67. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法-ZTO", json_encode($ZTOLogisticNumbers));
  68. foreach ($ZTOLogisticNumbers as $logisticNumber) {
  69. // LogService::log(OrderPackageReceivedSyncService::class, "同步ZTO快递单号", $logisticNumber);
  70. LogisticZopSync::dispatch($logisticNumber);
  71. }
  72. }
  73. //更新韵达
  74. if (array_key_exists('YUNDA', $logisticNumbers)) {
  75. $YDLogisticNumbers = $logisticNumbers['YUNDA'];
  76. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法-YUNDA", json_encode($YDLogisticNumbers));
  77. foreach ($YDLogisticNumbers as $logistic_number) {
  78. // LogService::log(OrderPackageReceivedSyncService::class, "同步YUNDA快递单号", $logisticNumber);
  79. LogisticYDSync::dispatch($logistic_number);
  80. }
  81. }
  82. //更新圆通
  83. if (array_key_exists('YTO', $logisticNumbers)) {
  84. $YTOLogisticNumbers = $logisticNumbers['YTO'];
  85. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法-YTO", json_encode($YTOLogisticNumbers));
  86. foreach ($YTOLogisticNumbers as $logistic_number) {
  87. if ($logistic_number) LogisticYTOSync::dispatch($logistic_number);
  88. }
  89. }
  90. });
  91. }
  92. public function syncLogisticRouteByAliJiSu($logistic_numbers = [])
  93. {
  94. if (empty($logistic_numbers)) {
  95. ini_set('max_execution_time', 2 * 60 * 60);
  96. $query = OrderPackage::query()
  97. ->select(['logistic_number', 'order_id', 'id'])
  98. ->whereIn('order_id', function ($query) {
  99. $query->from('orders')->selectRaw('id')->whereIn('logistic_id', function ($builder) {
  100. $builder->from('logistics')->selectRaw('id')->where('type', '=', '快递')->whereNotIn('belong_company', ['顺丰', '中通', '韵达', '圆通', '京东']);
  101. });
  102. });
  103. $query = $query->where('created_at', '>=', now()->subDays(config('api_logistic.querying_days')))
  104. ->whereNull('received_at');
  105. } else {
  106. $query = OrderPackage::query()
  107. ->select(['logistic_number', 'order_id', 'id'])
  108. ->whereIn('logistic_number', $logistic_numbers);
  109. }
  110. $query->chunkById(200, function ($orderPackages) {
  111. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法-阿里公用接口", json_encode($orderPackages));
  112. foreach ($orderPackages as $orderPackage) {
  113. if ($orderPackage && $orderPackage->logistic_number) LogisticAliJiSuSync::dispatch($orderPackage->logistic_number);
  114. }
  115. });
  116. //TODO 暂时不同步京东
  117. // $this->syncLogisticRouteJD();
  118. }
  119. public function syncLogisticRouteJD()
  120. {
  121. ini_set('max_execution_time', 60);
  122. $query = OrderPackage::query()
  123. ->select(['logistic_number', 'order_id', 'id'])
  124. ->whereIn('order_id', function ($query) {
  125. $query->from('orders')->selectRaw('id')->whereIn('logistic_id', function ($builder) {
  126. $builder->from('logistics')->selectRaw('id')->where('type', '!=', '物流')->where('belong_company', '京东');
  127. });
  128. });
  129. $query = $query->where('created_at', '>=', now()->subDays(config('api_logistic.querying_days')))
  130. ->whereNull('received_at')->where('logistic_number', 'like', 'JD%');
  131. $query->chunkById(200, function ($orderPackages) {
  132. // LogService::log(OrderPackageReceivedSyncService::class, "同步快递信息定时方法-JD", json_encode($orderPackages));
  133. foreach ($orderPackages as $orderPackage) {
  134. if ($orderPackage && $orderPackage->logistic_number) LogisticAliJiSuSync::dispatch($orderPackage->logistic_number);
  135. }
  136. });
  137. }
  138. /**
  139. * 根据传递的承运商与快递单号更新快递信息
  140. * @param array $logisticNumbers 快递单号
  141. * example: ['SF' => ['SF1038651915891', 'SF1038651413847', 'SF1038611050071'],'ZT'=>['75424148714142','548464120822', '75424147834290']....]
  142. * @throws Exception 快递接口调用或者返回的信息有误,无法更新指定的快递路由信息
  143. */
  144. public function syncLogisticRouteApi(array $logisticNumbers)
  145. {
  146. $this->update($this->getLogisticRoutes($logisticNumbers));
  147. }
  148. /**
  149. * 获取快件揽收信息
  150. * @param array $request [
  151. * 'SF' => ['SF1038651915891', 'SF1038651413847', 'SF1038611050071'],
  152. * 'ZT'=>['75424148714142','548464120822', '75424147834290']
  153. * ]
  154. * @return array
  155. * @throws Exception
  156. */
  157. public function getLogisticRoutes(array $request): array
  158. {
  159. $this->logisticSFService = new LogisticSFService();
  160. $resultSF = [];
  161. $resultYD = [];
  162. $resultYT = [];
  163. $resultOther = [];
  164. foreach ($request as $key => $logisticNums) {
  165. switch ($key) {
  166. case "SF":
  167. $resultSF = $this->logisticSFService->get($logisticNums);
  168. break;
  169. case "YD":
  170. $resultYD = [];
  171. break;
  172. case "YT":
  173. $resultYT = [];
  174. break;
  175. default:
  176. $resultOther = [];
  177. break;
  178. }
  179. }
  180. return array_merge($resultSF, $resultYD, $resultYT, $resultOther);
  181. }
  182. /**
  183. * 根据快递单号更新状态
  184. * @param array $logisticResponses
  185. */
  186. public function update(array $logisticResponses)
  187. {
  188. foreach ($logisticResponses as $logisticResponse) {
  189. if (empty($logisticResponse)) continue;
  190. $orderPackage = OrderPackage::query()->where('logistic_number', $logisticResponse['logistic_number'])->first();
  191. //如果已经收货,状态改为已签收
  192. if ($logisticResponse['received_at'] ?? false) {
  193. $logisticResponse['status'] = '已签收';
  194. }
  195. //标记为手动更新的 status不更新
  196. if ($orderPackage->is_manual_update) {
  197. if (OrderPackage::switchStatus($orderPackage->status) > OrderPackage::switchStatus($logisticResponse['status'] ?? '生成订单')) {
  198. unset($logisticResponse['status']);
  199. }
  200. }
  201. $logisticResponse = $this->setExceptionStatus($logisticResponse);
  202. if (Str::contains($orderPackage->logistic_number, ['SO', '#', '-'])) {
  203. $logisticResponse['exception_status'] = '单号异常';
  204. }
  205. if (((!isset($logisticResponse['status'])) || ($logisticResponse['status'] === '')) && $orderPackage->status === '') {
  206. $logisticResponse['status'] = '生成订单';
  207. if (!empty($orderPackage->sent_at)) {
  208. $logisticResponse['status'] = '已复核';
  209. }
  210. if (!empty($orderPackage->weighed_at)) {
  211. $logisticResponse['status'] = '已称重';
  212. }
  213. }
  214. if (isset($logisticResponse['exception_status'])) $logisticResponse['exception_status'] = OrderPackage::switchExceptionStatus($logisticResponse['exception_status']);
  215. if (isset($logisticResponse['status'])) $logisticResponse['status'] = OrderPackage::switchStatus($logisticResponse['status']);
  216. if (isset($logisticResponse['routes_length'])) unset($logisticResponse['routes_length']);
  217. if(empty($logisticResponse['transfer_status'])) unset($logisticResponse['transfer_status']);
  218. OrderPackage::query()->where('logistic_number', $logisticResponse['logistic_number'])
  219. ->update($logisticResponse);
  220. }
  221. }
  222. /**
  223. * 将orderPackage集合分类并摘取指定数据
  224. * @param Collection $orderPackages
  225. * @return array
  226. */
  227. private function buildData(Collection $orderPackages): array
  228. {
  229. $data = [];
  230. foreach ($orderPackages as $orderPackage) {
  231. try {
  232. $logisticCode = $orderPackage->order->logistic->code;
  233. } catch (Exception $e) {
  234. // LogService::log(OrderPackageReceivedSyncService::class, "快递同步按照承运商分组异常", json_encode($orderPackage??[]));
  235. continue;
  236. }
  237. $key = config('api_logistic.logistic.' . $logisticCode);
  238. if (!isset($data[$key])) {
  239. $data[$key] = [];
  240. }
  241. $data[$key][] = $orderPackage->logistic_number;
  242. }
  243. return $data;
  244. }
  245. }