## OneNET 本节介绍如何使用 RT-Thread MicroPython 来将设备接入 OneNET 云平台,本次示例使用的接入协议为 MQTT。 ### 准备工作 - 首先需要安装 `urequests` 模块和 `umqtt.simple` 模块,安装方法参考 [HttpClient ](./02-HttpClient.md) 和 [MQTT ](./04-MQTT.md) 章节。 - 本章实例代码在最后一节的附录中,可以在添加必要的注册信息后复制到 main.py 文件中在 MSH 中使用 python 命令来执行。 ### 产品创建 - 想要将开发板接入 OneNET 云平台,首先要进行产品的创建,创建分为两步,第一步是注册一个用户账号,第二步是创建一个基于特定协议的产品。 #### 用户注册 - 为了使用 OneNET 设备云的强大功能,首先要在 OneNET 上[注册开发者账号](https://open.iot.10086.cn/doc/art436.html#104),来创建专属的“开发者中心 。 #### 产品创建 - 接下来需要在 OneNET 平台上[创建产品](https://open.iot.10086.cn/doc/art437.html#104)。这里要注意的是在最后选择`设备接入方式`和`设备接入协议时`,因为本次示例使用的是 MQTT 协议,所以要在设备接入方式中选择`公开协议`,设备接入协议选择 `MQTT`。 ![1525764833130](../figures/choose_mqtt.png) ### 硬件接入 本章节将介绍如何将设备接入到 OneNET 云平台上。并且演示一个云平台向设备发送命令,设备向云平台返回命令发送次数的示例。 #### 设备的注册和接入 - 成功创建设备之后,将得到的产品 ID 记录下来供后面推送数据使用。 ![1525765493792](../figures/get_product_id.png) - 将设备的`正式环境注册码`记录下来用于注册新的设备。 ![1525765209683](../figures/reg_code.png) - 接下来打开例程中的 main.py,修改 sn 为设备唯一标识码,product_id 为上面得到的6位产品 ID,regKey 为上面记录下来的正式环境注册码。 ![1525766961043](../figures/change_code.png) - 在开发板中运行 main.py 即可在 OneNET 上看到我们注册的设备。 ![1525767092149](../figures/run_main.png) - 名为 `DeviceRT_Thread_Test_Product` 的设备已经注册完毕并且上线。 ![1525767167244](../figures/reg_done.png) #### 云平台向设备发送命令 - 可以通过发送命令功能来给开发板发送几组命令。 ![1525767264155](../figures/cmd1.png) ![1525767369050](../figures/cmd2.png) - 可以在设备端看到云平台下发的数据,同时设备端会上传命令发送次数的数据 ![1525767520609](../figures/cmd3.png) #### 设备向云平台上传数据 - 点击数据流管理功能来查看设备端上传的数据 ![1525767621708](../figures/look_data.png) - 可以在数据流管理中看到设备端上传的的命令发送次数 switch ![1525767740215](../figures/switch.png) - 这里的 switch 数据是在 mqtt.py 的 pubData 函数里面修改的,可以通过 value 对象的内容来给云平台上传不同的数据。 ![1525774755758](../figures/change_pub.png) - 至此设备和 OneNET 云平台就对接好了。 #### 添加独立应用 - 为了方便使用还可以给设备[添加独立的应用](https://open.iot.10086.cn/doc/art461.html#108),效果如下图: ![1525768143233](../figures/add_app.png) ### 代码讲解 - 通过修改 value 对象来修改向服务器发送的数据,这里是发送到特殊的系统 topic `$dp` ![1525774792209](../figures/code_review1.png) - 给服务器发送数据的触发条件是收到服务器下发的命令,这样就没有保证会一直有数据发送到服务器,所以在没有数据交换的一段时间后,MQTT 连接有可能断开。 ![1525768433046](../figures/code_review2.png) #### 附录示例代码 ```python import urequests as requests from umqtt.simple import MQTTClient import ujson as json import time class Register(): def __init__(self, url='', title='', sn='', mac=''): self.url = url self.title = title self.sn = sn self.mac = mac self.sock = None self.tjson = {} self.erron = 0 self.key = '' self.device_id = '' def regist(self): assert self.url is not None, "Url is not set" _, _, host, path = self.url.split('/', 3) if host == '': return device = {"mac":self.mac} if self.sn == '' else {"sn":self.sn} if self.title != '': device['title'] = self.title jdata = json.dumps(device) resp = requests.post(self.url, data=jdata) if resp: self.tjson = resp.json() if self.tjson['errno'] == 0: self.key = self.tjson['data']['key'] self.device_id = self.tjson['data']['device_id'] return 0 else: return -1 class OneNetMqtt: failed_count = 0 def __init__(self, client_id='', username='', password=''): self.server = "183.230.40.39" self.client_id = client_id self.username = username self.password = password self.topic = "topic_sub" # 填入测试 topic self.mqttClient = MQTTClient(self.client_id, self.server,6002,self.username,self.password) self.cmd_times = 0 # publish count def pubData(self, t): value = {'datastreams':[{"id":"switch","datapoints":[{"value": self.cmd_times }]}]} jdata = json.dumps(value) jlen = len(jdata) bdata = bytearray(jlen+3) bdata[0] = 1 # publish data in type of json bdata[1] = int(jlen / 256) # data lenght bdata[2] = jlen % 256 # data lenght bdata[3:jlen+4] = jdata.encode('ascii') # json data print('publish data', str(self.cmd_times + 1)) try: self.mqttClient.publish('$dp', bdata) # $dp 为特殊系统 topic,可以通过这个 topic 给系统推送信息,但是不能订阅这个 topic self.cmd_times += 1 self.failed_count = 0 except Exception as ex: self.failed_count += 1 print('publish failed:', ex.message()) if self.failed_count >= 3: print('publish failed three times, esp resetting...') reset() def sub_callback(self, topic, msg): print((topic,msg)) cmd = msg.decode('ascii').split(" ") print('sub_callback') def connect(self): self.mqttClient.set_callback(self.sub_callback) self.mqttClient.connect() self.mqttClient.subscribe(self.topic) print("Connected to %s, subscribed to %s topic." % (self.server, self.topic)) try: while True: self.mqttClient.check_msg() print("pubdata") self.pubData('x') finally: self.mqttClient.disconnect() print('MQTT closed') def main(): sn = 'RT_Thread_Test_Product' #1、填入设备唯一标识符 title = 'Device' + sn product_id = 'XXXXX' #2、填入创建设备时获得的产品 ID regKey = 'XXXXXXXX' #3、填入正式环境注册码 url = 'http://api.heclouds.com/register_de?register_code=' + regKey reg = Register(url=url, title=title, sn=sn) #根据上面的信息注册设备,如果已经注册不再重复注册 if reg.regist()==0: MQTT = OneNetMqtt(client_id=reg.device_id, username=product_id, password=reg.key) #开启 MQTT 服务 MQTT.connect() else: print('Error: No Client ID!') if __name__ == "__main__": main() ```