index-status.c revision 3c7a9aa32ddc1a930a75681a6c3cafb8d10fe810
/* Copyright (C) 2002 Timo Sirainen */
#include "lib.h"
#include "index-storage.h"
static unsigned int get_recent_count(MailIndex *index)
{
MailIndexHeader *hdr;
MailIndexRecord *rec;
hdr = mail_index_get_header(index);
if (index->first_recent_uid <= 1) {
/* all are recent */
return hdr->messages_count;
}
/* get the first recent message */
if (index->first_recent_uid >= hdr->next_uid)
return 0;
rec = index->lookup_uid_range(index, index->first_recent_uid,
hdr->next_uid - 1);
if (rec == NULL)
return 0;
/* now we know the record, but we'd still need to know how many
messages there's after this. there's two way to do this -
get the sequence number thus far (fast, unless there's deleted
messages) or just start reading messages forward until we're at
the end (fast assuming there's only a few recent messages).
it's a bit easier to use the first method and often it should be
faster too.. */
return hdr->messages_count - index->get_sequence(index, rec) + 1;
}
static unsigned int get_first_unseen_seq(MailIndex *index)
{
MailIndexHeader *hdr;
MailIndexRecord *rec;
unsigned int seq, lowwater_uid;
hdr = mail_index_get_header(index);
if (hdr->seen_messages_count == hdr->messages_count) {
/* no unseen messages */
return 0;
}
lowwater_uid = hdr->first_unseen_uid_lowwater;
if (lowwater_uid != 0) {
/* begin scanning from the low water mark */
rec = index->lookup_uid_range(index, lowwater_uid,
hdr->next_uid - 1);
if (rec == NULL) {
i_warning("index header's seen_messages_count or "
"first_unseen_uid_lowwater is invalid.");
INDEX_MARK_CORRUPTED(index);
return 0;
} else {
seq = index->get_sequence(index, rec);
}
} else {
/* begin scanning from the beginning */
rec = index->lookup(index, 1);
seq = 1;
}
while (rec != NULL && (rec->msg_flags & MAIL_SEEN)) {
rec = index->next(index, rec);
seq++;
}
if (rec != NULL && rec->uid != lowwater_uid) {
/* update the low water mark if we can get exclusive
lock immediately. */
if (index->try_lock(index, MAIL_LOCK_EXCLUSIVE))
hdr->first_unseen_uid_lowwater = rec->uid;
}
return rec == NULL ? 0 : seq;
}
int index_storage_get_status(Mailbox *box, MailboxStatusItems items,
MailboxStatus *status)
{
IndexMailbox *ibox = (IndexMailbox *) box;
MailIndexHeader *hdr;
if (!ibox->index->set_lock(ibox->index, MAIL_LOCK_SHARED))
return mail_storage_set_index_error(ibox);
/* we can get most of the status items without any trouble */
hdr = mail_index_get_header(ibox->index);
status->messages = hdr->messages_count;
status->unseen = hdr->messages_count - hdr->seen_messages_count;
status->uidvalidity = hdr->uid_validity;
status->uidnext = hdr->next_uid;
if (items & STATUS_FIRST_UNSEEN_SEQ) {
status->first_unseen_seq =
get_first_unseen_seq(ibox->index);
}
if (items & STATUS_RECENT)
status->recent = get_recent_count(ibox->index);
/* STATUS sends EXISTS, so we've synced it */
ibox->synced_messages_count = hdr->messages_count;
if (!ibox->index->set_lock(ibox->index, MAIL_LOCK_UNLOCK))
return mail_storage_set_index_error(ibox);
return hdr != NULL;
}