On this page
遥测
一、遥测数据规划
1.遥测规划
- 遥测数据上传ThingsBoard平台
- ThingsBoard通过规则引擎整合元数据
- ThingsBoard通过规则引擎按设备类型对数据分类
- ThingsBoard通过规则引擎把消息路由到消息队列(RabbitMQ或Kafka、EMQX),根据设备类型建topic
- 消息队列(RabbitMQ)根据topic消费数据,保存到数据库
- 根据微服务划分创建数据库,根据时序数据规则创建表
二、准备工作
1.上传遥测
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
| package com.iothub.telemetry;
import com.iothub.mqtt.EmqClient;
import com.iothub.mqtt.MqttProperties;
import com.iothub.mqtt.QosEnum;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
import javax.annotation.PostConstruct;
@Component
public class Upload {
@Autowired
private EmqClient emqClient;
@Autowired
private MqttProperties properties;
@PostConstruct
public void init(){
//连接服务端
emqClient.connect(properties.getUsername(),properties.getPassword());
//订阅一个主题
//emqClient.subscribe("testtopic/#", QosEnum.QoS1);
}
@Scheduled(fixedRate = 2000)
public void publish(){
String data = getData(2);
emqClient.publish("v1/devices/me/telemetry",data,
QosEnum.QoS1,false);
}
private String getData(Integer type){
if (type == 1) {
// 携带时间戳
String data = "{\n" +
"\t\"ts\": 1451649600512,\n" +
"\t\"values\": {\n" +
"\t\t\"stringKey\": \"value1\",\n" +
"\t\t\"booleanKey\": true,\n" +
"\t\t\"doubleKey\": 42.0,\n" +
"\t\t\"longKey\": 73,\n" +
"\t\t\"jsonKey\": {\n" +
"\t\t\t\"someNumber\": 42,\n" +
"\t\t\t\"someArray\": [1, 2, 3],\n" +
"\t\t\t\"someNestedObject\": {\n" +
"\t\t\t\t\"key\": \"value\"\n" +
"\t\t\t}\n" +
"\t\t}\n" +
"\t}\n" +
"}";
return data;
} else if (type == 2) {
// 不携带时间戳
String data = "{\n" +
" \"stringKey\": \"value1\",\n" +
" \"booleanKey\": true,\n" +
" \"doubleKey\": 42.0,\n" +
" \"longKey\": 73,\n" +
" \"jsonKey\": {\n" +
" \"someNumber\": 42,\n" +
" \"someArray\": [1,2,3],\n" +
" \"someNestedObject\": {\"key\": \"value\"}\n" +
" }\n" +
"}";
return data;
}else {
// 数组
String data = "[{\"key1\":\"value1\"}, {\"key2\":true}]";
return data;
}
}
}
|
1
2
3
4
5
6
7
8
9
10
11
| server:
port: 8999
spring:
application:
name: tb-telemetry
mqtt:
broker-url: tcp://82.157.166.86:1883
client-id: emq-client-telemetry
username: LuUoLXi3DMqOPqDjm20C
password:
|
遥测数据
1
2
3
4
5
6
7
8
9
10
11
12
13
| {
"stringKey": "value1",
"booleanKey": true,
"doubleKey": 42.0,
"longKey": 73,
"jsonKey": {
"someNumber": 42,
"someArray": [1, 2, 3],
"someNestedObject": {
"key": "value"
}
}
}
|
元数据
1
2
3
4
5
| {
"deviceName": "Test-telemetry",
"deviceType": "default",
"ts": "1696730983079"
}
|
2.RabbitMQ
2.1.Docker部署RabbitMQ
1
2
3
4
5
6
| docker run -d --network host --restart=always --name rabbitmq rabbitmq:management
# 访问地址
http://82.157.166.86:15672/
# 账户/密码:
guest/guest
|
2.2. 创建直连交换机
1.创建直连交换机
data:image/s3,"s3://crabby-images/e76c7/e76c7bc9eb85f62fb651067ab9d4f01f7ca3899c" alt=""
2.创建队列
data:image/s3,"s3://crabby-images/f9b05/f9b05d4711d62722c1969ad31c52667e6dd89a0e" alt=""
3.绑定队列
data:image/s3,"s3://crabby-images/c0e01/c0e0163db1bb2687e4ace2fb982f6ada377273c5" alt=""
2.3.创建主题交换机
1.创建topic交换机
data:image/s3,"s3://crabby-images/c5584/c55849f5ea4237d56aa16a622a29568b64c1ea7f" alt=""
2.创建队列
data:image/s3,"s3://crabby-images/613ea/613ea1eee684a956cf2aae089f822f024d930c43" alt=""
3.绑定交换机
data:image/s3,"s3://crabby-images/3040e/3040eff343da63c4bb14c441f54a2532ef82d45e" alt=""
3.创建规则链
3.1.创建遥测规则链
data:image/s3,"s3://crabby-images/89e1e/89e1ee64534310ac5f3835f3a7768363244946c7" alt=""
data:image/s3,"s3://crabby-images/cc0fa/cc0fa6e30ba5c8df1891696bfa9d54a0d12cfaaf" alt=""
data:image/s3,"s3://crabby-images/2591e/2591ec6f240ac106cf2fd3d26d4cf46d69fab102" alt=""
3.2.整合元数据
1
2
3
4
5
6
7
| # copy keys
{
"deviceName": "Test-telemetry",
"deviceType": "default",
"ts": "1696730983079"
}
|
data:image/s3,"s3://crabby-images/df824/df8243ae2326d2739dfc016d50a2de43b96891ee" alt=""
data:image/s3,"s3://crabby-images/8a24c/8a24cfb63cfe54a54aadc6d3a6b417f42bcbe7e8" alt=""
3.3.按设备类型消息分类
1
2
3
4
5
6
7
| # 按设备类型消息分类
if(msg.deviceType === 'test-telemetry') {
return ['test-telemetry'];
} else {
return ['default'];
}
|
data:image/s3,"s3://crabby-images/ab0d6/ab0d672d828f65e0363c9bb8eeb636296652316a" alt=""
data:image/s3,"s3://crabby-images/84e74/84e74bf7cb3a6ab9501ed7c696aefb41aca07749" alt=""
三、上传遥测
1.创建设备
data:image/s3,"s3://crabby-images/3a0f5/3a0f582c3948d9f167d07181a40f77161f173e01" alt=""
data:image/s3,"s3://crabby-images/fdd70/fdd7086a5f6ac32fc810d8d58f7fd69a917252b1" alt=""
2.上传遥测数据
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
| {
"stringKey": "value1",
"booleanKey": true,
"doubleKey": 42.0,
"longKey": 73,
"jsonKey": {
"someNumber": 42,
"someArray": [1, 2, 3],
"someNestedObject": {
"key": "value"
}
}
}
{
"stringKey": "value1",
"booleanKey": true,
"doubleKey": 42.0,
"longKey": 73
}
|
data:image/s3,"s3://crabby-images/95ba7/95ba70e3a3bf06d4067fce6d3370aa1db85e458b" alt=""
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
| @Component
public class Upload {
@Autowired
private EmqClient emqClient;
@Autowired
private MqttProperties properties;
@PostConstruct
public void init(){
//连接服务端
emqClient.connect(properties.getUsername(),properties.getPassword());
//订阅一个主题
//emqClient.subscribe("testtopic/#", QosEnum.QoS1);
}
@Scheduled(fixedRate = 2000)
public void publish(){
String data = getData(2);
emqClient.publish("v1/devices/me/telemetry",data,
QosEnum.QoS1,false);
}
private String getData(Integer type){
if (type == 1) {
// 携带时间戳
String data = "{\n" +
"\t\"ts\": 1451649600512,\n" +
"\t\"values\": {\n" +
"\t\t\"stringKey\": \"value1\",\n" +
"\t\t\"booleanKey\": true,\n" +
"\t\t\"doubleKey\": 42.0,\n" +
"\t\t\"longKey\": 73,\n" +
"\t\t\"jsonKey\": {\n" +
"\t\t\t\"someNumber\": 42,\n" +
"\t\t\t\"someArray\": [1, 2, 3],\n" +
"\t\t\t\"someNestedObject\": {\n" +
"\t\t\t\t\"key\": \"value\"\n" +
"\t\t\t}\n" +
"\t\t}\n" +
"\t}\n" +
"}";
return data;
} else if (type == 2) {
// 不携带时间戳
String data = "{\n" +
" \"stringKey\": \"value1\",\n" +
" \"booleanKey\": true,\n" +
" \"doubleKey\": 42.0,\n" +
" \"longKey\": 73,\n" +
" \"jsonKey\": {\n" +
" \"someNumber\": 42,\n" +
" \"someArray\": [1,2,3],\n" +
" \"someNestedObject\": {\"key\": \"value\"}\n" +
" }\n" +
"}";
return data;
}else {
// 数组
String data = "[{\"key1\":\"value1\"}, {\"key2\":true}]";
return data;
}
}
}
|
3.RabbitMQ
1.创建队列
telemetry-default-queue、telemetry-queue
data:image/s3,"s3://crabby-images/8f989/8f9899741c7fe3b63388c4242349495acbc27ff5" alt=""
2.创建交换机
data:image/s3,"s3://crabby-images/fcadd/fcaddd439eddd11c2eb429a87294ba85f8309ffd" alt=""
4.配置规则引擎
4.1.整体配置
data:image/s3,"s3://crabby-images/7585c/7585c8a074794a9a8938a8cc098f2b8ef6cbaa0c" alt=""
data:image/s3,"s3://crabby-images/f3436/f34361ea273378887b5b0c3df7e6a6edfb522910" alt=""
data:image/s3,"s3://crabby-images/3b518/3b518ce575496e47521d87c0b4984878205d8b4d" alt=""
4.2.整个元数据
1
2
3
4
5
| {
"deviceName": "Test-telemetry",
"deviceType": "default",
"ts": "1696730983079"
}
|
data:image/s3,"s3://crabby-images/e2f10/e2f100afb0dff011fbcadf5c9db1c134a0dc9bad" alt=""
4.3.按设备类型消息分类
1
2
3
4
5
| if(msg.deviceType === 'test-telemetry') {
return ['test-telemetry'];
} else {
return ['default'];
}
|
data:image/s3,"s3://crabby-images/400c1/400c1631db539c2f637e83eb71be33863cfc99ba" alt=""
5.测试
data:image/s3,"s3://crabby-images/e7a32/e7a32d234b828b36aec30013bd4826a1c64c884a" alt=""
data:image/s3,"s3://crabby-images/960ae/960ae84751ae3acc1f4178d15550d4834b59b29a" alt=""
data:image/s3,"s3://crabby-images/d6c66/d6c66d6d528d771591a22fc84ea3dc2367bc706f" alt=""
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
| {
"stringKey": "value1",
"booleanKey": true,
"doubleKey": 42.0,
"longKey": 73,
"jsonKey": {
"someNumber": 42,
"someArray": [1, 2, 3],
"someNestedObject": {
"key": "value"
}
},
"deviceType": "default",
"deviceName": "Test-telemetry",
"ts": "1451649600512"
}
|
1
2
3
4
5
6
7
8
9
| {
"stringKey": "value1",
"booleanKey": true,
"doubleKey": 42.0,
"longKey": 73,
"deviceType": "test-telemetry",
"deviceName": "Test-telemetry1",
"ts": "1696824491527"
}
|
四、REST API
1.遥测API
data:image/s3,"s3://crabby-images/df60d/df60db096fd6de4a21a15046bd5daa03b8ca5103" alt=""
data:image/s3,"s3://crabby-images/5424a/5424a4bb562a7804228aaaed3daa054cee4adaf5" alt=""