1
0
mirror of https://github.com/falk-werner/webfuse-provider synced 2024-10-27 20:44:10 +00:00
falk-werner_webfuse-provider/lib/webfuse/adapter/impl/operation/readdir.c

161 lines
3.7 KiB
C
Raw Normal View History

#include "webfuse/adapter/impl/operation/readdir.h"
#include "webfuse/adapter/impl/operation/context.h"
2019-01-29 22:11:46 +00:00
2019-02-03 18:10:05 +00:00
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
2019-01-29 22:11:46 +00:00
#include "webfuse/core/jsonrpc/proxy.h"
2019-03-26 22:04:53 +00:00
#include "webfuse/core/util.h"
#include "webfuse/core/json_util.h"
2019-02-03 18:10:05 +00:00
2019-03-26 22:04:53 +00:00
#define WF_DIRBUFFER_INITIAL_SIZE 1024
2019-02-03 18:10:05 +00:00
2019-03-26 22:04:53 +00:00
struct wf_impl_dirbuffer
2019-02-03 18:10:05 +00:00
{
char * data;
size_t position;
size_t capacity;
};
2019-03-26 22:04:53 +00:00
static void wf_impl_dirbuffer_init(
struct wf_impl_dirbuffer * buffer)
2019-02-03 18:10:05 +00:00
{
2019-03-26 22:04:53 +00:00
buffer->data = malloc(WF_DIRBUFFER_INITIAL_SIZE);
2019-02-03 18:10:05 +00:00
buffer->position = 0;
2019-03-26 22:04:53 +00:00
buffer->capacity = WF_DIRBUFFER_INITIAL_SIZE;
2019-02-03 18:10:05 +00:00
}
2019-03-26 22:04:53 +00:00
static void wf_impl_dirbuffer_dispose(
struct wf_impl_dirbuffer * buffer)
2019-02-03 18:10:05 +00:00
{
free(buffer->data);
}
2019-03-26 22:04:53 +00:00
static void wf_impl_dirbuffer_add(
2019-02-03 18:10:05 +00:00
fuse_req_t request,
2019-03-26 22:04:53 +00:00
struct wf_impl_dirbuffer * buffer,
2019-02-03 18:10:05 +00:00
char const * name,
fuse_ino_t inode)
{
size_t const size = fuse_add_direntry(request, NULL, 0, name, NULL, 0);
size_t remaining = buffer->capacity - buffer->position;
while (remaining < size)
{
buffer->capacity *= 2;
buffer->data = realloc(buffer->data, buffer->capacity);
remaining = buffer->capacity - buffer->position;
}
2019-01-29 22:11:46 +00:00
2019-02-03 18:10:05 +00:00
struct stat stat_buffer;
memset(&stat_buffer, 0, sizeof(struct stat));
stat_buffer.st_ino = inode;
fuse_add_direntry(request,
&buffer->data[buffer->position], remaining, name,
&stat_buffer, buffer->position + size);
buffer->position += size;
}
2019-03-26 22:04:53 +00:00
static size_t wf_impl_min(size_t a, size_t b)
2019-02-03 18:10:05 +00:00
{
return (a < b) ? a : b;
}
void wf_impl_operation_readdir_finished(
2019-02-09 02:08:02 +00:00
void * user_data,
json_t const * result,
json_t const * error)
2019-01-29 22:11:46 +00:00
{
wf_status status = wf_impl_jsonrpc_get_status(error);
2019-03-26 22:04:53 +00:00
struct wf_impl_operation_readdir_context * context = user_data;
2019-02-03 18:10:05 +00:00
2019-03-26 22:04:53 +00:00
struct wf_impl_dirbuffer buffer;
wf_impl_dirbuffer_init(&buffer);
2019-01-29 22:11:46 +00:00
if (json_is_array(result))
2019-01-29 22:11:46 +00:00
{
size_t const count = json_array_size(result);
for(size_t i = 0; i < count; i++)
2019-01-29 22:11:46 +00:00
{
json_t * entry =json_array_get(result, i);
if (json_is_object(entry))
2019-01-29 22:11:46 +00:00
{
json_t * name_holder = json_object_get(entry, "name");
json_t * inode_holder = json_object_get(entry, "inode");
if ((json_is_string(name_holder)) && (json_is_integer(inode_holder)))
{
char const * name = json_string_value(name_holder);
fuse_ino_t entry_inode = (fuse_ino_t) json_integer_value(inode_holder);
wf_impl_dirbuffer_add(context->request, &buffer, name, entry_inode);
}
else
2019-01-29 22:11:46 +00:00
{
status = WF_BAD_FORMAT;
break;
2019-01-29 22:11:46 +00:00
}
}
else
{
status = WF_BAD_FORMAT;
break;
}
2019-01-29 22:11:46 +00:00
}
}
else if (WF_GOOD == status)
{
status = WF_BAD_FORMAT;
}
2019-02-03 18:10:05 +00:00
2019-03-26 22:04:53 +00:00
if (WF_GOOD == status)
2019-02-03 18:10:05 +00:00
{
2019-02-09 02:08:02 +00:00
if (((size_t) context->offset) < buffer.position)
2019-02-03 18:10:05 +00:00
{
2019-02-09 02:08:02 +00:00
fuse_reply_buf(context->request, &buffer.data[context->offset],
2019-03-26 22:04:53 +00:00
wf_impl_min(buffer.position - context->offset, context->size));
2019-02-03 18:10:05 +00:00
}
else
{
2019-02-09 02:08:02 +00:00
fuse_reply_buf(context->request, NULL, 0);
2019-02-03 18:10:05 +00:00
}
}
else
{
2019-02-09 02:08:02 +00:00
fuse_reply_err(context->request, ENOENT);
2019-02-03 18:10:05 +00:00
}
2019-03-26 22:04:53 +00:00
wf_impl_dirbuffer_dispose(&buffer);
2019-02-09 02:08:02 +00:00
free(context);
}
2019-03-26 22:04:53 +00:00
void wf_impl_operation_readdir (
2019-02-09 02:08:02 +00:00
fuse_req_t request,
fuse_ino_t inode,
size_t size,
off_t offset,
2019-03-26 22:04:53 +00:00
struct fuse_file_info * WF_UNUSED_PARAM(file_info))
2019-02-09 02:08:02 +00:00
{
2020-04-04 21:27:34 +00:00
struct wf_impl_operation_context * user_data = fuse_req_userdata(request);
struct wf_jsonrpc_proxy * rpc = wf_impl_operation_context_get_proxy(user_data);
if (NULL != rpc)
{
2020-03-01 15:55:58 +00:00
struct wf_impl_operation_readdir_context * readdir_context = malloc(sizeof(struct wf_impl_operation_readdir_context));
readdir_context->request = request;
readdir_context->size = size;
readdir_context->offset = offset;
2019-02-09 02:08:02 +00:00
2020-03-01 15:55:58 +00:00
wf_jsonrpc_proxy_invoke(rpc, &wf_impl_operation_readdir_finished, readdir_context, "readdir", "si", user_data->name, inode);
}
else
{
fuse_reply_err(request, ENOENT);
}
2019-02-03 18:10:05 +00:00
}