virtual-config.c revision 428fb4dc39c6e9b2eb36216c396dad6096a65f8f
2454dfa32c93c20a8522c6ed42fe057baaac9f9aStephan Bosch/* Copyright (c) 2008-2011 Dovecot authors, see the included COPYING file */
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainenvirtual_search_args_parse(const string_t *rule, const char **error_r)
0dffa25d211be541ee3c953b23566a1a990789dfTimo Sirainen input = i_stream_create_from_data(str_data(rule), str_len(rule));
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen imap_parser = imap_parser_create(input, NULL, (size_t)-1);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen ret = imap_parser_finish_line(imap_parser, 0, 0, &args);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen *error_r = t_strdup(imap_parser_get_error(imap_parser, &fatal));
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen if (mail_search_build(mail_search_register_get_imap(),
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainenvirtual_config_add_rule(struct virtual_parse_context *ctx, const char **error_r)
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen unsigned int i, count;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen if (ctx->rule_idx == array_count(&ctx->mbox->backend_boxes)) {
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen crc32_str_more(ctx->mbox->search_args_crc32, str_c(ctx->rule));
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen search_args = virtual_search_args_parse(ctx->rule, error_r);
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen *error_r = t_strconcat("Previous search rule is invalid: ",
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen /* update at all the mailboxes that were introduced since the previous
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen bboxes = array_get(&ctx->mbox->backend_boxes, &count);
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen ctx->rule_idx = array_count(&ctx->mbox->backend_boxes);
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainenvirtual_config_parse_line(struct virtual_parse_context *ctx, const char *line,
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen const char **error_r)
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen struct mail_user *user = ctx->mbox->storage->storage.user;
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen /* continues the previous search rule */
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen if (ctx->rule_idx == array_count(&ctx->mbox->backend_boxes)) {
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen /* if there is no rule yet, it means we want the previous mailboxes
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen to use the rule that comes later */
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen if (virtual_config_add_rule(ctx, error_r) < 0)
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen /* new mailbox. the search args are added to it later. */
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen bbox = p_new(ctx->pool, struct virtual_backend_box, 1);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen if (*line == '-' || *line == '+' || *line == '!') line++;
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen *error_r = t_strdup_printf("Mailbox name not UTF-8: %s",
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen *error_r = t_strdup_printf("Namespace not found for %s",
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen name = bbox->name[0] == '-' ? bbox->name + 1 : bbox->name;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen bbox->glob = imap_match_init(ctx->pool, name, TRUE, ctx->sep);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen /* save messages here */
303a87c31cb4aa198326694e231df53a043e63c7Timo Sirainen array_append(&ctx->mbox->backend_boxes, &bbox, 1);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainenvirtual_mailbox_get_list_patterns(struct virtual_parse_context *ctx)
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen unsigned int i, count;
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes, &count);
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen p_array_init(&mbox->list_include_patterns, ctx->pool, count);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen p_array_init(&mbox->list_exclude_patterns, ctx->pool, count);
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen for (i = 0; i < count; i++) {
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainenseparate_wildcard_mailboxes(struct virtual_mailbox *mbox,
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen ARRAY_TYPE(virtual_backend_box) *wildcard_boxes,
39e6fcc3e8b1ccb13087c232cb6bdea04d1a20a4Timo Sirainen unsigned int i, count;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes, &count);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen t_array_init(wildcard_boxes, I_MIN(16, count));
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen for (i = 0; i < count;) {
3398d5e2b883812de5d569721c8294b581e1d9e6Timo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes,
8039af9679af6fb56116b353fe44f7dd4c08f031Timo Sirainenstatic void virtual_config_copy_expanded(struct virtual_parse_context *ctx,
06e72c658de3ce1252594b151313df90acf73271Timo Sirainen bbox = p_new(ctx->pool, struct virtual_backend_box, 1);
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen array_append(&ctx->mbox->backend_boxes, &bbox, 1);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainenstatic bool virtual_config_match(const struct mailbox_info *info,
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen unsigned int *idx_r)
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen unsigned int i, count;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen boxes = array_get_modifiable(boxes_arr, &count);
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen for (i = 0; i < count; i++) {
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen /* we match only one namespace for each pattern. */
07974f50bd55b06fd6d465f2c0e491794786e2faTimo Sirainen if (strcmp(boxes[i]->name + 1, info->name) == 0) {
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainenstatic int virtual_config_expand_wildcards(struct virtual_parse_context *ctx)
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen NAMESPACE_PRIVATE | NAMESPACE_SHARED | NAMESPACE_PUBLIC;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen const enum mailbox_list_iter_flags iter_flags =
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen struct mail_user *user = ctx->mbox->storage->storage.user;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen ARRAY_TYPE(virtual_backend_box) wildcard_boxes, neg_boxes;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen unsigned int i, j, count;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen separate_wildcard_mailboxes(ctx->mbox, &wildcard_boxes, &neg_boxes);
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen /* get patterns we want to list */
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen wboxes = array_get_modifiable(&wildcard_boxes, &count);
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen /* only negative wildcards - doesn't really make sense.
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen just ignore. */
17da42c31202b1b3e7e308121ea17d922c24da1bTimo Sirainen for (i = 0; i < count; i++)
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen /* match listed mailboxes to wildcards */
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen iter = mailbox_list_iter_init_namespaces(user->namespaces, patterns,
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen while ((info = mailbox_list_iter_next(iter)) != NULL) {
d5cebe7f98e63d4e2822863ef2faa4971e8b3a5dTimo Sirainen /* skip non-selectable mailboxes (especially mbox
7891195e3975d554df183670dba1fcecfa0a30c3Timo Sirainen directories) */
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen if (virtual_config_match(info, &wildcard_boxes, &i) &&
923eb3dde28e4d8841c14fd6b4a69635b7070c3eTimo Sirainen !virtual_config_match(info, &neg_boxes, &j) &&
51327f2489a4e0e615eb9f7d921473cf8512bb79Timo Sirainen for (i = 0; i < count; i++)
51327f2489a4e0e615eb9f7d921473cf8512bb79Timo Sirainen mail_search_args_unref(&wboxes[i]->search_args);
bd74402ca1a39ec303075fefb1212d7e18a71531Timo Sirainenstatic void virtual_config_search_args_dup(struct virtual_mailbox *mbox)
505561860cdfee4eac51469fd27a59983ef72e8eTimo Sirainen unsigned int i, count;
505561860cdfee4eac51469fd27a59983ef72e8eTimo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes, &count);
505561860cdfee4eac51469fd27a59983ef72e8eTimo Sirainen for (i = 0; i < count; i++) {
505561860cdfee4eac51469fd27a59983ef72e8eTimo Sirainen bboxes[i]->search_args = mail_search_args_dup(old_args);
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainenint virtual_config_read(struct virtual_mailbox *mbox)
17da42c31202b1b3e7e308121ea17d922c24da1bTimo Sirainen struct mail_storage *storage = mbox->box.storage;
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen unsigned int linenum = 0;
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen path = t_strconcat(box_path, "/"VIRTUAL_CONFIG_FNAME, NULL);
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen mail_storage_set_error(storage, MAIL_ERROR_NOTPOSSIBLE,
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen "Virtual mailbox missing configuration file");
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen mail_storage_set_error(storage, MAIL_ERROR_NOTFOUND,
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen T_MAIL_ERR_MAILBOX_NOT_FOUND(mbox->box.name));
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen ctx.sep = mail_namespaces_get_root_sep(storage->user->namespaces);
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen ctx.input = i_stream_create_fd(fd, (size_t)-1, FALSE);
270f00aeab7bede38764291e21a314211b884ab4Timo Sirainen i_stream_set_return_partial_line(ctx.input, TRUE);
51cbc45fc1ac5dde29bc2adbb175945df1b4f7d4Timo Sirainen while ((line = i_stream_read_next_line(ctx.input)) != NULL) {
3dd0679b6f24be0287cc42d7a60bbf59cdf8b637Timo Sirainen ret = virtual_config_parse_line(&ctx, line, &error);
c4267cf4c40fb1f866b5958ff122ef836b8c5dfbTimo Sirainen "%s: Error at line %u: %s",
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen "%s: Error at line %u: %s",
16c89b1260c9d07c01c83a9219424d3727069b2eTimo Sirainen if (ret == 0 && array_count(&mbox->backend_boxes) == 0) {
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainenvoid virtual_config_free(struct virtual_mailbox *mbox)
45c763dbebee268eda4a1e8bcc1ff82606b5ed0dTimo Sirainen unsigned int i, count;
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen if (!array_is_created(&mbox->backend_boxes)) {
28cd2599128e102198758cf6080588305feb6bcdTimo Sirainen /* mailbox wasn't opened */
a4bc2c3962b94f83c7bb7bebe7af364f4dee7883Timo Sirainen bboxes = array_get_modifiable(&mbox->backend_boxes, &count);
e34d170f8f0e084bd94bfbc1a7085ece67e508dfTimo Sirainen for (i = 0; i < count; i++) {