设备接入 IoTDA设备接入 IoTDA

计算
弹性云服务器 ECS
裸金属服务器 BMS
云手机 CPH
专属主机 DeH
弹性伸缩 AS
镜像服务 IMS
函数工作流 FunctionGraph
云耀云服务器 HECS
VR云渲游平台 CVR
特惠算力专区
存储
对象存储服务 OBS
云硬盘 EVS
云备份 CBR
弹性文件服务 SFS
存储容灾服务 SDRS
云硬盘备份 VBS
云服务器备份 CSBS
数据快递服务 DES
专属企业存储服务
云存储网关 CSG
专属分布式存储服务 DSS
CDN与智能边缘
内容分发网络 CDN
智能边缘云 IEC
智能边缘小站 IES
智能边缘平台 IEF
人工智能
AI开发平台ModelArts
华为HiLens
图引擎服务 GES
图像识别 Image
文字识别 OCR
自然语言处理 NLP
内容审核 Moderation
图像搜索 ImageSearch
医疗智能体 EIHealth
园区智能体 CampusGo
企业级AI应用开发专业套件 ModelArts Pro
人脸识别服务 FRS
对话机器人服务 CBS
视频分析服务 VAS
语音交互服务 SIS
知识图谱 KG
人证核身服务 IVS
IoT物联网
设备接入 IoTDA
设备管理 IoTDM(联通用户专用)
全球SIM联接 GSL
IoT数据分析
路网数字化服务 DRIS
IoT边缘 IoTEdge
设备发放 IoTDP
开发与运维
软件开发平台 DevCloud
项目管理 ProjectMan
代码托管 CodeHub
流水线 CloudPipeline
代码检查 CodeCheck
编译构建 CloudBuild
部署 CloudDeploy
云测 CloudTest
发布 CloudRelease
移动应用测试 MobileAPPTest
CloudIDE
Classroom
开源镜像站 Mirrors
应用魔方 AppCube
云性能测试服务 CPTS
应用管理与运维平台 ServiceStage
云应用引擎 CAE
视频
实时音视频 SparkRTC
视频直播 Live
视频点播 VOD
媒体处理 MPC
视频接入服务 VIS
管理与监管
统一身份认证服务 IAM
消息通知服务 SMN
云监控服务 CES
应用运维管理 AOM
应用性能管理 APM
云日志服务 LTS
云审计服务 CTS
标签管理服务 TMS
资源管理服务 RMS
应用身份管理服务 OneAccess
区块链
区块链服务 BCS
可信跨链服务 TCS
可信分布式身份服务
智能协作
IdeaHub
开发者工具
SDK开发指南
API签名指南
DevStar
HCloud CLI
Terraform
Ansible
云生态
云市场
合作伙伴中心
华为云培训中心
其他
管理控制台
消息中心
产品价格详情
系统权限
我的凭证
客户关联华为云合作伙伴须知
公共问题
宽限期保留期
奖励推广计划
活动
容器
云容器引擎 CCE
云容器实例 CCI
容器镜像服务 SWR
应用编排服务 AOS
多云容器平台 MCP
基因容器 GCS
容器洞察引擎 CIE
云原生服务中心 OSC
容器批量计算 BCE
容器交付流水线 ContainerOps
应用服务网格 ASM
网络
虚拟私有云 VPC
弹性公网IP EIP
弹性负载均衡 ELB
NAT网关 NAT
云专线 DC
虚拟专用网络 VPN
云连接 CC
VPC终端节点 VPCEP
数据库
云数据库 RDS
数据复制服务 DRS
文档数据库服务 DDS
分布式数据库中间件 DDM
云数据库 GaussDB (for openGauss)
云数据库 GaussDB(for MySQL)
云数据库 GaussDB NoSQL
数据管理服务 DAS
数据库和应用迁移 UGO
大数据
MapReduce服务 MRS
数据湖探索 DLI
表格存储服务 CloudTable
可信智能计算服务 TICS
推荐系统 RES
云搜索服务 CSS
数据可视化 DLV
数据湖治理中心 DGC
数据接入服务 DIS
数据仓库服务 GaussDB(DWS)
应用中间件
微服务引擎 CSE
分布式消息服务Kafka版
分布式消息服务RabbitMQ版
API网关 APIG
分布式缓存服务 DCS
分布式消息服务RocketMQ版
企业应用
域名注册服务 Domains
云解析服务 DNS
云速建站 CloudSite
网站备案
商标注册
华为云WeLink
会议
隐私保护通话 PrivateNumber
语音通话 VoiceCall
消息&短信 MSGSMS
云管理网络
SD-WAN 云服务
边缘数据中心管理 EDCM
云桌面 Workspace
应用与数据集成平台 ROMA Connect
ROMA资产中心 ROMAExchange
API全生命周期管理 ROMA API
安全与合规
安全技术与应用
DDoS防护 ADS
Web应用防火墙 WAF
云防火墙 CFW
应用信任中心 ATC
企业主机安全 HSS
容器安全服务 CGS
云堡垒机 CBH
数据库安全服务 DBSS
数据加密服务 DEW
数据安全中心 DSC
云证书管理服务 CCM
SSL证书管理 SCM
漏洞扫描服务 VSS
态势感知 SA
威胁检测服务 MTD
管理检测与响应 MDR
安全治理云图 Compass
认证测试中心 CTC
迁移
主机迁移服务 SMS
对象存储迁移服务 OMS
云数据迁移 CDM
专属云
专属计算集群 DCC
解决方案
高性能计算 HPC
SAP
混合云灾备
华为工业云平台 IMC
价格
成本优化最佳实践
专属云商业逻辑
用户服务
帐号中心
费用中心
成本中心
资源中心
企业管理
工单管理
客户运营能力
国际站常见问题
支持计划
专业服务
合作伙伴支持计划
文档首页> 设备接入 IoTDA> 最佳实践> 设备接入> 设备自定义TOPIC迁移上云
更新时间:2021-10-12 GMT+08:00
分享

设备自定义TOPIC迁移上云

场景说明

随着IoT终端设备规模不断增加,企业自建物联网面临服务器扩容,硬件投入和运维成本越来越高等诸多问题。为此,华为云物联网平台提供设备极简迁移上云解决方案,企业设备通过极少的改动,就可以快速将设备迁移到华为云物联网平台。

整体方案

假设企业终端设备接入自建MQTT集群,业务架构如下图所示。

基于MQTT协议的上行数据和下行指令的业务定义如下:

业务场景

通信Topic

报文Payload

设备上报数据

/aircondition/data/up

{ "temperature": 26.0 }

服务端控制指令

/aircondition/cmd

{ "switch": "off" }

为减少企业改造成本,华为云物联网平台提供如下迁移方案,设备侧不用改变原有的topic和payload报文格式,就可以快速迁移设备到华为云物联网平台。

企业设备迁移上云有三个核心变更点:

  • 设备侧修改接入域名为华为云物联网平台的接入点。

  • 配置规则引擎,把设备数据流转到应用服务器、AMQP消费组或华为云第三方云服务产品。

  • 应用服务器适配设备消息下发接口往指定topic下发消息。

操作步骤

  1. 在控制台配置产品、设备和数据流转方案,请参见云端配置开发

  2. 对设备端进行业务开发,请参见设备端开发

  3. 对服务端进行业务开发,实现接收设备数据和下发控制指令,请参见服务端开发

云端配置开发

  1. 访问设备接入服务,单击“立即使用”进入设备接入控制台。
  2. 创建产品。

    1. 选择左侧导航栏的“产品”,单击右上角下拉框,选择新建产品所属的资源空间。
    2. 单击右上角的“创建产品”,创建待迁移设备的产品,填写参数后,单击“立即创建”。

      基本信息

      产品名称

      自定义,如aircondition

      协议类型

      选择“MQTT”

      数据格式

      选择“JSON”

      厂商名称

      自定义,根据设备所属厂商填写。

      所属行业

      根据实际情况填写,如“智慧生活”

      设备类型

      根据实际情况填写,如“aircondition”

  3. 注册设备。

    1. 选择左侧导航栏的设备 > 设备注册,单击页面右上角的“注册设备”。

      参数名称

      说明

      所属产品

      选择在步骤3中创建的产品。

      设备标识码

      即node_id,填写为设备的IMEI、MAC地址或Serial No;若没有真实设备,填写自定义字符串,由英文字母和数字组成。

      设备名称

      自定义。

      设备认证类型

      根据企业设备现有的认证类型,选择对应的认证方式。本文以“密钥”方式为例。

      密钥

      设备密钥。填写待迁移设备的密钥。

    2. 填写设备参数后,单击“确定”

  4. 创建数据转发规则。

    1. 选择左侧导航栏的规则 > 数据转发,单击页面右上角的“创建规则”
    2. 参考下表填写参数后,单击“创建规则”

      参数名

      参数说明

      规则名称

      创建的规则名称。

      规则描述

      对该规则的描述。

      数据来源

      选择“设备消息”。设备迁移上云时,设备按原有的topic和payload进行上报,物联网平台默认按设备消息流程进行处理。

      触发事件

      选择数据来源后,自动匹配触发事件。

      资源空间

      您可以选择单个资源空间或所有资源空间。

    3. 在设置转发目标页面,单击“添加”,在弹出的页面中参考下表配置完参数后,单击“确认”。

      参数名

      参数说明

      转发目标

      选择数据转发目标,本文以“AMQP推送消息队列”为例。

      消息队列

      单击“选择”,选择消息队列。

      • 若没有消息队列,请新建消息队列,队列名称自定义且单个租户名下唯一,最大长度 128位,支持大小写英文字符串、数字、下划线(_)、中划线(-)和竖线(|),不支持斜杠(/)、空格等其他字符。
      • 若需要删除消息队列,单击消息队列右侧的“删除”即可。
        说明:

        已经订阅的队列不允许删除。

    4. 完成完整的规则定义后,单击“启动规则”,实现数据转发至AMQP消息队列。

设备端开发

完成云端配置后,需要进行设备端业务开发。完整的设备开发流程可参考设备侧开发。本章节以MQTT.fx为例,介绍在设备迁移场景下,设备侧如何在尽量少改动的情况下,实现设备建立MQTT连接、数据上报、指令接收等功能。

  1. 设备同物联网平台建立MQTT连接。

    1. 参考下表配置鉴权参数。

      参数

      必选/可选

      参数描述

      Broker Address

      必选

      华为云物联网平台的MQTT协议接入地址,请参考此处获取。

      Broker Port

      必选

      8883。若设备侧MQTT接入端口不是8883,且无法修改,可以开通企业版实例,参考此页面配置MQTT协议接入端口。

      Client ID

      可选

      使用设备迁移前的Client ID。

      User Name

      必选

      填写步骤3注册设备时生成的设备ID,默认通过控制台生成的设备ID会添加产品ID前缀。在设备迁移场景,设备侧User Name参数无法修改时可以调用创建设备接口,指定设备ID参数值同迁移前的User Name参数值保持一致。

      Password

      必选

      填写步骤3注册设备时指定的密钥,同迁移前的参数值保持一致。

    2. 参考下表配置“SSL/TLS”认证参数,然后单击“Apply”。

      参数

      必选/可选

      参数描述

      Enbale SSL/TLS

      必选

      选择“Enble SSL/TLS”

      CA certificate file

      必选

      上传证书资源页面获取的CA证书。如果设备侧的CA证书无法修改,可以开通企业版实例,参考此页面配置服务端证书。

  2. 设备同物联网平台建立连接后,设备沿用迁移前的topic和payload格式上报数据。物联网平台针对这类非系统预定义的topic,统一按照“设备消息”的处理流程将设备上报的数据转发给第三方应用或者华为云的其他云服务处理。

  3. 根据迁移前的topic进行订阅,接收应用服务器下发的指令。

服务端开发

对服务端进行业务开发,实现接收设备数据和下发控制指令。本文以Java脚本为例,演示接收设备数据和下发控制指令。

  • 业务服务器接收设备数据

    服务器通过AMQP客户端接收消息,详细说明请参见AMQP客户端接入说明Java SDK接入示例。设备迁移场景AMQP收到消息接口参考设备消息上报通知

    AMQP客户端收到的消息样例,body字段携带设备上报的原始topic和payload。

    {
    	"resource": "device.message",
    	"event": "report",
    	"event_time": "20201114T034403Z",
    	"notify_data": {
    		"header": {
    			"app_id": "QAksSBSBBQpWYtEKC3LrrOboNk0a",
    			"device_id": "5fae45e358115902ce609882_20201113",
    			"node_id": "20201113",
    			"product_id": "5fae45e358115902ce609882",
    			"gateway_id": "5fae45e358115902ce609882_20201113"
    		},
    		"body": {
    			"topic": "/aircondition/data/up",
    			"content": {
    				"temperature": 26.0
    			}
    		}
    	}
    }
    

    JAVA核心代码样例如下:

    package com.huawei.iot.amqp.jms;
    
    import org.apache.qpid.jms.JmsConnection;
    import org.apache.qpid.jms.JmsConnectionFactory;
    import org.apache.qpid.jms.JmsConnectionListener;
    import org.apache.qpid.jms.message.JmsInboundMessageDispatch;
    import org.apache.qpid.jms.transports.TransportOptions;
    import org.apache.qpid.jms.transports.TransportSupport;
    
    import javax.jms.*;
    import javax.naming.Context;
    import javax.naming.InitialContext;
    import java.net.URI;
    import java.util.Hashtable;
    import java.util.concurrent.ExecutorService;
    import java.util.concurrent.LinkedBlockingQueue;
    import java.util.concurrent.ThreadPoolExecutor;
    import java.util.concurrent.TimeUnit;
    
    public class HwIotAmqpJavaClientDemo{
        //异步线程池,参数可以根据业务特点作调整,也可以用其他异步方式来处理。
        private final static ExecutorService executorService = new ThreadPoolExecutor(Runtime.getRuntime().availableProcessors(),
                Runtime.getRuntime().availableProcessors() * 2, 60,
                TimeUnit.SECONDS, new LinkedBlockingQueue<>(5000));
    
        public static void main(String[] args) throws Exception{
            //连接凭证接入键值。
            String accessKey = "${yourAccessKey}";
            long timeStamp = System.currentTimeMillis();
            //UserName组装方法,请参见文档:AMQP客户端接入说明。
            String userName = "accessKey=" + accessKey + "|timestamp=" + timeStamp;
            //连接凭证接入码。
            String password = "${yourAccessCode}";
            //按照qpid-jms的规范,组装连接URL。
            String connectionUrl = "amqps://${UUCID}.iot-amqps.cn-north-4.myhuaweicloud.com:5671?amqp.vhost=default&amqp.idleTimeout=8000&amqp.saslMechanisms=PLAIN";
            Hashtable<String, String> hashtable = new Hashtable<>();
            hashtable.put("connectionfactory.HwConnectionURL", connectionUrl);
            //队列名,可以使用默认队列DefaultQueue
            String queueName = "${yourQueue}";
            hashtable.put("queue.HwQueueName", queueName);
            hashtable.put(Context.INITIAL_CONTEXT_FACTORY, "org.apache.qpid.jms.jndi.JmsInitialContextFactory");
            Context context = new InitialContext(hashtable);
            JmsConnectionFactory cf = (JmsConnectionFactory) context.lookup("HwConnectionURL");
            //同一个链接可创建多个queue,与前面queue.HwQueueName作好配对就行
            Destination queue = (Destination) context.lookup("HwQueueName");
    
            //信任服务端
            TransportOptions to = new TransportOptions(); to.setTrustAll(true);
            cf.setSslContext(TransportSupport.createJdkSslContext(to));
    
            // 创建连接
            Connection connection = cf.createConnection(userName, password);
            ((JmsConnection) connection).addConnectionListener(myJmsConnectionListener);
            // 创建 Session
            // Session.CLIENT_ACKNOWLEDGE: 收到消息后,需要手动调用message.acknowledge()。
            // Session.AUTO_ACKNOWLEDGE: SDK自动ACK(推荐)。
            Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
            connection.start();
            // 创建 Receiver Link
            MessageConsumer consumer = session.createConsumer(queue);
            //处理消息有两种方式
            // 1,主动拉数据(推荐),参照receiveMessage(consumer)
            // 2, 添加监听,参照consumer.setMessageListener(messageListener), 服务端主动推数据给客户端,但得考虑接受的数据速率是客户能力能够承受住的
            receiveMessage(consumer);
            // consumer.setMessageListener(messageListener);
        }
    
        private static void receiveMessage(MessageConsumer consumer) throws JMSException{
            while (true){
                try{
                    // 建议异步处理收到的消息,确保receiveMessage函数里没有耗时逻辑。
                    Message message = consumer.receive(); processMessage(message);
                } catch (Exception e) {
                    System.out.println("receiveMessage hand an exception: " + e.getMessage());
                    e.printStackTrace();
                }
            }
    
        }
    
        private static MessageListener messageListener = new MessageListener(){
            @Override
            public void onMessage(Message message){
                try {
                    // 建议异步处理收到的消息,确保onMessage函数里没有耗时逻辑。
                    // 如果业务处理耗时过程过长阻塞住线程,可能会影响SDK收到消息后的正常回调。
                    executorService.submit(() -> processMessage(message));
                } catch (Exception e){
                    System.out.println("submit task occurs exception: " + e.getMessage());
                    e.printStackTrace();
                }
            }
        };
    
        /**
         * 在这里处理您收到消息后的具体业务逻辑。
         */
        private static void processMessage(Message message) {
            try {
                String body = message.getBody(String.class); String content = new String(body);
                System.out.println("receive an message, the content is " + content);
            } catch (Exception e){
                System.out.println("processMessage occurs error: " + e.getMessage());
                e.printStackTrace();
            }
        }
    
        private static JmsConnectionListener myJmsConnectionListener = new JmsConnectionListener(){
            /**
             * 连接成功建立。
             */
            @Override
            public void onConnectionEstablished(URI remoteURI){
                System.out.println("onConnectionEstablished, remoteUri:" + remoteURI);
            }
    
            /**
             * 尝试过最大重试次数之后,最终连接失败。
             */
            @Override
            public void onConnectionFailure(Throwable error){
                System.out.println("onConnectionFailure, " + error.getMessage());
            }
    
            /**
             * 连接中断。
             */
            @Override
            public void onConnectionInterrupted(URI remoteURI){
                System.out.println("onConnectionInterrupted, remoteUri:" + remoteURI);
            }
    
            /**
             * 连接中断后又自动重连上。
             */
            @Override
            public void onConnectionRestored(URI remoteURI){
                System.out.println("onConnectionRestored, remoteUri:" + remoteURI);
            }
    
            @Override
            public void onInboundMessage(JmsInboundMessageDispatch envelope){
                System.out.println("onInboundMessage, " + envelope);
            }
    
            @Override
            public void onSessionClosed(Session session, Throwable cause){
                System.out.println("onSessionClosed, session=" + session + ", cause =" + cause);
            }
    
            @Override
            public void onConsumerClosed(MessageConsumer consumer, Throwable cause){
                System.out.println("MessageConsumer, consumer=" + consumer + ", cause =" + cause);
            }
    
            @Override
            public void onProducerClosed(MessageProducer producer, Throwable cause){
                System.out.println("MessageProducer, producer=" + producer + ", cause =" + cause);
            }
        };
    }
  • 业务服务器下发控制指令

    服务器通过调用消息下发接口下发控制指令,具体请参考下发设备消息。JAVA代码开发请参考Java Demo使用说明

    下发设备消息关键参数说明:

    名称

    必选/可选

    类型

    位置

    说明

    X-Auth-Token

    必选

    String

    Header

    用户Token。通过调用IAM服务 获取IAM用户Token接口获取,接口返回的响应消息头中“X-Subject-Token”就是需要获取的用户Token。简要的获取方法样例请参见 Token认证

    Instance-Id

    可选

    String

    Header

    实例ID。物理多租下各实例的唯一标识,一般华为云租户无需携带该参数,仅在物理多租场景下从管理面访问API时需要携带该参数。

    project_id

    必选

    String

    Path

    项目ID。获取方法请参见 获取项目ID

    device_id

    必选

    String

    Path

    填写迁移设备的设备ID。

    message

    必选

    String

    Body

    设备执行的消息,字符串,具体格式需要应用和设备约定。

    topic_full_name

    可选

    String(128)

    Body

    迁移设备的topic。

    接口样例:

    POST https://{Endpoint}/v5/iot/{project_id}/devices/{device_id}/messages
    Content-Type: application/json
    X-Auth-Token: ********
    Instance-Id: ********
    
    {
       "message": "{\"switch\":\"off\"}",
       "topic_full_name": "/aircondition/cmd"
    
    }

    JAVA核心代码样例如下:

    public class DeviceMessage {
        public static void main(String[] args) throws KeyManagementException, NoSuchAlgorithmException, IOException {
            String token = Authentication.getToken();
            Map<String, String> headers = new HashMap<String, String>();
            headers.put("Content-Type", "application/json");
            headers.put("X-Auth-Token", token);
    
            Message message = new Message();
            message.setMessage("{\"switch\" :\"off\"}");
            message.setTopic_full_name("/aircondition/cmd");
    
            String projectId = "11111";
            String deviceId = "5fae45e358115902ce609882_20201113";
            String url = "https://iotda.cn-north-4.myhuaweicloud.com/v5/iot/%s/devices/%s/messages";
            url = String.format(url, projectId, deviceId);
            HttpUtils httpUtils = new HttpUtils();
            httpUtils.initClient();
            StreamClosedHttpResponse httpResponse = httpUtils.doPost(url, headers, JsonUtils.Obj2String(message));
            System.out.println(httpResponse.getContent());
        }
    }
分享:

    相关文档

    相关产品

关闭导读