mbox-save.c revision 8ef44bc339434cbc94a8e9383b2badf6a3ff4757
/* Copyright (c) 2002-2007 Dovecot authors, see the included COPYING file */
#include "lib.h"
#include "ioloop.h"
#include "array.h"
#include "base64.h"
#include "hostpid.h"
#include "randgen.h"
#include "istream.h"
#include "ostream.h"
#include "str.h"
#include "write-full.h"
#include "istream-header-filter.h"
#include "ostream-crlf.h"
#include "message-parser.h"
#include "index-mail.h"
#include "mbox-storage.h"
#include "mbox-file.h"
#include "mbox-from.h"
#include "mbox-lock.h"
#include "mbox-md5.h"
#include "mbox-sync-private.h"
#include <stddef.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <netdb.h>
#include <utime.h>
struct mbox_save_context {
struct mail_save_context ctx;
struct mbox_mailbox *mbox;
struct mail_index_transaction *trans;
char last_char;
struct mbox_md5_context *mbox_md5_ctx;
unsigned int x_delivery_id_pos;
unsigned int synced:1;
unsigned int failed:1;
unsigned int finished:1;
};
{
return -1;
}
{
char ch;
int fd;
*offset = 0;
return 0;
}
return 0;
if (ch != '\n') {
return write_error(ctx);
*offset += 1;
}
return 0;
}
{
return write_error(ctx);
return 0;
}
const char *from_envelope)
{
int ret;
if (*my_hostdomain == '\0') {
/* failed, use just the hostname */
name = my_hostname;
}
}
t_push();
if (from_envelope == NULL) {
}
/* save in local timezone, no matter what it was given with */
t_pop();
return ret;
}
{
const char *str;
/* we can't seek, don't set Content-Length */
return 0;
}
/* write Content-Length headers */
/* flush manually here so that we don't confuse seek() errors with
buffer flushing errors */
return write_error(ctx);
return write_error(ctx);
return 0;
}
static void mbox_save_init_sync(struct mbox_transaction_context *t)
{
const struct mail_index_header *hdr;
struct mail_index_view *view;
/* open a new view to get the header. this is required if we just
synced the mailbox so we can get updated next_uid. */
t->mails_saved = TRUE;
}
const struct mbox_flag_type *flags_list)
{
int i;
for (i = 0; flags_list[i].chr != 0; i++) {
}
}
{
if ((flags & STATUS_FLAGS_MASK) != 0) {
}
if ((flags & XSTATUS_FLAGS_MASK) != 0) {
}
}
static void
struct mail_keywords *keywords)
{
const char *const *keyword_names;
unsigned int i, count, keyword_names_count;
for (i = 0; i < count; i++) {
}
}
static int
struct mbox_transaction_context *t, bool want_mail)
{
int ret;
"Read-only mbox");
return -1;
}
/* first appended mail in this transaction */
/* FIXME: we shouldn't fail here. it's just
a locking issue that should be possible to
fix.. */
"Can't copy mails inside same mailbox");
return -1;
}
return -1;
}
if (mbox_file_open(mbox) < 0)
return -1;
}
/* update mbox_sync_dirty state */
if (ret < 0)
return -1;
/* we're not required to assign UIDs for the appended
mails immediately. do it only if it doesn't require
syncing. */
}
}
/* we'll need to assign UID for the mail immediately. */
return -1;
}
/* the syncing above could have changed the append offset */
return -1;
FALSE);
}
return 0;
}
{
/* we can't allow From_-lines in headers. there's no
legitimate reason for allowing them in any case,
so just drop them. */
return;
}
}
}
{
unsigned char md5_result[MD5_RESULTLEN];
void *randbuf;
t_push();
sizeof(ioloop_timeval.tv_usec));
t_pop();
}
{
struct mbox_transaction_context *t =
(struct mbox_transaction_context *)_t;
enum mail_flags save_flags;
/* FIXME: we could write timezone_offset to From-line.. */
}
return -1;
}
/* writing the first mail. Insert X-IMAPbase as well. */
}
MAIL_STORAGE_FLAG_KEEP_HEADER_MD5) != 0) {
/* we're using MD5 sums to generate POP3 UIDLs.
clients don't like it much if there are duplicates,
so make sure that there can't be any by appending
our own X-Delivery-ID header. */
}
save_flags &= ~MAIL_RECENT;
}
/* parse and cache the mail headers as we read it */
}
}
else {
ctx->body_output =
MAIL_STORAGE_FLAG_SAVE_CRLF) != 0 ?
}
}
}
{
const unsigned char *data;
return -1;
/* writing body */
/* i_stream_read() may have returned 0 at EOF
because of this parser */
}
if (ret == 0)
return 0;
return write_error(ctx);
}
/* if mail doesn't end with LF, we'll do that.
otherwise some mbox parsers don't like the result.
this makes it impossible to save a mail that doesn't
end with LF though. */
return write_error(ctx);
}
return 0;
}
if (ret == 0)
return 0;
for (i = 0; i < size; i++) {
if (data[i] == '\n' &&
/* end of headers. we don't need to worry about
CRs because they're dropped */
break;
}
}
if (i != size) {
/* found end of headers. write the rest of them
(not including the finishing empty line) */
return write_error(ctx);
break;
}
return write_error(ctx);
}
return write_error(ctx);
}
if (ctx->mbox_md5_ctx) {
unsigned char hdr_md5_sum[16];
if (ctx->x_delivery_id_pos != 0) {
struct message_header_line hdr;
const unsigned char *p;
}
hdr_md5_sum, NULL);
}
/* append our own headers and ending empty line */
return write_error(ctx);
/* write body */
}
{
t_push();
if (mbox_write_content_length(ctx) < 0 ||
mbox_append_lf(ctx) < 0)
t_pop();
}
/* saving this mail failed - truncate back to beginning of it */
}
}
{
(void)mbox_save_finish(_ctx);
}
{
}
{
struct mbox_transaction_context *t =
int ret = 0;
ret = -1;
}
}
}
/* try to set atime back to its original value */
}
ret = -1;
}
}
return ret;
}
{
/* failed, truncate file back to original size.
output stream needs to be flushed before truncating
so unref() won't write anything. */
}
}