diff options
Diffstat (limited to '')
-rw-r--r-- | src/imap/imap-sync.c | 841 |
1 files changed, 841 insertions, 0 deletions
diff --git a/src/imap/imap-sync.c b/src/imap/imap-sync.c new file mode 100644 index 0000000..fc49ae7 --- /dev/null +++ b/src/imap/imap-sync.c @@ -0,0 +1,841 @@ +/* Copyright (c) 2002-2018 Dovecot authors, see the included COPYING file */ + +#include "imap-common.h" +#include "str.h" +#include "ostream.h" +#include "mail-user.h" +#include "mail-storage.h" +#include "mail-search-build.h" +#include "imap-quote.h" +#include "imap-util.h" +#include "imap-fetch.h" +#include "imap-notify.h" +#include "imap-commands.h" +#include "imap-sync-private.h" + +static void uids_to_seqs(struct mailbox *box, ARRAY_TYPE(seq_range) *uids) +{ + T_BEGIN { + ARRAY_TYPE(seq_range) seqs; + const struct seq_range *range; + uint32_t seq1, seq2; + + t_array_init(&seqs, array_count(uids)); + array_foreach(uids, range) { + mailbox_get_seq_range(box, range->seq1, range->seq2, + &seq1, &seq2); + /* since we have to notify about expunged messages, + we expect that all the referenced UIDs exist */ + i_assert(seq1 != 0); + i_assert(seq2 - seq1 == range->seq2 - range->seq1); + + seq_range_array_add_range(&seqs, seq1, seq2); + } + /* replace uids with seqs */ + array_clear(uids); + array_append_array(uids, &seqs); + + } T_END; +} + +static int search_update_fetch_more(const struct imap_search_update *update) +{ + int ret; + + if ((ret = imap_fetch_more_no_lock_update(update->fetch_ctx)) == 0) + return 0; + /* finished the FETCH */ + if (imap_fetch_end(update->fetch_ctx) < 0) + return -1; + return ret; +} + +static int +imap_sync_send_fetch_to_search_update(struct imap_sync_context *ctx, + const struct imap_search_update *update) +{ + struct mail_search_args *search_args; + struct mail_search_arg *arg; + ARRAY_TYPE(seq_range) seqs; + + if (ctx->search_update_notifying) + return search_update_fetch_more(update); + + i_assert(!update->fetch_ctx->state.fetching); + + if (array_count(&ctx->search_adds) == 0 || !ctx->have_new_mails) + return 1; + + search_args = mail_search_build_init(); + arg = mail_search_build_add(search_args, SEARCH_UIDSET); + p_array_init(&arg->value.seqset, search_args->pool, 1); + + /* find the newly appended messages: ctx->messages_count is the message + count before new messages found by sync, client->messages_count is + the number of messages after. */ + t_array_init(&seqs, 1); + seq_range_array_add_range(&seqs, ctx->messages_count+1, + ctx->client->messages_count); + mailbox_get_uid_range(ctx->client->mailbox, &seqs, &arg->value.seqset); + /* remove messages not in the search_adds list */ + seq_range_array_intersect(&arg->value.seqset, &ctx->search_adds); + + imap_fetch_begin(update->fetch_ctx, ctx->client->mailbox, search_args); + mail_search_args_unref(&search_args); + return search_update_fetch_more(update); +} + +static int +imap_sync_send_search_update(struct imap_sync_context *ctx, + const struct imap_search_update *update, + bool removes_only) +{ + string_t *cmd; + int ret = 1; + + if (!ctx->search_update_notifying) { + mailbox_search_result_sync(update->result, &ctx->search_removes, + &ctx->search_adds); + } + if (array_count(&ctx->search_adds) == 0 && + array_count(&ctx->search_removes) == 0) + return 1; + + i_assert(array_count(&ctx->search_adds) == 0 || !removes_only); + if (update->fetch_ctx != NULL) { + ret = imap_sync_send_fetch_to_search_update(ctx, update); + if (ret == 0) { + ctx->search_update_notifying = TRUE; + return 0; + } + } + ctx->search_update_notifying = FALSE; + + cmd = t_str_new(256); + str_append(cmd, "* ESEARCH (TAG "); + imap_append_string(cmd, update->tag); + str_append_c(cmd, ')'); + if (update->return_uids) + str_append(cmd, " UID"); + else { + /* convert to sequences */ + uids_to_seqs(ctx->client->mailbox, &ctx->search_removes); + uids_to_seqs(ctx->client->mailbox, &ctx->search_adds); + } + + if (array_count(&ctx->search_removes) != 0) { + str_printfa(cmd, " REMOVEFROM (0 "); + imap_write_seq_range(cmd, &ctx->search_removes); + str_append_c(cmd, ')'); + } + if (array_count(&ctx->search_adds) != 0) { + str_printfa(cmd, " ADDTO (0 "); + imap_write_seq_range(cmd, &ctx->search_adds); + str_append_c(cmd, ')'); + } + str_append(cmd, "\r\n"); + o_stream_nsend(ctx->client->output, str_data(cmd), str_len(cmd)); + return ret; +} + +static int +imap_sync_send_search_updates(struct imap_sync_context *ctx, bool removes_only) +{ + const struct imap_search_update *updates; + unsigned int i, count; + int ret = 1; + + if (!array_is_created(&ctx->client->search_updates)) + return 1; + + if (!array_is_created(&ctx->search_removes)) { + i_array_init(&ctx->search_removes, 64); + i_array_init(&ctx->search_adds, 128); + } + + updates = array_get(&ctx->client->search_updates, &count); + for (i = ctx->search_update_idx; i < count; i++) { + T_BEGIN { + ret = imap_sync_send_search_update(ctx, &updates[i], + removes_only); + } T_END; + if (ret <= 0) + break; + } + ctx->search_update_idx = i; + return ret; +} + +struct imap_sync_context * +imap_sync_init(struct client *client, struct mailbox *box, + enum imap_sync_flags imap_flags, enum mailbox_sync_flags flags) +{ + struct imap_sync_context *ctx; + + i_assert(client->mailbox == box); + + if (client->notify_immediate_expunges) { + /* NOTIFY enabled without SELECTED-DELAYED */ + flags &= ENUM_NEGATE(MAILBOX_SYNC_FLAG_NO_EXPUNGES); + } + + ctx = i_new(struct imap_sync_context, 1); + ctx->client = client; + ctx->box = box; + ctx->imap_flags = imap_flags; + i_array_init(&ctx->module_contexts, 5); + + /* make sure user can't DoS the system by causing Dovecot to create + tons of useless namespaces. */ + mail_user_drop_useless_namespaces(client->user); + + ctx->sync_ctx = mailbox_sync_init(box, flags); + ctx->t = mailbox_transaction_begin(box, 0, "Mailbox sync"); + ctx->mail = mail_alloc(ctx->t, MAIL_FETCH_FLAGS, NULL); + ctx->messages_count = client->messages_count; + i_array_init(&ctx->tmp_keywords, client->keywords.announce_count + 8); + + if (client_has_enabled(client, imap_feature_qresync)) { + i_array_init(&ctx->expunges, 128); + /* always send UIDs in FETCH replies */ + ctx->imap_flags |= IMAP_SYNC_FLAG_SEND_UID; + } + + client_send_mailbox_flags(client, FALSE); + /* send search updates the first time after sync is initialized. + it now contains expunged messages that must be sent before + EXPUNGE replies. */ + if (imap_sync_send_search_updates(ctx, TRUE) == 0) + i_unreached(); + ctx->search_update_idx = 0; + return ctx; +} + +static void +imap_sync_send_highestmodseq(struct imap_sync_context *ctx, + struct client_command_context *sync_cmd) +{ + struct client *client = ctx->client; + uint64_t send_modseq = 0; + + if (ctx->sync_status.sync_delayed_expunges && + client->highest_fetch_modseq > client->sync_last_full_modseq) { + /* if client updates highest-modseq using returned MODSEQs + it loses expunges. try to avoid this by sending it a lower + pre-expunge HIGHESTMODSEQ reply. */ + send_modseq = client->sync_last_full_modseq; + } else if (!ctx->sync_status.sync_delayed_expunges && + ctx->status.highest_modseq > client->sync_last_full_modseq && + ctx->status.highest_modseq > client->highest_fetch_modseq) { + /* we've probably sent some VANISHED or EXISTS replies which + increased the highest-modseq. notify the client about + this. */ + send_modseq = ctx->status.highest_modseq; + } + + if (send_modseq == 0) { + /* no sending */ + } else if (sync_cmd->sync != NULL && /* IDLE doesn't have ->sync */ + sync_cmd->sync->tagline != NULL && /* NOTIFY doesn't have tagline */ + str_begins(sync_cmd->sync->tagline, "OK ") && + sync_cmd->sync->tagline[3] != '[') { + /* modify the tagged reply directly */ + sync_cmd->sync->tagline = p_strdup_printf(sync_cmd->pool, + "OK [HIGHESTMODSEQ %"PRIu64"] %s", + send_modseq, sync_cmd->sync->tagline + 3); + } else { + /* send an untagged OK reply */ + client_send_line(client, t_strdup_printf( + "* OK [HIGHESTMODSEQ %"PRIu64"] Highest", + send_modseq)); + } + + if (!ctx->sync_status.sync_delayed_expunges) { + /* no delayed expunges, remember this for future */ + client->sync_last_full_modseq = ctx->status.highest_modseq; + } + client->highest_fetch_modseq = 0; +} + +static int imap_sync_finish(struct imap_sync_context *ctx, bool aborting) +{ + struct client *client = ctx->client; + int ret = ctx->failed ? -1 : 0; + + if (ctx->finished) + return ret; + ctx->finished = TRUE; + + mail_free(&ctx->mail); + /* the transaction is used only for fetching modseqs/flags. + it can't really fail.. */ + (void)mailbox_transaction_commit(&ctx->t); + + if (array_is_created(&ctx->expunges)) + array_free(&ctx->expunges); + + if (mailbox_sync_deinit(&ctx->sync_ctx, &ctx->sync_status) < 0 || + ctx->failed) { + ctx->failed = TRUE; + ret = -1; + } + mailbox_get_open_status(ctx->box, STATUS_UIDVALIDITY | + STATUS_MESSAGES | STATUS_RECENT | + STATUS_HIGHESTMODSEQ, &ctx->status); + + if (ctx->status.uidvalidity != client->uidvalidity) { + /* most clients would get confused by this. disconnect them. */ + client_disconnect_with_error(client, + "Mailbox UIDVALIDITY changed"); + } + if (mailbox_is_inconsistent(ctx->box)) { + client_disconnect_with_error(client, + "IMAP session state is inconsistent, please relogin."); + /* we can't trust status information anymore, so don't try to + sync message counts. */ + return -1; + } + if (!ctx->no_newmail && !aborting) { + if (ctx->status.messages < ctx->messages_count) + i_panic("Message count decreased"); + if (ctx->status.messages != ctx->messages_count && + client->notify_count_changes) { + client_send_line(client, + t_strdup_printf("* %u EXISTS", ctx->status.messages)); + ctx->have_new_mails = TRUE; + } + if (ctx->status.recent != client->recent_count && + client->notify_count_changes) { + client_send_line(client, + t_strdup_printf("* %u RECENT", ctx->status.recent)); + } + client->messages_count = ctx->status.messages; + client->recent_count = ctx->status.recent; + } + return ret; +} + +static int imap_sync_notify_more(struct imap_sync_context *ctx) +{ + int ret = 1; + + if (ctx->have_new_mails && ctx->client->notify_ctx != NULL) { + /* send FETCH replies for the new mails */ + if ((ret = imap_client_notify_newmails(ctx->client)) == 0) + return 0; + if (ret < 0) + ctx->failed = TRUE; + } + + /* send search updates the second time after syncing in done. + now it contains added/removed messages. */ + if ((ret = imap_sync_send_search_updates(ctx, FALSE)) < 0) + ctx->failed = TRUE; + + if (ret > 0) + ret = ctx->client->v.sync_notify_more(ctx); + return ret; +} + +int imap_sync_deinit(struct imap_sync_context *ctx, + struct client_command_context *sync_cmd) +{ + int ret; + + ret = imap_sync_finish(ctx, TRUE); + imap_client_notify_finished(ctx->client); + + if (client_has_enabled(ctx->client, imap_feature_qresync) && + !ctx->client->nonpermanent_modseqs) + imap_sync_send_highestmodseq(ctx, sync_cmd); + + if (array_is_created(&ctx->search_removes)) { + array_free(&ctx->search_removes); + array_free(&ctx->search_adds); + } + + array_free(&ctx->tmp_keywords); + array_free(&ctx->module_contexts); + i_free(ctx); + return ret; +} + +static void imap_sync_add_modseq(struct imap_sync_context *ctx, string_t *str) +{ + uint64_t modseq; + + modseq = mail_get_modseq(ctx->mail); + if (ctx->client->highest_fetch_modseq < modseq) + ctx->client->highest_fetch_modseq = modseq; + str_printfa(str, "MODSEQ (%"PRIu64")", modseq); +} + +static int imap_sync_send_flags(struct imap_sync_context *ctx, string_t *str) +{ + enum mail_flags flags; + const char *const *keywords; + + mail_set_seq(ctx->mail, ctx->seq); + flags = mail_get_flags(ctx->mail); + keywords = client_get_keyword_names(ctx->client, &ctx->tmp_keywords, + mail_get_keyword_indexes(ctx->mail)); + + if ((flags & MAIL_DELETED) != 0) + ctx->client->sync_seen_deletes = TRUE; + + str_truncate(str, 0); + str_printfa(str, "* %u FETCH (", ctx->seq); + if ((ctx->imap_flags & IMAP_SYNC_FLAG_SEND_UID) != 0) + str_printfa(str, "UID %u ", ctx->mail->uid); + if (client_has_enabled(ctx->client, imap_feature_condstore) && + !ctx->client->nonpermanent_modseqs) { + imap_sync_add_modseq(ctx, str); + str_append_c(str, ' '); + } + str_append(str, "FLAGS ("); + imap_write_flags(str, flags, keywords); + str_append(str, "))"); + return client_send_line_next(ctx->client, str_c(str)); +} + +static int imap_sync_send_modseq(struct imap_sync_context *ctx, string_t *str) +{ + mail_set_seq(ctx->mail, ctx->seq); + + str_truncate(str, 0); + str_printfa(str, "* %u FETCH (", ctx->seq); + if ((ctx->imap_flags & IMAP_SYNC_FLAG_SEND_UID) != 0) + str_printfa(str, "UID %u ", ctx->mail->uid); + imap_sync_add_modseq(ctx, str); + str_append_c(str, ')'); + return client_send_line_next(ctx->client, str_c(str)); +} + +static void imap_sync_vanished(struct imap_sync_context *ctx) +{ + const struct seq_range *seqs; + unsigned int i, count; + string_t *line; + uint32_t seq, prev_uid, start_uid; + bool comma = FALSE; + + /* Convert expunge sequences to UIDs and send them in VANISHED line. */ + seqs = array_get(&ctx->expunges, &count); + if (count == 0) + return; + + line = t_str_new(256); + str_append(line, "* VANISHED "); + for (i = 0; i < count; i++) { + start_uid = 0; prev_uid = 0; + for (seq = seqs[i].seq1; seq <= seqs[i].seq2; seq++) { + mail_set_seq(ctx->mail, seq); + if (prev_uid == 0 || prev_uid + 1 != ctx->mail->uid) { + if (start_uid != 0) { + if (!comma) + comma = TRUE; + else + str_append_c(line, ','); + str_printfa(line, "%u", start_uid); + if (start_uid != prev_uid) { + str_printfa(line, ":%u", + prev_uid); + } + } + start_uid = ctx->mail->uid; + } + prev_uid = ctx->mail->uid; + } + if (!comma) + comma = TRUE; + else + str_append_c(line, ','); + str_printfa(line, "%u", start_uid); + if (start_uid != prev_uid) + str_printfa(line, ":%u", prev_uid); + } + str_append(line, "\r\n"); + o_stream_nsend(ctx->client->output, str_data(line), str_len(line)); +} + +static int imap_sync_send_expunges(struct imap_sync_context *ctx, string_t *str) +{ + int ret = 1; + + if (!ctx->client->notify_count_changes) { + /* NOTIFY: MessageEvent not specified for selected mailbox */ + return 1; + } + + if (array_is_created(&ctx->expunges)) { + /* Use a single VANISHED line */ + seq_range_array_add_range(&ctx->expunges, + ctx->sync_rec.seq1, + ctx->sync_rec.seq2); + return 1; + } + if (ctx->seq == 0) + ctx->seq = ctx->sync_rec.seq2; + for (; ctx->seq >= ctx->sync_rec.seq1; ctx->seq--) { + if (ret == 0) { + /* buffer full, continue later */ + return 0; + } + + str_truncate(str, 0); + str_printfa(str, "* %u EXPUNGE", ctx->seq); + ret = client_send_line_next(ctx->client, str_c(str)); + } + return 1; +} + +int imap_sync_more(struct imap_sync_context *ctx) +{ + string_t *str; + int ret = 1; + + if (ctx->finished) + return imap_sync_notify_more(ctx); + + /* finish syncing even when client has disconnected. otherwise our + internal state (ctx->messages_count) can get messed up and unless + we immediately stop handling all commands and syncs we could end up + assert-crashing. */ + str = t_str_new(256); + for (;;) { + if (ctx->seq == 0) { + /* get next one */ + if (!mailbox_sync_next(ctx->sync_ctx, &ctx->sync_rec)) { + /* finished */ + ret = 1; + break; + } + } + + if (ctx->sync_rec.seq2 > ctx->messages_count) { + /* don't send change notifications of messages we + haven't even announced to client yet */ + if (ctx->sync_rec.seq1 > ctx->messages_count) { + ctx->seq = 0; + continue; + } + ctx->sync_rec.seq2 = ctx->messages_count; + } + + /* EXPUNGEs must come last */ + i_assert(!array_is_created(&ctx->expunges) || + array_count(&ctx->expunges) == 0 || + ctx->sync_rec.type == MAILBOX_SYNC_TYPE_EXPUNGE); + switch (ctx->sync_rec.type) { + case MAILBOX_SYNC_TYPE_FLAGS: + if (!ctx->client->notify_flag_changes) { + /* NOTIFY: FlagChange not specified for + selected mailbox */ + break; + } + if (ctx->seq == 0) + ctx->seq = ctx->sync_rec.seq1; + + ret = 1; + for (; ctx->seq <= ctx->sync_rec.seq2; ctx->seq++) { + if (ret == 0) + break; + + ret = imap_sync_send_flags(ctx, str); + } + break; + case MAILBOX_SYNC_TYPE_EXPUNGE: + ret = imap_sync_send_expunges(ctx, str); + if (ret > 0) { + /* update only after we're finished, so that + the seq2 > messages_count check above + doesn't break */ + ctx->messages_count -= + ctx->sync_rec.seq2 - + ctx->sync_rec.seq1 + 1; + } + break; + case MAILBOX_SYNC_TYPE_MODSEQ: + if (!client_has_enabled(ctx->client, imap_feature_condstore)) + break; + if (!ctx->client->notify_flag_changes) { + /* NOTIFY: FlagChange not specified for + selected mailbox. The RFC doesn't explicitly + specify MODSEQ changes, but they're close + enough to flag changes. */ + break; + } + + if (ctx->seq == 0) + ctx->seq = ctx->sync_rec.seq1; + + ret = 1; + for (; ctx->seq <= ctx->sync_rec.seq2; ctx->seq++) { + if (ret == 0) + break; + + ret = imap_sync_send_modseq(ctx, str); + } + break; + } + if (ret == 0) { + /* buffer full */ + break; + } + + ctx->seq = 0; + } + if (ret > 0) { + if (array_is_created(&ctx->expunges)) + imap_sync_vanished(ctx); + if (imap_sync_finish(ctx, FALSE) < 0) + return -1; + return imap_sync_more(ctx); + } + return ret; +} + +bool imap_sync_is_allowed(struct client *client) +{ + if (client->syncing) + return FALSE; + + if (client->mailbox != NULL && + mailbox_transaction_get_count(client->mailbox) > 0) + return FALSE; + + return TRUE; +} + +static bool cmd_finish_sync(struct client_command_context *cmd) +{ + if (cmd->sync->tagline != NULL) + client_send_tagline(cmd, cmd->sync->tagline); + return TRUE; +} + +static bool cmd_sync_continue(struct client_command_context *sync_cmd) +{ + struct client_command_context *cmd, *prev; + struct client *client = sync_cmd->client; + struct imap_sync_context *ctx = sync_cmd->context; + int ret; + + i_assert(ctx->client == client); + + if ((ret = imap_sync_more(ctx)) == 0) + return FALSE; + if (ret < 0) + ctx->failed = TRUE; + + client->syncing = FALSE; + if (imap_sync_deinit(ctx, sync_cmd) < 0) { + client_send_untagged_storage_error(client, + mailbox_get_storage(client->mailbox)); + } + sync_cmd->context = NULL; + + /* Finish all commands that waited for this sync. Go through the queue + backwards, so that tagged replies are sent in the same order as + they were received. This fixes problems with clients that rely on + this (Apple Mail 3.2) */ + for (cmd = client->command_queue; cmd->next != NULL; cmd = cmd->next) ; + for (; cmd != NULL; cmd = prev) { + prev = cmd->prev; + + if (cmd->state == CLIENT_COMMAND_STATE_WAIT_SYNC && + cmd != sync_cmd && + cmd->sync->counter+1 == client->sync_counter) { + cmd_finish_sync(cmd); + client_command_free(&cmd); + } + } + cmd_finish_sync(sync_cmd); + return TRUE; +} + +static void get_common_sync_flags(struct client *client, + enum mailbox_sync_flags *flags_r, + enum imap_sync_flags *imap_flags_r) +{ + struct client_command_context *cmd; + unsigned int count = 0, fast_count = 0, noexpunges_count = 0; + + *flags_r = 0; + *imap_flags_r = 0; + + for (cmd = client->command_queue; cmd != NULL; cmd = cmd->next) { + if (cmd->sync != NULL && + cmd->sync->counter == client->sync_counter) { + if ((cmd->sync->flags & MAILBOX_SYNC_FLAG_FAST) != 0) + fast_count++; + if ((cmd->sync->flags & MAILBOX_SYNC_FLAG_NO_EXPUNGES) != 0) + noexpunges_count++; + *flags_r |= cmd->sync->flags; + *imap_flags_r |= cmd->sync->imap_flags; + count++; + } + } + i_assert(noexpunges_count == 0 || noexpunges_count == count); + if (fast_count != count) + *flags_r &= ENUM_NEGATE(MAILBOX_SYNC_FLAG_FAST); + + i_assert((*flags_r & MAILBOX_SYNC_FLAG_FIX_INCONSISTENT) == 0); +} + +static bool cmd_sync_client(struct client_command_context *sync_cmd) +{ + struct client *client = sync_cmd->client; + struct imap_sync_context *ctx; + enum mailbox_sync_flags flags; + enum imap_sync_flags imap_flags; + bool no_newmail; + + /* there may be multiple commands waiting. use their combined flags */ + get_common_sync_flags(client, &flags, &imap_flags); + client->sync_counter++; + + no_newmail = (client->set->parsed_workarounds & WORKAROUND_DELAY_NEWMAIL) != 0 && + client->notify_ctx == NULL && /* always disabled with NOTIFY */ + (imap_flags & IMAP_SYNC_FLAG_SAFE) == 0; + if (no_newmail) { + /* expunges might break the client just as badly as new mail + notifications. */ + flags |= MAILBOX_SYNC_FLAG_NO_EXPUNGES; + } + + client->syncing = TRUE; + + ctx = imap_sync_init(client, client->mailbox, imap_flags, flags); + ctx->no_newmail = no_newmail; + + /* handle the syncing using sync_cmd. it doesn't actually matter which + one of the pending commands it is. */ + sync_cmd->func = cmd_sync_continue; + sync_cmd->context = ctx; + sync_cmd->state = CLIENT_COMMAND_STATE_WAIT_OUTPUT; + if (!cmd_sync_continue(sync_cmd)) { + o_stream_set_flush_pending(client->output, TRUE); + return FALSE; + } + + client_command_free(&sync_cmd); + cmd_sync_delayed(client); + return TRUE; +} + +bool cmd_sync(struct client_command_context *cmd, enum mailbox_sync_flags flags, + enum imap_sync_flags imap_flags, const char *tagline) +{ + struct client *client = cmd->client; + + i_assert(client->output_cmd_lock == NULL); + + if (cmd->cancel) + return TRUE; + + cmd->stats.last_run_timeval = ioloop_timeval; + if (client->mailbox == NULL) { + /* no mailbox selected, no point in delaying the sync */ + if (tagline != NULL) + client_send_tagline(cmd, tagline); + return TRUE; + } + cmd->tagline_reply = p_strdup(cmd->pool, tagline); + + cmd->sync = p_new(cmd->pool, struct imap_client_sync_context, 1); + cmd->sync->counter = client->sync_counter; + cmd->sync->flags = flags; + cmd->sync->imap_flags = imap_flags; + cmd->sync->tagline = cmd->tagline_reply; + cmd->state = CLIENT_COMMAND_STATE_WAIT_SYNC; + + cmd->func = NULL; + cmd->context = NULL; + + if (client->input_lock == cmd) + client->input_lock = NULL; + return FALSE; +} + +static bool cmd_sync_drop_fast(struct client *client) +{ + struct client_command_context *cmd, *prev; + bool ret = FALSE; + + if (client->command_queue == NULL) + return FALSE; + + for (cmd = client->command_queue; cmd->next != NULL; cmd = cmd->next) ; + for (; cmd != NULL; cmd = prev) { + prev = cmd->next; + + if (cmd->state != CLIENT_COMMAND_STATE_WAIT_SYNC) + continue; + + i_assert(cmd->sync != NULL); + if ((cmd->sync->flags & MAILBOX_SYNC_FLAG_FAST) != 0) { + cmd_finish_sync(cmd); + client_command_free(&cmd); + ret = TRUE; + } + } + return ret; +} + +static bool cmd_sync_delayed_real(struct client *client) +{ + struct client_command_context *cmd, *first_expunge, *first_nonexpunge; + + if (client->output_cmd_lock != NULL) { + /* wait until we can send output to client */ + return FALSE; + } + + if (!imap_sync_is_allowed(client)) { + /* wait until mailbox can be synced */ + return cmd_sync_drop_fast(client); + } + + /* separate syncs that can send expunges from those that can't */ + first_expunge = first_nonexpunge = NULL; + for (cmd = client->command_queue; cmd != NULL; cmd = cmd->next) { + if (cmd->sync != NULL && + cmd->sync->counter == client->sync_counter) { + if ((cmd->sync->flags & MAILBOX_SYNC_FLAG_NO_EXPUNGES) != 0) { + if (first_nonexpunge == NULL) + first_nonexpunge = cmd; + } else { + if (first_expunge == NULL) + first_expunge = cmd; + } + } + } + if (first_expunge != NULL && first_nonexpunge != NULL) { + /* sync expunges after nonexpunges */ + for (cmd = first_expunge; cmd != NULL; cmd = cmd->next) { + if (cmd->sync != NULL && + cmd->sync->counter == client->sync_counter && + (cmd->sync->flags & + MAILBOX_SYNC_FLAG_NO_EXPUNGES) == 0) + cmd->sync->counter++; + } + first_expunge = NULL; + } + cmd = first_nonexpunge != NULL ? first_nonexpunge : first_expunge; + + if (cmd == NULL) + return cmd_sync_drop_fast(client); + i_assert(client->mailbox != NULL); + return cmd_sync_client(cmd); +} + +bool cmd_sync_delayed(struct client *client) +{ + bool ret; + + T_BEGIN { + ret = cmd_sync_delayed_real(client); + } T_END; + return ret; +} |