| package com.zy.asrs.wcs.core.kernel.command; | 
|   | 
| import com.alibaba.fastjson.JSON; | 
| import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; | 
| import com.zy.asrs.framework.common.Cools; | 
| import com.zy.asrs.framework.exception.CoolException; | 
| import com.zy.asrs.wcs.core.action.LiftAction; | 
| import com.zy.asrs.wcs.core.action.ShuttleAction; | 
| import com.zy.asrs.wcs.core.entity.BasShuttle; | 
| import com.zy.asrs.wcs.core.entity.Loc; | 
| import com.zy.asrs.wcs.core.model.NavigateNode; | 
| import com.zy.asrs.wcs.core.model.command.ShuttleAssignCommand; | 
| import com.zy.asrs.wcs.core.model.command.ShuttleCommand; | 
| import com.zy.asrs.wcs.core.model.enums.*; | 
| import com.zy.asrs.wcs.core.service.BasShuttleService; | 
| import com.zy.asrs.wcs.core.service.LocService; | 
| import com.zy.asrs.wcs.core.service.TaskService; | 
| import com.zy.asrs.wcs.core.utils.*; | 
| import com.zy.asrs.wcs.rcs.News; | 
| import com.zy.asrs.wcs.rcs.cache.SlaveConnection; | 
| import com.zy.asrs.wcs.core.entity.Motion; | 
| import com.zy.asrs.wcs.rcs.model.enums.ShuttleProtocolStatusType; | 
| import com.zy.asrs.wcs.rcs.model.enums.SlaveType; | 
| import com.zy.asrs.wcs.rcs.model.protocol.LiftProtocol; | 
| import com.zy.asrs.wcs.rcs.model.protocol.ShuttleProtocol; | 
| import com.zy.asrs.wcs.core.service.MotionService; | 
| import com.zy.asrs.wcs.rcs.thread.LiftThread; | 
| import com.zy.asrs.wcs.rcs.thread.ShuttleThread; | 
| import lombok.extern.slf4j.Slf4j; | 
| import org.springframework.beans.factory.annotation.Autowired; | 
| import org.springframework.stereotype.Service; | 
|   | 
| import java.util.*; | 
|   | 
| /** | 
|  * Created by vincent on 2023/10/23 | 
|  */ | 
| @Slf4j | 
| @Service | 
| public class ShuttleCommandService { | 
|   | 
|     public static final Integer SHUTTLE_ADDITION_COMMAND_SPEED = 500; | 
|   | 
|     @Autowired | 
|     private RedisUtil redisUtil; | 
|     @Autowired | 
|     private MotionService motionService; | 
|     @Autowired | 
|     private TaskService taskService; | 
|     @Autowired | 
|     private BasShuttleService basShuttleService; | 
|     @Autowired | 
|     private LocService locService; | 
|     @Autowired | 
|     private NavigateMapUtils navigateMapUtils; | 
|     @Autowired | 
|     private ShuttleAction shuttleAction; | 
|     @Autowired | 
|     private LiftAction liftAction; | 
|   | 
|     // 计算 | 
|     public Boolean accept(Motion motion) { | 
|         Integer deviceNo = Integer.parseInt(motion.getDevice()); | 
|         ShuttleThread shuttleThread = (ShuttleThread) SlaveConnection.get(SlaveType.Shuttle, deviceNo); | 
|         if (shuttleThread == null) { | 
|             return false; | 
|         } | 
|         ShuttleProtocol shuttleProtocol = shuttleThread.getStatus(); | 
|         if (null == shuttleProtocol) { | 
|             return false; | 
|         } | 
|         if (!shuttleThread.isIdle()) {//设备不空闲 | 
|             return false; | 
|         } | 
|         if (motionService.count(new LambdaQueryWrapper<Motion>() | 
|                 .eq(Motion::getDeviceCtg, DeviceCtgType.SHUTTLE.val()) | 
|                 .eq(Motion::getDevice, motion.getDevice()) | 
|                 .eq(Motion::getMotionSts, MotionStsType.EXECUTING.val())) > 0) { | 
|             return false; | 
|         } | 
|   | 
|         ShuttleAssignCommand assignCommand = new ShuttleAssignCommand(); | 
|         assignCommand.setShuttleNo(deviceNo); | 
|         assignCommand.setTaskNo(motion.getTaskNo()); | 
|         assignCommand.setDeviceTaskNo(shuttleThread.generateDeviceTaskNo(motion.getTaskNo(), MotionCtgType.get(motion.getMotionCtgEl()))); | 
|         assignCommand.setSourceLocNo(motion.getOrigin()); | 
|         assignCommand.setLocNo(motion.getTarget()); | 
|         assignCommand.setDeviceId(Integer.parseInt(motion.getDevice())); | 
|   | 
|         List<ShuttleCommand> shuttleCommands = new ArrayList<>(); | 
|         ShuttleTaskModeType shuttleTaskModeType = null; | 
|   | 
|         LiftThread liftThread = null; | 
|         LiftProtocol liftProtocol = null; | 
|   | 
|         switch (Objects.requireNonNull(MotionCtgType.get(motion.getMotionCtgEl()))){ | 
|             case SHUTTLE_MOVE: | 
|                 // 如果已经在当前条码则过滤 | 
|                 if (String.valueOf(shuttleProtocol.getCurrentCode()).equals(locService.getOne(new LambdaQueryWrapper<Loc>() | 
|                         .eq(Loc::getLocNo, motion.getTarget()) | 
|                             .eq(Loc::getHostId, motion.getHostId())).getCode())) { | 
|                     return true; | 
|                 } | 
|                 shuttleCommands = this.shuttleAssignCommand(motion.getOrigin(), motion.getTarget(), NavigationMapType.NORMAL.id, assignCommand, shuttleThread); | 
|                 shuttleTaskModeType = ShuttleTaskModeType.SHUTTLE_MOVE_LOC_NO; | 
|                 break; | 
|             case SHUTTLE_MOVE_LIFT_PALLET://穿梭车顶升并移动 | 
|                 shuttleCommands = this.shuttleAssignCommand(motion.getOrigin(), motion.getTarget(), NavigationMapType.DFX.id, assignCommand, shuttleThread); | 
|                 shuttleTaskModeType = ShuttleTaskModeType.PAK_IN; | 
|                 shuttleCommands.add(0, shuttleThread.getLiftCommand(motion.getTaskNo(), true)); | 
|                 break; | 
|             case SHUTTLE_MOVE_DOWN_PALLET://穿梭车移动并托盘下降 | 
|                 shuttleCommands = this.shuttleAssignCommand(motion.getOrigin(), motion.getTarget(), NavigationMapType.DFX.id, assignCommand, shuttleThread); | 
|                 shuttleTaskModeType = ShuttleTaskModeType.PAK_IN; | 
|                 shuttleCommands.add(shuttleCommands.size(), shuttleThread.getLiftCommand(motion.getTaskNo(), false)); | 
|                 break; | 
|             case SHUTTLE_MOVE_FROM_LIFT://出提升机 | 
|                 // 判断提升机状态 | 
|                 liftThread = (LiftThread) SlaveConnection.get(SlaveType.Lift, Integer.parseInt(motion.getTemp())); | 
|                 if (liftThread == null) { | 
|                     return false; | 
|                 } | 
|   | 
|                 liftProtocol = liftThread.getStatus(); | 
|                 if (liftProtocol == null) { | 
|                     return false; | 
|                 } | 
|   | 
|                 // 判断提升机是否空闲 | 
|                 if (!liftThread.isIdle()) { | 
|                     return false; | 
|                 } | 
|   | 
|                 if (liftProtocol.getLev() != Utils.getLev(motion.getTarget())) {//判断提升机是否达到目标层 | 
|                     return false; | 
|                 } | 
|   | 
| //                //判断提升机是否被锁定 | 
| //                if (!liftProtocol.getLock()) { | 
| //                    //锁定提升机 | 
| //                    LiftCommand lockCommand = liftThread.getLockCommand(motion.getWrkNo(), true);//获取提升机锁定命令 | 
| // | 
| //                    LiftAssignCommand liftAssignCommand = new LiftAssignCommand(); | 
| //                    liftAssignCommand.setLiftNo(liftThread.getStatus().getLiftNo()); | 
| //                    liftAssignCommand.setTaskNo(motion.getWrkNo()); | 
| //                    ArrayList<LiftCommand> list = new ArrayList<>(); | 
| //                    list.add(lockCommand); | 
| //                    liftAssignCommand.setCommands(list); | 
| // | 
| //                    liftAction.assignWork(liftThread.getDevice(), liftAssignCommand); | 
| //                    return false;//等待下一次轮询 | 
| //                } | 
|   | 
|                 //判断提升机工作号是否和当前任务相同 | 
|                 if (liftProtocol.getTaskNo().intValue() != motion.getTaskNo()) { | 
|                     return false; | 
|                 } | 
|   | 
|                 shuttleCommands = this.shuttleAssignCommand(motion.getOrigin(), motion.getTarget(), NavigationMapType.NORMAL.id, assignCommand, shuttleThread); | 
|                 shuttleTaskModeType = ShuttleTaskModeType.SHUTTLE_MOVE_LOC_NO; | 
|                 break; | 
|             case SHUTTLE_MOVE_TO_LIFT://进提升机 | 
|                 // 判断提升机状态 | 
|                 liftThread = (LiftThread) SlaveConnection.get(SlaveType.Lift, Integer.parseInt(motion.getTemp())); | 
|                 if (liftThread == null) { | 
|                     return false; | 
|                 } | 
|                 liftProtocol = liftThread.getStatus(); | 
|                 if (liftProtocol == null) { | 
|                     return false; | 
|                 } | 
|                 // 判断提升机是否空闲 | 
|                 if (!liftThread.isIdle()) { | 
|                     return false; | 
|                 } | 
|   | 
|                 if (liftProtocol.getLev() != Utils.getLev(motion.getTarget())) {//判断提升机是否达到目标层 | 
|                     return false; | 
|                 } | 
|   | 
| //                //判断提升机是否被锁定 | 
| //                if (!liftProtocol.getLiftLock()) { | 
| //                    //锁定提升机 | 
| //                    LiftCommand lockCommand = liftThread.getLockCommand(true);//获取提升机锁定命令 | 
| //                    lockCommand.setLiftNo(liftProtocol.getLiftNo()); | 
| //                    lockCommand.setTaskNo(motion.getWrkNo().shortValue());//获取任务号 | 
| //                    liftThread.assignWork(lockCommand); | 
| //                    return false;//等待下一次轮询 | 
| //                } | 
|   | 
|                 //判断提升机工作号是否和当前任务相同 | 
|                 if (liftProtocol.getTaskNo().intValue() != motion.getTaskNo()) { | 
|                     return false; | 
|                 } | 
|   | 
|                 shuttleCommands = this.shuttleAssignCommand(motion.getOrigin(), motion.getTarget(), NavigationMapType.NORMAL.id, assignCommand, shuttleThread); | 
|                 shuttleTaskModeType = ShuttleTaskModeType.SHUTTLE_MOVE_LOC_NO; | 
|                 break; | 
|             case SHUTTLE_CHARGE_ON://充电开 | 
|                 shuttleTaskModeType = ShuttleTaskModeType.CHARGE; | 
|                 shuttleCommands.add(shuttleThread.getChargeCommand(motion.getTaskNo(), true)); | 
|                 assignCommand.setCharge(Boolean.TRUE); | 
|                 break; | 
|             case SHUTTLE_CHARGE_OFF://充电关 | 
|                 shuttleTaskModeType = ShuttleTaskModeType.CHARGE; | 
|                 shuttleCommands.add(shuttleThread.getChargeCommand(motion.getTaskNo(), false)); | 
|                 assignCommand.setCharge(Boolean.TRUE); | 
|                 break; | 
|             case SHUTTLE_PALLET_LIFT://托盘顶升 | 
|                 shuttleTaskModeType = ShuttleTaskModeType.PALLET_LIFT; | 
|                 shuttleCommands.add(shuttleThread.getLiftCommand(motion.getTaskNo(), true)); | 
|                 break; | 
|             case SHUTTLE_PALLET_DOWN://托盘下降 | 
|                 shuttleTaskModeType = ShuttleTaskModeType.PALLET_DOWN; | 
|                 shuttleCommands.add(shuttleThread.getLiftCommand(motion.getTaskNo(), false)); | 
|                 break; | 
|             default: | 
|                 throw new CoolException(motion.getMotionCtgEl() + "没有指定任务作业流程!!!"); | 
|         } | 
|   | 
|         if (Cools.isEmpty(shuttleCommands)) { | 
|             return false; | 
|         } | 
|   | 
|         assert null != shuttleTaskModeType; | 
|         assignCommand.setTaskMode(shuttleTaskModeType.id);//入出库模式 | 
|         assignCommand.setCommands(shuttleCommands); | 
|   | 
|         if (motion.getOrigin() != null && motion.getTarget() != null) { | 
|             //所使用的路径进行锁定禁用 | 
|             boolean lockResult = navigateMapUtils.writeNavigateNodeToRedisMap(Utils.getLev(motion.getTarget()), shuttleProtocol.getShuttleNo(), assignCommand.getNodes(), true);//所使用的路径进行锁定禁用 | 
|             if (!lockResult) { | 
|                 return false;//锁定失败 | 
|             } | 
|         } | 
|   | 
|         boolean result = shuttleAction.assignWork(shuttleThread.getDevice(), assignCommand); | 
|         return result; | 
|     } | 
|   | 
|     public Boolean finish(Motion motion) { | 
|         Integer deviceNo = Integer.parseInt(motion.getDevice()); | 
|         ShuttleThread shuttleThread = (ShuttleThread) SlaveConnection.get(SlaveType.Shuttle, deviceNo); | 
|         ShuttleProtocol shuttleProtocol = shuttleThread.getStatus(); | 
|         if (null == shuttleProtocol) { | 
|             return false; | 
|         } | 
|   | 
|         if (shuttleProtocol.getTaskNo() != 0 && shuttleProtocol.getTaskNo().intValue() != motion.getTaskNo()) { | 
|             return false; | 
|         } | 
|   | 
|         //充电任务 | 
|         if (Objects.requireNonNull(MotionCtgType.get(motion.getMotionCtgEl())).equals(MotionCtgType.SHUTTLE_CHARGE_ON)) { | 
|             // 复位穿梭车 | 
|             shuttleThread.setSyncTaskNo(0); | 
|             shuttleThread.setProtocolStatus(ShuttleProtocolStatusType.IDLE); | 
|             shuttleThread.setPakMk(true); | 
|             return true; | 
|         } | 
|   | 
|         if (!shuttleProtocol.getProtocolStatusType().equals(ShuttleProtocolStatusType.WAITING) | 
|             && !shuttleProtocol.getProtocolStatusType().equals(ShuttleProtocolStatusType.CHARGING_WAITING) | 
|         ) { | 
|             return false; | 
|         } | 
|   | 
|         //判断设备是否空闲 | 
|         if (!shuttleThread.isIdle()) { | 
|             return false; | 
|         } | 
|   | 
|         LiftThread liftThread = null; | 
|         LiftProtocol liftProtocol = null; | 
|   | 
|         switch (Objects.requireNonNull(MotionCtgType.get(motion.getMotionCtgEl()))){ | 
|             case SHUTTLE_MOVE: | 
|             case SHUTTLE_MOVE_LIFT_PALLET: | 
|             case SHUTTLE_MOVE_DOWN_PALLET: | 
|             case SHUTTLE_MOVE_TO_CONVEYOR: | 
|             case SHUTTLE_MOVE_FROM_CONVEYOR: | 
|                 if (!shuttleProtocol.getCurrentLocNo().equals(motion.getTarget())) { | 
|                     return false; | 
|                 } | 
|                 break; | 
|             case SHUTTLE_MOVE_TO_LIFT: | 
|             case SHUTTLE_MOVE_FROM_LIFT: | 
|             case SHUTTLE_TRANSPORT_FROM_LIFT: | 
|             case SHUTTLE_TRANSPORT_TO_LIFT: | 
|             case SHUTTLE_MOVE_FROM_LIFT_TO_CONVEYOR: | 
|                 liftThread = (LiftThread) SlaveConnection.get(SlaveType.Lift, Integer.parseInt(motion.getTemp())); | 
|                 if (liftThread == null) { | 
|                     return false; | 
|                 } | 
|                 liftProtocol = liftThread.getStatus(); | 
|   | 
|                 if (!shuttleProtocol.getCurrentLocNo().equals(motion.getTarget())) { | 
|                     return false; | 
|                 } | 
|   | 
| //                //判断提升机是否被锁定 | 
| //                if (liftProtocol.getLiftLock()) { | 
| //                    //解锁提升机 | 
| //                    LiftCommand lockCommand = liftThread.getLockCommand(false);//获取提升机解锁命令 | 
| //                    lockCommand.setLiftNo(liftProtocol.getLiftNo()); | 
| //                    lockCommand.setTaskNo(motion.getWrkNo().shortValue());//获取任务号 | 
| //                    liftThread.assignWork(lockCommand); | 
| //                    return false; | 
| //                } | 
|   | 
|                 //判断提升机工作号是否和当前任务相同 | 
|                 if (liftProtocol.getTaskNo().intValue() != motion.getTaskNo()) { | 
|                     return false; | 
|                 } | 
|   | 
| //                if (liftProtocol.getTaskNo() != 0) { | 
| //                    //清空提升机号 | 
| //                    liftThread.setTaskNo(0); | 
| //                } | 
|   | 
|                 break; | 
|             default: | 
|                 break; | 
|         } | 
|   | 
|         // 复位穿梭车 | 
|         shuttleThread.setSyncTaskNo(0); | 
|         shuttleThread.setProtocolStatus(ShuttleProtocolStatusType.IDLE); | 
|         shuttleThread.setPakMk(true); | 
|   | 
|         return true; | 
|     } | 
|   | 
|     public synchronized List<ShuttleCommand> shuttleAssignCommand(String startLocNo, String endLocNo, Integer mapType, ShuttleAssignCommand assignCommand, ShuttleThread shuttleThread) { | 
|         //获取小车移动速度 | 
|         Integer runSpeed = Optional.ofNullable(basShuttleService.getOne(new LambdaQueryWrapper<BasShuttle>().eq(BasShuttle::getDeviceId, assignCommand.getDeviceId())).getRunSpeed()).orElse(1000); | 
|         Long hostId = shuttleThread.getDevice().getHostId(); | 
|         List<NavigateNode> nodeList = NavigateUtils.calc(startLocNo, endLocNo, mapType, Utils.getShuttlePoints(Integer.parseInt(shuttleThread.getDevice().getDeviceNo()), Utils.getLev(startLocNo))); | 
|         if (nodeList == null) { | 
|             News.error("{} dash {} can't find navigate path!", startLocNo, endLocNo); | 
|             return null; | 
|         } | 
|   | 
|         List<NavigateNode> allNode = new ArrayList<>(); | 
|         for (NavigateNode node : nodeList) { | 
|             allNode.add(node.clone()); | 
|         } | 
|   | 
|         List<ShuttleCommand> commands = new ArrayList<>(); | 
|         //获取分段路径 | 
|         ArrayList<ArrayList<NavigateNode>> data = NavigateUtils.getSectionPath(nodeList); | 
|         //将每一段路径分成command指令 | 
|         for (ArrayList<NavigateNode> nodes : data) { | 
|             //开始路径 | 
|             NavigateNode startPath = nodes.get(0); | 
|   | 
|             //目标路径 | 
|             NavigateNode endPath = nodes.get(nodes.size() - 1); | 
|             Integer allDistance = NavigateUtils.getCurrentPathAllDistance(nodes);//计算当前路径行走总距离 | 
|             //通过xy坐标小车二维码 | 
|             String startCodeNum = NavigatePositionConvert.xyToPosition(startPath.getX(), startPath.getY(), startPath.getZ(), hostId); | 
|             //通过xy坐标小车二维码 | 
|             String distCodeNum = NavigatePositionConvert.xyToPosition(endPath.getX(), endPath.getY(), endPath.getZ(), hostId); | 
|             //获取移动命令 | 
|             ShuttleCommand command = shuttleThread.getMoveCommand(assignCommand.getDeviceTaskNo(), startCodeNum, distCodeNum, allDistance, ShuttleRunDirection.get(startPath.getDirection()).id.intValue(), runSpeed); | 
|             command.setNodes(nodes);//将行走节点添加到每一步命令中 | 
|             commands.add(command); | 
|         } | 
|   | 
|         assignCommand.setNodes(allNode);//当前任务所占用的节点list | 
|   | 
|         return commands; | 
|     } | 
|   | 
|   | 
| } |