通过物联网管理多台MQTT设备-基于米尔T527开发板

本篇测评由电子工程世界的优秀测评者“JerryZhen”提供。


本文将介绍基于米尔电子MYD-LT527开发板的网关方案测试。

一、系统概述

基于米尔-全志 T527设计一个简易的物联网网关,该网关能够管理多台MQTT设备,通过MQTT协议对设备进行读写操作,同时提供HTTP接口,允许用户通过HTTP协议与网关进行交互,并对设备进行读写操作。

二、系统架构

  1. 网关服务:基于FastAPI框架构建的Web服务,提供HTTP接口。

  2. MQTT客户端:负责与MQTT设备通信,管理设备连接、消息发布和订阅。

  3. 设备管理:维护一个设备列表,记录设备的基本信息和状态。

  4. 数据存储:使用内存或数据库存储设备数据,确保数据持久化。

三、组件设计

  1. MQTT组件

  • 负责与MQTT broker建立连接。
  • 订阅设备主题,接收设备发送的消息。
  • 发布消息到设备,实现远程控制。
  1. 设备管理组件:

  • 维护一个设备列表,记录设备的唯一标识符(如设备ID)、MQTT主题、连接状态等信息。
  • 提供设备增删改查的方法。
  1. HTTP组件:

  • 基于FastAPI定义HTTP接口。
  • 接收用户请求,调用MQTT组件和设备管理组件进行相应操作。
  • 返回操作结果给用户。

四、接口设计

  1. 设备列表:

  • GET /devices:返回所有设备的列表。
  • POST /devices:添加新设备到网关。
  • DELETE /devices/{device_id}:从网关中删除指定设备。
  1. 设备详情:

  • GET /devices/{device_id}:返回指定设备的详细信息。
  1. 设备数据:

  • GET /devices/{device_id}/data:获取指定设备的最新数据。
  • POST /devices/{device_id}/data:发送数据到指定设备。
  1. 设备控制:

  • POST /devices/{device_id}/control:发送控制命令到指定设备。

五、数据结构设计

  1. 设备信息:

  • 设备ID (device_id):唯一标识设备的字符串。
  • MQTT主题 (mqtt_topic):设备在MQTT broker上的主题。
  • 连接状态 (connection_status):表示设备是否在线的布尔值。
  • 其他设备属性(如名称、描述等)。
  1. 设备数据:

  • 设备ID (device_id):关联设备信息的设备ID。
  • 时间戳 (timestamp):数据发送或接收的时间。
  • 数据内容 (data):设备发送或接收的具体数据,可以是JSON格式或其他格式。

六、安全性考虑

  • 使用HTTPS协议提供安全的HTTP通信。

  • 实现用户认证和授权机制,确保只有授权用户可以访问和操作设备。

  • 对于敏感操作(如删除设备),要求用户进行二次确认或提供额外的安全措施。

七、部署与扩展

  • 使用Docker容器化部署网关服务,便于管理和扩展。

  • 根据需要,可以水平扩展网关实例以处理更多的设备连接和请求。

八、实现步骤

  1. 安装所需的Python库:fastapi, uvicorn, paho-mqtt等。

  2. 创建FastAPI应用并定义路由。

  3. 实现MQTT组件,包括与MQTT broker的连接、订阅、发布等功能。

  4. 实现设备管理组件,维护设备列表并提供增删改查的方法。

  5. 实现HTTP组件,调用MQTT组件和设备管理组件处理用户请求。

  6. 编写测试代码,验证网关的各项功能是否正常工作。

  7. 部署网关服务并监控其运行状态。

该设计方案仅仅是概述,具体实现细节可能需要根据实际需求和项目环境进行调整和优化。在实际开发中,还需要考虑异常处理、日志记录、性能优化等方面的问题。基于上述设计方案,以下是一个简化版的参考代码,展示了如何使用FastAPI和paho-mqtt库来创建一个物联网网关。需要注意,示例中不包含完整的错误处理、用户认证和授权机制,这些在实际生产环境中都是必不可少的。依赖的主要库版本:

fastapi==0.108.0

paho-mqtt==1.6.1

网关模拟代码gateway.py:

from fastapi import FastAPI, HTTPException, Body, status
from paho.mqtt.client import Client as MQTTClient
from typing import List, Dict, Any
import asyncio
import jsonapp = FastAPI()
mqtt_client = None
device_data = {}  subtopic="gateway/device/#"# MQTT回调函数
def on_message(client, userdata, msg): payload = msg.payload.decode()topic = msg.topicdevice_id = topic.split('/')[-1]device_data[device_id] = payloadprint(f"Received message from {device_id}: {payload}")  # MQTT连接和订阅
def mqtt_connect_and_subscribe(broker_url, broker_port):global mqtt_clientmqtt_client = MQTTClient()mqtt_client.on_message = on_messagemqtt_client.connect(broker_url, broker_port, 60)mqtt_client.subscribe(subtopic)mqtt_client.loop_start()# MQTT发布消息
async def mqtt_publish(topic: str, message: str):if mqtt_client is not None and mqtt_client.is_connected():mqtt_client.publish(topic, message)else: print("MQTT client is not connected!")# 设备管理:添加设备
@app.post("/devices/", status_code=status.HTTP_201_CREATED)
async def add_device(device_id: str):device_data[device_id] = Nonereturn {"message": f"Device {device_id} added"}# 设备管理:获取设备列表
@app.get("/devices/")
async def get_devices(): return list(device_data.keys())# 设备管理:获取设备数据
@app.get("/devices/{device_id}/data")
async def get_device_data(device_id: str):if device_id not in device_data:raise HTTPException(status_code=status.HTTP_404_NOT_FOUND, detail=f"Device {device_id} not found")return device_data.get(device_id)# 设备管理:发送数据到设备
@app.post("/devices/{device_id}/data")
async def send_data_to_device(device_id: str, data: Dict[str, Any] = Body(...)):topic = f"devices/{device_id}"message = json.dumps(data)await mqtt_publish(topic, message)return {"message": f"Data sent to {device_id}"}# 设备控制:发送控制命令到设备
@app.post("/devices/{device_id}/control")
async def control_device(device_id: str, command: str):topic = f"devices/device/{device_id}"await mqtt_publish(topic, command)return {"message": f"Control command sent to {device_id}"}# FastAPI启动事件
@app.on_event("startup")
async def startup_event():mqtt_connect_and_subscribe("127.0.0.1", 1883)# FastAPI关闭事件
@app.on_event("shutdown")
async def shutdown_event():if mqtt_client is not None:mqtt_client.loop_stop()mqtt_client.disconnect()# 运行FastAPI应用
if __name__ == "__main__":import uvicornuvicorn.run(app, host="127.0.0.1", port=8000)

设备1模拟代码 dev1.py:

import paho.mqtt.client as mqtt# 连接成功回调
def on_connect(client, userdata, flags, rc):print('Connected with result code '+str(rc))client.subscribe('devices/1')# 消息接收回调
def on_message(client, userdata, msg):print(msg.topic+" "+str(msg.payload))client.publish('gateway/device/1',payload=f'echo {msg.payload}',qos=0)client = mqtt.Client()# 指定回调函数
client.on_connect = on_connect
client.on_message = on_message# 建立连接
client.connect('127.0.0.1', 1883)
# 发布消息
client.publish('gateway/device/1',payload='Hello, I am device',qos=0)client.loop_forever()

设备2模拟代码 dev2.py

import paho.mqtt.client as mqtt# 连接成功回调
def on_connect(client, userdata, flags, rc):print('Connected with result code '+str(rc))client.subscribe('devices/2')# 消息接收回调
def on_message(client, userdata, msg):print(msg.topic+" "+str(msg.payload))client.publish('gateway/device/2',payload=f'echo {msg.payload}',qos=0)client = mqtt.Client()# 指定回调函数
client.on_connect = on_connect
client.on_message = on_message# 建立连接
client.connect('127.0.0.1', 1883)
# 发布消息
client.publish('gateway/device/2',payload='Hello, I am device',qos=0)client.loop_forever()

运行网关代码,打开网页得到api接口:

 通过api分别添加设备1和设备2,

在另外两个控制台中分别运行模拟设备1和模拟设备2的代码

通过网页API向设备1发送数据

通过网页API获得设备回复的数据,设备代码中只是简单的把网关发过来的数据进行回传

我们在网关的后台可以看到完整的数据流

至此一个简易的网关已经实现了,接下来将会尝试实现楼宇里的最常见的bacnet设备进行通讯管理。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/10540.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

i春秋-Test

题目 解题 参考WP https://blog.csdn.net/qq_40654505/article/details/107142533/目录扫描 复现wp payload为: search.php?searchtype5&tid&areaeval($_POST[cmd])使用蚁剑连接 http://eci-2ze4iyhwj7xvb68bsb2t.cloudeci1.ichunqiu.com:80/search.ph…

在 Navicat 17 中探索表配置文件

距离 Navicat 17(英文版)的发布还有不到一周的时间,现在是深入研究新的表配置文件功能的最佳时机。它允许我们保存经常用于表的筛选、排序和列显示的不同组合。所以,事不宜迟,让我们开始吧! 创建表配置文件…

leetcode——反转链表

206. 反转链表 - 力扣(LeetCode) 思路:创建三个指针n1,n2,n3,遍历原链表,通过三者之间的关系将链表反转。下面给出图示: 下面给出题解代码: typedef struct ListNode ListNode; struct List…

parallels desktop19最新免费Mac电脑虚拟机软件

Parallels Desktop是一款运行在Mac电脑上的虚拟机软件,它允许用户在Mac系统上同时运行多个操作系统,比如Windows、Linux等。通过这款软件,Mac用户可以轻松地在同一台电脑上体验不同操作系统的功能和应用程序,而无需额外的硬件设备…

自拍欺骗成为流行的身份证件欺诈技术

据 Socure 称,文档图像叠加是 2023 年最流行的身份证件欺诈技术,在所有被拒绝的身份证件中,有 63% 发生这种情况。 自拍欺骗和冒充在与文件相关的身份欺诈中占主导地位 当用户拍摄照片或使用 ID 的屏幕截图图像(而不是提供文档的…

26、Flink 的状态数据结构升级

状态数据结构升级 a)概述 Flink 流应用通常被设计为永远或者长时间运行,与所有长期运行的服务一样,应用程序需要随着业务的迭代而进行调整,应用所处理的数据 schema 也会随着进行变化。 升级状态类型的数据 schema &#xff0c…

初始Java篇(JavaSE基础语法)(8)认识String类(上)

找往期文章包括但不限于本期文章中不懂的知识点: 个人主页:我要学编程(ಥ_ಥ)-CSDN博客 所属专栏:JavaSE 简单介绍:在C语言中已经涉及到字符串了,但是在C语言中要表示字符串只能使用字符数组或者字符指针,可…

leetcode算法笔记-算法复杂度

对于时间复杂度,主要包括三种情况: 渐进紧确界: O渐进上界: 渐进下界: 加法原则:不同的时间复杂度相加取阶数最高的 乘法原则:不同的时间复杂度相乘,结果为时间复杂度的乘积 阶乘…

电脑nvidia驱动和合适版本的duda--自用 回忆版

参考文献:http://t.csdnimg.cn/ecDuG 内容很多抄的这个,主要害怕链接失效 一、Ubuntu 18.04 安装NVIDIA显卡驱动 1、查看本机显卡能够配置的驱动信息 ubuntu-drivers devices所以可以看出,推荐 nvidia-driver-530 - distro non-free 2、安…

Kubernetes学习-深入Pod篇(一) 创建Pod,Pod配置文件详解

🏷️个人主页:牵着猫散步的鼠鼠 🏷️系列专栏:Kubernetes渐进式学习-专栏 🏷️个人学习笔记,若有缺误,欢迎评论区指正 1.前言 我们在前面的文章讲解了Kubernetes的核心概念和服务部署&#x…

MyBatis-plus(一):快速入门

目录 一、MyBatis-plus 快速入门 1、原理 2、实体类命名规则 3、常见注解 4、主键 id 策略 5、使用 TableField 的常见场景 6、常用配置 二、核心功能 1、条件构造器 2、自定义 SQL 3、IService 接口 一、MyBatis-plus 快速入门 1、原理 MyBatisPlus 通过扫描实体…

算法学习012-不同路径 c++动态规划算法实现 中小学算法思维学习 信奥算法解析

目录 C不同路径 一、题目要求 1、编程实现 2、输入输出 二、算法分析 三、程序编写 四、运行结果 五、考点分析 六、推荐资料 C不同路径 一、题目要求 1、编程实现 一个机器人位于一个 m x n 网格的左上角 (起始点在下图中标记为 “Start” &#xff09…

Golang | Leetcode Golang题解之第78题子集

题目: 题解: func subsets(nums []int) (ans [][]int) {set : []int{}var dfs func(int)dfs func(cur int) {if cur len(nums) {ans append(ans, append([]int(nil), set...))return}set append(set, nums[cur])dfs(cur 1)set set[:len(set)-1]df…

计算概论学习笔记(2)

感谢北大李戈老师讲解的计算概论。 【道阻且长,行则将至】 很多年没有intensive coding,现在这个系列是coding retake,一点点回忆之前的知识,希望能重回到一线。主要内容包括C,C,Pytorch学术前沿项目学习和实践,预计…

iOS 安装cocoapds

注意 CocoaPods安装是基于ruby环境的,所以要安装CocoaPods先要安装Ruby环境,国内不能直接安装,只能通过VPN或淘宝的Ruby镜像来访问。 安装过程 gem sources --remove https://rubygems.org/ ** (注意是两个“-”,否则会移除失败) …

掌握文件重命名技巧:一次性处理多路径文件并赋予独立编号

在日常工作和生活中,我们经常需要处理大量的文件,而文件重命名则是一项非常常见的任务。如何高效地一次性处理多路径文件并赋予独立编号,成为许多用户关注的焦点。本文将介绍云炫文件管理器一些实用的文件重命名技巧,帮助您轻松应…

前后端完全开源!功能丰富的在线教室项目:Agora Flat

Agora Flat:高效集成的在线教室解决方案,重塑互动学习新体验。- 精选真开源,释放新价值。 概览 Agora Flat是在GitHub平台上公开分享的一个全面开源项目,它精心设计为一个高性能的在线教室解决方案,旨在便捷地搭建支持…

锐捷EWEB网管系统RCE漏洞

文章目录 免责声明漏洞描述漏洞原理影响版本漏洞复现修复建议 免责声明 该文章只为学习和交流,请不要做违法乱纪的事情,如有与本人无关 漏洞描述 锐捷网管系统是由北京锐捷数据时代科技有限公司开发的新一代基于云的网络管理软件,以"…

哈希表(unordered_set、unordered_map)

文章目录 一、unordered_set、unordered_map的介绍二、哈希表的建立方法2.1闭散列2.2开散列(哈希桶/拉链法) 三、闭散列代码(除留余数法)四、开散列代码(拉链法/哈希桶) 一、unordered_set、unordered_map的…

数据结构之——队列详解

目录 前言: 一、什么是队列 二、队列的实现 2.1 队列结构 2.2 队列初始化 2.3 队列销毁 2.4 入队列 2.5 出队列 2.6 获取队列头部元素 2.7 获取队列尾部元素 2.8 获取队列中有效元素个数 2.9 检测队列是否为空 三、 代码总览 Queue.h test.c 四、例题 前言…