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

org.simplejavamail.converter.internal.mimemessage.MimeMessageHelper Maven / Gradle / Ivy

/*
 * Copyright © 2009 Benny Bottema ([email protected])
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.simplejavamail.converter.internal.mimemessage;

import jakarta.activation.DataHandler;
import jakarta.activation.DataSource;
import jakarta.mail.Address;
import jakarta.mail.BodyPart;
import jakarta.mail.Message;
import jakarta.mail.MessagingException;
import jakarta.mail.Part;
import jakarta.mail.internet.InternetAddress;
import jakarta.mail.internet.MimeBodyPart;
import jakarta.mail.internet.MimeMessage;
import jakarta.mail.internet.MimeMultipart;
import jakarta.mail.internet.MimePart;
import jakarta.mail.internet.MimeUtility;
import jakarta.mail.internet.ParameterList;
import lombok.val;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.simplejavamail.api.email.AttachmentResource;
import org.simplejavamail.api.email.ContentTransferEncoding;
import org.simplejavamail.api.email.Email;
import org.simplejavamail.api.email.Recipient;
import org.simplejavamail.api.internal.general.MessageHeader;
import org.simplejavamail.internal.util.MiscUtil;
import org.simplejavamail.internal.util.NamedDataSource;

import java.io.UnsupportedEncodingException;
import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
import java.util.Collection;
import java.util.Map;
import java.util.UUID;

import static java.lang.Boolean.TRUE;
import static java.lang.String.format;
import static java.nio.charset.StandardCharsets.UTF_8;
import static java.util.Objects.requireNonNull;
import static java.util.Optional.ofNullable;
import static org.simplejavamail.internal.util.MiscUtil.valueNullOrEmpty;
import static org.simplejavamail.internal.util.Preconditions.checkNonEmptyArgument;

/**
 * Helper class that produces and populates a mime messages. Deals with jakarta.mail RFC MimeMessage stuff, as well as DKIM signing.
 */
public class MimeMessageHelper {

	/**
	 * Encoding used for setting body text, email address, headers, reply-to fields etc. ({@link StandardCharsets#UTF_8}).
	 */
	private static final Charset CHARACTER_ENCODING = UTF_8;

	static void setSubject(@NotNull final Email email, final MimeMessage message) throws MessagingException {
		message.setSubject(email.getSubject(), CHARACTER_ENCODING.name());
	}

	static void setFrom(@NotNull final Email email, final MimeMessage message) throws MessagingException {
		val fromRecipient = email.getFromRecipient();
		if (fromRecipient != null) {
			message.setFrom(MiscUtil.asInternetAddress(fromRecipient, CHARACTER_ENCODING));
		}
	}
	
	/**
	 * 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 MessagingException           See {@link Message#addRecipient(Message.RecipientType, Address)}
	 */
	static void setRecipients(final Email email, final Message message)
			throws MessagingException {
		for (final Recipient recipient : email.getRecipients()) {
			message.addRecipient(recipient.getType(), MiscUtil.asInternetAddress(recipient, CHARACTER_ENCODING));
		}
	}

	/**
	 * Fills the {@link Message} instance with reply-to address(es).
	 *
	 * @param email   The message in which the recipients are defined.
	 * @param message The javax message that needs to be filled with reply-to addresses.
	 * @throws MessagingException           See {@link Message#setReplyTo(Address[])}
	 */
	static void setReplyTo(@NotNull final Email email, final Message message)
			throws MessagingException {
		if (!email.getReplyToRecipients().isEmpty()) {
			val replyToAddresses = new Address[email.getReplyToRecipients().size()];
			int i = 0;
			for (val replyToRecipient : email.getReplyToRecipients()) {
				replyToAddresses[i++] = MiscUtil.asInternetAddress(replyToRecipient, CHARACTER_ENCODING);
			}
			message.setReplyTo(replyToAddresses);
		}
	}

	/**
	 * Fills the {@link Message} instance with the content bodies (text, html and calendar), with Content-Transfer-Encoding header taken from Email.
	 *
	 * @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)}.
	 */
	static void setTexts(@NotNull final Email email, final MimeMultipart multipartAlternativeMessages)
			throws MessagingException {
		if (email.getPlainText() != null) {
			val messagePart = new MimeBodyPart();
			messagePart.setText(email.getPlainText(), CHARACTER_ENCODING.name());
			messagePart.addHeader(MessageHeader.CONTENT_TRANSFER_ENCODING.getName(), determineContentTransferEncoder(email));
			multipartAlternativeMessages.addBodyPart(messagePart);
		}
		if (email.getHTMLText() != null) {
			val messagePartHTML = new MimeBodyPart();
			messagePartHTML.setContent(email.getHTMLText(), format("text/html; charset=\"%s\"", CHARACTER_ENCODING.name()));
			messagePartHTML.addHeader(MessageHeader.CONTENT_TRANSFER_ENCODING.getName(), determineContentTransferEncoder(email));
			multipartAlternativeMessages.addBodyPart(messagePartHTML);
		}
		if (email.getCalendarText() != null) {
			val calendarMethod = requireNonNull(email.getCalendarMethod(), "calendarMethod is required when calendarText is set");
			val messagePartCalendar = new MimeBodyPart();
			messagePartCalendar.setContent(email.getCalendarText(), format("text/calendar; charset=\"%s\"; method=\"%s\"", CHARACTER_ENCODING.name(), calendarMethod));
			messagePartCalendar.addHeader(MessageHeader.CONTENT_TRANSFER_ENCODING.getName(), determineContentTransferEncoder(email));
			multipartAlternativeMessages.addBodyPart(messagePartCalendar);
		}
	}

	private static String determineContentTransferEncoder(@NotNull Email email) {
		return (email.getContentTransferEncoding() != null
				? email.getContentTransferEncoding()
				: ContentTransferEncoding.getDefault()).getEncoder();
	}

	/**
	 * Fills the {@link MimeBodyPart} instance with the content body content (text, html and calendar), with Content-Transfer-Encoding header taken from Email.
	 *
	 * @param email       The message in which the content is defined.
	 * @param messagePart The {@link MimeBodyPart} that will contain the body content (either plain text, HTML text or iCalendar text)
	 *                    and the Content-Transfer-Encoding header.
	 * @throws MessagingException See {@link BodyPart#setText(String)}, {@link BodyPart#setContent(Object, String)}.
	 */
	static void setTexts(@NotNull final Email email, final MimePart messagePart)
			throws MessagingException {
		if (email.getPlainText() != null) {
			messagePart.setText(email.getPlainText(), CHARACTER_ENCODING.name());
		}
		if (email.getHTMLText() != null) {
			messagePart.setContent(email.getHTMLText(), format("text/html; charset=\"%s\"", CHARACTER_ENCODING.name()));
		}
		if (email.getCalendarText() != null) {
			val calendarMethod = requireNonNull(email.getCalendarMethod(), "CalendarMethod must be set when CalendarText is set");
			messagePart.setContent(email.getCalendarText(), format("text/calendar; charset=\"%s\"; method=\"%s\"", CHARACTER_ENCODING.name(), calendarMethod));
		}
		messagePart.addHeader(MessageHeader.CONTENT_TRANSFER_ENCODING.getName(), determineContentTransferEncoder(email));
	}
	
	/**
	 * If provided, adds the {@code emailToForward} as a MimeBodyPart to the mixed multipart root.
	 * 

* Note: this is done without setting {@code Content-Disposition} so email clients can choose * how to display embedded forwards. Most client will show the forward as inline, some may show it as attachment. */ static void configureForwarding(@NotNull final Email email, @NotNull final MimeMultipart multipartRootMixed) throws MessagingException { if (email.getEmailToForward() != null) { final BodyPart fordwardedMessage = new MimeBodyPart(); fordwardedMessage.setContent(email.getEmailToForward(), "message/rfc822"); multipartRootMixed.addBodyPart(fordwardedMessage); } } /** * 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)} */ static void setEmbeddedImages(@NotNull 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)} */ static void setAttachments(@NotNull final Email email, final MimeMultipart multipartRoot) throws MessagingException { for (final AttachmentResource attachment : email.getAttachments()) { multipartRoot.addBodyPart(getBodyPartFromDatasource(attachment, 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}. *

* Furthermore, sets the notification flags Disposition-Notification-To and Return-Receipt-To if provided. It used * JavaMail's built-in method for producing an RFC compliant email address (see {@link InternetAddress#toString()}). * * @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 MimeUtility#encodeText(String, String, String) * @see MimeUtility#fold(int, String) */ static void setHeaders(@NotNull final Email email, final Message message) throws UnsupportedEncodingException, MessagingException { // add headers (for raw message headers we need to 'fold' them using MimeUtility for (val header : email.getHeaders().entrySet()) { setHeader(message, header); } if (TRUE.equals(email.getUseDispositionNotificationTo())) { final Recipient dispositionTo = checkNonEmptyArgument(email.getDispositionNotificationTo(), "dispositionNotificationTo"); final Address address = MiscUtil.asInternetAddress(dispositionTo, CHARACTER_ENCODING); message.setHeader(MessageHeader.DISPOSITION_NOTIFICATION_TO.getName(), address.toString()); } if (TRUE.equals(email.getUseReturnReceiptTo())) { final Recipient returnReceiptTo = checkNonEmptyArgument(email.getReturnReceiptTo(), "returnReceiptTo"); final Address address = MiscUtil.asInternetAddress(returnReceiptTo, CHARACTER_ENCODING); message.setHeader(MessageHeader.RETURN_RECEIPT_TO.getName(), address.toString()); } } private static void setHeader(Message message, Map.Entry> header) throws UnsupportedEncodingException, MessagingException { for (final String headerValue : header.getValue()) { final String headerName = header.getKey(); final String headerValueEncoded = MimeUtility.encodeText(headerValue, CHARACTER_ENCODING.name(), null); final String foldedHeaderValue = MimeUtility.fold(headerName.length() + 2, headerValueEncoded); 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 attachmentResource 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 static BodyPart getBodyPartFromDatasource(final AttachmentResource attachmentResource, final String dispositionType) throws MessagingException { final BodyPart attachmentPart = new MimeBodyPart(); // setting headers isn't working nicely using the javax mail API, so let's do that manually final String fileName = determineResourceName(attachmentResource, dispositionType, false, false); final String contentID = determineResourceName(attachmentResource, dispositionType, true, true); attachmentPart.setDataHandler(new DataHandler(new NamedDataSource(fileName, attachmentResource.getDataSource()))); attachmentPart.setFileName(fileName); final String contentType = attachmentResource.getDataSource().getContentType(); ParameterList pl = new ParameterList(); pl.set("filename", fileName); pl.set("name", fileName); attachmentPart.setHeader("Content-Type", contentType + pl); attachmentPart.setHeader("Content-ID", format("<%s>", contentID)); attachmentPart.setHeader("Content-Description", determineAttachmentDescription(attachmentResource)); if (!valueNullOrEmpty(attachmentResource.getContentTransferEncoding())) { attachmentPart.setHeader("Content-Transfer-Encoding", attachmentResource.getContentTransferEncoding().getEncoder()); } attachmentPart.setDisposition(dispositionType); return attachmentPart; } /** * Determines the right resource name and optionally attaches the correct extension to the name. The result is mime encoded. */ static String determineResourceName(final AttachmentResource attachmentResource, String dispositionType, final boolean encodeResourceName, final boolean isContentID) { final String datasourceName = attachmentResource.getDataSource().getName(); String resourceName; if (!valueNullOrEmpty(attachmentResource.getName())) { resourceName = attachmentResource.getName(); } else if (!valueNullOrEmpty(datasourceName)) { resourceName = datasourceName; } else { resourceName = "resource" + UUID.randomUUID(); } // if ATTACHMENT, then add UUID to the name to prevent attachments with the same name to reference the same attachment content if (isContentID && dispositionType.equals(Part.ATTACHMENT)) { resourceName += "@" + UUID.randomUUID(); } // if there is no extension on the name, but there is on the datasource name, then add it to the name if (!valueNullOrEmpty(datasourceName) && dispositionType.equals(Part.ATTACHMENT)) { resourceName = possiblyAddExtension(datasourceName, resourceName); } return encodeResourceName ? MiscUtil.encodeText(resourceName) : resourceName; } @NotNull private static String possiblyAddExtension(final String datasourceName, String resourceName) { @SuppressWarnings("UnnecessaryLocalVariable") final String possibleFilename = datasourceName; if (!resourceName.contains(".") && possibleFilename.contains(".")) { final String extension = possibleFilename.substring(possibleFilename.lastIndexOf(".")); if (!resourceName.endsWith(extension)) { resourceName += extension; } } return resourceName; } @Nullable private static String determineAttachmentDescription(AttachmentResource attachmentResource) { return ofNullable(attachmentResource.getDescription()).map(MiscUtil::encodeText).orElse(null); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy