aboutsummaryrefslogblamecommitdiffstats
path: root/camel/camel-movemail.c
blob: d5cc33548102a9f392c2342b4208da22dcc05b17 (plain) (tree)


























                                                                           
                    




                   
                   
                   


                            




                                   





                                                                    


                                                                                                                









                                                                       
    
    
                                                                         
                        















                                                                      





                                                                          
                            
                       











                                                                     













                                                                          

                                                                                
                       

                      



















                                                                               



























































                                                                           
                       




                                                         





                                                      
                                           

                                                                            




                                                                   
                                                                            


                                                                                 







                                                                      
           
                                           





                                                                         
                 
              
                            





                                  
































































                                                                            
 




                                                                           
 
      






































































































































































































































                                                                                                                                                   
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/* camel-movemail.c: mbox copying function */

/*
 * Author:
 *  Dan Winship <danw@helixcode.com>
 *
 * Copyright 2000 Helix Code, Inc. (http://www.helixcode.com)
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
 * published by the Free Software Foundation; either version 2 of the
 * License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
 * USA
 */

#include <config.h>

#include <sys/stat.h>
#include <sys/uio.h>
#include <errno.h>
#include <fcntl.h>
#include <stdlib.h>
#include <stdio.h>
#include <time.h>
#include <unistd.h>
#include <string.h>
#include <signal.h>

#include "camel-movemail.h"
#include "camel-exception.h"

#include "camel-mime-parser.h"
#include "camel-mime-filter.h"
#include "camel-mime-filter-from.h"

#define d(x)

#ifdef MOVEMAIL_PATH
#include <sys/wait.h>

static void movemail_external (const char *source, const char *dest,
                   CamelException *ex);
#endif

/* these could probably be exposed as a utility? (but only mbox needs it) */
static int camel_movemail_copy_filter(int fromfd, int tofd, off_t start, size_t bytes, CamelMimeFilter *filter);
static int camel_movemail_copy(int fromfd, int tofd, off_t start, size_t bytes);

/**
 * camel_movemail: Copy an mbox file from a shared spool directory to a
 * new folder in a Camel store
 * @source: source file
 * @dest: destination file
 * @ex: a CamelException
 *
 * This copies an mbox file from a shared directory with multiple
 * readers and writers into a private (presumably Camel-controlled)
 * directory. Dot locking is used on the source file (but not the
 * destination).
 **/
void
camel_movemail (const char *source, const char *dest, CamelException *ex)
{
    gboolean locked;
    int sfd, dfd, tmpfd;
    char *locktmpfile, *lockfile;
    struct stat st;
    time_t now, timeout;
    int nread, nwrote;
    char buf[BUFSIZ];

    camel_exception_clear (ex);

    /* Stat and then open the spool file. If it doesn't exist or
     * is empty, the user has no mail. (There's technically a race
     * condition here in that an MDA might have just now locked it
     * to deliver a message, but we don't care. In that case,
     * assuming it's unlocked is equivalent to pretending we were
     * called a fraction earlier.)
     */
    if (stat (source, &st) == -1) {
        if (errno != ENOENT) {
            camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                          "Could not check mail file "
                          "%s: %s", source,
                          g_strerror (errno));
        }
        return;
    }
    if (st.st_size == 0)
        return;

    /* Create the unique lock file. */
    locktmpfile = g_strdup_printf ("%s.lock.XXXXXX", source);
#ifdef HAVE_MKSTEMP
    tmpfd = mkstemp (locktmpfile);
#else
    if (mktemp (locktmpfile)) {
        tmpfd = open (locktmpfile, O_RDWR | O_CREAT | O_EXCL,
                  S_IRUSR | S_IWUSR);
    } else
        tmpfd = -1;
#endif
    if (tmpfd == -1) {
        g_free (locktmpfile);
#ifdef MOVEMAIL_PATH
        if (errno == EACCES) {
            /* movemail_external will fail if the dest file
             * already exists, so if it does, return now,
             * let the fetch code process the mail that's
             * already there, and then the user can try again.
             */
            if (stat (dest, &st) == 0)
                return;

            movemail_external (source, dest, ex);
            return;
        }
#endif
        camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                      "Could not create lock file "
                      "for %s: %s", source, g_strerror (errno));
        return;
    }
    close (tmpfd);

    sfd = open (source, O_RDWR);
    if (sfd == -1) {
        camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                      "Could not open mail file %s: %s",
                      source, g_strerror (errno));
        unlink (locktmpfile);
        g_free (locktmpfile);
        return;
    }

    dfd = open (dest, O_WRONLY | O_CREAT | O_APPEND, S_IRUSR | S_IWUSR);
    if (dfd == -1) {
        camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                      "Could not open temporary mail "
                      "file %s: %s", dest, g_strerror (errno));
        close (sfd);
        unlink (locktmpfile);
        g_free (locktmpfile);
        return;
    }

    lockfile = g_strdup_printf ("%s.lock", source);
    locked = FALSE;
    time (&timeout);
    timeout += 30;

    /* Loop trying to lock the file for 30 seconds. */
    while (time (&now) < timeout) {
        /* Try to make the lock. */
        if (symlink (locktmpfile, lockfile) == 0) {
            locked = TRUE;
            break;
        }

        /* If we fail for a reason other than that someone
         * else has the lock, then abort.
         */
        if (errno != EEXIST) {
            camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                          "Could not create lock "
                          "file for %s: %s", source,
                          g_strerror (errno));
            break;
        }

        /* Check the modtime on the lock file. */
        if (stat (lockfile, &st) == -1) {
            /* If the lockfile disappeared, try again. */
            if (errno == ENOENT)
                continue;

            /* Some other error. Abort. */
            camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                          "Could not test lock "
                          "file for %s: %s", source,
                          g_strerror (errno));
            break;
        }

        /* If the lock file is stale, remove it and try again. */
        if (st.st_mtime < now - 60) {
            unlink (lockfile);
            continue;
        }

        /* Otherwise, sleep and try again. */
        sleep (5);
    }

    if (!locked) {
        /* Something has gone awry. */
        if (now >= timeout) {
            camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                          "Timed out trying to get "
                          "lock file on %s. Try again "
                          "later.", source);
        }
        g_free (lockfile);
        unlink (locktmpfile);
        g_free (locktmpfile);
        close (sfd);
        close (dfd);
        return;
    }

    /* OK. We have the file locked now. */

    /* FIXME: Set a timer to keep the file locked. */

    while (1) {
        int written = 0;

        nread = read (sfd, buf, sizeof (buf));
        if (nread == 0)
            break;
        else if (nread == -1) {
            if (errno == EINTR)
                continue;
            camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                          "Error reading mail file: %s",
                          g_strerror (errno));
            break;
        }

        while (nread) {
            nwrote = write (dfd, buf + written, nread);
            if (nwrote == -1) {
                if (errno == EINTR)
                    continue; /* continues inner loop */
                camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                              "Error writing "
                              "mail temp file: %s",
                              g_strerror (errno));
                break;
            }
            written += nwrote;
            nread -= nwrote;
        }
    }

    /* If no errors occurred copying the data, and we successfully
     * close the destination file, then truncate the source file.
     */
    if (!camel_exception_is_set (ex)) {
        if (close (dfd) == 0)
            ftruncate (sfd, 0);
        else {
            camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                          "Failed to store mail in "
                          "temp file %s: %s", dest,
                          g_strerror (errno));
        }
    } else
        close (dfd);
    close (sfd);

    /* Clean up lock files. */
    unlink (lockfile);
    g_free (lockfile);
    unlink (locktmpfile);
    g_free (locktmpfile);
}

#ifdef MOVEMAIL_PATH
static void
movemail_external (const char *source, const char *dest, CamelException *ex)
{
    sigset_t mask, omask;
    pid_t pid;
    int fd[2], len = 0, nread, status;
    char buf[BUFSIZ], *output = NULL;

    /* Block SIGCHLD so the app can't mess us up. */
    sigemptyset (&mask);
    sigaddset (&mask, SIGCHLD);
    sigprocmask (SIG_BLOCK, &mask, &omask);

    if (pipe (fd) == -1) {
        sigprocmask (SIG_SETMASK, &omask, NULL);
        camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                      "Could not create pipe: %s",
                      g_strerror (errno));
        return;
    }

    pid = fork ();
    switch (pid) {
    case -1:
        close (fd[0]);
        close (fd[1]);
        sigprocmask (SIG_SETMASK, &omask, NULL);
        camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                      "Could not fork: %s",
                      g_strerror (errno));
        return;

    case 0:
        /* Child */
        close (fd[0]);
        close (STDIN_FILENO);
        dup2 (fd[1], STDOUT_FILENO);
        dup2 (fd[1], STDERR_FILENO);

        execl (MOVEMAIL_PATH, MOVEMAIL_PATH, source, dest, NULL);
        _exit (255);
        break;

    default:
        break;
    }

    /* Parent */
    close (fd[1]);

    /* Read movemail's output. */
    while ((nread = read (fd[0], buf, sizeof (buf))) > 0) {
        output = g_realloc (output, len + nread + 1);
        memcpy (output + len, buf, nread);
        len += nread;
        output[len] = '\0';
    }
    close (fd[0]);

    /* Now get the exit status. */
    while (waitpid (pid, &status, 0) == -1 && errno == EINTR)
        ;
    sigprocmask (SIG_SETMASK, &omask, NULL);

    if (!WIFEXITED (status) || WEXITSTATUS (status) != 0) {
        camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                      "Movemail program failed: %s",
                      output ? output : "(Unknown error)");
    }
    g_free (output);
}
#endif


static int
camel_movemail_copy(int fromfd, int tofd, off_t start, size_t bytes)
{
        char buffer[4096];
        int written = 0;

    d(printf("writing %d bytes ... ", bytes));

    if (lseek(fromfd, start, SEEK_SET) != start)
        return -1;

        while (bytes>0) {
                int toread, towrite;

                toread = bytes;
                if (bytes>4096)
                        toread = 4096;
                else
                        toread = bytes;
        do {
            towrite = read(fromfd, buffer, toread);
        } while (towrite == -1 && errno == EINTR);

        if (towrite == -1)
            return -1;

                /* check for 'end of file' */
                if (towrite == 0) {
            d(printf("end of file?\n"));
                        break;
        }

        do {
            toread = write(tofd, buffer, towrite);
        } while (toread == -1 && errno == EINTR);

        if (toread == -1)
            return -1;

                written += toread;
                bytes -= toread;
        }

        d(printf("written %d bytes\n", written));

        return written;
}

#define PRE_SIZE (32)

static int
camel_movemail_copy_filter(int fromfd, int tofd, off_t start, size_t bytes, CamelMimeFilter *filter)
{
        char buffer[4096+PRE_SIZE];
        int written = 0;
    char *filterbuffer;
    int filterlen, filterpre;

    d(printf("writing %d bytes ... ", bytes));

    camel_mime_filter_reset(filter);

    if (lseek(fromfd, start, SEEK_SET) != start)
        return -1;

        while (bytes>0) {
                int toread, towrite;

                toread = bytes;
                if (bytes>4096)
                        toread = 4096;
                else
                        toread = bytes;
        do {
            towrite = read(fromfd, buffer+PRE_SIZE, toread);
        } while (towrite == -1 && errno == EINTR);

        if (towrite == -1)
            return -1;

                /* check for 'end of file' */
                if (towrite == 0) {
            d(printf("end of file?\n"));
            camel_mime_filter_complete(filter, buffer+PRE_SIZE, towrite, PRE_SIZE,
                           &filterbuffer, &filterlen, &filterpre);
            towrite = filterlen;
            if (towrite == 0)
                break;
        } else {
            camel_mime_filter_filter(filter, buffer+PRE_SIZE, towrite, PRE_SIZE,
                         &filterbuffer, &filterlen, &filterpre);
            towrite = filterlen;
        }

        do {
            toread = write(tofd, filterbuffer, towrite);
        } while (toread == -1 && errno == EINTR);

        if (toread == -1)
            return -1;

                written += toread;
                bytes -= toread;
        }

        d(printf("written %d bytes\n", written));

        return written;
}

/* write the headers back out again, but not he Content-Length header, because we dont
   want to maintain it! */
static int
solaris_header_write(int fd, struct _header_raw *header)
{
        struct iovec iv[4];
        int outlen = 0, len;

        iv[1].iov_base = ":";
        iv[1].iov_len = 1;
        iv[3].iov_base = "\n";
        iv[3].iov_len = 1;

        while (header) {
        if (strcasecmp(header->name, "Content-Length")) {
            iv[0].iov_base = header->name;
            iv[0].iov_len = strlen(header->name);
            iv[2].iov_base = header->value;
            iv[2].iov_len = strlen(header->value);
        
            do {
                len = writev(fd, iv, 4);
            } while (len == -1 && errno == EINTR);
            
            if (len == -1)
                return -1;
            outlen += len;
        }
                header = header->next;
        }

    do {
        len = write(fd, "\n", 1);
    } while (len == -1 && errno == EINTR);

    if (len == -1)
        return -1;

    outlen += 1;

    d(printf("Wrote %d bytes of headers\n", outlen));

        return outlen;
}

/* Well, since Solaris is a tad broken wrt its 'mbox' folder format,
   we must convert it to a real mbox format.  Thankfully this is
   mostly pretty easy */
static int
camel_movemail_solaris (int sfd, int dfd, CamelException *ex)
{
    CamelMimeParser *mp;
    char *buffer;
    int len;
    CamelMimeFilterFrom *ffrom;
    int ret = 1;

    mp = camel_mime_parser_new();
    camel_mime_parser_scan_from(mp, TRUE);
    camel_mime_parser_init_with_fd(mp, sfd);

    ffrom = camel_mime_filter_from_new();

    while (camel_mime_parser_step(mp, &buffer, &len) == HSCAN_FROM) {
        if (camel_mime_parser_step(mp, &buffer, &len) != HSCAN_FROM_END) {
            const char *cl;
            int length;
            int start, body;
            off_t newpos;

            ret = 0;

            start = camel_mime_parser_tell_start_from(mp);
            body = camel_mime_parser_tell(mp);

            /* write out headers, but NOT content-length header */
            solaris_header_write(dfd, camel_mime_parser_headers_raw(mp));

            cl = camel_mime_parser_header(mp, "content-length", NULL);
            if (cl == NULL) {
                g_warning("Required Content-Length header is missing from solaris mail box @ %d", (int)camel_mime_parser_tell(mp));
                camel_mime_parser_drop_step(mp);
                camel_mime_parser_drop_step(mp);
                camel_mime_parser_step(mp, &buffer, &len);
                camel_mime_parser_unstep(mp);
                length = camel_mime_parser_tell_start_from(mp) - body;
                newpos = -1;
            } else {
                length = atoi(cl);
                camel_mime_parser_drop_step(mp);
                camel_mime_parser_drop_step(mp);
                newpos = length+body;
            }
            /* copy body->length converting From lines */
            if (camel_movemail_copy_filter(sfd, dfd, body, length, (CamelMimeFilter *)ffrom) == -1)
                goto fail;
            if (newpos != -1)
                camel_mime_parser_seek(mp, newpos, SEEK_SET);
        } else {
            g_error("Inalid parser state: %d", camel_mime_parser_state(mp));
        }
    }

    gtk_object_unref((GtkObject *)mp);
    gtk_object_unref((GtkObject *)ffrom);

    return ret;

fail:
    camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
                  "Error copying "
                  "mail temp file: %s",
                  g_strerror (errno));


    gtk_object_unref((GtkObject *)mp);
    gtk_object_unref((GtkObject *)ffrom);

    return -1;
}