diff --git a/das/src/main/java/com/das/modules/auth/controller/SysMenusController.java b/das/src/main/java/com/das/modules/auth/controller/SysMenusController.java index 45b33017..75bd2071 100644 --- a/das/src/main/java/com/das/modules/auth/controller/SysMenusController.java +++ b/das/src/main/java/com/das/modules/auth/controller/SysMenusController.java @@ -41,6 +41,9 @@ public class SysMenusController { if(!hasPermission){ return R.fail("没有系统管理权限"); } + if (!sysMenuDto.getMenuName().matches("^[a-zA-Z]*$")){ + return R.fail("菜单名称,必须为英文"); + } return R.success(sysMenuService.createMenu(sysMenuDto)); } @@ -55,7 +58,7 @@ public class SysMenusController { if(!hasPermission){ return R.fail("没有系统管理权限"); } - if (StringUtils.isAllEnglishLetters(sysMenuDto.getMenuName())){ + if (!sysMenuDto.getMenuName().matches("^[a-zA-Z]*$")){ return R.fail("菜单名称,必须为英文"); } sysMenuService.updateMenu(sysMenuDto); diff --git a/das/src/main/java/com/das/modules/data/service/DataService.java b/das/src/main/java/com/das/modules/data/service/DataService.java index 7d1e75e6..4cd55fdc 100644 --- a/das/src/main/java/com/das/modules/data/service/DataService.java +++ b/das/src/main/java/com/das/modules/data/service/DataService.java @@ -49,12 +49,12 @@ public class DataService { //为空查全部 List sysIotModelFields = sysIotModelFieldMapper.queryAllFiledNames(Long.valueOf(snapshotValueQueryParam.getDeviceId())); for (String item : sysIotModelFields) { - String key = String.format("RT:[%s]:[%s]", snapshotValueQueryParam.getDeviceId(), item); + String key = String.format("RT:%s:%s", snapshotValueQueryParam.getDeviceId(), item); keyList.add(key); } } else { for (String item : attributes) { - String key = String.format("RT:[%s]:[%s]", snapshotValueQueryParam.getDeviceId(), item); + String key = String.format("RT:%s:%s", snapshotValueQueryParam.getDeviceId(), item); keyList.add(key); } } @@ -62,12 +62,10 @@ public class DataService { List dataList = adminRedisTemplate.mGet(keyList); for (int i = 0; i < keyList.size(); i++) { String key = keyList.get(i); - int firstColonIndex = key.indexOf('['); - int firstIndex = key.indexOf(']'); - int secondIndex = key.indexOf(']', firstIndex + 1); - int secondColonIndex = key.indexOf('[', firstColonIndex + 1); - String deviceId = key.substring(firstColonIndex + 1, firstIndex); - String fieldName = key.substring(secondColonIndex + 1, secondIndex); + int firstColonIndex = key.indexOf(':'); + int secondColonIndex = key.indexOf(':', firstColonIndex + 1); + String deviceId = key.substring(firstColonIndex + 1, secondColonIndex); + String fieldName = key.substring(secondColonIndex + 1); if (finalResult.get(deviceId) == null) { Map valueMap = new HashMap<>(); valueMap.put(fieldName, dataList.get(i)); diff --git a/das/src/main/java/com/das/modules/equipment/controller/SysEnumController.java b/das/src/main/java/com/das/modules/equipment/controller/SysEnumController.java index ba161a15..dda36d3f 100644 --- a/das/src/main/java/com/das/modules/equipment/controller/SysEnumController.java +++ b/das/src/main/java/com/das/modules/equipment/controller/SysEnumController.java @@ -62,6 +62,23 @@ public class SysEnumController { } + /** 更新枚举值 */ + @PostMapping("/updateEnumValues") + public R updateEnumValues(@RequestBody SysEnumValuesDto sysEnumValuesDto) { + //判断是否有权限 + boolean hasPermission = StpUtil.hasPermission(SysAuthorityIds.SYS_AUTHORITY_ID_DEVICE_MGR.toString()); + if(!hasPermission){ + return R.fail("没有设备管理权限"); + } + if (sysEnumValuesDto.getEnumTypeId() ==null && StringUtils.isBlank(sysEnumValuesDto.getValue()) + || sysEnumValuesDto.getOrderNumber() == null || sysEnumValuesDto.getIsActive() ==null || + sysEnumValuesDto.getId() ==null) { + throw new ServiceException("参数缺失"); + } + SysEnumValuesVo sysEnumValuesVo = sysEnumService.updateEnumValues(sysEnumValuesDto); + return R.success(sysEnumValuesVo); + } + /** * 查询枚举类型列表 * @@ -72,7 +89,7 @@ public class SysEnumController { } /** - * 查询枚举类型列表 + * 查询枚举值列表 * * */ @PostMapping("/queryEnumValuesList") diff --git a/das/src/main/java/com/das/modules/equipment/domain/vo/SysEnumValuesVo.java b/das/src/main/java/com/das/modules/equipment/domain/vo/SysEnumValuesVo.java index 836102ad..ff847d10 100644 --- a/das/src/main/java/com/das/modules/equipment/domain/vo/SysEnumValuesVo.java +++ b/das/src/main/java/com/das/modules/equipment/domain/vo/SysEnumValuesVo.java @@ -19,6 +19,10 @@ public class SysEnumValuesVo { * 关联到枚举类型表的ID */ private Long enumTypeId; + /** + * 关联到枚举类型表的ID字符串类型 + */ + private String enumTypeIdStr; /** * 枚举值的具体内容 diff --git a/das/src/main/java/com/das/modules/equipment/mapper/SysEnumTypesMapper.java b/das/src/main/java/com/das/modules/equipment/mapper/SysEnumTypesMapper.java index 1bbc8308..b1706778 100644 --- a/das/src/main/java/com/das/modules/equipment/mapper/SysEnumTypesMapper.java +++ b/das/src/main/java/com/das/modules/equipment/mapper/SysEnumTypesMapper.java @@ -1,9 +1,13 @@ package com.das.modules.equipment.mapper; import com.baomidou.mybatisplus.core.mapper.BaseMapper; +import com.das.modules.equipment.domain.dto.SysEnumTypesDto; import com.das.modules.equipment.entity.SysEnumTypes; import org.apache.ibatis.annotations.Mapper; +import org.apache.ibatis.annotations.Param; @Mapper public interface SysEnumTypesMapper extends BaseMapper { + + Long queryEnumTypesByName(@Param("sysDto") SysEnumTypesDto sysDto); } diff --git a/das/src/main/java/com/das/modules/equipment/service/SysEnumService.java b/das/src/main/java/com/das/modules/equipment/service/SysEnumService.java index 596cc9c9..ce4f13e1 100644 --- a/das/src/main/java/com/das/modules/equipment/service/SysEnumService.java +++ b/das/src/main/java/com/das/modules/equipment/service/SysEnumService.java @@ -14,6 +14,8 @@ public interface SysEnumService { SysEnumValuesVo addEnumValues(SysEnumValuesDto sysEnumValuesDto); + SysEnumValuesVo updateEnumValues(SysEnumValuesDto sysEnumValuesDto); + List queryEnumTypesList(); PageDataInfo queryEnumValuesList(SysEnumValuesDto sysEnumValuesDto); diff --git a/das/src/main/java/com/das/modules/equipment/service/impl/SysEnumServiceImpl.java b/das/src/main/java/com/das/modules/equipment/service/impl/SysEnumServiceImpl.java index 1d2c7abe..12e9b263 100644 --- a/das/src/main/java/com/das/modules/equipment/service/impl/SysEnumServiceImpl.java +++ b/das/src/main/java/com/das/modules/equipment/service/impl/SysEnumServiceImpl.java @@ -1,8 +1,10 @@ package com.das.modules.equipment.service.impl; import cn.dev33.satoken.stp.StpUtil; +import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; import com.baomidou.mybatisplus.core.metadata.IPage; import com.das.common.config.SessionUtil; +import com.das.common.exceptions.ServiceException; import com.das.common.utils.BeanCopyUtils; import com.das.common.utils.PageDataInfo; import com.das.common.utils.PageQuery; @@ -35,13 +37,18 @@ public class SysEnumServiceImpl implements SysEnumService { /** 新增枚举类型 */ @Override public SysEnumTypesVo addEnumTypes(SysEnumTypesDto sysEnumTypesDto) { + //判断枚举类型,是否重复 + Long count = sysEnumTypesMapper.queryEnumTypesByName(sysEnumTypesDto); + if (count >0){ + throw new ServiceException("枚举类型,不可重复!"); + } SysEnumTypes sysEnumTypes = new SysEnumTypes(); BeanCopyUtils.copy(sysEnumTypesDto, sysEnumTypes); SysUserVo sysUserVo = (SysUserVo) StpUtil.getTokenSession().get(SessionUtil.SESSION_USER_KEY); - sysEnumTypes.setCreatedTime(new Date()); - sysEnumTypes.setUpdatedTime(new Date()); sysEnumTypes.setCreatedBy(sysUserVo.getAccount()); sysEnumTypes.setUpdatedBy(sysUserVo.getAccount()); + sysEnumTypes.setCreatedTime(new Date()); + sysEnumTypes.setUpdatedTime(new Date()); sysEnumTypes.setRevision(1); sysEnumTypesMapper.insert(sysEnumTypes); SysEnumTypesVo sysEnumTypesVo = new SysEnumTypesVo(); @@ -54,11 +61,18 @@ public class SysEnumServiceImpl implements SysEnumService { public SysEnumValuesVo addEnumValues(SysEnumValuesDto sysEnumValuesDto) { SysEnumValues sysEnumValues = new SysEnumValues(); BeanCopyUtils.copy(sysEnumValuesDto,sysEnumValues); + QueryWrapper queryWrapper = new QueryWrapper<>(); + queryWrapper.eq("enum_type_id",sysEnumValuesDto.getEnumTypeId()); + queryWrapper.eq("description",sysEnumValuesDto.getDescription()); + Long count = sysEnumValuesMapper.selectCount(queryWrapper); + if (count >0){ + throw new ServiceException("枚举值描述,不可重复!"); + } SysUserVo sysUserVo = (SysUserVo) StpUtil.getTokenSession().get(SessionUtil.SESSION_USER_KEY); - sysEnumValues.setCreatedTime(new Date()); - sysEnumValues.setUpdatedTime(new Date()); sysEnumValues.setCreatedBy(sysUserVo.getAccount()); sysEnumValues.setUpdatedBy(sysUserVo.getAccount()); + sysEnumValues.setCreatedTime(new Date()); + sysEnumValues.setUpdatedTime(new Date()); sysEnumValues.setRevision(1); sysEnumValuesMapper.insert(sysEnumValues); SysEnumValuesVo sysEnumValuesVo = new SysEnumValuesVo(); @@ -66,6 +80,26 @@ public class SysEnumServiceImpl implements SysEnumService { return sysEnumValuesVo; } + @Override + public SysEnumValuesVo updateEnumValues(SysEnumValuesDto sysEnumValuesDto) { + SysEnumValues sysEnumValues = new SysEnumValues(); + BeanCopyUtils.copy(sysEnumValuesDto,sysEnumValues); + QueryWrapper queryWrapper = new QueryWrapper<>(); + queryWrapper.eq("enum_type_id",sysEnumValuesDto.getEnumTypeId()); + queryWrapper.eq("description",sysEnumValuesDto.getDescription()); + Long count = sysEnumValuesMapper.selectCount(queryWrapper); + if (count >0){ + throw new ServiceException("枚举值描述,不可重复!"); + } + SysUserVo sysUserVo = (SysUserVo) StpUtil.getTokenSession().get(SessionUtil.SESSION_USER_KEY); + sysEnumValues.setUpdatedBy(sysUserVo.getAccount()); + sysEnumValues.setUpdatedTime(new Date()); + sysEnumValuesMapper.updateById(sysEnumValues); + SysEnumValuesVo sysEnumValuesVo = new SysEnumValuesVo(); + BeanCopyUtils.copy(sysEnumValues, sysEnumValuesVo); + return sysEnumValuesVo; + } + /** * 查询枚举类型列表 * @return diff --git a/das/src/main/java/com/das/modules/equipment/service/impl/SysEquipmentServiceImpl.java b/das/src/main/java/com/das/modules/equipment/service/impl/SysEquipmentServiceImpl.java index c9be0752..159eb70a 100644 --- a/das/src/main/java/com/das/modules/equipment/service/impl/SysEquipmentServiceImpl.java +++ b/das/src/main/java/com/das/modules/equipment/service/impl/SysEquipmentServiceImpl.java @@ -60,6 +60,8 @@ public class SysEquipmentServiceImpl implements SysEquipmentService { @Override public SysEquipmentVo creatSysEquipment(SysEquipmentDto sysEquipmentDto) { + //去除空格 + sysEquipmentDto.setCode(sysEquipmentDto.getCode().replaceAll(" ", "")); SysEquipment sysEquipment = new SysEquipment(); BeanCopyUtils.copy(sysEquipmentDto, sysEquipment); @@ -81,6 +83,8 @@ public class SysEquipmentServiceImpl implements SysEquipmentService { @Override public SysEquipmentVo updateSysEquipment(SysEquipmentDto sysEquipmentDto) { + //去除空格 + sysEquipmentDto.setCode(sysEquipmentDto.getCode().replaceAll(" ", "")); SysEquipment sysEquipment = new SysEquipment(); BeanCopyUtils.copy(sysEquipmentDto, sysEquipment); SysUserVo sysUserVo = (SysUserVo) StpUtil.getTokenSession().get(SessionUtil.SESSION_USER_KEY); diff --git a/das/src/main/java/com/das/modules/equipment/service/impl/SysIotModelServiceImpl.java b/das/src/main/java/com/das/modules/equipment/service/impl/SysIotModelServiceImpl.java index 18aed703..cddb1293 100644 --- a/das/src/main/java/com/das/modules/equipment/service/impl/SysIotModelServiceImpl.java +++ b/das/src/main/java/com/das/modules/equipment/service/impl/SysIotModelServiceImpl.java @@ -1,6 +1,7 @@ package com.das.modules.equipment.service.impl; import cn.dev33.satoken.stp.StpUtil; +import cn.hutool.core.collection.ListUtil; import cn.hutool.poi.excel.ExcelReader; import cn.hutool.poi.excel.ExcelUtil; import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; @@ -25,6 +26,7 @@ import com.das.modules.equipment.mapper.SysIotModelFieldMapper; import com.das.modules.equipment.mapper.SysIotModelMapper; import com.das.modules.equipment.mapper.SysIotModelServiceMapper; import com.das.modules.equipment.service.SysIotModelService; +import com.das.modules.node.service.TDEngineService; import jakarta.servlet.http.HttpServletRequest; import jakarta.servlet.http.HttpServletResponse; import lombok.extern.slf4j.Slf4j; @@ -35,15 +37,13 @@ import org.springframework.transaction.annotation.Transactional; import org.springframework.web.multipart.MultipartFile; import java.io.InputStream; -import java.util.ArrayList; -import java.util.Date; -import java.util.LinkedHashMap; -import java.util.List; +import java.util.*; @Transactional(rollbackFor = Exception.class) @Service @Slf4j public class SysIotModelServiceImpl implements SysIotModelService { + public static final int COMMIT_COUNT = 1000; @Autowired private SysIotModelMapper sysIotModelMapper; @@ -54,6 +54,8 @@ public class SysIotModelServiceImpl implements SysIotModelService { @Autowired private SysIotModelServiceMapper sysIotModelServiceMapper; + @Autowired + private TDEngineService tdEngineService; public SysIotModelVo creatSysIotModel(SysIotModelDto sysIotModelDto) { SysIotModel sysIotModel = new SysIotModel(); @@ -162,6 +164,8 @@ public class SysIotModelServiceImpl implements SysIotModelService { sysIotModelField.setDataType("tinyint"); } + //动态执行创建td超级表,字段 + createTdStableOrColumn(sysIotModelField); sysIotModelFieldMapper.insert(sysIotModelField); SysIotModelFieldVo sysIotModelFieldVo = new SysIotModelFieldVo(); BeanCopyUtils.copy(sysIotModelField, sysIotModelFieldVo); @@ -188,6 +192,15 @@ public class SysIotModelServiceImpl implements SysIotModelService { } sysIotModelFieldMapper.updateById(sysIotModelField); + + SysIotModelField oldSysIotField = sysIotModelFieldMapper.selectById(sysIotModelFieldDto.getId()); + if (oldSysIotField == null){ + throw new ServiceException("未查找到该条记录"); + } + if (!oldSysIotField.getAttributeCode().equals(sysIotModelField.getAttributeCode()) && oldSysIotField.getDataType().equals(sysIotModelField.getDataType()) && Objects.equals(oldSysIotField.getHighSpeed(), sysIotModelField.getHighSpeed())){ + //更新td表结构 + updateTDStableOrColumn(sysIotModelField,oldSysIotField); + } SysIotModelFieldVo sysIotModelFieldVo = new SysIotModelFieldVo(); BeanCopyUtils.copy(sysIotModelField, sysIotModelFieldVo); return sysIotModelFieldVo; @@ -195,6 +208,8 @@ public class SysIotModelServiceImpl implements SysIotModelService { @Override public void deleteSysIotModelField(SysIotModelFieldDto sysIotModelFieldDto) { + SysIotModelField sysIotModelField = sysIotModelFieldMapper.selectById(sysIotModelFieldDto.getId()); + deleteTDStableOrColumn(sysIotModelField); sysIotModelFieldMapper.deleteById(sysIotModelFieldDto.getId()); } @@ -401,11 +416,35 @@ public class SysIotModelServiceImpl implements SysIotModelService { } } } - sysIotModelFieldMapper.insertBatch(sysIotModelFieldList); + if (CollectionUtils.isNotEmpty(sysIotModelFieldList)) { + sysIotModelFieldMapper.insertBatch(sysIotModelFieldList); + ListUtil.page(sysIotModelFieldList, COMMIT_COUNT, create -> { + for (SysIotModelField item : create){ + createTdStableOrColumn(item); + } + }); + } if (CollectionUtils.isNotEmpty(updateSysIotModelFieldList)) { + ListUtil.page(updateSysIotModelFieldList, COMMIT_COUNT, update -> { + for (SysIotModelField item : update){ + SysIotModelField oldSysIotField = sysIotModelFieldMapper.selectById(item.getId()); + if (oldSysIotField != null){ + if (!oldSysIotField.getAttributeCode().equals(item.getAttributeCode()) && oldSysIotField.getDataType().equals(item.getDataType()) && Objects.equals(oldSysIotField.getHighSpeed(), item.getHighSpeed())){ + //更新td表结构 + updateTDStableOrColumn(item,oldSysIotField); + } + } + } + }); sysIotModelFieldMapper.updateBatchById(updateSysIotModelFieldList); + } if (CollectionUtils.isNotEmpty(delSysIotModelFieldList)) { + ListUtil.page(updateSysIotModelFieldList, COMMIT_COUNT, delete -> { + for (SysIotModelField item : delete){ + deleteTDStableOrColumn(item); + } + }); sysIotModelFieldMapper.deleteBatchIds(delSysIotModelFieldList); } } @@ -428,4 +467,70 @@ public class SysIotModelServiceImpl implements SysIotModelService { return list; } + public void createTdStableOrColumn(SysIotModelField sysIotModelField){ + //如果新增的是第一条记录 创建tdengine超级表 分为高频和低频 + QueryWrapper queryWrapper = new QueryWrapper<>(); + queryWrapper.eq("iot_model_id",sysIotModelField.getIotModelId()); + queryWrapper.eq("highspeed",sysIotModelField.getHighSpeed()); + List sysIotModelFields = sysIotModelFieldMapper.selectList(queryWrapper); + if (CollectionUtils.isEmpty(sysIotModelFields)){ + if (sysIotModelField.getHighSpeed() == 0){ + SysIotModel sysIotModel = sysIotModelMapper.selectById(sysIotModelField.getIotModelId()); + Map map = new HashMap<>(); + map.put(sysIotModelField.getAttributeCode(), sysIotModelField.getDataType()); + //创建低频超级表 + tdEngineService.createStable(sysIotModel.getIotModelCode(),"l_",map); + } + if (sysIotModelField.getHighSpeed() == 1){ + SysIotModel sysIotModel = sysIotModelMapper.selectById(sysIotModelField.getIotModelId()); + Map map = new HashMap<>(); + map.put(sysIotModelField.getAttributeCode(), sysIotModelField.getDataType()); + //创建高频超级表 + tdEngineService.createStable(sysIotModel.getIotModelCode(),"h_",map); + } + }else { + //stable已经存在,新增stable列 + if (sysIotModelField.getHighSpeed() == 0){ + SysIotModel sysIotModel = sysIotModelMapper.selectById(sysIotModelField.getIotModelId()); + Map map = new HashMap<>(); + map.put(sysIotModelField.getAttributeCode(), sysIotModelField.getDataType()); + //创建低频超级表 + tdEngineService.addStableColumn(sysIotModel.getIotModelCode(),"l_",map); + } + if (sysIotModelField.getHighSpeed() == 1){ + SysIotModel sysIotModel = sysIotModelMapper.selectById(sysIotModelField.getIotModelId()); + Map map = new HashMap<>(); + map.put(sysIotModelField.getAttributeCode(), sysIotModelField.getDataType()); + //创建高频超级表 + tdEngineService.addStableColumn(sysIotModel.getIotModelCode(),"h_",map); + } + } + } + + public void updateTDStableOrColumn(SysIotModelField sysIotModelField,SysIotModelField oldIotModelField){ + deleteTDStableOrColumn(oldIotModelField); + createTdStableOrColumn(sysIotModelField); + } + + public void deleteTDStableOrColumn(SysIotModelField sysIotModelField){ + String stableName = null; + SysIotModel sysIotModel = sysIotModelMapper.selectById(sysIotModelField.getIotModelId()); + if(sysIotModelField.getHighSpeed() == 0) { + stableName = "l_" + sysIotModel.getIotModelCode(); + } + if(sysIotModelField.getHighSpeed() == 1) { + stableName = "h_" + sysIotModel.getIotModelCode(); + } + QueryWrapper queryWrapper = new QueryWrapper<>(); + queryWrapper.eq("iot_model_id",sysIotModelField.getIotModelId()); + queryWrapper.eq("highspeed",sysIotModelField.getHighSpeed()); + List sysIotModelFields = sysIotModelFieldMapper.selectList(queryWrapper); + if (CollectionUtils.isNotEmpty(sysIotModelFields) && sysIotModelFields.size() > 1){ + tdEngineService.deleteColumn(stableName,sysIotModelField.getAttributeCode()); + } + //物模型属性只剩下最后一个删除表 + if (CollectionUtils.isNotEmpty(sysIotModelFields) && sysIotModelFields.size() == 1){ + tdEngineService.deleteStable(stableName); + } + } } diff --git a/das/src/main/java/com/das/modules/node/domain/vo/ConfigUpdateVo.java b/das/src/main/java/com/das/modules/node/domain/vo/ConfigUpdateVo.java index ac8e33e2..96c4713c 100644 --- a/das/src/main/java/com/das/modules/node/domain/vo/ConfigUpdateVo.java +++ b/das/src/main/java/com/das/modules/node/domain/vo/ConfigUpdateVo.java @@ -19,7 +19,7 @@ public class ConfigUpdateVo { private List links; - private List equipments; + private List equipments; public ConfigUpdateVo() { diff --git a/das/src/main/java/com/das/modules/node/mapper/SysImptabmappingMapper.java b/das/src/main/java/com/das/modules/node/mapper/SysImptabmappingMapper.java index b4e1a9ce..7bba68ef 100644 --- a/das/src/main/java/com/das/modules/node/mapper/SysImptabmappingMapper.java +++ b/das/src/main/java/com/das/modules/node/mapper/SysImptabmappingMapper.java @@ -28,7 +28,9 @@ public interface SysImptabmappingMapper extends BaseMapperPlus getEquipmentId(Long linkId); - List getIotModelFieldByEquipmentId(Long equipmentId); + List getIotModelFieldByDeviceId(@Param("deviceId") Long deviceId); + + List getIotModelServiceByDeviceId(@Param("deviceId") Long deviceId); List getIotModelServiceByEquipmentId(Long equipmentId); diff --git a/das/src/main/java/com/das/modules/node/service/TDEngineService.java b/das/src/main/java/com/das/modules/node/service/TDEngineService.java index d97ef9e2..44276b15 100644 --- a/das/src/main/java/com/das/modules/node/service/TDEngineService.java +++ b/das/src/main/java/com/das/modules/node/service/TDEngineService.java @@ -61,6 +61,7 @@ public class TDEngineService { /** * 创建超级表 */ + @Async public void createStable(String iotModelCode, String stableType, Map fieldNameTypeMap) { try (Connection conn = hikariDataSource.getConnection(); Statement pstmt = conn.createStatement()) { @@ -94,6 +95,7 @@ public class TDEngineService { /** * 新增超级表列 */ + @Async public void addStableColumn(String iotModelCode, String stableType, Map fieldNameTypeMap) { try (Connection conn = hikariDataSource.getConnection(); Statement pstmt = conn.createStatement()) { @@ -124,6 +126,7 @@ public class TDEngineService { /** * 删除超级表列 */ + @Async public void deleteColumn(String stableName, String fieldCode) { try (Connection conn = hikariDataSource.getConnection(); Statement pstmt = conn.createStatement()) { @@ -131,7 +134,7 @@ public class TDEngineService { sb.setLength(0); sb.append("ALTER STABLE "); sb.append(stableName); - sb.append(" DROP COLUMN"); + sb.append(" DROP COLUMN "); ; sb.append(fieldCode); sb.append(";"); @@ -149,6 +152,7 @@ public class TDEngineService { /** * 删除超级表 */ + @Async public void deleteStable(String stableName) { try (Connection conn = hikariDataSource.getConnection(); Statement pstmt = conn.createStatement()) { diff --git a/das/src/main/java/com/das/modules/node/service/impl/DataServiceImpl.java b/das/src/main/java/com/das/modules/node/service/impl/DataServiceImpl.java index 22eb2074..4d5a30a7 100644 --- a/das/src/main/java/com/das/modules/node/service/impl/DataServiceImpl.java +++ b/das/src/main/java/com/das/modules/node/service/impl/DataServiceImpl.java @@ -141,13 +141,12 @@ public class DataServiceImpl implements DataService { ObjectNode equipJsonNode = equipObjectMapper.convertValue(map, ObjectNode.class); List newIotModelFieldList = new ArrayList<>(); List newIotModelServiceList = new ArrayList<>(); - // 获取设备IOT地址 - String iotAddr = sysImptabmappingMapper.getIotAddrByEquipmentId(equipmentId); // 根据设备Id获取对应的物模型属性和动作 - List iotModelFieldList = sysImptabmappingMapper.getIotModelFieldByEquipmentId(equipmentId); + List iotModelFieldList = sysImptabmappingMapper.getIotModelFieldByDeviceId(equipmentId); + List iotModelServiceList = sysImptabmappingMapper.getIotModelServiceByDeviceId(equipmentId); if (!CollectionUtils.isEmpty(iotModelFieldList)) { for (IotModelVo info : iotModelFieldList) { - if (info.getServiceType() == null) { + NewIotModelVo newIotModelVo = new NewIotModelVo(); newIotModelVo.setName(info.getEquipmentAttribute()); newIotModelVo.setHighSpeed(info.getHighSpeed()); @@ -164,21 +163,23 @@ public class DataServiceImpl implements DataService { newIotModelVo.setType("yx"); } newIotModelFieldList.add(newIotModelVo); - } else if (info.getAttributeType() == null) { - NewIotModelVo newIotModelVo = new NewIotModelVo(); - newIotModelVo.setName(info.getEquipmentService()); - if (info.getParams() == null) { - newIotModelVo.setParams(equipJsonNode); - } else { - newIotModelVo.setParams(equipObjectMapper.readTree(info.getParams().toString())); - } - if (info.getServiceType() == BaseIotModelType.TYPE_PSR_SETPOINT) { - newIotModelVo.setType("yt"); - } else if (info.getServiceType() == BaseIotModelType.TYPE_PSR_CONTROL) { - newIotModelVo.setType("yk"); - } - newIotModelServiceList.add(newIotModelVo); + } + } + if (!CollectionUtils.isEmpty(iotModelServiceList)){ + for (IotModelVo info : iotModelServiceList) { + NewIotModelVo newIotModelVo = new NewIotModelVo(); + newIotModelVo.setName(info.getEquipmentService()); + if (info.getParams() == null) { + newIotModelVo.setParams(equipJsonNode); + } else { + newIotModelVo.setParams(equipObjectMapper.readTree(info.getParams().toString())); } + if (info.getServiceType() == BaseIotModelType.TYPE_PSR_SETPOINT) { + newIotModelVo.setType("yt"); + } else if (info.getServiceType() == BaseIotModelType.TYPE_PSR_CONTROL) { + newIotModelVo.setType("yk"); + } + newIotModelServiceList.add(newIotModelVo); } } EquipmentVo equipment = new EquipmentVo(); @@ -247,9 +248,8 @@ public class DataServiceImpl implements DataService { Iterator keysHigh = values.fieldNames(); while (keysHigh.hasNext()) { String fieldName = keysHigh.next(); - String key = String.format("RT:[%s]:[%s]", deviceId, fieldName); - keyValueMap.put(key, values.get(fieldName).asDouble()); - + String key = String.format("RT:%s:%s", deviceId, fieldName); + keyValueMap.put(key,values.get(fieldName)); } adminRedisTemplate.mSet(keyValueMap); // Long dataTime = data.getTime(); diff --git a/das/src/main/resources/mapper/SysEnumTypesMapper.xml b/das/src/main/resources/mapper/SysEnumTypesMapper.xml new file mode 100644 index 00000000..c73e9a36 --- /dev/null +++ b/das/src/main/resources/mapper/SysEnumTypesMapper.xml @@ -0,0 +1,19 @@ + + + + + + + + + + + diff --git a/das/src/main/resources/mapper/SysEnumValuesMapper.xml b/das/src/main/resources/mapper/SysEnumValuesMapper.xml index 39c00d79..e950eda3 100644 --- a/das/src/main/resources/mapper/SysEnumValuesMapper.xml +++ b/das/src/main/resources/mapper/SysEnumValuesMapper.xml @@ -4,6 +4,7 @@ + @@ -22,6 +23,7 @@ and e.is_active =#{sysEnumValuesDto.isActive} + order by e.order_number asc diff --git a/das/src/main/resources/mapper/SysImptabmappingMapper.xml b/das/src/main/resources/mapper/SysImptabmappingMapper.xml index 595e8290..af03a126 100644 --- a/das/src/main/resources/mapper/SysImptabmappingMapper.xml +++ b/das/src/main/resources/mapper/SysImptabmappingMapper.xml @@ -58,14 +58,6 @@ where sc.id = #{linkId} - - + + diff --git a/docs/_sidebar.md b/docs/_sidebar.md index b18bdbdb..24c993d0 100644 --- a/docs/_sidebar.md +++ b/docs/_sidebar.md @@ -4,4 +4,7 @@ - [设备管理接口](api/equipment.md) - [节点管理接口](api/node.md) - [数据访问接口](api/data.md) -- [数据采集](datacollect/) \ No newline at end of file +- [数据采集](datacollect/) +- [系统部署](deploy/) + - [Linux系统部署](deploy/linux.md) + - [TDEngine3部署](deploy/tdengine3.md) \ No newline at end of file diff --git a/docs/deploy/README.md b/docs/deploy/README.md new file mode 100644 index 00000000..121bbfcb --- /dev/null +++ b/docs/deploy/README.md @@ -0,0 +1,36 @@ +# 润阳系统部署 + + +## 硬件环境 + +> CPU: 24核 + +> 内存: 32GB + +> 硬盘: 16T + + +## 软件环境 + +> 操作系统: OpenEuler 24.03 (LTS) + +> JDK: OpenJDK 17 + +> 关系数据库: PostgreSQL 15.8 + +> 缓存数据库: Redis 6 + +> 时序数据库: TDEngine 3.3.2.0 + + +## 目录结构 + +`/das` - 系统根目录 + +`/das/data` - 系统数据目录 + +`/das/data/tddata` - TDEngine3数据目录 + +`/das/data/pgdata` - PostgreSQL数据目录 + +`/das/data/redis` - Redis数据目录 diff --git a/docs/deploy/_sidebar.md b/docs/deploy/_sidebar.md new file mode 100644 index 00000000..4d99ba2c --- /dev/null +++ b/docs/deploy/_sidebar.md @@ -0,0 +1 @@ +- [首页](/) \ No newline at end of file diff --git a/docs/deploy/linux.md b/docs/deploy/linux.md new file mode 100644 index 00000000..9f0b819d --- /dev/null +++ b/docs/deploy/linux.md @@ -0,0 +1,3 @@ +# OpenEuler 24.03 (LTS) 安装部署 + +!> 等待更新 \ No newline at end of file diff --git a/docs/deploy/tdengine3.md b/docs/deploy/tdengine3.md new file mode 100644 index 00000000..c5eb53e6 --- /dev/null +++ b/docs/deploy/tdengine3.md @@ -0,0 +1,4 @@ +# TDEngine3 安装部署 + +> 当前版本: 3.3.2.0 + diff --git a/ui/dasadmin/.env.development b/ui/dasadmin/.env.development index d5455a0b..0349de0f 100644 --- a/ui/dasadmin/.env.development +++ b/ui/dasadmin/.env.development @@ -6,7 +6,7 @@ VITE_BASE_PATH = './' # 代理配置(开发使用),必须在一行中 # 本地 -# VITE_APP_PROXY=[["/api","http://192.168.130.12:8080/api"]] +# VITE_APP_PROXY=[["/api","http://192.168.130.22:8080/api"]] # 线上 VITE_APP_PROXY=[["/api","https://test.jsspisoft.com/api"]] diff --git a/ui/dasadmin/src/api/backend/Enumeration/request.ts b/ui/dasadmin/src/api/backend/Enumeration/request.ts index 8fd2cbf8..688a5736 100644 --- a/ui/dasadmin/src/api/backend/Enumeration/request.ts +++ b/ui/dasadmin/src/api/backend/Enumeration/request.ts @@ -34,7 +34,7 @@ export function enumValueAdd(params: object = {}) { }) } -// 枚举值新增 +// 枚举值删除 export function enumValueDelete(params: object = {}) { return createAxios({ url: '/api/enum/deleteEnumValues', @@ -42,4 +42,12 @@ export function enumValueDelete(params: object = {}) { data: params, }) } +// 枚举值编辑 +export function enumValuepageEdit(params: object = {}) { + return createAxios({ + url: '/api/enum/updateEnumValues', + method: 'POST', + data: params, + }) +} diff --git a/ui/dasadmin/src/api/backend/control/request.ts b/ui/dasadmin/src/api/backend/control/request.ts new file mode 100644 index 00000000..fc9f1e37 --- /dev/null +++ b/ui/dasadmin/src/api/backend/control/request.ts @@ -0,0 +1,9 @@ +import createAxios from '/@/utils/axios' + +export const sendValueReq = (data: any) => { + return createAxios({ + url: '/api/node/link/deviceControl', + method: 'post', + data: data, + }) +} \ No newline at end of file diff --git a/ui/dasadmin/src/api/backend/dashboard.ts b/ui/dasadmin/src/api/backend/dashboard.ts index 5caef9de..21f901fd 100644 --- a/ui/dasadmin/src/api/backend/dashboard.ts +++ b/ui/dasadmin/src/api/backend/dashboard.ts @@ -1,4 +1,5 @@ import createAxios from '/@/utils/axios' +import {addDataType, getDataReturnType, getDataType, getTreeDataReturnType} from "/@/views/backend/auth/org/type"; export const url = '/admin/Dashboard/' @@ -8,3 +9,11 @@ export function index() { method: 'get', }) } + +export const getInstitutionalListReq = (data: getDataType) => { + return createAxios>({ + url: '/api/org/query', + method: 'post', + data: data, + }) +} diff --git a/ui/dasadmin/src/api/backend/deviceModel/request.ts b/ui/dasadmin/src/api/backend/deviceModel/request.ts index dead7a32..7a3cd1c9 100644 --- a/ui/dasadmin/src/api/backend/deviceModel/request.ts +++ b/ui/dasadmin/src/api/backend/deviceModel/request.ts @@ -142,8 +142,16 @@ export const downloadModelReq = (data: { id: string }) => { } export const getDeviceTypeEnumReq = () => { - return createAxios>({ + return createAxios>({ url: '/api/equipment/type', method: 'post', }) } + +export const getRealValueListReq = (data: { deviceId: string, attributes: string[] }[]) => { + return createAxios>({ + url: '/api/data/snapshot', + method: 'post', + data: data, + }) +} diff --git a/ui/dasadmin/src/api/backend/linkMonitor/request.ts b/ui/dasadmin/src/api/backend/linkMonitor/request.ts new file mode 100644 index 00000000..707e53f5 --- /dev/null +++ b/ui/dasadmin/src/api/backend/linkMonitor/request.ts @@ -0,0 +1,24 @@ +import createAxios from '/@/utils/axios' +import { + RequestReturnType, + RequestReturnRowType, + LinkMonitorTreeType, + GetLinkMonitorTableParam, + LinkMonitorTableType, +} from '/@/views/backend/linkMonitor/type' + +// 左侧树节点树结构 +export const getNodeListReq = () => { + return createAxios>({ + url: '/api/node/list', + method: 'post', + }) +} +// 链路列表 +export const getLinkListReq = (data: GetLinkMonitorTableParam) => { + return createAxios>({ + url: '/api/node/link/list', + method: 'post', + data: data, + }) +} diff --git a/ui/dasadmin/src/assets/dashboard/Standby.png b/ui/dasadmin/src/assets/dashboard/Standby.png new file mode 100644 index 00000000..e898970f Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/Standby.png differ diff --git a/ui/dasadmin/src/assets/dashboard/bg1.png b/ui/dasadmin/src/assets/dashboard/bg1.png new file mode 100644 index 00000000..f937021f Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/bg1.png differ diff --git a/ui/dasadmin/src/assets/dashboard/biaogan.png b/ui/dasadmin/src/assets/dashboard/biaogan.png new file mode 100644 index 00000000..a6bfcffb Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/biaogan.png differ diff --git a/ui/dasadmin/src/assets/dashboard/fanlist1.png b/ui/dasadmin/src/assets/dashboard/fanlist1.png new file mode 100644 index 00000000..049ded0f Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/fanlist1.png differ diff --git a/ui/dasadmin/src/assets/dashboard/fanlist2.png b/ui/dasadmin/src/assets/dashboard/fanlist2.png new file mode 100644 index 00000000..e363af18 Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/fanlist2.png differ diff --git a/ui/dasadmin/src/assets/dashboard/fault.png b/ui/dasadmin/src/assets/dashboard/fault.png new file mode 100644 index 00000000..3c4036f6 Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/fault.png differ diff --git a/ui/dasadmin/src/assets/dashboard/fdl1.png b/ui/dasadmin/src/assets/dashboard/fdl1.png new file mode 100644 index 00000000..ac0b38d9 Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/fdl1.png differ diff --git a/ui/dasadmin/src/assets/dashboard/fdl2.png b/ui/dasadmin/src/assets/dashboard/fdl2.png new file mode 100644 index 00000000..07b083f8 Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/fdl2.png differ diff --git a/ui/dasadmin/src/assets/dashboard/fdl3.png b/ui/dasadmin/src/assets/dashboard/fdl3.png new file mode 100644 index 00000000..453ae972 Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/fdl3.png differ diff --git a/ui/dasadmin/src/assets/dashboard/fdl4.png b/ui/dasadmin/src/assets/dashboard/fdl4.png new file mode 100644 index 00000000..70069489 Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/fdl4.png differ diff --git a/ui/dasadmin/src/assets/dashboard/offline.png b/ui/dasadmin/src/assets/dashboard/offline.png new file mode 100644 index 00000000..e043ccc2 Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/offline.png differ diff --git a/ui/dasadmin/src/assets/dashboard/run.png b/ui/dasadmin/src/assets/dashboard/run.png new file mode 100644 index 00000000..c3e8cc4e Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/run.png differ diff --git a/ui/dasadmin/src/assets/dashboard/status1.png b/ui/dasadmin/src/assets/dashboard/status1.png new file mode 100644 index 00000000..ba7bde03 Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/status1.png differ diff --git a/ui/dasadmin/src/assets/dashboard/status2.png b/ui/dasadmin/src/assets/dashboard/status2.png new file mode 100644 index 00000000..c8d8b118 Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/status2.png differ diff --git a/ui/dasadmin/src/assets/dashboard/viewP.png b/ui/dasadmin/src/assets/dashboard/viewP.png new file mode 100644 index 00000000..29078a06 Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/viewP.png differ diff --git a/ui/dasadmin/src/assets/dashboard/viewR.png b/ui/dasadmin/src/assets/dashboard/viewR.png new file mode 100644 index 00000000..52484464 Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/viewR.png differ diff --git a/ui/dasadmin/src/assets/dashboard/viewW.png b/ui/dasadmin/src/assets/dashboard/viewW.png new file mode 100644 index 00000000..e78cbaf0 Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/viewW.png differ diff --git a/ui/dasadmin/src/assets/dashboard/viewY.png b/ui/dasadmin/src/assets/dashboard/viewY.png new file mode 100644 index 00000000..50b4ad9b Binary files /dev/null and b/ui/dasadmin/src/assets/dashboard/viewY.png differ diff --git a/ui/dasadmin/src/layouts/backend/components/navMenus.vue b/ui/dasadmin/src/layouts/backend/components/navMenus.vue index f22345b0..eaebdf11 100644 --- a/ui/dasadmin/src/layouts/backend/components/navMenus.vue +++ b/ui/dasadmin/src/layouts/backend/components/navMenus.vue @@ -1,10 +1,10 @@ - + - + @@ -182,13 +168,10 @@ import { enumTreeQuery, enumTypeAdd, enumValueAdd, - enumValueDelete + enumValueDelete, enumValuepageEdit } from "/@/api/backend/Enumeration/request"; - - - const activeName = ref('1') interface Tree { enumId: number, @@ -260,7 +243,6 @@ const handleNodeClick = (data: any) => { queryListData.description=queryName.value, queryListData.pageNum=currentPage.value, queryListData.pageSize=currentPageSize.value - console.log(queryListData) queryenumValueMethod(queryListData) } @@ -385,13 +367,6 @@ const queryenumValueMethod = (data: any) => { const onSubmitvalue = () => { formRef.value.validate((valid: any) => { if (valid) { - if(formInlineValue.isActive==true){ - formInlineValue.isActive=1 - }else{ - formInlineValue.isActive=0 - } - - console.log(formInlineValue) enumValueAdd(formInlineValue).then((res: any) => { if (res.code == 200) { queryListData.description =queryName.value @@ -459,40 +434,66 @@ const dialogVisibleDelete1 = (done: () => void) => { } }) } -// 菜单修改 +// 编辑枚举值 const fromUpDate = reactive({ id: '', enumTypeId: '', value:'', description : '', orderNumber: 0, - isActive: false + isActive: 0 }) +const addhandleSwitchChange = (value: any) => { + formInlineValue.isActive = value ? 1 : 0 +} +const edithandleSwitchChange = (value: any) => { + fromUpDate.isActive = value ? 1 : 0 +} const visibleValueEdit = ref(false) const enumValueEdit = (data: any) => { visibleValueEdit.value = true fromUpDate.id = data.row.id - fromUpDate.enumTypeId = data.row.enumTypeId + fromUpDate.enumTypeId = data.row.enumTypeIdStr fromUpDate.value = data.row.value fromUpDate.description = data.row.description fromUpDate.orderNumber = data.row.orderNumber - if(data.row.isActive==1){ - fromUpDate.isActive = true - }else{ - fromUpDate.isActive = false - } + fromUpDate.isActive = data.row.isActive + } + const edithandleClosevalue = (done: () => void) => { visibleValueEdit.value = false } +const EditonSubmitvalue = () => { + formRef.value.validate((valid: any) => { + if (valid) { + enumValuepageEdit(fromUpDate).then((res: any) => { + if (res.code == 200) { + queryListData.description =queryName.value + queryListData.enumTypeId = enumTypeId.value + queryListData.pageNum=currentPage.value, + queryListData.pageSize=currentPageSize.value + queryenumValueMethod(queryListData) + ElMessage({ + message: res.msg, + type: 'success', + }) + } else { + ElMessage.error({ + message: res.msg, + type: 'error', + }) + } + }) + + visibleValueEdit.value = false + } + }) +} onMounted(() => { enumTreeTypeList() - const firstNode = TypeTree.value.store.root.childNodes[0]; - /*if (firstNode) { - handleNodeClick(firstNode.data, null, TypeTree.value); - }*/ }) diff --git a/ui/dasadmin/src/views/backend/auth/model/index.vue b/ui/dasadmin/src/views/backend/auth/model/index.vue index bb831216..9fdcedd6 100644 --- a/ui/dasadmin/src/views/backend/auth/model/index.vue +++ b/ui/dasadmin/src/views/backend/auth/model/index.vue @@ -11,7 +11,7 @@ @@ -960,7 +960,7 @@ const downLoadModel = () => { } const currentPage = ref(1) -const currentPageSize = ref(10) +const currentPageSize = ref(20) const pageTotal = ref(0) const pagePagination = ref([10, 20, 30]) const getcurrentPage = () => { diff --git a/ui/dasadmin/src/views/backend/auth/org/index.vue b/ui/dasadmin/src/views/backend/auth/org/index.vue index b4f947bb..6548119b 100644 --- a/ui/dasadmin/src/views/backend/auth/org/index.vue +++ b/ui/dasadmin/src/views/backend/auth/org/index.vue @@ -379,9 +379,9 @@ const getInstitutionList = (data: getDataType = { name: null }) => { }) } -const pagePagination = [5, 10, 20, 30] +const pagePagination = [ 10, 20, 30] const currentPage = ref(1) -const currentPageSize = ref(pagePagination[0]) +const currentPageSize = ref(pagePagination[1]) const pageTotal = ref(0) const originData = ref() const tableData = computed(() => { diff --git a/ui/dasadmin/src/views/backend/auth/role/index.vue b/ui/dasadmin/src/views/backend/auth/role/index.vue index c9442f74..9c71a9ce 100644 --- a/ui/dasadmin/src/views/backend/auth/role/index.vue +++ b/ui/dasadmin/src/views/backend/auth/role/index.vue @@ -203,9 +203,9 @@ const tableData = computed(() => { return originData.value?.slice(start, end) }) -const pagePagination = [5, 10, 20, 30] +const pagePagination = [10, 20, 30] const currentPage = ref(1) -const currentPageSize = ref(pagePagination[0]) +const currentPageSize = ref(pagePagination[1]) const pageTotal = ref(0) const originData = ref[]>() const addClick = () => { diff --git a/ui/dasadmin/src/views/backend/auth/user/index.vue b/ui/dasadmin/src/views/backend/auth/user/index.vue index c26c7d44..4e0e44cf 100644 --- a/ui/dasadmin/src/views/backend/auth/user/index.vue +++ b/ui/dasadmin/src/views/backend/auth/user/index.vue @@ -546,9 +546,9 @@ const selectUpdataName = (data: string) => { formUserUpData.profilePicture = data } // 分页 -const pagePagination = [5, 10, 20, 30] +const pagePagination = [ 10, 20, 30] const currentPage = ref(1) -const currentPageSize = ref(pagePagination[0]) +const currentPageSize = ref(pagePagination[1]) const handleSizeChange = (val: number) => { formQuery.pageSize = val RyUserQuery(formQuery) diff --git a/ui/dasadmin/src/views/backend/dashboard.vue b/ui/dasadmin/src/views/backend/dashboard.vue index d50d1ea9..446a31c4 100644 --- a/ui/dasadmin/src/views/backend/dashboard.vue +++ b/ui/dasadmin/src/views/backend/dashboard.vue @@ -1,180 +1,301 @@ diff --git a/ui/dasadmin/src/views/backend/equipment/equipmentManagement/control.vue b/ui/dasadmin/src/views/backend/equipment/equipmentManagement/control.vue new file mode 100644 index 00000000..ba81cf05 --- /dev/null +++ b/ui/dasadmin/src/views/backend/equipment/equipmentManagement/control.vue @@ -0,0 +1,274 @@ + + + + + diff --git a/ui/dasadmin/src/views/backend/equipment/equipmentManagement/index.vue b/ui/dasadmin/src/views/backend/equipment/equipmentManagement/index.vue index cba1ed83..16e50248 100644 --- a/ui/dasadmin/src/views/backend/equipment/equipmentManagement/index.vue +++ b/ui/dasadmin/src/views/backend/equipment/equipmentManagement/index.vue @@ -49,9 +49,13 @@ @@ -418,11 +422,39 @@ + +
+ +
+
+ + +
+ +
+
diff --git a/ui/dasadmin/src/views/backend/equipment/equipmentManagement/measurement.vue b/ui/dasadmin/src/views/backend/equipment/equipmentManagement/measurement.vue new file mode 100644 index 00000000..c111b895 --- /dev/null +++ b/ui/dasadmin/src/views/backend/equipment/equipmentManagement/measurement.vue @@ -0,0 +1,233 @@ + + + + + diff --git a/ui/dasadmin/src/views/backend/home/windMatrix.vue b/ui/dasadmin/src/views/backend/home/windMatrix.vue new file mode 100644 index 00000000..7314248b --- /dev/null +++ b/ui/dasadmin/src/views/backend/home/windMatrix.vue @@ -0,0 +1,237 @@ + + + + + diff --git a/ui/dasadmin/src/views/backend/linkMonitor/index.vue b/ui/dasadmin/src/views/backend/linkMonitor/index.vue new file mode 100644 index 00000000..a2a79381 --- /dev/null +++ b/ui/dasadmin/src/views/backend/linkMonitor/index.vue @@ -0,0 +1,459 @@ + + + + + diff --git a/ui/dasadmin/src/views/backend/linkMonitor/type.ts b/ui/dasadmin/src/views/backend/linkMonitor/type.ts new file mode 100644 index 00000000..74d87393 --- /dev/null +++ b/ui/dasadmin/src/views/backend/linkMonitor/type.ts @@ -0,0 +1,45 @@ +export type RequestReturnType = Promise<{ + code: number + msg: string + success: boolean + data?: T +}> + +export type RequestReturnRowType = Promise<{ + code: number + msg: string + rows: T + total: number +}> +export type LinkMonitorTreeType = { + id?: string + nodeName: string + nodeIp: string + orgName?: string + orgId: string + revision?: number + children?: LinkMonitorTreeType[] +} +export type LinkMonitorTableType = { + id?: string + linkName: string + protocol?: number + protocolName?: string + params: string + nodeName: string + nodeId: string + revision: number +} +export type GetLinkMonitorTableParam = { + nodeId: string + linkName?: string + pageNum: number + pageSize: number +} + +export enum LinkMonitorFieldsEnums { + 'id' = 'id', + 'linkName' = '链路名称', + 'status' = '状态', + 'protocolName' = '协议类型', +} diff --git a/ui/dasadmin/src/views/backend/login.vue b/ui/dasadmin/src/views/backend/login.vue index a45b14c1..9c890874 100644 --- a/ui/dasadmin/src/views/backend/login.vue +++ b/ui/dasadmin/src/views/backend/login.vue @@ -103,7 +103,8 @@ let timer: number const config = useConfig() const adminInfo = useAdminInfo() -toggleDark(config.layout.isDark) + +// TODO 切换暗黑 toggleDark(config.layout.isDark) const formRef = ref() const usernameRef = ref() diff --git a/ui/dasadmin/src/views/backend/node/index.vue b/ui/dasadmin/src/views/backend/node/index.vue index bd6211cb..7093f10b 100644 --- a/ui/dasadmin/src/views/backend/node/index.vue +++ b/ui/dasadmin/src/views/backend/node/index.vue @@ -484,7 +484,7 @@ const submitLinkForm = () => { const paginationOptions = reactive({ current: 1, - pageSize: 10, + pageSize: 20, total: 0, pageSizes: [10, 20, 30], }) diff --git a/ui/dasadmin/vite.config.ts.timestamp-1721615163299-e1531267e82f8.mjs b/ui/dasadmin/vite.config.ts.timestamp-1721615163299-e1531267e82f8.mjs deleted file mode 100644 index c5c5ce65..00000000 --- a/ui/dasadmin/vite.config.ts.timestamp-1721615163299-e1531267e82f8.mjs +++ /dev/null @@ -1,177 +0,0 @@ -// vite.config.ts -import vue from "file:///E:/softstoneFile/ry-das/ui/dasadmin/node_modules/@vitejs/plugin-vue/dist/index.mjs"; -import { resolve } from "path"; -import { loadEnv } from "file:///E:/softstoneFile/ry-das/ui/dasadmin/node_modules/vite/dist/node/index.js"; - -// src/utils/vite.ts -var customHotUpdate = () => { - const closeHmr = /* @__PURE__ */ new Map(); - return { - name: "vite-plugin-custom-hot-update", - configureServer(server) { - server.ws.on("custom:close-hot", (data) => { - closeHmr.set(data.type, true); - server.watcher.removeAllListeners("add"); - server.watcher.removeAllListeners("unlink"); - }); - server.ws.on("custom:open-hot", (data) => { - closeHmr.set(data.type, false); - server.watcher.on("add", () => { - server.restart(); - }); - server.watcher.on("unlink", () => { - server.restart(); - }); - }); - server.ws.on("custom:reload-hot", () => { - server.restart(); - }); - }, - handleHotUpdate() { - const closeHmrs = Array.from(closeHmr.values()); - let closeHmrsBool = false; - for (const key in closeHmrs) { - closeHmrsBool = closeHmrsBool || closeHmrs[key]; - } - if (closeHmrsBool) - return []; - } - }; -}; - -// src/components/icon/svg/index.ts -import { readFileSync, readdirSync } from "fs"; -var idPerfix = ""; -var iconNames = []; -var svgTitle = /+].*?)>/; -var clearHeightWidth = /(width|height)="([^>+].*?)"/g; -var hasViewBox = /(viewBox="[^>+].*?")/g; -var clearReturn = /(\r)|(\n)/g; -var clearFill = /(fill="[^>+].*?")/g; -function findSvgFile(dir) { - const svgRes = []; - const dirents = readdirSync(dir, { - withFileTypes: true - }); - for (const dirent of dirents) { - iconNames.push(`${idPerfix}-${dirent.name.replace(".svg", "")}`); - if (dirent.isDirectory()) { - svgRes.push(...findSvgFile(dir + dirent.name + "/")); - } else { - const svg = readFileSync(dir + dirent.name).toString().replace(clearReturn, "").replace(clearFill, 'fill=""').replace(svgTitle, ($1, $2) => { - let width = 0; - let height = 0; - let content = $2.replace(clearHeightWidth, (s1, s2, s3) => { - if (s2 === "width") { - width = s3; - } else if (s2 === "height") { - height = s3; - } - return ""; - }); - if (!hasViewBox.test($2)) { - content += `viewBox="0 0 ${width} ${height}"`; - } - return ``; - }).replace("", ""); - svgRes.push(svg); - } - } - return svgRes; -} -var svgBuilder = (path, perfix = "local") => { - if (path === "") - return; - idPerfix = perfix; - const res = findSvgFile(path); - return { - name: "svg-transform", - transformIndexHtml(html) { - return html.replace( - "", - ` - - - ${res.join("")} - - ` - ); - } - }; -}; - -// vite.config.ts -var __vite_injected_original_dirname = "E:\\softstoneFile\\ry-das\\ui\\dasadmin"; -var pathResolve = (dir) => { - return resolve(__vite_injected_original_dirname, ".", dir); -}; -var viteConfig = ({ mode }) => { - const { VITE_PORT, VITE_OPEN, VITE_BASE_PATH, VITE_OUT_DIR, VITE_APP_PROXY } = loadEnv(mode, process.cwd()); - const alias = { - "/@": pathResolve("./src/"), - assets: pathResolve("./src/assets"), - "vue-i18n": "vue-i18n/dist/vue-i18n.cjs.js" - }; - const createProxy = (proxyList) => { - const ret = {}; - if (proxyList) { - proxyList = JSON.parse(proxyList); - for (const [prefix, target] of proxyList) { - ret[prefix] = { - target, - changeOrigin: true, - rewrite: (path) => path.replace(new RegExp(`^${prefix}`), "") - }; - } - } - return ret; - }; - return { - plugins: [vue(), svgBuilder("./src/assets/icons/"), customHotUpdate()], - root: process.cwd(), - resolve: { alias }, - base: VITE_BASE_PATH, - server: { - port: parseInt(VITE_PORT), - open: VITE_OPEN != "false", - proxy: { - ...createProxy(VITE_APP_PROXY) - } - }, - build: { - cssCodeSplit: false, - sourcemap: false, - outDir: VITE_OUT_DIR, - emptyOutDir: true, - chunkSizeWarningLimit: 1500, - terserOptions: { - compress: { - keep_infinity: true, - // Used to delete console in production environment - drop_console: true, - drop_debugger: true - }, - output: { - comments: true - // 去掉注释内容 - } - }, - rollupOptions: { - output: { - manualChunks: { - // 分包配置,配置完成自动按需加载 - vue: ["vue", "vue-router", "pinia", "vue-i18n", "element-plus"], - echarts: ["echarts"] - } - } - } - } - }; -}; -var vite_config_default = viteConfig; -export { - vite_config_default as default -}; -//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["vite.config.ts", "src/utils/vite.ts", "src/components/icon/svg/index.ts"],
  "sourcesContent": ["const __vite_injected_original_dirname = \"E:\\\\softstoneFile\\\\ry-das\\\\ui\\\\dasadmin\";const __vite_injected_original_filename = \"E:\\\\softstoneFile\\\\ry-das\\\\ui\\\\dasadmin\\\\vite.config.ts\";const __vite_injected_original_import_meta_url = \"file:///E:/softstoneFile/ry-das/ui/dasadmin/vite.config.ts\";import vue from '@vitejs/plugin-vue'\r\nimport { resolve } from 'path'\r\nimport { loadEnv } from 'vite'\r\nimport type { UserConfig, ConfigEnv, ProxyOptions } from 'vite'\r\nimport { isProd, customHotUpdate } from '/@/utils/vite'\r\nimport { svgBuilder } from '/@/components/icon/svg/index'\r\n\r\nconst pathResolve = (dir: string): any => {\r\n    return resolve(__dirname, '.', dir)\r\n}\r\n\r\n// https://vitejs.cn/config/\r\nconst viteConfig = ({ mode }: ConfigEnv): UserConfig => {\r\n    const { VITE_PORT, VITE_OPEN, VITE_BASE_PATH, VITE_OUT_DIR, VITE_APP_PROXY } = loadEnv(mode, process.cwd())\r\n\r\n    const alias: Record<string, string> = {\r\n        '/@': pathResolve('./src/'),\r\n        assets: pathResolve('./src/assets'),\r\n        'vue-i18n': 'vue-i18n/dist/vue-i18n.cjs.js',\r\n    }\r\n\r\n    const createProxy = (proxyList: any) => {\r\n        const ret = {} as Record<string, ProxyOptions>\r\n        if (proxyList) {\r\n            proxyList = JSON.parse(proxyList)\r\n            for (const [prefix, target] of proxyList) {\r\n                ret[prefix] = {\r\n                    target: target,\r\n                    changeOrigin: true,\r\n                    rewrite: (path) => path.replace(new RegExp(`^${prefix}`), ''),\r\n                }\r\n            }\r\n        }\r\n\r\n        return ret\r\n    }\r\n\r\n    // let proxy: Record<string, string | ProxyOptions> = {}\r\n    // if (VITE_PROXY_URL) {\r\n    //     proxy = {\r\n    //         '/': {\r\n    //             target: VITE_PROXY_URL,\r\n    //             changeOrigin: true,\r\n    //         },\r\n    //     }\r\n    // }\r\n\r\n    return {\r\n        plugins: [vue(), svgBuilder('./src/assets/icons/'), customHotUpdate()],\r\n        root: process.cwd(),\r\n        resolve: { alias },\r\n        base: VITE_BASE_PATH,\r\n        server: {\r\n            port: parseInt(VITE_PORT),\r\n            open: VITE_OPEN != 'false',\r\n            proxy: {\r\n                ...createProxy(VITE_APP_PROXY),\r\n            },\r\n        },\r\n        build: {\r\n            cssCodeSplit: false,\r\n            sourcemap: false,\r\n            outDir: VITE_OUT_DIR,\r\n            emptyOutDir: true,\r\n            chunkSizeWarningLimit: 1500,\r\n            terserOptions: {\r\n                compress: {\r\n                    keep_infinity: true,\r\n                    // Used to delete console in production environment\r\n                    drop_console: true,\r\n                    drop_debugger: true,\r\n                },\r\n                output: {\r\n                    comments: true, // \u53BB\u6389\u6CE8\u91CA\u5185\u5BB9\r\n                },\r\n            },\r\n            rollupOptions: {\r\n                output: {\r\n                    manualChunks: {\r\n                        // \u5206\u5305\u914D\u7F6E\uFF0C\u914D\u7F6E\u5B8C\u6210\u81EA\u52A8\u6309\u9700\u52A0\u8F7D\r\n                        vue: ['vue', 'vue-router', 'pinia', 'vue-i18n', 'element-plus'],\r\n                        echarts: ['echarts'],\r\n                    },\r\n                },\r\n            },\r\n        },\r\n    }\r\n}\r\n\r\nexport default viteConfig\r\n", "const __vite_injected_original_dirname = \"E:\\\\softstoneFile\\\\ry-das\\\\ui\\\\dasadmin\\\\src\\\\utils\";const __vite_injected_original_filename = \"E:\\\\softstoneFile\\\\ry-das\\\\ui\\\\dasadmin\\\\src\\\\utils\\\\vite.ts\";const __vite_injected_original_import_meta_url = \"file:///E:/softstoneFile/ry-das/ui/dasadmin/src/utils/vite.ts\";import { Plugin } from 'vite'\r\n\r\n/**\r\n * \u662F\u5426\u5728\u5F00\u53D1\u73AF\u5883\r\n */\r\nexport function isDev(mode: string): boolean {\r\n    return mode === 'development'\r\n}\r\n\r\n/**\r\n * \u662F\u5426\u5728\u751F\u4EA7\u73AF\u5883\r\n */\r\nexport function isProd(mode: string | undefined): boolean {\r\n    return mode === 'production'\r\n}\r\n\r\n/**\r\n * \u81EA\u5B9A\u4E49\u70ED\u66F4\u65B0\u5904\u7406\u63D2\u4EF6\r\n */\r\nexport const customHotUpdate = (): Plugin => {\r\n    const closeHmr = new Map<string, boolean>()\r\n\r\n    return {\r\n        name: 'vite-plugin-custom-hot-update',\r\n        configureServer(server) {\r\n            server.ws.on('custom:close-hot', (data) => {\r\n                closeHmr.set(data.type, true)\r\n\r\n                // \u5173\u95ED\u6587\u4EF6\u6DFB\u52A0\u548C\u5220\u9664\u7684\u76D1\u542C\r\n                server.watcher.removeAllListeners('add')\r\n                server.watcher.removeAllListeners('unlink')\r\n            })\r\n            server.ws.on('custom:open-hot', (data) => {\r\n                closeHmr.set(data.type, false)\r\n\r\n                server.watcher.on('add', () => {\r\n                    server.restart()\r\n                })\r\n                server.watcher.on('unlink', () => {\r\n                    server.restart()\r\n                })\r\n            })\r\n            server.ws.on('custom:reload-hot', () => {\r\n                server.restart()\r\n            })\r\n        },\r\n        handleHotUpdate() {\r\n            const closeHmrs = Array.from(closeHmr.values())\r\n            let closeHmrsBool = false\r\n            for (const key in closeHmrs) {\r\n                closeHmrsBool = closeHmrsBool || closeHmrs[key]\r\n            }\r\n            if (closeHmrsBool) return []\r\n        },\r\n    }\r\n}\r\n", "const __vite_injected_original_dirname = \"E:\\\\softstoneFile\\\\ry-das\\\\ui\\\\dasadmin\\\\src\\\\components\\\\icon\\\\svg\";const __vite_injected_original_filename = \"E:\\\\softstoneFile\\\\ry-das\\\\ui\\\\dasadmin\\\\src\\\\components\\\\icon\\\\svg\\\\index.ts\";const __vite_injected_original_import_meta_url = \"file:///E:/softstoneFile/ry-das/ui/dasadmin/src/components/icon/svg/index.ts\";import { readFileSync, readdirSync } from 'fs'\r\n\r\nlet idPerfix = ''\r\nconst iconNames: string[] = []\r\nconst svgTitle = /<svg([^>+].*?)>/\r\nconst clearHeightWidth = /(width|height)=\"([^>+].*?)\"/g\r\nconst hasViewBox = /(viewBox=\"[^>+].*?\")/g\r\nconst clearReturn = /(\\r)|(\\n)/g\r\n// \u6E05\u7406 svg \u7684 fill\r\nconst clearFill = /(fill=\"[^>+].*?\")/g\r\n\r\nfunction findSvgFile(dir: string): string[] {\r\n    const svgRes = []\r\n    const dirents = readdirSync(dir, {\r\n        withFileTypes: true,\r\n    })\r\n    for (const dirent of dirents) {\r\n        iconNames.push(`${idPerfix}-${dirent.name.replace('.svg', '')}`)\r\n        if (dirent.isDirectory()) {\r\n            svgRes.push(...findSvgFile(dir + dirent.name + '/'))\r\n        } else {\r\n            const svg = readFileSync(dir + dirent.name)\r\n                .toString()\r\n                .replace(clearReturn, '')\r\n                .replace(clearFill, 'fill=\"\"')\r\n                .replace(svgTitle, ($1, $2) => {\r\n                    let width = 0\r\n                    let height = 0\r\n                    let content = $2.replace(clearHeightWidth, (s1: string, s2: string, s3: number) => {\r\n                        if (s2 === 'width') {\r\n                            width = s3\r\n                        } else if (s2 === 'height') {\r\n                            height = s3\r\n                        }\r\n                        return ''\r\n                    })\r\n                    if (!hasViewBox.test($2)) {\r\n                        content += `viewBox=\"0 0 ${width} ${height}\"`\r\n                    }\r\n                    return `<symbol id=\"${idPerfix}-${dirent.name.replace('.svg', '')}\" ${content}>`\r\n                })\r\n                .replace('</svg>', '</symbol>')\r\n            svgRes.push(svg)\r\n        }\r\n    }\r\n    return svgRes\r\n}\r\n\r\nexport const svgBuilder = (path: string, perfix = 'local') => {\r\n    if (path === '') return\r\n    idPerfix = perfix\r\n    const res = findSvgFile(path)\r\n    return {\r\n        name: 'svg-transform',\r\n        transformIndexHtml(html: string) {\r\n            /* eslint-disable */\r\n            return html.replace(\r\n                '<body>',\r\n                `\r\n                <body>\r\n                <svg id=\"local-icon\" data-icon-name=\"${iconNames.join(\r\n                    ','\r\n                )}\" xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\" style=\"position: absolute; width: 0; height: 0\">\r\n                ${res.join('')}\r\n                </svg>\r\n                `\r\n            )\r\n            /* eslint-enable */\r\n        },\r\n    }\r\n}\r\n"],
  "mappings": "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",
  "names": []
}
