147 Commits

Author SHA1 Message Date
caozehui
391646d416 微调 2026-05-08 13:55:48 +08:00
caozehui
4f3f0833d5 微调 2026-05-08 13:15:28 +08:00
caozehui
924c2e8f45 微调 2026-05-08 10:39:27 +08:00
caozehui
8dca7bf537 微调 2026-05-08 10:37:58 +08:00
caozehui
7e6fb2d981 特性点顺序问题 2026-05-07 19:18:42 +08:00
caozehui
ec9a0ca236 微调 2026-05-07 16:27:49 +08:00
caozehui
7ab5b9a501 横向纵向实时打特性点 2026-05-07 15:21:35 +08:00
caozehui
3959b96040 微调 2026-05-07 10:55:01 +08:00
caozehui
ae5370abdf 微调 2026-05-07 10:49:01 +08:00
caozehui
1894cb07a2 微调 2026-05-07 09:03:34 +08:00
caozehui
08dff063c9 调整脚本换行、特性点入库 2026-05-07 08:52:17 +08:00
caozehui
f640afb4ed 特性点推送 2026-05-06 13:17:12 +08:00
caozehui
503018a721 起点 2026-04-30 09:34:44 +08:00
caozehui
391fd0cf4f 最后状态清空 2026-04-23 15:37:40 +08:00
caozehui
99c7448544 Merge remote-tracking branch 'origin/hainan' into hainan
# Conflicts:
#	detection/src/main/java/com/njcn/gather/detection/util/socket/FormalTestManager.java
2026-04-23 14:08:43 +08:00
caozehui
83296d257c 微调 2026-04-23 13:59:18 +08:00
caozehui
e020aa466e 调整变频器耐受状态字段 2026-04-22 20:39:53 +08:00
caozehui
f20e2c9b32 微调 2026-04-22 09:34:53 +08:00
caozehui
e03c3e3607 状态清空 2026-04-20 09:08:25 +08:00
caozehui
27f25d2404 微调 2026-04-17 08:56:15 +08:00
caozehui
a77313171c 向前端推送暂降点 2026-04-16 15:24:00 +08:00
caozehui
a658d6e81a 向前端推送暂降点 2026-04-16 15:02:53 +08:00
caozehui
ab11c91579 关于时间精度调整 2026-04-16 14:14:38 +08:00
caozehui
a2468f1353 微调 2026-04-16 11:43:41 +08:00
caozehui
d8bcca1ede 微调 2026-04-15 18:52:14 +08:00
caozehui
c5e77ee9b1 耐受结果、变频器重新连接Socket调整 2026-04-15 09:45:53 +08:00
caozehui
2293d81b71 动态创建表结构、入库 2026-04-14 14:51:33 +08:00
caozehui
97157a5ccf 变频器 2026-04-13 16:28:25 +08:00
caozehui
fd7c6ada6b 回滚 2026-04-13 15:41:37 +08:00
caozehui
c148bddfc9 Revert "微调"
This reverts commit 4110a835c8.
2026-04-13 15:39:15 +08:00
caozehui
f0857b7c46 Revert "微调"
This reverts commit ef757c52ea.
2026-04-13 15:38:55 +08:00
9b1c6f61e6 revert e78ce544e3
revert 微调
2026-04-13 15:31:24 +08:00
caozehui
3f72c52cdc Revert "微调"
This reverts commit e78ce544e3.
2026-04-13 15:27:55 +08:00
caozehui
ef757c52ea 微调 2026-04-10 15:13:22 +08:00
caozehui
4110a835c8 微调 2026-04-10 14:13:27 +08:00
caozehui
e78ce544e3 微调 2026-03-26 13:23:02 +08:00
caozehui
c10d54e79a 微调 2026-01-30 16:11:14 +08:00
caozehui
2796558040 重采方案 2026-01-26 10:16:48 +08:00
caozehui
920a808729 微调 2026-01-22 15:25:48 +08:00
caozehui
8e24ac4b71 脚本单独配置BC暂态时左侧树形列表展示不全问题 2026-01-22 15:09:02 +08:00
caozehui
3eb2736edb 微调 2026-01-22 11:26:25 +08:00
caozehui
8c3eba9224 平均值时移除误差较大的几组数据 2026-01-21 14:01:07 +08:00
caozehui
6288aa565e 系数下发 2026-01-19 14:23:23 +08:00
caozehui
0fa7ec91c4 微调 2025-12-30 08:36:09 +08:00
caozehui
fdb4b7060a 暂态前后延时间 2025-12-24 14:41:47 +08:00
3f47b0f008 微调 压缩文件的编解码保持一致 2025-12-23 10:29:40 +08:00
caozehui
cb431b5af1 Merge remote-tracking branch 'origin/master'
# Conflicts:
#	entrance/src/main/resources/application.yml
2025-12-23 10:22:04 +08:00
caozehui
153428b24f 合并qr_branch分支代码 2025-12-22 10:30:56 +08:00
caozehui
d92544f7c4 Merge branch 'qr_branch'
# Conflicts:
#	detection/src/main/java/com/njcn/gather/detection/handler/SocketDevResponseService.java
#	detection/src/main/java/com/njcn/gather/detection/handler/SocketSourceResponseService.java
#	detection/src/main/java/com/njcn/gather/detection/service/impl/DetectionServiceImpl.java
#	detection/src/main/java/com/njcn/gather/report/service/impl/PqReportServiceImpl.java
#	detection/src/main/java/com/njcn/gather/report/utils/BookmarkUtil.java
#	detection/src/main/java/com/njcn/gather/report/utils/Docx4jUtil.java
#	detection/src/main/java/com/njcn/gather/result/service/impl/ResultServiceImpl.java
#	entrance/src/main/resources/application.yml
2025-12-22 08:52:13 +08:00
caozehui
3f1ae1886a Merge remote-tracking branch 'origin/qr_branch' into qr_branch 2025-12-19 16:17:40 +08:00
caozehui
af4863af65 2楼报告封面调整:装置编码改成装置编号 2025-12-19 16:17:33 +08:00
caozehui
4b7c1259a7 ICD映射上传路径微调 2025-12-16 16:18:02 +08:00
caozehui
de1496389e 微调 2025-12-11 16:19:26 +08:00
79003cd0f4 微调 2025-12-10 18:48:28 +08:00
caozehui
c3443fcc91 切换数据处理原则 2025-12-10 11:15:43 +08:00
caozehui
5105e77823 微调 2025-12-10 09:17:31 +08:00
caozehui
eb068b76a4 报告模板路径调整 2025-12-09 16:18:53 +08:00
9f11f7ec11 河北报告定制化改动 2025-12-02 13:52:07 +08:00
2012221b73 河北报告定制化改动 2025-12-02 13:37:35 +08:00
caozehui
9ab5d42439 微调 2025-12-01 19:14:29 +08:00
7abcaefeb1 增加回路额定电流描述 2025-11-27 20:03:11 +08:00
caozehui
f4df52dd1c 暂态只有a项时,当a相无数据时判断为不符合 2025-11-27 18:36:38 +08:00
db115bb27d 调整电流的小数位 2025-11-27 17:08:49 +08:00
caozehui
68d96e67aa 重新计算、更换误差体系接口调整,cp95处理原则调整、检测结果原始数据调整 2025-11-26 18:33:35 +08:00
3f94012faa 修复数模式的相序校验电流加量问题 2025-11-26 10:34:25 +08:00
41c557118c Merge remote-tracking branch 'origin/qr_branch' into qr_branch 2025-11-26 10:22:15 +08:00
6596a572d6 修复数模式的相序校验电流加量问题 2025-11-26 10:17:33 +08:00
139c7b0651 微调 2025-11-26 08:36:25 +08:00
caozehui
ab236cd34f 微调 2025-11-21 10:09:48 +08:00
caozehui
786bd5d660 报告调整 2025-11-19 13:20:35 +08:00
caozehui
21d2c2b7a7 报告调整 2025-11-19 13:20:17 +08:00
caozehui
48fa4c2390 报告日期格式根据参数动态变化 2025-11-19 10:43:52 +08:00
caozehui
033330b005 微调 2025-11-19 09:35:49 +08:00
caozehui
f81123c3f7 监测点修改同步状态调整、脚本新增福禄克专用字段 2025-11-17 16:32:52 +08:00
caozehui
5539cb2887 微调 2025-11-14 10:16:08 +08:00
caozehui
1df9c8d703 微调 2025-11-13 14:20:50 +08:00
caozehui
1cbed2a620 icd上传映射文件 2025-11-13 08:37:52 +08:00
89667367ea 添加水印 2025-11-12 11:45:21 +08:00
92b95dd86d 微调 2025-11-11 13:18:42 +08:00
caozehui
6b7e38fef6 处理源未知异常 2025-11-10 14:45:23 +08:00
caozehui
f10debe2f2 比对-被检设备导入微调 2025-11-07 15:13:07 +08:00
caozehui
9d15351fba 微调 2025-11-06 09:25:09 +08:00
caozehui
2339a006ec 补充源未知异常推送消息 2025-11-06 08:51:54 +08:00
caozehui
7fd904ab79 系数校准接收数据个数调整 2025-11-05 16:02:53 +08:00
caozehui
61f149b562 系数校准接收数据个数调整 2025-11-05 13:51:47 +08:00
caozehui
57ee3a4d43 调整接口注释 2025-11-04 14:35:16 +08:00
563eb80b65 swagger调整 2025-11-04 10:34:08 +08:00
caozehui
dae10378dd 监测点变化时同步更新设备状态、计划状态 2025-11-04 09:36:42 +08:00
caozehui
40cb153656 报告模板调整 2025-11-03 08:38:57 +08:00
caozehui
af4f000b13 微调 2025-10-30 14:49:22 +08:00
caozehui
f922ee97aa 报告路径,报告名称修改 2025-10-30 09:06:40 +08:00
贾同学
321ec97130 fix(device):修改通道编号拼接逻辑 2025-10-29 10:07:14 +08:00
caozehui
0dd3502942 数模-检测项树表 2025-10-29 09:05:12 +08:00
caozehui
26c5e933f5 微调 2025-10-29 09:01:45 +08:00
贾同学
00ba09faae fix(activate): 处理许可文件读取异常 2025-10-28 13:44:58 +08:00
caozehui
40e39d651b 微调 2025-10-28 10:31:39 +08:00
贾同学
02c8164b7e fix(device):优化监测点线路号过滤与重复检查逻辑 2025-10-28 10:01:28 +08:00
贾同学
8c598aec1e fix(log):修正日志时间查询逻辑 2025-10-28 09:25:09 +08:00
caozehui
2c59defdc2 修改原始数据表结构、监测点检测结果加入统计数据逻辑 2025-10-27 13:58:46 +08:00
贾同学
5642bf2b31 Merge remote-tracking branch 'origin/master' 2025-10-24 16:31:01 +08:00
贾同学
ba76df66b0 UPDATE: 修改父依赖 2025-10-24 16:30:43 +08:00
caozehui
57c419eb70 微调 2025-10-24 16:25:21 +08:00
贾同学
98f4ecef6c Merge remote-tracking branch 'origin/master' 2025-10-24 09:28:40 +08:00
贾同学
20e07712cb ADD: 添加统计检测结果 2025-10-24 09:28:28 +08:00
caozehui
ac1d98efdc 微调 2025-10-23 15:15:49 +08:00
caozehui
5d161acfad 导入测试后数据,查看数据报错BUG 2025-10-23 13:28:14 +08:00
caozehui
1534327f6f Merge remote-tracking branch 'origin/master' 2025-10-23 13:27:56 +08:00
贾同学
04ada8740a UPDATE: 修改数据合并问题 2025-10-23 13:18:54 +08:00
caozehui
98cca582f6 Merge remote-tracking branch 'origin/master' 2025-10-22 15:49:14 +08:00
贾同学
a2de4b80a7 UPDATE: 修改激活逻辑 2025-10-22 13:27:24 +08:00
caozehui
c7d8fc3168 报告封面微调 2025-10-22 11:05:21 +08:00
caozehui
071c6e3d64 谐波(间谐波)结果展示调整 2025-10-22 09:19:31 +08:00
b4878d4a25 调整检测报告 2025-10-21 17:01:55 +08:00
caozehui
bb22857fc9 报告微调 2025-10-21 15:42:22 +08:00
caozehui
b0a4458c56 Bug修改 2025-10-21 15:23:13 +08:00
6357cde72b 调整检测报告 2025-10-21 15:09:05 +08:00
dddffe43cb 调整检测报告 2025-10-21 15:07:49 +08:00
贾同学
51fdf6bf59 UPDATE: 修复被检设备分页查询排序问题 2025-10-21 14:24:38 +08:00
贾同学
b53ef274cf UPDATE: 重新修改导入被检设备逻辑 2025-10-21 14:02:22 +08:00
贾同学
90d618b66f UPDATE: 新建计划可选择所有标准设备 2025-10-20 14:25:26 +08:00
e81413eaa9 增加报告模板友好提示信息 2025-10-20 13:37:38 +08:00
33cefdd0b1 端口调整 2025-10-20 10:30:53 +08:00
80a886a5eb 增加日志输出,以便观察实际波形信息 2025-10-20 09:01:37 +08:00
0a85b433ba 不采用降低采样率,严格对比对应窗口数据。避免没必要的降采抽点工作 2025-10-17 17:29:47 +08:00
caozehui
66786200bd 设备、计划状态修改 2025-10-16 15:40:20 +08:00
15e93b6734 调整检测报告 2025-10-16 13:36:14 +08:00
caozehui
6843497908 微调 2025-10-16 10:56:35 +08:00
caozehui
cd486f419f Merge remote-tracking branch 'origin/master' 2025-10-16 10:22:53 +08:00
caozehui
7394762e28 闪变、重新计算逻辑调整 2025-10-16 10:19:13 +08:00
贾同学
2f75fe062b UPDATE: 优化程序激活码工具。 2025-10-15 14:27:36 +08:00
贾同学
d4e09a09cf UPDATE: 优化程序激活码工具。 2025-10-15 14:25:39 +08:00
贾同学
d58452012d UPDATE: 程序激活码生成工具集成。 2025-10-14 20:29:55 +08:00
贾同学
014ac7931e ADD: 程序激活流程:1、生成设备申请码;2、验证设备激活码;3、读取许可信息;4、激活码生成工具。 2025-10-14 18:54:32 +08:00
3d4fb44b3a 移除event_smart模块 2025-10-11 10:39:07 +08:00
caozehui
1abba4a528 统计数据 2025-10-09 15:50:54 +08:00
贾同学
d06fa8476c UPDATE: 优化。 2025-09-30 09:51:49 +08:00
贾同学
20cb78eb06 UPDATE: 子计划绑定或解绑被检设备,检测状态更新。 2025-09-26 14:25:33 +08:00
贾同学
f3e9cb7171 UPDATE: 完善。 2025-09-25 15:12:59 +08:00
贾同学
6f5746861f UPDATE: 完善。 2025-09-25 14:40:13 +08:00
贾同学
dc34bd9b44 UPDATE: 完善。 2025-09-25 11:10:50 +08:00
贾同学
2dcf90d6c7 UPDATE: 1、子计划管理,筛选条件改成搜索、设备厂家、是否分配;
2、重复导入子计划时,增量被检设备并删除未检设备。
2025-09-25 08:59:36 +08:00
caozehui
ac1c5fd43e 微调 2025-09-25 08:49:44 +08:00
caozehui
dc0244d81d Merge remote-tracking branch 'origin/master'
# Conflicts:
#	detection/src/main/java/com/njcn/gather/device/service/impl/PqDevServiceImpl.java
2025-09-25 08:48:44 +08:00
449de8bbc5 删除不必要的文档 2025-09-25 08:37:35 +08:00
caozehui
69503c7ca9 修改检测计划态,录波校验调整 2025-09-23 19:35:47 +08:00
caozehui
6ef908ff60 微调 2025-09-03 20:32:14 +08:00
caozehui
7461801657 微调 2025-09-03 16:18:52 +08:00
312 changed files with 9055 additions and 12046 deletions

215
CLAUDE.md
View File

@@ -1,215 +0,0 @@
# CLAUDE.md
This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.
## 项目概述
CN_Gather是灿能公司的融合工具项目体专门用于电能质量设备检测的企业级应用系统。采用Spring Boot多模块Maven架构以detection模块为核心的检测业务系统。
## 项目架构
### 核心模块结构
- **entrance**: 应用入口模块端口18092整合所有其他模块
- **detection**: 核心检测业务模块,电能质量设备检测的完整业务流程
- **storage**: 数据存储模块,处理检测数据存储和谐波数据处理
- **system**: 基础系统模块,提供字典管理、日志管理、配置管理等基础功能
- **user**: 用户管理模块,处理认证授权和权限控制
### 模块依赖关系
```
entrance (启动入口)
├── system (基础服务层)
├── user (认证授权层)
├── detection (核心业务层) → 依赖 system, storage
└── storage (数据存储层) → 依赖 system
```
## 常用开发命令
### 构建和打包
```bash
# 编译整个项目
mvn clean compile
# 打包所有模块
mvn clean package
# 跳过测试打包
mvn clean package -DskipTests
# 安装到本地仓库
mvn clean install
```
### 运行应用
```bash
# 运行主入口应用 (端口18092)
cd entrance
mvn spring-boot:run
# 运行事件智能模块 (独立应用)
cd event_smart
mvn spring-boot:run
```
### 测试
```bash
# 运行所有测试
mvn test
# 运行特定模块测试
cd detection
mvn test
```
## 技术栈
### 核心框架
- **Spring Boot**: 2.3.12.RELEASE
- **MyBatis Plus**: 数据持久层框架
- **Maven**: 项目构建管理
- **Java**: 1.8
### 数据库
- **MySQL**: 主数据库 (192.168.1.24:13306/pqs9100)
- **Oracle**: event_smart模块使用
- **Druid**: 数据库连接池
### 通信技术
- **Netty**: Socket通信 (端口61000设备, 62000源)
- **WebSocket**: 实时数据推送 (端口7777)
- **RestTemplate**: HTTP客户端通信
### 其他关键技术
- **Apache POI + docx4j**: Word文档报告生成
- **FastJSON**: JSON数据处理
- **Spring Security + JWT**: 安全认证 (event_smart模块)
- **Redis**: 缓存服务 (event_smart模块)
## 关键配置
### 数据库配置
- 数据库URL: `jdbc:mysql://192.168.1.24:13306/pqs9100`
- MyBatis映射文件位置: `classpath*:com/njcn/**/mapping/*.xml`
- 主键生成策略: `assign_uuid`
### Socket通信配置
- 源设备Socket: 127.0.0.1:62000
- 被检设备Socket: 127.0.0.1:61000
- WebSocket端口: 7777
### 文件路径配置
- 日志目录: `D:\logs`
- 报告模板目录: `D:\template`
- 报告输出目录: `D:\report`
- Word模板位置: `entrance/src/main/resources/model/`
## detection模块核心架构
### 子模块功能划分
- **device**: 设备管理 - PqDev(被检设备)、PqStandardDev(标准设备)、PqDevSub(设备子表)
- **plan**: 检测计划管理 - AdPlan(检测计划)、AdPlanSource(计划源)、AdPlanStandardDev(计划标准设备)
- **script**: 检测脚本管理 - PqScript(检测脚本)、PqScriptDtls(脚本详情)、PqScriptCheckData(检测数据)
- **source**: 程控源管理 - PqSource(程控源设备)
- **err**: 误差体系管理 - PqErrSys(误差体系)、PqErrSysDtls(误差详情)
- **report**: 报告生成管理 - PqReport(报告模板)支持Word模板处理
- **monitor**: 监测管理 - PqMonitor(监测点管理)
- **icd**: ICD路径管理 - PqIcdPath(通信配置)
- **result**: 结果管理 - 检测结果查询和数据展示
- **type**: 设备类型管理 - DevType(设备类型字典)
### 核心检测流程 (PreDetectionController)
```java
// 主要检测接口
@PostMapping("/startPreTest") // 检测通用入口
@PostMapping("/ytxCheckSimulate") // 源通讯校验
@PostMapping("/startSimulateTest") // 启动程控源检测
@PostMapping("/coefficientCheck") // 系数校验
@PostMapping("/startContrastTest") // 比对检测
@PostMapping("/devPhaseSequence") // 设备相序检测
```
### Socket通信架构
- **SocketManager**: Socket会话管理存储userId与Channel映射
- **WebServiceManager**: WebSocket服务管理实时数据推送
- **通信处理器**:
- SocketSourceResponseService: 程控源响应处理
- SocketDevResponseService: 设备响应处理
- SocketContrastResponseService: 比对检测响应处理
- **通信工具**:
- CnSocketUtil: Socket连接工具
- FormalTestManager: 正式检测管理
- XiNumberManager: 系数管理
### 暂态检测参数
- 暂态前时间: 2秒
- 写入时间: 0.001秒
- 写出时间: 0.001秒
- 暂态后时间: 3秒
### 闪变参数
- 波形类型: CPM/SQU
- 占空比: 50%
## 开发注意事项
### detection模块包结构
```
detection/
├── controller/ # 控制层 - PreDetectionController
├── handler/ # Socket响应处理器
├── service/ # 业务逻辑层
│ └── impl/ # 服务实现
├── util/ # 工具类层
│ ├── business/ # 业务工具 - DetectionCommunicateUtil
│ └── socket/ # Socket通信工具
├── pojo/ # 数据模型层
│ ├── constant/ # 常量 - DetectionCommunicateConstant
│ ├── dto/ # 数据传输对象
│ ├── enums/ # 枚举 - DetectionCodeEnum等
│ ├── param/ # 请求参数
│ ├── po/ # 持久化对象
│ └── vo/ # 视图对象 - DetectionData等
└── [子模块]/ # device、plan、script等子模块
├── controller/ # 子模块控制器
├── service/ # 子模块服务
├── mapper/ # 数据访问层
│ └── mapping/ # MyBatis映射文件
└── pojo/ # 子模块数据对象
```
### 检测数据处理机制
- **任意值**: 取第一个满足条件的数据
- **部分值**: 去除最大最小值后取值
- **所有值**: 要求所有数据都合格
- **CP95值**: 取95%分位数
- **平均值**: 取算术平均值
### 检测项目类型
- **频率**: FREQ
- **电压**: V_RELATIVE(相对值)/V_ABSOLUTELY(绝对值)
- **电流**: I_RELATIVE/I_ABSOLUTELY
- **谐波**: HV/HI (2-50次谐波)
- **间谐波**: HSV/HSI
- **不平衡度**: IMBV/IMBA (三相不平衡)
- **闪变**: F (PST)
- **暂态**: VOLTAGE_MAG/VOLTAGE_DUR
### 检测模式
- **数字式检测**: 数字接口通信
- **模拟式检测**: 模拟信号输出
- **比对式检测**: 多台设备比对
### 报告生成机制
- **模板处理**: 使用POI和docx4j处理Word文档
- **模板位置**: `entrance/src/main/resources/model/`
- **支持模板**: NPQS-580、PQV-700、njcn_882系列等
- **功能**: 书签替换、表格填充、文档合并
- **云端上传**: 支持FTP批量上传报告
### 依赖组件
项目使用灿能公司自研组件:
- `njcn-common`: 通用工具包
- `mybatis-plus`: MyBatis增强包
- `spingboot2.3.12`: Spring Boot定制包
- `RestTemplate-plugin`: HTTP客户端插件

View File

@@ -1,238 +0,0 @@
# Gitea本地协作开发服务器配置指南
## 概述
本文档说明如何将本地安装的Gitea配置为团队协作开发服务器替代原有的物理服务器环境。
## 1. 网络配置
### 1.1 确认本机IP地址
```bash
# Windows系统
ipconfig
# 查找本机局域网IP地址通常形如 192.168.x.x 或 10.x.x.x
```
### 1.2 配置Gitea服务地址
编辑Gitea配置文件 `app.ini`
```ini
[server]
# 将localhost改为本机IP地址确保同事可以访问
HTTP_ADDR = 0.0.0.0
HTTP_PORT = 3000
# 外部访问URL替换为你的实际IP
ROOT_URL = http://192.168.x.x:3000/
```
### 1.3 防火墙配置
确保Windows防火墙允许Gitea端口通信
```bash
# 打开Windows防火墙入站规则
# 添加端口3000的TCP入站规则
```
或在Windows防火墙中
- 控制面板 → 系统和安全 → Windows Defender防火墙 → 高级设置
- 入站规则 → 新建规则 → 端口 → TCP → 特定本地端口: 3000
## 2. Gitea服务配置
### 2.1 启动Gitea服务
```bash
# 进入Gitea安装目录
cd C:\gitea # 或你的安装路径
gitea.exe web
```
### 2.2 配置为Windows服务推荐
创建Windows服务确保开机自启
1. 下载NSSM (Non-Sucking Service Manager)
2. 以管理员身份运行命令提示符:
```bash
nssm install Gitea
# 在弹出界面中配置:
# Path: C:\gitea\gitea.exe
# Arguments: web
# Working directory: C:\gitea
```
3. 启动服务:
```bash
net start Gitea
```
### 2.3 数据库配置优化
如果使用SQLite默认确保数据文件路径正确
```ini
[database]
DB_TYPE = sqlite3
PATH = data/gitea.db
```
如果需要更好性能考虑配置MySQL
```ini
[database]
DB_TYPE = mysql
HOST = 127.0.0.1:3306
NAME = gitea
USER = gitea
PASSWD = your_password
```
## 3. 同事访问配置
### 3.1 提供访问地址
向同事提供访问地址:
```
http://你的IP地址:3000
例如: http://192.168.1.100:3000
```
### 3.2 用户账号管理
1. 访问管理界面创建用户账号
2. 或开启用户自注册:
```ini
[service]
DISABLE_REGISTRATION = false
REQUIRE_SIGNIN_VIEW = false
```
### 3.3 权限配置
为协作项目设置适当权限:
- 项目所有者:完全控制权限
- 协作者:推送/拉取权限
- 读者:仅读取权限
## 4. 代码仓库迁移
### 4.1 从原服务器迁移仓库
如果原服务器数据可恢复:
```bash
# 在原服务器或备份中找到Git裸仓库
# 复制到新Gitea的repositories目录
# 通常位于 gitea-repositories/用户名/仓库名.git
```
### 4.2 重新创建仓库
如果需要重新创建:
1. 在Gitea界面创建新仓库
2. 本地添加新的远程地址:
```bash
git remote remove origin
git remote add origin http://你的IP:3000/用户名/仓库名.git
git push -u origin master
```
## 5. 开发工作流配置
### 5.1 分支保护规则
为主要分支设置保护规则:
- 设置 → 分支 → 分支保护规则
- 保护master分支要求代码审查
### 5.2 Webhook配置
如果需要CI/CD集成
```
设置 → Webhooks → 添加Webhook
配置自动构建触发器
```
## 6. 备份策略
### 6.1 定期备份
```bash
# 备份Gitea数据目录
# 包括repositories/, data/, log/, custom/
robocopy "C:\gitea" "D:\backup\gitea" /MIR /Z /R:3 /W:10
```
### 6.2 自动备份脚本
创建批处理文件实现定期备份:
```batch
@echo off
set BACKUP_DIR=D:\backup\gitea_%date:~0,4%%date:~5,2%%date:~8,2%
robocopy "C:\gitea" "%BACKUP_DIR%" /MIR /Z /R:3 /W:10
echo Backup completed to %BACKUP_DIR%
```
## 7. 常见问题排查
### 7.1 访问问题
- 检查防火墙设置
- 确认IP地址和端口正确
- 验证Gitea服务是否正常运行
### 7.2 权限问题
- 检查用户账号状态
- 确认仓库权限设置
- 验证SSH密钥配置如使用SSH
### 7.3 性能优化
```ini
[server]
# 调整并发连接数
HTTP_ADDR = 0.0.0.0
HTTP_PORT = 3000
[database]
# 数据库连接池配置
MAX_IDLE_CONNS = 30
MAX_OPEN_CONNS = 300
```
## 8. 安全建议
1. **网络安全**
- 仅在受信任的局域网环境中开放
- 考虑使用VPN访问
- 定期更新Gitea版本
2. **访问控制**
- 禁用不必要的公开注册
- 使用强密码策略
- 启用双因子认证
3. **数据安全**
- 定期备份重要数据
- 监控异常访问
- 记录操作日志
## 9. 同事操作指南
### 9.1 首次设置
```bash
# 克隆仓库
git clone http://你的IP:3000/用户名/CN_Gather.git
# 配置用户信息
git config user.name "姓名"
git config user.email "邮箱"
```
### 9.2 日常协作
```bash
# 拉取最新代码
git pull origin master
# 创建功能分支
git checkout -b feature/新功能
# 提交更改
git add .
git commit -m "描述信息"
git push origin feature/新功能
# 在Gitea界面创建Pull Request
```
---
**联系信息**
- Gitea服务地址http://你的IP:3000
- 管理员:[你的联系方式]
- 紧急联系:[备用联系方式]
**注意**:请确保定期备份重要代码,避免数据丢失。

View File

@@ -139,6 +139,12 @@
<artifactId>report-generator</artifactId> <artifactId>report-generator</artifactId>
<version>1.0.0</version> <version>1.0.0</version>
</dependency> </dependency>
<!--激活工具-->
<dependency>
<groupId>com.njcn.gather</groupId>
<artifactId>activate-tool</artifactId>
<version>1.0.0</version>
</dependency>
</dependencies> </dependencies>

View File

@@ -10,6 +10,7 @@ import com.njcn.gather.detection.pojo.param.ContrastDetectionParam;
import com.njcn.gather.detection.pojo.param.PreDetectionParam; import com.njcn.gather.detection.pojo.param.PreDetectionParam;
import com.njcn.gather.detection.pojo.param.SimulateDetectionParam; import com.njcn.gather.detection.pojo.param.SimulateDetectionParam;
import com.njcn.gather.detection.service.PreDetectionService; import com.njcn.gather.detection.service.PreDetectionService;
import com.njcn.gather.detection.util.socket.CnSocketUtil;
import com.njcn.web.controller.BaseController; import com.njcn.web.controller.BaseController;
import com.njcn.web.utils.HttpResultUtil; import com.njcn.web.utils.HttpResultUtil;
import io.swagger.annotations.Api; import io.swagger.annotations.Api;
@@ -155,4 +156,47 @@ public class PreDetectionController extends BaseController {
LogUtil.njcnDebug(log, "{}", methodDescribe); LogUtil.njcnDebug(log, "{}", methodDescribe);
preDetectionService.exportAlignData(); preDetectionService.exportAlignData();
} }
@OperateInfo(info = LogEnum.SYSTEM_COMMON)
@GetMapping("/canCoefficient")
@ApiOperation("比对模式是否能够进行系数校验")
public HttpResult<Boolean> canCoefficient() {
String methodDescribe = getMethodDescribe("canCoefficient");
LogUtil.njcnDebug(log, "{}", methodDescribe);
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, preDetectionService.getCanCoefficient(), methodDescribe);
}
@OperateInfo(info = LogEnum.SYSTEM_COMMON)
@GetMapping("/startCoefficient")
@ApiOperation("比对模式开启系数校验")
public HttpResult<String> startCoefficient() {
String methodDescribe = getMethodDescribe("startCoefficient");
LogUtil.njcnDebug(log, "{}", methodDescribe);
preDetectionService.startCoefficient();
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, null, methodDescribe);
}
@OperateInfo(info = LogEnum.SYSTEM_COMMON)
@GetMapping("/startFreqConverter")
@ApiOperation("开启变频器测试")
public HttpResult<String> startFreqConverter(@RequestParam("userId") String userId, @RequestParam("converterId") String converterId, @RequestParam("monitorId") String monitorId, @RequestParam("reset") Boolean reset) {
String methodDescribe = getMethodDescribe("startFreqConverter");
LogUtil.njcnDebug(log, "{}", methodDescribe);
preDetectionService.startFreqConverter(userId, converterId, monitorId,reset);
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, null, methodDescribe);
}
@OperateInfo(info = LogEnum.SYSTEM_COMMON)
@GetMapping("/stopFreqConverter")
@ApiOperation("关闭变频器测试")
public HttpResult<String> stopFreqConverter(@RequestParam("userId") String userId) {
String methodDescribe = getMethodDescribe("stopFreqConverter");
LogUtil.njcnDebug(log, "{}", methodDescribe);
preDetectionService.stopFreqConverter(userId + CnSocketUtil.FREQ_CONVERTER_TAG, userId + CnSocketUtil.DEV_TAG);
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, null, methodDescribe);
}
} }

View File

@@ -439,6 +439,7 @@ public class SocketDevResponseService {
//告诉前端当前项开始了 //告诉前端当前项开始了
WebSocketVO<Object> webSocketVO = new WebSocketVO<>(); WebSocketVO<Object> webSocketVO = new WebSocketVO<>();
FormalTestManager.currentIssue = sourceIssues.get(0);
String type = sourceIssues.get(0).getType(); String type = sourceIssues.get(0).getType();
if (ResultUnitEnum.P.getCode().equals(type)) { if (ResultUnitEnum.P.getCode().equals(type)) {
sourceIssues.get(0).setType(ResultUnitEnum.V_ABSOLUTELY.getCode()); sourceIssues.get(0).setType(ResultUnitEnum.V_ABSOLUTELY.getCode());
@@ -612,7 +613,7 @@ public class SocketDevResponseService {
private void assemblyEntity(List<DevData> deList, DevXiNumData.F F, DevXiNumData.GF startF, CoefficientVO coefficientVO, CoefficientVO.DevParameter devParameter) { private void assemblyEntity(List<DevData> deList, DevXiNumData.F F, DevXiNumData.GF startF, CoefficientVO coefficientVO, CoefficientVO.DevParameter devParameter) {
//表示接收完成,必须保证3个数 //表示接收完成,必须保证3个数
if (deList.size() >= 3) { if (deList.size() >= 7) {
List<Double> aList = deList.stream().map(it -> it.getSqlData().get(0).getList().getA()).collect(Collectors.toList()); List<Double> aList = deList.stream().map(it -> it.getSqlData().get(0).getList().getA()).collect(Collectors.toList());
List<Double> bList = deList.stream().map(it -> it.getSqlData().get(0).getList().getB()).collect(Collectors.toList()); List<Double> bList = deList.stream().map(it -> it.getSqlData().get(0).getList().getB()).collect(Collectors.toList());
List<Double> cList = deList.stream().map(it -> it.getSqlData().get(0).getList().getC()).collect(Collectors.toList()); List<Double> cList = deList.stream().map(it -> it.getSqlData().get(0).getList().getC()).collect(Collectors.toList());
@@ -687,7 +688,7 @@ public class SocketDevResponseService {
*/ */
private Double reduceList(List<Double> valList) { private Double reduceList(List<Double> valList) {
// valList.subList(0, 5).clear(); // valList.subList(0, 5).clear();
// valList.subList(valList.size() - 3, valList.size() - 1).clear(); valList.subList(valList.size() - 2, valList.size()).clear();
return valList.stream().mapToDouble(Double::doubleValue).average().getAsDouble(); return valList.stream().mapToDouble(Double::doubleValue).average().getAsDouble();
} }
@@ -849,7 +850,7 @@ public class SocketDevResponseService {
boolean isContinue = true; boolean isContinue = true;
List<String> descList = icdCheckData.getResultData().stream().map( List<String> descList = icdCheckData.getResultData().stream().map(
obj -> { obj -> {
if (DetectionCodeEnum.MAG.getCode().equals(obj.getDesc()) || DetectionCodeEnum.DUR.getCode().equals(obj.getDesc())) { if (DetectionCodeEnum.MAG.getCode().equals(obj.getDesc()) || DetectionCodeEnum.DUR.getCode().equals(obj.getDesc()) || DetectionCodeEnum.PST.getCode().equals(obj.getDesc())) {
return DetectionCodeEnum.AVG_PREFIX.getCode() + obj.getDesc(); return DetectionCodeEnum.AVG_PREFIX.getCode() + obj.getDesc();
} else { } else {
return DetectionCodeEnum.REAL_PREFIX.getCode() + obj.getDesc(); return DetectionCodeEnum.REAL_PREFIX.getCode() + obj.getDesc();
@@ -953,6 +954,7 @@ public class SocketDevResponseService {
} else { } else {
webSocketVO.setRequestId(sourceIssues.get(0).getType() + CnSocketUtil.START_TAG); webSocketVO.setRequestId(sourceIssues.get(0).getType() + CnSocketUtil.START_TAG);
} }
FormalTestManager.currentIssue = sourceIssues.get(0);
socketMsg.setData(JSON.toJSONString(sourceIssues.get(0))); socketMsg.setData(JSON.toJSONString(sourceIssues.get(0)));
socketMsg.setRequestId(SourceOperateCodeEnum.FORMAL_REAL.getValue() + CnSocketUtil.STEP_TAG + type); socketMsg.setRequestId(SourceOperateCodeEnum.FORMAL_REAL.getValue() + CnSocketUtil.STEP_TAG + type);
socketMsg.setOperateCode(SourceOperateCodeEnum.OPER_GATHER.getValue()); socketMsg.setOperateCode(SourceOperateCodeEnum.OPER_GATHER.getValue());
@@ -1210,6 +1212,7 @@ public class SocketDevResponseService {
} else { } else {
webSocketVO.setRequestId(sourceIssues.get(0).getType() + CnSocketUtil.START_TAG); webSocketVO.setRequestId(sourceIssues.get(0).getType() + CnSocketUtil.START_TAG);
} }
FormalTestManager.currentIssue = sourceIssues.get(0);
socketMsg.setData(JSON.toJSONString(sourceIssues.get(0))); socketMsg.setData(JSON.toJSONString(sourceIssues.get(0)));
socketMsg.setRequestId(SourceOperateCodeEnum.FORMAL_REAL.getValue() + CnSocketUtil.STEP_TAG + type); socketMsg.setRequestId(SourceOperateCodeEnum.FORMAL_REAL.getValue() + CnSocketUtil.STEP_TAG + type);
socketMsg.setOperateCode(SourceOperateCodeEnum.OPER_GATHER.getValue()); socketMsg.setOperateCode(SourceOperateCodeEnum.OPER_GATHER.getValue());
@@ -1371,7 +1374,7 @@ public class SocketDevResponseService {
checkDataParam.setIsValueTypeName(false); checkDataParam.setIsValueTypeName(false);
List<String> valueType = iPqScriptCheckDataService.getValueType(checkDataParam); List<String> valueType = iPqScriptCheckDataService.getValueType(checkDataParam);
iPqDevService.updateResult( param.getDevIds(), valueType, param.getCode(), param.getUserId(), param.getTemperature(), param.getHumidity()); iPqDevService.updateResult(param.getDevIds(), valueType, param.getCode(), param.getUserId(), param.getTemperature(), param.getHumidity());
CnSocketUtil.quitSend(param); CnSocketUtil.quitSend(param);
} }
successComm.clear(); successComm.clear();
@@ -1782,6 +1785,7 @@ public class SocketDevResponseService {
} else { } else {
dataRule = DictDataEnum.SECTION_VALUE; dataRule = DictDataEnum.SECTION_VALUE;
} }
FormalTestManager.currentTestPlan = plan;
String code = dictDataService.getDictDataById(plan.getPattern()).getCode(); String code = dictDataService.getDictDataById(plan.getPattern()).getCode();
FormalTestManager.patternEnum = PatternEnum.getEnum(code); FormalTestManager.patternEnum = PatternEnum.getEnum(code);

View File

@@ -0,0 +1,369 @@
package com.njcn.gather.detection.handler;
import cn.hutool.core.collection.CollUtil;
import cn.hutool.core.collection.ListUtil;
import cn.hutool.core.util.IdUtil;
import cn.hutool.core.util.StrUtil;
import com.alibaba.fastjson.JSON;
import com.njcn.db.mybatisplus.handler.DynamicTableNameHandler;
import com.njcn.gather.detection.pojo.enums.DetectionCodeEnum;
import com.njcn.gather.detection.pojo.enums.SourceOperateCodeEnum;
import com.njcn.gather.detection.pojo.enums.SourceResponseCodeEnum;
import com.njcn.gather.detection.pojo.param.DevPhaseSequenceParam;
import com.njcn.gather.detection.pojo.po.DevData;
import com.njcn.gather.detection.pojo.vo.SocketDataMsg;
import com.njcn.gather.detection.pojo.vo.SocketMsg;
import com.njcn.gather.detection.util.socket.CnSocketUtil;
import com.njcn.gather.detection.util.socket.FormalTestManager;
import com.njcn.gather.detection.util.socket.MsgUtil;
import com.njcn.gather.detection.util.socket.SocketManager;
import com.njcn.gather.detection.util.socket.cilent.NettyClient;
import com.njcn.gather.detection.util.socket.cilent.NettyFreqConverterDevClientHandler;
import com.njcn.gather.detection.util.socket.config.SocketConnectionConfig;
import com.njcn.gather.detection.util.socket.websocket.WebServiceManager;
import com.njcn.gather.device.pojo.po.PqDev;
import com.njcn.gather.device.pojo.vo.PreDetection;
import com.njcn.gather.device.service.IPqDevService;
import com.njcn.gather.dip.pojo.po.PqDipData;
import com.njcn.gather.dip.service.IPqDipDataService;
import com.njcn.gather.freqConverter.config.FreqConverterConfig;
import com.njcn.gather.freqConverter.pojo.po.FreqConverterStatus;
import com.njcn.gather.freqConverter.pojo.po.PqFreqConverterTestRes;
import com.njcn.gather.freqConverter.service.IFreqConverterService;
import com.njcn.gather.freqConverter.service.IPqFreqConverterTestResService;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import java.time.LocalDateTime;
import java.util.*;
import java.util.concurrent.CompletableFuture;
import java.util.stream.Collectors;
@Slf4j
@Service
@RequiredArgsConstructor
public class SocketFreqConverterDevService {
private final SocketConnectionConfig socketConnectionConfig;
private final IPqDevService pqDevService;
private final IPqDipDataService pqDipDataService;
private final IFreqConverterService freqConverterService;
private final IPqFreqConverterTestResService pqFreqConverterTestResService;
private final FreqConverterConfig freqConverterConfig;
private String monitorId;
private String userId;
public static final String DIP_DATA_SUFFIX = "&&VOLTAGE";
/**
* 连接设备Socket
*
* @param devTag 设备Channel唯一标识符
*/
public void connectSocket(String devTag) {
if (SocketManager.isChannelActive(devTag)) {
return;
}
String ip = socketConnectionConfig.getDevice().getIp();
Integer port = socketConnectionConfig.getDevice().getPort();
NettyFreqConverterDevClientHandler handler = new NettyFreqConverterDevClientHandler(devTag, this);
CompletableFuture.runAsync(() -> {
NettyClient.commonConnect(ip, port, devTag, handler);
});
}
private void init(String userId, String converterId, String monitorId, Boolean reset) {
FormalTestManager.freqConverterDevStep = null;
// FormalTestManager.stopFlag = false;
FormalTestManager.isRemoveSocket = false;
FormalTestManager.pendingDipTaskMap.clear();
if (reset) {
pqDipDataService.clearAllData(FormalTestManager.freqConverterTableSuffix);
pqFreqConverterTestResService.clearAllData(FormalTestManager.freqConverterTableSuffix);
}
this.userId = userId;
this.monitorId = monitorId;
}
/**
* 连接设备
*/
public void connectionDev(String userId, String devTag, String converterId, String monitorId, Boolean reset) {
this.init(userId, converterId, monitorId, reset);
String payload = buildSingleMonitorPayload(monitorId);
if (StrUtil.isBlank(payload)) {
return;
}
SocketMsg<String> socketMsg = new SocketMsg<>();
socketMsg.setRequestId(SourceOperateCodeEnum.YJC_SBTXJY.getValue());
socketMsg.setOperateCode(SourceOperateCodeEnum.DEV_INIT_GATHER_03.getValue());
socketMsg.setData(payload);
SocketManager.sendMsg(devTag, JSON.toJSONString(socketMsg));
FormalTestManager.freqConverterDevStep = SourceOperateCodeEnum.YJC_SBTXJY;
}
public void handleRead(String devTag, String msg) {
SocketDataMsg socketDataMsg = MsgUtil.socketDataMsg(msg);
switch (FormalTestManager.freqConverterDevStep) {
case YJC_SBTXJY:
handleYjcSbtxjy(devTag, socketDataMsg);
break;
case FORMAL_REAL:
handleFormalReal(devTag, socketDataMsg);
break;
case QUITE:
handleQuit(devTag, socketDataMsg);
break;
}
}
private void handleYjcSbtxjy(String devTag, SocketDataMsg socketDataMsg) {
SourceResponseCodeEnum responseCodeEnum = SourceResponseCodeEnum.getDictDataEnumByCode(socketDataMsg.getCode());
switch (Objects.requireNonNull(responseCodeEnum)) {
case UNPROCESSED_BUSINESS:
break;
case RE_OPERATE:
WebServiceManager.sendMsg(this.userId, JSON.toJSONString(socketDataMsg));
break;
case SUCCESS:
// 暂态协议触发后等待5秒将装置历史缓存的暂态数据给抛掉
CompletableFuture.runAsync(() -> {
try {
Thread.sleep(5000);
this.sendGetDipDataMsg(devTag);
FormalTestManager.freqConverterDevStep = SourceOperateCodeEnum.FORMAL_REAL;
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
log.error("异步调用sendGetDipDataMsg被中断", e);
}
});
break;
default:
log.warn("设备响应异常devTag={}, operateCode={}, code={}, data={}", devTag, socketDataMsg.getOperateCode(), socketDataMsg.getCode(), socketDataMsg.getData());
break;
}
}
private void handleFormalReal(String devTag, SocketDataMsg socketDataMsg) {
SourceResponseCodeEnum responseCodeEnum = SourceResponseCodeEnum.getDictDataEnumByCode(socketDataMsg.getCode());
switch (responseCodeEnum) {
case UNPROCESSED_BUSINESS:
break;
case SUCCESS:
case NORMAL_RESPONSE:
DevData devData = JSON.parseObject(socketDataMsg.getData(), DevData.class);
// 如果变频器不是处于 “故障中” 状态,就保存数据,反之,这段时期内的数据不保存
// if (!FormalTestManager.stopFlag) {
// saveDipData(devData);
// }
saveDipData(devData);
break;
case DEV_ERROR:
case DEV_TARGET:
case COMMUNICATION_ERR:
case DATA_RESOLVE:
case NO_INIT_DEV:
default:
log.warn("设备响应异常devTag={}, operateCode={}, code={}, data={}", devTag, socketDataMsg.getOperateCode(), socketDataMsg.getCode(), socketDataMsg.getData());
break;
}
}
private void handleQuit(String devTag, SocketDataMsg socketDataMsg) {
SourceResponseCodeEnum responseCodeEnum = SourceResponseCodeEnum.getDictDataEnumByCode(socketDataMsg.getCode());
switch (responseCodeEnum) {
case UNPROCESSED_BUSINESS:
break;
case SUCCESS:
cleanup(devTag);
break;
default:
log.warn("设备关闭响应失败devTag={}, operateCode={}, code={}, data={}", devTag, socketDataMsg.getOperateCode(), socketDataMsg.getCode(), socketDataMsg.getData());
break;
}
}
public void stopTest(String converterTag, String devTag) {
FormalTestManager.freqConverterDevStep = SourceOperateCodeEnum.QUITE;
sendQuitMsg(devTag, SourceOperateCodeEnum.QUIT_INIT_03);
}
private String buildSingleMonitorPayload(String monitorId) {
String[] split = monitorId.split(CnSocketUtil.SPLIT_TAG);
if (split.length < 2 || StrUtil.isBlank(split[0]) || StrUtil.isBlank(split[1])) {
return null;
}
List<PreDetection> preDetections = pqDevService.getDevInfo(Collections.singletonList(split[0]));
if (CollUtil.isEmpty(preDetections)) {
return null;
}
PreDetection preDetection = preDetections.get(0);
List<PreDetection.MonitorListDTO> monitorList = preDetection.getMonitorList();
if (CollUtil.isEmpty(monitorList)) {
return null;
}
List<PreDetection.MonitorListDTO> matchedMonitorList = monitorList.stream()
.filter(item -> split[1].equals(StrUtil.EMPTY + item.getLine()))
.collect(Collectors.toList());
if (CollUtil.isEmpty(matchedMonitorList)) {
return null;
}
preDetection.setMonitorList(matchedMonitorList);
Map<String, List<PreDetection>> payload = new HashMap<>(1);
payload.put("deviceList", Collections.singletonList(preDetection));
return JSON.toJSONString(payload);
}
private void sendGetDipDataMsg(String devTag) {
SocketMsg<String> socketMsg = new SocketMsg<>();
socketMsg.setRequestId(SourceOperateCodeEnum.FORMAL_REAL.getValue() + DIP_DATA_SUFFIX);
socketMsg.setOperateCode(SourceOperateCodeEnum.OPER_GATHER.getValue());
DevPhaseSequenceParam phaseSequenceParam = new DevPhaseSequenceParam();
String[] split = this.monitorId.split(String.valueOf(StrUtil.C_UNDERLINE));
PqDev dev = pqDevService.getById(split[0]);
// 设置监测点ID列表
phaseSequenceParam.setMoniterIdList(ListUtil.of(dev.getIp() + StrUtil.C_UNDERLINE + split[1]));
// 设置数据类型列表
phaseSequenceParam.setDataType(ListUtil.of("avg$MAG", "avg$DUR"));
// 设置读取次数
phaseSequenceParam.setReadCount(0);
// 设置忽略次数
phaseSequenceParam.setIgnoreCount(0);
socketMsg.setData(JSON.toJSONString(phaseSequenceParam));
socketMsg.setOperateCode(SourceOperateCodeEnum.DEV_DATA_REQUEST_03.getValue());
SocketManager.sendMsg(devTag, JSON.toJSONString(socketMsg));
}
private void sendQuitMsg(String devTag, SourceOperateCodeEnum operateCodeEnum) {
SocketMsg<String> socketMsg = new SocketMsg<>();
socketMsg.setRequestId(SourceOperateCodeEnum.QUITE.getValue());
socketMsg.setOperateCode(operateCodeEnum.getValue());
SocketManager.sendMsg(devTag, JSON.toJSONString(socketMsg));
}
private void saveDipData(DevData devData) {
if (Objects.isNull(devData) || CollUtil.isEmpty(devData.getSqlData())) {
return;
}
Double residualVoltage = null;
Integer durationMs = null;
for (DevData.SqlDataDTO sqlDataDTO : devData.getSqlData()) {
if (Objects.isNull(sqlDataDTO) || Objects.isNull(sqlDataDTO.getList())) {
continue;
}
Double value = getSqlDataValue(sqlDataDTO.getList());
if (Objects.isNull(value)) {
continue;
}
if (DetectionCodeEnum.MAG.getCode().equalsIgnoreCase(sqlDataDTO.getDesc())) {
residualVoltage = value;
} else if (DetectionCodeEnum.DUR.getCode().equalsIgnoreCase(sqlDataDTO.getDesc())) {
durationMs = (int) Math.round(value * 1000);
}
}
PqDipData pqDipData = new PqDipData();
pqDipData.setId(IdUtil.fastSimpleUUID());
pqDipData.setStartTime(LocalDateTime.parse(devData.getTime()));
pqDipData.setResidualVoltage(residualVoltage);
pqDipData.setDurationMs(durationMs);
DynamicTableNameHandler.setTableName("pq_dip_data_" + FormalTestManager.freqConverterTableSuffix);
pqDipDataService.save(pqDipData);
DynamicTableNameHandler.remove();
this.initDipTestRes(pqDipData);
}
private void initDipTestRes(PqDipData pqDipData) {
Integer suffix = FormalTestManager.freqConverterTableSuffix;
FreqConverterStatus lastStatusData = freqConverterService.getLastStatusData(suffix, pqDipData.getStartTime());
if (Objects.isNull(lastStatusData)) {
return;
}
List<FreqConverterStatus> statusList = freqConverterService.getDipDurationStatusData(suffix, lastStatusData.getTimestamp(), pqDipData.getStartTime().plusNanos(pqDipData.getDurationMs() * 1000_000L));
Integer originalTolerant = (lastStatusData.getStatusWord1() == freqConverterConfig.getTolerant()) ? 1 : 0;
LocalDateTime targetEndTime = pqDipData.getStartTime()
.plusNanos(pqDipData.getDurationMs() * 1000_000L)
.plusNanos(freqConverterConfig.getDt() * 1000_000L);
if (CollUtil.isNotEmpty(statusList)) {
FreqConverterStatus status = statusList.get(statusList.size() - 1);
PqFreqConverterTestRes testRes = new PqFreqConverterTestRes();
testRes.setId(IdUtil.fastSimpleUUID());
testRes.setTolerant(originalTolerant == 1 ?
(status.getStatusWord1() == freqConverterConfig.getTolerant() ? 1 : 0)
: 0);
testRes.setDurationMs(pqDipData.getDurationMs());
testRes.setResidualVoltage(pqDipData.getResidualVoltage());
testRes.setTime(LocalDateTime.now());
FormalTestManager.pendingDipTaskMap.put(testRes.getId(), new FormalTestManager.PendingDipTask(
pqDipData,
targetEndTime,
originalTolerant
));
pqFreqConverterTestResService.saveTestRes(suffix, Collections.singletonList(testRes));
}
}
private Double getSqlDataValue(DevData.SqlDataDTO.ListDTO listDTO) {
if (Objects.nonNull(listDTO.getA())) {
return listDTO.getA();
}
if (Objects.nonNull(listDTO.getB())) {
return listDTO.getB();
}
if (Objects.nonNull(listDTO.getC())) {
return listDTO.getC();
}
return listDTO.getT();
}
public void cleanup(String devTag) {
String currentUserId = this.userId;
FormalTestManager.freqConverterDevStep = null;
FormalTestManager.isRemoveSocket = true;
SocketManager.removeUser(devTag);
clearStateIfStopped(currentUserId);
this.userId = null;
this.monitorId = null;
}
/**
* 如果设备已停止,则清除共享的运行时状态
*
* @param currentUserId 当前用户ID
*/
private void clearStateIfStopped(String currentUserId) {
if (StrUtil.isBlank(currentUserId)) {
FormalTestManager.clearFreqConverterRuntimeState();
return;
}
String freqConverterTag = currentUserId + CnSocketUtil.FREQ_CONVERTER_TAG;
// 避免过早把 freqConverterTableSuffix 等全局值清掉,造成变频器无法使用全局变量
if (!SocketManager.isChannelActive(freqConverterTag)) {
FormalTestManager.freqConverterStep = null;
FormalTestManager.clearFreqConverterRuntimeState();
}
}
}

View File

@@ -0,0 +1,433 @@
package com.njcn.gather.detection.handler;
import cn.hutool.core.util.IdUtil;
import cn.hutool.core.util.ObjectUtil;
import cn.hutool.core.util.StrUtil;
import cn.hutool.json.JSONObject;
import cn.hutool.json.JSONUtil;
import com.alibaba.fastjson.JSON;
import com.njcn.gather.detection.pojo.dto.FreqConverterRespDTO;
import com.njcn.gather.detection.pojo.enums.SourceOperateCodeEnum;
import com.njcn.gather.detection.pojo.vo.SocketDataMsg;
import com.njcn.gather.detection.pojo.vo.SocketMsg;
import com.njcn.gather.detection.util.socket.CnSocketUtil;
import com.njcn.gather.detection.util.socket.FormalTestManager;
import com.njcn.gather.detection.util.socket.SocketManager;
import com.njcn.gather.detection.util.socket.cilent.NettyClient;
import com.njcn.gather.detection.util.socket.cilent.NettyFreqConverterClientHandler;
import com.njcn.gather.detection.util.socket.config.SocketConnectionConfig;
import com.njcn.gather.detection.util.socket.websocket.WebServiceManager;
import com.njcn.gather.freqConverter.config.FreqConverterConfig;
import com.njcn.gather.freqConverter.pojo.po.FreqConverterStatus;
import com.njcn.gather.freqConverter.pojo.po.PqFreqConverterConfig;
import com.njcn.gather.freqConverter.pojo.po.PqFreqConverterTestRes;
import com.njcn.gather.freqConverter.pojo.vo.TolerantPointVO;
import com.njcn.gather.freqConverter.service.IFreqConverterService;
import com.njcn.gather.freqConverter.service.IPqFreqConverterConfigService;
import com.njcn.gather.freqConverter.service.IPqFreqConverterTestResService;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import java.time.LocalDateTime;
import java.util.*;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
/**
* @author czh
* @version 1.0
*/
@Slf4j
@Service
@RequiredArgsConstructor
public class SocketFreqConverterService {
private final IFreqConverterService freqConverterService;
private final SocketConnectionConfig socketConnectionConfig;
private final IPqFreqConverterConfigService pqFreqConverterConfigService;
private final IPqFreqConverterTestResService pqFreqConverterTestResService;
private String userId;
/**
* 上一个暂降点耐受检测结果
*/
private TolerantPointVO lastTolerancePoint;
private final FreqConverterConfig freqConverterConfig;
/**
* 连接变频器Socket
*
* @param converterChannelTag 变频器Channel唯一标识符
*/
public void connectSocket(String converterChannelTag) {
if (SocketManager.isChannelActive(converterChannelTag)) {
return;
}
String ip = socketConnectionConfig.getFreqConverter().getIp();
Integer port = socketConnectionConfig.getFreqConverter().getPort();
NettyFreqConverterClientHandler handler = new NettyFreqConverterClientHandler(converterChannelTag, this);
CompletableFuture.runAsync(() -> {
NettyClient.commonConnect(ip, port, converterChannelTag, handler);
});
}
/**
* 重连
*
* @param converterChannelTag
*/
public void reconnect(String converterChannelTag) {
SocketManager.removeUser(converterChannelTag);
String ip = socketConnectionConfig.getFreqConverter().getIp();
Integer port = socketConnectionConfig.getFreqConverter().getPort();
NettyFreqConverterClientHandler handler = new NettyFreqConverterClientHandler(converterChannelTag, this);
CompletableFuture.runAsync(() -> {
NettyClient.commonConnect(ip, port, converterChannelTag, handler);
});
}
/**
* 重试连接成功,重新开启定时任务,获取变频器状态数据
*
* @param converterChannelTag
*/
public void onReconnectSuccess(String converterChannelTag) {
log.info("变频器重连成功恢复数据采集converterChannelTag={}", converterChannelTag);
// FormalTestManager.stopFlag = false;
if (FormalTestManager.scheduler == null) {
FormalTestManager.scheduler = Executors.newScheduledThreadPool(1);
FormalTestManager.scheduledFuture = FormalTestManager.scheduler.scheduleAtFixedRate(() -> {
this.sendGetDeviceStatusMsg(converterChannelTag);
}, 0l, 200l, TimeUnit.MILLISECONDS);
}
}
public void init(String userId, String converterId, String monitorId, Boolean reset) {
this.lastTolerancePoint = null;
this.userId = userId;
FormalTestManager.freqConverterStep = null;
FormalTestManager.currentFreqConverterId = converterId;
Integer suffix = pqFreqConverterConfigService.getSuffix(converterId);
if (reset) {
freqConverterService.clearAllData(suffix);
}
FormalTestManager.freqConverterTableSuffix = suffix;
FormalTestManager.isRemoveSocket = false;
FormalTestManager.pendingDipTaskMap.clear();
pqFreqConverterConfigService.updateTestStatus(converterId, 0);
clearScheduleTask();
}
/**
* 连接变频器
*/
public void connectionFreqConverter(String userId, String freqConverterTag, String converterId, String monitorId, Boolean reset) {
this.init(userId, converterId, monitorId, reset);
SocketMsg<Map<String, Object>> socketMsg = new SocketMsg<>();
socketMsg.setOperateCode(SourceOperateCodeEnum.CMD_INIT_SERIAL.getValue());
String requestId = IdUtil.fastSimpleUUID();
socketMsg.setRequestId(requestId);
PqFreqConverterConfig freqConverterConfig = pqFreqConverterConfigService.getById(converterId);
Map<String, Object> map = new HashMap<>();
map.put("portName", freqConverterConfig.getPortName());
map.put("slaveAddress", freqConverterConfig.getSlaveAddress());
map.put("baudRate", freqConverterConfig.getBaudRate());
map.put("parity", freqConverterConfig.getParity());
map.put("dataBits", freqConverterConfig.getDataBits());
map.put("stopBits", freqConverterConfig.getStopBits());
map.put("timeoutMs", freqConverterConfig.getTimeoutMs());
socketMsg.setData(map);
SocketManager.sendMsg(freqConverterTag, JSON.toJSONString(socketMsg));
FormalTestManager.freqConverterStep = SourceOperateCodeEnum.CMD_INIT_SERIAL;
}
public void handleRead(String converterChannelTag, String msg) {
FreqConverterRespDTO respDTO = JSON.parseObject(msg, FreqConverterRespDTO.class);
if (respDTO.getCode() != 0) {
SocketDataMsg socketDataMsg = new SocketDataMsg();
socketDataMsg.setRequestId(FormalTestManager.freqConverterStep.getValue());
socketDataMsg.setCode(respDTO.getCode());
WebServiceManager.sendMsg(this.userId, JSON.toJSONString(socketDataMsg));
} else {
switch (FormalTestManager.freqConverterStep) {
case CMD_INIT_SERIAL:
handleInitSerial(converterChannelTag, respDTO);
break;
case CMD_GET_DEVICE_STATUS:
handleGetDeviceStatus(converterChannelTag, respDTO);
break;
case CMD_CLOSE_SERIAL:
handleCloseSerial(converterChannelTag, respDTO);
break;
}
}
}
public void stopTest(String converterTag, String devTag) {
FormalTestManager.freqConverterStep = SourceOperateCodeEnum.CMD_CLOSE_SERIAL;
this.sendClose(converterTag);
}
public void cleanup(String converterChannelTag) {
String currentUserId = this.userId;
clearScheduleTask();
FormalTestManager.freqConverterStep = null;
// FormalTestManager.stopFlag = false;
FormalTestManager.isRemoveSocket = true;
SocketManager.removeUser(converterChannelTag);
updateCurrentTestStatus();
clearStateIfStopped(currentUserId);
this.userId = null;
this.lastTolerancePoint = null;
}
private void handleInitSerial(String converterChannelTag, FreqConverterRespDTO respDTO) {
if (respDTO.getCode() == 0 && respDTO.getSuccess()) {
FormalTestManager.freqConverterStep = SourceOperateCodeEnum.CMD_GET_DEVICE_STATUS;
if (Objects.isNull(FormalTestManager.scheduler)) {
FormalTestManager.scheduler = Executors.newScheduledThreadPool(1);
FormalTestManager.scheduledFuture = FormalTestManager.scheduler.scheduleAtFixedRate(() -> {
this.sendGetDeviceStatusMsg(converterChannelTag);
}, 0l, freqConverterConfig.getSchedulePeriod(), TimeUnit.MILLISECONDS);
}
} else {
log.warn("变频器初始化串口失败converterChannelTag={}, converterId={}, converterTag={}, msg={}", converterChannelTag, converterChannelTag, converterChannelTag, respDTO.getMessage());
}
}
private void handleGetDeviceStatus(String converterChannelTag, FreqConverterRespDTO respDTO) {
JSONObject obj = JSONUtil.parseObj(respDTO.getData().toString());
String timestamp = (String) obj.get("Timestamp");
timestamp = timestamp.replace("+08:00", StrUtil.EMPTY);
obj.set("Timestamp", timestamp);
FreqConverterStatus freqConverterStatus = JSON.parseObject(obj.toString(), FreqConverterStatus.class);
// 变频器故障中,移除这段时期内的设备数据
// if (freqConverterStatus.getStatusWord1() == freqConverterConfig.getNoTolerant()) {
// FormalTestManager.stopFlag = true;
// } else {
// FormalTestManager.stopFlag = false;
// }
this.consumePendingDipTasks(freqConverterStatus);
freqConverterService.saveFreqConverterStatus(FormalTestManager.freqConverterTableSuffix, freqConverterStatus);
}
private void handleCloseSerial(String converterChannelTag, FreqConverterRespDTO respDTO) {
if (respDTO.getCode() == 0 && respDTO.getSuccess()) {
if (FormalTestManager.currentFreqConverterId != null) {
pqFreqConverterConfigService.updateTestStatus(FormalTestManager.currentFreqConverterId, 1);
}
cleanup(converterChannelTag);
} else {
this.sendClose(converterChannelTag);
}
}
private void sendGetDeviceStatusMsg(String converterId) {
SocketMsg<Map<String, Object>> socketMsg = new SocketMsg<>();
socketMsg.setOperateCode(SourceOperateCodeEnum.CMD_GET_DEVICE_STATUS.getValue());
String requestId = IdUtil.fastSimpleUUID();
socketMsg.setRequestId(requestId);
Map<String, Object> map = new HashMap<>();
socketMsg.setData(map);
SocketManager.sendMsg(converterId, JSON.toJSONString(socketMsg));
}
private void sendClose(String converterTag) {
SocketMsg<Map<String, Object>> socketMsg = new SocketMsg<>();
socketMsg.setOperateCode(SourceOperateCodeEnum.CMD_CLOSE_SERIAL.getValue());
String requestId = IdUtil.fastSimpleUUID();
socketMsg.setRequestId(requestId);
Map<String, Object> map = new HashMap<>();
socketMsg.setData(map);
SocketManager.sendMsg(converterTag, JSON.toJSONString(socketMsg));
}
private void clearScheduleTask() {
if (FormalTestManager.scheduledFuture != null) {
FormalTestManager.scheduledFuture.cancel(true);
FormalTestManager.scheduledFuture = null;
}
if (FormalTestManager.scheduler != null) {
FormalTestManager.scheduler.shutdown();
FormalTestManager.scheduler = null;
}
}
private void updateCurrentTestStatus() {
if (StrUtil.isNotBlank(FormalTestManager.currentFreqConverterId)) {
pqFreqConverterConfigService.updateTestStatus(FormalTestManager.currentFreqConverterId, 1);
}
}
/**
* 如果变频器已停止,则清除共享的运行时状态
*
* @param currentUserId 当前用户ID
*/
private void clearStateIfStopped(String currentUserId) {
if (StrUtil.isBlank(currentUserId)) {
FormalTestManager.clearFreqConverterRuntimeState();
return;
}
String devTag = currentUserId + CnSocketUtil.DEV_TAG;
// 避免过早把 freqConverterTableSuffix 等全局值清掉,造成设备无法使用全局变量
if (!SocketManager.isChannelActive(devTag)) {
FormalTestManager.freqConverterDevStep = null;
FormalTestManager.clearFreqConverterRuntimeState();
}
}
private void consumePendingDipTasks(FreqConverterStatus freqConverterStatus) {
if (FormalTestManager.pendingDipTaskMap.isEmpty()) {
return;
}
Integer suffix = FormalTestManager.freqConverterTableSuffix;
List<String> finishedTestResIdList = new ArrayList<>();
List<PqFreqConverterTestRes> saveTestResList = new ArrayList<>();
List<PqFreqConverterTestRes> updateTestResList = new ArrayList<>();
FormalTestManager.pendingDipTaskMap.forEach((key, task) -> {
if (freqConverterStatus.getTimestamp().isAfter(task.getTargetEndTime())) {
PqFreqConverterTestRes testRes = new PqFreqConverterTestRes();
testRes.setId(key);
testRes.setDurationMs(task.getPqDipData().getDurationMs());
testRes.setResidualVoltage(task.getPqDipData().getResidualVoltage());
testRes.setTolerant(task.getOriginalTolerant() & (freqConverterStatus.getStatusWord1() == freqConverterConfig.getTolerant() ? 1 : 0));
finishedTestResIdList.add(key);
SocketDataMsg socketDataMsg = new SocketDataMsg();
socketDataMsg.setRequestId(SourceOperateCodeEnum.FORMAL_REAL + SocketFreqConverterDevService.DIP_DATA_SUFFIX);
TolerantPointVO newTolerantPointVO = new TolerantPointVO();
newTolerantPointVO.setResidualVoltage(task.getPqDipData().getResidualVoltage());
newTolerantPointVO.setDurationMs(task.getPqDipData().getDurationMs());
newTolerantPointVO.setTolerant(testRes.getTolerant());
socketDataMsg.setData(JSON.toJSONString(newTolerantPointVO));
WebServiceManager.sendMsg(this.userId, JSON.toJSONString(socketDataMsg));
if (testRes.getTolerant() == 0) {
if (ObjectUtil.isNotNull(this.lastTolerancePoint) && this.lastTolerancePoint.getTolerant() == 1) {
TolerantPointVO featurePointVO = new TolerantPointVO();
featurePointVO.setResidualVoltage(Math.round((task.getPqDipData().getResidualVoltage() + this.lastTolerancePoint.getResidualVoltage()) / 2D * 100) / 100D);
featurePointVO.setDurationMs(Integer.valueOf((task.getPqDipData().getDurationMs().intValue() + this.lastTolerancePoint.getDurationMs().intValue()) / 2));
featurePointVO.setTolerant(2);
socketDataMsg.setData(JSON.toJSONString(featurePointVO));
WebServiceManager.sendMsg(this.userId, JSON.toJSONString(socketDataMsg));
PqFreqConverterTestRes featureTestRes = new PqFreqConverterTestRes();
featureTestRes.setId(IdUtil.fastSimpleUUID());
featureTestRes.setDurationMs(featurePointVO.getDurationMs());
featureTestRes.setResidualVoltage(featurePointVO.getResidualVoltage());
featureTestRes.setTolerant(2);
featureTestRes.setTime(LocalDateTime.now());
saveTestResList.add(featureTestRes);
}
// 从数据库按照列查询距离该暂降点最近的一个暂降点
if (freqConverterConfig.getDirection() == 0) {
PqFreqConverterTestRes lastByDuration = pqFreqConverterTestResService.getLastByDuration(suffix, key, task.getPqDipData().getDurationMs());
if (ObjectUtil.isNotNull(lastByDuration) && lastByDuration.getTolerant() == 1) {
TolerantPointVO featurePointVO = new TolerantPointVO();
featurePointVO.setResidualVoltage(Math.round((task.getPqDipData().getResidualVoltage() + lastByDuration.getResidualVoltage()) / 2D * 100) / 100D);
featurePointVO.setDurationMs(Integer.valueOf((task.getPqDipData().getDurationMs().intValue() + lastByDuration.getDurationMs().intValue()) / 2));
featurePointVO.setTolerant(2);
socketDataMsg.setData(JSON.toJSONString(featurePointVO));
WebServiceManager.sendMsg(this.userId, JSON.toJSONString(socketDataMsg));
PqFreqConverterTestRes featureTestRes = new PqFreqConverterTestRes();
featureTestRes.setId(IdUtil.fastSimpleUUID());
featureTestRes.setDurationMs(featurePointVO.getDurationMs());
featureTestRes.setResidualVoltage(featurePointVO.getResidualVoltage());
featureTestRes.setTolerant(2);
featureTestRes.setTime(LocalDateTime.now());
saveTestResList.add(featureTestRes);
}
}
// 从数据库按照行查询距离该暂降点最近的一个暂降点
if (freqConverterConfig.getDirection() == 1) {
PqFreqConverterTestRes lastByResidualVoltage = pqFreqConverterTestResService.getLastByResidualVoltage(suffix, key, task.getPqDipData().getResidualVoltage());
if (ObjectUtil.isNotNull(lastByResidualVoltage) && lastByResidualVoltage.getTolerant() == 1) {
TolerantPointVO featurePointVO = new TolerantPointVO();
featurePointVO.setResidualVoltage(Math.round((task.getPqDipData().getResidualVoltage() + lastByResidualVoltage.getResidualVoltage()) / 2D * 100) / 100D);
featurePointVO.setDurationMs(Integer.valueOf((task.getPqDipData().getDurationMs().intValue() + lastByResidualVoltage.getDurationMs().intValue()) / 2));
featurePointVO.setTolerant(2);
socketDataMsg.setData(JSON.toJSONString(featurePointVO));
WebServiceManager.sendMsg(this.userId, JSON.toJSONString(socketDataMsg));
PqFreqConverterTestRes featureTestRes = new PqFreqConverterTestRes();
featureTestRes.setId(IdUtil.fastSimpleUUID());
featureTestRes.setDurationMs(featurePointVO.getDurationMs());
featureTestRes.setResidualVoltage(featurePointVO.getResidualVoltage());
featureTestRes.setTolerant(2);
featureTestRes.setTime(LocalDateTime.now());
saveTestResList.add(featureTestRes);
}
}
}
this.lastTolerancePoint = newTolerantPointVO;
updateTestResList.add(testRes);
}
});
if (!saveTestResList.isEmpty()) {
pqFreqConverterTestResService.saveTestRes(suffix, saveTestResList);
}
if (!updateTestResList.isEmpty()) {
pqFreqConverterTestResService.updateTestRes(suffix, updateTestResList);
}
for (String dipId : finishedTestResIdList) {
FormalTestManager.pendingDipTaskMap.remove(dipId);
}
}
/**
* 判断是否为新的一组测试脚本
*
* @param lastTolerancePoint 上一个暂降点
* @param newTolerantPointVO 最新的暂降点
* @return
*/
private boolean isNewGroup(TolerantPointVO lastTolerancePoint, TolerantPointVO newTolerantPointVO) {
// 横向分组
if (freqConverterConfig.getDirection() == 0) {
return lastTolerancePoint.getDurationMs() - newTolerantPointVO.getDurationMs() >= 10;
}
// 纵向分租
if (freqConverterConfig.getDirection() == 1) {
return lastTolerancePoint.getResidualVoltage() - newTolerantPointVO.getResidualVoltage() <= -2;
}
// if (Math.abs(lastTolerancePoint.getDurationMs() - newTolerantPointVO.getDurationMs()) >= 10 && Math.abs(lastTolerancePoint.getResidualVoltage() - newTolerantPointVO.getResidualVoltage()) >= 2) {
// return true;
// }
return false;
}
}

View File

@@ -2,6 +2,7 @@ package com.njcn.gather.detection.handler;
import cn.hutool.core.util.ObjectUtil; import cn.hutool.core.util.ObjectUtil;
import com.alibaba.fastjson.JSON; import com.alibaba.fastjson.JSON;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.njcn.gather.detection.pojo.enums.SourceOperateCodeEnum; import com.njcn.gather.detection.pojo.enums.SourceOperateCodeEnum;
import com.njcn.gather.detection.pojo.enums.SourceResponseCodeEnum; import com.njcn.gather.detection.pojo.enums.SourceResponseCodeEnum;
import com.njcn.gather.detection.pojo.param.DevPhaseSequenceParam; import com.njcn.gather.detection.pojo.param.DevPhaseSequenceParam;
@@ -14,7 +15,12 @@ import com.njcn.gather.detection.util.socket.*;
import com.njcn.gather.detection.util.socket.websocket.WebServiceManager; import com.njcn.gather.detection.util.socket.websocket.WebServiceManager;
import com.njcn.gather.device.pojo.vo.PreDetection; import com.njcn.gather.device.pojo.vo.PreDetection;
import com.njcn.gather.device.service.IPqDevService; import com.njcn.gather.device.service.IPqDevService;
import com.njcn.gather.plan.pojo.po.AdPlanSource;
import com.njcn.gather.plan.service.IAdPlanSourceService;
import com.njcn.gather.result.pojo.enums.ResultUnitEnum;
import com.njcn.gather.script.pojo.po.SourceIssue; import com.njcn.gather.script.pojo.po.SourceIssue;
import com.njcn.gather.source.pojo.po.SourceInitialize;
import com.njcn.gather.source.service.IPqSourceService;
import com.njcn.gather.system.pojo.enums.DicDataEnum; import com.njcn.gather.system.pojo.enums.DicDataEnum;
import lombok.RequiredArgsConstructor; import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
@@ -56,6 +62,8 @@ public class SocketSourceResponseService {
* 设备信息服务,提供设备基础信息查询功能 * 设备信息服务,提供设备基础信息查询功能
*/ */
private final IPqDevService iPqDevService; private final IPqDevService iPqDevService;
private final IAdPlanSourceService adPlanSourceService;
private final IPqSourceService pqSourceService;
/** /**
* Socket连接管理器负责管理设备和源的Socket连接 * Socket连接管理器负责管理设备和源的Socket连接
@@ -334,9 +342,9 @@ public class SocketSourceResponseService {
// 系数校验固定检测项:实时电压有效值和实时电流有效值 // 系数校验固定检测项:实时电压有效值和实时电流有效值
phaseSequenceParam.setDataType(Arrays.asList("real$VRMS", "real$IRMS")); phaseSequenceParam.setDataType(Arrays.asList("real$VRMS", "real$IRMS"));
// 读取3次数据用于系数计算 // 读取3次数据用于系数计算
phaseSequenceParam.setReadCount(3); phaseSequenceParam.setReadCount(7); //3
// 忽略前4次数据等待测量稳定 // 忽略前4次数据等待测量稳定
phaseSequenceParam.setIgnoreCount(4); phaseSequenceParam.setIgnoreCount(3); //4
socketMsg.setData(JSON.toJSONString(phaseSequenceParam)); socketMsg.setData(JSON.toJSONString(phaseSequenceParam));
SocketManager.sendMsg(s, JSON.toJSONString(socketMsg)); SocketManager.sendMsg(s, JSON.toJSONString(socketMsg));
@@ -363,18 +371,32 @@ public class SocketSourceResponseService {
SocketMsg<String> socketMsg = new SocketMsg<>(); SocketMsg<String> socketMsg = new SocketMsg<>();
switch (dictDataEnumByCode) { switch (dictDataEnumByCode) {
case SUCCESS: case SUCCESS:
//todo 前端推送收到的消息暂未处理好 if (FormalTestManager.unknownError) {
sendWebSocketMessage(param.getUserPageId(), socketDataMsg); FormalTestManager.unknownError = false;
//开始设备通讯检测(发送设备初始化)
Map<String, List<PreDetection>> map = new HashMap<>(1); //重新下发脚本
map.put("deviceList", FormalTestManager.devList); String type = FormalTestManager.currentIssue.getType();
String jsonString = JSON.toJSONString(map); if (ResultUnitEnum.P.getCode().equals(type)) {
socketMsg.setRequestId(SourceOperateCodeEnum.YJC_SBTXJY.getValue()); FormalTestManager.currentIssue.setType(ResultUnitEnum.V_ABSOLUTELY.getCode());
socketMsg.setOperateCode(SourceOperateCodeEnum.DEV_INIT_GATHER_01.getValue()); }
socketMsg.setData(jsonString); socketMsg.setOperateCode(SourceOperateCodeEnum.OPER_GATHER.getValue());
String json = JSON.toJSONString(socketMsg); socketMsg.setData(JSON.toJSONString(FormalTestManager.currentIssue));
// 使用智能发送工具类,自动管理设备连接 socketMsg.setRequestId(SourceOperateCodeEnum.FORMAL_REAL.getValue() + CnSocketUtil.STEP_TAG + type);
socketManager.smartSendToDevice(param, json); SocketManager.sendMsg(param.getUserPageId() + CnSocketUtil.SOURCE_TAG, JSON.toJSONString(socketMsg));
} else {
//todo 前端推送收到的消息暂未处理好
sendWebSocketMessage(param.getUserPageId(), socketDataMsg);
//开始设备通讯检测(发送设备初始化)
Map<String, List<PreDetection>> map = new HashMap<>(1);
map.put("deviceList", FormalTestManager.devList);
String jsonString = JSON.toJSONString(map);
socketMsg.setRequestId(SourceOperateCodeEnum.YJC_SBTXJY.getValue());
socketMsg.setOperateCode(SourceOperateCodeEnum.DEV_INIT_GATHER_01.getValue());
socketMsg.setData(jsonString);
String json = JSON.toJSONString(socketMsg);
// 使用智能发送工具类,自动管理设备连接
socketManager.smartSendToDevice(param, json);
}
break; break;
case UNPROCESSED_BUSINESS: case UNPROCESSED_BUSINESS:
sendWebSocketMessage(param.getUserPageId(), socketDataMsg); sendWebSocketMessage(param.getUserPageId(), socketDataMsg);
@@ -570,6 +592,10 @@ public class SocketSourceResponseService {
case UNPROCESSED_BUSINESS: case UNPROCESSED_BUSINESS:
sendWebSocketMessage(param.getUserPageId(), socketDataMsg); sendWebSocketMessage(param.getUserPageId(), socketDataMsg);
break; break;
case UNKNOWN_ERROR: //-1源未知异常
CnSocketUtil.quitSendSource(param);
FormalTestManager.unknownError = true;
break;
default: default:
sendErrorAndQuit(param, socketDataMsg, dictDataEnumByCode); sendErrorAndQuit(param, socketDataMsg, dictDataEnumByCode);
break; break;
@@ -601,9 +627,23 @@ public class SocketSourceResponseService {
SourceResponseCodeEnum dictDataEnumByCode = SourceResponseCodeEnum.getDictDataEnumByCode(socketDataMsg.getCode()); SourceResponseCodeEnum dictDataEnumByCode = SourceResponseCodeEnum.getDictDataEnumByCode(socketDataMsg.getCode());
switch (Objects.requireNonNull(dictDataEnumByCode)) { switch (Objects.requireNonNull(dictDataEnumByCode)) {
case SUCCESS: case SUCCESS:
//通讯校验成功 if (FormalTestManager.unknownError) {
SocketManager.removeUser(param.getUserPageId() + CnSocketUtil.SOURCE_TAG); //获取源初始化参数
sendWebSocketMessage(param.getUserPageId(), socketDataMsg); AdPlanSource planSource = adPlanSourceService.getOne(new LambdaQueryWrapper<AdPlanSource>().eq(AdPlanSource::getPlanId, param.getPlanId()));
SourceInitialize sourceParam = pqSourceService.getSourceInitializeParam(planSource.getSourceId());
if (ObjectUtil.isNotNull(sourceParam)) {
SocketMsg<String> socketMsg1 = new SocketMsg<>();
socketMsg1.setRequestId(SourceOperateCodeEnum.YJC_YTXJY.getValue());
socketMsg1.setOperateCode(SourceOperateCodeEnum.INIT_GATHER.getValue());
socketMsg1.setData(JSON.toJSONString(sourceParam));
//使用智能发送工具类自动管理与源控程序的socket连接
socketManager.smartSendToSource(param, JSON.toJSONString(socketMsg1));
}
} else {
//通讯校验成功
SocketManager.removeUser(param.getUserPageId() + CnSocketUtil.SOURCE_TAG);
sendWebSocketMessage(param.getUserPageId(), socketDataMsg);
}
break; break;
case UNPROCESSED_BUSINESS: case UNPROCESSED_BUSINESS:
break; break;
@@ -653,6 +693,8 @@ public class SocketSourceResponseService {
// 同步更新系数管理器中的设备列表 // 同步更新系数管理器中的设备列表
XiNumberManager.xiDevList = devList; XiNumberManager.xiDevList = devList;
FormalTestManager.unknownError = false;
FormalTestManager.currentIssue = null;
} }
} }

View File

@@ -0,0 +1,41 @@
package com.njcn.gather.detection.pojo.dto;
import com.fasterxml.jackson.annotation.JsonAlias;
import lombok.Data;
/**
* @author caozehui
* @data 2026-04-08
*/
@Data
public class FreqConverterRespDTO {
/**
* 请求编号
*/
@JsonAlias({"RequestId"})
private String requestId;
/**
* 是否成功
*/
@JsonAlias({"Success"})
private Boolean success;
/**
* 状态码
*/
@JsonAlias({"Code"})
private Integer code;
/**
* 消息
*/
@JsonAlias({"Message"})
private String message;
/**
* 数据
*/
@JsonAlias({"Data"})
private Object data;
}

View File

@@ -50,7 +50,10 @@ public enum DetectionCodeEnum {
DELTA("Delta", "角型接线"), DELTA("Delta", "角型接线"),
REAL_PREFIX("real$", "实时数据前缀"), REAL_PREFIX("real$", "实时数据前缀"),
AVG_PREFIX("avg$", "统计数据前缀"); CP_95_PREFIX("cp95$", "分钟统计-CP95值数据前缀"),
MAX_PREFIX("max$", "分钟统计-最大值数据前缀"),
MIN_PREFIX("min$", "分钟统计-最小值数据前缀"),
AVG_PREFIX("avg$", "分钟统计-平均值数据前缀");
private final String code; private final String code;
private final String message; private final String message;

View File

@@ -19,7 +19,8 @@ public enum DetectionResponseEnum {
SCRIPT_CHECK_DATA_NOT_EXIST("A020040","测试脚本项暂无配置" ), SCRIPT_CHECK_DATA_NOT_EXIST("A020040","测试脚本项暂无配置" ),
EXCEED_MAX_TIME("A020041","检测次数超出最大限制!" ); EXCEED_MAX_TIME("A020041","检测次数超出最大限制!" ),
SOCKET_CONNECT_TIMEOUT("A020042","socket连接超时");
private final String code; private final String code;
private final String message; private final String message;

View File

@@ -52,7 +52,7 @@ public enum SourceOperateCodeEnum {
YJC_MXYZXJY("yjc_mxyzxjy", "模型一致性校验"), YJC_MXYZXJY("yjc_mxyzxjy", "模型一致性校验"),
FORMAL_REAL("formal_real","正式检测"), FORMAL_REAL("formal_real","正式检测"),
RECORD_WAVE_STEP1("record_wave_step1","启动录波_step1"), RECORD_WAVE_STEP1("record_wave_step1","启动录波_step1"),
RECORD_WAVE_STEP2("record_wave_step2","启动录波_step2"), // RECORD_WAVE_STEP2("record_wave_step2","启动录波_step2"),
// SIMULATE_REAL("simulate_real","模拟检测"), // SIMULATE_REAL("simulate_real","模拟检测"),
Coefficient_Check("Coefficient_Check","系数校验"), Coefficient_Check("Coefficient_Check","系数校验"),
QUITE("quit","关闭设备通讯初始化"), QUITE("quit","关闭设备通讯初始化"),
@@ -67,8 +67,7 @@ public enum SourceOperateCodeEnum {
SERVER_ERROR("server_error","服务端主动关闭连接,请稍后再试"), SERVER_ERROR("server_error","服务端主动关闭连接,请稍后再试"),
DEVICE_ERROR("device_error","设备主动关闭连接,请稍后再试"), DEVICE_ERROR("device_error","设备主动关闭连接,请稍后再试"),
FLICKER_DATA_CHECK("flicker_data_check","闪变数据校验"),
/** /**
@@ -98,12 +97,17 @@ public enum SourceOperateCodeEnum {
small_comp_start("small_comp_start","小电压校准开始"), small_comp_start("small_comp_start","小电压校准开始"),
small_comp_end("small_comp_end","小电压校准结束"), small_comp_end("small_comp_end","小电压校准结束"),
/** /**
* ftp文件传送指令 * ftp文件传送指令
*/ */
FTP_SEND_01("FTP_SEND$01", "发送文件"), FTP_SEND_01("FTP_SEND$01", "发送文件"),
RDRE$01("RDRE$01", "启动录波"); RDRE$01("RDRE$01", "启动录波"),
CMD_PING("ping", "检查 TCP 服务是否在线"),
CMD_INIT_SERIAL("initSerial", "初始化并打开串口连接"),
CMD_GET_SERIAL_CONFIG("getSerialConfig", "获取当前串口配置"),
CMD_GET_DEVICE_STATUS("getDeviceStatus", "读取变频器运行状态"),
CMD_CLOSE_SERIAL("closeSerial", "关闭串口"),;
private final String value; private final String value;
private final String msg; private final String msg;

View File

@@ -42,7 +42,9 @@ public enum SourceResponseCodeEnum {
FAIL(25002,"失败"), FAIL(25002,"失败"),
ALL_FAIL(25003,"校验失败"), ALL_FAIL(25003,"校验失败"),
RECEIVE_DATA_TIME_OUT(25004,"接收数据超时"), RECEIVE_DATA_TIME_OUT(25004,"接收数据超时"),
REAL_DATA_CHECK_FAIL(25005,"实时数据校验失败") REAL_DATA_CHECK_FAIL(25005,"实时数据不符合"),
STATISTICS_DATA_CHECK_FAIL(25006,"统计数据不符合"),
FLICKER_DATA_START(25007,"开始接收闪变数据")

View File

@@ -43,4 +43,9 @@ public class ContrastDetectionParam {
private List<Boolean> testItemList; private List<Boolean> testItemList;
private String userId; private String userId;
/**
* 是否进行相许校验
*/
private Integer phaseCheck;
} }

View File

@@ -53,4 +53,9 @@ public class DetectionData {
* 误差体系详情ID(比对式使用) * 误差体系详情ID(比对式使用)
*/ */
private String errorDtlId; private String errorDtlId;
/**
* 有效组数
*/
private Integer validGroupNum;
} }

View File

@@ -4,8 +4,6 @@ import com.njcn.gather.detection.pojo.param.ContrastDetectionParam;
import com.njcn.gather.detection.pojo.param.PreDetectionParam; import com.njcn.gather.detection.pojo.param.PreDetectionParam;
import com.njcn.gather.detection.pojo.param.SimulateDetectionParam; import com.njcn.gather.detection.pojo.param.SimulateDetectionParam;
import java.util.Map;
/** /**
* @author wr * @author wr
@@ -54,7 +52,6 @@ public interface PreDetectionService {
void closeTestSimulate(SimulateDetectionParam param); void closeTestSimulate(SimulateDetectionParam param);
/** /**
*
* @param param * @param param
*/ */
void startContrastTest(ContrastDetectionParam param); void startContrastTest(ContrastDetectionParam param);
@@ -63,4 +60,12 @@ public interface PreDetectionService {
* 导出实时数据对齐过程中的数据 * 导出实时数据对齐过程中的数据
*/ */
void exportAlignData(); void exportAlignData();
boolean getCanCoefficient();
void startCoefficient();
void startFreqConverter(String userId, String converterId, String monitorId,Boolean reset);
void stopFreqConverter(String converterId, String monitorId);
} }

View File

@@ -6,9 +6,7 @@ import cn.hutool.core.util.ObjectUtil;
import com.alibaba.fastjson.JSON; import com.alibaba.fastjson.JSON;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.njcn.common.pojo.exception.BusinessException; import com.njcn.common.pojo.exception.BusinessException;
import com.njcn.gather.detection.handler.SocketContrastResponseService; import com.njcn.gather.detection.handler.*;
import com.njcn.gather.detection.handler.SocketDevResponseService;
import com.njcn.gather.detection.handler.SocketSourceResponseService;
import com.njcn.gather.detection.pojo.constant.DetectionCommunicateConstant; import com.njcn.gather.detection.pojo.constant.DetectionCommunicateConstant;
import com.njcn.gather.detection.pojo.enums.DetectionResponseEnum; import com.njcn.gather.detection.pojo.enums.DetectionResponseEnum;
import com.njcn.gather.detection.pojo.enums.SourceOperateCodeEnum; import com.njcn.gather.detection.pojo.enums.SourceOperateCodeEnum;
@@ -21,10 +19,13 @@ import com.njcn.gather.detection.util.business.DetectionCommunicateUtil;
import com.njcn.gather.detection.util.socket.CnSocketUtil; import com.njcn.gather.detection.util.socket.CnSocketUtil;
import com.njcn.gather.detection.util.socket.FormalTestManager; import com.njcn.gather.detection.util.socket.FormalTestManager;
import com.njcn.gather.detection.util.socket.SocketManager; import com.njcn.gather.detection.util.socket.SocketManager;
import com.njcn.gather.detection.util.socket.XiNumberManager;
import com.njcn.gather.detection.util.socket.cilent.NettyContrastClientHandler;
import com.njcn.gather.detection.util.socket.websocket.WebServiceManager; import com.njcn.gather.detection.util.socket.websocket.WebServiceManager;
import com.njcn.gather.device.pojo.po.PqDev; import com.njcn.gather.device.pojo.po.PqDev;
import com.njcn.gather.device.pojo.vo.PreDetection; import com.njcn.gather.device.pojo.vo.PreDetection;
import com.njcn.gather.device.service.IPqDevService; import com.njcn.gather.device.service.IPqDevService;
import com.njcn.gather.plan.pojo.enums.DataSourceEnum;
import com.njcn.gather.plan.pojo.po.AdPlan; import com.njcn.gather.plan.pojo.po.AdPlan;
import com.njcn.gather.plan.pojo.po.AdPlanSource; import com.njcn.gather.plan.pojo.po.AdPlanSource;
import com.njcn.gather.plan.service.IAdPlanService; import com.njcn.gather.plan.service.IAdPlanService;
@@ -37,6 +38,7 @@ import com.njcn.gather.script.service.IPqScriptCheckDataService;
import com.njcn.gather.script.service.IPqScriptDtlsService; import com.njcn.gather.script.service.IPqScriptDtlsService;
import com.njcn.gather.source.pojo.po.SourceInitialize; import com.njcn.gather.source.pojo.po.SourceInitialize;
import com.njcn.gather.source.service.IPqSourceService; import com.njcn.gather.source.service.IPqSourceService;
import com.njcn.gather.system.cfg.service.ISysTestConfigService;
import com.njcn.gather.system.dictionary.pojo.enums.DictDataEnum; import com.njcn.gather.system.dictionary.pojo.enums.DictDataEnum;
import com.njcn.gather.system.dictionary.service.IDictDataService; import com.njcn.gather.system.dictionary.service.IDictDataService;
import com.njcn.web.utils.HttpServletUtil; import com.njcn.web.utils.HttpServletUtil;
@@ -72,10 +74,14 @@ public class PreDetectionServiceImpl implements PreDetectionService {
private final SocketDevResponseService socketDevResponseService; private final SocketDevResponseService socketDevResponseService;
private final SocketSourceResponseService socketSourceResponseService; private final SocketSourceResponseService socketSourceResponseService;
private final SocketContrastResponseService socketContrastResponseService; private final SocketContrastResponseService socketContrastResponseService;
private final SocketFreqConverterService socketFreqConverterService;
private final SocketFreqConverterDevService socketFreqConverterDevService;
private final IPqScriptCheckDataService iPqScriptCheckDataService; private final IPqScriptCheckDataService iPqScriptCheckDataService;
private final SocketManager socketManager; private final SocketManager socketManager;
private final ISysTestConfigService sysTestConfigService;
@Value("${log.homeDir}")
@Value("${report.reportDir}")
private String alignDataFilePath; private String alignDataFilePath;
@@ -336,7 +342,7 @@ public class PreDetectionServiceImpl implements PreDetectionService {
@Override @Override
public void exportAlignData() { public void exportAlignData() {
String fileName = "实时数据.xlsx"; String fileName = "对齐数据.xlsx";
HttpServletResponse response = HttpServletUtil.getResponse(); HttpServletResponse response = HttpServletUtil.getResponse();
response.reset(); response.reset();
response.setHeader("Content-Disposition", "attachment; filename=\"" + fileName + "\""); response.setHeader("Content-Disposition", "attachment; filename=\"" + fileName + "\"");
@@ -357,6 +363,68 @@ public class PreDetectionServiceImpl implements PreDetectionService {
} }
} }
@Override
public boolean getCanCoefficient() {
return CollUtil.isNotEmpty(DetectionServiceImpl.vAndIResultMap);
}
@Override
public void startCoefficient() {
if (CollUtil.isNotEmpty(DetectionServiceImpl.vAndIResultMap)) {
// 是否进行系数校准
//万一录波了这里就进不来了但是在误差处理的时候DetectionServiceImpl.vAndIResultMap该集合会保留不符合的结果
XiNumberManager.xiDevList.clear();
XiNumberManager.devXiNumDataMap.clear();
List<String> stdIpList = DetectionServiceImpl.vAndIResultMap.values().stream().map(DetectionServiceImpl.VAndIResult::getIp).collect(Collectors.toList());
XiNumberManager.xiDevList = FormalTestManager.standardDevList.stream().filter(d -> stdIpList.contains(d.getDevIP())).collect(Collectors.toList());
// 逐一设备向通讯模块要原始系数
SocketMsg<String> socketMsg = new SocketMsg<>();
socketMsg.setRequestId(SourceOperateCodeEnum.Coefficient_Check.getValue());
socketMsg.setOperateCode(SourceOperateCodeEnum.DATA_CHNFACTOR$01.getValue());
PreDetection preDetection = XiNumberManager.xiDevList.get(0);
Map<String, Object> map = new HashMap<>();
map.put("devIP", preDetection.getDevIP());
map.put("chnNum", preDetection.getDevChns());
socketMsg.setData(JSON.toJSONString(map));
SocketManager.sendMsg(NettyContrastClientHandler.param.getUserPageId() + CnSocketUtil.CONTRAST_DEV_TAG, JSON.toJSONString(socketMsg));
FormalTestManager.currentStep = SourceOperateCodeEnum.Coefficient_Check;
}
}
@Override
public void startFreqConverter(String userId, String converterId, String monitorId, Boolean reset) {
String freqConverterTag = userId + CnSocketUtil.FREQ_CONVERTER_TAG;
String devTag = userId + CnSocketUtil.DEV_TAG;
// socketFreqConverterService.init(userId, converterId, monitorId);
socketFreqConverterService.connectSocket(freqConverterTag);
socketFreqConverterDevService.connectSocket(devTag);
long startTime = System.currentTimeMillis();
long timeout = 3000; // 3秒超时时间
while (true) {
if (SocketManager.isChannelActive(freqConverterTag) && SocketManager.isChannelActive(devTag)) {
// if (SocketManager.isChannelActive(devTag)) {
socketFreqConverterService.connectionFreqConverter(userId, freqConverterTag, converterId, monitorId, reset);
socketFreqConverterDevService.connectionDev(userId, devTag, converterId, monitorId,reset);
break;
}
// 检查是否超时
if (System.currentTimeMillis() - startTime > timeout) {
throw new BusinessException(DetectionResponseEnum.SOCKET_CONNECT_TIMEOUT);
}
}
}
@Override
public void stopFreqConverter(String converterTag, String devTag) {
socketFreqConverterService.stopTest(converterTag, devTag);
socketFreqConverterDevService.stopTest(converterTag, devTag);
}
/** /**
* 比对式-与通信模块进行连接 * 比对式-与通信模块进行连接
* *
@@ -385,7 +453,15 @@ public class PreDetectionServiceImpl implements PreDetectionService {
String jsonString = JSON.toJSONString(map); String jsonString = JSON.toJSONString(map);
SocketMsg<String> socketMsg = new SocketMsg<>(); SocketMsg<String> socketMsg = new SocketMsg<>();
socketMsg.setRequestId(SourceOperateCodeEnum.YJC_SBTXJY.getValue()); socketMsg.setRequestId(SourceOperateCodeEnum.YJC_SBTXJY.getValue());
socketMsg.setOperateCode(SourceOperateCodeEnum.DEV_INIT_GATHER_02.getValue()); AdPlan currentTestPlan = FormalTestManager.currentTestPlan;
String[] datasource = currentTestPlan.getDatasourceId().split(",");
if (Arrays.stream(datasource).anyMatch(x -> x.equals(DataSourceEnum.REAL_DATA.getValue()))) {
socketMsg.setOperateCode(SourceOperateCodeEnum.DEV_INIT_GATHER_02.getValue());
} else if (Arrays.stream(datasource).anyMatch(x -> x.equals(DataSourceEnum.MINUTE_STATISTICS_AVG.getValue()) || x.equals(DataSourceEnum.MINUTE_STATISTICS_MAX.getValue()) || x.equals(DataSourceEnum.MINUTE_STATISTICS_MIN.getValue()) || x.equals(DataSourceEnum.MINUTE_STATISTICS_CP95.getValue()))) {
socketMsg.setOperateCode(SourceOperateCodeEnum.DEV_INIT_GATHER_01.getValue());
} else if (Arrays.stream(datasource).anyMatch(x -> x.equals(DataSourceEnum.WAVE_DATA.getValue()))) {
socketMsg.setOperateCode(SourceOperateCodeEnum.DEV_INIT_GATHER_02.getValue());
}
socketMsg.setData(jsonString); socketMsg.setData(jsonString);
PreDetectionParam preDetectionParam = new PreDetectionParam(); PreDetectionParam preDetectionParam = new PreDetectionParam();
preDetectionParam.setUserPageId(param.getLoginName()); preDetectionParam.setUserPageId(param.getLoginName());
@@ -407,4 +483,4 @@ public class PreDetectionServiceImpl implements PreDetectionService {
} }
} }
} }

View File

@@ -1,6 +1,7 @@
package com.njcn.gather.detection.util; package com.njcn.gather.detection.util;
import cn.hutool.core.collection.CollUtil; import cn.hutool.core.collection.CollUtil;
import cn.hutool.core.util.NumberUtil;
import cn.hutool.core.util.ObjectUtil; import cn.hutool.core.util.ObjectUtil;
import com.njcn.gather.detection.pojo.po.DevData; import com.njcn.gather.detection.pojo.po.DevData;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
@@ -51,6 +52,11 @@ public class DetectionUtil {
*/ */
private static final long MILLIS_TO_SECONDS = 1000L; private static final long MILLIS_TO_SECONDS = 1000L;
/**
* 毫秒转分钟的转换因子
*/
private static final long MILLIS_TO_MINUTES = 1000 * 60L;
/** /**
* 时间对齐判断的容差毫秒数 * 时间对齐判断的容差毫秒数
* 当两个时间戳差值小于此值时,认为时间是对齐的 * 当两个时间戳差值小于此值时,认为时间是对齐的
@@ -123,24 +129,33 @@ public class DetectionUtil {
* @param standardDevData 标准设备数据,包含时间戳信息 * @param standardDevData 标准设备数据,包含时间戳信息
* @return true表示数据时间对齐false表示不对齐 * @return true表示数据时间对齐false表示不对齐
*/ */
public static boolean isAlignData(DevData devData, DevData standardDevData) { public static boolean isAlignData(DevData devData, DevData standardDevData, int type) {
if (ObjectUtil.isNotNull(devData) && ObjectUtil.isNotNull(standardDevData)) { if (ObjectUtil.isNotNull(devData) && ObjectUtil.isNotNull(standardDevData)) {
// 获取两个设备数据的时间戳(毫秒) // 获取两个设备数据的时间戳(毫秒)
long devMillis = getMillis(devData.getTime()); long devMillis = getMillis(devData.getTime());
long standardMillis = getMillis(standardDevData.getTime()); long standardMillis = getMillis(standardDevData.getTime());
if (type == 1) { // 实时数据
// 方式1将时间戳转换为秒级精度进行比较处理跨秒边界情况
BigDecimal devSeconds = BigDecimal.valueOf(devMillis).divide(BigDecimal.valueOf(MILLIS_TO_SECONDS), 0, RoundingMode.HALF_UP);
BigDecimal standardSeconds = BigDecimal.valueOf(standardMillis).divide(BigDecimal.valueOf(MILLIS_TO_SECONDS), 0, RoundingMode.HALF_UP);
if (devSeconds.compareTo(standardSeconds) == 0) {
return true;
}
// 方式1将时间戳转换为秒级精度进行比较处理跨秒边界情况 // 方式2毫秒级时间差小于容差值也认为是对齐的处理精确对齐
BigDecimal devSeconds = BigDecimal.valueOf(devMillis).divide(BigDecimal.valueOf(MILLIS_TO_SECONDS), 0, RoundingMode.HALF_UP); if (Math.abs(devMillis - standardMillis) < TIME_ALIGNMENT_TOLERANCE_MS) {
BigDecimal standardSeconds = BigDecimal.valueOf(standardMillis).divide(BigDecimal.valueOf(MILLIS_TO_SECONDS), 0, RoundingMode.HALF_UP); return true;
if (devSeconds.compareTo(standardSeconds) == 0) { }
return true;
} }
if (type == 2) { //统计数据
// 方式2毫秒级时间差小于容差值也认为是对齐的处理精确对齐 BigDecimal devSeconds = BigDecimal.valueOf(devMillis).divide(BigDecimal.valueOf(MILLIS_TO_MINUTES), 0, RoundingMode.HALF_UP);
if (Math.abs(devMillis - standardMillis) < TIME_ALIGNMENT_TOLERANCE_MS) { BigDecimal standardSeconds = BigDecimal.valueOf(standardMillis).divide(BigDecimal.valueOf(MILLIS_TO_MINUTES), 0, RoundingMode.HALF_UP);
return true; if (devSeconds.compareTo(standardSeconds) == 0) {
return true;
}
} }
} }
return false; return false;
} }
@@ -407,7 +422,7 @@ public class DetectionUtil {
} }
return newList; return newList;
} }
return null; return Collections.emptyList();
} }
/** /**
@@ -423,4 +438,94 @@ public class DetectionUtil {
System.out.println(" 文件大小: " + file.length() + " bytes"); System.out.println(" 文件大小: " + file.length() + " bytes");
} }
} }
/**
* 获取皮尔逊系数
*
* <p>相关系数绝对值|r|所代表的相关强度</p>
* <ul>
* <li>0.8 - 1.0:极强相关</li>
* <li>0.6 - 0.8:强相关</li>
* <li>0.4 - 0.6:中等程度相关</li>
* <li>0.2 - 0.4:弱相关</li>
* <li>0.0 - 0.2:极弱相关或无相关</li>
* </ul>
*
* @return
*/
public static BigDecimal getPearsonCorrelationCoefficient(List<Double> x, List<Double> y) {
Double xAvg = x.parallelStream().mapToDouble(Double::doubleValue).average().orElse(0.0);
Double yAvg = y.parallelStream().mapToDouble(Double::doubleValue).average().orElse(0.0);
// 协方差求和
BigDecimal covarianceSum = BigDecimal.ZERO;
// x的方差
BigDecimal varianceX = BigDecimal.ZERO;
// y的方差
BigDecimal varianceY = BigDecimal.ZERO;
for (int i = 0; i < x.size(); i++) {
covarianceSum = covarianceSum.add(BigDecimal.valueOf((x.get(i) - xAvg) * (y.get(i) - yAvg)));
varianceX = varianceX.add(BigDecimal.valueOf(Math.pow(x.get(i) - xAvg, 2)));
varianceY = varianceY.add(BigDecimal.valueOf(Math.pow(y.get(i) - yAvg, 2)));
}
BigDecimal fm = BigDecimal.valueOf(Math.sqrt(varianceX.doubleValue()) * Math.sqrt(varianceY.doubleValue()));
if (NumberUtil.equals(fm.doubleValue(), BigDecimal.ZERO.doubleValue())) {
if (NumberUtil.equals(covarianceSum, BigDecimal.ZERO)) {
return BigDecimal.valueOf(1.0);
} else {
return BigDecimal.ZERO;
}
} else {
return BigDecimal.valueOf(Math.abs(covarianceSum.divide(fm, 8, BigDecimal.ROUND_HALF_UP).doubleValue())); //相关系数
}
}
/**
* 比值一致性获取CV系数
*
* <p>一致性等级</p>
* <ul>
* <li>CV<1%:非常优秀</li>
* <li>1%≤CV5%:优良</li>
* <li>5%≤CV10%:良好</li>
* <li>10%≤CV20%:中等</li>
* <li>20%≤CV较差</li>
* </ul>
*
* @param x
* @param y
* @return
*/
public static BigDecimal getCvCoefficient(List<Double> x, List<Double> y) {
List<BigDecimal> ratioList = new ArrayList<>();
for (int i = 0; i < x.size(); i++) {
ratioList.add(BigDecimal.valueOf(x.get(i) / y.get(i)));
}
BigDecimal avg = BigDecimal.valueOf(ratioList.parallelStream().mapToDouble(BigDecimal::doubleValue).average().orElse(0.0));
BigDecimal variance = BigDecimal.valueOf(
ratioList.parallelStream().mapToDouble(BigDecimal::doubleValue)
.map(v -> Math.pow(v - avg.doubleValue(), 2)).average().orElse(0.0));
return BigDecimal.valueOf(Math.sqrt(variance.doubleValue()) / avg.doubleValue());
}
/**
* 计算新的系数
*
* @param devList
* @param stdDevList
* @return
*/
public static Double getNewXi(List<Double> devList, List<Double> stdDevList) {
ArrayList<Double> xiList = new ArrayList<>();
for (int i = 0; i < stdDevList.size(); i++) {
if (!stdDevList.get(i).equals(0)) {
BigDecimal bigDecimalDev = BigDecimal.valueOf(devList.get(i));
BigDecimal bigDecimalStdDev = BigDecimal.valueOf(stdDevList.get(i));
xiList.add((bigDecimalDev.subtract(bigDecimalStdDev)).divide(BigDecimal.valueOf(2.0).multiply(bigDecimalDev), 8, BigDecimal.ROUND_HALF_UP).doubleValue() + 1.0);
}
}
return xiList.stream().mapToDouble(Double::doubleValue).average().orElse(1.0);
}
} }

View File

@@ -14,6 +14,8 @@ import com.njcn.gather.detection.util.socket.websocket.WebServiceManager;
*/ */
public class CnSocketUtil { public class CnSocketUtil {
public final static String FREQ_CONVERTER_TAG="_FreqConverter";
public final static String DEV_TAG = "_Dev"; public final static String DEV_TAG = "_Dev";
public final static String CONTRAST_DEV_TAG = "_Contrast_Dev"; public final static String CONTRAST_DEV_TAG = "_Contrast_Dev";
@@ -58,7 +60,7 @@ public class CnSocketUtil {
* 比对式-退出检测 * 比对式-退出检测
*/ */
public static void contrastSendquit(String loginName, SourceOperateCodeEnum operateCode, boolean isRemoveSocket) { public static void contrastSendquit(String loginName, SourceOperateCodeEnum operateCode, boolean isRemoveSocket) {
System.out.println("比对式-发送关闭备通讯模块指令。。。。。。。。"); System.out.println("比对式-发送" + operateCode.getMsg() + "指令。。。。。。。。");
SocketMsg<String> socketMsg = new SocketMsg<>(); SocketMsg<String> socketMsg = new SocketMsg<>();
socketMsg.setRequestId(SourceOperateCodeEnum.QUITE.getValue()); socketMsg.setRequestId(SourceOperateCodeEnum.QUITE.getValue());
socketMsg.setOperateCode(operateCode.getValue()); socketMsg.setOperateCode(operateCode.getValue());

View File

@@ -7,15 +7,22 @@ import com.njcn.gather.detection.pojo.po.DevData;
import com.njcn.gather.detection.pojo.vo.DevLineTestResult; import com.njcn.gather.detection.pojo.vo.DevLineTestResult;
import com.njcn.gather.device.pojo.enums.PatternEnum; import com.njcn.gather.device.pojo.enums.PatternEnum;
import com.njcn.gather.device.pojo.vo.PreDetection; import com.njcn.gather.device.pojo.vo.PreDetection;
import com.njcn.gather.dip.pojo.po.PqDipData;
import com.njcn.gather.plan.pojo.enums.DataSourceEnum;
import com.njcn.gather.plan.pojo.po.AdPlan; import com.njcn.gather.plan.pojo.po.AdPlan;
import com.njcn.gather.plan.pojo.po.AdPlanTestConfig; import com.njcn.gather.plan.pojo.po.AdPlanTestConfig;
import com.njcn.gather.script.pojo.po.SourceIssue;
import com.njcn.gather.system.dictionary.pojo.enums.DictDataEnum; import com.njcn.gather.system.dictionary.pojo.enums.DictDataEnum;
import lombok.Data;
import java.time.LocalDateTime;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.HashMap; import java.util.HashMap;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
/** /**
* @Author: cdf * @Author: cdf
@@ -27,6 +34,20 @@ public class FormalTestManager {
// 当前步骤 // 当前步骤
public static SourceOperateCodeEnum currentStep; public static SourceOperateCodeEnum currentStep;
public static SourceOperateCodeEnum freqConverterStep;
public static SourceOperateCodeEnum freqConverterDevStep;
/**
* 变频器存放数据的表后缀
*/
public static Integer freqConverterTableSuffix;
public static String currentFreqConverterId;
/**
* 待采集后续变频器状态的Dip任务
*/
public static Map<String, PendingDipTask> pendingDipTaskMap = new ConcurrentHashMap<>();
/** /**
* key:设备ip,value:当前设备下面的监测点ID(ip_通道号) * key:设备ip,value:当前设备下面的监测点ID(ip_通道号)
*/ */
@@ -87,6 +108,17 @@ public class FormalTestManager {
*/ */
public static AdPlan currentTestPlan; public static AdPlan currentTestPlan;
/**
* 非录波数据源枚举
*/
public static DataSourceEnum nonWaveDataSourceEnum;
/**
* 统计数据最大超时时间
*/
public static Long maxTime;
public static AdPlanTestConfig curretntTestPlanConfig; public static AdPlanTestConfig curretntTestPlanConfig;
/** /**
@@ -154,4 +186,66 @@ public class FormalTestManager {
*/ */
public static List<DevLineTestResult> preNumTestResultList = new ArrayList<>(); public static List<DevLineTestResult> preNumTestResultList = new ArrayList<>();
/**
* 定时器
*/
public static ScheduledExecutorService scheduler;
/**
* 定时器任务
*/
public static ScheduledFuture<?> scheduledFuture;
public static boolean isWaveCheck;
public static List<String> pstDataType;
public static boolean isPstData;
/**
* 是否在检测中
*/
public static boolean isTesting;
public static boolean realProtocol;
public static boolean statisticsProtocol;
public static boolean voltageProtocol;
public static boolean unknownError;
/**
* 当前下发的脚本
*/
public static SourceIssue currentIssue;
/**
* 是否进行相序校验
*/
public static boolean isXu;
/**
* 清理变频器耐受实验运行态数据
*/
public static void clearFreqConverterRuntimeState() {
currentFreqConverterId = null;
freqConverterTableSuffix = null;
pendingDipTaskMap.clear();
}
@Data
public static class PendingDipTask {
private PqDipData pqDipData;
private LocalDateTime targetEndTime;
private Integer originalTolerant;
public PendingDipTask(PqDipData pqDipData, LocalDateTime targetEndTime, Integer originalTolerant) {
this.pqDipData = pqDipData;
this.targetEndTime = targetEndTime;
this.originalTolerant = originalTolerant;
}
}
} }

View File

@@ -84,6 +84,15 @@ public class MsgUtil {
} }
} }
public static String getMonitorInfo(String monitorId, Map<String, String> devMap) {
if (StrUtil.isBlank(monitorId)) {
return "";
} else {
String[] split1 = monitorId.split("_");
return devMap.get(split1[0]) + CnSocketUtil.SPLIT_TAG + split1[1] + "\"";
}
}
/** /**
* 获取消息 * 获取消息
* *

View File

@@ -315,4 +315,4 @@ public class SocketManager {
} }

View File

@@ -193,7 +193,7 @@ public class HeartbeatHandler extends SimpleChannelInboundHandler<String> {
} else if (CnSocketUtil.SOURCE_TAG.equals(handlerType)) { } else if (CnSocketUtil.SOURCE_TAG.equals(handlerType)) {
CnSocketUtil.quitSendSource(param); CnSocketUtil.quitSendSource(param);
} else { } else {
if (FormalTestManager.currentStep == SourceOperateCodeEnum.RECORD_WAVE_STEP1 || FormalTestManager.currentStep == SourceOperateCodeEnum.RECORD_WAVE_STEP2) { if (FormalTestManager.currentStep == SourceOperateCodeEnum.RECORD_WAVE_STEP1) {
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_03, true); CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_03, true);
} else { } else {
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, true); CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, true);

View File

@@ -56,6 +56,109 @@ public class NettyClient {
*/ */
private static NettyClient instance; private static NettyClient instance;
/**
* 静态方法:智能连接变频器设备(兼容性包装)
*
* @param ip IP地址
* @param port 端口号
* @param ChannelId Channel唯一标识
* @param handler 变频器处理器
*/
public static void commonConnect(String ip, Integer port, String ChannelId,
SimpleChannelInboundHandler handler) {
if (instance != null) {
instance.executeCommonConnect(ip, port, ChannelId, handler);
} else {
log.error("NettyClient未初始化无法创建连接");
}
}
/**
* 执行变频器Socket连接建立流程
*
* @param ip 目标服务器IP地址
* @param port 目标服务器端口号
* @param ChannelId Channel唯一标识id
* @param handler 变频器业务处理器
*/
private static void executeCommonConnect(String ip, Integer port,
String ChannelId,
SimpleChannelInboundHandler handler) {
NioEventLoopGroup group = createEventLoopGroup();
try {
Bootstrap bootstrap = configureBootstrap(group);
ChannelInitializer<NioSocketChannel> initializer = createCommonChannelInitializer(ChannelId, handler);
bootstrap.handler(initializer);
ChannelFuture channelFuture = bootstrap.connect(ip, port).sync();
handleCommonConnectionResult(channelFuture, ChannelId, handler, group);
} catch (Exception e) {
handleCommonConnectionException(e, ChannelId, handler, group);
}
}
/**
* 创建通用通道初始化器
*
* @param channelId Channel唯一标识id
* @param handler 通用业务处理器
* @return ChannelInitializer 通道初始化器
*/
private static ChannelInitializer<NioSocketChannel> createCommonChannelInitializer(
String channelId, SimpleChannelInboundHandler handler) {
return new ChannelInitializer<NioSocketChannel>() {
@Override
protected void initChannel(NioSocketChannel ch) {
ch.pipeline()
.addLast(new LineBasedFrameDecoder(10240 * 2))
.addLast(new StringDecoder(CharsetUtil.UTF_8))
.addLast(new StringEncoder(CharsetUtil.UTF_8))
.addLast(new IdleStateHandler(60, 0, 0, TimeUnit.SECONDS))
.addLast(handler);
}
};
}
/**
* 处理通用连接结果
*
* @param channelFuture 连接Future对象
* @param channelId Channel唯一标识符
* @param handler 通用业务处理器
* @param group 事件循环组
*/
private static void handleCommonConnectionResult(ChannelFuture channelFuture,
String channelId,
SimpleChannelInboundHandler handler,
NioEventLoopGroup group) {
channelFuture.addListener((ChannelFutureListener) ch -> {
if (!ch.isSuccess()) {
log.error("连接Socket失败channelId={}", channelId);
group.shutdownGracefully();
} else {
log.info("连接Socket成功channel={}, channelId={}",
channelId, channelFuture.channel().id());
SocketManager.addGroup(channelId, group);
SocketManager.addUser(channelId, channelFuture.channel());
}
});
}
/**
* 处理通用连接异常
*
* @param e 异常对象
* @param channelId Channel唯一标识id
* @param handler 通用业务处理器
* @param group 事件循环组
*/
private static void handleCommonConnectionException(Exception e, String channelId,
SimpleChannelInboundHandler handler,
NioEventLoopGroup group) {
log.error("连接Socket服务端发生异常channelId={}, error={}", channelId, e.getMessage(), e);
group.shutdownGracefully();
}
@PostConstruct @PostConstruct
public void init() { public void init() {

View File

@@ -1,6 +1,6 @@
package com.njcn.gather.detection.util.socket.cilent; package com.njcn.gather.detection.util.socket.cilent;
import com.alibaba.fastjson.JSON; import cn.hutool.core.util.ObjectUtil;
import com.njcn.gather.detection.handler.SocketContrastResponseService; import com.njcn.gather.detection.handler.SocketContrastResponseService;
import com.njcn.gather.detection.pojo.enums.SourceOperateCodeEnum; import com.njcn.gather.detection.pojo.enums.SourceOperateCodeEnum;
import com.njcn.gather.detection.pojo.enums.SourceResponseCodeEnum; import com.njcn.gather.detection.pojo.enums.SourceResponseCodeEnum;
@@ -60,11 +60,14 @@ public class NettyContrastClientHandler extends SimpleChannelInboundHandler<Stri
socketContrastResponseService.deal(param, msg); socketContrastResponseService.deal(param, msg);
} catch (Exception e) { } catch (Exception e) {
log.error("处理服务端消息异常", e); log.error("处理服务端消息异常", e);
if (FormalTestManager.currentStep == SourceOperateCodeEnum.RECORD_WAVE_STEP1 || FormalTestManager.currentStep == SourceOperateCodeEnum.RECORD_WAVE_STEP2) { // if (FormalTestManager.currentStep == SourceOperateCodeEnum.RECORD_WAVE_STEP1) {
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_03, true); // CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_03, true);
} else { // } else {
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, true); // CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, true);
} // }
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_01, false);
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, false);
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_03, true);
} }
} }
@@ -85,18 +88,35 @@ public class NettyContrastClientHandler extends SimpleChannelInboundHandler<Stri
public void userEventTriggered(ChannelHandlerContext ctx, Object evt) { public void userEventTriggered(ChannelHandlerContext ctx, Object evt) {
if (evt instanceof IdleStateEvent) { //IdleState.在一段时间内没有收到任何消息时,会触发该事件 if (evt instanceof IdleStateEvent) { //IdleState.在一段时间内没有收到任何消息时,会触发该事件
if (((IdleStateEvent) evt).state() == IdleState.READER_IDLE) { if (((IdleStateEvent) evt).state() == IdleState.READER_IDLE) {
System.out.println(LocalDateTime.now() + "contrastClientHandler触发读超时函数**************************************"); if (FormalTestManager.isTesting) {
SocketManager.contrastClockMap.put(DataSourceEnum.REAL_DATA, SocketManager.contrastClockMap.get(DataSourceEnum.REAL_DATA) + 60L); System.out.println(LocalDateTime.now() + "contrastClientHandler触发读超时函数**************************************");
if (FormalTestManager.isRemoveSocket) { if (!FormalTestManager.isRemoveSocket && ObjectUtil.isNotNull(FormalTestManager.nonWaveDataSourceEnum)) {
//实时数据 long time = SocketManager.contrastClockMap.get(FormalTestManager.nonWaveDataSourceEnum) + 60L;
if (SocketManager.contrastClockMap.get(DataSourceEnum.REAL_DATA) >= 60) { SocketManager.contrastClockMap.put(FormalTestManager.nonWaveDataSourceEnum, time);
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, false);
System.out.println("超时处理-----》" + "实时数据已超时----------------关闭"); if (FormalTestManager.isPstData) {
timeoutSend(); if (time > 60 * 10) {
if (DataSourceEnum.REAL_DATA == FormalTestManager.nonWaveDataSourceEnum) {
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, true);
}
if (FormalTestManager.isWaveCheck) {
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_03, true);
}
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_01, true);
timeoutSend(SourceOperateCodeEnum.QUIT_INIT_01);
}
} else if (DataSourceEnum.REAL_DATA == FormalTestManager.nonWaveDataSourceEnum) {
if (time >= 60) {
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, true);
timeoutSend(SourceOperateCodeEnum.QUIT_INIT_02);
}
} else if (time >= FormalTestManager.maxTime) {
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_01, true);
timeoutSend(SourceOperateCodeEnum.QUIT_INIT_01);
}
} }
} }
} }
} }
} }
@@ -126,8 +146,9 @@ public class NettyContrastClientHandler extends SimpleChannelInboundHandler<Stri
System.out.println("Unknown exception caught: " + cause.getMessage()); System.out.println("Unknown exception caught: " + cause.getMessage());
WebServiceManager.sendDetectionErrorMessage(param.getUserPageId(), SourceOperateCodeEnum.DEVICE_ERROR); WebServiceManager.sendDetectionErrorMessage(param.getUserPageId(), SourceOperateCodeEnum.DEVICE_ERROR);
} }
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, true); CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_01, false);
// socketContrastResponseService.backCheckState(param); CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, false);
CnSocketUtil.contrastSendquit(param.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_03, true);
ctx.close(); ctx.close();
} }
@@ -135,13 +156,14 @@ public class NettyContrastClientHandler extends SimpleChannelInboundHandler<Stri
/** /**
* 接收数据超时处理 * 接收数据超时处理
*/ */
private void timeoutSend() { private void timeoutSend(SourceOperateCodeEnum sourceOperateCodeEnum) {
System.out.println("超时处理-----》" + "统计数据已超时----------------关闭");
// 向前端推送超时消息 // 向前端推送超时消息
SocketDataMsg webSend = new SocketDataMsg(); SocketDataMsg webSend = new SocketDataMsg();
webSend.setRequestId(sourceOperateCodeEnum.getValue());
webSend.setData(sourceOperateCodeEnum.getMsg() + SourceResponseCodeEnum.RECEIVE_DATA_TIME_OUT.getMessage());
webSend.setCode(SourceResponseCodeEnum.RECEIVE_DATA_TIME_OUT.getCode()); webSend.setCode(SourceResponseCodeEnum.RECEIVE_DATA_TIME_OUT.getCode());
WebServiceManager.sendMsg(param.getUserPageId(), MsgUtil.msgToWebData(webSend, FormalTestManager.devNameMapComm, 0)); WebServiceManager.sendMsg(param.getUserPageId(), MsgUtil.msgToWebData(webSend, FormalTestManager.devNameMapComm, 0));
WebServiceManager.sendMsg(param.getUserPageId(), JSON.toJSONString(webSend));
} }
} }

View File

@@ -336,7 +336,7 @@ public class NettyDevClientHandler extends SimpleChannelInboundHandler<String> {
devLineTestResult.setDeviceName(dev.getDevName()); devLineTestResult.setDeviceName(dev.getDevName());
Integer[] resultFlags = dev.getMonitorList().stream() Integer[] resultFlags = dev.getMonitorList().stream()
.map(monitor -> ResultEnum.NETWORK_TIMEOUT) .map(monitor -> ResultEnum.NETWORK_TIMEOUT.getValue())
.toArray(Integer[]::new); .toArray(Integer[]::new);
devLineTestResult.setChnResult(resultFlags); devLineTestResult.setChnResult(resultFlags);

View File

@@ -0,0 +1,147 @@
package com.njcn.gather.detection.util.socket.cilent;
import cn.hutool.core.util.StrUtil;
import com.njcn.gather.detection.handler.SocketFreqConverterService;
import com.njcn.gather.detection.util.socket.FormalTestManager;
import com.njcn.gather.detection.util.socket.SocketManager;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;
import io.netty.handler.timeout.IdleState;
import io.netty.handler.timeout.IdleStateEvent;
import lombok.extern.slf4j.Slf4j;
/**
* 变频器Netty客户端处理器
*/
@Slf4j
public class NettyFreqConverterClientHandler extends SimpleChannelInboundHandler<String> {
/**
* 变频器Channel唯一标识符
*/
private final String converterChannelTag;
/**
* 变频器Socket响应服务
*/
private final SocketFreqConverterService socketFreqConverterService;
/**
* 重连次数
*/
private int reconnectAttempts = 0;
/**
* 最大重连次数
*/
private static final int MAX_RECONNECT_ATTEMPTS = 3;
/**
* 重连间隔(毫秒)
*/
private static final long RECONNECT_INTERVAL_MS = 5000;
/**
* 是否正在重连
*/
private volatile boolean isReconnecting = false;
/**
* 构造方法
*
* @param converterChannelTag 变频器Chanel唯一标识符
* @param socketFreqConverterService 变频器Socket响应服务
*/
public NettyFreqConverterClientHandler(String converterChannelTag, SocketFreqConverterService socketFreqConverterService) {
this.converterChannelTag = converterChannelTag;
this.socketFreqConverterService = socketFreqConverterService;
}
@Override
public void channelActive(ChannelHandlerContext ctx) throws Exception {
log.info("变频器连接已建立converterChannelTag={}, channelId={}", converterChannelTag, ctx.channel().id());
// 注册Channel到SocketManager
SocketManager.addUser(converterChannelTag, ctx.channel());
if (reconnectAttempts > 0) {
log.info("变频器重连成功converterChannelTag={}, 重连次数={}", converterChannelTag, reconnectAttempts);
reconnectAttempts = 0;
isReconnecting = false;
socketFreqConverterService.onReconnectSuccess(converterChannelTag);
}
super.channelActive(ctx);
}
@Override
protected void channelRead0(ChannelHandlerContext ctx, String msg) throws Exception {
if (StrUtil.isBlank(msg)) {
log.debug("收到空消息忽略converterChannelTag={}", converterChannelTag);
return;
}
log.info("收到变频器消息converterChannelTag={}, msg={}", converterChannelTag, msg);
// 处理状态数据
socketFreqConverterService.handleRead(converterChannelTag, msg);
}
@Override
public void channelInactive(ChannelHandlerContext ctx) throws Exception {
log.warn("变频器连接已断开converterChannelTag={}", converterChannelTag);
socketFreqConverterService.cleanup(converterChannelTag);
if (!isReconnecting && reconnectAttempts < MAX_RECONNECT_ATTEMPTS && !FormalTestManager.isRemoveSocket) {
attemptReconnect(ctx);
} else if (reconnectAttempts >= MAX_RECONNECT_ATTEMPTS) {
log.error("变频器重连失败,已达到最大重连次数{}次converterChannelTag={}", MAX_RECONNECT_ATTEMPTS, converterChannelTag);
}
super.channelInactive(ctx);
}
@Override
public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
log.error("变频器连接发生异常converterChannelTag={}, error={}", converterChannelTag, cause.getMessage(), cause);
ctx.close();
}
@Override
public void userEventTriggered(ChannelHandlerContext ctx, Object evt) throws Exception {
if (evt instanceof IdleStateEvent) {
IdleStateEvent event = (IdleStateEvent) evt;
if (event.state() == IdleState.READER_IDLE) {
log.warn("变频器连接读空闲converterChannelTag={}", converterChannelTag);
// 可以选择发送心跳或关闭连接
}
}
super.userEventTriggered(ctx, evt);
}
private void attemptReconnect(ChannelHandlerContext ctx) {
isReconnecting = true;
reconnectAttempts++;
log.info("准备重连变频器converterChannelTag={}, 第{}/{}次重连,{}秒后开始",
converterChannelTag, reconnectAttempts, MAX_RECONNECT_ATTEMPTS, RECONNECT_INTERVAL_MS / 1000);
ctx.executor().schedule(() -> {
try {
log.info("开始执行变频器重连converterChannelTag={}", converterChannelTag);
socketFreqConverterService.reconnect(converterChannelTag);
} catch (Exception e) {
log.error("变频器重连触发失败converterChannelTag={}, error={}", converterChannelTag, e.getMessage(), e);
isReconnecting = false;
if (reconnectAttempts < MAX_RECONNECT_ATTEMPTS) {
log.warn("将在{}秒后进行下一次重试", RECONNECT_INTERVAL_MS / 1000);
ctx.executor().schedule(() -> attemptReconnect(ctx), RECONNECT_INTERVAL_MS, java.util.concurrent.TimeUnit.MILLISECONDS);
}
}
}, RECONNECT_INTERVAL_MS, java.util.concurrent.TimeUnit.MILLISECONDS);
}
}

View File

@@ -0,0 +1,63 @@
package com.njcn.gather.detection.util.socket.cilent;
import com.njcn.gather.detection.handler.SocketFreqConverterDevService;
import com.njcn.gather.detection.util.socket.FormalTestManager;
import com.njcn.gather.detection.util.socket.SocketManager;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler;
import io.netty.handler.timeout.IdleState;
import io.netty.handler.timeout.IdleStateEvent;
import lombok.extern.slf4j.Slf4j;
/**
* 设备 Netty 客户端处理器
*/
@Slf4j
public class NettyFreqConverterDevClientHandler extends SimpleChannelInboundHandler<String> {
private final String devChannelTag;
private final SocketFreqConverterDevService socketFreqConverterDevService;
public NettyFreqConverterDevClientHandler(String devChannelTag, SocketFreqConverterDevService socketFreqConverterDevService) {
this.devChannelTag = devChannelTag;
this.socketFreqConverterDevService = socketFreqConverterDevService;
}
@Override
public void channelActive(ChannelHandlerContext ctx) throws Exception {
log.info("设备连接已建立devChannelTag={}, channelId={}", devChannelTag, ctx.channel().id());
SocketManager.addUser(devChannelTag, ctx.channel());
super.channelActive(ctx);
}
@Override
protected void channelRead0(ChannelHandlerContext ctx, String msg) throws Exception {
log.info("收到设备消息devChannelTag={}, msg={}", devChannelTag, msg);
socketFreqConverterDevService.handleRead(devChannelTag, msg);
}
@Override
public void channelInactive(ChannelHandlerContext ctx) throws Exception {
log.warn("设备连接已断开devChannelTag={}", devChannelTag);
socketFreqConverterDevService.cleanup(devChannelTag);
super.channelInactive(ctx);
}
@Override
public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
log.error("设备连接发生异常devChannelTag={}, error={}", devChannelTag, cause.getMessage(), cause);
ctx.close();
}
@Override
public void userEventTriggered(ChannelHandlerContext ctx, Object evt) throws Exception {
if (evt instanceof IdleStateEvent) {
IdleStateEvent event = (IdleStateEvent) evt;
if (event.state() == IdleState.READER_IDLE) {
log.warn("设备连接读空闲devChannelTag={}", devChannelTag);
}
}
super.userEventTriggered(ctx, evt);
}
}

View File

@@ -18,29 +18,21 @@ import java.util.Set;
@Component @Component
@ConfigurationProperties(prefix = "socket") @ConfigurationProperties(prefix = "socket")
public class SocketConnectionConfig { public class SocketConnectionConfig {
/**
* 被检设备配置
*/
private DeviceConfig device = new DeviceConfig();
/** /**
* 程控源设备配置 * 程控源设备配置
*/ */
private SourceConfig source = new SourceConfig(); private SourceConfig source = new SourceConfig();
/**
* 被检设备配置
*/
private DeviceConfig device = new DeviceConfig();
@Data /**
public static class SourceConfig { * 变频器配置
/** */
* 程控源IP地址 private DeviceConfig freqConverter = new DeviceConfig();
*/
private String ip;
/**
* 程控源端口号
*/
private Integer port;
}
@Data @Data
public static class DeviceConfig { public static class DeviceConfig {
@@ -48,13 +40,33 @@ public class SocketConnectionConfig {
* 被检设备IP地址 * 被检设备IP地址
*/ */
private String ip; private String ip;
/** /**
* 被检设备端口号 * 被检设备端口号
*/ */
private Integer port; private Integer port;
} }
@Data
public static class SourceConfig {
/**
* 程控源IP地址
*/
private String ip;
/**
* 程控源端口号
*/
private Integer port;
}
/**
* 获取被检设备配置
*/
public DeviceConfig getDevice() {
return device;
}
/** /**
* 获取程控源配置 * 获取程控源配置
*/ */
@@ -63,10 +75,10 @@ public class SocketConnectionConfig {
} }
/** /**
* 获取被检设备配置 * 获取变频器配置
*/ */
public DeviceConfig getDevice() { public DeviceConfig getFreqConverter() {
return device; return freqConverter;
} }
/** /**

View File

@@ -120,6 +120,7 @@ public class WebServiceManager {
channel.writeAndFlush(frame); channel.writeAndFlush(frame);
} else { } else {
log.error("WebSocket推送消息失败用户连接已断开时间: {}, userId: {}", LocalDateTime.now(), userId); log.error("WebSocket推送消息失败用户连接已断开时间: {}, userId: {}", LocalDateTime.now(), userId);
WebSocketHandler.cleanupSocketResources(userId);
} }
} }
@@ -137,6 +138,7 @@ public class WebServiceManager {
channel.writeAndFlush(frame); channel.writeAndFlush(frame);
} else { } else {
log.error("WebSocket推送结构化消息失败用户连接已断开时间: {}, userId: {}", LocalDateTime.now(), userId); log.error("WebSocket推送结构化消息失败用户连接已断开时间: {}, userId: {}", LocalDateTime.now(), userId);
WebSocketHandler.cleanupSocketResources(userId);
} }
} }

View File

@@ -5,7 +5,6 @@ import com.njcn.gather.detection.pojo.enums.SourceOperateCodeEnum;
import com.njcn.gather.detection.pojo.param.PreDetectionParam; import com.njcn.gather.detection.pojo.param.PreDetectionParam;
import com.njcn.gather.detection.util.socket.CnSocketUtil; import com.njcn.gather.detection.util.socket.CnSocketUtil;
import com.njcn.gather.detection.util.socket.FormalTestManager; import com.njcn.gather.detection.util.socket.FormalTestManager;
import com.njcn.gather.detection.util.socket.SocketManager;
import com.njcn.gather.device.pojo.enums.PatternEnum; import com.njcn.gather.device.pojo.enums.PatternEnum;
import io.netty.channel.ChannelHandlerContext; import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.SimpleChannelInboundHandler; import io.netty.channel.SimpleChannelInboundHandler;
@@ -360,7 +359,7 @@ public class WebSocketHandler extends SimpleChannelInboundHandler<TextWebSocketF
* *
* @param userId 用户ID * @param userId 用户ID
*/ */
private void cleanupSocketResources(String userId) { public static void cleanupSocketResources(String userId) {
if (userId == null || userId.trim().isEmpty()) { if (userId == null || userId.trim().isEmpty()) {
log.warn("userId为空无法进行Socket连接清理"); log.warn("userId为空无法进行Socket连接清理");
return; return;
@@ -372,20 +371,43 @@ public class WebSocketHandler extends SimpleChannelInboundHandler<TextWebSocketF
// 使用该用户的检测参数关闭Socket连接 // 使用该用户的检测参数关闭Socket连接
log.info("使用用户检测参数关闭Socket连接userId: {}", userId); log.info("使用用户检测参数关闭Socket连接userId: {}", userId);
if (FormalTestManager.patternEnum.equals(PatternEnum.CONTRAST)) { if (FormalTestManager.patternEnum.equals(PatternEnum.CONTRAST)) {
if (!FormalTestManager.isRemoveSocket) { CnSocketUtil.contrastSendquit(preDetectionParam.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_01, false);
if (FormalTestManager.currentStep == SourceOperateCodeEnum.RECORD_WAVE_STEP1 || FormalTestManager.currentStep == SourceOperateCodeEnum.RECORD_WAVE_STEP2) { CnSocketUtil.contrastSendquit(preDetectionParam.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, false);
CnSocketUtil.contrastSendquit(preDetectionParam.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_03, true); CnSocketUtil.contrastSendquit(preDetectionParam.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_03, false);
} else if (FormalTestManager.currentStep != SourceOperateCodeEnum.QUITE) { // if (FormalTestManager.isRemoveSocket) {
CnSocketUtil.contrastSendquit(preDetectionParam.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, true); // boolean channelActive = SocketManager.isChannelActive(preDetectionParam.getUserPageId() + CnSocketUtil.CONTRAST_DEV_TAG);
} else { // if (channelActive) {
SocketManager.removeUser(preDetectionParam.getUserPageId() + CnSocketUtil.CONTRAST_DEV_TAG); // SocketManager.removeUser(preDetectionParam.getUserPageId() + CnSocketUtil.CONTRAST_DEV_TAG);
} // }
} else { // } else {
boolean channelActive = SocketManager.isChannelActive(preDetectionParam.getUserPageId() + CnSocketUtil.CONTRAST_DEV_TAG); // if (FormalTestManager.currentStep == SourceOperateCodeEnum.RECORD_WAVE_STEP1) {
if (channelActive) { // CnSocketUtil.contrastSendquit(preDetectionParam.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_03, false);
SocketManager.removeUser(preDetectionParam.getUserPageId() + CnSocketUtil.CONTRAST_DEV_TAG); // } else if (FormalTestManager.currentStep != SourceOperateCodeEnum.QUITE) {
} // if (ObjectUtil.isNotNull(FormalTestManager.nonWaveDataSourceEnum)) {
} // if (FormalTestManager.nonWaveDataSourceEnum == DataSourceEnum.REAL_DATA) {
// CnSocketUtil.contrastSendquit(preDetectionParam.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, false);
// if (FormalTestManager.statisticsProtocol) {
// CnSocketUtil.contrastSendquit(preDetectionParam.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_01, false);
// }
// } else {
// CnSocketUtil.contrastSendquit(preDetectionParam.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_01, false);
// }
// if (FormalTestManager.isWaveCheck) {
// CnSocketUtil.contrastSendquit(preDetectionParam.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_03, false);
// }
// } else {
// if (FormalTestManager.statisticsProtocol) {
// CnSocketUtil.contrastSendquit(preDetectionParam.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_01, false);
// }
// if (FormalTestManager.isWaveCheck) {
// CnSocketUtil.contrastSendquit(preDetectionParam.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_02, false);
// CnSocketUtil.contrastSendquit(preDetectionParam.getUserPageId(), SourceOperateCodeEnum.QUIT_INIT_03, false);
// }
// }
// } else {
// SocketManager.removeUser(preDetectionParam.getUserPageId() + CnSocketUtil.CONTRAST_DEV_TAG);
// }
// }
} else { } else {
CnSocketUtil.quitSendSource(preDetectionParam); CnSocketUtil.quitSendSource(preDetectionParam);
CnSocketUtil.quitSend(preDetectionParam); CnSocketUtil.quitSend(preDetectionParam);

View File

@@ -12,7 +12,6 @@ import com.njcn.common.utils.LogUtil;
import com.njcn.gather.device.pojo.param.PqDevParam; import com.njcn.gather.device.pojo.param.PqDevParam;
import com.njcn.gather.device.pojo.vo.PqDevVO; import com.njcn.gather.device.pojo.vo.PqDevVO;
import com.njcn.gather.device.service.IPqDevService; import com.njcn.gather.device.service.IPqDevService;
import com.njcn.gather.monitor.pojo.po.PqMonitor;
import com.njcn.gather.type.service.IDevTypeService; import com.njcn.gather.type.service.IDevTypeService;
import com.njcn.web.controller.BaseController; import com.njcn.web.controller.BaseController;
import com.njcn.web.utils.FileUtil; import com.njcn.web.utils.FileUtil;
@@ -101,7 +100,7 @@ public class PqDevController extends BaseController {
@OperateInfo(operateType = OperateType.DELETE) @OperateInfo(operateType = OperateType.DELETE)
@PostMapping("/delete") @PostMapping("/delete")
@ApiOperation("删除被检设备") @ApiOperation("删除被检设备")
@ApiImplicitParam(name = "ids", value = "被检设备id", required = true) @ApiImplicitParam(name = "param", value = "删除参数", required = true)
public HttpResult<Boolean> delete(@RequestBody @Validated PqDevParam.DeleteParam param) { public HttpResult<Boolean> delete(@RequestBody @Validated PqDevParam.DeleteParam param) {
String methodDescribe = getMethodDescribe("delete"); String methodDescribe = getMethodDescribe("delete");
LogUtil.njcnDebug(log, "{}删除ID数据为{}", methodDescribe, String.join(StrUtil.COMMA, param.getIds())); LogUtil.njcnDebug(log, "{}删除ID数据为{}", methodDescribe, String.join(StrUtil.COMMA, param.getIds()));
@@ -137,7 +136,7 @@ public class PqDevController extends BaseController {
@ApiImplicitParam(name = "file", value = "被检设备数据文件", required = true), @ApiImplicitParam(name = "file", value = "被检设备数据文件", required = true),
@ApiImplicitParam(name = "patternId", value = "模式id", required = true) @ApiImplicitParam(name = "patternId", value = "模式id", required = true)
}) })
public HttpResult<Boolean> importDev(@RequestParam("file") MultipartFile file, @RequestParam("patternId") String patternId, @RequestParam("planId") String planId, HttpServletResponse response) { public HttpResult importDev(@RequestParam("file") MultipartFile file, @RequestParam("patternId") String patternId, @RequestParam("planId") String planId, @RequestParam(value = "cover", defaultValue = "0") Integer cover, HttpServletResponse response) {
String methodDescribe = getMethodDescribe("importDev"); String methodDescribe = getMethodDescribe("importDev");
LogUtil.njcnDebug(log, "{},上传文件为:{}", methodDescribe, file.getOriginalFilename()); LogUtil.njcnDebug(log, "{},上传文件为:{}", methodDescribe, file.getOriginalFilename());
boolean fileType = FileUtil.judgeFileIsExcel(file.getOriginalFilename()); boolean fileType = FileUtil.judgeFileIsExcel(file.getOriginalFilename());
@@ -147,12 +146,7 @@ public class PqDevController extends BaseController {
if ("null".equals(planId)) { if ("null".equals(planId)) {
planId = null; planId = null;
} }
Boolean result = pqDevService.importDev(file, patternId, planId, response); return pqDevService.importDev(file, patternId, planId, response, cover);
if (result) {
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, true, methodDescribe);
} else {
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.FAIL, false, methodDescribe);
}
} }
@OperateInfo(info = LogEnum.BUSINESS_COMMON) @OperateInfo(info = LogEnum.BUSINESS_COMMON)
@@ -168,10 +162,41 @@ public class PqDevController extends BaseController {
@OperateInfo @OperateInfo
@GetMapping("/getSelectOptions") @GetMapping("/getSelectOptions")
@ApiOperation("根据历史记录信息来获取下拉框内容") @ApiOperation("从历史数据中查询下拉框选项")
@ApiImplicitParam(name = "pattern", value = "模式id", required = true)
public HttpResult<Map<String, List<String>>> getSelectOptions(@RequestParam("pattern") String pattern) { public HttpResult<Map<String, List<String>>> getSelectOptions(@RequestParam("pattern") String pattern) {
String methodDescribe = getMethodDescribe("getSelectOptions"); String methodDescribe = getMethodDescribe("getSelectOptions");
Map<String, List<String>> result = pqDevService.listSelectOptions(pattern); Map<String, List<String>> result = pqDevService.listSelectOptions(pattern);
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, result, methodDescribe); return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, result, methodDescribe);
} }
@OperateInfo(info = LogEnum.BUSINESS_COMMON, operateType = OperateType.UPLOAD)
@PostMapping(value = "/ttt")
@ApiOperation("批量导入被检设备")
@ApiImplicitParams({
@ApiImplicitParam(name = "file", value = "被检设备数据文件", required = true),
@ApiImplicitParam(name = "patternId", value = "模式id", required = true)
})
public HttpResult ttt(@RequestParam("file") MultipartFile file, @RequestParam("patternId") String patternId, @RequestParam("planId") String planId, @RequestParam(value = "cover", defaultValue = "0") Integer cover, HttpServletResponse response) {
String methodDescribe = getMethodDescribe("ttt");
LogUtil.njcnDebug(log, "{},上传文件为:{}", methodDescribe, file.getOriginalFilename());
boolean fileType = FileUtil.judgeFileIsExcel(file.getOriginalFilename());
if (!fileType) {
throw new BusinessException(CommonResponseEnum.FILE_XLSX_ERROR);
}
if ("null".equals(planId)) {
planId = null;
}
return pqDevService.importDev(file, patternId, planId, response, cover);
}
@OperateInfo
@GetMapping("/listAll")
@ApiOperation("查询所有未删除设备数据")
public HttpResult<List<PqDevVO>> listAll() {
String methodDescribe = getMethodDescribe("listAll");
LogUtil.njcnDebug(log, "{},查询所有设备", methodDescribe);
List<PqDevVO> result = pqDevService.listAll();
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, result, methodDescribe);
}
} }

View File

@@ -12,6 +12,7 @@ import com.njcn.common.pojo.response.HttpResult;
import com.njcn.common.utils.LogUtil; import com.njcn.common.utils.LogUtil;
import com.njcn.gather.device.pojo.param.PqStandardDevParam; import com.njcn.gather.device.pojo.param.PqStandardDevParam;
import com.njcn.gather.device.pojo.po.PqStandardDev; import com.njcn.gather.device.pojo.po.PqStandardDev;
import com.njcn.gather.device.pojo.vo.PqStandardDevVO;
import com.njcn.gather.device.service.IPqStandardDevService; import com.njcn.gather.device.service.IPqStandardDevService;
import com.njcn.web.controller.BaseController; import com.njcn.web.controller.BaseController;
import com.njcn.web.utils.FileUtil; import com.njcn.web.utils.FileUtil;
@@ -55,10 +56,10 @@ public class PqStandardDevController extends BaseController {
@GetMapping("/getById") @GetMapping("/getById")
@ApiOperation("根据id查询标准设备") @ApiOperation("根据id查询标准设备")
@ApiImplicitParam(name = "id", value = "标准设备id", required = true) @ApiImplicitParam(name = "id", value = "标准设备id", required = true)
public HttpResult<PqStandardDev> getById(@RequestParam("id") String id) { public HttpResult<PqStandardDevVO> getById(@RequestParam("id") String id) {
String methodDescribe = getMethodDescribe("getById"); String methodDescribe = getMethodDescribe("getById");
LogUtil.njcnDebug(log, "{}查询ID为{}", methodDescribe, id); LogUtil.njcnDebug(log, "{}查询ID为{}", methodDescribe, id);
PqStandardDev result = pqStandardDevService.getPqStandardDevById(id); PqStandardDevVO result = pqStandardDevService.getPqStandardDevById(id);
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, result, methodDescribe); return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, result, methodDescribe);
} }
@@ -149,17 +150,17 @@ public class PqStandardDevController extends BaseController {
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, result, methodDescribe); return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, result, methodDescribe);
} }
@OperateInfo(info = LogEnum.BUSINESS_COMMON) // @OperateInfo(info = LogEnum.BUSINESS_COMMON)
@PostMapping("/listByPlanId") // @PostMapping("/listByPlanId")
@ApiOperation("查询出指定计划已关联的标准设备") // @ApiOperation("查询出指定计划已关联的标准设备")
@ApiImplicitParam(name = "planId", value = "计划id", required = true) // @ApiImplicitParam(name = "planId", value = "计划id", required = true)
public HttpResult<List<PqStandardDev>> listByPlanId(@RequestParam("planId") String planId) { // public HttpResult<List<PqStandardDev>> listByPlanId(@RequestParam("planId") String planId) {
String methodDescribe = getMethodDescribe("listByPlanId"); // String methodDescribe = getMethodDescribe("listByPlanId");
LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, planId); // LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, planId);
List<PqStandardDev> pqDevVOList = pqStandardDevService.listByPlanId(planId); // List<PqStandardDev> pqDevVOList = pqStandardDevService.listByPlanId(planId);
//
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, pqDevVOList, methodDescribe); // return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, pqDevVOList, methodDescribe);
} // }
@OperateInfo(info = LogEnum.BUSINESS_COMMON) @OperateInfo(info = LogEnum.BUSINESS_COMMON)

View File

@@ -0,0 +1,13 @@
package com.njcn.gather.device.mapper;
import com.github.yulichang.base.MPJBaseMapper;
import com.njcn.gather.device.pojo.po.PqStandardDevGain;
/**
* @author caozehui
* @date 2026-01-12
*/
public interface PqStandardDevGainMapper extends MPJBaseMapper<PqStandardDevGain> {
}

View File

@@ -0,0 +1,13 @@
package com.njcn.gather.device.mapper;
import com.github.yulichang.base.MPJBaseMapper;
import com.njcn.gather.device.pojo.po.PqStandardDevGainRecord;
/**
* @author caozehui
* @date 2026-01-15
*/
public interface PqStandardDevGainRecordMapper extends MPJBaseMapper<PqStandardDevGainRecord> {
}

View File

@@ -134,6 +134,7 @@
#{item} #{item}
</foreach> </foreach>
</if> </if>
order by dev.Create_Time DESC, dev.Name ASC
</select> </select>
</mapper> </mapper>

View File

@@ -0,0 +1,7 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.njcn.gather.device.mapper.PqStandardDevGainMapper">
</mapper>

View File

@@ -0,0 +1,7 @@
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.njcn.gather.device.mapper.PqStandardDevGainRecordMapper">
</mapper>

View File

@@ -6,14 +6,12 @@ import com.njcn.gather.pojo.constant.DetectionValidMessage;
import com.njcn.web.pojo.annotation.DateTimeStrValid; import com.njcn.web.pojo.annotation.DateTimeStrValid;
import com.njcn.web.pojo.param.BaseParam; import com.njcn.web.pojo.param.BaseParam;
import io.swagger.annotations.ApiModelProperty; import io.swagger.annotations.ApiModelProperty;
import io.swagger.models.auth.In;
import lombok.Data; import lombok.Data;
import lombok.EqualsAndHashCode; import lombok.EqualsAndHashCode;
import org.hibernate.validator.constraints.Range; import org.hibernate.validator.constraints.Range;
import javax.validation.Valid; import javax.validation.Valid;
import javax.validation.constraints.*; import javax.validation.constraints.*;
import java.time.LocalDate;
import java.util.List; import java.util.List;
/** /**
@@ -137,6 +135,7 @@ public class PqDevParam {
@ApiModelProperty("是否为导入设备") @ApiModelProperty("是否为导入设备")
private Integer importFlag; private Integer importFlag;
/** /**
* 更新操作实体 * 更新操作实体
*/ */
@@ -197,6 +196,12 @@ public class PqDevParam {
@ApiModelProperty("是否分配") @ApiModelProperty("是否分配")
private Integer assign; private Integer assign;
@ApiModelProperty("关键词")
private String keywords;
@ApiModelProperty("主计划ID")
private String planId;
@ApiModelProperty("是否分配子计划")
private Integer assignSub;
} }
@Data @Data

View File

@@ -1,5 +1,6 @@
package com.njcn.gather.device.pojo.po; package com.njcn.gather.device.pojo.po;
import com.baomidou.mybatisplus.annotation.TableField;
import com.baomidou.mybatisplus.annotation.TableName; import com.baomidou.mybatisplus.annotation.TableName;
import com.njcn.db.mybatisplus.bo.BaseEntity; import com.njcn.db.mybatisplus.bo.BaseEntity;
import lombok.Data; import lombok.Data;
@@ -72,5 +73,8 @@ public class PqStandardDev extends BaseEntity implements Serializable {
private String devKey; private String devKey;
private Integer state; private Integer state;
@TableField(exist = false)
private boolean disabled;
} }

View File

@@ -0,0 +1,61 @@
package com.njcn.gather.device.pojo.po;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableName;
import com.njcn.db.mybatisplus.bo.BaseEntity;
import lombok.Data;
import lombok.EqualsAndHashCode;
import java.io.Serializable;
/**
* @author caozehui
* @date 2026-01-12
*/
@Data
@EqualsAndHashCode(callSuper = true)
@TableName("pq_standard_dev_gain")
public class PqStandardDevGain extends BaseEntity implements Serializable {
private static final long serialVersionUID = -46280767885558804L;
/**
* 标准设备Id_通道号
*/
@TableId
private String stdDevMonitorId;
private Integer uaGain;
private Integer ubGain;
private Integer ucGain;
private Integer u0Gain;
private Integer iaGain;
private Integer ibGain;
private Integer icGain;
private Integer i0Gain;
private Integer uabGain;
private Integer ubcGain;
private Integer ucaGain;
private Integer state;
}

View File

@@ -0,0 +1,35 @@
package com.njcn.gather.device.pojo.po;
import com.baomidou.mybatisplus.annotation.TableName;
import com.njcn.db.mybatisplus.bo.BaseEntity;
import lombok.Data;
import lombok.EqualsAndHashCode;
import java.io.Serializable;
/**
* @author caozehui
* @date 2026-01-15
*/
@Data
@TableName("pq_standard_dev_gain_record")
public class PqStandardDevGainRecord implements Serializable {
private static final long serialVersionUID = 477276215572686991L;
/**
* 标准设备监测点id,由设备id_通道号组成
*/
private String stdDevMonitorId;
/**
* 被检设备监测点id,由设备id_通道号组成
*/
private String devMonitorId;
/**
* 系数下发次数
*/
private Integer num;
}

View File

@@ -38,8 +38,8 @@ public class ContrastDevExcel implements Serializable {
@Pattern(regexp = PatternRegex.DEV_NAME_REGEX, message = DetectionValidMessage.NAME_FORMAT_ERROR) @Pattern(regexp = PatternRegex.DEV_NAME_REGEX, message = DetectionValidMessage.NAME_FORMAT_ERROR)
private String name; private String name;
@Excel(name = "设备序列号*", width = 20, needMerge = true, orderNum = "5") @Excel(name = "设备序列号", width = 20, needMerge = true, orderNum = "5")
@NotBlank(message = DetectionValidMessage.FACTORYNO_NOT_BLANK) // @NotBlank(message = DetectionValidMessage.FACTORYNO_NOT_BLANK)
private String createId; private String createId;
@Excel(name = "设备类型*", width = 20, needMerge = true, orderNum = "6") @Excel(name = "设备类型*", width = 20, needMerge = true, orderNum = "6")
@@ -91,8 +91,8 @@ public class ContrastDevExcel implements Serializable {
@NotNull(message = DetectionValidMessage.INSPECT_DATE_NOT_NULL) @NotNull(message = DetectionValidMessage.INSPECT_DATE_NOT_NULL)
private LocalDate inspectDate; private LocalDate inspectDate;
@Excel(name = "谐波系统设备id*", width = 30, needMerge = true, orderNum = "18") @Excel(name = "谐波系统设备id", width = 30, needMerge = true, orderNum = "18")
@NotBlank(message = DetectionValidMessage.HARM_SYS_ID_NOT_BLANK) // @NotBlank(message = DetectionValidMessage.HARM_SYS_ID_NOT_BLANK)
private String harmSysId; private String harmSysId;
@ExcelCollection(name = "监测点信息", orderNum = "19") @ExcelCollection(name = "监测点信息", orderNum = "19")

View File

@@ -111,4 +111,9 @@ public class PqDevVO extends PqDev {
* 是否已经分配。0-未分配、1-已分配、2-所有 * 是否已经分配。0-未分配、1-已分配、2-所有
*/ */
private Integer assign; private Integer assign;
/**
* 检测点结果
*/
private List<Integer> monitorResults;
} }

View File

@@ -0,0 +1,20 @@
package com.njcn.gather.device.pojo.vo;
import com.njcn.gather.device.pojo.po.PqStandardDev;
import com.njcn.gather.device.pojo.po.PqStandardDevGain;
import lombok.Data;
import java.util.List;
/**
* @author caozehui
* @data 2026-01-19
*/
@Data
public class PqStandardDevVO extends PqStandardDev {
/**
* 通道系数
*/
List<PqStandardDevGain> gainList;
}

View File

@@ -3,6 +3,7 @@ package com.njcn.gather.device.service;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.IService; import com.baomidou.mybatisplus.extension.service.IService;
import com.njcn.common.pojo.poi.PullDown; import com.njcn.common.pojo.poi.PullDown;
import com.njcn.common.pojo.response.HttpResult;
import com.njcn.gather.device.pojo.enums.TimeCheckResultEnum; import com.njcn.gather.device.pojo.enums.TimeCheckResultEnum;
import com.njcn.gather.device.pojo.param.PqDevParam; import com.njcn.gather.device.pojo.param.PqDevParam;
import com.njcn.gather.device.pojo.po.PqDev; import com.njcn.gather.device.pojo.po.PqDev;
@@ -124,7 +125,7 @@ public interface IPqDevService extends IService<PqDev> {
* @param devId * @param devId
* @param userId * @param userId
*/ */
void updateResult(String devId,String userId); void updateResult(String devId, String userId);
void updatePqDevReportState(String devId, int i); void updatePqDevReportState(String devId, int i);
@@ -160,7 +161,7 @@ public interface IPqDevService extends IService<PqDev> {
* @param planId 计划Id * @param planId 计划Id
* @param response 响应 * @param response 响应
*/ */
boolean importDev(MultipartFile file, String patternId, String planId, HttpServletResponse response); HttpResult importDev(MultipartFile file, String patternId, String planId, HttpServletResponse response, Integer cover);
/** /**
* 导入灿能二楼设备数据 * 导入灿能二楼设备数据
@@ -254,7 +255,7 @@ public interface IPqDevService extends IService<PqDev> {
* @param planId 计划Id * @param planId 计划Id
* @param response 响应 * @param response 响应
*/ */
boolean importContrastDev(MultipartFile file, String patternId, String planId, HttpServletResponse response); HttpResult importContrastDev(MultipartFile file, String patternId, String planId, HttpServletResponse response, Integer cover);
/** /**
* 导入比对式设备数据 * 导入比对式设备数据
@@ -262,7 +263,7 @@ public interface IPqDevService extends IService<PqDev> {
* @param contrastDevExcelList * @param contrastDevExcelList
* @param patternId * @param patternId
*/ */
boolean importContrastDev(List<ContrastDevExcel> contrastDevExcelList, String patternId, String planId); HttpResult importContrastDev(List<ContrastDevExcel> contrastDevExcelList, String patternId, String planId, Integer cover);
/** /**
* 获取比对式设备导出、导出文件模板的下拉列表 * 获取比对式设备导出、导出文件模板的下拉列表
@@ -287,4 +288,6 @@ public interface IPqDevService extends IService<PqDev> {
* @return * @return
*/ */
List<ContrastDevExcel> getExportContrastDevData(List<PqDevVO> pqDevVOList); List<ContrastDevExcel> getExportContrastDevData(List<PqDevVO> pqDevVOList);
List<PqDevVO> listAll();
} }

View File

@@ -0,0 +1,29 @@
package com.njcn.gather.device.service;
import com.baomidou.mybatisplus.extension.service.IService;
import com.njcn.gather.device.pojo.po.PqStandardDevGainRecord;
import java.util.List;
/**
* @author caozehui
* @date 2026-01-15
*/
public interface IPqStandardDevGainRecordService extends IService<PqStandardDevGainRecord> {
/**
* 获取最大下发系数的次数
*
* @param stdDevMonitorId
* @param devMonitorId
* @return
*/
int getMaxNum(String stdDevMonitorId, String devMonitorId);
/**
* 批量新增或更新记录
* @param recordList
* @return
*/
boolean addOrUpdate(List<PqStandardDevGainRecord> recordList);
}

View File

@@ -0,0 +1,19 @@
package com.njcn.gather.device.service;
import com.baomidou.mybatisplus.extension.service.IService;
import com.njcn.gather.device.pojo.po.PqStandardDevGain;
import java.util.List;
/**
* @author caozehui
* @date 2026-01-12
*/
public interface IPqStandardDevGainService extends IService<PqStandardDevGain> {
boolean add(List<PqStandardDevGain> pqStandardDevGainList);
PqStandardDevGain get(String stdDevMonitorId);
List<PqStandardDevGain> list(String stdDevId);
}

View File

@@ -4,6 +4,7 @@ import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.IService; import com.baomidou.mybatisplus.extension.service.IService;
import com.njcn.gather.device.pojo.param.PqStandardDevParam; import com.njcn.gather.device.pojo.param.PqStandardDevParam;
import com.njcn.gather.device.pojo.po.PqStandardDev; import com.njcn.gather.device.pojo.po.PqStandardDev;
import com.njcn.gather.device.pojo.vo.PqStandardDevVO;
import com.njcn.gather.device.pojo.vo.PreDetection; import com.njcn.gather.device.pojo.vo.PreDetection;
import org.springframework.web.multipart.MultipartFile; import org.springframework.web.multipart.MultipartFile;
@@ -30,7 +31,7 @@ public interface IPqStandardDevService extends IService<PqStandardDev> {
* @param id 设备id * @param id 设备id
* @return 设备对象 * @return 设备对象
*/ */
PqStandardDev getPqStandardDevById(String id); PqStandardDevVO getPqStandardDevById(String id);
/** /**
* 新增标准设备 * 新增标准设备
@@ -82,7 +83,7 @@ public interface IPqStandardDevService extends IService<PqStandardDev> {
* @param planId * @param planId
* @return * @return
*/ */
List<PqStandardDev> listByPlanId(String planId); // List<PqStandardDev> listByPlanId(String planId);
/** /**
* 查询出标准设备所需的检测信息 * 查询出标准设备所需的检测信息

View File

@@ -11,15 +11,19 @@ import cn.hutool.core.util.ObjectUtil;
import cn.hutool.core.util.ReflectUtil; import cn.hutool.core.util.ReflectUtil;
import cn.hutool.core.util.StrUtil; import cn.hutool.core.util.StrUtil;
import com.baomidou.mybatisplus.core.conditions.Wrapper; import com.baomidou.mybatisplus.core.conditions.Wrapper;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper; import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
import com.baomidou.mybatisplus.extension.conditions.update.LambdaUpdateChainWrapper;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.github.yulichang.wrapper.MPJLambdaWrapper; import com.github.yulichang.wrapper.MPJLambdaWrapper;
import com.njcn.common.pojo.constant.PatternRegex; import com.njcn.common.pojo.constant.PatternRegex;
import com.njcn.common.pojo.enums.common.DataStateEnum; import com.njcn.common.pojo.enums.common.DataStateEnum;
import com.njcn.common.pojo.enums.response.CommonResponseEnum;
import com.njcn.common.pojo.exception.BusinessException; import com.njcn.common.pojo.exception.BusinessException;
import com.njcn.common.pojo.poi.PullDown; import com.njcn.common.pojo.poi.PullDown;
import com.njcn.common.pojo.response.HttpResult;
import com.njcn.common.utils.EncryptionUtil; import com.njcn.common.utils.EncryptionUtil;
import com.njcn.db.mybatisplus.constant.DbConstant; import com.njcn.db.mybatisplus.constant.DbConstant;
import com.njcn.gather.device.mapper.PqDevMapper; import com.njcn.gather.device.mapper.PqDevMapper;
@@ -33,6 +37,7 @@ import com.njcn.gather.device.service.IPqDevSubService;
import com.njcn.gather.monitor.pojo.po.PqMonitor; import com.njcn.gather.monitor.pojo.po.PqMonitor;
import com.njcn.gather.monitor.pojo.vo.PqMonitorExcel; import com.njcn.gather.monitor.pojo.vo.PqMonitorExcel;
import com.njcn.gather.monitor.service.IPqMonitorService; import com.njcn.gather.monitor.service.IPqMonitorService;
import com.njcn.gather.plan.pojo.enums.DataSourceEnum;
import com.njcn.gather.pojo.enums.DetectionResponseEnum; import com.njcn.gather.pojo.enums.DetectionResponseEnum;
import com.njcn.gather.storage.service.DetectionDataDealService; import com.njcn.gather.storage.service.DetectionDataDealService;
import com.njcn.gather.system.cfg.pojo.enums.SceneEnum; import com.njcn.gather.system.cfg.pojo.enums.SceneEnum;
@@ -48,6 +53,7 @@ import com.njcn.gather.user.user.pojo.po.SysUser;
import com.njcn.gather.user.user.service.ISysUserService; import com.njcn.gather.user.user.service.ISysUserService;
import com.njcn.web.factory.PageFactory; import com.njcn.web.factory.PageFactory;
import com.njcn.web.utils.ExcelUtil; import com.njcn.web.utils.ExcelUtil;
import com.njcn.web.utils.HttpResultUtil;
import com.njcn.web.utils.PoiUtil; import com.njcn.web.utils.PoiUtil;
import lombok.RequiredArgsConstructor; import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
@@ -62,6 +68,7 @@ import java.time.LocalDateTime;
import java.util.*; import java.util.*;
import java.util.regex.Pattern; import java.util.regex.Pattern;
import java.util.stream.Collectors; import java.util.stream.Collectors;
import java.util.stream.Stream;
/** /**
* @author caozehui * @author caozehui
@@ -211,7 +218,8 @@ public class PqDevServiceImpl extends ServiceImpl<PqDevMapper, PqDev> implements
if (PatternEnum.CONTRAST.getValue().equals(dictDataService.getDictDataById(param.getPattern()).getCode())) { if (PatternEnum.CONTRAST.getValue().equals(dictDataService.getDictDataById(param.getPattern()).getCode())) {
for (String id : param.getIds()) { for (String id : param.getIds()) {
if (ObjectUtils.isNotEmpty(pqMonitorService.listPqMonitorByDevIds(Collections.singletonList(id)))) { if (ObjectUtils.isNotEmpty(pqMonitorService.listPqMonitorByDevIds(Collections.singletonList(id)))) {
throw new BusinessException(DetectionResponseEnum.PQ_DEV_HAS_MONITOR); // throw new BusinessException(DetectionResponseEnum.PQ_DEV_HAS_MONITOR);
pqMonitorService.removeByDevId(id);
} }
} }
} }
@@ -285,6 +293,33 @@ public class PqDevServiceImpl extends ServiceImpl<PqDevMapper, PqDev> implements
return Collections.emptyList(); return Collections.emptyList();
} }
List<PqDevVO> pqDevList = this.baseMapper.selectByQueryParam(param); List<PqDevVO> pqDevList = this.baseMapper.selectByQueryParam(param);
pqDevList.forEach(pqDev -> {
List<PqMonitor> monitorList = pqMonitorService.listPqMonitorByDevIds(Collections.singletonList(pqDev.getId()));
List<PqMonitor> enabledMonitorList = monitorList.stream().filter(x -> x.getCheckFlag() == 1).collect(Collectors.toList());
pqDev.setMonitorResults(enabledMonitorList.stream().map(x -> {
if (ObjectUtil.isNull(x.getResultType())) {
return CheckResultEnum.UNCHECKED.getValue();
} else {
DataSourceEnum dataSourceEnum = DataSourceEnum.ofByValue(x.getResultType());
switch (dataSourceEnum) {
case REAL_DATA:
return x.getRealtimeResult();
case MINUTE_STATISTICS_AVG:
case MINUTE_STATISTICS_CP95:
case MINUTE_STATISTICS_MIN:
case MINUTE_STATISTICS_MAX:
return x.getStatisticsResult();
case WAVE_DATA:
return x.getRecordedResult();
default:
break;
}
}
return CheckResultEnum.UNCHECKED.getValue();
}).collect(Collectors.toList()));
});
return pqDevList; return pqDevList;
} }
@@ -395,12 +430,21 @@ public class PqDevServiceImpl extends ServiceImpl<PqDevMapper, PqDev> implements
.in(CollectionUtil.isNotEmpty(queryParam.getPlanIdList()), "pq_dev.Plan_Id", queryParam.getPlanIdList()) .in(CollectionUtil.isNotEmpty(queryParam.getPlanIdList()), "pq_dev.Plan_Id", queryParam.getPlanIdList())
.isNotNull(DataStateEnum.ENABLE.getCode().equals(queryParam.getAssign()), "pq_dev.Plan_Id") .isNotNull(DataStateEnum.ENABLE.getCode().equals(queryParam.getAssign()), "pq_dev.Plan_Id")
.isNull(DataStateEnum.DELETED.getCode().equals(queryParam.getAssign()), "pq_dev.Plan_Id") .isNull(DataStateEnum.DELETED.getCode().equals(queryParam.getAssign()), "pq_dev.Plan_Id")
.eq(DataStateEnum.DELETED.getCode().equals(queryParam.getAssignSub()), "pq_dev.Plan_Id", queryParam.getPlanId())
.ne(DataStateEnum.ENABLE.getCode().equals(queryParam.getAssignSub()), "pq_dev.Plan_Id", queryParam.getPlanId())
.between(ObjectUtil.isAllNotEmpty(queryParam.getSearchBeginTime(), queryParam.getSearchEndTime()), "pq_dev.Create_Date", queryParam.getSearchBeginTime(), queryParam.getSearchEndTime()); .between(ObjectUtil.isAllNotEmpty(queryParam.getSearchBeginTime(), queryParam.getSearchEndTime()), "pq_dev.Create_Date", queryParam.getSearchBeginTime(), queryParam.getSearchEndTime());
if (StrUtil.isNotBlank(queryParam.getRegion())) { if (StrUtil.isNotBlank(queryParam.getRegion())) {
queryWrapper.and(w -> w.like(StrUtil.isNotBlank(queryParam.getRegion()), "pq_dev.City_Name", queryParam.getRegion()) queryWrapper.and(w -> w.like(StrUtil.isNotBlank(queryParam.getRegion()), "pq_dev.City_Name", queryParam.getRegion())
.or().like(StrUtil.isNotBlank(queryParam.getRegion()), "pq_dev.Gd_Name", queryParam.getRegion()) .or().like(StrUtil.isNotBlank(queryParam.getRegion()), "pq_dev.Gd_Name", queryParam.getRegion())
.or().like(StrUtil.isNotBlank(queryParam.getRegion()), "pq_dev.Sub_Name", queryParam.getRegion())); .or().like(StrUtil.isNotBlank(queryParam.getRegion()), "pq_dev.Sub_Name", queryParam.getRegion()));
} }
if (StrUtil.isNotBlank(queryParam.getKeywords())) {
queryWrapper.and(w -> w.like(StrUtil.isNotBlank(queryParam.getKeywords()), "pq_dev.City_Name", queryParam.getKeywords())
.or().like(StrUtil.isNotBlank(queryParam.getKeywords()), "pq_dev.Gd_Name", queryParam.getKeywords())
.or().like(StrUtil.isNotBlank(queryParam.getKeywords()), "pq_dev.Sub_Name", queryParam.getKeywords())
.or().like(StrUtil.isNotBlank(queryParam.getKeywords()), "pq_dev.name", queryParam.getKeywords())
);
}
//排序 //排序
if (ObjectUtil.isAllNotEmpty(queryParam.getSortBy(), queryParam.getOrderBy())) { if (ObjectUtil.isAllNotEmpty(queryParam.getSortBy(), queryParam.getOrderBy())) {
queryWrapper.orderBy(true, queryParam.getOrderBy().equals(DbConstant.ASC), StrUtil.toUnderlineCase(queryParam.getSortBy())); queryWrapper.orderBy(true, queryParam.getOrderBy().equals(DbConstant.ASC), StrUtil.toUnderlineCase(queryParam.getSortBy()));
@@ -519,12 +563,30 @@ public class PqDevServiceImpl extends ServiceImpl<PqDevMapper, PqDev> implements
PqDev dev = this.getById(devId); PqDev dev = this.getById(devId);
Integer checkState = pqMonitorService.getDevCheckState(devId); Integer checkState = pqMonitorService.getDevCheckState(devId);
Integer checkResult = pqMonitorService.getDevCheckResult(devId); Integer checkResult = pqMonitorService.getDevCheckResult(devId);
if (checkResult == 1) {
checkResult = CheckResultEnum.ACCORD.getValue();
}
if (checkResult == 2) {
checkResult = CheckResultEnum.NOT_ACCORD.getValue();
}
pqDevSubService.lambdaUpdate()
SysUser user = userService.getById(userId);
LambdaUpdateChainWrapper<PqDevSub> w = pqDevSubService.lambdaUpdate()
.set(PqDevSub::getCheckState, checkState) .set(PqDevSub::getCheckState, checkState)
.set(PqDevSub::getCheckResult, checkResult) .set(PqDevSub::getCheckResult, checkResult)
.set(StrUtil.isNotBlank(userId), PqDevSub::getCheckBy, userId) .set(PqDevSub::getReportState, DevReportStateEnum.NOT_GENERATED.getValue())
.eq(PqDevSub::getDevId, devId).update(); .set(PqDevSub::getCheckTime, LocalDateTime.now())
.eq(PqDevSub::getDevId, devId);
if (ObjectUtil.isNotNull(user)) {
w.set(PqDevSub::getCheckBy, user.getName());
}
if (checkState.equals(CheckStateEnum.CHECKED.getValue())) {
w.set(PqDevSub::getReportState, DevReportStateEnum.NOT_GENERATED.getValue());
}
w.update();
PqDevParam.QueryParam param = new PqDevParam.QueryParam(); PqDevParam.QueryParam param = new PqDevParam.QueryParam();
String planId = dev.getPlanId(); String planId = dev.getPlanId();
@@ -532,18 +594,30 @@ public class PqDevServiceImpl extends ServiceImpl<PqDevMapper, PqDev> implements
List<PqDevVO> pqDevVOList = this.baseMapper.selectByQueryParam(param); List<PqDevVO> pqDevVOList = this.baseMapper.selectByQueryParam(param);
if (CollUtil.isNotEmpty(pqDevVOList)) { if (CollUtil.isNotEmpty(pqDevVOList)) {
Set<Integer> set = pqDevVOList.stream().map(PqDevVO::getCheckResult).collect(Collectors.toSet()); Set<Integer> set = pqDevVOList.stream().filter(obj -> CheckStateEnum.CHECKED.getValue().equals(obj.getCheckState()) || CheckStateEnum.DOCUMENTED.getValue().equals(obj.getCheckState())).map(PqDevVO::getCheckResult).collect(Collectors.toSet());
if (set.contains(CheckResultEnum.NOT_ACCORD.getValue())) { if (checkState == CheckStateEnum.CHECKED.getValue()) {
this.baseMapper.updatePlanCheckResult(planId, CheckResultEnum.NOT_ACCORD.getValue()); set.add(checkResult);
} else if (set.contains(CheckResultEnum.UNCHECKED.getValue())) { }
if (CollUtil.isEmpty(set)) {
this.baseMapper.updatePlanCheckResult(planId, CheckResultEnum.UNCHECKED.getValue()); this.baseMapper.updatePlanCheckResult(planId, CheckResultEnum.UNCHECKED.getValue());
} else { } else {
this.baseMapper.updatePlanCheckResult(planId, CheckResultEnum.ACCORD.getValue()); if (set.contains(CheckResultEnum.NOT_ACCORD.getValue())) {
this.baseMapper.updatePlanCheckResult(planId, CheckResultEnum.NOT_ACCORD.getValue());
} else if (set.contains(CheckResultEnum.UNCHECKED.getValue())) {
this.baseMapper.updatePlanCheckResult(planId, CheckResultEnum.UNCHECKED.getValue());
} else {
this.baseMapper.updatePlanCheckResult(planId, CheckResultEnum.ACCORD.getValue());
}
} }
set = pqDevVOList.stream().map(PqDevVO::getCheckState).collect(Collectors.toSet()); set = pqDevVOList.stream().map(PqDevVO::getCheckState).collect(Collectors.toSet());
if (set.contains(CheckStateEnum.UNCHECKED.getValue())) { if (set.contains(CheckStateEnum.UNCHECKED.getValue())) {
this.baseMapper.updatePlanTestState(planId, CheckStateEnum.CHECKING.getValue()); this.baseMapper.updatePlanTestState(planId, CheckStateEnum.CHECKING.getValue());
} else if (set.contains(CheckStateEnum.CHECKING.getValue())) {
this.baseMapper.updatePlanTestState(planId, CheckStateEnum.CHECKING.getValue());
} else {
this.baseMapper.updatePlanTestState(planId, CheckStateEnum.CHECKED.getValue());
} }
} }
} }
@@ -619,25 +693,31 @@ public class PqDevServiceImpl extends ServiceImpl<PqDevMapper, PqDev> implements
} }
@Override @Override
public boolean importDev(MultipartFile file, String patternId, String planId, HttpServletResponse response) { public HttpResult importDev(MultipartFile file, String patternId, String planId, HttpServletResponse response, Integer cover) {
DictData dictData = dictDataService.getDictDataById(patternId); DictData dictData = dictDataService.getDictDataById(patternId);
if (PatternEnum.CONTRAST.getValue().equals(dictData.getCode())) { if (PatternEnum.CONTRAST.getValue().equals(dictData.getCode())) {
return this.importContrastDev(file, patternId, planId, response); return this.importContrastDev(file, patternId, planId, response, cover);
} else { } else {
String currrentScene = sysTestConfigService.getCurrrentScene(); String currrentScene = sysTestConfigService.getCurrrentScene();
SceneEnum sceneEnum = SceneEnum.getSceneEnum(currrentScene); SceneEnum sceneEnum = SceneEnum.getSceneEnum(currrentScene);
switch (sceneEnum) { switch (sceneEnum) {
case PROVINCE_PLATFORM: case PROVINCE_PLATFORM:
return this.importProvinceDev(file, patternId, planId, response); boolean result = this.importProvinceDev(file, patternId, planId, response);
if (result) {
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, result, "导入成功");
}
case LEAVE_FACTORY_TEST: case LEAVE_FACTORY_TEST:
return this.importCNDev(file, patternId, planId, response); result = this.importCNDev(file, patternId, planId, response);
if (result) {
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, result, "导入成功");
}
case SELF_TEST: case SELF_TEST:
break; break;
default: default:
break; break;
} }
} }
return false; return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.FAIL, false, "导入失败");
} }
@Override @Override
@@ -764,21 +844,15 @@ public class PqDevServiceImpl extends ServiceImpl<PqDevMapper, PqDev> implements
* @param isExcludeSelf 是否排除自己 * @param isExcludeSelf 是否排除自己
*/ */
private void checkRepeat(PqDevParam param, boolean isExcludeSelf) { private void checkRepeat(PqDevParam param, boolean isExcludeSelf) {
QueryWrapper<PqDev> queryWrapper = new QueryWrapper<>(); LambdaQueryWrapper<PqDev> queryWrapper = new LambdaQueryWrapper<PqDev>()
queryWrapper .eq(PqDev::getState, DataStateEnum.ENABLE.getCode())
.eq("state", DataStateEnum.ENABLE.getCode()) .eq(StrUtil.isNotBlank(param.getIp()), PqDev::getIp, param.getIp())
.eq(StrUtil.isNotBlank(param.getDevType()), "Dev_Type", param.getDevType()) .eq(ObjectUtil.isNotNull(param.getPort()), PqDev::getPort, param.getPort())
.eq(StrUtil.isNotBlank(param.getPattern()), "pattern", param.getPattern()) .eq(StrUtil.isNotBlank(param.getPattern()), PqDev::getPattern, param.getPattern())
.eq(StrUtil.isNotBlank(param.getCityName()), "City_Name", param.getCityName()) .eq(StrUtil.isNotBlank(param.getCreateId()), PqDev::getCreateId, param.getCreateId());
.eq(StrUtil.isNotBlank(param.getGdName()), "Gd_Name", param.getGdName())
.eq(StrUtil.isNotBlank(param.getSubName()), "Sub_Name", param.getSubName())
.and(q -> q.eq(StrUtil.isNotBlank(param.getName()), "name", param.getName()).or()
.eq(StrUtil.isNotBlank(param.getCreateId()), "Create_Id", param.getCreateId())); //设备序列号重复
// .eq("manufacturer", param.getManufacturer())
// .eq("Dev_Type", param.getDevType()).or()
if (isExcludeSelf) { if (isExcludeSelf) {
if (param instanceof PqDevParam.UpdateParam) { if (param instanceof PqDevParam.UpdateParam) {
queryWrapper.ne("id", ((PqDevParam.UpdateParam) param).getId()); queryWrapper.ne(PqDev::getId, ((PqDevParam.UpdateParam) param).getId());
} }
} }
int count = this.count(queryWrapper); int count = this.count(queryWrapper);
@@ -1157,7 +1231,7 @@ public class PqDevServiceImpl extends ServiceImpl<PqDevMapper, PqDev> implements
} }
@Override @Override
public boolean importContrastDev(MultipartFile file, String patternId, String planId, HttpServletResponse response) { public HttpResult importContrastDev(MultipartFile file, String patternId, String planId, HttpServletResponse response, Integer cover) {
ImportParams params = new ImportParams(); ImportParams params = new ImportParams();
params.setStartSheetIndex(0); params.setStartSheetIndex(0);
params.setSheetNum(1); params.setSheetNum(1);
@@ -1177,52 +1251,56 @@ public class PqDevServiceImpl extends ServiceImpl<PqDevMapper, PqDev> implements
} catch (Exception e) { } catch (Exception e) {
throw new BusinessException(DetectionResponseEnum.IMPORT_DATA_FAIL); throw new BusinessException(DetectionResponseEnum.IMPORT_DATA_FAIL);
} }
return this.importContrastDev(contrastDevExcelList, patternId, planId); return this.importContrastDev(contrastDevExcelList, patternId, planId, cover);
} }
@Override @Override
@Transactional @Transactional
public boolean importContrastDev(List<ContrastDevExcel> contrastDevExcelList, String patternId, String planId) { public HttpResult importContrastDev(List<ContrastDevExcel> contrastDevExcelList, String patternId, String planId, Integer cover) {
if (CollUtil.isNotEmpty(contrastDevExcelList)) { if (CollUtil.isNotEmpty(contrastDevExcelList)) {
// 根据设备名称分组 // 根据设备名称分组
Map<String, List<ContrastDevExcel>> listMap = contrastDevExcelList.stream() Map<String, List<ContrastDevExcel>> listMap = contrastDevExcelList.stream()
.collect(Collectors.groupingBy(ContrastDevExcel::getName, LinkedHashMap::new, Collectors.toList())); .collect(Collectors.groupingBy(ContrastDevExcel::getName, LinkedHashMap::new, Collectors.toList()));
List<PqDev> oldDevList = new ArrayList<>(listMap.size()); List<PqDev> newDevList = new ArrayList<>(listMap.size());
List<PqMonitor> finalMonitorList = new ArrayList<>(); List<PqDev> updateDevList = new ArrayList<>();
List<PqDev> finalUpdateDevList = new ArrayList<>();
for (Map.Entry<String, List<ContrastDevExcel>> entry : listMap.entrySet()) { for (Map.Entry<String, List<ContrastDevExcel>> entry : listMap.entrySet()) {
String name = entry.getKey(); String name = entry.getKey();
List<ContrastDevExcel> devExcelList = entry.getValue(); List<ContrastDevExcel> devExcelList = entry.getValue();
// 监测点数据 // 监测点数据
List<PqMonitorExcel> pqMonitorExcelList = devExcelList.stream() List<PqMonitorExcel> pqMonitorExcelList = devExcelList.stream()
.map(ContrastDevExcel::getPqMonitorExcelList) .map(ContrastDevExcel::getPqMonitorExcelList)
.filter(Objects::nonNull) .filter(Objects::nonNull)
.flatMap(List::stream) .flatMap(List::stream)
// 过滤掉没有线路号的数据
.filter(item -> ObjectUtil.isNotNull(item.getNum()))
.collect(Collectors.toList()); .collect(Collectors.toList());
// 取第一条为设备基本信息 // 取第一条为设备基本信息
ContrastDevExcel devExcel = devExcelList.get(0); ContrastDevExcel devExcel = devExcelList.get(0);
PqDev pqDev = BeanUtil.copyProperties(devExcel, PqDev.class); PqDev importPqDev = BeanUtil.copyProperties(devExcel, PqDev.class);
if (pqDev.getEncryptionFlag() == 1) { if (importPqDev.getEncryptionFlag() == 1) {
if (StrUtil.isNotBlank(pqDev.getSeries()) && StrUtil.isNotBlank(pqDev.getDevKey())) { if (StrUtil.isNotBlank(importPqDev.getSeries()) && StrUtil.isNotBlank(importPqDev.getDevKey())) {
pqDev.setSeries(EncryptionUtil.encodeString(1, pqDev.getSeries())); importPqDev.setSeries(EncryptionUtil.encodeString(1, importPqDev.getSeries()));
pqDev.setDevKey(EncryptionUtil.encodeString(1, pqDev.getDevKey())); importPqDev.setDevKey(EncryptionUtil.encodeString(1, importPqDev.getDevKey()));
} else { } else {
throw new BusinessException(DetectionResponseEnum.SERIES_AND_DEVKEY_NOT_BLANK); throw new BusinessException(DetectionResponseEnum.SERIES_AND_DEVKEY_NOT_BLANK);
} }
} }
DevType devType = devTypeService.getByName(pqDev.getDevType()); DevType devType = devTypeService.getByName(importPqDev.getDevType());
if (ObjectUtil.isNull(devType)) { if (ObjectUtil.isNull(devType)) {
throw new BusinessException(DetectionResponseEnum.DEV_TYPE_NOT_EXIST); throw new BusinessException(DetectionResponseEnum.DEV_TYPE_NOT_EXIST);
} }
// 校验监测点数量 // 校验监测点数量
int devChns = devType.getDevChns(); int devChns = devType.getDevChns();
if (pqMonitorExcelList.size() != devChns) { // if (pqMonitorExcelList.size() != devChns) {
throw new BusinessException(DetectionResponseEnum.IMPORT_DATA_FAIL, "" + name + "】的设备类型必须具备" + devChns + "个监测点信息!"); // throw new BusinessException(DetectionResponseEnum.IMPORT_DATA_FAIL, "【" + name + "】的设备类型必须具备" + devChns + "个监测点信息!");
} // }
List<Integer> numList = pqMonitorExcelList.stream().map(PqMonitorExcel::getNum).collect(Collectors.toList()); List<Integer> numList = pqMonitorExcelList.stream().map(PqMonitorExcel::getNum).collect(Collectors.toList());
// 判断是否有重复的num // 判断是否有重复的num
Set<Integer> uniqueNumSet = new HashSet<>(numList); Set<Integer> uniqueNumSet = new HashSet<>(numList);
if (uniqueNumSet.size() != numList.size()) { if (uniqueNumSet.size() != numList.size()) {
throw new BusinessException(DetectionResponseEnum.MONITOR_NUM_REPEAT); throw new BusinessException(DetectionResponseEnum.MONITOR_NUM_REPEAT, "" + name + "】该被检设备下存在相同线路号的监测点!");
} }
Integer max = CollectionUtil.max(numList); Integer max = CollectionUtil.max(numList);
Integer min = CollectionUtil.min(numList); Integer min = CollectionUtil.min(numList);
@@ -1230,90 +1308,144 @@ public class PqDevServiceImpl extends ServiceImpl<PqDevMapper, PqDev> implements
throw new BusinessException(DetectionResponseEnum.MONITOR_NUM_OUT_OF_RANGE); throw new BusinessException(DetectionResponseEnum.MONITOR_NUM_OUT_OF_RANGE);
} }
pqDev.setDevType(devType.getId()); importPqDev.setDevType(devType.getId());
pqDev.setImportFlag(1); importPqDev.setImportFlag(1);
pqDev.setId(UUID.randomUUID().toString().replaceAll("-", "")); importPqDev.setId(UUID.randomUUID().toString().replaceAll("-", ""));
pqDev.setCreateId(pqDev.getName()); //导入时设备序列号默认与设备名称相同
List<PqMonitor> monitorList = new ArrayList<>(); if (StrUtil.isEmpty(importPqDev.getCreateId())) {
// 与设备名称相同
importPqDev.setCreateId(importPqDev.getName());
}
// 根据num排序 // 根据num排序
pqMonitorExcelList.sort(Comparator.comparingInt(PqMonitorExcel::getNum)); pqMonitorExcelList.sort(Comparator.comparingInt(PqMonitorExcel::getNum));
for (PqMonitorExcel pqMonitorExcel : pqMonitorExcelList) {
PqMonitor monitor = BeanUtil.copyProperties(pqMonitorExcel, PqMonitor.class);
monitor.setDevId(pqDev.getId());
monitorList.add(monitor);
}
StringBuilder inspectChannelBuilder = new StringBuilder(); StringBuilder inspectChannelBuilder = new StringBuilder();
for (int i = 1; i <= devChns; i++) { for (int i = 0; i < numList.size(); i++) {
inspectChannelBuilder.append(i); inspectChannelBuilder.append(numList.get(i));
if (i < devChns) { if (i < numList.size() - 1) {
inspectChannelBuilder.append(","); inspectChannelBuilder.append(StrUtil.COMMA);
} }
} }
pqDev.setInspectChannel(inspectChannelBuilder.toString()); importPqDev.setInspectChannel(inspectChannelBuilder.toString());
oldDevList.add(pqDev); List<PqMonitor> monitorList = new ArrayList<>();
finalMonitorList.addAll(monitorList); // 1、先判断是否有谐波系统ID
} boolean hasHarmSys = false;
//逆向可视化 if (StrUtil.isNotBlank(importPqDev.getHarmSysId())) {
this.reverseVisualizeProvinceDev(oldDevList, patternId); // 1.1、如果有则判断是否已存在
List<PqDev> hasList = this.lambdaQuery()
List<PqDev> newDevList = new ArrayList<>(); .eq(PqDev::getState, DataStateEnum.ENABLE.getCode())
oldDevList.forEach(pqDev -> { .eq(PqDev::getPattern, patternId)
PqDevParam param = BeanUtil.copyProperties(pqDev, PqDevParam.class); .eq(PqDev::getHarmSysId, importPqDev.getHarmSysId())
this.checkRepeat(param, false); .isNull(PqDev::getPlanId)
long count = newDevList.stream().filter(dev -> .orderByDesc(PqDev::getCreateTime).list();
dev.getDevType().equals(pqDev.getDevType()) // 1.2、 存在则放入强制更新list
&& dev.getCityName().equals(pqDev.getCityName()) if (CollUtil.isNotEmpty(hasList)) {
&& dev.getGdName().equals(pqDev.getGdName()) importPqDev.setId(hasList.get(0).getId());
&& dev.getSubName().equals(pqDev.getSubName()) for (PqMonitorExcel pqMonitorExcel : pqMonitorExcelList) {
&& (dev.getName().equals(pqDev.getName()) || dev.getCreateId().equals(pqDev.getCreateId()))) PqMonitor monitor = BeanUtil.copyProperties(pqMonitorExcel, PqMonitor.class);
.count(); monitor.setDevId(importPqDev.getId());
if (count == 0) { monitor.setPt(pqMonitorExcel.getPt1() + StrUtil.C_COLON + pqMonitorExcel.getPt2());
pqDev.setPlanId(planId); monitor.setCt(pqMonitorExcel.getCt1() + StrUtil.C_COLON + pqMonitorExcel.getCt2());
newDevList.add(pqDev); monitorList.add(monitor);
} }
}); importPqDev.setMonitorList(monitorList);
QueryWrapper<PqDev> wrapper1 = new QueryWrapper<PqDev>() finalUpdateDevList.add(importPqDev);
.eq("pq_dev.State", DataStateEnum.ENABLE.getCode()) hasHarmSys = true;
.in("pq_dev.Harm_Sys_Id", newDevList.stream().map(PqDev::getHarmSysId).collect(Collectors.toList()));
List<PqDev> oldDevList1 = this.list(wrapper1);
if (CollUtil.isNotEmpty(oldDevList1)) {
oldDevList1.stream().forEach(oldDev -> {
PqDev newDev = newDevList.stream().filter(dev -> dev.getHarmSysId().equals(oldDev.getHarmSysId())).findFirst().orElse(null);
if (ObjectUtil.isNotNull(newDev)) {
newDevList.remove(newDev);
finalMonitorList.stream()
.filter(monitor -> monitor.getDevId().equals(newDev.getId()))
.forEach(monitor -> monitor.setDevId(oldDev.getId()));
BeanUtil.copyProperties(newDev, oldDev, "id");
} }
});
this.updateBatchById(oldDevList1);
}
this.saveBatch(newDevList);
List<PqDevSub> pqDevSubList = new ArrayList<>();
for (PqDev dev : newDevList) {
PqDevSub pqDevSub = new PqDevSub();
pqDevSub.setDevId(dev.getId());
pqDevSub.setCheckState(CheckStateEnum.UNCHECKED.getValue());
pqDevSub.setCheckResult(CheckResultEnum.UNCHECKED.getValue());
pqDevSub.setReportState(DevReportStateEnum.UNCHECKED.getValue());
pqDevSub.setTimeCheckResult(TimeCheckResultEnum.UNKNOWN.getValue());
pqDevSub.setFactorCheckResult(FactorCheckResultEnum.UNKNOWN.getValue());
pqDevSubList.add(pqDevSub);
}
pqDevSubService.saveBatch(pqDevSubList);
List<String> devIdList = oldDevList1.stream().map(PqDev::getId).collect(Collectors.toList()); }
if (CollUtil.isNotEmpty(devIdList)) { if (!hasHarmSys) {
QueryWrapper<PqMonitor> wrapper = new QueryWrapper<PqMonitor>() // 2、查询 ip + 端口 + 序列号 + patternId + Plan_Id为空 是否存在
.in("pq_monitor.Dev_Id", devIdList); List<PqDev> hasList = this.lambdaQuery()
pqMonitorService.remove(wrapper); .eq(PqDev::getState, DataStateEnum.ENABLE.getCode())
.eq(PqDev::getIp, devExcel.getIp())
.eq(PqDev::getPort, devExcel.getPort())
.eq(PqDev::getPattern, patternId)
.eq(PqDev::getCreateId, devExcel.getCreateId())
.isNull(PqDev::getPlanId)
.orderByDesc(PqDev::getCreateTime).list();
// 2.1、存在则放入更新list
if (CollUtil.isNotEmpty(hasList)) {
importPqDev.setId(hasList.get(0).getId());
for (PqMonitorExcel pqMonitorExcel : pqMonitorExcelList) {
PqMonitor monitor = BeanUtil.copyProperties(pqMonitorExcel, PqMonitor.class);
monitor.setDevId(importPqDev.getId());
monitor.setPt(pqMonitorExcel.getPt1() + StrUtil.C_COLON + pqMonitorExcel.getPt2());
monitor.setCt(pqMonitorExcel.getCt1() + StrUtil.C_COLON + pqMonitorExcel.getCt2());
monitorList.add(monitor);
}
importPqDev.setMonitorList(monitorList);
updateDevList.add(importPqDev);
} else {
//2.2、不存在则放入新增list
for (PqMonitorExcel pqMonitorExcel : pqMonitorExcelList) {
PqMonitor monitor = BeanUtil.copyProperties(pqMonitorExcel, PqMonitor.class);
monitor.setDevId(importPqDev.getId());
monitor.setPt(pqMonitorExcel.getPt1() + StrUtil.C_COLON + pqMonitorExcel.getPt2());
monitor.setCt(pqMonitorExcel.getCt1() + StrUtil.C_COLON + pqMonitorExcel.getCt2());
monitorList.add(monitor);
}
importPqDev.setMonitorList(monitorList);
newDevList.add(importPqDev);
}
}
} }
pqMonitorService.reverseVisualizeMonitor(finalMonitorList); // 3、是否有重复的设备
pqMonitorService.saveBatch(finalMonitorList); if (CollUtil.isNotEmpty(updateDevList)) {
return true; // 3.1、有则让用户确认是否覆盖
if (cover == 0) {
List<String> existDevList = new ArrayList<>();
updateDevList.forEach(pqDev -> {
existDevList.add(pqDev.getName() + "(" + pqDev.getIp() + ":" + pqDev.getPort() + ")");
});
return HttpResultUtil.assembleResult(DetectionResponseEnum.DEV_IP_PORT_EXIST.getCode(), existDevList, "请确认是否覆盖");
} else {
// 3.2、确认覆盖则放入强制更新list
finalUpdateDevList.addAll(updateDevList);
}
}
// 4、新增list
if (CollUtil.isNotEmpty(newDevList)) {
//逆向可视化
this.reverseVisualizeProvinceDev(newDevList, patternId);
this.saveBatch(newDevList);
List<PqDevSub> pqDevSubList = new ArrayList<>();
List<PqMonitor> newMonitorList = new ArrayList<>();
for (PqDev dev : newDevList) {
PqDevSub pqDevSub = new PqDevSub();
pqDevSub.setDevId(dev.getId());
pqDevSub.setCheckState(CheckStateEnum.UNCHECKED.getValue());
pqDevSub.setCheckResult(CheckResultEnum.UNCHECKED.getValue());
pqDevSub.setReportState(DevReportStateEnum.UNCHECKED.getValue());
pqDevSub.setTimeCheckResult(TimeCheckResultEnum.UNKNOWN.getValue());
pqDevSub.setFactorCheckResult(FactorCheckResultEnum.UNKNOWN.getValue());
pqDevSubList.add(pqDevSub);
newMonitorList.addAll(dev.getMonitorList());
}
pqMonitorService.reverseVisualizeMonitor(newMonitorList);
pqMonitorService.saveBatch(newMonitorList);
pqDevSubService.saveBatch(pqDevSubList);
}
// 5、更新list
if (CollUtil.isNotEmpty(finalUpdateDevList)) {
//逆向可视化
this.reverseVisualizeProvinceDev(finalUpdateDevList, patternId);
this.updateBatchById(finalUpdateDevList);
List<PqMonitor> updateMonitorList = new ArrayList<>();
for (PqDev dev : finalUpdateDevList) {
updateMonitorList.addAll(dev.getMonitorList());
}
pqMonitorService.reverseVisualizeMonitor(updateMonitorList);
for (PqMonitor monitor : updateMonitorList) {
pqMonitorService.update(monitor, new LambdaUpdateWrapper<PqMonitor>().eq(PqMonitor::getDevId, monitor.getDevId()));
}
}
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, true, "导入成功");
} }
return false; return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.FAIL, false, "导入失败");
} }
@Override @Override
@@ -1371,13 +1503,19 @@ public class PqDevServiceImpl extends ServiceImpl<PqDevMapper, PqDev> implements
if (ObjectUtil.isNotNull(dictType)) { if (ObjectUtil.isNotNull(dictType)) {
dictDataList = dictDataService.getDictDataByTypeId(dictType.getId()); dictDataList = dictDataService.getDictDataByTypeId(dictType.getId());
pullDown = new PullDown(); pullDown = new PullDown();
pullDown.setFirstCol(startCol + 24); pullDown.setFirstCol(startCol + 26);
pullDown.setLastCol(startCol + 24); pullDown.setLastCol(startCol + 26);
pullDown.setStrings(dictDataList.stream().map(DictData::getName).collect(Collectors.toList())); pullDown.setStrings(dictDataList.stream().map(DictData::getName).collect(Collectors.toList()));
pullDowns.add(pullDown); pullDowns.add(pullDown);
} }
pullDown = new PullDown();
pullDown.setFirstCol(startCol + 27);
pullDown.setLastCol(startCol + 27);
pullDown.setStrings(Stream.iterate(1, x -> x + 1).limit(10).map(String::valueOf).collect(Collectors.toList()));
pullDowns.add(pullDown);
return pullDowns; return pullDowns;
} }
@@ -1439,7 +1577,22 @@ public class PqDevServiceImpl extends ServiceImpl<PqDevMapper, PqDev> implements
contrastDevExcels.forEach(contrastDevExcel -> { contrastDevExcels.forEach(contrastDevExcel -> {
List<PqMonitor> monitorList = pqMonitorService.listPqMonitorByDevIds(Collections.singletonList(contrastDevExcel.getId())); List<PqMonitor> monitorList = pqMonitorService.listPqMonitorByDevIds(Collections.singletonList(contrastDevExcel.getId()));
pqMonitorService.visualizeMonitor(monitorList); pqMonitorService.visualizeMonitor(monitorList);
List<PqMonitorExcel> pqMonitorExcelList = BeanUtil.copyToList(monitorList, PqMonitorExcel.class); List<PqMonitorExcel> pqMonitorExcelList = new ArrayList<>();
for (int i = 0; i < monitorList.size(); i++) {
PqMonitor pqMonitor = monitorList.get(i);
PqMonitorExcel pqMonitorExcel = BeanUtil.copyProperties(pqMonitor, PqMonitorExcel.class);
String pt = pqMonitor.getPt();
String[] split1 = pt.split(String.valueOf(StrUtil.C_COLON));
pqMonitorExcel.setPt1(split1[0]);
pqMonitorExcel.setPt2(split1[1]);
String ct = pqMonitor.getCt();
String[] split2 = ct.split(String.valueOf(StrUtil.C_COLON));
pqMonitorExcel.setCt1(split2[0]);
pqMonitorExcel.setCt2(split2[1]);
pqMonitorExcelList.add(pqMonitorExcel);
}
contrastDevExcel.setPqMonitorExcelList(pqMonitorExcelList); contrastDevExcel.setPqMonitorExcelList(pqMonitorExcelList);
}); });
} }
@@ -1472,4 +1625,31 @@ public class PqDevServiceImpl extends ServiceImpl<PqDevMapper, PqDev> implements
} }
return result; return result;
} }
@Override
public List<PqDevVO> listAll() {
return this.lambdaQuery()
.eq(PqDev::getState, DataStateEnum.ENABLE.getCode())
.orderByDesc(PqDev::getCreateTime)
.list().stream().map(pqDev -> {
PqDevVO pqDevVO = new PqDevVO();
BeanUtil.copyProperties(pqDev, pqDevVO);
// 解密序列号和密钥
if (StrUtil.isNotBlank(pqDevVO.getSeries())) {
pqDevVO.setSeries(EncryptionUtil.decoderString(1, pqDevVO.getSeries()));
}
if (StrUtil.isNotBlank(pqDevVO.getDevKey())) {
pqDevVO.setDevKey(EncryptionUtil.decoderString(1, pqDevVO.getDevKey()));
}
// 填充设备类型信息
DevType devType = devTypeService.getById(pqDevVO.getDevType());
if (ObjectUtil.isNotNull(devType)) {
pqDevVO.setDevType(devType.getName());
pqDevVO.setDevChns(devType.getDevChns());
pqDevVO.setDevVolt(devType.getDevVolt());
pqDevVO.setDevCurr(devType.getDevCurr());
}
return pqDevVO;
}).collect(Collectors.toList());
}
} }

View File

@@ -0,0 +1,45 @@
package com.njcn.gather.device.service.impl;
import cn.hutool.core.collection.CollectionUtil;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.njcn.gather.device.mapper.PqStandardDevGainRecordMapper;
import com.njcn.gather.device.pojo.po.PqStandardDevGainRecord;
import com.njcn.gather.device.service.IPqStandardDevGainRecordService;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import java.util.List;
/**
* @author caozehui
* @date 2026-01-15
*/
@Slf4j
@Service
@RequiredArgsConstructor
public class PqStandardDevGainRecordServiceImpl extends ServiceImpl<PqStandardDevGainRecordMapper, PqStandardDevGainRecord> implements IPqStandardDevGainRecordService {
@Override
public int getMaxNum(String stdDevMonitorId, String devMonitorId) {
return this.lambdaQuery().eq(PqStandardDevGainRecord::getStdDevMonitorId, stdDevMonitorId)
.eq(PqStandardDevGainRecord::getDevMonitorId, devMonitorId)
.list().stream().mapToInt(PqStandardDevGainRecord::getNum).max().orElse(0);
}
@Override
public boolean addOrUpdate(List<PqStandardDevGainRecord> recordList) {
recordList.forEach(record -> {
List<PqStandardDevGainRecord> list = this.lambdaQuery().eq(PqStandardDevGainRecord::getStdDevMonitorId, record.getStdDevMonitorId())
.eq(PqStandardDevGainRecord::getDevMonitorId, record.getDevMonitorId()).list();
if (CollectionUtil.isNotEmpty(list)) {
this.lambdaUpdate().eq(PqStandardDevGainRecord::getStdDevMonitorId, record.getStdDevMonitorId())
.eq(PqStandardDevGainRecord::getDevMonitorId, record.getDevMonitorId())
.set(PqStandardDevGainRecord::getNum, record.getNum()).update();
} else {
this.save(record);
}
});
return true;
}
}

View File

@@ -0,0 +1,42 @@
package com.njcn.gather.device.service.impl;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.njcn.gather.device.mapper.PqStandardDevGainMapper;
import com.njcn.gather.device.pojo.po.PqStandardDevGain;
import com.njcn.gather.device.service.IPqStandardDevGainService;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import java.util.List;
import java.util.stream.Collectors;
/**
* @author caozehui
* @date 2026-01-12
*/
@Slf4j
@Service
@RequiredArgsConstructor
public class PqStandardDevGainServiceImpl extends ServiceImpl<PqStandardDevGainMapper, PqStandardDevGain> implements IPqStandardDevGainService {
@Override
public boolean add(List<PqStandardDevGain> pqStandardDevGainList) {
log.info("add PqStandardDevGainList: {}", pqStandardDevGainList.size());
List<String> stdMonitorIdList = pqStandardDevGainList.stream().map(PqStandardDevGain::getStdDevMonitorId).distinct().collect(Collectors.toList());
List<String> existingStdMonitorIdList = this.listByIds(stdMonitorIdList).stream().map(PqStandardDevGain::getStdDevMonitorId).collect(Collectors.toList());
pqStandardDevGainList = pqStandardDevGainList.stream().filter(pqStandardDevGain -> !existingStdMonitorIdList.contains(pqStandardDevGain.getStdDevMonitorId())).collect(Collectors.toList());
return this.saveBatch(pqStandardDevGainList);
}
@Override
public PqStandardDevGain get(String stdDevMonitorId) {
return this.lambdaQuery().eq(PqStandardDevGain::getStdDevMonitorId, stdDevMonitorId).one();
}
@Override
public List<PqStandardDevGain> list(String stdDevId) {
return this.lambdaQuery().likeRight(PqStandardDevGain::getStdDevMonitorId, stdDevId).orderByAsc(PqStandardDevGain::getStdDevMonitorId).list();
}
}

View File

@@ -8,7 +8,6 @@ import cn.hutool.core.bean.BeanUtil;
import cn.hutool.core.collection.CollectionUtil; import cn.hutool.core.collection.CollectionUtil;
import cn.hutool.core.util.ObjectUtil; import cn.hutool.core.util.ObjectUtil;
import cn.hutool.core.util.StrUtil; import cn.hutool.core.util.StrUtil;
import cn.hutool.extra.spring.SpringUtil;
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper; import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
@@ -20,13 +19,13 @@ import com.njcn.common.utils.EncryptionUtil;
import com.njcn.gather.device.mapper.PqStandardDevMapper; import com.njcn.gather.device.mapper.PqStandardDevMapper;
import com.njcn.gather.device.pojo.param.PqStandardDevParam; import com.njcn.gather.device.pojo.param.PqStandardDevParam;
import com.njcn.gather.device.pojo.po.PqStandardDev; import com.njcn.gather.device.pojo.po.PqStandardDev;
import com.njcn.gather.device.pojo.po.PqStandardDevGain;
import com.njcn.gather.device.pojo.vo.PqStandardDevExcel; import com.njcn.gather.device.pojo.vo.PqStandardDevExcel;
import com.njcn.gather.device.pojo.vo.PqStandardDevVO;
import com.njcn.gather.device.pojo.vo.PreDetection; import com.njcn.gather.device.pojo.vo.PreDetection;
import com.njcn.gather.device.service.IPqStandardDevGainService;
import com.njcn.gather.device.service.IPqStandardDevService; import com.njcn.gather.device.service.IPqStandardDevService;
import com.njcn.gather.plan.mapper.AdPlanStandardDevMapper; import com.njcn.gather.plan.mapper.AdPlanStandardDevMapper;
import com.njcn.gather.plan.pojo.po.AdPlan;
import com.njcn.gather.plan.pojo.po.AdPlanStandardDev;
import com.njcn.gather.plan.service.IAdPlanService;
import com.njcn.gather.plan.service.IAdPlanStandardDevService; import com.njcn.gather.plan.service.IAdPlanStandardDevService;
import com.njcn.gather.pojo.enums.DetectionResponseEnum; import com.njcn.gather.pojo.enums.DetectionResponseEnum;
import com.njcn.gather.system.dictionary.pojo.po.DictData; import com.njcn.gather.system.dictionary.pojo.po.DictData;
@@ -62,8 +61,7 @@ public class PqStandardDevServiceImpl extends ServiceImpl<PqStandardDevMapper, P
private final IDevTypeService devTypeService; private final IDevTypeService devTypeService;
private final IDictDataService dictDataService; private final IDictDataService dictDataService;
private final IDictTypeService dictTypeService; private final IDictTypeService dictTypeService;
private final AdPlanStandardDevMapper adPlanStandardDevMapper; private final IPqStandardDevGainService pqStandardDevGainService;
private final IAdPlanStandardDevService adPlanStandardDevService;
@Override @Override
public Page<PqStandardDev> listPqStandardDevs(PqStandardDevParam.QueryParam queryParam) { public Page<PqStandardDev> listPqStandardDevs(PqStandardDevParam.QueryParam queryParam) {
@@ -71,18 +69,23 @@ public class PqStandardDevServiceImpl extends ServiceImpl<PqStandardDevMapper, P
wrapper.like(StrUtil.isNotBlank(queryParam.getName()), "name", queryParam.getName()) wrapper.like(StrUtil.isNotBlank(queryParam.getName()), "name", queryParam.getName())
.eq(StrUtil.isNotBlank(queryParam.getManufacturer()), "manufacturer", queryParam.getManufacturer()) .eq(StrUtil.isNotBlank(queryParam.getManufacturer()), "manufacturer", queryParam.getManufacturer())
.eq(StrUtil.isNotBlank(queryParam.getDevType()), "dev_type", queryParam.getDevType()) .eq(StrUtil.isNotBlank(queryParam.getDevType()), "dev_type", queryParam.getDevType())
.eq("state", DataStateEnum.ENABLE.getCode()); .eq("state", DataStateEnum.ENABLE.getCode())
.orderByDesc("create_time");
return this.page(new Page<>(PageFactory.getPageNum(queryParam), PageFactory.getPageSize(queryParam)), wrapper); return this.page(new Page<>(PageFactory.getPageNum(queryParam), PageFactory.getPageSize(queryParam)), wrapper);
} }
@Override @Override
public PqStandardDev getPqStandardDevById(String id) { public PqStandardDevVO getPqStandardDevById(String id) {
PqStandardDev standardDev = this.getById(id); PqStandardDev standardDev = this.getById(id);
PqStandardDevVO pqStandardDevVO = BeanUtil.copyProperties(standardDev, PqStandardDevVO.class);
if (standardDev.getEncryptionFlag() == 1) { if (standardDev.getEncryptionFlag() == 1) {
standardDev.setSeries(EncryptionUtil.decoderString(1, standardDev.getSeries())); pqStandardDevVO.setSeries(EncryptionUtil.decoderString(1, standardDev.getSeries()));
standardDev.setDevKey(EncryptionUtil.decoderString(1, standardDev.getDevKey())); pqStandardDevVO.setDevKey(EncryptionUtil.decoderString(1, standardDev.getDevKey()));
} }
return standardDev;
List<PqStandardDevGain> gainList = pqStandardDevGainService.list(id);
pqStandardDevVO.setGainList(gainList);
return pqStandardDevVO;
} }
@Override @Override
@@ -157,10 +160,10 @@ public class PqStandardDevServiceImpl extends ServiceImpl<PqStandardDevMapper, P
this.importData(contrastDevExcelList); this.importData(contrastDevExcelList);
} }
@Override // @Override
public List<PqStandardDev> listByPlanId(String planId) { // public List<PqStandardDev> listByPlanId(String planId) {
return adPlanStandardDevMapper.listByPlanId(Collections.singletonList(planId)); // return adPlanStandardDevMapper.listByPlanId(Collections.singletonList(planId));
} // }
@Override @Override
public List<PreDetection> listStandardDevPreDetection(List<String> ids) { public List<PreDetection> listStandardDevPreDetection(List<String> ids) {
@@ -309,34 +312,37 @@ public class PqStandardDevServiceImpl extends ServiceImpl<PqStandardDevMapper, P
@Override @Override
public List<PqStandardDev> canBindingList() { public List<PqStandardDev> canBindingList() {
List<String> excludeStandardDevIds = new ArrayList<>(); /*List<String> excludeStandardDevIds = new ArrayList<>();
// 获取所有已绑定的标准设备 // 获取所有已绑定的标准设备
List<AdPlanStandardDev> boundList = adPlanStandardDevService.list(); List<AdPlanStandardDev> boundList = adPlanStandardDevService.list();
if (CollectionUtil.isNotEmpty(boundList)) { if (CollectionUtil.isNotEmpty(boundList)) {
// 获取对应检测计划 // 获取对应检测计划
List<String> planIds = boundList.stream().map(AdPlanStandardDev::getPlanId).collect(Collectors.toList()); List<String> planIds = boundList.stream().map(AdPlanStandardDev::getPlanId).collect(Collectors.toList());
IAdPlanService adPlanService = SpringUtil.getBean(IAdPlanService.class); IAdPlanService adPlanService = SpringUtil.getBean(IAdPlanService.class);
List<AdPlan> planList = adPlanService.listByIds(planIds); List<AdPlan> planList = adPlanService.lambdaQuery().in(AdPlan::getId, planIds).eq(AdPlan::getState, DataStateEnum.ENABLE.getCode()).list();
// 区分主计划和子计划 // 区分主计划和子计划
List<AdPlan> mainPlanList = planList.stream().filter(plan -> plan.getFatherPlanId() == null).collect(Collectors.toList()); List<AdPlan> mainPlanList = planList.stream().filter(plan -> plan.getFatherPlanId().equals(CommonEnum.FATHER_ID.getValue())).collect(Collectors.toList());
List<AdPlan> subPlanList = planList.stream().filter(plan -> plan.getFatherPlanId() != null).collect(Collectors.toList()); List<AdPlan> subPlanList = planList.stream().filter(plan -> !plan.getFatherPlanId().equals(CommonEnum.FATHER_ID.getValue())).collect(Collectors.toList());
List<String> excludePlanIds = new ArrayList<>(); List<String> excludePlanIds = new ArrayList<>();
// 主计划直接排除 // 主计划未完成直接排除
if (CollectionUtil.isNotEmpty(mainPlanList)) { if (CollectionUtil.isNotEmpty(mainPlanList)) {
List<String> excludeMainPlanIds = mainPlanList.stream().filter(plan -> plan.getTestState() != 2).map(plan -> plan.getId()).collect(Collectors.toList()); List<String> excludeMainPlanIds = mainPlanList.stream().filter(plan -> !plan.getTestState().equals(CheckStateEnum.CHECKED.getValue())).map(AdPlan::getId).collect(Collectors.toList());
excludePlanIds.addAll(excludeMainPlanIds); if (CollectionUtil.isNotEmpty(excludeMainPlanIds)) {
excludePlanIds.addAll(excludeMainPlanIds);
}
} }
// 子计划需要判断其主计划, 如果主计划未完成则排除 // 子计划需要判断其主计划, 如果主计划未完成则排除
if (CollectionUtil.isNotEmpty(subPlanList)) { if (CollectionUtil.isNotEmpty(subPlanList)) {
List<String> fatherPlanIds = subPlanList.stream().map(plan -> plan.getFatherPlanId()).collect(Collectors.toList()); List<String> fatherPlanIds = subPlanList.stream().map(AdPlan::getFatherPlanId).distinct().collect(Collectors.toList());
List<AdPlan> fatherPlanList = adPlanService.listByIds(fatherPlanIds); List<AdPlan> fatherPlanList = adPlanService.listByIds(fatherPlanIds);
List<String> excludeFatherPlanIds = fatherPlanList.stream() List<String> excludeFatherPlanIds = fatherPlanList.stream()
.filter(plan -> plan.getTestState() != 2) .filter(plan -> !plan.getTestState().equals(CheckStateEnum.CHECKED.getValue()))
.map(plan -> plan.getId()).collect(Collectors.toList()); .map(AdPlan::getId).collect(Collectors.toList());
List<String> excludeSubPlanIds = subPlanList.stream() List<String> excludeSubPlanIds = subPlanList.stream()
.filter(plan -> excludeFatherPlanIds.contains(plan.getFatherPlanId())) .filter(plan -> excludeFatherPlanIds.contains(plan.getFatherPlanId()))
.map(plan -> plan.getId()).collect(Collectors.toList()); .map(AdPlan::getId).collect(Collectors.toList());
excludePlanIds.addAll(excludeSubPlanIds); excludePlanIds.addAll(excludeSubPlanIds);
} }
if (CollectionUtil.isNotEmpty(excludePlanIds)) { if (CollectionUtil.isNotEmpty(excludePlanIds)) {
@@ -345,11 +351,13 @@ public class PqStandardDevServiceImpl extends ServiceImpl<PqStandardDevMapper, P
.collect(Collectors.toList()); .collect(Collectors.toList());
excludeStandardDevIds = excludeBoundList.stream().map(AdPlanStandardDev::getStandardDevId).collect(Collectors.toList()); excludeStandardDevIds = excludeBoundList.stream().map(AdPlanStandardDev::getStandardDevId).collect(Collectors.toList());
} }
} }*/
List<PqStandardDev> list = this.lambdaQuery()
return this.lambdaQuery()
.eq(PqStandardDev::getState, DataStateEnum.ENABLE.getCode()) .eq(PqStandardDev::getState, DataStateEnum.ENABLE.getCode())
.notIn(CollectionUtil.isNotEmpty(excludeStandardDevIds), PqStandardDev::getId, excludeStandardDevIds)
.list(); .list();
/*for (PqStandardDev pqStandardDev : list) {
pqStandardDev.setDisabled(excludeStandardDevIds.contains(pqStandardDev.getId()));
}*/
return list;
} }
} }

View File

@@ -0,0 +1,11 @@
package com.njcn.gather.dip.mapper;
import com.github.yulichang.base.MPJBaseMapper;
import com.njcn.gather.dip.pojo.po.PqDipData;
/**
* @author caozehui
* @date 2026-04-09
*/
public interface PqDipDataMapper extends MPJBaseMapper<PqDipData> {
}

View File

@@ -0,0 +1,37 @@
package com.njcn.gather.dip.pojo.po;
import com.baomidou.mybatisplus.annotation.TableName;
import lombok.Data;
import java.time.LocalDateTime;
/**
* 电压暂降数据
*
* @author caozehui
* @date 2026-04-09
*/
@Data
@TableName(value = "ad_harmonic_xx")
public class PqDipData {
/**
* 主键ID
*/
private String id;
/**
* 起始时间戳
*/
private LocalDateTime startTime;
/**
* 残余电压,单位:%Ur
*/
private Double residualVoltage;
/**
* 持续时间单位ms
*/
private Integer durationMs;
}

View File

@@ -0,0 +1,26 @@
package com.njcn.gather.dip.pojo.po.vo;
import lombok.Data;
/**
* @author caozehui
* @data 2026-04-16
*/
@Data
public class DipPoint {
/**
* 残余电压,单位:%Ur
*/
private Double residualVoltage;
/**
* 持续时间单位ms
*/
private Integer durationMs;
/**
* 0为不耐受1为耐受
*/
private Boolean tolerant;
}

View File

@@ -0,0 +1,23 @@
package com.njcn.gather.dip.service;
import com.baomidou.mybatisplus.extension.service.IService;
import com.njcn.gather.dip.pojo.po.PqDipData;
import java.util.List;
/**
* @author caozehui
* @date 2026-04-09
*/
public interface IPqDipDataService extends IService<PqDipData> {
/**
* 查询指定变频器所对应的电压暂降数据
*
* @param suffix 表后缀
* @return
*/
List<PqDipData> listDipData(Integer suffix);
void clearAllData(Integer suffix);
}

View File

@@ -0,0 +1,34 @@
package com.njcn.gather.dip.service.impl;
import com.baomidou.mybatisplus.extension.conditions.query.LambdaQueryChainWrapper;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.njcn.db.mybatisplus.handler.DynamicTableNameHandler;
import com.njcn.gather.dip.mapper.PqDipDataMapper;
import com.njcn.gather.dip.pojo.po.PqDipData;
import com.njcn.gather.dip.service.IPqDipDataService;
import org.springframework.stereotype.Service;
import java.util.List;
/**
* @author caozehui
* @date 2026-04-09
*/
@Service
public class PqDipDataServiceImpl extends ServiceImpl<PqDipDataMapper, PqDipData> implements IPqDipDataService {
@Override
public List<PqDipData> listDipData(Integer suffix) {
DynamicTableNameHandler.setTableName("pq_dip_data_" + suffix);
LambdaQueryChainWrapper<PqDipData> wrapper = this.lambdaQuery().orderByAsc(PqDipData::getStartTime);
List<PqDipData> result = wrapper.list();
DynamicTableNameHandler.remove();
return result;
}
@Override
public void clearAllData(Integer suffix) {
DynamicTableNameHandler.setTableName("pq_dip_data_" + suffix);
this.remove(null);
DynamicTableNameHandler.remove();
}
}

View File

@@ -9,6 +9,7 @@ import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.njcn.common.pojo.enums.common.DataStateEnum; import com.njcn.common.pojo.enums.common.DataStateEnum;
import com.njcn.common.pojo.exception.BusinessException; import com.njcn.common.pojo.exception.BusinessException;
import com.njcn.gather.detection.pojo.enums.DetectionCodeEnum;
import com.njcn.gather.err.mapper.PqErrSysMapper; import com.njcn.gather.err.mapper.PqErrSysMapper;
import com.njcn.gather.err.pojo.param.PqErrSysDtlsParam; import com.njcn.gather.err.pojo.param.PqErrSysDtlsParam;
import com.njcn.gather.err.pojo.param.PqErrSysParam; import com.njcn.gather.err.pojo.param.PqErrSysParam;
@@ -18,6 +19,7 @@ import com.njcn.gather.err.pojo.vo.PqErrSysDtlsVO;
import com.njcn.gather.err.service.IPqErrSysDtlsService; import com.njcn.gather.err.service.IPqErrSysDtlsService;
import com.njcn.gather.err.service.IPqErrSysService; import com.njcn.gather.err.service.IPqErrSysService;
import com.njcn.gather.pojo.enums.DetectionResponseEnum; import com.njcn.gather.pojo.enums.DetectionResponseEnum;
import com.njcn.gather.report.pojo.enums.PowerIndexEnum;
import com.njcn.gather.system.dictionary.pojo.po.DictData; import com.njcn.gather.system.dictionary.pojo.po.DictData;
import com.njcn.gather.system.dictionary.pojo.po.DictTree; import com.njcn.gather.system.dictionary.pojo.po.DictTree;
import com.njcn.gather.system.dictionary.service.IDictDataService; import com.njcn.gather.system.dictionary.service.IDictDataService;
@@ -219,7 +221,9 @@ public class PqErrSysServiceImpl extends ServiceImpl<PqErrSysMapper, PqErrSys> i
List<DictTree> parentDictTreeList = dictTreeService.listByIds(pids); List<DictTree> parentDictTreeList = dictTreeService.listByIds(pids);
Map<String, String> map = new HashMap<>(); Map<String, String> map = new HashMap<>();
parentDictTreeList.forEach(dictTree -> { parentDictTreeList.forEach(dictTree -> {
map.put(dictTree.getId(), dictTree.getName()); if(!dictTree.getCode().equals(PowerIndexEnum.VOLTAGE.getKey())&&!dictTree.getCode().equals(PowerIndexEnum.P.getKey())){
map.put(dictTree.getId(), dictTree.getName());
}
}); });
return map; return map;
} }

View File

@@ -0,0 +1,21 @@
package com.njcn.gather.freqConverter.config;
import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;
/**
* @author caozehui
* @data 2026-04-15
*/
@Data
@Component
@ConfigurationProperties(prefix = "freq-converter")
public class FreqConverterConfig {
private Long schedulePeriod;
private Integer tolerant;
private Integer dt;
private Integer direction;
private Integer allowErrorDuration;
private Double allowErrorResidualVoltage;
}

View File

@@ -0,0 +1,116 @@
package com.njcn.gather.freqConverter.controller;
import cn.hutool.core.util.StrUtil;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.njcn.common.pojo.annotation.OperateInfo;
import com.njcn.common.pojo.constant.OperateType;
import com.njcn.common.pojo.enums.response.CommonResponseEnum;
import com.njcn.common.pojo.response.HttpResult;
import com.njcn.common.utils.LogUtil;
import com.njcn.gather.freqConverter.pojo.param.PqFreqConverterParam;
import com.njcn.gather.freqConverter.pojo.po.PqFreqConverterConfig;
import com.njcn.gather.freqConverter.pojo.vo.TolerantPointVO;
import com.njcn.gather.freqConverter.service.IFreqConverterService;
import com.njcn.gather.freqConverter.service.IPqFreqConverterConfigService;
import com.njcn.web.controller.BaseController;
import com.njcn.web.utils.HttpResultUtil;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiImplicitParam;
import io.swagger.annotations.ApiOperation;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.bind.annotation.*;
import java.util.List;
/**
* @author caozehui
* @data 2026-04-13
*/
@Slf4j
@Api(tags = "变频器")
@RestController
@RequestMapping("/freqConverter")
@RequiredArgsConstructor
public class FreqConverterController extends BaseController {
private final IFreqConverterService freqConverterService;
private final IPqFreqConverterConfigService pqFreqConverterConfigService;
@OperateInfo
@PostMapping("/list")
@ApiOperation("分页查询变频器列表")
@ApiImplicitParam(name = "queryParam", value = "查询参数", required = true)
public HttpResult<Page<PqFreqConverterConfig>> list(@RequestBody @Validated PqFreqConverterParam.QueryParam queryParam) {
String methodDescribe = getMethodDescribe("list");
LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, queryParam);
Page<PqFreqConverterConfig> result = pqFreqConverterConfigService.listPqFreqConverterConfigs(queryParam);
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, result, methodDescribe);
}
@OperateInfo(operateType = OperateType.ADD)
@PostMapping("/add")
@ApiOperation("新增变频器")
@ApiImplicitParam(name = "pqDevParam", value = "被检设备", required = true)
public HttpResult<Boolean> add(@RequestBody @Validated PqFreqConverterParam param) {
String methodDescribe = getMethodDescribe("add");
LogUtil.njcnDebug(log, "{},新增数据为:{}", methodDescribe, param);
boolean result = pqFreqConverterConfigService.addPqFreqConverterConfig(param);
if (result) {
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, true, methodDescribe);
} else {
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.FAIL, false, methodDescribe);
}
}
@OperateInfo(operateType = OperateType.UPDATE)
@PostMapping("/update")
@ApiOperation("修改变频器")
@ApiImplicitParam(name = "updateParam", value = "被检设备", required = true)
public HttpResult<Boolean> update(@RequestBody @Validated PqFreqConverterParam.UpdateParam param) {
String methodDescribe = getMethodDescribe("update");
LogUtil.njcnDebug(log, "{},修改数据为:{}", methodDescribe, param);
boolean result = pqFreqConverterConfigService.updatePqFreqConverterConfig(param);
if (result) {
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, true, methodDescribe);
} else {
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.FAIL, false, methodDescribe);
}
}
@OperateInfo(operateType = OperateType.DELETE)
@PostMapping("/delete")
@ApiOperation("删除变频器")
@ApiImplicitParam(name = "param", value = "删除参数", required = true)
public HttpResult<Boolean> delete(@RequestBody @Validated List<String> ids) {
String methodDescribe = getMethodDescribe("delete");
LogUtil.njcnDebug(log, "{}删除ID数据为{}", methodDescribe, String.join(StrUtil.COMMA, ids));
boolean result = pqFreqConverterConfigService.deletePqFreqConverterConfig(ids);
if (result) {
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, true, methodDescribe);
} else {
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.FAIL, false, methodDescribe);
}
}
@GetMapping("/result")
@ApiOperation("查询变频器测试结果")
@ApiImplicitParam(name = "param", value = "查询参数", required = true)
public HttpResult<List<TolerantPointVO>> result(@RequestParam("converterId") String converterId) {
String methodDescribe = getMethodDescribe("result");
LogUtil.njcnDebug(log, "{}查询ID数据为{}", methodDescribe, converterId);
List<TolerantPointVO> tolerantPoints = freqConverterService.getDipPoints(converterId);
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, tolerantPoints, methodDescribe);
}
// @GetMapping("/scurve")
// @ApiOperation("获取绘制特性曲线的point点")
// @ApiImplicitParam(name = "converterId", value = "变频器ID", required = true)
// public HttpResult<List<TolerantPointVO>> getScurvePoints(@RequestParam("converterId") String converterId) {
// String methodDescribe = getMethodDescribe("getScurvePoints");
// LogUtil.njcnDebug(log, "{}查询ID数据为{}", methodDescribe, converterId);
// List<TolerantPointVO> tolerantPoints = freqConverterService.getFeaturesScurvePoints(converterId);
// return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, tolerantPoints, methodDescribe);
// }
}

View File

@@ -0,0 +1,11 @@
package com.njcn.gather.freqConverter.mapper;
import com.github.yulichang.base.MPJBaseMapper;
import com.njcn.gather.freqConverter.pojo.po.FreqConverterStatus;
/**
* @author caozehui
* @data 2026-04-07
*/
public interface FreqConverterStatusMapper extends MPJBaseMapper<FreqConverterStatus> {
}

View File

@@ -0,0 +1,11 @@
package com.njcn.gather.freqConverter.mapper;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.njcn.gather.freqConverter.pojo.po.PqFreqConverterConfig;
/**
* @author caozehui
* @data 2026-04-08
*/
public interface PqFreqConverterConfigMapper extends BaseMapper<PqFreqConverterConfig> {
}

View File

@@ -0,0 +1,11 @@
package com.njcn.gather.freqConverter.mapper;
import com.github.yulichang.base.MPJBaseMapper;
import com.njcn.gather.freqConverter.pojo.po.PqFreqConverterTestRes;
/**
* @author caozehui
* @data 2026-04-14
*/
public interface PqFreqConverterTestResMapper extends MPJBaseMapper<PqFreqConverterTestRes> {
}

View File

@@ -0,0 +1,66 @@
package com.njcn.gather.freqConverter.pojo.param;
import com.njcn.common.pojo.constant.PatternRegex;
import com.njcn.gather.pojo.constant.DetectionValidMessage;
import com.njcn.web.pojo.param.BaseParam;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;
import lombok.EqualsAndHashCode;
import javax.validation.constraints.NotBlank;
import javax.validation.constraints.Pattern;
/**
* @author caozehui
* @data 2026-04-13
*/
@Data
public class PqFreqConverterParam {
@ApiModelProperty(value = "名称", required = true)
private String name;
@ApiModelProperty(value = "电脑串口名", required = true)
private String portName;
@ApiModelProperty(value = "变频器设置从机地址", required = true)
private Integer slaveAddress;
@ApiModelProperty(value = "变频器设置波特率", required = true)
private Integer baudRate;
@ApiModelProperty(value = "奇偶校验类型: None, Even, Odd", required = true)
private String parity;
@ApiModelProperty(value = "变频器数据位", required = true)
private Integer dataBits;
@ApiModelProperty(value = "变频器停止位,当前只支持 1 或 2", required = true)
private Integer stopBits;
@ApiModelProperty(value = "串口读写超时,单位毫秒", required = true)
private Integer timeoutMs;
private Integer testStatus;
/**
* 分页查询实体
*/
@Data
@EqualsAndHashCode(callSuper = true)
public static class QueryParam extends BaseParam {
@ApiModelProperty(value = "名称", required = true)
private String name;
}
/**
* 更新操作实体
*/
@Data
@EqualsAndHashCode(callSuper = true)
public static class UpdateParam extends PqFreqConverterParam {
@ApiModelProperty(value = "id", required = true)
@NotBlank(message = DetectionValidMessage.ID_NOT_BLANK)
@Pattern(regexp = PatternRegex.SYSTEM_ID, message = DetectionValidMessage.ID_FORMAT_ERROR)
private String id;
}
}

View File

@@ -0,0 +1,31 @@
package com.njcn.gather.freqConverter.pojo.po;
import com.baomidou.mybatisplus.annotation.TableName;
import lombok.Data;
import java.time.LocalDateTime;
/**
* @author caozehui
* @data 2026-04-07
*/
@Data
@TableName(value = "ad_harmonic_xx")
public class FreqConverterStatus {
/**
* 主键ID
*/
private String id;
private Integer slaveAddress;
private Integer statusWord1;
private String statusWord1Hex;
/**
* 状态记录时刻(时间戳)
*/
private LocalDateTime timestamp;
}

View File

@@ -0,0 +1,73 @@
package com.njcn.gather.freqConverter.pojo.po;
import com.baomidou.mybatisplus.annotation.TableName;
import com.njcn.db.mybatisplus.bo.BaseEntity;
import lombok.Data;
/**
* 变频器配置实体类
*/
@Data
@TableName("pq_freq_converter_config")
public class PqFreqConverterConfig extends BaseEntity {
/**
* 主键ID
*/
private String id;
/**
* 变频器名称
*/
private String name;
/**
* 电脑串口名如COM1
*/
private String portName;
/**
* 变频器设置从机地址范围1~127
*/
private Integer slaveAddress;
/**
* 变频器设置波特率如19200
*/
private Integer baudRate;
/**
* 奇偶校验类型: None, Even, Odd
*/
private String parity;
/**
* 变频器数据位
*/
private Integer dataBits;
/**
* 变频器停止位,当前只支持 1 或 2
*/
private Integer stopBits;
/**
* 串口读写超时,单位毫秒
*/
private Integer timeoutMs;
/**
* 数据表后缀
*/
private Integer suffix;
/**
* 检测状态0未检测1检测完成
*/
private Integer testStatus;
/**
* 状态0-删除 1-正常
*/
private Integer state;
}

View File

@@ -0,0 +1,41 @@
package com.njcn.gather.freqConverter.pojo.po;
import com.baomidou.mybatisplus.annotation.TableName;
import com.fasterxml.jackson.annotation.JsonFormat;
import lombok.Data;
import java.time.LocalDateTime;
/**
* @author caozehui
* @data 2026-04-14
*/
@Data
@TableName(value = "ad_harmonic_xx")
public class PqFreqConverterTestRes {
/**
* 主键ID
*/
private String id;
/**
* 残余电压,单位:%Ur
*/
private Double residualVoltage;
/**
* 持续时间单位ms
*/
private Integer durationMs;
/**
* 0为不耐受1为耐受2为特性曲线点
*/
private Integer tolerant;
/**
* 时间
*/
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss.SSS", timezone = "GMT+8")
private LocalDateTime time;
}

View File

@@ -0,0 +1,34 @@
package com.njcn.gather.freqConverter.pojo.vo;
import com.fasterxml.jackson.annotation.JsonFormat;
import lombok.Data;
import java.time.LocalDateTime;
/**
* @author caozehui
* @data 2026-04-13
*/
@Data
public class TolerantPointVO {
/**
* 持续时间单位ms
*/
private Integer durationMs;
/**
* 残余电压,单位:%Ur
*/
private Double residualVoltage;
/**
* 是否耐受。0-否1-是2-表示特性曲线点
*/
private Integer tolerant;
/**
* 时间
*/
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss.SSS", timezone = "GMT+8")
private LocalDateTime time;
}

View File

@@ -0,0 +1,62 @@
package com.njcn.gather.freqConverter.service;
import com.baomidou.mybatisplus.extension.service.IService;
import com.njcn.gather.freqConverter.pojo.po.FreqConverterStatus;
import com.njcn.gather.freqConverter.pojo.vo.TolerantPointVO;
import java.time.LocalDateTime;
import java.util.List;
/**
* @author caozehui
* @data 2026-04-07
*/
public interface IFreqConverterService extends IService<FreqConverterStatus> {
/**
* 保存变频器状态数据
*
* @param suffix 表后缀
* @param status 变频器状态数据
* @return 是否保存成功
*/
boolean saveFreqConverterStatus(Integer suffix, FreqConverterStatus status);
/**
* 查询指定变频器的状态历史
*
* @param converterId 变频器ID
* @return 状态数据列表
*/
List<FreqConverterStatus> listStatusData(String converterId);
/**
* 清空所有数据
*
* @param converterId 变频器ID
* @return
*/
void clearAllData(Integer converterId);
/**
* 根据设备暂降数据判断变频器是否耐受
*
* @param converterId 变频器Id
* @return 是否耐受
*/
List<TolerantPointVO> getDipPoints(String converterId);
List<FreqConverterStatus> getDipDurationStatusData(Integer suffix, LocalDateTime startTime, LocalDateTime endTime);
FreqConverterStatus getLastStatusData(Integer suffix, LocalDateTime startTime);
/**
* 获取变频器特性曲线点
*
* @param converterId 变频器ID
* @return
*/
List<TolerantPointVO> getFeaturesScurvePoints(String converterId);
}

View File

@@ -0,0 +1,27 @@
package com.njcn.gather.freqConverter.service;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.IService;
import com.njcn.gather.freqConverter.pojo.param.PqFreqConverterParam;
import com.njcn.gather.freqConverter.pojo.po.PqFreqConverterConfig;
import java.util.List;
/**
* @author caozehui
* @data 2026-04-08
*/
public interface IPqFreqConverterConfigService extends IService<PqFreqConverterConfig> {
Page<PqFreqConverterConfig> listPqFreqConverterConfigs(PqFreqConverterParam.QueryParam queryParam);
Integer getSuffix(String converterId);
boolean addPqFreqConverterConfig(PqFreqConverterParam param);
boolean updatePqFreqConverterConfig(PqFreqConverterParam.UpdateParam param);
boolean deletePqFreqConverterConfig(List<String> ids);
boolean updateTestStatus(String converterId, Integer testStatus);
}

View File

@@ -0,0 +1,50 @@
package com.njcn.gather.freqConverter.service;
import com.baomidou.mybatisplus.extension.service.IService;
import com.njcn.gather.freqConverter.pojo.po.PqFreqConverterTestRes;
import java.util.List;
/**
* @author caozehui
* @data 2026-04-14
*/
public interface IPqFreqConverterTestResService extends IService<PqFreqConverterTestRes> {
/**
* 清空所有数据
*
* @param suffix 表后缀
* @return
*/
void clearAllData(Integer suffix);
/**
* 新增结果记录
*
* @param suffix 表后缀
* @param testResList 结果数据
* @return 是否成功
*/
boolean saveTestRes(Integer suffix, List<PqFreqConverterTestRes> testResList);
/**
* 更新结果记录
*
* @param suffix 表后缀
* @param testResList 结果数据
* @return 是否成功
*/
boolean updateTestRes(Integer suffix, List<PqFreqConverterTestRes> testResList);
/**
* 查询结果记录
*
* @param suffix 表后缀
* @return 结果列表
*/
List<PqFreqConverterTestRes> listTestRes(Integer suffix);
PqFreqConverterTestRes getLastByDuration(Integer suffix, String id, Integer durationMs);
PqFreqConverterTestRes getLastByResidualVoltage(Integer suffix, String id, Double residualVoltage);
}

View File

@@ -0,0 +1,172 @@
package com.njcn.gather.freqConverter.service.impl;
import cn.hutool.core.util.IdUtil;
import com.baomidou.mybatisplus.extension.conditions.query.LambdaQueryChainWrapper;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.njcn.db.mybatisplus.handler.DynamicTableNameHandler;
import com.njcn.gather.dip.service.IPqDipDataService;
import com.njcn.gather.freqConverter.mapper.FreqConverterStatusMapper;
import com.njcn.gather.freqConverter.pojo.po.FreqConverterStatus;
import com.njcn.gather.freqConverter.pojo.po.PqFreqConverterTestRes;
import com.njcn.gather.freqConverter.pojo.vo.TolerantPointVO;
import com.njcn.gather.freqConverter.service.IFreqConverterService;
import com.njcn.gather.freqConverter.service.IPqFreqConverterConfigService;
import com.njcn.gather.freqConverter.service.IPqFreqConverterTestResService;
import lombok.AllArgsConstructor;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import java.time.LocalDateTime;
import java.util.*;
import java.util.stream.Collectors;
/**
* 变频器状态数据Service实现类
* <p>
* 实现变频器状态数据的存储、查询和清理功能。
* 当数据量超过阈值时,自动清理无效或过期数据,避免内存和数据库资源浪费。
* </p>
*
* @author CN_Gather Detection Team
* @version 1.0
* @since 2026
*/
@Service
@AllArgsConstructor
public class FreqConverterServiceImpl extends ServiceImpl<FreqConverterStatusMapper, FreqConverterStatus> implements IFreqConverterService {
private final IPqFreqConverterConfigService pqFreqConverterConfigService;
private final IPqDipDataService dipDataService;
private final IPqFreqConverterTestResService pqFreqConverterTestResService;
@Override
@Transactional(rollbackFor = Exception.class)
public boolean saveFreqConverterStatus(Integer suffix, FreqConverterStatus status) {
if (status.getId() == null) {
status.setId(IdUtil.fastSimpleUUID());
}
DynamicTableNameHandler.setTableName(PqFreqConverterConfigServiceImpl.PQ_FREQ_CONVERTER_STATUS_TB_PREFIX + suffix);
boolean result = this.save(status);
DynamicTableNameHandler.remove();
return result;
}
@Override
public List<FreqConverterStatus> listStatusData(String converterId) {
Integer suffix = pqFreqConverterConfigService.getSuffix(converterId);
DynamicTableNameHandler.setTableName(PqFreqConverterConfigServiceImpl.PQ_FREQ_CONVERTER_STATUS_TB_PREFIX + suffix);
LambdaQueryChainWrapper<FreqConverterStatus> wrapper = this.lambdaQuery().orderByAsc(FreqConverterStatus::getTimestamp);
List<FreqConverterStatus> result = wrapper.list();
DynamicTableNameHandler.remove();
return result;
}
@Override
@Transactional(rollbackFor = Exception.class)
public void clearAllData(Integer suffix) {
DynamicTableNameHandler.setTableName(PqFreqConverterConfigServiceImpl.PQ_FREQ_CONVERTER_STATUS_TB_PREFIX + suffix);
this.remove(null);
DynamicTableNameHandler.remove();
}
@Override
public List<TolerantPointVO> getDipPoints(String converterId) {
Integer suffix = pqFreqConverterConfigService.getSuffix(converterId);
DynamicTableNameHandler.setTableName(PqFreqConverterConfigServiceImpl.PQ_FREQ_CONVERTER_TEST_RES_TB_PREFIX + suffix);
List<PqFreqConverterTestRes> pqFreqConverterTestResList = pqFreqConverterTestResService.list();
DynamicTableNameHandler.remove();
List<TolerantPointVO> result = pqFreqConverterTestResList.stream().map(item -> {
TolerantPointVO tolerantPointVO = new TolerantPointVO();
tolerantPointVO.setDurationMs(item.getDurationMs());
tolerantPointVO.setResidualVoltage(item.getResidualVoltage());
tolerantPointVO.setTolerant(item.getTolerant());
tolerantPointVO.setTime(item.getTime());
return tolerantPointVO;
})
.sorted(Comparator.comparingInt(TolerantPointVO::getDurationMs))
.sorted(Comparator.comparingDouble(TolerantPointVO::getResidualVoltage)).collect(Collectors.toList());
return result;
}
@Override
public List<FreqConverterStatus> getDipDurationStatusData(Integer suffix, LocalDateTime startTime, LocalDateTime endTime) {
DynamicTableNameHandler.setTableName(PqFreqConverterConfigServiceImpl.PQ_FREQ_CONVERTER_STATUS_TB_PREFIX + suffix);
List<FreqConverterStatus> result = this.lambdaQuery()
.between(FreqConverterStatus::getTimestamp, startTime, endTime)
.orderByAsc(FreqConverterStatus::getTimestamp)
.list();
DynamicTableNameHandler.remove();
return result;
}
@Override
public FreqConverterStatus getLastStatusData(Integer suffix, LocalDateTime startTime) {
DynamicTableNameHandler.setTableName(PqFreqConverterConfigServiceImpl.PQ_FREQ_CONVERTER_STATUS_TB_PREFIX + suffix);
FreqConverterStatus one = this.lambdaQuery().le(FreqConverterStatus::getTimestamp, startTime)
.orderByDesc(FreqConverterStatus::getTimestamp)
.last("limit 1")
.one();
DynamicTableNameHandler.remove();
return one;
}
@Override
public List<TolerantPointVO> getFeaturesScurvePoints(String converterId) {
List<TolerantPointVO> dipPoints = this.getDipPoints(converterId);
if (dipPoints == null || dipPoints.isEmpty()) {
return new ArrayList<>();
}
Map<Integer, List<TolerantPointVO>> durationPointMap = dipPoints.stream()
.collect(Collectors.groupingBy(TolerantPointVO::getDurationMs, TreeMap::new, Collectors.toList()));
if (durationPointMap.size() < 2) {
return new ArrayList<>();
}
List<TolerantPointVO> result = new ArrayList<>();
TolerantPointVO firstPoint = new TolerantPointVO();
List<Integer> keyList = durationPointMap.keySet().stream().collect(Collectors.toList());
Integer i1 = keyList.get(0);
List<TolerantPointVO> tolerantPointVOS1 = durationPointMap.get(i1);
Collections.sort(tolerantPointVOS1, Comparator.comparing(TolerantPointVO::getResidualVoltage));
TolerantPointVO tolerantPointVO1 = tolerantPointVOS1.get(0);
Integer i2 = keyList.get(1);
List<TolerantPointVO> tolerantPointVOS2 = durationPointMap.get(i2);
Collections.sort(tolerantPointVOS2, Comparator.comparing(TolerantPointVO::getResidualVoltage));
TolerantPointVO tolerantPointVO2 = tolerantPointVOS2.get(0);
firstPoint.setDurationMs((i1 + i2) / 2);
firstPoint.setResidualVoltage((tolerantPointVO1.getResidualVoltage() + tolerantPointVO2.getResidualVoltage()) / 2D);
firstPoint.setTolerant(2);
result.add(firstPoint);
durationPointMap.entrySet().stream()
.forEach(entry -> {
List<TolerantPointVO> sameDurationPoints = entry.getValue().stream()
.sorted(Comparator.comparing(TolerantPointVO::getResidualVoltage, Comparator.reverseOrder()))
.collect(Collectors.toList());
for (int index = 0; index < sameDurationPoints.size() - 1; index++) {
TolerantPointVO currentPoint = sameDurationPoints.get(index);
TolerantPointVO nextPoint = sameDurationPoints.get(index + 1);
if (!Objects.equals(currentPoint.getTolerant(), nextPoint.getTolerant())) {
TolerantPointVO featurePoint = new TolerantPointVO();
featurePoint.setDurationMs(entry.getKey());
featurePoint.setResidualVoltage((currentPoint.getResidualVoltage() + nextPoint.getResidualVoltage()) / 2D);
featurePoint.setTolerant(2);
result.add(featurePoint);
break;
}
}
});
return result;
}
}

View File

@@ -0,0 +1,136 @@
package com.njcn.gather.freqConverter.service.impl;
import cn.hutool.core.bean.BeanUtil;
import cn.hutool.core.collection.CollectionUtil;
import cn.hutool.core.util.ObjectUtil;
import cn.hutool.core.util.StrUtil;
import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.njcn.common.pojo.enums.common.DataStateEnum;
import com.njcn.gather.freqConverter.mapper.PqFreqConverterConfigMapper;
import com.njcn.gather.freqConverter.pojo.param.PqFreqConverterParam;
import com.njcn.gather.freqConverter.pojo.po.PqFreqConverterConfig;
import com.njcn.gather.freqConverter.service.IPqFreqConverterConfigService;
import com.njcn.gather.storage.mapper.TableGenMapper;
import lombok.AllArgsConstructor;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import java.util.List;
/**
* @author caozehui
* @data 2026-04-08
*/
@Service
@AllArgsConstructor
public class PqFreqConverterConfigServiceImpl extends ServiceImpl<PqFreqConverterConfigMapper, PqFreqConverterConfig> implements IPqFreqConverterConfigService {
private final TableGenMapper tableGenMapper;
/**
* 表名前缀
*/
public static final String PQ_FREQ_CONVERTER_STATUS_TB_PREFIX = "pq_freq_converter_status_";
public static final String PQ_DIP_DATA_TB_PREFIX = "pq_dip_data_";
public static final String PQ_FREQ_CONVERTER_TEST_RES_TB_PREFIX = "pq_freq_converter_test_res_";
@Override
public Page<PqFreqConverterConfig> listPqFreqConverterConfigs(PqFreqConverterParam.QueryParam queryParam) {
QueryWrapper wrapper = new QueryWrapper<>();
wrapper.like(StrUtil.isNotBlank(queryParam.getName()), "name", queryParam.getName());
wrapper.eq("state", 1);
return this.page(new Page<>(queryParam.getPageNum(), queryParam.getPageSize()), wrapper);
}
@Override
public Integer getSuffix(String converterId) {
PqFreqConverterConfig freqConverterConfig = this.lambdaQuery().eq(PqFreqConverterConfig::getId, converterId).one();
return freqConverterConfig.getSuffix();
}
@Override
@Transactional(rollbackFor = Exception.class)
public boolean addPqFreqConverterConfig(PqFreqConverterParam param) {
// 建表
PqFreqConverterConfig freqConverterConfig = this.lambdaQuery().orderByDesc(PqFreqConverterConfig::getSuffix)
.last("limit 1").one();
Integer maxSuffix = 1;
if (ObjectUtil.isNotNull(freqConverterConfig)) {
maxSuffix = freqConverterConfig.getSuffix() + 1;
}
String tableSql = "CREATE TABLE `" + PQ_FREQ_CONVERTER_STATUS_TB_PREFIX + maxSuffix + "`(" +
" `id` char(32) NOT NULL COMMENT '主键ID'," +
" `slave_address` int(11) DEFAULT NULL COMMENT '从机地址'," +
" `status_word1` int(11) DEFAULT NULL COMMENT '状态字1'," +
" `status_word1_hex` varchar(20) DEFAULT NULL COMMENT '状态字1十六进制'," +
" `timestamp` datetime(3) NOT NULL COMMENT '状态记录时刻(时间戳)'," +
" PRIMARY KEY (`id`) USING BTREE" +
") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='变频器状态表';";
tableGenMapper.genTable(tableSql);
tableSql = "CREATE TABLE `" + PQ_DIP_DATA_TB_PREFIX + maxSuffix + "` (" +
" `id` char(32) NOT NULL COMMENT '主键ID'," +
" `start_time` datetime(3) NOT NULL COMMENT '起始时间戳'," +
" `residual_voltage` decimal(6,2) NOT NULL COMMENT '残余电压(%Ur)'," +
" `duration_ms` int(11) NOT NULL COMMENT '持续时间(ms)'," +
" PRIMARY KEY (`id`)" +
") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='电压暂降数据表';";
tableGenMapper.genTable(tableSql);
tableSql = "CREATE TABLE `" + PQ_FREQ_CONVERTER_TEST_RES_TB_PREFIX + maxSuffix + "` (" +
" `id` char(32) NOT NULL COMMENT '主键ID'," +
" `residual_voltage` decimal(6,2) NOT NULL COMMENT '残余电压(%Ur)'," +
" `duration_ms` int(11) NOT NULL COMMENT '持续时间(ms)'," +
" `tolerant` tinyInt(1) NOT NULL COMMENT '0为不耐受1为耐受2为特征点'," +
" `time` datetime(3) DEFAULT NULL COMMENT '时间',"+
" PRIMARY KEY (`id`)" +
") ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='变频器耐受实验结果';";
tableGenMapper.genTable(tableSql);
PqFreqConverterConfig pqFreqConverterConfig = BeanUtil.copyProperties(param, PqFreqConverterConfig.class);
pqFreqConverterConfig.setSuffix(maxSuffix);
pqFreqConverterConfig.setTestStatus(0);
pqFreqConverterConfig.setState(DataStateEnum.ENABLE.getCode());
return this.save(pqFreqConverterConfig);
}
@Override
@Transactional(rollbackFor = Exception.class)
public boolean updatePqFreqConverterConfig(PqFreqConverterParam.UpdateParam param) {
PqFreqConverterConfig pqFreqConverterConfig = BeanUtil.copyProperties(param, PqFreqConverterConfig.class);
return this.updateById(pqFreqConverterConfig);
}
@Override
@Transactional(rollbackFor = Exception.class)
public boolean deletePqFreqConverterConfig(List<String> ids) {
StringBuffer sql = new StringBuffer();
sql.append("drop table if exists ");
List<PqFreqConverterConfig> pqFreqConverterConfigs = this.listByIds(ids);
for (PqFreqConverterConfig pqFreqConverterConfig : pqFreqConverterConfigs) {
Integer suffix = pqFreqConverterConfig.getSuffix();
sql.append(PQ_FREQ_CONVERTER_STATUS_TB_PREFIX + suffix + ",")
.append(PQ_DIP_DATA_TB_PREFIX + suffix + ",")
.append(PQ_FREQ_CONVERTER_TEST_RES_TB_PREFIX + suffix + ",");
}
sql.deleteCharAt(sql.length() - 1);
// 删除表
tableGenMapper.genTable(sql.toString());
return this.lambdaUpdate()
.in(CollectionUtil.isNotEmpty(ids), PqFreqConverterConfig::getId, ids)
.set(PqFreqConverterConfig::getState, 0).update();
}
@Override
@Transactional(rollbackFor = Exception.class)
public boolean updateTestStatus(String converterId, Integer testStatus) {
return this.lambdaUpdate()
.eq(PqFreqConverterConfig::getId, converterId)
.set(PqFreqConverterConfig::getTestStatus, testStatus)
.update();
}
}

View File

@@ -0,0 +1,81 @@
package com.njcn.gather.freqConverter.service.impl;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.njcn.db.mybatisplus.handler.DynamicTableNameHandler;
import com.njcn.gather.freqConverter.config.FreqConverterConfig;
import com.njcn.gather.freqConverter.mapper.PqFreqConverterTestResMapper;
import com.njcn.gather.freqConverter.pojo.po.PqFreqConverterTestRes;
import com.njcn.gather.freqConverter.service.IPqFreqConverterTestResService;
import lombok.AllArgsConstructor;
import org.springframework.stereotype.Service;
import java.util.List;
/**
* @author caozehui
* @data 2026-04-14
*/
@Service
@AllArgsConstructor
public class PqFreqConverterTestResServiceImpl extends ServiceImpl<PqFreqConverterTestResMapper, PqFreqConverterTestRes> implements IPqFreqConverterTestResService {
private final FreqConverterConfig freqConverterConfig;
@Override
public void clearAllData(Integer suffix) {
DynamicTableNameHandler.setTableName(PqFreqConverterConfigServiceImpl.PQ_FREQ_CONVERTER_TEST_RES_TB_PREFIX + suffix);
this.remove(null);
DynamicTableNameHandler.remove();
}
@Override
public boolean saveTestRes(Integer suffix, List<PqFreqConverterTestRes> testResList) {
DynamicTableNameHandler.setTableName(PqFreqConverterConfigServiceImpl.PQ_FREQ_CONVERTER_TEST_RES_TB_PREFIX + suffix);
this.saveBatch(testResList);
DynamicTableNameHandler.remove();
return true;
}
@Override
public boolean updateTestRes(Integer suffix, List<PqFreqConverterTestRes> testResList) {
DynamicTableNameHandler.setTableName(PqFreqConverterConfigServiceImpl.PQ_FREQ_CONVERTER_TEST_RES_TB_PREFIX + suffix);
this.updateBatchById(testResList);
DynamicTableNameHandler.remove();
return true;
}
@Override
public List<PqFreqConverterTestRes> listTestRes(Integer suffix) {
DynamicTableNameHandler.setTableName(PqFreqConverterConfigServiceImpl.PQ_FREQ_CONVERTER_TEST_RES_TB_PREFIX + suffix);
List<PqFreqConverterTestRes> result = this.list();
DynamicTableNameHandler.remove();
return result;
}
@Override
public PqFreqConverterTestRes getLastByDuration(Integer suffix, String id, Integer durationMs) {
DynamicTableNameHandler.setTableName(PqFreqConverterConfigServiceImpl.PQ_FREQ_CONVERTER_TEST_RES_TB_PREFIX + suffix);
LambdaQueryWrapper<PqFreqConverterTestRes> queryWrapper = new LambdaQueryWrapper<>();
queryWrapper.between(PqFreqConverterTestRes::getDurationMs, durationMs - freqConverterConfig.getAllowErrorDuration(), durationMs + freqConverterConfig.getAllowErrorDuration())
.ne(PqFreqConverterTestRes::getId, id)
.orderByAsc(PqFreqConverterTestRes::getResidualVoltage)
.last("limit 1");
PqFreqConverterTestRes result = this.getOne(queryWrapper);
DynamicTableNameHandler.remove();
return result;
}
@Override
public PqFreqConverterTestRes getLastByResidualVoltage(Integer suffix, String id, Double residualVoltage) {
DynamicTableNameHandler.setTableName(PqFreqConverterConfigServiceImpl.PQ_FREQ_CONVERTER_TEST_RES_TB_PREFIX + suffix);
LambdaQueryWrapper<PqFreqConverterTestRes> queryWrapper = new LambdaQueryWrapper<>();
queryWrapper.between(PqFreqConverterTestRes::getResidualVoltage, residualVoltage - freqConverterConfig.getAllowErrorResidualVoltage(), residualVoltage + freqConverterConfig.getAllowErrorResidualVoltage())
.ne(PqFreqConverterTestRes::getId, id)
.orderByDesc(PqFreqConverterTestRes::getDurationMs)
.last("limit 1");
PqFreqConverterTestRes result = this.getOne(queryWrapper);
DynamicTableNameHandler.remove();
return result;
}
}

View File

@@ -60,7 +60,7 @@ public class PqIcdPathController extends BaseController {
@PostMapping("/add") @PostMapping("/add")
@ApiOperation("新增icd") @ApiOperation("新增icd")
@ApiImplicitParam(name = "param", value = "icd新增参数", required = true) @ApiImplicitParam(name = "param", value = "icd新增参数", required = true)
public HttpResult<Boolean> add(@RequestBody @Validated PqIcdPathParam param) { public HttpResult<Boolean> add(PqIcdPathParam param) {
String methodDescribe = getMethodDescribe("add"); String methodDescribe = getMethodDescribe("add");
LogUtil.njcnDebug(log, "{},新增数据为:{}", methodDescribe, param); LogUtil.njcnDebug(log, "{},新增数据为:{}", methodDescribe, param);
@@ -76,7 +76,7 @@ public class PqIcdPathController extends BaseController {
@PostMapping("/update") @PostMapping("/update")
@ApiOperation("修改icd") @ApiOperation("修改icd")
@ApiImplicitParam(name = "param", value = "icd修改参数", required = true) @ApiImplicitParam(name = "param", value = "icd修改参数", required = true)
public HttpResult<Boolean> update(@RequestBody @Validated PqIcdPathParam.UpdateParam param) { public HttpResult<Boolean> update(PqIcdPathParam.UpdateParam param) {
String methodDescribe = getMethodDescribe("update"); String methodDescribe = getMethodDescribe("update");
LogUtil.njcnDebug(log, "{},修改数据为:{}", methodDescribe, param); LogUtil.njcnDebug(log, "{},修改数据为:{}", methodDescribe, param);

View File

@@ -0,0 +1,19 @@
package com.njcn.gather.icd.pojo.enums;
/**
* @author caozehui
* @data 2025-11-11
*/
public enum IcdResponseEnum {
FILE_NOT_NULL("A018001", "映射文件不能为空"),
FILE_TYPE_ERROR("A018002", "映射文件类型错误"),
FILE_SIZE_ERROR("A018003", "映射文件大小超出限制");
private String code;
private String message;
IcdResponseEnum(String code, String message) {
this.code = code;
this.message = message;
}
}

View File

@@ -6,6 +6,7 @@ import com.njcn.web.pojo.param.BaseParam;
import io.swagger.annotations.ApiModelProperty; import io.swagger.annotations.ApiModelProperty;
import lombok.Data; import lombok.Data;
import lombok.EqualsAndHashCode; import lombok.EqualsAndHashCode;
import org.springframework.web.multipart.MultipartFile;
import javax.validation.constraints.NotBlank; import javax.validation.constraints.NotBlank;
import javax.validation.constraints.Pattern; import javax.validation.constraints.Pattern;
@@ -32,6 +33,9 @@ public class PqIcdPathParam {
@ApiModelProperty(value = "角型接线时是否使用相别的指标来进行检测0表示否1表示是", required = true) @ApiModelProperty(value = "角型接线时是否使用相别的指标来进行检测0表示否1表示是", required = true)
private Integer usePhaseIndex; private Integer usePhaseIndex;
@ApiModelProperty(value = "映射文件", required = true)
private MultipartFile mappingFile;
/** /**
* 分页查询实体 * 分页查询实体
*/ */

View File

@@ -1,5 +1,6 @@
package com.njcn.gather.icd.pojo.po; package com.njcn.gather.icd.pojo.po;
import com.baomidou.mybatisplus.annotation.TableField;
import com.baomidou.mybatisplus.annotation.TableName; import com.baomidou.mybatisplus.annotation.TableName;
import com.njcn.db.mybatisplus.bo.BaseEntity; import com.njcn.db.mybatisplus.bo.BaseEntity;
import lombok.Data; import lombok.Data;
@@ -46,5 +47,18 @@ public class PqIcdPath extends BaseEntity implements Serializable {
* 角型接线时是否使用相别的指标来进行检测0表示否1表示是 * 角型接线时是否使用相别的指标来进行检测0表示否1表示是
*/ */
private Integer usePhaseIndex; private Integer usePhaseIndex;
/**
* 映射文件路径
*/
@TableField(exist = false)
private FileVO mappingFile;
@Data
public static class FileVO{
private String name;
private String url;
}
} }

View File

@@ -1,5 +1,6 @@
package com.njcn.gather.icd.service.impl; package com.njcn.gather.icd.service.impl;
import cn.hutool.core.util.ObjectUtil;
import cn.hutool.core.util.StrUtil; import cn.hutool.core.util.StrUtil;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
@@ -7,17 +8,26 @@ import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.njcn.common.pojo.enums.common.DataStateEnum; import com.njcn.common.pojo.enums.common.DataStateEnum;
import com.njcn.common.pojo.exception.BusinessException; import com.njcn.common.pojo.exception.BusinessException;
import com.njcn.gather.icd.mapper.PqIcdPathMapper; import com.njcn.gather.icd.mapper.PqIcdPathMapper;
import com.njcn.gather.icd.pojo.enums.IcdResponseEnum;
import com.njcn.gather.icd.pojo.param.PqIcdPathParam; import com.njcn.gather.icd.pojo.param.PqIcdPathParam;
import com.njcn.gather.icd.pojo.po.PqIcdPath; import com.njcn.gather.icd.pojo.po.PqIcdPath;
import com.njcn.gather.icd.service.IPqIcdPathService; import com.njcn.gather.icd.service.IPqIcdPathService;
import com.njcn.gather.pojo.enums.DetectionResponseEnum; import com.njcn.gather.pojo.enums.DetectionResponseEnum;
import com.njcn.gather.report.pojo.enums.ReportResponseEnum;
import com.njcn.web.factory.PageFactory; import com.njcn.web.factory.PageFactory;
import lombok.RequiredArgsConstructor; import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.BeanUtils; import org.springframework.beans.BeanUtils;
import org.springframework.stereotype.Service; import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional; import org.springframework.transaction.annotation.Transactional;
import org.springframework.web.multipart.MultipartFile;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.List; import java.util.List;
/** /**
@@ -41,6 +51,13 @@ public class PqIcdPathServiceImpl extends ServiceImpl<PqIcdPathMapper, PqIcdPath
.like(StrUtil.isNotBlank(param.getName()), PqIcdPath::getName, param.getName()) .like(StrUtil.isNotBlank(param.getName()), PqIcdPath::getName, param.getName())
.orderByDesc(PqIcdPath::getCreateTime); .orderByDesc(PqIcdPath::getCreateTime);
Page<PqIcdPath> page = this.page(new Page<>(PageFactory.getPageNum(param), PageFactory.getPageSize(param)), wrapper); Page<PqIcdPath> page = this.page(new Page<>(PageFactory.getPageNum(param), PageFactory.getPageSize(param)), wrapper);
String commInstallPath = this.getCommInstallPath();
page.getRecords().forEach(pqIcdPath -> {
PqIcdPath.FileVO fileVO = new PqIcdPath.FileVO();
fileVO.setUrl(commInstallPath + "\\DeviceControl\\Config\\" + pqIcdPath.getName() + ".txt");
fileVO.setName(pqIcdPath.getName() + ".txt");
pqIcdPath.setMappingFile(fileVO);
});
return page; return page;
} }
@@ -52,9 +69,81 @@ public class PqIcdPathServiceImpl extends ServiceImpl<PqIcdPathMapper, PqIcdPath
PqIcdPath pqIcdPath = new PqIcdPath(); PqIcdPath pqIcdPath = new PqIcdPath();
BeanUtils.copyProperties(param, pqIcdPath); BeanUtils.copyProperties(param, pqIcdPath);
pqIcdPath.setState(DataStateEnum.ENABLE.getCode()); pqIcdPath.setState(DataStateEnum.ENABLE.getCode());
String commInstallPath = this.getCommInstallPath();
System.out.println("commInstallPath = " + commInstallPath);
long FILE_SIZE_LIMIT = 1 * 1024 * 1024;
MultipartFile mappingFile = param.getMappingFile();
System.out.println("mappingFile = " + ObjectUtil.isNotNull(mappingFile) + " " + !mappingFile.isEmpty());
if (ObjectUtil.isNotNull(mappingFile) && !mappingFile.isEmpty()) {
String mappingFilename = mappingFile.getOriginalFilename();
if (!mappingFilename.endsWith(".txt")) {
throw new BusinessException(IcdResponseEnum.FILE_TYPE_ERROR);
}
if (mappingFile.getSize() > FILE_SIZE_LIMIT) {
throw new BusinessException(IcdResponseEnum.FILE_SIZE_ERROR);
}
try {
// 如果文件存在,则先删除
String mappingFilePath = commInstallPath + "\\DeviceControl\\Config\\" + mappingFilename;
System.out.println("mappingFilePath = " + mappingFilePath);
Path path = Paths.get(mappingFilePath);
File file = path.toFile();
if (file.exists()) {
file.delete();
}
// 保存文件
byte[] bytes = mappingFile.getBytes();
BufferedOutputStream bufferedOutputStream = new BufferedOutputStream(new FileOutputStream(mappingFilePath));
bufferedOutputStream.write(bytes);
bufferedOutputStream.flush();
bufferedOutputStream.close();
System.out.println("File saved successfully");
} catch (IOException e) {
throw new BusinessException(ReportResponseEnum.FILE_UPLOAD_FAILED);
}
} else {
System.out.println("mappingFile is null or empty");
throw new BusinessException(IcdResponseEnum.FILE_NOT_NULL);
}
this.executeRestartCmd(commInstallPath);
return this.save(pqIcdPath); return this.save(pqIcdPath);
} }
/**
* 执行重启通讯服务脚本
*
* @param commInstallPath
*/
private void executeRestartCmd(String commInstallPath) {
// 以管理员身份运行bat脚本
String batFilePath = commInstallPath + "\\重启所有服务.bat";
try {
Runtime.getRuntime().exec(batFilePath);
} catch (IOException e) {
log.error("重启通讯服务失败", e);
}
}
private String getCommInstallPath() {
String workDir = System.getProperty("user.dir");
// String workDir = "D:\\program\\CN_Gather";
// 获取映射文件存放文件夹
String dirPath = workDir + "\\9100";
int index = workDir.indexOf("\\resources\\extraResources\\java");
if (index != -1) {
dirPath = workDir.substring(0, workDir.indexOf("\\resources\\extraResources\\java")) + "\\9100";
}
return dirPath;
}
@Override @Override
@Transactional @Transactional
public boolean updateIcd(PqIcdPathParam.UpdateParam param) { public boolean updateIcd(PqIcdPathParam.UpdateParam param) {
@@ -62,12 +151,60 @@ public class PqIcdPathServiceImpl extends ServiceImpl<PqIcdPathMapper, PqIcdPath
this.checkRepeat(param, true); this.checkRepeat(param, true);
PqIcdPath pqIcdPath = new PqIcdPath(); PqIcdPath pqIcdPath = new PqIcdPath();
BeanUtils.copyProperties(param, pqIcdPath); BeanUtils.copyProperties(param, pqIcdPath);
String commInstallPath = this.getCommInstallPath();
long FILE_SIZE_LIMIT = 1 * 1024 * 1024;
MultipartFile mappingFile = param.getMappingFile();
if (ObjectUtil.isNotNull(mappingFile) && !mappingFile.isEmpty()) {
String mappingFilename = mappingFile.getOriginalFilename();
if (!mappingFilename.endsWith(".txt")) {
throw new BusinessException(IcdResponseEnum.FILE_TYPE_ERROR);
}
if (mappingFile.getSize() > FILE_SIZE_LIMIT) {
throw new BusinessException(IcdResponseEnum.FILE_SIZE_ERROR);
}
try {
// 如果文件存在,则先删除
String mappingFilePath = commInstallPath + "\\DeviceControl\\Config\\" + mappingFilename;
Path path = Paths.get(mappingFilePath);
File file = path.toFile();
if (file.exists()) {
file.delete();
}
// 保存文件
byte[] bytes = mappingFile.getBytes();
BufferedOutputStream bufferedOutputStream = new BufferedOutputStream(new FileOutputStream(mappingFilePath));
bufferedOutputStream.write(bytes);
bufferedOutputStream.flush();
bufferedOutputStream.close();
} catch (IOException e) {
throw new BusinessException(ReportResponseEnum.FILE_UPLOAD_FAILED);
}
}
this.executeRestartCmd(commInstallPath);
return this.updateById(pqIcdPath); return this.updateById(pqIcdPath);
} }
@Override @Override
@Transactional @Transactional
public boolean deleteIcd(List<String> ids) { public boolean deleteIcd(List<String> ids) {
List<PqIcdPath> pqIcdPaths = this.listByIds(ids);
String commInstallPath = this.getCommInstallPath();
pqIcdPaths.forEach(pqIcdPath -> {
String mappingFilePath = commInstallPath + "\\DeviceControl\\Config\\" + pqIcdPath.getName() + ".txt";
Path path = Paths.get(mappingFilePath);
File file = path.toFile();
if (file.exists()) {
file.delete();
}
});
return this.lambdaUpdate().in(PqIcdPath::getId, ids).set(PqIcdPath::getState, DataStateEnum.DELETED.getCode()).update(); return this.lambdaUpdate().in(PqIcdPath::getId, ids).set(PqIcdPath::getState, DataStateEnum.DELETED.getCode()).update();
} }

View File

@@ -1,8 +1,10 @@
package com.njcn.gather.monitor.mapper; package com.njcn.gather.monitor.mapper;
import com.github.yulichang.base.MPJBaseMapper; import com.github.yulichang.base.MPJBaseMapper;
import com.njcn.gather.device.pojo.po.PqDevSub;
import com.njcn.gather.device.pojo.vo.PreDetection; import com.njcn.gather.device.pojo.vo.PreDetection;
import com.njcn.gather.monitor.pojo.po.PqMonitor; import com.njcn.gather.monitor.pojo.po.PqMonitor;
import com.njcn.gather.plan.pojo.po.AdPlan;
import org.apache.ibatis.annotations.Param; import org.apache.ibatis.annotations.Param;
import java.util.List; import java.util.List;
@@ -33,5 +35,21 @@ public interface PqMonitorMapper extends MPJBaseMapper<PqMonitor> {
PqMonitor getByDevIdAndNum(@Param("devId") String devId, @Param("num") Integer num); PqMonitor getByDevIdAndNum(@Param("devId") String devId, @Param("num") Integer num);
List<PqMonitor> listByDevIds(@Param("devIds") List<String> devIds); List<PqMonitor> listByDevIds(@Param("devIds") List<String> devIds);
void updateDeviceCheckState(@Param("devId")String devId, @Param("value") Integer value);
void updateDeviceCheckResult(@Param("devId")String devId, @Param("value") Integer value);
void updateDeviceReportRState(@Param("devId")String devId, @Param("value") Integer value);
AdPlan getPlanByDevId(@Param("devId") String devId);
List<PqDevSub> listDevSubByPlanId(@Param("planId") String planId);
void updatePlanCheckState(@Param("planId") String planId, @Param("value") Integer value);
void updatePlanCheckResult(@Param("planId") String planId, @Param("value") Integer value);
void updatePlanReportRState(@Param("planId") String planId, @Param("value") Integer value);
} }

View File

@@ -2,6 +2,7 @@
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd"> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.njcn.gather.monitor.mapper.PqMonitorMapper"> <mapper namespace="com.njcn.gather.monitor.mapper.PqMonitorMapper">
<select id="selectMonitorInfo" <select id="selectMonitorInfo"
resultType="com.njcn.gather.device.pojo.vo.PreDetection$MonitorListDTO"> resultType="com.njcn.gather.device.pojo.vo.PreDetection$MonitorListDTO">
SELECT CONCAT(pq_dev.IP, '_', Num) as lineId, SELECT CONCAT(pq_dev.IP, '_', Num) as lineId,
@@ -41,5 +42,55 @@
</foreach> </foreach>
order by Num order by Num
</select> </select>
<update id="updateDeviceCheckState">
update pq_dev_sub
set Check_State = #{value}
where dev_Id = #{devId}
</update>
<update id="updateDeviceCheckResult">
update pq_dev_sub
set Check_Result = #{value}
where dev_Id = #{devId}
</update>
<update id="updateDeviceReportRState">
update pq_dev_sub
set Report_State = #{value}
where dev_Id = #{devId}
</update>
<select id="getPlanByDevId" resultType="com.njcn.gather.plan.pojo.po.AdPlan">
select *
from ad_plan
inner join pq_dev on ad_plan.id = pq_dev.Plan_Id
where pq_dev.Id = #{devId}
</select>
<select id="listDevSubByPlanId" resultType="com.njcn.gather.device.pojo.po.PqDevSub">
select *
from pq_dev_sub
inner join pq_dev on pq_dev_sub.dev_Id = pq_dev.Id
where pq_dev.Plan_Id = #{planId}
</select>
<update id="updatePlanCheckState">
update ad_plan
set Test_State = #{value}
where id = #{planId}
</update>
<update id="updatePlanCheckResult">
update ad_plan
set Result = #{value}
where id = #{planId}
</update>
<update id="updatePlanReportRState">
update ad_plan
set Report_State = #{value}
where id = #{planId}
</update>
</mapper> </mapper>

View File

@@ -52,7 +52,7 @@ public class PqMonitorParam {
private Integer statInterval; private Integer statInterval;
@ApiModelProperty(value = "谐波系统监测点id") @ApiModelProperty(value = "谐波系统监测点id")
@NotBlank(message = DetectionValidMessage.MONITOR_ID_NOT_BLANK) // @NotBlank(message = DetectionValidMessage.MONITOR_ID_NOT_BLANK)
private String harmSysId; private String harmSysId;
@ApiModelProperty(value = "是否做检测") @ApiModelProperty(value = "是否做检测")

View File

@@ -14,11 +14,11 @@ import javax.validation.constraints.NotNull;
@Data @Data
public class PqMonitorExcel { public class PqMonitorExcel {
@Excel(name = "谐波系统监测点ID*", width = 20, orderNum = "1") @Excel(name = "谐波系统监测点ID", width = 20, orderNum = "1")
@NotBlank(message = DetectionValidMessage.MONITOR_ID_NOT_BLANK) // @NotBlank(message = DetectionValidMessage.MONITOR_ID_NOT_BLANK)
private String harmSysId; private String harmSysId;
@Excel(name = "所属母线*", width = 20, orderNum = "2") @Excel(name = "母线名称*", width = 20, orderNum = "2")
@NotBlank(message = DetectionValidMessage.BELONG_LINE_NOT_BLANK) @NotBlank(message = DetectionValidMessage.BELONG_LINE_NOT_BLANK)
private String busbar; private String busbar;
@@ -30,19 +30,27 @@ public class PqMonitorExcel {
@NotNull(message = DetectionValidMessage.MONITOR_NUM_NOT_NULL) @NotNull(message = DetectionValidMessage.MONITOR_NUM_NOT_NULL)
private Integer num; private Integer num;
@Excel(name = "PT变比(pt1:pt2)*", width = 20, orderNum = "5") @Excel(name = "PT一次变比*", width = 20, orderNum = "5")
@NotBlank(message = DetectionValidMessage.PT_NOT_BLANK) @NotBlank(message = DetectionValidMessage.PT_NOT_BLANK)
private String pt; private String pt1;
@Excel(name = "CT变比(ct1:ct2)*", width = 20, orderNum = "6") @Excel(name = "PT二次变比*", width = 20, orderNum = "6")
@NotBlank(message = DetectionValidMessage.PT_NOT_BLANK)
private String pt2;
@Excel(name = "CT一次变比*", width = 20, orderNum = "7")
@NotBlank(message = DetectionValidMessage.CT_NOT_BLANK) @NotBlank(message = DetectionValidMessage.CT_NOT_BLANK)
private String ct; private String ct1;
@Excel(name = "接线方式*", width = 20, orderNum = "7") @Excel(name = "CT二次变比*", width = 20, orderNum = "8")
@NotBlank(message = DetectionValidMessage.CT_NOT_BLANK)
private String ct2;
@Excel(name = "接线方式*", width = 20, orderNum = "9")
@NotBlank(message = DetectionValidMessage.CONNECTION_NOT_BLANK) @NotBlank(message = DetectionValidMessage.CONNECTION_NOT_BLANK)
private String connection; private String connection;
@Excel(name = "统计间隔*", width = 10, orderNum = "8") @Excel(name = "统计间隔*", width = 20, orderNum = "10")
@NotNull(message = DetectionValidMessage.STAT_INTERVAL_NOT_NULL) @NotNull(message = DetectionValidMessage.STAT_INTERVAL_NOT_NULL)
private Integer statInterval; private Integer statInterval;
} }

View File

@@ -6,7 +6,6 @@ import com.njcn.gather.monitor.pojo.po.PqMonitor;
import com.njcn.gather.plan.pojo.enums.DataSourceEnum; import com.njcn.gather.plan.pojo.enums.DataSourceEnum;
import java.util.List; import java.util.List;
import java.util.Map;
/** /**
* @author caozehui * @author caozehui
@@ -112,4 +111,18 @@ public interface IPqMonitorService extends IService<PqMonitor> {
*/ */
Integer getDevCheckResult(String devId); Integer getDevCheckResult(String devId);
/**
* 根据被检设备id删除监测点信息
*
* @param devId
* @return
*/
boolean removeByDevId(String devId);
/**
* 根据被检设备id和监测点编号获取监测点信息
* @param devId 设备id
* @param monitorNum 监测点编号
*/
PqMonitor getByDevIdAndNum(String devId, int monitorNum);
} }

View File

@@ -10,11 +10,15 @@ import com.njcn.common.pojo.exception.BusinessException;
import com.njcn.gather.detection.util.socket.CnSocketUtil; import com.njcn.gather.detection.util.socket.CnSocketUtil;
import com.njcn.gather.device.pojo.enums.CheckResultEnum; import com.njcn.gather.device.pojo.enums.CheckResultEnum;
import com.njcn.gather.device.pojo.enums.CheckStateEnum; import com.njcn.gather.device.pojo.enums.CheckStateEnum;
import com.njcn.gather.device.pojo.enums.DevReportStateEnum;
import com.njcn.gather.device.pojo.po.PqDevSub;
import com.njcn.gather.monitor.mapper.PqMonitorMapper; import com.njcn.gather.monitor.mapper.PqMonitorMapper;
import com.njcn.gather.monitor.pojo.param.PqMonitorParam; import com.njcn.gather.monitor.pojo.param.PqMonitorParam;
import com.njcn.gather.monitor.pojo.po.PqMonitor; import com.njcn.gather.monitor.pojo.po.PqMonitor;
import com.njcn.gather.monitor.service.IPqMonitorService; import com.njcn.gather.monitor.service.IPqMonitorService;
import com.njcn.gather.plan.pojo.enums.DataSourceEnum; import com.njcn.gather.plan.pojo.enums.DataSourceEnum;
import com.njcn.gather.plan.pojo.enums.PlanReportStateEnum;
import com.njcn.gather.plan.pojo.po.AdPlan;
import com.njcn.gather.pojo.enums.DetectionResponseEnum; import com.njcn.gather.pojo.enums.DetectionResponseEnum;
import com.njcn.gather.storage.service.DetectionDataDealService; import com.njcn.gather.storage.service.DetectionDataDealService;
import com.njcn.gather.storage.service.impl.DetectionDataServiceImpl; import com.njcn.gather.storage.service.impl.DetectionDataServiceImpl;
@@ -26,6 +30,7 @@ import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional; import org.springframework.transaction.annotation.Transactional;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.concurrent.atomic.AtomicReference; import java.util.concurrent.atomic.AtomicReference;
@@ -62,11 +67,110 @@ public class PqMonitorServiceImpl extends ServiceImpl<PqMonitorMapper, PqMonitor
@Transactional @Transactional
public boolean updatePqMonitorByDevId(String devId, List<PqMonitorParam> paramList) { public boolean updatePqMonitorByDevId(String devId, List<PqMonitorParam> paramList) {
// 先删除原有数据 // 先删除原有数据
this.deletePqMonitorByDevId(devId); // this.deletePqMonitorByDevId(devId);
// 再添加新数据
// 添加新数据
List<PqMonitor> pqMonitorList = BeanUtil.copyToList(paramList, PqMonitor.class); List<PqMonitor> pqMonitorList = BeanUtil.copyToList(paramList, PqMonitor.class);
pqMonitorList.forEach(pqMonitor -> pqMonitor.setDevId(devId)); pqMonitorList.forEach(pqMonitor -> pqMonitor.setDevId(devId));
return this.saveBatch(pqMonitorList);
List<PqMonitor> existedMonitorList = this.listPqMonitorByDevIds(Collections.singletonList(devId));
Map<Integer, List<PqMonitor>> map = pqMonitorList.stream().collect(Collectors.groupingBy(PqMonitor::getNum));
List<PqMonitor> newMonitorList = new ArrayList<>();
map.forEach((num, monitorList) -> {
PqMonitor pqMonitor = existedMonitorList.stream().filter(monitor -> monitor.getNum() == num).findFirst().orElse(null);
if (ObjectUtil.isNotNull(pqMonitor)) {
BeanUtil.copyProperties(monitorList.get(0), pqMonitor, "id", "realtimeResult", "statisticsResult", "recordedResult", "realtimeNum", "statisticsNum", "recordedNum", "resultType", "qualifiedNum");
newMonitorList.add(pqMonitor);
existedMonitorList.remove(pqMonitor);
} else {
newMonitorList.addAll(monitorList);
}
});
// 同步更新设备的状态
List<PqMonitor> enableCheckMonitorList = newMonitorList.stream().filter(pqMonitor -> pqMonitor.getCheckFlag() == 1).collect(Collectors.toList());
List<PqMonitor> checkedMonitorList = enableCheckMonitorList.stream().filter(pqMonitor -> ObjectUtil.isNotNull(pqMonitor.getResultType())).collect(Collectors.toList());
if (enableCheckMonitorList.size() == checkedMonitorList.size() && checkedMonitorList.size() > 0) {
this.baseMapper.updateDeviceCheckState(devId, CheckStateEnum.CHECKED.getValue());
this.baseMapper.updateDeviceReportRState(devId, DevReportStateEnum.NOT_GENERATED.getValue());
}
if (enableCheckMonitorList.size() > checkedMonitorList.size() && checkedMonitorList.size() > 0) {
this.baseMapper.updateDeviceCheckState(devId, CheckStateEnum.CHECKING.getValue());
this.baseMapper.updateDeviceReportRState(devId, DevReportStateEnum.NOT_GENERATED.getValue());
}
if (enableCheckMonitorList.size() > 0 && checkedMonitorList.size() == 0) {
this.baseMapper.updateDeviceCheckState(devId, CheckStateEnum.UNCHECKED.getValue());
}
List<Integer> monitorResultList = checkedMonitorList.stream().map(pqMonitor -> {
String resultType = pqMonitor.getResultType();
DataSourceEnum dataSourceEnum = DataSourceEnum.ofByValue(resultType);
switch (dataSourceEnum) {
case REAL_DATA:
return pqMonitor.getRealtimeResult();
case WAVE_DATA:
return pqMonitor.getRecordedResult();
case MINUTE_STATISTICS_MAX:
case MINUTE_STATISTICS_MIN:
case MINUTE_STATISTICS_AVG:
case MINUTE_STATISTICS_CP95:
return pqMonitor.getStatisticsResult();
default:
break;
}
return null;
}).collect(Collectors.toList());
long qualifiedCount = monitorResultList.stream().filter(result -> result != null && result == 1).count();
if (qualifiedCount == monitorResultList.size() && monitorResultList.size() > 0) {
this.baseMapper.updateDeviceCheckResult(devId, CheckResultEnum.ACCORD.getValue());
} else if (monitorResultList.size() > 0) {
this.baseMapper.updateDeviceCheckResult(devId, CheckResultEnum.NOT_ACCORD.getValue());
} else {
this.baseMapper.updateDeviceCheckResult(devId, CheckResultEnum.UNCHECKED.getValue());
}
// 同步更新计划的状态
AdPlan plan = this.baseMapper.getPlanByDevId(devId);
if (ObjectUtil.isNotNull(plan)) {
List<PqDevSub> devSubList = this.baseMapper.listDevSubByPlanId(plan.getId());
List<PqDevSub> checkedDevSubList = devSubList.stream().filter(pqDevSub -> pqDevSub.getCheckState() == CheckStateEnum.CHECKED.getValue()).collect(Collectors.toList());
List<PqDevSub> checkingDevSubList = devSubList.stream().filter(pqDevSub -> pqDevSub.getCheckState() == CheckStateEnum.CHECKING.getValue()).collect(Collectors.toList());
if (checkedDevSubList.size() == devSubList.size() && devSubList.size() > 0) {
this.baseMapper.updatePlanCheckState(plan.getId(), CheckStateEnum.CHECKED.getValue());
} else if (checkedDevSubList.size() > 0 || checkingDevSubList.size() > 0) {
this.baseMapper.updatePlanCheckState(plan.getId(), CheckStateEnum.CHECKING.getValue());
} else {
this.baseMapper.updatePlanCheckState(plan.getId(), CheckStateEnum.UNCHECKED.getValue());
}
List<PqDevSub> accordDevSubList = checkedDevSubList.stream().filter(pqDevSub -> pqDevSub.getCheckResult() == CheckResultEnum.ACCORD.getValue()).collect(Collectors.toList());
if (accordDevSubList.size() == checkedDevSubList.size() && checkedDevSubList.size() > 0) {
this.baseMapper.updatePlanCheckResult(plan.getId(), CheckResultEnum.ACCORD.getValue());
} else if (accordDevSubList.size() > 0) {
this.baseMapper.updatePlanCheckResult(plan.getId(), CheckResultEnum.NOT_ACCORD.getValue());
} else {
this.baseMapper.updatePlanCheckResult(plan.getId(), CheckResultEnum.UNCHECKED.getValue());
}
List<PqDevSub> generatedDevSubList = checkedDevSubList.stream().filter(pqDevSub -> pqDevSub.getReportState() == DevReportStateEnum.GENERATED.getValue()).collect(Collectors.toList());
if (generatedDevSubList.size() == checkedDevSubList.size() && checkedDevSubList.size() > 0) {
this.baseMapper.updatePlanReportRState(plan.getId(), PlanReportStateEnum.REPORT_STATE_ALL_GENERATED.getValue());
} else if (generatedDevSubList.size() > 0) {
this.baseMapper.updatePlanReportRState(plan.getId(), PlanReportStateEnum.REPORT_STATE_PARTIALLY_GENERATED.getValue());
} else {
this.baseMapper.updatePlanReportRState(plan.getId(), PlanReportStateEnum.REPORT_STATE_NOT_GENERATED.getValue());
}
}
this.removeByIds(existedMonitorList.stream().map(PqMonitor::getId).collect(Collectors.toList()));
return this.saveOrUpdateBatch(newMonitorList);
} }
@Override @Override
@@ -148,12 +252,9 @@ public class PqMonitorServiceImpl extends ServiceImpl<PqMonitorMapper, PqMonitor
Integer newMonitorResult = CheckResultEnum.UNCHECKED.getValue(); Integer newMonitorResult = CheckResultEnum.UNCHECKED.getValue();
AtomicReference<Integer> newWaveNum = new AtomicReference<>(-1); AtomicReference<Integer> newWaveNum = new AtomicReference<>(-1);
switch (dataSourceEnum) { switch (dataSourceEnum) {
case REAL_DATA:
newMonitorResult = detectionDataDealService.getMonitorResult(monitorId, adTypes, dataSourceEnum.getValue(), num, null, code);
break;
case WAVE_DATA: case WAVE_DATA:
Map<Integer, Integer> waveNumResultMap = detectionDataDealService.getWaveNumResultMap(monitorId, adTypes, num, code); Map<Integer, Integer> waveNumResultMap = detectionDataDealService.getWaveNumResultMap(monitorId, adTypes, num, code);
if(CollUtil.isEmpty(waveNumResultMap)){ if (CollUtil.isEmpty(waveNumResultMap)) {
return true; return true;
} }
waveNumResultMap.forEach((key, value) -> { waveNumResultMap.forEach((key, value) -> {
@@ -166,9 +267,19 @@ public class PqMonitorServiceImpl extends ServiceImpl<PqMonitorMapper, PqMonitor
} else { } else {
newWaveNum.set(waveNum); newWaveNum.set(waveNum);
newMonitorResult = waveNumResultMap.get(waveNum); newMonitorResult = waveNumResultMap.get(waveNum);
if (newMonitorResult == 2) {
newMonitorResult = CheckResultEnum.NOT_ACCORD.getValue();
}
} }
break; break;
default: default:
newMonitorResult = detectionDataDealService.getMonitorResult(monitorId, adTypes, dataSourceEnum.getValue(), num, null, code);
if (newMonitorResult == 2) {
newMonitorResult = CheckResultEnum.NOT_ACCORD.getValue();
}
if (newMonitorResult == 1) {
newMonitorResult = CheckResultEnum.ACCORD.getValue();
}
break; break;
} }
@@ -190,7 +301,18 @@ public class PqMonitorServiceImpl extends ServiceImpl<PqMonitorMapper, PqMonitor
if (CheckResultEnum.ACCORD.getValue().equals(oldMonitorResult)) { if (CheckResultEnum.ACCORD.getValue().equals(oldMonitorResult)) {
if (CheckResultEnum.ACCORD.getValue().equals(newMonitorResult)) { if (CheckResultEnum.ACCORD.getValue().equals(newMonitorResult)) {
qualifiedNum += 1; String oldNum = "";
if (DataSourceEnum.REAL_DATA.getValue().equals(resultType)) {
oldNum = monitor.getRealtimeNum();
} else if (DataSourceEnum.WAVE_DATA.getValue().equals(resultType)) {
oldNum = monitor.getRecordedNum();
} else {
oldNum = monitor.getStatisticsNum();
}
String[] split1 = oldNum.split(CnSocketUtil.SPLIT_TAG);
if (!split1[0].equals(num.toString())) {
qualifiedNum += 1;
}
updateFlag = true; updateFlag = true;
} }
} else { } else {
@@ -206,22 +328,28 @@ public class PqMonitorServiceImpl extends ServiceImpl<PqMonitorMapper, PqMonitor
switch (dataSourceEnum) { switch (dataSourceEnum) {
case REAL_DATA: case REAL_DATA:
monitor.setRealtimeResult(newMonitorResult);
monitor.setRealtimeNum(String.valueOf(num)); monitor.setRealtimeNum(String.valueOf(num));
monitor.setRecordedResult(null); monitor.setRealtimeResult(newMonitorResult);
monitor.setRecordedNum(null); monitor.setRecordedNum(null);
monitor.setStatisticsResult(null); monitor.setRecordedResult(null);
monitor.setStatisticsNum(null); monitor.setStatisticsNum(null);
monitor.setStatisticsResult(null);
break; break;
case WAVE_DATA: case WAVE_DATA:
monitor.setRealtimeResult(null);
monitor.setRealtimeNum(null); monitor.setRealtimeNum(null);
monitor.setRecordedResult(newMonitorResult); monitor.setRealtimeResult(null);
monitor.setRecordedNum(num + CnSocketUtil.SPLIT_TAG + newWaveNum.get()); monitor.setRecordedNum(num + CnSocketUtil.SPLIT_TAG + newWaveNum.get());
monitor.setStatisticsResult(null); monitor.setRecordedResult(newMonitorResult);
monitor.setStatisticsNum(null); monitor.setStatisticsNum(null);
monitor.setStatisticsResult(null);
break; break;
default: default:
monitor.setRealtimeResult(null);
monitor.setRealtimeNum(null);
monitor.setRecordedResult(null);
monitor.setRecordedNum(null);
monitor.setStatisticsNum(String.valueOf(num));
monitor.setStatisticsResult(newMonitorResult);
break; break;
} }
monitor.setQualifiedNum(qualifiedNum); monitor.setQualifiedNum(qualifiedNum);
@@ -265,15 +393,61 @@ public class PqMonitorServiceImpl extends ServiceImpl<PqMonitorMapper, PqMonitor
String resultType = monitor.getResultType(); String resultType = monitor.getResultType();
if (StrUtil.isNotBlank(resultType)) { if (StrUtil.isNotBlank(resultType)) {
if (DataSourceEnum.REAL_DATA.getValue().equals(resultType)) { if (DataSourceEnum.REAL_DATA.getValue().equals(resultType)) {
allResultFlags.add(monitor.getRealtimeResult()); //allResultFlags.add(monitor.getRealtimeResult());
if (monitor.getRealtimeResult() == 0) {
allResultFlags.add(2);
}
if (monitor.getRealtimeResult() == 1) {
allResultFlags.add(1);
}
if (monitor.getRealtimeResult() == 4) {
allResultFlags.add(4);
}
} else if (DataSourceEnum.WAVE_DATA.getValue().equals(resultType)) { } else if (DataSourceEnum.WAVE_DATA.getValue().equals(resultType)) {
allResultFlags.add(monitor.getRecordedResult()); //allResultFlags.add(monitor.getRecordedResult());
if (monitor.getRecordedResult() == 0) {
allResultFlags.add(2);
}
if (monitor.getRecordedResult() == 1) {
allResultFlags.add(1);
}
if (monitor.getRecordedResult() == 4) {
allResultFlags.add(4);
}
} else { } else {
allResultFlags.add(monitor.getStatisticsResult()); //allResultFlags.add(monitor.getStatisticsResult());
if (monitor.getStatisticsResult() == 0) {
allResultFlags.add(2);
}
if (monitor.getStatisticsResult() == 1) {
allResultFlags.add(1);
}
if (monitor.getStatisticsResult() == 4) {
allResultFlags.add(4);
}
} }
} }
} }
return DetectionDataServiceImpl.isResultFlag(allResultFlags); return DetectionDataServiceImpl.isResultFlag(allResultFlags);
} }
@Override
public boolean removeByDevId(String devId) {
QueryWrapper<PqMonitor> wrapper = new QueryWrapper<>();
wrapper.eq("pq_monitor.Dev_Id", devId);
return this.remove(wrapper);
}
@Override
public PqMonitor getByDevIdAndNum(String devId, int monitorNum) {
QueryWrapper<PqMonitor> wrapper = new QueryWrapper<>();
wrapper.eq("pq_monitor.Dev_Id", devId)
.eq("pq_monitor.Num", monitorNum);
List<PqMonitor> pqMonitors = this.list(wrapper);
if (CollUtil.isNotEmpty(pqMonitors)) {
return pqMonitors.get(0);
}
return null;
}
} }

View File

@@ -42,6 +42,8 @@ import org.springframework.web.bind.annotation.*;
import org.springframework.web.multipart.MultipartFile; import org.springframework.web.multipart.MultipartFile;
import javax.servlet.http.HttpServletResponse; import javax.servlet.http.HttpServletResponse;
import java.io.File;
import java.io.IOException;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
@@ -178,7 +180,7 @@ public class AdPlanController extends BaseController {
@OperateInfo(info = LogEnum.BUSINESS_COMMON) @OperateInfo(info = LogEnum.BUSINESS_COMMON)
@PostMapping("/getBigTestItem") @PostMapping("/getBigTestItem")
@ApiOperation("获取检测大项数据") @ApiOperation("获取检测大项数据")
@ApiImplicitParam(name = "id", value = "检测计划id", required = true) @ApiImplicitParam(name = "checkParam", value = "检测计划id", required = true)
public HttpResult<List<Map<String, String>>> getBigTestItem(@RequestBody AdPlanParam.CheckParam checkParam) { public HttpResult<List<Map<String, String>>> getBigTestItem(@RequestBody AdPlanParam.CheckParam checkParam) {
String methodDescribe = getMethodDescribe("getBigTestItem"); String methodDescribe = getMethodDescribe("getBigTestItem");
LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, checkParam); LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, checkParam);
@@ -189,7 +191,7 @@ public class AdPlanController extends BaseController {
@OperateInfo(info = LogEnum.BUSINESS_COMMON, operateType = OperateType.DOWNLOAD) @OperateInfo(info = LogEnum.BUSINESS_COMMON, operateType = OperateType.DOWNLOAD)
@PostMapping("/analyse") @PostMapping("/analyse")
@ApiOperation("检测数据分析") @ApiOperation("检测数据分析")
@ApiImplicitParam(name = "planId", value = "检测计划id", required = true) @ApiImplicitParam(name = "ids", value = "检测计划id", required = true)
public void analyse(@RequestBody List<String> ids) { public void analyse(@RequestBody List<String> ids) {
String methodDescribe = getMethodDescribe("analyse"); String methodDescribe = getMethodDescribe("analyse");
LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, String.join(StrUtil.COMMA, ids)); LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, String.join(StrUtil.COMMA, ids));
@@ -199,7 +201,7 @@ public class AdPlanController extends BaseController {
@OperateInfo(info = LogEnum.BUSINESS_COMMON) @OperateInfo(info = LogEnum.BUSINESS_COMMON)
@PostMapping("/listByPlanId") @PostMapping("/listByPlanId")
@ApiOperation("查询出所有已绑定的设备") @ApiOperation("查询出所有已绑定的设备")
@ApiImplicitParam(name = "planId", value = "计划id", required = true) @ApiImplicitParam(name = "param", value = "查询参数", required = true)
public HttpResult<List<PqDevVO>> listByPlanId(@RequestBody @Validated PqDevParam.QueryParam param) { public HttpResult<List<PqDevVO>> listByPlanId(@RequestBody @Validated PqDevParam.QueryParam param) {
String methodDescribe = getMethodDescribe("listByPlanId"); String methodDescribe = getMethodDescribe("listByPlanId");
LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, param); LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, param);
@@ -239,7 +241,7 @@ public class AdPlanController extends BaseController {
@OperateInfo(info = LogEnum.BUSINESS_COMMON) @OperateInfo(info = LogEnum.BUSINESS_COMMON)
@PostMapping("/listDevByPlanId") @PostMapping("/listDevByPlanId")
@ApiOperation("根据计划id分页查询被检设备") @ApiOperation("根据计划id分页查询被检设备")
@ApiImplicitParam(name = "queryParam", value = "查询参数", required = true) @ApiImplicitParam(name = "param", value = "查询参数", required = true)
public HttpResult<Page<PqDevVO>> listDevByPlanId(@RequestBody @Validated PqDevParam.QueryParam param) { public HttpResult<Page<PqDevVO>> listDevByPlanId(@RequestBody @Validated PqDevParam.QueryParam param) {
String methodDescribe = getMethodDescribe("listDevByPlanId"); String methodDescribe = getMethodDescribe("listDevByPlanId");
LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, param); LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, param);
@@ -250,6 +252,7 @@ public class AdPlanController extends BaseController {
List<String> planIdList = planList.stream().map(AdPlan::getId).collect(Collectors.toList()); List<String> planIdList = planList.stream().map(AdPlan::getId).collect(Collectors.toList());
param.getPlanIdList().addAll(planIdList); param.getPlanIdList().addAll(planIdList);
} }
param.setPlanId(plan.getId());
Page<PqDevVO> pqDevVOPage = pqDevService.listPqDevs(param); Page<PqDevVO> pqDevVOPage = pqDevService.listPqDevs(param);
List<AdPlan> planList = adPlanService.listByIds(param.getPlanIdList()); List<AdPlan> planList = adPlanService.listByIds(param.getPlanIdList());
@@ -451,7 +454,17 @@ public class AdPlanController extends BaseController {
fileTypeError.setMessage("请上传zip文件"); fileTypeError.setMessage("请上传zip文件");
throw new BusinessException(fileTypeError); throw new BusinessException(fileTypeError);
} }
asyncPlanHandler.importAndMergePlanCheckData(file, getUserId(), planId); // 创建临时文件
File tempFile = cn.hutool.core.io.FileUtil.createTempFile();
// 将MultipartFile内容写入临时文件
try {
file.transferTo(tempFile);
} catch (IOException e) {
throw new BusinessException(CommonResponseEnum.FAIL, "文件保存失败");
}
// 获取文件路径
String filePath = tempFile.getAbsolutePath();
asyncPlanHandler.importAndMergePlanCheckData(filePath, getUserId(), planId);
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, true, methodDescribe); return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, true, methodDescribe);
} }

View File

@@ -9,6 +9,7 @@ import cn.hutool.json.JSONConfig;
import cn.hutool.json.JSONUtil; import cn.hutool.json.JSONUtil;
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper; import com.baomidou.mybatisplus.core.conditions.update.LambdaUpdateWrapper;
import com.njcn.common.pojo.enums.common.DataStateEnum;
import com.njcn.common.pojo.enums.response.CommonResponseEnum; import com.njcn.common.pojo.enums.response.CommonResponseEnum;
import com.njcn.gather.detection.pojo.po.AdPair; import com.njcn.gather.detection.pojo.po.AdPair;
import com.njcn.gather.detection.service.IAdPariService; import com.njcn.gather.detection.service.IAdPariService;
@@ -36,7 +37,6 @@ import org.springframework.scheduling.annotation.Async;
import org.springframework.scheduling.annotation.EnableAsync; import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.stereotype.Component; import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.Transactional; import org.springframework.transaction.annotation.Transactional;
import org.springframework.web.multipart.MultipartFile;
import java.io.File; import java.io.File;
import java.time.Duration; import java.time.Duration;
@@ -63,6 +63,8 @@ public class AsyncPlanHandler {
private final IAdPariService adPairService; private final IAdPariService adPairService;
private final JdbcTemplate jdbcTemplate; private final JdbcTemplate jdbcTemplate;
@Value("${report.reportDir}") @Value("${report.reportDir}")
private String reportPath; private String reportPath;
@Value("${data.homeDir}") @Value("${data.homeDir}")
@@ -89,6 +91,12 @@ public class AsyncPlanHandler {
List<String> devIdList = devList.stream().map(PqDev::getId).collect(Collectors.toList()); List<String> devIdList = devList.stream().map(PqDev::getId).collect(Collectors.toList());
// 被检设备状态统计 // 被检设备状态统计
List<PqDevSub> devSubList = pqDevSubService.list(new LambdaQueryWrapper<PqDevSub>().in(PqDevSub::getDevId, devIdList)); List<PqDevSub> devSubList = pqDevSubService.list(new LambdaQueryWrapper<PqDevSub>().in(PqDevSub::getDevId, devIdList));
for (PqDevSub devSub : devSubList) {
// 不导出报告,设置报告状态为未生成
if (ObjectUtil.isNull(report) || report.equals(DataStateEnum.DELETED.getCode())) {
devSub.setReportState(DataStateEnum.DELETED.getCode());
}
}
planCheckDataVO.setDevSubList(devSubList); planCheckDataVO.setDevSubList(devSubList);
// 被检设备监测点信息 // 被检设备监测点信息
List<PqMonitor> monitorList = pqMonitorService.list(new LambdaQueryWrapper<PqMonitor>().in(PqMonitor::getDevId, devIdList)); List<PqMonitor> monitorList = pqMonitorService.list(new LambdaQueryWrapper<PqMonitor>().in(PqMonitor::getDevId, devIdList));
@@ -227,28 +235,28 @@ public class AsyncPlanHandler {
} }
@Transactional
@Async @Async
public void importAndMergePlanCheckData(MultipartFile file, String uid, String planId) { public void importAndMergePlanCheckData(String zipFilePath, String uid, String planId) {
importAndMergePlanCheckDataLogic(zipFilePath, uid, planId);
}
@Transactional
public void importAndMergePlanCheckDataLogic(String zipFilePath, String uid, String planId) {
NonWebAutoFillValueHandler.setCurrentUserId(uid); NonWebAutoFillValueHandler.setCurrentUserId(uid);
LocalDateTime startTime = LocalDateTime.now(); LocalDateTime startTime = LocalDateTime.now();
AtomicInteger progress = new AtomicInteger(0); AtomicInteger progress = new AtomicInteger(0);
AtomicInteger currentProgress = new AtomicInteger(0); AtomicInteger currentProgress = new AtomicInteger(0);
AtomicInteger dataCount = new AtomicInteger(0); AtomicInteger dataCount = new AtomicInteger(0);
try { try {
sseClient.sendMessage(uid, planId, HttpResultUtil.assembleResult(CommonResponseEnum.SUCCESS.getCode(), progress, "开始保存文件,请耐心等待..."));
// 创建临时目录用于解压文件 // 创建临时目录用于解压文件
File tempDir = FileUtil.mkdir(FileUtil.getTmpDirPath() + "import_plan_check_data_" + System.currentTimeMillis() + "/"); File tempDir = FileUtil.mkdir(FileUtil.getTmpDirPath() + "import_plan_check_data_" + System.currentTimeMillis() + "/");
// 将上传的zip文件保存到临时目录
File zipFile = FileUtil.file(tempDir, file.getOriginalFilename());
file.transferTo(zipFile);
progress.addAndGet(1); progress.addAndGet(1);
sseClient.sendMessage(uid, planId, HttpResultUtil.assembleResult(CommonResponseEnum.SUCCESS.getCode(), progress, "开始解压文件,请耐心等待...")); sseClient.sendMessage(uid, planId, HttpResultUtil.assembleResult(CommonResponseEnum.SUCCESS.getCode(), progress, "开始解压文件,请耐心等待..."));
// 解压zip文件 // 解压zip文件
File unzipDir = FileUtil.mkdir(FileUtil.file(tempDir, "unzip")); File unzipDir = FileUtil.mkdir(FileUtil.file(tempDir, "unzip"));
ZipUtil.unzip(zipFile.getAbsolutePath(), unzipDir.getAbsolutePath()); ZipUtil.unzip(zipFilePath, unzipDir.getAbsolutePath());
// 查找解压目录中的json文件 // 查找解压目录中的json文件
File[] files = unzipDir.listFiles(); File[] files = unzipDir.listFiles();
@@ -310,11 +318,13 @@ public class AsyncPlanHandler {
adPlanService.updateById(subPlan); adPlanService.updateById(subPlan);
progress.addAndGet(1); progress.addAndGet(1);
sseClient.sendMessage(uid, planId, HttpResultUtil.assembleResult(CommonResponseEnum.SUCCESS.getCode(), progress, "开始同步计划设备信息,请耐心等待...")); sseClient.sendMessage(uid, planId, HttpResultUtil.assembleResult(CommonResponseEnum.SUCCESS.getCode(), progress, "开始同步计划设备信息,请耐心等待..."));
// 更新监测点数据
// 批量更新被检设备信息 List<PqMonitor> monitorList = planCheckDataVO.getMonitorList();
pqMonitorService.updateBatchById(monitorList);
// 批量更新被检设备信息,不需要更新
// 不更新导入标志 // 不更新导入标志
devList.forEach(dev -> dev.setImportFlag(null)); /*devList.forEach(dev -> dev.setImportFlag(null));
pqDevService.updateBatchById(devList); pqDevService.updateBatchById(devList);*/
List<PqDevSub> devSubList = planCheckDataVO.getDevSubList(); List<PqDevSub> devSubList = planCheckDataVO.getDevSubList();
for (PqDevSub devSub : devSubList) { for (PqDevSub devSub : devSubList) {
@@ -325,7 +335,7 @@ public class AsyncPlanHandler {
// 同步检测数据 // 同步检测数据
List<AdPair> pairList = planCheckDataVO.getPairList(); List<AdPair> pairList = planCheckDataVO.getPairList();
adPairService.updateBatchById(pairList); adPairService.saveOrUpdateBatch(pairList);
// 主计划 // 主计划
AdPlan plan = adPlanService.getById(planId); AdPlan plan = adPlanService.getById(planId);
if (CollUtil.isNotEmpty(docxFiles)) { if (CollUtil.isNotEmpty(docxFiles)) {
@@ -447,14 +457,14 @@ public class AsyncPlanHandler {
sseClient.sendMessage(uid, planId, HttpResultUtil.assembleResult(CommonResponseEnum.SUCCESS.getCode(), progress, "数据合并完成")); sseClient.sendMessage(uid, planId, HttpResultUtil.assembleResult(CommonResponseEnum.SUCCESS.getCode(), progress, "数据合并完成"));
} catch (Exception e) { } catch (Exception e) {
log.error("导入数据失败", e); log.error("导入数据失败", e);
sseClient.sendMessage(uid, planId, HttpResultUtil.assembleResult(CommonResponseEnum.FAIL.getCode(), progress.get() + currentProgress.get(), "导入失败")); sseClient.sendMessage(uid, planId, HttpResultUtil.assembleResult(CommonResponseEnum.FAIL.getCode(), progress.get() + currentProgress.get(), "导入失败"));
} finally { } finally {
NonWebAutoFillValueHandler.clearCurrentUserId(); NonWebAutoFillValueHandler.clearCurrentUserId();
} }
FileUtil.del(zipFilePath);
sseClient.closeSse(uid); sseClient.closeSse(uid);
} }
// 构建分页查询SQL // 构建分页查询SQL

View File

@@ -193,11 +193,21 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
adPlanVO.setSourceName(pqSourceList.stream().map(PqSource::getName).collect(Collectors.toList())); adPlanVO.setSourceName(pqSourceList.stream().map(PqSource::getName).collect(Collectors.toList()));
PqDevParam.QueryParam queryParam1 = new PqDevParam.QueryParam(); PqDevParam.QueryParam queryParam1 = new PqDevParam.QueryParam();
queryParam1.setPlanIdList(Collections.singletonList(adPlan.getId())); if (adPlan.getFatherPlanId().equals(CommonEnum.FATHER_ID.getValue())) {
List<String> planIds = this.lambdaQuery()
.eq(AdPlan::getFatherPlanId, adPlan.getId())
.eq(AdPlan::getState, DataStateEnum.ENABLE.getCode())
.select(AdPlan::getId)
.list()
.stream().map(AdPlan::getId).collect(Collectors.toList());
planIds.add(adPlan.getId());
queryParam1.setPlanIdList(planIds);
} else {
queryParam1.setPlanIdList(Collections.singletonList(adPlan.getId()));
}
List<PqDevVO> pqDevVOList = pqDevMapper.selectByQueryParam(queryParam1); List<PqDevVO> pqDevVOList = pqDevMapper.selectByQueryParam(queryParam1);
if (CollUtil.isNotEmpty(pqDevVOList)) { if (CollUtil.isNotEmpty(pqDevVOList)) {
long count = pqDevVOList.stream().filter(pqDev -> CheckStateEnum.CHECKED.getValue().equals(pqDev.getCheckState())).count(); long count = pqDevVOList.stream().filter(pqDev -> CheckStateEnum.CHECKED.getValue().equals(pqDev.getCheckState()) || CheckStateEnum.DOCUMENTED.getValue().equals(pqDev.getCheckState())).count();
adPlanVO.setProgress((float) count / pqDevVOList.size()); adPlanVO.setProgress((float) count / pqDevVOList.size());
} else { } else {
adPlanVO.setProgress(0.0f); adPlanVO.setProgress(0.0f);
@@ -355,7 +365,6 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
} }
this.updateBindStandardDev(param.getId(), param.getStandardDevIds()); this.updateBindStandardDev(param.getId(), param.getStandardDevIds());
List<AdPlan> list = new ArrayList<>(); List<AdPlan> list = new ArrayList<>();
AdPlan fatherPlan = this.getById(fatherPlanId);
// 原始绑定被检设备 // 原始绑定被检设备
List<PqDev> oldDevList = pqDevService.list(new QueryWrapper<PqDev>().eq("Plan_Id", param.getId()).eq("State", DataStateEnum.ENABLE.getCode())); List<PqDev> oldDevList = pqDevService.list(new QueryWrapper<PqDev>().eq("Plan_Id", param.getId()).eq("State", DataStateEnum.ENABLE.getCode()));
List<String> oldDevIds = oldDevList.stream().map(PqDev::getId).collect(Collectors.toList()); List<String> oldDevIds = oldDevList.stream().map(PqDev::getId).collect(Collectors.toList());
@@ -371,8 +380,7 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
// 把排除绑定到主计划 // 把排除绑定到主计划
fatherNowDevIds.addAll(unionDevIds); fatherNowDevIds.addAll(unionDevIds);
fatherPlan.setTestState(pqDevService.bind(fatherPlanId, fatherNowDevIds)); pqDevService.bind(fatherPlanId, fatherNowDevIds);
list.add(fatherPlan);
} else { } else {
if (CollUtil.isNotEmpty(oldDevIds)) { if (CollUtil.isNotEmpty(oldDevIds)) {
@@ -380,10 +388,23 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
List<String> fatherNowDevIds = fatherNowDevList.stream().map(PqDev::getId).collect(Collectors.toList()); List<String> fatherNowDevIds = fatherNowDevList.stream().map(PqDev::getId).collect(Collectors.toList());
fatherNowDevIds.addAll(oldDevIds); fatherNowDevIds.addAll(oldDevIds);
// 把原始都绑定到主计划 // 把原始都绑定到主计划
fatherPlan.setTestState(pqDevService.bind(fatherPlanId, fatherNowDevIds)); pqDevService.bind(fatherPlanId, fatherNowDevIds);
list.add(fatherPlan);
} }
} }
// 非未检测状态
if (!plan1.getTestState().equals(CheckStateEnum.UNCHECKED.getValue())) {
PqDevParam.QueryParam queryParam = new PqDevParam.QueryParam();
queryParam.setPlanIdList(Collections.singletonList(plan1.getId()));
List<PqDevVO> pqDevVOS = pqDevService.listByPlanId(queryParam);
long checkedCount = pqDevVOS.stream().filter(pqDevVO -> pqDevVO.getCheckState().equals(CheckStateEnum.CHECKED.getValue())).count();
if (checkedCount == pqDevVOS.size()) {
plan1.setTestState(CheckStateEnum.CHECKED.getValue());
} else if (checkedCount > 0) {
plan1.setTestState(CheckStateEnum.CHECKING.getValue());
}
}
list.add(plan1); list.add(plan1);
return this.updateBatchById(list); return this.updateBatchById(list);
} }
@@ -510,7 +531,7 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
@Override @Override
public List<Map<String, Object>> listByPattern(String pattern) { public List<Map<String, Object>> listByPattern(String pattern) {
List<AdPlan> adPlanList = this.lambdaQuery().eq(AdPlan::getPattern, pattern).eq(AdPlan::getState, DataStateEnum.ENABLE.getCode()).list(); List<AdPlan> adPlanList = this.lambdaQuery().eq(AdPlan::getPattern, pattern).eq(AdPlan::getState, DataStateEnum.ENABLE.getCode()).orderByDesc(AdPlan::getCreateTime).list();
Map<Integer, List<AdPlan>> map1 = adPlanList.stream().collect(Collectors.groupingBy(AdPlan::getTestState)); Map<Integer, List<AdPlan>> map1 = adPlanList.stream().collect(Collectors.groupingBy(AdPlan::getTestState));
List<Map<String, Object>> result = new ArrayList<>(); List<Map<String, Object>> result = new ArrayList<>();
@@ -533,6 +554,7 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
child.put("pid", adPlan.getFatherPlanId()); child.put("pid", adPlan.getFatherPlanId());
child.put("name", adPlan.getName()); child.put("name", adPlan.getName());
child.put("timeCheck", adPlan.getTimeCheck()); child.put("timeCheck", adPlan.getTimeCheck());
child.put("dataRule", adPlan.getDataRule());
List<PqStandardDev> pqStandardDevs = adPlanStandardDevMapper.listByPlanId(Collections.singletonList(adPlan.getId())); List<PqStandardDev> pqStandardDevs = adPlanStandardDevMapper.listByPlanId(Collections.singletonList(adPlan.getId()));
List<String> devTypeIdList = pqStandardDevs.stream().map(PqStandardDev::getDevType).collect(Collectors.toList()); List<String> devTypeIdList = pqStandardDevs.stream().map(PqStandardDev::getDevType).collect(Collectors.toList());
@@ -602,16 +624,24 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
} }
DictData dictData = dictDataService.getById(patternId); DictData dictData = dictDataService.getById(patternId);
if (PatternEnum.CONTRAST.getValue().equals(dictData.getCode())) { if (PatternEnum.CONTRAST.getValue().equals(dictData.getCode())) {
String[] split = adPlan.getTestItem().split(StrUtil.COMMA); String datasourceId = adPlan.getDatasourceId();
List<DictTree> dictTreeList = dictTreeService.list(new QueryWrapper<DictTree>().in("id", split).eq("state", DataStateEnum.DELETED.getCode())); if (datasourceId.contains(DataSourceEnum.REAL_DATA.getValue())
|| datasourceId.contains(DataSourceEnum.MINUTE_STATISTICS_CP95.getValue())
|| datasourceId.contains(DataSourceEnum.MINUTE_STATISTICS_AVG.getValue())
|| datasourceId.contains(DataSourceEnum.MINUTE_STATISTICS_MAX.getValue())
|| datasourceId.contains(DataSourceEnum.MINUTE_STATISTICS_MIN.getValue())) {
String[] split = adPlan.getTestItem().split(StrUtil.COMMA);
List<DictTree> dictTreeList = dictTreeService.list(new QueryWrapper<DictTree>().in("id", split).eq("state", DataStateEnum.DELETED.getCode()).orderByAsc("sort"));
for (DictTree dictTree : dictTreeList) { for (DictTree dictTree : dictTreeList) {
Map<String, String> map = new HashMap<>(); Map<String, String> map = new HashMap<>();
map.put("id", dictTree.getId()); map.put("id", dictTree.getId());
map.put("code", dictTree.getCode()); map.put("code", dictTree.getCode());
map.put("scriptName", dictTree.getName()); map.put("scriptName", dictTree.getName());
result.add(map); result.add(map);
}
} }
if (StrUtil.isBlank(scriptType)) { if (StrUtil.isBlank(scriptType)) {
if (adPlan.getDatasourceId().contains(DataSourceEnum.WAVE_DATA.getValue())) { if (adPlan.getDatasourceId().contains(DataSourceEnum.WAVE_DATA.getValue())) {
Map<String, String> map = new HashMap<>(); Map<String, String> map = new HashMap<>();
@@ -911,8 +941,25 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
// 0-解绑、1-绑定 // 0-解绑、1-绑定
if (param.getBindFlag().equals(1)) { if (param.getBindFlag().equals(1)) {
pqDevService.lambdaUpdate().set(PqDev::getPlanId, plan.getId()).in(PqDev::getId, param.getDevIds()).update(); pqDevService.lambdaUpdate().set(PqDev::getPlanId, plan.getId()).in(PqDev::getId, param.getDevIds()).update();
// 绑定被检设备,若检测计划已检测完成,则修改为检测中
if (plan.getTestState().equals(CheckStateEnum.CHECKED.getValue())) {
plan.setTestState(CheckStateEnum.CHECKING.getValue());
this.updateById(plan);
}
} else { } else {
pqDevService.lambdaUpdate().set(PqDev::getPlanId, plan.getFatherPlanId()).in(PqDev::getId, param.getDevIds()).update(); pqDevService.lambdaUpdate().set(PqDev::getPlanId, plan.getFatherPlanId()).in(PqDev::getId, param.getDevIds()).update();
// 解绑被检设备,若检测计划检测中,且所有设备已检测完成,则修改为检测完成
if (plan.getTestState().equals(CheckStateEnum.CHECKING.getValue())) {
PqDevParam.QueryParam queryParam = new PqDevParam.QueryParam();
queryParam.setPlanIdList(Collections.singletonList(plan.getId()));
List<PqDevVO> pqDevVOS = pqDevService.listByPlanId(queryParam);
long checkedCount = pqDevVOS.stream().filter(pqDevVO -> pqDevVO.getCheckState().equals(CheckStateEnum.CHECKED.getValue())).count();
if (checkedCount == pqDevVOS.size()) {
plan.setTestState(CheckStateEnum.CHECKED.getValue());
this.updateById(plan);
}
}
} }
return true; return true;
} }
@@ -1218,7 +1265,7 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
for (int i = 1; i <= maxTime; i++) { for (int i = 1; i <= maxTime; i++) {
row2[i] = i + "次检测"; row2[i] = i + "次检测";
int tempI = i; int tempI = i;
List<PqDevVO> tempDevList = devList.stream().filter(dev -> dev.getRecheckNum() <= tempI).collect(Collectors.toList()); List<PqDevVO> tempDevList = devList.stream().filter(dev -> dev.getRecheckNum() == tempI).collect(Collectors.toList());
long passCount = tempDevList.stream().filter(dev -> dev.getCheckResult() == CheckResultEnum.ACCORD.getValue()).count(); long passCount = tempDevList.stream().filter(dev -> dev.getCheckResult() == CheckResultEnum.ACCORD.getValue()).count();
row3[i] = passCount + ""; row3[i] = passCount + "";
row4[i] = total + ""; row4[i] = total + "";
@@ -1508,8 +1555,8 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
} }
// 检测脚本 // 检测脚本
List<Map<String, Object>> maps1 = pqScriptService.listAllPqScript(null); List<PqScript> pqScriptList = pqScriptService.listAllPqScript(null);
List<String> scriptNameList = maps1.stream().map(m -> (String) m.get("name")).collect(Collectors.toList()); List<String> scriptNameList = pqScriptList.stream().map(m -> m.getName()).collect(Collectors.toList());
pullDown = new PullDown(); pullDown = new PullDown();
pullDown.setFirstCol(3); pullDown.setFirstCol(3);
pullDown.setLastCol(3); pullDown.setLastCol(3);
@@ -1675,8 +1722,8 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
String zipFileName = URLEncoder.encode(subPlan.getName() + ".zip", "UTF-8"); String zipFileName = URLEncoder.encode(subPlan.getName() + ".zip", "UTF-8");
File zipFile = FileUtil.file(tempDir, zipFileName); File zipFile = FileUtil.file(tempDir, zipFileName);
// 先将json文件添加到zip中 // 先将json文件添加到zip中使用UTF-8编码
ZipUtil.zip(jsonFile.getAbsolutePath(), zipFile.getAbsolutePath()); ZipUtil.zip(zipFile, CharsetUtil.CHARSET_UTF_8, false, jsonFile);
// 创建一个临时目录存放两个文件 // 创建一个临时目录存放两个文件
@@ -1698,8 +1745,8 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
} }
// 重新创建zip文件包含所有文件 // 重新创建zip文件包含所有文件使用UTF-8编码
ZipUtil.zip(tempZipDir.getAbsolutePath(), zipFile.getAbsolutePath()); ZipUtil.zip(zipFile, CharsetUtil.CHARSET_UTF_8, false, FileUtil.file(tempZipDir));
// 删除临时目录 // 删除临时目录
FileUtil.del(tempZipDir); FileUtil.del(tempZipDir);
@@ -1733,9 +1780,9 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
File zipFile = FileUtil.file(tempDir, file.getOriginalFilename()); File zipFile = FileUtil.file(tempDir, file.getOriginalFilename());
file.transferTo(zipFile); file.transferTo(zipFile);
// 解压zip文件 // 解压zip文件使用UTF-8编码
File unzipDir = FileUtil.mkdir(FileUtil.file(tempDir, "unzip")); File unzipDir = FileUtil.mkdir(FileUtil.file(tempDir, "unzip"));
ZipUtil.unzip(zipFile.getAbsolutePath(), unzipDir.getAbsolutePath()); ZipUtil.unzip(zipFile, unzipDir, CharsetUtil.CHARSET_UTF_8);
// 查找解压目录中的json文件 // 查找解压目录中的json文件
File[] files = unzipDir.listFiles(); File[] files = unzipDir.listFiles();
@@ -1769,7 +1816,12 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
// 更新检测计划信息 // 更新检测计划信息
plan.setImportFlag(1); plan.setImportFlag(1);
plan.setFatherPlanId(CommonEnum.FATHER_ID.getValue()); plan.setFatherPlanId(CommonEnum.FATHER_ID.getValue());
saveOrUpdate(plan); AdPlan oldPlan = this.getById(plan.getId());
// 不存在,则入库
if (oldPlan == null) {
this.save(plan);
}
List<PqDev> devList = subPlanMetaDataVO.getDevList(); List<PqDev> devList = subPlanMetaDataVO.getDevList();
List<PqStandardDev> standardDevList = subPlanMetaDataVO.getStandardDevList(); List<PqStandardDev> standardDevList = subPlanMetaDataVO.getStandardDevList();
@@ -1781,28 +1833,59 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
// 批量更新被检设备信息 // 批量更新被检设备信息
// 设置导入标志为 1 // 设置导入标志为 1
devList.forEach(dev -> dev.setImportFlag(1)); devList.forEach(dev -> dev.setImportFlag(1));
pqDevService.saveOrUpdateBatch(devList); // 排除已存在的,再入库
devList.forEach(dev -> {
pqMonitorService.saveOrUpdateBatch(dev.getMonitorList());
// 新增时默认设置
PqDevSub pqDevSub = new PqDevSub();
pqDevSub.setDevId(dev.getId());
pqDevSub.setTimeCheckResult(TimeCheckResultEnum.UNKNOWN.getValue());
pqDevSub.setFactorCheckResult(FactorCheckResultEnum.UNKNOWN.getValue());
pqDevSub.setCheckState(CheckStateEnum.UNCHECKED.getValue());
pqDevSub.setReportState(DevReportStateEnum.UNCHECKED.getValue());
pqDevSub.setCheckResult(CheckResultEnum.UNCHECKED.getValue());
Integer count = pqDevSubService.lambdaQuery().eq(PqDevSub::getDevId, dev.getId()).count();
if (count.intValue() == 0) {
pqDevSubService.save(pqDevSub);
} else {
pqDevSubService.update(pqDevSub, new LambdaUpdateWrapper<PqDevSub>().eq(PqDevSub::getDevId, dev.getId()));
}
});
List<String> devIds = devList.stream().map(PqDev::getId).collect(Collectors.toList()); List<String> devIds = devList.stream().map(PqDev::getId).collect(Collectors.toList());
// 守时检测 List<PqDevSub> pqDevSubs = pqDevSubService.lambdaQuery().in(PqDevSub::getDevId, devIds).list();
pqDevService.updatePqDevTimeCheckResult(devIds, TimeCheckResultEnum.UNKNOWN); Map<String, Integer> hadDevMaps = pqDevSubs.stream().collect(Collectors.toMap(PqDevSub::getDevId, PqDevSub::getCheckState));
List<PqDev> updateDevList = new ArrayList<>();
// 更新被检设备
for (Map.Entry<String, Integer> entry : hadDevMaps.entrySet()) {
String devId = entry.getKey();
Integer checkState = entry.getValue();
if (checkState.equals(CheckStateEnum.UNCHECKED.getValue())) {
devList.stream().filter(dev -> dev.getId().equals(devId))
.findFirst().ifPresent(dev -> updateDevList.add(dev));
}
}
if (CollUtil.isNotEmpty(updateDevList)) {
pqDevService.updateBatchById(updateDevList);
// 更新监测点数据
for (PqDev dev : updateDevList) {
pqMonitorService.saveOrUpdateBatch(dev.getMonitorList());
}
}
// 新增被检设备
List<PqDev> saveDevList = devList.stream().filter(dev -> !hadDevMaps.keySet().contains(dev.getId())).collect(Collectors.toList());
if (CollUtil.isNotEmpty(saveDevList)) {
pqDevService.saveBatch(saveDevList);
saveDevList.forEach(dev -> {
pqMonitorService.saveBatch(dev.getMonitorList());
// 新增时默认设置
PqDevSub pqDevSub = new PqDevSub();
pqDevSub.setDevId(dev.getId());
pqDevSub.setTimeCheckResult(TimeCheckResultEnum.UNKNOWN.getValue());
pqDevSub.setFactorCheckResult(FactorCheckResultEnum.UNKNOWN.getValue());
pqDevSub.setCheckState(CheckStateEnum.UNCHECKED.getValue());
pqDevSub.setReportState(DevReportStateEnum.UNCHECKED.getValue());
pqDevSub.setCheckResult(CheckResultEnum.UNCHECKED.getValue());
pqDevSubService.save(pqDevSub);
});
}
// 移除未检测的被检设备
List<String> uncheckedDevIds = pqDevSubs.stream()
.filter(pqDevSub -> pqDevSub.getCheckState().equals(CheckStateEnum.UNCHECKED.getValue()))
.map(PqDevSub::getDevId)
.collect(Collectors.toList());
List<String> removeDevIds = uncheckedDevIds.stream()
.filter(devId -> !devIds.contains(devId))
.collect(Collectors.toList());
if (CollUtil.isNotEmpty(removeDevIds)) {
pqDevService.removeByIds(removeDevIds);
pqDevSubService.remove(new LambdaQueryWrapper<PqDevSub>().in(PqDevSub::getDevId, removeDevIds));
pqMonitorService.remove(new LambdaQueryWrapper<PqMonitor>().in(PqMonitor::getDevId, removeDevIds));
}
// 批量更新标准设备信息 // 批量更新标准设备信息
pqStandardDevService.saveOrUpdateBatch(standardDevList); pqStandardDevService.saveOrUpdateBatch(standardDevList);
List<String> standardDevIds = standardDevList.stream().map(PqStandardDev::getId).collect(Collectors.toList()); List<String> standardDevIds = standardDevList.stream().map(PqStandardDev::getId).collect(Collectors.toList());
@@ -2017,8 +2100,8 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
} }
} }
// 重新创建zip文件包含所有文件 // 重新创建zip文件包含所有文件使用UTF-8编码
ZipUtil.zip(tempZipDir.getAbsolutePath(), zipFile.getAbsolutePath()); ZipUtil.zip(zipFile, CharsetUtil.CHARSET_UTF_8, false, FileUtil.file(tempZipDir));
// 删除临时目录 // 删除临时目录
FileUtil.del(tempZipDir); FileUtil.del(tempZipDir);
@@ -2045,6 +2128,7 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
/** /**
* 比对模式下计划的检测项获取 * 比对模式下计划的检测项获取
*
* @param planId 计划ID * @param planId 计划ID
* @return 检测项 * @return 检测项
*/ */
@@ -2055,7 +2139,7 @@ public class AdPlanServiceImpl extends ServiceImpl<AdPlanMapper, AdPlan> impleme
String pattern = adPlan.getPattern(); String pattern = adPlan.getPattern();
DictData dictData = dictDataService.getDictDataById(pattern); DictData dictData = dictDataService.getDictDataById(pattern);
if (ObjectUtil.isNotNull(dictData)) { if (ObjectUtil.isNotNull(dictData)) {
if(dictData.getCode().equalsIgnoreCase("Contrast")){ if (dictData.getCode().equalsIgnoreCase("Contrast")) {
String[] items = adPlan.getTestItem().split(","); String[] items = adPlan.getTestItem().split(",");
scriptList = new ArrayList<>(Arrays.asList(items)); scriptList = new ArrayList<>(Arrays.asList(items));
} }

View File

@@ -73,8 +73,9 @@ public enum DetectionResponseEnum {
MONITOR_NUM_REPEAT("A02094", "该被检设备下存在相同线路号的监测点"), MONITOR_NUM_REPEAT("A02094", "该被检设备下存在相同线路号的监测点"),
PLAN_HAS_CHILDREN("A02095", "该计划下存在子计划,请先删除子计划"), PLAN_HAS_CHILDREN("A02095", "该计划下存在子计划,请先删除子计划"),
PLAN_REPEATED_IN_SAME_LEVEL("A02096", "该父计划下存在同名的子计划"), PLAN_REPEATED_IN_SAME_LEVEL("A02096", "该父计划下存在同名的子计划"),
PLEASE_UNASSIGN_STANDARD_DEV("A02097","存在已分配给子计划的标准设备,请先解除分配" ), PLEASE_UNASSIGN_STANDARD_DEV("A02097", "存在已分配给子计划的标准设备,请先解除分配"),
PLEASE_UNASSIGN_DEVICE("A02098", "存在已分配给计划的被检设备,请先解除分配"); PLEASE_UNASSIGN_DEVICE("A02098", "存在已分配给计划的被检设备,请先解除分配"),
DEV_IP_PORT_EXIST("A02099", "存在重复被检设备");
private final String code; private final String code;

View File

@@ -31,22 +31,22 @@ import java.util.List;
* @data 2025/1/9 14:02 * @data 2025/1/9 14:02
*/ */
@Slf4j @Slf4j
@Api(tags = "管理") @Api(tags = "管理")
@RestController @RestController
@RequestMapping("/report") @RequestMapping("/report")
@RequiredArgsConstructor @RequiredArgsConstructor
public class ReportController extends BaseController { public class ReportController extends BaseController {
private final IPqReportService pqReportService; private final IPqReportService pqReportService;
/** /**
* 此方法临时的,给楼下使用,实际需要优化 * 此方法临时的,给楼下使用,实际需要优化
* 1、不同的设备需要不同的模板 * 1、不同的设备需要不同的模板
* 2、数据页的内容暂时是固定的后期可能是动态的 * 2、数据页的内容暂时是固定的后期可能是动态 的;
*/ */
@OperateInfo @OperateInfo
@PostMapping("/generateReport") @PostMapping("/generateReport")
@ApiOperation("生成测试报告") @ApiOperation("生成报告")
@ApiImplicitParam(name = "param", value = "查询参数", required = true) @ApiImplicitParam(name = "param", value = "查询参数", required = true)
public HttpResult<Object> generateReport(@RequestBody DevReportParam devReportParam) { public HttpResult<Object> generateReport(@RequestBody DevReportParam devReportParam) {
String methodDescribe = getMethodDescribe("generateReport"); String methodDescribe = getMethodDescribe("generateReport");
@@ -60,7 +60,7 @@ public class ReportController extends BaseController {
*/ */
@OperateInfo @OperateInfo
@PostMapping("/downloadReport") @PostMapping("/downloadReport")
@ApiOperation("下载测试报告") @ApiOperation("下载报告")
@ApiImplicitParam(name = "param", value = "查询参数", required = true) @ApiImplicitParam(name = "param", value = "查询参数", required = true)
public void downloadReport(@RequestBody DevReportParam devReportParam, HttpServletResponse response) { public void downloadReport(@RequestBody DevReportParam devReportParam, HttpServletResponse response) {
String methodDescribe = getMethodDescribe("downloadReport"); String methodDescribe = getMethodDescribe("downloadReport");
@@ -91,7 +91,7 @@ public class ReportController extends BaseController {
@OperateInfo(operateType = OperateType.ADD) @OperateInfo(operateType = OperateType.ADD)
@PostMapping("/add") @PostMapping("/add")
@ApiOperation("新增报告模板") @ApiOperation("新增报告模板")
@ApiImplicitParam(name = "param", value = "报告模板参数", required = true) @ApiImplicitParam(name = "reportParam", value = "报告模板参数", required = true)
public HttpResult<Boolean> add(ReportParam reportParam) { public HttpResult<Boolean> add(ReportParam reportParam) {
String methodDescribe = getMethodDescribe("add"); String methodDescribe = getMethodDescribe("add");
LogUtil.njcnDebug(log, "{},新增参数为:{}", methodDescribe, reportParam); LogUtil.njcnDebug(log, "{},新增参数为:{}", methodDescribe, reportParam);
@@ -106,7 +106,7 @@ public class ReportController extends BaseController {
@OperateInfo(operateType = OperateType.UPDATE) @OperateInfo(operateType = OperateType.UPDATE)
@PostMapping("/update") @PostMapping("/update")
@ApiOperation("更新报告模板") @ApiOperation("更新报告模板")
@ApiImplicitParam(name = "param", value = "报告模板参数", required = true) @ApiImplicitParam(name = "reportParam", value = "报告模板参数", required = true)
public HttpResult<Boolean> update(ReportParam.UpdateParam reportParam) { public HttpResult<Boolean> update(ReportParam.UpdateParam reportParam) {
String methodDescribe = getMethodDescribe("update"); String methodDescribe = getMethodDescribe("update");
LogUtil.njcnDebug(log, "{},修改参数为:{}", methodDescribe, reportParam); LogUtil.njcnDebug(log, "{},修改参数为:{}", methodDescribe, reportParam);
@@ -136,7 +136,6 @@ public class ReportController extends BaseController {
@OperateInfo @OperateInfo
@GetMapping("/listAllName") @GetMapping("/listAllName")
@ApiOperation("查询所有报告模板名称") @ApiOperation("查询所有报告模板名称")
@ApiImplicitParam(name = "param", value = "查询参数", required = true)
public HttpResult<List<String>> listAllName() { public HttpResult<List<String>> listAllName() {
String methodDescribe = getMethodDescribe("listAllName"); String methodDescribe = getMethodDescribe("listAllName");
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, pqReportService.listAllName(), methodDescribe); return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, pqReportService.listAllName(), methodDescribe);
@@ -145,7 +144,7 @@ public class ReportController extends BaseController {
@OperateInfo(info = LogEnum.BUSINESS_COMMON, operateType = OperateType.UPDATE) @OperateInfo(info = LogEnum.BUSINESS_COMMON, operateType = OperateType.UPDATE)
@PostMapping("/documented") @PostMapping("/documented")
@ApiOperation("设备归档") @ApiOperation("设备归档")
@ApiImplicitParam(name = "id", value = "设备id", required = true) @ApiImplicitParam(name = "ids", value = "设备id", required = true)
public HttpResult<Boolean> documented(@RequestBody List<String> ids) { public HttpResult<Boolean> documented(@RequestBody List<String> ids) {
String methodDescribe = getMethodDescribe("documented"); String methodDescribe = getMethodDescribe("documented");
LogUtil.njcnDebug(log, "{}设备id为{}", methodDescribe, ids); LogUtil.njcnDebug(log, "{}设备id为{}", methodDescribe, ids);

View File

@@ -31,6 +31,7 @@ public enum BaseReportKeyEnum {
DAY("day",""), DAY("day",""),
YEAR_MONTH_DAY("year-month-day","年-月-日"), YEAR_MONTH_DAY("year-month-day","年-月-日"),
REPORT_DATE("reportDate","年-月-日"), REPORT_DATE("reportDate","年-月-日"),
GD_NAME("gdName","供电部门"),
SUB_NAME("subName","变电站"), SUB_NAME("subName","变电站"),
CHECK_BY("checkBy","检测人"), CHECK_BY("checkBy","检测人"),
AUDIT_BY("auditBy","负责人、审核人"), AUDIT_BY("auditBy","负责人、审核人"),
@@ -38,7 +39,10 @@ public enum BaseReportKeyEnum {
CREATE_DATE("createDate","生产日期"), CREATE_DATE("createDate","生产日期"),
TEMPERATURE("temp","温度"), TEMPERATURE("temp","温度"),
HUMIDITY("hum","相对湿度"), HUMIDITY("hum","相对湿度"),
DELEGATE("delegate","委托方"); DELEGATE("delegate","委托方"),
CREATEDATE("createDate","出厂日期"),
HW_VERSION("hardwareVersion","硬件版本"),
SW_VERSION("softwareVersion","软件版本");
private String key; private String key;

View File

@@ -26,7 +26,8 @@ public enum PowerIndexEnum {
HI("HI", "谐波电流"), HI("HI", "谐波电流"),
HSV("HSV", "间谐波电压"), HSV("HSV", "间谐波电压"),
HSI("HSI", "间谐波电流"), HSI("HSI", "间谐波电流"),
VOLTAGE("VOLTAGE", "电压暂降、暂升及短时中断"); VOLTAGE("VOLTAGE", "电压暂降、暂升及短时中断"),
PST("PST", "短时间闪变");
private String key; private String key;

View File

@@ -39,6 +39,8 @@ import com.njcn.gather.device.service.IPqDevService;
import com.njcn.gather.device.service.IPqDevSubService; import com.njcn.gather.device.service.IPqDevSubService;
import com.njcn.gather.err.pojo.po.PqErrSys; import com.njcn.gather.err.pojo.po.PqErrSys;
import com.njcn.gather.err.service.IPqErrSysService; import com.njcn.gather.err.service.IPqErrSysService;
import com.njcn.gather.monitor.pojo.po.PqMonitor;
import com.njcn.gather.monitor.service.IPqMonitorService;
import com.njcn.gather.plan.pojo.enums.DataSourceEnum; import com.njcn.gather.plan.pojo.enums.DataSourceEnum;
import com.njcn.gather.plan.pojo.enums.PlanReportStateEnum; import com.njcn.gather.plan.pojo.enums.PlanReportStateEnum;
import com.njcn.gather.plan.pojo.po.AdPlan; import com.njcn.gather.plan.pojo.po.AdPlan;
@@ -59,6 +61,7 @@ import com.njcn.gather.report.service.IPqReportService;
import com.njcn.gather.result.service.IResultService; import com.njcn.gather.result.service.IResultService;
import com.njcn.gather.script.pojo.vo.PqScriptDtlDataVO; import com.njcn.gather.script.pojo.vo.PqScriptDtlDataVO;
import com.njcn.gather.script.service.IPqScriptDtlsService; import com.njcn.gather.script.service.IPqScriptDtlsService;
import com.njcn.gather.storage.pojo.param.SingleNonHarmParam;
import com.njcn.gather.storage.pojo.po.SimAndDigHarmonicResult; import com.njcn.gather.storage.pojo.po.SimAndDigHarmonicResult;
import com.njcn.gather.storage.pojo.po.SimAndDigNonHarmonicResult; import com.njcn.gather.storage.pojo.po.SimAndDigNonHarmonicResult;
import com.njcn.gather.storage.service.SimAndDigHarmonicService; import com.njcn.gather.storage.service.SimAndDigHarmonicService;
@@ -67,6 +70,7 @@ import com.njcn.gather.system.cfg.pojo.enums.SceneEnum;
import com.njcn.gather.system.cfg.service.ISysTestConfigService; import com.njcn.gather.system.cfg.service.ISysTestConfigService;
import com.njcn.gather.system.dictionary.pojo.po.DictData; import com.njcn.gather.system.dictionary.pojo.po.DictData;
import com.njcn.gather.system.dictionary.service.IDictDataService; import com.njcn.gather.system.dictionary.service.IDictDataService;
import com.njcn.gather.system.pojo.enums.SystemResponseEnum;
import com.njcn.gather.tools.report.model.constant.ReportConstant; import com.njcn.gather.tools.report.model.constant.ReportConstant;
import com.njcn.gather.tools.report.service.IWordReportService; import com.njcn.gather.tools.report.service.IWordReportService;
import com.njcn.gather.tools.report.util.BookmarkUtil; import com.njcn.gather.tools.report.util.BookmarkUtil;
@@ -85,11 +89,11 @@ import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.Slf4j;
import org.apache.commons.net.ftp.FTPClient; import org.apache.commons.net.ftp.FTPClient;
import org.apache.commons.net.ftp.FTPReply; import org.apache.commons.net.ftp.FTPReply;
import org.apache.poi.xwpf.usermodel.XWPFDocument;
import org.docx4j.jaxb.Context; import org.docx4j.jaxb.Context;
import org.docx4j.openpackaging.packages.WordprocessingMLPackage; import org.docx4j.openpackaging.packages.WordprocessingMLPackage;
import org.docx4j.openpackaging.parts.WordprocessingML.MainDocumentPart; import org.docx4j.openpackaging.parts.WordprocessingML.MainDocumentPart;
import org.docx4j.wml.*; import org.docx4j.wml.*;
import org.docx4j.wml.Color;
import org.springframework.beans.BeanUtils; import org.springframework.beans.BeanUtils;
import org.springframework.beans.factory.annotation.Value; import org.springframework.beans.factory.annotation.Value;
import org.springframework.core.io.ClassPathResource; import org.springframework.core.io.ClassPathResource;
@@ -109,9 +113,11 @@ import java.math.BigInteger;
import java.math.RoundingMode; import java.math.RoundingMode;
import java.net.URLEncoder; import java.net.URLEncoder;
import java.nio.file.Files; import java.nio.file.Files;
import java.nio.file.NoSuchFileException;
import java.nio.file.Path; import java.nio.file.Path;
import java.nio.file.Paths; import java.nio.file.Paths;
import java.time.LocalDateTime; import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
import java.util.List; import java.util.List;
import java.util.*; import java.util.*;
import java.util.concurrent.CompletableFuture; import java.util.concurrent.CompletableFuture;
@@ -161,6 +167,8 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
@Value("${qr.gcDev.path}") @Value("${qr.gcDev.path}")
private String gcDevPath; private String gcDevPath;
@Value("${report.dateFormat}")
private String dateFormat;
private final IPqDevService iPqDevService; private final IPqDevService iPqDevService;
private final PqDevMapper pqDevMapper; private final PqDevMapper pqDevMapper;
@@ -178,6 +186,7 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
private final ISysUserService sysUserService; private final ISysUserService sysUserService;
private final IPqErrSysService pqErrSysService; private final IPqErrSysService pqErrSysService;
private final IAdPlanTestConfigService adPlanTestConfigService; private final IAdPlanTestConfigService adPlanTestConfigService;
private final IPqMonitorService pqMonitorService;
@Resource @Resource
private RestTemplateUtil restTemplateUtil; private RestTemplateUtil restTemplateUtil;
@@ -306,7 +315,8 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
private void uploadFile(ReportParam reportParam, PqReport pqReport, boolean isAdd) { private void uploadFile(ReportParam reportParam, PqReport pqReport, boolean isAdd) {
MultipartFile baseFile = reportParam.getBaseFile(); MultipartFile baseFile = reportParam.getBaseFile();
MultipartFile detailFile = reportParam.getDetailFile(); MultipartFile detailFile = reportParam.getDetailFile();
String newDir = templatePath + File.separator + reportParam.getName() + File.separator + reportParam.getVersion() + File.separator; String relativePath = reportParam.getName() + File.separator + reportParam.getVersion() + File.separator;
String newDir = templatePath + File.separator + relativePath;
long FILE_SIZE_LIMIT = 5 * 1024 * 1024; long FILE_SIZE_LIMIT = 5 * 1024 * 1024;
if (isAdd) { if (isAdd) {
@@ -325,13 +335,13 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
throw new BusinessException(ReportResponseEnum.FILE_SIZE_ERROR); throw new BusinessException(ReportResponseEnum.FILE_SIZE_ERROR);
} }
pqReport.setBasePath(newDir + baseOriginalFilename); pqReport.setBasePath(relativePath + baseOriginalFilename);
pqReport.setDetailPath(newDir + detailOriginalFilename); pqReport.setDetailPath(relativePath + detailOriginalFilename);
this.createDirectory(newDir); this.createDirectory(newDir);
this.clearDirectory(newDir); this.clearDirectory(newDir);
this.uploadFile(baseFile, pqReport.getBasePath()); this.uploadFile(baseFile, newDir + baseOriginalFilename);
this.uploadFile(detailFile, pqReport.getDetailPath()); this.uploadFile(detailFile, newDir + detailOriginalFilename);
} else { } else {
throw new BusinessException(ReportResponseEnum.FILE_NOT_NULL); throw new BusinessException(ReportResponseEnum.FILE_NOT_NULL);
} }
@@ -379,34 +389,34 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
//若修改了文件名称、版本号,则需要重命名文件 //若修改了文件名称、版本号,则需要重命名文件
this.createDirectory(newDir); this.createDirectory(newDir);
if (!oldDir.equals(newDir)) { if (!oldDir.equals(relativePath)) {
// 文件夹重命名 // 文件夹重命名
String oldBasePathStr = oldPqReport.getBasePath(); String oldBasePathStr = oldPqReport.getBasePath();
String baseName = oldBasePathStr.substring(oldBasePathStr.lastIndexOf(File.separator) + 1); String baseName = oldBasePathStr.substring(oldBasePathStr.lastIndexOf(File.separator) + 1);
Path oldBasePath = Paths.get(oldBasePathStr); Path oldBasePath = Paths.get(templatePath + File.separator + oldBasePathStr);
Path newBasePath = Paths.get(newDir + baseName); Path newBasePath = Paths.get(newDir + baseName);
pqReport.setBasePath(newDir + baseName); pqReport.setBasePath(relativePath + baseName);
String oldDetailPathStr = oldPqReport.getDetailPath(); String oldDetailPathStr = oldPqReport.getDetailPath();
String detailName = oldDetailPathStr.substring(oldDetailPathStr.lastIndexOf(File.separator) + 1); String detailName = oldDetailPathStr.substring(oldDetailPathStr.lastIndexOf(File.separator) + 1);
Path oldDetailPath = Paths.get(oldDetailPathStr); Path oldDetailPath = Paths.get(templatePath + File.separator + oldDetailPathStr);
Path newDetailPath = Paths.get(newDir + detailName); Path newDetailPath = Paths.get(newDir + detailName);
pqReport.setDetailPath(newDir + detailName); pqReport.setDetailPath(relativePath + detailName);
try { try {
// windows下文件夹名称不区分大小写 // windows下文件夹名称不区分大小写
if (!oldDir.equalsIgnoreCase(newDir)) { if (!oldDir.equalsIgnoreCase(relativePath)) {
this.clearDirectory(newDir); this.clearDirectory(newDir);
Files.move(oldBasePath, newBasePath); Files.move(oldBasePath, newBasePath);
Files.move(oldDetailPath, newDetailPath); Files.move(oldDetailPath, newDetailPath);
if (!oldPqReport.getName().equals(reportParam.getName()) && !this.existSameName(pqReport.getId(), oldPqReport.getName())) { if (!oldPqReport.getName().equals(reportParam.getName()) && !this.existSameName(pqReport.getId(), oldPqReport.getName())) {
this.recursionDeleteDirectory(templatePath + File.separator + oldPqReport.getName()); this.recursionDeleteDirectory(templatePath + File.separator + oldPqReport.getName());
} else { } else {
Paths.get(oldDir).toFile().delete(); Paths.get(templatePath + oldDir).toFile().delete();
} }
} else { } else {
// 文件夹重命名 // 文件夹重命名
Paths.get(oldDir).toFile().renameTo(Paths.get(newDir).toFile()); Paths.get(templatePath + oldDir).toFile().renameTo(Paths.get(newDir).toFile());
} }
} catch (IOException e) { } catch (IOException e) {
throw new BusinessException(ReportResponseEnum.FILE_RENAME_FAILED); throw new BusinessException(ReportResponseEnum.FILE_RENAME_FAILED);
@@ -414,16 +424,16 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
} }
if (!baseFileOriginalFilename.isEmpty()) { if (!baseFileOriginalFilename.isEmpty()) {
pqReport.setBasePath(newDir + baseFileOriginalFilename); pqReport.setBasePath(relativePath + baseFileOriginalFilename);
Paths.get(oldPqReport.getBasePath()).toFile().delete(); Paths.get(templatePath + File.separator + oldPqReport.getBasePath()).toFile().delete();
Paths.get(newDir + oldPqReport.getBasePath().substring(oldPqReport.getBasePath().lastIndexOf(File.separator) + 1)).toFile().delete(); Paths.get(newDir + oldPqReport.getBasePath().substring(oldPqReport.getBasePath().lastIndexOf(File.separator) + 1)).toFile().delete();
this.uploadFile(baseFile, pqReport.getBasePath()); this.uploadFile(baseFile, newDir + baseFileOriginalFilename);
} }
if (!detailFileOriginalFilename.isEmpty()) { if (!detailFileOriginalFilename.isEmpty()) {
pqReport.setDetailPath(newDir + detailFileOriginalFilename); pqReport.setDetailPath(relativePath + detailFileOriginalFilename);
Paths.get(oldPqReport.getDetailPath()).toFile().delete(); Paths.get(templatePath + File.separator + oldPqReport.getDetailPath()).toFile().delete();
Paths.get(newDir + oldPqReport.getDetailPath().substring(oldPqReport.getDetailPath().lastIndexOf(File.separator) + 1)).toFile().delete(); Paths.get(newDir + oldPqReport.getDetailPath().substring(oldPqReport.getDetailPath().lastIndexOf(File.separator) + 1)).toFile().delete();
this.uploadFile(detailFile, pqReport.getDetailPath()); this.uploadFile(detailFile, newDir + detailFileOriginalFilename);
} }
} }
} }
@@ -571,7 +581,6 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
/** /**
*
* 根据设备类型生成报告 * 根据设备类型生成报告
* 注:该方法目前仅支持楼下出厂检测场景,属于模板占位符替换方式,后期可能会有调整 * 注:该方法目前仅支持楼下出厂检测场景,属于模板占位符替换方式,后期可能会有调整
* *
@@ -641,6 +650,13 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
} }
} }
}); });
} catch (NoSuchFileException e) {
String filePath = e.getFile() != null ? e.getFile().replaceAll("\\\\", "/") : "未知文件";
log.error("报告模板文件不存在 - 文件路径: {}", filePath, e);
throw new BusinessException(SystemResponseEnum.FILE_NOT_FOUND);
} catch (IOException e) {
log.error("报告文件读写异常", e);
throw new BusinessException(SystemResponseEnum.FILE_IO_ERROR);
} catch (Exception e) { } catch (Exception e) {
log.error(ReportResponseEnum.GENERATE_REPORT_ERROR.getMessage(), e); log.error(ReportResponseEnum.GENERATE_REPORT_ERROR.getMessage(), e);
throw new BusinessException(ReportResponseEnum.GENERATE_REPORT_ERROR); throw new BusinessException(ReportResponseEnum.GENERATE_REPORT_ERROR);
@@ -779,46 +795,63 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
if (Objects.isNull(pqDevVO)) { if (Objects.isNull(pqDevVO)) {
throw new BusinessException(ReportResponseEnum.DEVICE_NOT_EXIST); throw new BusinessException(ReportResponseEnum.DEVICE_NOT_EXIST);
} }
devReportParam.setDevId(devId); if (CheckStateEnum.CHECKED.getValue().compareTo(pqDevVO.getCheckState()) <= 0) {
// 获取设备型号 devReportParam.setDevId(devId);
DevType devType = devTypeService.getById(pqDevVO.getDevType()); // 获取设备型号
if (Objects.isNull(devType)) { DevType devType = devTypeService.getById(pqDevVO.getDevType());
throw new BusinessException(ReportResponseEnum.DEVICE_TYPE_NOT_EXIST); if (Objects.isNull(devType)) {
} throw new BusinessException(ReportResponseEnum.DEVICE_TYPE_NOT_EXIST);
PqReport report = this.lambdaQuery().eq(PqReport::getId, plan.getReportTemplateId()).eq(PqReport::getState, DataStateEnum.ENABLE.getCode()).one(); }
if (Objects.isNull(report)) { PqReport report = this.lambdaQuery().eq(PqReport::getId, plan.getReportTemplateId()).eq(PqReport::getState, DataStateEnum.ENABLE.getCode()).one();
throw new BusinessException(ReportResponseEnum.REPORT_TEMPLATE_NOT_EXIST); if (Objects.isNull(report)) {
} throw new BusinessException(ReportResponseEnum.REPORT_TEMPLATE_NOT_EXIST);
Path basePath = Paths.get(report.getBasePath()); }
Path detailPath = Paths.get(report.getDetailPath()); Path basePath = Paths.get(templatePath + File.separator + report.getBasePath());
try (InputStream baseInputStream = Files.newInputStream(basePath); Path detailPath = Paths.get(templatePath + File.separator + report.getDetailPath());
InputStream detailInputStream = Files.newInputStream(detailPath)) { try (InputStream baseInputStream = Files.newInputStream(basePath);
WordprocessingMLPackage detailModelDocument = WordprocessingMLPackage.load(detailInputStream); InputStream detailInputStream = Files.newInputStream(detailPath)) {
// 获取文档基础部分,并替换占位符 WordprocessingMLPackage detailModelDocument = WordprocessingMLPackage.load(detailInputStream);
Map<String, String> baseModelDataMap = dealBaseModelContrastData(plan, pqDevVO, devType); // 获取文档基础部分,并替换占位符
InputStream wordFinishInputStream = wordReportService.replacePlaceholders(baseInputStream, baseModelDataMap); Map<String, String> baseModelDataMap = dealBaseModelContrastData(plan, pqDevVO, devType);
WordprocessingMLPackage baseModelDocument = WordprocessingMLPackage.load(wordFinishInputStream); InputStream wordFinishInputStream = wordReportService.replacePlaceholders(baseInputStream, baseModelDataMap);
MainDocumentPart baseDocumentPart = baseModelDocument.getMainDocumentPart(); WordprocessingMLPackage baseModelDocument = WordprocessingMLPackage.load(wordFinishInputStream);
MainDocumentPart baseDocumentPart = baseModelDocument.getMainDocumentPart();
// 获取数据模版页内容,根据脚本动态组装数据页内容 // 获取数据模版页内容,根据脚本动态组装数据页内容
MainDocumentPart detailDocumentPart = detailModelDocument.getMainDocumentPart(); MainDocumentPart detailDocumentPart = detailModelDocument.getMainDocumentPart();
dealDataModelScatteredByBookmarkByPlanContrast(baseDocumentPart, detailDocumentPart, devReportParam, pqDevVO); dealDataModelScatteredByBookmarkByPlanContrast(baseDocumentPart, detailDocumentPart, devReportParam, pqDevVO);
// 保存新的文档 // 保存新的文档
String dirPath = reportPath.concat(File.separator).concat(plan.getName()); String dirPath = reportPath.concat(File.separator).concat(plan.getName());
// 确保目录存在 // 确保目录存在
ensureDirectoryExists(dirPath); ensureDirectoryExists(dirPath);
// 构建文件名cityName_gdName_subName_name.docx // 构建文件名cityName_gdName_subName_name.docx
String fileName = String.format("%s_%s_%s_%s.docx", String fileName = String.format("%s_%s_%s_%s.docx",
pqDevVO.getCityName() != null ? pqDevVO.getCityName() : "未知地市", pqDevVO.getCityName() != null ? pqDevVO.getCityName() : "未知地市",
pqDevVO.getGdName() != null ? pqDevVO.getGdName() : "未知供电公司", pqDevVO.getGdName() != null ? pqDevVO.getGdName() : "未知供电公司",
pqDevVO.getSubName() != null ? pqDevVO.getSubName() : "未知电站", pqDevVO.getSubName() != null ? pqDevVO.getSubName() : "未知电站",
pqDevVO.getName() != null ? pqDevVO.getName() : "未知设备"); pqDevVO.getName() != null ? pqDevVO.getName() : "未知设备");
Docx4jUtil.cleanBlankPagesAndRedundantPageBreaks(baseModelDocument); // 判断是否需要在报告上输出水印
baseModelDocument.save(new File(dirPath.concat(File.separator).concat(fileName))); String leader = baseModelDataMap.get(BaseReportKeyEnum.AUDIT_BY.getKey());
this.updateDevAndPlanState(devId, devReportParam.getPlanId()); String loginName = RequestUtil.getLoginNameByToken();
} catch (Exception e) { SysUser user = sysUserService.getUserByLoginName(loginName);
log.error(ReportResponseEnum.GENERATE_REPORT_ERROR.getMessage(), e); if (!leader.equals(user.getName())) {
throw new BusinessException(ReportResponseEnum.GENERATE_REPORT_ERROR); log.info("当前用户不是审核人,添加非正式水印");
Docx4jUtil.addWatermarkToDocument(baseModelDocument, "非正式");
}
Docx4jUtil.cleanBlankPagesAndRedundantPageBreaks(baseModelDocument);
baseModelDocument.save(new File(dirPath.concat(File.separator).concat(fileName)));
this.updateDevAndPlanState(devId, devReportParam.getPlanId());
} catch (NoSuchFileException e) {
String filePath = e.getFile() != null ? e.getFile().replaceAll("\\\\", "/") : "未知文件";
log.error("报告模板文件不存在 - 文件路径: {}", filePath, e);
throw new BusinessException(SystemResponseEnum.FILE_NOT_FOUND);
} catch (IOException e) {
log.error("报告文件读写异常", e);
throw new BusinessException(SystemResponseEnum.FILE_IO_ERROR);
} catch (Exception e) {
log.error(ReportResponseEnum.GENERATE_REPORT_ERROR.getMessage(), e);
throw new BusinessException(ReportResponseEnum.GENERATE_REPORT_ERROR);
}
} }
}); });
@@ -850,8 +883,8 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
if (Objects.isNull(report)) { if (Objects.isNull(report)) {
throw new BusinessException(ReportResponseEnum.REPORT_TEMPLATE_NOT_EXIST); throw new BusinessException(ReportResponseEnum.REPORT_TEMPLATE_NOT_EXIST);
} }
Path basePath = Paths.get(report.getBasePath()); Path basePath = Paths.get(templatePath + File.separator + report.getBasePath());
Path detailPath = Paths.get(report.getDetailPath()); Path detailPath = Paths.get(templatePath + File.separator + report.getDetailPath());
try (InputStream baseInputStream = Files.newInputStream(basePath); try (InputStream baseInputStream = Files.newInputStream(basePath);
InputStream detailInputStream = Files.newInputStream(detailPath)) { InputStream detailInputStream = Files.newInputStream(detailPath)) {
WordprocessingMLPackage detailModelDocument = WordprocessingMLPackage.load(detailInputStream); WordprocessingMLPackage detailModelDocument = WordprocessingMLPackage.load(detailInputStream);
@@ -871,6 +904,13 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
ensureDirectoryExists(dirPath); ensureDirectoryExists(dirPath);
baseModelDocument.save(new File(dirPath.concat(File.separator).concat(pqDevVO.getCreateId()).concat(ReportConstant.DOCX))); baseModelDocument.save(new File(dirPath.concat(File.separator).concat(pqDevVO.getCreateId()).concat(ReportConstant.DOCX)));
this.updateDevAndPlanState(devId, devReportParam.getPlanId()); this.updateDevAndPlanState(devId, devReportParam.getPlanId());
} catch (NoSuchFileException e) {
String filePath = e.getFile() != null ? e.getFile().replaceAll("\\\\", "/") : "未知文件";
log.error("报告模板文件不存在 - 文件路径: {}", filePath, e);
throw new BusinessException(SystemResponseEnum.FILE_NOT_FOUND);
} catch (IOException e) {
log.error("报告文件读写异常", e);
throw new BusinessException(SystemResponseEnum.FILE_IO_ERROR);
} catch (Exception e) { } catch (Exception e) {
log.error(ReportResponseEnum.GENERATE_REPORT_ERROR.getMessage(), e); log.error(ReportResponseEnum.GENERATE_REPORT_ERROR.getMessage(), e);
throw new BusinessException(ReportResponseEnum.GENERATE_REPORT_ERROR); throw new BusinessException(ReportResponseEnum.GENERATE_REPORT_ERROR);
@@ -1065,7 +1105,6 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
* 实时数据3秒数据150周波数据 * 实时数据3秒数据150周波数据
* 录波:录波数据 * 录波:录波数据
* 分钟统计数据:分钟统计数据-最大值、分钟统计数据-最小值、分钟统计数据-平均值、分钟统计数据-CP95值 * 分钟统计数据:分钟统计数据-最大值、分钟统计数据-最小值、分钟统计数据-平均值、分钟统计数据-CP95值
*
*/ */
private String getDataTypeFromParam(AdPlan adPlan) { private String getDataTypeFromParam(AdPlan adPlan) {
String dataSource = adPlan.getDatasourceId(); String dataSource = adPlan.getDatasourceId();
@@ -1088,11 +1127,11 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
AdPlanTestConfig adPlanTestConfig = adPlanTestConfigService.getByPlanId(adPlan.getId()); AdPlanTestConfig adPlanTestConfig = adPlanTestConfigService.getByPlanId(adPlan.getId());
StringBuilder sampleCount = new StringBuilder(); StringBuilder sampleCount = new StringBuilder();
for (String item : dataSourceArray) { for (String item : dataSourceArray) {
if(item.equalsIgnoreCase(DataSourceEnum.REAL_DATA.getValue())){ if (item.equalsIgnoreCase(DataSourceEnum.REAL_DATA.getValue())) {
sampleCount.append("实时数据采集").append(adPlanTestConfig.getRealTime()).append("组,"); sampleCount.append("实时数据采集").append(adPlanTestConfig.getRealTime()).append("组,");
}else if(item.equalsIgnoreCase(DataSourceEnum.WAVE_DATA.getValue())){ } else if (item.equalsIgnoreCase(DataSourceEnum.WAVE_DATA.getValue())) {
sampleCount.append("录波数据采集").append(adPlanTestConfig.getWaveRecord()).append("组,"); sampleCount.append("录波数据采集").append(adPlanTestConfig.getWaveRecord()).append("组,");
}else{ } else {
sampleCount.append("统计数据采集").append(adPlanTestConfig.getStatistics()).append("组,"); sampleCount.append("统计数据采集").append(adPlanTestConfig.getStatistics()).append("组,");
} }
} }
@@ -1134,10 +1173,11 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
Integer monitorNum = next.getKey(); Integer monitorNum = next.getKey();
// 线路下的指标数据 // 线路下的指标数据
List<ContrastTestResult> contrastTestResults = next.getValue(); List<ContrastTestResult> contrastTestResults = next.getValue();
PqMonitor pqMonitor = pqMonitorService.getByDevIdAndNum(pqDevVO.getId(), monitorNum);
// 插入回路号前,先换个页 // 插入回路号前,先换个页
todoInsertList.add(Docx4jUtil.createPageBreakParagraph()); todoInsertList.add(Docx4jUtil.createPageBreakParagraph());
// 回路标题 // 回路标题
todoInsertList.add(getContrastLineTitle(contentMap, monitorNum, stepIndex, factory)); todoInsertList.add(getContrastLineTitle(contentMap, monitorNum, stepIndex, factory, pqDevVO));
int scriptIndex = 1; int scriptIndex = 1;
for (ContrastTestResult contrastTestResult : contrastTestResults) { for (ContrastTestResult contrastTestResult : contrastTestResults) {
// 比如电压 V // 比如电压 V
@@ -1148,7 +1188,7 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
// 根据code获取对应需要填充的内容 // 根据code获取对应需要填充的内容
List<Docx4jUtil.HeadingContent> tempContent = contentMap.get(scriptCode); List<Docx4jUtil.HeadingContent> tempContent = contentMap.get(scriptCode);
// 需要区分下谐波类和非谐波类 // 需要区分下谐波类和非谐波类
List<Object> tempList = fillContentInTemplateContrast(tempContent, factory, contrastTestResult); List<Object> tempList = fillContentInTemplateContrast(tempContent, factory, contrastTestResult, pqMonitor);
todoInsertList.addAll(tempList); todoInsertList.addAll(tempList);
scriptIndex++; scriptIndex++;
} }
@@ -1566,18 +1606,37 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
// 获取现有行的样式 // 获取现有行的样式
Tr existingRow = (Tr) tbl.getContent().get(rows.size() - 1); Tr existingRow = (Tr) tbl.getContent().get(rows.size() - 1);
// 获取现有样式 // 获取现有样式
TrPr trPr = existingRow.getTrPr(); TrPr trPr = existingRow.getTrPr();
JAXBElement<Tc> element = (JAXBElement<Tc>) existingRow.getContent().get(0); // 获取每个单元格的TcPr保留各单元格独立的边框设置
TcPr tcPr = element.getValue().getTcPr(); List<TcPr> tcPrList = new ArrayList<>();
TblWidth cellWidth = factory.createTblWidth(); RPr templateRPr = null;
cellWidth.setType("dxa"); for (Object cellObj : existingRow.getContent()) {
cellWidth.setW(BigInteger.valueOf(5000 / tableKeys.size())); if (cellObj instanceof JAXBElement) {
tcPr.setTcW(cellWidth); JAXBElement<Tc> cellElement = (JAXBElement<Tc>) cellObj;
Tc templateCell = cellElement.getValue();
TcPr tcPr = templateCell.getTcPr();
// 设置单元格宽度
if (tcPr == null) {
tcPr = factory.createTcPr();
}
TblWidth cellWidth = factory.createTblWidth();
cellWidth.setType("dxa");
cellWidth.setW(BigInteger.valueOf(5000 / tableKeys.size()));
tcPr.setTcW(cellWidth);
tcPrList.add(tcPr);
// 从第一个单元格获取字体样式
if (templateRPr == null && !templateCell.getContent().isEmpty() && templateCell.getContent().get(0) instanceof P) {
P templateP = (P) templateCell.getContent().get(0);
templateRPr = Docx4jUtil.getTcPrFromParagraph(templateP);
}
}
}
tbl.getContent().remove(existingRow); tbl.getContent().remove(existingRow);
// 迭代增加行,需要填充的表格keys在tableKeys集合中 // 迭代增加行,需要填充的表格keys在tableKeys集合中
for (Map<String, String> stringStringMap : dataList) { for (Map<String, String> stringStringMap : dataList) {
Tr newRow = Docx4jUtil.createCustomRow(factory, stringStringMap, tableKeys, trPr, tcPr, true); Tr newRow = Docx4jUtil.createCustomRow(factory, stringStringMap, tableKeys, trPr, tcPrList, templateRPr, true);
tbl.getContent().add(newRow); tbl.getContent().add(newRow);
} }
} else { } else {
@@ -1604,7 +1663,7 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
* *
* @return * @return
*/ */
private List<Object> fillContentInTemplateContrast(List<Docx4jUtil.HeadingContent> tempContent, ObjectFactory factory, ContrastTestResult contrastTestResult) { private List<Object> fillContentInTemplateContrast(List<Docx4jUtil.HeadingContent> tempContent, ObjectFactory factory, ContrastTestResult contrastTestResult, PqMonitor pqMonitor) {
List<Object> todoInsertList = new ArrayList<>(); List<Object> todoInsertList = new ArrayList<>();
Docx4jUtil.HeadingContent headingContent = tempContent.get(0); Docx4jUtil.HeadingContent headingContent = tempContent.get(0);
List<String> tableKeys = Docx4jUtil.getTableFillKeys(tempContent); List<String> tableKeys = Docx4jUtil.getTableFillKeys(tempContent);
@@ -1718,9 +1777,47 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
// 纵向表格暂不考虑 // 纵向表格暂不考虑
} }
// 如果存在特殊说明,在表格后添加一个段落 StringBuilder description = new StringBuilder();
if (StrUtil.isNotBlank(contrastTestResult.getSpecialCase())) { if (contrastTestResult.getScriptCode().equalsIgnoreCase("I")) {
P specialCaseP = Docx4jUtil.createSpecialCaseParagraph(factory, contrastTestResult.getSpecialCase()); // 获取该通道的额定电流
String ct = pqMonitor.getCt();
// 理论上ct是 xxx:xxx的格式。比如10:1100:5
if (ct.indexOf(":") > 0) {
String[] ctArray = ct.split(":");
description.append("注:当前回路额定电流为:").append(ctArray[1]).append("A。");
}
} else if (contrastTestResult.getScriptCode().equalsIgnoreCase("V")) {
// 获取该通道的额定电压
String pt = pqMonitor.getPt();
// 理论上pt是 xxx:xxx的格式。比如380:38010000:100
if (pt.indexOf(":") > 0) {
String[] ptArray = pt.split(":");
// 电压需要特殊处理下,处理为相电压值
String voltage = ptArray[1];
if (voltage.equalsIgnoreCase("100")) {
voltage = "57.74";
} else if (voltage.equalsIgnoreCase("380")) {
voltage = "220";
} else {
// 其他场景下就除以根号3
double result = Double.parseDouble(voltage) / Math.sqrt(3);
voltage = doubleRound(2, result);
}
description.append("注:当前回路额定电流为:").append(voltage).append("V。");
}
}
// 如果存在特殊说明,在表格后添加一个段落 如果有电压和电流需要把额定电压和电流标注进来 pqMonitor
if (StrUtil.isNotBlank(contrastTestResult.getSpecialCase()) || StrUtil.isNotBlank(description.toString())) {
P specialCaseP;
if (StrUtil.isNotBlank(description.toString())) {
if (StrUtil.isNotBlank(contrastTestResult.getSpecialCase())) {
description.append(contrastTestResult.getSpecialCase().replace("注:", ""));
}
specialCaseP = Docx4jUtil.createSpecialCaseParagraph(factory, description.toString());
} else {
specialCaseP = Docx4jUtil.createSpecialCaseParagraph(factory, contrastTestResult.getSpecialCase());
}
todoInsertList.add(specialCaseP); todoInsertList.add(specialCaseP);
} }
} }
@@ -1805,10 +1902,10 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
if (dictDataById != null && "Contrast".equals(dictDataById.getCode())) { if (dictDataById != null && "Contrast".equals(dictDataById.getCode())) {
// 比对模式:使用新的路径结构 // 比对模式:使用新的路径结构
String fileName = String.format("%s_%s_%s_%s.docx", String fileName = String.format("%s_%s_%s_%s.docx",
pqDevVO.getCityName() != null ? pqDevVO.getCityName() : "未知地市", pqDevVO.getCityName() != null ? pqDevVO.getCityName() : "未知地市",
pqDevVO.getGdName() != null ? pqDevVO.getGdName() : "未知供电公司", pqDevVO.getGdName() != null ? pqDevVO.getGdName() : "未知供电公司",
pqDevVO.getSubName() != null ? pqDevVO.getSubName() : "未知电站", pqDevVO.getSubName() != null ? pqDevVO.getSubName() : "未知电站",
pqDevVO.getName() != null ? pqDevVO.getName() : "未知设备"); pqDevVO.getName() != null ? pqDevVO.getName() : "未知设备");
filePath = reportPath.concat(File.separator).concat(plan.getName()).concat(File.separator).concat(fileName); filePath = reportPath.concat(File.separator).concat(plan.getName()).concat(File.separator).concat(fileName);
downloadFileName = fileName; downloadFileName = fileName;
} else { } else {
@@ -1837,7 +1934,7 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
try { try {
// 对中文文件名进行URL编码// 将+号替换为%20空格的正确编码 // 对中文文件名进行URL编码// 将+号替换为%20空格的正确编码
String encodedFileName = URLEncoder.encode(downloadFileName, "UTF-8") String encodedFileName = URLEncoder.encode(downloadFileName, "UTF-8")
.replaceAll("\\+", "%20"); .replaceAll("\\+", "%20");
// 使用RFC 5987标准格式 // 使用RFC 5987标准格式
response.setHeader("Content-Disposition", response.setHeader("Content-Disposition",
@@ -1879,12 +1976,12 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
for (int i = 0; i < pqDevVOList.size(); i++) { for (int i = 0; i < pqDevVOList.size(); i++) {
pqDevVO = pqDevVOList.get(i); pqDevVO = pqDevVOList.get(i);
tempPqDevVO = pqDevVO;
if (!pqDevVO.getCheckState().equals(CheckStateEnum.CHECKED.getValue())) { if (!pqDevVO.getCheckState().equals(CheckStateEnum.CHECKED.getValue())) {
throw new BusinessException(DetectionResponseEnum.DEV_UN_CHECKED); throw new BusinessException(DetectionResponseEnum.DEV_UN_CHECKED);
} }
if (!pqDevVO.getReportState().equals(DevReportStateEnum.GENERATED.getValue())) { if (!pqDevVO.getReportState().equals(DevReportStateEnum.GENERATED.getValue())) {
devIds.add(pqDevVO.getId()); devIds.add(pqDevVO.getId());
tempPqDevVO = pqDevVO;
} }
} }
@@ -1928,6 +2025,18 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
baseModelMap.put(BaseReportKeyEnum.DEV_TYPE.getKey(), devType.getName()); baseModelMap.put(BaseReportKeyEnum.DEV_TYPE.getKey(), devType.getName());
// 检测员 // 检测员
baseModelMap.put(BaseReportKeyEnum.INSPECTOR.getKey(), pqDevVO.getCheckBy() + ""); baseModelMap.put(BaseReportKeyEnum.INSPECTOR.getKey(), pqDevVO.getCheckBy() + "");
baseModelMap.put(BaseReportKeyEnum.INSPECTOR.getKey(), pqDevVO.getCheckBy() + "");
String datePattern = DatePattern.NORM_DATE_PATTERN;
if (DatePattern.CHINESE_DATE_PATTERN.equals(dateFormat)) {
datePattern = DatePattern.CHINESE_DATE_PATTERN;
}
// 出厂日期
baseModelMap.put(BaseReportKeyEnum.CREATEDATE.getKey(), pqDevVO.getCreateDate().format(DateTimeFormatter.ofPattern(datePattern)));
// 硬件版本
baseModelMap.put(BaseReportKeyEnum.HW_VERSION.getKey(), StrUtil.isNotBlank(pqDevVO.getHardwareVersion()) ? pqDevVO.getHardwareVersion() : StrUtil.EMPTY);
// 软件版本
baseModelMap.put(BaseReportKeyEnum.SW_VERSION.getKey(), StrUtil.isNotBlank(pqDevVO.getSoftwareVersion()) ? pqDevVO.getSoftwareVersion() : StrUtil.EMPTY);
// 调试日期 // 调试日期
if (pqDevVO.getCheckTime() != null) { if (pqDevVO.getCheckTime() != null) {
baseModelMap.put(BaseReportKeyEnum.TEST_DATE.getKey(), DateUtil.format(pqDevVO.getCheckTime(), DatePattern.CHINESE_DATE_PATTERN)); baseModelMap.put(BaseReportKeyEnum.TEST_DATE.getKey(), DateUtil.format(pqDevVO.getCheckTime(), DatePattern.CHINESE_DATE_PATTERN));
@@ -1986,7 +2095,6 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
} }
/** /**
*
* 比对模式下需要获取的数据 * 比对模式下需要获取的数据
* 处理基础模版中的信息,非数据页报告 * 处理基础模版中的信息,非数据页报告
* 因为Docx4j工具包替换时会默认增加${} * 因为Docx4j工具包替换时会默认增加${}
@@ -1997,7 +2105,7 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
// 委托方 // 委托方
String delegate = pqDevVO.getDelegate(); String delegate = pqDevVO.getDelegate();
if (StrUtil.isNotBlank(delegate)) { if (StrUtil.isNotBlank(delegate)) {
DictData delegateDictData = dictDataService.getDictDataById(pqDevVO.getManufacturer()); DictData delegateDictData = dictDataService.getDictDataById(pqDevVO.getDelegate());
if (ObjectUtil.isNotNull(delegateDictData)) { if (ObjectUtil.isNotNull(delegateDictData)) {
baseModelMap.put(BaseReportKeyEnum.DELEGATE.getKey(), delegateDictData.getName()); baseModelMap.put(BaseReportKeyEnum.DELEGATE.getKey(), delegateDictData.getName());
} else { } else {
@@ -2010,6 +2118,8 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
baseModelMap.put(BaseReportKeyEnum.SAMPLE_ID.getKey(), StrUtil.isEmpty(pqDevVO.getName()) ? StrPool.TAB : pqDevVO.getName()); baseModelMap.put(BaseReportKeyEnum.SAMPLE_ID.getKey(), StrUtil.isEmpty(pqDevVO.getName()) ? StrPool.TAB : pqDevVO.getName());
// 报告日期 // 报告日期
baseModelMap.put(BaseReportKeyEnum.REPORT_DATE.getKey(), DateUtil.format(new Date(), DatePattern.CHINESE_DATE_PATTERN)); baseModelMap.put(BaseReportKeyEnum.REPORT_DATE.getKey(), DateUtil.format(new Date(), DatePattern.CHINESE_DATE_PATTERN));
// 供电部门
baseModelMap.put(BaseReportKeyEnum.GD_NAME.getKey(), pqDevVO.getGdName());
// 变电站名称 // 变电站名称
baseModelMap.put(BaseReportKeyEnum.SUB_NAME.getKey(), pqDevVO.getSubName()); baseModelMap.put(BaseReportKeyEnum.SUB_NAME.getKey(), pqDevVO.getSubName());
// 检测人 // 检测人
@@ -2067,7 +2177,19 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
private void dealDataModel(List<InputStream> wordFileInputStreams, DevReportParam devReportParam, PqDevVO pqDevVO) throws Exception { private void dealDataModel(List<InputStream> wordFileInputStreams, DevReportParam devReportParam, PqDevVO pqDevVO) throws Exception {
Integer devChns = pqDevVO.getDevChns(); Integer devChns = pqDevVO.getDevChns();
for (int i = 1; i <= devChns; i++) { for (int i = 1; i <= devChns; i++) {
ClassPathResource resource = new ClassPathResource("/model/report_table.docx"); String path = "/model/report_table.docx";
DevType devType = devTypeService.getById(pqDevVO.getDevType());
if (devType.getName().equals("PQ-COM")) {
path = "/model/report_table - PQ-COM.docx";
}
ClassPathResource resource = new ClassPathResource(path);
XWPFDocument dataModelDocumentTemp = new XWPFDocument(resource.getInputStream());
SingleNonHarmParam singleNonHarmParam = new SingleNonHarmParam();
singleNonHarmParam.setPlanCode(devReportParam.getPlanCode());
singleNonHarmParam.setDevId(pqDevVO.getId());
singleNonHarmParam.setChannelNo(i);
// 获取数据 // 获取数据
Map<String, String> dataModelMap = new HashMap<>(16); Map<String, String> dataModelMap = new HashMap<>(16);
// 读取模板文件中的占位符 // 读取模板文件中的占位符
@@ -2085,11 +2207,20 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
List<SimAndDigHarmonicResult> adHarmonicResultList = adHarmonicService.listAllResultData(devReportParam.getScriptId(), devReportParam.getPlanCode(), devReportParam.getDevId() + "_" + i); List<SimAndDigHarmonicResult> adHarmonicResultList = adHarmonicService.listAllResultData(devReportParam.getScriptId(), devReportParam.getPlanCode(), devReportParam.getDevId() + "_" + i);
// 填充数据 // 填充数据
int finalI = i;
indexKeysMap.forEach((index, keys) -> { indexKeysMap.forEach((index, keys) -> {
String s1 = keys.stream().findFirst().get(); String s1 = keys.stream().findFirst().get();
// 谐波类 // 谐波类
if (DetectionCodeEnum.V2_50.getCode().equals(s1) || DetectionCodeEnum.I2_50.getCode().equals(s1) || DetectionCodeEnum.SV_1_49.getCode().equals(s1) || DetectionCodeEnum.SI_1_49.getCode().equals(s1)) { if (DetectionCodeEnum.V2_50.getCode().equals(s1) || DetectionCodeEnum.I2_50.getCode().equals(s1) || DetectionCodeEnum.SV_1_49.getCode().equals(s1) || DetectionCodeEnum.SI_1_49.getCode().equals(s1)) {
fillMapValueHarm(adHarmonicResultList, dataModelMap, keys, index); // 查找一下U1
double u1 = 57.74;
List<SimAndDigHarmonicResult> rawData = adHarmonicService.listAllRawData(devReportParam.getScriptId(), devReportParam.getPlanCode(), devReportParam.getDevId() + "_" + finalI);
SimAndDigHarmonicResult adHarmonicResult = rawData.stream().filter(obj -> obj.getAdType().equals(DetectionCodeEnum.V2_50.getCode())).sorted((obj1, obj2) -> obj2.getTimeId().compareTo(obj1.getTimeId())).findFirst().orElse(null);
if (ObjectUtil.isNotNull(adHarmonicResult)) {
String aValue1 = adHarmonicResult.getAValue1();
u1 = Double.parseDouble(aValue1);
}
fillMapValueHarm(adHarmonicResultList, dataModelMap, keys, index, u1);
} else { } else {
// 非谐波类 // 非谐波类
if (DetectionCodeEnum.V_UNBAN.getCode().equals(s1) || DetectionCodeEnum.FREQ.getCode().equals(s1)) { if (DetectionCodeEnum.V_UNBAN.getCode().equals(s1) || DetectionCodeEnum.FREQ.getCode().equals(s1)) {
@@ -2157,14 +2288,14 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
* @param keys key * @param keys key
* @param index index * @param index index
*/ */
private void fillMapValueHarm(List<SimAndDigHarmonicResult> allHarmonicResultList, Map<String, String> dataModelMap, Set<String> keys, String index) { private void fillMapValueHarm(List<SimAndDigHarmonicResult> allHarmonicResultList, Map<String, String> dataModelMap, Set<String> keys, String index, double baseValue) {
keys.forEach(key -> { keys.forEach(key -> {
List<SimAndDigHarmonicResult> resultList = allHarmonicResultList.stream().filter(obj -> obj.getAdType().equals(key) && obj.getSort().toString().equals(index)).collect(Collectors.toList()); List<SimAndDigHarmonicResult> resultList = allHarmonicResultList.stream().filter(obj -> obj.getAdType().equals(key) && obj.getSort().toString().equals(index)).collect(Collectors.toList());
if (CollectionUtil.isNotEmpty(resultList)) { if (CollectionUtil.isNotEmpty(resultList)) {
SimAndDigHarmonicResult adHarmonicResult = resultList.get(0); SimAndDigHarmonicResult adHarmonicResult = resultList.get(0);
if (Objects.nonNull(adHarmonicResult)) { if (Objects.nonNull(adHarmonicResult)) {
if (DetectionCodeEnum.V2_50.getCode().equals(key) || DetectionCodeEnum.SV_1_49.getCode().equals(key)) { if (DetectionCodeEnum.V2_50.getCode().equals(key) || DetectionCodeEnum.SV_1_49.getCode().equals(key)) {
fillHarm(dataModelMap, adHarmonicResult, index, key, 57.74, 100); fillHarm(dataModelMap, adHarmonicResult, index, key, baseValue, 100);
} }
if (DetectionCodeEnum.I2_50.getCode().equals(key) || DetectionCodeEnum.SI_1_49.getCode().equals(key)) { if (DetectionCodeEnum.I2_50.getCode().equals(key) || DetectionCodeEnum.SI_1_49.getCode().equals(key)) {
fillHarm(dataModelMap, adHarmonicResult, index, key, 1, 1); fillHarm(dataModelMap, adHarmonicResult, index, key, 1, 1);
@@ -2371,16 +2502,21 @@ public class PqReportServiceImpl extends ServiceImpl<PqReportMapper, PqReport> i
/** /**
* 创建回路标题到报告中 * 创建回路标题到报告中
*/ */
private P getContrastLineTitle(Map<String, List<Docx4jUtil.HeadingContent>> contentMap, int monitorNum, int index, ObjectFactory factory) { private P getContrastLineTitle(Map<String, List<Docx4jUtil.HeadingContent>> contentMap, int monitorNum, int index, ObjectFactory factory, PqDevVO pqDevVO) {
PqMonitor pqMonitor = pqMonitorService.getByDevIdAndNum(pqDevVO.getId(), monitorNum);
String monitorInfoName = "";
if (StrUtil.isNotBlank(pqDevVO.getSubName()) && Objects.nonNull(pqMonitor)) {
monitorInfoName = "(" + pqDevVO.getSubName() + "-" + pqMonitor.getName() + ")";
}
List<Docx4jUtil.HeadingContent> headingContents = contentMap.get(PowerIndexEnum.LINE_TITLE.getKey()); List<Docx4jUtil.HeadingContent> headingContents = contentMap.get(PowerIndexEnum.LINE_TITLE.getKey());
// 如果contentMap中有指定内容创建大纲级别为2的标题 // 如果contentMap中有指定内容创建大纲级别为2的标题
if (CollUtil.isNotEmpty(headingContents)) { if (CollUtil.isNotEmpty(headingContents)) {
return Docx4jUtil.createTitle(factory, 2, index + ".测量回路" + monitorNum, return Docx4jUtil.createTitle(factory, 2, index + ".测量回路" + monitorNum + monitorInfoName,
"SimSun", 30, true); "SimSun", 30, true);
} }
// 没有模板配置时,创建默认样式段落 // 没有模板配置时,创建默认样式段落
P titleParagraph = factory.createP(); P titleParagraph = factory.createP();
Docx4jUtil.createTitle(factory, titleParagraph, "测量回路" + monitorNum, Docx4jUtil.createTitle(factory, titleParagraph, "测量回路" + monitorNum + monitorInfoName,
28, true); 28, true);
return titleParagraph; return titleParagraph;
} }

View File

@@ -90,7 +90,7 @@ public class ResultController extends BaseController {
@OperateInfo(info = LogEnum.BUSINESS_COMMON, operateType = OperateType.UPDATE) @OperateInfo(info = LogEnum.BUSINESS_COMMON, operateType = OperateType.UPDATE)
@PostMapping("/reCalculate") @PostMapping("/reCalculate")
@ApiOperation("重新计算检测结果") @ApiOperation("重新计算检测结果")
@ApiImplicitParam(name = "queryParam", value = "查询参数", required = true) @ApiImplicitParam(name = "param", value = "重新计算参数", required = true)
public HttpResult<Object> reCalculate(@RequestBody @Validated ResultParam.ChangeErrorSystemParam param) { public HttpResult<Object> reCalculate(@RequestBody @Validated ResultParam.ChangeErrorSystemParam param) {
String methodDescribe = getMethodDescribe("reCalculate"); String methodDescribe = getMethodDescribe("reCalculate");
LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, param); LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, param);
@@ -115,7 +115,7 @@ public class ResultController extends BaseController {
@OperateInfo(info = LogEnum.BUSINESS_COMMON, operateType = OperateType.DELETE) @OperateInfo(info = LogEnum.BUSINESS_COMMON, operateType = OperateType.DELETE)
@GetMapping("/deleteTempTable") @GetMapping("/deleteTempTable")
@ApiOperation("删除临时表") @ApiOperation("删除临时表")
@ApiImplicitParam(name = "param", value = "删除参数", required = true) @ApiImplicitParam(name = "code", value = "计划对应的表后缀code", required = true)
public HttpResult<Object> deleteTempTable(@RequestParam("code") String code) { public HttpResult<Object> deleteTempTable(@RequestParam("code") String code) {
String methodDescribe = getMethodDescribe("deleteTempTable"); String methodDescribe = getMethodDescribe("deleteTempTable");
LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, code); LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, code);
@@ -127,7 +127,7 @@ public class ResultController extends BaseController {
@OperateInfo(info = LogEnum.BUSINESS_COMMON) @OperateInfo(info = LogEnum.BUSINESS_COMMON)
@PostMapping("/getContrastFormContent") @PostMapping("/getContrastFormContent")
@ApiOperation("获取比对式检测结果-表单内容") @ApiOperation("获取比对式检测结果-表单内容")
@ApiImplicitParam(name = "param", value = "查询参数", required = true) @ApiImplicitParam(name = "queryParam", value = "查询参数", required = true)
public HttpResult<FormContentVO> getContrastFormContent(@RequestBody @Validated ResultParam.QueryParam queryParam) { public HttpResult<FormContentVO> getContrastFormContent(@RequestBody @Validated ResultParam.QueryParam queryParam) {
String methodDescribe = getMethodDescribe("getContrastFormContent"); String methodDescribe = getMethodDescribe("getContrastFormContent");
LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, queryParam); LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, queryParam);
@@ -139,7 +139,7 @@ public class ResultController extends BaseController {
@OperateInfo(info = LogEnum.BUSINESS_COMMON) @OperateInfo(info = LogEnum.BUSINESS_COMMON)
@PostMapping("/getContrastResult") @PostMapping("/getContrastResult")
@ApiOperation("获取比对式检测结果") @ApiOperation("获取比对式检测结果")
@ApiImplicitParam(name = "param", value = "查询参数", required = true) @ApiImplicitParam(name = "queryParam", value = "查询参数", required = true)
public HttpResult<ContrastResultVO> getContrastResult(@RequestBody @Validated ResultParam.QueryParam queryParam) { public HttpResult<ContrastResultVO> getContrastResult(@RequestBody @Validated ResultParam.QueryParam queryParam) {
String methodDescribe = getMethodDescribe("getContrastResult"); String methodDescribe = getMethodDescribe("getContrastResult");
LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, queryParam); LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, queryParam);
@@ -154,7 +154,7 @@ public class ResultController extends BaseController {
@OperateInfo(info = LogEnum.BUSINESS_COMMON) @OperateInfo(info = LogEnum.BUSINESS_COMMON)
@PostMapping("/getMonitorResult") @PostMapping("/getMonitorResult")
@ApiOperation("获取监测点的检测结果") @ApiOperation("获取监测点的检测结果")
@ApiImplicitParam(name = "param", value = "查询参数", required = true) @ApiImplicitParam(name = "devId", value = "设备id", required = true)
public HttpResult<List<MonitorResultVO>> getMonitorResult(@RequestParam("devId") String devId) { public HttpResult<List<MonitorResultVO>> getMonitorResult(@RequestParam("devId") String devId) {
String methodDescribe = getMethodDescribe("getMonitorResult"); String methodDescribe = getMethodDescribe("getMonitorResult");
LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, devId); LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, devId);
@@ -180,7 +180,7 @@ public class ResultController extends BaseController {
@OperateInfo(info = LogEnum.BUSINESS_COMMON) @OperateInfo(info = LogEnum.BUSINESS_COMMON)
@PostMapping("/updateMonitorResult") @PostMapping("/updateMonitorResult")
@ApiOperation("更新监测点的检测结果") @ApiOperation("更新监测点的检测结果")
@ApiImplicitParam(name = "result", value = "更新内容", required = true) @ApiImplicitParam(name = "resultParams", value = "更新内容", required = true)
public HttpResult<Boolean> updateMonitorResult(@RequestBody @Validated MonitorResultVO resultParams) { public HttpResult<Boolean> updateMonitorResult(@RequestBody @Validated MonitorResultVO resultParams) {
String methodDescribe = getMethodDescribe("updateMonitorResult"); String methodDescribe = getMethodDescribe("updateMonitorResult");
LogUtil.njcnDebug(log, "{},更新数据为:{}", methodDescribe, resultParams); LogUtil.njcnDebug(log, "{},更新数据为:{}", methodDescribe, resultParams);
@@ -198,7 +198,7 @@ public class ResultController extends BaseController {
String methodDescribe = getMethodDescribe("getCheckItem"); String methodDescribe = getMethodDescribe("getCheckItem");
LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, monitorQueryParam); LogUtil.njcnDebug(log, "{},查询数据为:{}", methodDescribe, monitorQueryParam);
List<ContrastTestItemVO> result = resultService.getCheckItem(monitorQueryParam.getDevId(),monitorQueryParam.getChnNum(), monitorQueryParam.getNum()); List<ContrastTestItemVO> result = resultService.getCheckItem(monitorQueryParam.getPlanId(), monitorQueryParam.getDevId(), monitorQueryParam.getChnNum(), monitorQueryParam.getNum(), monitorQueryParam.getCode());
return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, result, methodDescribe); return HttpResultUtil.assembleCommonResponseResult(CommonResponseEnum.SUCCESS, result, methodDescribe);
} }
} }

View File

@@ -15,4 +15,8 @@ public class MonitorQueryParam {
private Integer num; private Integer num;
private Integer waveNum; private Integer waveNum;
private String planId;
private String code;
} }

View File

@@ -2,6 +2,7 @@ package com.njcn.gather.result.pojo.param;
import com.njcn.common.pojo.constant.PatternRegex; import com.njcn.common.pojo.constant.PatternRegex;
import com.njcn.gather.pojo.constant.DetectionValidMessage; import com.njcn.gather.pojo.constant.DetectionValidMessage;
import io.swagger.annotations.ApiImplicitParams;
import io.swagger.annotations.ApiModelProperty; import io.swagger.annotations.ApiModelProperty;
import lombok.Data; import lombok.Data;
@@ -74,6 +75,8 @@ public class ResultParam {
// 模式id // 模式id
private String patternId; private String patternId;
private String code;
} }
@Data @Data
@@ -83,7 +86,7 @@ public class ResultParam {
@Pattern(regexp = PatternRegex.SYSTEM_ID, message = DetectionValidMessage.PLAN_ID_FORMAT_ERROR) @Pattern(regexp = PatternRegex.SYSTEM_ID, message = DetectionValidMessage.PLAN_ID_FORMAT_ERROR)
private String planId; private String planId;
@ApiModelProperty(value = "脚本Id", required = false) @ApiModelProperty(value = "脚本Id")
private String scriptId; private String scriptId;
@ApiModelProperty(value = "误差体系Id", required = true) @ApiModelProperty(value = "误差体系Id", required = true)
@@ -95,9 +98,16 @@ public class ResultParam {
@Pattern(regexp = PatternRegex.SYSTEM_ID, message = DetectionValidMessage.DEV_ID_FORMAT_ERROR) @Pattern(regexp = PatternRegex.SYSTEM_ID, message = DetectionValidMessage.DEV_ID_FORMAT_ERROR)
private String deviceId; private String deviceId;
@ApiModelProperty(value = "计划对应表后缀", required = true)
private String code; private String code;
@ApiModelProperty(value = "模式Id", required = true) @ApiModelProperty(value = "模式Id", required = true)
private String patternId; private String patternId;
@ApiModelProperty(value = "通道号", required = true)
private String chnNum;
@ApiModelProperty(value = "数据处理原则", required = true)
private String dataRuleId;
} }
} }

View File

@@ -14,58 +14,38 @@ import javax.validation.constraints.NotNull;
@Data @Data
public class MonitorResultVO implements Comparable<MonitorResultVO> { public class MonitorResultVO implements Comparable<MonitorResultVO> {
/**
* 监测点id
*/
@ApiModelProperty(value = "监测点id", required = true) @ApiModelProperty(value = "监测点id", required = true)
@NotBlank(message = DetectionValidMessage.DEV_MONITOR_ID_NOT_BLANK) @NotBlank(message = DetectionValidMessage.DEV_MONITOR_ID_NOT_BLANK)
private String monitorId; private String monitorId;
/** @ApiModelProperty(value = "监测点序号")
* 监测点序号
*/
private Integer monitorNum; private Integer monitorNum;
/** @ApiModelProperty(value = "总检测次数")
* 总检测次数
*/
private Integer totalNum; private Integer totalNum;
/** @ApiModelProperty(value = "合格检测次数")
* 合格检测次数
*/
private Integer qualifiedNum; private Integer qualifiedNum;
/** @ApiModelProperty(value = "不合格检测次数")
* 不合格检测次数
*/
private Integer unQualifiedNum; private Integer unQualifiedNum;
/** @ApiModelProperty(value = "误差体系名称")
* 误差体系名称
*/
private String errorSysName; private String errorSysName;
/**
* 检测结果
*/
@ApiModelProperty(value = "检测结果", required = true) @ApiModelProperty(value = "检测结果", required = true)
@NotNull(message = DetectionValidMessage.DEV_MONITOR_RESULT_NOT_NULL) @NotNull(message = DetectionValidMessage.DEV_MONITOR_RESULT_NOT_NULL)
private Integer checkResult; private Integer checkResult;
/**
* 结论来源 @ApiModelProperty(value = "结论来源")
*/
private String resultOrigin; private String resultOrigin;
/**
* 哪次
*/
@ApiModelProperty(value = "使用哪次检测结果", required = true) @ApiModelProperty(value = "使用哪次检测结果", required = true)
@NotNull(message = DetectionValidMessage.DEV_MONITOR_RESULT_NUM_NOT_BLANK) @NotNull(message = DetectionValidMessage.DEV_MONITOR_RESULT_NUM_NOT_BLANK)
private String whichTime; private String whichTime;
/**
* 数据源类型
*/
@ApiModelProperty(value = "数据源类型", required = true) @ApiModelProperty(value = "数据源类型", required = true)
@NotBlank(message = DetectionValidMessage.DEV_MONITOR_RESULT_TYPE_NOT_BLANK) @NotBlank(message = DetectionValidMessage.DEV_MONITOR_RESULT_TYPE_NOT_BLANK)
private String resultType; private String resultType;

Some files were not shown because too many files have changed in this diff Show More