virtual-config.c revision 77a2367051db01c6fc8cc575242d8a7f5b2c1d42
76b43e4417bab52e913da39b5f5bc2a130d3f149Timo Sirainen/* Copyright (c) 2008-2016 Dovecot authors, see the included COPYING file */
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen#include "lib.h"
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen#include "array.h"
c96eb61168670cfdd7596baba18856d3f086a093Timo Sirainen#include "crc32.h"
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen#include "istream.h"
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen#include "str.h"
16c89b1260c9d07c01c83a9219424d3727069b2eTimo Sirainen#include "unichar.h"
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen#include "imap-parser.h"
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen#include "imap-match.h"
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen#include "mail-namespace.h"
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen#include "mail-search-build.h"
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen#include "mail-search-parser.h"
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen#include "mailbox-list-iter.h"
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen#include "virtual-storage.h"
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen#include "virtual-plugin.h"
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen#include <unistd.h>
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen#include <fcntl.h>
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainenstruct virtual_parse_context {
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen struct virtual_mailbox *mbox;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen struct istream *input;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen
39e6fcc3e8b1ccb13087c232cb6bdea04d1a20a4Timo Sirainen pool_t pool;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen string_t *rule;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen unsigned int rule_idx;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen
fdc557286bc9f92c5f3bb49096ff6e2bcec0ea79Timo Sirainen char sep;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen bool have_wildcards;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen bool have_mailbox_defines;
8039af9679af6fb56116b353fe44f7dd4c08f031Timo Sirainen};
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainenstatic struct mail_search_args *
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainenvirtual_search_args_parse(const string_t *rule, const char **error_r)
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen{
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen struct istream *input;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen struct imap_parser *imap_parser;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen const struct imap_arg *args;
150e64c376365becf1ec5c9d45912ecb840eea96Timo Sirainen struct mail_search_parser *parser;
150e64c376365becf1ec5c9d45912ecb840eea96Timo Sirainen struct mail_search_args *sargs;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen const char *charset = "UTF-8";
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen bool fatal;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen int ret;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen if (str_len(rule) == 0) {
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen sargs = mail_search_build_init();
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen mail_search_build_add_all(sargs);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen return sargs;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen }
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen input = i_stream_create_from_data(str_data(rule), str_len(rule));
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen (void)i_stream_read(input);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen
5afa8e2edf4f313cd56e5909f92f39c3b5b7b4d3Timo Sirainen imap_parser = imap_parser_create(input, NULL, (size_t)-1);
5afa8e2edf4f313cd56e5909f92f39c3b5b7b4d3Timo Sirainen ret = imap_parser_finish_line(imap_parser, 0, 0, &args);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen if (ret < 0) {
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen sargs = NULL;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen *error_r = t_strdup(imap_parser_get_error(imap_parser, &fatal));
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen } else {
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen parser = mail_search_parser_init_imap(args);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen if (mail_search_build(mail_search_register_get_imap(),
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen parser, &charset, &sargs, error_r) < 0)
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen sargs = NULL;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen mail_search_parser_deinit(&parser);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen }
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen imap_parser_unref(&imap_parser);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen i_stream_destroy(&input);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen return sargs;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen}
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainenstatic int
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainenvirtual_config_add_rule(struct virtual_parse_context *ctx, const char **error_r)
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen{
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen struct virtual_backend_box *const *bboxes;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen struct mail_search_args *search_args;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen unsigned int i, count;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen *error_r = NULL;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen if (ctx->rule_idx == array_count(&ctx->mbox->backend_boxes)) {
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen i_assert(str_len(ctx->rule) == 0);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen return 0;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen }
303a87c31cb4aa198326694e231df53a043e63c7Timo Sirainen
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen ctx->mbox->search_args_crc32 =
303a87c31cb4aa198326694e231df53a043e63c7Timo Sirainen crc32_str_more(ctx->mbox->search_args_crc32, str_c(ctx->rule));
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen search_args = virtual_search_args_parse(ctx->rule, error_r);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen str_truncate(ctx->rule, 0);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen if (search_args == NULL) {
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen i_assert(*error_r != NULL);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen *error_r = t_strconcat("Previous search rule is invalid: ",
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen *error_r, NULL);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen return -1;
303a87c31cb4aa198326694e231df53a043e63c7Timo Sirainen }
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen
303a87c31cb4aa198326694e231df53a043e63c7Timo Sirainen /* update at all the mailboxes that were introduced since the previous
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen rule. */
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen bboxes = array_get(&ctx->mbox->backend_boxes, &count);
303a87c31cb4aa198326694e231df53a043e63c7Timo Sirainen i_assert(ctx->rule_idx < count);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen for (i = ctx->rule_idx; i < count; i++) {
303a87c31cb4aa198326694e231df53a043e63c7Timo Sirainen i_assert(bboxes[i]->search_args == NULL);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen mail_search_args_ref(search_args);
f2bd9e507b8befdd95a983f86664febf5c19bd95Timo Sirainen bboxes[i]->search_args = search_args;
303a87c31cb4aa198326694e231df53a043e63c7Timo Sirainen }
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen mail_search_args_unref(&search_args);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen ctx->rule_idx = array_count(&ctx->mbox->backend_boxes);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen return 0;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen}
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainenstatic int
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainenvirtual_config_parse_line(struct virtual_parse_context *ctx, const char *line,
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen const char **error_r)
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen{
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen struct mail_user *user = ctx->mbox->storage->storage.user;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen struct virtual_backend_box *bbox;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen if (*line == ' ' || *line == '\t') {
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen /* continues the previous search rule */
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen if (ctx->rule_idx == array_count(&ctx->mbox->backend_boxes)) {
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen *error_r = "Search rule without a mailbox";
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen return -1;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen }
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen while (*line == ' ' || *line == '\t') line++;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen str_append_c(ctx->rule, ' ');
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen str_append(ctx->rule, line);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen return 0;
39e6fcc3e8b1ccb13087c232cb6bdea04d1a20a4Timo Sirainen }
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen /* if there is no rule yet, it means we want the previous mailboxes
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen to use the rule that comes later */
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen if (str_len(ctx->rule) > 0) {
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen if (virtual_config_add_rule(ctx, error_r) < 0)
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen return -1;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen }
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen if (!uni_utf8_str_is_valid(line)) {
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen *error_r = t_strdup_printf("Mailbox name not UTF-8: %s",
39e6fcc3e8b1ccb13087c232cb6bdea04d1a20a4Timo Sirainen line);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen return -1;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen }
fdc557286bc9f92c5f3bb49096ff6e2bcec0ea79Timo Sirainen
fdc557286bc9f92c5f3bb49096ff6e2bcec0ea79Timo Sirainen /* new mailbox. the search args are added to it later. */
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen bbox = p_new(ctx->pool, struct virtual_backend_box, 1);
8039af9679af6fb56116b353fe44f7dd4c08f031Timo Sirainen bbox->virtual_mbox = ctx->mbox;
bc3698b8892df8003b410daea6f5bbcd20433808Timo Sirainen if (strcasecmp(line, "INBOX") == 0)
06e72c658de3ce1252594b151313df90acf73271Timo Sirainen line = "INBOX";
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen bbox->name = p_strdup(ctx->pool, line);
06e72c658de3ce1252594b151313df90acf73271Timo Sirainen switch (bbox->name[0]) {
06e72c658de3ce1252594b151313df90acf73271Timo Sirainen case '+':
06e72c658de3ce1252594b151313df90acf73271Timo Sirainen bbox->name++;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen bbox->clear_recent = TRUE;
8039af9679af6fb56116b353fe44f7dd4c08f031Timo Sirainen break;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen case '-':
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen bbox->name++;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen bbox->negative_match = TRUE;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen break;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen }
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen if (strchr(bbox->name, '*') != NULL ||
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen strchr(bbox->name, '%') != NULL) {
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen bbox->glob = imap_match_init(ctx->pool, bbox->name, TRUE, ctx->sep);
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen ctx->have_wildcards = TRUE;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen } else if (bbox->name[0] == '!') {
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen /* save messages here */
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen if (ctx->mbox->save_bbox != NULL) {
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen *error_r = "Multiple save mailboxes defined";
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen return -1;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen }
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen bbox->name++;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen ctx->mbox->save_bbox = bbox;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen }
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen /* now that the prefix characters have been processed,
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen find the namespace */
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen bbox->ns = strcasecmp(bbox->name, "INBOX") == 0 ?
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen mail_namespace_find_inbox(user->namespaces) :
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen mail_namespace_find(user->namespaces, bbox->name);
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen if (bbox->ns == NULL) {
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen *error_r = t_strdup_printf("Namespace not found for %s",
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen bbox->name);
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen return -1;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen }
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen if (strcmp(bbox->name, ctx->mbox->box.vname) == 0) {
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen *error_r = "Virtual mailbox can't point to itself";
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen return -1;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen }
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen ctx->have_mailbox_defines = TRUE;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen array_append(&ctx->mbox->backend_boxes, &bbox, 1);
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen return 0;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen}
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainenstatic void
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainenvirtual_mailbox_get_list_patterns(struct virtual_parse_context *ctx)
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen{
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen struct virtual_mailbox *mbox = ctx->mbox;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen ARRAY_TYPE(mailbox_virtual_patterns) *dest;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen struct mailbox_virtual_pattern pattern;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen struct virtual_backend_box *const *bboxes;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen unsigned int i, count;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen memset(&pattern, 0, sizeof(pattern));
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes, &count);
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen p_array_init(&mbox->list_include_patterns, ctx->pool, count);
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen p_array_init(&mbox->list_exclude_patterns, ctx->pool, count);
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen for (i = 0; i < count; i++) {
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen pattern.ns = bboxes[i]->ns;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen pattern.pattern = bboxes[i]->name;
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen if (bboxes[i]->negative_match)
17da42c31202b1b3e7e308121ea17d922c24da1bTimo Sirainen dest = &mbox->list_include_patterns;
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen else {
ccb77e2f63626ec46e5745ef4f38baa8e8e504fcTimo Sirainen dest = &mbox->list_exclude_patterns;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen pattern.pattern++;
d5cebe7f98e63d4e2822863ef2faa4971e8b3a5dTimo Sirainen }
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen array_append(dest, &pattern, 1);
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen }
fdc557286bc9f92c5f3bb49096ff6e2bcec0ea79Timo Sirainen}
bd74402ca1a39ec303075fefb1212d7e18a71531Timo Sirainen
17da42c31202b1b3e7e308121ea17d922c24da1bTimo Sirainenstatic void
17da42c31202b1b3e7e308121ea17d922c24da1bTimo Sirainenseparate_wildcard_mailboxes(struct virtual_mailbox *mbox,
5fb3bff645380804c9db2510940c41db6b8fdb01Timo Sirainen ARRAY_TYPE(virtual_backend_box) *wildcard_boxes,
d5cebe7f98e63d4e2822863ef2faa4971e8b3a5dTimo Sirainen ARRAY_TYPE(virtual_backend_box) *neg_boxes)
f5a24412980cb19b07cb0cd12dba75886f281875Timo Sirainen{
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen struct virtual_backend_box *const *bboxes;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen ARRAY_TYPE(virtual_backend_box) *dest;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen unsigned int i, count;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen
63f36c2b47217fc2dc4ed49cfc1907311d5ed366Timo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes, &count);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen t_array_init(wildcard_boxes, I_MIN(16, count));
17da42c31202b1b3e7e308121ea17d922c24da1bTimo Sirainen t_array_init(neg_boxes, 4);
bd74402ca1a39ec303075fefb1212d7e18a71531Timo Sirainen for (i = 0; i < count;) {
17da42c31202b1b3e7e308121ea17d922c24da1bTimo Sirainen if (bboxes[i]->negative_match)
bd74402ca1a39ec303075fefb1212d7e18a71531Timo Sirainen dest = neg_boxes;
bd74402ca1a39ec303075fefb1212d7e18a71531Timo Sirainen else if (bboxes[i]->glob != NULL)
ef11d3930c3602fc86349a4e3a53442df470b601Timo Sirainen dest = wildcard_boxes;
c6ae908f6a2313573625d782bdd4e0ff3882c44aTimo Sirainen else {
c6ae908f6a2313573625d782bdd4e0ff3882c44aTimo Sirainen dest = NULL;
17da42c31202b1b3e7e308121ea17d922c24da1bTimo Sirainen i++;
ef11d3930c3602fc86349a4e3a53442df470b601Timo Sirainen }
17da42c31202b1b3e7e308121ea17d922c24da1bTimo Sirainen
ef11d3930c3602fc86349a4e3a53442df470b601Timo Sirainen if (dest != NULL) {
ef11d3930c3602fc86349a4e3a53442df470b601Timo Sirainen array_append(dest, &bboxes[i], 1);
17da42c31202b1b3e7e308121ea17d922c24da1bTimo Sirainen array_delete(&mbox->backend_boxes, i, 1);
17b03c9db961e1c004284907d969eb11b08a795eTimo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes,
17da42c31202b1b3e7e308121ea17d922c24da1bTimo Sirainen &count);
17da42c31202b1b3e7e308121ea17d922c24da1bTimo Sirainen }
ef11d3930c3602fc86349a4e3a53442df470b601Timo Sirainen }
ef11d3930c3602fc86349a4e3a53442df470b601Timo Sirainen}
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainenstatic void virtual_config_copy_expanded(struct virtual_parse_context *ctx,
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen struct virtual_backend_box *wbox,
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen const char *name)
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen{
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen struct virtual_backend_box *bbox;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen bbox = p_new(ctx->pool, struct virtual_backend_box, 1);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen *bbox = *wbox;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen bbox->name = p_strdup(ctx->pool, name);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen bbox->glob = NULL;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen bbox->wildcard = TRUE;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen mail_search_args_ref(bbox->search_args);
8039af9679af6fb56116b353fe44f7dd4c08f031Timo Sirainen array_append(&ctx->mbox->backend_boxes, &bbox, 1);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen}
ccb77e2f63626ec46e5745ef4f38baa8e8e504fcTimo Sirainen
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainenstatic bool virtual_ns_match(struct mail_namespace *config_ns,
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen struct mail_namespace *iter_ns)
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen{
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen /* we match only one namespace for each pattern, except with shared
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen namespaces match also autocreated children */
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen if (config_ns == iter_ns)
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen return TRUE;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen if (config_ns->type == iter_ns->type &&
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen (config_ns->flags & NAMESPACE_FLAG_AUTOCREATED) == 0 &&
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen (iter_ns->flags & NAMESPACE_FLAG_AUTOCREATED) != 0)
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen return TRUE;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen if ((iter_ns->flags & NAMESPACE_FLAG_INBOX_USER) != 0 &&
8039af9679af6fb56116b353fe44f7dd4c08f031Timo Sirainen (config_ns->flags & NAMESPACE_FLAG_AUTOCREATED) != 0 &&
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen config_ns->prefix_len == 0) {
16c89b1260c9d07c01c83a9219424d3727069b2eTimo Sirainen /* prefix="" namespace was autocreated, so e.g. "*" would match
8039af9679af6fb56116b353fe44f7dd4c08f031Timo Sirainen only that empty namespace. but we want "*" to also match
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen the inbox=yes namespace, so check it here separately. */
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen return TRUE;
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen }
8039af9679af6fb56116b353fe44f7dd4c08f031Timo Sirainen return FALSE;
8039af9679af6fb56116b353fe44f7dd4c08f031Timo Sirainen}
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainenstatic bool virtual_config_match(const struct mailbox_info *info,
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen ARRAY_TYPE(virtual_backend_box) *boxes_arr,
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen unsigned int *idx_r)
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen{
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen struct virtual_backend_box *const *boxes;
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen unsigned int i, count;
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen boxes = array_get_modifiable(boxes_arr, &count);
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen for (i = 0; i < count; i++) {
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen if (boxes[i]->glob != NULL) {
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen if (virtual_ns_match(boxes[i]->ns, info->ns) &&
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen imap_match(boxes[i]->glob,
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen info->vname) == IMAP_MATCH_YES) {
8039af9679af6fb56116b353fe44f7dd4c08f031Timo Sirainen *idx_r = i;
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen return TRUE;
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen }
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen } else {
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen if (strcmp(boxes[i]->name, info->vname) == 0) {
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen *idx_r = i;
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen return TRUE;
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen }
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen }
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen }
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen return FALSE;
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen}
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainenstatic int virtual_config_expand_wildcards(struct virtual_parse_context *ctx,
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen const char **error_r)
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen{
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen const enum mail_namespace_type iter_ns_types =
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen MAIL_NAMESPACE_TYPE_MASK_ALL;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen const enum mailbox_list_iter_flags iter_flags =
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen MAILBOX_LIST_ITER_RETURN_NO_FLAGS;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen struct mail_user *user = ctx->mbox->storage->storage.user;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen ARRAY_TYPE(virtual_backend_box) wildcard_boxes, neg_boxes;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen struct mailbox_list_iterate_context *iter;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen struct virtual_backend_box *const *wboxes;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen const char **patterns;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen const struct mailbox_info *info;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen unsigned int i, j, count;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen separate_wildcard_mailboxes(ctx->mbox, &wildcard_boxes, &neg_boxes);
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen /* get patterns we want to list */
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen wboxes = array_get_modifiable(&wildcard_boxes, &count);
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen if (count == 0) {
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen /* only negative wildcards - doesn't really make sense.
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen just ignore. */
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen return 0;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen }
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen patterns = t_new(const char *, count + 1);
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen for (i = 0; i < count; i++)
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen patterns[i] = wboxes[i]->name;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen /* match listed mailboxes to wildcards */
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen iter = mailbox_list_iter_init_namespaces(user->namespaces, patterns,
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen iter_ns_types, iter_flags);
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen while ((info = mailbox_list_iter_next(iter)) != NULL) {
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen /* skip non-selectable mailboxes (especially mbox
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen directories) */
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen if ((info->flags & MAILBOX_NOSELECT) != 0)
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen continue;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen if (strcmp(info->vname, ctx->mbox->box.vname) == 0) {
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen /* don't allow virtual folder to point to itself */
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen continue;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen }
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen if (virtual_config_match(info, &wildcard_boxes, &i) &&
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen !virtual_config_match(info, &neg_boxes, &j) &&
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen virtual_backend_box_lookup_name(ctx->mbox,
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen info->vname) == NULL) {
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen virtual_config_copy_expanded(ctx, wboxes[i],
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen info->vname);
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen }
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen }
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen for (i = 0; i < count; i++)
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen mail_search_args_unref(&wboxes[i]->search_args);
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen if (mailbox_list_iter_deinit(&iter) < 0) {
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen *error_r = mailbox_list_get_last_error(user->namespaces->list, NULL);
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen return -1;
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen }
68efcccb384f2d6871164b072457e87473502c51Timo Sirainen return 0;
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen}
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainenstatic void virtual_config_search_args_dup(struct virtual_mailbox *mbox)
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen{
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen struct virtual_backend_box *const *bboxes;
c979eeda1f46483d9c963e265786b701d7683d77Timo Sirainen struct mail_search_args *old_args;
c979eeda1f46483d9c963e265786b701d7683d77Timo Sirainen unsigned int i, count;
c979eeda1f46483d9c963e265786b701d7683d77Timo Sirainen
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes, &count);
68efcccb384f2d6871164b072457e87473502c51Timo Sirainen for (i = 0; i < count; i++) {
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen old_args = bboxes[i]->search_args;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen bboxes[i]->search_args = mail_search_args_dup(old_args);
a54be2bd26d6f0860d194d3aeedfa6b7fc14d24cTimo Sirainen mail_search_args_unref(&old_args);
a54be2bd26d6f0860d194d3aeedfa6b7fc14d24cTimo Sirainen }
a54be2bd26d6f0860d194d3aeedfa6b7fc14d24cTimo Sirainen}
10cfe8a2bdc5ccfc05380689c71c27209327538fTimo Sirainen
10cfe8a2bdc5ccfc05380689c71c27209327538fTimo Sirainenint virtual_config_read(struct virtual_mailbox *mbox)
a54be2bd26d6f0860d194d3aeedfa6b7fc14d24cTimo Sirainen{
10cfe8a2bdc5ccfc05380689c71c27209327538fTimo Sirainen struct mail_storage *storage = mbox->box.storage;
a54be2bd26d6f0860d194d3aeedfa6b7fc14d24cTimo Sirainen struct virtual_parse_context ctx;
a54be2bd26d6f0860d194d3aeedfa6b7fc14d24cTimo Sirainen const char *box_path, *path, *line, *error;
a54be2bd26d6f0860d194d3aeedfa6b7fc14d24cTimo Sirainen unsigned int linenum = 0;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen int fd, ret = 0;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen i_array_init(&mbox->backend_boxes, 8);
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen mbox->search_args_crc32 = (uint32_t)-1;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen box_path = mailbox_get_path(&mbox->box);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen path = t_strconcat(box_path, "/"VIRTUAL_CONFIG_FNAME, NULL);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen fd = open(path, O_RDONLY);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen if (fd == -1) {
88286b0527bcc0711e312e9db65ca121a45213e3Timo Sirainen if (errno == EACCES) {
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen mail_storage_set_critical(storage, "%s",
88286b0527bcc0711e312e9db65ca121a45213e3Timo Sirainen mail_error_eacces_msg("open", path));
88286b0527bcc0711e312e9db65ca121a45213e3Timo Sirainen } else if (errno != ENOENT) {
88286b0527bcc0711e312e9db65ca121a45213e3Timo Sirainen mail_storage_set_critical(storage,
8039af9679af6fb56116b353fe44f7dd4c08f031Timo Sirainen "open(%s) failed: %m", path);
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen } else if (errno == ENOENT) {
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen mail_storage_set_error(storage, MAIL_ERROR_NOTFOUND,
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen T_MAIL_ERR_MAILBOX_NOT_FOUND(mbox->box.vname));
e30e068c8fac372ae217b3b31791a0c8c8046b7fTimo Sirainen } else {
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen mail_storage_set_critical(storage,
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen "stat(%s) failed: %m", box_path);
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen }
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen return -1;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen }
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen memset(&ctx, 0, sizeof(ctx));
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen ctx.sep = mail_namespaces_get_root_sep(storage->user->namespaces);
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen ctx.mbox = mbox;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen ctx.pool = mbox->box.pool;
d42c9a8f362b76740418c4f9f9441eb7fc661e57Timo Sirainen ctx.rule = t_str_new(256);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen ctx.input = i_stream_create_fd(fd, (size_t)-1, FALSE);
88286b0527bcc0711e312e9db65ca121a45213e3Timo Sirainen i_stream_set_return_partial_line(ctx.input, TRUE);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen while ((line = i_stream_read_next_line(ctx.input)) != NULL) {
88286b0527bcc0711e312e9db65ca121a45213e3Timo Sirainen linenum++;
88286b0527bcc0711e312e9db65ca121a45213e3Timo Sirainen if (*line == '#')
88286b0527bcc0711e312e9db65ca121a45213e3Timo Sirainen continue;
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen if (*line == '\0')
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen ret = virtual_config_add_rule(&ctx, &error);
68efcccb384f2d6871164b072457e87473502c51Timo Sirainen else
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen ret = virtual_config_parse_line(&ctx, line, &error);
d42c9a8f362b76740418c4f9f9441eb7fc661e57Timo Sirainen if (ret < 0) {
d42c9a8f362b76740418c4f9f9441eb7fc661e57Timo Sirainen mail_storage_set_critical(storage,
d42c9a8f362b76740418c4f9f9441eb7fc661e57Timo Sirainen "%s: Error at line %u: %s",
d42c9a8f362b76740418c4f9f9441eb7fc661e57Timo Sirainen path, linenum, error);
d42c9a8f362b76740418c4f9f9441eb7fc661e57Timo Sirainen break;
d42c9a8f362b76740418c4f9f9441eb7fc661e57Timo Sirainen }
d42c9a8f362b76740418c4f9f9441eb7fc661e57Timo Sirainen }
d42c9a8f362b76740418c4f9f9441eb7fc661e57Timo Sirainen if (ret == 0) {
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen ret = virtual_config_add_rule(&ctx, &error);
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen if (ret < 0) {
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen mail_storage_set_critical(storage,
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen "%s: Error at line %u: %s",
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen path, linenum, error);
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen }
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen }
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen virtual_mailbox_get_list_patterns(&ctx);
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen if (ret == 0 && ctx.have_wildcards) {
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen ret = virtual_config_expand_wildcards(&ctx, &error);
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen if (ret < 0)
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen mail_storage_set_critical(storage, "%s: %s", path, error);
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen }
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen if (ret == 0 && !ctx.have_mailbox_defines) {
a2f250a332dfc1e6cd4ffd196c621eb9dbf7b8a1Timo Sirainen mail_storage_set_critical(storage,
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen "%s: No mailboxes defined", path);
88286b0527bcc0711e312e9db65ca121a45213e3Timo Sirainen ret = -1;
88286b0527bcc0711e312e9db65ca121a45213e3Timo Sirainen }
88286b0527bcc0711e312e9db65ca121a45213e3Timo Sirainen if (ret == 0)
88286b0527bcc0711e312e9db65ca121a45213e3Timo Sirainen virtual_config_search_args_dup(mbox);
88286b0527bcc0711e312e9db65ca121a45213e3Timo Sirainen i_stream_unref(&ctx.input);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen i_close_fd(&fd);
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen return ret;
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen}
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen
68efcccb384f2d6871164b072457e87473502c51Timo Sirainenvoid virtual_config_free(struct virtual_mailbox *mbox)
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen{
d756ebcfa96bd7cff02097c8f26df9df368b81b1Timo Sirainen struct virtual_backend_box *const *bboxes;
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen unsigned int i, count;
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen if (!array_is_created(&mbox->backend_boxes)) {
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen /* mailbox wasn't opened */
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen return;
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen }
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes, &count);
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen for (i = 0; i < count; i++) {
ccd83028a34cc6e2b6370eb7ecf1cf25e717c2d3Timo Sirainen if (bboxes[i]->search_args != NULL)
c96eb61168670cfdd7596baba18856d3f086a093Timo Sirainen mail_search_args_unref(&bboxes[i]->search_args);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen }
1939d1843ee6c7ca5e5baa3967b0332341440005Timo Sirainen array_free(&mbox->backend_boxes);
c96eb61168670cfdd7596baba18856d3f086a093Timo Sirainen}
c96eb61168670cfdd7596baba18856d3f086a093Timo Sirainen