doveconf.c revision 33bd898e7756b289e65f43133312d9637afc1371
/* Copyright (c) 2005-2014 Dovecot authors, see the included COPYING file */
#include "lib.h"
#include "array.h"
#include "abspath.h"
#include "module-dir.h"
#include "env-util.h"
#include "guid.h"
#include "hash.h"
#include "hostpid.h"
#include "ostream.h"
#include "str.h"
#include "strescape.h"
#include "settings-parser.h"
#include "master-service.h"
#include "all-settings.h"
#include "sysinfo-get.h"
#include "config-connection.h"
#include "config-parser.h"
#include "config-request.h"
#include "dovecot-version.h"
#include <stdio.h>
#include <stdlib.h>
#include <ctype.h>
#include <unistd.h>
#include <sysexits.h>
struct prefix_stack {
unsigned int prefix_idx;
unsigned int str_pos;
};
struct config_dump_human_context {
struct config_export_context *export_ctx;
unsigned int list_prefix_sent:1;
};
#define LIST_KEY_PREFIX "\001"
#define UNIQUE_KEY_SUFFIX "\xff"
static const char *indent_str = " !!!!";
static void
{
const char *p;
switch (type) {
case CONFIG_KEY_NORMAL:
break;
case CONFIG_KEY_LIST:
break;
case CONFIG_KEY_UNIQUE_KEY:
break;
case CONFIG_KEY_ERROR:
return;
}
}
{
unsigned int i = 0;
return 0;
i++;
}
if (s1[i] == '=')
return -1;
if (s2[i] == '=')
return 1;
}
{
const struct prefix_stack *s;
struct prefix_stack sc;
unsigned int count;
if (count == 1) {
} else {
}
return sc;
}
{
struct prefix_stack *s;
}
static struct config_dump_human_context *
bool check_settings)
{
struct config_dump_human_context *ctx;
enum config_dump_flags flags;
if (check_settings)
return ctx;
}
{
}
static bool value_need_quote(const char *value)
{
if (len == 0)
return FALSE;
return TRUE;
return TRUE;
return FALSE;
}
static int ATTR_NULL(4)
const char *setting_name_filter)
{
struct prefix_stack prefix;
bool unique_key;
int ret = 0;
return -1;
/* strings are sorted so that all lists come first */
if (p[1] == '\0') {
/* "strlist=" */
} else {
/* string is in format: "list=0 1 2" */
*args);
}
}
} T_END;
unique_key = FALSE;
p + 2, NULL);
unique_key = TRUE;
}
if (setting_name_filter_len > 0) {
/* see if this setting matches the name filter */
setting_name_filter_len) == 0 &&
goto end;
}
j = 0;
/* if there are open sections and this key isn't in it,
close the sections */
while (prefix_idx != UINT_MAX) {
indent--;
else {
}
} else {
/* keep the prefix */
j = prefix_idx + 1;
break;
}
}
/* see if this key is in some section */
for (; j < prefix_count; j++) {
prefix_idx = j;
if (p != NULL)
else
else
}
indent++;
if (unique_key)
goto end;
else
goto again;
}
}
if (unique_key) key++;
else {
}
end: ;
} T_END;
while (prefix_idx != UINT_MAX) {
break;
indent--;
}
/* flush output before writing errors */
ret = -1;
}
return ret;
}
static unsigned int
const struct config_filter *filter)
{
unsigned int indent = 0;
if (filter->local_bits > 0) {
} else {
}
indent++;
}
indent++;
}
if (filter->remote_bits > 0) {
} else {
}
indent++;
}
indent++;
}
return indent;
}
static void
{
while (indent > 0) {
indent--;
}
}
static int
const struct config_filter *filter,
const char *const *modules)
{
struct config_filter_parser *const *filters;
static struct config_dump_human_context *ctx;
unsigned int indent;
int ret = 0;
/* first filter should be the global one */
filters++;
FALSE);
ret = -1;
if (ctx->list_prefix_sent)
}
return ret;
}
static int ATTR_NULL(4)
{
static struct config_dump_human_context *ctx;
int ret;
if (setting_name_filter == NULL)
return ret;
}
static int
{
static struct config_dump_human_context *ctx;
const char *const *str;
unsigned int len;
bool dump_section = FALSE;
return -1;
continue;
if (hide_key)
else {
}
break;
dump_section = TRUE;
}
}
if (dump_section)
return 0;
}
void *context)
{
char **setting_name_filters = context;
unsigned int i, filter_len;
if (setting_name_filters == NULL) {
return;
}
for (i = 0; setting_name_filters[i] != NULL; i++) {
}
}
void *context ATTR_UNUSED)
{
T_BEGIN {
} T_END;
}
{
struct config_module_parser *l;
const struct setting_define *def;
const char *const *value;
const void *set;
continue;
return *value;
}
}
}
return "";
}
{
else {
value = "";
}
} else {
}
}
struct hostname_format {
unsigned int numcount;
bool zeropadding;
};
static void
unsigned int num)
{
str_truncate(str, 0);
if (!fmt->zeropadding)
else
}
static void hostname_verify_format(const char *arg)
{
struct hostname_format fmt;
const char *p;
unsigned char hash[GUID_128_HOST_HASH_SIZE];
HASH_TABLE(void *, void *) hosts;
const char *host2;
bool duplicates = FALSE;
/* host%d, host%2d, host%02d */
if (p == NULL)
i_fatal("Host parameter missing %%d");
if (*p == '0') {
p++;
}
if (!i_isdigit(*p))
else
if (*p++ != 'd')
i_fatal("Host parameter missing %%d");
} else {
/* detect host1[suffix] vs host01[suffix] */
len--;
len--;
}
i_fatal("Hostname '%s' has no digits, can't verify",
}
}
limit *= 10;
for (n = 0; n < limit; n++) {
i_error("Duplicate host hashes: %s and %s",
duplicates = TRUE;
} else {
}
}
if (duplicates)
else {
printf("No duplicate host hashes in %s .. %s\n",
exit(0);
}
}
static void check_wrong_config(const char *config_path)
{
return;
i_warning("Dovecot was last started using %s, "
}
}
static void failure_exit_callback(int *status)
{
/* don't use EX_CONFIG, because it often causes MTAs to bounce
the mails back. */
*status = EX_TEMPFAIL;
}
{
ARRAY(const char *) module_names;
struct config_filter filter;
const char *const *wanted_modules, *error;
unsigned int i;
/* we're coming from (e.g.) LDA */
}
i_set_failure_prefix("doveconf: ");
while ((c = master_getopt(master_service)) > 0) {
if (c == 'e') {
expand_vars = TRUE;
break;
}
switch (c) {
case 'a':
break;
case 'd':
break;
case 'f':
break;
case 'h':
break;
case 'H':
host_verify = TRUE;
break;
case 'm':
break;
case 'n':
break;
case 'N':
break;
case 'p':
break;
case 'S':
break;
case 'x':
expand_vars = TRUE;
break;
default:
return FATAL_DEFAULT;
}
}
array_idx(&module_names, 0);
/* use strcmp() instead of !=, because dovecot -n always gives us
-c parameter */
if (host_verify)
if (c == 'e') {
i_fatal("Missing command for -e");
/* print only a single config setting */
} else if (!simple_output) {
/* print the config file path before parsing it, so in case
of errors it's still shown */
}
&error)) == 0 &&
error);
}
if (simple_output) {
struct config_export_context *ctx;
} else if (setting_name_filters != NULL) {
ret2 = 0;
/* ignore settings-check failures in configuration. this allows
using doveconf to lookup settings for things like install or
uninstall scripts where the configuration might
(temporarily) not be fully usable */
ret = 0;
for (i = 0; setting_name_filters[i] != NULL; i++) {
setting_name_filters[i]) < 0)
ret2 = -1;
}
const char *info;
if (*info != '\0')
if (!config_path_specified)
if (scope == CONFIG_DUMP_SCOPE_ALL)
printf("# NOTE: Send doveconf -n output instead when asking for help.\n");
} else {
struct config_export_context *ctx;
env_put("DOVECONF_ENV=1");
if (config_export_finish(&ctx) < 0)
i_fatal("Invalid configuration");
}
if (ret < 0) {
/* delayed error */
}
if (ret2 < 0)
i_fatal("Errors in configuration");
return 0;
}