package com.zy.acs.manager.core.scheduler.test; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.zy.acs.common.constant.RedisConstant; import com.zy.acs.common.utils.RedisSupport; import com.zy.acs.framework.common.Cools; import com.zy.acs.framework.common.SnowflakeIdWorker; import com.zy.acs.manager.common.domain.TaskDto; import com.zy.acs.manager.core.service.*; import com.zy.acs.manager.core.integrate.dto.OpenBusSubmitParam; import com.zy.acs.manager.manager.entity.*; import com.zy.acs.manager.manager.enums.*; import com.zy.acs.manager.manager.service.*; import com.zy.acs.manager.manager.service.impl.CodeServiceImpl; import com.zy.acs.manager.manager.service.impl.TaskServiceImpl; import com.zy.acs.manager.system.service.ConfigService; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Component; import java.util.*; import java.util.stream.Collectors; @Slf4j @Component public class AutoRunScheduler { private static final AgvModelType DEFAULT_AGV_MODEL = AgvModelType.CTU_BOX_TRANSPORT_AGV; private final RedisSupport redis = RedisSupport.defaultRedisSupport; @Autowired private AgvService agvService; @Autowired private BusService busService; @Autowired private MainService mainService; @Autowired private ConfigService configService; @Autowired private LocService locService; @Autowired private StaService staService; @Autowired private AgvModelService agvModelService; @Autowired private SnowflakeIdWorker snowflakeIdWorker; @Autowired private AreaGovernService areaGovernService; @Autowired private CodeServiceImpl codeService; @Autowired private AreaAgvService areaAgvService; @Autowired private AreaService areaService; @Autowired private TaskServiceImpl taskService; // @Scheduled(fixedRate = 500) // 固定频率执行,不同步 @Scheduled(fixedDelay = 1000) // 固定频率执行,同步 // @Scheduled(cron = "0/1 * * * * ? ") private void execute() { if (!configService.getVal("TaskAssignMode", Boolean.class)) { return; } Boolean maintainLocSts = configService.getVal("maintainLocSts", Boolean.class); if (!maintainLocSts) { log.error("failed to run auto agv task, the maintainLocSts is false"); return; } AgvModel agvModel = agvModelService.getOne(new LambdaQueryWrapper().eq(AgvModel::getType, DEFAULT_AGV_MODEL.toString())); if (null == agvModel) { return; } this.autoRun(agvModel); } private void autoRun(AgvModel agvModel) { int availableAgvCount = this.getAvailableAgvCount(); if (0 == availableAgvCount) { log.error("failed to run auto agv task, 跑库区域无可用小车"); return; } // List staPreNos = getStaPrefixes(staGroupList); List staPreNos = new ArrayList<>(); String memo = "DEMO_STA_" + String.join("-", staPreNos); List codes = getTestAreaCodes(); if (codes.isEmpty()) { log.error("failed to run auto agv task, 未配置跑库区域"); return; } // 入库 this.runStaToLoc(agvModel, codes, memo); // 出库 this.runLocToSta(agvModel, codes, memo); // 移库 this.runLocToLoc(agvModel, codes, memo); } // 入库 private void runStaToLoc(AgvModel agvModel, List codes, String memo) { // STOCK STA List stockList = staService.queryAvailableOutSta(1); List staList = new ArrayList<>(); for (Sta sta : stockList) { if (sta.getCode$() != null && codes.contains(sta.getCode$())) { List list = taskService.list(new LambdaQueryWrapper().eq(Task::getOriSta, sta.getCode()).or().eq(Task::getDestSta, sta.getCode())); if ((sta.getCapacity() - sta.getRsvInCnt()) > (list.isEmpty() ? 0 : list.size())) { staList.add(sta); } } } if (Cools.isEmpty(staList)) { return; } Collections.shuffle(staList); OpenBusSubmitParam param = new OpenBusSubmitParam(); param.setBatchNo(String.valueOf(snowflakeIdWorker.nextId()).substring(13, 19)); for (int i = 0; i < Math.min(agvModel.getBackpack(), staList.size()); i++) { Sta stockSta = staList.get(i); Loc idleLoc = getLocByLocSts(LocStsType.IDLE.val(), codes); if (null == idleLoc) { break; } TaskDto taskDto = new TaskDto(); taskDto.setOriSta(stockSta.getStaNo()); taskDto.setDestLoc(idleLoc.getLocNo()); taskDto.setPriority(100); taskDto.setTaskNo(String.valueOf(snowflakeIdWorker.nextId()).substring(15, 19)); param.getTasks().add(taskDto); } if (Cools.isEmpty(param.getTasks())) { return; } mainService.generateBusAndTask(param, memo); } // 出库 private void runLocToSta(AgvModel agvModel, List codes, String memo) { // IDLE STA List idleList = staService.queryAvailableInSta(1); List staList = new ArrayList<>(); for (Sta sta : idleList) { if (sta.getCode$() != null && codes.contains(sta.getCode$())) { List list = taskService.list(new LambdaQueryWrapper().eq(Task::getOriSta, sta.getCode()).or().eq(Task::getDestSta, sta.getCode())); if ((sta.getCapacity() - sta.getRsvOutCnt()) > (list.isEmpty() ? 0 : list.size())) { staList.add(sta); } } } if (Cools.isEmpty(staList)) { return; } Collections.shuffle(staList); OpenBusSubmitParam param = new OpenBusSubmitParam(); param.setBatchNo(String.valueOf(snowflakeIdWorker.nextId()).substring(13, 19)); for (int i = 0; i < Math.min(agvModel.getBackpack(), staList.size()); i++) { Sta idleSta = staList.get(i); Loc stockLoc = getLocByLocSts(LocStsType.STOCK.val(), codes); if (null == stockLoc) { break; } TaskDto taskDto = new TaskDto(); taskDto.setOriLoc(stockLoc.getLocNo()); taskDto.setDestSta(idleSta.getStaNo()); taskDto.setPriority(100); taskDto.setTaskNo(String.valueOf(snowflakeIdWorker.nextId()).substring(15, 19)); param.getTasks().add(taskDto); } if (Cools.isEmpty(param.getTasks())) { return; } mainService.generateBusAndTask(param, memo); } // 移库 private void runLocToLoc(AgvModel agvModel, List codes, String staTaskMemo) { String memo = "DEMO_LOC"; int availableAgvCount = this.getAvailableAgvCount(); // 最多 ? 组bus运行 if (availableAgvCount <= busService.count(new LambdaQueryWrapper() .in(Bus::getBusSts, BusStsType.RECEIVE.val(), BusStsType.PROGRESS.val()) .in(Bus::getMemo, memo, staTaskMemo) )) { return; } int maxCapacity = agvModel.getBackpack(); // STOCK List stockLocList = getAllLocByLocSts(LocStsType.STOCK.val(), codes); if (Cools.isEmpty(stockLocList)) { return; } Collections.shuffle(stockLocList); // IDLE List idleLocList = getAllLocByLocSts(LocStsType.IDLE.val(), codes); if (Cools.isEmpty(idleLocList)) { return; } Collections.shuffle(idleLocList); OpenBusSubmitParam param = new OpenBusSubmitParam(); param.setBatchNo(String.valueOf(snowflakeIdWorker.nextId()).substring(13, 19)); for (int i = 0; i < Math.min(maxCapacity, Math.min(stockLocList.size(), idleLocList.size())); i++) { Loc stockLoc = stockLocList.get(i); Loc idleLoc = idleLocList.get(i); TaskDto taskDto = new TaskDto(); taskDto.setOriLoc(stockLoc.getLocNo()); taskDto.setDestLoc(idleLoc.getLocNo()); taskDto.setTaskNo(String.valueOf(snowflakeIdWorker.nextId()).substring(15, 19)); param.getTasks().add(taskDto); } if (Cools.isEmpty(param.getTasks())) { return; } mainService.generateBusAndTask(param, memo); } public Loc getLocByLocSts(Long locSts, List codes) { List list = locService.list(new LambdaQueryWrapper().eq(Loc::getLocSts, locSts)); Collections.shuffle(list); for (Loc loc : list) { if (codes.contains(loc.getCode$())) { return loc; } } return null; } public List getAllLocByLocSts(Long locSts, List codes) { List data = new ArrayList<>(); List list = locService.list(new LambdaQueryWrapper().eq(Loc::getLocSts, locSts)); Collections.shuffle(list); for (Loc loc : list) { if (codes.contains(loc.getCode$())) { data.add(loc); } } return data; } public List getTestAreaCodes() { String taskAssignModeArea = configService.getVal("TaskAssignModeArea", String.class); if (Cools.isEmpty(taskAssignModeArea)) { List list = codeService.list(new LambdaQueryWrapper().eq(Code::getStatus, 1)); return list.stream().map(Code::getData).collect(Collectors.toList()); } List list = areaService.list(new LambdaQueryWrapper().eq(Area::getName, taskAssignModeArea)); if (list.isEmpty()) { return Collections.emptyList(); } return areaGovernService.queryCodes(list.stream().map(Area::getId).collect(Collectors.toList())); } public List getTestAreaAgvs() { String taskAssignModeArea = configService.getVal("TaskAssignModeArea", String.class); if (Cools.isEmpty(taskAssignModeArea)) { List list = agvService.list(new LambdaQueryWrapper().eq(Agv::getStatus, 1)); return list.stream().map(Agv::getId).collect(Collectors.toList()); } List list = areaService.list(new LambdaQueryWrapper().eq(Area::getName, taskAssignModeArea)); if (list.isEmpty()) { return Collections.emptyList(); } List list1 = areaAgvService.list(new LambdaQueryWrapper().in(AreaAgv::getAreaId, list.stream().map(Area::getId).collect(Collectors.toList()))); return list1.stream().map(AreaAgv::getAgvId).collect(Collectors.toList()); } public static List getStaPrefixes(List staGroupList) { Set rowSet = new HashSet<>(); for (String s : staGroupList) { rowSet.add(s.split("-")[0]); } List result = new ArrayList<>(rowSet); result.sort(Comparator.comparingInt(Integer::parseInt)); return result; } private int getAvailableAgvCount() { int res = 0; List testAreaAgvs = getTestAreaAgvs(); for (Long agvId : testAreaAgvs) { String agvNo = agvService.getAgvNo(agvId); Agv agv = agvService.selectByUuid(agvNo); if (null == redis.getObject(RedisConstant.AGV_ONLINE_FLAG, agvNo)) { continue; } if (!agv.getStatusBool()) { continue; } res++; } return res; } }