virtual-config.c revision 4307c886579381dbb1897ea1388ae6978c96f560
2454dfa32c93c20a8522c6ed42fe057baaac9f9aStephan Bosch/* Copyright (c) 2008-2012 Dovecot authors, see the included COPYING file */
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen#include "lib.h"
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen#include "array.h"
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen#include "crc32.h"
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen#include "istream.h"
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen#include "str.h"
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen#include "unichar.h"
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen#include "imap-parser.h"
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen#include "imap-match.h"
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen#include "mail-namespace.h"
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen#include "mail-search-build.h"
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen#include "mail-search-parser.h"
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen#include "virtual-storage.h"
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen#include "virtual-plugin.h"
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen#include <unistd.h>
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen#include <fcntl.h>
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
5a470af15391d6fc73bebb41512a671b3d847644Timo Sirainenstruct virtual_parse_context {
5a470af15391d6fc73bebb41512a671b3d847644Timo Sirainen struct virtual_mailbox *mbox;
5a470af15391d6fc73bebb41512a671b3d847644Timo Sirainen struct istream *input;
5a470af15391d6fc73bebb41512a671b3d847644Timo Sirainen
5a470af15391d6fc73bebb41512a671b3d847644Timo Sirainen pool_t pool;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen string_t *rule;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen unsigned int rule_idx;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen char sep;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bool have_wildcards;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen};
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenstatic struct mail_search_args *
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenvirtual_search_args_parse(const string_t *rule, const char **error_r)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen{
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct istream *input;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct imap_parser *imap_parser;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen const struct imap_arg *args;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct mail_search_parser *parser;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct mail_search_args *sargs;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen const char *charset = "UTF-8";
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bool fatal;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen int ret;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (str_len(rule) == 0) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen sargs = mail_search_build_init();
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_search_build_add_all(sargs);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return sargs;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen input = i_stream_create_from_data(str_data(rule), str_len(rule));
191153d1a5b0eb0c129139570e3aa5212f28d2acJosef 'Jeff' Sipek (void)i_stream_read(input);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen imap_parser = imap_parser_create(input, NULL, (size_t)-1);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ret = imap_parser_finish_line(imap_parser, 0, 0, &args);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (ret < 0) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen sargs = NULL;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen *error_r = t_strdup(imap_parser_get_error(imap_parser, &fatal));
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen } else {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen parser = mail_search_parser_init_imap(args);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (mail_search_build(mail_search_register_get_imap(),
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen parser, &charset, &sargs, error_r) < 0)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen sargs = NULL;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_search_parser_deinit(&parser);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen imap_parser_unref(&imap_parser);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen i_stream_destroy(&input);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return sargs;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen}
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenstatic int
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenvirtual_config_add_rule(struct virtual_parse_context *ctx, const char **error_r)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen{
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct virtual_backend_box *const *bboxes;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct mail_search_args *search_args;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen unsigned int i, count;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (ctx->rule_idx == array_count(&ctx->mbox->backend_boxes)) {
191153d1a5b0eb0c129139570e3aa5212f28d2acJosef 'Jeff' Sipek i_assert(str_len(ctx->rule) == 0);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return 0;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ctx->mbox->search_args_crc32 =
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen crc32_str_more(ctx->mbox->search_args_crc32, str_c(ctx->rule));
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen search_args = virtual_search_args_parse(ctx->rule, error_r);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen str_truncate(ctx->rule, 0);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (search_args == NULL) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen *error_r = t_strconcat("Previous search rule is invalid: ",
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen *error_r, NULL);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return -1;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen /* update at all the mailboxes that were introduced since the previous
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen rule. */
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bboxes = array_get(&ctx->mbox->backend_boxes, &count);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen i_assert(ctx->rule_idx < count);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen for (i = ctx->rule_idx; i < count; i++) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen i_assert(bboxes[i]->search_args == NULL);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_search_args_ref(search_args);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bboxes[i]->search_args = search_args;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_search_args_unref(&search_args);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ctx->rule_idx = array_count(&ctx->mbox->backend_boxes);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return 0;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen}
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenstatic int
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenvirtual_config_parse_line(struct virtual_parse_context *ctx, const char *line,
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen const char **error_r)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen{
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct mail_user *user = ctx->mbox->storage->storage.user;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct virtual_backend_box *bbox;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen const char *name;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (*line == ' ' || *line == '\t') {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen /* continues the previous search rule */
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (ctx->rule_idx == array_count(&ctx->mbox->backend_boxes)) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen *error_r = "Search rule without a mailbox";
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return -1;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen while (*line == ' ' || *line == '\t') line++;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen str_append_c(ctx->rule, ' ');
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen str_append(ctx->rule, line);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return 0;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen /* if there is no rule yet, it means we want the previous mailboxes
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen to use the rule that comes later */
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (str_len(ctx->rule) > 0) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (virtual_config_add_rule(ctx, error_r) < 0)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return -1;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen /* new mailbox. the search args are added to it later. */
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bbox = p_new(ctx->pool, struct virtual_backend_box, 1);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (strcasecmp(line, "INBOX") == 0)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen line = "INBOX";
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bbox->name = p_strdup(ctx->pool, line);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (*line == '-' || *line == '+' || *line == '!') line++;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bbox->ns = strcasecmp(line, "INBOX") == 0 ?
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_namespace_find_inbox(user->namespaces) :
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_namespace_find(user->namespaces, line);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (!uni_utf8_str_is_valid(bbox->name)) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen *error_r = t_strdup_printf("Mailbox name not UTF-8: %s",
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bbox->name);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return -1;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (bbox->ns == NULL) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen *error_r = t_strdup_printf("Namespace not found for %s",
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bbox->name);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return -1;
f0339f522dc9c8e2e8a29ef9a3f937c431c6bd1bTimo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (bbox->name[0] == '+') {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bbox->name++;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bbox->clear_recent = TRUE;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (strchr(bbox->name, '*') != NULL ||
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen strchr(bbox->name, '%') != NULL) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen name = bbox->name[0] == '-' ? bbox->name + 1 : bbox->name;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bbox->glob = imap_match_init(ctx->pool, name, TRUE, ctx->sep);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ctx->have_wildcards = TRUE;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen } else if (bbox->name[0] == '!') {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen /* save messages here */
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (ctx->mbox->save_bbox != NULL) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen *error_r = "Multiple save mailboxes defined";
28542ffed61d0b860e0d1f9c837e0eb622df6b51Timo Sirainen return -1;
28542ffed61d0b860e0d1f9c837e0eb622df6b51Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bbox->name++;
28542ffed61d0b860e0d1f9c837e0eb622df6b51Timo Sirainen ctx->mbox->save_bbox = bbox;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen array_append(&ctx->mbox->backend_boxes, &bbox, 1);
e3df4d9063a06e0cd228a1713677ec105b0a4aa2Timo Sirainen return 0;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen}
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenstatic void
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenvirtual_mailbox_get_list_patterns(struct virtual_parse_context *ctx)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen{
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct virtual_mailbox *mbox = ctx->mbox;
e3df4d9063a06e0cd228a1713677ec105b0a4aa2Timo Sirainen ARRAY_TYPE(mailbox_virtual_patterns) *dest;
e3df4d9063a06e0cd228a1713677ec105b0a4aa2Timo Sirainen struct mailbox_virtual_pattern pattern;
e83126866761632b437e532dfdc30be01d14039dTimo Sirainen struct virtual_backend_box *const *bboxes;
d12ea923bd1d532c6f7fee44304dde8d0f321dd0Timo Sirainen unsigned int i, count;
d12ea923bd1d532c6f7fee44304dde8d0f321dd0Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen memset(&pattern, 0, sizeof(pattern));
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes, &count);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen p_array_init(&mbox->list_include_patterns, ctx->pool, count);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen p_array_init(&mbox->list_exclude_patterns, ctx->pool, count);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen for (i = 0; i < count; i++) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen pattern.ns = bboxes[i]->ns;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen pattern.pattern = bboxes[i]->name;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (*pattern.pattern != '-')
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen dest = &mbox->list_include_patterns;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen else {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen dest = &mbox->list_exclude_patterns;
e3df4d9063a06e0cd228a1713677ec105b0a4aa2Timo Sirainen pattern.pattern++;
e3df4d9063a06e0cd228a1713677ec105b0a4aa2Timo Sirainen }
e83126866761632b437e532dfdc30be01d14039dTimo Sirainen array_append(dest, &pattern, 1);
be2bc5677a16a3547d5cbfd21c9deedc96a759adTimo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen}
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenstatic void
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenseparate_wildcard_mailboxes(struct virtual_mailbox *mbox,
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ARRAY_TYPE(virtual_backend_box) *wildcard_boxes,
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ARRAY_TYPE(virtual_backend_box) *neg_boxes)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen{
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct virtual_backend_box *const *bboxes;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ARRAY_TYPE(virtual_backend_box) *dest;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen unsigned int i, count;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes, &count);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen t_array_init(wildcard_boxes, I_MIN(16, count));
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen t_array_init(neg_boxes, 4);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen for (i = 0; i < count;) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (*bboxes[i]->name == '-')
28542ffed61d0b860e0d1f9c837e0eb622df6b51Timo Sirainen dest = neg_boxes;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen else if (bboxes[i]->glob != NULL)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen dest = wildcard_boxes;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen else {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen dest = NULL;
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen i++;
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen }
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen if (dest != NULL) {
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen array_append(dest, &bboxes[i], 1);
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen array_delete(&mbox->backend_boxes, i, 1);
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes,
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen &count);
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen }
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen }
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen}
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainenstatic void virtual_config_copy_expanded(struct virtual_parse_context *ctx,
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen struct virtual_backend_box *wbox,
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen const char *name)
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen{
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen struct virtual_backend_box *bbox;
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen bbox = p_new(ctx->pool, struct virtual_backend_box, 1);
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen *bbox = *wbox;
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen bbox->name = p_strdup(ctx->pool, name);
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen bbox->glob = NULL;
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen bbox->wildcard = TRUE;
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen mail_search_args_ref(bbox->search_args);
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen array_append(&ctx->mbox->backend_boxes, &bbox, 1);
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen}
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainenstatic bool virtual_ns_match(struct mail_namespace *config_ns,
28542ffed61d0b860e0d1f9c837e0eb622df6b51Timo Sirainen struct mail_namespace *iter_ns)
28542ffed61d0b860e0d1f9c837e0eb622df6b51Timo Sirainen{
28542ffed61d0b860e0d1f9c837e0eb622df6b51Timo Sirainen /* we match only one namespace for each pattern, except with shared
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen namespaces match also autocreated children */
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen if (config_ns == iter_ns)
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen return TRUE;
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen if (config_ns->type == iter_ns->type &&
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen (config_ns->flags & NAMESPACE_FLAG_AUTOCREATED) == 0 &&
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen (iter_ns->flags & NAMESPACE_FLAG_AUTOCREATED) != 0)
28542ffed61d0b860e0d1f9c837e0eb622df6b51Timo Sirainen return TRUE;
28542ffed61d0b860e0d1f9c837e0eb622df6b51Timo Sirainen return FALSE;
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen}
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainenstatic bool virtual_config_match(const struct mailbox_info *info,
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen ARRAY_TYPE(virtual_backend_box) *boxes_arr,
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen unsigned int *idx_r)
7676a044f800a539bb0cbbc59fb9d1ad3f30ba7fTimo Sirainen{
28542ffed61d0b860e0d1f9c837e0eb622df6b51Timo Sirainen struct virtual_backend_box *const *boxes;
28542ffed61d0b860e0d1f9c837e0eb622df6b51Timo Sirainen unsigned int i, count;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen boxes = array_get_modifiable(boxes_arr, &count);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen for (i = 0; i < count; i++) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (boxes[i]->glob != NULL) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (virtual_ns_match(boxes[i]->ns, info->ns) &&
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen imap_match(boxes[i]->glob,
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen info->name) == IMAP_MATCH_YES) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen *idx_r = i;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return TRUE;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen } else {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen i_assert(boxes[i]->name[0] == '-');
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (strcmp(boxes[i]->name + 1, info->name) == 0) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen *idx_r = i;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return TRUE;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return FALSE;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen}
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenstatic int virtual_config_expand_wildcards(struct virtual_parse_context *ctx)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen{
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen const enum namespace_type iter_ns_types =
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen NAMESPACE_PRIVATE | NAMESPACE_SHARED | NAMESPACE_PUBLIC;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen const enum mailbox_list_iter_flags iter_flags =
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen MAILBOX_LIST_ITER_RETURN_NO_FLAGS;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct mail_user *user = ctx->mbox->storage->storage.user;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ARRAY_TYPE(virtual_backend_box) wildcard_boxes, neg_boxes;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct mailbox_list_iterate_context *iter;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct virtual_backend_box *const *wboxes;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen const char **patterns;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen const struct mailbox_info *info;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen unsigned int i, j, count;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen separate_wildcard_mailboxes(ctx->mbox, &wildcard_boxes, &neg_boxes);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen /* get patterns we want to list */
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen wboxes = array_get_modifiable(&wildcard_boxes, &count);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (count == 0) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen /* only negative wildcards - doesn't really make sense.
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen just ignore. */
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return 0;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen patterns = t_new(const char *, count + 1);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen for (i = 0; i < count; i++)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen patterns[i] = wboxes[i]->name;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen /* match listed mailboxes to wildcards */
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen iter = mailbox_list_iter_init_namespaces(user->namespaces, patterns,
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen iter_ns_types, iter_flags);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen while ((info = mailbox_list_iter_next(iter)) != NULL) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen /* skip non-selectable mailboxes (especially mbox
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen directories) */
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if ((info->flags & MAILBOX_NOSELECT) != 0)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen continue;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (virtual_config_match(info, &wildcard_boxes, &i) &&
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen !virtual_config_match(info, &neg_boxes, &j) &&
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen virtual_backend_box_lookup_name(ctx->mbox,
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen info->name) == NULL) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen virtual_config_copy_expanded(ctx, wboxes[i],
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen info->name);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen for (i = 0; i < count; i++)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_search_args_unref(&wboxes[i]->search_args);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return mailbox_list_iter_deinit(&iter);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen}
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenstatic void virtual_config_search_args_dup(struct virtual_mailbox *mbox)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen{
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct virtual_backend_box *const *bboxes;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct mail_search_args *old_args;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen unsigned int i, count;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes, &count);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen for (i = 0; i < count; i++) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen old_args = bboxes[i]->search_args;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bboxes[i]->search_args = mail_search_args_dup(old_args);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_search_args_unref(&old_args);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen}
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenint virtual_config_read(struct virtual_mailbox *mbox)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen{
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct mail_storage *storage = mbox->box.storage;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct virtual_parse_context ctx;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct stat st;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen const char *box_path, *path, *line, *error;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen unsigned int linenum = 0;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen int fd, ret = 0;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen i_array_init(&mbox->backend_boxes, 8);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mbox->search_args_crc32 = (uint32_t)-1;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen box_path = mailbox_get_path(&mbox->box);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen path = t_strconcat(box_path, "/"VIRTUAL_CONFIG_FNAME, NULL);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen fd = open(path, O_RDONLY);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (fd == -1) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (errno == EACCES) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_storage_set_critical(storage, "%s",
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_error_eacces_msg("open", path));
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen } else if (errno != ENOENT) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_storage_set_critical(storage,
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen "open(%s) failed: %m", path);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen } else if (stat(box_path, &st) == 0) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_storage_set_error(storage, MAIL_ERROR_NOTPOSSIBLE,
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen "Virtual mailbox missing configuration file");
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen } else if (errno == ENOENT) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_storage_set_error(storage, MAIL_ERROR_NOTFOUND,
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen T_MAIL_ERR_MAILBOX_NOT_FOUND(mbox->box.name));
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen } else {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_storage_set_critical(storage,
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen "stat(%s) failed: %m", box_path);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return -1;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen memset(&ctx, 0, sizeof(ctx));
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ctx.sep = mail_namespaces_get_root_sep(storage->user->namespaces);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ctx.mbox = mbox;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ctx.pool = mbox->box.pool;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ctx.rule = t_str_new(256);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ctx.input = i_stream_create_fd(fd, (size_t)-1, FALSE);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen i_stream_set_return_partial_line(ctx.input, TRUE);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen while ((line = i_stream_read_next_line(ctx.input)) != NULL) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen linenum++;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (*line == '#')
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen continue;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (*line == '\0')
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ret = virtual_config_add_rule(&ctx, &error);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen else
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ret = virtual_config_parse_line(&ctx, line, &error);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (ret < 0) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_storage_set_critical(storage,
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen "%s: Error at line %u: %s",
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen path, linenum, error);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen break;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (ret == 0) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ret = virtual_config_add_rule(&ctx, &error);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (ret < 0) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_storage_set_critical(storage,
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen "%s: Error at line %u: %s",
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen path, linenum, error);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen virtual_mailbox_get_list_patterns(&ctx);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (ret == 0 && ctx.have_wildcards)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ret = virtual_config_expand_wildcards(&ctx);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (ret == 0 && array_count(&mbox->backend_boxes) == 0) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_storage_set_critical(storage,
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen "%s: No mailboxes defined", path);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen ret = -1;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (ret == 0)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen virtual_config_search_args_dup(mbox);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen i_stream_unref(&ctx.input);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen i_close_fd(&fd);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return ret;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen}
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainenvoid virtual_config_free(struct virtual_mailbox *mbox)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen{
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen struct virtual_backend_box *const *bboxes;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen unsigned int i, count;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (!array_is_created(&mbox->backend_boxes)) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen /* mailbox wasn't opened */
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen return;
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes, &count);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen for (i = 0; i < count; i++) {
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen if (bboxes[i]->search_args != NULL)
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen mail_search_args_unref(&bboxes[i]->search_args);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen }
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen array_free(&mbox->backend_boxes);
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen}
39ee82dad4d4fa61e3ed074d191afc6a9b82e249Timo Sirainen