/* Copyright (c) 2010-2018 Dovecot authors, see the included COPYING file */
#include "lib.h"
#include "array.h"
#include "var-expand.h"
#include "settings-parser.h"
#include "service-settings.h"
#include "mail-storage-settings.h"
#include "master-service.h"
#include "master-service-settings.h"
#include "master-service-ssl-settings.h"
#include "iostream-ssl.h"
#include "doveadm-settings.h"
ARRAY_TYPE(doveadm_setting_root) doveadm_setting_roots;
bool doveadm_verbose_proctitle;
static pool_t doveadm_settings_pool = NULL;
static bool doveadm_settings_check(void *_set, pool_t pool, const char **error_r);
/* */
static struct file_listener_settings doveadm_unix_listeners_array[] = {
{ "doveadm-server", 0600, "", "" }
};
static struct file_listener_settings *doveadm_unix_listeners[] = {
&doveadm_unix_listeners_array[0]
};
static buffer_t doveadm_unix_listeners_buf = {
{ { doveadm_unix_listeners, sizeof(doveadm_unix_listeners) } }
};
/* */
struct service_settings doveadm_service_settings = {
.name = "doveadm",
.protocol = "",
.type = "",
.executable = "doveadm-server",
.user = "",
.group = "",
.privileged_group = "",
.extra_groups = "$default_internal_group",
.chroot = "",
.drop_priv_before_exec = FALSE,
.process_min_avail = 0,
.process_limit = 0,
.client_limit = 1,
.service_count = 1,
.idle_kill = 0,
.vsz_limit = UOFF_T_MAX,
.unix_listeners = { { &doveadm_unix_listeners_buf,
sizeof(doveadm_unix_listeners[0]) } },
.fifo_listeners = ARRAY_INIT,
.inet_listeners = ARRAY_INIT
};
#undef DEF
#define DEF(type, name) \
SETTING_DEFINE_STRUCT_##type(#name, name, struct doveadm_settings)
static const struct setting_define doveadm_setting_defines[] = {
DEF(STR, base_dir),
DEF(STR, libexec_dir),
DEF(STR, mail_plugins),
DEF(STR, mail_plugin_dir),
DEF(STR_VARS, mail_temp_dir),
DEF(BOOL, auth_debug),
DEF(STR, auth_socket_path),
DEF(STR, doveadm_socket_path),
DEF(UINT, doveadm_worker_count),
DEF(IN_PORT, doveadm_port),
{ .type = SET_ALIAS, .key = "doveadm_proxy_port" },
DEF(ENUM, doveadm_ssl),
DEF(STR, doveadm_username),
DEF(STR, doveadm_password),
DEF(STR, doveadm_allowed_commands),
DEF(STR, dsync_alt_char),
DEF(STR, dsync_remote_cmd),
DEF(STR, director_username_hash),
DEF(STR, doveadm_api_key),
DEF(STR, dsync_features),
DEF(UINT, dsync_commit_msgs_interval),
DEF(STR, doveadm_http_rawlog_dir),
DEF(STR, dsync_hashed_headers),
{ .type = SET_STRLIST, .key = "plugin",
.offset = offsetof(struct doveadm_settings, plugin_envs) },
SETTING_DEFINE_LIST_END
};
const struct doveadm_settings doveadm_default_settings = {
.base_dir = PKG_RUNDIR,
.libexec_dir = PKG_LIBEXECDIR,
.mail_plugins = "",
.mail_plugin_dir = MODULEDIR,
.mail_temp_dir = "/tmp",
.auth_debug = FALSE,
.auth_socket_path = "auth-userdb",
.doveadm_socket_path = "doveadm-server",
.doveadm_worker_count = 0,
.doveadm_port = 0,
.doveadm_ssl = "no:ssl:starttls",
.doveadm_username = "doveadm",
.doveadm_password = "",
.doveadm_allowed_commands = "",
.dsync_alt_char = "_",
.dsync_remote_cmd = "ssh -l%{login} %{host} doveadm dsync-server -u%u -U",
.dsync_features = "",
.dsync_hashed_headers = "Date Message-ID",
.dsync_commit_msgs_interval = 100,
.director_username_hash = "%Lu",
.doveadm_api_key = "",
.doveadm_http_rawlog_dir = "",
.plugin_envs = ARRAY_INIT
};
static const struct setting_parser_info *doveadm_setting_dependencies[] = {
&mail_user_setting_parser_info,
NULL
};
const struct setting_parser_info doveadm_setting_parser_info = {
.module_name = "doveadm",
.defines = doveadm_setting_defines,
.defaults = &doveadm_default_settings,
.type_offset = SIZE_MAX,
.struct_size = sizeof(struct doveadm_settings),
.parent_offset = SIZE_MAX,
.check_func = doveadm_settings_check,
.dependencies = doveadm_setting_dependencies
};
struct doveadm_settings *doveadm_settings;
const struct master_service_settings *service_set;
static void
fix_base_path(struct doveadm_settings *set, pool_t pool, const char **str)
{
if (*str != NULL && **str != '\0' && **str != '/')
*str = p_strconcat(pool, set->base_dir, "/", *str, NULL);
}
/* */
struct dsync_feature_list {
const char *name;
enum dsync_features num;
};
static const struct dsync_feature_list dsync_feature_list[] = {
{ "empty-header-workaround", DSYNC_FEATURE_EMPTY_HDR_WORKAROUND },
{ "no-header-hashes", DSYNC_FEATURE_NO_HEADER_HASHES },
{ NULL, 0 }
};
static int
dsync_settings_parse_features(struct doveadm_settings *set,
const char **error_r)
{
enum dsync_features features = 0;
const struct dsync_feature_list *list;
const char *const *str;
str = t_strsplit_spaces(set->dsync_features, " ,");
for (; *str != NULL; str++) {
list = dsync_feature_list;
for (; list->name != NULL; list++) {
if (strcasecmp(*str, list->name) == 0) {
features |= list->num;
break;
}
}
if (list->name == NULL) {
*error_r = t_strdup_printf("dsync_features: "
"Unknown feature: %s", *str);
return -1;
}
}
set->parsed_features = features;
return 0;
}
static bool doveadm_settings_check(void *_set, pool_t pool ATTR_UNUSED,
const char **error_r)
{
struct doveadm_settings *set = _set;
#ifndef CONFIG_BINARY
fix_base_path(set, pool, &set->auth_socket_path);
fix_base_path(set, pool, &set->doveadm_socket_path);
#endif
if (*set->dsync_hashed_headers == '\0') {
*error_r = "dsync_hashed_headers must not be empty";
return FALSE;
}
if (*set->dsync_alt_char == '\0') {
*error_r = "dsync_alt_char must not be empty";
return FALSE;
}
if (dsync_settings_parse_features(set, error_r) != 0)
return FALSE;
return TRUE;
}
/* */
const struct master_service_ssl_settings *doveadm_ssl_set = NULL;
void doveadm_get_ssl_settings(struct ssl_iostream_settings *set_r, pool_t pool)
{
master_service_ssl_client_settings_to_iostream_set(doveadm_ssl_set,
pool, set_r);
}
void doveadm_settings_expand(struct doveadm_settings *set, pool_t pool)
{
struct var_expand_table tab[] = { { '\0', NULL, NULL } };
const char *error;
if (settings_var_expand(&doveadm_setting_parser_info, set,
pool, tab, &error) <= 0)
i_fatal("Failed to expand settings: %s", error);
}
void doveadm_read_settings(void)
{
static const struct setting_parser_info *default_set_roots[] = {
&master_service_ssl_setting_parser_info,
&doveadm_setting_parser_info,
};
struct master_service_settings_input input;
struct master_service_settings_output output;
const struct doveadm_settings *set;
struct doveadm_setting_root *root;
ARRAY(const struct setting_parser_info *) set_roots;
ARRAY_TYPE(const_string) module_names;
void **sets;
const char *error;
t_array_init(&set_roots, N_ELEMENTS(default_set_roots) +
array_count(&doveadm_setting_roots) + 1);
array_append(&set_roots, default_set_roots,
N_ELEMENTS(default_set_roots));
t_array_init(&module_names, 4);
array_foreach_modifiable(&doveadm_setting_roots, root) {
array_push_back(&module_names, &root->info->module_name);
array_push_back(&set_roots, &root->info);
}
array_append_zero(&module_names);
array_append_zero(&set_roots);
i_zero(&input);
input.roots = array_front(&set_roots);
input.module = "doveadm";
input.extra_modules = array_front(&module_names);
input.service = "doveadm";
input.preserve_user = TRUE;
input.preserve_home = TRUE;
if (master_service_settings_read(master_service, &input,
&output, &error) < 0)
i_fatal("Error reading configuration: %s", error);
doveadm_settings_pool = pool_alloconly_create("doveadm settings", 1024);
service_set = master_service_settings_get(master_service);
service_set = settings_dup(&master_service_setting_parser_info,
service_set, doveadm_settings_pool);
doveadm_verbose_proctitle = service_set->verbose_proctitle;
sets = master_service_settings_get_others(master_service);
set = sets[1];
doveadm_settings = settings_dup(&doveadm_setting_parser_info, set,
doveadm_settings_pool);
doveadm_ssl_set = settings_dup(&master_service_ssl_setting_parser_info,
master_service_ssl_settings_get(master_service),
doveadm_settings_pool);
doveadm_settings_expand(doveadm_settings, doveadm_settings_pool);
doveadm_settings->parsed_features = set->parsed_features; /* copy this value by hand */
array_foreach_modifiable(&doveadm_setting_roots, root) {
unsigned int idx =
array_foreach_idx(&doveadm_setting_roots, root);
root->settings = settings_dup(root->info, sets[2+idx],
doveadm_settings_pool);
}
}
void doveadm_setting_roots_add(const struct setting_parser_info *info)
{
struct doveadm_setting_root *root;
root = array_append_space(&doveadm_setting_roots);
root->info = info;
}
void *doveadm_setting_roots_get_settings(const struct setting_parser_info *info)
{
const struct doveadm_setting_root *root;
array_foreach(&doveadm_setting_roots, root) {
if (root->info == info)
return root->settings;
}
i_panic("Failed to find settings for module %s", info->module_name);
}
void doveadm_settings_init(void)
{
i_array_init(&doveadm_setting_roots, 8);
}
void doveadm_settings_deinit(void)
{
array_free(&doveadm_setting_roots);
pool_unref(&doveadm_settings_pool);
}