master-service-settings.c revision b7fa1f01c411634198e4341bb937b5eb52326efb
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen/* Copyright (c) 2005-2009 Dovecot authors, see the included COPYING file */
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen#define DOVECOT_CONFIG_BIN_PATH BINDIR"/doveconf"
c6260b961ca7b469a75ab842eeb53d3f904090e3Timo Sirainen#define CONFIG_HANDSHAKE "VERSION\tconfig\t1\t0\n"
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen { type, #name, offsetof(struct master_service_settings, name), NULL }
7744586e3e0fd60158abfbb03a233d3bd8d6c48bTimo Sirainenstatic const struct setting_define master_service_setting_defines[] = {
7744586e3e0fd60158abfbb03a233d3bd8d6c48bTimo Sirainenstatic const struct master_service_settings master_service_default_settings = {
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen MEMBER(log_timestamp) DEFAULT_FAILURE_STAMP_FORMAT,
7744586e3e0fd60158abfbb03a233d3bd8d6c48bTimo Sirainenconst struct setting_parser_info master_service_setting_parser_info = {
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen MEMBER(defines) master_service_setting_defines,
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen MEMBER(defaults) &master_service_default_settings,
383d0e8c24451468d6bea17e4b55d74de744abe6Timo Sirainen MEMBER(struct_size) sizeof(struct master_service_settings),
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainenmaster_service_exec_config(struct master_service *service, bool preserve_home)
d41ebbd7ae8cfe36a570c644dd7c9c55073fbb20Timo Sirainen const char **conf_argv, *path, *const *paths, *binary_path;
8bf9eec2783abd32643578460cc6ba4ab0a647f9Timo Sirainen /* already have the path */
8bf9eec2783abd32643578460cc6ba4ab0a647f9Timo Sirainen } else if (strchr(service->argv[0], '/') != NULL) {
8bf9eec2783abd32643578460cc6ba4ab0a647f9Timo Sirainen /* relative to current directory */
8bf9eec2783abd32643578460cc6ba4ab0a647f9Timo Sirainen if (realpath(service->argv[0], full_path) == NULL)
8bf9eec2783abd32643578460cc6ba4ab0a647f9Timo Sirainen i_fatal("realpath(%s) failed: %m", service->argv[0]);
d41ebbd7ae8cfe36a570c644dd7c9c55073fbb20Timo Sirainen /* we have to find our executable from path */
d41ebbd7ae8cfe36a570c644dd7c9c55073fbb20Timo Sirainen path = t_strconcat(*paths, "/", binary_path, NULL);
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen /* @UNSAFE */
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen conf_argv = t_new(const char *, 6 + (service->argc + 1) + 1);
4970e8dfcb18c1353e44b82e4ec836d165b7de82Timo Sirainen conf_argv[2] = t_strconcat("service=", service->name, NULL);
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen i_fatal("execv(%s) failed: %m", conf_argv[0]);
bad5fa318c6c1384ab83bd72d53ce06593274c18Timo Sirainenmaster_service_read_config(struct master_service *service, const char *path,
44aac2d461b4cb8e05e8c07f2f209372997a8719Timo Sirainen const struct master_service_settings_input *input,
d176f84ce5ca2073f4dfbafb457b9c74f6bf0d76Timo Sirainen const char **error_r)
21c1655dbc5fe861a152dc9a8a388d0d64f5ae20Timo Sirainen fd = net_connect_unix_with_retries(path, 1000);
b7fa1f01c411634198e4341bb937b5eb52326efbTimo Sirainen !S_ISSOCK(st.st_mode) && !S_ISFIFO(st.st_mode)) {
b7fa1f01c411634198e4341bb937b5eb52326efbTimo Sirainen /* it's a file, not a socket/pipe */
1358e2c58ce29231485a5cfa454756d429ad3d2cTimo Sirainen str_printfa(str, "\tmodule=%s", input->module);
1358e2c58ce29231485a5cfa454756d429ad3d2cTimo Sirainen str_printfa(str, "\tservice=%s", input->service);
44aac2d461b4cb8e05e8c07f2f209372997a8719Timo Sirainen str_printfa(str, "\tuser=%s", input->username);
44aac2d461b4cb8e05e8c07f2f209372997a8719Timo Sirainen ret = write_full(fd, str_data(str), str_len(str));
d176f84ce5ca2073f4dfbafb457b9c74f6bf0d76Timo Sirainen *error_r = t_strdup_printf("write_full(%s) failed: %m", path);
d39a04db2f4d0599cb9b5f03a9aa10a3c234453cTimo Sirainenmaster_service_apply_config_overrides(struct master_service *service,
d39a04db2f4d0599cb9b5f03a9aa10a3c234453cTimo Sirainen const char **error_r)
d39a04db2f4d0599cb9b5f03a9aa10a3c234453cTimo Sirainen const char *const *overrides;
d39a04db2f4d0599cb9b5f03a9aa10a3c234453cTimo Sirainen unsigned int i, count;
d39a04db2f4d0599cb9b5f03a9aa10a3c234453cTimo Sirainen overrides = array_get(&service->config_overrides, &count);
d39a04db2f4d0599cb9b5f03a9aa10a3c234453cTimo Sirainen for (i = 0; i < count; i++) {
d39a04db2f4d0599cb9b5f03a9aa10a3c234453cTimo Sirainen if (settings_parse_line(parser, overrides[i]) < 0) {
d39a04db2f4d0599cb9b5f03a9aa10a3c234453cTimo Sirainen settings_parse_set_key_expandeded(parser, service->set_pool,
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainenint master_service_settings_read(struct master_service *service,
44aac2d461b4cb8e05e8c07f2f209372997a8719Timo Sirainen const struct master_service_settings_input *input,
44aac2d461b4cb8e05e8c07f2f209372997a8719Timo Sirainen const char **error_r)
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen ARRAY_DEFINE(all_roots, const struct setting_parser_info *);
27ca0e561c70933da8834e57f967dee9b41896baTimo Sirainen const char *path = NULL, *error, *env, *const *keys;
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen unsigned int i;
047c00cd3f7f403672f81569413669238df8c15aTimo Sirainen (service->flags & MASTER_SERVICE_FLAG_NO_CONFIG_SETTINGS) == 0) {
bad5fa318c6c1384ab83bd72d53ce06593274c18Timo Sirainen path = input->config_path != NULL ? input->config_path :
bad5fa318c6c1384ab83bd72d53ce06593274c18Timo Sirainen fd = master_service_read_config(service, path, input, error_r);
e774b2f08b0909382ddc3d2869ae7e3b97205184Timo Sirainen pool_alloconly_create("master service settings", 8192);
44aac2d461b4cb8e05e8c07f2f209372997a8719Timo Sirainen settings_parser_info_update(service->set_pool,
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen p_array_init(&all_roots, service->set_pool, 8);
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen tmp_root = &master_service_setting_parser_info;
44aac2d461b4cb8e05e8c07f2f209372997a8719Timo Sirainen array_append(&all_roots, &input->roots[i], 1);
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen parser = settings_parser_init_list(service->set_pool,
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen array_idx(&all_roots, 0), array_count(&all_roots),
44aac2d461b4cb8e05e8c07f2f209372997a8719Timo Sirainen istream = i_stream_create_fd(fd, (size_t)-1, FALSE);
27ca0e561c70933da8834e57f967dee9b41896baTimo Sirainen ret = settings_parse_stream_read(parser, istream);
27ca0e561c70933da8834e57f967dee9b41896baTimo Sirainen /* most likely timed out, but just in case some other
27ca0e561c70933da8834e57f967dee9b41896baTimo Sirainen signal was delivered early check if we need to
13fc05b4fae8583f090901702031c4f0e8bf135dTimo Sirainen if ((service->flags & MASTER_SERVICE_FLAG_KEEP_CONFIG_OPEN) != 0)
047c00cd3f7f403672f81569413669238df8c15aTimo Sirainen if ((service->flags & MASTER_SERVICE_FLAG_NO_ENV_SETTINGS) == 0) {
047c00cd3f7f403672f81569413669238df8c15aTimo Sirainen /* let environment override settings. especially useful for the
047c00cd3f7f403672f81569413669238df8c15aTimo Sirainen settings from userdb. */
d39a04db2f4d0599cb9b5f03a9aa10a3c234453cTimo Sirainen if (array_is_created(&service->config_overrides)) {
d39a04db2f4d0599cb9b5f03a9aa10a3c234453cTimo Sirainen if (master_service_apply_config_overrides(service, parser,
8c4b2e0734326ef8e534f25e8ee78ace1fc22d15Timo Sirainen if (!settings_parser_check(parser, service->set_pool, &error)) {
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen *error_r = t_strdup_printf("Invalid settings: %s", error);
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen (service->flags & MASTER_SERVICE_FLAG_STANDALONE) != 0) {
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen /* running standalone. we want to ignore plugin versions. */
86791365b10f45982c88e70f2eb94fd6c3fea151Timo Sirainen master_service_set_die_with_master(master_service, TRUE);
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen /* if we change any settings afterwards, they're in expanded form.
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen especially all settings from userdb are already expanded. */
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen settings_parse_set_expanded(service->set_parser, TRUE);
44aac2d461b4cb8e05e8c07f2f209372997a8719Timo Sirainenint master_service_settings_read_simple(struct master_service *service,
44aac2d461b4cb8e05e8c07f2f209372997a8719Timo Sirainen const char **error_r)
44aac2d461b4cb8e05e8c07f2f209372997a8719Timo Sirainen return master_service_settings_read(service, &input, error_r);
bad5fa318c6c1384ab83bd72d53ce06593274c18Timo Sirainenpool_t master_service_settings_detach(struct master_service *service)
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainenmaster_service_settings_get(struct master_service *service)
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen sets = settings_parser_get_list(service->set_parser);
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainenvoid **master_service_settings_get_others(struct master_service *service)
8d3278a82b964217d95c340ec6f82037cdc59d19Timo Sirainen return settings_parser_get_list(service->set_parser) + 1;
2eb2cf8eeb763bd5ca9b6848dce32f0303e88ec1Timo Sirainenint master_service_set(struct master_service *service, const char *line)