RyTask.java 112 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038
  1. package com.slibra.quartz.task;
  2. import cn.hutool.core.date.DateUtil;
  3. import cn.hutool.http.HttpRequest;
  4. import com.alibaba.fastjson2.JSON;
  5. import com.alibaba.fastjson2.JSONArray;
  6. import com.alibaba.fastjson2.JSONObject;
  7. import com.alibaba.fastjson2.JSONWriter;
  8. import com.google.protobuf.ByteString;
  9. import com.slibra.business.domain.*;
  10. import com.slibra.business.mapper.*;
  11. import com.slibra.business.req.ChatReq;
  12. import com.slibra.business.req.DecisionReq;
  13. import com.slibra.business.res.ChartBasic;
  14. import com.slibra.business.res.ShowValueChartBasic;
  15. import com.slibra.common.constant.MyConstants;
  16. import com.slibra.common.enums.BusinessEnum;
  17. import com.slibra.common.enums.DataSourceType;
  18. import com.slibra.common.utils.DateUtils;
  19. import com.slibra.common.utils.uuid.IdUtils;
  20. import com.slibra.framework.datasource.DynamicDataSourceContextHolder;
  21. import inference.InferenceAPIsServiceGrpc;
  22. import inference.PredictionResponse;
  23. import inference.PredictionsRequest;
  24. import io.grpc.ManagedChannel;
  25. import io.grpc.ManagedChannelBuilder;
  26. import lombok.extern.slf4j.Slf4j;
  27. import org.springframework.beans.factory.annotation.Autowired;
  28. import org.springframework.stereotype.Component;
  29. import com.slibra.common.utils.StringUtils;
  30. import org.springframework.util.CollectionUtils;
  31. import java.io.IOException;
  32. import java.math.BigDecimal;
  33. import java.math.RoundingMode;
  34. import java.time.LocalDateTime;
  35. import java.time.format.DateTimeFormatter;
  36. import java.util.*;
  37. import java.util.stream.Collectors;
  38. import static com.slibra.common.constant.MyConstants.*;
  39. /**
  40. * 定时任务调度测试
  41. *
  42. *
  43. */
  44. @Component("ryTask")
  45. @Slf4j
  46. public class RyTask
  47. {
  48. public void ryMultipleParams(String s, Boolean b, Long l, Double d, Integer i)
  49. {
  50. System.out.println(StringUtils.format("执行多参方法: 字符串类型{},布尔类型{},长整型{},浮点型{},整形{}", s, b, l, d, i));
  51. }
  52. public void ryParams(String params)
  53. {
  54. System.out.println("执行有参方法:" + params);
  55. }
  56. public void ryNoParams()
  57. {
  58. System.out.println("执行无参方法");
  59. }
  60. //----------------------------------------------下面是新增的方法----------------------------------------------
  61. @Autowired
  62. private TXinyiIndustryMapper xinyiIndustryMapper;
  63. @Autowired
  64. private TXinyiRobotMapper xinyiRobotMapper;
  65. @Autowired
  66. private TXinyiNormConfigMapper xinyiNormConfigMapper;
  67. @Autowired
  68. private TXinyiWarningRecordMapper xinyiWarningRecordMapper;
  69. @Autowired
  70. private TXinyiChatRecordMapper xinyiChatRecordMapper;
  71. @Autowired
  72. private TXinyiDailyMapper xinyiDailyMapper;
  73. @Autowired
  74. private AsyncTask asyncTask;
  75. // public final static StopWatch watch = new StopWatch("task");
  76. public static final String[] queryTags = {"信义污水厂JS_COD_Value","信义污水厂JS_PH_Value","信义污水厂JS_SS_Value","信义污水厂JS_ZL_Value","信义污水厂JS_ZA_Value","信义污水厂JS_AD_Value","信义污水厂JS_T_Value","信义污水厂进水泵房液位","信义污水厂出水瞬时流量","信义污水厂升级出水COD","信义污水厂升级出水PH","信义污水厂升级出水SS","信义污水厂升级出水TN","信义污水厂升级出水TP","信义污水厂升级出水氨氮","信义污水厂AIT202_Value","信义污水厂AIT203_Value","信义污水厂AIT207_Value","信义污水厂AIT206_Value","信义污水厂AIT209_Value","信义污水厂AIT210_Value","信义污水厂进水TDS","信义污水厂FT101_Value","信义污水厂SWCHHYHLB1_R_Value","信义污水厂SWCHHYHLB2_R_Value","信义污水厂SWCHHYHLB3_R_Value","信义污水厂SWCHHYHLB4_R_Value","信义污水厂SWCHHYHLB5_R_Value","信义污水厂SWCHHYHLB6_R_Value","信义污水厂SWCWNHLB1_R_Value","信义污水厂SWCWNHLB2_R_Value","信义污水厂SWCWNHLB3_R_Value","信义污水厂SWCWNHLB4_R_Value","信义污水厂SWCWNHLB5_R_Value","信义污水厂GFJ1_R_Value","信义污水厂GFJ2_R_Value","信义污水厂GFJ3_R_Value","信义污水厂GFJ4_R_Value","信义污水厂GFJ5_R_Value","信义污水厂GFJ6_R_Value","信义污水厂GFJ1_KQLL_Value","信义污水厂GFJ2_KQLL_Value","信义污水厂GFJ3_KQLL_Value","信义污水厂GFJ4_KQLL_Value","信义污水厂GFJ5_KQLL_Value","信义污水厂GFJ6_KQLL_Value","信义污水厂实际碳源加药量","信义污水厂除磷加药瞬时流量", "信义污水厂_除磷P04预测值_"};
  77. public static final String[] predictorArr = {"出水COD", "出水SS", "出水总氮", "出水总磷", "出水氨氮", "xsy1", "xsy2"};
  78. public static final String[] predictorArrStr = {"出水COD", "出水SS", "出水总氮", "出水总磷", "出水氨氮", "硝酸盐#1", "硝酸盐#2"};
  79. /**
  80. * 定时从工业库获取数据
  81. *
  82. * 2024年4月17日17:44:15 调整逻辑:考虑到因断电等情况导致服务断电,所以不再同步最近一小时,而是同步从上次成功的最后一条数据开始。
  83. */
  84. public void getIndustryData(){
  85. log.info("进入了定时同步工业库数据的任务");
  86. //耗时工具
  87. // watch.start("parseJob");
  88. // 给定时间段的起始时间和结束时间
  89. LocalDateTime endTime = LocalDateTime.now();
  90. // LocalDateTime startTime = endTime.plusMinutes(-60);
  91. //获取上次最后一条同步的数据的日期到 分钟维度
  92. String lastDateHour = this.xinyiIndustryMapper.getLastMinute();
  93. log.info("获取上次同步工业库的最后一条记录的时间是{}", lastDateHour);
  94. lastDateHour = lastDateHour + ":00";
  95. //开始时间
  96. LocalDateTime startTime = LocalDateTime.parse(lastDateHour.replaceAll("/", "-").replace(" ", "T"));
  97. startTime = startTime.plusMinutes(60L);//加一分钟 从上次最后一条记录的下一分钟开始
  98. /*LocalDateTime startTime = LocalDateTime.parse("2024-02-26T00:00:00");
  99. LocalDateTime endTime = LocalDateTime.parse("2024-02-27T00:00:00");*/
  100. // 每个小时的时间格式
  101. DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
  102. // 循环按小时分割
  103. LocalDateTime currentHour = startTime;
  104. //最终获取的数据
  105. Map<String, TXinyiIndustry> needMap = new LinkedHashMap<>();
  106. while (currentHour.isBefore(endTime)) {
  107. String begin = currentHour.format(formatter);
  108. String end = currentHour.plusMinutes(5).format(formatter);
  109. // 输出当前小时的起始时间和结束时间
  110. System.out.println("起始时间:" + begin);
  111. System.out.println("结束时间:" + end);
  112. // 当前小时加一小时,作为下一个小时的起始时间
  113. currentHour = currentHour.plusMinutes(5);
  114. //每个小时查询一次数据
  115. String url = "http://10.0.0.27:4568/api/v1/khquerydata";
  116. HashMap<String, Object> req = new HashMap<>();
  117. req.put("tagNames", queryTags);
  118. req.put("startTime", begin);
  119. req.put("endTime", end);
  120. req.put("recordNumbers", 100000);
  121. String body = HttpRequest.post(url).header("Authorization", "c2E6c2E=").header("clientName", "hongshan").body(JSON.toJSONString(req)).execute().body();
  122. // System.out.println("body = " + body);
  123. List<HashMap<String, String>> list = new ArrayList<>();
  124. //行转列数据处理
  125. for (String queryTag : queryTags) {
  126. JSONArray array = JSON.parseObject(body).getJSONArray(queryTag);
  127. //特殊数据处理一
  128. if(Objects.isNull(array) || array.isEmpty()){
  129. System.out.println(queryTag + "查询到了空的数据,跳过本次循环");
  130. continue;
  131. }
  132. int size = array.size();
  133. //特殊数据处理二
  134. if("0".equals(array.get(1) + "")){
  135. System.out.println(queryTag + "查询到了数据,但是数据集合只有一条,且都是0");
  136. continue;
  137. }
  138. //结合至少62个数据才满足条件(有可能获取不到)
  139. /*if(size < 62){
  140. System.out.println(queryTag + "查询到了不符合条件的数据,跳过本次循环");
  141. continue;
  142. }*/
  143. //存放的数据集
  144. //利用map去重
  145. HashMap<String, String> map = new LinkedHashMap<>();
  146. for (int i = 2; i < size; i++) {
  147. // System.out.println(i + "" + array.get(i));
  148. JSONArray oneRecord = JSON.parseArray(JSON.toJSONString(array.get(i)));
  149. //处理为空或者为0的数据
  150. Object timeStampValue = oneRecord.get(2);
  151. if(Objects.isNull(timeStampValue) || "0".equals(timeStampValue + ""))
  152. continue;
  153. BigDecimal value = Objects.isNull(oneRecord.get(0)) ? null : new BigDecimal(oneRecord.get(0) + "");
  154. long timestamp = (long) timeStampValue;
  155. String format = DateUtil.format(new Date(timestamp), DateUtils.YYYYMMDDHH_TS);
  156. map.put(format, queryTag + "-" + value);
  157. }
  158. list.add(map);
  159. }
  160. Set<String> recordTimeSet = new HashSet<>();
  161. Map<String, TXinyiIndustry> recordMap = new HashMap<>();
  162. for (int i = 0; i < list.size(); i++) {
  163. HashMap<String, String> map = list.get(i);
  164. int finalJ = i;
  165. map.forEach((k, v) ->{
  166. TXinyiIndustry industry = null;
  167. if(!recordTimeSet.contains(k)){//第一次
  168. industry = new TXinyiIndustry();
  169. recordTimeSet.add(k);
  170. recordMap.put(k, industry);
  171. }else{
  172. industry = recordMap.get(k);
  173. }
  174. industry.setTestTime(k + ":00");
  175. //2024年4月15日11:19:52 额外增加2个字段
  176. industry.setTestDate(k.substring(0,10));
  177. industry.setTestHour(k.substring(0,13));
  178. //解析值
  179. String[] split = v.split("-");
  180. String type = split[0];
  181. BigDecimal value = new BigDecimal(split[1]);
  182. if ("信义污水厂JS_COD_Value".equals(type)) {
  183. industry.setJsCod(value);
  184. } else if ("信义污水厂JS_PH_Value".equals(type)) {
  185. industry.setJsPh(value);
  186. } else if ("信义污水厂JS_SS_Value".equals(type)) {
  187. // log.info("************----------真实的进水SS的值为{}\n时间为{}", value, k);
  188. industry.setJsSs(value);
  189. } else if ("信义污水厂JS_ZL_Value".equals(type)) {
  190. industry.setJsTp(value);
  191. } else if ("信义污水厂JS_ZA_Value".equals(type)) {
  192. industry.setJsTn(value);
  193. } else if ("信义污水厂JS_AD_Value".equals(type)) {
  194. industry.setJsNh3(value);
  195. } else if ("信义污水厂JS_T_Value".equals(type)) {
  196. industry.setJsSwPh(value);
  197. } else if ("信义污水厂进水泵房液位".equals(type)) {
  198. industry.setJsBfyw(value);
  199. } else if ("信义污水厂出水瞬时流量".equals(type)) {
  200. industry.setCsSlqc(value);
  201. } else if ("信义污水厂升级出水COD".equals(type)) {
  202. industry.setCsCod(value);
  203. } else if ("信义污水厂升级出水PH".equals(type)) {
  204. industry.setCsPh(value);
  205. } else if ("信义污水厂升级出水SS".equals(type)) {
  206. industry.setCsSs(value);
  207. } else if ("信义污水厂升级出水TN".equals(type)) {
  208. // log.info("************----------真实的出水总氮的值为{}\n时间为{}", value, k);
  209. // industry.setCsTn(new BigDecimal(17));
  210. industry.setCsTn(value);
  211. } else if ("信义污水厂升级出水TP".equals(type)) {
  212. industry.setCsTp(value);
  213. } else if ("信义污水厂升级出水氨氮".equals(type)) {
  214. industry.setCsNh3(value);
  215. } else if ("信义污水厂AIT202_Value".equals(type)) {
  216. industry.setOneHyzdDo(value);
  217. } else if ("信义污水厂AIT203_Value".equals(type)) {
  218. industry.setOneHymdDo(value);
  219. } else if ("信义污水厂AIT207_Value".equals(type)) {
  220. industry.setTwoHyzdDo(value);
  221. } else if ("信义污水厂AIT206_Value".equals(type)) {
  222. industry.setTwoHymdDo(value);
  223. } else if ("信义污水厂AIT209_Value".equals(type)) {
  224. industry.setOneMlss(value);
  225. } else if ("信义污水厂AIT210_Value".equals(type)) {
  226. industry.setTwoMlss(value);
  227. } else if ("信义污水厂进水TDS".equals(type)) {
  228. industry.setJsTds(value);
  229. } else if ("信义污水厂FT101_Value".equals(type)) {
  230. industry.setJsSlq(value);
  231. } else if ("信义污水厂SWCHHYHLB1_R_Value".equals(type)) {
  232. industry.setNHlbOneGp(value);
  233. } else if ("信义污水厂SWCHHYHLB2_R_Value".equals(type)) {
  234. industry.setNHlbTwoGp(value);
  235. } else if ("信义污水厂SWCHHYHLB3_R_Value".equals(type)) {
  236. industry.setNHlbThreeGp(value);
  237. } else if ("信义污水厂SWCHHYHLB4_R_Value".equals(type)) {
  238. industry.setNHlbFourGp(value);
  239. } else if ("信义污水厂SWCHHYHLB5_R_Value".equals(type)) {
  240. industry.setNhlBFiveGp(value);
  241. } else if ("信义污水厂SWCHHYHLB6_R_Value".equals(type)) {
  242. industry.setNHlbSixGp(value);
  243. } else if ("信义污水厂SWCWNHLB1_R_Value".equals(type)) {
  244. industry.setWHlbOneGp(value);
  245. } else if ("信义污水厂SWCWNHLB2_R_Value".equals(type)) {
  246. industry.setWHlbTwoGp(value);
  247. } else if ("信义污水厂SWCWNHLB3_R_Value".equals(type)) {
  248. industry.setWHlbThreeGp(value);
  249. } else if ("信义污水厂SWCWNHLB4_R_Value".equals(type)) {
  250. industry.setWHlbFourGp(value);
  251. } else if ("信义污水厂SWCWNHLB5_R_Value".equals(type)) {
  252. industry.setWHlbFiveGp(value);
  253. } else if ("信义污水厂GFJ1_R_Value".equals(type)) {
  254. industry.setFjOne(value);
  255. } else if ("信义污水厂GFJ2_R_Value".equals(type)) {
  256. industry.setFjTwo(value);
  257. } else if ("信义污水厂GFJ3_R_Value".equals(type)) {
  258. industry.setFjThree(value);
  259. } else if ("信义污水厂GFJ4_R_Value".equals(type)) {
  260. industry.setFjFour(value);
  261. } else if ("信义污水厂GFJ5_R_Value".equals(type)) {
  262. industry.setFjFive(value);
  263. } else if ("信义污水厂GFJ6_R_Value".equals(type)) {
  264. industry.setFjSix(value);
  265. } else if ("信义污水厂GFJ1_KQLL_Value".equals(type)) {
  266. industry.setKqllOne(value);
  267. } else if ("信义污水厂GFJ2_KQLL_Value".equals(type)) {
  268. industry.setKqllTwo(value);
  269. } else if ("信义污水厂GFJ3_KQLL_Value".equals(type)) {
  270. industry.setKqllThree(value);
  271. } else if ("信义污水厂GFJ4_KQLL_Value".equals(type)) {
  272. industry.setKqllFour(value);
  273. } else if ("信义污水厂GFJ5_KQLL_Value".equals(type)) {
  274. industry.setKqllFive(value);
  275. } else if ("信义污水厂GFJ6_KQLL_Value".equals(type)) {
  276. industry.setKqllSix(value);
  277. }else if ("信义污水厂实际碳源加药量".equals(type)) {
  278. industry.setSJTYJLY(value);
  279. }else if ("信义污水厂除磷加药瞬时流量".equals(type)) {
  280. industry.setCLJYSSLL(value);
  281. } else if ("信义污水厂_除磷P04预测值_".equals(type)) {
  282. industry.setCLP04YCZ(value);
  283. }
  284. //只有最后一次才执行数据库添加
  285. if(finalJ == list.size()-1){
  286. needMap.put(industry.getTestHour(), industry);
  287. }
  288. });
  289. }
  290. }
  291. //保存数据 触发告警 决策 问答记录等等
  292. needMap.forEach((k, industry) ->{
  293. //2024年4月22日15:45:24 额外保存两个字段 数组
  294. List<BigDecimal> extraList = new ArrayList<>();
  295. extraList.add(industry.getOneHymdDo());
  296. extraList.add(industry.getTwoHymdDo());
  297. industry.setHycRjyAll(JSON.toJSONString(extraList));
  298. extraList.clear();
  299. extraList.add(industry.getOneHyzdDo());
  300. extraList.add(industry.getTwoHyzdDo());
  301. industry.setHycRjyZdAll(JSON.toJSONString(extraList));
  302. extraList.clear();
  303. extraList.add(industry.getOneMlss());
  304. extraList.add(industry.getTwoMlss());
  305. industry.setHycWnndAll(JSON.toJSONString(extraList));
  306. //插入数据库
  307. xinyiIndustryMapper.insertTXinyiIndustry(industry);
  308. //判断是否触发告警、接触告警、保存决策等等
  309. // this.handleWarning(industry);
  310. //2024年5月30日18:29:20 改为异步处理
  311. asyncTask.handleWarning(industry);
  312. });
  313. //执行完成 测试执行时间
  314. //计时结束
  315. // watch.stop();
  316. // System.out.println(watch.getLastTaskName() + " 执行耗时:" + watch.getLastTaskTimeMillis() + " ms");
  317. }
  318. private void handleWarning(TXinyiIndustry tXinyiIndustry) {
  319. log.info("进入了定时任务保存工业库数据并触发报警操作");
  320. // tXinyiIndustry.setCsTn(new BigDecimal(17));
  321. //获取配置表
  322. List<TXinyiNormConfig> tXinyiNormConfigs = this.xinyiNormConfigMapper.selectTXinyiNormConfigList(null);
  323. if(CollectionUtils.isEmpty(tXinyiNormConfigs))
  324. throw new RuntimeException("未查询到配置信息");
  325. TXinyiNormConfig normConfig = tXinyiNormConfigs.get(0);
  326. //水质报警
  327. this.handleSZWarning(tXinyiIndustry, normConfig);
  328. //2024年5月28日14:14:26 下面是新增的 生化报警处理
  329. this.handleSHWarning(tXinyiIndustry, normConfig);
  330. }
  331. private void handleSHWarning(TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig) {
  332. //判断对应指标是否报警 然后调研大模型获取决策信息
  333. BigDecimal jsSlq = tXinyiIndustry.getJsSlq();
  334. //内回流比报警
  335. BigDecimal nhlbSjz = normConfig.getNhlbSjz();//400
  336. BigDecimal nhlbnkSxz = normConfig.getNhlbnkSxz();//360
  337. BigDecimal nhlbqdsl = normConfig.getNHLBQDSL();
  338. BigDecimal nhlbdsjll = normConfig.getNHLBDSJLL();
  339. BigDecimal nhlbgzxl = normConfig.getNHLBGZXL();
  340. if(!Objects.isNull(jsSlq) && !Objects.isNull(nhlbnkSxz) && !Objects.isNull(nhlbqdsl)
  341. && !Objects.isNull(nhlbdsjll) && !Objects.isNull(nhlbgzxl)){
  342. BigDecimal divide = nhlbqdsl.multiply(nhlbdsjll).multiply(nhlbgzxl).divide(jsSlq, 4, RoundingMode.HALF_UP);
  343. if(!Objects.isNull(nhlbSjz)){
  344. handleXinYiWarningsSH(nhlbSjz, divide, nhlbnkSxz, BusinessEnum.WarningCategoryEnum.NHLB.getCode(), tXinyiIndustry, normConfig);
  345. }
  346. }
  347. //外回流比报警
  348. BigDecimal whlbqdsl = normConfig.getWHLBQDSL();
  349. BigDecimal whlbdsjll = normConfig.getWHLBDSJLL();
  350. BigDecimal whlbgzxl = normConfig.getWHLBGZXL();
  351. BigDecimal whlbnkXxz = normConfig.getWhlbnkXxz();
  352. BigDecimal whlbSjz = normConfig.getWhlbSjz();
  353. if(!Objects.isNull(whlbqdsl) && !Objects.isNull(whlbdsjll) && !Objects.isNull(whlbgzxl)){
  354. BigDecimal divide = whlbqdsl.multiply(whlbdsjll).multiply(whlbgzxl).divide(jsSlq, 4, RoundingMode.HALF_UP);
  355. if(!Objects.isNull(whlbSjz)){
  356. handleXinYiWarningsSH(whlbSjz, divide, whlbnkXxz, BusinessEnum.WarningCategoryEnum.WHLB.getCode(), tXinyiIndustry, normConfig);
  357. }
  358. }
  359. //最新的一条日报数据
  360. List<TXinyiDaily> tXinyiDailies = this.xinyiDailyMapper.selectTXinyiDailyList(null);
  361. if(CollectionUtils.isEmpty(tXinyiDailies))
  362. return;//肯定不会出现这种情况 因为有很多历史数据了
  363. TXinyiDaily tXinyiDaily = tXinyiDailies.get(0);
  364. log.info("生化报警获取日报的最新的一条数据为{}", JSON.toJSONString(tXinyiDaily));
  365. BigDecimal jsBod5 = tXinyiDaily.getJsBod5();
  366. //污泥负荷(需要从日报获取数据) 计算 + 部分数据从日报获取
  367. //污泥负荷=[Ls]=24*([BOD_in]-[BOD_off])*[Q_in]/[MLSS]/([V_hao]+[V_que]+[V_yan])/2 kgBOD/(kgMLSS·d)
  368. //todo 后面再加 计算太复杂
  369. //碳氮比(需要从日报获取数据)
  370. //进水碳氮比=[b]=[BOD_in]/[TN_in]
  371. BigDecimal jsTn = tXinyiIndustry.getJsTn();
  372. if(!Objects.isNull(jsBod5) && !Objects.isNull(jsTn)){
  373. BigDecimal divide = jsBod5.divide(jsTn, 4, RoundingMode.HALF_UP);
  374. BigDecimal jstdbnkzSxz = normConfig.getJstdbnkzSxz();
  375. BigDecimal cstdbSjz = normConfig.getCstdbSjz();
  376. if(!Objects.isNull(whlbSjz)){
  377. handleXinYiWarningsSH(cstdbSjz, divide, jstdbnkzSxz, BusinessEnum.WarningCategoryEnum.TDB.getCode(), tXinyiIndustry, normConfig);
  378. }
  379. }
  380. //碳磷比(需要从日报获取数据)
  381. //进水碳磷比=[c]=[BOD_in]/[TP_in]
  382. BigDecimal jsTp = tXinyiIndustry.getJsTp();
  383. BigDecimal jstlbSjz = normConfig.getJstlbSjz();
  384. if(!Objects.isNull(jsBod5) && !Objects.isNull(jsTp)){
  385. BigDecimal divide = jsBod5.divide(jsTp, 4, RoundingMode.HALF_UP);
  386. BigDecimal jstlbNkz = normConfig.getJstlbNkz();
  387. if(!Objects.isNull(jstlbSjz)){
  388. handleXinYiWarningsSH(jstlbSjz, divide, jstlbNkz, BusinessEnum.WarningCategoryEnum.TLB.getCode(), tXinyiIndustry, normConfig);
  389. }
  390. }
  391. //BOD比COD(需要从日报获取数据)
  392. //进水BOD与COD比值数据=[d]=[BOD_in]/[COD_in]
  393. BigDecimal jsCod = tXinyiIndustry.getJsCod();
  394. if(!Objects.isNull(jsBod5) && !Objects.isNull(jsCod)){
  395. BigDecimal jsbodycodbzSzj = normConfig.getJsbodycodbzSzj();
  396. String jsbodycodbzGkz = normConfig.getJsbodycodbzGkz();
  397. if(StringUtils.isNotBlank(jsbodycodbzGkz) && jsbodycodbzGkz.contains("-")){
  398. String[] split = jsbodycodbzGkz.split("-");
  399. BigDecimal gkz = new BigDecimal(split[1]);
  400. BigDecimal divide = jsBod5.divide(jsCod, 4, RoundingMode.HALF_UP);
  401. handleXinYiWarningsSH(gkz, divide, jsbodycodbzSzj, BusinessEnum.WarningCategoryEnum.BODCODB.getCode(), tXinyiIndustry, normConfig);
  402. }
  403. }
  404. //好氧区DO(一池)
  405. BigDecimal shcHyOneDo = tXinyiIndustry.getOneHymdDo();
  406. BigDecimal hycrjysjzSxz = normConfig.getHycrjysjzSxz();
  407. BigDecimal hycrjyNkz = normConfig.getHycrjyNkz();
  408. if(!Objects.isNull(shcHyOneDo) && !Objects.isNull(hycrjysjzSxz)){
  409. handleXinYiWarningsSH(hycrjysjzSxz, shcHyOneDo, hycrjyNkz, BusinessEnum.WarningCategoryEnum.HYQDO_ONE.getCode(), tXinyiIndustry, normConfig);
  410. }
  411. //好氧区DO(二池)
  412. BigDecimal shcHyTwoDo = tXinyiIndustry.getTwoHymdDo();
  413. if(!Objects.isNull(shcHyTwoDo) && !Objects.isNull(hycrjysjzSxz)){
  414. handleXinYiWarningsSH(hycrjysjzSxz, shcHyTwoDo, hycrjyNkz, BusinessEnum.WarningCategoryEnum.HYQDO_TWO.getCode(), tXinyiIndustry, normConfig);
  415. }
  416. //气水比(需要从日报获取数据)
  417. BigDecimal gfjgzts = normConfig.getGFJGZTS();
  418. BigDecimal gfjckll = normConfig.getGFJCKLL();
  419. BigDecimal shcqbSjz = normConfig.getShcqbSjz();
  420. if(!Objects.isNull(gfjgzts) && !Objects.isNull(gfjckll) && !Objects.isNull(shcqbSjz) && !Objects.isNull(jsSlq)){
  421. BigDecimal qsb = gfjgzts.multiply(gfjckll).divide(jsSlq, 4, RoundingMode.HALF_UP);
  422. BigDecimal shcqbNkz = normConfig.getShcqbNkz();
  423. handleXinYiWarningsSH(shcqbSjz, qsb, shcqbNkz, BusinessEnum.WarningCategoryEnum.WHLB.getCode(), tXinyiIndustry, normConfig);
  424. }
  425. }
  426. /**
  427. * 处理信义生化报警的逻辑 统一处理
  428. * @param bzz
  429. * @param currentVal
  430. * @param gkz
  431. * @param category
  432. * @param tXinyiIndustry
  433. * @param normConfig
  434. */
  435. private void handleXinYiWarningsSH(BigDecimal bzz, BigDecimal currentVal, BigDecimal gkz, String category, TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig) {
  436. BigDecimal multiply = bzz.multiply(new BigDecimal(MyConstants.SCALE_VALUE));
  437. TXinyiWarningRecord tXinyiWarningRecord = new TXinyiWarningRecord();
  438. /*String category = BusinessEnum.WarningCategoryEnum.CS_AD.getCode();*/
  439. tXinyiWarningRecord.setStatus(0);
  440. tXinyiWarningRecord.setType(1);
  441. tXinyiWarningRecord.setCategory(category);
  442. tXinyiWarningRecord.setTime(DateUtils.getNowDate());
  443. tXinyiWarningRecord.setWarningVal(currentVal);
  444. tXinyiWarningRecord.setDesignVal(bzz);
  445. tXinyiWarningRecord.setControlVal(gkz);
  446. tXinyiWarningRecord.setCreateBy(WARNING_DEFAULT_CREATE);
  447. tXinyiWarningRecord.setCreateTime(DateUtils.getNowDate());
  448. //2024年5月25日17:52:33 如果工业库获取不到数据,也触发报警,但是不调用决策接口
  449. if (Objects.isNull(currentVal)) {
  450. tXinyiWarningRecord.setReason(category + EXCEPTION_WARNING);
  451. tXinyiWarningRecord.setLevel(WARNING_LEVEL_NO_DATE);
  452. } else if (currentVal.compareTo(multiply) > 0) {//一级
  453. tXinyiWarningRecord.setReason(category + CHAOBIAO_WARNING);
  454. tXinyiWarningRecord.setLevel(WARNING_LEVEL_ONE);
  455. } else if (currentVal.compareTo(bzz) >= 0 && currentVal.compareTo(multiply) <= 0) {//二级
  456. tXinyiWarningRecord.setReason(category + CHAOBIAO_WARNING);
  457. tXinyiWarningRecord.setLevel(WARNING_LEVEL_TWO);
  458. } else if (!Objects.isNull(gkz) && currentVal.compareTo(gkz) > 0) {
  459. tXinyiWarningRecord.setReason(category + CHAOGUANKONG_WARNING);
  460. tXinyiWarningRecord.setLevel(WARNING_LEVEL_THREE);
  461. } else {
  462. tXinyiWarningRecord = null;//这种的无需处理
  463. }
  464. //当前状态正常 需要查询历史有无正在报警的数据,如果有,将报警状态改完2(系统自动关闭)
  465. List<TXinyiWarningRecord> tXinyiWarningRecords = this.xinyiWarningRecordMapper.selectTXinyiWarningRecordList(TXinyiWarningRecord.builder().delFlag(0).type(1).category(category).status(0).build());
  466. if(Objects.isNull(tXinyiWarningRecord)){//数据正常,无告警信息
  467. if(!CollectionUtils.isEmpty(tXinyiWarningRecords)){
  468. log.info( "{}:现在恢复正常,历史报警数据为{}", category,JSON.toJSONString(tXinyiWarningRecords));
  469. for (TXinyiWarningRecord xinyiWarningRecord : tXinyiWarningRecords) {
  470. xinyiWarningRecord.setStatus(2);
  471. Date nowDate = DateUtils.getNowDate();
  472. xinyiWarningRecord.setOffTime(nowDate);
  473. xinyiWarningRecord.setUpdateTime(nowDate);
  474. xinyiWarningRecord.setUpdateBy(WARNING_DEFAULT_CREATE);
  475. this.xinyiWarningRecordMapper.updateTXinyiWarningRecord(xinyiWarningRecord);
  476. }
  477. }
  478. }else{//有新的告警信息
  479. if(CollectionUtils.isEmpty(tXinyiWarningRecords)){//之前没有告警记录
  480. //保存到数据库中
  481. this.xinyiWarningRecordMapper.insertTXinyiWarningRecord(tXinyiWarningRecord);
  482. if(WARNING_LEVEL_NO_DATE.equals(tXinyiWarningRecord.getLevel())){
  483. //只保存一个普通的问答记录 不需要调用决策信息,但是实时数据还是要记录的
  484. this.addChatRecordByWarning(tXinyiWarningRecord, tXinyiIndustry, normConfig);
  485. }else {
  486. //继续调用决策(普通问答)
  487. this.askBigModelForSHWarning(tXinyiWarningRecord, tXinyiIndustry, normConfig);
  488. }
  489. }else{
  490. log.info("{}:之前已经有过告警记录了,且还是继续报警,无需重复添加报警,但是决策仍然要调用", category);
  491. for (TXinyiWarningRecord xinyiWarningRecord : tXinyiWarningRecords) {//理论上只有一个的
  492. if(WARNING_LEVEL_NO_DATE.equals(tXinyiWarningRecord.getLevel())){
  493. //只保存一个普通的问答记录 不需要调用决策信息,但是实时数据还是要记录的
  494. this.addChatRecordByWarning(tXinyiWarningRecord, tXinyiIndustry, normConfig);
  495. }else {
  496. //继续调用决策(普通问答)
  497. this.askBigModelForSHWarning(xinyiWarningRecord, tXinyiIndustry, normConfig);
  498. }
  499. }
  500. }
  501. }
  502. }
  503. private void askBigModelForSHWarning(TXinyiWarningRecord xinyiWarningRecord, TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig) {
  504. log.info("进入了后台接口调⽤⼤模型获取问答结果处理");
  505. StringBuilder sb = new StringBuilder();
  506. String sessionId = IdUtils.simpleUUID();
  507. ChatReq chatReq = new ChatReq();
  508. // String ipAddr = IpUtils.getIpAddr();//获取用户的ip地址 传给大模型
  509. String ipAddr = "";//获取用户的ip地址 传给大模型 定时任务获取不到ip地址
  510. int counts = 1;//默认是第一次
  511. //这种问答 没有历史问答的概念 直接把问题扔进去就行 无需查询历史记录
  512. List<String> historyDates = new ArrayList<>();
  513. //构建问题(替换提示词中的占位符)
  514. String shWarningPrompt = SH_WARNING_PROMPT;
  515. shWarningPrompt =shWarningPrompt.replace("#{0}", xinyiWarningRecord.getReason());
  516. shWarningPrompt =shWarningPrompt.replace("#{1}", String.valueOf(xinyiWarningRecord.getDesignVal()));
  517. shWarningPrompt =shWarningPrompt.replace("#{2}", String.valueOf(xinyiWarningRecord.getControlVal()));
  518. shWarningPrompt =shWarningPrompt.replace("#{3}", String.valueOf(xinyiWarningRecord.getWarningVal()));
  519. historyDates.add(shWarningPrompt);
  520. // 获取输出流
  521. ManagedChannel channel = null;
  522. try {
  523. channel = ManagedChannelBuilder.forAddress("10.0.0.24", 17070)
  524. .usePlaintext()
  525. .build();
  526. InferenceAPIsServiceGrpc.InferenceAPIsServiceBlockingStub stub = InferenceAPIsServiceGrpc.newBlockingStub(channel);
  527. String dataJson = "{\"bot_id\":\"721\",\"exp_id\":\"721\",\"session_id\":\"" + sessionId + "\",\"use_rag\":\"true\",\"prompt\":\"你是⼀个资深⽔务领域专家,能回答各种⽔务相关问题\",\"history_dia\":" + JSON.toJSONString(historyDates) + ",\"generate_args\":{\"max_new_tokens\":2048,\"max_length\":4096,\"num_beams\":1,\"do_sample\":true,\"top_p\":0.7,\"temperature\":0.95},\"extra\":{ \"ip_address\": \"" + ipAddr + "\" },\"strengthen\":" + (true) + "}";
  528. log.info("请求大模型的问答参数为{}", dataJson);
  529. PredictionsRequest request = PredictionsRequest.newBuilder()
  530. .setModelName("slibra_bot")
  531. .putInput("method", ByteString.copyFrom("infer_stream", "utf-8"))//推理
  532. .putInput("data", ByteString.copyFrom(dataJson, "utf-8"))
  533. .buildPartial();
  534. Iterator<PredictionResponse> predictions = stub.streamPredictions(request);
  535. //将结果记录到问答表
  536. while (predictions.hasNext()) {
  537. String responseStr = predictions.next().getPrediction().toStringUtf8();
  538. log.info("大模型问答返回的原始结果为{}", responseStr);
  539. responseStr = JSON.parseObject(responseStr).getString("message");
  540. if("complete".equals(responseStr)){
  541. log.info("结尾语句并且是非JSON,无需处理");
  542. }else{
  543. sb.append(responseStr);
  544. }
  545. }
  546. //将问答更新到数据库中
  547. chatReq.setSessionId(sessionId);
  548. chatReq.setType(1);//0问答 1决策 2本地 3仿真预测
  549. chatReq.setModule(3);//0专家问答 1智能工单 2智能体助手 3告警 4简报
  550. String showVal = this.buildShowValue(xinyiWarningRecord, tXinyiIndustry, normConfig);
  551. chatReq.setShowVal(showVal);
  552. chatReq.setQuestion(shWarningPrompt);
  553. chatReq.setAnswer(sb.toString());
  554. chatReq.setWarningId(String.valueOf(xinyiWarningRecord.getId()));
  555. chatReq.setCounts(counts);//问答次数
  556. chatReq.setUserId(WARNING_DEFAULT_CREATE);
  557. chatReq.setCreateBy(WARNING_DEFAULT_CREATE);
  558. chatReq.setCreateTime(DateUtils.getNowDate());
  559. this.xinyiChatRecordMapper.insertTXinyiChatRecord(chatReq);
  560. } catch (IOException e) {
  561. throw new RuntimeException(e);
  562. } finally {
  563. // 关闭输出流
  564. channel.shutdown();
  565. }
  566. }
  567. private void handleSZWarning(TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig) {
  568. //出水相关
  569. //出水COD报警
  570. BigDecimal csCod = tXinyiIndustry.getCsCod();
  571. BigDecimal cscodBzz = normConfig.getCscodBzz();
  572. BigDecimal cscodGkz = normConfig.getCscodGkz();
  573. if(!Objects.isNull(cscodBzz)){
  574. handleXinYiWarningsCs(cscodBzz, csCod, cscodGkz, BusinessEnum.WarningCategoryEnum.CS_COD.getCode(), tXinyiIndustry, normConfig);
  575. }
  576. //出水总磷超标报警
  577. BigDecimal csTp = tXinyiIndustry.getCsTp();
  578. BigDecimal cszlBzz = normConfig.getCszlBzz();
  579. BigDecimal cszlGkz = normConfig.getCszlGkz();
  580. if(!Objects.isNull(cszlBzz)){
  581. handleXinYiWarningsCs(cszlBzz, csTp, cszlGkz, BusinessEnum.WarningCategoryEnum.CS_ZL.getCode(), tXinyiIndustry, normConfig);
  582. }
  583. //出水总氮超标报警
  584. BigDecimal csTn = tXinyiIndustry.getCsTn();
  585. BigDecimal cszzBzz = normConfig.getCszzBzz();
  586. BigDecimal cszzGkz = normConfig.getCszzGkz();
  587. if(!Objects.isNull(cszzBzz)){
  588. handleXinYiWarningsCs(cszzBzz, csTn, cszzGkz, BusinessEnum.WarningCategoryEnum.CS_ZD.getCode(), tXinyiIndustry, normConfig);
  589. }
  590. //出水氨氮超标报警
  591. BigDecimal csNh3 = tXinyiIndustry.getCsNh3();
  592. BigDecimal csadBzz = normConfig.getCsadBzz();
  593. BigDecimal csadGkz = normConfig.getCsadGkz();
  594. if(!Objects.isNull(csadBzz)){
  595. handleXinYiWarningsCs(csadBzz, csNh3, csadGkz, BusinessEnum.WarningCategoryEnum.CS_AD.getCode(), tXinyiIndustry, normConfig);
  596. }
  597. //出水SS超标报警
  598. BigDecimal csSS = tXinyiIndustry.getCsSs();
  599. BigDecimal csSSBzz = normConfig.getCsssBzz();
  600. BigDecimal csssGkz = normConfig.getCsssGkz();
  601. if(!Objects.isNull(csSSBzz)){
  602. handleXinYiWarningsCs(csSSBzz, csSS, csssGkz, BusinessEnum.WarningCategoryEnum.CS_SS.getCode(), tXinyiIndustry, normConfig);
  603. }
  604. //进水相关报警
  605. //进水总磷超标报警
  606. BigDecimal jsTp = tXinyiIndustry.getJsTp();
  607. BigDecimal jszlSjz = normConfig.getJszlSjz();
  608. if(!Objects.isNull(jszlSjz)){
  609. handleXinYiWarningRecordJS(jszlSjz, jsTp, BusinessEnum.WarningCategoryEnum.JS_ZL.getCode(), tXinyiIndustry, normConfig);
  610. }
  611. //进水COD超标报警
  612. BigDecimal jsCod = tXinyiIndustry.getJsCod();
  613. BigDecimal jscodSjz = normConfig.getJscodSjz();
  614. if(!Objects.isNull(jscodSjz)){
  615. handleXinYiWarningRecordJS(jscodSjz, jsCod, BusinessEnum.WarningCategoryEnum.JS_COD.getCode(), tXinyiIndustry, normConfig);
  616. }
  617. //进水总氮超标报警
  618. BigDecimal jsTn = tXinyiIndustry.getJsTn();
  619. BigDecimal jszdSjz = normConfig.getJszdSjz();
  620. if(!Objects.isNull(jszdSjz)){
  621. handleXinYiWarningRecordJS(jszdSjz, jsTn, BusinessEnum.WarningCategoryEnum.JS_ZD.getCode(), tXinyiIndustry, normConfig);
  622. }
  623. //进水氨氮超标报警
  624. BigDecimal jsNh3 = tXinyiIndustry.getJsNh3();
  625. BigDecimal jsadSjz = normConfig.getJsadSjz();
  626. if(!Objects.isNull(jsadSjz)){
  627. handleXinYiWarningRecordJS(jsadSjz, jsNh3, BusinessEnum.WarningCategoryEnum.JS_AD.getCode(), tXinyiIndustry, normConfig);
  628. }
  629. //进水SS超标报警
  630. BigDecimal jsSS = tXinyiIndustry.getJsSs();
  631. BigDecimal jsSSSjz = normConfig.getJsssSjz();
  632. if(!Objects.isNull(jsSSSjz)){
  633. handleXinYiWarningRecordJS(jsSSSjz, jsSS, BusinessEnum.WarningCategoryEnum.JS_SS.getCode(), tXinyiIndustry, normConfig);
  634. }
  635. }
  636. /**
  637. * 调用大模型获取决策结果 并同时记录对应信息到聊天记录表中
  638. *
  639. * @param tXinyiWarningRecord
  640. * @param tXinyiIndustry
  641. * @param normConfig
  642. */
  643. private void handleDecision(TXinyiWarningRecord tXinyiWarningRecord, TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig) {
  644. log.info("进入了调⽤大模型决策接口");
  645. ChatReq chatReq = new ChatReq();
  646. // StringBuilder sb = new StringBuilder();
  647. //大模型结果 放入一个结合中
  648. List<String> resultData = new ArrayList<>();
  649. //决策和问答不一样 没有历史的概念 所以sessionId都是新的 次数都是1
  650. String sessionId = IdUtils.simpleUUID();
  651. String feedback = chatReq.getFeedback();
  652. //决策请求的业务参数
  653. // List<HashMap<String, Object>> list = this.xinyiIndustryMapper.selectLast10RecordsForDecision();
  654. //2024年5月21日15:23:07 这里不能用关联查询处理,日报要获取最新的一条而不是今日的数据。
  655. List<DecisionReq> decisionReqs = this.xinyiIndustryMapper.selectLast10RecordsForDecisionOnlyIndustry();
  656. if(!CollectionUtils.isEmpty(decisionReqs)){
  657. for (DecisionReq decisionReq : decisionReqs) {
  658. //处理日报数据
  659. TXinyiDaily daily = this.xinyiDailyMapper.selectNewestData();
  660. if(!Objects.isNull(daily)){
  661. decisionReq.setT(daily.getJsSw());
  662. decisionReq.setSVI(daily.getWntjzsAll());
  663. decisionReq.setSV(daily.getWncjbAll());
  664. decisionReq.setMlvss(daily.getHfxwnndAll());
  665. decisionReq.setDoAna(daily.getYycRjyAll());
  666. decisionReq.setDoQue(daily.getQycRjyAll());
  667. //2024年5月26日11:59:02 干污泥量数据做了同步
  668. decisionReq.setGwnl(daily.getGWNL());
  669. BigDecimal jsBod5 = daily.getJsBod5();
  670. decisionReq.setBodIn(jsBod5);
  671. BigDecimal tpIn = decisionReq.getTpIn();
  672. BigDecimal tnIn = decisionReq.getTnIn();
  673. if(!Objects.isNull(jsBod5)){
  674. if(!Objects.isNull(tpIn) && tpIn.compareTo(new BigDecimal("0")) >0){
  675. decisionReq.setC(jsBod5.divide(tpIn, 4, RoundingMode.HALF_UP));
  676. }
  677. if(!Objects.isNull(tnIn) && tnIn.compareTo(new BigDecimal("0")) >0){
  678. decisionReq.setB(jsBod5.divide(tnIn, 4, RoundingMode.HALF_UP));
  679. }
  680. }
  681. }
  682. }
  683. }
  684. String rows = JSON.toJSONString(decisionReqs, JSONWriter.Feature.WriteNulls);
  685. // 获取输出流
  686. ManagedChannel channel = null;
  687. String dataJson = "";
  688. try {
  689. channel = ManagedChannelBuilder.forAddress("10.0.0.24", 17070)
  690. .usePlaintext()
  691. .build();
  692. InferenceAPIsServiceGrpc.InferenceAPIsServiceBlockingStub stub = InferenceAPIsServiceGrpc.newBlockingStub(channel);
  693. dataJson = "{\"bot_id\":\"b00001\",\"exp_id\":\"721\",\"norm\":\"" + tXinyiWarningRecord.getCategory() + "\",\"feedback\":" + feedback + ",\"session_id\":" + "\"" + sessionId + "\"" + ",\"generate_args\":{\"max_new_tokens\":1024,\"max_length\":4096,\"num_beams\":1,\"do_sample\":true,\"top_p\":0.7,\"temperature\":0.95},\"extra\":{\"rows\":" + rows + "}}";
  694. log.info("请求大模型的决策的参数为{}", dataJson);
  695. PredictionsRequest request = PredictionsRequest.newBuilder()
  696. .setModelName("slibra_bot")
  697. .putInput("method", ByteString.copyFrom("decision_stream", "utf-8"))//推理
  698. .putInput("data", ByteString.copyFrom(dataJson, "utf-8"))
  699. .buildPartial();
  700. Iterator<PredictionResponse> predictions = stub.streamPredictions(request);
  701. while (predictions.hasNext()) {
  702. String responseStr = predictions.next().getPrediction().toStringUtf8();
  703. log.info("决策流式返回的结果是{}", responseStr);
  704. //2024年5月25日16:37:16 按照大模型返回的类型解析数据
  705. String biz = JSON.parseObject(responseStr).getString("biz");
  706. if(BusinessEnum.BigModelBizEnum.OK.getCode().equals(biz)){
  707. log.info("结尾语句并且是非JSON,无需处理");
  708. //结束语句也流式输出,但是并不记录下来 2024年5月24日11:15:23 也不返回前端
  709. /*outputStream.write(responseStr.getBytes());
  710. outputStream.flush();*/
  711. }else if(BusinessEnum.BigModelBizEnum.DECISION_DEBUGGER.getCode().equals(biz)){
  712. log.info("中间过程,目前只打印日志,不记录数据,也不返回给前端,返回数据为{}", responseStr);
  713. //结束语句也流式输出,但是并不记录下来 2024年5月24日11:15:23 也不返回前端
  714. /*outputStream.write(responseStr.getBytes());
  715. outputStream.flush();*/
  716. }else{//其他 要么错误 要么alert 要么出的报告
  717. // sb.append(responseStr);
  718. resultData.add(responseStr);
  719. }
  720. }
  721. } catch (Exception e) {
  722. // throw new RuntimeException(e);
  723. log.error("定时任务处理告警调用决策异常,异常信息为{}", JSON.toJSONString(e));
  724. resultData.add("{\"biz\":\"ERROR\",\"message\":\"大模型分析数据异常,请稍后再试\"}");
  725. } finally {
  726. log.info("决策最终要保存的数据是{}", JSON.toJSONString(resultData));
  727. //保存聊天记录
  728. //将问答更新到数据库中
  729. chatReq.setSessionId(sessionId);
  730. chatReq.setType(1);//0问答 1决策
  731. chatReq.setModule(3);
  732. /*String userId = SecurityUtils.getUserId().toString();
  733. String username = SecurityUtils.getUsername();*/
  734. chatReq.setUserId(WARNING_DEFAULT_CREATE);
  735. String showVal = this.buildShowValue(tXinyiWarningRecord, tXinyiIndustry, normConfig);
  736. chatReq.setShowVal(showVal);//前端展示的数据和提问的数据不一致
  737. chatReq.setQuestion(dataJson);
  738. chatReq.setAnswer(JSON.toJSONString(resultData));
  739. chatReq.setWarningId(String.valueOf(tXinyiWarningRecord.getId()));
  740. chatReq.setCounts(1);//问答次数
  741. chatReq.setCreateBy(WARNING_DEFAULT_CREATE);
  742. chatReq.setCreateTime(DateUtils.getNowDate());
  743. this.xinyiChatRecordMapper.insertTXinyiChatRecord(chatReq);
  744. // 关闭输出流
  745. if(!Objects.isNull(channel))
  746. channel.shutdown();
  747. }
  748. }
  749. private String buildShowValue(TXinyiWarningRecord tXinyiWarningRecord, TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig) {
  750. JSONObject result = new JSONObject();
  751. JSONObject basic = new JSONObject();
  752. Integer status = tXinyiWarningRecord.getStatus();
  753. Date warningTime = tXinyiWarningRecord.getTime();
  754. int count = DateUtils.differentHoursByMillisecond(warningTime, DateUtils.getNowDate()) + 1;
  755. basic.put("title", tXinyiWarningRecord.getReason());
  756. basic.put("报警时间", DateUtils.parseDateToStr(DateUtils.YYYY_MM_DD_HH_MM ,warningTime));
  757. basic.put("报警值", tXinyiWarningRecord.getWarningVal());
  758. basic.put("标准值", tXinyiWarningRecord.getDesignVal());
  759. basic.put("管控值", tXinyiWarningRecord.getControlVal());
  760. basic.put("报警次数", Math.min(count, MAX_COUNT));
  761. if(tXinyiWarningRecord.getType() != 2)
  762. basic.put("状态", status == 0 ? "报警中" : status == 1 ? "用户关闭" : status == 2 ? "系统关闭" : "应急处理中");
  763. else
  764. basic.put("状态", status == 0 ? "预警中" : "已完成");
  765. //2024年5月27日14:04:22 额外返回2个字段 [管控值 和 告警级别] 返回的json没有数据是因为value没有值
  766. // basic.put("管控值", tXinyiWarningRecord.getControlVal());
  767. basic.put("告警级别", tXinyiWarningRecord.getLevel());
  768. result.put("basic", basic);
  769. JSONObject jsData = getJsonObject(tXinyiIndustry, normConfig);//进水数据
  770. result.put("jsData", jsData);
  771. JSONObject csData = getCsonObject(tXinyiIndustry, normConfig);//出水数据
  772. result.put("csData", csData);
  773. return JSON.toJSONString(result, JSONWriter.Feature.WriteNulls);
  774. }
  775. /**
  776. * 带图形的构建展示数据
  777. *
  778. * @param tXinyiWarningRecord
  779. * @param tXinyiIndustry
  780. * @param normConfig
  781. * @param date
  782. * @param yjHour
  783. * @return
  784. */
  785. private String buildShowValueWithCharts(TXinyiWarningRecord tXinyiWarningRecord, TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig, List<ChartBasic> chartsDataList, String date, String yjHour) {
  786. JSONObject result = new JSONObject();
  787. // JSONObject basic = new JSONObject();
  788. Integer status = tXinyiWarningRecord.getStatus();
  789. Date warningTime = tXinyiWarningRecord.getTime();
  790. String remark = tXinyiWarningRecord.getRemark();
  791. int count = DateUtils.differentHoursByMillisecond(warningTime, DateUtils.getNowDate()) + 1;
  792. ShowValueChartBasic showValueChartBasic = new ShowValueChartBasic();
  793. if("0".equals(remark)){//水质报警
  794. showValueChartBasic.setH("在线仪表");
  795. }else if("1".equals(remark)){//生化报警
  796. showValueChartBasic.setH("在线仪表");
  797. }else if("2".equals(remark)){//预测报错
  798. showValueChartBasic.setH("预测");
  799. }else {//机器人化验室报警
  800. showValueChartBasic.setH("化验室");
  801. }
  802. //通用的
  803. // showValueBasic.setA(tXinyiWarningRecord.getReason());
  804. showValueChartBasic.setB(DateUtils.parseDateToStr(DateUtils.YYYY_MM_DD_HH_MM ,warningTime));
  805. showValueChartBasic.setC(date + " " + yjHour);//预计超标时间
  806. showValueChartBasic.setD(tXinyiWarningRecord.getWarningVal());
  807. showValueChartBasic.setE(tXinyiWarningRecord.getForecastVal());
  808. showValueChartBasic.setF(tXinyiWarningRecord.getControlVal());
  809. showValueChartBasic.setG(tXinyiWarningRecord.getDesignVal());
  810. // showValueChartBasic.setH(Math.min(count, MAX_COUNT));
  811. if(tXinyiWarningRecord.getType() != 2)
  812. showValueChartBasic.setI(status == 0 ? "报警中" : status == 1 ? "用户关闭" : status == 2 ? "系统关闭" : "应急处理中");
  813. else
  814. showValueChartBasic.setI(status == 0 ? "预警中" : "已完成");
  815. result.put("basic", showValueChartBasic);
  816. JSONObject jsData = getJsonObject(tXinyiIndustry, normConfig);//进水数据
  817. result.put("jsData", jsData);
  818. JSONObject csData = getCsonObject(tXinyiIndustry, normConfig);//出水数据
  819. result.put("csData", csData);
  820. // JSONObject chartsData = getChartsCsonObject(tXinyiIndustry, normConfig);//图表数据 因为展示不同,所以要传过来或者判断
  821. result.put("chartsData", chartsDataList);
  822. result.put("chartsTitle", tXinyiWarningRecord.getCategory() + "数据趋势图");
  823. return JSON.toJSONString(result, JSONWriter.Feature.WriteNulls);
  824. }
  825. @Deprecated
  826. private JSONObject getChartsCsonObject(TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig) {
  827. JSONObject jsData = new JSONObject();
  828. //处理图表数据
  829. return jsData;
  830. }
  831. private static JSONObject getJsonObject(TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig) {
  832. JSONObject jsData = new JSONObject();
  833. HashMap<Object, Object> temp1 = new HashMap<>();
  834. BigDecimal jsSlq = tXinyiIndustry.getJsSlq();
  835. temp1.put("value", jsSlq);
  836. temp1.put("exceed", jsSlq.compareTo(normConfig.getJsslSjz()) >0);
  837. jsData.put("流量", temp1);
  838. HashMap<Object, Object> temp2 = new HashMap<>();
  839. BigDecimal jsCod = tXinyiIndustry.getJsCod();
  840. temp2.put("value", jsCod);
  841. temp2.put("exceed", jsCod.compareTo(normConfig.getJscodSjz()) > 0);
  842. jsData.put("COD", temp2);
  843. HashMap<Object, Object> temp3 = new HashMap<>();
  844. BigDecimal jsNh3 = tXinyiIndustry.getJsNh3();
  845. temp3.put("value", jsNh3);
  846. temp3.put("exceed", jsNh3.compareTo(normConfig.getJsadSjz()) > 0);
  847. jsData.put("NH3-N", temp3);
  848. HashMap<Object, Object> temp4 = new HashMap<>();
  849. BigDecimal jsTp = tXinyiIndustry.getJsTp();
  850. temp4.put("value", jsTp);
  851. temp4.put("exceed", jsTp.compareTo(normConfig.getJszlSjz()) > 0);
  852. jsData.put("TP", temp4);
  853. HashMap<Object, Object> temp5 = new HashMap<>();
  854. BigDecimal jsSs = tXinyiIndustry.getJsSs();
  855. temp5.put("value", jsSs);
  856. temp5.put("exceed", jsSs.compareTo(normConfig.getJsssSjz()) > 0);
  857. jsData.put("SS", temp5);
  858. HashMap<Object, Object> temp6 = new HashMap<>();
  859. BigDecimal jsTn = tXinyiIndustry.getJsTn();
  860. temp6.put("value", jsTn);
  861. temp6.put("exceed", jsTn.compareTo(normConfig.getJszdSjz()) > 0);
  862. jsData.put("TN", temp6);
  863. return jsData;
  864. }
  865. private static JSONObject getCsonObject(TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig) {
  866. JSONObject csData = new JSONObject();
  867. HashMap<Object, Object> temp1 = new HashMap<>();
  868. BigDecimal csSlq = tXinyiIndustry.getCsSlqc();
  869. temp1.put("value", csSlq);
  870. temp1.put("exceed", false);//出水水量没有管控值
  871. csData.put("流量", temp1);
  872. HashMap<Object, Object> temp2 = new HashMap<>();
  873. BigDecimal csCod = tXinyiIndustry.getCsCod();
  874. temp2.put("value", csCod);
  875. temp2.put("exceed", csCod.compareTo(normConfig.getCscodGkz()) > 0);
  876. csData.put("COD", temp2);
  877. HashMap<Object, Object> temp3 = new HashMap<>();
  878. BigDecimal csNh3 = tXinyiIndustry.getCsNh3();
  879. temp3.put("value", csNh3);
  880. temp3.put("exceed", csNh3.compareTo(normConfig.getCsadGkz()) > 0);
  881. csData.put("NH3-N", temp3);
  882. HashMap<Object, Object> temp4 = new HashMap<>();
  883. BigDecimal csTp = tXinyiIndustry.getCsTp();
  884. temp4.put("value", csTp);
  885. temp4.put("exceed", csTp.compareTo(normConfig.getCszlGkz()) > 0);
  886. csData.put("TP", temp4);
  887. HashMap<Object, Object> temp5 = new HashMap<>();
  888. BigDecimal csSs = tXinyiIndustry.getCsSs();
  889. temp5.put("value", csSs);
  890. temp5.put("exceed", csSs.compareTo(normConfig.getCsssGkz()) > 0);
  891. csData.put("SS", temp5);
  892. HashMap<Object, Object> temp6 = new HashMap<>();
  893. BigDecimal csTn = tXinyiIndustry.getCsTn();
  894. temp6.put("value", csTn);
  895. temp6.put("exceed", csTn.compareTo(normConfig.getCszzGkz()) > 0);
  896. csData.put("TN", temp6);
  897. return csData;
  898. }
  899. /**
  900. * 通过输入的值 生成对应类型的报警对象(出水)
  901. *
  902. * @param csBzz
  903. * @param currentVal
  904. * @param csGkz
  905. * @param category
  906. * @param tXinyiIndustry
  907. * @param normConfig
  908. * @return
  909. */
  910. private void handleXinYiWarningsCs(BigDecimal csBzz, BigDecimal currentVal, BigDecimal csGkz, String category, TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig) {
  911. BigDecimal multiply = csBzz.multiply(new BigDecimal(MyConstants.SCALE_VALUE));
  912. TXinyiWarningRecord tXinyiWarningRecord = new TXinyiWarningRecord();
  913. /*String category = BusinessEnum.WarningCategoryEnum.CS_AD.getCode();*/
  914. tXinyiWarningRecord.setStatus(0);
  915. tXinyiWarningRecord.setType(0);
  916. tXinyiWarningRecord.setCategory(category);
  917. tXinyiWarningRecord.setTime(DateUtils.getNowDate());
  918. tXinyiWarningRecord.setWarningVal(currentVal);
  919. tXinyiWarningRecord.setDesignVal(csBzz);
  920. tXinyiWarningRecord.setControlVal(csGkz);
  921. tXinyiWarningRecord.setCreateBy(WARNING_DEFAULT_CREATE);
  922. tXinyiWarningRecord.setCreateTime(DateUtils.getNowDate());
  923. //2024年5月25日17:52:33 如果工业库获取不到数据,也触发报警,但是不调用决策接口
  924. if(Objects.isNull(currentVal)){
  925. tXinyiWarningRecord.setReason(category + EXCEPTION_WARNING);
  926. tXinyiWarningRecord.setLevel(WARNING_LEVEL_NO_DATE);
  927. }else if(currentVal.compareTo(multiply) > 0){//一级
  928. tXinyiWarningRecord.setReason(category + CHAOBIAO_WARNING);
  929. tXinyiWarningRecord.setLevel(WARNING_LEVEL_ONE);
  930. }else if(currentVal.compareTo(csBzz) >= 0 && currentVal.compareTo(multiply) <= 0){//二级
  931. tXinyiWarningRecord.setReason(category + CHAOBIAO_WARNING);
  932. tXinyiWarningRecord.setLevel(WARNING_LEVEL_TWO);
  933. }else if(!Objects.isNull(csGkz) && currentVal.compareTo(csGkz) > 0){
  934. tXinyiWarningRecord.setReason(category + CHAOGUANKONG_WARNING);
  935. tXinyiWarningRecord.setLevel(WARNING_LEVEL_THREE);
  936. }else{
  937. tXinyiWarningRecord = null;//这种的无需处理
  938. }
  939. //当前状态正常 需要查询历史有无正在报警的数据,如果有,将报警状态改完2(系统自动关闭)
  940. List<TXinyiWarningRecord> tXinyiWarningRecords = this.xinyiWarningRecordMapper.selectTXinyiWarningRecordList(TXinyiWarningRecord.builder().delFlag(0).type(0).category(category).warningStatus(0).build());
  941. if(Objects.isNull(tXinyiWarningRecord)){//数据正常,无告警信息
  942. if(!CollectionUtils.isEmpty(tXinyiWarningRecords)){
  943. log.info( "{}:现在恢复正常,历史报警数据为{}", category,JSON.toJSONString(tXinyiWarningRecords));
  944. for (TXinyiWarningRecord xinyiWarningRecord : tXinyiWarningRecords) {
  945. xinyiWarningRecord.setStatus(2);
  946. Date nowDate = DateUtils.getNowDate();
  947. xinyiWarningRecord.setOffTime(nowDate);
  948. xinyiWarningRecord.setUpdateTime(nowDate);
  949. xinyiWarningRecord.setUpdateBy(WARNING_DEFAULT_CREATE);
  950. this.xinyiWarningRecordMapper.updateTXinyiWarningRecord(xinyiWarningRecord);
  951. }
  952. }
  953. }else{//有告警信息
  954. if(CollectionUtils.isEmpty(tXinyiWarningRecords)){//之前没有告警记录
  955. //保存到数据库中
  956. this.xinyiWarningRecordMapper.insertTXinyiWarningRecord(tXinyiWarningRecord);
  957. if(WARNING_LEVEL_NO_DATE.equals(tXinyiWarningRecord.getLevel())){
  958. //只保存一个普通的问答记录 不需要调用决策信息,但是实时数据还是要记录的
  959. this.addChatRecordByWarning(tXinyiWarningRecord, tXinyiIndustry, normConfig);
  960. }else {
  961. //继续调用决策
  962. this.handleDecision(tXinyiWarningRecord, tXinyiIndustry, normConfig);
  963. }
  964. }else{
  965. log.info("{}:之前已经有过告警记录了,且还是继续报警,无需重复添加报警,但是决策仍然要调用", category);
  966. for (TXinyiWarningRecord xinyiWarningRecord : tXinyiWarningRecords) {//理论上只有一个的
  967. if(WARNING_LEVEL_NO_DATE.equals(tXinyiWarningRecord.getLevel())){
  968. //只保存一个普通的问答记录 不需要调用决策信息,但是实时数据还是要记录的
  969. this.addChatRecordByWarning(tXinyiWarningRecord, tXinyiIndustry, normConfig);
  970. }else {
  971. //继续调用决策
  972. this.handleDecision(xinyiWarningRecord, tXinyiIndustry, normConfig);
  973. }
  974. }
  975. }
  976. }
  977. }
  978. /**
  979. * 通过输入的值 生成对应类型的报警对象(进水)
  980. *
  981. * @param jsBzz
  982. * @param currentVal
  983. * @param category
  984. * @param tXinyiIndustry
  985. * @param normConfig
  986. * @return
  987. */
  988. private void handleXinYiWarningRecordJS(BigDecimal jsBzz, BigDecimal currentVal, String category, TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig) {
  989. BigDecimal multiply = jsBzz.multiply(new BigDecimal(MyConstants.SCALE_VALUE));
  990. TXinyiWarningRecord tXinyiWarningRecord = new TXinyiWarningRecord();
  991. /*String category = BusinessEnum.WarningCategoryEnum.CS_AD.getCode();*/
  992. tXinyiWarningRecord.setStatus(0);
  993. tXinyiWarningRecord.setType(0);
  994. tXinyiWarningRecord.setCategory(category);
  995. tXinyiWarningRecord.setTime(DateUtils.getNowDate());
  996. tXinyiWarningRecord.setWarningVal(currentVal);
  997. tXinyiWarningRecord.setDesignVal(jsBzz);
  998. // tXinyiWarningRecord.setControlVal(csGkz);
  999. tXinyiWarningRecord.setCreateBy(WARNING_DEFAULT_CREATE);
  1000. tXinyiWarningRecord.setCreateTime(DateUtils.getNowDate());
  1001. //2024年5月25日17:52:33 如果工业库获取不到数据,也触发报警,但是不调用决策接口
  1002. if(Objects.isNull(currentVal)){
  1003. tXinyiWarningRecord.setReason(category + EXCEPTION_WARNING);
  1004. tXinyiWarningRecord.setLevel(WARNING_LEVEL_NO_DATE);
  1005. }else if(currentVal.compareTo(multiply) > 0){//一级
  1006. tXinyiWarningRecord.setReason(category + CHAOBIAO_WARNING);
  1007. tXinyiWarningRecord.setLevel(WARNING_LEVEL_ONE);
  1008. }else if(currentVal.compareTo(jsBzz) >= 0 && currentVal.compareTo(multiply) <= 0){//二级
  1009. tXinyiWarningRecord.setReason(category + CHAOBIAO_WARNING);
  1010. tXinyiWarningRecord.setLevel(WARNING_LEVEL_TWO);
  1011. }else{
  1012. tXinyiWarningRecord = null;//这种的无需处理
  1013. }
  1014. /*else if(!Objects.isNull(csGkz) && currentVal.compareTo(csGkz) > 0){
  1015. tXinyiWarningRecord.setReason(category + CHAOGUANKONG_WARNING);
  1016. tXinyiWarningRecord.setLevel(WARNING_LEVEL_THREE);
  1017. }*/
  1018. //当前状态正常 需要查询历史有无正在报警的数据,如果有,将报警状态改完2(系统自动关闭)
  1019. List<TXinyiWarningRecord> tXinyiWarningRecords = this.xinyiWarningRecordMapper.selectTXinyiWarningRecordList(TXinyiWarningRecord.builder().delFlag(0).type(0).category(category).status(0).build());
  1020. if(Objects.isNull(tXinyiWarningRecord)){//数据正常,无告警信息
  1021. if(!CollectionUtils.isEmpty(tXinyiWarningRecords)){
  1022. log.info( "{}:现在恢复正常,历史报警数据为{}", category,JSON.toJSONString(tXinyiWarningRecords));
  1023. for (TXinyiWarningRecord xinyiWarningRecord : tXinyiWarningRecords) {
  1024. xinyiWarningRecord.setStatus(2);
  1025. Date nowDate = DateUtils.getNowDate();
  1026. xinyiWarningRecord.setOffTime(nowDate);
  1027. xinyiWarningRecord.setUpdateTime(nowDate);
  1028. xinyiWarningRecord.setUpdateBy(WARNING_DEFAULT_CREATE);
  1029. this.xinyiWarningRecordMapper.updateTXinyiWarningRecord(xinyiWarningRecord);
  1030. }
  1031. }
  1032. }else{//有告警信息
  1033. if(CollectionUtils.isEmpty(tXinyiWarningRecords)){//之前没有告警记录
  1034. //保存到数据库中
  1035. this.xinyiWarningRecordMapper.insertTXinyiWarningRecord(tXinyiWarningRecord);
  1036. if(WARNING_LEVEL_NO_DATE.equals(tXinyiWarningRecord.getLevel())){
  1037. //只保存一个普通的问答记录 不需要调用决策信息,但是实时数据还是要记录的
  1038. this.addChatRecordByWarning(tXinyiWarningRecord, tXinyiIndustry, normConfig);
  1039. }else {
  1040. //继续调用决策
  1041. this.handleDecision(tXinyiWarningRecord, tXinyiIndustry, normConfig);
  1042. }
  1043. }else{
  1044. log.info("{}:之前已经有过告警记录了,且还是继续报警,无需重复添加报警,但是决策仍然要调用", category);
  1045. for (TXinyiWarningRecord xinyiWarningRecord : tXinyiWarningRecords) {//理论上只有一个的
  1046. if(WARNING_LEVEL_NO_DATE.equals(tXinyiWarningRecord.getLevel())){
  1047. //只保存一个普通的问答记录 不需要调用决策信息,但是实时数据还是要记录的
  1048. this.addChatRecordByWarning(tXinyiWarningRecord, tXinyiIndustry, normConfig);
  1049. }else {
  1050. //继续调用决策
  1051. this.handleDecision(xinyiWarningRecord, tXinyiIndustry, normConfig);
  1052. }
  1053. }
  1054. }
  1055. }
  1056. }
  1057. private void addChatRecordByWarning(TXinyiWarningRecord tXinyiWarningRecord, TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig) {
  1058. ChatReq chatReq = new ChatReq();
  1059. //保存聊天记录
  1060. //将问答更新到数据库中
  1061. chatReq.setSessionId(IdUtils.simpleUUID());
  1062. chatReq.setType(1);//0问答 1决策 2本地 3仿真预测
  1063. chatReq.setModule(3);
  1064. /*String userId = SecurityUtils.getUserId().toString();
  1065. String username = SecurityUtils.getUsername();*/
  1066. chatReq.setUserId(WARNING_DEFAULT_CREATE);
  1067. String showVal = this.buildShowValue(tXinyiWarningRecord, tXinyiIndustry, normConfig);
  1068. chatReq.setShowVal(showVal);//前端展示的数据和提问的数据不一致
  1069. chatReq.setQuestion(WARNING_DEFAULT_QUESTION);//本地问题
  1070. chatReq.setAnswer(tXinyiWarningRecord.getReason() + ",请检查设备是否正常运行");
  1071. chatReq.setWarningId(String.valueOf(tXinyiWarningRecord.getId()));
  1072. chatReq.setCounts(1);//问答次数
  1073. chatReq.setCreateBy(WARNING_DEFAULT_CREATE);
  1074. chatReq.setCreateTime(DateUtils.getNowDate());
  1075. this.xinyiChatRecordMapper.insertTXinyiChatRecord(chatReq);
  1076. }
  1077. /**
  1078. * 定时从sqlserver获取数据
  1079. */
  1080. public void sqlserverData(){
  1081. log.info("进入了定时同步SqlServer的任务");
  1082. //主库获取上次最新的同步日期
  1083. String lastTime = this.xinyiRobotMapper.selectLastTime();
  1084. log.info("上次同步的日期是{}", lastTime);
  1085. //从
  1086. DynamicDataSourceContextHolder.setDataSourceType(DataSourceType.SLAVE.name());
  1087. List<TXinyiRobot> tXinyiRobots = xinyiRobotMapper.selectTXinyiRobotListByTime(lastTime);
  1088. DynamicDataSourceContextHolder.clearDataSourceType();
  1089. // System.out.println(JSON.toJSONString(tXinyiRobots));
  1090. // System.out.println("-------------");
  1091. //主
  1092. List<TXinyiRobot> needHandleList = new ArrayList<>();
  1093. if(!CollectionUtils.isEmpty(tXinyiRobots)){
  1094. for (TXinyiRobot tXinyiRobot : tXinyiRobots) {
  1095. String date = handleDate(tXinyiRobot.getVDate().replaceAll(" ", ""));//有空格
  1096. String time = handleDate(tXinyiRobot.getVTime().replaceAll(" ", ""));//有空格
  1097. tXinyiRobot.setVDate(date);
  1098. tXinyiRobot.setVTime(time);
  1099. tXinyiRobot.setVDateTime(date + " " + time);
  1100. //处理给前端展示的字段
  1101. tXinyiRobot.setTestDate(date);//日期
  1102. tXinyiRobot.setTestHour(date + " " + time.substring(0, 2));//小时
  1103. tXinyiRobot.setTestTime(date + " " + time.substring(0, 5));//分钟
  1104. tXinyiRobot.setCreatedTime(new Date());
  1105. //2024年5月29日10:33:32 额外处理几个新增的字段 多个池子数据合并一个
  1106. List<BigDecimal> extraList = new ArrayList<>();
  1107. extraList.add(tXinyiRobot.getNo3Hlj1Jqr());
  1108. extraList.add(tXinyiRobot.getNo3Hlj2Jqr());
  1109. tXinyiRobot.setHycxsyAll(JSON.toJSONString(extraList));
  1110. extraList.clear();
  1111. extraList.add(tXinyiRobot.getNh31Jqr());
  1112. extraList.add(tXinyiRobot.getNh32Jqr());
  1113. tXinyiRobot.setQyanAll(JSON.toJSONString(extraList));
  1114. extraList.clear();
  1115. extraList.add(tXinyiRobot.getNo3Qyc1Jqr());
  1116. extraList.add(tXinyiRobot.getNo3Qyc2Jqr());
  1117. tXinyiRobot.setQyckxsyAll(JSON.toJSONString(extraList));
  1118. extraList.clear();
  1119. extraList.add(tXinyiRobot.getTpHl1Jqr());
  1120. extraList.add(tXinyiRobot.getTpHl2Jqr());
  1121. tXinyiRobot.setHyzlsyAll(JSON.toJSONString(extraList));
  1122. //按照小时填充集合 然后处理报警相关逻辑
  1123. //2024年6月17日13:32:59 因为机器人化验室数据基本几个小时的都一致 改成一小时取一次
  1124. needHandleList.add(tXinyiRobot);
  1125. // this.xinyiRobotMapper.insertTXinyiRobot(tXinyiRobot);
  1126. }
  1127. }
  1128. if(!CollectionUtils.isEmpty(needHandleList)){
  1129. List<TXinyiRobot> uniqueList = needHandleList.stream()
  1130. .collect(Collectors.collectingAndThen(
  1131. Collectors.toCollection(() -> new TreeSet<>(Comparator.comparing(TXinyiRobot::getTestHour))),
  1132. ArrayList::new
  1133. ));
  1134. if(!CollectionUtils.isEmpty(uniqueList))
  1135. for (TXinyiRobot tXinyiRobot : uniqueList) {
  1136. this.xinyiRobotMapper.insertTXinyiRobot(tXinyiRobot);
  1137. }
  1138. asyncTask.handleRobotWarning(uniqueList);
  1139. }
  1140. }
  1141. /**
  1142. *
  1143. * 定时生成每日简报数据
  1144. *
  1145. */
  1146. public void generageShortReport(){
  1147. log.info("进入了定时生成每日简报数据");
  1148. List<TXinyiDaily> dailyTwoRecords = this.xinyiDailyMapper.selectNRecords(2);
  1149. //正常不会有这种问题 因为日报有很多条
  1150. if(CollectionUtils.isEmpty(dailyTwoRecords) || dailyTwoRecords.size() < 2){
  1151. log.error("进入了定时生成每日简报数据 获取最新的2条数据不足,终止");
  1152. return;
  1153. }
  1154. //暂时不考虑因为没有填写日报 导致生成重复数据的问题(后续需要的话再添加)
  1155. //处理数据 并 拼装
  1156. String queryData = buildShortReportQueryData(dailyTwoRecords);
  1157. log.info("定时生成简报,组装好的请求大模型的参数为:{}", queryData);
  1158. if(StringUtils.isBlank(queryData)){
  1159. log.error("无法拼装请求数据!!!!!!");
  1160. return;
  1161. }
  1162. String showVal = formateDateStr(dailyTwoRecords.get(0).getTestDate()) + JIAN_BAO_END;
  1163. //调用模型 并保存结果
  1164. this.askBigModel(queryData, showVal);
  1165. log.info("定时生成简报任务结束~~~~~~~~~~~~~~");
  1166. }
  1167. /**
  1168. *
  1169. * 每小时发一一次预测
  1170. *
  1171. */
  1172. public void predictor(){
  1173. //获取最新的6条工业科数据
  1174. List<TXinyiIndustry> tXinyiIndustries6 = this.xinyiIndustryMapper.selectNIndustry(6);
  1175. List<TXinyiRobot> tXinyiRobots6 = this.xinyiRobotMapper.selectNRobot(6);
  1176. for (int i = 0; i < predictorArr.length; i++) {
  1177. String result = testPredictor(predictorArr[i]);
  1178. if(StringUtils.isBlank(result) || "error".equals(result)){
  1179. log.info("预测数据返回结果不符合解析条件,返回结果为{}", result);
  1180. continue;
  1181. }
  1182. JSONObject jsonObject = JSON.parseObject(result);
  1183. String task = jsonObject.getString("task");
  1184. String hour = jsonObject.getString("hour");
  1185. String pred = jsonObject.getString("pred");
  1186. String date = jsonObject.getString("date");
  1187. if(StringUtils.isNotBlank(pred) && pred.contains(",")){
  1188. String[] split = pred.split(",");
  1189. if(split.length != 3){
  1190. log.error("预测数据返回结果为{},长度不是3,无法正常解析", result);
  1191. //因为部分预测还不支持 所以不需要
  1192. // handleXinYiWarningsYC(cscodBzz, split, cscodGkz, BusinessEnum.WarningCategoryEnum.CS_COD_YC.getCode(), hour, normConfig);
  1193. continue;
  1194. }
  1195. //解析数据 处理报警 调研prompt 保存等
  1196. this.handlePredictorWarning(split, hour, task, tXinyiIndustries6, tXinyiRobots6, date);
  1197. }else {
  1198. log.error("预测数据返回结果为{},无法正常解析", result);
  1199. }
  1200. }
  1201. }
  1202. private void handlePredictorWarning(String[] split, String hour, String task, List<TXinyiIndustry> tXinyiIndustries6, List<TXinyiRobot> tXinyiRobots6, String date) {
  1203. //获取配置表
  1204. List<TXinyiNormConfig> tXinyiNormConfigs = this.xinyiNormConfigMapper.selectTXinyiNormConfigList(null);
  1205. if(CollectionUtils.isEmpty(tXinyiNormConfigs)) {
  1206. log.error( "未查询到配置信息");
  1207. return;
  1208. }
  1209. TXinyiNormConfig normConfig = tXinyiNormConfigs.get(0);
  1210. //获取最新的工业库的数据
  1211. TXinyiIndustry tXinyiIndustry = this.xinyiIndustryMapper.selectTXinyiIndustryNewest();
  1212. List<ChartBasic> chartsDataList = new ArrayList<>(9);
  1213. if("cod".equals(task)){
  1214. BigDecimal cscodBzz = normConfig.getCscodBzz();
  1215. BigDecimal cscodGkz = normConfig.getCscodGkz();
  1216. //处理展示数据
  1217. for (int i = 0; i < tXinyiIndustries6.size(); i++) {
  1218. TXinyiIndustry tXinyiIndustryTemp = tXinyiIndustries6.get(5 - i);
  1219. ChartBasic chartBasic = new ChartBasic();
  1220. chartBasic.setTime(tXinyiIndustryTemp.getTestHour().substring(11));
  1221. chartBasic.setVal(tXinyiIndustryTemp.getCsCod());
  1222. chartsDataList.add(chartBasic);
  1223. }
  1224. String[] hours = hour.split(",");
  1225. for (int i = 0; i < split.length; i++) {
  1226. String val = split[i];
  1227. ChartBasic chartBasic = new ChartBasic();
  1228. chartBasic.setTime(hours[i]);
  1229. chartBasic.setVal(new BigDecimal(val));
  1230. chartsDataList.add(chartBasic);
  1231. }
  1232. handleXinYiWarningsYC(cscodBzz, split, cscodGkz, BusinessEnum.WarningCategoryEnum.CS_COD_YC.getCode(), hour, normConfig,tXinyiIndustry.getCsCod(), tXinyiIndustry, chartsDataList, date);
  1233. }else if("ss".equals(task)){
  1234. BigDecimal csSSBzz = normConfig.getCsssBzz();
  1235. BigDecimal csssGkz = normConfig.getCsssGkz();
  1236. //处理展示数据
  1237. for (int i = 0; i < tXinyiIndustries6.size(); i++) {
  1238. TXinyiIndustry tXinyiIndustryTemp = tXinyiIndustries6.get(5 - i);
  1239. ChartBasic chartBasic = new ChartBasic();
  1240. chartBasic.setTime(tXinyiIndustryTemp.getTestHour().substring(11));
  1241. chartBasic.setVal(tXinyiIndustryTemp.getCsSs());
  1242. chartsDataList.add(chartBasic);
  1243. }
  1244. String[] hours = hour.split(",");
  1245. for (int i = 0; i < split.length; i++) {
  1246. String val = split[i];
  1247. ChartBasic chartBasic = new ChartBasic();
  1248. chartBasic.setTime(hours[i]);
  1249. chartBasic.setVal(new BigDecimal(val));
  1250. chartsDataList.add(chartBasic);
  1251. }
  1252. handleXinYiWarningsYC(csSSBzz, split, csssGkz, BusinessEnum.WarningCategoryEnum.CS_SS_YC.getCode(), hour, normConfig,tXinyiIndustry.getCsCod(), tXinyiIndustry, chartsDataList, date);
  1253. }else if("tn".equals(task)){
  1254. BigDecimal cszzBzz = normConfig.getCszzBzz();
  1255. BigDecimal cszzGkz = normConfig.getCszzGkz();
  1256. //处理展示数据
  1257. for (int i = 0; i < tXinyiIndustries6.size(); i++) {
  1258. TXinyiIndustry tXinyiIndustryTemp = tXinyiIndustries6.get(5 - i);
  1259. ChartBasic chartBasic = new ChartBasic();
  1260. chartBasic.setTime(tXinyiIndustryTemp.getTestHour().substring(11));
  1261. chartBasic.setVal(tXinyiIndustryTemp.getCsTn());
  1262. chartsDataList.add(chartBasic);
  1263. }
  1264. String[] hours = hour.split(",");
  1265. for (int i = 0; i < split.length; i++) {
  1266. String val = split[i];
  1267. ChartBasic chartBasic = new ChartBasic();
  1268. chartBasic.setTime(hours[i]);
  1269. chartBasic.setVal(new BigDecimal(val));
  1270. chartsDataList.add(chartBasic);
  1271. }
  1272. handleXinYiWarningsYC(cszzBzz, split, cszzGkz, BusinessEnum.WarningCategoryEnum.CS_TN_YC.getCode(), hour, normConfig,tXinyiIndustry.getCsCod(), tXinyiIndustry, chartsDataList, date);
  1273. }else if("tp".equals(task)){
  1274. BigDecimal cszlBzz = normConfig.getCszlBzz();
  1275. BigDecimal cszlGkz = normConfig.getCszlGkz();
  1276. //处理展示数据
  1277. for (int i = 0; i < tXinyiIndustries6.size(); i++) {
  1278. TXinyiIndustry tXinyiIndustryTemp = tXinyiIndustries6.get(5 - i);
  1279. ChartBasic chartBasic = new ChartBasic();
  1280. chartBasic.setTime(tXinyiIndustryTemp.getTestHour().substring(11));
  1281. chartBasic.setVal(tXinyiIndustryTemp.getCsTp());
  1282. chartsDataList.add(chartBasic);
  1283. }
  1284. String[] hours = hour.split(",");
  1285. for (int i = 0; i < split.length; i++) {
  1286. String val = split[i];
  1287. ChartBasic chartBasic = new ChartBasic();
  1288. chartBasic.setTime(hours[i]);
  1289. chartBasic.setVal(new BigDecimal(val));
  1290. chartsDataList.add(chartBasic);
  1291. }
  1292. handleXinYiWarningsYC(cszlBzz, split, cszlGkz, BusinessEnum.WarningCategoryEnum.CS_TP_YC.getCode(), hour, normConfig,tXinyiIndustry.getCsCod(), tXinyiIndustry, chartsDataList, date);
  1293. }else if("nh3".equals(task)){
  1294. BigDecimal csadBzz = normConfig.getCsadBzz();
  1295. BigDecimal csadGkz = normConfig.getCsadGkz();
  1296. //处理展示数据
  1297. for (int i = 0; i < tXinyiIndustries6.size(); i++) {
  1298. TXinyiIndustry tXinyiIndustryTemp = tXinyiIndustries6.get(5 - i);
  1299. ChartBasic chartBasic = new ChartBasic();
  1300. chartBasic.setTime(tXinyiIndustryTemp.getTestHour().substring(11));
  1301. chartBasic.setVal(tXinyiIndustryTemp.getCsNh3());
  1302. chartsDataList.add(chartBasic);
  1303. }
  1304. String[] hours = hour.split(",");
  1305. for (int i = 0; i < split.length; i++) {
  1306. String val = split[i];
  1307. ChartBasic chartBasic = new ChartBasic();
  1308. chartBasic.setTime(hours[i]);
  1309. chartBasic.setVal(new BigDecimal(val));
  1310. chartsDataList.add(chartBasic);
  1311. }
  1312. handleXinYiWarningsYC(csadBzz, split, csadGkz, BusinessEnum.WarningCategoryEnum.CS_AD_YC.getCode(), hour, normConfig,tXinyiIndustry.getCsCod(), tXinyiIndustry, chartsDataList, date);
  1313. }else if("xsy1".equals(task)){
  1314. BigDecimal cscodBzz = new BigDecimal(XIAOSUANYAN_BZK);
  1315. BigDecimal cscodGkz = new BigDecimal(XIAOSUANYAN_GKZ);
  1316. //处理展示数据
  1317. for (int i = 0; i < tXinyiRobots6.size(); i++) {
  1318. TXinyiRobot tXinyiRobotTemp = tXinyiRobots6.get(5 - i);
  1319. ChartBasic chartBasic = new ChartBasic();
  1320. chartBasic.setTime(tXinyiRobotTemp.getTestHour().substring(11));
  1321. chartBasic.setVal(tXinyiRobotTemp.getNo3Hlj1Jqr());
  1322. chartsDataList.add(chartBasic);
  1323. }
  1324. String[] hours = hour.split(",");
  1325. for (int i = 0; i < split.length; i++) {
  1326. String val = split[i];
  1327. ChartBasic chartBasic = new ChartBasic();
  1328. chartBasic.setTime(hours[i]);
  1329. chartBasic.setVal(new BigDecimal(val));
  1330. chartsDataList.add(chartBasic);
  1331. }
  1332. handleXinYiWarningsYC(cscodBzz, split, cscodGkz, BusinessEnum.WarningCategoryEnum.CS_XSY_1_YC.getCode(), hour, normConfig,tXinyiIndustry.getCsCod(), tXinyiIndustry, chartsDataList, date);
  1333. }else if("xsy2".equals(task)){
  1334. BigDecimal cscodBzz = new BigDecimal(XIAOSUANYAN_BZK);
  1335. BigDecimal cscodGkz = new BigDecimal(XIAOSUANYAN_GKZ);
  1336. //处理展示数据
  1337. for (int i = 0; i < tXinyiRobots6.size(); i++) {
  1338. TXinyiRobot tXinyiRobotTemp = tXinyiRobots6.get(5 - i);
  1339. ChartBasic chartBasic = new ChartBasic();
  1340. chartBasic.setTime(tXinyiRobotTemp.getTestHour().substring(11));
  1341. chartBasic.setVal(tXinyiRobotTemp.getNo3Hlj2Jqr());
  1342. chartsDataList.add(chartBasic);
  1343. }
  1344. String[] hours = hour.split(",");
  1345. for (int i = 0; i < split.length; i++) {
  1346. String val = split[i];
  1347. ChartBasic chartBasic = new ChartBasic();
  1348. chartBasic.setTime(hours[i]);
  1349. chartBasic.setVal(new BigDecimal(val));
  1350. chartsDataList.add(chartBasic);
  1351. }
  1352. handleXinYiWarningsYC(cscodBzz, split, cscodGkz, BusinessEnum.WarningCategoryEnum.CS_XSY_2_YC.getCode(), hour, normConfig,tXinyiIndustry.getCsCod(), tXinyiIndustry, chartsDataList, date);
  1353. }else {
  1354. log.error("暂未支持的类型{}", task);
  1355. }
  1356. }
  1357. private void handleXinYiWarningsYC(BigDecimal csBzz, String[] split, BigDecimal csGkz, String category, String hour, TXinyiNormConfig normConfig, BigDecimal currentVal, TXinyiIndustry tXinyiIndustry, List<ChartBasic> chartsDataList, String date) {
  1358. BigDecimal multiply = csBzz.multiply(new BigDecimal(MyConstants.SCALE_VALUE));
  1359. TXinyiWarningRecord tXinyiWarningRecord = null;
  1360. String yjHour = null;
  1361. for (int i = 0, splitLength = split.length; i < splitLength; i++) {
  1362. String forecast = split[i];
  1363. BigDecimal forecastVal = new BigDecimal("10000");
  1364. tXinyiWarningRecord = new TXinyiWarningRecord();
  1365. /*String category = BusinessEnum.WarningCategoryEnum.CS_AD.getCode();*/
  1366. tXinyiWarningRecord.setStatus(0);
  1367. tXinyiWarningRecord.setType(2);
  1368. tXinyiWarningRecord.setCategory(category);
  1369. tXinyiWarningRecord.setTime(DateUtils.getNowDate());
  1370. tXinyiWarningRecord.setWarningVal(currentVal);//当前值
  1371. tXinyiWarningRecord.setForecastVal(forecastVal);
  1372. tXinyiWarningRecord.setDesignVal(csBzz);
  1373. tXinyiWarningRecord.setControlVal(csGkz);
  1374. tXinyiWarningRecord.setCreateBy(WARNING_DEFAULT_CREATE);
  1375. tXinyiWarningRecord.setCreateTime(DateUtils.getNowDate());
  1376. tXinyiWarningRecord.setRemark("2");
  1377. //2024年5月25日17:52:33 如果工业库获取不到数据,也触发报警,但是不调用决策接口
  1378. if (Objects.isNull(forecastVal)) {
  1379. tXinyiWarningRecord.setReason(category + EXCEPTION_WARNING_YC);
  1380. tXinyiWarningRecord.setLevel(WARNING_LEVEL_NO_DATE);
  1381. } else if (forecastVal.compareTo(multiply) > 0) {//一级
  1382. tXinyiWarningRecord.setReason(category + CHAOBIAO_WARNING_YC);
  1383. tXinyiWarningRecord.setLevel(WARNING_LEVEL_ONE);
  1384. } else if (forecastVal.compareTo(csBzz) >= 0 && forecastVal.compareTo(multiply) <= 0) {//二级
  1385. tXinyiWarningRecord.setReason(category + CHAOBIAO_WARNING_YC);
  1386. tXinyiWarningRecord.setLevel(WARNING_LEVEL_TWO);
  1387. } else if (!Objects.isNull(csGkz) && forecastVal.compareTo(csGkz) > 0) {
  1388. tXinyiWarningRecord.setReason(category + CHAOGUANKONG_WARNING_YC);
  1389. tXinyiWarningRecord.setLevel(WARNING_LEVEL_THREE);
  1390. } else {
  1391. tXinyiWarningRecord = null;//这种的无需处理
  1392. }
  1393. if (!Objects.isNull(tXinyiWarningRecord)) {
  1394. log.info("预测报警中出现了超标的情况,循环可以退出了");
  1395. yjHour = hour.split(",")[i];
  1396. break;
  1397. }
  1398. }
  1399. //当前状态正常 需要查询历史有无正在报警的数据,如果有,将报警状态改完2(系统自动关闭)
  1400. List<TXinyiWarningRecord> tXinyiWarningRecords = this.xinyiWarningRecordMapper.selectTXinyiWarningRecordList(TXinyiWarningRecord.builder().delFlag(0).type(2).category(category).warningStatus(0).build());
  1401. if(Objects.isNull(tXinyiWarningRecord)){//数据正常,无告警信息
  1402. if(!CollectionUtils.isEmpty(tXinyiWarningRecords)){
  1403. log.info( "{}:现在恢复正常,历史报警数据为{}", category,JSON.toJSONString(tXinyiWarningRecords));
  1404. for (TXinyiWarningRecord xinyiWarningRecord : tXinyiWarningRecords) {
  1405. xinyiWarningRecord.setStatus(2);
  1406. Date nowDate = DateUtils.getNowDate();
  1407. xinyiWarningRecord.setOffTime(nowDate);
  1408. xinyiWarningRecord.setUpdateTime(nowDate);
  1409. xinyiWarningRecord.setUpdateBy(WARNING_DEFAULT_CREATE);
  1410. this.xinyiWarningRecordMapper.updateTXinyiWarningRecord(xinyiWarningRecord);
  1411. }
  1412. }
  1413. }else{//有告警信息
  1414. if(CollectionUtils.isEmpty(tXinyiWarningRecords)){//之前没有告警记录
  1415. //保存到数据库中
  1416. this.xinyiWarningRecordMapper.insertTXinyiWarningRecord(tXinyiWarningRecord);
  1417. if(WARNING_LEVEL_NO_DATE.equals(tXinyiWarningRecord.getLevel())){
  1418. //只保存一个普通的问答记录 不需要调用决策信息,但是实时数据还是要记录的
  1419. this.addChatRecordByWarning(tXinyiWarningRecord, tXinyiIndustry, normConfig);
  1420. }else {
  1421. //继续调用大模型prompt
  1422. this.askBigModelForYC(tXinyiWarningRecord, tXinyiIndustry, normConfig, chartsDataList, date, yjHour);
  1423. }
  1424. }else{
  1425. log.info("{}:之前已经有过告警记录了,且还是继续报警,无需重复添加报警,但是决策仍然要调用", category);
  1426. for (TXinyiWarningRecord xinyiWarningRecord : tXinyiWarningRecords) {//理论上只有一个的
  1427. if(WARNING_LEVEL_NO_DATE.equals(tXinyiWarningRecord.getLevel())){
  1428. //只保存一个普通的问答记录 不需要调用决策信息,但是实时数据还是要记录的
  1429. this.addChatRecordByWarning(tXinyiWarningRecord, tXinyiIndustry, normConfig);
  1430. }else {
  1431. //继续调用决策
  1432. this.askBigModelForYC(xinyiWarningRecord, tXinyiIndustry, normConfig, chartsDataList, date, yjHour);
  1433. }
  1434. }
  1435. }
  1436. }
  1437. }
  1438. private void askBigModelForYC(TXinyiWarningRecord xinyiWarningRecord, TXinyiIndustry tXinyiIndustry, TXinyiNormConfig normConfig, List<ChartBasic> chartsDataList, String date, String yjHour) {
  1439. log.info("预测进入了后台接口调⽤⼤模型获取问答结果处理(预测)");
  1440. StringBuilder sb = new StringBuilder();
  1441. String sessionId = IdUtils.simpleUUID();
  1442. ChatReq chatReq = new ChatReq();
  1443. // String ipAddr = IpUtils.getIpAddr();//获取用户的ip地址 传给大模型
  1444. String ipAddr = "";//获取用户的ip地址 传给大模型 定时任务获取不到ip地址
  1445. int counts = 1;//默认是第一次
  1446. //这种问答 没有历史问答的概念 直接把问题扔进去就行 无需查询历史记录
  1447. List<String> historyDates = new ArrayList<>();
  1448. //构建问题(替换提示词中的占位符)
  1449. String shWarningPrompt = YC_WARNING_PROMPT;
  1450. shWarningPrompt =shWarningPrompt.replace("#{0}", xinyiWarningRecord.getReason());
  1451. shWarningPrompt =shWarningPrompt.replace("#{1}", String.valueOf(xinyiWarningRecord.getDesignVal()));
  1452. shWarningPrompt =shWarningPrompt.replace("#{2}", String.valueOf(xinyiWarningRecord.getWarningVal()));
  1453. shWarningPrompt =shWarningPrompt.replace("#{3}", String.valueOf(xinyiWarningRecord.getForecastVal()));
  1454. historyDates.add(shWarningPrompt);
  1455. // 获取输出流
  1456. ManagedChannel channel = null;
  1457. try {
  1458. channel = ManagedChannelBuilder.forAddress("10.0.0.24", 17070)
  1459. .usePlaintext()
  1460. .build();
  1461. InferenceAPIsServiceGrpc.InferenceAPIsServiceBlockingStub stub = InferenceAPIsServiceGrpc.newBlockingStub(channel);
  1462. String dataJson = "{\"bot_id\":\"721\",\"exp_id\":\"721\",\"session_id\":\"" + sessionId + "\",\"use_rag\":\"true\",\"prompt\":\"你是⼀个资深⽔务领域专家,能回答各种⽔务相关问题\",\"history_dia\":" + JSON.toJSONString(historyDates) + ",\"generate_args\":{\"max_new_tokens\":2048,\"max_length\":4096,\"num_beams\":1,\"do_sample\":true,\"top_p\":0.7,\"temperature\":0.95},\"extra\":{ \"ip_address\": \"" + ipAddr + "\" },\"strengthen\":" + (true) + "}";
  1463. log.info("请求大模型的问答参数为{}", dataJson);
  1464. PredictionsRequest request = PredictionsRequest.newBuilder()
  1465. .setModelName("slibra_bot")
  1466. .putInput("method", ByteString.copyFrom("infer_stream", "utf-8"))//推理
  1467. .putInput("data", ByteString.copyFrom(dataJson, "utf-8"))
  1468. .buildPartial();
  1469. Iterator<PredictionResponse> predictions = stub.streamPredictions(request);
  1470. //将结果记录到问答表
  1471. while (predictions.hasNext()) {
  1472. String responseStr = predictions.next().getPrediction().toStringUtf8();
  1473. log.info("大模型问答返回的原始结果为{}", responseStr);
  1474. responseStr = JSON.parseObject(responseStr).getString("message");
  1475. if("complete".equals(responseStr)){
  1476. log.info("结尾语句并且是非JSON,无需处理");
  1477. }else{
  1478. sb.append(responseStr);
  1479. }
  1480. }
  1481. //将问答更新到数据库中
  1482. chatReq.setSessionId(sessionId);
  1483. chatReq.setType(3);//0问答 1决策 2本地 3仿真预测
  1484. chatReq.setModule(3);//0专家问答 1智能工单 2智能体助手 3告警 4简报
  1485. String showVal = this.buildShowValueWithCharts(xinyiWarningRecord, tXinyiIndustry, normConfig, chartsDataList, date, yjHour);
  1486. chatReq.setShowVal(showVal);
  1487. chatReq.setQuestion(shWarningPrompt);
  1488. chatReq.setAnswer(sb.toString());
  1489. chatReq.setWarningId(String.valueOf(xinyiWarningRecord.getId()));
  1490. chatReq.setCounts(counts);//问答次数
  1491. chatReq.setUserId(WARNING_DEFAULT_CREATE);
  1492. chatReq.setCreateBy(WARNING_DEFAULT_CREATE);
  1493. chatReq.setCreateTime(DateUtils.getNowDate());
  1494. this.xinyiChatRecordMapper.insertTXinyiChatRecord(chatReq);
  1495. } catch (IOException e) {
  1496. throw new RuntimeException(e);
  1497. } finally {
  1498. // 关闭输出流
  1499. channel.shutdown();
  1500. }
  1501. }
  1502. public static String testPredictor(String type){
  1503. // 获取输出流
  1504. ManagedChannel channel = null;
  1505. try {
  1506. channel = ManagedChannelBuilder.forAddress("10.0.0.24", 17070)
  1507. .usePlaintext()
  1508. .build();
  1509. InferenceAPIsServiceGrpc.InferenceAPIsServiceBlockingStub stub = InferenceAPIsServiceGrpc.newBlockingStub(channel);
  1510. String dataJson = "{\"bot_id\":\"b00001\",\"exp_id\":\"721\",\"norm\":\"" + type + "\",\"session_id\":\" " + IdUtils.simpleUUID() + " \",\"extra\":{}}";
  1511. log.info("请求大模型的预测的参数为{}", dataJson);
  1512. PredictionsRequest request = PredictionsRequest.newBuilder()
  1513. .setModelName("slibra_bot")
  1514. .putInput("method", ByteString.copyFrom("predictor", "utf-8"))//推理
  1515. .putInput("data", ByteString.copyFrom(dataJson, "utf-8"))
  1516. .buildPartial();
  1517. Iterator<PredictionResponse> predictions = stub.streamPredictions(request);
  1518. //将结果记录到问答表
  1519. String responseStr = predictions.next().getPrediction().toStringUtf8();
  1520. log.info("大模型的预测的返回结果为{}", responseStr);
  1521. // return JSON.parseObject(responseStr).getString("pred");
  1522. return responseStr;
  1523. } catch (IOException e) {
  1524. throw new RuntimeException(e);
  1525. } finally {
  1526. assert channel != null;
  1527. channel.shutdown();
  1528. }
  1529. }
  1530. /**
  1531. *
  1532. * 2022/01/01 转成2022年01月01日 数据
  1533. * @param testDate
  1534. * @return
  1535. */
  1536. private String formateDateStr(String testDate) {
  1537. if(StringUtils.isBlank(testDate))
  1538. return "";
  1539. if(!testDate.contains("/"))
  1540. return testDate;
  1541. String[] split = testDate.split("/");
  1542. return split[0] + "年" + split[1] + "月" + split[2] + "日";
  1543. }
  1544. private void askBigModel(String question, String showVal) {
  1545. log.info("进入了后台接口调⽤⼤模型获取问答结果处理");
  1546. StringBuilder sb = new StringBuilder();
  1547. String sessionId = IdUtils.simpleUUID();
  1548. ChatReq chatReq = new ChatReq();
  1549. // String ipAddr = IpUtils.getIpAddr();//获取用户的ip地址 传给大模型
  1550. String ipAddr = "";//获取用户的ip地址 传给大模型 定时任务获取不到ip地址
  1551. int counts = 1;//默认是第一次
  1552. //这种问答 没有历史问答的概念 直接把问题扔进去就行 无需查询历史记录
  1553. List<String> historyDates = new ArrayList<>();
  1554. historyDates.add(question);
  1555. // 获取输出流
  1556. ManagedChannel channel = null;
  1557. try {
  1558. channel = ManagedChannelBuilder.forAddress("10.0.0.24", 17070)
  1559. .usePlaintext()
  1560. .build();
  1561. InferenceAPIsServiceGrpc.InferenceAPIsServiceBlockingStub stub = InferenceAPIsServiceGrpc.newBlockingStub(channel);
  1562. String dataJson = "{\"bot_id\":\"721\",\"exp_id\":\"721\",\"session_id\":\"" + sessionId + "\",\"use_rag\":\"true\",\"prompt\":\"你是⼀个资深⽔务领域专家,能回答各种⽔务相关问题\",\"history_dia\":" + JSON.toJSONString(historyDates) + ",\"generate_args\":{\"max_new_tokens\":2048,\"max_length\":4096,\"num_beams\":1,\"do_sample\":true,\"top_p\":0.7,\"temperature\":0.95},\"extra\":{ \"ip_address\": \"" + ipAddr + "\" },\"strengthen\":" + (true) + "}";
  1563. log.info("请求大模型的问答参数为{}", dataJson);
  1564. PredictionsRequest request = PredictionsRequest.newBuilder()
  1565. .setModelName("slibra_bot")
  1566. .putInput("method", ByteString.copyFrom("infer_stream", "utf-8"))//推理
  1567. .putInput("data", ByteString.copyFrom(dataJson, "utf-8"))
  1568. .buildPartial();
  1569. Iterator<PredictionResponse> predictions = stub.streamPredictions(request);
  1570. //将结果记录到问答表
  1571. while (predictions.hasNext()) {
  1572. String responseStr = predictions.next().getPrediction().toStringUtf8();
  1573. log.info("大模型问答返回的原始结果为{}", responseStr);
  1574. responseStr = JSON.parseObject(responseStr).getString("message");
  1575. if("complete".equals(responseStr)){
  1576. System.out.println("结尾语句并且是非JSON,无需处理");
  1577. //结束语句也流式输出,但是并不记录下来 2024年5月24日11:15:23 也不返回前端
  1578. /*outputStream.write(responseStr.getBytes());
  1579. outputStream.flush();*/
  1580. }else{
  1581. sb.append(responseStr);
  1582. }
  1583. }
  1584. //将问答更新到数据库中
  1585. chatReq.setSessionId(sessionId);
  1586. chatReq.setType(2);//0问答 1决策 2本地 3仿真预测
  1587. chatReq.setModule(4);//0专家问答 1智能工单 2智能体助手 3告警 4简报
  1588. chatReq.setShowVal(showVal);
  1589. chatReq.setQuestion(question);
  1590. chatReq.setAnswer(sb.toString());
  1591. chatReq.setCounts(counts);//问答次数
  1592. chatReq.setUserId(WARNING_DEFAULT_CREATE);
  1593. chatReq.setCreateBy(WARNING_DEFAULT_CREATE);
  1594. chatReq.setCreateTime(DateUtils.getNowDate());
  1595. this.xinyiChatRecordMapper.insertTXinyiChatRecord(chatReq);
  1596. } catch (IOException e) {
  1597. throw new RuntimeException(e);
  1598. } finally {
  1599. // 关闭输出流
  1600. channel.shutdown();
  1601. }
  1602. }
  1603. private String buildShortReportQueryData(List<TXinyiDaily> dailyTwoRecords) {
  1604. //查询配置信息
  1605. List<TXinyiNormConfig> tXinyiNormConfigs = this.xinyiNormConfigMapper.selectTXinyiNormConfigList(null);
  1606. if(CollectionUtils.isEmpty(tXinyiNormConfigs))
  1607. return null;
  1608. TXinyiNormConfig normConfig = tXinyiNormConfigs.get(0);
  1609. //获取数据
  1610. TXinyiDaily yesterdayData = dailyTwoRecords.get(0);
  1611. TXinyiDaily beforeYesterdayData = dailyTwoRecords.get(1);
  1612. String originStr = JIAN_BAO_PROMPT;
  1613. String yesterdayStr = yesterdayData.getTestDate().substring(5).replace("/", "月") + "日";
  1614. String beforeYesterdayStr = beforeYesterdayData.getTestDate().substring(5).replace("/", "月") + "日";
  1615. originStr = originStr.replace("#{0}", yesterdayStr);
  1616. originStr = originStr.replace("#{1}", beforeYesterdayStr);
  1617. originStr = originStr.replace("#{2}", String.valueOf(yesterdayData.getJsCod()));
  1618. originStr = originStr.replace("#{3}", String.valueOf(yesterdayData.getJsTn()));
  1619. originStr = originStr.replace("#{4}", String.valueOf(yesterdayData.getJsTp()));
  1620. originStr = originStr.replace("#{5}", String.valueOf(yesterdayData.getJsNh3()));
  1621. originStr = originStr.replace("#{6}", String.valueOf(yesterdayData.getJsSs()));
  1622. originStr = originStr.replace("#{7}", String.valueOf(yesterdayData.getJSL()));
  1623. originStr = originStr.replace("#{8}", String.valueOf(yesterdayData.getCsCod()));
  1624. originStr = originStr.replace("#{9}", String.valueOf(yesterdayData.getCsTn()));
  1625. originStr = originStr.replace("#{10}", String.valueOf(yesterdayData.getCsTp()));
  1626. originStr = originStr.replace("#{11}", String.valueOf(yesterdayData.getCsNh3()));
  1627. originStr = originStr.replace("#{12}", String.valueOf(yesterdayData.getCsSs()));
  1628. originStr = originStr.replace("#{13}", String.valueOf(yesterdayData.getCSL()));
  1629. originStr = originStr.replace("#{14}", String.valueOf(beforeYesterdayData.getJsCod()));
  1630. originStr = originStr.replace("#{15}", String.valueOf(beforeYesterdayData.getJsTn()));
  1631. originStr = originStr.replace("#{16}", String.valueOf(beforeYesterdayData.getJsTp()));
  1632. originStr = originStr.replace("#{17}", String.valueOf(beforeYesterdayData.getJsNh3()));
  1633. originStr = originStr.replace("#{18}", String.valueOf(beforeYesterdayData.getJsSs()));
  1634. originStr = originStr.replace("#{19}", String.valueOf(beforeYesterdayData.getJSL()));
  1635. originStr = originStr.replace("#{20}", String.valueOf(beforeYesterdayData.getCsCod()));
  1636. originStr = originStr.replace("#{21}", String.valueOf(beforeYesterdayData.getCsTn()));
  1637. originStr = originStr.replace("#{22}", String.valueOf(beforeYesterdayData.getCsTp()));
  1638. originStr = originStr.replace("#{23}", String.valueOf(beforeYesterdayData.getCsNh3()));
  1639. originStr = originStr.replace("#{24}", String.valueOf(beforeYesterdayData.getCsSs()));
  1640. originStr = originStr.replace("#{25}", String.valueOf(beforeYesterdayData.getCSL()));
  1641. originStr = originStr.replace("#{26}", String.valueOf(normConfig.getJscodSjz()));
  1642. originStr = originStr.replace("#{27}", String.valueOf(normConfig.getJszdSjz()));
  1643. originStr = originStr.replace("#{28}", String.valueOf(normConfig.getJszlSjz()));
  1644. originStr = originStr.replace("#{29}", String.valueOf(normConfig.getJsadSjz()));
  1645. originStr = originStr.replace("#{30}", String.valueOf(normConfig.getJsssSjz()));
  1646. originStr = originStr.replace("#{31}", String.valueOf(normConfig.getCscodBzz()));
  1647. originStr = originStr.replace("#{32}", String.valueOf(normConfig.getCszzBzz()));
  1648. originStr = originStr.replace("#{33}", String.valueOf(normConfig.getCszlBzz()));
  1649. originStr = originStr.replace("#{34}", String.valueOf(normConfig.getCsadBzz()));
  1650. originStr = originStr.replace("#{35}", String.valueOf(normConfig.getCsssBzz()));
  1651. return originStr;
  1652. }
  1653. public static String handleDate(String str){
  1654. StringBuilder sb = new StringBuilder();
  1655. if(str.contains(" ")){//包含空格 就是年月日时分秒了
  1656. String[] split = str.split(" ");
  1657. addBeforeZero(sb, split[0], "/");
  1658. sb.append(" ");
  1659. addBeforeZero(sb, split[1], ":");
  1660. }else{
  1661. if(str.contains("/")){//年月日
  1662. addBeforeZero(sb, str, "/");
  1663. }else if(str.contains(":")){//时分秒
  1664. addBeforeZero(sb, str, ":");
  1665. }else {
  1666. sb.append(str);
  1667. }
  1668. }
  1669. return sb.toString();
  1670. }
  1671. public static StringBuilder addBeforeZero(StringBuilder sb, String str, String tag){
  1672. String[] split = str.split(tag);
  1673. int length = split.length;
  1674. for (int i = 0; i < length; i++) {
  1675. String value = split[i];
  1676. Integer intValue = Integer.parseInt(value);
  1677. if(intValue < 10 && value.length() == 1){////防止有正确的情况 额外再补充字符串
  1678. sb.append(0).append(value);
  1679. }else{
  1680. sb.append(value);
  1681. }
  1682. if(i < length-1){
  1683. sb.append(tag);
  1684. }
  1685. }
  1686. return sb;
  1687. }
  1688. public static void main(String[] args) {
  1689. /*LocalDateTime endTime = LocalDateTime.now();
  1690. System.out.println("endTime = " + endTime);
  1691. endTime = endTime.plusMinutes(60);
  1692. System.out.println("endTime = " + endTime);*/
  1693. /*String str = "2024/04/15 09:55";
  1694. System.out.println(str);
  1695. System.out.println(str.substring(0,10));
  1696. System.out.println(str.substring(0,13));*/
  1697. /*String str = "2024/04/18 08:00";
  1698. str = str + ":00";
  1699. System.out.println(str);
  1700. LocalDateTime startTime = LocalDateTime.parse(str.replaceAll("/", "-").replace(" ", "T"));
  1701. System.out.println(startTime.plusMinutes(1L).format(DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss")));*/
  1702. /*ArrayList<Integer> objects = new ArrayList<>();
  1703. objects.add(1);
  1704. objects.add(2);
  1705. objects.add(3);
  1706. System.out.println(objects);
  1707. objects.clear();
  1708. System.out.println(objects);
  1709. // test();
  1710. Date date = new Date();
  1711. System.out.println(DateUtils.differentHoursByMillisecond(date, date));
  1712. HashMap<Object, Object> map = new HashMap<>();
  1713. map.put("a", null);
  1714. map.put("b", "ab");
  1715. map.put("c", "");
  1716. map.put("d", '1');
  1717. System.out.println(JSON.toJSONString(map, JSONWriter.Feature.WriteNulls));
  1718. TXinyiIndustry tXinyiIndustry = new TXinyiIndustry();
  1719. System.out.println(JSON.toJSONString(tXinyiIndustry, JSONWriter.Feature.WriteNulls));*/
  1720. String s = "2022/01/01".substring(5).replace("/", "月") + "日";
  1721. System.out.println("s = " + s);
  1722. // System.out.println(new BigDecimal("1").compareTo(null));//空指针 要判断
  1723. System.out.println("2022/01/01 10".substring(11));
  1724. }
  1725. //测试工业库 没小时保存一条记录是否可行
  1726. public static void test() {
  1727. // 每个小时的时间格式
  1728. DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss");
  1729. LocalDateTime startTime = LocalDateTime.parse("2024-05-18T00:00:00");
  1730. LocalDateTime endTime = LocalDateTime.now();
  1731. // 循环按小时分割
  1732. LocalDateTime currentHour = startTime;
  1733. //最终获取的数据
  1734. Map<String, TXinyiIndustry> needMap = new LinkedHashMap<>();
  1735. while (currentHour.isBefore(endTime)) {
  1736. String begin = currentHour.format(formatter);
  1737. String end = currentHour.plusMinutes(5).format(formatter);
  1738. // 输出当前小时的起始时间和结束时间
  1739. System.out.println("起始时间:" + begin);
  1740. System.out.println("结束时间:" + end);
  1741. // 当前小时加一小时,作为下一个小时的起始时间
  1742. currentHour = currentHour.plusMinutes(5);
  1743. //每个小时查询一次数据
  1744. String url = "http://10.0.0.27:4568/api/v1/khquerydata";
  1745. HashMap<String, Object> req = new HashMap<>();
  1746. req.put("tagNames", queryTags);
  1747. req.put("startTime", begin);
  1748. req.put("endTime", end);
  1749. req.put("recordNumbers", 100000);
  1750. String body = HttpRequest.post(url).header("Authorization", "c2E6c2E=").header("clientName", "hongshan").body(JSON.toJSONString(req)).execute().body();
  1751. // System.out.println("body = " + body);
  1752. List<HashMap<String, String>> list = new ArrayList<>();
  1753. //行转列数据处理
  1754. for (String queryTag : queryTags) {
  1755. JSONArray array = JSON.parseObject(body).getJSONArray(queryTag);
  1756. //特殊数据处理一
  1757. if(Objects.isNull(array) || array.isEmpty()){
  1758. System.out.println(queryTag + "查询到了空的数据,跳过本次循环");
  1759. continue;
  1760. }
  1761. int size = array.size();
  1762. //特殊数据处理二
  1763. if("0".equals(array.get(1) + "")){
  1764. System.out.println(queryTag + "查询到了数据,但是数据集合只有一条,且都是0");
  1765. continue;
  1766. }
  1767. //结合至少62个数据才满足条件(有可能获取不到)
  1768. /*if(size < 62){
  1769. System.out.println(queryTag + "查询到了不符合条件的数据,跳过本次循环");
  1770. continue;
  1771. }*/
  1772. //存放的数据集
  1773. //利用map去重
  1774. HashMap<String, String> map = new LinkedHashMap<>();
  1775. for (int i = 2; i < size; i++) {
  1776. // System.out.println(i + "" + array.get(i));
  1777. JSONArray oneRecord = JSON.parseArray(JSON.toJSONString(array.get(i)));
  1778. //处理为空或者为0的数据
  1779. Object timeStampValue = oneRecord.get(2);
  1780. if(Objects.isNull(timeStampValue) || "0".equals(timeStampValue + ""))
  1781. continue;
  1782. BigDecimal value = Objects.isNull(oneRecord.get(0)) ? null : new BigDecimal(oneRecord.get(0) + "");
  1783. long timestamp = (long) timeStampValue;
  1784. String format = DateUtil.format(new Date(timestamp), DateUtils.YYYYMMDDHH_TS);
  1785. map.put(format, queryTag + "-" + value);
  1786. }
  1787. list.add(map);
  1788. }
  1789. Set<String> recordTimeSet = new HashSet<>();
  1790. Map<String, TXinyiIndustry> recordMap = new HashMap<>();
  1791. for (int i = 0; i < list.size(); i++) {
  1792. HashMap<String, String> map = list.get(i);
  1793. int finalJ = i;
  1794. map.forEach((k, v) ->{
  1795. TXinyiIndustry industry = null;
  1796. if(!recordTimeSet.contains(k)){//第一次
  1797. industry = new TXinyiIndustry();
  1798. recordTimeSet.add(k);
  1799. recordMap.put(k, industry);
  1800. }else{
  1801. industry = recordMap.get(k);
  1802. }
  1803. industry.setTestTime(k + ":00");
  1804. //2024年4月15日11:19:52 额外增加2个字段
  1805. industry.setTestDate(k.substring(0,10));
  1806. industry.setTestHour(k.substring(0,13));
  1807. //解析值
  1808. String[] split = v.split("-");
  1809. String type = split[0];
  1810. BigDecimal value = new BigDecimal(split[1]);
  1811. if ("信义污水厂JS_COD_Value".equals(type)) {
  1812. industry.setJsCod(value);
  1813. } else if ("信义污水厂JS_PH_Value".equals(type)) {
  1814. industry.setJsPh(value);
  1815. } else if ("信义污水厂JS_SS_Value".equals(type)) {
  1816. industry.setJsSs(value);
  1817. } else if ("信义污水厂JS_ZL_Value".equals(type)) {
  1818. industry.setJsTp(value);
  1819. } else if ("信义污水厂JS_ZA_Value".equals(type)) {
  1820. industry.setJsTn(value);
  1821. } else if ("信义污水厂JS_AD_Value".equals(type)) {
  1822. industry.setJsNh3(value);
  1823. } else if ("信义污水厂JS_T_Value".equals(type)) {
  1824. industry.setJsSwPh(value);
  1825. } else if ("信义污水厂进水泵房液位".equals(type)) {
  1826. industry.setJsBfyw(value);
  1827. } else if ("信义污水厂出水瞬时流量".equals(type)) {
  1828. industry.setCsSlqc(value);
  1829. } else if ("信义污水厂升级出水COD".equals(type)) {
  1830. industry.setCsCod(value);
  1831. } else if ("信义污水厂升级出水PH".equals(type)) {
  1832. industry.setCsPh(value);
  1833. } else if ("信义污水厂升级出水SS".equals(type)) {
  1834. industry.setCsSs(value);
  1835. } else if ("信义污水厂升级出水TN".equals(type)) {
  1836. industry.setCsTn(value);
  1837. } else if ("信义污水厂升级出水TP".equals(type)) {
  1838. industry.setCsTp(value);
  1839. } else if ("信义污水厂升级出水氨氮".equals(type)) {
  1840. industry.setCsNh3(value);
  1841. } else if ("信义污水厂AIT202_Value".equals(type)) {
  1842. industry.setOneHyzdDo(value);
  1843. } else if ("信义污水厂AIT203_Value".equals(type)) {
  1844. industry.setOneHymdDo(value);
  1845. } else if ("信义污水厂AIT207_Value".equals(type)) {
  1846. industry.setTwoHyzdDo(value);
  1847. } else if ("信义污水厂AIT206_Value".equals(type)) {
  1848. industry.setTwoHymdDo(value);
  1849. } else if ("信义污水厂AIT209_Value".equals(type)) {
  1850. industry.setOneMlss(value);
  1851. } else if ("信义污水厂AIT210_Value".equals(type)) {
  1852. industry.setTwoMlss(value);
  1853. } else if ("信义污水厂进水TDS".equals(type)) {
  1854. industry.setJsTds(value);
  1855. } else if ("信义污水厂FT101_Value".equals(type)) {
  1856. industry.setJsSlq(value);
  1857. } else if ("信义污水厂SWCHHYHLB1_R_Value".equals(type)) {
  1858. industry.setNHlbOneGp(value);
  1859. } else if ("信义污水厂SWCHHYHLB2_R_Value".equals(type)) {
  1860. industry.setNHlbTwoGp(value);
  1861. } else if ("信义污水厂SWCHHYHLB3_R_Value".equals(type)) {
  1862. industry.setNHlbThreeGp(value);
  1863. } else if ("信义污水厂SWCHHYHLB4_R_Value".equals(type)) {
  1864. industry.setNHlbFourGp(value);
  1865. } else if ("信义污水厂SWCHHYHLB5_R_Value".equals(type)) {
  1866. industry.setNhlBFiveGp(value);
  1867. } else if ("信义污水厂SWCHHYHLB6_R_Value".equals(type)) {
  1868. industry.setNHlbSixGp(value);
  1869. } else if ("信义污水厂SWCWNHLB1_R_Value".equals(type)) {
  1870. industry.setWHlbOneGp(value);
  1871. } else if ("信义污水厂SWCWNHLB2_R_Value".equals(type)) {
  1872. industry.setWHlbTwoGp(value);
  1873. } else if ("信义污水厂SWCWNHLB3_R_Value".equals(type)) {
  1874. industry.setWHlbThreeGp(value);
  1875. } else if ("信义污水厂SWCWNHLB4_R_Value".equals(type)) {
  1876. industry.setWHlbFourGp(value);
  1877. } else if ("信义污水厂SWCWNHLB5_R_Value".equals(type)) {
  1878. industry.setWHlbFiveGp(value);
  1879. } else if ("信义污水厂GFJ1_R_Value".equals(type)) {
  1880. industry.setFjOne(value);
  1881. } else if ("信义污水厂GFJ2_R_Value".equals(type)) {
  1882. industry.setFjTwo(value);
  1883. } else if ("信义污水厂GFJ3_R_Value".equals(type)) {
  1884. industry.setFjThree(value);
  1885. } else if ("信义污水厂GFJ4_R_Value".equals(type)) {
  1886. industry.setFjFour(value);
  1887. } else if ("信义污水厂GFJ5_R_Value".equals(type)) {
  1888. industry.setFjFive(value);
  1889. } else if ("信义污水厂GFJ6_R_Value".equals(type)) {
  1890. industry.setFjSix(value);
  1891. } else if ("信义污水厂GFJ1_KQLL_Value".equals(type)) {
  1892. industry.setKqllOne(value);
  1893. } else if ("信义污水厂GFJ2_KQLL_Value".equals(type)) {
  1894. industry.setKqllTwo(value);
  1895. } else if ("信义污水厂GFJ3_KQLL_Value".equals(type)) {
  1896. industry.setKqllThree(value);
  1897. } else if ("信义污水厂GFJ4_KQLL_Value".equals(type)) {
  1898. industry.setKqllFour(value);
  1899. } else if ("信义污水厂GFJ5_KQLL_Value".equals(type)) {
  1900. industry.setKqllFive(value);
  1901. } else if ("信义污水厂GFJ6_KQLL_Value".equals(type)) {
  1902. industry.setKqllSix(value);
  1903. }else if ("信义污水厂实际碳源加药量".equals(type)) {
  1904. industry.setSJTYJLY(value);
  1905. }else if ("信义污水厂除磷加药瞬时流量".equals(type)) {
  1906. industry.setCLJYSSLL(value);
  1907. } else if ("信义污水厂_除磷P04预测值_".equals(type)) {
  1908. industry.setCLP04YCZ(value);
  1909. }
  1910. //只有最后一次才执行数据库添加
  1911. if(finalJ == list.size()-1){
  1912. needMap.put(industry.getTestHour(), industry);
  1913. }
  1914. });
  1915. }
  1916. }
  1917. //保存数据 触发告警 决策 问答记录等等
  1918. needMap.forEach((k, industry) ->{
  1919. System.out.println(JSON.toJSONString(industry));
  1920. });
  1921. }
  1922. }