/* * 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 * * @} */ #include #include #include #include #include "net/nanocoap_sock.h" #include "net/nanocoap_vfs.h" #include "shell.h" #include "vfs_default.h" #include "vfs_util.h" /** * @brief Default download location for ncget */ #ifndef CONFIG_NCGET_DEFAULT_DATA_DIR #define CONFIG_NCGET_DEFAULT_DATA_DIR VFS_DEFAULT_DATA #endif 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); } static int _nanocoap_get_handler(int argc, char **argv) { int res; char buffer[CONFIG_NANOCOAP_URI_MAX]; char *dst, *url = argv[1]; if (argc < 2) { printf("Usage: %s [destination]\n", argv[0]); printf("Default destination: %s\n", CONFIG_NCGET_DEFAULT_DATA_DIR); 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", CONFIG_NCGET_DEFAULT_DATA_DIR, dst) >= (int)sizeof(buffer)) { printf("Output file path too long\n"); return -ENOBUFS; } dst = buffer; } else { char *filename = strrchr(url, '/'); dst = argv[2]; if (vfs_is_dir(dst) > 0 && filename) { if (snprintf(buffer, sizeof(buffer), "%s%s", dst, filename) >= (int)sizeof(buffer)) { printf("Output file path too long\n"); return -ENOBUFS; } dst = buffer; } } /* 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; } 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 \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; } SHELL_COMMAND(ncget, "download a file from a CoAP server", _nanocoap_get_handler); SHELL_COMMAND(ncput, "upload a file to a CoAP server", _nanocoap_put_handler);