Browse Source

feat:simulator

alwayssuper 7 months ago
parent
commit
8a7e146445

+ 16 - 3
yudao-module-iot/yudao-module-iot-biz/src/main/java/cn/iocoder/yudao/module/iot/controller/admin/device/IotDeviceDataController.java

@@ -3,19 +3,21 @@ package cn.iocoder.yudao.module.iot.controller.admin.device;
 import cn.iocoder.yudao.framework.common.pojo.CommonResult;
 import cn.iocoder.yudao.framework.common.pojo.PageResult;
 import cn.iocoder.yudao.framework.common.util.object.BeanUtils;
+import cn.iocoder.yudao.module.iot.controller.admin.device.vo.device.IotDeviceSaveReqVO;
 import cn.iocoder.yudao.module.iot.controller.admin.device.vo.deviceData.IotDeviceDataPageReqVO;
 import cn.iocoder.yudao.module.iot.controller.admin.device.vo.deviceData.IotDeviceDataRespVO;
+import cn.iocoder.yudao.module.iot.controller.admin.device.vo.deviceData.IotDeviceDataSimulatorSaveReqVO;
 import cn.iocoder.yudao.module.iot.dal.dataobject.device.IotDeviceDataDO;
 import cn.iocoder.yudao.module.iot.controller.admin.device.vo.deviceData.IotTimeDataRespVO;
+import cn.iocoder.yudao.module.iot.service.device.IotDeviceLogDataService;
 import cn.iocoder.yudao.module.iot.service.device.IotDevicePropertyDataService;
 import io.swagger.v3.oas.annotations.Operation;
 import io.swagger.v3.oas.annotations.tags.Tag;
 import jakarta.annotation.Resource;
 import jakarta.validation.Valid;
+import org.springframework.security.access.prepost.PreAuthorize;
 import org.springframework.validation.annotation.Validated;
-import org.springframework.web.bind.annotation.GetMapping;
-import org.springframework.web.bind.annotation.RequestMapping;
-import org.springframework.web.bind.annotation.RestController;
+import org.springframework.web.bind.annotation.*;
 
 import java.util.List;
 import java.util.Map;
@@ -31,6 +33,9 @@ public class IotDeviceDataController {
     @Resource
     private IotDevicePropertyDataService deviceDataService;
 
+    @Resource
+    private IotDeviceLogDataService iotDeviceLogDataService;
+
     // TODO @浩浩:这里的 /latest-list,包括方法名。
     @GetMapping("/latest")
     @Operation(summary = "获取设备属性最新数据")
@@ -47,4 +52,12 @@ public class IotDeviceDataController {
         return success(BeanUtils.toBean(list, IotTimeDataRespVO.class));
     }
 
+    @PostMapping("/simulator")
+    @Operation(summary = "模拟设备")
+    public CommonResult<Boolean> simulatorDevice(@Valid @RequestBody IotDeviceDataSimulatorSaveReqVO simulatorReqVO) {
+        //TODO:先生成一下日志  后续完善模拟设备代码逻辑
+        iotDeviceLogDataService.createDeviceLog(simulatorReqVO);
+        return success(true);
+    }
+
 }

+ 40 - 0
yudao-module-iot/yudao-module-iot-biz/src/main/java/cn/iocoder/yudao/module/iot/controller/admin/device/vo/deviceData/IotDeviceDataSimulatorSaveReqVO.java

@@ -0,0 +1,40 @@
+package cn.iocoder.yudao.module.iot.controller.admin.device.vo.deviceData;
+
+import io.swagger.v3.oas.annotations.media.Schema;
+import jakarta.validation.constraints.NotEmpty;
+import lombok.Data;
+
+
+import java.time.LocalDateTime;
+
+@Schema(description = "管理后台 - IoT 模拟设备数据 Request VO")
+@Data
+public class IotDeviceDataSimulatorSaveReqVO {
+
+    @Schema(description = "消息ID", requiredMode = Schema.RequiredMode.REQUIRED, example = "msg123")
+    private String id;
+
+    @Schema(description = "产品ID", requiredMode = Schema.RequiredMode.REQUIRED, example = "product123")
+    @NotEmpty(message = "产品ID不能为空")
+    private String productKey;
+
+    @Schema(description = "设备ID", requiredMode = Schema.RequiredMode.REQUIRED, example = "device123")
+    @NotEmpty(message = "设备ID不能为空")
+    private String deviceKey;
+
+    @Schema(description = "消息/日志类型", requiredMode = Schema.RequiredMode.REQUIRED, example = "property")
+    @NotEmpty(message = "消息类型不能为空")
+    private String type;
+
+    @Schema(description = "标识符", requiredMode = Schema.RequiredMode.REQUIRED, example = "temperature")
+    @NotEmpty(message = "标识符不能为空")
+    private String subType;
+
+    @Schema(description = "数据内容", requiredMode = Schema.RequiredMode.REQUIRED, example = "{\"value\": 25.6}")
+    @NotEmpty(message = "数据内容不能为空")
+    private String content;
+
+    @Schema(description = "上报时间", requiredMode = Schema.RequiredMode.REQUIRED)
+    private LocalDateTime reportTime;
+
+}

+ 61 - 4
yudao-module-iot/yudao-module-iot-biz/src/main/java/cn/iocoder/yudao/module/iot/dal/dataobject/device/IotDeviceLogDO.java

@@ -1,5 +1,62 @@
-package cn.iocoder.yudao.module.iot.dal.dataobject.device;/**
- * @Author alwayssuper
- * @Date 2025/1/5 12:32
-*/public class IotDeviceLogDO {
+package cn.iocoder.yudao.module.iot.dal.dataobject.device;
+
+import cn.hutool.core.date.DateTime;
+import lombok.AllArgsConstructor;
+import lombok.Builder;
+import lombok.Data;
+import lombok.NoArgsConstructor;
+
+import java.time.LocalDateTime;
+
+/**
+ * IoT 设备日志数据 DO
+ *
+ * @author alwayssuper
+ */
+@Data
+@Builder
+@NoArgsConstructor
+@AllArgsConstructor
+public class IotDeviceLogDO {
+    /**
+     * 消息ID
+     */
+    private String id;
+
+    /**
+     * 产品ID
+     */
+    private String productKey;
+
+    /**
+     * 设备ID
+     */
+    private String deviceKey;
+
+    /**
+     * 消息/日志类型
+     */
+    private String type;
+
+    /**
+     * 标识符:用于标识具体的属性、事件或服务
+     */
+    private String subType;
+
+    /**
+     * 数据内容:存储具体的消息数据内容,通常是JSON格式
+     */
+    private String content;
+
+    /**
+     * 上报时间戳
+     */
+    private DateTime reportTime;
+
+    /**
+     * 时序时间
+     */
+    private DateTime ts;
+
+
 }

+ 1 - 0
yudao-module-iot/yudao-module-iot-biz/src/main/java/cn/iocoder/yudao/module/iot/dal/tdengine/TdThingModelMessageMapper.java

@@ -13,6 +13,7 @@ import org.apache.ibatis.annotations.Param;
  * 处理 TD 中物模型消息日志的操作
  */
 @Mapper
+@Deprecated
 @TDengineDS
 @InterceptorIgnore(tenantLine = "true") // 避免 SQL 解析,因为 JSqlParser 对 TDengine 的 SQL 解析会报错
 public interface TdThingModelMessageMapper {

+ 39 - 4
yudao-module-iot/yudao-module-iot-biz/src/main/java/cn/iocoder/yudao/module/iot/framework/tdengine/config/TDengineTableInitConfiguration.java

@@ -1,5 +1,40 @@
-package cn.iocoder.yudao.module.iot.framework.tdengine.config;/**
- * @Author alwayssuper
- * @Date 2025/1/5 17:37
-*/public class TDengineTableInitConfiguration {
+package cn.iocoder.yudao.module.iot.framework.tdengine.config;
+
+import cn.iocoder.yudao.module.iot.service.device.IotDeviceLogDataService;
+import lombok.RequiredArgsConstructor;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.boot.ApplicationArguments;
+import org.springframework.boot.ApplicationRunner;
+import org.springframework.context.annotation.Configuration;
+import org.springframework.core.annotation.Order;
+
+/**
+ * TDengine 表初始化的 Configuration
+ *
+ * @author alwayssuper
+ */
+@Slf4j
+@RequiredArgsConstructor
+@Configuration
+@Order(Integer.MAX_VALUE) // 保证在最后执行
+public class TDengineTableInitConfiguration implements ApplicationRunner {
+
+    private final IotDeviceLogDataService deviceLogService;
+
+    @Override
+    public void run(ApplicationArguments args) {
+        try {
+            // 初始化设备日志表
+            deviceLogService.initTDengineSTable();
+            log.info("初始化 设备日志表 TDengine 表结构成功");
+        } catch (Exception ex) {
+            if (ex.getMessage().contains("Table already exists")) {
+                log.info("TDengine 设备日志超级表已存在,跳过创建");
+                return;
+            }else{
+                log.error("初始化 设备日志表 TDengine  表结构失败", ex);
+            }
+            throw ex;
+        }
+    }
 }

+ 44 - 4
yudao-module-iot/yudao-module-iot-biz/src/main/java/cn/iocoder/yudao/module/iot/service/device/IotDeviceLogDataServiceImpl.java

@@ -1,5 +1,45 @@
-package cn.iocoder.yudao.module.iot.service.device;/**
- * @Author alwayssuper
- * @Date 2025/1/5 11:50
-*/public class IotDeviceLogDataServiceImpl {
+package cn.iocoder.yudao.module.iot.service.device;
+
+import cn.hutool.core.date.DateTime;
+import cn.iocoder.yudao.framework.common.util.object.BeanUtils;
+import cn.iocoder.yudao.module.iot.controller.admin.device.vo.deviceData.IotDeviceDataSimulatorSaveReqVO;
+import cn.iocoder.yudao.module.iot.dal.dataobject.device.IotDeviceLogDO;
+import cn.iocoder.yudao.module.iot.dal.tdengine.IotDeviceLogDataMapper;
+import jakarta.annotation.Resource;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.stereotype.Service;
+import org.springframework.validation.annotation.Validated;
+
+import java.time.LocalDateTime;
+
+/**
+ * IoT 设备日志数据 Service 实现了
+ *
+ * @author alwayssuper
+ */
+@Service
+@Slf4j
+public class IotDeviceLogDataServiceImpl implements IotDeviceLogDataService{
+
+    @Resource
+    private IotDeviceLogDataMapper iotDeviceLogDataMapper;
+
+
+    @Override
+    public void initTDengineSTable() {
+        try {
+            // 创建设备日志超级表
+            iotDeviceLogDataMapper.createDeviceLogSTable();
+            log.info("创建设备日志超级表成功");
+        } catch (Exception ex) {
+            throw ex;
+        }
+    }
+
+    @Override
+    public void createDeviceLog(IotDeviceDataSimulatorSaveReqVO simulatorReqVO) {
+        IotDeviceLogDO iotDeviceLogDO = BeanUtils.toBean(simulatorReqVO, IotDeviceLogDO.class);
+        iotDeviceLogDO.setTs(DateTime.now());
+        iotDeviceLogDataMapper.insert(iotDeviceLogDO);
+    }
 }

+ 0 - 4
yudao-module-iot/yudao-module-iot-biz/src/main/java/cn/iocoder/yudao/module/iot/service/device/IotDevicePropertyDataServiceImpl.java

@@ -85,8 +85,6 @@ public class IotDevicePropertyDataServiceImpl implements IotDevicePropertyDataSe
     @Resource
     private IotDevicePropertyDataMapper devicePropertyDataMapper;
 
-    @Resource
-    private TdThingModelMessageMapper tdThingModelMessageMapper;
 
     @Override
     public void defineDevicePropertyData(Long productId) {
@@ -114,8 +112,6 @@ public class IotDevicePropertyDataServiceImpl implements IotDevicePropertyDataSe
             newFields.add(0, new TDengineTableField(TDengineTableField.FIELD_TS, TDengineTableField.TYPE_TIMESTAMP));
             // 2.1.1 创建产品超级表
             devicePropertyDataMapper.createProductPropertySTable(product.getProductKey(), newFields);
-            // 2.1.2 创建物模型日志超级表
-            tdThingModelMessageMapper.createSuperTable(product.getProductKey());
             return;
         }
         // 2.2 情况二:如果是修改的时候,需要更新表

+ 1 - 12
yudao-module-iot/yudao-module-iot-biz/src/main/java/cn/iocoder/yudao/module/iot/service/tdengine/IotThingModelMessageServiceImpl.java

@@ -78,8 +78,6 @@ public class IotThingModelMessageServiceImpl implements IotThingModelMessageServ
             createDeviceTable(device.getDeviceType(), device.getProductKey(), device.getDeviceName(), device.getDeviceKey());
             iotDeviceService.updateDeviceStatus(new IotDeviceStatusUpdateReqVO()
                     .setId(device.getId()).setStatus(IotDeviceStatusEnum.ONLINE.getStatus()));
-            // 1.2 创建物模型日志设备表
-            tdThingModelMessageMapper.createTableWithTag(device.getProductKey(), device.getDeviceKey());
         }
 
         // 2. 获取设备属性并进行物模型校验,过滤非物模型属性
@@ -231,16 +229,7 @@ public class IotThingModelMessageServiceImpl implements IotThingModelMessageServ
                 .setTags(tagsFieldValues));
     }
 
-    /**
-     * 创建物模型日志设备数据表
-     *
-     * @param productKey 产品 Key
-     * @param deviceKey  设备 Key
-     *
-     */
-    private void createThingModelMessageDeviceTable(String productKey,  String deviceKey){
-        tdThingModelMessageMapper.createTableWithTag(productKey, deviceKey);
-    }
+
 
     /**
      * 获取数据库名称

+ 44 - 0
yudao-module-iot/yudao-module-iot-biz/src/main/resources/mapper/device/IotDeviceLogDataMapper.xml

@@ -0,0 +1,44 @@
+<?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="cn.iocoder.yudao.module.iot.dal.tdengine.IotDeviceLogDataMapper">
+
+    <!-- 创建设备日志超级表   初始化只创建一次-->
+    <update id="createDeviceLogSTable">
+        CREATE STABLE device_log(
+        ts TIMESTAMP,
+        id NCHAR(50),
+        product_key NCHAR(50),
+        type NCHAR(50),
+        subType NCHAR(50),
+        content NCHAR(1024),
+        report_time TIMESTAMP
+        )TAGS (
+        device_key NCHAR(50)
+        )
+    </update>
+
+
+    <!-- 创建设备日志子表  讨论:TDengine 在子表不存在的情况下 可在数据插入时 自动建表  要不要去掉创建子表的逻辑  由第一次插入数据时自动创建-->
+    <update id="createDeviceLogTable">
+        CREATE TABLE device_log_${deviceKey} USING device_log TAGS('${deviceKey}')
+    </update>
+
+    <!-- 插入设备日志数据 在子表不存在的情况下 可在数据插入时 自动建表 -->
+    <insert id="insert">
+        INSERT INTO device_log_${log.deviceKey} (ts, id, product_key, type, subType, content, report_time)
+        USING device_log
+        TAGS ('${log.deviceKey}')
+        VALUES (
+            #{log.ts},
+            #{log.id},
+            #{log.productKey},
+            #{log.type},
+            #{log.subType},
+            #{log.content},
+            #{log.reportTime}
+        )
+    </insert>
+
+</mapper>