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

org.codemonkey.simplejavamail.Mailer Maven / Gradle / Ivy

package org.codemonkey.simplejavamail;

import java.io.UnsupportedEncodingException;
import java.util.Date;
import java.util.Map;
import java.util.Properties;

import javax.activation.DataHandler;
import javax.activation.DataSource;
import javax.mail.Address;
import javax.mail.Authenticator;
import javax.mail.BodyPart;
import javax.mail.Message;
import javax.mail.MessagingException;
import javax.mail.Part;
import javax.mail.PasswordAuthentication;
import javax.mail.Session;
import javax.mail.Transport;
import javax.mail.internet.InternetAddress;
import javax.mail.internet.MimeBodyPart;
import javax.mail.internet.MimeMessage;
import javax.mail.internet.MimeMultipart;
import javax.mail.internet.MimeUtility;

import org.apache.log4j.Logger;

/**
 * Mailing tool aimed for simplicity, for sending e-mails of any complexity. This includes e-mails with plain text and/or html content,
 * embedded images and separate attachments, SMTP, SMTPS / SSL and SMTP + SSL
*
* This mailing tool abstracts the javax.mail API to a higher level easy to use API. For public use, this tool only works with {@link Email} * instances.
*
* The e-mail message structure is built to work with all e-mail clients and has been tested with many different webclients as well as some * mainstream client applications such as MS Outlook or Mozilla Thunderbird.
*
* Technically, the resulting email structure is as follows:
* *
 * - root
 * 	- related
 * 		- alternative
 * 			- mail text
 * 			- mail html text
 * 		- embedded images
 * 	- attachments
 * 
* *
* Usage example:
* *
 * Email email = new Email();
 * email.setFromAddress("lollypop", "[email protected]");
 * email.addRecipient("Sugar Cane", "[email protected]", RecipientType.TO);
 * email.setText("We should meet up!!");
 * email.setTextHTML("<b>We should meet up!</b>");
 * email.setSubject("Hey");
 * new Mailer(preconfiguredMailSession).sendMail(email);
 * // or:
 * new Mailer("smtp.someserver.com", 25, "username", "password").sendMail(email);
 * 
* * @author Benny Bottema * @see MimeEmailMessageWrapper * @see Email */ public class Mailer { private static final Logger logger = Logger.getLogger(Mailer.class); /** * Encoding used for setting body text, email address, headers, reply-to fields etc. ({@value #CHARACTER_ENCODING}). */ private static final String CHARACTER_ENCODING = "UTF-8"; /** * Used to actually send the email. This session can come from being passed in the default constructor, or made by Mailer * directly, when no Session instance was provided. * * @see #Mailer(Session) * @see #Mailer(String, Integer, String, String, TransportStrategy) */ private final Session session; /** * The transport protocol strategy enum that actually handles the session configuration. Session configuration meaning setting the right * properties for the appropriate transport type (ie. "mail.smtp.host" for SMTP, "mail.smtps.host" for SMTPS). */ private TransportStrategy transportStrategy; /** * Email address restriction flags set either by constructor or overridden by getter by user. * * @see EmailAddressValidationCriteria */ private EmailAddressValidationCriteria emailAddressValidationCriteria; /** * Default constructor, stores the given mail session for later use. Assumes that *all* properties used to make a connection are * configured (host, port, authentication and transport protocol settings). *

* Also defines a default email address validation criteria object, which remains true to RFC 2822, meaning allowing both domain * literals and quoted identifiers (see {@link EmailAddressValidationCriteria#EmailAddressValidationCriteria(boolean, boolean)}). * * @param session A preconfigured mail {@link Session} object with which a {@link Message} can be produced. */ public Mailer(final Session session) { this.session = session; this.emailAddressValidationCriteria = new EmailAddressValidationCriteria(true, true); } /** * Overloaded constructor which produces a new {@link Session} on the fly. Use this if you don't have a mail session configured in your * web container, or Spring context etc. *

* Also defines a default email address validation criteria object, which remains true to RFC 2822, meaning allowing both domain * literals and quoted identifiers (see {@link EmailAddressValidationCriteria#EmailAddressValidationCriteria(boolean, boolean)}). * * @param host The address URL of the SMTP server to be used. * @param port The port of the SMTP server. * @param username An optional username, may be null. * @param password An optional password, may be null, but only if username is null as well. * @param transportStrategy The transport protocol configuration type for handling SSL or TLS (or vanilla SMTP) */ public Mailer(final String host, final Integer port, final String username, final String password, final TransportStrategy transportStrategy) { // we're doing these validations manually instead of using Apache Commons to avoid another dependency if (host == null || host.trim().equals("")) { throw new MailException(MailException.MISSING_HOST); } else if ((password != null && !password.trim().equals("")) && (username == null || username.trim().equals(""))) { throw new MailException(MailException.MISSING_USERNAME); } this.transportStrategy = transportStrategy; this.session = createMailSession(host, port, username, password); this.emailAddressValidationCriteria = new EmailAddressValidationCriteria(true, true); } /** * Actually instantiates and configures the {@link Session} instance. Delegates resolving transport protocol specific properties to the * {@link #transportStrategy} in two ways: *

    *
  1. request an initial property list which the strategy may pre-populate
  2. *
  3. by requesting the property names according to the respective transport protocol it handles (for the host property for example it * would be "mail.smtp.host" for SMTP and "mail.smtps.host" for SMTPS)
  4. *
* * @param host The address URL of the SMTP server to be used. * @param port The port of the SMTP server. * @param username An optional username, may be null. * @param password An optional password, may be null. * @return A fully configured Session instance complete with transport protocol settings. * @see TransportStrategy#generateProperties() * @see TransportStrategy#propertyNameHost() * @see TransportStrategy#propertyNamePort() * @see TransportStrategy#propertyNameUsername() * @see TransportStrategy#propertyNameAuthenticate() */ public Session createMailSession(final String host, final Integer port, final String username, final String password) { if (transportStrategy == null) { logger.warn("Transport Strategy not set, using plain SMTP strategy instead!"); transportStrategy = TransportStrategy.SMTP_PLAIN; } Properties props = transportStrategy.generateProperties(); props.put(transportStrategy.propertyNameHost(), host); if (port != null) { props.put(transportStrategy.propertyNamePort(), String.valueOf(port)); } else { // let JavaMail's Transport objects determine deault port base don the used protocol } if (username != null) { props.put(transportStrategy.propertyNameUsername(), username); } if (password != null) { props.put(transportStrategy.propertyNameAuthenticate(), "true"); return Session.getInstance(props, new Authenticator() { @Override protected PasswordAuthentication getPasswordAuthentication() { return new PasswordAuthentication(username, password); } }); } else { return Session.getInstance(props); } } /** * Overloaded constructor which produces a new {@link Session} on the fly, using default vanilla SMTP transport protocol. * * @param host The address URL of the SMTP server to be used. * @param port The port of the SMTP server. * @param username An optional username, may be null. * @param password An optional password, may be null, but only if username is null as well. * @see #Mailer(String, Integer, String, String, TransportStrategy) */ public Mailer(final String host, final Integer port, final String username, final String password) { this(host, port, username, password, TransportStrategy.SMTP_PLAIN); } /** * Actually sets {@link Session#setDebug(boolean)} so that it generates debug information. * * @param debug Flag to indicate debug mode yes/no. */ public void setDebug(boolean debug) { session.setDebug(debug); } /** * Processes an {@link Email} instance into a completely configured {@link Message}. *

* Sends the Sun JavaMail {@link Message} object using {@link Session#getTransport()}. It will call {@link Transport#connect()} assuming * all connection details have been configured in the provided {@link Session} instance. *

* Performs a call to {@link Message#saveChanges()} as the Sun JavaMail API indicates it is needed to configure the message headers and * providing a message id. * * @param email The information for the email to be sent. * @throws MailException Can be thrown if an email isn't validating correctly, or some other problem occurs during connection, sending * etc. * @see #validate(Email) * @see #prepareMessage(Email, MimeEmailMessageWrapper) * @see #setRecipients(Email, Message) * @see #setTexts(Email, MimeMultipart) * @see #setEmbeddedImages(Email, MimeMultipart) * @see #setAttachments(Email, MimeMultipart) */ public final void sendMail(final Email email) throws MailException { if (validate(email)) { try { // create new wrapper for each mail being sent (enable sending multiple emails with one mailer) final MimeEmailMessageWrapper messageRoot = new MimeEmailMessageWrapper(); // fill and send wrapped mime message parts final Message message = prepareMessage(email, messageRoot); logSession(session, transportStrategy); message.saveChanges(); // some headers and id's will be set for this specific message Transport transport = session.getTransport(); transport.connect(); transport.sendMessage(message, message.getAllRecipients()); transport.close(); } catch (final UnsupportedEncodingException e) { logger.error(e.getMessage(), e); throw new MailException(String.format(MailException.INVALID_ENCODING, e.getMessage())); } catch (final MessagingException e) { logger.error(e.getMessage(), e); throw new MailException(String.format(MailException.GENERIC_ERROR, e.getMessage()), e); } } } /** * Simply logs host details, credentials used and whether authentication will take place and finally the transport protocol used. */ private void logSession(Session session, TransportStrategy transportStrategy) { final String logmsg = "starting mail session (host: %s, port: %s, username: %s, authenticate: %s, transport: %s)"; Properties properties = session.getProperties(); logger.debug(String.format(logmsg, properties.get(transportStrategy.propertyNameHost()), properties.get(transportStrategy.propertyNamePort()), properties.get(transportStrategy.propertyNameUsername()), properties.get(transportStrategy.propertyNameAuthenticate()), transportStrategy)); } /** * Validates an {@link Email} instance. Validation fails if the subject is missing, content is missing, or no recipients are defined. * * @param email The email that needs to be configured correctly. * @return Always true (throws a {@link MailException} exception if validation fails). * @throws MailException Is being thrown in any of the above causes. * @see EmailValidationUtil */ public boolean validate(final Email email) throws MailException { if (email.getText() == null && email.getTextHTML() == null) { throw new MailException(MailException.MISSING_CONTENT); } else if (email.getSubject() == null || email.getSubject().equals("")) { throw new MailException(MailException.MISSING_SUBJECT); } else if (email.getRecipients().size() == 0) { throw new MailException(MailException.MISSING_RECIPIENT); } else if (email.getFromRecipient() == null) { throw new MailException(MailException.MISSING_SENDER); } else { if (!EmailValidationUtil.isValid(email.getFromRecipient().getAddress(), emailAddressValidationCriteria)) { throw new MailException(String.format(MailException.INVALID_SENDER, email)); } for (final Recipient recipient : email.getRecipients()) { if (!EmailValidationUtil.isValid(recipient.getAddress(), emailAddressValidationCriteria)) { throw new MailException(String.format(MailException.INVALID_RECIPIENT, email)); } } if (email.getReplyToRecipient() != null) { if (!EmailValidationUtil.isValid(email.getReplyToRecipient().getAddress(), emailAddressValidationCriteria)) { throw new MailException(String.format(MailException.INVALID_REPLYTO, email)); } } } return true; } /** * Creates a new {@link MimeMessage} instance and prepares it in the email structure, so that it can be filled and send. *

* Fills subject, from,reply-to, content, sent-date, recipients, texts, embedded images, attachments, content and adds all headers. * * @param email The email message from which the subject and From-address are extracted. * @param messageRoot The root of the email which holds everything (filled with some email data). * @return A fully preparated {@link Message} instance, ready to be sent. * @throws MessagingException Kan gegooid worden als het message niet goed behandelt wordt. * @throws UnsupportedEncodingException Zie {@link InternetAddress#InternetAddress(String, String)}. */ private Message prepareMessage(final Email email, final MimeEmailMessageWrapper messageRoot) throws MessagingException, UnsupportedEncodingException { final MimeMessage message = new MimeMessage(session); // set basic email properties message.setSubject(email.getSubject(), CHARACTER_ENCODING); message.setFrom(new InternetAddress(email.getFromRecipient().getAddress(), email.getFromRecipient().getName(), CHARACTER_ENCODING)); setReplyTo(email, message); setRecipients(email, message); // fill multipart structure setTexts(email, messageRoot.multipartAlternativeMessages); setEmbeddedImages(email, messageRoot.multipartRelated); setAttachments(email, messageRoot.multipartRoot); message.setContent(messageRoot.multipartRoot); setHeaders(email, message); message.setSentDate(new Date()); return message; } /** * Fills the {@link Message} instance with recipients from the {@link Email}. * * @param email The message in which the recipients are defined. * @param message The javax message that needs to be filled with recipients. * @throws UnsupportedEncodingException See {@link InternetAddress#InternetAddress(String, String)}. * @throws MessagingException See {@link Message#addRecipient(javax.mail.Message.RecipientType, Address)} */ private void setRecipients(final Email email, final Message message) throws UnsupportedEncodingException, MessagingException { for (final Recipient recipient : email.getRecipients()) { final Address address = new InternetAddress(recipient.getAddress(), recipient.getName()); message.addRecipient(recipient.getType(), address); } } /** * Fills the {@link Message} instance with reply-to address. * * @param email The message in which the recipients are defined. * @param message The javax message that needs to be filled with reply-to address. * @throws UnsupportedEncodingException See {@link InternetAddress#InternetAddress(String, String)}. * @throws MessagingException See {@link Message#setReplyTo(Address[])} */ private void setReplyTo(final Email email, final Message message) throws UnsupportedEncodingException, MessagingException { final Recipient replyToRecipient = email.getReplyToRecipient(); if (replyToRecipient != null) { InternetAddress replyToAddress = new InternetAddress(replyToRecipient.getAddress(), replyToRecipient.getName(), CHARACTER_ENCODING); message.setReplyTo(new Address[] { replyToAddress }); } } /** * Fills the {@link Message} instance with the content bodies (text and html). * * @param email The message in which the content is defined. * @param multipartAlternativeMessages See {@link MimeMultipart#addBodyPart(BodyPart)} * @throws MessagingException See {@link BodyPart#setText(String)}, {@link BodyPart#setContent(Object, String)} and * {@link MimeMultipart#addBodyPart(BodyPart)}. */ private void setTexts(final Email email, final MimeMultipart multipartAlternativeMessages) throws MessagingException { if (email.getText() != null) { final MimeBodyPart messagePart = new MimeBodyPart(); messagePart.setText(email.getText(), CHARACTER_ENCODING); multipartAlternativeMessages.addBodyPart(messagePart); } if (email.getTextHTML() != null) { final MimeBodyPart messagePartHTML = new MimeBodyPart(); messagePartHTML.setContent(email.getTextHTML(), "text/html; charset=\"" + CHARACTER_ENCODING + "\""); multipartAlternativeMessages.addBodyPart(messagePartHTML); } } /** * Fills the {@link Message} instance with the embedded images from the {@link Email}. * * @param email The message in which the embedded images are defined. * @param multipartRelated The branch in the email structure in which we'll stuff the embedded images. * @throws MessagingException See {@link MimeMultipart#addBodyPart(BodyPart)} and * {@link #getBodyPartFromDatasource(AttachmentResource, String)} */ private void setEmbeddedImages(final Email email, final MimeMultipart multipartRelated) throws MessagingException { for (final AttachmentResource embeddedImage : email.getEmbeddedImages()) { multipartRelated.addBodyPart(getBodyPartFromDatasource(embeddedImage, Part.INLINE)); } } /** * Fills the {@link Message} instance with the attachments from the {@link Email}. * * @param email The message in which the attachments are defined. * @param multipartRoot The branch in the email structure in which we'll stuff the attachments. * @throws MessagingException See {@link MimeMultipart#addBodyPart(BodyPart)} and * {@link #getBodyPartFromDatasource(AttachmentResource, String)} */ private void setAttachments(final Email email, final MimeMultipart multipartRoot) throws MessagingException { for (final AttachmentResource resource : email.getAttachments()) { multipartRoot.addBodyPart(getBodyPartFromDatasource(resource, Part.ATTACHMENT)); } } /** * Sets all headers on the {@link Message} instance. Since we're not using a high-level JavaMail method, the JavaMail library says we * need to do some encoding and 'folding' manually, to get the value right for the headers (see {@link MimeUtility}. * * @param email The message in which the headers are defined. * @param message The {@link Message} on which to set the raw, encoded and folded headers. * @throws UnsupportedEncodingException See {@link MimeUtility#encodeText(String, String, String)} * @throws MessagingException See {@link Message#addHeader(String, String)} * @see {@link MimeUtility#encodeText(String, String, String)} * @see MimeUtility#fold(int, String) */ private void setHeaders(final Email email, final Message message) throws UnsupportedEncodingException, MessagingException { // add headers (for raw message headers we need to 'fold' them using MimeUtility for (Map.Entry header : email.getHeaders().entrySet()) { String headerName = header.getKey(); String headerValue = MimeUtility.encodeText(header.getValue(), CHARACTER_ENCODING, null); String foldedHeaderValue = MimeUtility.fold(headerName.length() + 2, headerValue); message.addHeader(header.getKey(), foldedHeaderValue); } } /** * Helper method which generates a {@link BodyPart} from an {@link AttachmentResource} (from its {@link DataSource}) and a disposition * type ({@link Part#INLINE} or {@link Part#ATTACHMENT}). With this the attachment data can be converted into objects that fit in the * email structure.
*
* For every attachment and embedded image a header needs to be set. * * @param resource An object that describes the attachment and contains the actual content data. * @param dispositionType The type of attachment, {@link Part#INLINE} or {@link Part#ATTACHMENT} . * @return An object with the attachment data read for placement in the email structure. * @throws MessagingException All BodyPart setters. */ private BodyPart getBodyPartFromDatasource(final AttachmentResource resource, final String dispositionType) throws MessagingException { final BodyPart attachmentPart = new MimeBodyPart(); final DataSource ds = resource.getDataSource(); // setting headers isn't working nicely using the javax mail API, so let's do that manually attachmentPart.setDataHandler(new DataHandler(resource.getDataSource())); attachmentPart.setFileName(resource.getName()); attachmentPart.setHeader("Content-Type", ds.getContentType() + "; filename=" + ds.getName() + "; name=" + ds.getName()); attachmentPart.setHeader("Content-ID", String.format("<%s>", ds.getName())); attachmentPart.setDisposition(dispositionType + "; size=0"); return attachmentPart; } /** * This class conveniently wraps all necessary mimemessage parts that need to be filled with content, attachments etc. The root is * ultimately sent using JavaMail.
*
* The constructor creates a new email message constructed from {@link MimeMultipart} as follows: * *

	 * - root
	 * 	- related
	 * 		- alternative
	 * 			- mail tekst
	 * 			- mail html tekst
	 * 		- embedded images
	 * 	- attachments
	 * 
* * @author Benny Bottema */ private class MimeEmailMessageWrapper { private final MimeMultipart multipartRoot; private final MimeMultipart multipartRelated; private final MimeMultipart multipartAlternativeMessages; /** * Creates an email skeleton structure, so that embedded images, attachments and (html) texts are being processed properly. */ MimeEmailMessageWrapper() { multipartRoot = new MimeMultipart("mixed"); final MimeBodyPart contentRelated = new MimeBodyPart(); multipartRelated = new MimeMultipart("related"); final MimeBodyPart contentAlternativeMessages = new MimeBodyPart(); multipartAlternativeMessages = new MimeMultipart("alternative"); try { // construct mail structure multipartRoot.addBodyPart(contentRelated); contentRelated.setContent(multipartRelated); multipartRelated.addBodyPart(contentAlternativeMessages); contentAlternativeMessages.setContent(multipartAlternativeMessages); } catch (final MessagingException e) { logger.error(e.getMessage(), e); throw new RuntimeException(e.getMessage(), e); } } } /** * Overrides the default email address validation restrictions when validating and sending emails using the current Mailer * instance. * * @param emailAddressValidationCriteria Refer to * {@link EmailAddressValidationCriteria#EmailAddressValidationCriteria(boolean, boolean)}. */ public void setEmailAddressValidationCriteria(EmailAddressValidationCriteria emailAddressValidationCriteria) { this.emailAddressValidationCriteria = emailAddressValidationCriteria; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy