maildir-uidlist.c revision 85452efe6f9edd85e2ce45bfc4f198e9cb9b82bd
bcb4e51a409d94ae670de96afb8483a4f7855294Stephan Bosch/* Copyright (C) 2003 Timo Sirainen */
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include "lib.h"
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include "ioloop.h"
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include "array.h"
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include "hash.h"
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include "istream.h"
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include "str.h"
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include "file-dotlock.h"
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include "close-keep-errno.h"
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include "nfs-workarounds.h"
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include "write-full.h"
2b1a3f0eb782068490f90a2c25379b6c0e15b0b1Aki Tuomi#include "maildir-storage.h"
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include "maildir-sync.h"
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include "maildir-filename.h"
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include "maildir-uidlist.h"
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include <stdio.h>
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include <stdlib.h>
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#include <sys/stat.h>
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen/* NFS: How many times to retry reading dovecot-uidlist file if ESTALE
8aa049fd90cd03a66ec43018554bf3ba06815a10Timo Sirainen error occurs in the middle of reading it */
8aa049fd90cd03a66ec43018554bf3ba06815a10Timo Sirainen#define UIDLIST_ESTALE_RETRY_COUNT NFS_ESTALE_RETRY_COUNT
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen/* how many seconds to wait before overriding uidlist.lock */
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#define UIDLIST_LOCK_STALE_TIMEOUT (60*2)
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen#define UIDLIST_IS_LOCKED(uidlist) \
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen ((uidlist)->lock_count > 0)
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainenstruct maildir_uidlist_rec {
c4478af52de63804efef2055580adf1dfc8679c6Timo Sirainen uint32_t uid;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen uint32_t flags;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen char *filename;
477082109e06b912d868a2c05674f44cb880875bTimo Sirainen};
477082109e06b912d868a2c05674f44cb880875bTimo SirainenARRAY_DEFINE_TYPE(maildir_uidlist_rec_p, struct maildir_uidlist_rec *);
0d1b8b6bec79746c5d89d57dd8c1688946bd9237Josef 'Jeff' Sipek
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainenstruct maildir_uidlist {
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen struct maildir_mailbox *mbox;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen char *fname;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen int fd;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen dev_t fd_dev;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen ino_t fd_ino;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen int lock_fd;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen unsigned int lock_count;
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi pool_t record_pool;
661998e2ccd772ad92a9d4a75cb712692a8c94b3Timo Sirainen ARRAY_TYPE(maildir_uidlist_rec_p) records;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen struct hash_table *files;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen unsigned int change_counter;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen struct dotlock_settings dotlock_settings;
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi struct dotlock *dotlock;
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi unsigned int version;
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi unsigned int uid_validity, next_uid, prev_read_uid, last_seen_uid;
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi uint32_t first_recent_uid;
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi unsigned int initial_read:1;
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi unsigned int initial_sync:1;
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi unsigned int need_rewrite:1;
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi unsigned int delayed_rewrite:1;
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi};
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomi
f9258697fe2573b087d09645dfade044dfb27052Aki Tuomistruct maildir_uidlist_sync_ctx {
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen struct maildir_uidlist *uidlist;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen enum maildir_uidlist_sync_flags sync_flags;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen pool_t record_pool;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen ARRAY_TYPE(maildir_uidlist_rec_p) records;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen struct hash_table *files;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen unsigned int first_new_pos;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen unsigned int new_files_count;
9346506a9f4dd9a6285fe8595588e73161849235Timo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen unsigned int partial:1;
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen unsigned int finished:1;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen unsigned int failed:1;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen};
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainenstruct maildir_uidlist_iter_ctx {
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen struct maildir_uidlist *uidlist;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen struct maildir_uidlist_rec *const *next, *const *end;
0c5ed44cf93523f5ae6d118501a79fff6dddf7d7Timo Sirainen
0c5ed44cf93523f5ae6d118501a79fff6dddf7d7Timo Sirainen unsigned int change_counter;
0c5ed44cf93523f5ae6d118501a79fff6dddf7d7Timo Sirainen uint32_t prev_uid;
0c5ed44cf93523f5ae6d118501a79fff6dddf7d7Timo Sirainen};
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainenstatic int maildir_uidlist_lock_timeout(struct maildir_uidlist *uidlist,
ce3a152df54ff6bcb08af5d489a873bd85fcecddTimo Sirainen bool nonblock)
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen{
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen struct maildir_mailbox *mbox = uidlist->mbox;
efe78d3ba24fc866af1c79b9223dc0809ba26cadStephan Bosch const char *path;
9346506a9f4dd9a6285fe8595588e73161849235Timo Sirainen mode_t old_mask;
9346506a9f4dd9a6285fe8595588e73161849235Timo Sirainen int fd;
9346506a9f4dd9a6285fe8595588e73161849235Timo Sirainen
9346506a9f4dd9a6285fe8595588e73161849235Timo Sirainen if (uidlist->lock_count > 0) {
20e04227229970d148801c507946666e2a9bd838Timo Sirainen uidlist->lock_count++;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen return 1;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen }
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen path = t_strconcat(mbox->control_dir, "/" MAILDIR_UIDLIST_NAME, NULL);
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen old_mask = umask(0777 & ~mbox->mail_create_mode);
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen fd = file_dotlock_open(&uidlist->dotlock_settings, path,
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen nonblock ? DOTLOCK_CREATE_FLAG_NONBLOCK : 0,
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen &uidlist->dotlock);
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen umask(old_mask);
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen if (fd == -1) {
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen if (errno == EAGAIN) {
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen mail_storage_set_error(&mbox->storage->storage,
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen MAIL_ERROR_TEMP, MAIL_ERRSTR_LOCK_TIMEOUT);
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen return 0;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen }
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen mail_storage_set_critical(&mbox->storage->storage,
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen "file_dotlock_open(%s) failed: %m", path);
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen return -1;
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen }
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen uidlist->lock_fd = fd;
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen if (mbox->mail_create_gid != (gid_t)-1) {
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen if (fchown(fd, (uid_t)-1, mbox->mail_create_gid) < 0) {
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen mail_storage_set_critical(&mbox->storage->storage,
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen "fchown(%s) failed: %m", path);
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen }
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen }
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen /* our view of uidlist must be up-to-date if we plan on changing it */
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen if (maildir_uidlist_update(uidlist) < 0)
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen return -1;
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen uidlist->lock_count++;
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen return 1;
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen}
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainenint maildir_uidlist_lock(struct maildir_uidlist *uidlist)
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen{
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen return maildir_uidlist_lock_timeout(uidlist, FALSE);
4209bd278ff41fe352e8b993d849a061152829c8Timo Sirainen}
8f7e81b327b8b5bf34262f6755df6d4481760d23Timo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainenint maildir_uidlist_try_lock(struct maildir_uidlist *uidlist)
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen{
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen return maildir_uidlist_lock_timeout(uidlist, TRUE);
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen}
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainenint maildir_uidlist_lock_touch(struct maildir_uidlist *uidlist)
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen{
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen i_assert(UIDLIST_IS_LOCKED(uidlist));
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen return file_dotlock_touch(uidlist->dotlock);
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen}
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainenbool maildir_uidlist_is_locked(struct maildir_uidlist *uidlist)
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen{
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen return UIDLIST_IS_LOCKED(uidlist);
3008b30900f4de3e76f1a6d289444eb694f3d17cTimo Sirainen}
void maildir_uidlist_unlock(struct maildir_uidlist *uidlist)
{
i_assert(uidlist->lock_count > 0);
if (--uidlist->lock_count > 0)
return;
if (!uidlist->delayed_rewrite) {
(void)file_dotlock_delete(&uidlist->dotlock);
} else {
if (file_dotlock_replace(&uidlist->dotlock, 0) <= 0) {
const char *db_path;
db_path = t_strconcat(uidlist->mbox->control_dir,
"/" MAILDIR_UIDLIST_NAME, NULL);
mail_storage_set_critical(
&uidlist->mbox->storage->storage,
"file_dotlock_replace(%s) failed: %m", db_path);
}
uidlist->delayed_rewrite = FALSE;
}
uidlist->lock_fd = -1;
}
struct maildir_uidlist *maildir_uidlist_init(struct maildir_mailbox *mbox)
{
struct maildir_uidlist *uidlist;
uidlist = i_new(struct maildir_uidlist, 1);
uidlist->fd = -1;
uidlist->mbox = mbox;
uidlist->fname =
i_strconcat(mbox->control_dir, "/" MAILDIR_UIDLIST_NAME, NULL);
uidlist->lock_fd = -1;
i_array_init(&uidlist->records, 128);
uidlist->files = hash_create(default_pool, default_pool, 4096,
maildir_filename_base_hash,
maildir_filename_base_cmp);
uidlist->next_uid = 1;
uidlist->dotlock_settings.use_excl_lock =
(mbox->storage->storage.flags &
MAIL_STORAGE_FLAG_DOTLOCK_USE_EXCL) != 0;
uidlist->dotlock_settings.timeout = UIDLIST_LOCK_STALE_TIMEOUT + 2;
uidlist->dotlock_settings.stale_timeout = UIDLIST_LOCK_STALE_TIMEOUT;
uidlist->dotlock_settings.temp_prefix = mbox->storage->temp_prefix;
return uidlist;
}
static void maildir_uidlist_close(struct maildir_uidlist *uidlist)
{
if (uidlist->fd != -1) {
if (close(uidlist->fd) < 0)
i_error("close(%s) failed: %m", uidlist->fname);
uidlist->fd = -1;
uidlist->fd_ino = 0;
}
}
void maildir_uidlist_deinit(struct maildir_uidlist *uidlist)
{
i_assert(!UIDLIST_IS_LOCKED(uidlist));
maildir_uidlist_close(uidlist);
hash_destroy(uidlist->files);
if (uidlist->record_pool != NULL)
pool_unref(uidlist->record_pool);
array_free(&uidlist->records);
i_free(uidlist->fname);
i_free(uidlist);
}
static void
maildir_uidlist_mark_recent(struct maildir_uidlist *uidlist, uint32_t uid)
{
if (uidlist->first_recent_uid == 0 ||
uid < uidlist->first_recent_uid)
uidlist->first_recent_uid = uid;
}
static int maildir_uidlist_next(struct maildir_uidlist *uidlist,
const char *line)
{
struct maildir_uidlist_rec *rec;
uint32_t uid, flags;
uid = flags = 0;
while (*line >= '0' && *line <= '9') {
uid = uid*10 + (*line - '0');
line++;
}
if (uid == 0 || *line != ' ') {
/* invalid file */
mail_storage_set_critical(&uidlist->mbox->storage->storage,
"Invalid data in file %s", uidlist->fname);
return 0;
}
if (uid <= uidlist->prev_read_uid) {
mail_storage_set_critical(&uidlist->mbox->storage->storage,
"UIDs not ordered in file %s (%u > %u)",
uidlist->fname, uid, uidlist->prev_read_uid);
return 0;
}
uidlist->prev_read_uid = uid;
if (uid <= uidlist->last_seen_uid) {
/* we already have this */
return 1;
}
uidlist->last_seen_uid = uid;
if (uid >= uidlist->next_uid) {
mail_storage_set_critical(&uidlist->mbox->storage->storage,
"UID larger than next_uid in file %s (%u >= %u)",
uidlist->fname, uid, uidlist->next_uid);
return 0;
}
while (*line == ' ') line++;
if (uidlist->version == 2) {
/* skip flags parameter */
while (*line != ' ') line++;
while (*line == ' ') line++;
}
if (hash_lookup_full(uidlist->files, line, NULL, NULL)) {
mail_storage_set_critical(&uidlist->mbox->storage->storage,
"Duplicate file in uidlist file %s: %s",
uidlist->fname, line);
return 0;
}
rec = p_new(uidlist->record_pool, struct maildir_uidlist_rec, 1);
rec->uid = uid;
rec->flags = MAILDIR_UIDLIST_REC_FLAG_NONSYNCED;
rec->filename = p_strdup(uidlist->record_pool, line);
hash_insert(uidlist->files, rec->filename, rec);
array_append(&uidlist->records, &rec, 1);
return 1;
}
static int
maildir_uidlist_update_read(struct maildir_uidlist *uidlist,
bool *retry_r, bool try_retry)
{
struct mail_storage *storage = &uidlist->mbox->storage->storage;
const char *line;
unsigned int uid_validity, next_uid;
struct istream *input;
struct stat st;
int fd, ret;
*retry_r = FALSE;
maildir_uidlist_close(uidlist);
fd = nfs_safe_open(uidlist->fname, O_RDONLY);
if (fd == -1) {
if (errno != ENOENT) {
mail_storage_set_critical(storage,
"open(%s) failed: %m", uidlist->fname);
return -1;
}
return 0;
}
if (fstat(fd, &st) < 0) {
close_keep_errno(fd);
if (errno == ESTALE && try_retry) {
*retry_r = TRUE;
return -1;
}
mail_storage_set_critical(storage,
"fstat(%s) failed: %m", uidlist->fname);
return -1;
}
if (uidlist->record_pool == NULL) {
uidlist->record_pool =
pool_alloconly_create(MEMPOOL_GROWING
"uidlist record_pool",
nearest_power(st.st_size -
st.st_size/8));
}
uidlist->version = 0;
input = i_stream_create_file(fd, default_pool, 4096, FALSE);
/* get header */
line = i_stream_read_next_line(input);
if (line == NULL) {
/* I/O error / empty file */
ret = input->stream_errno == 0 ? 0 : -1;
} else if (sscanf(line, "%u %u %u", &uidlist->version,
&uid_validity, &next_uid) != 3 ||
uidlist->version < 1 || uidlist->version > 2) {
/* broken file */
mail_storage_set_critical(storage,
"Corrupted header in file %s (version = %u)",
uidlist->fname, uidlist->version);
ret = 0;
} else if (uid_validity == uidlist->uid_validity &&
next_uid < uidlist->next_uid) {
mail_storage_set_critical(storage,
"%s: next_uid was lowered (%u -> %u)",
uidlist->fname, uidlist->next_uid, next_uid);
ret = 0;
} else if (uid_validity == 0 || next_uid == 0) {
mail_storage_set_critical(storage,
"%s: Broken header (uidvalidity = %u, next_uid=%u)",
uidlist->fname, uid_validity, next_uid);
ret = 0;
} else {
uidlist->uid_validity = uid_validity;
uidlist->next_uid = next_uid;
uidlist->prev_read_uid = 0;
uidlist->change_counter++;
ret = 1;
while ((line = i_stream_read_next_line(input)) != NULL) {
if (!maildir_uidlist_next(uidlist, line)) {
ret = 0;
break;
}
}
if (input->stream_errno != 0)
ret = -1;
}
if (ret == 0) {
/* file is broken */
(void)unlink(uidlist->fname);
} else if (ret > 0) {
/* success */
uidlist->fd = fd;
uidlist->fd_dev = st.st_dev;
uidlist->fd_ino = st.st_ino;
} else {
/* I/O error */
if (input->stream_errno == ESTALE && try_retry)
*retry_r = TRUE;
else {
errno = input->stream_errno;
mail_storage_set_critical(storage,
"read(%s) failed: %m", uidlist->fname);
}
}
i_stream_destroy(&input);
if (ret <= 0) {
if (close(fd) < 0)
i_error("close(%s) failed: %m", uidlist->fname);
}
return ret;
}
int maildir_uidlist_update(struct maildir_uidlist *uidlist)
{
struct mail_storage *storage = &uidlist->mbox->storage->storage;
struct stat st;
unsigned int i;
bool retry;
int ret;
if (uidlist->fd != -1) {
if (nfs_safe_stat(uidlist->fname, &st) < 0) {
if (errno != ENOENT) {
mail_storage_set_critical(storage,
"stat(%s) failed: %m", uidlist->fname);
return -1;
}
return 0;
}
if (st.st_ino == uidlist->fd_ino &&
CMP_DEV_T(st.st_dev, uidlist->fd_dev)) {
/* unchanged */
return 1;
}
}
for (i = 0; ; i++) {
ret = maildir_uidlist_update_read(uidlist, &retry,
i < UIDLIST_ESTALE_RETRY_COUNT);
if (!retry) {
if (ret >= 0)
uidlist->initial_read = TRUE;
break;
}
/* ESTALE - try reopening and rereading */
}
return ret;
}
static const struct maildir_uidlist_rec *
maildir_uidlist_lookup_rec(struct maildir_uidlist *uidlist, uint32_t uid,
unsigned int *idx_r)
{
struct maildir_uidlist_rec *const *recs;
unsigned int idx, left_idx, right_idx;
if (!uidlist->initial_read) {
/* first time we need to read uidlist */
if (maildir_uidlist_update(uidlist) < 0)
return NULL;
}
idx = left_idx = 0;
recs = array_get(&uidlist->records, &right_idx);
while (left_idx < right_idx) {
idx = (left_idx + right_idx) / 2;
if (recs[idx]->uid < uid)
left_idx = idx+1;
else if (recs[idx]->uid > uid)
right_idx = idx;
else {
*idx_r = idx;
return recs[idx];
}
}
if (idx > 0) idx--;
*idx_r = idx;
return NULL;
}
const char *
maildir_uidlist_lookup(struct maildir_uidlist *uidlist, uint32_t uid,
enum maildir_uidlist_rec_flag *flags_r)
{
const struct maildir_uidlist_rec *rec;
unsigned int idx;
rec = maildir_uidlist_lookup_rec(uidlist, uid, &idx);
if (rec == NULL) {
if (uidlist->fd != -1)
return NULL;
/* the uidlist doesn't exist. */
if (maildir_storage_sync_force(uidlist->mbox) < 0)
return NULL;
/* try again */
rec = maildir_uidlist_lookup_rec(uidlist, uid, &idx);
if (rec == NULL)
return NULL;
}
*flags_r = rec->flags;
return rec->filename;
}
bool maildir_uidlist_is_recent(struct maildir_uidlist *uidlist, uint32_t uid)
{
enum maildir_uidlist_rec_flag flags;
if (uidlist->first_recent_uid == 0 || uid < uidlist->first_recent_uid)
return FALSE;
if (maildir_uidlist_lookup(uidlist, uid, &flags) == NULL)
return FALSE;
i_assert(uidlist->first_recent_uid != uid ||
(flags & MAILDIR_UIDLIST_REC_FLAG_RECENT) != 0);
return (flags & MAILDIR_UIDLIST_REC_FLAG_RECENT) != 0;
}
uint32_t maildir_uidlist_get_recent_count(struct maildir_uidlist *uidlist)
{
struct maildir_uidlist_rec *const *recs;
unsigned int idx, count;
uint32_t recent_count;
if (!uidlist->initial_sync) {
/* we haven't synced yet, trust index */
const struct mail_index_header *hdr;
hdr = mail_index_get_header(uidlist->mbox->ibox.view);
return hdr->recent_messages_count;
}
/* all recent messages were in new/ dir, so even if we did only
a partial sync we should know all the recent messages. */
if (uidlist->first_recent_uid == 0)
return 0;
recs = array_get(&uidlist->records, &count);
maildir_uidlist_lookup_rec(uidlist, uidlist->first_recent_uid, &idx);
for (recent_count = 0; idx < count; idx++) {
if ((recs[idx]->flags & MAILDIR_UIDLIST_REC_FLAG_RECENT) != 0)
recent_count++;
}
return recent_count;
}
uint32_t maildir_uidlist_get_uid_validity(struct maildir_uidlist *uidlist)
{
return uidlist->uid_validity;
}
void maildir_uidlist_set_uid_validity(struct maildir_uidlist *uidlist,
uint32_t uid_validity, uint32_t next_uid)
{
uidlist->uid_validity = uid_validity;
/* set next_uid only if we know newer UIDs haven't been added yet */
if (uidlist->next_uid < next_uid)
uidlist->next_uid = next_uid;
}
uint32_t maildir_uidlist_get_next_uid(struct maildir_uidlist *uidlist)
{
return !uidlist->initial_read ? 0 : uidlist->next_uid;
}
static int maildir_uidlist_rewrite_fd(struct maildir_uidlist *uidlist,
const char *temp_path)
{
struct mail_storage *storage = &uidlist->mbox->storage->storage;
struct maildir_uidlist_iter_ctx *iter;
string_t *str;
uint32_t uid;
enum maildir_uidlist_rec_flag flags;
const char *filename;
int ret = 0;
if (uidlist->delayed_rewrite) {
/* already written, truncate */
if (lseek(uidlist->lock_fd, 0, SEEK_SET) < 0) {
mail_storage_set_critical(storage,
"lseek(%s) failed: %m", temp_path);
return -1;
}
if (ftruncate(uidlist->lock_fd, 0) < 0) {
mail_storage_set_critical(storage,
"ftruncate(%s) failed: %m", temp_path);
return -1;
}
}
uidlist->version = 1;
if (uidlist->uid_validity == 0) {
/* Get UIDVALIDITY from index */
const struct mail_index_header *hdr;
hdr = mail_index_get_header(uidlist->mbox->ibox.view);
uidlist->uid_validity = hdr->uid_validity;
}
str = t_str_new(4096);
str_printfa(str, "%u %u %u\n", uidlist->version,
uidlist->uid_validity, uidlist->next_uid);
iter = maildir_uidlist_iter_init(uidlist->mbox->uidlist);
while (maildir_uidlist_iter_next(iter, &uid, &flags, &filename)) {
/* avoid overflowing str buffer so we don't eat more memory
than we need. */
if (str_len(str) + MAX_INT_STRLEN +
strlen(filename) + 5 + 10 >= 4096) {
/* flush buffer */
if (write_full(uidlist->lock_fd,
str_data(str), str_len(str)) < 0) {
mail_storage_set_critical(storage,
"write_full(%s) failed: %m", temp_path);
ret = -1;
break;
}
str_truncate(str, 0);
}
str_printfa(str, "%u %s\n", uid, filename);
}
maildir_uidlist_iter_deinit(iter);
if (ret < 0)
return -1;
if (write_full(uidlist->lock_fd, str_data(str), str_len(str)) < 0) {
mail_storage_set_critical(storage,
"write_full(%s) failed: %m", temp_path);
return -1;
}
if (!uidlist->mbox->ibox.fsync_disable) {
if (fsync(uidlist->lock_fd) < 0) {
mail_storage_set_critical(storage,
"fsync(%s) failed: %m", temp_path);
return -1;
}
}
return 0;
}
static int maildir_uidlist_rewrite(struct maildir_uidlist *uidlist)
{
struct maildir_mailbox *mbox = uidlist->mbox;
const char *temp_path, *db_path;
struct stat st;
int ret;
i_assert(uidlist->lock_count ==
1 + (uidlist->mbox->ibox.keep_locked ? 1 : 0));
temp_path = t_strconcat(mbox->control_dir,
"/" MAILDIR_UIDLIST_NAME ".lock", NULL);
ret = maildir_uidlist_rewrite_fd(uidlist, temp_path);
if (ret == 0 && !uidlist->mbox->ibox.keep_locked) {
db_path = t_strconcat(mbox->control_dir,
"/" MAILDIR_UIDLIST_NAME, NULL);
if (file_dotlock_replace(&uidlist->dotlock,
DOTLOCK_REPLACE_FLAG_DONT_CLOSE_FD) <= 0) {
mail_storage_set_critical(&mbox->storage->storage,
"file_dotlock_replace(%s) failed: %m", db_path);
(void)unlink(temp_path);
(void)close(uidlist->lock_fd);
ret = -1;
} else {
maildir_uidlist_close(uidlist);
uidlist->fd = uidlist->lock_fd;
if (fstat(uidlist->fd, &st) < 0) {
i_error("fstat(%s) failed: %m", uidlist->fname);
uidlist->fd_ino = 0;
} else {
uidlist->fd_dev = st.st_dev;
uidlist->fd_ino = st.st_ino;
}
}
uidlist->lock_fd = -1;
uidlist->lock_count--;
} else {
if (uidlist->mbox->ibox.keep_locked)
uidlist->delayed_rewrite = TRUE;
maildir_uidlist_unlock(uidlist);
}
return ret;
}
static void maildir_uidlist_mark_all(struct maildir_uidlist *uidlist,
bool nonsynced)
{
struct maildir_uidlist_rec **recs;
unsigned int i, count;
recs = array_get_modifiable(&uidlist->records, &count);
if (nonsynced) {
for (i = 0; i < count; i++)
recs[i]->flags |= MAILDIR_UIDLIST_REC_FLAG_NONSYNCED;
} else {
for (i = 0; i < count; i++)
recs[i]->flags &= ~MAILDIR_UIDLIST_REC_FLAG_NONSYNCED;
}
}
int maildir_uidlist_sync_init(struct maildir_uidlist *uidlist,
enum maildir_uidlist_sync_flags sync_flags,
struct maildir_uidlist_sync_ctx **sync_ctx_r)
{
struct maildir_uidlist_sync_ctx *ctx;
int ret;
if ((ret = maildir_uidlist_lock(uidlist)) <= 0)
return ret;
*sync_ctx_r = ctx = i_new(struct maildir_uidlist_sync_ctx, 1);
ctx->uidlist = uidlist;
ctx->sync_flags = sync_flags;
ctx->partial = (sync_flags & MAILDIR_UIDLIST_SYNC_PARTIAL) != 0;
if (ctx->partial) {
/* initially mark all nonsynced */
maildir_uidlist_mark_all(uidlist, TRUE);
return 1;
}
ctx->record_pool = pool_alloconly_create(MEMPOOL_GROWING
"maildir_uidlist_sync", 16384);
ctx->files = hash_create(default_pool, ctx->record_pool, 4096,
maildir_filename_base_hash,
maildir_filename_base_cmp);
i_array_init(&ctx->records, array_count(&uidlist->records));
return 1;
}
static int
maildir_uidlist_sync_next_partial(struct maildir_uidlist_sync_ctx *ctx,
const char *filename,
enum maildir_uidlist_rec_flag flags)
{
struct maildir_uidlist *uidlist = ctx->uidlist;
struct maildir_uidlist_rec *rec;
/* we'll update uidlist directly */
rec = hash_lookup(uidlist->files, filename);
i_assert(rec != NULL || UIDLIST_IS_LOCKED(uidlist));
if (rec == NULL) {
if (ctx->new_files_count == 0)
ctx->first_new_pos = array_count(&uidlist->records);
ctx->new_files_count++;
if (uidlist->record_pool == NULL) {
uidlist->record_pool =
pool_alloconly_create(MEMPOOL_GROWING
"uidlist record_pool",
1024);
}
rec = p_new(uidlist->record_pool,
struct maildir_uidlist_rec, 1);
rec->uid = (uint32_t)-1;
array_append(&uidlist->records, &rec, 1);
uidlist->change_counter++;
}
if ((flags & MAILDIR_UIDLIST_REC_FLAG_RECENT) != 0 &&
rec->uid != (uint32_t)-1)
maildir_uidlist_mark_recent(uidlist, rec->uid);
rec->flags = (rec->flags | flags) & ~MAILDIR_UIDLIST_REC_FLAG_NONSYNCED;
rec->filename = p_strdup(uidlist->record_pool, filename);
hash_insert(uidlist->files, rec->filename, rec);
return 1;
}
int maildir_uidlist_sync_next_pre(struct maildir_uidlist_sync_ctx *ctx,
const char *filename)
{
if (!UIDLIST_IS_LOCKED(ctx->uidlist) &&
hash_lookup(ctx->uidlist->files, filename) == NULL &&
(ctx->partial || hash_lookup(ctx->files, filename) == NULL)) {
if (!ctx->uidlist->initial_read) {
/* first time reading the uidlist */
if (maildir_uidlist_update(ctx->uidlist) < 0) {
ctx->failed = TRUE;
return -1;
}
return maildir_uidlist_sync_next_pre(ctx, filename);
}
return 0;
}
return 1;
}
int maildir_uidlist_sync_next(struct maildir_uidlist_sync_ctx *ctx,
const char *filename,
enum maildir_uidlist_rec_flag flags)
{
struct maildir_uidlist *uidlist = ctx->uidlist;
struct maildir_uidlist_rec *rec, *old_rec;
if (ctx->failed)
return -1;
if (ctx->partial)
return maildir_uidlist_sync_next_partial(ctx, filename, flags);
rec = hash_lookup(ctx->files, filename);
if (rec != NULL) {
if ((rec->flags & (MAILDIR_UIDLIST_REC_FLAG_NEW_DIR |
MAILDIR_UIDLIST_REC_FLAG_MOVED)) == 0) {
/* possibly duplicate */
return 0;
}
rec->flags &= ~(MAILDIR_UIDLIST_REC_FLAG_NEW_DIR |
MAILDIR_UIDLIST_REC_FLAG_MOVED);
} else {
old_rec = hash_lookup(uidlist->files, filename);
i_assert(old_rec != NULL || UIDLIST_IS_LOCKED(uidlist));
rec = p_new(ctx->record_pool, struct maildir_uidlist_rec, 1);
if (old_rec != NULL)
*rec = *old_rec;
else {
rec->uid = (uint32_t)-1;
ctx->new_files_count++;
}
array_append(&ctx->records, &rec, 1);
}
if ((flags & MAILDIR_UIDLIST_REC_FLAG_RECENT) != 0 &&
rec->uid != (uint32_t)-1)
maildir_uidlist_mark_recent(uidlist, rec->uid);
rec->flags = (rec->flags | flags) & ~MAILDIR_UIDLIST_REC_FLAG_NONSYNCED;
rec->filename = p_strdup(ctx->record_pool, filename);
hash_insert(ctx->files, rec->filename, rec);
return 1;
}
const char *
maildir_uidlist_sync_get_full_filename(struct maildir_uidlist_sync_ctx *ctx,
const char *filename)
{
struct maildir_uidlist_rec *rec;
rec = hash_lookup(ctx->files, filename);
return rec == NULL ? NULL : rec->filename;
}
const char *
maildir_uidlist_get_full_filename(struct maildir_uidlist *uidlist,
const char *filename)
{
struct maildir_uidlist_rec *rec;
rec = hash_lookup(uidlist->files, filename);
return rec == NULL ? NULL : rec->filename;
}
static int maildir_time_cmp(const void *p1, const void *p2)
{
const struct maildir_uidlist_rec *const *rec1 = p1, *const *rec2 = p2;
return maildir_filename_sort_cmp((*rec1)->filename, (*rec2)->filename);
}
static void maildir_uidlist_assign_uids(struct maildir_uidlist_sync_ctx *ctx,
unsigned int first_new_pos)
{
struct maildir_uidlist_rec **recs;
unsigned int dest, count;
i_assert(UIDLIST_IS_LOCKED(ctx->uidlist));
recs = array_get_modifiable(&ctx->uidlist->records, &count);
/* sort new files and assign UIDs for them */
qsort(recs + first_new_pos, count - first_new_pos,
sizeof(*recs), maildir_time_cmp);
for (dest = first_new_pos; dest < count; dest++) {
i_assert(recs[dest]->uid == (uint32_t)-1);
recs[dest]->uid = ctx->uidlist->next_uid++;
recs[dest]->flags &= ~MAILDIR_UIDLIST_REC_FLAG_MOVED;
if ((recs[dest]->flags &
MAILDIR_UIDLIST_REC_FLAG_RECENT) != 0) {
maildir_uidlist_mark_recent(ctx->uidlist,
recs[dest]->uid);
}
}
ctx->uidlist->last_seen_uid = ctx->uidlist->next_uid-1;
}
static int maildir_uid_cmp(const void *p1, const void *p2)
{
const struct maildir_uidlist_rec *const *rec1 = p1, *const *rec2 = p2;
return (*rec1)->uid < (*rec2)->uid ? -1 :
(*rec1)->uid > (*rec2)->uid ? 1 : 0;
}
static void maildir_uidlist_swap(struct maildir_uidlist_sync_ctx *ctx)
{
struct maildir_uidlist *uidlist = ctx->uidlist;
struct maildir_uidlist_rec **recs;
unsigned int count;
/* buffer is unsorted, sort it by UID */
recs = array_get_modifiable(&ctx->records, &count);
qsort(recs, count, sizeof(*recs), maildir_uid_cmp);
array_free(&uidlist->records);
uidlist->records = ctx->records;
ctx->records.arr.buffer = NULL;
hash_destroy(uidlist->files);
uidlist->files = ctx->files;
ctx->files = NULL;
if (uidlist->record_pool != NULL)
pool_unref(uidlist->record_pool);
uidlist->record_pool = ctx->record_pool;
ctx->record_pool = NULL;
if (ctx->new_files_count != 0)
maildir_uidlist_assign_uids(ctx, count - ctx->new_files_count);
ctx->uidlist->change_counter++;
}
void maildir_uidlist_sync_finish(struct maildir_uidlist_sync_ctx *ctx)
{
if (ctx->uidlist->uid_validity == 0) {
/* saving a message to a newly created maildir */
ctx->uidlist->uid_validity = ioloop_time;
}
if (!ctx->partial) {
if (!ctx->failed)
maildir_uidlist_swap(ctx);
} else {
if (ctx->new_files_count != 0)
maildir_uidlist_assign_uids(ctx, ctx->first_new_pos);
}
ctx->finished = TRUE;
ctx->uidlist->initial_sync = TRUE;
}
int maildir_uidlist_sync_deinit(struct maildir_uidlist_sync_ctx **_ctx)
{
struct maildir_uidlist_sync_ctx *ctx = *_ctx;
bool unlocked = FALSE;
int ret = ctx->failed ? -1 : 0;
*_ctx = NULL;
if (!ctx->finished)
maildir_uidlist_sync_finish(ctx);
if (ctx->partial)
maildir_uidlist_mark_all(ctx->uidlist, FALSE);
if (ctx->uidlist->need_rewrite ||
(ctx->new_files_count != 0 && !ctx->failed)) {
unsigned int nonrecursive_lock_count = 1;
if (ctx->uidlist->mbox->ibox.keep_locked)
nonrecursive_lock_count++;
if (ctx->uidlist->lock_count > nonrecursive_lock_count) {
/* recursive sync. let the root syncing do
the rewrite */
ctx->uidlist->need_rewrite = TRUE;
} else {
t_push();
ret = maildir_uidlist_rewrite(ctx->uidlist);
t_pop();
unlocked = TRUE;
if (ret == 0)
ctx->uidlist->need_rewrite = FALSE;
}
}
if (!unlocked)
maildir_uidlist_unlock(ctx->uidlist);
if (ctx->files != NULL)
hash_destroy(ctx->files);
if (ctx->record_pool != NULL)
pool_unref(ctx->record_pool);
if (array_is_created(&ctx->records))
array_free(&ctx->records);
i_free(ctx);
return ret;
}
void maildir_uidlist_add_flags(struct maildir_uidlist *uidlist,
const char *filename,
enum maildir_uidlist_rec_flag flags)
{
struct maildir_uidlist_rec *rec;
rec = hash_lookup(uidlist->files, filename);
i_assert(rec != NULL);
rec->flags |= flags;
}
struct maildir_uidlist_iter_ctx *
maildir_uidlist_iter_init(struct maildir_uidlist *uidlist)
{
struct maildir_uidlist_iter_ctx *ctx;
unsigned int count;
ctx = i_new(struct maildir_uidlist_iter_ctx, 1);
ctx->uidlist = uidlist;
ctx->next = array_get(&uidlist->records, &count);
ctx->end = ctx->next + count;
ctx->change_counter = ctx->uidlist->change_counter;
return ctx;
}
static void
maildir_uidlist_iter_update_idx(struct maildir_uidlist_iter_ctx *ctx)
{
unsigned int old_rev_idx, idx, count;
old_rev_idx = ctx->end - ctx->next;
ctx->next = array_get(&ctx->uidlist->records, &count);
ctx->end = ctx->next + count;
idx = old_rev_idx >= count ? 0 :
count - old_rev_idx;
while (idx < count && ctx->next[idx]->uid <= ctx->prev_uid)
idx++;
while (idx > 0 && ctx->next[idx-1]->uid > ctx->prev_uid)
idx--;
ctx->next += idx;
}
int maildir_uidlist_iter_next(struct maildir_uidlist_iter_ctx *ctx,
uint32_t *uid_r,
enum maildir_uidlist_rec_flag *flags_r,
const char **filename_r)
{
if (ctx->change_counter != ctx->uidlist->change_counter)
maildir_uidlist_iter_update_idx(ctx);
if (ctx->next == ctx->end)
return 0;
*uid_r = (*ctx->next)->uid;
*flags_r = (*ctx->next)->flags;
*filename_r = (*ctx->next)->filename;
ctx->next++;
ctx->prev_uid = *uid_r;
return 1;
}
void maildir_uidlist_iter_deinit(struct maildir_uidlist_iter_ctx *ctx)
{
i_free(ctx);
}