1
0
mirror of https://github.com/falk-werner/webfused synced 2024-09-28 20:50:44 +00:00
falk-werner_webfused/src/webfused/config/factory.c

386 lines
9.5 KiB
C
Raw Normal View History

2020-03-08 15:14:55 +00:00
#include "webfused/config/factory.h"
#include "webfused/config/config_intern.h"
#include "webfused/config/config.h"
2020-03-18 09:17:17 +00:00
#include "webfused/config/settings_intern.h"
2020-03-08 19:29:08 +00:00
#include "webfused/log/log.h"
2020-03-08 15:14:55 +00:00
#include <libconfig.h>
#include <stdlib.h>
#include <stdbool.h>
2020-11-11 21:24:03 +00:00
#if ((LIBCONFIG_VER_MAJOR != 1) || (LIBCONFIG_VER_MINOR < 5))
2020-11-11 20:56:11 +00:00
#error "libconfig 1.5 or higher needed"
2020-03-08 15:14:55 +00:00
#endif
#define WFD_CONFIG_VERSION_MAJOR 1
#define WFD_CONFIG_VERSION_MINOR 0
static bool
wfd_config_check_version(
config_t * config)
{
int version_major;
int rc = config_lookup_int(config, "version.major", &version_major);
if (CONFIG_TRUE != rc)
{
2020-03-08 19:29:08 +00:00
WFD_ERROR("failed to load config: missing version.major");
2020-03-08 15:14:55 +00:00
return false;
}
if (WFD_CONFIG_VERSION_MAJOR != version_major)
{
2020-11-11 20:56:11 +00:00
WFD_ERROR("failed to load config: "
2020-03-08 19:29:08 +00:00
"incompatible versions: expected %d, but war %d",
WFD_CONFIG_VERSION_MAJOR, version_major);
2020-03-08 15:14:55 +00:00
return false;
}
int version_minor;
rc = config_lookup_int(config, "version.minor", &version_minor);
if (CONFIG_TRUE != rc)
{
2020-03-08 19:29:08 +00:00
WFD_ERROR("failed to load config: missing version.minor");
2020-03-08 15:14:55 +00:00
return false;
}
if (WFD_CONFIG_VERSION_MINOR < version_minor)
{
2020-03-08 19:29:08 +00:00
WFD_WARN("newer config detected: some features might be disabled");
2020-03-08 15:14:55 +00:00
}
else if (WFD_CONFIG_VERSION_MINOR > version_minor)
{
2020-03-08 19:29:08 +00:00
WFD_INFO("old config detected: some features might use default values");
2020-03-08 15:14:55 +00:00
}
return true;
}
2020-03-18 16:33:31 +00:00
static bool
wfd_config_read_logger(
config_t * config,
struct wfd_config * builder)
2020-03-18 16:33:31 +00:00
{
bool result = true;
bool hasLogger = (NULL != config_lookup(config, "log"));
if (hasLogger)
{
char const * provider;
int rc = config_lookup_string(config, "log.provider", &provider);
if (CONFIG_TRUE != rc)
{
2020-03-20 09:26:41 +00:00
WFD_ERROR("failed to load config: missing log provider");
2020-03-18 16:33:31 +00:00
result = false;
}
char const * level_str;
if (result)
{
rc = config_lookup_string(config, "log.level", &level_str);
if (CONFIG_TRUE != rc)
{
2020-03-20 09:26:41 +00:00
WFD_ERROR("failed to load config: missing log level");
2020-03-18 16:33:31 +00:00
result = false;
}
}
int level;
if (result)
{
bool success = wfd_log_level_parse(level_str, &level);
if (!success)
{
2020-03-20 09:26:41 +00:00
WFD_ERROR("failed to parse log level: unknown value \'%s\'", level_str);
2020-03-18 16:33:31 +00:00
result = false;
}
}
if (result)
{
config_setting_t * setting = config_lookup(config, "log.settings");
struct wfd_settings settings;
wfd_settings_init(&settings, setting);
result = wfd_config_set_logger(builder, provider, level, &settings);
2020-03-18 16:33:31 +00:00
wfd_settings_cleanup(&settings);
}
}
return result;
}
2020-03-20 09:26:41 +00:00
static void
2020-03-09 22:59:36 +00:00
wfd_config_read_server(
config_t * config,
struct wfd_config * builder)
2020-03-09 22:59:36 +00:00
{
char const * vhost_name;
int rc = config_lookup_string(config, "server.vhost_name", &vhost_name);
if (CONFIG_TRUE == rc)
{
wfd_config_set_server_vhostname(builder, vhost_name);
2020-03-09 22:59:36 +00:00
}
int port;
rc = config_lookup_int(config, "server.port", &port);
if (CONFIG_TRUE == rc)
{
wfd_config_set_server_port(builder, port);
2020-03-09 22:59:36 +00:00
}
char const * cert;
rc = config_lookup_string(config, "server.tls.certificate", &cert);
if (CONFIG_TRUE == rc)
{
wfd_config_set_server_cert(builder, cert);
2020-03-09 22:59:36 +00:00
}
char const * key;
rc = config_lookup_string(config, "server.tls.key", &key);
if (CONFIG_TRUE == rc)
{
wfd_config_set_server_key(builder, key);
2020-03-09 22:59:36 +00:00
}
char const * doc_root;
rc = config_lookup_string(config, "server.document_root", &doc_root);
if (CONFIG_TRUE == rc)
{
wfd_config_set_server_document_root(builder, doc_root);
2020-03-09 22:59:36 +00:00
}
}
2020-03-16 20:50:31 +00:00
static bool
wfd_config_read_authenticator(
config_setting_t * authenticator,
struct wfd_config * builder)
2020-03-16 20:50:31 +00:00
{
bool result = (NULL != authenticator);
2020-03-20 09:26:41 +00:00
if (!result)
{
WFD_ERROR("failed to load config: invalid authentication section");
}
2020-03-16 20:50:31 +00:00
char const * provider_name = NULL;
2020-11-11 20:56:11 +00:00
if (result)
2020-03-16 20:50:31 +00:00
{
int rc = config_setting_lookup_string(authenticator, "provider", &provider_name);
if (CONFIG_TRUE != rc)
2020-03-16 20:50:31 +00:00
{
2020-03-20 09:26:41 +00:00
WFD_ERROR("failed to load config: missing authentication provider");
result = false;
2020-03-16 20:50:31 +00:00
}
}
2020-03-16 20:50:31 +00:00
struct config_setting_t * settings = NULL;
if (result)
{
settings = config_setting_lookup(authenticator, "settings");
if (NULL == settings)
2020-03-16 20:50:31 +00:00
{
2020-03-20 09:26:41 +00:00
WFD_ERROR("failed to load config: missing authentication settings");
result = false;
2020-03-16 20:50:31 +00:00
}
}
2020-03-16 20:50:31 +00:00
if (result)
{
struct wfd_settings auth_settings;
wfd_settings_init(&auth_settings, settings);
result = wfd_config_add_auth_provider(builder, provider_name, &auth_settings);
wfd_settings_cleanup(&auth_settings);
}
2020-03-16 20:50:31 +00:00
return result;
}
static bool
wfd_config_read_authentication(
config_t * config,
struct wfd_config * builder)
{
bool result = true;
config_setting_t * authentication = config_lookup(config, "authentication");
if (NULL != authentication)
{
int length = config_setting_length(authentication);
for (int i = 0; (result) && (i < length); i++)
{
config_setting_t * authenticator = config_setting_get_elem(authentication, i);
result = wfd_config_read_authenticator(authenticator, builder);
2020-03-16 20:50:31 +00:00
}
}
2020-11-11 20:56:11 +00:00
2020-03-16 20:50:31 +00:00
return result;
}
2020-03-17 20:51:04 +00:00
static bool
wfd_config_read_filesystems(
config_t * config,
struct wfd_config * builder)
2020-03-17 20:51:04 +00:00
{
bool result = true;
config_setting_t * filesystems = config_lookup(config, "filesystems");
if (NULL != filesystems)
{
int length = config_setting_length(filesystems);
for (int i = 0; i < length; i++)
{
config_setting_t * fs = config_setting_get_elem(filesystems, i);
if (NULL == fs)
{
WFD_ERROR("failed to load config: invalid filesystem section");
result = false;
break;
}
char const * name;
int rc = config_setting_lookup_string(fs, "name", &name);
if (rc != CONFIG_TRUE)
{
WFD_ERROR("failed to load config: missing required filesystem property \'name\'");
result = false;
break;
}
char const * mount_point;
rc = config_setting_lookup_string(fs, "mount_point", &mount_point);
if (rc != CONFIG_TRUE)
{
WFD_ERROR("failed to load config: missing required filesystem property \'mount_point\'");
result = false;
break;
}
result = wfd_config_add_filesystem(builder, name, mount_point);
2020-03-17 20:51:04 +00:00
if (!result)
{
break;
}
}
}
return result;
}
static bool
wfd_config_read_user(
config_t * config,
struct wfd_config * builder)
{
bool result = true;
2020-11-11 20:56:11 +00:00
bool has_user = (NULL != config_lookup(config, "user"));
if (has_user)
{
char const * user;
{
int rc = config_lookup_string(config, "user.name", &user);
if (CONFIG_TRUE != rc)
{
WFD_ERROR("failed to load config: missing required user propert: \'name\'");
result = false;
}
}
char const * group;
if (result)
{
int rc = config_lookup_string(config, "user.group", &group);
if (CONFIG_TRUE != rc)
{
WFD_ERROR("failed to load config: missing required user propert: \'group\'");
result = false;
}
}
if (result)
{
wfd_config_set_user(builder, user, group);
}
}
return result;
}
static struct wfd_config *
2020-03-15 20:48:45 +00:00
wfd_config_load(
config_t * config)
2020-03-08 15:14:55 +00:00
{
struct wfd_config * result = wfd_config_create();
bool success = wfd_config_check_version(config)
&& wfd_config_read_logger(config, result)
&& wfd_config_read_authentication(config, result)
&& wfd_config_read_filesystems(config, result)
&& wfd_config_read_user(config, result)
2020-03-09 22:59:36 +00:00
;
2020-11-11 20:56:11 +00:00
2020-03-20 09:26:41 +00:00
if (success)
{
wfd_config_read_server(config, result);
}
2020-11-11 20:56:11 +00:00
if (!success)
{
wfd_config_dispose(result);
result = NULL;
}
2020-03-08 15:14:55 +00:00
return result;
}
struct wfd_config *
2020-03-08 15:14:55 +00:00
wfd_config_load_file(
char const * filename)
{
struct wfd_config * result = NULL;
2020-03-08 15:14:55 +00:00
config_t config;
config_init(&config);
int rc = config_read_file(&config, filename);
if (CONFIG_TRUE == rc)
{
result = wfd_config_load(&config);
2020-03-08 15:14:55 +00:00
}
2020-03-09 22:25:47 +00:00
else
{
2020-11-11 20:56:11 +00:00
WFD_ERROR("failed to load config: %s: %d: %s",
2020-03-09 22:25:47 +00:00
config_error_file(&config),
config_error_line(&config),
config_error_text(&config));
}
config_destroy(&config);
2020-11-11 20:56:11 +00:00
2020-03-08 15:14:55 +00:00
return result;
}
struct wfd_config *
2020-03-08 15:14:55 +00:00
wfd_config_load_string(
char const * contents)
{
struct wfd_config * result = NULL;
2020-03-08 15:14:55 +00:00
config_t config;
config_init(&config);
int rc = config_read_string(&config, contents);
if (CONFIG_TRUE == rc)
{
result = wfd_config_load(&config);
2020-03-08 15:14:55 +00:00
}
2020-03-09 22:25:47 +00:00
else
{
2020-11-11 20:56:11 +00:00
WFD_ERROR("failed to load config: %d: %s",
2020-03-09 22:25:47 +00:00
config_error_line(&config),
config_error_text(&config));
}
config_destroy(&config);
2020-11-11 20:56:11 +00:00
2020-03-08 15:14:55 +00:00
return result;
}