jiazx0107@163.com
2023-12-14 231cdeb9b7a55ba08fb8cffda35f5535b9292873
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
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
package com.fzzy.protocol.zldz.analysis;
 
import com.fzzy.api.utils.BytesUtil;
import com.fzzy.api.utils.ContextUtil;
import com.fzzy.gateway.data.BaseReqData;
import com.fzzy.gateway.entity.GatewayDevice;
import com.fzzy.io.request.BaseRequest;
import com.fzzy.protocol.ProtocolUtils;
import com.fzzy.protocol.zldz.data.ReMessage;
import com.fzzy.protocol.zldz.service.ZldzGatewayGrainService;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.Transactional;
 
import javax.annotation.Resource;
import javax.persistence.criteria.CriteriaBuilder;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
 
/**
 * 粮情解析
 *
 * @author Andy
 */
@Slf4j
@Component(AnalysisGrain.BEAN_ID)
public class AnalysisGrain {
 
    public static final String BEAN_ID = "zldz.analysisGrain";
 
    @Resource
    private ZldzGatewayGrainService zldzGatewayGrainService;
 
    public static Map<String, Map<String, String>> contextMap = new HashMap<>();
 
    public static double ERROR_CHECK_TAG = -100.0;
 
    public static double FAULT_CHECK_TAG = 85.0;
 
    public static double ERROR_CHECK_TAG2 = 50;
 
    public static double MAX_TEMP = -50.0;
    public static double MIN_TEMP = 50.0;
 
    /**
     * 后台发送请求后,终端的响应
     *
     * @param reMessage
     * @param device
     */
    public void analysis8815(ReMessage reMessage, GatewayDevice device) {
        BaseRequest request = new BaseRequest();
        request.setSerId(device.getId());
        request.setCompanyId(device.getOrgId());
        request.setIp(reMessage.getIp());
        request.setPort(reMessage.getPort());
        request.setSerName(device.getDeviceName());
 
        zldzGatewayGrainService.reply8815(request);
    }
 
    /**
     * 接收到粮情的回复
     *
     * @param reMessage
     * @param device
     */
    public void reply8817(ReMessage reMessage, GatewayDevice device) {
        String bodyMsg = reMessage.getBody().getContent();
 
        log.info("分机------>>>平台:收到的粮情信息-{}-{}",
                device.getDeviceName(), reMessage.getStrMsg());
 
        // Step1根据需要简化需要的信息
        String ua = bodyMsg.substring(0 * 2, 1 * 2);
        log.debug("获取到的电压信息,系统不保留={}", ua);
 
        // 总包数
        // int sumPacket = BytesUtil.hexToInt(bodyMsg.substring(1 * 2, 1 * 2 + 1
        // * 2));
        // 当前包
        String hexCurPacket = bodyMsg.substring(2 * 2, 2 * 2 + 1 * 2);
        // int curPacket = BytesUtil.hexToInt(hexCurPacket);
 
        BaseRequest request = new BaseRequest();
        request.setSerId(device.getDeviceSn());
        request.setCompanyId(device.getOrgId());
        request.setIp(reMessage.getIp());
        request.setPort(reMessage.getPort());
        request.setSerName(device.getDeviceName());
 
        zldzGatewayGrainService.reply8817(request, hexCurPacket);
    }
 
 
    public void analysis8817(ReMessage msg, GatewayDevice device) {
        // 接到信息回复--主机
        reply8817(msg, device);
 
 
        BaseReqData reqData = ProtocolUtils.getSyncReq(device.getDepotIdSys());
 
        if (null == reqData) {
            log.warn("---------没有获取到当前仓库的请求,不执行解析-----{}", device.getDeviceName());
            return;
        }
 
        analysisStep(msg, device, reqData);
    }
 
 
    /**
     * 开始解析
     *
     * @param device
     * @param msg
     * @throws Exception
     */
    private void analysisStep(ReMessage msg, GatewayDevice device, BaseReqData reqData) {
        // 粮情的批次号重新根据频率调整
        msg.setBatchId(ContextUtil.getDefaultBatchId());
 
        String bodyMsg = msg.getBody().getContent();
 
        log.debug("{}=收到的粮情信息={}", device.getDeviceName(),
                msg.getStrMsg());
 
        // Step1根据需要简化需要的信息
        String ua = bodyMsg.substring(0 * 2, 1 * 2);
        log.debug("获取到的电压信息,系统不保留={}", ua);
 
        // 总包数
        int sumPacket = BytesUtil.hexToInt(bodyMsg.substring(1 * 2,
                1 * 2 + 1 * 2));
        // 当前包
        String hexCurPacket = bodyMsg.substring(2 * 2, 2 * 2 + 1 * 2);
        int curPacket = BytesUtil.hexToInt(hexCurPacket);
 
        log.debug("分机={},包总数={},当前包={}", device.getDeviceName(), sumPacket, curPacket);
 
        // 粮食温度信息
        String grainStr = bodyMsg.substring(3 * 2);
 
        String contextKey = this.buildContextKey(msg, device.getDepotIdSys());
        String curKey = this.buildCurKey(msg, curPacket);
 
        Map<String, String> curMap = contextMap.get(contextKey);
        if (null == curMap) {
            curMap = new HashMap<>();
            contextMap.put(contextKey, curMap);
        }
        contextMap.get(contextKey).put(curKey, grainStr);
 
        // 数据准备好,开始执行解析
        if (curPacket == sumPacket) {
            analysisStep0(msg, device, reqData, contextKey, sumPacket);
            log.info("分机------>>>平台:命令类型=8817--粮情全部收到,开始解析-{}",
                    device.getDeviceName());
        }
    }
 
    private void analysisStep0(ReMessage msg, GatewayDevice device, BaseReqData reqData, String contextKey, int sumPacket) {
 
        Map<String, String> curMap = contextMap.get(contextKey);
 
        if (StringUtils.isEmpty(device.getCableCir())) {
            analysisStep1(msg, device, reqData, curMap, sumPacket);
        } else {
            analysisStep2(msg, device, reqData, curMap, sumPacket);
        }
    }
 
    /**
     * 圆筒仓的粮情解析
     *
     * @param packetMap
     * @param sumPackets
     */
    private void analysisStep2(ReMessage msg,
                               GatewayDevice device, BaseReqData reqData,
                               Map<String, String> packetMap, int sumPackets) {
 
 
        String[] cableRuleAtt = device.getCableRule().split("-");
        String[] cableCirAtt = device.getCableCir().split("-");
 
        if (cableRuleAtt.length != cableCirAtt.length) {
            log.error("分机------>>>平台:当前仓库:{},布线规则不正确,无法解析粮情信息……", device.getDeviceName());
            return;
        }
 
        // 获取最大的层配置--默认每一圈都一样
        int layMax = Integer.valueOf(cableCirAtt[0]);
        for (int i = 0; i < cableCirAtt.length; i++) {
            if (Integer.valueOf(cableCirAtt[i]) >= layMax) layMax = Integer.valueOf(cableCirAtt[i]);
        }
 
        // 获取完整的粮情包信息
        String strPoints = "";
        for (int i = 1; i <= sumPackets; i++) {
            strPoints += packetMap.get(buildCurKey(msg, i));
        }
 
        log.info("------筒仓完整的粮情报文={}------", strPoints);
 
        int sumNum = 0, cableZ = 1;// sumNum 共多少根电缆;cableZ 层的最大值,锥形仓补齐最大层
        for (int i = 0; i < cableCirAtt.length; i++) {
            if (Integer.valueOf(cableCirAtt[i]) > cableZ) {
                cableZ = Integer.valueOf(cableCirAtt[i]);
            }
            sumNum += Integer.valueOf(cableRuleAtt[i]);
        }
 
        // 根据层行列获取指定长度
        int start = 0;
        int len = 4 * cableZ * sumNum;
 
        strPoints = strPoints.substring(start, start + len);
        log.info("分机------>>>平台:返回粮情完整信息,分机={}", device.getDeviceName());
 
        // 将粮情解析成数组
        List<Double> temps = new ArrayList<>();
        double tempValue;
        String temp;
        int curLay = 1;//所在层从1开始
        int curRoot = 1;//所在根
        int curCir = 1;//所在圈
        int cirLay = 1;//当前圈的层
        for (int i = 0; i < strPoints.length() / 4; i++) {
            temp = strPoints.substring(i * 4, i * 4 + 4);
            if (temp == null) {
                temp = "0000";
            }
            tempValue = BytesUtil.hexToInt(BytesUtil.tran_LH(temp)) / 10.0;
 
            //非正常值
            if (tempValue > ERROR_CHECK_TAG2) {
                tempValue = ProtocolUtils.ERROR_TEMP;
            }
 
            // 故障值处理
            if (tempValue >= FAULT_CHECK_TAG) {
                tempValue = ProtocolUtils.FAULT_TEMP;
            }
 
            // 备用值
            if (tempValue == ERROR_CHECK_TAG) {
                tempValue = ProtocolUtils.ERROR_TEMP;
 
 
                //验证是不是锥形仓补偿值
                curLay = (i % layMax) + 1;
                curRoot = (i / layMax) + 1;
                curCir = getCurCir(curRoot, cableRuleAtt);
 
                cirLay = Integer.valueOf(cableCirAtt[curCir - 1]);
 
                //比如配置了5层但是当前是6层,说明当前点为补偿点
                if (curLay > cirLay) {
                    tempValue = ProtocolUtils.ADD_TEMP;
                } else {
                    temps.add(tempValue);
                }
 
            } else {
                temps.add(tempValue);
            }
        }
 
        // 将集合解析成坐标数据
        addPoint2(temps, msg, device, reqData);
    }
 
    /**
     * 获取当前跟所在圈
     *
     * @param curRoot
     * @param cableRuleAtt
     * @return
     */
    private int getCurCir(int curRoot, String[] cableRuleAtt) {
 
        int sum = 0;
        for (int i = 0; i < cableRuleAtt.length; i++) {
            sum += Integer.valueOf(cableRuleAtt[i]);
            if (curRoot <= sum) return i + 1;
        }
 
        return 1;
    }
 
    /**
     * 平方仓的解析,解析需要考虑当前是否启用的一分机多仓
     *
     * @param packetMap
     * @param sumPackets
     */
    private void analysisStep1(ReMessage msg,
                               GatewayDevice device, BaseReqData reqData,
                               Map<String, String> packetMap, int sumPackets) {
        // 获取完整的粮情包信息
        String strPoints = "";
        for (int i = 1; i <= sumPackets; i++) {
            strPoints += packetMap.get(buildCurKey(msg, i));
        }
 
        String[] attCable = device.getCableRule().split("-");
        int cableZ = Integer.valueOf(attCable[0]);
        int cableY = Integer.valueOf(attCable[1]);
        int cableX = Integer.valueOf(attCable[2]);
 
        // 根据层行列获取指定长度
        int start = 4 * (Integer.valueOf(device.getCableStart()) - 1) * cableZ * cableY;
        int len = 4 * cableZ * cableY * cableX;
 
        log.info("分机------>>>平台:返回粮情完整信息,分机={}", device.getDeviceName());
 
        strPoints = strPoints.substring(start, start + len);
 
        // 将粮情解析成数组
        List<Double> temps = new ArrayList<>();
        double tempValue;
        String temp;
        for (int i = 0; i < strPoints.length() / 4; i++) {
            temp = strPoints.substring(i * 4, i * 4 + 4);
            if (temp == null) {
                temp = "0000";
            }
            tempValue = BytesUtil.hexToInt(BytesUtil.tran_LH(temp)) / 10.0;
            // 说明解析的数据有问题
            if (tempValue == ERROR_CHECK_TAG || tempValue == ERROR_CHECK_TAG2) {
                tempValue = ProtocolUtils.ERROR_TEMP;
            }
            // 故障值处理
            if (tempValue >= FAULT_CHECK_TAG) {
                tempValue = ProtocolUtils.FAULT_TEMP;
            }
            temps.add(tempValue);
        }
 
 
        // 将集合解析成坐标数据
        addPoint1(temps, msg, device, reqData);
    }
 
    private String buildCurKey(ReMessage msg, int curPacket) {
        return msg.getSourceId() + "_" + curPacket;
    }
 
    private String buildContextKey(ReMessage msg, String depotId) {
        return msg.getCompanyId() + "_" + msg.getSourceId() + "_" + depotId;
    }
 
    /**
     * 平房仓,解析第二步,解析到坐标数据
     *
     * @param temps
     * @throws Exception
     */
    private void addPoint1(List<Double> temps, ReMessage msg,
                           GatewayDevice device, BaseReqData reqData) {
        //TODO
    }
 
    /**
     * 圆筒仓解析步骤
     *
     * @param temps
     * @param msg
     */
    private void addPoint2(List<Double> temps, ReMessage msg,
                           GatewayDevice device, BaseReqData reqData) {
 
        //TODO
    }
 
 
}