2022-04-12 22:35:34 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2022 ML!PA Consulting GmbH
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup sys_shell_commands
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief NanoCoAP commands that interact with the filesystem
|
|
|
|
*
|
|
|
|
* @author Benjamin Valentin <benjamin.valentin@ml-pa.com>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <inttypes.h>
|
|
|
|
|
|
|
|
#include "net/nanocoap_sock.h"
|
2022-05-31 23:27:27 +02:00
|
|
|
#include "net/nanocoap_vfs.h"
|
|
|
|
#include "shell.h"
|
|
|
|
#include "vfs_default.h"
|
2022-04-12 22:35:34 +02:00
|
|
|
|
2022-08-04 14:54:44 +02:00
|
|
|
/**
|
|
|
|
* @brief Default download location for ncget
|
|
|
|
*/
|
|
|
|
#ifndef CONFIG_NCGET_DEFAULT_DATA_DIR
|
|
|
|
#define CONFIG_NCGET_DEFAULT_DATA_DIR VFS_DEFAULT_DATA
|
|
|
|
#endif
|
|
|
|
|
2022-04-12 22:35:34 +02:00
|
|
|
struct dir_list_ctx {
|
|
|
|
char *buf;
|
|
|
|
char *cur;
|
|
|
|
char *end;
|
|
|
|
};
|
|
|
|
|
|
|
|
static bool _is_dir(const char *url)
|
|
|
|
{
|
|
|
|
int len = strlen(url);
|
|
|
|
return url[len - 1] == '/';
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _print_cb(void *arg, size_t offset, uint8_t *buf, size_t len, int more)
|
|
|
|
{
|
|
|
|
(void)arg;
|
|
|
|
(void)offset;
|
|
|
|
|
|
|
|
write(STDOUT_FILENO, buf, len);
|
|
|
|
if (!more) {
|
|
|
|
puts("");
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _print_dir_cb(void *arg, size_t offset, uint8_t *buf, size_t len, int more)
|
|
|
|
{
|
|
|
|
(void)offset;
|
|
|
|
(void)more;
|
|
|
|
|
|
|
|
struct dir_list_ctx *ctx = arg;
|
|
|
|
|
|
|
|
char *end = (char *)buf + len;
|
|
|
|
for (char *c = (char *)buf; c < end; ++c) {
|
|
|
|
if (ctx->cur) {
|
|
|
|
if (*c == '>' || ctx->cur == ctx->end) {
|
|
|
|
*ctx->cur = 0;
|
|
|
|
puts(ctx->buf);
|
|
|
|
ctx->cur = NULL;
|
|
|
|
} else {
|
|
|
|
*ctx->cur++ = *c;
|
|
|
|
}
|
|
|
|
} else if (*c == '<') {
|
|
|
|
ctx->cur = ctx->buf;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _print_dir(const char *url, char *buf, size_t len)
|
|
|
|
{
|
|
|
|
struct dir_list_ctx ctx = {
|
|
|
|
.buf = buf,
|
|
|
|
.end = buf + len,
|
|
|
|
};
|
|
|
|
return nanocoap_get_blockwise_url(url, CONFIG_NANOCOAP_BLOCKSIZE_DEFAULT,
|
|
|
|
_print_dir_cb, &ctx);
|
|
|
|
}
|
|
|
|
|
2022-04-18 03:10:09 +02:00
|
|
|
static int _nanocoap_get_handler(int argc, char **argv)
|
2022-04-12 22:35:34 +02:00
|
|
|
{
|
|
|
|
int res;
|
2022-04-18 03:10:09 +02:00
|
|
|
char buffer[CONFIG_NANOCOAP_URI_MAX];
|
2022-04-12 22:35:34 +02:00
|
|
|
char *dst, *url = argv[1];
|
|
|
|
|
|
|
|
if (argc < 2) {
|
|
|
|
printf("Usage: %s <url> [destination]\n", argv[0]);
|
2022-08-04 14:54:44 +02:00
|
|
|
printf("Default destination: %s\n", CONFIG_NCGET_DEFAULT_DATA_DIR);
|
2022-04-12 22:35:34 +02:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_is_dir(url) && argc < 3) {
|
|
|
|
res = _print_dir(url, buffer, sizeof(buffer));
|
|
|
|
if (res) {
|
|
|
|
printf("Request failed: %s\n", strerror(-res));
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (argc < 3) {
|
|
|
|
dst = strrchr(url, '/');
|
|
|
|
if (dst == NULL) {
|
|
|
|
printf("invalid url: '%s'\n", url);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
if (snprintf(buffer, sizeof(buffer), "%s%s",
|
2022-08-04 14:54:44 +02:00
|
|
|
CONFIG_NCGET_DEFAULT_DATA_DIR, dst) >= (int)sizeof(buffer)) {
|
2022-04-12 22:35:34 +02:00
|
|
|
printf("Output file path too long\n");
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
dst = buffer;
|
|
|
|
} else {
|
2022-08-01 17:06:43 +02:00
|
|
|
char *filename = strrchr(url, '/');
|
2022-04-12 22:35:34 +02:00
|
|
|
dst = argv[2];
|
2022-08-01 17:06:43 +02:00
|
|
|
if (_is_dir(dst) && filename) {
|
|
|
|
if (snprintf(buffer, sizeof(buffer), "%s%s",
|
|
|
|
dst, filename + 1) >= (int)sizeof(buffer)) {
|
|
|
|
printf("Output file path too long\n");
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
dst = buffer;
|
|
|
|
}
|
2022-04-12 22:35:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* alternatively write the file to stdout */
|
|
|
|
if (strcmp(dst, "-") == 0) {
|
|
|
|
return nanocoap_get_blockwise_url(url, CONFIG_NANOCOAP_BLOCKSIZE_DEFAULT,
|
|
|
|
_print_cb, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
res = nanocoap_vfs_get_url(url, dst);
|
|
|
|
if (res < 0) {
|
|
|
|
printf("Download failed: %s\n", strerror(-res));
|
|
|
|
} else {
|
|
|
|
printf("Saved as %s\n", dst);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
2022-05-31 23:27:27 +02:00
|
|
|
|
2022-04-18 03:10:09 +02:00
|
|
|
static int _nanocoap_put_handler(int argc, char **argv)
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
char *file, *url;
|
|
|
|
char buffer[CONFIG_NANOCOAP_URI_MAX];
|
|
|
|
char work_buf[coap_szx2size(CONFIG_NANOCOAP_BLOCKSIZE_DEFAULT) + 1];
|
|
|
|
|
|
|
|
if (argc < 3) {
|
|
|
|
printf("Usage: %s <file> <url>\n", argv[0]);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
file = argv[1];
|
|
|
|
url = argv[2];
|
|
|
|
|
|
|
|
if (_is_dir(url)) {
|
|
|
|
const char *basename = strrchr(file, '/');
|
|
|
|
if (basename == NULL) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
if (snprintf(buffer, sizeof(buffer), "%s%s",
|
|
|
|
url, basename + 1) >= (int)sizeof(buffer)) {
|
|
|
|
puts("Constructed URI too long");
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
url = buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
res = nanocoap_vfs_put_url(url, file, work_buf, sizeof(work_buf));
|
|
|
|
if (res < 0) {
|
|
|
|
printf("Upload failed: %s\n", strerror(-res));
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
printf("Saved to %s\n", url);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2022-05-31 23:27:27 +02:00
|
|
|
SHELL_COMMAND(ncget, "download a file from a CoAP server", _nanocoap_get_handler);
|
2022-04-18 03:10:09 +02:00
|
|
|
SHELL_COMMAND(ncput, "upload a file to a CoAP server", _nanocoap_put_handler);
|