From 04914a9997afbbead6f8adbb9d9c40e05b2edbd1 Mon Sep 17 00:00:00 2001
From: huang <1532065656@qq.com>
Date: 星期三, 17 十二月 2025 17:04:34 +0800
Subject: [PATCH] 修复调用导入工程失败 重复保存;修复分批出片逻辑

---
 mes-processes/mes-plcSend/src/main/java/com/mes/task/service/TaskExecutionEngine.java |  825 ++++++++++++++++++++++++++++++++++++++++++++++------------
 1 files changed, 648 insertions(+), 177 deletions(-)

diff --git a/mes-processes/mes-plcSend/src/main/java/com/mes/task/service/TaskExecutionEngine.java b/mes-processes/mes-plcSend/src/main/java/com/mes/task/service/TaskExecutionEngine.java
index e2d0469..196e3d7 100644
--- a/mes-processes/mes-plcSend/src/main/java/com/mes/task/service/TaskExecutionEngine.java
+++ b/mes-processes/mes-plcSend/src/main/java/com/mes/task/service/TaskExecutionEngine.java
@@ -247,17 +247,11 @@
                 // 4. 澶х悊鐗囩璁惧锛氬惎鍔ㄥ畾鏃跺櫒閫昏緫澶勭悊锛堜笉娑夊強PLC浜や簰锛屽彧璐熻矗閫昏緫澶勭悊锛�
                 if (isLargeGlass) {
                     TaskStepDetail step = createStepRecord(task, device, currentOrder);
-                    
-                    ScheduledFuture<?> largeGlassTask = startLargeGlassTimer(task, step, device, context);
-                    if (largeGlassTask != null) {
-                        registerScheduledTask(task.getTaskId(), largeGlassTask);
-                        stepSummaries.add(createStepSummary(device.getDeviceName(), true, "澶х悊鐗囩瀹氭椂鍣ㄥ凡鍚姩锛岄�昏緫澶勭悊涓�"));
-                    } else {
-                        stepSummaries.add(createStepSummary(device.getDeviceName(), false, "鍚姩瀹氭椂鍣ㄥけ璐�"));
-                        success = false;
-                        failureMessage = "澶х悊鐗囩璁惧鍚姩瀹氭椂鍣ㄥけ璐�";
-                        break;
-                    }
+                  
+                    // 绛夎繘鐗囧ぇ杞︽楠ょ湡姝e畬鎴愬悗鍐嶅惎鍔ㄥぇ鐞嗙墖绗煎畾鏃跺櫒锛屼繚璇佹墽琛岄『搴忎负锛氳繘鐗囧ぇ杞� -> 澶х悊鐗囩
+                    context.getSharedData().put("largeGlassStepId", step.getId());
+                    context.getSharedData().put("largeGlassDeviceId", device.getId());
+                    stepSummaries.add(createStepSummary(device.getDeviceName(), true, "宸插垱寤哄ぇ鐞嗙墖绗兼楠わ紝绛夊緟杩涚墖澶ц溅瀹屾垚鍚庡惎鍔ㄥ畾鏃跺櫒"));
                     currentOrder++;
                     continue;
                 }
@@ -676,21 +670,82 @@
                                 task.getTaskId(), device.getId());
                         return;
                     }
+                    // 宸插畬鎴�/澶辫触鐨勬楠や笉鍐嶅洖閫�鐘舵��
+                    if (TaskStepDetail.Status.COMPLETED.name().equals(step.getStatus())
+                            || TaskStepDetail.Status.FAILED.name().equals(step.getStatus())) {
+                        return;
+                    }
                     // 杩涚墖澶ц溅璁惧锛氬彧鏈夊湪鐪熸寮�濮嬪鐞嗘椂鎵嶈缃负RUNNING
+                    // 鍏堟鏌ュ崸杞珛璁惧鏄惁宸插畬鎴愶紝濡傛灉杩樺湪鎵ц涓紝涓嶅簲璇ュ紑濮嬪ぇ杞︾殑宸ヤ綔
+                    boolean transferCompleted = isTransferDeviceCompleted(task.getTaskId(), context);
+                    if (!transferCompleted) {
+                        // 鍗ц浆绔嬭繕鍦ㄦ墽琛屼腑锛岀瓑寰呭崸杞珛瀹屾垚
+                        if (!TaskStepDetail.Status.PENDING.name().equals(step.getStatus())) {
+                            step.setStatus(TaskStepDetail.Status.PENDING.name());
+                            step.setSuccessMessage("绛夊緟鍗ц浆绔嬪畬鎴�");
+                            if (step.getStartTime() == null) {
+                                step.setStartTime(new Date());
+                            }
+                            taskStepDetailMapper.updateById(step);
+                            notificationService.notifyStepUpdate(task.getTaskId(), step);
+                        }
+                        log.debug("鍗ц浆绔嬭繕鍦ㄦ墽琛屼腑锛岃繘鐗囧ぇ杞︾瓑寰�: taskId={}, deviceId={}", 
+                                task.getTaskId(), device.getId());
+                        return;
+                    }
+                    
                     // 妫�鏌ユ槸鍚︽湁鍗ц浆绔嬩富浣撳凡杈撳嚭銆佸噯澶囦笂澶ц溅鐨勭幓鐠冧俊鎭�
                     List<String> readyGlassIds = getTransferReadyGlassIds(context);
                     
-                    // 濡傛灉褰撳墠娌℃湁鏂扮殑鐜荤拑锛屾棤璁烘楠ゆ槸鍚﹀凡杩涘叆RUNNING锛岄兘搴旇杞MES浠诲姟/纭鐘舵��
+                    // 濡傛灉褰撳墠娌℃湁鏂扮殑鐜荤拑锛屼絾鍗ц浆绔嬪凡瀹屾垚锛屽彲浠ヨ疆璇ES浠诲姟/纭鐘舵��
                     if (CollectionUtils.isEmpty(readyGlassIds)) {
-                        // 杞MES浠诲姟/纭锛岄伩鍏嶉敊杩嘙ES渚у悗鍐欏叆鐨勪换鍔�
-                        pollMesForVehicle(task, step, device, context);
-                        // 濡傛灉浠嶇劧娌℃湁鍗ц浆绔嬭緭鍑虹殑鐜荤拑锛屼繚鎸�/鏇存柊涓篜ENDING鎻愮ず
-                        if (!TaskStepDetail.Status.RUNNING.name().equals(step.getStatus())
-                                && !TaskStepDetail.Status.PENDING.name().equals(step.getStatus())) {
-                            step.setStatus(TaskStepDetail.Status.PENDING.name());
-                            step.setSuccessMessage("绛夊緟鍗ц浆绔嬭緭鍑虹幓鐠�");
-                            taskStepDetailMapper.updateById(step);
-                            notificationService.notifyStepUpdate(task.getTaskId(), step);
+                        // 妫�鏌ユ槸鍚︽墍鏈夊垵濮嬬幓鐠冮兘宸茶杞�
+                        @SuppressWarnings("unchecked")
+                        List<String> initialGlassIds = (List<String>) context.getSharedData().get("initialGlassIds");
+                        List<String> loadedGlassIds = getLoadedGlassIds(context);
+                        
+                        if (initialGlassIds != null && !initialGlassIds.isEmpty()) {
+                            // 濡傛灉鎵�鏈夊垵濮嬬幓鐠冮兘宸茶杞斤紝璇存槑宸茬粡澶勭悊瀹岋紝涓嶅簲璇ュ啀鍙樺洖绛夊緟
+                            if (loadedGlassIds != null && !loadedGlassIds.isEmpty() 
+                                    && loadedGlassIds.containsAll(initialGlassIds)) {
+                                // 鎵�鏈夌幓鐠冮兘宸茶杞斤紝淇濇寔RUNNING鐘舵�侊紝缁х画杞MES浠诲姟/纭
+                                if (!TaskStepDetail.Status.RUNNING.name().equals(step.getStatus())) {
+                                    step.setStatus(TaskStepDetail.Status.RUNNING.name());
+                                    if (step.getStartTime() == null) {
+                                        step.setStartTime(new Date());
+                                    }
+                                    taskStepDetailMapper.updateById(step);
+                                    notificationService.notifyStepUpdate(task.getTaskId(), step);
+                                }
+                                // 缁х画杞MES浠诲姟/纭鐘舵�侊紝鑻ES纭瀹屾垚浼氬湪鍐呴儴鏇存柊姝ラ涓篊OMPLETED
+                                pollMesForVehicle(task, step, device, context);
+                                // 濡傛灉杩涚墖澶ц溅姝ラ鍦ㄨ疆璇㈠悗宸插畬鎴愶紝鍒欏皾璇曞惎鍔ㄥぇ鐞嗙墖绗煎畾鏃跺櫒
+                                if (TaskStepDetail.Status.COMPLETED.name().equals(step.getStatus())) {
+                                    startLargeGlassTimerIfNeeded(task, context);
+                                }
+                                return;
+                            }
+                        }
+                        
+                        // 濡傛灉澶ц溅宸茬粡瑁呰浇杩囩幓鐠冿紙RUNNING鐘舵�侊級锛岃疆璇ES浠诲姟/纭鐘舵��
+                        if (TaskStepDetail.Status.RUNNING.name().equals(step.getStatus())) {
+                            // 杞MES浠诲姟/纭鐘舵�侊紝鑻ES纭瀹屾垚浼氬湪鍐呴儴鏇存柊姝ラ涓篊OMPLETED
+                            pollMesForVehicle(task, step, device, context);
+                            // 濡傛灉杩涚墖澶ц溅姝ラ鍦ㄨ疆璇㈠悗宸插畬鎴愶紝鍒欏皾璇曞惎鍔ㄥぇ鐞嗙墖绗煎畾鏃跺櫒
+                            if (TaskStepDetail.Status.COMPLETED.name().equals(step.getStatus())) {
+                                startLargeGlassTimerIfNeeded(task, context);
+                            }
+                        } else {
+                            // 濡傛灉杩樻病鏈夎杞借繃鐜荤拑锛岀瓑寰呭崸杞珛杈撳嚭
+                            if (!TaskStepDetail.Status.PENDING.name().equals(step.getStatus())) {
+                                step.setStatus(TaskStepDetail.Status.PENDING.name());
+                                step.setSuccessMessage("绛夊緟鍗ц浆绔嬭緭鍑虹幓鐠�");
+                                if (step.getStartTime() == null) {
+                                    step.setStartTime(new Date());
+                                }
+                                taskStepDetailMapper.updateById(step);
+                                notificationService.notifyStepUpdate(task.getTaskId(), step);
+                            }
                         }
                         return;
                     }
@@ -778,15 +833,56 @@
                         // 鍙湁鍦ㄤ换鍔″凡寮�濮嬫墽琛岋紙鏈変换鍔¤褰曪級鏃舵墠妫�鏌ES纭
                         DevicePlcVO.OperationResult mesResult = null;
                         try {
-                            mesResult = handler.execute(device, "checkMesConfirm", Collections.emptyMap());
+                            Map<String, Object> confirmParams = new HashMap<>();
+                            confirmParams.put("_taskContext", context);
+                            mesResult = handler.execute(device, "checkMesConfirm", confirmParams);
                         } catch (Exception e) {
                             log.warn("杩涚墖澶ц溅璁惧妫�鏌ES纭鐘舵�佸紓甯�: taskId={}, deviceId={}, error={}",
                                     task.getTaskId(), device.getId(), e.getMessage());
                         }
                         
+                        // 濡傛灉MES宸茬‘璁ゅ畬鎴愶紙mesConfirm=1锛夛紝妫�鏌ュ崸杞珛璁惧鐘舵�佸拰鐜荤拑淇℃伅
+                        // 濡傛灉鍗ц浆绔嬪凡瀹屾垚涓旀墍鏈夌幓鐠冮兘宸茶杞斤紝鍙互鏍囪涓哄畬鎴�
+                        if (mesResult != null && mesResult.getData() != null) {
+                            Object completedFlag = mesResult.getData().get("completed");
+                            boolean mesConfirmed = false;
+                            if (completedFlag instanceof Boolean) {
+                                mesConfirmed = (Boolean) completedFlag;
+                            } else if (completedFlag != null) {
+                                mesConfirmed = "true".equalsIgnoreCase(String.valueOf(completedFlag));
+                            }
+                            
+                            if (mesConfirmed) {
+                                // MES宸茬‘璁ゅ畬鎴愶紝妫�鏌ュ崸杞珛璁惧鏄惁宸插畬鎴�
+                                boolean transferCompletedForMes = isTransferDeviceCompleted(task.getTaskId(), context);
+                                if (transferCompletedForMes) {
+                                    // 妫�鏌ヤ换鍔′笂涓嬫枃涓殑鍒濆鐜荤拑ID鍜屽凡瑁呰浇鐨勭幓鐠僆D
+                                    @SuppressWarnings("unchecked")
+                                    List<String> initialGlassIds = (List<String>) context.getSharedData().get("initialGlassIds");
+                                    List<String> loadedGlassIds = getLoadedGlassIds(context);
+                                    
+                                    if (initialGlassIds != null && !initialGlassIds.isEmpty() 
+                                            && loadedGlassIds != null && !loadedGlassIds.isEmpty()) {
+                                        // 妫�鏌ユ槸鍚︽墍鏈夊垵濮嬬幓鐠冮兘宸茶杞�
+                                        boolean allGlassesLoaded = loadedGlassIds.containsAll(initialGlassIds);
+                                        if (allGlassesLoaded) {
+                                            // 鍗ц浆绔嬪凡瀹屾垚涓旀墍鏈夌幓鐠冮兘宸茶杞斤紝鏍囪涓哄畬鎴�
+                                            log.info("MES宸茬‘璁や笖鍗ц浆绔嬪凡瀹屾垚涓旀墍鏈夌幓鐠冨凡瑁呰浇锛屼换鍔¤嚜鍔ㄥ畬鎴�: taskId={}, deviceId={}, initialCount={}, loadedCount={}",
+                                                    task.getTaskId(), device.getId(), initialGlassIds.size(), loadedGlassIds.size());
+                                            // mesResult宸茬粡鍖呭惈completed=true锛屼笉闇�瑕佷慨鏀�
+                                        }
+                                    }
+                                } else {
+                                    // 鍗ц浆绔嬭繕鏈畬鎴愶紝涓嶅簲璇ユ爣璁颁负瀹屾垚
+                                    log.debug("MES宸茬‘璁や絾鍗ц浆绔嬫湭瀹屾垚锛岀瓑寰呭崸杞珛瀹屾垚: taskId={}, deviceId={}", 
+                                            task.getTaskId(), device.getId());
+                                }
+                            }
+                        }
+                        
                         // 鏇存柊姝ラ鐘舵�侊紙澶ц溅璁惧淇濇寔RUNNING锛岀洿鍒癕ES纭瀹屾垚鎴栦换鍔″彇娑堬級
                         if (mesResult != null) {
-                            updateStepStatusForVehicle(step, mesResult);
+                            updateStepStatusForVehicle(task.getTaskId(), step, mesResult);
                             boolean opSuccess = Boolean.TRUE.equals(mesResult.getSuccess());
                             updateTaskProgress(task, step.getStepOrder(), opSuccess);
                             if (!opSuccess) {
@@ -794,13 +890,18 @@
                                         DeviceCoordinationService.DeviceStatus.FAILED, context);
                             }
                         } else {
-                            updateStepStatusForVehicle(step, feedResult);
+                            updateStepStatusForVehicle(task.getTaskId(), step, feedResult);
                             boolean opSuccess = Boolean.TRUE.equals(feedResult.getSuccess());
                             updateTaskProgress(task, step.getStepOrder(), opSuccess);
                             if (!opSuccess) {
                                 deviceCoordinationService.syncDeviceStatus(device,
                                         DeviceCoordinationService.DeviceStatus.FAILED, context);
                             }
+                        }
+
+                        // 褰撹繘鐗囧ぇ杞︽楠ょ湡姝e畬鎴愬悗锛屽啀鍚姩澶х悊鐗囩瀹氭椂鍣紝淇濊瘉鎵ц椤哄簭
+                        if (TaskStepDetail.Status.COMPLETED.name().equals(step.getStatus())) {
+                            startLargeGlassTimerIfNeeded(task, context);
                         }
                     }
                 } catch (Exception e) {
@@ -827,11 +928,10 @@
                                                          TaskExecutionContext context) {
         try {
             final long MONITOR_INTERVAL_MS = 2_000; // 2绉掔洃鎺т竴娆�
-            
+
             log.debug("鍚姩鍑虹墖澶ц溅璁惧瀹氭椂鍣�: taskId={}, deviceId={}, interval={}s",
                     task.getTaskId(), device.getId(), MONITOR_INTERVAL_MS / 1000);
-            
-            // 鍚姩瀹氭椂浠诲姟
+
             ScheduledFuture<?> future = scheduledExecutor.scheduleWithFixedDelay(() -> {
                 try {
                     if (isTaskCancelled(context)) {
@@ -839,164 +939,242 @@
                                 task.getTaskId(), device.getId());
                         return;
                     }
-                    // 鍑虹墖澶ц溅璁惧锛氬彧鏈夊湪鐪熸寮�濮嬪鐞嗘椂鎵嶈缃负RUNNING
-                    // 妫�鏌ユ槸鍚︽湁宸插鐞嗙殑鐜荤拑淇℃伅锛堜粠澶х悊鐗囩鏉ョ殑锛�
-                    List<String> processedGlassIds = getProcessedGlassIds(context);
-                    boolean isRunning = TaskStepDetail.Status.RUNNING.name().equals(step.getStatus());
-                    
-                    // 濡傛灉璁惧宸茬粡鍦ㄨ繍琛屼腑锛屽嵆浣挎病鏈夋柊鐜荤拑锛屼篃瑕佺户缁洃鎺ES浠诲姟鍜岀‘璁ょ姸鎬�
-                    if (CollectionUtils.isEmpty(processedGlassIds)) {
-                        if (isRunning) {
-                            // 璁惧姝e湪杩愯涓紝鍏堟鏌ES浠诲姟锛岀劧鍚庣洃鎺ES纭鐘舵��
-                            DeviceLogicHandler handler = handlerFactory.getHandler(device.getDeviceType());
-                            if (handler != null) {
-                                Map<String, Object> logicParams = parseLogicParams(device);
-                                
-                                // 鍏堟鏌ES浠诲姟锛堝鏋渕esSend=1锛屼細鍒涘缓浠诲姟骞跺紑濮嬫墽琛岋級
-                                DevicePlcVO.OperationResult mesTaskResult = null;
-                                try {
-                                    mesTaskResult = handler.execute(device, "checkMesTask", Collections.emptyMap());
-                                    if (mesTaskResult != null && Boolean.TRUE.equals(mesTaskResult.getSuccess())) {
-                                        log.info("鍑虹墖澶ц溅璁惧宸叉鏌ES浠诲姟骞跺紑濮嬫墽琛�: taskId={}, deviceId={}, message={}",
-                                                task.getTaskId(), device.getId(), mesTaskResult.getMessage());
-                                    }
-                                } catch (Exception e) {
-                                    log.warn("鍑虹墖澶ц溅璁惧妫�鏌ES浠诲姟寮傚父: taskId={}, deviceId={}, error={}",
-                                            task.getTaskId(), device.getId(), e.getMessage());
-                                }
-                                
-                                // 鐒跺悗妫�鏌ES纭鐘舵�侊紙鍙湁鍦ㄤ换鍔″凡寮�濮嬫墽琛屾椂鎵嶆鏌ワ級
-                                DevicePlcVO.OperationResult mesResult = null;
-                                try {
-                                    mesResult = handler.execute(device, "checkMesConfirm", Collections.emptyMap());
-                                } catch (Exception e) {
-                                    log.warn("鍑虹墖澶ц溅璁惧妫�鏌ES纭鐘舵�佸紓甯�: taskId={}, deviceId={}, error={}",
-                                            task.getTaskId(), device.getId(), e.getMessage());
-                                }
-                                
-                                // 鏇存柊姝ラ鐘舵�侊紙澶ц溅璁惧淇濇寔RUNNING锛岀洿鍒癕ES纭瀹屾垚鎴栦换鍔″彇娑堬級
-                                if (mesResult != null) {
-                                    updateStepStatusForVehicle(step, mesResult);
-                                    boolean opSuccess = Boolean.TRUE.equals(mesResult.getSuccess());
-                                    updateTaskProgress(task, step.getStepOrder(), opSuccess);
-                                    if (!opSuccess) {
-                                        deviceCoordinationService.syncDeviceStatus(device,
-                                                DeviceCoordinationService.DeviceStatus.FAILED, context);
-                                    }
-                                }
+
+                    final String lastMsgKey = "outboundVehicleLastMessage:" + device.getId();
+
+                    // 1) 鎬荤洰鏍囷細澶х悊鐗囩浜у嚭鐨勨�滃簲鍑虹墖鐜荤拑鍒楄〃鈥�
+                    List<String> processedGlassIdsRaw = getProcessedGlassIds(context);
+                    if (CollectionUtils.isEmpty(processedGlassIdsRaw)) {
+                        // 灏氭湭鏈夊ぇ鐞嗙墖绗艰緭鍑猴紝淇濇寔绛夊緟
+                        deviceCoordinationService.syncDeviceStatus(device,
+                                DeviceCoordinationService.DeviceStatus.WAITING, context);
+                        if (!TaskStepDetail.Status.PENDING.name().equals(step.getStatus())
+                                && !TaskStepDetail.Status.COMPLETED.name().equals(step.getStatus())) {
+                            step.setStatus(TaskStepDetail.Status.PENDING.name());
+                            step.setSuccessMessage("绛夊緟澶х悊鐗囩澶勭悊瀹屾垚");
+                            if (step.getStartTime() == null) {
+                                step.setStartTime(new Date());
                             }
-                            return;
-                        } else {
-                            // 娌℃湁鏁版嵁锛屼笖璁惧鏈繍琛岋紝淇濇寔PENDING鐘舵��
-                            if (!TaskStepDetail.Status.PENDING.name().equals(step.getStatus())) {
-                                step.setStatus(TaskStepDetail.Status.PENDING.name());
-                                step.setSuccessMessage("绛夊緟澶х悊鐗囩澶勭悊瀹屾垚");
-                                taskStepDetailMapper.updateById(step);
-                                notificationService.notifyStepUpdate(task.getTaskId(), step);
-                            }
-                            log.debug("鍑虹墖澶ц溅璁惧瀹氭椂鍣細鏆傛棤宸插鐞嗙殑鐜荤拑淇℃伅: taskId={}, deviceId={}",
-                                    task.getTaskId(), device.getId());
-                            return;
+                            taskStepDetailMapper.updateById(step);
+                            notificationService.notifyStepUpdate(task.getTaskId(), step);
+                        }
+                        return;
+                    }
+
+                    // 2) 宸插畬鎴愮疮绉細outboundGlassIds锛堜换鍔′晶缁存姢锛屾寜mesConfirm=1绱姞锛�
+                    Set<String> processedSet = new LinkedHashSet<>();
+                    for (String id : processedGlassIdsRaw) {
+                        if (StringUtils.hasText(id)) {
+                            processedSet.add(id);
                         }
                     }
-                    
-                    log.debug("鍑虹墖澶ц溅璁惧瀹氭椂鍣ㄦ娴嬪埌宸插鐞嗙殑鐜荤拑淇℃伅: taskId={}, deviceId={}, glassCount={}",
-                            task.getTaskId(), device.getId(), processedGlassIds.size());
-                    
-                    // 鎵ц鍑虹墖鎿嶄綔
-                    Map<String, Object> checkParams = new HashMap<>();
-                    checkParams.put("glassIds", new ArrayList<>(processedGlassIds));
-                    checkParams.put("_taskContext", context);
-                    
+                    Set<String> outboundSet = new HashSet<>();
+                    for (String id : getOutboundGlassIds(context)) {
+                        if (StringUtils.hasText(id)) {
+                            outboundSet.add(id);
+                        }
+                    }
+
+                    List<String> remaining = new ArrayList<>();
+                    for (String id : processedSet) {
+                        if (!outboundSet.contains(id)) {
+                            remaining.add(id);
+                        }
+                    }
+                    int total = processedSet.size();
+                    int done = total - remaining.size();
+
+                    // 3) 鑻ュ凡鍏ㄩ儴瀹屾垚锛岀洿鎺ユ敹灏炬楠�
+                    if (total > 0 && remaining.isEmpty()) {
+                        if (!TaskStepDetail.Status.COMPLETED.name().equals(step.getStatus())) {
+                            step.setStatus(TaskStepDetail.Status.COMPLETED.name());
+                            String lastMsg = null;
+                            Object lastObj = context.getSharedData().get(lastMsgKey);
+                            if (lastObj != null && StringUtils.hasText(String.valueOf(lastObj))) {
+                                lastMsg = String.valueOf(lastObj);
+                            }
+                            step.setSuccessMessage(StringUtils.hasText(lastMsg)
+                                    ? String.format("鍑虹墖瀹屾垚锛�%d/%d锛�%s", done, total, lastMsg)
+                                    : String.format("鍑虹墖瀹屾垚锛�%d/%d", done, total));
+                            step.setErrorMessage(null);
+                            Date now = new Date();
+                            if (step.getStartTime() == null) {
+                                step.setStartTime(now);
+                            }
+                            if (step.getEndTime() == null) {
+                                step.setEndTime(now);
+                            }
+                            if (step.getStartTime() != null && step.getEndTime() != null) {
+                                step.setDurationMs(step.getEndTime().getTime() - step.getStartTime().getTime());
+                            }
+                            taskStepDetailMapper.updateById(step);
+                            notificationService.notifyStepUpdate(task.getTaskId(), step);
+                            checkAndCompleteTaskIfDone(step.getTaskId());
+                        }
+                        deviceCoordinationService.syncDeviceStatus(device,
+                                DeviceCoordinationService.DeviceStatus.COMPLETED, context);
+                        return;
+                    }
+
+                    // 4) 杩涘叆杩愯鎬侊紙鍙湪鐪熸寮�濮嬪嚭鐗囨椂锛�
+                    if (!TaskStepDetail.Status.RUNNING.name().equals(step.getStatus())) {
+                        step.setStatus(TaskStepDetail.Status.RUNNING.name());
+                        if (step.getStartTime() == null) {
+                            step.setStartTime(new Date());
+                        }
+                    }
+                    // 鏇存柊杩涘害淇℃伅锛堜究浜庡墠绔疄鏃跺睍绀猴級
+                    Date now = new Date();
+                    if (step.getStartTime() != null) {
+                        step.setDurationMs(now.getTime() - step.getStartTime().getTime());
+                    }
+                    taskStepDetailMapper.updateById(step);
+                    notificationService.notifyStepUpdate(task.getTaskId(), step);
+
                     DeviceLogicHandler handler = handlerFactory.getHandler(device.getDeviceType());
-                    if (handler != null) {
-                        // 灏唋ogicParams鍚堝苟鍒癱heckParams涓�
-                        Map<String, Object> logicParams = parseLogicParams(device);
-                        if (logicParams != null && !logicParams.isEmpty()) {
-                            checkParams.put("_logicParams", logicParams);
-                        }
-                        // 绗竴姝ワ細鍐欏叆澶ц溅鍑虹墖璇锋眰
-                        DevicePlcVO.OperationResult feedResult = handler.execute(device, "feedGlass", checkParams);
-                        
-                        if (Boolean.TRUE.equals(feedResult.getSuccess())) {
-                            // 鐪熸寮�濮嬪鐞嗭紝璁剧疆涓篟UNNING
-                            deviceCoordinationService.syncDeviceStatus(device,
-                                    DeviceCoordinationService.DeviceStatus.RUNNING, context);
-                            // 姝ラ鐘舵�佷篃璁剧疆涓篟UNNING
-                            if (!TaskStepDetail.Status.RUNNING.name().equals(step.getStatus())) {
-                                step.setStatus(TaskStepDetail.Status.RUNNING.name());
-                                if (step.getStartTime() == null) {
-                                    step.setStartTime(new Date());
+                    if (handler == null) {
+                        log.warn("鏈壘鍒板嚭鐗囧ぇ杞andler: deviceId={}, deviceType={}", device.getId(), device.getDeviceType());
+                        return;
+                    }
+
+                    Map<String, Object> logicParams = parseLogicParams(device);
+                    Map<String, Object> baseParams = new HashMap<>();
+                    baseParams.put("_taskContext", context);
+                    if (logicParams != null && !logicParams.isEmpty()) {
+                        baseParams.put("_logicParams", logicParams);
+                    }
+
+                    String latestInteractionMsg = null;
+
+                    // 5) 鍏堟鏌ES浠诲姟锛坢esSend=1鏃朵細鎶婃湰鎵规鐜荤拑ID鍐欏叆currentMesBatchGlassIds锛�
+                    DevicePlcVO.OperationResult mesTaskResult = null;
+                    try {
+                        mesTaskResult = handler.execute(device, "checkMesTask", new HashMap<>(baseParams));
+                    } catch (Exception e) {
+                        log.warn("鍑虹墖澶ц溅妫�鏌ES浠诲姟寮傚父: taskId={}, deviceId={}, error={}",
+                                task.getTaskId(), device.getId(), e.getMessage());
+                    }
+                    if (mesTaskResult != null && StringUtils.hasText(mesTaskResult.getMessage())) {
+                        latestInteractionMsg = mesTaskResult.getMessage();
+                    }
+
+                    // 浠巆heckMesTask杩斿洖鍊间腑璇诲彇鏈壒娆$幓鐠僆D锛堜笉渚濊禆璁惧渚у啓_taskContext锛�
+                    try {
+                        if (mesTaskResult != null && mesTaskResult.getData() != null) {
+                            Object batchObj = mesTaskResult.getData().get("batchGlassIds");
+                            if (batchObj instanceof List) {
+                                @SuppressWarnings("unchecked")
+                                List<Object> raw = (List<Object>) batchObj;
+                                List<String> batchIds = new ArrayList<>();
+                                for (Object o : raw) {
+                                    if (o != null && StringUtils.hasText(String.valueOf(o))) {
+                                        batchIds.add(String.valueOf(o));
+                                    }
                                 }
-                                taskStepDetailMapper.updateById(step);
-                                notificationService.notifyStepUpdate(task.getTaskId(), step);
-                            }
-                            log.debug("鍑虹墖澶ц溅璁惧瀹氭椂鍣ㄦ墽琛屾垚鍔�: taskId={}, deviceId={}, glassCount={}",
-                                    task.getTaskId(), device.getId(), processedGlassIds.size());
-                            // 娓呯┖宸插鐞嗙殑鐜荤拑ID鍒楄〃锛堝凡澶勭悊锛�
-                            clearProcessedGlassIds(context);
-                            
-                            // feedGlass鎴愬姛鍚庯紝鍏堟鏌ES浠诲姟锛坈heckMesTask锛夋潵寮�濮嬫墽琛屼换鍔�
-                            DevicePlcVO.OperationResult mesTaskResult = null;
-                            try {
-                                mesTaskResult = handler.execute(device, "checkMesTask", Collections.emptyMap());
-                                if (mesTaskResult != null && Boolean.TRUE.equals(mesTaskResult.getSuccess())) {
-                                    log.info("鍑虹墖澶ц溅璁惧宸叉鏌ES浠诲姟骞跺紑濮嬫墽琛�: taskId={}, deviceId={}, message={}",
-                                            task.getTaskId(), device.getId(), mesTaskResult.getMessage());
+                                if (!batchIds.isEmpty()) {
+                                    context.getSharedData().put("currentMesBatchGlassIds", batchIds);
                                 }
-                            } catch (Exception e) {
-                                log.warn("鍑虹墖澶ц溅璁惧妫�鏌ES浠诲姟寮傚父: taskId={}, deviceId={}, error={}",
-                                        task.getTaskId(), device.getId(), e.getMessage());
                             }
-                        } else {
-                            // 娌℃湁鏁版嵁锛屼繚鎸乄AITING鐘舵�佸拰PENDING姝ラ鐘舵��
-                            deviceCoordinationService.syncDeviceStatus(device,
-                                    DeviceCoordinationService.DeviceStatus.WAITING, context);
-                            if (!TaskStepDetail.Status.PENDING.name().equals(step.getStatus())) {
-                                step.setStatus(TaskStepDetail.Status.PENDING.name());
-                                step.setSuccessMessage("绛夊緟涓�");
-                                taskStepDetailMapper.updateById(step);
-                                notificationService.notifyStepUpdate(task.getTaskId(), step);
-                            }
-                            log.debug("鍑虹墖澶ц溅璁惧瀹氭椂鍣ㄦ墽琛屽け璐�: taskId={}, deviceId={}, message={}",
-                                    task.getTaskId(), device.getId(), feedResult.getMessage());
                         }
-                        
-                        // 绗簩姝ワ細妫�鏌ES纭鐘舵�侊紙濡傛灉澶ц溅澶勭悊鍣ㄦ敮鎸佺殑璇濓級
-                        // 鍙湁鍦ㄤ换鍔″凡寮�濮嬫墽琛岋紙鏈変换鍔¤褰曪級鏃舵墠妫�鏌ES纭
-                        DevicePlcVO.OperationResult mesResult = null;
+                    } catch (Exception ignore) {
+                        // 涓嶅奖鍝嶄富娴佺▼
+                    }
+
+                    boolean mesSendIsZero = false;
+                    if (mesTaskResult != null && mesTaskResult.getData() != null) {
+                        Object reason = mesTaskResult.getData().get("waitingReason");
+                        if ("mesSend=0".equals(String.valueOf(reason))) {
+                            mesSendIsZero = true;
+                        }
+                    } else if (mesTaskResult != null && StringUtils.hasText(mesTaskResult.getMessage())
+                            && mesTaskResult.getMessage().contains("mesSend=0")) {
+                        mesSendIsZero = true;
+                    }
+
+                    // 6) 鑻ES灏氭湭涓嬪彂浠诲姟锛坢esSend=0锛夛紝瑙﹀彂涓�娆″嚭鐗囪姹傦紙feedGlass锛�
+                    if (mesSendIsZero && !remaining.isEmpty()) {
+                        Map<String, Object> feedParams = new HashMap<>(baseParams);
+                        feedParams.put("glassIds", new ArrayList<>(remaining));
                         try {
-                            mesResult = handler.execute(device, "checkMesConfirm", Collections.emptyMap());
+                            DevicePlcVO.OperationResult feedResult = handler.execute(device, "feedGlass", feedParams);
+                            if (feedResult != null && StringUtils.hasText(feedResult.getMessage())) {
+                                latestInteractionMsg = feedResult.getMessage();
+                            }
+                            if (Boolean.TRUE.equals(feedResult.getSuccess())) {
+                                deviceCoordinationService.syncDeviceStatus(device,
+                                        DeviceCoordinationService.DeviceStatus.RUNNING, context);
+                            } else {
+                                deviceCoordinationService.syncDeviceStatus(device,
+                                        DeviceCoordinationService.DeviceStatus.WAITING, context);
+                            }
                         } catch (Exception e) {
-                            log.warn("鍑虹墖澶ц溅璁惧妫�鏌ES纭鐘舵�佸紓甯�: taskId={}, deviceId={}, error={}",
+                            log.warn("鍑虹墖澶ц溅瑙﹀彂feedGlass寮傚父: taskId={}, deviceId={}, error={}",
                                     task.getTaskId(), device.getId(), e.getMessage());
                         }
-                        
-                        // 鏇存柊姝ラ鐘舵�侊紙澶ц溅璁惧淇濇寔RUNNING锛岀洿鍒癕ES纭瀹屾垚鎴栦换鍔″彇娑堬級
-                        if (mesResult != null) {
-                            updateStepStatusForVehicle(step, mesResult);
-                            boolean opSuccess = Boolean.TRUE.equals(mesResult.getSuccess());
-                            updateTaskProgress(task, step.getStepOrder(), opSuccess);
-                            if (!opSuccess) {
-                                deviceCoordinationService.syncDeviceStatus(device,
-                                        DeviceCoordinationService.DeviceStatus.FAILED, context);
-                            }
-                        } else {
-                            updateStepStatusForVehicle(step, feedResult);
-                            boolean opSuccess = Boolean.TRUE.equals(feedResult.getSuccess());
-                            updateTaskProgress(task, step.getStepOrder(), opSuccess);
-                            if (!opSuccess) {
-                                deviceCoordinationService.syncDeviceStatus(device,
-                                        DeviceCoordinationService.DeviceStatus.FAILED, context);
-                            }
+                    }
+
+                    // 7) 鍐嶆鏌ES纭锛坢esConfirm=1琛ㄧず鏈浜や簰瀹屾垚锛�
+                    DevicePlcVO.OperationResult mesConfirmResult = null;
+                    try {
+                        mesConfirmResult = handler.execute(device, "checkMesConfirm", new HashMap<>(baseParams));
+                    } catch (Exception e) {
+                        log.warn("鍑虹墖澶ц溅妫�鏌ES纭寮傚父: taskId={}, deviceId={}, error={}",
+                                task.getTaskId(), device.getId(), e.getMessage());
+                    }
+                    if (mesConfirmResult != null && StringUtils.hasText(mesConfirmResult.getMessage())) {
+                        // 纭鎻愮ず浼樺厛绾ф渶楂�
+                        latestInteractionMsg = mesConfirmResult.getMessage();
+                    }
+
+                    boolean interactionCompleted = false;
+                    if (mesConfirmResult != null && mesConfirmResult.getData() != null) {
+                        Object completedFlag = mesConfirmResult.getData().get("completed");
+                        if (completedFlag instanceof Boolean) {
+                            interactionCompleted = (Boolean) completedFlag;
+                        } else if (completedFlag != null) {
+                            interactionCompleted = "true".equalsIgnoreCase(String.valueOf(completedFlag));
                         }
+                    }
+
+                    // 8) 鏇存柊鈥滄渶杩戜竴娆′氦浜掓彁绀衡�濅笌姝ラ鎻愮ず鏂囨锛堥伩鍏嶈绾繘搴﹁鐩栵級
+                    if (StringUtils.hasText(latestInteractionMsg)) {
+                        context.getSharedData().put(lastMsgKey, latestInteractionMsg);
+                    }
+                    String lastMsg = null;
+                    Object lastObj = context.getSharedData().get(lastMsgKey);
+                    if (lastObj != null && StringUtils.hasText(String.valueOf(lastObj))) {
+                        lastMsg = String.valueOf(lastObj);
+                    }
+                    step.setSuccessMessage(StringUtils.hasText(lastMsg)
+                            ? String.format("鍑虹墖杩涜涓細%d/%d锛�%s", done, total, lastMsg)
+                            : String.format("鍑虹墖杩涜涓細%d/%d", done, total));
+                    taskStepDetailMapper.updateById(step);
+                    notificationService.notifyStepUpdate(task.getTaskId(), step);
+
+                    if (interactionCompleted) {
+                        // 鏈浜や簰瀹屾垚锛氬皢鏈壒娆�(currentMesBatchGlassIds)绱姞鍒板凡鍑虹墖(outboundGlassIds)
+                        Object batchObj = context.getSharedData().get("currentMesBatchGlassIds");
+                        if (batchObj instanceof List) {
+                            @SuppressWarnings("unchecked")
+                            List<String> batchIds = new ArrayList<>((List<String>) batchObj);
+                            // 浠呯疮鍔犫�滅洰鏍囬泦鍚堚�濆唴鐨勭幓鐠冿紝閬垮厤鑴忔暟鎹�
+                            List<String> filtered = new ArrayList<>();
+                            for (String id : batchIds) {
+                                if (StringUtils.hasText(id) && processedSet.contains(id)) {
+                                    filtered.add(id);
+                                }
+                            }
+                            addOutboundGlassIds(context, filtered);
+                        }
+                        // 娓呯┖鏈壒娆¤褰曪紝閬垮厤涓嬩竴娆¤疆璇㈤噸澶嶄娇鐢ㄦ棫鎵规
+                        context.getSharedData().put("currentMesBatchGlassIds", new ArrayList<>());
                     }
                 } catch (Exception e) {
                     log.error("鍑虹墖澶ц溅璁惧瀹氭椂鍣ㄦ墽琛屽紓甯�: taskId={}, deviceId={}", task.getTaskId(), device.getId(), e);
                 }
             }, 0, MONITOR_INTERVAL_MS, TimeUnit.MILLISECONDS);
-            
-            // 鍦ㄤ覆琛屾墽琛屾ā寮忎笅锛岃澶囧惎鍔ㄥ畾鏃跺櫒鏃跺厛璁剧疆涓� WAITING锛屽畾鏃跺櫒绗竴娆℃墽琛屾椂鍐嶈缃负 RUNNING
+
+            // 鍚姩鏃朵繚鎸乄AITING锛屽悗缁敱瀹氭椂鍣ㄩ�昏緫鍒囨崲涓篟UNNING/COMPLETED
             deviceCoordinationService.syncDeviceStatus(device,
                     DeviceCoordinationService.DeviceStatus.WAITING, context);
             return future;
@@ -1030,14 +1208,57 @@
                                 task.getTaskId(), device.getId());
                         return;
                     }
+                    // 濡傛灉姝ラ宸茬粡瀹屾垚锛屼笉鍐嶅鐞�
+                    if (TaskStepDetail.Status.COMPLETED.name().equals(step.getStatus())) {
+                        // 妫�鏌ユ槸鍚︽墍鏈夊垵濮嬬幓鐠冮兘宸插鐞嗗畬
+                        @SuppressWarnings("unchecked")
+                        List<String> initialGlassIds = (List<String>) context.getSharedData().get("initialGlassIds");
+                        List<String> processedGlassIds = getProcessedGlassIds(context);
+                        
+                        if (initialGlassIds != null && !initialGlassIds.isEmpty() 
+                                && processedGlassIds != null && !processedGlassIds.isEmpty()
+                                && processedGlassIds.containsAll(initialGlassIds)) {
+                            // 鎵�鏈夌幓鐠冮兘宸插鐞嗗畬锛屼繚鎸佸畬鎴愮姸鎬�
+                            log.debug("澶х悊鐗囩璁惧宸插畬鎴愪笖鎵�鏈夌幓鐠冨凡澶勭悊: taskId={}, deviceId={}, initialCount={}, processedCount={}",
+                                    task.getTaskId(), device.getId(), initialGlassIds.size(), processedGlassIds.size());
+                            return;
+                        }
+                    }
+                    
                     // 澶х悊鐗囩璁惧锛氬彧鏈夊湪鐪熸寮�濮嬪鐞嗘椂鎵嶈缃负RUNNING
                     // 妫�鏌ユ槸鍚︽湁宸茶杞界殑鐜荤拑淇℃伅锛堜粠杩涚墖澶ц溅鏉ョ殑锛�
                     List<String> loadedGlassIds = getLoadedGlassIds(context);
                     if (CollectionUtils.isEmpty(loadedGlassIds)) {
-                        // 娌℃湁鏁版嵁锛屼繚鎸乄AITING鐘舵�佸拰PENDING姝ラ鐘舵��
+                        // 娌℃湁鏁版嵁锛屾鏌ユ槸鍚︽墍鏈夌幓鐠冮兘宸插鐞嗗畬
+                        @SuppressWarnings("unchecked")
+                        List<String> initialGlassIds = (List<String>) context.getSharedData().get("initialGlassIds");
+                        List<String> processedGlassIds = getProcessedGlassIds(context);
+                        
+                        if (initialGlassIds != null && !initialGlassIds.isEmpty() 
+                                && processedGlassIds != null && !processedGlassIds.isEmpty()
+                                && processedGlassIds.containsAll(initialGlassIds)) {
+                            // 鎵�鏈夌幓鐠冮兘宸插鐞嗗畬锛屾爣璁颁负瀹屾垚
+                            if (!TaskStepDetail.Status.COMPLETED.name().equals(step.getStatus())) {
+                                step.setStatus(TaskStepDetail.Status.COMPLETED.name());
+                                step.setSuccessMessage("鎵�鏈夌幓鐠冨凡澶勭悊瀹屾垚");
+                                if (step.getEndTime() == null) {
+                                    step.setEndTime(new Date());
+                                }
+                                if (step.getStartTime() != null && step.getEndTime() != null) {
+                                    step.setDurationMs(step.getEndTime().getTime() - step.getStartTime().getTime());
+                                }
+                                taskStepDetailMapper.updateById(step);
+                                notificationService.notifyStepUpdate(task.getTaskId(), step);
+                                checkAndCompleteTaskIfDone(step.getTaskId());
+                            }
+                            return;
+                        }
+                        
+                        // 娌℃湁鏁版嵁涓旀湭瀹屾垚锛屼繚鎸乄AITING鐘舵�佸拰PENDING姝ラ鐘舵��
                         deviceCoordinationService.syncDeviceStatus(device,
                                 DeviceCoordinationService.DeviceStatus.WAITING, context);
-                        if (!TaskStepDetail.Status.PENDING.name().equals(step.getStatus())) {
+                        if (!TaskStepDetail.Status.PENDING.name().equals(step.getStatus())
+                                && !TaskStepDetail.Status.COMPLETED.name().equals(step.getStatus())) {
                             step.setStatus(TaskStepDetail.Status.PENDING.name());
                             step.setSuccessMessage("绛夊緟杩涚墖澶ц溅瑁呰浇鐜荤拑");
                             taskStepDetailMapper.updateById(step);
@@ -1056,7 +1277,17 @@
                     Long processStartTime = getProcessStartTime(context);
                     if (processStartTime == null) {
                         // 绗竴娆℃娴嬪埌鐜荤拑锛岃褰曞紑濮嬪鐞嗘椂闂�
-                        setProcessStartTime(context, System.currentTimeMillis());
+                        long now = System.currentTimeMillis();
+                        setProcessStartTime(context, now);
+                        // 琛ラ綈姝ラ鐨勫紑濮嬫椂闂翠笌鐘舵�侊紝纭繚鍓嶇鑰楁椂姝e父鏄剧ず
+                        if (!TaskStepDetail.Status.RUNNING.name().equals(step.getStatus())) {
+                            step.setStatus(TaskStepDetail.Status.RUNNING.name());
+                        }
+                        if (step.getStartTime() == null) {
+                            step.setStartTime(new Date(now));
+                        }
+                        taskStepDetailMapper.updateById(step);
+                        notificationService.notifyStepUpdate(task.getTaskId(), step);
                         log.debug("澶х悊鐗囩璁惧寮�濮嬪鐞�: taskId={}, deviceId={}, glassCount={}, processTime={}s",
                                 task.getTaskId(), device.getId(), loadedGlassIds.size(), processTimeSeconds);
                         return;
@@ -1082,8 +1313,21 @@
                     // 鏇存柊姝ラ鐘舵��
                     step.setStatus(TaskStepDetail.Status.COMPLETED.name());
                     step.setErrorMessage(null);
+                    // 璁板綍缁撴潫鏃堕棿涓庤�楁椂
+                    if (step.getEndTime() == null) {
+                        step.setEndTime(new Date());
+                    }
+                    if (step.getStartTime() == null) {
+                        // 濡傛灉寮�濮嬫椂闂寸己澶憋紝鐢ㄥ鐞嗗紑濮嬫椂闂存垨缁撴潫鏃堕棿鍏滃簳
+                        step.setStartTime(new Date(processStartTime));
+                    }
+                    if (step.getStartTime() != null && step.getEndTime() != null) {
+                        step.setDurationMs(step.getEndTime().getTime() - step.getStartTime().getTime());
+                    }
                     step.setOutputData(toJson(Collections.singletonMap("glassIds", loadedGlassIds)));
                     taskStepDetailMapper.updateById(step);
+                    // 閫氱煡鍓嶇鐘舵�佹洿鏂�
+                    notificationService.notifyStepUpdate(task.getTaskId(), step);
                     // 澶х悊鐗囩瀹屾垚鍚庡皾璇曡嚜鍔ㄦ敹灏炬暣涓换鍔�
                     checkAndCompleteTaskIfDone(step.getTaskId());
                     
@@ -1138,10 +1382,16 @@
     
     /**
      * 璁剧疆宸茶杞界殑鐜荤拑ID鍒楄〃
+     * 瀵逛簬鍒嗘壒鍑虹墖鍦烘櫙锛屾瘡娆¤缃簲鏇挎崲涓哄綋鍓嶆壒娆$殑鐜荤拑ID
      */
     private void setLoadedGlassIds(TaskExecutionContext context, List<String> glassIds) {
         if (context != null) {
-            context.getSharedData().put("loadedGlassIds", new ArrayList<>(glassIds));
+            // 鏇挎崲涓哄綋鍓嶆壒娆$殑鐜荤拑ID
+            List<String> currentBatch = new ArrayList<>();
+            if (glassIds != null) {
+                currentBatch.addAll(glassIds);
+            }
+            context.getSharedData().put("loadedGlassIds", currentBatch);
         }
     }
     
@@ -1184,6 +1434,33 @@
     private void clearProcessedGlassIds(TaskExecutionContext context) {
         if (context != null) {
             context.getSharedData().put("processedGlassIds", new ArrayList<>());
+        }
+    }
+    
+    /**
+     * 鑾峰彇宸插嚭鐗囩殑鐜荤拑ID鍒楄〃
+     */
+    @SuppressWarnings("unchecked")
+    private List<String> getOutboundGlassIds(TaskExecutionContext context) {
+        if (context == null) {
+            return Collections.emptyList();
+        }
+        Object glassIds = context.getSharedData().get("outboundGlassIds");
+        if (glassIds instanceof List) {
+            return new ArrayList<>((List<String>) glassIds);
+        }
+        return Collections.emptyList();
+    }
+    
+    /**
+     * 娣诲姞宸插嚭鐗囩殑鐜荤拑ID鍒楄〃
+     */
+    private void addOutboundGlassIds(TaskExecutionContext context, List<String> glassIds) {
+        if (context != null && glassIds != null && !glassIds.isEmpty()) {
+            List<String> existing = getOutboundGlassIds(context);
+            Set<String> allOutbound = new HashSet<>(existing);
+            allOutbound.addAll(glassIds);
+            context.getSharedData().put("outboundGlassIds", new ArrayList<>(allOutbound));
         }
     }
     
@@ -1393,6 +1670,8 @@
             
             // 鎵�鏈夋楠ら兘宸插畬鎴愶紝鏀跺熬浠诲姟
             task.setStatus(MultiDeviceTask.Status.COMPLETED.name());
+            // 鍏抽敭锛氬悓姝ヨ繘搴﹀埌鏈�缁堝�硷紝閬垮厤鍓嶇浠嶆樉绀衡��3/5鈥濊繖绫绘棫杩涘害
+            task.setCurrentStep(totalSteps);
             task.setEndTime(new Date());
             multiDeviceTaskMapper.updateById(task);
             
@@ -1438,6 +1717,9 @@
         }
         step.setOutputData(toJson(result));
         taskStepDetailMapper.updateById(step);
+        if (StringUtils.hasText(step.getTaskId())) {
+            notificationService.notifyStepUpdate(step.getTaskId(), step);
+        }
     }
     
     /**
@@ -1508,8 +1790,14 @@
     /**
      * 鏇存柊澶ц溅璁惧姝ラ鐘舵�侊紙淇濇寔RUNNING锛岀洿鍒版墜鍔ㄥ仠姝㈡垨浠诲姟鍙栨秷锛涘け璐ユ椂鏍囪涓篎AILED锛�
      */
-    private void updateStepStatusForVehicle(TaskStepDetail step, DevicePlcVO.OperationResult result) {
+    private void updateStepStatusForVehicle(String taskId, TaskStepDetail step, DevicePlcVO.OperationResult result) {
         if (step == null || result == null) {
+            return;
+        }
+        // 濡傛灉姝ラ宸茬粡澶勪簬瀹屾垚鎴栧け璐ョ姸鎬侊紝鍒欎笉鍐嶈閲嶅鏇存柊锛堥槻姝㈢姸鎬佸弽澶嶅垏鎹級
+        if (TaskStepDetail.Status.COMPLETED.name().equals(step.getStatus())
+                || TaskStepDetail.Status.FAILED.name().equals(step.getStatus())) {
+            log.debug("姝ラ宸插畬鎴愭垨澶辫触锛屼笉鍐嶆洿鏂扮姸鎬�: stepId={}, status={}", step.getId(), step.getStatus());
             return;
         }
         boolean success = Boolean.TRUE.equals(result.getSuccess());
@@ -1546,16 +1834,20 @@
         
         if (success && !completed) {
             // 鎴愬姛浣嗘湭瀹屾垚锛氭牴鎹畐aiting鐘舵�佸喅瀹氭樉绀轰负绛夊緟杩樻槸鎵ц涓�
-            if (waiting) {
+            // 娉ㄦ剰锛氬鏋滄楠ゅ凡缁忔槸RUNNING鐘舵�侊紙璇存槑宸茬粡瑁呰浇杩囩幓鐠冿級锛屼笉搴旇鏀瑰洖PENDING
+            boolean isAlreadyRunning = TaskStepDetail.Status.RUNNING.name().equals(step.getStatus());
+            if (waiting && !isAlreadyRunning) {
+                // 鍙湁鍦ㄨ繕娌℃湁寮�濮嬭繍琛屾椂锛屾墠璁剧疆涓篜ENDING
                 step.setStatus(TaskStepDetail.Status.PENDING.name());
             } else {
+                // 濡傛灉宸茬粡杩愯杩囷紝鎴栬�呬笉鏄瓑寰呯姸鎬侊紝淇濇寔鎴栬缃负RUNNING
                 step.setStatus(TaskStepDetail.Status.RUNNING.name());
             }
             String message = result.getMessage();
             if (!StringUtils.hasText(message) && waiting) {
                 message = "澶ц溅璁惧绛夊緟涓�" + (StringUtils.hasText(waitingReason) ? "锛�" + waitingReason + "锛�" : "");
             }
-            step.setSuccessMessage(StringUtils.hasText(message) ? message : (waiting ? "澶ц溅璁惧绛夊緟涓�" : "澶ц溅璁惧杩愯涓�"));
+            step.setSuccessMessage(StringUtils.hasText(message) ? message : (waiting && !isAlreadyRunning ? "澶ц溅璁惧绛夊緟涓�" : "澶ц溅璁惧杩愯涓�"));
             step.setErrorMessage(null);
             if (step.getStartTime() != null) {
                 step.setDurationMs(now.getTime() - step.getStartTime().getTime());
@@ -1589,6 +1881,8 @@
 
         step.setOutputData(toJson(result));
         taskStepDetailMapper.updateById(step);
+        // 閫氱煡鍓嶇姝ラ鐘舵�佸凡鏇存柊
+        notificationService.notifyStepUpdate(taskId, step);
     }
 
     /**
@@ -1610,7 +1904,9 @@
             // 鍏堟鏌ES浠诲姟锛堝鏋渕esSend=1锛屼細鍒涘缓浠诲姟骞跺紑濮嬫墽琛岋級
             DevicePlcVO.OperationResult mesTaskResult = null;
             try {
-                mesTaskResult = handler.execute(device, "checkMesTask", Collections.emptyMap());
+                Map<String, Object> taskParams = new HashMap<>();
+                taskParams.put("_taskContext", context);
+                mesTaskResult = handler.execute(device, "checkMesTask", taskParams);
                 if (mesTaskResult != null && Boolean.TRUE.equals(mesTaskResult.getSuccess())) {
                     log.info("澶ц溅璁惧宸叉鏌ES浠诲姟骞跺紑濮嬫墽琛�: taskId={}, deviceId={}, message={}",
                             task.getTaskId(), device.getId(), mesTaskResult.getMessage());
@@ -1623,15 +1919,56 @@
             // 鐒跺悗妫�鏌ES纭鐘舵��
             DevicePlcVO.OperationResult mesResult = null;
             try {
-                mesResult = handler.execute(device, "checkMesConfirm", Collections.emptyMap());
+                Map<String, Object> checkParams = new HashMap<>();
+                checkParams.put("_taskContext", context);
+                mesResult = handler.execute(device, "checkMesConfirm", checkParams);
             } catch (Exception e) {
                 log.warn("澶ц溅璁惧妫�鏌ES纭鐘舵�佸紓甯�: taskId={}, deviceId={}, error={}",
                         task.getTaskId(), device.getId(), e.getMessage());
             }
 
+            // 濡傛灉MES宸茬‘璁ゅ畬鎴愶紙mesConfirm=1锛夛紝妫�鏌ュ崸杞珛璁惧鐘舵�佸拰鐜荤拑淇℃伅
+            // 濡傛灉鍗ц浆绔嬪凡瀹屾垚涓旀墍鏈夌幓鐠冮兘宸茶杞斤紝鍙互鏍囪涓哄畬鎴�
+            if (mesResult != null && mesResult.getData() != null) {
+                Object completedFlag = mesResult.getData().get("completed");
+                boolean mesConfirmed = false;
+                if (completedFlag instanceof Boolean) {
+                    mesConfirmed = (Boolean) completedFlag;
+                } else if (completedFlag != null) {
+                    mesConfirmed = "true".equalsIgnoreCase(String.valueOf(completedFlag));
+                }
+                
+                if (mesConfirmed) {
+                    // MES宸茬‘璁ゅ畬鎴愶紝妫�鏌ュ崸杞珛璁惧鏄惁宸插畬鎴�
+                    boolean transferCompleted = isTransferDeviceCompleted(task.getTaskId(), context);
+                    if (transferCompleted) {
+                        // 妫�鏌ヤ换鍔′笂涓嬫枃涓殑鍒濆鐜荤拑ID鍜屽凡瑁呰浇鐨勭幓鐠僆D
+                        @SuppressWarnings("unchecked")
+                        List<String> initialGlassIds = (List<String>) context.getSharedData().get("initialGlassIds");
+                        List<String> loadedGlassIds = getLoadedGlassIds(context);
+                        
+                        if (initialGlassIds != null && !initialGlassIds.isEmpty() 
+                                && loadedGlassIds != null && !loadedGlassIds.isEmpty()) {
+                            // 妫�鏌ユ槸鍚︽墍鏈夊垵濮嬬幓鐠冮兘宸茶杞�
+                            boolean allGlassesLoaded = loadedGlassIds.containsAll(initialGlassIds);
+                            if (allGlassesLoaded) {
+                                // 鍗ц浆绔嬪凡瀹屾垚涓旀墍鏈夌幓鐠冮兘宸茶杞斤紝鏍囪涓哄畬鎴�
+                                log.info("MES宸茬‘璁や笖鍗ц浆绔嬪凡瀹屾垚涓旀墍鏈夌幓鐠冨凡瑁呰浇锛屼换鍔¤嚜鍔ㄥ畬鎴�: taskId={}, deviceId={}, initialCount={}, loadedCount={}",
+                                        task.getTaskId(), device.getId(), initialGlassIds.size(), loadedGlassIds.size());
+                                // mesResult宸茬粡鍖呭惈completed=true锛屼笉闇�瑕佷慨鏀�
+                            }
+                        }
+                    } else {
+                        // 鍗ц浆绔嬭繕鏈畬鎴愶紝涓嶅簲璇ユ爣璁颁负瀹屾垚
+                        log.debug("MES宸茬‘璁や絾鍗ц浆绔嬫湭瀹屾垚锛岀瓑寰呭崸杞珛瀹屾垚: taskId={}, deviceId={}", 
+                                task.getTaskId(), device.getId());
+                    }
+                }
+            }
+
             // 鏇存柊姝ラ鐘舵��
             if (mesResult != null) {
-                updateStepStatusForVehicle(step, mesResult);
+                updateStepStatusForVehicle(task.getTaskId(), step, mesResult);
                 boolean opSuccess = Boolean.TRUE.equals(mesResult.getSuccess());
                 updateTaskProgress(task, step.getStepOrder(), opSuccess);
                 if (!opSuccess) {
@@ -2455,6 +2792,140 @@
         }
     }
 
+    /**
+     * 褰撹繘鐗囧ぇ杞︽楠ゅ畬鎴愬悗锛屽鏋滃瓨鍦ㄥぇ鐞嗙墖绗艰澶囦笖灏氭湭鍚姩鍏跺畾鏃跺櫒锛屽垯鍚姩澶х悊鐗囩瀹氭椂鍣�
+     * 淇濊瘉鎵ц椤哄簭涓猴細杩涚墖澶ц溅 -> 澶х悊鐗囩
+     */
+    private void startLargeGlassTimerIfNeeded(MultiDeviceTask task, TaskExecutionContext context) {
+        if (task == null || context == null) {
+            return;
+        }
+        // 闃叉閲嶅鍚姩
+        Object startedFlag = context.getSharedData().get("largeGlassTimerStarted");
+        if (startedFlag instanceof Boolean && (Boolean) startedFlag) {
+            return;
+        }
+
+        try {
+            // 浠庝笂涓嬫枃涓幏鍙栬澶囧垪琛�
+            @SuppressWarnings("unchecked")
+            List<DeviceConfig> devices = (List<DeviceConfig>) context.getSharedData().get("devices");
+            if (devices == null || devices.isEmpty()) {
+                return;
+            }
+
+            DeviceConfig largeGlassDevice = null;
+            for (DeviceConfig device : devices) {
+                if (DeviceConfig.DeviceType.LARGE_GLASS.equals(device.getDeviceType())) {
+                    largeGlassDevice = device;
+                    break;
+                }
+            }
+            if (largeGlassDevice == null) {
+                log.warn("鏈湪璁惧鍒楄〃涓壘鍒板ぇ鐞嗙墖绗艰澶�: taskId={}", task.getTaskId());
+                return;
+            }
+
+            // 閲嶆柊鍔犺浇澶х悊鐗囩姝ラ锛岀‘淇濇嬁鍒版渶鏂扮姸鎬侊紙鍙栬璁惧鏈�鏂扮殑涓�鏉℃楠よ褰曪級
+            List<TaskStepDetail> largeGlassSteps = taskStepDetailMapper.selectList(
+                    Wrappers.<TaskStepDetail>lambdaQuery()
+                            .eq(TaskStepDetail::getTaskId, task.getTaskId())
+                            .eq(TaskStepDetail::getDeviceId, largeGlassDevice.getId())
+                            .orderByDesc(TaskStepDetail::getStepOrder)
+                            .last("LIMIT 1")
+            );
+            TaskStepDetail largeGlassStep = (largeGlassSteps != null && !largeGlassSteps.isEmpty())
+                    ? largeGlassSteps.get(0)
+                    : null;
+            if (largeGlassStep == null) {
+                log.warn("鏈壘鍒板ぇ鐞嗙墖绗兼楠よ褰�: taskId={}, deviceId={}", task.getTaskId(), largeGlassDevice.getId());
+                return;
+            }
+
+            // 濡傛灉姝ラ宸茬粡瀹屾垚鎴栧け璐ワ紝鍒欎笉鍐嶅惎鍔ㄥ畾鏃跺櫒
+            String status = largeGlassStep.getStatus();
+            if (TaskStepDetail.Status.COMPLETED.name().equals(status)
+                    || TaskStepDetail.Status.FAILED.name().equals(status)) {
+                log.debug("澶х悊鐗囩姝ラ宸茬粨鏉燂紝涓嶅啀鍚姩瀹氭椂鍣�: taskId={}, stepId={}, status={}", task.getTaskId(), largeGlassStep.getId(), status);
+                context.getSharedData().put("largeGlassTimerStarted", true);
+                return;
+            }
+
+            ScheduledFuture<?> largeGlassTask = startLargeGlassTimer(task, largeGlassStep, largeGlassDevice, context);
+            if (largeGlassTask != null) {
+                registerScheduledTask(task.getTaskId(), largeGlassTask);
+                context.getSharedData().put("largeGlassTimerStarted", true);
+                log.info("宸插湪杩涚墖澶ц溅瀹屾垚鍚庡惎鍔ㄥぇ鐞嗙墖绗煎畾鏃跺櫒: taskId={}, largeGlassDeviceId={}, largeGlassStepId={}",
+                        task.getTaskId(), largeGlassDevice.getId(), largeGlassStep.getId());
+            } else {
+                log.warn("鍦ㄨ繘鐗囧ぇ杞﹀畬鎴愬悗鍚姩澶х悊鐗囩瀹氭椂鍣ㄥけ璐�: taskId={}, largeGlassDeviceId={}, largeGlassStepId={}",
+                        task.getTaskId(), largeGlassDevice.getId(), largeGlassStep.getId());
+            }
+        } catch (Exception e) {
+            log.warn("鍦ㄨ繘鐗囧ぇ杞﹀畬鎴愬悗鍚姩澶х悊鐗囩瀹氭椂鍣ㄥ紓甯�: taskId={}", task.getTaskId(), e);
+        }
+    }
+
+    /**
+     * 妫�鏌ュ崸杞珛璁惧鏄惁宸插畬鎴�
+     * 杩斿洖true琛ㄧず鍗ц浆绔嬪凡瀹屾垚锛圕OMPLETED锛夛紝鍙互鍒ゆ柇澶ц溅鏄惁瀹屾垚
+     * 杩斿洖false琛ㄧず鍗ц浆绔嬭繕鍦ㄨ繍琛屼腑锛圧UNNING锛夋垨绛夊緟涓紙PENDING锛夛紝涓嶅簲璇ユ爣璁板ぇ杞︿负瀹屾垚
+     */
+    private boolean isTransferDeviceCompleted(String taskId, TaskExecutionContext context) {
+        if (taskId == null || context == null) {
+            return false;
+        }
+        try {
+            // 浠庝笂涓嬫枃涓幏鍙栬澶囧垪琛�
+            @SuppressWarnings("unchecked")
+            List<DeviceConfig> devices = (List<DeviceConfig>) context.getSharedData().get("devices");
+            if (devices == null || devices.isEmpty()) {
+                return false;
+            }
+            
+            // 鏌ユ壘鍗ц浆绔嬭澶�
+            DeviceConfig transferDevice = null;
+            for (DeviceConfig device : devices) {
+                if (DeviceConfig.DeviceType.WORKSTATION_TRANSFER.equals(device.getDeviceType())) {
+                    transferDevice = device;
+                    break;
+                }
+            }
+            
+            if (transferDevice == null) {
+                // 娌℃湁鍗ц浆绔嬭澶囷紝杩斿洖true锛堜笉褰卞搷鍒ゆ柇锛�
+                return true;
+            }
+            
+            // 鏌ユ壘鍗ц浆绔嬭澶囩殑姝ラ锛堝簲璇ュ彧鏈変竴涓楠わ級
+            List<TaskStepDetail> transferSteps = taskStepDetailMapper.selectList(
+                Wrappers.<TaskStepDetail>lambdaQuery()
+                    .eq(TaskStepDetail::getTaskId, taskId)
+                    .eq(TaskStepDetail::getDeviceId, transferDevice.getId())
+                    .orderByDesc(TaskStepDetail::getStepOrder)
+                    .last("LIMIT 1")
+            );
+            
+            if (transferSteps == null || transferSteps.isEmpty()) {
+                // 娌℃湁鎵惧埌姝ラ锛岃繑鍥瀎alse锛堝崸杞珛鍙兘杩樻病寮�濮嬶級
+                return false;
+            }
+            
+            // 妫�鏌ユ楠ょ姸鎬侊細鍙湁COMPLETED鎵嶇畻瀹屾垚锛孯UNNING鎴朠ENDING閮戒笉绠楀畬鎴�
+            TaskStepDetail transferStep = transferSteps.get(0);
+            String status = transferStep.getStatus();
+            boolean isCompleted = TaskStepDetail.Status.COMPLETED.name().equals(status);
+            
+            log.debug("妫�鏌ュ崸杞珛璁惧鐘舵��: taskId={}, deviceId={}, status={}, isCompleted={}", 
+                    taskId, transferDevice.getId(), status, isCompleted);
+            
+            return isCompleted;
+        } catch (Exception e) {
+            log.warn("妫�鏌ュ崸杞珛璁惧鐘舵�佸け璐�: taskId={}", taskId, e);
+            return false;
+        }
+    }
+
     private String determineOperation(DeviceConfig device, Map<String, Object> params) {
         if (params != null && params.containsKey("operation")) {
             Object op = params.get("operation");

--
Gitblit v1.8.0