From cac27c083bacfcb1fe602d722729676f87ee7023 Mon Sep 17 00:00:00 2001
From: zwl <1051256694@qq.com>
Date: 星期五, 24 四月 2026 15:20:25 +0800
Subject: [PATCH] 完善电视机出库托数延时高的问题

---
 src/main/java/com/zy/api/service/impl/WcsApiServiceImpl.java |  346 +++++++++++++++++++++++++++++++++++++++++++++++++--------
 1 files changed, 294 insertions(+), 52 deletions(-)

diff --git a/src/main/java/com/zy/api/service/impl/WcsApiServiceImpl.java b/src/main/java/com/zy/api/service/impl/WcsApiServiceImpl.java
index dacbf9f..4fc62b5 100644
--- a/src/main/java/com/zy/api/service/impl/WcsApiServiceImpl.java
+++ b/src/main/java/com/zy/api/service/impl/WcsApiServiceImpl.java
@@ -17,6 +17,7 @@
 import com.zy.api.service.WcsApiService;
 import com.zy.asrs.entity.*;
 import com.zy.asrs.service.*;
+import com.zy.asrs.task.support.OutboundBatchSeqReleaseGuard;
 import com.zy.asrs.utils.Utils;
 import com.zy.common.constant.MesConstant;
 import com.zy.common.model.LocTypeDto;
@@ -35,6 +36,7 @@
 
 import java.io.IOException;
 import java.math.BigDecimal;
+import java.math.BigInteger;
 import java.util.*;
 import java.util.concurrent.TimeUnit;
 import java.util.stream.Collectors;
@@ -45,6 +47,12 @@
     private static final Long WCS_SYNC_USER = 9999L;
     private static final String YES = "Y";
     private static final String NO = "N";
+    private static final long OUT_LOCK_REPORT_PENDING_WRK_STS = 13L;
+    private static final long OUT_LOCK_REPORT_SUCCESS_WRK_STS = 21L;
+    private static final long OUT_LOCK_REPORT_FAIL_WRK_STS = 22L;
+    private static final long OUTBOUND_CRN_COMPLETE_WRK_STS = 25L;
+    private static final long OUTBOUND_STATION_COMPLETE_WRK_STS = 26L;
+    private static final String OUT_LOCK_REPORT_PENDING_FLAG = "P";
 
     /** 鍚屼竴 WCS 璺緞銆佸悓涓�鍗曞彿涓嬩竴缁勪笅鍙戠殑浠诲姟鏉℃暟涓婇檺 */
     private static final int WCS_PUB_BATCH_SIZE = 20;
@@ -103,6 +111,8 @@
     private RowLastnoService rowLastnoService;
     @Autowired
     private RedisUtil redisUtil;
+    @Autowired
+    private OutboundBatchSeqReleaseGuard outboundBatchSeqReleaseGuard;
 
 
     /**
@@ -122,6 +132,10 @@
         String validateMsg = validatePubTask(params, wrkMast);
         if (!Cools.isEmpty(validateMsg)) {
             return R.error(validateMsg);
+        }
+        String batchBlockMsg = validateOutboundBatchSeqReady(params, wrkMast);
+        if (!Cools.isEmpty(batchBlockMsg)) {
+            return R.error(batchBlockMsg);
         }
         String url = resolveTaskPath(params);
         String requestJson = JSON.toJSONString(params);
@@ -199,6 +213,7 @@
         List<String> failMsgs = new ArrayList<>();
         List<WorkTaskParams> lastSentChunk = null;
         String skipGroupKey = null;
+        Set<String> blockedOutboundUserKeys = new HashSet<>();
 
         for (List<WorkTaskParams> chunk : chunks) {
             if (chunk == null || chunk.isEmpty()) {
@@ -207,8 +222,22 @@
             WorkTaskParams head = chunk.get(0);
             WrkMast headMast = wrkMastMap.get(head.getTaskNo());
             String key = buildBatchGroupKey(head, headMast);
+            String outboundUserKey = buildOutboundUserKey(head, headMast);
 
             if (skipGroupKey != null && skipGroupKey.equals(key)) {
+                continue;
+            }
+            if (outboundUserKey != null && blockedOutboundUserKeys.contains(outboundUserKey)) {
+                continue;
+            }
+
+            String batchBlockMsg = validateOutboundBatchSeqReady(chunk, wrkMastMap);
+            if (!Cools.isEmpty(batchBlockMsg)) {
+                skipMsgs.add(batchBlockMsg);
+                skipGroupKey = key;
+                if (outboundUserKey != null) {
+                    blockedOutboundUserKeys.add(outboundUserKey);
+                }
                 continue;
             }
 
@@ -302,7 +331,7 @@
     }
 
     /**
-     * 鍑哄簱锛氫粎褰撳崟鍙枫�佸簭鍙峰潎鏈夋晥鏃跺仛璺冲彿鏍¢獙锛涘崟鍙风┖鎴栧簭鍙锋棤鏁堜粛涓嬪彂銆傚叆搴�/绉诲簱涓嶅鐞嗐��
+     * 鍑哄簱锛氫粎褰撳崟鍙枫�佹壒娆°�佸簭鍙峰潎鏈夋晥鏃跺仛鎵规鍐呰烦鍙锋牎楠岋紱鏃犳晥鏃朵粛涓嬪彂銆傚叆搴�/绉诲簱涓嶅鐞嗐��
      */
     private List<WorkTaskParams> filterOutboundByContiguousPlt(List<WorkTaskParams> accepted, Map<String, WrkMast> wrkMastMap, List<String> skipMsgs) {
         Map<String, Integer> reachCache = new HashMap<>();
@@ -314,12 +343,14 @@
             }
             WrkMast w = wrkMastMap.get(p.getTaskNo());
             String userNo = sortUserNoForPub(p, w);
+            String batchGroup = sortBatchGroupForPub(p, w);
             Integer plt = sortPltForPub(p, w);
             if (Cools.isEmpty(userNo) || plt == null || plt <= 0) {
                 kept.add(p);
                 continue;
             }
-            int maxReach = reachCache.computeIfAbsent(userNo, wrkMastService::outboundSeqMaxContiguousPlt);
+            String cacheKey = buildOutboundBatchCacheKey(userNo, batchGroup);
+            int maxReach = reachCache.computeIfAbsent(cacheKey, key -> wrkMastService.outboundSeqMaxContiguousPlt(userNo, batchGroup));
             if (plt > maxReach) {
                 skipMsgs.add(buildTaskMsg(p, "鍑哄簱搴忓彿璺冲彿锛岃烦杩�"));
                 continue;
@@ -363,7 +394,7 @@
     }
 
     /**
-     * 鍚屽崟涓嬩竴缁勶細浼樺厛 WCS queryTask锛涘け璐ユ垨鏃犳暟鎹垯涓昏〃宸查潪 11 鎴栧凡杩涘巻鍙茶〃銆�
+     * 鍚屽崟鍚屾壒涓嬩竴缁勶細浼樺厛 WCS queryTask锛涘け璐ユ垨鏃犳暟鎹垯涓昏〃宸查潪 11 鎴栧凡杩涘巻鍙茶〃銆�
      */
     private boolean sameOrderNextChunkAllowed(List<WorkTaskParams> lastSentChunk) {
         if (lastSentChunk == null || lastSentChunk.isEmpty()) {
@@ -426,7 +457,7 @@
     }
 
     /**
-     * 鍑哄簱姣忕粍涓嬪彂鍓嶏細鏈粍鏈夋湁鏁堟渶灏忓簭鍙蜂笖&gt;1 鏃讹紝鍙牎楠屻�屾渶灏忓簭鍙�-1銆嶄竴妗o紱搴忓彿鍏ㄦ棤鍒欒烦杩囨湰鏉′欢銆�
+     * 鍑哄簱姣忕粍涓嬪彂鍓嶏細鏈粍鏈夋湁鏁堟渶灏忓簭鍙蜂笖&gt;1 鏃讹紝鍙牎楠屻�屽悓鍗曞悓鎵圭殑鏈�灏忓簭鍙�-1銆嶄竴妗o紱搴忓彿鍏ㄦ棤鍒欒烦杩囨湰鏉′欢銆�
      */
     private boolean outboundChunkPredecessorPltReady(List<WorkTaskParams> chunk, Map<String, WrkMast> wrkMastMap) {
         if (chunk == null || chunk.isEmpty()) {
@@ -438,6 +469,7 @@
         }
         WrkMast headMast = wrkMastMap.get(head.getTaskNo());
         String userNo = sortUserNoForPub(head, headMast);
+        String batchGroup = sortBatchGroupForPub(head, headMast);
         if (Cools.isEmpty(userNo)) {
             return true;
         }
@@ -454,14 +486,40 @@
         if (minPlt == Integer.MAX_VALUE || minPlt <= 1) {
             return true;
         }
-        return outboundPltSlotReleasedInWms(userNo, minPlt - 1);
+        return outboundPltSlotReleasedInWms(userNo, batchGroup, minPlt - 1);
     }
 
-    private boolean outboundPltSlotReleasedInWms(String userNo, int pltType) {
-        List<WrkMast> rows = wrkMastService.selectList(new EntityWrapper<WrkMast>()
-                .eq("user_no", userNo)
-                .eq("io_type", 101)
-                .eq("plt_type", pltType));
+    private String validateOutboundBatchSeqReady(List<WorkTaskParams> chunk, Map<String, WrkMast> wrkMastMap) {
+        if (chunk == null || chunk.isEmpty()) {
+            return null;
+        }
+        WorkTaskParams head = chunk.get(0);
+        return validateOutboundBatchSeqReady(head, wrkMastMap.get(head.getTaskNo()));
+    }
+
+    private String validateOutboundBatchSeqReady(WorkTaskParams params, WrkMast wrkMast) {
+        if (params == null || !"out".equalsIgnoreCase(params.getType())) {
+            return null;
+        }
+        if (wrkMast == null || !Objects.equals(wrkMast.getIoType(), 101)) {
+            return null;
+        }
+        return outboundBatchSeqReleaseGuard.validateReady(
+                sortUserNoForPub(params, wrkMast),
+                sortBatchGroupForPub(params, wrkMast));
+    }
+
+    private boolean outboundPltSlotReleasedInWms(String userNo, String batchSeq, int pltType) {
+        EntityWrapper<WrkMast> mastWrapper = new EntityWrapper<>();
+        mastWrapper.eq("user_no", userNo);
+        mastWrapper.eq("io_type", 101);
+        mastWrapper.eq("plt_type", pltType);
+        if (batchSeq == null) {
+            mastWrapper.isNull("batch_seq");
+        } else {
+            mastWrapper.eq("batch_seq", batchSeq);
+        }
+        List<WrkMast> rows = wrkMastService.selectList(mastWrapper);
         if (rows != null && !rows.isEmpty()) {
             for (WrkMast m : rows) {
                 if (m != null && m.getWrkSts() != null && Objects.equals(m.getWrkSts(), 11L)) {
@@ -470,10 +528,16 @@
             }
             return true;
         }
-        int logCnt = wrkMastLogService.selectCount(new EntityWrapper<WrkMastLog>()
-                .eq("user_no", userNo)
-                .eq("io_type", 101)
-                .eq("plt_type", pltType));
+        EntityWrapper<WrkMastLog> logWrapper = new EntityWrapper<>();
+        logWrapper.eq("user_no", userNo);
+        logWrapper.eq("io_type", 101);
+        logWrapper.eq("plt_type", pltType);
+        if (batchSeq == null) {
+            logWrapper.isNull("batch_seq");
+        } else {
+            logWrapper.eq("batch_seq", batchSeq);
+        }
+        int logCnt = wrkMastLogService.selectCount(logWrapper);
         return logCnt > 0;
     }
 
@@ -502,6 +566,7 @@
         return Comparator
                 .comparing((WorkTaskParams p) -> Optional.ofNullable(p.getType()).orElse(""), String.CASE_INSENSITIVE_ORDER)
                 .thenComparing(p -> sortUserNoForPub(p, wrkMastMap.get(p.getTaskNo())), Comparator.nullsLast(String::compareTo))
+                .thenComparing(p -> sortBatchGroupForPub(p, wrkMastMap.get(p.getTaskNo())), WcsApiServiceImpl::compareBatchSeqNatural)
                 .thenComparing(p -> sortPltForPub(p, wrkMastMap.get(p.getTaskNo())), Comparator.nullsLast(Integer::compareTo));
     }
 
@@ -518,6 +583,77 @@
             return wrkMast.getPltType();
         }
         return p.getBatchSeq();
+    }
+
+    private static String sortBatchGroupForPub(WorkTaskParams p, WrkMast wrkMast) {
+        if (wrkMast != null) {
+            return wrkMast.getBatchSeq();
+        }
+        return null;
+    }
+
+    private static String buildOutboundUserKey(WorkTaskParams params, WrkMast wrkMast) {
+        if (params == null || !"out".equalsIgnoreCase(params.getType())) {
+            return null;
+        }
+        String userNo = sortUserNoForPub(params, wrkMast);
+        if (Cools.isEmpty(userNo)) {
+            return null;
+        }
+        return resolveSafeKey(userNo);
+    }
+
+    private static int compareBatchSeqNatural(String left, String right) {
+        String safeLeft = normalizeBatchSeq(left);
+        String safeRight = normalizeBatchSeq(right);
+        int leftIndex = 0;
+        int rightIndex = 0;
+        while (leftIndex < safeLeft.length() && rightIndex < safeRight.length()) {
+            char leftChar = safeLeft.charAt(leftIndex);
+            char rightChar = safeRight.charAt(rightIndex);
+            if (Character.isDigit(leftChar) && Character.isDigit(rightChar)) {
+                int leftStart = leftIndex;
+                int rightStart = rightIndex;
+                while (leftIndex < safeLeft.length() && Character.isDigit(safeLeft.charAt(leftIndex))) {
+                    leftIndex++;
+                }
+                while (rightIndex < safeRight.length() && Character.isDigit(safeRight.charAt(rightIndex))) {
+                    rightIndex++;
+                }
+                String leftNumber = safeLeft.substring(leftStart, leftIndex);
+                String rightNumber = safeRight.substring(rightStart, rightIndex);
+                int compare = new BigInteger(leftNumber).compareTo(new BigInteger(rightNumber));
+                if (compare != 0) {
+                    return compare;
+                }
+                compare = Integer.compare(leftNumber.length(), rightNumber.length());
+                if (compare != 0) {
+                    return compare;
+                }
+                continue;
+            }
+            int compare = Character.compare(leftChar, rightChar);
+            if (compare != 0) {
+                return compare;
+            }
+            leftIndex++;
+            rightIndex++;
+        }
+        return Integer.compare(safeLeft.length(), safeRight.length());
+    }
+
+    private static String normalizeBatchSeq(String value) {
+        return Cools.isEmpty(value) ? "" : value;
+    }
+
+    private static String resolveSafeKey(String value) {
+        return Cools.isEmpty(value) ? "_EMPTY_" : value;
+    }
+
+    private static String buildOutboundBatchCacheKey(String userNo, String batchSeq) {
+        String safeUserNo = Cools.isEmpty(userNo) ? "_NO_USER_" : userNo;
+        String safeBatchSeq = Cools.isEmpty(batchSeq) ? "_NO_BATCH_" : batchSeq;
+        return safeUserNo + "#" + safeBatchSeq;
     }
 
     /**
@@ -544,13 +680,44 @@
         }
 
 
-        if (params.getNotifyType().equals("task")) {
+        if (isOutboundCrnTaskRun(params)) {
+            // WCS鍑哄簱浠诲姟寮�濮嬶細鍫嗗灈鏈哄紑濮嬫墽琛屽嚭搴撲换鍔★紝宸ヤ綔鐘舵�� 12 -> 13銆�
+            if (isOutboundTask(mast) && Objects.equals(mast.getWrkSts(), 12L)) {
+                mast.setWrkSts(OUT_LOCK_REPORT_PENDING_WRK_STS);
+                mast.setExpTime(0D);
+                mast.setLogMk(OUT_LOCK_REPORT_PENDING_FLAG);
+                mast.setLogErrMemo(null);
+                mast.setLogErrTime(null);
+                mast.setModiTime(new Date());
+                if (!wrkMastService.updateById(mast)) {
+                    throw new CoolException("浠诲姟鐘舵�佷慨鏀瑰け璐ワ紒锛�");
+                }
+            }
+        } else if (isOutboundCrnTaskComplete(params)) {
+            // WCS鍑哄簱浠诲姟瀹屾垚锛氬爢鍨涙満鍑哄簱浠诲姟鎵ц瀹屾垚锛屽伐浣滅姸鎬� -> 25銆�
+            if (isOutboundTask(mast) && canMarkOutboundCrnComplete(mast)) {
+                mast.setWrkSts(OUTBOUND_CRN_COMPLETE_WRK_STS);
+                mast.setModiTime(new Date());
+                if (!wrkMastService.updateById(mast)) {
+                    throw new CoolException("浠诲姟鐘舵�佷慨鏀瑰け璐ワ紒锛�");
+                }
+            }
+        } else if (isOutboundStationTaskRunComplete(params)) {
+            // WCS杈撻�佺珯鐐瑰嚭搴撲换鍔¤繍琛屽畬鎴愶細鎵樼洏宸插埌鐩殑鍦帮紝宸ヤ綔鐘舵�� -> 26銆�
+            if (isOutboundTask(mast) && canMarkOutboundStationComplete(mast)) {
+                mast.setWrkSts(OUTBOUND_STATION_COMPLETE_WRK_STS);
+                mast.setModiTime(new Date());
+                if (!wrkMastService.updateById(mast)) {
+                    throw new CoolException("浠诲姟鐘舵�佷慨鏀瑰け璐ワ紒锛�");
+                }
+            }
+        } else if ("task".equalsIgnoreCase(params.getNotifyType())) {
             //浠诲姟
-            if (params.getMsgType().equals("task_complete")) {
+            if ("task_complete".equalsIgnoreCase(params.getMsgType())) {
 
                 if (mast.getIoType() == 1 || mast.getIoType() == 2 ||mast.getIoType() == 10) {
                     mast.setWrkSts(4L);
-                } else if ((mast.getIoType() == 101||mast.getIoType()==110) && mast.getWrkSts()<14) {
+                } else if (isOutboundTask(mast) && canMarkOutboundTaskComplete(mast)) {
                     mast.setWrkSts(14L);
                     if(Cools.isEmpty(mast.getStaNo())){
                         mast.setOveMk("Y");
@@ -560,27 +727,63 @@
                     throw new CoolException("浠诲姟鐘舵�佷慨鏀瑰け璐ワ紒锛�");
                 }
                 //wcs浠诲姟鍙栨秷鎺ュ彛
-            } else if (params.getMsgType().equals("task_cancel")) {
+            } else if ("task_cancel".equalsIgnoreCase(params.getMsgType())) {
                 workService.cancelWrkMast(String.valueOf(mast.getWrkNo()), 9955L);
-            } else if (params.getMsgType().equals("task_arrive")) {
-                //鍒拌揪鐩殑鍦�
-                //濡傛灉鍑哄簱浠诲姟鏄法鍖哄垯闇�瑕佺敓鎴愭柊鐨勫叆搴撲换鍔″叆搴�
-                if(!Cools.isEmpty(mast.getLocNo())){
-                    mast.setOnlineYn("N");//绛夊緟鐢熸垚璺ㄥ尯鍏ュ簱浠诲姟
-                }
-                mast.setWrkSts(14L);
-                if(Cools.isEmpty(mast.getStaNo())){
-                    mast.setOveMk("Y");
-                }
-                mast.setModiTime(new Date());
-                if (!wrkMastService.updateById(mast)) {
-                    throw new CoolException("浠诲姟鐘舵�佷慨鏀瑰け璐ワ紒锛�");
-                }
             }
-        } else if (params.getNotifyType().equals("weight")) {
-
         }
         return R.ok();
+    }
+
+    private boolean isOutboundCrnTaskRun(ReceviceTaskParams params) {
+        return params != null
+                && "Crn".equalsIgnoreCase(params.getNotifyType())
+                && "crn_out_task_run".equalsIgnoreCase(params.getMsgType());
+    }
+
+    private boolean isOutboundCrnTaskComplete(ReceviceTaskParams params) {
+        return params != null
+                && "Crn".equalsIgnoreCase(params.getNotifyType())
+                && "crn_out_task_complete".equalsIgnoreCase(params.getMsgType());
+    }
+
+    private boolean isOutboundStationTaskRunComplete(ReceviceTaskParams params) {
+        return params != null
+                && "Devp".equalsIgnoreCase(params.getNotifyType())
+                && "station_out_task_run_complete".equalsIgnoreCase(params.getMsgType());
+    }
+
+    private boolean isOutboundTask(WrkMast mast) {
+        return mast != null && mast.getIoType() != null && (mast.getIoType() == 101 || mast.getIoType() == 110);
+    }
+
+    private boolean canMarkOutboundCrnComplete(WrkMast mast) {
+        if (mast == null || mast.getWrkSts() == null) {
+            return false;
+        }
+        return mast.getWrkSts() < 14
+                || mast.getWrkSts().equals(OUT_LOCK_REPORT_SUCCESS_WRK_STS)
+                || mast.getWrkSts().equals(OUT_LOCK_REPORT_FAIL_WRK_STS);
+    }
+
+    private boolean canMarkOutboundTaskComplete(WrkMast mast) {
+        if (mast == null || mast.getWrkSts() == null) {
+            return false;
+        }
+        return mast.getWrkSts() < 14
+                || mast.getWrkSts().equals(OUT_LOCK_REPORT_SUCCESS_WRK_STS)
+                || mast.getWrkSts().equals(OUT_LOCK_REPORT_FAIL_WRK_STS)
+                || mast.getWrkSts().equals(OUTBOUND_CRN_COMPLETE_WRK_STS)
+                || mast.getWrkSts().equals(OUTBOUND_STATION_COMPLETE_WRK_STS);
+    }
+
+    private boolean canMarkOutboundStationComplete(WrkMast mast) {
+        if (mast == null || mast.getWrkSts() == null) {
+            return false;
+        }
+        return mast.getWrkSts() < 14
+                || mast.getWrkSts().equals(OUT_LOCK_REPORT_SUCCESS_WRK_STS)
+                || mast.getWrkSts().equals(OUT_LOCK_REPORT_FAIL_WRK_STS)
+                || mast.getWrkSts().equals(OUTBOUND_CRN_COMPLETE_WRK_STS);
     }
 
     @Override
@@ -641,19 +844,26 @@
             return R.error("褰撳墠鐩爣搴撲綅涓嶅瓨鍦�");
         }
 
-        Integer preferredArea = resolveReassignArea(wrkMast, currentLoc);
-        if (preferredArea == null) {
+        LocTypeDto locTypeDto = buildReassignLocTypeDto(currentLoc);
+        List<Integer> areaOrder = buildReassignAreaOrder(wrkMast, currentLoc);
+        if (Cools.isEmpty(areaOrder)) {
             return R.error("鏃犳硶纭畾浠诲姟鎵�灞炲簱鍖�");
         }
 
-        List<Integer> candidateCrnNos = buildReassignCandidateCrnNos(preferredArea, wrkMast.getCrnNo());
-        if (candidateCrnNos.isEmpty()) {
-            return R.error("褰撳墠搴撳尯娌℃湁鍏朵粬鍫嗗灈鏈哄彲渚涢噸鍒嗛厤");
+        StartupDto startupDto = null;
+        Integer preferredArea = null;
+        for (Integer area : areaOrder) {
+            List<Integer> candidateCrnNos = buildReassignCandidateCrnNos(area, wrkMast.getCrnNo());
+            if (candidateCrnNos.isEmpty()) {
+                continue;
+            }
+            startupDto = commonService.findRun2InboundLocByCandidateCrnNos(
+                    wrkMast.getSourceStaNo(), wrkMast.getIoType(), area, candidateCrnNos, locTypeDto);
+            if (startupDto != null && !Cools.isEmpty(startupDto.getLocNo())) {
+                preferredArea = area;
+                break;
+            }
         }
-
-        LocTypeDto locTypeDto = buildReassignLocTypeDto(currentLoc);
-        StartupDto startupDto = commonService.findRun2InboundLocByCandidateCrnNos(
-                wrkMast.getSourceStaNo(), wrkMast.getIoType(), preferredArea, candidateCrnNos, locTypeDto);
         if (startupDto == null || Cools.isEmpty(startupDto.getLocNo())) {
             return R.error("褰撳墠搴撳尯娌℃湁鍙噸鏂板垎閰嶇殑绌哄簱浣�");
         }
@@ -706,6 +916,9 @@
             if ("Y".equalsIgnoreCase(wrkMast.getPauseMk())) {
                 return "task paused";
             }
+            if (Objects.equals(wrkMast.getIoType(), 101) && Cools.isEmpty(wrkMast.getBatchSeq())) {
+                return "鍑哄簱杩涗粨缂栧彿(batchSeq)涓虹┖锛岃烦杩囦笅鍙�";
+            }
             if (requiresOutboundErpConfirm(wrkMast) && !"Y".equalsIgnoreCase(wrkMast.getPdcType())) {
                 return "task not confirmed by erp";
             }
@@ -736,15 +949,19 @@
     }
 
     private Integer resolveReassignArea(WrkMast wrkMast, LocMast currentLoc) {
-        Integer stationArea = Utils.getStationStorageArea(wrkMast.getSourceStaNo());
-        if (belongsToArea(stationArea, wrkMast.getCrnNo(), currentLoc)) {
-            return stationArea;
+        List<Integer> stationAreas = Utils.getStationStorageAreas(wrkMast.getSourceStaNo());
+        if (!Cools.isEmpty(stationAreas)) {
+            for (Integer area : stationAreas) {
+                if (belongsToArea(area, wrkMast.getCrnNo(), currentLoc)) {
+                    return area;
+                }
+            }
         }
         Integer fallbackArea = findAreaByCurrentTask(wrkMast.getCrnNo(), currentLoc);
         if (fallbackArea != null) {
             return fallbackArea;
         }
-        return stationArea;
+        return Utils.getStationStorageArea(wrkMast.getSourceStaNo());
     }
 
     private Integer findAreaByCurrentTask(Integer currentCrnNo, LocMast currentLoc) {
@@ -754,6 +971,25 @@
             }
         }
         return null;
+    }
+
+    private List<Integer> buildReassignAreaOrder(WrkMast wrkMast, LocMast currentLoc) {
+        LinkedHashSet<Integer> areaOrder = new LinkedHashSet<>();
+        List<Integer> stationAreas = Utils.getStationStorageAreas(wrkMast.getSourceStaNo());
+        if (!Cools.isEmpty(stationAreas)) {
+            areaOrder.addAll(stationAreas);
+        }
+        Integer currentArea = findAreaByCurrentTask(wrkMast.getCrnNo(), currentLoc);
+        if (currentArea != null) {
+            areaOrder.add(currentArea);
+        }
+        if (areaOrder.isEmpty()) {
+            Integer resolvedArea = resolveReassignArea(wrkMast, currentLoc);
+            if (resolvedArea != null) {
+                areaOrder.add(resolvedArea);
+            }
+        }
+        return new ArrayList<>(areaOrder);
     }
 
     private boolean belongsToArea(Integer area, Integer currentCrnNo, LocMast currentLoc) {
@@ -993,21 +1229,23 @@
      * <p>
      * 鍒嗙粍瑙勫垯锛�
      * 1. 鍏堟寜鎺ュ彛璺緞鍖哄垎锛岄伩鍏嶄笉鍚屼换鍔$被鍨嬫贩鐢ㄥ悓涓�涓� WCS 鎺ュ彛锛�
-     * 2. 鍐嶆寜 userNo 鍖哄垎锛岀‘淇濈浉鍚� userNo 鐨勪换鍔′竴璧蜂笂鎶ャ��
+     * 2. 鍐嶆寜 userNo + batchSeq 鍖哄垎锛岀‘淇濈浉鍚岃鍗曞悓鎵规鐨勪换鍔′竴璧蜂笂鎶ャ��
      * <p>
      * 姝e父鎯呭喌涓� userNo 鍙栬嚜 work_mast.user_no锛�
-     * 濡傛灉褰撳墠娌℃煡鍒板伐浣滄。锛屽垯鍥為��鍒拌姹傞噷鐨� batch 瀛楁锛屼繚璇佸吋瀹瑰凡鏈夎皟鐢ㄣ��
+     * batchSeq 鍙栬嚜 work_mast.batch_seq锛涘鏋滃綋鍓嶆病鏌ュ埌宸ヤ綔妗o紝鍒欏彧鎸� userNo 鍥為��鍏煎宸叉湁璋冪敤銆�
      */
     private String buildBatchGroupKey(WorkTaskParams params, WrkMast wrkMast) {
         String path = resolveTaskPath(params);
         String userNo = wrkMast == null ? null : wrkMast.getUserNo();
+        String batchGroup = wrkMast == null ? null : wrkMast.getBatchSeq();
         if (Cools.isEmpty(userNo)) {
             userNo = params.getBatch();
         }
         if (Cools.isEmpty(userNo)) {
             userNo = "_NO_USER_";
         }
-        return path + "#" + userNo;
+        String batchKey = Cools.isEmpty(batchGroup) ? "_NO_BATCH_" : batchGroup;
+        return path + "#" + userNo + "#" + batchKey;
     }
 
     /**
@@ -1048,10 +1286,12 @@
         if (!Cools.isEmpty(params.getStaNo())) {
             task.put("staNo", params.getStaNo());
         }
-        if (!Cools.isEmpty(params.getBatch())) {
+        boolean includeOutBatch = !"out".equalsIgnoreCase(params.getType())
+                || (params.getBatchSeq() != null && params.getBatchSeq() > 0);
+        if (includeOutBatch && !Cools.isEmpty(params.getBatch())) {
             task.put("batch", params.getBatch());
         }
-        if (!Objects.isNull(params.getBatchSeq())) {
+        if (includeOutBatch && !Objects.isNull(params.getBatchSeq())) {
             task.put("batchSeq", params.getBatchSeq());
         }
         return task;
@@ -1195,6 +1435,8 @@
             // crn_sts 鏈湴琛ㄥ瓨鐨勬槸鈥滃爢鍨涙満妯″紡(鎵嬪姩/鑷姩/鐢佃剳)鈥濓紝鍥犳蹇呴』鍐� mode锛屼笉鑳藉啓 status銆�
             basCrnp.setCrnSts(defaultZero(crnProtocol.getMode()));
             basCrnp.setWrkNo(defaultZero(crnProtocol.getTaskNo()));
+            basCrnp.setBay(crnProtocol.getBay());
+            basCrnp.setLevel(crnProtocol.getLevel());
             basCrnp.setCrnErr(crnProtocol.getAlarm() == null ? 0L : Long.valueOf(crnProtocol.getAlarm()));
             basCrnp.setModiUser(WCS_SYNC_USER);
             basCrnp.setModiTime(now);

--
Gitblit v1.9.1