2022-03-22 16:29:24 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2015-2017 Ken Bannister. All rights reserved.
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief gcoap block server handler example
|
|
|
|
*
|
|
|
|
* @author Ken Bannister <kb2ma@runbox.com>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "fmt.h"
|
|
|
|
#include "hashes/sha256.h"
|
|
|
|
#include "net/gcoap.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
|
|
#include "debug.h"
|
|
|
|
|
2022-04-15 21:18:22 +02:00
|
|
|
static ssize_t _sha256_handler(coap_pkt_t* pdu, uint8_t *buf, size_t len, coap_request_ctx_t *ctx);
|
|
|
|
static ssize_t _riot_block2_handler(coap_pkt_t *pdu, uint8_t *buf, size_t len, coap_request_ctx_t *ctx);
|
2022-03-22 16:29:24 +01:00
|
|
|
|
|
|
|
/* CoAP resources */
|
|
|
|
static const coap_resource_t _resources[] = {
|
|
|
|
{ "/riot/ver", COAP_GET, _riot_block2_handler, NULL },
|
|
|
|
{ "/sha256", COAP_POST, _sha256_handler, NULL },
|
|
|
|
};
|
|
|
|
|
|
|
|
static gcoap_listener_t _listener = {
|
|
|
|
.resources = _resources,
|
|
|
|
.resources_len = ARRAY_SIZE(_resources),
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Constants for /riot/ver. */
|
|
|
|
static const uint8_t block2_intro[] = "This is RIOT (Version: ";
|
|
|
|
static const uint8_t block2_board[] = " running on a ";
|
|
|
|
static const uint8_t block2_mcu[] = " board with a ";
|
|
|
|
|
2022-04-15 21:18:22 +02:00
|
|
|
static ssize_t _riot_block2_handler(coap_pkt_t *pdu, uint8_t *buf, size_t len, coap_request_ctx_t *ctx)
|
2022-03-22 16:29:24 +01:00
|
|
|
{
|
|
|
|
(void)ctx;
|
|
|
|
coap_block_slicer_t slicer;
|
|
|
|
coap_block2_init(pdu, &slicer);
|
|
|
|
|
|
|
|
gcoap_resp_init(pdu, buf, len, COAP_CODE_CONTENT);
|
|
|
|
coap_opt_add_format(pdu, COAP_FORMAT_TEXT);
|
|
|
|
coap_opt_add_block2(pdu, &slicer, 1);
|
|
|
|
ssize_t plen = coap_opt_finish(pdu, COAP_OPT_FINISH_PAYLOAD);
|
|
|
|
|
|
|
|
/* Add actual content */
|
|
|
|
plen += coap_blockwise_put_bytes(&slicer, buf+plen, block2_intro, sizeof(block2_intro)-1);
|
|
|
|
plen += coap_blockwise_put_bytes(&slicer, buf+plen, (uint8_t*)RIOT_VERSION, strlen(RIOT_VERSION));
|
|
|
|
plen += coap_blockwise_put_char(&slicer, buf+plen, ')');
|
|
|
|
plen += coap_blockwise_put_bytes(&slicer, buf+plen, block2_board, sizeof(block2_board)-1);
|
|
|
|
plen += coap_blockwise_put_bytes(&slicer, buf+plen, (uint8_t*)RIOT_BOARD, strlen(RIOT_BOARD));
|
|
|
|
plen += coap_blockwise_put_bytes(&slicer, buf+plen, block2_mcu, sizeof(block2_mcu)-1);
|
|
|
|
plen += coap_blockwise_put_bytes(&slicer, buf+plen, (uint8_t*)RIOT_MCU, strlen(RIOT_MCU));
|
|
|
|
/* To demonstrate individual chars */
|
|
|
|
plen += coap_blockwise_put_char(&slicer, buf+plen, ' ');
|
|
|
|
plen += coap_blockwise_put_char(&slicer, buf+plen, 'M');
|
|
|
|
plen += coap_blockwise_put_char(&slicer, buf+plen, 'C');
|
|
|
|
plen += coap_blockwise_put_char(&slicer, buf+plen, 'U');
|
|
|
|
plen += coap_blockwise_put_char(&slicer, buf+plen, '.');
|
|
|
|
|
|
|
|
coap_block2_finish(&slicer);
|
|
|
|
|
|
|
|
return plen;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Uses block1 POSTs to generate an sha256 digest. */
|
2022-04-15 21:18:22 +02:00
|
|
|
static ssize_t _sha256_handler(coap_pkt_t* pdu, uint8_t *buf, size_t len, coap_request_ctx_t *ctx)
|
2022-03-22 16:29:24 +01:00
|
|
|
{
|
|
|
|
(void)ctx;
|
|
|
|
|
|
|
|
/* using a shared sha256 context *will* break if two requests are handled
|
|
|
|
* at the same time. doing it anyways, as this is meant to showcase block1
|
|
|
|
* support, not proper synchronisation. */
|
|
|
|
static sha256_context_t sha256;
|
|
|
|
uint8_t digest[SHA256_DIGEST_LENGTH];
|
|
|
|
coap_block1_t block1;
|
|
|
|
|
|
|
|
int blockwise = coap_get_block1(pdu, &block1);
|
|
|
|
|
|
|
|
printf("_sha256_handler: received data: offset=%u len=%u blockwise=%i more=%i\n",
|
|
|
|
(unsigned)block1.offset, pdu->payload_len, blockwise, block1.more);
|
|
|
|
|
|
|
|
/* initialize sha256 calculation and add payload bytes */
|
|
|
|
if (block1.blknum == 0) {
|
|
|
|
puts("_sha256_handler: init");
|
|
|
|
sha256_init(&sha256);
|
|
|
|
}
|
|
|
|
sha256_update(&sha256, pdu->payload, pdu->payload_len);
|
|
|
|
|
|
|
|
unsigned resp_code = COAP_CODE_CHANGED;
|
|
|
|
if (block1.more) {
|
|
|
|
resp_code = COAP_CODE_CONTINUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* start response */
|
|
|
|
gcoap_resp_init(pdu, buf, len, resp_code);
|
|
|
|
|
|
|
|
/* has payload */
|
|
|
|
if (!blockwise || !block1.more) {
|
|
|
|
coap_opt_add_format(pdu, COAP_FORMAT_TEXT);
|
|
|
|
}
|
|
|
|
if (blockwise) {
|
|
|
|
coap_opt_add_block1_control(pdu, &block1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* include digest if done, otherwise response code above asks for next block */
|
|
|
|
size_t pdu_len = 0;
|
|
|
|
if (!blockwise || !block1.more) {
|
|
|
|
puts("_sha256_handler: finish");
|
|
|
|
sha256_final(&sha256, digest);
|
|
|
|
pdu_len = coap_opt_finish(pdu, COAP_OPT_FINISH_PAYLOAD);
|
|
|
|
|
|
|
|
pdu_len += fmt_bytes_hex((char *)pdu->payload, digest, sizeof(digest));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
pdu_len = coap_opt_finish(pdu, COAP_OPT_FINISH_NONE);
|
|
|
|
}
|
|
|
|
|
|
|
|
return pdu_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
int gcoap_cli_cmd(int argc, char **argv)
|
|
|
|
{
|
|
|
|
if (argc == 1) {
|
|
|
|
/* show help for main commands */
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp(argv[1], "info") == 0) {
|
|
|
|
uint8_t open_reqs = gcoap_op_state();
|
|
|
|
|
|
|
|
printf("CoAP server is listening on port %u\n", CONFIG_GCOAP_PORT);
|
|
|
|
printf("CoAP open requests: %u\n", open_reqs);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
end:
|
|
|
|
printf("usage: %s <info>\n", argv[0]);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void gcoap_cli_init(void)
|
|
|
|
{
|
|
|
|
gcoap_register_listener(&_listener);
|
|
|
|
}
|