From aa710969e00e9d7e56a276066a239f74d5c49310 Mon Sep 17 00:00:00 2001
From: Junjie <fallin.jie@qq.com>
Date: 星期二, 31 三月 2026 21:47:07 +0800
Subject: [PATCH] #

---
 src/main/java/com/zy/core/thread/impl/ZyStationV5Thread.java |  978 +++------------------------------------------------------
 1 files changed, 58 insertions(+), 920 deletions(-)

diff --git a/src/main/java/com/zy/core/thread/impl/ZyStationV5Thread.java b/src/main/java/com/zy/core/thread/impl/ZyStationV5Thread.java
index 6f633a9..17bd7f8 100644
--- a/src/main/java/com/zy/core/thread/impl/ZyStationV5Thread.java
+++ b/src/main/java/com/zy/core/thread/impl/ZyStationV5Thread.java
@@ -2,26 +2,14 @@
 
 import com.alibaba.fastjson.JSON;
 import com.alibaba.fastjson.JSONObject;
-import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
-import com.core.common.Cools;
-import com.core.common.DateUtils;
 import com.core.common.SpringUtils;
-import com.zy.asrs.domain.vo.StationCycleCapacityVo;
-import com.zy.asrs.domain.vo.StationCycleLoopVo;
-import com.zy.asrs.entity.BasDevp;
 import com.zy.asrs.entity.BasStationOpt;
 import com.zy.asrs.entity.DeviceConfig;
-import com.zy.asrs.entity.DeviceDataLog;
-import com.zy.asrs.service.BasDevpService;
 import com.zy.asrs.service.BasStationOptService;
-import com.zy.asrs.service.StationCycleCapacityService;
-import com.zy.asrs.utils.Utils;
 import com.zy.common.model.NavigateNode;
 import com.zy.common.utils.NavigateUtils;
 import com.zy.common.utils.RedisUtil;
 import com.zy.core.cache.MessageQueue;
-import com.zy.core.cache.OutputQueue;
-import com.zy.core.enums.RedisKeyType;
 import com.zy.core.enums.SlaveType;
 import com.zy.core.enums.StationCommandType;
 import com.zy.core.model.CommandResponse;
@@ -33,25 +21,19 @@
 import com.zy.core.network.ZyStationConnectDriver;
 import com.zy.core.network.entity.ZyStationStatusEntity;
 import com.zy.core.service.StationTaskLoopService;
+import com.zy.core.thread.impl.v5.StationV5RunBlockReroutePlanner;
 import com.zy.core.thread.impl.v5.StationV5SegmentExecutor;
-import com.zy.core.trace.StationTaskTraceRegistry;
-import com.zy.core.utils.DeviceLogRedisKeyBuilder;
+import com.zy.core.thread.impl.v5.StationV5StatusReader;
+import com.zy.core.thread.support.RecentStationArrivalTracker;
 import lombok.Data;
 import lombok.extern.slf4j.Slf4j;
 
-import java.text.MessageFormat;
 import java.util.ArrayList;
-import java.util.ArrayDeque;
-import java.util.Collections;
 import java.util.Date;
-import java.util.Deque;
 import java.util.HashMap;
-import java.util.HashSet;
-import java.util.LinkedHashSet;
 import java.util.List;
 import java.util.Map;
 import java.util.Objects;
-import java.util.Set;
 import java.util.concurrent.ExecutorService;
 import java.util.concurrent.Executors;
 
@@ -59,40 +41,35 @@
 @Slf4j
 public class ZyStationV5Thread implements Runnable, com.zy.core.thread.StationThread {
 
-    private static final int RUN_BLOCK_REROUTE_STATE_EXPIRE_SECONDS = 60 * 60 * 24;
-    private static final int SHORT_PATH_REPEAT_AVOID_THRESHOLD = 2;
-    private static final int LOOP_REPEAT_TRIGGER_COUNT = 3;
-    private static final int LOCAL_LOOP_NEIGHBOR_HOP = 3;
+    private static final int SEGMENT_EXECUTOR_POOL_SIZE = 64;
 
-    private List<StationProtocol> statusList = new ArrayList<>();
     private DeviceConfig deviceConfig;
     private RedisUtil redisUtil;
     private ZyStationConnectDriver zyStationConnectDriver;
-    private int deviceLogCollectTime = 200;
-    private boolean initStatus = false;
-    private long deviceDataLogTime = System.currentTimeMillis();
-    private ExecutorService executor = Executors.newFixedThreadPool(9999);
+    private final ExecutorService executor = Executors.newFixedThreadPool(SEGMENT_EXECUTOR_POOL_SIZE);
     private StationV5SegmentExecutor segmentExecutor;
+    private final RecentStationArrivalTracker recentArrivalTracker;
+    private final StationV5StatusReader statusReader;
+    private final StationV5RunBlockReroutePlanner runBlockReroutePlanner;
 
     public ZyStationV5Thread(DeviceConfig deviceConfig, RedisUtil redisUtil) {
         this.deviceConfig = deviceConfig;
         this.redisUtil = redisUtil;
+        this.recentArrivalTracker = new RecentStationArrivalTracker(redisUtil);
         this.segmentExecutor = new StationV5SegmentExecutor(deviceConfig, redisUtil, this::sendCommand);
+        this.statusReader = new StationV5StatusReader(deviceConfig, redisUtil, recentArrivalTracker);
+        this.runBlockReroutePlanner = new StationV5RunBlockReroutePlanner(redisUtil);
     }
 
     @Override
     @SuppressWarnings("InfiniteLoopStatement")
     public void run() {
         this.connect();
-        deviceLogCollectTime = Utils.getDeviceLogCollectTime();
 
         Thread readThread = new Thread(() -> {
             while (true) {
                 try {
-                    if (initStatus) {
-                        deviceLogCollectTime = Utils.getDeviceLogCollectTime();
-                    }
-                    readStatus();
+                    statusReader.readStatus(zyStationConnectDriver);
                     Thread.sleep(100);
                 } catch (Exception e) {
                     log.error("StationV5Thread Fail", e);
@@ -104,15 +81,7 @@
         Thread processThread = new Thread(() -> {
             while (true) {
                 try {
-                    int step = 1;
-                    Task task = MessageQueue.poll(SlaveType.Devp, deviceConfig.getDeviceNo());
-                    if (task != null) {
-                        step = task.getStep();
-                    }
-                    if (step == 2) {
-                        StationCommand cmd = (StationCommand) task.getData();
-                        executor.submit(() -> segmentExecutor.execute(cmd));
-                    }
+                    pollAndDispatchQueuedCommand();
                     Thread.sleep(100);
                 } catch (Exception e) {
                     log.error("StationV5Process Fail", e);
@@ -120,83 +89,6 @@
             }
         }, "DevpProcess-" + deviceConfig.getDeviceNo());
         processThread.start();
-    }
-
-    private void readStatus() {
-        if (zyStationConnectDriver == null) {
-            return;
-        }
-
-        if (statusList.isEmpty()) {
-            BasDevpService basDevpService = null;
-            try {
-                basDevpService = SpringUtils.getBean(BasDevpService.class);
-            } catch (Exception ignore) {
-            }
-            if (basDevpService == null) {
-                return;
-            }
-
-            BasDevp basDevp = basDevpService
-                    .getOne(new QueryWrapper<BasDevp>().eq("devp_no", deviceConfig.getDeviceNo()));
-            if (basDevp == null) {
-                return;
-            }
-
-            List<ZyStationStatusEntity> list = JSONObject.parseArray(basDevp.getStationList(), ZyStationStatusEntity.class);
-            for (ZyStationStatusEntity entity : list) {
-                StationProtocol stationProtocol = new StationProtocol();
-                stationProtocol.setStationId(entity.getStationId());
-                statusList.add(stationProtocol);
-            }
-            initStatus = true;
-        }
-
-        List<ZyStationStatusEntity> zyStationStatusEntities = zyStationConnectDriver.getStatus();
-        for (ZyStationStatusEntity statusEntity : zyStationStatusEntities) {
-            for (StationProtocol stationProtocol : statusList) {
-                if (stationProtocol.getStationId().equals(statusEntity.getStationId())) {
-                    stationProtocol.setTaskNo(statusEntity.getTaskNo());
-                    stationProtocol.setTargetStaNo(statusEntity.getTargetStaNo());
-                    stationProtocol.setAutoing(statusEntity.isAutoing());
-                    stationProtocol.setLoading(statusEntity.isLoading());
-                    stationProtocol.setInEnable(statusEntity.isInEnable());
-                    stationProtocol.setOutEnable(statusEntity.isOutEnable());
-                    stationProtocol.setEmptyMk(statusEntity.isEmptyMk());
-                    stationProtocol.setFullPlt(statusEntity.isFullPlt());
-                    stationProtocol.setPalletHeight(statusEntity.getPalletHeight());
-                    stationProtocol.setError(statusEntity.getError());
-                    stationProtocol.setErrorMsg(statusEntity.getErrorMsg());
-                    stationProtocol.setBarcode(statusEntity.getBarcode());
-                    stationProtocol.setRunBlock(statusEntity.isRunBlock());
-                    stationProtocol.setEnableIn(statusEntity.isEnableIn());
-                    stationProtocol.setWeight(statusEntity.getWeight());
-                    stationProtocol.setTaskWriteIdx(statusEntity.getTaskWriteIdx());
-                    stationProtocol.setTaskBufferItems(statusEntity.getTaskBufferItems());
-                }
-
-                if (!Cools.isEmpty(stationProtocol.getSystemWarning())) {
-                    if (stationProtocol.isAutoing() && !stationProtocol.isLoading()) {
-                        stationProtocol.setSystemWarning("");
-                    }
-                }
-            }
-        }
-
-        OutputQueue.DEVP.offer(MessageFormat.format("銆恵0}銆慬id:{1}] <<<<< 瀹炴椂鏁版嵁鏇存柊鎴愬姛",
-                DateUtils.convert(new Date()), deviceConfig.getDeviceNo()));
-
-        if (System.currentTimeMillis() - deviceDataLogTime > deviceLogCollectTime) {
-            DeviceDataLog deviceDataLog = new DeviceDataLog();
-            deviceDataLog.setOriginData(JSON.toJSONString(zyStationStatusEntities));
-            deviceDataLog.setWcsData(JSON.toJSONString(statusList));
-            deviceDataLog.setType(String.valueOf(SlaveType.Devp));
-            deviceDataLog.setDeviceNo(deviceConfig.getDeviceNo());
-            deviceDataLog.setCreateTime(new Date());
-
-            redisUtil.set(DeviceLogRedisKeyBuilder.build(deviceDataLog), deviceDataLog, 60 * 60 * 24);
-            deviceDataLogTime = System.currentTimeMillis();
-        }
     }
 
     @Override
@@ -222,16 +114,36 @@
 
     @Override
     public List<StationProtocol> getStatus() {
-        return statusList;
+        return statusReader.getStatusList();
     }
 
     @Override
     public Map<Integer, StationProtocol> getStatusMap() {
         Map<Integer, StationProtocol> map = new HashMap<>();
-        for (StationProtocol stationProtocol : statusList) {
+        for (StationProtocol stationProtocol : statusReader.getStatusList()) {
             map.put(stationProtocol.getStationId(), stationProtocol);
         }
         return map;
+    }
+
+    private void pollAndDispatchQueuedCommand() {
+        Task task = MessageQueue.poll(SlaveType.Devp, deviceConfig.getDeviceNo());
+        if (task == null || task.getStep() == null || task.getStep() != 2) {
+            return;
+        }
+        submitSegmentCommand((StationCommand) task.getData());
+    }
+
+    private void submitSegmentCommand(StationCommand command) {
+        if (command == null || executor == null || segmentExecutor == null) {
+            return;
+        }
+        executor.submit(() -> segmentExecutor.execute(command));
+    }
+
+    @Override
+    public boolean hasRecentArrival(Integer stationId, Integer taskNo) {
+        return recentArrivalTracker.hasRecentArrival(stationId, taskNo);
     }
 
     @Override
@@ -285,7 +197,6 @@
             return getCommand(StationCommandType.MOVE, taskNo, stationId, targetStationId, palletSize, pathLenFactor);
         }
 
-        RunBlockRerouteState rerouteState = loadRunBlockRerouteState(taskNo, stationId);
         StationTaskLoopService taskLoopService = loadStationTaskLoopService();
         StationTaskLoopService.LoopEvaluation loopEvaluation = taskLoopService == null
                 ? new StationTaskLoopService.LoopEvaluation(taskNo, stationId, StationTaskLoopService.LoopIdentitySnapshot.empty(), 0, 0, false)
@@ -296,61 +207,44 @@
                 loopEvaluation.getLoopIdentity().getScopeType(),
                 loopEvaluation.getLoopIdentity().getLocalStationCount(),
                 loopEvaluation.getLoopIdentity().getSourceLoopStationCount());
-        rerouteState.setTaskNo(taskNo);
-        rerouteState.setBlockStationId(stationId);
-        rerouteState.setLastTargetStationId(targetStationId);
-        rerouteState.setPlanCount((rerouteState.getPlanCount() == null ? 0 : rerouteState.getPlanCount()) + 1);
-        rerouteState.setLastPlanTime(System.currentTimeMillis());
-
         List<List<NavigateNode>> candidatePathList = calcCandidatePathNavigateNodes(taskNo, stationId, targetStationId, pathLenFactor);
-        if (candidatePathList.isEmpty()) {
-            saveRunBlockRerouteState(rerouteState);
+        List<StationCommand> candidateCommandList = new ArrayList<>();
+        for (List<NavigateNode> candidatePath : candidatePathList) {
+            StationCommand rerouteCommand = buildMoveCommand(taskNo, stationId, targetStationId, palletSize, candidatePath);
+            if (rerouteCommand == null || rerouteCommand.getNavigatePath() == null || rerouteCommand.getNavigatePath().isEmpty()) {
+                continue;
+            }
+            candidateCommandList.add(rerouteCommand);
+        }
+
+        StationV5RunBlockReroutePlanner.PlanResult planResult = runBlockReroutePlanner.plan(
+                taskNo,
+                stationId,
+                loopEvaluation,
+                candidateCommandList
+        );
+        if (candidateCommandList.isEmpty()) {
             log.warn("杈撻�佺嚎鍫靛閲嶈鍒掑け璐ワ紝鍊欓�夎矾寰勪负绌猴紝taskNo={}, planCount={}, stationId={}, targetStationId={}",
-                    taskNo, rerouteState.getPlanCount(), stationId, targetStationId);
+                    taskNo, planResult.getPlanCount(), stationId, targetStationId);
             return null;
         }
 
-        StationCommand rerouteCommand = selectAvailableRerouteCommand(
-                rerouteState,
-                loopEvaluation,
-                candidatePathList,
-                taskNo,
-                stationId,
-                targetStationId,
-                palletSize
-        );
-        if (rerouteCommand == null) {
-            log.info("杈撻�佺嚎鍫靛閲嶈鍒掑�欓�夎矾绾垮凡鍏ㄩ儴璇曡繃锛岄噸缃矾绾垮巻鍙插悗閲嶆柊寮�濮嬶紝taskNo={}, planCount={}, stationId={}, targetStationId={}",
-                    taskNo, rerouteState.getPlanCount(), stationId, targetStationId);
-            rerouteState.resetIssuedRoutes();
-            rerouteCommand = selectAvailableRerouteCommand(
-                    rerouteState,
-                    loopEvaluation,
-                    candidatePathList,
-                    taskNo,
-                    stationId,
-                    targetStationId,
-                    palletSize
-            );
-        }
-
+        StationCommand rerouteCommand = planResult.getCommand();
         if (rerouteCommand != null) {
-            saveRunBlockRerouteState(rerouteState);
             if (taskLoopService != null) {
                 taskLoopService.recordLoopIssue(loopEvaluation, "RUN_BLOCK_REROUTE");
             }
             log.info("杈撻�佺嚎鍫靛閲嶈鍒掗�変腑鍊欓�夎矾绾匡紝taskNo={}, planCount={}, stationId={}, targetStationId={}, route={}",
-                    taskNo, rerouteState.getPlanCount(), stationId, targetStationId, JSON.toJSONString(rerouteCommand.getNavigatePath()));
+                    taskNo, planResult.getPlanCount(), stationId, targetStationId, JSON.toJSONString(rerouteCommand.getNavigatePath()));
             return rerouteCommand;
         }
 
-        saveRunBlockRerouteState(rerouteState);
         log.warn("杈撻�佺嚎鍫靛閲嶈鍒掓湭鎵惧埌鍙笅鍙戣矾绾匡紝taskNo={}, planCount={}, stationId={}, targetStationId={}, triedRoutes={}",
                 taskNo,
-                rerouteState.getPlanCount(),
+                planResult.getPlanCount(),
                 stationId,
                 targetStationId,
-                JSON.toJSONString(rerouteState.getIssuedRoutePathList()));
+                JSON.toJSONString(planResult.getIssuedRoutePathList()));
         return null;
     }
 
@@ -386,6 +280,8 @@
                             stationId, item.getSlotIdx(), item.getTaskNo());
                     continue;
                 }else {
+                    item.setTaskNo(0);
+                    item.setTargetStaNo(0);
                     success = true;
                     log.warn("杈撻�佺珯缂撳瓨鍖烘畫鐣欒矾寰勬竻鐞嗘垚鍔熴�俿tationId={}, slotIdx={}, taskNo={}",
                             stationId, item.getSlotIdx(), item.getTaskNo());
@@ -520,769 +416,11 @@
         return stationCommand;
     }
 
-    private StationCommand selectAvailableRerouteCommand(RunBlockRerouteState rerouteState,
-                                                         StationTaskLoopService.LoopEvaluation loopEvaluation,
-                                                         List<List<NavigateNode>> candidatePathList,
-                                                         Integer taskNo,
-                                                         Integer stationId,
-                                                         Integer targetStationId,
-                                                         Integer palletSize) {
-        if (rerouteState == null || candidatePathList == null || candidatePathList.isEmpty()) {
-            return null;
-        }
-
-        Set<String> issuedRouteSignatureSet = rerouteState.getIssuedRouteSignatureSet();
-        List<RerouteCandidateCommand> candidateCommandList = new ArrayList<>();
-        for (List<NavigateNode> candidatePath : candidatePathList) {
-            StationCommand rerouteCommand = buildMoveCommand(taskNo, stationId, targetStationId, palletSize, candidatePath);
-            if (rerouteCommand == null || rerouteCommand.getNavigatePath() == null || rerouteCommand.getNavigatePath().isEmpty()) {
-                continue;
-            }
-            String routeSignature = buildPathSignature(rerouteCommand.getNavigatePath());
-            if (Cools.isEmpty(routeSignature)) {
-                continue;
-            }
-            RerouteCandidateCommand candidateCommand = new RerouteCandidateCommand();
-            candidateCommand.setCommand(rerouteCommand);
-            candidateCommand.setRouteSignature(routeSignature);
-            candidateCommand.setPathLength(rerouteCommand.getNavigatePath().size());
-            candidateCommand.setIssuedCount(rerouteState.getRouteIssueCountMap().getOrDefault(routeSignature, 0));
-            candidateCommand.setLoopFingerprint(loopEvaluation.getLoopIdentity().getLoopFingerprint());
-            candidateCommand.setLoopIssuedCount(loopEvaluation.getExpectedLoopIssueCount());
-            candidateCommand.setLoopTriggered(loopEvaluation.isLargeLoopTriggered());
-            candidateCommand.setCurrentLoopHitCount(countCurrentLoopStationHit(
-                    rerouteCommand.getNavigatePath(),
-                    loopEvaluation.getLoopIdentity().getStationIdSet()
-            ));
-            candidateCommandList.add(candidateCommand);
-        }
-        if (candidateCommandList.isEmpty()) {
-            return null;
-        }
-
-        List<RerouteCandidateCommand> orderedCandidateCommandList = reorderCandidateCommandsForLoopRelease(candidateCommandList);
-        for (RerouteCandidateCommand candidateCommand : orderedCandidateCommandList) {
-            if (candidateCommand == null || candidateCommand.getCommand() == null) {
-                continue;
-            }
-            if (issuedRouteSignatureSet.contains(candidateCommand.getRouteSignature())) {
-                continue;
-            }
-
-            StationCommand rerouteCommand = candidateCommand.getCommand();
-            issuedRouteSignatureSet.add(candidateCommand.getRouteSignature());
-            rerouteState.getIssuedRoutePathList().add(new ArrayList<>(rerouteCommand.getNavigatePath()));
-            rerouteState.setLastSelectedRoute(new ArrayList<>(rerouteCommand.getNavigatePath()));
-            rerouteState.getRouteIssueCountMap().put(
-                    candidateCommand.getRouteSignature(),
-                    rerouteState.getRouteIssueCountMap().getOrDefault(candidateCommand.getRouteSignature(), 0) + 1
-            );
-            return rerouteCommand;
-        }
-        return null;
-    }
-
-    private List<RerouteCandidateCommand> reorderCandidateCommandsForLoopRelease(List<RerouteCandidateCommand> candidateCommandList) {
-        if (candidateCommandList == null || candidateCommandList.isEmpty()) {
-            return new ArrayList<>();
-        }
-
-        int shortestPathLength = Integer.MAX_VALUE;
-        int shortestPathLoopHitCount = Integer.MAX_VALUE;
-        boolean shortestPathOverused = false;
-        boolean currentLoopOverused = false;
-        boolean hasLongerCandidate = false;
-        for (RerouteCandidateCommand candidateCommand : candidateCommandList) {
-            if (candidateCommand == null || candidateCommand.getPathLength() == null || candidateCommand.getPathLength() <= 0) {
-                continue;
-            }
-            shortestPathLength = Math.min(shortestPathLength, candidateCommand.getPathLength());
-        }
-        if (shortestPathLength == Integer.MAX_VALUE) {
-            return candidateCommandList;
-        }
-
-        for (RerouteCandidateCommand candidateCommand : candidateCommandList) {
-            if (candidateCommand == null || candidateCommand.getPathLength() == null || candidateCommand.getPathLength() <= 0) {
-                continue;
-            }
-            if (candidateCommand.getPathLength() == shortestPathLength) {
-                shortestPathLoopHitCount = Math.min(shortestPathLoopHitCount, safeInt(candidateCommand.getCurrentLoopHitCount()));
-            }
-            if (candidateCommand.getPathLength() > shortestPathLength) {
-                hasLongerCandidate = true;
-            }
-            if (candidateCommand.getPathLength() == shortestPathLength
-                    && candidateCommand.getIssuedCount() != null
-                    && candidateCommand.getIssuedCount() >= SHORT_PATH_REPEAT_AVOID_THRESHOLD) {
-                shortestPathOverused = true;
-            }
-            if (!Cools.isEmpty(candidateCommand.getLoopFingerprint())
-                    && Boolean.TRUE.equals(candidateCommand.getLoopTriggered())) {
-                currentLoopOverused = true;
-            }
-        }
-        if (!shortestPathOverused && !currentLoopOverused) {
-            return candidateCommandList;
-        }
-        if (shortestPathLoopHitCount == Integer.MAX_VALUE) {
-            shortestPathLoopHitCount = 0;
-        }
-
-        boolean hasLoopExitCandidate = false;
-        for (RerouteCandidateCommand candidateCommand : candidateCommandList) {
-            if (candidateCommand == null) {
-                continue;
-            }
-            if (safeInt(candidateCommand.getCurrentLoopHitCount()) < shortestPathLoopHitCount) {
-                hasLoopExitCandidate = true;
-                break;
-            }
-        }
-        if (!hasLongerCandidate && !hasLoopExitCandidate) {
-            return candidateCommandList;
-        }
-
-        List<RerouteCandidateCommand> reorderedList = new ArrayList<>();
-        if (currentLoopOverused && hasLoopExitCandidate) {
-            for (RerouteCandidateCommand candidateCommand : candidateCommandList) {
-                if (candidateCommand == null) {
-                    continue;
-                }
-                if (safeInt(candidateCommand.getCurrentLoopHitCount()) < shortestPathLoopHitCount) {
-                    appendCandidateIfAbsent(reorderedList, candidateCommand);
-                }
-            }
-        }
-        for (RerouteCandidateCommand candidateCommand : candidateCommandList) {
-            if (candidateCommand != null
-                    && candidateCommand.getPathLength() != null
-                    && candidateCommand.getPathLength() > shortestPathLength) {
-                appendCandidateIfAbsent(reorderedList, candidateCommand);
-            }
-        }
-        for (RerouteCandidateCommand candidateCommand : candidateCommandList) {
-            if (candidateCommand == null || candidateCommand.getPathLength() == null) {
-                continue;
-            }
-            appendCandidateIfAbsent(reorderedList, candidateCommand);
-        }
-        return reorderedList;
-    }
-
-    private void appendCandidateIfAbsent(List<RerouteCandidateCommand> reorderedList,
-                                         RerouteCandidateCommand candidateCommand) {
-        if (reorderedList == null || candidateCommand == null) {
-            return;
-        }
-        if (!reorderedList.contains(candidateCommand)) {
-            reorderedList.add(candidateCommand);
-        }
-    }
-
-    private RunBlockRerouteState loadRunBlockRerouteState(Integer taskNo, Integer blockStationId) {
-        if (redisUtil == null || taskNo == null || taskNo <= 0 || blockStationId == null || blockStationId <= 0) {
-            return new RunBlockRerouteState();
-        }
-        Object stateObj = redisUtil.get(buildRunBlockRerouteStateKey(taskNo, blockStationId));
-        if (stateObj == null) {
-            return new RunBlockRerouteState();
-        }
-        try {
-            RunBlockRerouteState state = JSON.parseObject(String.valueOf(stateObj), RunBlockRerouteState.class);
-            return state == null ? new RunBlockRerouteState() : state.normalize();
-        } catch (Exception ignore) {
-            return new RunBlockRerouteState();
-        }
-    }
-
-    private void saveRunBlockRerouteState(RunBlockRerouteState rerouteState) {
-        if (redisUtil == null
-                || rerouteState == null
-                || rerouteState.getTaskNo() == null
-                || rerouteState.getTaskNo() <= 0
-                || rerouteState.getBlockStationId() == null
-                || rerouteState.getBlockStationId() <= 0) {
-            return;
-        }
-        rerouteState.normalize();
-        redisUtil.set(
-                buildRunBlockRerouteStateKey(rerouteState.getTaskNo(), rerouteState.getBlockStationId()),
-                JSON.toJSONString(rerouteState),
-                RUN_BLOCK_REROUTE_STATE_EXPIRE_SECONDS
-        );
-    }
-
-    private TaskLoopRerouteState loadTaskLoopRerouteState(Integer taskNo) {
-        if (redisUtil == null || taskNo == null || taskNo <= 0) {
-            return new TaskLoopRerouteState();
-        }
-        Object stateObj = redisUtil.get(RedisKeyType.STATION_RUN_BLOCK_TASK_LOOP_STATE_.key + taskNo);
-        if (stateObj == null) {
-            return new TaskLoopRerouteState();
-        }
-        try {
-            TaskLoopRerouteState state = JSON.parseObject(String.valueOf(stateObj), TaskLoopRerouteState.class);
-            return state == null ? new TaskLoopRerouteState() : state.normalize();
-        } catch (Exception ignore) {
-            return new TaskLoopRerouteState();
-        }
-    }
-
-    private void saveTaskLoopRerouteState(TaskLoopRerouteState taskLoopRerouteState) {
-        if (redisUtil == null
-                || taskLoopRerouteState == null
-                || taskLoopRerouteState.getTaskNo() == null
-                || taskLoopRerouteState.getTaskNo() <= 0) {
-            return;
-        }
-        taskLoopRerouteState.normalize();
-        redisUtil.set(
-                RedisKeyType.STATION_RUN_BLOCK_TASK_LOOP_STATE_.key + taskLoopRerouteState.getTaskNo(),
-                JSON.toJSONString(taskLoopRerouteState),
-                RUN_BLOCK_REROUTE_STATE_EXPIRE_SECONDS
-        );
-    }
-
-    private void touchTaskLoopRerouteState(TaskLoopRerouteState taskLoopRerouteState,
-                                           LoopIdentity currentLoopIdentity) {
-        if (taskLoopRerouteState == null || currentLoopIdentity == null || Cools.isEmpty(currentLoopIdentity.getLoopFingerprint())) {
-            return;
-        }
-        taskLoopRerouteState.getLoopIssueCountMap().put(
-                currentLoopIdentity.getLoopFingerprint(),
-                taskLoopRerouteState.getLoopIssueCountMap().getOrDefault(currentLoopIdentity.getLoopFingerprint(), 0) + 1
-        );
-        taskLoopRerouteState.setLastLoopFingerprint(currentLoopIdentity.getLoopFingerprint());
-        taskLoopRerouteState.setLastIssueTime(System.currentTimeMillis());
-    }
-
-    private int resolveCurrentLoopIssuedCount(TaskLoopRerouteState taskLoopRerouteState,
-                                              LoopIdentity currentLoopIdentity) {
-        if (taskLoopRerouteState == null || currentLoopIdentity == null || Cools.isEmpty(currentLoopIdentity.getLoopFingerprint())) {
-            return 0;
-        }
-        return taskLoopRerouteState.getLoopIssueCountMap().getOrDefault(currentLoopIdentity.getLoopFingerprint(), 0);
-    }
-
-    private int resolveExpectedLoopIssuedCount(TaskLoopRerouteState taskLoopRerouteState,
-                                               LoopIdentity currentLoopIdentity) {
-        if (currentLoopIdentity == null || Cools.isEmpty(currentLoopIdentity.getLoopFingerprint())) {
-            return 0;
-        }
-        return resolveCurrentLoopIssuedCount(taskLoopRerouteState, currentLoopIdentity) + 1;
-    }
-
-    private boolean isLoopRepeatTriggered(Integer loopIssuedCount) {
-        return loopIssuedCount != null && loopIssuedCount >= LOOP_REPEAT_TRIGGER_COUNT;
-    }
-
-    private void syncTaskTraceLoopAlert(Integer taskNo,
-                                        Integer blockedStationId,
-                                        LoopIdentity currentLoopIdentity,
-                                        int loopIssuedCount) {
-        StationTaskTraceRegistry traceRegistry;
-        try {
-            traceRegistry = SpringUtils.getBean(StationTaskTraceRegistry.class);
-        } catch (Exception ignore) {
-            return;
-        }
-        if (traceRegistry == null || taskNo == null || taskNo <= 0) {
-            return;
-        }
-
-        boolean active = currentLoopIdentity != null
-                && !Cools.isEmpty(currentLoopIdentity.getLoopFingerprint())
-                && isLoopRepeatTriggered(loopIssuedCount);
-        Map<String, Object> details = new HashMap<>();
-        details.put("blockedStationId", blockedStationId);
-        details.put("loopScopeType", currentLoopIdentity == null ? "" : currentLoopIdentity.getScopeType());
-        details.put("loopStationCount", currentLoopIdentity == null ? 0 : currentLoopIdentity.getLocalStationCount());
-        details.put("sourceLoopStationCount", currentLoopIdentity == null ? 0 : currentLoopIdentity.getSourceLoopStationCount());
-        details.put("loopRepeatCount", loopIssuedCount);
-        String loopAlertType = resolveLoopAlertType(currentLoopIdentity);
-        String loopAlertText = buildLoopAlertText(loopAlertType, currentLoopIdentity, loopIssuedCount);
-        traceRegistry.updateLoopHint(taskNo, active, loopAlertType, loopAlertText, loopIssuedCount, details);
-    }
-
-    private String resolveLoopAlertType(LoopIdentity currentLoopIdentity) {
-        if (currentLoopIdentity == null || Cools.isEmpty(currentLoopIdentity.getLoopFingerprint())) {
-            return "";
-        }
-        return "wholeLoop".equals(currentLoopIdentity.getScopeType()) ? "LARGE_LOOP" : "SMALL_LOOP";
-    }
-
-    private String buildLoopAlertText(String loopAlertType,
-                                      LoopIdentity currentLoopIdentity,
-                                      int loopIssuedCount) {
-        if (Cools.isEmpty(loopAlertType) || currentLoopIdentity == null || !isLoopRepeatTriggered(loopIssuedCount)) {
-            return "";
-        }
-        String typeLabel = "LARGE_LOOP".equals(loopAlertType) ? "澶х幆绾�" : "灏忕幆绾�";
-        return typeLabel + "缁曞湀棰勮锛岀疮璁¢噸瑙勫垝" + loopIssuedCount + "娆★紝褰撳墠璇嗗埆鑼冨洿"
-                + currentLoopIdentity.getLocalStationCount() + "绔�";
-    }
-
-    private int countCurrentLoopStationHit(List<Integer> path, Set<Integer> currentLoopStationIdSet) {
-        if (path == null || path.isEmpty() || currentLoopStationIdSet == null || currentLoopStationIdSet.isEmpty()) {
-            return 0;
-        }
-        int hitCount = 0;
-        for (Integer stationId : path) {
-            if (stationId != null && currentLoopStationIdSet.contains(stationId)) {
-                hitCount++;
-            }
-        }
-        return hitCount;
-    }
-
-    private String buildPathSignature(List<Integer> path) {
-        if (path == null || path.isEmpty()) {
-            return "";
-        }
-        StringBuilder builder = new StringBuilder();
-        for (Integer stationNo : path) {
-            if (stationNo == null) {
-                continue;
-            }
-            if (builder.length() > 0) {
-                builder.append("->");
-            }
-            builder.append(stationNo);
-        }
-        return builder.toString();
-    }
-
     private StationTaskLoopService loadStationTaskLoopService() {
         try {
             return SpringUtils.getBean(StationTaskLoopService.class);
         } catch (Exception ignore) {
             return null;
         }
-    }
-
-    private String buildRunBlockRerouteStateKey(Integer taskNo, Integer blockStationId) {
-        return RedisKeyType.STATION_RUN_BLOCK_REROUTE_STATE_.key + taskNo + "_" + blockStationId;
-    }
-
-    private LoopIdentity resolveStationLoopIdentity(Integer stationId) {
-        if (stationId == null || stationId <= 0) {
-            return LoopIdentity.empty();
-        }
-        try {
-            StationCycleCapacityService stationCycleCapacityService = SpringUtils.getBean(StationCycleCapacityService.class);
-            if (stationCycleCapacityService == null) {
-                return LoopIdentity.empty();
-            }
-            StationCycleCapacityVo capacityVo = stationCycleCapacityService.getLatestSnapshot();
-            if (capacityVo == null || capacityVo.getLoopList() == null || capacityVo.getLoopList().isEmpty()) {
-                return LoopIdentity.empty();
-            }
-            for (StationCycleLoopVo loopVo : capacityVo.getLoopList()) {
-                List<Integer> loopStationIdList = normalizeLoopStationIdList(loopVo == null ? null : loopVo.getStationIdList());
-                if (loopStationIdList.isEmpty() || !loopStationIdList.contains(stationId)) {
-                    continue;
-                }
-                Set<Integer> loopStationIdSet = new HashSet<>(loopStationIdList);
-                Map<Integer, Set<Integer>> stationGraph = loadUndirectedStationGraph();
-                List<Integer> localCycleStationIdList = resolveLocalCycleStationIdList(stationId, loopStationIdSet, stationGraph);
-                if (localCycleStationIdList.size() >= 3) {
-                    return buildLoopIdentity(localCycleStationIdList, loopStationIdList.size(), "localCycle");
-                }
-
-                List<Integer> localNeighborhoodStationIdList = resolveLocalNeighborhoodStationIdList(stationId, loopStationIdSet, stationGraph);
-                if (localNeighborhoodStationIdList.size() >= 3 && localNeighborhoodStationIdList.size() < loopStationIdList.size()) {
-                    return buildLoopIdentity(localNeighborhoodStationIdList, loopStationIdList.size(), "localNeighborhood");
-                }
-                return buildLoopIdentity(loopStationIdList, loopStationIdList.size(), "wholeLoop");
-            }
-        } catch (Exception ignore) {
-        }
-        return LoopIdentity.empty();
-    }
-
-    private LoopIdentity buildLoopIdentity(List<Integer> stationIdList,
-                                           int sourceLoopStationCount,
-                                           String scopeType) {
-        List<Integer> normalizedStationIdList = normalizeLoopStationIdList(stationIdList);
-        if (normalizedStationIdList.isEmpty()) {
-            return LoopIdentity.empty();
-        }
-        return new LoopIdentity(
-                buildLoopFingerprint(normalizedStationIdList),
-                new HashSet<>(normalizedStationIdList),
-                sourceLoopStationCount,
-                normalizedStationIdList.size(),
-                scopeType
-        );
-    }
-
-    private List<Integer> resolveLocalCycleStationIdList(Integer stationId,
-                                                         Set<Integer> loopStationIdSet,
-                                                         Map<Integer, Set<Integer>> stationGraph) {
-        if (stationId == null
-                || stationId <= 0
-                || loopStationIdSet == null
-                || loopStationIdSet.isEmpty()
-                || stationGraph == null
-                || stationGraph.isEmpty()) {
-            return new ArrayList<>();
-        }
-        Set<Integer> localNeighborhoodStationIdSet = collectLoopNeighborhoodStationIdSet(
-                stationId,
-                loopStationIdSet,
-                stationGraph,
-                LOCAL_LOOP_NEIGHBOR_HOP
-        );
-        if (localNeighborhoodStationIdSet.size() < 3) {
-            return new ArrayList<>();
-        }
-
-        Set<Integer> directNeighborStationIdSet = filterLoopNeighborStationIdSet(
-                stationGraph.getOrDefault(stationId, Collections.emptySet()),
-                localNeighborhoodStationIdSet,
-                stationId
-        );
-        if (directNeighborStationIdSet.size() < 2) {
-            return new ArrayList<>();
-        }
-
-        List<Integer> bestCycleStationIdList = new ArrayList<>();
-        List<Integer> neighborStationIdList = new ArrayList<>(directNeighborStationIdSet);
-        for (int i = 0; i < neighborStationIdList.size(); i++) {
-            Integer leftNeighborStationId = neighborStationIdList.get(i);
-            if (leftNeighborStationId == null) {
-                continue;
-            }
-            for (int j = i + 1; j < neighborStationIdList.size(); j++) {
-                Integer rightNeighborStationId = neighborStationIdList.get(j);
-                if (rightNeighborStationId == null) {
-                    continue;
-                }
-                List<Integer> pathBetweenNeighbors = findShortestScopePath(
-                        leftNeighborStationId,
-                        rightNeighborStationId,
-                        stationId,
-                        localNeighborhoodStationIdSet,
-                        stationGraph
-                );
-                if (pathBetweenNeighbors.isEmpty()) {
-                    continue;
-                }
-                List<Integer> cycleStationIdList = new ArrayList<>();
-                cycleStationIdList.add(stationId);
-                cycleStationIdList.addAll(pathBetweenNeighbors);
-                cycleStationIdList = normalizeLoopStationIdList(cycleStationIdList);
-                if (cycleStationIdList.size() < 3) {
-                    continue;
-                }
-                if (bestCycleStationIdList.isEmpty() || cycleStationIdList.size() < bestCycleStationIdList.size()) {
-                    bestCycleStationIdList = cycleStationIdList;
-                }
-            }
-        }
-        return bestCycleStationIdList;
-    }
-
-    private List<Integer> resolveLocalNeighborhoodStationIdList(Integer stationId,
-                                                                Set<Integer> loopStationIdSet,
-                                                                Map<Integer, Set<Integer>> stationGraph) {
-        return normalizeLoopStationIdList(new ArrayList<>(collectLoopNeighborhoodStationIdSet(
-                stationId,
-                loopStationIdSet,
-                stationGraph,
-                LOCAL_LOOP_NEIGHBOR_HOP
-        )));
-    }
-
-    private Set<Integer> collectLoopNeighborhoodStationIdSet(Integer stationId,
-                                                             Set<Integer> loopStationIdSet,
-                                                             Map<Integer, Set<Integer>> stationGraph,
-                                                             int maxHop) {
-        Set<Integer> neighborhoodStationIdSet = new LinkedHashSet<>();
-        if (stationId == null
-                || stationId <= 0
-                || loopStationIdSet == null
-                || loopStationIdSet.isEmpty()
-                || !loopStationIdSet.contains(stationId)
-                || stationGraph == null
-                || stationGraph.isEmpty()
-                || maxHop < 0) {
-            return neighborhoodStationIdSet;
-        }
-
-        Deque<StationHopNode> queue = new ArrayDeque<>();
-        queue.offer(new StationHopNode(stationId, 0));
-        neighborhoodStationIdSet.add(stationId);
-        while (!queue.isEmpty()) {
-            StationHopNode current = queue.poll();
-            if (current == null || current.getHop() >= maxHop) {
-                continue;
-            }
-            Set<Integer> neighborStationIdSet = filterLoopNeighborStationIdSet(
-                    stationGraph.getOrDefault(current.getStationId(), Collections.emptySet()),
-                    loopStationIdSet,
-                    null
-            );
-            for (Integer neighborStationId : neighborStationIdSet) {
-                if (neighborStationId == null || !neighborhoodStationIdSet.add(neighborStationId)) {
-                    continue;
-                }
-                queue.offer(new StationHopNode(neighborStationId, current.getHop() + 1));
-            }
-        }
-        return neighborhoodStationIdSet;
-    }
-
-    private Set<Integer> filterLoopNeighborStationIdSet(Set<Integer> candidateNeighborStationIdSet,
-                                                        Set<Integer> allowedStationIdSet,
-                                                        Integer excludedStationId) {
-        Set<Integer> result = new LinkedHashSet<>();
-        if (candidateNeighborStationIdSet == null || candidateNeighborStationIdSet.isEmpty()
-                || allowedStationIdSet == null || allowedStationIdSet.isEmpty()) {
-            return result;
-        }
-        for (Integer stationId : candidateNeighborStationIdSet) {
-            if (stationId == null
-                    || !allowedStationIdSet.contains(stationId)
-                    || (excludedStationId != null && excludedStationId.equals(stationId))) {
-                continue;
-            }
-            result.add(stationId);
-        }
-        return result;
-    }
-
-    private List<Integer> findShortestScopePath(Integer startStationId,
-                                                Integer endStationId,
-                                                Integer excludedStationId,
-                                                Set<Integer> allowedStationIdSet,
-                                                Map<Integer, Set<Integer>> stationGraph) {
-        if (startStationId == null
-                || endStationId == null
-                || Objects.equals(startStationId, excludedStationId)
-                || Objects.equals(endStationId, excludedStationId)
-                || allowedStationIdSet == null
-                || !allowedStationIdSet.contains(startStationId)
-                || !allowedStationIdSet.contains(endStationId)
-                || stationGraph == null
-                || stationGraph.isEmpty()) {
-            return new ArrayList<>();
-        }
-
-        Deque<Integer> queue = new ArrayDeque<>();
-        Map<Integer, Integer> parentMap = new HashMap<>();
-        Set<Integer> visitedStationIdSet = new HashSet<>();
-        queue.offer(startStationId);
-        visitedStationIdSet.add(startStationId);
-        while (!queue.isEmpty()) {
-            Integer currentStationId = queue.poll();
-            if (currentStationId == null) {
-                continue;
-            }
-            if (Objects.equals(currentStationId, endStationId)) {
-                break;
-            }
-            Set<Integer> neighborStationIdSet = filterLoopNeighborStationIdSet(
-                    stationGraph.getOrDefault(currentStationId, Collections.emptySet()),
-                    allowedStationIdSet,
-                    excludedStationId
-            );
-            for (Integer neighborStationId : neighborStationIdSet) {
-                if (neighborStationId == null || !visitedStationIdSet.add(neighborStationId)) {
-                    continue;
-                }
-                parentMap.put(neighborStationId, currentStationId);
-                queue.offer(neighborStationId);
-            }
-        }
-        if (!visitedStationIdSet.contains(endStationId)) {
-            return new ArrayList<>();
-        }
-
-        List<Integer> pathStationIdList = new ArrayList<>();
-        Integer cursorStationId = endStationId;
-        while (cursorStationId != null) {
-            pathStationIdList.add(cursorStationId);
-            if (Objects.equals(cursorStationId, startStationId)) {
-                break;
-            }
-            cursorStationId = parentMap.get(cursorStationId);
-        }
-        if (pathStationIdList.isEmpty()
-                || !Objects.equals(pathStationIdList.get(pathStationIdList.size() - 1), startStationId)) {
-            return new ArrayList<>();
-        }
-        Collections.reverse(pathStationIdList);
-        return pathStationIdList;
-    }
-
-    private Map<Integer, Set<Integer>> loadUndirectedStationGraph() {
-        NavigateUtils navigateUtils = SpringUtils.getBean(NavigateUtils.class);
-        if (navigateUtils == null) {
-            return new HashMap<>();
-        }
-        Map<Integer, Set<Integer>> stationGraph = navigateUtils.loadUndirectedStationGraphSnapshot();
-        return stationGraph == null ? new HashMap<>() : stationGraph;
-    }
-
-    private List<Integer> normalizeLoopStationIdList(List<Integer> stationIdList) {
-        if (stationIdList == null || stationIdList.isEmpty()) {
-            return new ArrayList<>();
-        }
-        List<Integer> normalizedList = new ArrayList<>();
-        Set<Integer> seenStationIdSet = new HashSet<>();
-        for (Integer stationId : stationIdList) {
-            if (stationId == null || stationId <= 0 || !seenStationIdSet.add(stationId)) {
-                continue;
-            }
-            normalizedList.add(stationId);
-        }
-        Collections.sort(normalizedList);
-        return normalizedList;
-    }
-
-    private String buildLoopFingerprint(List<Integer> stationIdList) {
-        if (stationIdList == null || stationIdList.isEmpty()) {
-            return "";
-        }
-        StringBuilder builder = new StringBuilder();
-        for (Integer stationId : stationIdList) {
-            if (stationId == null) {
-                continue;
-            }
-            if (builder.length() > 0) {
-                builder.append("|");
-            }
-            builder.append(stationId);
-        }
-        return builder.toString();
-    }
-
-    private int safeInt(Integer value) {
-        return value == null ? 0 : value;
-    }
-
-    @Data
-    private static class RunBlockRerouteState {
-        private Integer taskNo;
-        private Integer blockStationId;
-        private Integer planCount = 0;
-        private Integer lastTargetStationId;
-        private Long lastPlanTime;
-        private List<List<Integer>> issuedRoutePathList = new ArrayList<>();
-        private List<Integer> lastSelectedRoute = new ArrayList<>();
-        private Set<String> issuedRouteSignatureSet = new LinkedHashSet<>();
-        private Map<String, Integer> routeIssueCountMap = new HashMap<>();
-
-        private RunBlockRerouteState normalize() {
-            if (planCount == null || planCount < 0) {
-                planCount = 0;
-            }
-            if (issuedRoutePathList == null) {
-                issuedRoutePathList = new ArrayList<>();
-            }
-            if (lastSelectedRoute == null) {
-                lastSelectedRoute = new ArrayList<>();
-            }
-            if (issuedRouteSignatureSet == null) {
-                issuedRouteSignatureSet = new LinkedHashSet<>();
-            }
-            if (routeIssueCountMap == null) {
-                routeIssueCountMap = new HashMap<>();
-            }
-            for (List<Integer> routePath : issuedRoutePathList) {
-                if (routePath == null || routePath.isEmpty()) {
-                    continue;
-                }
-                String pathSignature = buildPathSignatureText(routePath);
-                if (!Cools.isEmpty(pathSignature)) {
-                    issuedRouteSignatureSet.add(pathSignature);
-                    routeIssueCountMap.putIfAbsent(pathSignature, 1);
-                }
-            }
-            return this;
-        }
-
-        private void resetIssuedRoutes() {
-            this.issuedRoutePathList = new ArrayList<>();
-            this.lastSelectedRoute = new ArrayList<>();
-            this.issuedRouteSignatureSet = new LinkedHashSet<>();
-        }
-
-        private static String buildPathSignatureText(List<Integer> routePath) {
-            if (routePath == null || routePath.isEmpty()) {
-                return "";
-            }
-            StringBuilder builder = new StringBuilder();
-            for (Integer stationId : routePath) {
-                if (stationId == null) {
-                    continue;
-                }
-                if (builder.length() > 0) {
-                    builder.append("->");
-                }
-                builder.append(stationId);
-            }
-            return builder.toString();
-        }
-    }
-
-    @Data
-    private static class RerouteCandidateCommand {
-        private StationCommand command;
-        private String routeSignature;
-        private Integer pathLength;
-        private Integer issuedCount;
-        private String loopFingerprint;
-        private Integer loopIssuedCount;
-        private Boolean loopTriggered;
-        private Integer currentLoopHitCount;
-    }
-
-    @Data
-    private static class TaskLoopRerouteState {
-        private Integer taskNo;
-        private String lastLoopFingerprint;
-        private Long lastIssueTime;
-        private Map<String, Integer> loopIssueCountMap = new HashMap<>();
-
-        private TaskLoopRerouteState normalize() {
-            if (loopIssueCountMap == null) {
-                loopIssueCountMap = new HashMap<>();
-            }
-            return this;
-        }
-    }
-
-    @Data
-    private static class LoopIdentity {
-        private String loopFingerprint;
-        private Set<Integer> stationIdSet = new HashSet<>();
-        private int sourceLoopStationCount;
-        private int localStationCount;
-        private String scopeType;
-
-        private LoopIdentity(String loopFingerprint,
-                             Set<Integer> stationIdSet,
-                             int sourceLoopStationCount,
-                             int localStationCount,
-                             String scopeType) {
-            this.loopFingerprint = loopFingerprint;
-            this.stationIdSet = stationIdSet == null ? new HashSet<>() : stationIdSet;
-            this.sourceLoopStationCount = sourceLoopStationCount;
-            this.localStationCount = localStationCount;
-            this.scopeType = scopeType == null ? "" : scopeType;
-        }
-
-        private static LoopIdentity empty() {
-            return new LoopIdentity("", new HashSet<>(), 0, 0, "none");
-        }
-    }
-
-    @Data
-    private static class StationHopNode {
-        private final Integer stationId;
-        private final int hop;
     }
 }

--
Gitblit v1.9.1