| New file | 
|  |  |  | 
|---|
|  |  |  | package com.dy.common.mw.protocol.p206V2; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | import com.dy.common.mw.channel.tcp.TcpIoSessionAttrIdIsRtuAddr; | 
|---|
|  |  |  | import com.dy.common.mw.protocol.p206V1.ProtocolConstantV206V1; | 
|---|
|  |  |  | import com.dy.common.util.ByteUtil; | 
|---|
|  |  |  | import com.dy.common.util.ByteUtilUnsigned; | 
|---|
|  |  |  | import com.dy.common.util.CRC16; | 
|---|
|  |  |  | import com.dy.common.util.CRC8_for_2_0; | 
|---|
|  |  |  | import org.apache.mina.core.session.IoSession; | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public class CommonV2 { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 在Io会话中设置协议名称及版本号 | 
|---|
|  |  |  | * @param ioSession 会话 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public void setThisProtocolArr2IoSession(IoSession ioSession){ | 
|---|
|  |  |  | ioSession.setAttribute(TcpIoSessionAttrIdIsRtuAddr.sessionArrProtocolName, ProtocolConstantV206V1.protocolName) ; | 
|---|
|  |  |  | ioSession.setAttribute(TcpIoSessionAttrIdIsRtuAddr.sessionArrProtocolName, ProtocolConstantV206V1.protocolVer) ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 检查头 | 
|---|
|  |  |  | * @param bs 上行字节数组 | 
|---|
|  |  |  | * @return true是,false否 | 
|---|
|  |  |  | * @throws Exception  异常 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public Boolean[] isThisProtocolHead(byte[] bs) throws Exception{ | 
|---|
|  |  |  | if(bs == null){ | 
|---|
|  |  |  | return null ; | 
|---|
|  |  |  | }else if(bs.length >= (ProtocolConstantV206V2.ctrlIndex) | 
|---|
|  |  |  | && bs[ProtocolConstantV206V2.headFlag1Index] == ProtocolConstantV206V2.P_Head_Byte | 
|---|
|  |  |  | && bs[ProtocolConstantV206V2.headFlag2Index] == ProtocolConstantV206V2.P_Head_Byte){ | 
|---|
|  |  |  | Short vs = this.parseVersion(bs) ; | 
|---|
|  |  |  | if(vs.shortValue() == ProtocolConstantV206V2.protocolVer){ | 
|---|
|  |  |  | return new Boolean[]{true, true}; | 
|---|
|  |  |  | }else{ | 
|---|
|  |  |  | return new Boolean[]{false, true}; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }else if(bs.length >= (ProtocolConstantV206V2.UG_codeIndex) | 
|---|
|  |  |  | && bs[0] == ProtocolConstantV206V2.UG_P_Head_Byte | 
|---|
|  |  |  | && bs[3] == ProtocolConstantV206V2.UG_P_Head_Byte){ | 
|---|
|  |  |  | return new Boolean[]{true, false} ; | 
|---|
|  |  |  | }else{ | 
|---|
|  |  |  | return new Boolean[]{false, false} ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 检查协议类型 | 
|---|
|  |  |  | * @param bs 上行字节数组 | 
|---|
|  |  |  | * @return 协议类型 | 
|---|
|  |  |  | * @throws Exception 异常 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public Boolean protocolType_p206TrueUgFalse(byte[] bs){ | 
|---|
|  |  |  | if(bs == null){ | 
|---|
|  |  |  | return null ; | 
|---|
|  |  |  | }else if(bs.length >= (ProtocolConstantV206V2.ctrlIndex) | 
|---|
|  |  |  | && bs[0] == ProtocolConstantV206V2.P_Head_Byte | 
|---|
|  |  |  | && bs[2] == ProtocolConstantV206V2.P_Head_Byte){ | 
|---|
|  |  |  | return true ; | 
|---|
|  |  |  | }else if(bs.length >= (ProtocolConstantV206V2.UG_codeIndex) | 
|---|
|  |  |  | && bs[0] == ProtocolConstantV206V2.UG_P_Head_Byte | 
|---|
|  |  |  | && bs[3] == ProtocolConstantV206V2.UG_P_Head_Byte){ | 
|---|
|  |  |  | return false ; | 
|---|
|  |  |  | }else{ | 
|---|
|  |  |  | return null ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 分析版本号 | 
|---|
|  |  |  | * @param bs | 
|---|
|  |  |  | * @return | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public Short parseVersion(byte[] bs)throws Exception{ | 
|---|
|  |  |  | return ByteUtil.byte2PlusInt(bs[ProtocolConstantV206V2.versionIndex]) ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 检查尾 | 
|---|
|  |  |  | * @param bs 上行字节数组 | 
|---|
|  |  |  | * @throws Exception 异常 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public void checkTail(byte[] bs) throws Exception{ | 
|---|
|  |  |  | if(bs.length < 1 || bs[bs.length - 1] != ProtocolConstantV206V2.P_Tail_Byte){ | 
|---|
|  |  |  | throw new Exception("上行数据尾不正确!") ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 分析帧长度 | 
|---|
|  |  |  | * @param bs 上行字节数组 | 
|---|
|  |  |  | * @param p206TrueUgFalse 206协议为true,升级协议为false | 
|---|
|  |  |  | * @return 数据长度 | 
|---|
|  |  |  | * @throws Exception 异常 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public int parseFrameLen(byte[] bs, boolean p206TrueUgFalse)throws Exception{ | 
|---|
|  |  |  | if(p206TrueUgFalse) { | 
|---|
|  |  |  | int len = ByteUtilUnsigned.byte2Byte(bs, ProtocolConstantV206V2.dataLenIndex); | 
|---|
|  |  |  | return len + ProtocolConstantV206V2.lenHead2ctrl + ProtocolConstantV206V2.lenTail; | 
|---|
|  |  |  | }else{ | 
|---|
|  |  |  | int len = ByteUtilUnsigned.bytes2Short_BE(bs, ProtocolConstantV206V2.UG_dataLenIndex_start); | 
|---|
|  |  |  | return len + ProtocolConstantV206V2.UG_lenHead2Cmd + ProtocolConstantV206V2.UG_lenTail; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 分析用户数据域字节数(默认是非升级协议) | 
|---|
|  |  |  | * @param bs 上行字节数组 | 
|---|
|  |  |  | * @return 数据长度 | 
|---|
|  |  |  | * @throws Exception 异常 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public int parseDataLen4P206(byte[] bs)throws Exception{ | 
|---|
|  |  |  | int len = ByteUtilUnsigned.byte2Byte(bs, ProtocolConstantV206V2.dataLenIndex) ; | 
|---|
|  |  |  | return len - ProtocolConstantV206V2.lenCtrl - ProtocolConstantV206V2.lenRtuAddr ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 分析用户数据域字节数(升级协议) | 
|---|
|  |  |  | * @param bs 上行字节数组 | 
|---|
|  |  |  | * @return 数据长度 | 
|---|
|  |  |  | * @throws Exception 异常 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public int parseDataLen4Ug(byte[] bs)throws Exception{ | 
|---|
|  |  |  | int len = ByteUtilUnsigned.bytes2Short_LE(bs, ProtocolConstantV206V2.UG_dataLenIndex_start) ; | 
|---|
|  |  |  | return len - ProtocolConstantV206V2.UG_lenCmd - ProtocolConstantV206V2.UG_lenRtuAddr ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 分析Rtu地址 | 
|---|
|  |  |  | * @param bs 上行字节数组 | 
|---|
|  |  |  | * @return 控制器地址 | 
|---|
|  |  |  | * @throws Exception 异常 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public String parseRtuAddr(byte[] bs)throws Exception{ | 
|---|
|  |  |  | String rtuAddrBCD = "" + ByteUtil.BCD2Long_BE(bs, ProtocolConstantV206V2.rtuAddr1Index_start, ProtocolConstantV206V2.rtuAddr1Index_end) ; | 
|---|
|  |  |  | String rtuAddrStr = "" + ByteUtilUnsigned.bytes2Short_LE(bs, ProtocolConstantV206V2.rtuAddr2Index_start) ; | 
|---|
|  |  |  | while(rtuAddrStr.length() <= 5){ | 
|---|
|  |  |  | rtuAddrStr = "0" + rtuAddrStr ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return rtuAddrBCD + rtuAddrStr ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 分析Rtu地址 | 
|---|
|  |  |  | * @param bs 上行字节数组 | 
|---|
|  |  |  | * @param index 启始位 | 
|---|
|  |  |  | * @return 控制器地址 | 
|---|
|  |  |  | * @throws Exception 异常 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public String parseRtuAddr(byte[] bs, int index)throws Exception{ | 
|---|
|  |  |  | String rtuAddrBCD = "" + ByteUtil.BCD2Long_BE(bs, index, index + 2) ;//地址是大端模式 | 
|---|
|  |  |  | String rtuAddrStr = "" + ByteUtilUnsigned.bytes2Short_LE(bs, index + 3) ; | 
|---|
|  |  |  | while(rtuAddrStr.length() <= 5){ | 
|---|
|  |  |  | rtuAddrStr = "0" + rtuAddrStr ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return rtuAddrBCD + rtuAddrStr ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 分析功能码 | 
|---|
|  |  |  | * @param bs 上行字节数组 | 
|---|
|  |  |  | * @return 功能码 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public String parseCode(byte[] bs){ | 
|---|
|  |  |  | return ByteUtil.bytes2Hex(bs, false, ProtocolConstantV206V2.codeIndex, 1) ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 分析功能码 | 
|---|
|  |  |  | * @param bs 上行字节数组 | 
|---|
|  |  |  | * @return 功能码 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public String parseCode(byte[] bs, boolean p206TrueUgFalse){ | 
|---|
|  |  |  | if(p206TrueUgFalse) { | 
|---|
|  |  |  | return ByteUtil.bytes2Hex(bs, false, ProtocolConstantV206V1.codeIndex, 1); | 
|---|
|  |  |  | }else{ | 
|---|
|  |  |  | return ByteUtil.bytes2Hex(bs, false, ProtocolConstantV206V1.UG_codeIndex, 2); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 校验和检查 | 
|---|
|  |  |  | * @param bs  上行字节数组 | 
|---|
|  |  |  | * @return 通过null,未通过返回原因 | 
|---|
|  |  |  | * @throws Exception 异常 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public String checkCrc_str(byte[] bs) throws Exception { | 
|---|
|  |  |  | byte crcCompute = (byte)new CRC8_for_2_0().CRC8(bs, ProtocolConstantV206V2.ctrlIndex, bs.length - 3) ; | 
|---|
|  |  |  | byte crcInBs = bs[bs.length - 2] ; | 
|---|
|  |  |  | if(crcCompute == crcInBs){ | 
|---|
|  |  |  | return null ; | 
|---|
|  |  |  | }else{ | 
|---|
|  |  |  | return "计算CRC是:" + crcCompute + ",上传CRC是" + crcInBs ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 校验和检查 | 
|---|
|  |  |  | * @param bs  上行字节数组 | 
|---|
|  |  |  | * @param p206TrueUgFalse 206协议为true,升级协议为false | 
|---|
|  |  |  | * @return 通过null,未通过返回原因 | 
|---|
|  |  |  | * @throws Exception 异常 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public String checkCrc_str(byte[] bs, boolean p206TrueUgFalse) throws Exception { | 
|---|
|  |  |  | if(p206TrueUgFalse){ | 
|---|
|  |  |  | byte crcCompute = (byte)new CRC8_for_2_0().CRC8(bs, ProtocolConstantV206V1.ctrlIndex, bs.length - 3) ; | 
|---|
|  |  |  | byte crcInBs = bs[bs.length - 2] ; | 
|---|
|  |  |  | if(crcCompute == crcInBs){ | 
|---|
|  |  |  | return null ; | 
|---|
|  |  |  | }else{ | 
|---|
|  |  |  | return "计算CRC是:" + crcCompute + ",上传CRC是" + crcInBs ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }else{ | 
|---|
|  |  |  | short crcCompute = new CRC16().CRC(bs, 0, bs.length - 4) ; | 
|---|
|  |  |  | short crcInBs = ByteUtil.bytes2Short_BE(bs,bs.length - 3) ; | 
|---|
|  |  |  | //int crcInBs = ByteUtilUnsigned.bytes2Short_BE(bs, bs.length - 3) ; | 
|---|
|  |  |  | if(crcCompute == crcInBs){ | 
|---|
|  |  |  | return null ; | 
|---|
|  |  |  | }else{ | 
|---|
|  |  |  | return "计算CRC是:" + crcCompute + ",上传CRC是" + crcInBs ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | /* | 
|---|
|  |  |  | 构造控制域 | 
|---|
|  |  |  | D7                  D6                  D5~D4            D3~D0 | 
|---|
|  |  |  | 传输方向位 DIR        拆分标志位 DIV       帧计数位 FCB       功能码 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public byte createCtrl(byte dir, byte funcCode){ | 
|---|
|  |  |  | byte b = dir;//(byte)0x80//控制域:DIR=1,表示此帧报文是由终端发出的上行报文; | 
|---|
|  |  |  | b = (byte)(b | funcCode) ; | 
|---|
|  |  |  | //FCB == 3 | 
|---|
|  |  |  | b = (byte)(b | (byte)0x18) ; | 
|---|
|  |  |  | //DIV = 0 | 
|---|
|  |  |  | //DIR = 0 | 
|---|
|  |  |  | return b ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 得到IC卡类型名称 ( 卡类型(1:用户卡;2:管理员卡;3:调试卡;4:巡检卡;5:清空卡)) | 
|---|
|  |  |  | * @param type 字节 | 
|---|
|  |  |  | * @return 名称 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public static String icCardType(byte type){ | 
|---|
|  |  |  | return switch (type) { | 
|---|
|  |  |  | case 0 -> "无卡"; | 
|---|
|  |  |  | case 1 -> "用户卡"; | 
|---|
|  |  |  | case 2 -> "管理员卡"; | 
|---|
|  |  |  | case 3 -> "调试卡"; | 
|---|
|  |  |  | case 4 -> "巡检卡"; | 
|---|
|  |  |  | case 5 -> "清空卡"; | 
|---|
|  |  |  | default -> "未知"; | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /** | 
|---|
|  |  |  | * 得到关开阀类型名称 | 
|---|
|  |  |  | * 开关阀类型( | 
|---|
|  |  |  | * 1:刷卡开阀; | 
|---|
|  |  |  | * 2:刷卡关阀; | 
|---|
|  |  |  | * 3:中心站开阀; | 
|---|
|  |  |  | * 4:中心站关阀; | 
|---|
|  |  |  | * 5:余额不足关阀; | 
|---|
|  |  |  | * 6:流量计故障关阀; | 
|---|
|  |  |  | * 7:紧急关闭; | 
|---|
|  |  |  | * 8:用户远程开阀; | 
|---|
|  |  |  | * 9:用户远程关阀; | 
|---|
|  |  |  | * 10:巡检卡关阀; | 
|---|
|  |  |  | * 11:巡检卡刷卡卡开阀; | 
|---|
|  |  |  | * 12:黑名单命令关阀; | 
|---|
|  |  |  | * 13:远程定时关阀; | 
|---|
|  |  |  | * 14:远程定量关阀; | 
|---|
|  |  |  | * 16:管道无水自动关阀;(王江海协议) | 
|---|
|  |  |  | * ) | 
|---|
|  |  |  | * @param type 字节 | 
|---|
|  |  |  | * @return 名称 | 
|---|
|  |  |  | */ | 
|---|
|  |  |  | public static String openCloseValveType(Byte type){ | 
|---|
|  |  |  | if(type == null){ | 
|---|
|  |  |  | return "" ; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return switch (type) { | 
|---|
|  |  |  | case 1 -> "刷卡开阀"; | 
|---|
|  |  |  | case 2 -> "刷卡关阀"; | 
|---|
|  |  |  | case 3 -> "中心站开阀"; | 
|---|
|  |  |  | case 4 -> "中心站关阀"; | 
|---|
|  |  |  | case 5 -> "余额不足关阀"; | 
|---|
|  |  |  | case 6 -> "流量计故障关阀"; | 
|---|
|  |  |  | case 7 -> "紧急关阀"; | 
|---|
|  |  |  | case 8 -> "用户远程开阀"; | 
|---|
|  |  |  | case 9 -> "用户远程关阀"; | 
|---|
|  |  |  | case 10 -> "巡检卡关阀"; | 
|---|
|  |  |  | case 11 -> "巡检卡开阀"; | 
|---|
|  |  |  | case 12 -> "黑名单命令关阀"; | 
|---|
|  |  |  | case 13 -> "远程定时关阀"; | 
|---|
|  |  |  | case 14 -> "远程定量关阀"; | 
|---|
|  |  |  | case 16 -> "管道无水自动关阀"; | 
|---|
|  |  |  | default -> "未知"; | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | public static Boolean isCloseValveType(byte type){ | 
|---|
|  |  |  | return switch (type) { | 
|---|
|  |  |  | case 1, 3, 8, 11 -> false ; | 
|---|
|  |  |  | case 2, 4, 5, 6, 7, 9, 10, 12, 13, 14 -> true ; | 
|---|
|  |  |  | default -> null ; | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | } | 
|---|