All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.manydesigns.mail.queue.FileSystemMailQueue Maven / Gradle / Ivy

There is a newer version: 5.3.4
Show newest version
/*
 * Copyright (C) 2005-2015 ManyDesigns srl.  All rights reserved.
 * http://www.manydesigns.com/
 *
 * This is free software; you can redistribute it and/or modify it
 * under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation; either version 3 of
 * the License, or (at your option) any later version.
 *
 * This software 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this software; if not, write to the Free
 * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
 * 02110-1301 USA, or see the FSF site: http://www.fsf.org.
 */

package com.manydesigns.mail.queue;

import com.manydesigns.elements.util.ElementsFileUtils;
import com.manydesigns.elements.util.RandomUtil;
import com.manydesigns.mail.queue.model.Attachment;
import com.manydesigns.mail.queue.model.Email;
import com.manydesigns.mail.queue.model.Recipient;
import org.apache.commons.io.FileUtils;
import org.apache.commons.io.IOUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBException;
import javax.xml.bind.Marshaller;
import javax.xml.bind.Unmarshaller;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * @author Paolo Predonzani     - [email protected]
 * @author Angelo Lupo          - [email protected]
 * @author Giampiero Granatella - [email protected]
 * @author Alessio Stalla       - [email protected]
 */
public class FileSystemMailQueue implements MailQueue {
    public static final String copyright =
            "Copyright (c) 2005-2015, ManyDesigns srl";

    protected final File queuedDirectory, sentDirectory, failedDirectory;
    protected final JAXBContext jaxbContext;
    protected boolean keepSent;

    protected static final Logger logger = LoggerFactory.getLogger(FileSystemMailQueue.class);

    public FileSystemMailQueue(File directory) {
        this.queuedDirectory = new File(directory, "queue");
        this.sentDirectory = new File(directory, "sent");
        this.failedDirectory = new File(directory, "failed");

        if(!ElementsFileUtils.ensureDirectoryExistsAndWritable(this.queuedDirectory)) {
            logger.warn("Directory does not exist or is not writable: {}", this.queuedDirectory);
        }
        if(!ElementsFileUtils.ensureDirectoryExistsAndWritable(this.sentDirectory)) {
            logger.warn("Directory does not exist or is not writable: {}", this.sentDirectory);
        }
        if(!ElementsFileUtils.ensureDirectoryExistsAndWritable(this.failedDirectory)) {
            logger.warn("Directory does not exist or is not writable: {}", this.failedDirectory);
        }
        try {
            jaxbContext = JAXBContext.newInstance(Email.class, Recipient.class);
        } catch (JAXBException e) {
            throw new Error("Couldn't create jaxb context", e);
        }
    }

    protected void checkDirectory(File file) throws QueueException {
        if(!ElementsFileUtils.ensureDirectoryExistsAndWritable(file)) {
            throw new QueueException("Invalid directory " + file.getAbsolutePath());
        }
    }

    protected void checkDirectories() throws QueueException {
        checkDirectory(queuedDirectory);
        checkDirectory(sentDirectory);
        checkDirectory(failedDirectory);
    }

    public String enqueue(Email email) throws QueueException {
        try {
            Marshaller marshaller = jaxbContext.createMarshaller();
            marshaller.setProperty(javax.xml.bind.Marshaller.JAXB_FORMATTED_OUTPUT, Boolean.TRUE);
            String emailId = RandomUtil.createRandomId(20);
            File destinationFile = getEmailFile(emailId);
            checkDirectory(queuedDirectory);
            if(!email.getAttachments().isEmpty()) {
                File attachDir = getEmailAttachmentsDirectory(emailId);
                checkDirectory(attachDir);
                for(Attachment attachment : email.getAttachments()) {
                    String attachmentId = RandomUtil.createRandomId(20);
                    File attachmentFile = new File(attachDir, attachmentId + ".bin");
                    FileOutputStream fos = new FileOutputStream(attachmentFile);
                    IOUtils.copy(attachment.getInputStream(), fos);
                    IOUtils.closeQuietly(fos);
                    IOUtils.closeQuietly(attachment.getInputStream());
                    attachment.setFilePath(attachmentFile.getAbsolutePath());
                }
            }
            marshaller.marshal(email, destinationFile);
            return emailId;
        } catch (Exception e) {
            throw new QueueException("Couldn't enqueue mail", e);
        }
    }

    protected File getEmailFile(String emailId) {
        return RandomUtil.getCodeFile(queuedDirectory, "email-{0}.xml", emailId);
    }

    protected File getEmailAttachmentsDirectory(String emailId) {
        return RandomUtil.getCodeFile(queuedDirectory, "email-{0}-attachments", emailId);
    }

    public List getEnqueuedEmailIds() throws QueueException {
        checkDirectory(queuedDirectory);
        List ids = new ArrayList();
        Pattern pattern = Pattern.compile("^email-(.*)\\.xml$");
        for(String filename : queuedDirectory.list()) {
            Matcher matcher = pattern.matcher(filename);
            if(matcher.matches()) {
                logger.debug("Path matched: {}", filename);
                ids.add(matcher.group(1));
            }
        }
        return ids;
    }

    public Email loadEmail(String id) throws QueueException {
        try {
            Unmarshaller unmarshaller = jaxbContext.createUnmarshaller();
            File emailFile = getEmailFile(id);
            if(emailFile.exists()) {
                logger.debug("Found email with id: {}", id);
                Email email = (Email) unmarshaller.unmarshal(emailFile);
                return email;
            } else {
                logger.debug("Email with id {} not found", id);
                return null;
            }
        } catch (JAXBException e) {
            throw new MailParseException("Couldn't parse email", e);
        } catch (Exception e) {
            throw new QueueException("Couldn't load email", e);
        }
    }

    public void markSent(String id) throws QueueException {
        checkDirectories();
        try {
            File emailFile = getEmailFile(id);
            if(emailFile.exists()) {
                File attachmentsDir = getEmailAttachmentsDirectory(id);
                if(keepSent) {
                    logger.info("Moving email with id {} to sent directory", id);
                    FileUtils.moveToDirectory(emailFile, sentDirectory, false);
                    if(attachmentsDir.exists()) {
                        FileUtils.moveToDirectory(attachmentsDir, sentDirectory, false);
                    }
                } else {
                    logger.info("Deleting sent email with id {}", id);
                    boolean success = emailFile.delete();
                    if(attachmentsDir.exists()) {
                        for(File attachmentFile : attachmentsDir.listFiles()) {
                            success = attachmentFile.delete() && success;
                        }
                        success = attachmentsDir.delete() && success;
                    }
                    if(!success) {
                        throw new QueueException("Couldn't mark mail as sent");
                    }
                }
            } else {
                logger.debug("Not marking email with id {} as sent", id);
            }
        } catch (IOException e) {
            throw new Error("Couldn't mark mail as sent", e);
        }
    }

    public void markFailed(String id) throws QueueException {
        checkDirectories();
        try {
            File emailFile = getEmailFile(id);
            if(emailFile.exists()) {
                File attachmentsDir = getEmailAttachmentsDirectory(id);
                logger.info("Marking email with id {} as failed", id);
                FileUtils.moveToDirectory(emailFile, failedDirectory, false);
                if(attachmentsDir.exists()) {
                    FileUtils.moveToDirectory(attachmentsDir, failedDirectory, false);
                }
            } else {
                logger.debug("Not marking email with id {} as failed", id);
            }
        } catch (IOException e) {
            throw new QueueException("Couldn't mark mail as failed", e);
        }
    }

    public File getQueuedDirectory() {
        return queuedDirectory;
    }

    public File getSentDirectory() {
        return sentDirectory;
    }

    public File getFailedDirectory() {
        return failedDirectory;
    }

    public boolean isKeepSent() {
        return keepSent;
    }

    public void setKeepSent(boolean keepSent) {
        this.keepSent = keepSent;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy