From 9ed9cd2e6f619c84732ae6715699b160c404684c Mon Sep 17 00:00:00 2001
From: Junjie <fallin.jie@qq.com>
Date: 星期二, 24 三月 2026 12:42:19 +0800
Subject: [PATCH] #V5 beta
---
src/main/java/com/zy/core/model/command/StationCommand.java | 3
src/main/java/com/zy/core/utils/StationOperateProcessUtils.java | 154 ++++
src/main/java/com/zy/core/move/StationMoveSession.java | 64 ++
src/main/java/com/zy/core/thread/impl/v5/StationMoveSegmentExecutor.java | 18
src/main/java/com/zy/core/thread/impl/ZyStationV4Thread.java | 5
src/main/java/com/zy/core/move/StationMoveTriggerType.java | 9
src/main/java/com/zy/core/thread/impl/station/StationSegmentExecutor.java | 456 +++++++++++++++
src/main/java/com/zy/core/move/StationMoveDispatchMode.java | 8
src/main/java/com/zy/core/move/StationMoveSessionRegistry.java | 243 ++++++++
src/main/java/com/zy/core/move/StationMoveCoordinator.java | 357 ++++++++++++
src/main/java/com/zy/core/thread/impl/station/StationSegmentExecutionPlan.java | 15
src/main/java/com/zy/core/thread/impl/station/StationSegmentPlanner.java | 96 +++
src/main/java/com/zy/core/enums/RedisKeyType.java | 1
src/main/java/com/zy/core/thread/impl/v5/StationV5SegmentExecutor.java | 298 ----------
14 files changed, 1,417 insertions(+), 310 deletions(-)
diff --git a/src/main/java/com/zy/core/enums/RedisKeyType.java b/src/main/java/com/zy/core/enums/RedisKeyType.java
index 3a60047..fef34ca 100644
--- a/src/main/java/com/zy/core/enums/RedisKeyType.java
+++ b/src/main/java/com/zy/core/enums/RedisKeyType.java
@@ -47,6 +47,7 @@
STATION_RUN_BLOCK_TASK_LOOP_STATE_("station_run_block_task_loop_state_"),
CHECK_STATION_IDLE_RECOVER_LIMIT_("check_station_idle_recover_limit_"),
STATION_COMMAND_DISPATCH_DEDUP_("station_command_dispatch_dedup_"),
+ STATION_MOVE_SESSION_("station_move_session_"),
CHECK_SHALLOW_LOC_STATUS_LIMIT("check_shallow_loc_status_limit_"),
GENERATE_ENABLE_IN_STATION_DATA_LIMIT("generate_enable_in_station_data_limit_"),
GENERATE_STATION_BACK_LIMIT("generate_station_back_limit_"),
diff --git a/src/main/java/com/zy/core/model/command/StationCommand.java b/src/main/java/com/zy/core/model/command/StationCommand.java
index 262abc8..32ef7aa 100644
--- a/src/main/java/com/zy/core/model/command/StationCommand.java
+++ b/src/main/java/com/zy/core/model/command/StationCommand.java
@@ -40,4 +40,7 @@
private Integer traceVersion;
+ // 鍚屼竴 task 鐨� MOVE 璺敱鐗堟湰锛岀敤浜庢帶鍒堕噸涓嬪彂鍜屾棫璺緞澶辨晥
+ private Integer routeVersion;
+
}
diff --git a/src/main/java/com/zy/core/move/StationMoveCoordinator.java b/src/main/java/com/zy/core/move/StationMoveCoordinator.java
new file mode 100644
index 0000000..1adc417
--- /dev/null
+++ b/src/main/java/com/zy/core/move/StationMoveCoordinator.java
@@ -0,0 +1,357 @@
+package com.zy.core.move;
+
+import com.alibaba.fastjson.JSON;
+import com.alibaba.fastjson.serializer.SerializerFeature;
+import com.zy.common.utils.RedisUtil;
+import com.zy.core.enums.RedisKeyType;
+import com.zy.core.model.command.StationCommand;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Component;
+
+import java.nio.charset.StandardCharsets;
+import java.security.MessageDigest;
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.LinkedHashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.Objects;
+
+@Component
+public class StationMoveCoordinator {
+
+ private static final int SESSION_EXPIRE_SECONDS = 60 * 60 * 24;
+
+ @Autowired
+ private StationMoveSessionRegistry sessionRegistry;
+ @Autowired
+ private RedisUtil redisUtil;
+
+ public StationMoveSession loadSession(Integer taskNo) {
+ if (sessionRegistry == null) {
+ return null;
+ }
+ return sessionRegistry.load(taskNo);
+ }
+
+ public boolean isActiveRoute(Integer taskNo, Integer routeVersion) {
+ return sessionRegistry != null && sessionRegistry.isActiveRoute(taskNo, routeVersion);
+ }
+
+ public void markSegmentIssued(Integer taskNo, Integer routeVersion) {
+ if (sessionRegistry != null) {
+ sessionRegistry.markSegmentIssued(taskNo, routeVersion);
+ }
+ }
+
+ public void updateCurrentStation(Integer taskNo, Integer routeVersion, Integer currentStationId) {
+ if (sessionRegistry != null) {
+ sessionRegistry.updateCurrentStation(taskNo, routeVersion, currentStationId);
+ }
+ }
+
+ public void markCancelled(Integer taskNo, Integer routeVersion, Integer currentStationId, String cancelReason) {
+ updateTerminal(taskNo, routeVersion, currentStationId, StationMoveSession.STATUS_CANCELLED, cancelReason);
+ }
+
+ public void markBlocked(Integer taskNo, Integer routeVersion, Integer currentStationId) {
+ updateTerminal(taskNo, routeVersion, currentStationId, StationMoveSession.STATUS_BLOCKED, null);
+ }
+
+ public void markTimeout(Integer taskNo, Integer routeVersion, Integer currentStationId) {
+ updateTerminal(taskNo, routeVersion, currentStationId, StationMoveSession.STATUS_TIMEOUT, null);
+ }
+
+ public void markFinished(Integer taskNo, Integer routeVersion, Integer currentStationId) {
+ updateTerminal(taskNo, routeVersion, currentStationId, StationMoveSession.STATUS_FINISHED, null);
+ }
+
+ public void markCancelPending(Integer taskNo, String cancelReason) {
+ StationMoveSession session = loadSession(taskNo);
+ if (session == null || !session.isActive()) {
+ return;
+ }
+ session.setStatus(StationMoveSession.STATUS_CANCEL_PENDING);
+ session.setCancelReason(cancelReason);
+ saveSession(session);
+ }
+
+ public boolean shouldSuppressDispatch(Integer taskNo, Integer currentStationId, StationCommand candidateCommand) {
+ if (taskNo == null || taskNo <= 0 || currentStationId == null || candidateCommand == null) {
+ return false;
+ }
+
+ StationMoveSession session = loadSession(taskNo);
+ if (session == null || !session.isActive()) {
+ return false;
+ }
+
+ String candidateSignature = buildPathSignature(candidateCommand);
+ if (!isBlank(candidateSignature) && Objects.equals(candidateSignature, session.getPathSignature())) {
+ return true;
+ }
+
+ if (Objects.equals(currentStationId, session.getNextDecisionStationId())) {
+ return false;
+ }
+
+ return session.containsStation(currentStationId);
+ }
+
+ public StationMoveSession recordDispatch(Integer taskNo,
+ Integer dispatchStationId,
+ String triggerName,
+ StationCommand command,
+ boolean circleRoute) {
+ if (taskNo == null || taskNo <= 0 || command == null) {
+ return null;
+ }
+
+ StationMoveSession current = loadSession(taskNo);
+ long now = System.currentTimeMillis();
+ String pathSignature = buildPathSignature(command);
+ List<Integer> fullPathStationIds = resolveFullPathStationIds(command);
+ boolean reuseCurrent = current != null
+ && current.isActive()
+ && Objects.equals(current.getDispatchStationId(), dispatchStationId)
+ && Objects.equals(current.getNextDecisionStationId(), command.getTargetStaNo())
+ && Objects.equals(current.getPathSignature(), pathSignature);
+
+ StationMoveSession session = reuseCurrent ? current : new StationMoveSession();
+ if (!reuseCurrent) {
+ session.setRouteVersion(current == null || current.getRouteVersion() == null
+ ? 1
+ : current.getRouteVersion() + 1);
+ session.setCreatedAt(now);
+ } else if (session.getRouteVersion() == null) {
+ session.setRouteVersion(1);
+ }
+
+ session.setTaskNo(taskNo);
+ session.setThreadImpl(resolveThreadImpl(triggerName));
+ session.setCurrentStationId(dispatchStationId);
+ session.setBusinessTargetStationId(command.getTargetStaNo());
+ session.setCurrentRouteTargetStationId(command.getTargetStaNo());
+ session.setTriggerType(resolveTriggerType(triggerName, circleRoute));
+ session.setDispatchMode(resolveDispatchMode(triggerName, circleRoute));
+ session.setStatus(StationMoveSession.STATUS_RUNNING);
+ session.setDispatchStationId(dispatchStationId);
+ session.setNextDecisionStationId(command.getTargetStaNo());
+ session.setFullPathStationIds(fullPathStationIds);
+ session.setPathSignature(pathSignature);
+ session.setCancelReason(null);
+ session.setUpdatedAt(now);
+ session.setLastIssuedAt(now);
+
+ command.setRouteVersion(session.getRouteVersion());
+ saveSession(session);
+
+ if (circleRoute) {
+ saveLegacyCircleCommand(taskNo, command);
+ } else {
+ clearLegacyCircleCommand(taskNo);
+ }
+
+ return session;
+ }
+
+ public StationMoveSession cancelSession(Integer taskNo) {
+ StationMoveSession session = loadSession(taskNo);
+ if (session == null) {
+ clearLegacyCircleCommand(taskNo);
+ return null;
+ }
+ session.setStatus(StationMoveSession.STATUS_CANCELLED);
+ session.setCancelReason("reroute_cancelled");
+ saveSession(session);
+ clearLegacyCircleCommand(taskNo);
+ return session;
+ }
+
+ public StationMoveSession finishSession(Integer taskNo) {
+ StationMoveSession session = loadSession(taskNo);
+ if (session == null) {
+ clearLegacyCircleCommand(taskNo);
+ return null;
+ }
+ session.setStatus(StationMoveSession.STATUS_FINISHED);
+ saveSession(session);
+ clearLegacyCircleCommand(taskNo);
+ return session;
+ }
+
+ public String buildPathSignature(StationCommand command) {
+ if (command == null) {
+ return "";
+ }
+ Map<String, Object> signature = new LinkedHashMap<>();
+ signature.put("commandType", command.getCommandType() == null ? null : command.getCommandType().name());
+ signature.put("stationId", command.getStationId());
+ signature.put("targetStaNo", command.getTargetStaNo());
+ signature.put("navigatePath", copyIntegerList(command.getNavigatePath()));
+ signature.put("liftTransferPath", copyIntegerList(command.getLiftTransferPath()));
+ signature.put("originalNavigatePath", copyIntegerList(command.getOriginalNavigatePath()));
+ return JSON.toJSONString(signature, SerializerFeature.DisableCircularReferenceDetect);
+ }
+
+ public String buildPathSignatureHash(StationCommand command) {
+ String signature = buildPathSignature(command);
+ if (isBlank(signature)) {
+ return "";
+ }
+ return digest(signature);
+ }
+
+ private void updateTerminal(Integer taskNo,
+ Integer routeVersion,
+ Integer currentStationId,
+ String status,
+ String cancelReason) {
+ if (sessionRegistry == null) {
+ return;
+ }
+ StationMoveSession session = sessionRegistry.load(taskNo);
+ if (session == null || !Objects.equals(session.getRouteVersion(), routeVersion)) {
+ return;
+ }
+ session.setCurrentStationId(currentStationId);
+ session.setStatus(status);
+ session.setCancelReason(cancelReason);
+ saveSession(session);
+ if (StationMoveSession.STATUS_CANCELLED.equals(status)
+ || StationMoveSession.STATUS_BLOCKED.equals(status)
+ || StationMoveSession.STATUS_TIMEOUT.equals(status)
+ || StationMoveSession.STATUS_FINISHED.equals(status)) {
+ clearLegacyCircleCommand(taskNo);
+ }
+ }
+
+ private void saveSession(StationMoveSession session) {
+ if (sessionRegistry != null) {
+ sessionRegistry.save(session);
+ }
+ }
+
+ private void saveLegacyCircleCommand(Integer taskNo, StationCommand command) {
+ if (redisUtil == null || taskNo == null || taskNo <= 0 || command == null) {
+ return;
+ }
+ redisUtil.set(RedisKeyType.WATCH_CIRCLE_STATION_.key + taskNo,
+ JSON.toJSONString(command, SerializerFeature.DisableCircularReferenceDetect),
+ SESSION_EXPIRE_SECONDS);
+ }
+
+ private void clearLegacyCircleCommand(Integer taskNo) {
+ if (redisUtil == null || taskNo == null || taskNo <= 0) {
+ return;
+ }
+ redisUtil.del(RedisKeyType.WATCH_CIRCLE_STATION_.key + taskNo);
+ }
+
+ private List<Integer> resolveFullPathStationIds(StationCommand command) {
+ if (command == null) {
+ return new ArrayList<>();
+ }
+ List<Integer> source = command.getOriginalNavigatePath();
+ if (source == null || source.isEmpty()) {
+ source = command.getNavigatePath();
+ }
+ if (source != null && !source.isEmpty()) {
+ return new ArrayList<>(source);
+ }
+ List<Integer> path = new ArrayList<>();
+ if (command.getStationId() != null) {
+ path.add(command.getStationId());
+ }
+ if (command.getTargetStaNo() != null && !Objects.equals(command.getTargetStaNo(), command.getStationId())) {
+ path.add(command.getTargetStaNo());
+ }
+ return path;
+ }
+
+ private List<Integer> copyIntegerList(List<Integer> source) {
+ if (source == null || source.isEmpty()) {
+ return Collections.emptyList();
+ }
+ return new ArrayList<>(source);
+ }
+
+ private String resolveThreadImpl(String triggerName) {
+ if (triggerName == null) {
+ return null;
+ }
+ if (triggerName.contains("crn")) {
+ return "crn";
+ }
+ if (triggerName.contains("watchCircle")) {
+ return "station";
+ }
+ if (triggerName.contains("checkStationRunBlock")) {
+ return "station";
+ }
+ if (triggerName.contains("checkStationIdleRecover")) {
+ return "station";
+ }
+ if (triggerName.contains("checkStationOutOrder")) {
+ return "station";
+ }
+ return triggerName;
+ }
+
+ private StationMoveTriggerType resolveTriggerType(String triggerName, boolean circleRoute) {
+ if (circleRoute) {
+ return StationMoveTriggerType.WATCH_CIRCLE;
+ }
+ if (triggerName == null) {
+ return StationMoveTriggerType.INITIAL_OUTBOUND;
+ }
+ if (triggerName.contains("checkStationOutOrder")) {
+ return StationMoveTriggerType.OUT_ORDER;
+ }
+ if (triggerName.contains("watchCircle")) {
+ return StationMoveTriggerType.WATCH_CIRCLE;
+ }
+ if (triggerName.contains("checkStationRunBlock")) {
+ return StationMoveTriggerType.RUN_BLOCK;
+ }
+ if (triggerName.contains("checkStationIdleRecover")) {
+ return StationMoveTriggerType.IDLE_RECOVER;
+ }
+ return StationMoveTriggerType.INITIAL_OUTBOUND;
+ }
+
+ private StationMoveDispatchMode resolveDispatchMode(String triggerName, boolean circleRoute) {
+ if (circleRoute) {
+ return StationMoveDispatchMode.CIRCLE;
+ }
+ if (triggerName != null && triggerName.contains("checkStationRunBlock")) {
+ return StationMoveDispatchMode.RUN_BLOCK_REROUTE;
+ }
+ if (triggerName != null && triggerName.contains("checkStationIdleRecover")) {
+ return StationMoveDispatchMode.IDLE_RECOVER_REROUTE;
+ }
+ return StationMoveDispatchMode.DIRECT;
+ }
+
+ private boolean isBlank(String value) {
+ return value == null || value.trim().isEmpty();
+ }
+
+ private String digest(String value) {
+ try {
+ MessageDigest digest = MessageDigest.getInstance("SHA-256");
+ byte[] bytes = digest.digest(value.getBytes(StandardCharsets.UTF_8));
+ StringBuilder builder = new StringBuilder(bytes.length * 2);
+ for (byte b : bytes) {
+ String hex = Integer.toHexString(b & 0xff);
+ if (hex.length() == 1) {
+ builder.append('0');
+ }
+ builder.append(hex);
+ }
+ return builder.toString();
+ } catch (Exception ignore) {
+ return Integer.toHexString(value.hashCode());
+ }
+ }
+}
diff --git a/src/main/java/com/zy/core/move/StationMoveDispatchMode.java b/src/main/java/com/zy/core/move/StationMoveDispatchMode.java
new file mode 100644
index 0000000..6fce095
--- /dev/null
+++ b/src/main/java/com/zy/core/move/StationMoveDispatchMode.java
@@ -0,0 +1,8 @@
+package com.zy.core.move;
+
+public enum StationMoveDispatchMode {
+ DIRECT,
+ CIRCLE,
+ RUN_BLOCK_REROUTE,
+ IDLE_RECOVER_REROUTE
+}
diff --git a/src/main/java/com/zy/core/move/StationMoveSession.java b/src/main/java/com/zy/core/move/StationMoveSession.java
new file mode 100644
index 0000000..fbd953a
--- /dev/null
+++ b/src/main/java/com/zy/core/move/StationMoveSession.java
@@ -0,0 +1,64 @@
+package com.zy.core.move;
+
+import lombok.Data;
+
+import java.util.ArrayList;
+import java.util.List;
+
+@Data
+public class StationMoveSession {
+
+ public static final String STATUS_WAITING = "WAITING";
+ public static final String STATUS_RUNNING = "RUNNING";
+ public static final String STATUS_CANCEL_PENDING = "CANCEL_PENDING";
+ public static final String STATUS_CANCELLED = "CANCELLED";
+ public static final String STATUS_BLOCKED = "BLOCKED";
+ public static final String STATUS_TIMEOUT = "TIMEOUT";
+ public static final String STATUS_FINISHED = "FINISHED";
+
+ private Integer taskNo;
+
+ private Integer routeVersion;
+
+ private String threadImpl;
+
+ private Integer currentStationId;
+
+ private StationMoveTriggerType triggerType;
+
+ private StationMoveDispatchMode dispatchMode;
+
+ private String status;
+
+ private Integer dispatchStationId;
+
+ private Integer businessTargetStationId;
+
+ private Integer currentRouteTargetStationId;
+
+ private Integer nextDecisionStationId;
+
+ private List<Integer> fullPathStationIds = new ArrayList<>();
+
+ private String pathSignature;
+
+ private String cancelReason;
+
+ private Long createdAt;
+
+ private Long updatedAt;
+
+ private Long lastIssuedAt;
+
+ public boolean isActive() {
+ return STATUS_WAITING.equals(status)
+ || STATUS_RUNNING.equals(status)
+ || STATUS_CANCEL_PENDING.equals(status);
+ }
+
+ public boolean containsStation(Integer stationId) {
+ return stationId != null
+ && fullPathStationIds != null
+ && fullPathStationIds.contains(stationId);
+ }
+}
diff --git a/src/main/java/com/zy/core/move/StationMoveSessionRegistry.java b/src/main/java/com/zy/core/move/StationMoveSessionRegistry.java
new file mode 100644
index 0000000..82bdad5
--- /dev/null
+++ b/src/main/java/com/zy/core/move/StationMoveSessionRegistry.java
@@ -0,0 +1,243 @@
+package com.zy.core.move;
+
+import com.alibaba.fastjson.JSON;
+import com.alibaba.fastjson.serializer.SerializerFeature;
+import com.zy.common.utils.RedisUtil;
+import com.zy.core.enums.RedisKeyType;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Component;
+
+import java.util.ArrayList;
+import java.util.List;
+import java.util.Objects;
+
+@Component
+public class StationMoveSessionRegistry {
+
+ private static final int SESSION_EXPIRE_SECONDS = 60 * 60 * 24;
+
+ @Autowired
+ private RedisUtil redisUtil;
+
+ public synchronized StationMoveSession load(Integer taskNo) {
+ if (taskNo == null || taskNo <= 0 || redisUtil == null) {
+ return null;
+ }
+ Object sessionObj = redisUtil.get(buildKey(taskNo));
+ if (sessionObj == null) {
+ return null;
+ }
+ try {
+ return JSON.parseObject(String.valueOf(sessionObj), StationMoveSession.class);
+ } catch (Exception ignore) {
+ return null;
+ }
+ }
+
+ public synchronized void save(StationMoveSession session) {
+ if (session == null || session.getTaskNo() == null || session.getTaskNo() <= 0 || redisUtil == null) {
+ return;
+ }
+ long now = System.currentTimeMillis();
+ if (session.getCreatedAt() == null || session.getCreatedAt() <= 0L) {
+ session.setCreatedAt(now);
+ }
+ session.setUpdatedAt(now);
+ redisUtil.set(buildKey(session.getTaskNo()),
+ JSON.toJSONString(session, SerializerFeature.DisableCircularReferenceDetect),
+ SESSION_EXPIRE_SECONDS);
+ }
+
+ public synchronized void delete(Integer taskNo) {
+ if (taskNo == null || taskNo <= 0 || redisUtil == null) {
+ return;
+ }
+ redisUtil.del(buildKey(taskNo));
+ }
+
+ public synchronized boolean isActiveRoute(Integer taskNo, Integer routeVersion) {
+ StationMoveSession session = load(taskNo);
+ return session != null
+ && session.isActive()
+ && routeVersion != null
+ && Objects.equals(routeVersion, session.getRouteVersion());
+ }
+
+ public synchronized boolean shouldSkipOutOrderDecision(Integer taskNo, Integer currentStationId) {
+ StationMoveSession session = load(taskNo);
+ if (session == null || !session.isActive() || currentStationId == null) {
+ return false;
+ }
+ List<Integer> fullPathStationIds = session.getFullPathStationIds();
+ if (fullPathStationIds == null || !fullPathStationIds.contains(currentStationId)) {
+ return false;
+ }
+ if (StationMoveDispatchMode.CIRCLE == session.getDispatchMode()) {
+ return true;
+ }
+ return !Objects.equals(currentStationId, session.getCurrentRouteTargetStationId());
+ }
+
+ public synchronized boolean isCircleDecisionStation(Integer taskNo, Integer currentStationId) {
+ StationMoveSession session = load(taskNo);
+ return session != null
+ && session.isActive()
+ && StationMoveDispatchMode.CIRCLE == session.getDispatchMode()
+ && currentStationId != null
+ && Objects.equals(currentStationId, session.getNextDecisionStationId());
+ }
+
+ public synchronized boolean isCircleTransitStation(Integer taskNo, Integer currentStationId) {
+ StationMoveSession session = load(taskNo);
+ if (session == null
+ || !session.isActive()
+ || StationMoveDispatchMode.CIRCLE != session.getDispatchMode()
+ || currentStationId == null
+ || Objects.equals(currentStationId, session.getNextDecisionStationId())) {
+ return false;
+ }
+ List<Integer> fullPathStationIds = session.getFullPathStationIds();
+ return fullPathStationIds != null && fullPathStationIds.contains(currentStationId);
+ }
+
+ public synchronized StationMoveSession registerPlan(Integer taskNo,
+ String threadImpl,
+ Integer currentStationId,
+ Integer businessTargetStationId,
+ StationMoveTriggerType triggerType,
+ StationMoveDispatchMode dispatchMode,
+ Integer currentRouteTargetStationId,
+ Integer nextDecisionStationId,
+ List<Integer> fullPathStationIds,
+ boolean cancelActive,
+ String cancelReason) {
+ if (taskNo == null || taskNo <= 0) {
+ return null;
+ }
+ StationMoveSession current = load(taskNo);
+ StationMoveSession next = new StationMoveSession();
+ next.setTaskNo(taskNo);
+ next.setThreadImpl(threadImpl);
+ next.setRouteVersion(current == null || current.getRouteVersion() == null ? 1 : current.getRouteVersion() + 1);
+ next.setCurrentStationId(currentStationId);
+ next.setBusinessTargetStationId(businessTargetStationId);
+ next.setTriggerType(triggerType);
+ next.setDispatchMode(dispatchMode);
+ next.setCurrentRouteTargetStationId(currentRouteTargetStationId);
+ next.setNextDecisionStationId(nextDecisionStationId);
+ next.setStatus(StationMoveSession.STATUS_WAITING);
+ next.setCancelReason(cancelReason);
+ next.setFullPathStationIds(copyIntegerList(fullPathStationIds));
+ next.setPathSignature(buildPathSignature(fullPathStationIds));
+ save(next);
+ return next;
+ }
+
+ public synchronized boolean isSameActivePath(Integer taskNo, List<Integer> fullPathStationIds) {
+ StationMoveSession session = load(taskNo);
+ if (session == null || !session.isActive()) {
+ return false;
+ }
+ return Objects.equals(session.getPathSignature(), buildPathSignature(fullPathStationIds));
+ }
+
+ public synchronized void markSegmentIssued(Integer taskNo, Integer routeVersion) {
+ StationMoveSession session = load(taskNo);
+ if (session == null || !Objects.equals(session.getRouteVersion(), routeVersion)) {
+ return;
+ }
+ session.setStatus(StationMoveSession.STATUS_RUNNING);
+ session.setLastIssuedAt(System.currentTimeMillis());
+ save(session);
+ }
+
+ public synchronized void updateCurrentStation(Integer taskNo, Integer routeVersion, Integer currentStationId) {
+ StationMoveSession session = load(taskNo);
+ if (session == null || !Objects.equals(session.getRouteVersion(), routeVersion)) {
+ return;
+ }
+ session.setCurrentStationId(currentStationId);
+ if (!StationMoveSession.STATUS_CANCEL_PENDING.equals(session.getStatus())) {
+ session.setStatus(StationMoveSession.STATUS_RUNNING);
+ }
+ save(session);
+ }
+
+ public synchronized void markCancelPending(Integer taskNo, String cancelReason) {
+ StationMoveSession session = load(taskNo);
+ if (session == null || !session.isActive()) {
+ return;
+ }
+ session.setStatus(StationMoveSession.STATUS_CANCEL_PENDING);
+ session.setCancelReason(cancelReason);
+ save(session);
+ }
+
+ public synchronized void markCancelled(Integer taskNo, Integer routeVersion, Integer currentStationId, String cancelReason) {
+ updateTerminal(taskNo, routeVersion, currentStationId, StationMoveSession.STATUS_CANCELLED, cancelReason);
+ }
+
+ public synchronized void markBlocked(Integer taskNo, Integer routeVersion, Integer currentStationId) {
+ updateTerminal(taskNo, routeVersion, currentStationId, StationMoveSession.STATUS_BLOCKED, null);
+ }
+
+ public synchronized void markTimeout(Integer taskNo, Integer routeVersion, Integer currentStationId) {
+ updateTerminal(taskNo, routeVersion, currentStationId, StationMoveSession.STATUS_TIMEOUT, null);
+ }
+
+ public synchronized void markFinished(Integer taskNo, Integer routeVersion, Integer currentStationId) {
+ updateTerminal(taskNo, routeVersion, currentStationId, StationMoveSession.STATUS_FINISHED, null);
+ }
+
+ public synchronized StationMoveSession buildCircleSessionCommandSource(Integer taskNo) {
+ StationMoveSession session = load(taskNo);
+ if (session == null || !session.isActive() || StationMoveDispatchMode.CIRCLE != session.getDispatchMode()) {
+ return null;
+ }
+ return session;
+ }
+
+ private void updateTerminal(Integer taskNo,
+ Integer routeVersion,
+ Integer currentStationId,
+ String status,
+ String cancelReason) {
+ StationMoveSession session = load(taskNo);
+ if (session == null || !Objects.equals(session.getRouteVersion(), routeVersion)) {
+ return;
+ }
+ session.setCurrentStationId(currentStationId);
+ session.setStatus(status);
+ session.setCancelReason(cancelReason);
+ save(session);
+ }
+
+ private String buildKey(Integer taskNo) {
+ return RedisKeyType.STATION_MOVE_SESSION_.key + taskNo;
+ }
+
+ private String buildPathSignature(List<Integer> fullPathStationIds) {
+ if (fullPathStationIds == null || fullPathStationIds.isEmpty()) {
+ return "";
+ }
+ StringBuilder builder = new StringBuilder();
+ for (Integer stationId : fullPathStationIds) {
+ if (stationId == null) {
+ continue;
+ }
+ if (builder.length() > 0) {
+ builder.append("->");
+ }
+ builder.append(stationId);
+ }
+ return builder.toString();
+ }
+
+ private List<Integer> copyIntegerList(List<Integer> source) {
+ List<Integer> result = new ArrayList<>();
+ if (source != null) {
+ result.addAll(source);
+ }
+ return result;
+ }
+}
diff --git a/src/main/java/com/zy/core/move/StationMoveTriggerType.java b/src/main/java/com/zy/core/move/StationMoveTriggerType.java
new file mode 100644
index 0000000..afcd2a3
--- /dev/null
+++ b/src/main/java/com/zy/core/move/StationMoveTriggerType.java
@@ -0,0 +1,9 @@
+package com.zy.core.move;
+
+public enum StationMoveTriggerType {
+ INITIAL_OUTBOUND,
+ OUT_ORDER,
+ WATCH_CIRCLE,
+ RUN_BLOCK,
+ IDLE_RECOVER
+}
diff --git a/src/main/java/com/zy/core/thread/impl/ZyStationV4Thread.java b/src/main/java/com/zy/core/thread/impl/ZyStationV4Thread.java
index 1a8a4b7..d8ec936 100644
--- a/src/main/java/com/zy/core/thread/impl/ZyStationV4Thread.java
+++ b/src/main/java/com/zy/core/thread/impl/ZyStationV4Thread.java
@@ -30,6 +30,7 @@
import com.zy.core.network.DeviceConnectPool;
import com.zy.core.network.ZyStationConnectDriver;
import com.zy.core.network.entity.ZyStationStatusEntity;
+import com.zy.core.thread.impl.v5.StationMoveSegmentExecutor;
import com.zy.core.utils.DeviceLogRedisKeyBuilder;
import com.zy.system.entity.Config;
import com.zy.system.service.ConfigService;
@@ -51,6 +52,7 @@
private DeviceConfig deviceConfig;
private RedisUtil redisUtil;
private ZyStationConnectDriver zyStationConnectDriver;
+ private StationMoveSegmentExecutor segmentExecutor;
private int deviceLogCollectTime = 200;
private boolean initStatus = false;
private long deviceDataLogTime = System.currentTimeMillis();
@@ -59,6 +61,7 @@
public ZyStationV4Thread(DeviceConfig deviceConfig, RedisUtil redisUtil) {
this.deviceConfig = deviceConfig;
this.redisUtil = redisUtil;
+ this.segmentExecutor = new StationMoveSegmentExecutor(deviceConfig, redisUtil, this::sendCommand);
}
@Override
@@ -92,7 +95,7 @@
}
if (step == 2) {
StationCommand cmd = (StationCommand) task.getData();
- executor.submit(() -> executeMoveWithSeg(cmd));
+ executor.submit(() -> segmentExecutor.execute(cmd));
}
Thread.sleep(100);
} catch (Exception e) {
diff --git a/src/main/java/com/zy/core/thread/impl/station/StationSegmentExecutionPlan.java b/src/main/java/com/zy/core/thread/impl/station/StationSegmentExecutionPlan.java
new file mode 100644
index 0000000..59f3f12
--- /dev/null
+++ b/src/main/java/com/zy/core/thread/impl/station/StationSegmentExecutionPlan.java
@@ -0,0 +1,15 @@
+package com.zy.core.thread.impl.station;
+
+import com.zy.core.model.command.StationCommand;
+import lombok.Data;
+
+import java.util.ArrayList;
+import java.util.List;
+
+@Data
+public class StationSegmentExecutionPlan {
+
+ private List<Integer> fullPathStationIds = new ArrayList<>();
+
+ private List<StationCommand> segmentCommands = new ArrayList<>();
+}
diff --git a/src/main/java/com/zy/core/thread/impl/station/StationSegmentExecutor.java b/src/main/java/com/zy/core/thread/impl/station/StationSegmentExecutor.java
new file mode 100644
index 0000000..3da25e3
--- /dev/null
+++ b/src/main/java/com/zy/core/thread/impl/station/StationSegmentExecutor.java
@@ -0,0 +1,456 @@
+package com.zy.core.thread.impl.station;
+
+import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
+import com.core.common.Cools;
+import com.core.common.SpringUtils;
+import com.zy.asrs.domain.vo.StationTaskTraceSegmentVo;
+import com.zy.asrs.entity.DeviceConfig;
+import com.zy.common.utils.RedisUtil;
+import com.zy.core.cache.SlaveConnection;
+import com.zy.core.enums.RedisKeyType;
+import com.zy.core.enums.SlaveType;
+import com.zy.core.enums.StationCommandType;
+import com.zy.core.model.CommandResponse;
+import com.zy.core.model.command.StationCommand;
+import com.zy.core.model.protocol.StationProtocol;
+import com.zy.core.move.StationMoveCoordinator;
+import com.zy.core.trace.StationTaskTraceRegistry;
+import com.zy.system.entity.Config;
+import com.zy.system.service.ConfigService;
+
+import java.util.ArrayList;
+import java.util.LinkedHashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.function.Function;
+
+public class StationSegmentExecutor {
+
+ private static final String CFG_STATION_COMMAND_SEGMENT_ADVANCE_RATIO = "stationCommandSegmentAdvanceRatio";
+ private static final double DEFAULT_STATION_COMMAND_SEGMENT_ADVANCE_RATIO = 0.3d;
+ private static final long CURRENT_STATION_TIMEOUT_MS = 1000L * 60L;
+
+ private final DeviceConfig deviceConfig;
+ private final RedisUtil redisUtil;
+ private final Function<StationCommand, CommandResponse> commandSender;
+ private final StationSegmentPlanner segmentPlanner = new StationSegmentPlanner();
+
+ public StationSegmentExecutor(DeviceConfig deviceConfig,
+ RedisUtil redisUtil,
+ Function<StationCommand, CommandResponse> commandSender) {
+ this.deviceConfig = deviceConfig;
+ this.redisUtil = redisUtil;
+ this.commandSender = commandSender;
+ }
+
+ public void execute(StationCommand original) {
+ if (original == null) {
+ return;
+ }
+ if (original.getCommandType() != StationCommandType.MOVE) {
+ commandSender.apply(original);
+ return;
+ }
+
+ StationSegmentExecutionPlan localPlan = segmentPlanner.buildPlan(original);
+ if (localPlan.getSegmentCommands().isEmpty()) {
+ return;
+ }
+
+ StationTaskTraceRegistry traceRegistry = SpringUtils.getBean(StationTaskTraceRegistry.class);
+ StationTaskTraceRegistry.TraceRegistration traceRegistration = traceRegistry == null
+ ? new StationTaskTraceRegistry.TraceRegistration()
+ : traceRegistry.registerPlan(original.getTaskNo(), deviceConfig.getThreadImpl(),
+ original.getStationId(), original.getStationId(), original.getTargetStaNo(),
+ localPlan.getFullPathStationIds(), buildTraceSegments(localPlan.getSegmentCommands()));
+ int traceVersion = traceRegistration.getTraceVersion() == null ? 1 : traceRegistration.getTraceVersion();
+ int pathOffset = traceRegistration.getPathOffset() == null ? 0 : traceRegistration.getPathOffset();
+ bindCommands(localPlan.getSegmentCommands(), traceVersion, pathOffset, original.getRouteVersion());
+ List<Integer> effectiveFullPath = traceRegistration.getFullPathStationIds() == null
+ || traceRegistration.getFullPathStationIds().isEmpty()
+ ? copyIntegerList(localPlan.getFullPathStationIds())
+ : copyIntegerList(traceRegistration.getFullPathStationIds());
+
+ StationCommand firstCommand = localPlan.getSegmentCommands().get(0);
+ if (!sendSegmentWithRetry(firstCommand, traceRegistry, traceVersion, null)) {
+ return;
+ }
+ if (traceRegistry != null) {
+ traceRegistry.markSegmentIssued(original.getTaskNo(), traceVersion, firstCommand,
+ "FIRST_SEGMENT_SENT", "杈撻�佷换鍔¢娈典笅鍙戞垚鍔�", buildSegmentDetails(firstCommand));
+ }
+
+ long lastSeenAt = System.currentTimeMillis();
+ int segCursor = 0;
+ Integer lastCurrentStationId = null;
+ boolean firstRun = true;
+ while (true) {
+ try {
+ if (!isRouteActive(original.getTaskNo(), original.getRouteVersion())) {
+ if (traceRegistry != null) {
+ traceRegistry.markCancelled(original.getTaskNo(), traceVersion, lastCurrentStationId,
+ buildDetails("reason", "route_version_replaced", "routeVersion", original.getRouteVersion()));
+ }
+ markCancelled(original.getTaskNo(), original.getRouteVersion(), lastCurrentStationId, "route_version_replaced");
+ break;
+ }
+
+ Object cancel = redisUtil.get(RedisKeyType.DEVICE_STATION_MOVE_RESET.key + original.getTaskNo());
+ if (cancel != null) {
+ if (traceRegistry != null) {
+ traceRegistry.markCancelled(original.getTaskNo(), traceVersion, lastCurrentStationId,
+ buildDetails("reason", "redis_cancel_signal", "routeVersion", original.getRouteVersion()));
+ }
+ markCancelled(original.getTaskNo(), original.getRouteVersion(), lastCurrentStationId, "redis_cancel_signal");
+ break;
+ }
+
+ StationProtocol currentStation = findCurrentStationByTask(original.getTaskNo());
+ if (currentStation == null) {
+ if (System.currentTimeMillis() - lastSeenAt > CURRENT_STATION_TIMEOUT_MS) {
+ if (traceRegistry != null) {
+ traceRegistry.markTimeout(original.getTaskNo(), traceVersion, lastCurrentStationId,
+ buildDetails("timeoutMs", CURRENT_STATION_TIMEOUT_MS, "routeVersion", original.getRouteVersion()));
+ }
+ markTimeout(original.getTaskNo(), original.getRouteVersion(), lastCurrentStationId);
+ break;
+ }
+ Thread.sleep(500L);
+ continue;
+ }
+
+ lastSeenAt = System.currentTimeMillis();
+ Integer previousCurrentStationId = lastCurrentStationId;
+ updateCurrentStation(original.getTaskNo(), original.getRouteVersion(), currentStation.getStationId());
+ if (traceRegistry != null) {
+ traceRegistry.updateProgress(original.getTaskNo(), traceVersion, currentStation.getStationId(),
+ equalsInteger(previousCurrentStationId, currentStation.getStationId()) ? null : "CURRENT_STATION_CHANGE",
+ "杈撻�佷换鍔″綋鍓嶄綅缃凡鏇存柊",
+ buildDetails("stationId", currentStation.getStationId(), "routeVersion", original.getRouteVersion()));
+ }
+ lastCurrentStationId = currentStation.getStationId();
+ if (!firstRun && currentStation.isRunBlock()) {
+ if (traceRegistry != null) {
+ traceRegistry.markBlocked(original.getTaskNo(), traceVersion, currentStation.getStationId(),
+ buildDetails("blockedStationId", currentStation.getStationId(), "routeVersion", original.getRouteVersion()));
+ }
+ markBlocked(original.getTaskNo(), original.getRouteVersion(), currentStation.getStationId());
+ break;
+ }
+
+ int currentIndex = effectiveFullPath.indexOf(currentStation.getStationId());
+ if (currentIndex < 0) {
+ Thread.sleep(500L);
+ firstRun = false;
+ continue;
+ }
+
+ int remaining = effectiveFullPath.size() - currentIndex - 1;
+ if (remaining <= 0) {
+ if (traceRegistry != null) {
+ traceRegistry.markFinished(original.getTaskNo(), traceVersion, currentStation.getStationId(),
+ buildDetails("targetStationId", original.getTargetStaNo(), "routeVersion", original.getRouteVersion()));
+ }
+ markFinished(original.getTaskNo(), original.getRouteVersion(), currentStation.getStationId());
+ break;
+ }
+
+ StationCommand currentSegmentCommand = localPlan.getSegmentCommands().get(segCursor);
+ int currentSegEndIndex = safeIndex(currentSegmentCommand.getSegmentEndIndex());
+ int currentSegStartIndex = segCursor == 0
+ ? 0
+ : safeIndex(localPlan.getSegmentCommands().get(segCursor - 1).getSegmentEndIndex());
+ int segLen = currentSegEndIndex - currentSegStartIndex + 1;
+ int remainingSegment = Math.max(0, currentSegEndIndex - currentIndex);
+ int thresholdSegment = (int) Math.ceil(segLen * loadSegmentAdvanceRatio());
+ thresholdSegment = Math.max(1, thresholdSegment);
+ if (remainingSegment <= thresholdSegment
+ && segCursor < localPlan.getSegmentCommands().size() - 1) {
+ StationCommand nextCommand = localPlan.getSegmentCommands().get(segCursor + 1);
+ if (sendSegmentWithRetry(nextCommand, traceRegistry, traceVersion, currentStation.getStationId())) {
+ segCursor++;
+ if (traceRegistry != null) {
+ traceRegistry.markSegmentIssued(original.getTaskNo(), traceVersion, nextCommand,
+ "NEXT_SEGMENT_SENT", "杈撻�佷换鍔′笅涓�娈靛凡鎻愬墠涓嬪彂",
+ buildSegmentDetails(nextCommand));
+ }
+ } else {
+ break;
+ }
+ }
+ Thread.sleep(500L);
+ firstRun = false;
+ } catch (Exception ignore) {
+ break;
+ }
+ }
+ }
+
+ private boolean sendSegmentWithRetry(StationCommand command,
+ StationTaskTraceRegistry traceRegistry,
+ Integer traceVersion,
+ Integer currentStationId) {
+ while (true) {
+ if (!isRouteActive(command == null ? null : command.getTaskNo(), command == null ? null : command.getRouteVersion())) {
+ if (traceRegistry != null && command != null) {
+ traceRegistry.markCancelled(command.getTaskNo(), traceVersion, currentStationId,
+ buildDetails("reason", "route_version_replaced", "routeVersion", command.getRouteVersion()));
+ }
+ markCancelled(command == null ? null : command.getTaskNo(),
+ command == null ? null : command.getRouteVersion(),
+ currentStationId,
+ "route_version_replaced");
+ return false;
+ }
+ if (isTaskMoveReset(command == null ? null : command.getTaskNo())) {
+ if (traceRegistry != null && command != null) {
+ traceRegistry.markCancelled(command.getTaskNo(), traceVersion, currentStationId,
+ buildDetails("reason", "redis_cancel_signal", "routeVersion", command.getRouteVersion()));
+ }
+ markCancelled(command == null ? null : command.getTaskNo(),
+ command == null ? null : command.getRouteVersion(),
+ currentStationId,
+ "redis_cancel_signal");
+ return false;
+ }
+ CommandResponse commandResponse = commandSender.apply(command);
+ if (commandResponse == null) {
+ sleepQuietly(200L);
+ continue;
+ }
+ if (commandResponse.getResult()) {
+ markSegmentIssued(command == null ? null : command.getTaskNo(), command == null ? null : command.getRouteVersion());
+ return true;
+ }
+ sleepQuietly(200L);
+ }
+ }
+
+ private double loadSegmentAdvanceRatio() {
+ try {
+ ConfigService configService = SpringUtils.getBean(ConfigService.class);
+ if (configService == null) {
+ return DEFAULT_STATION_COMMAND_SEGMENT_ADVANCE_RATIO;
+ }
+ Config config = configService.getOne(new QueryWrapper<Config>()
+ .eq("code", CFG_STATION_COMMAND_SEGMENT_ADVANCE_RATIO));
+ if (config == null || Cools.isEmpty(config.getValue())) {
+ return DEFAULT_STATION_COMMAND_SEGMENT_ADVANCE_RATIO;
+ }
+ return normalizeSegmentAdvanceRatio(config.getValue());
+ } catch (Exception ignore) {
+ return DEFAULT_STATION_COMMAND_SEGMENT_ADVANCE_RATIO;
+ }
+ }
+
+ private double normalizeSegmentAdvanceRatio(String valueText) {
+ if (valueText == null) {
+ return DEFAULT_STATION_COMMAND_SEGMENT_ADVANCE_RATIO;
+ }
+ String text = valueText.trim();
+ if (text.isEmpty()) {
+ return DEFAULT_STATION_COMMAND_SEGMENT_ADVANCE_RATIO;
+ }
+ if (text.endsWith("%")) {
+ text = text.substring(0, text.length() - 1).trim();
+ }
+ try {
+ double ratio = Double.parseDouble(text);
+ if (ratio > 1d && ratio <= 100d) {
+ ratio = ratio / 100d;
+ }
+ if (ratio < 0d) {
+ return 0d;
+ }
+ if (ratio > 1d) {
+ return 1d;
+ }
+ return ratio;
+ } catch (Exception ignore) {
+ return DEFAULT_STATION_COMMAND_SEGMENT_ADVANCE_RATIO;
+ }
+ }
+
+ private boolean isTaskMoveReset(Integer taskNo) {
+ if (taskNo == null || redisUtil == null) {
+ return false;
+ }
+ Object cancel = redisUtil.get(RedisKeyType.DEVICE_STATION_MOVE_RESET.key + taskNo);
+ return cancel != null;
+ }
+
+ private StationProtocol findCurrentStationByTask(Integer taskNo) {
+ try {
+ com.zy.asrs.service.DeviceConfigService deviceConfigService = SpringUtils.getBean(com.zy.asrs.service.DeviceConfigService.class);
+ if (deviceConfigService == null) {
+ return null;
+ }
+ List<DeviceConfig> devpList = deviceConfigService.list(new QueryWrapper<DeviceConfig>()
+ .eq("device_type", String.valueOf(SlaveType.Devp)));
+ for (DeviceConfig dc : devpList) {
+ com.zy.core.thread.StationThread thread = (com.zy.core.thread.StationThread) SlaveConnection.get(SlaveType.Devp, dc.getDeviceNo());
+ if (thread == null) {
+ continue;
+ }
+ Map<Integer, StationProtocol> statusMap = thread.getStatusMap();
+ if (statusMap == null || statusMap.isEmpty()) {
+ continue;
+ }
+ for (StationProtocol protocol : statusMap.values()) {
+ if (protocol.getTaskNo() != null && protocol.getTaskNo().equals(taskNo) && protocol.isLoading()) {
+ return protocol;
+ }
+ }
+ }
+ } catch (Exception ignore) {
+ return null;
+ }
+ return null;
+ }
+
+ private List<StationTaskTraceSegmentVo> buildTraceSegments(List<StationCommand> segmentCommands) {
+ List<StationTaskTraceSegmentVo> result = new ArrayList<>();
+ if (segmentCommands == null) {
+ return result;
+ }
+ for (StationCommand command : segmentCommands) {
+ if (command == null) {
+ continue;
+ }
+ StationTaskTraceSegmentVo item = new StationTaskTraceSegmentVo();
+ item.setSegmentNo(command.getSegmentNo());
+ item.setSegmentCount(command.getSegmentCount());
+ item.setStationId(command.getStationId());
+ item.setTargetStationId(command.getTargetStaNo());
+ item.setSegmentStartIndex(command.getSegmentStartIndex());
+ item.setSegmentEndIndex(command.getSegmentEndIndex());
+ item.setSegmentPath(copyIntegerList(command.getNavigatePath()));
+ item.setIssued(Boolean.FALSE);
+ result.add(item);
+ }
+ return result;
+ }
+
+ private void bindCommands(List<StationCommand> segmentCommands, int traceVersion, int pathOffset, Integer routeVersion) {
+ if (segmentCommands == null) {
+ return;
+ }
+ for (StationCommand command : segmentCommands) {
+ if (command == null) {
+ continue;
+ }
+ command.setTraceVersion(traceVersion);
+ command.setRouteVersion(routeVersion);
+ if (command.getSegmentStartIndex() != null) {
+ command.setSegmentStartIndex(command.getSegmentStartIndex() + pathOffset);
+ }
+ if (command.getSegmentEndIndex() != null) {
+ command.setSegmentEndIndex(command.getSegmentEndIndex() + pathOffset);
+ }
+ }
+ }
+
+ private Map<String, Object> buildSegmentDetails(StationCommand command) {
+ Map<String, Object> details = new LinkedHashMap<>();
+ if (command != null) {
+ details.put("segmentNo", command.getSegmentNo());
+ details.put("segmentCount", command.getSegmentCount());
+ details.put("segmentPath", copyIntegerList(command.getNavigatePath()));
+ details.put("segmentStartIndex", command.getSegmentStartIndex());
+ details.put("segmentEndIndex", command.getSegmentEndIndex());
+ details.put("traceVersion", command.getTraceVersion());
+ details.put("routeVersion", command.getRouteVersion());
+ }
+ return details;
+ }
+
+ private Map<String, Object> buildDetails(Object... keyValues) {
+ Map<String, Object> details = new LinkedHashMap<>();
+ if (keyValues == null) {
+ return details;
+ }
+ for (int i = 0; i + 1 < keyValues.length; i += 2) {
+ Object key = keyValues[i];
+ if (key != null) {
+ details.put(String.valueOf(key), keyValues[i + 1]);
+ }
+ }
+ return details;
+ }
+
+ private List<Integer> copyIntegerList(List<Integer> source) {
+ List<Integer> result = new ArrayList<>();
+ if (source != null) {
+ result.addAll(source);
+ }
+ return result;
+ }
+
+ private int safeIndex(Integer value) {
+ return value == null ? -1 : value;
+ }
+
+ private boolean equalsInteger(Integer a, Integer b) {
+ return a != null && a.equals(b);
+ }
+
+ private void sleepQuietly(long millis) {
+ try {
+ Thread.sleep(millis);
+ } catch (Exception ignore) {
+ }
+ }
+
+ private boolean isRouteActive(Integer taskNo, Integer routeVersion) {
+ // Legacy direct-enqueue commands (for example FakeProcess/stationInExecute)
+ // do not register a move session and therefore have no routeVersion.
+ // They should keep the historical behavior and execute normally.
+ if (taskNo == null || routeVersion == null) {
+ return true;
+ }
+ StationMoveCoordinator moveCoordinator = SpringUtils.getBean(StationMoveCoordinator.class);
+ return moveCoordinator == null || moveCoordinator.isActiveRoute(taskNo, routeVersion);
+ }
+
+ private void markSegmentIssued(Integer taskNo, Integer routeVersion) {
+ StationMoveCoordinator moveCoordinator = SpringUtils.getBean(StationMoveCoordinator.class);
+ if (moveCoordinator != null) {
+ moveCoordinator.markSegmentIssued(taskNo, routeVersion);
+ }
+ }
+
+ private void updateCurrentStation(Integer taskNo, Integer routeVersion, Integer currentStationId) {
+ StationMoveCoordinator moveCoordinator = SpringUtils.getBean(StationMoveCoordinator.class);
+ if (moveCoordinator != null) {
+ moveCoordinator.updateCurrentStation(taskNo, routeVersion, currentStationId);
+ }
+ }
+
+ private void markCancelled(Integer taskNo, Integer routeVersion, Integer currentStationId, String cancelReason) {
+ StationMoveCoordinator moveCoordinator = SpringUtils.getBean(StationMoveCoordinator.class);
+ if (moveCoordinator != null) {
+ moveCoordinator.markCancelled(taskNo, routeVersion, currentStationId, cancelReason);
+ }
+ }
+
+ private void markBlocked(Integer taskNo, Integer routeVersion, Integer currentStationId) {
+ StationMoveCoordinator moveCoordinator = SpringUtils.getBean(StationMoveCoordinator.class);
+ if (moveCoordinator != null) {
+ moveCoordinator.markBlocked(taskNo, routeVersion, currentStationId);
+ }
+ }
+
+ private void markTimeout(Integer taskNo, Integer routeVersion, Integer currentStationId) {
+ StationMoveCoordinator moveCoordinator = SpringUtils.getBean(StationMoveCoordinator.class);
+ if (moveCoordinator != null) {
+ moveCoordinator.markTimeout(taskNo, routeVersion, currentStationId);
+ }
+ }
+
+ private void markFinished(Integer taskNo, Integer routeVersion, Integer currentStationId) {
+ StationMoveCoordinator moveCoordinator = SpringUtils.getBean(StationMoveCoordinator.class);
+ if (moveCoordinator != null) {
+ moveCoordinator.markFinished(taskNo, routeVersion, currentStationId);
+ }
+ }
+}
diff --git a/src/main/java/com/zy/core/thread/impl/station/StationSegmentPlanner.java b/src/main/java/com/zy/core/thread/impl/station/StationSegmentPlanner.java
new file mode 100644
index 0000000..3c9328c
--- /dev/null
+++ b/src/main/java/com/zy/core/thread/impl/station/StationSegmentPlanner.java
@@ -0,0 +1,96 @@
+package com.zy.core.thread.impl.station;
+
+import com.alibaba.fastjson.JSON;
+import com.alibaba.fastjson.serializer.SerializerFeature;
+import com.zy.core.model.command.StationCommand;
+
+import java.util.ArrayList;
+import java.util.List;
+import java.util.Objects;
+
+public class StationSegmentPlanner {
+
+ public StationSegmentExecutionPlan buildPlan(StationCommand original) {
+ StationSegmentExecutionPlan plan = new StationSegmentExecutionPlan();
+ if (original == null) {
+ return plan;
+ }
+
+ List<Integer> path = copyIntegerList(original.getNavigatePath());
+ List<Integer> liftTransferPath = copyIntegerList(original.getLiftTransferPath());
+ Integer startStationId = original.getStationId();
+ Integer targetStationId = original.getTargetStaNo();
+
+ if ((path == null || path.isEmpty()) && Objects.equals(startStationId, targetStationId) && startStationId != null) {
+ path = new ArrayList<>();
+ path.add(startStationId);
+ }
+ if (path == null || path.isEmpty()) {
+ return plan;
+ }
+
+ plan.setFullPathStationIds(copyIntegerList(path));
+
+ int total = path.size();
+ List<Integer> segmentEndIndices = new ArrayList<>();
+ if (liftTransferPath != null) {
+ for (Integer liftTransferStationId : liftTransferPath) {
+ int endIndex = path.indexOf(liftTransferStationId);
+ if (endIndex <= 0) {
+ continue;
+ }
+ if (segmentEndIndices.isEmpty() || endIndex > segmentEndIndices.get(segmentEndIndices.size() - 1)) {
+ segmentEndIndices.add(endIndex);
+ }
+ }
+ }
+ if (segmentEndIndices.isEmpty() || !Objects.equals(segmentEndIndices.get(segmentEndIndices.size() - 1), total - 1)) {
+ segmentEndIndices.add(total - 1);
+ }
+
+ List<StationCommand> segmentCommands = new ArrayList<>();
+ int buildStartIdx = 0;
+ for (Integer endIdx : segmentEndIndices) {
+ if (endIdx == null || endIdx < buildStartIdx) {
+ continue;
+ }
+ List<Integer> segmentPath = new ArrayList<>(path.subList(buildStartIdx, endIdx + 1));
+ if (segmentPath.isEmpty()) {
+ buildStartIdx = endIdx + 1;
+ continue;
+ }
+
+ StationCommand segmentCommand = new StationCommand();
+ segmentCommand.setTaskNo(original.getTaskNo());
+ segmentCommand.setCommandType(original.getCommandType());
+ segmentCommand.setPalletSize(original.getPalletSize());
+ segmentCommand.setBarcode(original.getBarcode());
+ segmentCommand.setOriginalNavigatePath(copyIntegerList(path));
+ segmentCommand.setNavigatePath(segmentPath);
+ segmentCommand.setStationId(segmentPath.get(0));
+ segmentCommand.setTargetStaNo(segmentPath.get(segmentPath.size() - 1));
+ segmentCommand.setSegmentStartIndex(buildStartIdx);
+ segmentCommand.setSegmentEndIndex(endIdx);
+ segmentCommand.setRouteVersion(original.getRouteVersion());
+ segmentCommands.add(segmentCommand);
+
+ buildStartIdx = endIdx;
+ }
+
+ int segmentCount = segmentCommands.size();
+ for (int i = 0; i < segmentCommands.size(); i++) {
+ StationCommand segmentCommand = segmentCommands.get(i);
+ segmentCommand.setSegmentNo(i + 1);
+ segmentCommand.setSegmentCount(segmentCount);
+ }
+ plan.setSegmentCommands(segmentCommands);
+ return plan;
+ }
+
+ private List<Integer> copyIntegerList(List<Integer> source) {
+ if (source == null) {
+ return new ArrayList<>();
+ }
+ return JSON.parseArray(JSON.toJSONString(source, SerializerFeature.DisableCircularReferenceDetect), Integer.class);
+ }
+}
diff --git a/src/main/java/com/zy/core/thread/impl/v5/StationMoveSegmentExecutor.java b/src/main/java/com/zy/core/thread/impl/v5/StationMoveSegmentExecutor.java
new file mode 100644
index 0000000..d0a2618
--- /dev/null
+++ b/src/main/java/com/zy/core/thread/impl/v5/StationMoveSegmentExecutor.java
@@ -0,0 +1,18 @@
+package com.zy.core.thread.impl.v5;
+
+import com.zy.asrs.entity.DeviceConfig;
+import com.zy.common.utils.RedisUtil;
+import com.zy.core.model.CommandResponse;
+import com.zy.core.model.command.StationCommand;
+import com.zy.core.thread.impl.station.StationSegmentExecutor;
+
+import java.util.function.Function;
+
+public class StationMoveSegmentExecutor extends StationSegmentExecutor {
+
+ public StationMoveSegmentExecutor(DeviceConfig deviceConfig,
+ RedisUtil redisUtil,
+ Function<StationCommand, CommandResponse> commandSender) {
+ super(deviceConfig, redisUtil, commandSender);
+ }
+}
diff --git a/src/main/java/com/zy/core/thread/impl/v5/StationV5SegmentExecutor.java b/src/main/java/com/zy/core/thread/impl/v5/StationV5SegmentExecutor.java
index fecd517..c8a86fa 100644
--- a/src/main/java/com/zy/core/thread/impl/v5/StationV5SegmentExecutor.java
+++ b/src/main/java/com/zy/core/thread/impl/v5/StationV5SegmentExecutor.java
@@ -1,309 +1,17 @@
package com.zy.core.thread.impl.v5;
-import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
-import com.core.common.Cools;
-import com.core.common.SpringUtils;
-import com.zy.asrs.domain.vo.StationTaskTraceSegmentVo;
+
import com.zy.asrs.entity.DeviceConfig;
import com.zy.common.utils.RedisUtil;
-import com.zy.core.cache.SlaveConnection;
-import com.zy.core.enums.RedisKeyType;
-import com.zy.core.enums.SlaveType;
-import com.zy.core.enums.StationCommandType;
import com.zy.core.model.CommandResponse;
import com.zy.core.model.command.StationCommand;
-import com.zy.core.model.protocol.StationProtocol;
-import com.zy.core.trace.StationTaskTraceRegistry;
-import com.zy.system.entity.Config;
-import com.zy.system.service.ConfigService;
-import java.util.ArrayList;
-import java.util.LinkedHashMap;
-import java.util.List;
-import java.util.Map;
import java.util.function.Function;
-public class StationV5SegmentExecutor {
-
- private static final int NEXT_SEGMENT_ADVANCE_NODE_COUNT = 1;
- private static final long CURRENT_STATION_TIMEOUT_MS = 1000L * 60L;
-
- private final DeviceConfig deviceConfig;
- private final RedisUtil redisUtil;
- private final Function<StationCommand, CommandResponse> commandSender;
- private final StationV5SegmentPlanner segmentPlanner = new StationV5SegmentPlanner();
+public class StationV5SegmentExecutor extends StationMoveSegmentExecutor {
public StationV5SegmentExecutor(DeviceConfig deviceConfig,
RedisUtil redisUtil,
Function<StationCommand, CommandResponse> commandSender) {
- this.deviceConfig = deviceConfig;
- this.redisUtil = redisUtil;
- this.commandSender = commandSender;
- }
-
- public void execute(StationCommand original) {
- if (original == null) {
- return;
- }
- if (original.getCommandType() != StationCommandType.MOVE) {
- commandSender.apply(original);
- return;
- }
-
- StationV5SegmentExecutionPlan localPlan = segmentPlanner.buildPlan(original);
- if (localPlan.getSegmentCommands().isEmpty()) {
- return;
- }
-
- StationTaskTraceRegistry traceRegistry = SpringUtils.getBean(StationTaskTraceRegistry.class);
- StationTaskTraceRegistry.TraceRegistration traceRegistration = traceRegistry == null
- ? new StationTaskTraceRegistry.TraceRegistration()
- : traceRegistry.registerPlan(original.getTaskNo(), deviceConfig.getThreadImpl(),
- original.getStationId(), original.getStationId(), original.getTargetStaNo(),
- localPlan.getFullPathStationIds(), buildTraceSegments(localPlan.getSegmentCommands()));
- int traceVersion = traceRegistration.getTraceVersion() == null ? 1 : traceRegistration.getTraceVersion();
- int pathOffset = traceRegistration.getPathOffset() == null ? 0 : traceRegistration.getPathOffset();
- bindCommands(localPlan.getSegmentCommands(), traceVersion, pathOffset);
- List<Integer> effectiveFullPath = traceRegistration.getFullPathStationIds() == null
- || traceRegistration.getFullPathStationIds().isEmpty()
- ? copyIntegerList(localPlan.getFullPathStationIds())
- : copyIntegerList(traceRegistration.getFullPathStationIds());
-
- StationCommand firstCommand = localPlan.getSegmentCommands().get(0);
- if (!sendSegmentWithRetry(firstCommand)) {
- return;
- }
- if (traceRegistry != null) {
- traceRegistry.markSegmentIssued(original.getTaskNo(), traceVersion, firstCommand,
- "FIRST_SEGMENT_SENT", "杈撻�佷换鍔¢娈典笅鍙戞垚鍔�", buildSegmentDetails(firstCommand));
- }
-
- long lastSeenAt = System.currentTimeMillis();
- int segCursor = 0;
- Integer lastCurrentStationId = null;
- boolean firstRun = true;
- while (true) {
- try {
- Object cancel = redisUtil.get(RedisKeyType.DEVICE_STATION_MOVE_RESET.key + original.getTaskNo());
- if (cancel != null) {
- if (traceRegistry != null) {
- traceRegistry.markCancelled(original.getTaskNo(), traceVersion, lastCurrentStationId,
- buildDetails("reason", "redis_cancel_signal"));
- }
- break;
- }
-
- StationProtocol currentStation = findCurrentStationByTask(original.getTaskNo());
- if (currentStation == null) {
- if (System.currentTimeMillis() - lastSeenAt > CURRENT_STATION_TIMEOUT_MS) {
- if (traceRegistry != null) {
- traceRegistry.markTimeout(original.getTaskNo(), traceVersion, lastCurrentStationId,
- buildDetails("timeoutMs", CURRENT_STATION_TIMEOUT_MS));
- }
- break;
- }
- Thread.sleep(500L);
- continue;
- }
-
- lastSeenAt = System.currentTimeMillis();
- Integer previousCurrentStationId = lastCurrentStationId;
- if (traceRegistry != null) {
- traceRegistry.updateProgress(original.getTaskNo(), traceVersion, currentStation.getStationId(),
- equalsInteger(previousCurrentStationId, currentStation.getStationId()) ? null : "CURRENT_STATION_CHANGE",
- "杈撻�佷换鍔″綋鍓嶄綅缃凡鏇存柊",
- buildDetails("stationId", currentStation.getStationId()));
- }
- lastCurrentStationId = currentStation.getStationId();
- if (!firstRun && currentStation.isRunBlock()) {
- if (traceRegistry != null) {
- traceRegistry.markBlocked(original.getTaskNo(), traceVersion, currentStation.getStationId(),
- buildDetails("blockedStationId", currentStation.getStationId()));
- }
- break;
- }
-
- int currentIndex = effectiveFullPath.indexOf(currentStation.getStationId());
- if (currentIndex < 0) {
- Thread.sleep(500L);
- firstRun = false;
- continue;
- }
-
- int remaining = effectiveFullPath.size() - currentIndex - 1;
- if (remaining <= 0) {
- if (traceRegistry != null) {
- traceRegistry.markFinished(original.getTaskNo(), traceVersion, currentStation.getStationId(),
- buildDetails("targetStationId", original.getTargetStaNo()));
- }
- break;
- }
-
- StationCommand currentSegmentCommand = localPlan.getSegmentCommands().get(segCursor);
- int currentSegEndIndex = safeIndex(currentSegmentCommand.getSegmentEndIndex());
- int remainingSegment = Math.max(0, currentSegEndIndex - currentIndex);
- if (remainingSegment <= NEXT_SEGMENT_ADVANCE_NODE_COUNT
- && segCursor < localPlan.getSegmentCommands().size() - 1) {
- StationCommand nextCommand = localPlan.getSegmentCommands().get(segCursor + 1);
- if (sendSegmentWithRetry(nextCommand)) {
- segCursor++;
- if (traceRegistry != null) {
- traceRegistry.markSegmentIssued(original.getTaskNo(), traceVersion, nextCommand,
- "NEXT_SEGMENT_SENT", "杈撻�佷换鍔′笅涓�娈靛凡鎻愬墠涓嬪彂",
- buildSegmentDetails(nextCommand));
- }
- }
- }
- Thread.sleep(500L);
- firstRun = false;
- } catch (Exception ignore) {
- break;
- }
- }
- }
-
- private boolean sendSegmentWithRetry(StationCommand command) {
- while (true) {
- if (isTaskMoveReset(command == null ? null : command.getTaskNo())) {
- return false;
- }
- CommandResponse commandResponse = commandSender.apply(command);
- if (commandResponse == null) {
- sleepQuietly(200L);
- continue;
- }
- if (commandResponse.getResult()) {
- return true;
- }
- sleepQuietly(200L);
- }
- }
-
- private boolean isTaskMoveReset(Integer taskNo) {
- if (taskNo == null || redisUtil == null) {
- return false;
- }
- Object cancel = redisUtil.get(RedisKeyType.DEVICE_STATION_MOVE_RESET.key + taskNo);
- return cancel != null;
- }
-
- private StationProtocol findCurrentStationByTask(Integer taskNo) {
- try {
- com.zy.asrs.service.DeviceConfigService deviceConfigService = SpringUtils.getBean(com.zy.asrs.service.DeviceConfigService.class);
- if (deviceConfigService == null) {
- return null;
- }
- List<DeviceConfig> devpList = deviceConfigService.list(new QueryWrapper<DeviceConfig>()
- .eq("device_type", String.valueOf(SlaveType.Devp)));
- for (DeviceConfig dc : devpList) {
- com.zy.core.thread.StationThread thread = (com.zy.core.thread.StationThread) SlaveConnection.get(SlaveType.Devp, dc.getDeviceNo());
- if (thread == null) {
- continue;
- }
- Map<Integer, StationProtocol> statusMap = thread.getStatusMap();
- if (statusMap == null || statusMap.isEmpty()) {
- continue;
- }
- for (StationProtocol protocol : statusMap.values()) {
- if (protocol.getTaskNo() != null && protocol.getTaskNo().equals(taskNo) && protocol.isLoading()) {
- return protocol;
- }
- }
- }
- } catch (Exception ignore) {
- return null;
- }
- return null;
- }
-
- private List<StationTaskTraceSegmentVo> buildTraceSegments(List<StationCommand> segmentCommands) {
- List<StationTaskTraceSegmentVo> result = new ArrayList<>();
- if (segmentCommands == null) {
- return result;
- }
- for (StationCommand command : segmentCommands) {
- if (command == null) {
- continue;
- }
- StationTaskTraceSegmentVo item = new StationTaskTraceSegmentVo();
- item.setSegmentNo(command.getSegmentNo());
- item.setSegmentCount(command.getSegmentCount());
- item.setStationId(command.getStationId());
- item.setTargetStationId(command.getTargetStaNo());
- item.setSegmentStartIndex(command.getSegmentStartIndex());
- item.setSegmentEndIndex(command.getSegmentEndIndex());
- item.setSegmentPath(copyIntegerList(command.getNavigatePath()));
- item.setIssued(Boolean.FALSE);
- result.add(item);
- }
- return result;
- }
-
- private void bindCommands(List<StationCommand> segmentCommands, int traceVersion, int pathOffset) {
- if (segmentCommands == null) {
- return;
- }
- for (StationCommand command : segmentCommands) {
- if (command == null) {
- continue;
- }
- command.setTraceVersion(traceVersion);
- if (command.getSegmentStartIndex() != null) {
- command.setSegmentStartIndex(command.getSegmentStartIndex() + pathOffset);
- }
- if (command.getSegmentEndIndex() != null) {
- command.setSegmentEndIndex(command.getSegmentEndIndex() + pathOffset);
- }
- }
- }
-
- private Map<String, Object> buildSegmentDetails(StationCommand command) {
- Map<String, Object> details = new LinkedHashMap<>();
- if (command != null) {
- details.put("segmentNo", command.getSegmentNo());
- details.put("segmentCount", command.getSegmentCount());
- details.put("segmentPath", copyIntegerList(command.getNavigatePath()));
- details.put("segmentStartIndex", command.getSegmentStartIndex());
- details.put("segmentEndIndex", command.getSegmentEndIndex());
- details.put("traceVersion", command.getTraceVersion());
- }
- return details;
- }
-
- private Map<String, Object> buildDetails(Object... keyValues) {
- Map<String, Object> details = new LinkedHashMap<>();
- if (keyValues == null) {
- return details;
- }
- for (int i = 0; i + 1 < keyValues.length; i += 2) {
- Object key = keyValues[i];
- if (key != null) {
- details.put(String.valueOf(key), keyValues[i + 1]);
- }
- }
- return details;
- }
-
- private List<Integer> copyIntegerList(List<Integer> source) {
- List<Integer> result = new ArrayList<>();
- if (source == null) {
- return result;
- }
- result.addAll(source);
- return result;
- }
-
- private int safeIndex(Integer value) {
- return value == null ? -1 : value;
- }
-
- private boolean equalsInteger(Integer a, Integer b) {
- return a != null && a.equals(b);
- }
-
- private void sleepQuietly(long millis) {
- try {
- Thread.sleep(millis);
- } catch (Exception ignore) {
- }
+ super(deviceConfig, redisUtil, commandSender);
}
}
diff --git a/src/main/java/com/zy/core/utils/StationOperateProcessUtils.java b/src/main/java/com/zy/core/utils/StationOperateProcessUtils.java
index ea10e5d..94b122b 100644
--- a/src/main/java/com/zy/core/utils/StationOperateProcessUtils.java
+++ b/src/main/java/com/zy/core/utils/StationOperateProcessUtils.java
@@ -19,6 +19,8 @@
import com.zy.common.service.CommonService;
import com.zy.common.utils.NavigateUtils;
import com.zy.common.utils.RedisUtil;
+import com.zy.core.move.StationMoveCoordinator;
+import com.zy.core.move.StationMoveSession;
import com.zy.core.News;
import com.zy.core.cache.MessageQueue;
import com.zy.core.cache.SlaveConnection;
@@ -74,6 +76,8 @@
private StationTaskLoopService stationTaskLoopService;
@Autowired
private WrkAnalysisService wrkAnalysisService;
+ @Autowired
+ private StationMoveCoordinator stationMoveCoordinator;
//鎵ц杈撻�佺珯鐐瑰叆搴撲换鍔�
public synchronized void stationInExecute() {
@@ -252,7 +256,16 @@
wrkMast.setModiTime(now);
if (wrkMastService.updateById(wrkMast)) {
wrkAnalysisService.markOutboundStationStart(wrkMast, now);
- offerDevpCommandWithDedup(stationObjModel.getDeviceNo(), command, "crnStationOutExecute");
+ boolean offered = offerDevpCommandWithDedup(stationObjModel.getDeviceNo(), command, "crnStationOutExecute");
+ if (offered && stationMoveCoordinator != null) {
+ stationMoveCoordinator.recordDispatch(
+ wrkMast.getWrkNo(),
+ stationProtocol.getStationId(),
+ "crnStationOutExecute",
+ command,
+ false
+ );
+ }
News.info("杈撻�佺珯鐐瑰嚭搴撳懡浠や笅鍙戞垚鍔燂紝绔欑偣鍙�={}锛屽伐浣滃彿={}锛屽懡浠ゆ暟鎹�={}", stationProtocol.getStationId(), wrkMast.getWrkNo(), JSON.toJSONString(command));
redisUtil.set(RedisKeyType.STATION_OUT_EXECUTE_LIMIT.key + stationProtocol.getStationId(), "lock", 5);
redisUtil.del(RedisKeyType.CRN_OUT_TASK_COMPLETE_STATION_INFO.key + wrkMast.getWrkNo());
@@ -372,6 +385,9 @@
if (wrkMast == null || wrkMast.getWrkNo() == null) {
return;
}
+ if (stationMoveCoordinator != null) {
+ stationMoveCoordinator.finishSession(wrkMast.getWrkNo());
+ }
Date now = new Date();
wrkMast.setWrkSts(WrkStsType.STATION_RUN_COMPLETE.sts);
wrkMast.setIoTime(now);
@@ -419,6 +435,9 @@
}
if (complete) {
+ if (stationMoveCoordinator != null) {
+ stationMoveCoordinator.finishSession(wrkNo);
+ }
wrkMast.setWrkSts(WrkStsType.COMPLETE_OUTBOUND.sts);
wrkMast.setIoTime(new Date());
wrkMastService.updateById(wrkMast);
@@ -476,6 +495,9 @@
stationProtocol.getTaskNo(),
currentTaskBufferCommandCount);
continue;
+ }
+ if (stationMoveCoordinator != null) {
+ stationMoveCoordinator.cancelSession(wrkMast.getWrkNo());
}
//杩愯鍫靛锛岄噸鏂扮敵璇蜂换鍔�
String response = wmsOperateUtils.applyReassignTaskLocNo(wrkMast.getWrkNo(), stationProtocol.getStationId());
@@ -554,7 +576,19 @@
}
if (wrkMastService.updateById(wrkMast)) {
- offerDevpCommandWithDedup(basDevp.getDevpNo(), command, "checkStationRunBlock_direct");
+ boolean offered = offerDevpCommandWithDedup(basDevp.getDevpNo(), command, "checkStationRunBlock_direct");
+ if (!offered) {
+ continue;
+ }
+ if (stationMoveCoordinator != null) {
+ stationMoveCoordinator.recordDispatch(
+ wrkMast.getWrkNo(),
+ stationProtocol.getStationId(),
+ "checkStationRunBlock_direct",
+ command,
+ false
+ );
+ }
}
} else {
News.error("璇锋眰WMS鎺ュ彛澶辫触锛侊紒锛乺esponse锛歿}", response);
@@ -592,9 +626,24 @@
continue;
}
+ if (stationMoveCoordinator != null) {
+ stationMoveCoordinator.cancelSession(wrkMast.getWrkNo());
+ }
resetSegmentMoveCommandsBeforeReroute(wrkMast.getWrkNo());
- offerDevpCommandWithDedup(basDevp.getDevpNo(), command, "checkStationRunBlock_reroute");
+ boolean offered = offerDevpCommandWithDedup(basDevp.getDevpNo(), command, "checkStationRunBlock_reroute");
+ if (!offered) {
+ continue;
+ }
syncOutOrderWatchState(wrkMast, stationProtocol.getStationId(), outOrderStationIds, dispatchDecision, command);
+ if (stationMoveCoordinator != null) {
+ stationMoveCoordinator.recordDispatch(
+ wrkMast.getWrkNo(),
+ stationProtocol.getStationId(),
+ "checkStationRunBlock_reroute",
+ command,
+ dispatchDecision != null && dispatchDecision.isCircle()
+ );
+ }
News.info("杈撻�佺珯鐐瑰牭濉炲悗閲嶆柊璁$畻璺緞鍛戒护涓嬪彂鎴愬姛锛岀珯鐐瑰彿={}锛屽伐浣滃彿={}锛屽懡浠ゆ暟鎹�={}", stationProtocol.getStationId(), wrkMast.getWrkNo(), JSON.toJSONString(command));
}
}
@@ -696,13 +745,6 @@
if (countCurrentTaskBufferCommands(stationProtocol.getTaskBufferItems(), stationProtocol.getTaskNo()) > 0) {
continue;
}
- if (isWatchingCircleTransit(wrkMast.getWrkNo(), stationProtocol.getStationId())) {
- continue;
- }
-
- if (isWatchingCircleArrival(wrkMast.getWrkNo(), stationProtocol.getStationId())) {
- continue;
- }
Double pathLenFactor = resolveOutboundPathLenFactor(wrkMast);
OutOrderDispatchDecision dispatchDecision = resolveOutboundDispatchDecision(
@@ -727,12 +769,28 @@
News.taskInfo(wrkMast.getWrkNo(), "鑾峰彇杈撻�佺嚎鍛戒护澶辫触");
continue;
}
+ if (stationMoveCoordinator != null
+ && stationMoveCoordinator.shouldSuppressDispatch(wrkMast.getWrkNo(), stationProtocol.getStationId(), command)) {
+ continue;
+ }
if (!tryAcquireOutOrderDispatchLock(wrkMast.getWrkNo(), stationProtocol.getStationId())) {
continue;
}
resetSegmentMoveCommandsBeforeReroute(wrkMast.getWrkNo());
+ boolean offered = offerDevpCommandWithDedup(stationObjModel.getDeviceNo(), command, "checkStationOutOrder");
+ if (!offered) {
+ continue;
+ }
syncOutOrderWatchState(wrkMast, stationProtocol.getStationId(), outOrderStationIds, dispatchDecision, command);
- offerDevpCommandWithDedup(stationObjModel.getDeviceNo(), command, "checkStationOutOrder");
+ if (stationMoveCoordinator != null) {
+ stationMoveCoordinator.recordDispatch(
+ wrkMast.getWrkNo(),
+ stationProtocol.getStationId(),
+ "checkStationOutOrder",
+ command,
+ dispatchDecision != null && dispatchDecision.isCircle()
+ );
+ }
News.info(dispatchDecision.isCircle() ? "{}浠诲姟杩涜缁曞湀" : "{}浠诲姟鐩存帴鍘荤洰鏍囩偣", wrkMast.getWrkNo());
}
}
@@ -807,12 +865,28 @@
News.taskInfo(wrkMast.getWrkNo(), "鑾峰彇杈撻�佺嚎鍛戒护澶辫触");
continue;
}
+ if (stationMoveCoordinator != null
+ && stationMoveCoordinator.shouldSuppressDispatch(wrkMast.getWrkNo(), stationProtocol.getStationId(), command)) {
+ continue;
+ }
if (!tryAcquireOutOrderDispatchLock(wrkMast.getWrkNo(), stationProtocol.getStationId())) {
continue;
}
resetSegmentMoveCommandsBeforeReroute(wrkMast.getWrkNo());
+ boolean offered = offerDevpCommandWithDedup(basDevp.getDevpNo(), command, "watchCircleStation");
+ if (!offered) {
+ continue;
+ }
syncOutOrderWatchState(wrkMast, stationProtocol.getStationId(), outOrderList, dispatchDecision, command);
- offerDevpCommandWithDedup(basDevp.getDevpNo(), command, "watchCircleStation");
+ if (stationMoveCoordinator != null) {
+ stationMoveCoordinator.recordDispatch(
+ wrkMast.getWrkNo(),
+ stationProtocol.getStationId(),
+ "watchCircleStation",
+ command,
+ dispatchDecision != null && dispatchDecision.isCircle()
+ );
+ }
}
}
}
@@ -1283,11 +1357,33 @@
}
private boolean isWatchingCircleArrival(Integer wrkNo, Integer stationId) {
+ if (stationMoveCoordinator != null) {
+ StationMoveSession session = stationMoveCoordinator.loadSession(wrkNo);
+ if (session != null && session.isActive() && stationId != null) {
+ if (stationId.equals(session.getNextDecisionStationId())) {
+ return true;
+ }
+ if (session.containsStation(stationId)) {
+ return false;
+ }
+ }
+ }
StationCommand command = getWatchCircleCommand(wrkNo);
return command != null && stationId != null && stationId.equals(command.getTargetStaNo());
}
private boolean isWatchingCircleTransit(Integer wrkNo, Integer stationId) {
+ if (stationMoveCoordinator != null) {
+ StationMoveSession session = stationMoveCoordinator.loadSession(wrkNo);
+ if (session != null && session.isActive() && stationId != null) {
+ if (stationId.equals(session.getNextDecisionStationId())) {
+ return false;
+ }
+ if (session.containsStation(stationId)) {
+ return true;
+ }
+ }
+ }
StationCommand command = getWatchCircleCommand(wrkNo);
if (command == null || stationId == null || Objects.equals(stationId, command.getTargetStaNo())) {
return false;
@@ -1376,6 +1472,9 @@
return;
}
+ if (stationMoveCoordinator != null) {
+ stationMoveCoordinator.cancelSession(stationProtocol.getTaskNo());
+ }
redisUtil.set(RedisKeyType.CHECK_STATION_IDLE_RECOVER_LIMIT_.key + stationProtocol.getTaskNo(), "lock", STATION_IDLE_RECOVER_LIMIT_SECONDS);
resetSegmentMoveCommandsBeforeReroute(stationProtocol.getTaskNo());
int clearedCommandCount = clearIssuedMoveCommandsDuringIdleStay(idleTrack, stationProtocol.getTaskNo(), stationProtocol.getStationId());
@@ -1392,8 +1491,20 @@
return;
}
- offerDevpCommandWithDedup(basDevp.getDevpNo(), command, "checkStationIdleRecover");
+ boolean offered = offerDevpCommandWithDedup(basDevp.getDevpNo(), command, "checkStationIdleRecover");
+ if (!offered) {
+ return;
+ }
syncOutOrderWatchState(wrkMast, stationProtocol.getStationId(), outOrderList, dispatchDecision, command);
+ if (stationMoveCoordinator != null) {
+ stationMoveCoordinator.recordDispatch(
+ wrkMast.getWrkNo(),
+ stationProtocol.getStationId(),
+ "checkStationIdleRecover",
+ command,
+ dispatchDecision != null && dispatchDecision.isCircle()
+ );
+ }
saveStationTaskIdleTrack(new StationTaskIdleTrack(wrkMast.getWrkNo(), stationProtocol.getStationId(), System.currentTimeMillis()));
News.info("杈撻�佺珯鐐逛换鍔″仠鐣檣}绉掓湭杩愯锛屽凡閲嶆柊璁$畻璺緞骞堕噸鍚繍琛岋紝绔欑偣鍙�={}锛岀洰鏍囩珯={}锛屽伐浣滃彿={}锛屾竻鐞嗘棫鍒嗘鍛戒护鏁�={}锛屽懡浠ゆ暟鎹�={}",
STATION_IDLE_RECOVER_SECONDS, stationProtocol.getStationId(), moveStaNo, wrkMast.getWrkNo(), clearedCommandCount, JSON.toJSONString(command));
@@ -1469,8 +1580,23 @@
private String buildStationCommandDispatchDedupKey(Integer deviceNo, StationCommand command) {
return RedisKeyType.STATION_COMMAND_DISPATCH_DEDUP_.key
+ + deviceNo + "_"
+ command.getTaskNo() + "_"
- + command.getStationId();
+ + command.getStationId() + "_"
+ + (stationMoveCoordinator == null ? Integer.toHexString(buildFallbackPathSignature(command).hashCode())
+ : stationMoveCoordinator.buildPathSignatureHash(command));
+ }
+
+ private String buildFallbackPathSignature(StationCommand command) {
+ if (command == null) {
+ return "";
+ }
+ return String.valueOf(command.getCommandType())
+ + "_" + command.getStationId()
+ + "_" + command.getTargetStaNo()
+ + "_" + command.getNavigatePath()
+ + "_" + command.getLiftTransferPath()
+ + "_" + command.getOriginalNavigatePath();
}
private int clearIssuedMoveCommandsDuringIdleStay(StationTaskIdleTrack idleTrack,
--
Gitblit v1.9.1