2022-04-12 22:34:42 +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 net_nanocoap
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
2023-06-08 16:29:12 +02:00
|
|
|
* @brief nanoCoAP VFS helpers
|
2022-04-12 22:34:42 +02:00
|
|
|
*
|
|
|
|
* @author Benjamin Valentin <benjamin.valentin@ml-pa.com>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <fcntl.h>
|
2022-04-18 03:01:35 +02:00
|
|
|
#include "net/nanocoap_vfs.h"
|
2022-04-12 22:34:42 +02:00
|
|
|
#include "net/sock/util.h"
|
|
|
|
#include "vfs.h"
|
|
|
|
|
|
|
|
#define ENABLE_DEBUG 0
|
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
static int _2file(void *arg, size_t offset, uint8_t *buf, size_t len, int more)
|
|
|
|
{
|
|
|
|
(void)more;
|
|
|
|
int *fd = arg;
|
|
|
|
|
|
|
|
vfs_lseek(*fd, offset, SEEK_SET);
|
|
|
|
return vfs_write(*fd, buf, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _prepare_file(const char *dst, char *dst_tmp, size_t len)
|
|
|
|
{
|
|
|
|
/* download to temp file, rename it later */
|
|
|
|
if (snprintf(dst_tmp, len, "%s.t", dst) > (int)len) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
|
|
|
|
return vfs_open(dst_tmp, O_CREAT | O_WRONLY | O_TRUNC, 0644);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _finalize_file(int fd, int res, const char *dst, const char *dst_tmp)
|
|
|
|
{
|
|
|
|
vfs_close(fd);
|
|
|
|
|
|
|
|
/* move file to it's final location */
|
|
|
|
if (res >= 0) {
|
|
|
|
DEBUG("nanocoap: moving %s to %s\n", dst_tmp, dst);
|
|
|
|
vfs_unlink(dst);
|
|
|
|
res = vfs_rename(dst_tmp, dst);
|
|
|
|
}
|
|
|
|
|
|
|
|
vfs_unlink(dst_tmp);
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
int nanocoap_vfs_get(nanocoap_sock_t *sock, const char *path, const char *dst)
|
|
|
|
{
|
|
|
|
int fd, res;
|
|
|
|
char dst_tmp[CONFIG_SOCK_URLPATH_MAXLEN];
|
|
|
|
|
|
|
|
fd = _prepare_file(dst, dst_tmp, sizeof(dst_tmp));
|
|
|
|
if (fd < 0) {
|
|
|
|
return fd;
|
|
|
|
}
|
2022-06-21 18:18:39 +02:00
|
|
|
|
|
|
|
DEBUG("nanocoap: downloading %s to %s\n", path, dst_tmp);
|
2022-04-12 22:34:42 +02:00
|
|
|
res = nanocoap_sock_get_blockwise(sock, path, CONFIG_NANOCOAP_BLOCKSIZE_DEFAULT,
|
|
|
|
_2file, &fd);
|
|
|
|
return _finalize_file(fd, res, dst, dst_tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
int nanocoap_vfs_get_url(const char *url, const char *dst)
|
|
|
|
{
|
|
|
|
int fd, res;
|
|
|
|
char dst_tmp[CONFIG_SOCK_URLPATH_MAXLEN];
|
|
|
|
|
|
|
|
fd = _prepare_file(dst, dst_tmp, sizeof(dst_tmp));
|
|
|
|
if (fd < 0) {
|
|
|
|
return fd;
|
|
|
|
}
|
2022-06-21 18:18:39 +02:00
|
|
|
|
|
|
|
DEBUG("nanocoap: downloading %s to %s\n", url, dst_tmp);
|
2022-04-12 22:34:42 +02:00
|
|
|
res = nanocoap_get_blockwise_url(url, CONFIG_NANOCOAP_BLOCKSIZE_DEFAULT,
|
|
|
|
_2file, &fd);
|
|
|
|
return _finalize_file(fd, res, dst, dst_tmp);
|
|
|
|
}
|
2022-04-18 03:01:35 +02:00
|
|
|
|
|
|
|
static int _vfs_put(coap_block_request_t *ctx, const char *file, void *buffer)
|
|
|
|
{
|
|
|
|
int res, fd = vfs_open(file, O_RDONLY, 0644);
|
|
|
|
if (fd < 0) {
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* buffer is at least one larger than SZX value */
|
|
|
|
int buffer_len = coap_szx2size(ctx->blksize) + 1;
|
|
|
|
|
|
|
|
bool more = true;
|
|
|
|
while (more && (res = vfs_read(fd, buffer, buffer_len)) > 0) {
|
|
|
|
more = res == buffer_len;
|
|
|
|
res = nanocoap_sock_block_request(ctx, buffer,
|
|
|
|
res, more, NULL, NULL);
|
|
|
|
if (res < 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
vfs_lseek(fd, -1, SEEK_CUR);
|
|
|
|
}
|
|
|
|
|
|
|
|
vfs_close(fd);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
int nanocoap_vfs_put(nanocoap_sock_t *sock, const char *path, const char *src,
|
|
|
|
void *work_buf, size_t work_buf_len)
|
|
|
|
{
|
|
|
|
DEBUG("nanocoap: uploading %s to %s\n", src, path);
|
|
|
|
|
|
|
|
if (work_buf_len < coap_szx2size(0) + 1) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
|
|
|
|
coap_block_request_t ctx = {
|
|
|
|
.path = path,
|
|
|
|
.method = COAP_METHOD_PUT,
|
|
|
|
.blksize = coap_size2szx(work_buf_len - 1),
|
2022-11-01 00:02:31 +01:00
|
|
|
.sock = sock,
|
2022-04-18 03:01:35 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
return _vfs_put(&ctx, src, work_buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
int nanocoap_vfs_put_url(const char *url, const char *src,
|
|
|
|
void *work_buf, size_t work_buf_len)
|
|
|
|
{
|
|
|
|
DEBUG("nanocoap: uploading %s to %s\n", src, url);
|
|
|
|
|
|
|
|
if (work_buf_len < coap_szx2size(0) + 1) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
|
2022-11-01 00:02:31 +01:00
|
|
|
nanocoap_sock_t sock;
|
2022-04-18 03:01:35 +02:00
|
|
|
coap_block_request_t ctx;
|
2022-11-01 00:02:31 +01:00
|
|
|
int res = nanocoap_block_request_connect_url(&ctx, &sock, url, COAP_METHOD_PUT,
|
|
|
|
coap_size2szx(work_buf_len - 1));
|
2022-10-31 15:56:10 +01:00
|
|
|
if (res == 0) {
|
|
|
|
res = _vfs_put(&ctx, src, work_buf);
|
2022-11-01 00:02:31 +01:00
|
|
|
nanocoap_sock_close(&sock);
|
2022-04-18 03:01:35 +02:00
|
|
|
}
|
|
|
|
|
2022-10-31 15:56:10 +01:00
|
|
|
return res;
|
2022-04-18 03:01:35 +02:00
|
|
|
}
|