1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
package com.dy.common.mw.protocol.p206V2.parse;
 
import com.dy.common.mw.protocol.*;
import com.dy.common.mw.protocol.p206V2.*;
import com.dy.common.mw.protocol.p206V2.parse.global.GlParse;
import com.dy.common.mw.protocol.p206V2.upVos.DataCd83CloseVo;
import com.dy.common.mw.protocol.p206V2.upVos.DataCd83OpenVo;
import com.dy.common.mw.protocol.rtuState.ValveStateInfo;
import com.dy.common.util.ByteUtil;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
 
@AnnotationCodeUp(ifAny={
        CodeV2.cd_83
})
@SuppressWarnings("unused")
public class Cd_83_Up implements CodeParse {
 
    private static final Logger log = LogManager.getLogger(Cd_83_Up.class);
 
    /**
     * 分析上行数据
     */
    @Override
    public MidResult[] parse(Boolean isLowPower, CodeParseParams params, CodeParseCallback callback)throws Exception {
        ParseParamsForUpV2 para = (ParseParamsForUpV2)params ;
        int bsLen = new CommonV2().parseDataLen4P206(para.upBuffer) ;
        if(bsLen > 0){
            this.doParse(para.upBuffer,
                    bsLen,
                    para.upCode,
                    para.data) ;
        }
        log.info("\n分析上行数据<" + CodeV2.getCodeName(para.upCode) + " RTU地址=" + para.rtuAddr + ">:\n" + para.data.toString());
 
        MidResultFromRtu midRs = new MidResultFromRtu() ;
        midRs.protocolName = para.protocolName ;//协议名称
        midRs.protocolVersion = para.protocolVersion ;//协议版本号
        midRs.rtuAddr = para.rtuAddr ;//Rtu地址
        midRs.upCode = para.upCode ;//上行数据中的功能码
        midRs.upHex = para.upHex ;//上行数据十六进制形式
        midRs.upBuffer = para.upBuffer ;//上行数据字节数组
        midRs.data = para.data ;//解析后的数据
 
        midRs.reportOrResponse_trueOrFalse = true ;//主动上报
 
        String confirmComCode = para.upCode ;
        ParseParamsForDownV2 downCpParams  = new ParseParamsForDownV2() ;
        downCpParams.setValue(
                null,
                ProtocolConstantV206V2.protocolName,
                para.protocolVersion,
                para.rtuAddr,
                Command.defaultId,
                confirmComCode,
                null,
                null);
        //构造应答
        byte[] data = new Cd_83_Down().doParse(downCpParams) ;
 
        MidResultToRtu confirmCommand = new MidResultToRtu() ;
        confirmCommand.protocolName = para.protocolName ;//协议名称
        confirmCommand.protocolVersion = para.protocolVersion ;//协议版本号
        confirmCommand.rtuAddr = para.rtuAddr ;//Rtu地址
        confirmCommand.commandId = Command.defaultId ;//命令ID,发起命令的客户端(web端)生成,以匹配命令结果
        confirmCommand.downCode = confirmComCode ;//下行命令功能码;
        confirmCommand.downCodeName = CodeV2.getCodeName(confirmComCode) ;//下行命令功能码名称;
        confirmCommand.downBuffer = data ;//下行命令数据
        confirmCommand.downBufHex = ByteUtil.bytes2Hex(data, true) ;//下行命令数据十六进制形式
        confirmCommand.hasResponse = false ;//是否有应答
        confirmCommand.maxSendTimes = 1 ;//命令最大发送次数(当收不到应答时,将重发2次),如果不设置,命令缓存器进行补充设置
        confirmCommand.isCachForOffLine = false ;//RTU不在线,命令是否缓存
 
        confirmCommand.isSendFirst = true ;//确认命令,优先发送
        if(isLowPower != null && isLowPower.booleanValue()){
            //低功耗时,尽快发送
            confirmCommand.isQuickSend = true ;
        }
 
        callback.callback(midRs.reportOrResponse_trueOrFalse, para.data.subData==null?null:((DataV2)(para.data.subData)).subData);
        if(para.data != null && para.data.getSubData() != null){
            Object subData = ((DataV2)para.data.getSubData()).subData ;
            if(subData instanceof DataCd83OpenVo){
                callback.notify(new ValveStateInfo(true));
            }else if(subData instanceof DataCd83CloseVo){
                callback.notify(new ValveStateInfo(false));
            }
        }
        return new MidResult[]{confirmCommand, midRs} ;
    }
    /**
     * 执行分析
     * @param bs 字节数组
     * @param bsLen 字节长度(总包长,包括包头和包尾)
     * @param dataCode 功能码
     * @param data 数据
     * @throws Exception 异常
     */
    protected void doParse(byte[] bs, int bsLen, String dataCode, Data data) throws Exception {
        byte opType = (byte)ByteUtil.BCD2Int_LE(bs[ProtocolConstantV206V2.dataIndex]) ;
        Boolean isCloseType = CommonV2.isCloseValveType(opType) ;
        if(isCloseType != null && isCloseType.booleanValue()){
            this.doParseClose(opType, bs, bsLen, dataCode, data);
        }else if(isCloseType != null && !isCloseType.booleanValue()){
            this.doParseOpen(opType, bs, bsLen, dataCode, data);
        }else{
            throw new Exception("开关阀类型[" + ByteUtil.bytes2Hex(new byte[]{opType}, false) + "(hex)]不可识别" ) ;
        }
    }
    private void doParseOpen(byte opType, byte[] bs, int bsLen, String dataCode, Data data) throws Exception {
        DataV2 dV2 = (DataV2)data.getSubData() ;
        DataCd83OpenVo cdData = new DataCd83OpenVo() ;
        dV2.subData = cdData ;
 
        cdData.type = opType ;
 
        short index = ProtocolConstantV206V2.dataIndex + 1 ;
        //累计流量:5字节BCD码,取值范围0~99999999.99,单位为m3。
        int tpInt = ByteUtil.BCD2Int_LE(bs, index, index + 4) ;
        cdData.totalAmount = tpInt/100.0 ;
 
        index += 5 ;
        //用水户号数据格式:8字节低位在前高位在后。
        cdData.icCardNo = GlParse.parseIcCardNo(bs, index) ;
 
        index += 8 ;
        //IC卡号格式:4字节HEX码低位在前高位在后。
        cdData.icCardAddr = ByteUtil.bytes2Hex_LE(bs, false,  index, 4) ;
 
        index += 4 ;
        //用水户余额:用户余额4字节BCD码,取值范围0.00~999999.99,单位为元。
        tpInt = ByteUtil.BCD2Int_LE(bs, index, index + 3) ;
        cdData.remainMoney = tpInt/100.0 ;
 
        index += 4 ;
        //用水户用水开始时间:6字节BCD码,顺序是年月日时分秒,其中公元年=2000+年。
        cdData.openDt = GlParse.parseTp(bs, index) ;
 
        //index += 6 ;
        //控制器时钟
        //cdData.rtuDt = GlParse.parseTp(bs, index) ;
        //2024-10-27 刘润玉:苏有勋把协议中的控制器时钟给删除掉了(目的是节约存储空间)
        //处理办法是把开阀时间作为控制器时钟,这两个时间相差不到一分钟
        cdData.rtuDt = cdData.openDt;
        //index += 6 ;
    }
 
    private void doParseClose(byte opType, byte[] bs, int bsLen, String dataCode, Data data) throws Exception {
        DataV2 dV2 = (DataV2)data.getSubData() ;
        DataCd83CloseVo cdData = new DataCd83CloseVo() ;
        dV2.subData = cdData ;
 
        cdData.type = opType ;
 
        short index = ProtocolConstantV206V2.dataIndex + 1 ;
        //累计流量:5字节BCD码,取值范围0~9999999999,单位为m3。
        int tpInt = ByteUtil.BCD2Int_LE(bs, index, index + 4) ;
        cdData.totalAmount = tpInt/100.0 ;
 
        index += 5 ;
        //用水户号数据格式:8字节低位在前高位在后。
        cdData.icCardNo = GlParse.parseIcCardNo(bs, index) ;
 
        index += 8 ;
        //IC卡号格式:4字节HEX码低位在前高位在后。
        cdData.icCardAddr = ByteUtil.bytes2Hex_LE(bs, false,  index, 4) ;
 
        index += 4 ;
        //用水户余额:用户余额4字节BCD码,取值范围0.00~999999.99,单位为元。
        tpInt = ByteUtil.BCD2Int_LE(bs, index, index + 3) ;
        cdData.remainMoney = tpInt/100.0 ;
 
        index += 4 ;
        //用水户用水开始时间:6字节BCD码,顺序是年月日时分秒,其中公元年=2000+年。
        cdData.openDt = GlParse.parseTp(bs, index) ;
 
        index += 6 ;
        //用水户用水结束时间:6字节BCD码,顺序是年月日时分秒,其中公元年=2000+年。
        cdData.closeDt = GlParse.parseTp(bs, index) ;
 
        index += 6 ;
        //用水户本次用水量:累计流量5字节BCD码,取值范围0~9999999999,单位为m3。
        tpInt = ByteUtil.BCD2Int_LE(bs, index, index + 4) ;
        cdData.thisAmount = tpInt/100.0 ;
 
        index += 5 ;
        //用水户本次消费金额:用户余额4字节BCD码,取值范围0.00~999999.99,单位为元。
        tpInt = ByteUtil.BCD2Int_LE(bs, index, index + 3) ;
        cdData.thisMoney = tpInt/100.0 ;
 
        index += 4 ;
        //用水户本次用水时长:用水时长2字节BCD码,取值范围0~9999,单位为分钟。
        tpInt = ByteUtil.BCD2Int_LE(bs, index, index + 1) ;
        cdData.thisTime = tpInt;
 
        index += 2 ;
        cdData.priceType = bs[index] ;
 
        index++ ;
        tpInt = ByteUtil.BCD2Int_LE(bs, index, index + 1) ;
        cdData.price = tpInt/100.0 ;
 
        index += 2 ;
        cdData.cardType = bs[index] ;
 
        //index++ ;
        //控制器时钟
        //cdData.rtuDt = GlParse.parseTp(bs, index) ;
        //2024-10-27 刘润玉:苏有勋把协议中的控制器时钟给删除掉了(目的是节约存储空间)
        //处理办法是把关阀时间作为控制器时钟,这两个时间相差不到一分钟
        cdData.rtuDt = cdData.closeDt ;
        //index += 6 ;
    }
 
    public static void main(String[] args) throws Exception {
        Cd_83_Up obj = new Cd_83_Up() ;
        //下面两条上报数据,IC卡编码都是非BCD编码而异常
        //String hex = "683C68B08485353448830200000000001000282353FE739444000001000313000101211615000101210000000000000000000200019000011518000101210A7B16";
        String hex = "683C68B05301154CEA8306001000000004343638483BBBB9E0001000001603000101215907000101210000000000000000000500019000015308000101210AF716";
        byte[] bs = ByteUtil.hex2Bytes(hex) ;
 
        Data data = new Data() ;
        data.setSubData(new DataV2()) ;
 
        int bsLen = new CommonV2().parseDataLen4P206(bs) ;
        if(bsLen > 0){
            try {
            obj.doParse(bs,
                        bsLen,
                        "83",
                        data) ;
            }catch (Exception e){
                e.printStackTrace();
            }
        }
        System.out.println("data = " + data);
    }
}