| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526 |
- <?php
- namespace App\Services;
- use App\Exceptions\ErrorException;
- use App\Exceptions\Exception;
- use App\Jobs\CacheShelfTaskJob;
- use App\MaterialBox;
- use App\Station;
- use App\StationTask;
- use App\StationTaskMaterialBox;
- use Carbon\Carbon;
- use Illuminate\Support\Collection;
- use Illuminate\Support\Facades\Cache;
- use Illuminate\Support\Facades\DB;
- use Illuminate\Support\Facades\Http;
- use App\Traits\ServiceAppAop;
- class ForeignHaiRoboticsService
- {
- use ServiceAppAop;
- // protected $modelClass=ForeignHaiRobotics::class;
- /** @var $stationTaskMaterialBoxService StationTaskMaterialBoxService */
- private $stationTaskMaterialBoxService;
- /** @var $stationTaskBatchService StationTaskBatchService */
- private $stationTaskBatchService;
- /** @var $stationTaskCommoditiesService StationTaskCommodityService */
- private $stationTaskCommoditiesService;
- /** @var $materialBoxService MaterialBoxService */
- private $materialBoxService;
- /** @var $stationTaskService StationTaskService */
- private $stationTaskService;
- /** @var $stationService StationService */
- private $stationService;
- public function __construct()
- {
- $this->stationTaskMaterialBoxService=null;
- $this->materialBoxService=null;
- $this->stationService=null;
- }
- /**
- * @param string $modeName '输送线入立架'|'立架出至输送线'|'移动立架内位置'|'缓存架入立架'|'立架出至缓存架'
- * @param string $fromLocation
- * @param string $toLocation
- * @param Collection $taskMaterialBoxes
- * @param string $groupId
- * @param int $priority
- * @param int $isSequenced
- * @return array
- */
- private function makeJson_move(
- Collection $taskMaterialBoxes,
- string $modeName,
- string $fromLocation='',
- string $toLocation='',
- $groupId=''
- , $priority=10
- , $isSequenced=1
- ): array
- {
- $timestampSuffix = microtime(true);
- $taskMode=$this->getTaskMode($modeName);
- $bins=$taskMaterialBoxes->map(function (StationTaskMaterialBox $taskMaterialBox)use($timestampSuffix,$fromLocation,$toLocation){
- return [
- "taskCode" =>$taskMaterialBox['id'].'_'.$timestampSuffix,
- "binCode" => $taskMaterialBox['materialBox']['code'],
- "fromLocCode" => $fromLocation??'',
- "toLocCode" => $toLocation??'',
- ];
- });
- return [[
- "taskMode" =>$taskMode,
- "bins"=>$bins,
- "groupCode"=>$groupId.'_'.$timestampSuffix,
- "priority"=>$priority,
- "sequenceFlag"=>$isSequenced,
- ]];
- }
- public function makeJson_move_multi(
- Collection $taskMaterialBoxes,
- string $modeName,
- Collection $fromLocation=null,
- Collection $toLocations=null,
- $groupId=''
- , $priority=10
- , $isSequenced=1
- ): array
- {
- if((!$toLocations||($toLocations->count()!=$taskMaterialBoxes->count())) && (!$fromLocation || ($fromLocation->count()!=$taskMaterialBoxes->count())))throw new Exception('元素数量必须和料箱任务不一致');
- $timestampSuffix = microtime(true);
- $taskMode=$this->getTaskMode($modeName);
- $bins=$taskMaterialBoxes->map(function (StationTaskMaterialBox $taskMaterialBox,$i)use($timestampSuffix,$fromLocation,$toLocations){
- return [
- "taskCode" =>$taskMaterialBox['id'].'_'.$timestampSuffix,
- "binCode" => $taskMaterialBox['materialBox']['code'],
- "fromLocCode" => $fromLocation ? $fromLocation[$i] : '',
- "toLocCode" => $toLocations ? $toLocations[$i] : '',
- ];
- });
- return [[
- "taskMode" =>$taskMode,
- "bins"=>$bins,
- "groupCode"=>$groupId.'_'.$timestampSuffix,
- "priority"=>$priority,
- "sequenceFlag"=>$isSequenced,
- ]];
- }
- private function getTaskMode($modeName){
- switch ($modeName){
- case '输送线入立架': return 1;
- case '立架出至输送线': return 2;
- case '移动立架内位置': return 3;
- case '立架出至缓存架':
- case '缓存架入立架':return 6;
- default: throw new \Exception('发至海柔的移料箱请求,模式不存在');
- }
- }
- public function fetchGroup($toLocation, Collection $taskMaterialBoxes, $groupIdPrefix='',$mode='立架出至输送线'): bool
- {
- $dataToPost=$this->makeJson_move(
- $taskMaterialBoxes,
- $mode,
- '',
- $toLocation??'',
- $groupIdPrefix
- );
- return $this->controlHaiRobot($dataToPost,$taskMaterialBoxes,$mode);
- }
- /**
- * @param Collection $toLocations //库位和料箱任务一一对应,一个库位对应一个料箱
- * @param Collection $taskMaterialBoxes
- * @param string $groupIdPrefix
- * @param string $mode
- * @return bool
- * @throws ErrorException
- */
- public function fetchGroup_multiLocation(Collection $toLocations, Collection $taskMaterialBoxes, $groupIdPrefix='',$mode='立架出至输送线'): bool
- {
- $dataToPost=$this->makeJson_move_multi(
- $taskMaterialBoxes,
- $mode,
- null,
- $toLocations,
- $groupIdPrefix
- );
- return $this->controlHaiRobot($dataToPost,$taskMaterialBoxes,$mode);
- }
- public function moveBin(){
- }
- public function markBinProcessed(
- $workStation,
- $binCode,
- $success,
- $created_at,
- $exception,
- $is_in_plan
- ): bool
- {
- LogService::log('海柔请求','markBinProcessed1.1',
- $binCode.'|'.$success.'|'.$exception.'|'.$is_in_plan);
- $this->instant($this->stationService,'StationService');
- $this->instant($this->materialBoxService,'MaterialBoxService');
- $this->instant($this->stationTaskMaterialBoxService,'StationTaskMaterialBoxService');
- $this->instant($this->stationTaskCommoditiesService,'StationTaskCommodityService');
- $this->instant($this->stationTaskBatchService,'StationTaskBatchService');
- try{
- LogService::log('海柔请求','markBinProcessed1.2',
- json_encode([$binCode,$success,$exception,$is_in_plan]));
- if($failed
- =!$success)
- throw new ErrorException('海柔任务失败:'.$exception);
- LogService::log('海柔请求','markBinProcessed1.3',
- $failed);
- // if($NotInPlan
- // =!$is_in_plan)
- // throw new ErrorException('海柔认为是计划外的料箱:'.$exception);
- LogService::log('海柔请求','markBinProcessed1.4',
- '$NotInPlan');
- $materialBox=
- $this->materialBoxService->get(['code'=>$binCode])->first();
- LogService::log('海柔请求','markBinProcessed1.5',
- json_encode($materialBox));
- /** @var StationTaskMaterialBox $stationTaskMaterialBox */
- $stationTaskMaterialBox
- =(function()use($materialBox){
- return $stationTaskMaterialBox=
- StationTaskMaterialBox::query()
- ->where('material_box_id',$materialBox['id'])
- ->where('created_at','>',Carbon::now()->subDay())
- ->whereIn('status',['处理中','待处理','异常','处理队列'])
- ->orderBy('id','desc')
- ->first();
- })();
- LogService::log('海柔请求','markBinProcessed1.6',
- json_encode($stationTaskMaterialBox).'|'.$binCode);
- if(!$stationTaskMaterialBox){
- throw new ErrorException($binCode.'该料箱没有安排在处理队列中.');
- }
- LogService::log('海柔请求','markBinProcessed1.7',
- json_encode($stationTaskMaterialBox).'|'.$binCode);
- DB::transaction(function ()use($stationTaskMaterialBox,$binCode){
- $stationTaskMaterialBox_next=
- $this->stationTaskMaterialBoxService
- ->processNextQueued($stationTaskMaterialBox); //找到队列中下一个料箱,并标记为处理中
- $this->stationTaskCommoditiesService
- ->markProcessed($stationTaskMaterialBox['stationTaskCommodities']);
- LogService::log('海柔请求','markBinProcessed1.8',
- json_encode($stationTaskMaterialBox).'|'.$binCode);
- if($stationTaskMaterialBox_next)
- $this->stationTaskCommoditiesService
- ->markProcessing($stationTaskMaterialBox_next['stationTaskCommodities']);//因为上边商品任务被标记完成了,所以这里要将队列中找出正在处理的料箱对应的标记为“处理中”
- $this->stationTaskMaterialBoxService
- ->markProcessed($stationTaskMaterialBox);
- $notProcessedBoxTasks = $this->stationTaskMaterialBoxService->getNotProcessedSiblings($stationTaskMaterialBox);
- if($notProcessedBoxTasks->isEmpty()){
- $this->instant($this->stationTaskService,'StationTaskService');
- LogService::log('海柔请求','markBinProcessed1.81',
- json_encode($stationTaskMaterialBox['stationTaskBatch']).'|'.$binCode);
- $stationTaskMaterialBox->loadMissing('stationTaskBatch');
- $this->stationTaskBatchService->markProcessed($stationTaskMaterialBox['stationTaskBatch']);
- LogService::log('海柔请求','markBinProcessed1.82',
- json_encode($stationTaskMaterialBox['stationTaskBatch']).'|'.$binCode);
- $this->stationTaskService->markProcessed($stationTaskMaterialBox['stationTask']);
- }
- $this->storeBox($stationTaskMaterialBox)
- ?true
- :(function(){throw new ErrorException('呼叫机器人回收U型线料箱失败');})();
- LogService::log('海柔请求','markBinProcessed1.9',
- json_encode($stationTaskMaterialBox).'|'.$binCode);
- $this->stationService->broadcastBinMonitor($stationTaskMaterialBox['station_id'],$stationTaskMaterialBox['stationTask']);
- LogService::log('海柔请求','markBinProcessed1.99',
- json_encode($stationTaskMaterialBox).'|'.$binCode);
- });
- return true;
- }catch (\Exception $e){
- LogService::log('海柔请求','markBinProcessed E1',
- $binCode);
- $this->instant($this->stationTaskMaterialBoxService,'StationTaskMaterialBoxService');
- $this->instant($this->materialBoxService,'MaterialBoxService');
- $box=$this->materialBoxService->firstOrCreate(['code'=>$binCode]);
- $stationTaskMaterialBox_toStore=
- $this->stationTaskMaterialBoxService->create([
- 'station_id' => $this->stationService->getStation_byType('立库')['id'],
- 'material_box_id' => $box['id'],
- 'status' => '处理中'
- ] );
- $dataToPost=$this->makeJson_move(
- collect([$stationTaskMaterialBox_toStore]),
- '输送线入立架',
- 'BIN-IN1',//TODO:这里应该是动态取得,参考出立架getULineExit()方法,不然不能从站获得对应的出口,而且要改Station的child为children
- '',
- $stationTaskMaterialBox_toStore['stationTaskBatch']['id']
- );
- $this->controlHaiRobot($dataToPost,collect([$stationTaskMaterialBox_toStore]),'输送线入立架');
- LogService::log('海柔请求','markBinProcessed E2',
- $binCode);
- $stationTaskMaterialBox = $stationTaskMaterialBox_toStore??$materialBox??null;
- if($stationTaskMaterialBox && get_class($stationTaskMaterialBox)==MaterialBox::class){
- $stationTaskMaterialBox = StationTaskMaterialBox::query()
- ->where('material_box_id',$stationTaskMaterialBox['id'])
- ->where('status','<>','完成')
- ->where('created_at','>',now()->format('Y-m-d'))
- ->first();
- }
- if($stationTaskMaterialBox)
- $this->stationTaskMaterialBoxService
- ->excepted($stationTaskMaterialBox);
- return $e->getMessage();
- }
- }
- public function storeBox(?StationTaskMaterialBox $stationTaskMaterialBox): bool
- {
- LogService::log('海柔请求','putBinToStore1',
- '');
- $this->instant($this->stationService,'StationService');
- $this->instant($this->stationTaskMaterialBoxService,'StationTaskMaterialBoxService');
- $stationTaskMaterialBox_toStore=
- $this->stationTaskMaterialBoxService->firstOrCreate([
- 'station_id' => $this->stationService->getStation_byType('立库')['id'],
- 'material_box_id' => $stationTaskMaterialBox['materialBox']['id'],
- 'status' => '处理中',
- 'type' => '放',
- ]);
- LogService::log('海柔请求','putBinToStore2',
- json_encode($stationTaskMaterialBox));
- $dataToPost=$this->makeJson_move(
- collect([$stationTaskMaterialBox_toStore]),
- '输送线入立架',
- 'BIN-IN1',//TODO:这里应该是动态取得,参考出立架getULineExit()方法,不然不能从站获得对应的出口,而且要改Station的child为children
- '',
- $stationTaskMaterialBox['stationTaskBatch']['id']
- );
- LogService::log('海柔请求','putBinToStore3',
- json_encode($dataToPost));
- $controlSuccess = $this->controlHaiRobot($dataToPost,collect([$stationTaskMaterialBox_toStore]),'输送线入立架');
- return $controlSuccess;
- }
- /** 缓存架入立架 料箱 任务
- * @param StationTaskMaterialBox|null $stationTaskMaterialBox
- * @param Station|\stdClass|null $station
- * @return bool
- * @throws ErrorException
- */
- public function putBinToStore_fromCacheShelf(?StationTaskMaterialBox $stationTaskMaterialBox,
- $station): bool
- {
- $formLocation = $station->code;
- if ($station && $station->parent_id){
- //缓存标记 推入延时队列 等待三秒决定是否执行
- $key = "cacheShelfTask_".$station->parent_id;
- if (Cache::has($key)){
- list($task,$location) = Cache::get($key);
- $task->add($stationTaskMaterialBox);
- $location->add($formLocation);
- }else{
- $task = collect([$stationTaskMaterialBox]);
- $location = collect([$formLocation]);
- }
- Cache::forever($key,array($task,$location));
- CacheShelfTaskJob::dispatch($key,$task->count())->delay(now()->addSeconds(config("haiRou.cacheShelf.callAwait")));
- return true;
- }
- $dataToPost=$this->makeJson_move(collect([$stationTaskMaterialBox]), '缓存架入立架', $formLocation, '');
- $controlSuccess = $this->controlHaiRobot($dataToPost,collect([$stationTaskMaterialBox]),'缓存架入立架');
- if($controlSuccess){
- $this->instant($this->stationTaskMaterialBoxService,'StationTaskMaterialBoxService');
- $this->stationTaskMaterialBoxService->set($stationTaskMaterialBox,['status' => '处理中']);
- }
- return $controlSuccess;
- }
- public function taskUpdate(
- // $groupCode,
- $stationTaskMaterialBox_id, //实际对应传入的字段 taskCode, 这里用料箱任务号做taskCode
- $updateEventType, //0:task_begin(取货)1:task_end(放货)
- $status, //0:任务成功1:任务失败
- $binCode
- ):bool{
- LogService::log('海柔请求','taskUpdateIn',
- json_encode([
- $stationTaskMaterialBox_id,
- $updateEventType,
- $status,
- $binCode
- ]));
- $this->instant($this->stationTaskMaterialBoxService,'StationTaskMaterialBoxService');
- try{
- if(($failed
- =$status)==1){
- throw new ErrorException('海柔任务失败');
- }
- if($料箱不匹配=
- !$stationTaskMaterialBox
- =(function()use($stationTaskMaterialBox_id,$binCode){
- $stationTaskMaterialBox=StationTaskMaterialBox::query()->find($id=$stationTaskMaterialBox_id);
- if($stationTaskMaterialBox['materialBox']['code']==$binCode)return $stationTaskMaterialBox;
- return null;
- })()){
- throw new ErrorException('发回的料箱和任务号(ID)不匹配:$stationTaskMaterialBox_id:'
- .$stationTaskMaterialBox_id.' $binCode:'.$binCode. ' '.
- StationTaskMaterialBox::query()
- ->where('id', $id=$stationTaskMaterialBox_id)
- ->get()
- ->toJson());
- }
- if(($标记已放置=
- function()use($updateEventType,$stationTaskMaterialBox){
- if(($isPut
- =$updateEventType)==1){
- $this->stationTaskMaterialBoxService->markHasPut($stationTaskMaterialBox);
- return true;
- }return false;
- })())
- return true;
- ($标记已取出=
- function()use($updateEventType,$stationTaskMaterialBox){
- if(($isGet
- =$updateEventType)==0){
- $this->stationTaskMaterialBoxService->markHasTaken($stationTaskMaterialBox);
- }
- })();
- }catch (\Exception $e){
- $this->excepted($stationTaskMaterialBox_id, $binCode, $e->getMessage());
- return false;
- }
- return true;
- }
- public function excepted($taskCode='',$binCode='', $msg=''):bool{
- try{
- throw new ErrorException(
- "taskCode任务号:$taskCode , binCode箱号:$binCode 海柔运行报错: $msg"
- );
- }catch (\Exception $e){
- return true;
- }
- }
- /**
- * @param array $dataToPost
- * @return bool
- */
- public function controlHaiRobot(array $dataToPost,Collection $taskMaterialBoxes,$modeName): bool
- {
- LogService::log('海柔请求','runMany','波次任务分配6.r5f2c1:'.json_encode($dataToPost));
- try{
- LogService::log('海柔请求','runMany','波次任务分配6.r5f2c1.51:');
- $response = Http::post(config('api.haiq.storage.moveBin'), $dataToPost);
- if(isset($response->json()['code'])&&$response->json()['code']==500)
- throw new ErrorException('机器人500错误:'.json_encode($response->json()));
- LogService::log('海柔请求','runMany','波次任务分配6.r5f2c1.52:');
- LogService::log(__METHOD__,'runMany','波次任务分配6.r5f2c1.53:'.json_encode($response->json()));
- }catch (\Exception $e){
- LogService::log('海柔请求','runMany','波次任务分配6.r5f2c1.54:'.json_encode($dataToPost).$e->getMessage());
- throw new ErrorException('海柔机器人任务执行失败:'.json_encode($dataToPost).$e->getMessage());
- }
- LogService::log('海柔请求','runMany','波次任务分配6.r5f2c2:'.json_encode($dataToPost));
- $errMsg = (function () use ($response) {
- if ($response->ok()) return '';
- $errMsg = '错误: ';
- if (!$response) {
- return $errMsg . '没有返回内容,检查连接或目标服务器';
- }
- switch (((string)$response["code"])[0]) {
- case 5:
- $errMsg .= '目标服务器代码错误,请联系对方';
- break;
- case 4:
- $errMsg .= '权限不足以请求资源,请检查对方服务器规范';
- break;
- default:
- $errMsg .= '出现未知请求错误';
- break;
- }
- $responseDetails = ' code:' . $response["code"]
- . ' header:' . $response->body()
- . ' response:' . json_encode($response->headers());
- return $errMsg . $responseDetails;
- })();
- LogService::log('海柔请求','runMany','波次任务分配6.r5f2c3:'.json_encode($errMsg));
- LogService::log(__METHOD__, __FUNCTION__,
- $errMsg ?? ''
- . '请求:' . json_encode($dataToPost)
- . '调用堆栈c:' . json_encode(array_slice(debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS), 0, 3))
- );
- $isSuccess = !$errMsg;
- $标记料箱状态=(function() use ($taskMaterialBoxes,$modeName){
- foreach ($taskMaterialBoxes as $taskMaterialBox){
- switch ($modeName){
- case '缓存架入立架':
- case '输送线入立架':
- case '移动立架内位置':
- $taskMaterialBox->materialBox['status']='在入库中';break;
- case '立架出至输送线':
- case '立架出至缓存架':
- $taskMaterialBox->materialBox['status']='在出库中';break;
- default:
- $taskMaterialBox->materialBox['status']='未知';break;
- }
- $taskMaterialBox->materialBox->update();
- }
- })();
- return $isSuccess;
- }
- /**
- * 填充缓存架
- *
- * @param \Illuminate\Database\Eloquent\Collection $stations
- *
- * @return bool
- *
- * @throws
- */
- public function paddingCacheShelf($stations):?bool
- {
- $collection = new Collection();
- $stationCollection = new Collection();
- $blacklist = [];
- foreach ($stations as $station){
- $box = app("MaterialBoxService")->getAnEmptyBoxSortedByOwner(null,$blacklist);
- if (!$box)break;
- $task = StationTask::query()->create([
- 'status' => "待处理",
- 'station_id' => $station->id,
- ]);
- $collection->add(StationTaskMaterialBox::query()->create([
- 'station_id' => $station->id,
- 'material_box_id'=>$box->id,
- 'status'=>"待处理",
- 'type' => '放',
- 'station_task_id' => $task->id,
- ]));
- $stationCollection->add($station->code);
- $blacklist[] = $box->id;
- }
- if ($stationCollection->count()>0){
- if (!$this->fetchGroup_multiLocation($stationCollection,$collection,'','立架出至缓存架')) return null;
- }
- return $stations->count()==$stationCollection->count();
- }
- }
|