/**
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.camel.component.mail;
import java.io.IOException;
import java.io.InputStream;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import javax.mail.BodyPart;
import javax.mail.Message;
import javax.mail.MessagingException;
import javax.mail.Multipart;
import javax.mail.internet.MimeMultipart;
import javax.mail.search.SearchTerm;
import com.sun.mail.imap.SortTerm;
import org.apache.camel.Converter;
import org.apache.camel.Exchange;
import org.apache.camel.FallbackConverter;
import org.apache.camel.NoTypeConversionAvailableException;
import org.apache.camel.TypeConverter;
import org.apache.camel.converter.IOConverter;
import org.apache.camel.spi.TypeConverterRegistry;
/**
* JavaMail specific converters.
*
* @version
*/
@Converter
public final class MailConverters {
private static final String NOW_DATE_FORMAT = "yyyy-MM-dd HH:mm:SS";
// the now syntax: "now-24h" or "now - 24h" = the last 24 hours etc.
private static final Pattern NOW_PATTERN = Pattern.compile("now\\s?(\\+|\\-)\\s?(.*)");
private MailConverters() {
//Utility Class
}
/**
* Converts the given JavaMail message to a String body.
* Can return null.
*/
@Converter
public static String toString(Message message) throws MessagingException, IOException {
Object content = message.getContent();
while (content instanceof MimeMultipart) {
MimeMultipart multipart = (MimeMultipart) content;
if (multipart.getCount() > 0) {
BodyPart part = multipart.getBodyPart(0);
content = part.getContent();
}
}
if (content != null) {
return content.toString();
}
return null;
}
/**
* Converts the given JavaMail multipart to a String body, where the content-type of the multipart
* must be text based (ie start with text). Can return null.
*/
@Converter
public static String toString(Multipart multipart) throws MessagingException, IOException {
int size = multipart.getCount();
for (int i = 0; i < size; i++) {
BodyPart part = multipart.getBodyPart(i);
Object content = part.getContent();
while (content instanceof MimeMultipart) {
if (multipart.getCount() < 1) {
break;
}
part = ((MimeMultipart)content).getBodyPart(0);
content = part.getContent();
}
if (part.getContentType().toLowerCase().startsWith("text")) {
return part.getContent().toString();
}
}
return null;
}
/**
* Converts the given JavaMail message to an InputStream.
*/
@Converter
public static InputStream toInputStream(Message message) throws IOException, MessagingException {
return message.getInputStream();
}
/**
* Converts the given JavaMail multipart to a InputStream body, where the content-type of the multipart
* must be text based (ie start with text). Can return null.
*/
@Converter
public static InputStream toInputStream(Multipart multipart, Exchange exchange) throws IOException, MessagingException {
String s = toString(multipart);
if (s == null) {
return null;
}
return IOConverter.toInputStream(s, exchange);
}
/**
* Converts a JavaMail multipart into a body of any type a String can be
* converted into. The content-type of the part must be text based.
*/
@FallbackConverter
public static <T> T convertTo(Class<T> type, Exchange exchange, Object value, TypeConverterRegistry registry) throws MessagingException, IOException {
if (Multipart.class.isAssignableFrom(value.getClass())) {
TypeConverter tc = registry.lookup(type, String.class);
if (tc != null) {
String s = toString((Multipart)value);
if (s != null) {
return tc.convertTo(type, s);
}
}
}
return null;
}
public static SearchTerm toSearchTerm(SimpleSearchTerm simple, TypeConverter typeConverter) throws ParseException, NoTypeConversionAvailableException {
SearchTermBuilder builder = new SearchTermBuilder();
if (simple.isUnseen()) {
builder = builder.unseen();
}
if (simple.getSubjectOrBody() != null) {
String text = simple.getSubjectOrBody();
SearchTermBuilder builderTemp = new SearchTermBuilder();
builderTemp = builderTemp.subject(text).body(SearchTermBuilder.Op.or, text);
builder = builder.and(builderTemp.build());
}
if (simple.getSubject() != null) {
builder = builder.subject(simple.getSubject());
}
if (simple.getBody() != null) {
builder = builder.body(simple.getBody());
}
if (simple.getFrom() != null) {
builder = builder.from(simple.getFrom());
}
if (simple.getTo() != null) {
builder = builder.recipient(Message.RecipientType.TO, simple.getTo());
}
if (simple.getFromSentDate() != null) {
String s = simple.getFromSentDate();
if (s.startsWith("now")) {
long offset = extractOffset(s, typeConverter);
builder = builder.and(new NowSearchTerm(SearchTermBuilder.Comparison.GE.asNum(), true, offset));
} else {
SimpleDateFormat sdf = new SimpleDateFormat(NOW_DATE_FORMAT);
Date date = sdf.parse(s);
builder = builder.sent(SearchTermBuilder.Comparison.GE, date);
}
}
if (simple.getToSentDate() != null) {
String s = simple.getToSentDate();
if (s.startsWith("now")) {
long offset = extractOffset(s, typeConverter);
builder = builder.and(new NowSearchTerm(SearchTermBuilder.Comparison.LE.asNum(), true, offset));
} else {
SimpleDateFormat sdf = new SimpleDateFormat(NOW_DATE_FORMAT);
Date date = sdf.parse(s);
builder = builder.sent(SearchTermBuilder.Comparison.LE, date);
}
}
if (simple.getFromReceivedDate() != null) {
String s = simple.getFromReceivedDate();
if (s.startsWith("now")) {
long offset = extractOffset(s, typeConverter);
builder = builder.and(new NowSearchTerm(SearchTermBuilder.Comparison.GE.asNum(), false, offset));
} else {
SimpleDateFormat sdf = new SimpleDateFormat(NOW_DATE_FORMAT);
Date date = sdf.parse(s);
builder = builder.received(SearchTermBuilder.Comparison.GE, date);
}
}
if (simple.getToReceivedDate() != null) {
String s = simple.getToReceivedDate();
if (s.startsWith("now")) {
long offset = extractOffset(s, typeConverter);
builder = builder.and(new NowSearchTerm(SearchTermBuilder.Comparison.LE.asNum(), false, offset));
} else {
SimpleDateFormat sdf = new SimpleDateFormat(NOW_DATE_FORMAT);
Date date = sdf.parse(s);
builder = builder.received(SearchTermBuilder.Comparison.LE, date);
}
}
return builder.build();
}
/*
* Converts from comma separated list of sort terms to SortTerm obj array
*/
@Converter
public static SortTerm[] toSortTerm(String sortTerm) {
ArrayList<SortTerm> result = new ArrayList<SortTerm>();
if (sortTerm == null) {
return null;
}
String[] sortTerms = sortTerm.split(",");
for (String key : sortTerms) {
if ("arrival".equals(key)) {
result.add(SortTerm.ARRIVAL);
} else if ("cc".equals(key)) {
result.add(SortTerm.CC);
} else if ("date".equals(key)) {
result.add(SortTerm.DATE);
} else if ("from".equals(key)) {
result.add(SortTerm.FROM);
} else if ("reverse".equals(key)) {
result.add(SortTerm.REVERSE);
} else if ("size".equals(key)) {
result.add(SortTerm.SIZE);
} else if ("subject".equals(key)) {
result.add(SortTerm.SUBJECT);
} else if ("to".equals(key)) {
result.add(SortTerm.TO);
}
}
if (result.size() > 0) {
return result.toArray(new SortTerm[result.size()]);
} else {
return null;
}
}
private static long extractOffset(String now, TypeConverter typeConverter) throws NoTypeConversionAvailableException {
Matcher matcher = NOW_PATTERN.matcher(now);
if (matcher.matches()) {
String op = matcher.group(1);
String remainder = matcher.group(2);
// convert remainder to a time millis (eg we have a String -> long converter that supports
// syntax with hours, days, minutes: eg 5h30m for 5 hours and 30 minutes).
long offset = typeConverter.mandatoryConvertTo(long.class, remainder);
if ("+".equals(op)) {
return offset;
} else {
return -1 * offset;
}
}
return 0;
}
}