| | |
| | | package com.iailab.module.shasteel.mq.consumer; |
| | | |
| | | import com.alibaba.fastjson.JSONArray; |
| | | import com.alibaba.fastjson.JSONObject; |
| | | import com.iailab.framework.common.util.date.DateUtils; |
| | | import com.iailab.module.model.api.mcs.McsApi; |
| | | import com.iailab.module.model.api.mcs.dto.AlarmConfigRespDTO; |
| | | import com.iailab.module.model.api.mcs.dto.AlarmMessageRespDTO; |
| | | import com.iailab.module.model.api.mcs.dto.PreDataJsonReqVO; |
| | | import com.iailab.module.shasteel.mq.common.constant.CommonConstant; |
| | | import com.iailab.module.shasteel.mq.common.constant.RoutingConstant; |
| | | import com.iailab.module.shasteel.mq.config.QueuePredictFinishConfig; |
| | | import lombok.extern.slf4j.Slf4j; |
| | | import org.springframework.amqp.core.Message; |
| | | import org.springframework.amqp.rabbit.annotation.RabbitListener; |
| | | import org.springframework.amqp.rabbit.core.RabbitTemplate; |
| | | import org.springframework.stereotype.Component; |
| | | import org.springframework.util.CollectionUtils; |
| | | |
| | | import javax.annotation.Resource; |
| | | import java.math.BigDecimal; |
| | | import java.util.ArrayList; |
| | | import java.util.HashMap; |
| | | import java.util.List; |
| | | import java.util.Map; |
| | | import java.util.*; |
| | | import java.util.concurrent.ConcurrentHashMap; |
| | | import java.util.stream.Collectors; |
| | | |
| | | /** |
| | | * 监听预测完成 |
| | | * 根据预警配置产生预警信息 |
| | | * |
| | | * @author PanZhibao |
| | | * @Description |
| | | * @createTime 2024年12月11日 |
| | |
| | | private McsApi mcsApi; |
| | | |
| | | @Resource |
| | | private AlarmConsumer alarmConsumer; |
| | | private RabbitTemplate rabbitTemplate; |
| | | |
| | | private static Map<String, Long> lastRunTime = new ConcurrentHashMap<>(); |
| | | |
| | | public static Map<Long, List<String>> finishModuleMap = new ConcurrentHashMap<>(); |
| | | |
| | | /** |
| | | * 监听预测完成,产生预警消息 |
| | | * |
| | | * @param message |
| | | */ |
| | | @RabbitListener(queues = QueuePredictFinishConfig.QUEUE_NAME) |
| | | public void listen(Message message) { |
| | | String routingKey = message.getMessageProperties().getReceivedRoutingKey(); |
| | | log.info("routingKey:" + routingKey); |
| | | String messageBody = new String(message.getBody()); |
| | | log.info("messageBody:" + messageBody); |
| | | JSONObject messageJson = JSONObject.parseObject(messageBody); |
| | | List<AlarmConfigRespDTO> configList = mcsApi.listAlarmConfig(new HashMap<String, Object>()); |
| | | if (CollectionUtils.isEmpty(configList)) { |
| | | return; |
| | | } |
| | | List<String> OutputIdList = new ArrayList<>(); |
| | | configList.forEach(item -> { |
| | | OutputIdList.add(item.getOutId()); |
| | | }); |
| | | configList.forEach(item -> { |
| | | PreDataJsonReqVO reqVO = new PreDataJsonReqVO(); |
| | | reqVO.setPredictTime(DateUtils.parse(messageJson.get("predictTime").toString(), DateUtils.FORMAT_YEAR_MONTH_DAY_HOUR_MINUTE_SECOND)); |
| | | reqVO.setOutputIdList(OutputIdList); |
| | | Map<String, List<Object[]>> preData = mcsApi.getPreDataCur(reqVO); |
| | | if (CollectionUtils.isEmpty(preData)) { |
| | | try { |
| | | String routingKey = message.getMessageProperties().getReceivedRoutingKey(); |
| | | log.info("routingKey:" + routingKey); |
| | | String messageBody = new String(message.getBody()); |
| | | log.info("messageBody:" + messageBody); |
| | | JSONObject messageJson = JSONObject.parseObject(messageBody); |
| | | if (CollectionUtils.isEmpty(messageJson)) { |
| | | return; |
| | | } |
| | | List<Object[]> result = preData.get(item.getOutId()); |
| | | int toIndex = result.size(); |
| | | if (toIndex <= 0) { |
| | | return; |
| | | // 预测时间 |
| | | Date predictTime = DateUtils.parse(messageJson.get("predictTime").toString(), DateUtils.FORMAT_YEAR_MONTH_DAY_HOUR_MINUTE_SECOND); |
| | | // 预测管网 |
| | | String moduleType = messageJson.get("moduleType").toString(); |
| | | if (!finishModuleMap.containsKey(predictTime.getTime())) { |
| | | List<String> mList = new ArrayList<>(); |
| | | finishModuleMap.put(predictTime.getTime(), mList); |
| | | } |
| | | int fromIndex = result.size() - item.getCompLength(); |
| | | List<Object[]> predictList = result.subList(fromIndex, toIndex); |
| | | for (Object[] data : predictList) { |
| | | BigDecimal dataValue = new BigDecimal(Double.parseDouble(data[1].toString())).setScale(2, BigDecimal.ROUND_HALF_UP); |
| | | if (!(dataValue.compareTo(item.getLowerLimit()) >= 0 && dataValue.compareTo(item.getUpperLimit()) <= 0)) { |
| | | AlarmMessage alarmMessage = new AlarmMessage(); |
| | | AlarmMessageRespDTO alarmMessageRespDTO = new AlarmMessageRespDTO(); |
| | | alarmMessageRespDTO.setConfigId(item.getId()); |
| | | if (dataValue.compareTo(item.getLowerLimit()) < 0) { |
| | | alarmMessageRespDTO.setAlarmType("1");//超下限 |
| | | } else if (dataValue.compareTo(item.getUpperLimit()) > 0) { |
| | | alarmMessageRespDTO.setAlarmType("2");//超上限 |
| | | } |
| | | alarmMessageRespDTO.setAlarmTime(DateUtils.parse(data[0].toString(), DateUtils.FORMAT_YEAR_MONTH_DAY_HOUR_MINUTE_SECOND)); |
| | | JSONObject content = new JSONObject(); |
| | | content.put("alarmObj", item.getAlarmObj()); |
| | | content.put("alarmType", alarmMessageRespDTO.getAlarmType()); |
| | | content.put("unit", item.getUnit()); |
| | | content.put("overLimitTime", data[0]); |
| | | content.put("overLimitValue", dataValue); |
| | | alarmMessageRespDTO.setContent(JSONObject.toJSONString(content)); |
| | | System.out.println("预警消息=" + alarmMessageRespDTO); |
| | | alarmMessage.setAlarmMessageRespDTO(alarmMessageRespDTO); |
| | | try { |
| | | alarmConsumer.process(alarmMessage); |
| | | } catch (InterruptedException e) { |
| | | throw new RuntimeException(e); |
| | | } |
| | | break; |
| | | finishModuleMap.get(predictTime.getTime()).add(moduleType); |
| | | log.info("finishModuleMap:" + JSONObject.toJSONString(finishModuleMap)); |
| | | |
| | | if (moduleType.equals(CommonConstant.NET_BFG) || moduleType.equals(CommonConstant.NET_COG) || |
| | | moduleType.equals(CommonConstant.NET_LDG) || |
| | | moduleType.equals(CommonConstant.NET_LDG1) || |
| | | moduleType.equals(CommonConstant.NET_LDG2) || |
| | | moduleType.equals(CommonConstant.NET_LDG3)) { |
| | | log.info("moduleType:" + moduleType); |
| | | if (lastRunTime.containsKey(moduleType) && lastRunTime.get(moduleType) == predictTime.getTime()) { |
| | | log.info("moduleType return"); |
| | | return; |
| | | } |
| | | lastRunTime.put(moduleType, predictTime.getTime()); |
| | | log.info("lastRunTime=" + JSONObject.toJSONString(lastRunTime)); |
| | | log.info("开始处理预警"); |
| | | Map<String, Object> params = new HashMap<>(); |
| | | params.put("alarmObj", moduleType); |
| | | List<AlarmConfigRespDTO> configList = mcsApi.listAlarmConfig(params); |
| | | if (CollectionUtils.isEmpty(configList)) { |
| | | log.info("AlarmConfigList is empty"); |
| | | return; |
| | | } |
| | | log.info("configList=" + JSONArray.toJSONString(configList)); |
| | | |
| | | // 预警信息列表 |
| | | List<AlarmMessageRespDTO> alarmList = new ArrayList<>(); |
| | | List<String> outputIdList = configList.stream().map(item -> { |
| | | return item.getOutId(); |
| | | }).collect(Collectors.toList()); |
| | | |
| | | // 查询预测结果 |
| | | PreDataJsonReqVO reqVO = new PreDataJsonReqVO(); |
| | | reqVO.setPredictTime(predictTime); |
| | | reqVO.setOutputIdList(outputIdList); |
| | | Map<String, List<Object[]>> preData = mcsApi.getPreDataCur(reqVO); |
| | | if (CollectionUtils.isEmpty(preData)) { |
| | | return; |
| | | } |
| | | outerLoop: |
| | | for (AlarmConfigRespDTO configItem : configList) { |
| | | log.info("AlarmConfigItem: " + configItem); |
| | | List<Object[]> result = preData.get(configItem.getOutId()); |
| | | if (CollectionUtils.isEmpty(result)) { |
| | | continue; |
| | | } |
| | | log.info("AlarmPreData: " + JSONArray.toJSONString(result)); |
| | | |
| | | // 对比预测值是否超限 |
| | | int toIndex = result.size(); |
| | | int fromIndex = result.size() - configItem.getCompLength(); |
| | | List<Object[]> predictList = result.subList(fromIndex, toIndex); |
| | | for (Object[] data : predictList) { |
| | | BigDecimal dataValue = new BigDecimal(Double.parseDouble(data[1].toString())).setScale(2, BigDecimal.ROUND_HALF_UP); |
| | | if (dataValue.compareTo(configItem.getLowerLimit()) >= 0 && dataValue.compareTo(configItem.getUpperLimit()) <= 0) { |
| | | continue; |
| | | } |
| | | log.info("AlarmOutValue: " + dataValue); |
| | | // 预警记录 |
| | | AlarmMessageRespDTO alarmMessage = new AlarmMessageRespDTO(); |
| | | alarmMessage.setConfigId(configItem.getId()); |
| | | alarmMessage.setTitle(configItem.getTitle()); |
| | | alarmMessage.setAlarmObj(configItem.getAlarmObj()); |
| | | alarmMessage.setAlarmTime(predictTime); |
| | | // 设置超出时间 |
| | | alarmMessage.setOutTime(DateUtils.parse(data[0].toString(), DateUtils.FORMAT_YEAR_MONTH_DAY_HOUR_MINUTE_SECOND)); |
| | | // 预警消息 |
| | | StringBuilder content = new StringBuilder(); |
| | | content.append(configItem.getTitle().replace("预警", "")); |
| | | content.append(DateUtils.format(alarmMessage.getOutTime(), DateUtils.FORMAT_SIMPLE_TIME)); |
| | | content.append(","); |
| | | content.append("即将"); |
| | | if (dataValue.compareTo(configItem.getLowerLimit()) < 0) { |
| | | content.append("低于下限"); |
| | | alarmMessage.setAlarmType(CommonConstant.EXCEEDING_LOWER_LIMIT); |
| | | |
| | | } else if (dataValue.compareTo(configItem.getUpperLimit()) > 0) { |
| | | content.append("超出上限"); |
| | | alarmMessage.setAlarmType(CommonConstant.EXCEEDING_UPPER_LIMIT); |
| | | } |
| | | alarmMessage.setContent(content.toString()); |
| | | mcsApi.createAlarmMessage(alarmMessage); |
| | | alarmList.add(alarmMessage); |
| | | continue outerLoop; |
| | | } |
| | | |
| | | // 对比累计值是否超限 |
| | | if (configItem.getCulUpper() != null && configItem.getCulLower() != null) { |
| | | Double culValue = Double.valueOf(0.0); |
| | | for (Object[] data : result) { |
| | | culValue += Double.parseDouble(data[1].toString()); |
| | | if (culValue.compareTo(configItem.getCulLower().doubleValue()) >= 0 && culValue.compareTo(configItem.getCulUpper().doubleValue()) <= 0) { |
| | | continue; |
| | | } |
| | | |
| | | // 生成预警信息 |
| | | AlarmMessageRespDTO alarmMessage = new AlarmMessageRespDTO(); |
| | | alarmMessage.setConfigId(configItem.getId()); |
| | | alarmMessage.setTitle(configItem.getTitle()); |
| | | alarmMessage.setAlarmObj(configItem.getAlarmObj()); |
| | | alarmMessage.setAlarmTime(predictTime); |
| | | // 设置超出时间 |
| | | alarmMessage.setOutTime(DateUtils.parse(data[0].toString(), DateUtils.FORMAT_YEAR_MONTH_DAY_HOUR_MINUTE_SECOND)); |
| | | StringBuilder content = new StringBuilder(); |
| | | content.append(configItem.getTitle().replace("预警", "")); |
| | | content.append(DateUtils.format(alarmMessage.getOutTime(), DateUtils.FORMAT_SIMPLE_TIME)); |
| | | content.append(","); |
| | | content.append("即将"); |
| | | |
| | | if (culValue.compareTo(configItem.getCulUpper().doubleValue()) > 0) { |
| | | content.append("超出累计值上限"); |
| | | alarmMessage.setAlarmType(CommonConstant.EXCEEDING_UPPER_LIMIT); |
| | | } |
| | | if (culValue.compareTo(configItem.getCulLower().doubleValue()) < 0) { |
| | | content.append("低于累计值下限"); |
| | | alarmMessage.setAlarmType(CommonConstant.EXCEEDING_LOWER_LIMIT); |
| | | } |
| | | alarmMessage.setContent(content.toString()); |
| | | mcsApi.createAlarmMessage(alarmMessage); |
| | | alarmList.add(alarmMessage); |
| | | continue outerLoop; |
| | | } |
| | | } |
| | | } |
| | | |
| | | if (CollectionUtils.isEmpty(alarmList)) { |
| | | log.info("alarmList is empty"); |
| | | return; |
| | | } |
| | | log.info("发送预警消息"); |
| | | Map<String, Object> msg = new HashMap<>(2); |
| | | msg.put("predictTime", DateUtils.format(predictTime, DateUtils.FORMAT_YEAR_MONTH_DAY_HOUR_MINUTE_SECOND)); |
| | | msg.put("alarmList", alarmList); |
| | | msg.put("moduleType", moduleType); |
| | | rabbitTemplate.convertAndSend(RoutingConstant.EXCHANGE, RoutingConstant.Iailab_Model_Alarm, msg); |
| | | } |
| | | System.out.println(preData); |
| | | }); |
| | | } catch (Exception e) { |
| | | e.printStackTrace(); |
| | | } |
| | | } |
| | | } |