1
0
mirror of https://github.com/falk-werner/webfuse-provider synced 2024-10-27 20:44:10 +00:00
falk-werner_webfuse-provider/example/provider/main.c

381 lines
8.8 KiB
C
Raw Normal View History

2019-02-16 15:28:14 +00:00
#include <stdio.h>
#include <stdlib.h>
2019-02-17 13:31:04 +00:00
#include <string.h>
#include <signal.h>
#include <stdbool.h>
2019-02-16 15:28:14 +00:00
2019-03-03 15:29:02 +00:00
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <getopt.h>
2019-03-03 15:29:02 +00:00
2019-03-26 22:04:53 +00:00
#include "webfuse_provider.h"
2019-02-16 15:28:14 +00:00
struct config
{
char * url;
2019-03-26 22:04:53 +00:00
struct wfp_client_config * client_config;
bool show_help;
};
enum fs_entry_type
{
FS_FILE,
FS_DIR
};
struct fs_entry
{
ino_t parent;
ino_t inode;
char const * name;
int mode;
enum fs_entry_type type;
size_t content_length;
char const * content;
};
2019-02-17 13:31:04 +00:00
struct fs
{
struct fs_entry const * entries;
2019-02-17 13:31:04 +00:00
};
static void show_help()
{
printf(
2019-03-26 22:04:53 +00:00
"webfuse-provider, Copyright (c) 2019, webfuse authors <https://github.com/falk-werner/webfuse>\n"
"Example for websocket file system provider\n"
"\n"
2019-03-26 22:04:53 +00:00
"Usage: webfuse-provider -u <url> [-k <key_path>] [-c <cert_path>]\n"
"\n"
"Options:\n"
2019-03-26 22:04:53 +00:00
"\t-u, --url URL of webfuse server (required)\n"
"\t-k, --key_path Path to private key of provider (default: not set, TLS disabled)\n"
"\t-c, --cert_path Path to certificate of provider (defautl: not set, TLS disabled)\n"
"\t-h, --help prints this message\n"
"\n"
"Example:\n"
2019-03-26 22:04:53 +00:00
"\twebfuse-provider -u ws://localhost:8080/\n"
"\n"
);
}
static int parse_arguments(
int argc,
char* argv[],
struct config * config)
{
static struct option const options[] =
{
{"url", required_argument, NULL, 'u'},
{"key_path", required_argument, NULL, 'k'},
{"cert_path", required_argument, NULL, 'c'},
{"help", no_argument, NULL, 'h'},
{NULL, 0, NULL, 0}
};
int result = EXIT_SUCCESS;
bool finished = false;
while (!finished)
{
int option_index = 0;
int const c = getopt_long(argc, argv, "u:k:c:h", options, &option_index);
switch (c)
{
case -1:
finished = true;
break;
case 'h':
config->show_help = true;
finished = true;
break;
case 'u':
free(config->url);
config->url = strdup(optarg);
break;
case 'k':
2019-03-26 22:04:53 +00:00
wfp_client_config_set_keypath(config->client_config, optarg);
break;
case 'c':
2019-03-26 22:04:53 +00:00
wfp_client_config_set_certpath(config->client_config, optarg);
break;
default:
fprintf(stderr, "error: unknown argument\n");
finished = true;
result = EXIT_FAILURE;
break;
}
if (NULL == config->url)
{
fprintf(stderr, "error: missing required argument \"-u\"\n");
result = EXIT_FAILURE;
}
if (result != EXIT_SUCCESS)
{
config->show_help = true;
}
}
return result;
}
static struct fs_entry const * fs_getentry(
struct fs * fs,
ino_t inode)
{
for (size_t i = 0; 0 != fs->entries[i].inode; i++)
{
struct fs_entry const * entry = &fs->entries[i];
if (inode == entry->inode)
{
return entry;
}
}
return NULL;
}
static struct fs_entry const * fs_getentry_byname(
struct fs * fs,
ino_t parent,
char const * name)
{
for( size_t i = 0; 0 != fs->entries[i].inode; i++)
{
struct fs_entry const * entry = &fs->entries[i];
if ((parent == entry->parent) && (0 == strcmp(name, entry->name)))
{
return entry;
}
}
return NULL;
}
static void fs_stat(
struct fs_entry const * entry,
struct stat * stat)
{
memset(stat, 0, sizeof(struct stat));
stat->st_ino = entry->inode;
stat->st_mode = entry->mode;
if (FS_DIR == entry->type)
{
stat->st_mode |= S_IFDIR;
}
if (FS_FILE == entry->type)
{
stat->st_mode |= S_IFREG;
stat->st_size = entry->content_length;
}
}
2019-02-17 13:31:04 +00:00
static void fs_lookup(
2019-03-26 22:04:53 +00:00
struct wfp_request * request,
2019-02-17 13:31:04 +00:00
ino_t parent,
char const * name,
void * user_data)
{
struct fs * fs = (struct fs*) user_data;
struct fs_entry const * entry = fs_getentry_byname(fs, parent, name);
if (NULL != entry)
{
struct stat stat;
fs_stat(entry, &stat);
2019-03-26 22:04:53 +00:00
wfp_respond_lookup(request, &stat);
}
else
{
2019-03-26 22:04:53 +00:00
wfp_respond_error(request, WF_BAD_NOENTRY);
}
2019-02-17 13:31:04 +00:00
}
2019-02-17 13:31:04 +00:00
static void fs_getattr(
2019-03-26 22:04:53 +00:00
struct wfp_request * request,
2019-02-17 13:31:04 +00:00
ino_t inode,
void * user_data)
{
struct fs * fs = (struct fs*) user_data;
struct fs_entry const * entry = fs_getentry(fs, inode);
2019-03-03 12:34:43 +00:00
if (NULL != entry)
{
struct stat stat;
fs_stat(entry, &stat);
2019-03-26 22:04:53 +00:00
wfp_respond_getattr(request, &stat);
}
2019-03-03 12:34:43 +00:00
else
{
2019-03-26 22:04:53 +00:00
wfp_respond_error(request, WF_BAD_NOENTRY);
2019-03-03 12:34:43 +00:00
}
2019-02-17 13:31:04 +00:00
}
static void fs_readdir(
2019-03-26 22:04:53 +00:00
struct wfp_request * request,
2019-02-17 13:31:04 +00:00
ino_t directory,
void * user_data)
{
2019-03-03 12:34:43 +00:00
struct fs * fs = (struct fs*) user_data;
2019-02-17 13:31:04 +00:00
struct fs_entry const * dir = fs_getentry(fs, directory);
if ((NULL != dir) && (FS_DIR == dir->type))
2019-03-03 12:34:43 +00:00
{
2019-03-26 22:04:53 +00:00
struct wfp_dirbuffer * buffer = wfp_dirbuffer_create();
wfp_dirbuffer_add(buffer, ".", dir->inode);
wfp_dirbuffer_add(buffer, "..", dir->inode);
2019-03-03 12:34:43 +00:00
for(size_t i = 0; 0 != fs->entries[i].inode; i++)
2019-03-03 12:34:43 +00:00
{
struct fs_entry const * entry = &fs->entries[i];
if (directory == entry->parent)
2019-03-03 12:34:43 +00:00
{
2019-03-26 22:04:53 +00:00
wfp_dirbuffer_add(buffer, entry->name, entry->inode);
2019-03-03 12:34:43 +00:00
}
}
2019-03-26 22:04:53 +00:00
wfp_respond_readdir(request, buffer);
wfp_dirbuffer_dispose(buffer);
2019-03-03 12:34:43 +00:00
}
else
{
2019-03-26 22:04:53 +00:00
wfp_respond_error(request, WF_BAD_NOENTRY);
2019-03-03 12:34:43 +00:00
}
2019-02-17 13:31:04 +00:00
}
static void fs_open(
2019-03-26 22:04:53 +00:00
struct wfp_request * request,
2019-02-17 13:31:04 +00:00
ino_t inode,
int flags,
void * user_data)
{
2019-03-03 15:29:02 +00:00
struct fs * fs = (struct fs*) user_data;
2019-02-17 13:31:04 +00:00
2019-03-03 15:29:02 +00:00
struct fs_entry const * entry = fs_getentry(fs, inode);
if ((NULL != entry) && (FS_FILE == entry->type))
{
if (O_RDONLY == (flags & O_ACCMODE))
{
2019-03-26 22:04:53 +00:00
wfp_respond_open(request, 0U);
2019-03-03 15:29:02 +00:00
}
else
{
wfp_respond_error(request, WF_BAD_ACCESS_DENIED);
2019-03-03 15:29:02 +00:00
}
}
else
{
2019-03-26 22:04:53 +00:00
wfp_respond_error(request, WF_BAD_NOENTRY);
2019-03-03 15:29:02 +00:00
}
2019-02-17 13:31:04 +00:00
}
2019-03-03 17:02:30 +00:00
static size_t min(size_t const a, size_t const b)
{
return (a < b) ? a : b;
}
2019-02-17 13:31:04 +00:00
static void fs_read(
2019-03-26 22:04:53 +00:00
struct wfp_request * request,
2019-02-17 13:31:04 +00:00
ino_t inode,
uint32_t handle,
size_t offset,
size_t length,
void * user_data)
{
(void) handle;
2019-03-03 17:02:30 +00:00
struct fs * fs = (struct fs*) user_data;
struct fs_entry const * entry = fs_getentry(fs, inode);
if ((NULL != entry) && (FS_FILE == entry->type))
{
if (entry->content_length > offset)
{
size_t const remaining = entry->content_length - offset;
size_t const count = min(remaining, length);
2019-03-26 22:04:53 +00:00
wfp_respond_read(request, &entry->content[offset], count);
2019-03-03 17:02:30 +00:00
}
else
{
2019-03-26 22:04:53 +00:00
wfp_respond_error(request, WF_BAD);
2019-03-03 17:02:30 +00:00
}
}
else
{
2019-03-26 22:04:53 +00:00
wfp_respond_error(request, WF_BAD_NOENTRY);
2019-03-03 17:02:30 +00:00
}
2019-02-17 13:31:04 +00:00
}
2019-03-26 22:04:53 +00:00
static struct wfp_client * client;
2019-02-17 13:31:04 +00:00
static void on_interrupt(int signal_id)
{
(void) signal_id;
2019-03-26 22:04:53 +00:00
wfp_client_shutdown(client);
2019-02-17 13:31:04 +00:00
}
2019-02-16 15:28:14 +00:00
int main(int argc, char* argv[])
{
struct config config;
config.url = NULL;
config.show_help = false;
2019-03-26 22:04:53 +00:00
config.client_config = wfp_client_config_create();
int result = parse_arguments(argc, argv, &config);
2019-02-16 15:28:14 +00:00
if (EXIT_SUCCESS == result)
2019-02-17 13:31:04 +00:00
{
static struct fs_entry const entries[]=
2019-02-17 13:31:04 +00:00
{
{.parent = 0, .inode = 1, .name = "<root>", .mode = 0555, .type = FS_DIR},
{
.parent = 1,
.inode = 2,
.name = "hello.txt",
feat(webfuse): add multiclient support (#23) * fixes verbosity option when set through command line * adds support for build type and allows to run gdb in container * adds missing toolchain headers to project * renames container macros * adds gdbserver * fixes verbosity option when set through command line * adds support for build type and allows to run gdb in container * adds missing toolchain headers to project * renames container macros * adds gdbserver * removes language settings, which contains alternating values * adds wrapper script to launch gdbserver * fix docker command in wrapper script * fixes run in dind setup * replaces docker's init through dump-init * moves filesystem to session * fixes verbosity option when set through command line * adds support for build type and allows to run gdb in container * renames container macros * adds gdbserver * fixes verbosity option when set through command line * adds support for build type and allows to run gdb in container * renames container macros * adds gdbserver * adds wrapper script to launch gdbserver * fix docker command in wrapper script * fixes run in dind setup * replaces docker's init through dump-init * moves filesystem to session * adds container_of * added dlist * allows multiple clients to connect * removes directory when session is closed * adds dependecy to uuid-dev * allow clients to register filesystems * updates documentation * moves mountpoint handling into filesystem: mountpoints are removed during session cleanup * adds filesystem name/id to request parameters * fixes security issue: add_filesystem did not check name * removes default link, if it is broken * recreates symlink "default", if filesystem is gone * updates documentation * fixes memory leak * makes authentication work .. again * updates provider to support changed protocol * removes execute right of hello.txt * fixes style issues * fixes javascript style issues * fixes flase positive from Flawfinder * fixes some javascript style issues * removes use of PATH_MAX * removes use of GNU extensions in container_of implementation * ignores findings of flawfinder * replaces dlist by slist * removes duplicate implementation of slist (message_queue)
2019-04-17 20:51:16 +00:00
.mode = 0444,
.type = FS_FILE,
.content="hello, world!",
.content_length = 13,
},
{.parent = 0, .inode = 0, .name = NULL}
};
struct fs fs =
{
.entries = entries
};
2019-02-17 13:31:04 +00:00
signal(SIGINT, &on_interrupt);
2019-02-17 13:31:04 +00:00
2019-03-26 22:04:53 +00:00
wfp_client_config_set_userdata(config.client_config, &fs);
wfp_client_config_set_onlookup(config.client_config, &fs_lookup);
wfp_client_config_set_ongetattr(config.client_config, &fs_getattr);
wfp_client_config_set_onreaddir(config.client_config, &fs_readdir);
wfp_client_config_set_onopen(config.client_config, &fs_open);
wfp_client_config_set_onread(config.client_config, &fs_read);
2019-02-17 13:31:04 +00:00
2019-03-26 22:04:53 +00:00
client = wfp_client_create(config.client_config);
wfp_client_connect(client, config.url);
2019-02-17 13:31:04 +00:00
2019-03-26 22:04:53 +00:00
wfp_client_run(client);
2019-03-26 22:04:53 +00:00
wfp_client_dispose(client);
}
if (config.show_help)
{
show_help();
}
2019-02-17 13:31:04 +00:00
free(config.url);
2019-03-26 22:04:53 +00:00
wfp_client_config_dispose(config.client_config);
return result;
2019-02-16 15:28:14 +00:00
}