mail-index-view-sync.c revision e06c0b65c16ccce69bbee009ead14d7d3d17a256
45312f52ff3a3d4c137447be4c7556500c2f8bf2Timo Sirainen/* Copyright (C) 2003-2004 Timo Sirainen */
0cb2e8eb55e70f8ebe1e8349bdf49e4cbe5d8834Timo Sirainenview_sync_get_expunges(struct mail_index_view *view, buffer_t **expunges_r)
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen struct mail_transaction_expunge *src, *src_end, *dest;
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen *expunges_r = buffer_create_dynamic(default_pool, 512, (size_t)-1);
0cb2e8eb55e70f8ebe1e8349bdf49e4cbe5d8834Timo Sirainen /* with mask 0 we don't get anything, we'll just read the expunges
c979eeda1f46483d9c963e265786b701d7683d77Timo Sirainen while seeking to end */
5a58037ad75b88356d82240fab2bc604de03107eTimo Sirainen if (mail_transaction_log_view_set(view->log_view,
2a6af811ea3de3cf9e2f15e446674dd21b0705f3Timo Sirainen while ((ret = mail_transaction_log_view_next(view->log_view,
94a8cb0ee1d85569ad1a2acacd92d3ce22f8a1cbTimo Sirainen mail_transaction_log_sort_expunges(*expunges_r,
b032dc80e358f09893f09999f172ff12f5dbbb8eTimo Sirainen /* convert to sequences */
b032dc80e358f09893f09999f172ff12f5dbbb8eTimo Sirainen src = dest = buffer_get_modifyable_data(*expunges_r, &size);
b032dc80e358f09893f09999f172ff12f5dbbb8eTimo Sirainen ret = mail_index_lookup_uid_range(view, src->uid1,
b032dc80e358f09893f09999f172ff12f5dbbb8eTimo Sirainen mail_transaction_log_view_unset(view->log_view);
b032dc80e358f09893f09999f172ff12f5dbbb8eTimo Sirainenint mail_index_view_sync_begin(struct mail_index_view *view,
b032dc80e358f09893f09999f172ff12f5dbbb8eTimo Sirainen /* We must sync flags as long as view is mmap()ed, as the flags may
94a8cb0ee1d85569ad1a2acacd92d3ce22f8a1cbTimo Sirainen have already changed under us. */
94a8cb0ee1d85569ad1a2acacd92d3ce22f8a1cbTimo Sirainen i_assert((sync_mask & MAIL_INDEX_SYNC_TYPE_FLAGS) != 0);
b032dc80e358f09893f09999f172ff12f5dbbb8eTimo Sirainen if (mail_index_view_lock_head(view, TRUE) < 0)
b032dc80e358f09893f09999f172ff12f5dbbb8eTimo Sirainen if ((sync_mask & MAIL_INDEX_SYNC_TYPE_EXPUNGE) != 0) {
b032dc80e358f09893f09999f172ff12f5dbbb8eTimo Sirainen /* get list of all expunges first */
b032dc80e358f09893f09999f172ff12f5dbbb8eTimo Sirainen if (view_sync_get_expunges(view, &expunges) < 0)
b032dc80e358f09893f09999f172ff12f5dbbb8eTimo Sirainen mask = mail_transaction_type_mask_get(sync_mask);
8d80659e504ffb34bb0c6a633184fece35751b18Timo Sirainen if (mail_transaction_log_view_set(view->log_view,
a050ca9def13949dbaa67bd6574a41c4f397ae26Timo Sirainen ctx = i_new(struct mail_index_view_sync_ctx, 1);
a050ca9def13949dbaa67bd6574a41c4f397ae26Timo Sirainen if ((sync_mask & MAIL_INDEX_SYNC_TYPE_EXPUNGE) != 0) {
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen /* keep the old mapping without expunges until we're
ae8817f05005f57bba32479a610b52d083e2b6ebTimo Sirainen fully synced */
ae8817f05005f57bba32479a610b52d083e2b6ebTimo Sirainen /* we need a private copy of the map if we don't want to
ae8817f05005f57bba32479a610b52d083e2b6ebTimo Sirainen sync expunges */
ae8817f05005f57bba32479a610b52d083e2b6ebTimo Sirainenstatic int view_is_transaction_synced(struct mail_index_view *view,
ae8817f05005f57bba32479a610b52d083e2b6ebTimo Sirainen data = buffer_get_data(view->log_syncs, &size);
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen *((const uint32_t *)(data + sizeof(uoff_t))) == seq)
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainenstatic int mail_index_view_sync_next_trans(struct mail_index_view_sync_ctx *ctx,
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen struct mail_transaction_log_view *log_view = ctx->view->log_view;
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen ret = mail_transaction_log_view_next(log_view, &ctx->hdr, &ctx->data,
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen mail_transaction_log_view_get_prev_pos(log_view, seq_r, offset_r);
659fe5d24825b160cae512538088020d97a60239Timo Sirainen /* skip flag changes that we committed ourself or have already synced */
659fe5d24825b160cae512538088020d97a60239Timo Sirainen if (view_is_transaction_synced(view, *seq_r, *offset_r))
5cdd348121e62a6244ba2f93db781731f7129a71Timo Sirainen /* expunges have to be synced afterwards so that caller can still get
5cdd348121e62a6244ba2f93db781731f7129a71Timo Sirainen information of the messages. otherwise caller most likely wants to
b43bb773ee6534c1013b01a62fbd5703e3b0d17dTimo Sirainen see only updated information. */
5f78b33aa505b17e23cdf27b071a24e127b3db54Timo Sirainen (ctx->hdr->type & MAIL_TRANSACTION_EXPUNGE) == 0) {
2a6af811ea3de3cf9e2f15e446674dd21b0705f3Timo Sirainen (((u)->add_flags | (u)->remove_flags) == MAIL_INDEX_MAIL_FLAG_DIRTY && \
2a6af811ea3de3cf9e2f15e446674dd21b0705f3Timo Sirainen memcmp((u)->add_keywords, empty, INDEX_KEYWORDS_BYTE_COUNT) == 0 && \
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen memcmp((u)->add_keywords, empty, INDEX_KEYWORDS_BYTE_COUNT) == 0)
b43bb773ee6534c1013b01a62fbd5703e3b0d17dTimo Sirainenmail_index_view_sync_get_rec(struct mail_index_view_sync_ctx *ctx,
7b3bf1de3fa7eee2185d1e404812b50c295e2b93Timo Sirainen static keywords_mask_t empty_keywords = { 0, };
0cb2e8eb55e70f8ebe1e8349bdf49e4cbe5d8834Timo Sirainen const struct mail_transaction_header *hdr = ctx->hdr;
88187ee880b4829443e0d55ea7d145d9d5880217Timo Sirainen switch (hdr->type & MAIL_TRANSACTION_TYPE_MASK) {
ba94d6d7e2f7b07d51f5c27e7532f6502ac9a298Timo Sirainen const struct mail_transaction_flag_update *update =
95a1a5195d56f3cf5d1e529aad668f87ad3b979bTimo Sirainen if (!FLAG_UPDATE_IS_INTERNAL(update, empty_keywords))
88187ee880b4829443e0d55ea7d145d9d5880217Timo Sirainenint mail_index_view_sync_next(struct mail_index_view_sync_ctx *ctx,
fd3f33bdb57170d63aea66ecacc8bea0f0145d6aTimo Sirainen if (ctx->hdr == NULL || ctx->data_offset == ctx->hdr->size) {
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen ret = mail_index_view_sync_next_trans(ctx, &seq,
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen } while (ret == 0);
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen } while (!mail_index_view_sync_get_rec(ctx, sync_rec));
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainenmail_index_view_sync_get_expunges(struct mail_index_view_sync_ctx *ctx,
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainenvoid mail_index_view_sync_end(struct mail_index_view_sync_ctx *ctx)
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen /* we didn't sync everything */
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen if ((ctx->sync_mask & MAIL_INDEX_SYNC_TYPE_APPEND) != 0)
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen view->messages_count = view->map->records_count;
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen mail_transaction_log_view_unset(view->log_view);
b6612c334604eeb27e1ca2bd804ac66dcbc2eaadTimo Sirainenvoid mail_index_view_add_synced_transaction(struct mail_index_view *view,
919875067c26fb261a15b3f7afdb67d7eeddb226Timo Sirainen view->log_syncs = buffer_create_dynamic(default_pool,
20c892309312df8f4f73cfcaf8acd2ededda8b05Timo Sirainen buffer_append(view->log_syncs, &log_file_offset,