mail-index-view-sync.c revision 0c909e3461607eadcd66f4eac69b7f34e37fccf1
5f5870385cff47efd2f58e7892f251cf13761528Timo Sirainen/* Copyright (c) 2003-2009 Dovecot authors, see the included COPYING file */
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen#include "lib.h"
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen#include "array.h"
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen#include "buffer.h"
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen#include "mail-index-view-private.h"
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen#include "mail-index-sync-private.h"
5733207dc3ec10e6e5a6e0a8b30fbd1b061062b9Timo Sirainen#include "mail-index-modseq.h"
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen#include "mail-transaction-log.h"
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen#include <stdlib.h>
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainenstruct mail_index_view_sync_ctx {
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen struct mail_index_view *view;
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen enum mail_index_view_sync_flags flags;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen struct mail_index_sync_map_ctx sync_map_ctx;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen /* After syncing view, map is replaced with sync_new_map. */
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen struct mail_index_map *sync_new_map;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen ARRAY_TYPE(seq_range) expunges;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen unsigned int finish_min_msg_count;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen const struct mail_transaction_header *hdr;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen const void *data;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen /* temporary variables while handling lost transaction logs: */
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen ARRAY_TYPE(keyword_indexes) lost_old_kw, lost_new_kw;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen buffer_t *lost_kw_buf;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen uint32_t lost_new_ext_idx;
5733207dc3ec10e6e5a6e0a8b30fbd1b061062b9Timo Sirainen /* result of lost transaction logs: */
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen ARRAY_TYPE(seq_range) lost_flags;
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen unsigned int lost_flag_idx;
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen size_t data_offset;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen unsigned int failed:1;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen unsigned int sync_map_update:1;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen unsigned int skipped_expunges:1;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen unsigned int last_read:1;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen unsigned int log_was_lost:1;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen unsigned int hidden:1;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen};
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainenstatic int
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainenview_sync_set_log_view_range(struct mail_index_view *view, bool sync_expunges,
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen bool *reset_r)
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen{
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen const struct mail_index_header *hdr = &view->index->map->hdr;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen uint32_t start_seq, end_seq;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen uoff_t start_offset, end_offset;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen int ret;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen start_seq = view->log_file_expunge_seq;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen start_offset = view->log_file_expunge_offset;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen end_seq = hdr->log_file_seq;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen end_offset = hdr->log_file_head_offset;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen for (;;) {
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen /* the view begins from the first non-synced transaction */
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen ret = mail_transaction_log_view_set(view->log_view,
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen start_seq, start_offset,
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen end_seq, end_offset,
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen reset_r);
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen if (ret <= 0)
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen return ret;
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen if (!*reset_r || sync_expunges)
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen break;
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen /* log was reset, but we don't want to sync expunges.
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen we can't do this, so sync only up to the reset. */
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen mail_transaction_log_view_get_prev_pos(view->log_view,
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen &end_seq, &end_offset);
2726de0d31ed0f1a1425e4c8d167f7681e07fc25Timo Sirainen end_seq--; end_offset = (uoff_t)-1;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen if (end_seq < start_seq) {
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen /* we have only this reset log */
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen mail_transaction_log_view_clear(view->log_view,
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen view->log_file_expunge_seq);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen break;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen }
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen }
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen return 1;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen}
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainenstatic unsigned int
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainenview_sync_expunges2seqs(struct mail_index_view_sync_ctx *ctx)
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen{
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen struct mail_index_view *view = ctx->view;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen struct seq_range *src, *src_end, *dest;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen unsigned int count, expunge_count = 0;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen uint32_t prev_seq = 0;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen /* convert UIDs to sequences */
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen src = dest = array_get_modifiable(&ctx->expunges, &count);
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen src_end = src + count;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen for (; src != src_end; src++) {
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen if (!mail_index_lookup_seq_range(view, src->seq1, src->seq2,
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen &dest->seq1, &dest->seq2))
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen count--;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen else {
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen i_assert(dest->seq1 > prev_seq);
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen prev_seq = dest->seq2;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen expunge_count += dest->seq2 - dest->seq1 + 1;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen dest++;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen }
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen }
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen array_delete(&ctx->expunges, count,
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen array_count(&ctx->expunges) - count);
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen return expunge_count;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen}
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainenstatic void
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainenview_sync_add_expunge_range(ARRAY_TYPE(seq_range) *dest,
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen const struct seq_range *src, size_t src_size)
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen{
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen unsigned int i, src_count;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen i_assert(src_size % sizeof(*src) == 0);
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen src_count = src_size / sizeof(*src);
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen for (i = 0; i < src_count; i++)
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen seq_range_array_add_range(dest, src[i].seq1, src[i].seq2);
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen}
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainenstatic void
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainenview_sync_add_expunge_guids(ARRAY_TYPE(seq_range) *dest,
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen const struct mail_transaction_expunge_guid *src,
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen size_t src_size)
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen{
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen unsigned int i, src_count;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen i_assert(src_size % sizeof(*src) == 0);
32a93320fd2b6ada5ac8027166819463c1a007b6Timo Sirainen
32a93320fd2b6ada5ac8027166819463c1a007b6Timo Sirainen src_count = src_size / sizeof(*src);
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen for (i = 0; i < src_count; i++)
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen seq_range_array_add(dest, 0, src[i].uid);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen}
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainenstatic int
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainenview_sync_get_expunges(struct mail_index_view_sync_ctx *ctx,
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen unsigned int *expunge_count_r)
b75feeb580a78d76188f66d589939b298da4b61bTimo Sirainen{
00b6a40fd9f032914b41fe2cfe52d902db29829bTimo Sirainen struct mail_index_view *view = ctx->view;
b75feeb580a78d76188f66d589939b298da4b61bTimo Sirainen const struct mail_transaction_header *hdr;
b75feeb580a78d76188f66d589939b298da4b61bTimo Sirainen const void *data;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen int ret;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen /* get a list of expunge transactions. there may be some that we have
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen already synced, but it doesn't matter because they'll get dropped
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen out when converting to sequences. the uid ranges' validity has
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen already been verified, so we can use them directly. */
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen mail_transaction_log_view_mark(view->log_view);
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen while ((ret = mail_transaction_log_view_next(view->log_view,
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen &hdr, &data)) > 0) {
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen if ((hdr->type & MAIL_TRANSACTION_EXTERNAL) == 0) {
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen /* skip expunge requests */
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen continue;
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen }
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen if ((hdr->type & MAIL_TRANSACTION_EXPUNGE_GUID) != 0) {
b75feeb580a78d76188f66d589939b298da4b61bTimo Sirainen view_sync_add_expunge_guids(&ctx->expunges,
b75feeb580a78d76188f66d589939b298da4b61bTimo Sirainen data, hdr->size);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen } else if ((hdr->type & MAIL_TRANSACTION_EXPUNGE) != 0) {
447bf65ddb82ec279e7386828748ef47e199a6afTimo Sirainen view_sync_add_expunge_range(&ctx->expunges,
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen data, hdr->size);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen }
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen }
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen mail_transaction_log_view_rewind(view->log_view);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen *expunge_count_r = view_sync_expunges2seqs(ctx);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen return 0;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen}
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainenstatic bool have_existing_expunges(struct mail_index_view *view,
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen const struct seq_range *range, size_t size)
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen{
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen const struct seq_range *range_end;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen uint32_t seq1, seq2;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen range_end = CONST_PTR_OFFSET(range, size);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen for (; range != range_end; range++) {
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen if (mail_index_lookup_seq_range(view, range->seq1, range->seq2,
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen &seq1, &seq2))
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen return TRUE;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen }
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen return FALSE;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen}
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainenstatic bool
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainenhave_existing_guid_expunge(struct mail_index_view *view,
5733207dc3ec10e6e5a6e0a8b30fbd1b061062b9Timo Sirainen const struct mail_transaction_expunge_guid *expunges,
5733207dc3ec10e6e5a6e0a8b30fbd1b061062b9Timo Sirainen size_t size)
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen{
5733207dc3ec10e6e5a6e0a8b30fbd1b061062b9Timo Sirainen const struct mail_transaction_expunge_guid *expunges_end;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen uint32_t seq;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen expunges_end = CONST_PTR_OFFSET(expunges, size);
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen for (; expunges != expunges_end; expunges++) {
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen if (mail_index_lookup_seq(view, expunges->uid, &seq))
7dd73e056c7d9976080393d9cd731f69111818eeTimo Sirainen return TRUE;
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen }
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen return FALSE;
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen}
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainenstatic bool view_sync_have_expunges(struct mail_index_view *view)
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen{
dc1bc1685e4a0d58ae7bacaecc282d0ebde2d7daTimo Sirainen const struct mail_transaction_header *hdr;
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen const void *data;
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen bool have_expunges = FALSE;
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen int ret;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen mail_transaction_log_view_mark(view->log_view);
5733207dc3ec10e6e5a6e0a8b30fbd1b061062b9Timo Sirainen
5733207dc3ec10e6e5a6e0a8b30fbd1b061062b9Timo Sirainen while ((ret = mail_transaction_log_view_next(view->log_view,
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen &hdr, &data)) > 0) {
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen if ((hdr->type & MAIL_TRANSACTION_EXTERNAL) == 0) {
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen /* skip expunge requests */
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen continue;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen }
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen if ((hdr->type & MAIL_TRANSACTION_EXPUNGE_GUID) != 0) {
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen /* we have an expunge. see if it still exists. */
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen if (have_existing_expunges(view, data, hdr->size)) {
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen have_expunges = TRUE;
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen break;
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen }
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen } else if ((hdr->type & MAIL_TRANSACTION_EXPUNGE) != 0) {
5733207dc3ec10e6e5a6e0a8b30fbd1b061062b9Timo Sirainen /* we have an expunge. see if it still exists. */
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen if (have_existing_guid_expunge(view, data, hdr->size)) {
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen have_expunges = TRUE;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen break;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen }
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen }
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen }
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen mail_transaction_log_view_rewind(view->log_view);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen /* handle failures as having expunges (which is safer).
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen we'll probably fail later. */
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen return ret < 0 || have_expunges;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen}
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainenstatic int uint_cmp(const void *p1, const void *p2)
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen{
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen const unsigned int *u1 = p1, *u2 = p2;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
5733207dc3ec10e6e5a6e0a8b30fbd1b061062b9Timo Sirainen if (*u1 < *u2)
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen return -1;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen if (*u1 > *u2)
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen return 1;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen return 0;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen}
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainenstatic bool view_sync_lost_keywords_equal(struct mail_index_view_sync_ctx *ctx)
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen{
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen unsigned int *old_idx, *new_idx;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen unsigned int old_count, new_count;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen old_idx = array_get_modifiable(&ctx->lost_old_kw, &old_count);
b6b9c99fefbbc662bd9a0006566133c4480bf0e8Timo Sirainen new_idx = array_get_modifiable(&ctx->lost_new_kw, &new_count);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen if (old_count != new_count)
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen return FALSE;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen qsort(old_idx, old_count, sizeof(*old_idx), uint_cmp);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen qsort(new_idx, new_count, sizeof(*new_idx), uint_cmp);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen return memcmp(old_idx, new_idx, old_count * sizeof(old_idx)) == 0;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen}
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainenstatic int view_sync_update_keywords(struct mail_index_view_sync_ctx *ctx,
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen uint32_t uid)
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen{
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen struct mail_transaction_header thdr;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen struct mail_transaction_keyword_update kw_up;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen const unsigned int *kw_idx;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen const char *const *kw_names;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen unsigned int i, count;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen kw_idx = array_get(&ctx->lost_new_kw, &count);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen if (count == 0)
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen return 0;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen kw_names = array_idx(&ctx->view->index->keywords, 0);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen memset(&thdr, 0, sizeof(thdr));
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen thdr.type = MAIL_TRANSACTION_KEYWORD_UPDATE | MAIL_TRANSACTION_EXTERNAL;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen memset(&kw_up, 0, sizeof(kw_up));
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen kw_up.modify_type = MODIFY_ADD;
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen /* add new flags one by one */
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen for (i = 0; i < count; i++) {
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen kw_up.name_size = strlen(kw_names[kw_idx[i]]);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen buffer_set_used_size(ctx->lost_kw_buf, 0);
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen buffer_append(ctx->lost_kw_buf, &kw_up, sizeof(kw_up));
2670cd577aa57eb9f915a4f4220ae48c9b4fc5fbTimo Sirainen buffer_append(ctx->lost_kw_buf, kw_names[kw_idx[i]],
kw_up.name_size);
if (ctx->lost_kw_buf->used % 4 != 0) {
buffer_append_zero(ctx->lost_kw_buf,
4 - ctx->lost_kw_buf->used % 4);
}
buffer_append(ctx->lost_kw_buf, &uid, sizeof(uid));
buffer_append(ctx->lost_kw_buf, &uid, sizeof(uid));
thdr.size = ctx->lost_kw_buf->used;
if (mail_index_sync_record(&ctx->sync_map_ctx, &thdr,
ctx->lost_kw_buf->data) < 0)
return -1;
}
return 0;
}
static int view_sync_apply_lost_changes(struct mail_index_view_sync_ctx *ctx,
uint32_t old_seq, uint32_t new_seq)
{
struct mail_index_map *old_map = ctx->view->map;
struct mail_index_map *new_map = ctx->view->index->map;
const struct mail_index_record *old_rec, *new_rec;
struct mail_transaction_header thdr;
const struct mail_index_ext *ext;
const uint64_t *modseqp;
uint64_t new_modseq;
bool changed = FALSE;
old_rec = MAIL_INDEX_MAP_IDX(old_map, old_seq - 1);
new_rec = MAIL_INDEX_MAP_IDX(new_map, new_seq - 1);
memset(&thdr, 0, sizeof(thdr));
if (old_rec->flags != new_rec->flags) {
struct mail_transaction_flag_update flag_update;
/* check this before syncing the record, since it updates
old_rec. */
if ((old_rec->flags & MAIL_INDEX_FLAGS_MASK) !=
(new_rec->flags & MAIL_INDEX_FLAGS_MASK))
changed = TRUE;
thdr.type = MAIL_TRANSACTION_FLAG_UPDATE |
MAIL_TRANSACTION_EXTERNAL;
thdr.size = sizeof(flag_update);
memset(&flag_update, 0, sizeof(flag_update));
flag_update.uid1 = flag_update.uid2 = new_rec->uid;
flag_update.add_flags = new_rec->flags;
flag_update.remove_flags = ~new_rec->flags & 0xff;
if (mail_index_sync_record(&ctx->sync_map_ctx, &thdr,
&flag_update) < 0)
return -1;
}
mail_index_map_lookup_keywords(old_map, old_seq, &ctx->lost_old_kw);
mail_index_map_lookup_keywords(new_map, new_seq, &ctx->lost_new_kw);
if (!view_sync_lost_keywords_equal(ctx)) {
struct mail_transaction_keyword_reset kw_reset;
thdr.type = MAIL_TRANSACTION_KEYWORD_RESET |
MAIL_TRANSACTION_EXTERNAL;
thdr.size = sizeof(kw_reset);
/* remove all old flags by resetting them */
memset(&kw_reset, 0, sizeof(kw_reset));
kw_reset.uid1 = kw_reset.uid2 = new_rec->uid;
if (mail_index_sync_record(&ctx->sync_map_ctx, &thdr,
&kw_reset) < 0)
return -1;
view_sync_update_keywords(ctx, new_rec->uid);
changed = TRUE;
}
if (changed) {
/* flags or keywords changed */
} else if (ctx->view->highest_modseq != 0 &&
ctx->lost_new_ext_idx != (uint32_t)-1) {
/* if modseq has changed include this message in changed flags
list, even if we didn't see any changes above. */
ext = array_idx(&new_map->extensions, ctx->lost_new_ext_idx);
modseqp = CONST_PTR_OFFSET(new_rec, ext->record_offset);
new_modseq = *modseqp;
if (new_modseq > ctx->view->highest_modseq)
changed = TRUE;
}
/* without modseqs lost_flags isn't updated perfectly correctly, because
by the time we're comparing old flags it may have changed from what
we last sent to the client (because the map is shared). This could
be avoided by always keeping a private copy of the map in the view,
but that's a waste of memory for as rare of a problem as this. */
if (changed)
seq_range_array_add(&ctx->lost_flags, 0, new_rec->uid);
return 0;
}
static int
view_sync_get_log_lost_changes(struct mail_index_view_sync_ctx *ctx,
unsigned int *expunge_count_r)
{
struct mail_index_view *view = ctx->view;
struct mail_index_map *old_map = view->map;
struct mail_index_map *new_map = view->index->map;
const unsigned int old_count = old_map->hdr.messages_count;
const unsigned int new_count = new_map->hdr.messages_count;
const struct mail_index_record *old_rec, *new_rec;
struct mail_transaction_header thdr;
unsigned int i, j;
/* we don't update the map in the same order as it's typically done.
map->rec_map may already have some messages appended that we don't
want. get an atomic map to make sure these get removed. */
(void)mail_index_sync_get_atomic_map(&ctx->sync_map_ctx);
if (!mail_index_map_get_ext_idx(new_map, view->index->modseq_ext_id,
&ctx->lost_new_ext_idx))
ctx->lost_new_ext_idx = (uint32_t)-1;
i_array_init(&ctx->lost_flags, 64);
t_array_init(&ctx->lost_old_kw, 32);
t_array_init(&ctx->lost_new_kw, 32);
ctx->lost_kw_buf = buffer_create_dynamic(pool_datastack_create(), 128);
/* handle expunges and sync flags */
i = j = 0;
while (i < old_count && j < new_count) {
old_rec = MAIL_INDEX_MAP_IDX(old_map, i);
new_rec = MAIL_INDEX_MAP_IDX(new_map, j);
if (old_rec->uid == new_rec->uid) {
/* message found - check if flags have changed */
if (view_sync_apply_lost_changes(ctx, i + 1, j + 1) < 0)
return -1;
i++; j++;
} else if (old_rec->uid < new_rec->uid) {
/* message expunged */
seq_range_array_add(&ctx->expunges, 0, old_rec->uid);
i++;
} else {
/* new message appeared out of nowhere */
mail_index_set_error(view->index,
"%s view is inconsistent: "
"uid=%u inserted in the middle of mailbox",
view->index->filepath, new_rec->uid);
return -1;
}
}
/* if there are old messages left, they're all expunged */
for (; i < old_count; i++) {
old_rec = MAIL_INDEX_MAP_IDX(old_map, i);
seq_range_array_add(&ctx->expunges, 0, old_rec->uid);
}
/* if there are new messages left, they're all new messages */
thdr.type = MAIL_TRANSACTION_APPEND | MAIL_TRANSACTION_EXTERNAL;
thdr.size = sizeof(*new_rec);
for (; j < new_count; j++) {
new_rec = MAIL_INDEX_MAP_IDX(new_map, j);
if (mail_index_sync_record(&ctx->sync_map_ctx,
&thdr, new_rec) < 0)
return -1;
mail_index_map_lookup_keywords(new_map, j + 1,
&ctx->lost_new_kw);
view_sync_update_keywords(ctx, new_rec->uid);
}
*expunge_count_r = view_sync_expunges2seqs(ctx);
/* we have no idea how far we've synced - make sure these aren't used */
old_map->hdr.log_file_seq = 0;
old_map->hdr.log_file_head_offset = 0;
old_map->hdr.log_file_tail_offset = 0;
if ((ctx->flags & MAIL_INDEX_VIEW_SYNC_FLAG_NOEXPUNGES) != 0) {
array_clear(&ctx->expunges);
ctx->skipped_expunges = *expunge_count_r > 0;
} else {
view->log_file_head_seq = new_map->hdr.log_file_seq;
view->log_file_head_offset = new_map->hdr.log_file_head_offset;
}
return 0;
}
static int mail_index_view_sync_init_fix(struct mail_index_view_sync_ctx *ctx)
{
struct mail_index_view *view = ctx->view;
uint32_t seq;
uoff_t offset;
bool reset;
/* replace the view's map */
view->index->map->refcount++;
mail_index_unmap(&view->map);
view->map = view->index->map;
/* update log positions */
view->log_file_head_seq = seq = view->map->hdr.log_file_seq;
view->log_file_head_offset = offset =
view->map->hdr.log_file_head_offset;
if (mail_transaction_log_view_set(view->log_view, seq, offset,
seq, offset, &reset) < 0)
return -1;
view->inconsistent = FALSE;
return 0;
}
struct mail_index_view_sync_ctx *
mail_index_view_sync_begin(struct mail_index_view *view,
enum mail_index_view_sync_flags flags)
{
struct mail_index_view_sync_ctx *ctx;
struct mail_index_map *map;
unsigned int expunge_count = 0;
bool reset, sync_expunges, have_expunges;
int ret;
i_assert(!view->syncing);
i_assert(view->transactions == 0);
view->syncing = TRUE;
/* Syncing the view invalidates all previous looked up records.
Unreference the mappings this view keeps because of them. */
mail_index_view_unref_maps(view);
ctx = i_new(struct mail_index_view_sync_ctx, 1);
ctx->view = view;
ctx->flags = flags;
sync_expunges = (flags & MAIL_INDEX_VIEW_SYNC_FLAG_NOEXPUNGES) == 0;
if (sync_expunges)
i_array_init(&ctx->expunges, 64);
if ((flags & MAIL_INDEX_VIEW_SYNC_FLAG_FIX_INCONSISTENT) != 0) {
/* just get this view synced - don't return anything */
i_assert(sync_expunges);
if (mail_index_view_sync_init_fix(ctx) < 0)
ctx->failed = TRUE;
return ctx;
}
if (mail_index_view_is_inconsistent(view)) {
mail_index_set_error(view->index, "%s view is inconsistent",
view->index->filepath);
ctx->failed = TRUE;
return ctx;
}
ret = view_sync_set_log_view_range(view, sync_expunges, &reset);
if (ret < 0) {
ctx->failed = TRUE;
return ctx;
}
if (ret == 0) {
ctx->log_was_lost = TRUE;
if (!sync_expunges)
i_array_init(&ctx->expunges, 64);
mail_index_sync_map_init(&ctx->sync_map_ctx, view,
MAIL_INDEX_SYNC_HANDLER_VIEW);
ret = view_sync_get_log_lost_changes(ctx, &expunge_count);
mail_index_modseq_sync_end(&ctx->sync_map_ctx.modseq_ctx);
mail_index_sync_map_deinit(&ctx->sync_map_ctx);
if (ret < 0) {
mail_index_set_error(view->index,
"%s view syncing failed to apply changes",
view->index->filepath);
view->inconsistent = TRUE;
ctx->failed = TRUE;
return ctx;
}
have_expunges = expunge_count > 0;
} else if (sync_expunges) {
/* get list of all expunges first */
if (view_sync_get_expunges(ctx, &expunge_count) < 0) {
ctx->failed = TRUE;
return ctx;
}
have_expunges = expunge_count > 0;
} else {
have_expunges = view_sync_have_expunges(view);
}
ctx->finish_min_msg_count = reset ? 0 :
view->map->hdr.messages_count - expunge_count;
if (reset && view->map->hdr.messages_count > 0) {
view->inconsistent = TRUE;
mail_index_set_error(view->index,
"%s reset, view is now inconsistent",
view->index->filepath);
}
if (!have_expunges) {
/* no expunges, we can just replace the map */
if (view->index->map->hdr.messages_count <
ctx->finish_min_msg_count) {
mail_index_set_error(view->index,
"Index %s lost messages without expunging "
"(%u -> %u)", view->index->filepath,
view->map->hdr.messages_count,
view->index->map->hdr.messages_count);
ctx->finish_min_msg_count = 0;
view->inconsistent = TRUE;
}
view->index->map->refcount++;
mail_index_unmap(&view->map);
view->map = view->index->map;
} else {
/* expunges seen. create a private map which we update.
if we're syncing expunges the map will finally be replaced
with the head map to remove the expunged messages. */
ctx->sync_map_update = TRUE;
if (view->map->refcount > 1) {
map = mail_index_map_clone(view->map);
mail_index_unmap(&view->map);
view->map = map;
} else {
map = view->map;
}
if (sync_expunges) {
ctx->sync_new_map = view->index->map;
ctx->sync_new_map->refcount++;
}
}
mail_index_sync_map_init(&ctx->sync_map_ctx, view,
MAIL_INDEX_SYNC_HANDLER_VIEW);
#ifdef DEBUG
mail_index_map_check(view->map);
#endif
return ctx;
}
static bool
view_sync_is_hidden(struct mail_index_view *view, uint32_t seq, uoff_t offset)
{
const struct mail_index_view_log_sync_area *syncs;
unsigned int i, count;
if (!array_is_created(&view->syncs_hidden))
return FALSE;
syncs = array_get(&view->syncs_hidden, &count);
for (i = 0; i < count; i++) {
if (syncs[i].log_file_offset <= offset &&
offset - syncs[i].log_file_offset < syncs[i].length &&
syncs[i].log_file_seq == seq)
return TRUE;
}
return FALSE;
}
static bool
mail_index_view_sync_want(struct mail_index_view_sync_ctx *ctx,
const struct mail_transaction_header *hdr)
{
struct mail_index_view *view = ctx->view;
uint32_t seq;
uoff_t offset, next_offset;
mail_transaction_log_view_get_prev_pos(view->log_view, &seq, &offset);
next_offset = offset + sizeof(*hdr) + hdr->size;
if ((hdr->type & (MAIL_TRANSACTION_EXPUNGE |
MAIL_TRANSACTION_EXPUNGE_GUID)) != 0 &&
(hdr->type & MAIL_TRANSACTION_EXTERNAL) != 0) {
if ((ctx->flags & MAIL_INDEX_VIEW_SYNC_FLAG_NOEXPUNGES) != 0) {
i_assert(!LOG_IS_BEFORE(seq, offset,
view->log_file_expunge_seq,
view->log_file_expunge_offset));
if (!ctx->skipped_expunges) {
view->log_file_expunge_seq = seq;
view->log_file_expunge_offset = offset;
ctx->skipped_expunges = TRUE;
}
return FALSE;
}
if (LOG_IS_BEFORE(seq, offset, view->log_file_expunge_seq,
view->log_file_expunge_offset)) {
/* already synced */
return FALSE;
}
}
if (LOG_IS_BEFORE(seq, offset, view->log_file_head_seq,
view->log_file_head_offset)) {
/* already synced */
return FALSE;
}
view->log_file_head_seq = seq;
view->log_file_head_offset = next_offset;
return TRUE;
}
static int
mail_index_view_sync_get_next_transaction(struct mail_index_view_sync_ctx *ctx)
{
struct mail_transaction_log_view *log_view = ctx->view->log_view;
struct mail_index_view *view = ctx->view;
const struct mail_transaction_header *hdr;
uint32_t seq;
uoff_t offset;
int ret;
bool synced_to_map;
do {
/* Get the next transaction from log. */
ret = mail_transaction_log_view_next(log_view, &ctx->hdr,
&ctx->data);
if (ret <= 0) {
if (ret < 0)
return -1;
ctx->hdr = NULL;
ctx->last_read = TRUE;
return 0;
}
hdr = ctx->hdr;
/* skip records we've already synced */
} while (!mail_index_view_sync_want(ctx, hdr));
mail_transaction_log_view_get_prev_pos(log_view, &seq, &offset);
/* If we started from a map that we didn't create ourself,
some of the transactions may already be synced. at the end
of this view sync we'll update file_seq=0 so that this check
always becomes FALSE for subsequent syncs. */
synced_to_map = view->map->hdr.log_file_seq != 0 &&
LOG_IS_BEFORE(seq, offset, view->map->hdr.log_file_seq,
view->map->hdr.log_file_head_offset);
/* Apply transaction to view's mapping if needed (meaning we
didn't just re-map the view to head mapping). */
if (ctx->sync_map_update && !synced_to_map) {
if ((hdr->type & (MAIL_TRANSACTION_EXPUNGE |
MAIL_TRANSACTION_EXPUNGE_GUID)) == 0) {
T_BEGIN {
ret = mail_index_sync_record(&ctx->sync_map_ctx,
hdr, ctx->data);
} T_END;
}
if (ret < 0)
return -1;
}
ctx->hidden = view_sync_is_hidden(view, seq, offset);
return 1;
}
static bool
mail_index_view_sync_get_rec(struct mail_index_view_sync_ctx *ctx,
struct mail_index_view_sync_rec *rec)
{
const struct mail_transaction_header *hdr = ctx->hdr;
const void *data = ctx->data;
switch (hdr->type & MAIL_TRANSACTION_TYPE_MASK) {
case MAIL_TRANSACTION_FLAG_UPDATE: {
const struct mail_transaction_flag_update *update =
CONST_PTR_OFFSET(data, ctx->data_offset);
/* data contains mail_transaction_flag_update[] */
for (;;) {
ctx->data_offset += sizeof(*update);
if (!MAIL_TRANSACTION_FLAG_UPDATE_IS_INTERNAL(update))
break;
/* skip internal flag changes */
if (ctx->data_offset == ctx->hdr->size)
return 0;
update = CONST_PTR_OFFSET(data, ctx->data_offset);
}
rec->type = MAIL_INDEX_VIEW_SYNC_TYPE_FLAGS;
rec->uid1 = update->uid1;
rec->uid2 = update->uid2;
break;
}
case MAIL_TRANSACTION_KEYWORD_UPDATE: {
const struct mail_transaction_keyword_update *update = data;
const uint32_t *uids;
/* data contains mail_transaction_keyword_update header,
the keyword name and an array of { uint32_t uid1, uid2; } */
if (ctx->data_offset == 0) {
/* skip over the header and name */
ctx->data_offset = sizeof(*update) + update->name_size;
if ((ctx->data_offset % 4) != 0)
ctx->data_offset += 4 - (ctx->data_offset % 4);
}
uids = CONST_PTR_OFFSET(data, ctx->data_offset);
rec->type = MAIL_INDEX_VIEW_SYNC_TYPE_FLAGS;
rec->uid1 = uids[0];
rec->uid2 = uids[1];
ctx->data_offset += sizeof(uint32_t) * 2;
break;
}
case MAIL_TRANSACTION_KEYWORD_RESET: {
const struct mail_transaction_keyword_reset *reset =
CONST_PTR_OFFSET(data, ctx->data_offset);
/* data contains mail_transaction_keyword_reset[] */
rec->type = MAIL_INDEX_VIEW_SYNC_TYPE_FLAGS;
rec->uid1 = reset->uid1;
rec->uid2 = reset->uid2;
ctx->data_offset += sizeof(*reset);
break;
}
default:
ctx->hdr = NULL;
return FALSE;
}
rec->hidden = ctx->hidden;
return TRUE;
}
static bool
mail_index_view_sync_next_lost(struct mail_index_view_sync_ctx *ctx,
struct mail_index_view_sync_rec *sync_rec)
{
const struct seq_range *range;
unsigned int count;
range = array_get(&ctx->lost_flags, &count);
if (ctx->lost_flag_idx == count) {
ctx->last_read = TRUE;
return FALSE;
}
sync_rec->type = MAIL_INDEX_VIEW_SYNC_TYPE_FLAGS;
sync_rec->uid1 = range[ctx->lost_flag_idx].seq1;
sync_rec->uid2 = range[ctx->lost_flag_idx].seq2;
ctx->lost_flag_idx++;
return TRUE;
}
bool mail_index_view_sync_next(struct mail_index_view_sync_ctx *ctx,
struct mail_index_view_sync_rec *sync_rec)
{
int ret;
if (ctx->log_was_lost)
return mail_index_view_sync_next_lost(ctx, sync_rec);
do {
if (ctx->hdr == NULL || ctx->data_offset == ctx->hdr->size) {
ret = mail_index_view_sync_get_next_transaction(ctx);
if (ret <= 0) {
if (ret < 0)
ctx->failed = TRUE;
return FALSE;
}
ctx->data_offset = 0;
}
} while (!mail_index_view_sync_get_rec(ctx, sync_rec));
return TRUE;
}
void mail_index_view_sync_get_expunges(struct mail_index_view_sync_ctx *ctx,
const ARRAY_TYPE(seq_range) **expunges_r)
{
*expunges_r = &ctx->expunges;
}
static void
mail_index_view_sync_clean_log_syncs(struct mail_index_view *view)
{
const struct mail_index_view_log_sync_area *syncs;
unsigned int i, count;
if (!array_is_created(&view->syncs_hidden))
return;
/* Clean up to view's tail */
syncs = array_get(&view->syncs_hidden, &count);
for (i = 0; i < count; i++) {
if ((syncs[i].log_file_offset +
syncs[i].length > view->log_file_expunge_offset &&
syncs[i].log_file_seq == view->log_file_expunge_seq) ||
syncs[i].log_file_seq > view->log_file_expunge_seq)
break;
}
if (i > 0)
array_delete(&view->syncs_hidden, 0, i);
}
int mail_index_view_sync_commit(struct mail_index_view_sync_ctx **_ctx,
bool *delayed_expunges_r)
{
struct mail_index_view_sync_ctx *ctx = *_ctx;
struct mail_index_view *view = ctx->view;
int ret = ctx->failed ? -1 : 0;
i_assert(view->syncing);
*_ctx = NULL;
*delayed_expunges_r = ctx->skipped_expunges;
if ((!ctx->last_read || view->inconsistent) &&
(ctx->flags & MAIL_INDEX_VIEW_SYNC_FLAG_FIX_INCONSISTENT) == 0) {
/* we didn't sync everything */
view->inconsistent = TRUE;
ret = -1;
}
if (ctx->sync_map_ctx.modseq_ctx != NULL)
mail_index_modseq_sync_end(&ctx->sync_map_ctx.modseq_ctx);
if (ctx->sync_new_map != NULL) {
mail_index_unmap(&view->map);
view->map = ctx->sync_new_map;
} else if (ctx->sync_map_update) {
/* log offsets have no meaning in views. make sure they're not
tried to be used wrong by setting them to zero. */
view->map->hdr.log_file_seq = 0;
view->map->hdr.log_file_head_offset = 0;
view->map->hdr.log_file_tail_offset = 0;
}
i_assert(view->map->hdr.messages_count >= ctx->finish_min_msg_count);
if (!ctx->skipped_expunges) {
view->log_file_expunge_seq = view->log_file_head_seq;
view->log_file_expunge_offset = view->log_file_head_offset;
}
if (ctx->sync_map_ctx.view != NULL)
mail_index_sync_map_deinit(&ctx->sync_map_ctx);
mail_index_view_sync_clean_log_syncs(ctx->view);
#ifdef DEBUG
mail_index_map_check(view->map);
#endif
/* set log view to empty range so unneeded memory gets freed */
mail_transaction_log_view_clear(view->log_view,
view->log_file_expunge_seq);
if (array_is_created(&ctx->expunges))
array_free(&ctx->expunges);
if (array_is_created(&ctx->lost_flags))
array_free(&ctx->lost_flags);
view->highest_modseq = mail_index_map_modseq_get_highest(view->map);
view->syncing = FALSE;
i_free(ctx);
return ret;
}
void mail_index_view_add_hidden_transaction(struct mail_index_view *view,
uint32_t log_file_seq,
uoff_t log_file_offset,
unsigned int length)
{
struct mail_index_view_log_sync_area *area;
if (!array_is_created(&view->syncs_hidden))
i_array_init(&view->syncs_hidden, 32);
area = array_append_space(&view->syncs_hidden);
area->log_file_seq = log_file_seq;
area->log_file_offset = log_file_offset;
area->length = length;
}