ForeignHaiRoboticsService.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331
  1. <?php
  2. namespace App\Services;
  3. use App\Exceptions\ErrorException;
  4. use App\MaterialBox;
  5. use App\StationTaskMaterialBox;
  6. use Illuminate\Support\Collection;
  7. use Illuminate\Support\Facades\Http;
  8. use App\Traits\ServiceAppAop;
  9. class ForeignHaiRoboticsService
  10. {
  11. use ServiceAppAop;
  12. // protected $modelClass=ForeignHaiRobotics::class;
  13. /** @var $stationTaskMaterialBoxService StationTaskMaterialBoxService */
  14. private $stationTaskMaterialBoxService;
  15. /** @var $stationTaskCommoditiesService StationTaskCommodityService */
  16. private $stationTaskCommoditiesService;
  17. /** @var $materialBoxService MaterialBoxService */
  18. private $materialBoxService;
  19. /** @var $stationTaskService StationTaskService */
  20. private $stationTaskService;
  21. /** @var $stationService StationService */
  22. private $stationService;
  23. public function __construct()
  24. {
  25. $this->stationTaskMaterialBoxService=null;
  26. $this->materialBoxService=null;
  27. $this->stationService=null;
  28. }
  29. /**
  30. * @param string $modeName '输送线入立架'|'立架出至输送线'|'移动立架内位置'|'缓存架入立架'|'立架出至缓存架'
  31. * @param string $fromLocation
  32. * @param string $toLocation
  33. * @param Collection $taskMaterialBoxes
  34. * @param string $groupId
  35. * @param int $priority
  36. * @param int $isSequenced
  37. * @return array
  38. */
  39. private function makeJson_move(
  40. Collection $taskMaterialBoxes,
  41. string $modeName,
  42. string $fromLocation='',
  43. string $toLocation='',
  44. $groupId=''
  45. , $priority=10
  46. , $isSequenced=1
  47. ): array
  48. {
  49. $timestampSuffix = microtime(true);
  50. $taskMode=(function()use($modeName){
  51. switch ($modeName){
  52. case '输送线入立架': return 1;
  53. case '立架出至输送线': return 2;
  54. case '移动立架内位置': return 3;
  55. case '缓存架入立架':
  56. case '立架出至缓存架': return 6;
  57. default: throw new \Exception('发至海柔的移料箱请求,模式不存在');
  58. }
  59. })();
  60. $bins=$taskMaterialBoxes->map(function (StationTaskMaterialBox $taskMaterialBox)use($timestampSuffix,$fromLocation,$toLocation){
  61. return [
  62. "taskCode" =>$taskMaterialBox['id'],
  63. "binCode" => $taskMaterialBox['materialBox']['code'],
  64. "fromLocCode" => $fromLocation??'',
  65. "toLocCode" => $toLocation??'',
  66. ];
  67. });
  68. return [[
  69. "taskMode" =>$taskMode,
  70. "bins"=>$bins,
  71. "groupCode"=>$groupId.$timestampSuffix,
  72. "priority"=>$priority,
  73. "sequenceFlag"=>$isSequenced,
  74. ]];
  75. }
  76. public function fetchGroup($toLocation, Collection $taskMaterialBoxes, $groupIdPrefix=''): bool
  77. {
  78. LogService::log(__METHOD__,'runMany','波次任务分配6.r5f0:');
  79. LogService::log(__METHOD__,'runMany','波次任务分配6.r5f1:'.json_encode($toLocation).json_encode($taskMaterialBoxes).json_encode($groupIdPrefix));
  80. $dataToPost=$this->makeJson_move(
  81. $taskMaterialBoxes,
  82. '立架出至输送线',
  83. '',
  84. $toLocation??'',
  85. $groupIdPrefix
  86. );
  87. LogService::log(__METHOD__,'runMany','波次任务分配6.r5f2:'.json_encode($dataToPost));
  88. return $this->controlHaiRobot($dataToPost);
  89. }
  90. public function moveBin(){
  91. }
  92. public function markBinProcessed(
  93. $workStation,
  94. $binCode,
  95. $success,
  96. $created_at,
  97. $exception,
  98. $is_in_plan
  99. ): bool
  100. {
  101. $this->instant($this->stationService,'StationService');
  102. $this->instant($this->materialBoxService,'MaterialBoxService');
  103. $this->instant($this->stationTaskMaterialBoxService,'StationTaskMaterialBoxService');
  104. $this->instant($this->stationTaskCommoditiesService,'StationTaskCommodityService');
  105. try{
  106. if($failed
  107. =!$success)
  108. throw new ErrorException('海柔任务失败:'.$exception);
  109. if($NotInPlan
  110. =!$is_in_plan)
  111. throw new ErrorException('海柔认为是计划外的料箱:'.$exception);
  112. $materialBox=
  113. $this->materialBoxService->get(['code'=>$binCode])->first();
  114. $stationTaskMaterialBox
  115. =(function()use($materialBox){
  116. return $stationTaskMaterialBox=
  117. $this->stationTaskMaterialBoxService
  118. ->get(
  119. [
  120. 'material_box_id'=>$materialBox['id'],
  121. 'status'=>['处理中','待处理','异常','处理队列'],
  122. ]
  123. )->first();
  124. })();
  125. if(!$stationTaskMaterialBox){
  126. throw new ErrorException('该料箱没有安排在处理队列中');
  127. }
  128. $this->putBinToStore($stationTaskMaterialBox)
  129. ?true
  130. :(function(){throw new ErrorException('呼叫机器人回收U型线料箱失败');})();
  131. $this->stationTaskMaterialBoxService
  132. ->markProcessed($stationTaskMaterialBox);
  133. $stationTaskMaterialBox_next=
  134. $this->stationTaskMaterialBoxService
  135. ->processNextQueued($stationTaskMaterialBox); //找到队列中下一个料箱,并标记为处理中
  136. $this->stationTaskCommoditiesService
  137. ->markProcessed($stationTaskMaterialBox['stationTaskCommodities']);
  138. if($stationTaskMaterialBox_next)
  139. $this->stationTaskCommoditiesService
  140. ->markProcessing($stationTaskMaterialBox_next['stationTaskCommodities']);//因为上边商品任务被标记完成了,所以这里要将队列中找出正在处理的料箱对应的标记为“处理中”
  141. if(!$stationTaskMaterialBox_next){
  142. $this->instant($this->stationTaskService,'StationTaskService');
  143. $this->stationTaskService->markProcessed($stationTaskMaterialBox['stationTask']);
  144. }
  145. $this->stationService->broadcastBinMonitor($stationTaskMaterialBox['station_id'],$stationTaskMaterialBox['stationTask']);
  146. return true;
  147. }catch (\Exception $e){
  148. $stationTaskMaterialBox = $stationTaskMaterialBox??$materialBox??null;
  149. if($stationTaskMaterialBox && get_class($stationTaskMaterialBox)==MaterialBox::class){
  150. $stationTaskMaterialBox = StationTaskMaterialBox::query()
  151. ->where('material_box_id',$stationTaskMaterialBox['id'])
  152. ->where('status','<>','完成')
  153. ->where('created_at','>',now()->format('Y-m-d'))
  154. ->first();
  155. }
  156. if($stationTaskMaterialBox)
  157. $this->stationTaskMaterialBoxService
  158. ->excepted($stationTaskMaterialBox);
  159. return $e->getMessage();
  160. }
  161. }
  162. public function putBinToStore(?StationTaskMaterialBox $stationTaskMaterialBox): bool
  163. {
  164. $this->instant($this->stationService,'StationService');
  165. $dataToPost=$this->makeJson_move(
  166. collect([$stationTaskMaterialBox]),
  167. '缓存架入立架',
  168. 'BIN-IN1',//TODO:这里应该是动态取得,参考出立架getULineExit()方法,不然不能从站获得对应的出口,而且要改Station的child为children
  169. '',
  170. $stationTaskMaterialBox['stationTaskBatch']['id']
  171. );
  172. return $this->controlHaiRobot($dataToPost);
  173. }
  174. public function taskUpdate(
  175. // $groupCode,
  176. $stationTaskMaterialBox_id, //实际对应传入的字段 taskCode, 这里用料箱任务号做taskCode
  177. $updateEventType, //0:task_begin(取货)1:task_end(放货)
  178. $status, //0:任务成功1:任务失败
  179. $binCode
  180. ):bool{
  181. LogService::log('海柔请求','taskUpdateIn1',
  182. json_encode([
  183. $stationTaskMaterialBox_id,
  184. $updateEventType,
  185. $status,
  186. $binCode
  187. ]));
  188. $this->instant($this->stationTaskMaterialBoxService,'StationTaskMaterialBoxService');
  189. try{
  190. if(($failed
  191. =$status)==1){
  192. throw new ErrorException('海柔任务失败');
  193. }
  194. LogService::log('海柔请求','taskUpdateIn2',
  195. $failed);
  196. if($料箱不匹配=
  197. !$stationTaskMaterialBox
  198. =(function()use($stationTaskMaterialBox_id,$binCode){
  199. $stationTaskMaterialBox=StationTaskMaterialBox::query()->find($id=$stationTaskMaterialBox_id);
  200. if($stationTaskMaterialBox['materialBox']['code']==$binCode)return $stationTaskMaterialBox;
  201. return null;
  202. })()){
  203. throw new ErrorException('发回的料箱和任务号(ID)不匹配:$stationTaskMaterialBox_id:'
  204. .$stationTaskMaterialBox_id.' $binCode:'.$binCode. ' '.
  205. StationTaskMaterialBox::query()
  206. ->where('id', $id=$stationTaskMaterialBox_id)
  207. ->get()
  208. ->toJson());
  209. }
  210. LogService::log('海柔请求','taskUpdateIn2',
  211. $料箱不匹配);
  212. ($标记已放置在库外=
  213. function()use($updateEventType,$stationTaskMaterialBox){
  214. if(($isPut
  215. =$updateEventType)==0){
  216. $this->stationTaskMaterialBoxService->markHasTaken($stationTaskMaterialBox);
  217. }
  218. })();
  219. LogService::log('海柔请求','taskUpdateIn3',
  220. $标记已放置在库外);
  221. ($标记已入立架=
  222. function()use($updateEventType,$stationTaskMaterialBox){
  223. if(($isGet
  224. =$updateEventType)==1){
  225. $this->stationTaskMaterialBoxService->markPutStored($stationTaskMaterialBox);
  226. }
  227. })();
  228. LogService::log('海柔请求','taskUpdateIn4',
  229. $标记已入立架);
  230. }catch (\Exception $e){
  231. LogService::log('海柔请求','taskUpdateIn555',
  232. $e->getMessage());
  233. $this->excepted($stationTaskMaterialBox_id, $binCode, $e->getMessage());
  234. return false;
  235. }
  236. return true;
  237. }
  238. // public function markHasPut($taskCode,$binCode):bool{
  239. // try{
  240. // //标记料箱进入位置
  241. //// $taskMaterialBoxesService->markDone();//
  242. // }catch (\Exception $e){
  243. // switch ($e->getCode()){
  244. // case 'taskBinNotMatch';
  245. // case 'taskGetFailed';
  246. // }
  247. // }
  248. //
  249. // }
  250. public function excepted($taskCode='',$binCode='', $msg=''):bool{
  251. try{
  252. throw new ErrorException(
  253. "taskCode任务号:$taskCode , binCode箱号:$binCode 海柔运行报错: $msg"
  254. );
  255. }catch (\Exception $e){
  256. return true;
  257. }
  258. }
  259. /**
  260. * @param array $dataToPost
  261. * @return bool
  262. */
  263. public function controlHaiRobot(array $dataToPost): bool
  264. {
  265. LogService::log(__METHOD__,'runMany','波次任务分配6.r5f2c1:'.json_encode($dataToPost));
  266. try{
  267. LogService::log(__METHOD__,'runMany','波次任务分配6.r5f2c1.51:');
  268. $a = json_encode($dataToPost);
  269. $response = Http::post(config('api.haiq.storage.moveBin'), $dataToPost);
  270. if(isset($response->json()['code'])&&$response->json()['code']==500)
  271. throw new ErrorException('机器人500错误:'.json_encode($response->json()));
  272. $b=json_encode($response->json());
  273. LogService::log(__METHOD__,'runMany','波次任务分配6.r5f2c1.52:');
  274. LogService::log(__METHOD__,'runMany','波次任务分配6.r5f2c1.53:'.json_encode($response->json()));
  275. }catch (\Exception $e){
  276. LogService::log(__METHOD__,'runMany','波次任务分配6.r5f2c1.54:');
  277. throw new ErrorException('海柔机器人任务执行失败:'.json_encode($dataToPost).$e->getMessage());
  278. }
  279. LogService::log(__METHOD__,'runMany','波次任务分配6.r5f2c2:'.json_encode($dataToPost));
  280. $errMsg = (function () use ($response) {
  281. if ($response->ok()) return '';
  282. $errMsg = '错误: ';
  283. if (!$response) {
  284. return $errMsg . '没有返回内容,检查连接或目标服务器';
  285. }
  286. switch (((string)$response["code"])[0]) {
  287. case 5:
  288. $errMsg .= '目标服务器代码错误,请联系对方';
  289. break;
  290. case 4:
  291. $errMsg .= '权限不足以请求资源,请检查对方服务器规范';
  292. break;
  293. default:
  294. $errMsg .= '出现未知请求错误';
  295. break;
  296. }
  297. $responseDetails = ' code:' . $response["code"]
  298. . ' header:' . $response->body()
  299. . ' response:' . json_encode($response->headers());
  300. return $errMsg . $responseDetails;
  301. })();
  302. LogService::log(__METHOD__,'runMany','波次任务分配6.r5f2c3:'.json_encode($errMsg));
  303. LogService::log(__METHOD__, __FUNCTION__,
  304. $errMsg ?? ''
  305. . '请求:' . json_encode($dataToPost)
  306. . '调用堆栈:' . json_encode(array_slice(debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS), 0, 3))
  307. );
  308. return $isSuccess = !$errMsg;
  309. }
  310. }