往期知识点记录:
- 鸿蒙(HarmonyOS)应用层开发(北向)知识点汇总
- OpenHarmony之分布式软总线json_payload.c(一)
- OpenHarmony之分布式软总线json_payload.c(二)
- OpenHarmony之分布式软总线json_payload.c(三)
- OpenHarmony之分布式软总线json_payload.c(四)
- OpenHarmony之分布式软总线nstackx_common.c分析
- OpenHarmony之分布式软总线coap_discover.c(一)
- 持续更新中……
前言
coap_discover.c实现了基于coap的设备发现功能的细节。在之前说过coap_service.c在实现coap服务发现的前期工作中做了许多前期初始化功能和注册服务的工作,这些功能函数是调用nstackx_device.c提供的接口函数,而nstackx_common.c中的接口函数又调用coap_discover.c中提供的函数接口。所以在coap_discover.c中实现了具体的初始化细节,并且在这个文件中定义了发送设备信息服务和接收对端设备信息服务功能。
coap_discover.c
请求和响应接口
首先定义了请求/响应结构
/*COAP 请求(响应)结构 , 响应和请求共用统一首部*/
typedef struct CoapRequest {
const char *remoteUrl; //URL
char *data; // 数据
size_t dataLength; //数据长度
const char *remoteIp; //远端iP
} CoapRequest;
发送coap请求报文
/*
函数功能:发送coap请求报文
函数参数: coapRequest : 待发送的请求数据结构
函数返回值: 成功返回0,否则返回非零
详细:
1.判断请求报文,报文中url和IP地址是否为空
2.socket通用操作,将IP字符串格式的ip地址转换为一个uint32_t数字格式,接口转为网络字节序、选择IPv4
3.创建一个客户端socket并且connect地址:端口
4.获取创建好的客户端socket套接字
5.发送请求报文
*/
static int CoapSendRequest(const CoapRequest *coapRequest)
{
if (coapRequest == NULL || coapRequest->remoteUrl == NULL) {
return NSTACKX_EFAILED;
}
struct sockaddr_in sockAddr = {0};
if (coapRequest->remoteIp == NULL) {
return NSTACKX_EFAILED;
}
//将IP字符串格式的ip地址转换为一个uint32_t数字格式
sockAddr.sin_addr.s_addr = inet_addr(coapRequest->remoteIp);
//将接口转为网络字节序
sockAddr.sin_port = htons(COAP_DEFAULT_PORT);
//选择IPv4
sockAddr.sin_family = AF_INET;
//创建一个客户端socket并且connect地址:端口
int ret = CoapCreateUdpClient(&sockAddr);
if (ret != NSTACKX_EOK) {
return NSTACKX_EFAILED;
}
//socket信息,获取创建好的客户端socket套接字
SocketInfo socket = {0};
socket.cliendFd = GetCoapClientSocket();
socket.dstAddr = sockAddr;
//发送数据
if (CoapSocketSend(&socket, (uint8_t *)coapRequest->data, coapRequest->dataLength) == -1) {
SOFTBUS_PRINT("[DISCOVERY]reponse coap failed.\r\n");
return NSTACKX_EFAILED;
}
return NSTACKX_EOK;
}
发送响应报文
/*
函数功能: 发送响应报文
函数参数:
pkt: coap 报文
remoteUrl : 远端Url
remoteIp : 远端IP
函数返回值:成功返回0,否则非零
详细:
1. 声明一个响应报文结构,并且初始化
2. 设置url和ip
3. 调用PrepareServiceDiscover,cJSON封装本地数据(设备信息,iP地址,设备功能)
4. 声明coap 读写缓冲区,并分配空间1*COAP_MAX_PDU_SIZE,且初始化为0
5. 设置缓存区长度为COAP_MAX_PDU_SIZE,当前还没有数据所以长度为0
6. 调用coap_adapter.c中BuildSendPkt函数创建发送coap数据包,组装成的coap数据包存储在sndPktBuff读写缓存区中
7. 将存储在sndPktBuff读写缓存区中的coap作为报文结构中的数据主体部分,并设置报文数据长度
8. 最后释放缓冲区空间,并将指针置为空,防止野指针
*/
static int CoapResponseService(const COAP_Packet *pkt, const char* remoteUrl, const char* remoteIp)
{
int ret;
CoapRequest coapRequest; //响应报文结构
(void)memset_s(&coapRequest, sizeof(coapRequest), 0, sizeof(coapRequest)); //初始化
coapRequest.remoteUrl = remoteUrl; //远端url
coapRequest.remoteIp = remoteIp; //远端ip
char *payload = PrepareServiceDiscover(); //payload 具有本地设备信息 IP地址,设备功能
if (payload == NULL) {
return NSTACKX_EFAILED;
}
//发送读写缓冲区
COAP_ReadWriteBuffer sndPktBuff = {0};
sndPktBuff.readWriteBuf = calloc(1, COAP_MAX_PDU_SIZE);
//分配不成功
if (sndPktBuff.readWriteBuf == NULL) {
free(payload);
return NSTACKX_EFAILED;
}
sndPktBuff.size = COAP_MAX_PDU_SIZE;
sndPktBuff.len = 0;
//创建响应coap数据包,组装成的coap数据包存储在sndPktBuff读写缓存区中
ret = BuildSendPkt(pkt, remoteIp, payload, &sndPktBuff);
free(payload);
if (ret != DISCOVERY_ERR_SUCCESS) {
free(sndPktBuff.readWriteBuf);
sndPktBuff.readWriteBuf = NULL;
return ret;
}
//将存储在sndPktBuff读写缓存区中的coap作为报文结构中的数据主体部分
coapRequest.data = sndPktBuff.readWriteBuf;
//设置报文数据长度
coapRequest.dataLength = sndPktBuff.len;
//发送报文
ret = CoapSendRequest(&coapRequest);
//最后释放readWriteBuf空间,并置空防止野指针
free(sndPktBuff.readWriteBuf);
sndPktBuff.readWriteBuf = NULL;
return ret;
}
写在最后
如果你觉得这篇内容对你还蛮有帮助,我想邀请你帮我三个小忙:
- 点赞,转发,有你们的 『点赞和评论』,才是我创造的动力。
- 关注小编,同时可以期待后续文章ing🚀,不定期分享原创知识。
- 想要获取更多完整鸿蒙最新学习资源,请看下图提示: