maildir-mail.c revision d9e5b8e97506db35c357684eec08306c8464a55a
/* Copyright (C) 2003 Timo Sirainen */
#include "lib.h"
#include "istream.h"
#include "index-mail.h"
#include "maildir-storage.h"
#include "maildir-uidlist.h"
#include <fcntl.h>
#include <unistd.h>
#include <sys/stat.h>
static int
do_open(struct maildir_mailbox *mbox, const char *path, void *context)
{
int *fd = context;
*fd = open(path, O_RDONLY);
if (*fd != -1)
return 1;
if (errno == ENOENT)
return 0;
mail_storage_set_critical(STORAGE(mbox->storage),
"open(%s) failed: %m", path);
return -1;
}
static int
do_stat(struct maildir_mailbox *mbox, const char *path, void *context)
{
struct stat *st = context;
if (stat(path, st) == 0)
return 1;
if (errno == ENOENT)
return 0;
mail_storage_set_critical(STORAGE(mbox->storage),
"stat(%s) failed: %m", path);
return -1;
}
static struct istream *
maildir_open_mail(struct maildir_mailbox *mbox, uint32_t uid, bool *deleted)
{
int fd;
*deleted = FALSE;
fd = -1;
if (maildir_file_do(mbox, uid, do_open, &fd) < 0)
return NULL;
if (fd == -1) {
*deleted = TRUE;
return NULL;
}
if (mbox->ibox.mail_read_mmaped) {
return i_stream_create_mmap(fd, default_pool,
MAIL_MMAP_BLOCK_SIZE, 0, 0, TRUE);
} else {
return i_stream_create_file(fd, default_pool,
MAIL_READ_BLOCK_SIZE, TRUE);
}
}
static time_t maildir_mail_get_received_date(struct mail *_mail)
{
struct index_mail *mail = (struct index_mail *)_mail;
struct maildir_mailbox *mbox = (struct maildir_mailbox *)mail->ibox;
struct index_mail_data *data = &mail->data;
struct stat st;
int fd;
(void)index_mail_get_received_date(_mail);
if (data->received_date != (time_t)-1)
return data->received_date;
if (data->access_part != 0 && data->stream == NULL) {
/* we're going to open the mail anyway */
(void)mail_get_stream(_mail, NULL, NULL);
}
if (data->stream != NULL) {
fd = i_stream_get_fd(data->stream);
i_assert(fd != -1);
if (fstat(fd, &st) < 0) {
mail_storage_set_critical(STORAGE(mbox->storage),
"fstat(maildir) failed: %m");
return (time_t)-1;
}
} else {
if (maildir_file_do(mbox, _mail->uid, do_stat, &st) <= 0)
return (time_t)-1;
}
data->received_date = st.st_mtime;
index_mail_cache_add(mail, MAIL_CACHE_RECEIVED_DATE,
&data->received_date, sizeof(data->received_date));
return data->received_date;
}
static uoff_t maildir_mail_get_virtual_size(struct mail *_mail)
{
struct index_mail *mail = (struct index_mail *)_mail;
struct maildir_mailbox *mbox = (struct maildir_mailbox *)mail->ibox;
struct index_mail_data *data = &mail->data;
const char *fname, *p;
uoff_t virtual_size;
enum maildir_uidlist_rec_flag flags;
if (data->virtual_size != (uoff_t)-1)
return data->virtual_size;
if ((mail->wanted_fields & MAIL_FETCH_VIRTUAL_SIZE) == 0) {
data->virtual_size = index_mail_get_cached_virtual_size(mail);
if (data->virtual_size != (uoff_t)-1)
return data->virtual_size;
}
fname = maildir_uidlist_lookup(mbox->uidlist, _mail->uid, &flags);
if (fname == NULL)
return (uoff_t)-1;
/* size can be included in filename */
p = strstr(fname, MAILDIR_EXTRA_SEP_S MAILDIR_EXTRA_VIRTUAL_SIZE "=");
if (p != NULL) {
p += 3;
virtual_size = 0;
while (*p >= '0' && *p <= '9') {
virtual_size = virtual_size * 10 + (*p - '0');
p++;
}
if (*p == MAILDIR_INFO_SEP || *p == MAILDIR_EXTRA_SEP ||
*p == '\0') {
index_mail_cache_add(mail, MAIL_CACHE_VIRTUAL_FULL_SIZE,
&virtual_size,
sizeof(virtual_size));
return virtual_size;
}
}
return index_mail_get_virtual_size(_mail);
}
static const char *
maildir_mail_get_special(struct mail *_mail, enum mail_fetch_field field)
{
struct index_mail *mail = (struct index_mail *)_mail;
struct maildir_mailbox *mbox = (struct maildir_mailbox *)mail->ibox;
enum maildir_uidlist_rec_flag flags;
const char *fname, *end;
if (field == MAIL_FETCH_UIDL_FILE_NAME) {
fname = maildir_uidlist_lookup(mbox->uidlist,
_mail->uid, &flags);
end = strchr(fname, MAILDIR_INFO_SEP);
return end == NULL ? fname : t_strdup_until(fname, end);
}
return index_mail_get_special(_mail, field);
}
static uoff_t maildir_mail_get_physical_size(struct mail *_mail)
{
struct index_mail *mail = (struct index_mail *)_mail;
struct maildir_mailbox *mbox = (struct maildir_mailbox *)mail->ibox;
struct index_mail_data *data = &mail->data;
struct stat st;
const char *fname, *p;
uoff_t size;
enum maildir_uidlist_rec_flag flags;
size = index_mail_get_physical_size(_mail);
if (size != (uoff_t)-1)
return size;
fname = maildir_uidlist_lookup(mbox->uidlist, _mail->uid, &flags);
if (fname == NULL)
return (uoff_t)-1;
/* size can be included in filename */
p = strstr(fname, MAILDIR_EXTRA_SEP_S MAILDIR_EXTRA_FILE_SIZE "=");
if (p != NULL) {
p += 3;
size = 0;
while (*p >= '0' && *p <= '9') {
size = size * 10 + (*p - '0');
p++;
}
if (*p != MAILDIR_INFO_SEP &&
*p != MAILDIR_EXTRA_SEP && *p != '\0')
size = (uoff_t)-1;
}
if (size == (uoff_t)-1) {
if (maildir_file_do(mbox, _mail->uid, do_stat, &st) <= 0)
return (uoff_t)-1;
size = st.st_size;
}
index_mail_cache_add(mail, MAIL_CACHE_PHYSICAL_FULL_SIZE,
&size, sizeof(size));
data->physical_size = size;
return size;
}
static struct istream *maildir_mail_get_stream(struct mail *_mail,
struct message_size *hdr_size,
struct message_size *body_size)
{
struct index_mail *mail = (struct index_mail *)_mail;
struct maildir_mailbox *mbox = (struct maildir_mailbox *)mail->ibox;
struct index_mail_data *data = &mail->data;
bool deleted;
if (data->stream == NULL) {
data->stream = maildir_open_mail(mbox, _mail->uid, &deleted);
if (data->stream == NULL) {
_mail->expunged = deleted;
return NULL;
}
}
return index_mail_init_stream(mail, hdr_size, body_size);
}
struct mail_vfuncs maildir_mail_vfuncs = {
index_mail_free,
index_mail_set_seq,
index_mail_get_flags,
index_mail_get_keywords,
index_mail_get_parts,
maildir_mail_get_received_date,
index_mail_get_date,
maildir_mail_get_virtual_size,
maildir_mail_get_physical_size,
index_mail_get_first_header,
index_mail_get_headers,
index_mail_get_header_stream,
maildir_mail_get_stream,
maildir_mail_get_special,
index_mail_update_flags,
index_mail_update_keywords,
index_mail_expunge
};