| | |
| | | import com.dy.common.util.Callback; |
| | | import com.dy.common.util.DateTime; |
| | | import lombok.Data; |
| | | import org.apache.logging.log4j.LogManager; |
| | | import org.apache.logging.log4j.Logger; |
| | | |
| | | import java.util.*; |
| | | import java.util.concurrent.ConcurrentHashMap; |
| | | import java.util.concurrent.atomic.AtomicBoolean; |
| | | |
| | | /** |
| | | * @Author: liurunyu |
| | |
| | | */ |
| | | @Data |
| | | public class UpgradeTask { |
| | | |
| | | private static final Logger log = LogManager.getLogger(UpgradeTask.class.getName()); |
| | | |
| | | protected static final String TaskOverType_Natural = "自然" ; |
| | | protected static final String TaskOverType_Force = "强制" ; |
| | |
| | | protected byte[][] softFileDataGrp ;//以512字节为单位把升级程序数据分组 |
| | | |
| | | @JSONField(serialize = false) |
| | | protected Map<String, UpgradeRtu> upgradeRtus;//升级状态 |
| | | protected ConcurrentHashMap<String, UpgradeRtu> upgradeRtus;//升级状态 |
| | | |
| | | public boolean taskIsOver = false ;//任务是否完成 |
| | | public String taskOverType = "" ;//任务完成方式(自然,强制) |
| | |
| | | /////////////////////////////////////////////////// |
| | | //以下内部控制用 |
| | | @JSONField(serialize = false) |
| | | private int curUgRunningRtuTotal = 0 ;//当前正在升级的RTU个数 |
| | | protected int curUgRunningRtuTotal = 0 ;//当前正在升级的RTU个数 |
| | | |
| | | public UpgradeTask() { |
| | | this.curUgRunningRtuTotal = 0 ; |
| | |
| | | this.setupDtLong = System.currentTimeMillis() ; |
| | | this.taskVo = taskVo ; |
| | | |
| | | this.upgradeRtus = new HashMap<>(); |
| | | if(taskVo.softFileData64 != null && taskVo.softFileData64.trim().equals("")){ |
| | | this.upgradeRtus = new ConcurrentHashMap<>(); |
| | | if(taskVo.softFileData64 != null && !taskVo.softFileData64.trim().equals("")){ |
| | | taskVo.softFileData = Base64.getDecoder().decode(taskVo.softFileData64); |
| | | List<byte[]> listBytes = new HexFileParse().splitBytesByUnit512(taskVo.softFileData); |
| | | this.softFileDataGrp = listBytes.toArray(new byte[0][]); |
| | | for(String rtuAddr : this.taskVo.rtuAddrList){ |
| | | //此时状态设置成离线状态 |
| | | UpgradeRtuDev ugRtu = new UpgradeRtuDev(this, rtuAddr, this.softFileDataGrp.length, UpgradeRtuDev.STATE_OFFLINE) ; |
| | | this.upgradeRtus.put(rtuAddr, ugRtu) ; |
| | | try{ |
| | | for(String rtuAddr : this.taskVo.rtuAddrList){ |
| | | //此时状态设置成离线状态 |
| | | UpgradeRtuDev ugRtu = new UpgradeRtuDev(this.taskVo, this.failTryTimes, rtuAddr, this.softFileDataGrp.length, UpgradeRtuDev.STATE_OFFLINE) ; |
| | | this.upgradeRtus.put(rtuAddr, ugRtu) ; |
| | | } |
| | | }catch (Exception e){ |
| | | log.error(e); |
| | | } |
| | | } |
| | | } |
| | |
| | | //根据方法setTask的逻辑,只要RTU在升级之列,此处ugRtuState一定不为null |
| | | //为保险,实现下面逻辑 |
| | | if(taskVo.rtuAddrList.contains(rtuAddr)){ |
| | | ugRtu = new UpgradeRtuDev(this, rtuAddr, this.taskVo.softFileData.length) ; |
| | | ugRtu = new UpgradeRtuDev(this.taskVo, this.failTryTimes, rtuAddr, this.taskVo.softFileData.length) ; |
| | | upgradeRtus.put(rtuAddr, ugRtu) ; |
| | | }else{ |
| | | //rtu不在升级之列 |
| | | return ; |
| | | } |
| | | } |
| | | if(ugRtu != null){ |
| | | }else{ |
| | | if(ugRtu.isOver){ |
| | | //当前RTU已经升级完成,无需再升级 |
| | | return; |
| | |
| | | ugRtu.trigger(code, protocolName, protocolVersion, this.softFileDataGrp, callbackCom) ; |
| | | } |
| | | } |
| | | }else{ |
| | | //rtu不在升级之列 |
| | | return ; |
| | | } |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 强制结束升级任务 |
| | | */ |
| | |
| | | } |
| | | } |
| | | |
| | | |
| | | /** |
| | | * 当前升级状态 |
| | | * @return |
| | |
| | | if(this.taskVo.rtuAddrList != null && this.taskVo.rtuAddrList.size() > 0){ |
| | | state.rtuTotal = this.taskVo.rtuAddrList.size() ; |
| | | if(this.upgradeRtus != null && this.upgradeRtus.size() > 0){ |
| | | Collection<UpgradeRtu> col = this.upgradeRtus.values() ; |
| | | for(UpgradeRtu info : col){ |
| | | AtomicBoolean hasRunning = new AtomicBoolean(false); |
| | | this.upgradeRtus.values().forEach(info ->{ |
| | | if(info.state == UpgradeRtu.STATE_OFFLINE){ |
| | | state.offLineTotal ++ ; |
| | | state.offLineTotal++ ; |
| | | }else if(info.state == UpgradeRtu.STATE_UNSTART){ |
| | | state.unStartTotal ++ ; |
| | | }else if(info.state == UpgradeRtu.STATE_RUNNING){ |
| | |
| | | }else if(info.state == UpgradeRtu.STATE_SUCCESS) { |
| | | state.successTotal++; |
| | | }else if(info.state == UpgradeRtu.STATE_FAILONE) { |
| | | state.failOneTotal++; |
| | | state.failTotal++; |
| | | state.dieOneTotal++; |
| | | if(info.isOver){ |
| | | state.failTotal++; |
| | | } |
| | | }else if(info.state == UpgradeRtu.STATE_FAIL) { |
| | | state.dieMultiTotal++; |
| | | if(info.isOver) { |
| | | state.failTotal++; |
| | | } |
| | | }else if(info.state == UpgradeRtu.STATE_FAILOFFLINE) { |
| | | state.failTotal++; |
| | | state.failOffTotal++; |
| | | } |
| | | if(info.isOver){ |
| | | state.overTotal++; |
| | | }else{ |
| | | hasRunning.set(true); |
| | | } |
| | | }); |
| | | if(!hasRunning.get()){ |
| | | state.allOver = true ; |
| | | }else{ |
| | | state.allOver = false ; |
| | | } |
| | | } |
| | | } |
| | |
| | | } |
| | | |
| | | /////////////////////////////////////////////////////////// |
| | | //以下方法为内部服务,不对外提供服务 |
| | | // |
| | | // 以下方法为内部服务,不对外提供服务 |
| | | // |
| | | /////////////////////////////////////////////////////////// |
| | | |
| | | /** |
| | | * 统计当前正在升级的RTU数量,为受限同时升级数量做准备 |
| | | * 判断是否没用任何一个RTU进行过升级,而且超过了时限 |
| | | * @return -1:无一RTU升级且超时,0:无RTU升级但未超时等待,1有RTU升级正常执行 |
| | | */ |
| | | protected void statisticsRunningRtuCount(){ |
| | | protected int countNoOneRtuUpgradeInDuration(){ |
| | | if(this.upgradeRtus == null || upgradeRtus.size() == 0){ |
| | | //当前没有任何一个设备进行过升级 |
| | | Long now = System.currentTimeMillis() ; |
| | | if(now - this.setupDtLong > UpgradeUnit.confVo.noOneRtuUpgradeMaxDuration){ |
| | | return -1 ; |
| | | } |
| | | }else{ |
| | | Collection<UpgradeRtu> col = this.upgradeRtus.values() ; |
| | | for(UpgradeRtu info : col){ |
| | | if(info.currentPackage > 0){ |
| | | //当前有设备进行过升级 |
| | | return 1 ; |
| | | } |
| | | } |
| | | Long now = System.currentTimeMillis() ; |
| | | if(now - this.setupDtLong > UpgradeUnit.confVo.noOneRtuUpgradeMaxDuration){ |
| | | return -1 ; |
| | | } |
| | | } |
| | | return 0 ; |
| | | } |
| | | |
| | | /** |
| | | * 统计当前正在升级的RTU数量,为同时升级数量限制做准备 |
| | | */ |
| | | protected void countRunningRtuCount(){ |
| | | int runningTotal = 0 ; |
| | | Collection<UpgradeRtu> col = this.upgradeRtus.values() ; |
| | | for(UpgradeRtu info : col){ |
| | |
| | | } |
| | | |
| | | /** |
| | | * 统计需要升级但当前离线RTU的情况 |
| | | * 统计需要升级但当前离线RTU的情况,超过时限的设备为升级完成 |
| | | * @return -1:没有超时,0超时了且无因离线被强制设置升级完成的RTU,>0离线被强制设置升级完成的RTU数量 |
| | | */ |
| | | protected void statisticsOffRtuCountAndSet() { |
| | | protected int countOffRtuAndSetIfOver() { |
| | | Long now = System.currentTimeMillis() ; |
| | | if(now - this.setupDtLong > UpgradeUnit.confVo.rtuOffLineWaitDuration){ |
| | | //rtu离线,等待其升级的时长(毫秒),超过配置的最大时长,设置其升级失败,且设置升级任务完成 |
| | | int count = 0 ; |
| | | if (this.taskVo.rtuAddrList != null && this.taskVo.rtuAddrList.size() > 0) { |
| | | Collection<UpgradeRtu> col = this.upgradeRtus.values() ; |
| | | for(UpgradeRtu info : col){ |
| | | if(info.state == UpgradeRtu.STATE_OFFLINE){ |
| | | info.isOver = true ; |
| | | info.state = UpgradeRtu.STATE_FAILOFFLINE ; |
| | | count ++ ; |
| | | } |
| | | } |
| | | } |
| | | return count ; |
| | | }else{ |
| | | return -1 ; |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 统计是否升级全部结束 |
| | | */ |
| | | protected boolean statisticsIsAllOver() { |
| | | protected boolean countIsAllOver() { |
| | | if (this.taskVo.rtuAddrList != null && this.taskVo.rtuAddrList.size() > 0) { |
| | | Collection<UpgradeRtu> col = this.upgradeRtus.values() ; |
| | | for(UpgradeRtu info : col){ |