libcoap3对接华为云平台

文章目录

  • 前言
  • 一、平台注册
  • 二、引入源码库
    • 1.libcoap仓库编译
    • 2.分析网络报文
    • 3.案例代码
    • 4.编译&运行
  • 总结


前言

通过libcoap3开源代码库对接华为云平台,本文章将讨论加密与不加密的方式对接华为云平台。


一、平台注册

首先,你需要在华为云平台上创建一个coap协议的设备,并制定好数据格式,这个自行百度。

二、引入源码库

1.libcoap仓库编译

步骤如下(linux环境):

#参考https://raw.githubusercontent.com/obgm/libcoap/develop/BUILDING
git clone https://github.com/obgm/libcoap.git
cd libcoap
cd ext
# v0.9-rc1 这个版本的加密库没啥问题,其它版本没试
git clone --branch v0.9-rc1 --single-branch https://github.com/eclipse/tinydtls.git
cd ..
#这一步要做,不然报错
cp autogen.sh ext/tinydtls/ 
cmake -E remove_directory build
cmake -E make_directory build
cd build
cmake --build . -- install
cp  lib/libtinydtls.so /usr/local/lib/

2.分析网络报文

在这里插入图片描述
(1)连接报文
注意这几个选项,在下面的案例代码将体现
在这里插入图片描述
(2)服务端资源请求报文
重点关注这个token,后面主动上传全靠它
在这里插入图片描述
(3)数据上传报文
在这里插入图片描述

3.案例代码

先找到设备标识码

在这里插入图片描述

#include <threads.h>
#include <coap3/coap.h>

#include <ctype.h>
#include <stdio.h>


#define COAP_CLIENT_URI "coap://015f8fcbf7.iot-coaps.cn-north-4.myhuaweicloud.com"

#define COAP_USE_PSK_ID "561342" //修改这个地方的标识码

#define RD_ROOT_STR   "t/d"

typedef unsigned char method_t;
unsigned char msgtype = COAP_MESSAGE_NON;
static coap_context_t *main_coap_context = NULL;
static int quit = 0;
static int is_mcast = 0;

uint8_t sensor_data[2]={0x00,0x33};	

unsigned int wait_seconds = 5; /* default timeout in seconds */

static unsigned char _token_data[24]={0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38}; /* With support for RFC8974 */
coap_binary_t the_token = { 8, _token_data };

static void handle_observe_request(coap_resource_t *resource COAP_UNUSED,
           coap_session_t *session,
           const coap_pdu_t *request,
           const coap_string_t *query COAP_UNUSED,
           coap_pdu_t *response) {

  unsigned char observe_op=0;
  
  coap_log_info("handle_observe_request  \n");
  /* 打印服务端的报文信息 */
  coap_show_pdu(COAP_LOG_INFO, request);

  
  coap_pdu_set_code(response, COAP_RESPONSE_CODE_CHANGED);
  

  /* 存储服务端发来的token   ,后面数据上传需要这个token */
  coap_bin_const_t token = coap_pdu_get_token(request);
  memcpy(the_token.s,token.s,token.length);
  the_token.length = token.length;
 
  coap_add_option(response,
                  COAP_OPTION_OBSERVE,
                  1,
                  &observe_op);
  
}

  
static void
free_xmit_data(coap_session_t *session COAP_UNUSED, void *app_ptr) {
  coap_free(app_ptr);
  return;
}

/* 上报设备信息 */
static int 
coap_report(coap_context_t *ctx,
                 coap_session_t *session,
                 method_t m,
                 unsigned char *data,
                 size_t length) {
  coap_pdu_t *pdu;
  (void)ctx;

  coap_log_debug("sending CoAP request\n");
  if (!(pdu = coap_new_pdu(COAP_MESSAGE_NON, m, session))) {
    free_xmit_data(session, data);
    return 1;
  }
  
  if (!coap_add_token(pdu, the_token.length,the_token.s)) {
    coap_log_debug("cannot add token to request\n");
  }

  if (!length) {
  		return 1;
  }

  coap_add_data( pdu,  length,  data);
  if (coap_send(session, pdu) == COAP_INVALID_MID) {
     coap_log_err("cannot send CoAP pdu\n");
     return 1;
  }



  return 0;
}


static coap_response_t
message_handler(coap_session_t *session COAP_UNUSED,
				const coap_pdu_t *sent,
				const coap_pdu_t *received,
				const coap_mid_t id COAP_UNUSED) {

  coap_opt_t *block_opt;
  coap_opt_iterator_t opt_iter;
  size_t len;
  const uint8_t *databuf;
  size_t offset;
  size_t total;
  coap_pdu_code_t rcv_code = coap_pdu_get_code(received);
  coap_pdu_type_t rcv_type = coap_pdu_get_type(received);
  coap_bin_const_t token = coap_pdu_get_token(received);
  coap_string_t *query = coap_get_query(received);

  coap_log_debug("** process incoming %d.%02d response:\n",
				 COAP_RESPONSE_CLASS(rcv_code), rcv_code & 0x1F);
  
  coap_show_pdu(COAP_LOG_INFO, received);

  
  return COAP_RESPONSE_OK;

 
}

static void init_resources(coap_context_t * ctx)
{
  coap_resource_t *r;
  /* 创建设备资源,后面服务器要访问这个资源 */
  r = coap_resource_init(coap_make_str_const(RD_ROOT_STR), 0);
  /* 绑定get 方法,这个是通过抓包发现的 */
  coap_register_request_handler(r, COAP_REQUEST_GET, handle_observe_request);
  coap_add_resource(ctx, r);

}

static int
resolve_address(const char *host, const char *service, coap_address_t *dst,
				int scheme_hint_bits)
{
  uint16_t port = service ? atoi(service) : 0;
  int ret = 0;
  coap_str_const_t str_host;
  coap_addr_info_t *addr_info;

  str_host.s = (const uint8_t *)host;
  str_host.length = strlen(host);
  addr_info = coap_resolve_address_info(&str_host, port, port, port, port,
										AF_UNSPEC, scheme_hint_bits,
										COAP_RESOLVE_TYPE_REMOTE);
  if (addr_info)
  {
	ret = 1;
	*dst = addr_info->addr;
	is_mcast = coap_is_mcast(dst);
  }
  
  coap_free_address_info(addr_info);
  return ret;
}


void client_coap_init(void)
{
  coap_session_t *session = NULL;
  coap_pdu_t *pdu;
  coap_address_t dst;
  coap_mid_t mid;
  int len;
  coap_uri_t uri;
  char portbuf[8];
  char uri_query_op[20]={0};
  unsigned int wait_ms = 0;
  int result = -1;
#define BUFSIZE 100
  unsigned char buf[BUFSIZE];
  int res;
  const char *coap_uri = COAP_CLIENT_URI;

  /* Initialize libcoap library */
  coap_startup();
  coap_set_log_level(COAP_MAX_LOGGING_LEVEL);

  /* Parse the URI */
  len = coap_split_uri((const unsigned char *)coap_uri, strlen(coap_uri), &uri);
  if (len != 0)
  {
    coap_log_warn("Failed to parse uri %s\n", coap_uri);
    goto fail;
  }

  snprintf(portbuf, sizeof(portbuf), "%d", uri.port);
  snprintf((char *)buf, sizeof(buf), "%*.*s", (int)uri.host.length,
           (int)uri.host.length, (const char *)uri.host.s);

  /* resolve destination address where packet should be sent */
  len = resolve_address((const char *)buf, portbuf, &dst, 1 << uri.scheme);
  if (len <= 0)
  {
    coap_log_warn("Failed to resolve address %*.*s\n", (int)uri.host.length,
                  (int)uri.host.length, (const char *)uri.host.s);
    goto fail;
  }
 
  main_coap_context = coap_new_context(NULL);
  if (!main_coap_context)
  {
    coap_log_warn("Failed to initialize context\n");
    goto fail;
  }
  init_resources(main_coap_context);
  coap_context_set_block_mode(main_coap_context, COAP_BLOCK_USE_LIBCOAP);
  coap_context_set_keepalive(main_coap_context, 60);

  session = coap_new_client_session(main_coap_context, NULL, &dst,
									   COAP_PROTO_UDP);
  
  coap_session_init_token(session, the_token.length, the_token.s);
  coap_register_response_handler(main_coap_context, message_handler);

  /* construct CoAP message */
  pdu = coap_pdu_init(COAP_MESSAGE_CON,
					  COAP_REQUEST_CODE_POST,
					  coap_new_message_id(session),
					  coap_session_max_pdu_size(session));
  if (!pdu) {
	  coap_log_warn("Failed to create PDU\n");
	  goto fail;
  }

  if (!coap_add_token(pdu, the_token.length, the_token.s)) {
    coap_log_debug("cannot add token to request\n");
  }
  coap_add_option(pdu, COAP_OPTION_URI_PATH, 1, "t");
  
  coap_add_option(pdu, COAP_OPTION_URI_PATH, 1, "r");
  
  unsigned char opbuf[40];
  coap_add_option(pdu, COAP_OPTION_CONTENT_FORMAT,
				  coap_encode_var_safe(opbuf, sizeof(opbuf),
									   COAP_MEDIATYPE_APPLICATION_OCTET_STREAM),
				  opbuf);
  sprintf(uri_query_op,"ep=%s",COAP_USE_PSK_ID);
  coap_add_option(pdu, COAP_OPTION_URI_QUERY, strlen(uri_query_op), uri_query_op);
  
  if (is_mcast)
  {
    wait_seconds = coap_session_get_default_leisure(session).integer_part + 1;
  }
  wait_ms = wait_seconds * 1000;

  /* and send the PDU */
  mid = coap_send(session, pdu);
  if (mid == COAP_INVALID_MID)
  {
    coap_log_warn("Failed to send PDU\n");
    goto fail;
  }


  while (!quit || is_mcast)
  {
    result = coap_io_process(main_coap_context, 1000);
	coap_log_info("result  %d  wait_ms %d\n",result,wait_ms);
    if (result >= 0)
    {
      if (wait_ms > 0)
      {
        if ((unsigned)result >= wait_ms)
        {
          //产生一个随机值
          sensor_data[1] = wait_ms/100+result-wait_ms;
		  if ( 1 == coap_report(main_coap_context, session, COAP_RESPONSE_CODE_CONTENT,
                                       sensor_data, 2)) {
            goto fail;
          }
		  wait_ms = wait_seconds * 1000;
        }
        else
        {
          wait_ms -= result;
        }
      }
    }
  }
fail:
   /* Clean up library usage so client can be run again */
  quit = 0;
  coap_session_release(session);
  session = NULL;
  coap_free_context(main_coap_context);
  main_coap_context = NULL;
  coap_cleanup();
}

int main()
{
  client_coap_init();  

}

4.编译&运行

(1)编译
gcc -o test_coap_nodtls test_coap_nodtls.c -I/usr/local/include/coap3/ -lcoap-3 -ltinydtls -I/usr/local/include/tinydtls/
(2)运行
在这里插入图片描述
在这里插入图片描述

总结

加密对接的代码就不放出来,这个先参考着搞吧。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Matlab提取excel数据及处理的实操举例

实现目的 当excel数据量庞大的时候&#xff0c;如果采用人工处理数据的方法就会成为非常出力不讨好的事&#xff0c;既容易出错&#xff0c;又容易抑郁。 利用matlab处理成为既简单又高效的方式。 例如&#xff0c;以GD32F7xx系列的管脚复用表格为例&#xff0c;在169x19的e…

昇思MindSpore学习笔记6-06计算机视觉--Vision Transormer图像分类

摘要&#xff1a; 记录MindSpore AI框架使用ViT模型在ImageNet图像数据分类上进行训练、验证、推理的过程和方法。包括环境准备、下载数据集、数据集加载、模型解析与构建、模型训练与推理等。 一、概念 1. ViT模型 Vision Transformer 自注意结构模型 Self-Attention Tran…

ubuntu部署minio集群

minio集群介绍 官方文档&#xff1a;https://min.io/docs/minio/linux/operations/install-deploy-manage/deploy-minio-multi-node-multi-drive.html 本方案采用在多节点多驱动器 (MNMD) 或“分布式”配置部署 MinIO。 MNMD 部署提供企业级性能、可用​​性和可扩展性&#…

力扣 203反转链表

思路 用cur->next指向pre,把链表倒转 cur后移&#xff0c;cur指向原链表的下一个 注意用tmp存储原链表中cur的后一个 class Solution { public: ListNode* reverseList(ListNode* head) { ListNode *cur head; ListNode *pre nullptr; ListNode *tmp; while (cur ! nul…

书生·浦语2.5开源,推理能力再创新标杆

2024 年 7 月 3 日&#xff0c;上海人工智能实验室与商汤科技联合香港中文大学和复旦大学正式发布新一代大语言模型书⽣浦语2.5&#xff08;InternLM2.5&#xff09;。相比上一代模型&#xff0c;InternLM2.5 有三项突出亮点&#xff1a; 推理能力大幅提升&#xff0c;领先于国…

【代码随想录】【算法训练营】【第59天】 [卡码110]字符串接龙 [卡码105]有向图的完全可达性 [卡码106]岛屿的周长

前言 思路及算法思维&#xff0c;指路 代码随想录。 题目来自 卡码网。 day 59&#xff0c;周五&#xff0c;继续ding~ 题目详情 [卡码110] 字符串接龙 题目描述 卡码110 字符串接龙 解题思路 前提&#xff1a; 思路&#xff1a; 重点&#xff1a; 代码实现 C语言 […

光伏仿真系统推荐

在全球能源转型和绿色能源发展的背景下&#xff0c;光伏行业作为重要的绿色能源组成部分&#xff0c;其智能化、数字化的发展显得尤为关键。光伏仿真系统作为提升光伏项目设计、运维效率的重要工具&#xff0c;在行业中扮演着不可或缺的角色。在众多光伏仿真系统中&#xff0c;…

自动化(二正)

Java接口自动化用到的技术栈 技术栈汇总&#xff1a; ①Java基础&#xff08;封装、反射、泛型、jdbc&#xff09; ②配置文件解析(properties) ③httpclient&#xff08;发送http请求&#xff09; ④fastjson、jsonpath处理数据的 ⑤testng自动化测试框架重点 ⑥allure测试报…

从0开始的STM32HAL库学习4

对射式红外传感器计数复现 配置工程 我们直接复制oled的工程&#xff0c;但是要重命名。 将PB14设置为中断引脚 自定义命名为sensorcount 设置为上升沿触发 打开中断 配置NVCI 都为默认就可以了 修改代码 修改stm32f1xx_it.c 文件 找到中断函数并修改 void EXTI15_10_I…

element plus 实现跨页面+跨tab栏多选

文章目录 element plus 层面数据层面 菜鸟好久没写博客了&#xff0c;主要是没遇见什么很难的问题&#xff0c;今天碰见了一个没有思路的问题&#xff0c;解决后立马来和大家伙分享了&#xff01; 菜鸟今天要实现一个需求&#xff0c;就是&#xff1a;实现跨页面跨 tab栏 多选…

Linux 程序卡死的特殊处理

一、前言 Linux环境。 我们在日常编写的程序中&#xff0c;可能会出现一些细节问题&#xff0c;导致程序卡死&#xff0c;即程序没法正常运行&#xff0c;界面卡住&#xff0c;也不会闪退... 当这种问题出现在客户现场&#xff0c;那就是大问题了。。。 当我们暂时还无法排…

USB转RS485+RS232+TTL串口电路

USB转RS485RS232TTL电路 USB转RS485RS232TTL电路如下图所示&#xff0c;可实现USB转RS485RS232TTL串口&#xff0c;一个电路模块即可实现电路调试过程中用到常用接口。 电路模块上留有2.54MM单排针接口和接线端子两种接线方式&#xff0c;可接线和跳线。电路模块同时有5V和3.3V…

不仅是输出信息,console.log 也能玩出花

console.log 是 JavaScript 中一个常用的函数&#xff0c;用于向控制台输出信息。 console.log 虽然主要用于调试目的&#xff0c;但也包含了一些有趣的用法&#xff0c; console.log 不仅能输出文本&#xff0c;还能以更丰富的方式展示信息。 比如我们打开 B 站&#xff0c;然…

计算机网络体系结构解析

OSI参考模型 与 TCP/IP模型 如图所示 TCP/IP模型有几层 应用层&#xff1a;只需要专注于为用户提供应用功能 HTTP、SMTP、Telnet等&#xff0c;工作在操作系统中的用户态&#xff0c;传输层及以下工作在内核态传输层&#xff1a;为应用层提供网络支持&#xff08;TCP、UDP传…

c++多态的定义和原理

目录 1、多态的定义和实现 1.多态的构成条件 2.虚函数 3.虚函数的重写(覆盖) 4.虚函数重写的两个例外 5.c11 override和final 6.重载&#xff0c;覆盖(重写)和隐藏(重定义) 2、抽象类 概念 接口继承和实现继承 3、多态的原理 1.虚函数表 2.多态的原理 4、多继承中的虚…

武夷山细节决定成败抓质量求生存

在当今竞争激烈的市场环境中&#xff0c;细节决定成败&#xff0c;质量求生存的理念已成为企业发展的关键。蓝鹏测控科技有限公司&#xff0c;一家专业从事工业测量领域的高新技术企业&#xff0c;正是秉持这一理念&#xff0c;在工业测径仪领域取得了显著成就。 蓝鹏测控科技…

Ozon俄罗斯哪些产品热销中?Ozon7月市场热卖趋势放送

Ozon俄罗斯哪些产品热销工具&#xff1a;D。DDqbt。COm/74rD 据Ozon数据&#xff0c;2023年&#xff0c;在自提服务方面&#xff0c;Ozon投资了100亿扩展自提网络&#xff0c;自提点数量激增至超过5万个&#xff0c;是之前的2.6倍。 物流基础设施方面&#xff0c;Ozon在仓库建…

BGP第二日

上图为今日所用拓扑 &#xff0c;其中R1和R4&#xff0c;R3和R5为EBGP邻居&#xff0c;R1和R3为IBGP邻居&#xff0c;AS200区域做OSPF动态路由 一.BGP建立邻居的六种状态 1.idle 空闲状态&#xff1a;建立邻居最初的状态 2.Connect 连接状态&#xff1a;在…

360安全浏览器就是不行-python秒破解

下面画框都很容易破解&#xff0c;大家试试

ZGC在三色指针中的应用

ZGC基于颜色指针的并发处理算法 ZGC初始化之后&#xff0c;整个内存空间的地址视图被设置为Remapped&#xff0c;当进入标记阶段时的视图转变为Marked0&#xff08;也称为M0&#xff09;或者Marked1&#xff08;也称为M1&#xff09;&#xff0c;从标记阶段结束进入转移阶段时…