package com.zy.acs.manager.core;
|
|
import com.alibaba.fastjson.JSON;
|
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.common.utils.Utils;
|
import com.zy.acs.framework.common.Cools;
|
import com.zy.acs.framework.common.R;
|
import com.zy.acs.framework.common.SnowflakeIdWorker;
|
import com.zy.acs.manager.common.annotation.OperationLog;
|
import com.zy.acs.manager.common.domain.param.HandlerPublishParam;
|
import com.zy.acs.manager.common.exception.BusinessException;
|
import com.zy.acs.manager.core.service.*;
|
import com.zy.acs.manager.core.service.astart.MapDataDispatcher;
|
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.system.controller.BaseController;
|
import lombok.extern.slf4j.Slf4j;
|
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.jdbc.core.JdbcTemplate;
|
import org.springframework.security.access.prepost.PreAuthorize;
|
import org.springframework.transaction.annotation.Transactional;
|
import org.springframework.web.bind.annotation.*;
|
|
import java.util.Date;
|
import java.util.List;
|
import java.util.concurrent.CompletableFuture;
|
import java.util.concurrent.ExecutionException;
|
import java.util.stream.Collectors;
|
|
/**
|
* Created by vincent on 8/1/2024
|
*/
|
@Slf4j
|
@RestController
|
@RequestMapping("/api/handler")
|
public class HandlerController extends BaseController {
|
|
private final RedisSupport redis = RedisSupport.defaultRedisSupport;
|
|
public static final String APP_KEY = "xltys1995";
|
|
@Autowired
|
private AgvService agvService;
|
@Autowired
|
private AgvDetailService agvDetailService;
|
@Autowired
|
private TaskService taskService;
|
@Autowired
|
private MainService mainService;
|
@Autowired
|
private MainLockWrapService mainLockWrapService;
|
@Autowired
|
private SnowflakeIdWorker snowflakeIdWorker;
|
@Autowired
|
private CodeService codeService;
|
@Autowired
|
private JdbcTemplate jdbcTemplate;
|
@Autowired
|
private LocService locService;
|
@Autowired
|
private StaService staService;
|
@Autowired
|
private TrafficService trafficService;
|
@Autowired
|
private ThreadPoolRegulator threadPoolRegulator;
|
@Autowired
|
private MapDataDispatcher mapDataDispatcher;
|
@Autowired
|
private AvoidWaveCalculator avoidWaveCalculator;
|
@Autowired
|
private PatrolService patrolService;
|
|
@PreAuthorize("hasAuthority('manager:agv:update')")
|
@OperationLog("Locate All Agv")
|
@PostMapping("/locateAllAgv")
|
public synchronized R locateAllAgv() {
|
final Integer MAP_DEFAULT_LEV = 1;
|
redis.deleteValue(RedisConstant.AGV_MAP_ASTAR_DYNAMIC_FLAG, String.valueOf(MAP_DEFAULT_LEV));
|
avoidWaveCalculator.calcDynamicNodeWhenBoot();
|
return R.ok();
|
}
|
|
@PreAuthorize("hasAuthority('manager:agv:update')")
|
@PostMapping("/patrol/batch/startup")
|
public synchronized R patrolBatchStartup() {
|
List<Agv> list = agvService.list(new LambdaQueryWrapper<Agv>().eq(Agv::getStatus, StatusType.ENABLE.val));
|
int result = 0;
|
for (Agv agv : list) {
|
patrolService.startupPatrol(agv.getUuid());
|
result++;
|
}
|
return R.ok().add(result);
|
}
|
|
@PreAuthorize("hasAuthority('manager:agv:update')")
|
@PostMapping("/patrol/batch/shutdown")
|
public synchronized R patrolBatchShutdown() {
|
List<Agv> list = agvService.list(new LambdaQueryWrapper<Agv>());
|
for (String agvNo : list.stream().map(Agv::getUuid).collect(Collectors.toList())) {
|
if (patrolService.isPatrolling(agvNo)) {
|
patrolService.shutdownPatrol(agvNo);
|
}
|
}
|
return R.ok();
|
}
|
|
@RequestMapping(value = "/control/agv", method = {RequestMethod.GET, RequestMethod.POST})
|
@Transactional
|
public R controlAgv(@RequestHeader String appKey,
|
@RequestBody HandlerPublishParam param) {
|
if (Cools.isEmpty(param.getAgvNo(), param.getTaskMode(), appKey)) {
|
return R.error();
|
}
|
if (!APP_KEY.equals(appKey)) {
|
return R.error();
|
}
|
|
Date now = new Date();
|
Agv agv = agvService.selectByUuid(param.getAgvNo());
|
AgvDetail agvDetail = agvDetailService.selectByAgvId(agv.getId());
|
|
// check
|
if (taskService.count(new LambdaQueryWrapper<Task>()
|
.eq(Task::getAgvId, agv.getId())
|
.and(i -> {
|
i.eq(Task::getTaskSts, TaskStsType.WAITING.val())
|
.or()
|
.eq(Task::getTaskSts, TaskStsType.ASSIGN.val())
|
.or().eq(Task::getTaskSts, TaskStsType.PROGRESS.val());
|
})) > 0) {
|
log.info(agv.getUuid() + "号AGV不可用,已经存在进行中的任务...");
|
return R.error();
|
}
|
if (!agvService.judgeEnable(agv.getId())) {
|
log.info(agv.getUuid() + "号AGV不可用,任务无法计算...");
|
return R.error();
|
}
|
|
// generate
|
Loc oriLoc = null;
|
Loc destLoc = null;
|
Sta oriSta = null;
|
Sta destSta = null;
|
Task task = null;
|
switch (param.getTaskMode()) {
|
case MOVE:
|
Code endCode = null;
|
if (!Cools.isEmpty(param.getEndCode())) {
|
endCode = codeService.getCacheById(param.getEndCode());
|
}
|
if (!Cools.isEmpty(param.getEndCodeStr())) {
|
endCode = codeService.getCacheByData(param.getEndCodeStr());
|
}
|
if (null == endCode) {
|
return R.error();
|
}
|
if (!mainLockWrapService.buildMinorTask(agv.getId(), param.getTaskMode(), endCode.getData(), null)) {
|
return R.error();
|
}
|
break;
|
case TO_CHARGE:
|
case TO_STANDBY:
|
if (!mainLockWrapService.buildMinorTask(agv.getId(), param.getTaskMode(), null, null)) {
|
return R.error();
|
}
|
break;
|
case LOC_TO_LOC:
|
// oriLoc
|
if (!Cools.isEmpty(param.getStartLocNo())) {
|
oriLoc = locService.getById(param.getStartLocNo());
|
}
|
if (!Cools.isEmpty(param.getStartLocNoStr())) {
|
oriLoc = locService.selecatByLocNo(param.getStartLocNoStr());
|
}
|
if (null == oriLoc) {
|
return R.error();
|
}
|
if (!oriLoc.getLocSts().equals(LocStsType.STOCK.val())) {
|
throw new BusinessException("oriLoc:" + oriLoc.getLocNo() + " 不是在库状态");
|
}
|
oriLoc.setLocSts(LocStsType.PAKOUT.val());
|
oriLoc.setUpdateTime(now);
|
if (!locService.updateById(oriLoc)) {
|
throw new BusinessException("oriLoc:" + oriLoc.getLocNo() + " 修改库位状态失败");
|
}
|
|
// destLoc
|
if (!Cools.isEmpty(param.getEndLocNo())) {
|
destLoc = locService.getById(param.getEndLocNo());
|
}
|
if (!Cools.isEmpty(param.getEndLocNoStr())) {
|
destLoc = locService.selecatByLocNo(param.getEndLocNoStr());
|
}
|
if (null == destLoc) {
|
return R.error();
|
}
|
if (!destLoc.getLocSts().equals(LocStsType.IDLE.val())) {
|
throw new BusinessException("destLoc:" + destLoc.getLocNo() + " 不是空闲状态");
|
}
|
destLoc.setLocSts(LocStsType.PAKIN.val());
|
destLoc.setUpdateTime(now);
|
if (!locService.updateById(destLoc)) {
|
throw new BusinessException("destLoc:" + destLoc.getLocNo() + " 修改库位状态失败");
|
}
|
|
// task
|
task = new Task();
|
|
task.setOriLoc(oriLoc.getId());
|
task.setOriCode(oriLoc.getCode());
|
task.setDestLoc(destLoc.getId());
|
task.setDestCode(destLoc.getCode());
|
break;
|
case LOC_TO_STA:
|
// oriLoc
|
if (!Cools.isEmpty(param.getStartLocNo())) {
|
oriLoc = locService.getById(param.getStartLocNo());
|
}
|
if (!Cools.isEmpty(param.getStartLocNoStr())) {
|
oriLoc = locService.selecatByLocNo(param.getStartLocNoStr());
|
}
|
if (null == oriLoc) {
|
return R.error();
|
}
|
if (!oriLoc.getLocSts().equals(LocStsType.STOCK.val())) {
|
throw new BusinessException("oriLoc:" + oriLoc.getLocNo() + " 不是在库状态");
|
}
|
oriLoc.setLocSts(LocStsType.PAKOUT.val());
|
oriLoc.setUpdateTime(now);
|
if (!locService.updateById(oriLoc)) {
|
throw new BusinessException("oriLoc:" + oriLoc.getLocNo() + " 修改库位状态失败");
|
}
|
|
// destSta
|
if (!Cools.isEmpty(param.getEndStaNo())) {
|
destSta = staService.getById(param.getEndStaNo());
|
}
|
if (!Cools.isEmpty(param.getEndStaNoStr())) {
|
destSta = staService.selectByStaNo(param.getEndStaNoStr());
|
}
|
if (null == destSta) {
|
return R.error();
|
}
|
if (!destSta.getStaSts().equals(StaStsType.IDLE.val())) {
|
throw new BusinessException("destSta:" + destSta.getStaNo() + " 不是无货状态");
|
}
|
destSta.setStaSts(StaStsType.READY_RELEASE.val());
|
destSta.setUpdateTime(now);
|
if (!staService.updateById(destSta)) {
|
throw new BusinessException("destSta:" + destSta.getStaNo() + " 修改站点状态失败");
|
}
|
|
// task
|
task = new Task();
|
|
task.setOriLoc(oriLoc.getId());
|
task.setOriCode(oriLoc.getCode());
|
task.setDestSta(destSta.getId());
|
task.setDestCode(destSta.getCode());
|
break;
|
case STA_TO_LOC:
|
// oriSta
|
if (!Cools.isEmpty(param.getStartStaNo())) {
|
oriSta = staService.getById(param.getStartStaNo());
|
}
|
if (!Cools.isEmpty(param.getStartStaNoStr())) {
|
oriSta = staService.selectByStaNo(param.getStartStaNoStr());
|
}
|
if (null == oriSta) {
|
return R.error();
|
}
|
if (!oriSta.getStaSts().equals(StaStsType.STOCK.val())) {
|
throw new BusinessException("oriSta:" + oriSta.getStaNo() + " 不是有货状态");
|
}
|
oriSta.setStaSts(StaStsType.READY_TAKE.val());
|
oriSta.setUpdateTime(now);
|
if (!staService.updateById(oriSta)) {
|
throw new BusinessException("oriSta:" + oriSta.getStaNo() + " 修改站点状态失败");
|
}
|
|
// destLoc
|
if (!Cools.isEmpty(param.getEndLocNo())) {
|
destLoc = locService.getById(param.getEndLocNo());
|
}
|
if (!Cools.isEmpty(param.getEndLocNoStr())) {
|
destLoc = locService.selecatByLocNo(param.getEndLocNoStr());
|
}
|
if (null == destLoc) {
|
return R.error();
|
}
|
if (!destLoc.getLocSts().equals(LocStsType.IDLE.val())) {
|
throw new BusinessException("destLoc:" + destLoc.getLocNo() + " 不是空闲状态");
|
}
|
destLoc.setLocSts(LocStsType.PAKIN.val());
|
destLoc.setUpdateTime(now);
|
if (!locService.updateById(destLoc)) {
|
throw new BusinessException("destLoc:" + destLoc.getLocNo() + " 修改库位状态失败");
|
}
|
|
// task
|
task = new Task();
|
|
task.setOriSta(oriSta.getId());
|
task.setOriCode(oriSta.getCode());
|
task.setDestLoc(destLoc.getId());
|
task.setDestCode(destLoc.getCode());
|
break;
|
case STA_TO_STA:
|
// oriSta
|
if (!Cools.isEmpty(param.getStartStaNo())) {
|
oriSta = staService.getById(param.getStartStaNo());
|
}
|
if (!Cools.isEmpty(param.getStartStaNoStr())) {
|
oriSta = staService.selectByStaNo(param.getStartStaNoStr());
|
}
|
if (null == oriSta) {
|
return R.error();
|
}
|
if (!oriSta.getStaSts().equals(StaStsType.STOCK.val())) {
|
throw new BusinessException("oriSta:" + oriSta.getStaNo() + " 不是有货状态");
|
}
|
oriSta.setStaSts(StaStsType.READY_TAKE.val());
|
oriSta.setUpdateTime(now);
|
if (!staService.updateById(oriSta)) {
|
throw new BusinessException("oriSta:" + oriSta.getStaNo() + " 修改站点状态失败");
|
}
|
|
// destSta
|
if (!Cools.isEmpty(param.getEndStaNo())) {
|
destSta = staService.getById(param.getEndStaNo());
|
}
|
if (!Cools.isEmpty(param.getEndStaNoStr())) {
|
destSta = staService.selectByStaNo(param.getEndStaNoStr());
|
}
|
if (null == destSta) {
|
return R.error();
|
}
|
if (!destSta.getStaSts().equals(StaStsType.IDLE.val())) {
|
throw new BusinessException("destSta:" + destSta.getStaNo() + " 不是无货状态");
|
}
|
destSta.setStaSts(StaStsType.READY_RELEASE.val());
|
destSta.setUpdateTime(now);
|
if (!staService.updateById(destSta)) {
|
throw new BusinessException("destSta:" + destSta.getStaNo() + " 修改站点状态失败");
|
}
|
|
// task
|
task = new Task();
|
|
task.setOriSta(oriSta.getId());
|
task.setOriCode(oriSta.getCode());
|
task.setDestSta(destSta.getId());
|
task.setDestCode(destSta.getCode());
|
break;
|
default:
|
break;
|
}
|
|
if (null != task) {
|
task.setAgvId(agv.getId());
|
task.setUuid(String.valueOf(snowflakeIdWorker.nextId()).substring(3));
|
List<Task> lastTasks = taskService.list(new LambdaQueryWrapper<Task>().orderByDesc(Task::getId));
|
task.setSeqNum(Utils.generateSeqNum(Cools.isEmpty(lastTasks)?null:lastTasks.get(0).getSeqNum()));
|
|
task.setTaskType(param.getTaskMode().val());
|
task.setTaskSts(TaskStsType.WAITING.val());
|
|
task.setPriority(999);
|
task.setIoTime(now);
|
task.setStartTime(now);
|
if (!taskService.save(task)) {
|
throw new BusinessException(task.getSeqNum() + "任务保存失败");
|
}
|
}
|
|
return R.ok();
|
}
|
|
@RequestMapping(value = "/restore/agv", method = {RequestMethod.GET, RequestMethod.POST})
|
public R restoreAgv(@RequestHeader String appKey, @RequestBody HandlerPublishParam param) {
|
if (Cools.isEmpty(appKey)) {
|
return R.error();
|
}
|
if (!APP_KEY.equals(appKey)) {
|
return R.error();
|
}
|
|
Agv agv = null;
|
if (!Cools.isEmpty(param.getAgvNo())) {
|
agv = agvService.selectByUuid(param.getAgvNo());
|
List<Task> tasks = taskService.selectInSts(agv.getId(), TaskStsType.WAITING, TaskStsType.ASSIGN, TaskStsType.PROGRESS);
|
if (!Cools.isEmpty(tasks)) {
|
for (Task task : tasks) {
|
if (TaskTypeType.LOC_TO_LOC.toString().equals(task.getTaskTypeEl())
|
|| TaskTypeType.LOC_TO_STA.toString().equals(task.getTaskTypeEl())
|
|| TaskTypeType.STA_TO_LOC.toString().equals(task.getTaskTypeEl())
|
|| TaskTypeType.STA_TO_STA.toString().equals(task.getTaskTypeEl())
|
) {
|
return R.error("Restore Failed, because the agv has Tasks that are associated with Loc!");
|
}
|
}
|
}
|
}
|
|
if (null != agv) {
|
mapDataDispatcher.modifyDynamicMatrix(null, null, agv.getUuid(), true);
|
avoidWaveCalculator.calcDynamicNodeByVehicle(agv, null);
|
|
jdbcTemplate.update("UPDATE man_task SET task_sts = " + TaskStsType.COMPLETE.val() + " where agv_id = " + agv.getId());
|
jdbcTemplate.update("UPDATE man_travel SET state = 'FINISH'" + " where agv_id = " + agv.getId());
|
jdbcTemplate.update("UPDATE man_segment SET state = 'FINISH'" + " where agv_id = " + agv.getId());
|
jdbcTemplate.update("UPDATE man_action SET action_sts = " + ActionStsType.FINISH.val() + " where agv_id = " + agv.getId());
|
} else {
|
List<Agv> agvList = agvService.list(new LambdaQueryWrapper<Agv>());
|
for (Agv one : agvList) {
|
mapDataDispatcher.modifyDynamicMatrix(null, null, one.getUuid(), true);
|
avoidWaveCalculator.calcDynamicNodeByVehicle(one, null);
|
}
|
|
jdbcTemplate.update("UPDATE man_task SET task_sts = " + TaskStsType.COMPLETE.val());
|
jdbcTemplate.update("UPDATE man_travel SET state = 'FINISH'");
|
jdbcTemplate.update("UPDATE man_segment SET state = 'FINISH'");
|
jdbcTemplate.update("UPDATE man_action SET action_sts = " + ActionStsType.FINISH.val());
|
}
|
|
return R.ok();
|
}
|
|
@RequestMapping(value = "/unlock", method = {RequestMethod.GET, RequestMethod.POST})
|
public R unlockPath(@RequestHeader String appKey,
|
@RequestBody HandlerPublishParam param) throws ExecutionException, InterruptedException {
|
if (Cools.isEmpty(param.getAgvNo(), appKey)) {
|
return R.error();
|
}
|
if (!APP_KEY.equals(appKey)) {
|
return R.error();
|
}
|
|
Agv agv = agvService.selectByUuid(param.getAgvNo());
|
|
// block
|
// Future<R> future = threadPoolRegulator.getInstance().submit(() -> {
|
// mapDataDispatcher.modifyDynamicMatrix(null, null, param.getAgvNo(), true);
|
// return success();
|
// });
|
// System.out.println(JSON.toJSONString(future.get()));
|
|
// non-block
|
CompletableFuture<?> completableFuture = CompletableFuture.supplyAsync(() -> {
|
mapDataDispatcher.modifyDynamicMatrix(null, null, param.getAgvNo(), true);
|
avoidWaveCalculator.calcDynamicNodeByVehicle(agv, null);
|
return R.ok();
|
}, threadPoolRegulator.getInstance());
|
|
completableFuture.thenAccept(result -> {
|
System.out.println(JSON.toJSONString(result));
|
}).exceptionally(e -> {
|
log.error("unlockPath", e);
|
return null;
|
});
|
|
return R.ok();
|
}
|
|
@RequestMapping(value = "/agv/patrol", method = {RequestMethod.GET, RequestMethod.POST})
|
public R agvPatrol(@RequestHeader String appKey,
|
@RequestBody HandlerPublishParam param) throws ExecutionException, InterruptedException {
|
if (Cools.isEmpty(param.getAgvNo(), appKey)) {
|
return R.error();
|
}
|
if (!APP_KEY.equals(appKey)) {
|
return R.error();
|
}
|
String agvNo = param.getAgvNo();
|
if (patrolService.isPatrolling(agvNo)) {
|
return patrolService.shutdownPatrol(agvNo);
|
} else {
|
return patrolService.startupPatrol(agvNo);
|
}
|
}
|
|
}
|