Ensure envelope/body

pull/178/head
M66B 5 years ago
parent 0ba03a5020
commit 5fc2b08db9

@ -166,7 +166,7 @@ public class ActivityEML extends ActivityBase {
Session isession = Session.getInstance(props, null); Session isession = Session.getInstance(props, null);
MimeMessage imessage = new MimeMessage(isession, is); MimeMessage imessage = new MimeMessage(isession, is);
MessageHelper helper = new MessageHelper(imessage); MessageHelper helper = new MessageHelper(imessage, context);
result.from = MessageHelper.formatAddresses(helper.getFrom()); result.from = MessageHelper.formatAddresses(helper.getFrom());
result.to = MessageHelper.formatAddresses(helper.getTo()); result.to = MessageHelper.formatAddresses(helper.getTo());
@ -176,7 +176,7 @@ public class ActivityEML extends ActivityBase {
result.sent = helper.getSent(); result.sent = helper.getSent();
result.received = helper.getReceived(); result.received = helper.getReceived();
result.subject = helper.getSubject(); result.subject = helper.getSubject();
result.parts = helper.getMessageParts(context); result.parts = helper.getMessageParts();
String html = result.parts.getHtml(context); String html = result.parts.getHtml(context);
if (html != null) { if (html != null) {

@ -1061,7 +1061,7 @@ class Core {
boolean found = false; boolean found = false;
for (Message imessage : imessages) { for (Message imessage : imessages) {
MessageHelper helper = new MessageHelper((MimeMessage) imessage); MessageHelper helper = new MessageHelper((MimeMessage) imessage, context);
String msgid; String msgid;
if (caps.containsKey("UIDL")) if (caps.containsKey("UIDL"))
@ -1145,7 +1145,7 @@ class Core {
if (imessage == null) if (imessage == null)
throw new MessageRemovedException(); throw new MessageRemovedException();
MessageHelper helper = new MessageHelper(imessage); MessageHelper helper = new MessageHelper(imessage, context);
db.message().setMessageHeaders(message.id, helper.getHeaders()); db.message().setMessageHeaders(message.id, helper.getHeaders());
} }
@ -1198,8 +1198,8 @@ class Core {
if (imessage == null) if (imessage == null)
throw new MessageRemovedException(); throw new MessageRemovedException();
MessageHelper helper = new MessageHelper((MimeMessage) imessage); MessageHelper helper = new MessageHelper((MimeMessage) imessage, context);
MessageHelper.MessageParts parts = helper.getMessageParts(context); MessageHelper.MessageParts parts = helper.getMessageParts();
String body = parts.getHtml(context); String body = parts.getHtml(context);
File file = message.getFile(context); File file = message.getFile(context);
Helper.writeText(file, body); Helper.writeText(file, body);
@ -1232,8 +1232,8 @@ class Core {
throw new MessageRemovedException(); throw new MessageRemovedException();
// Get message parts // Get message parts
MessageHelper helper = new MessageHelper((MimeMessage) imessage); MessageHelper helper = new MessageHelper((MimeMessage) imessage, context);
MessageHelper.MessageParts parts = helper.getMessageParts(context); MessageHelper.MessageParts parts = helper.getMessageParts();
// Download attachment // Download attachment
parts.downloadAttachment(context, attachment); parts.downloadAttachment(context, attachment);
@ -1547,7 +1547,7 @@ class Core {
if (!state.isRunning()) if (!state.isRunning())
return; return;
MessageHelper helper = new MessageHelper((MimeMessage) imessage); MessageHelper helper = new MessageHelper((MimeMessage) imessage, context);
String uidl = caps.containsKey("UIDL") String uidl = caps.containsKey("UIDL")
? ifolder.getUID(imessage) ? ifolder.getUID(imessage)
: helper.getMessageID(); : helper.getMessageID();
@ -1580,7 +1580,7 @@ class Core {
sent = 0L; sent = 0L;
String authentication = helper.getAuthentication(); String authentication = helper.getAuthentication();
MessageHelper.MessageParts parts = helper.getMessageParts(context); MessageHelper.MessageParts parts = helper.getMessageParts();
EntityMessage message = new EntityMessage(); EntityMessage message = new EntityMessage();
message.account = folder.account; message.account = folder.account;
@ -1986,10 +1986,11 @@ class Core {
try { try {
// Some providers erroneously return old messages // Some providers erroneously return old messages
if (full.contains(isub[j])) { if (full.contains(isub[j])) {
Date received = isub[j].getReceivedDate(); MessageHelper helper = new MessageHelper((MimeMessage) isub[j], context);
long received = helper.getReceived();
boolean unseen = (sync_unseen && !isub[j].isSet(Flags.Flag.SEEN)); boolean unseen = (sync_unseen && !isub[j].isSet(Flags.Flag.SEEN));
boolean flagged = (sync_flagged && isub[j].isSet(Flags.Flag.FLAGGED)); boolean flagged = (sync_flagged && isub[j].isSet(Flags.Flag.FLAGGED));
if (received != null && received.getTime() < keep_time && !unseen && !flagged) { if (received < keep_time && !unseen && !flagged) {
long uid = ifolder.getUID(isub[j]); long uid = ifolder.getUID(isub[j]);
Log.i(folder.name + " Skipping old uid=" + uid + " date=" + received); Log.i(folder.name + " Skipping old uid=" + uid + " date=" + received);
ids[from + j] = null; ids[from + j] = null;
@ -2127,7 +2128,7 @@ class Core {
throw new MessageRemovedException("Flagged deleted"); throw new MessageRemovedException("Flagged deleted");
} }
MessageHelper helper = new MessageHelper(imessage); MessageHelper helper = new MessageHelper(imessage, context);
boolean seen = helper.getSeen(); boolean seen = helper.getSeen();
boolean answered = helper.getAnsered(); boolean answered = helper.getAnsered();
boolean flagged = helper.getFlagged(); boolean flagged = helper.getFlagged();
@ -2197,7 +2198,7 @@ class Core {
} }
String authentication = helper.getAuthentication(); String authentication = helper.getAuthentication();
MessageHelper.MessageParts parts = helper.getMessageParts(context); MessageHelper.MessageParts parts = helper.getMessageParts();
message = new EntityMessage(); message = new EntityMessage();
message.account = folder.account; message.account = folder.account;
@ -2762,8 +2763,8 @@ class Core {
// fp.add(GmailFolder.FetchProfileItem.THRID); // fp.add(GmailFolder.FetchProfileItem.THRID);
//ifolder.fetch(new Message[]{imessage}, fp); //ifolder.fetch(new Message[]{imessage}, fp);
MessageHelper helper = new MessageHelper(imessage); MessageHelper helper = new MessageHelper(imessage, context);
MessageHelper.MessageParts parts = helper.getMessageParts(context); MessageHelper.MessageParts parts = helper.getMessageParts();
if (!message.content) { if (!message.content) {
if (state.getNetworkState().isUnmetered() || if (state.getNetworkState().isUnmetered() ||

@ -5314,8 +5314,8 @@ public class FragmentMessages extends FragmentBase implements SharedPreferences.
Session isession = Session.getInstance(props, null); Session isession = Session.getInstance(props, null);
try (InputStream fis = new FileInputStream(plain)) { try (InputStream fis = new FileInputStream(plain)) {
MimeMessage imessage = new MimeMessage(isession, fis); MimeMessage imessage = new MimeMessage(isession, fis);
MessageHelper helper = new MessageHelper(imessage); MessageHelper helper = new MessageHelper(imessage, context);
parts = helper.getMessageParts(context); parts = helper.getMessageParts();
} }
try { try {
@ -5917,8 +5917,8 @@ public class FragmentMessages extends FragmentBase implements SharedPreferences.
Properties props = MessageHelper.getSessionProperties(); Properties props = MessageHelper.getSessionProperties();
Session isession = Session.getInstance(props, null); Session isession = Session.getInstance(props, null);
MimeMessage imessage = new MimeMessage(isession, is); MimeMessage imessage = new MimeMessage(isession, is);
MessageHelper helper = new MessageHelper(imessage); MessageHelper helper = new MessageHelper(imessage, context);
MessageHelper.MessageParts parts = helper.getMessageParts(context); MessageHelper.MessageParts parts = helper.getMessageParts();
DB db = DB.getInstance(context); DB db = DB.getInstance(context);
try { try {

@ -29,6 +29,7 @@ import androidx.documentfile.provider.DocumentFile;
import androidx.preference.PreferenceManager; import androidx.preference.PreferenceManager;
import com.sun.mail.gimap.GmailMessage; import com.sun.mail.gimap.GmailMessage;
import com.sun.mail.imap.IMAPMessage;
import com.sun.mail.util.ASCIIUtility; import com.sun.mail.util.ASCIIUtility;
import com.sun.mail.util.BASE64DecoderStream; import com.sun.mail.util.BASE64DecoderStream;
import com.sun.mail.util.FolderClosedIOException; import com.sun.mail.util.FolderClosedIOException;
@ -100,8 +101,12 @@ import biweekly.Biweekly;
import biweekly.ICalendar; import biweekly.ICalendar;
public class MessageHelper { public class MessageHelper {
private boolean ensuredEnvelope = false;
private boolean ensuredBody = false;
private MimeMessage imessage; private MimeMessage imessage;
private static File cacheDir = null;
static final int SMALL_MESSAGE_SIZE = 64 * 1024; // bytes static final int SMALL_MESSAGE_SIZE = 64 * 1024; // bytes
static final int DEFAULT_ATTACHMENT_DOWNLOAD_SIZE = 256 * 1024; // bytes static final int DEFAULT_ATTACHMENT_DOWNLOAD_SIZE = 256 * 1024; // bytes
@ -747,7 +752,9 @@ public class MessageHelper {
} }
} }
MessageHelper(MimeMessage message) { MessageHelper(MimeMessage message, Context context) {
if (cacheDir == null)
cacheDir = context.getCacheDir();
this.imessage = message; this.imessage = message;
} }
@ -779,6 +786,8 @@ public class MessageHelper {
} }
String getMessageID() throws MessagingException { String getMessageID() throws MessagingException {
ensureMessage(false);
// Outlook outbox -> sent // Outlook outbox -> sent
String header = imessage.getHeader("X-Correlation-ID", null); String header = imessage.getHeader("X-Correlation-ID", null);
if (header == null) if (header == null)
@ -787,6 +796,8 @@ public class MessageHelper {
} }
String[] getReferences() throws MessagingException { String[] getReferences() throws MessagingException {
ensureMessage(false);
List<String> result = new ArrayList<>(); List<String> result = new ArrayList<>();
String refs = imessage.getHeader("References", null); String refs = imessage.getHeader("References", null);
if (refs != null) if (refs != null)
@ -847,6 +858,8 @@ public class MessageHelper {
} }
String getDeliveredTo() throws MessagingException { String getDeliveredTo() throws MessagingException {
ensureMessage(false);
String header = imessage.getHeader("Delivered-To", null); String header = imessage.getHeader("Delivered-To", null);
if (header == null) if (header == null)
header = imessage.getHeader("X-Delivered-To", null); header = imessage.getHeader("X-Delivered-To", null);
@ -861,11 +874,15 @@ public class MessageHelper {
} }
String getInReplyTo() throws MessagingException { String getInReplyTo() throws MessagingException {
ensureMessage(false);
String header = imessage.getHeader("In-Reply-To", null); String header = imessage.getHeader("In-Reply-To", null);
return (header == null ? null : MimeUtility.unfold(header)); return (header == null ? null : MimeUtility.unfold(header));
} }
String getThreadId(Context context, long account, long uid) throws MessagingException { String getThreadId(Context context, long account, long uid) throws MessagingException {
ensureMessage(false);
if (imessage instanceof GmailMessage) { if (imessage instanceof GmailMessage) {
// https://developers.google.com/gmail/imap/imap-extensions#access_to_the_gmail_thread_id_x-gm-thrid // https://developers.google.com/gmail/imap/imap-extensions#access_to_the_gmail_thread_id_x-gm-thrid
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context); SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
@ -927,6 +944,8 @@ public class MessageHelper {
Integer getPriority() throws MessagingException { Integer getPriority() throws MessagingException {
Integer priority = null; Integer priority = null;
ensureMessage(false);
// https://docs.microsoft.com/en-us/openspecs/exchange_server_protocols/ms-oxcmail/2bb19f1b-b35e-4966-b1cb-1afd044e83ab // https://docs.microsoft.com/en-us/openspecs/exchange_server_protocols/ms-oxcmail/2bb19f1b-b35e-4966-b1cb-1afd044e83ab
String header = imessage.getHeader("Importance", null); String header = imessage.getHeader("Importance", null);
if (header == null) if (header == null)
@ -985,15 +1004,21 @@ public class MessageHelper {
} }
boolean getReceiptRequested() throws MessagingException { boolean getReceiptRequested() throws MessagingException {
ensureMessage(false);
return (imessage.getHeader("Return-Receipt-To") != null || return (imessage.getHeader("Return-Receipt-To") != null ||
imessage.getHeader("Disposition-Notification-To") != null); imessage.getHeader("Disposition-Notification-To") != null);
} }
Address[] getReceiptTo() throws MessagingException { Address[] getReceiptTo() throws MessagingException {
ensureMessage(false);
return getAddressHeader("Disposition-Notification-To"); return getAddressHeader("Disposition-Notification-To");
} }
String getAuthentication() throws MessagingException { String getAuthentication() throws MessagingException {
ensureMessage(false);
String header = imessage.getHeader("Authentication-Results", null); String header = imessage.getHeader("Authentication-Results", null);
return (header == null ? null : MimeUtility.unfold(header)); return (header == null ? null : MimeUtility.unfold(header));
} }
@ -1024,6 +1049,8 @@ public class MessageHelper {
} }
String getReceivedFromHost() throws MessagingException { String getReceivedFromHost() throws MessagingException {
ensureMessage(false);
String[] received = imessage.getHeader("Received"); String[] received = imessage.getHeader("Received");
if (received == null || received.length == 0) if (received == null || received.length == 0)
return null; return null;
@ -1062,6 +1089,8 @@ public class MessageHelper {
} }
private Address[] getAddressHeader(String name) throws MessagingException { private Address[] getAddressHeader(String name) throws MessagingException {
ensureMessage(false);
String header = imessage.getHeader(name, ","); String header = imessage.getHeader(name, ",");
if (header == null) if (header == null)
return null; return null;
@ -1121,6 +1150,8 @@ public class MessageHelper {
} }
Address[] getListPost() throws MessagingException { Address[] getListPost() throws MessagingException {
ensureMessage(false);
String list; String list;
try { try {
// https://www.ietf.org/rfc/rfc2369.txt // https://www.ietf.org/rfc/rfc2369.txt
@ -1159,6 +1190,8 @@ public class MessageHelper {
} }
String getListUnsubscribe() throws MessagingException { String getListUnsubscribe() throws MessagingException {
ensureMessage(false);
String list; String list;
try { try {
// https://www.ietf.org/rfc/rfc2369.txt // https://www.ietf.org/rfc/rfc2369.txt
@ -1203,6 +1236,8 @@ public class MessageHelper {
} }
String getAutocrypt() throws MessagingException { String getAutocrypt() throws MessagingException {
ensureMessage(false);
String autocrypt = imessage.getHeader("Autocrypt", null); String autocrypt = imessage.getHeader("Autocrypt", null);
if (autocrypt == null) if (autocrypt == null)
return null; return null;
@ -1211,6 +1246,8 @@ public class MessageHelper {
} }
String getSubject() throws MessagingException { String getSubject() throws MessagingException {
ensureMessage(false);
String subject = imessage.getHeader("Subject", null); String subject = imessage.getHeader("Subject", null);
if (subject == null) if (subject == null)
return null; return null;
@ -1227,11 +1264,15 @@ public class MessageHelper {
} }
Long getSize() throws MessagingException { Long getSize() throws MessagingException {
ensureMessage(false);
long size = imessage.getSize(); long size = imessage.getSize();
return (size < 0 ? null : size); return (size < 0 ? null : size);
} }
long getReceived() throws MessagingException { long getReceived() throws MessagingException {
ensureMessage(false);
Date received = imessage.getReceivedDate(); Date received = imessage.getReceivedDate();
if (received == null) if (received == null)
received = imessage.getSentDate(); received = imessage.getSentDate();
@ -1239,11 +1280,15 @@ public class MessageHelper {
} }
Long getSent() throws MessagingException { Long getSent() throws MessagingException {
ensureMessage(false);
Date date = imessage.getSentDate(); Date date = imessage.getSentDate();
return (date == null ? null : date.getTime()); return (date == null ? null : date.getTime());
} }
String getHeaders() throws MessagingException { String getHeaders() throws MessagingException {
ensureMessage(false);
StringBuilder sb = new StringBuilder(); StringBuilder sb = new StringBuilder();
Enumeration<Header> headers = imessage.getAllHeaders(); Enumeration<Header> headers = imessage.getAllHeaders();
while (headers.hasMoreElements()) { while (headers.hasMoreElements()) {
@ -1817,33 +1862,10 @@ public class MessageHelper {
EntityAttachment attachment; EntityAttachment attachment;
} }
MessageParts getMessageParts(Context context) throws IOException, MessagingException { MessageParts getMessageParts() throws IOException, MessagingException {
MessageParts parts = new MessageParts(); MessageParts parts = new MessageParts();
try { ensureMessage(true);
imessage.getContentType();
} catch (MessagingException ex) {
// https://javaee.github.io/javamail/FAQ#imapserverbug
if ("Failed to load IMAP envelope".equals(ex.getMessage()) ||
"Unable to load BODYSTRUCTURE".equals(ex.getMessage())) {
Log.w("Fetching raw message");
File file = File.createTempFile("serverbug", null, context.getCacheDir());
try (OutputStream os = new BufferedOutputStream(new FileOutputStream(file))) {
imessage.writeTo(os);
}
Properties props = MessageHelper.getSessionProperties();
Session isession = Session.getInstance(props, null);
Log.w("Decoding raw message");
try (InputStream is = new BufferedInputStream(new FileInputStream(file))) {
imessage = new MimeMessageEx(isession, is, imessage);
}
file.delete();
} else
throw ex;
}
try { try {
MimePart part = imessage; MimePart part = imessage;
@ -2044,6 +2066,53 @@ public class MessageHelper {
} }
} }
private void ensureMessage(boolean body) throws MessagingException {
if (body ? ensuredBody : ensuredEnvelope)
return;
if (body)
ensuredBody = true;
else
ensuredEnvelope = true;
Log.i("Ensure body=" + body);
try {
if (imessage instanceof IMAPMessage) {
if (body)
imessage.getContentType(); // force loadBODYSTRUCTURE
else
imessage.getMessageID(); // force loadEnvelope
}
} catch (MessagingException ex) {
// https://javaee.github.io/javamail/FAQ#imapserverbug
if ("Failed to load IMAP envelope".equals(ex.getMessage()) ||
"Unable to load BODYSTRUCTURE".equals(ex.getMessage()))
try {
Log.w("Fetching raw message");
File file = File.createTempFile("serverbug", null, cacheDir);
try (OutputStream os = new BufferedOutputStream(new FileOutputStream(file))) {
imessage.writeTo(os);
}
Properties props = MessageHelper.getSessionProperties();
Session isession = Session.getInstance(props, null);
Log.w("Decoding raw message");
try (InputStream is = new BufferedInputStream(new FileInputStream(file))) {
imessage = new MimeMessageEx(isession, is, imessage);
}
file.delete();
} catch (IOException ex1) {
Log.e(ex1);
throw ex;
}
else
throw ex;
}
}
static String sanitizeKeyword(String keyword) { static String sanitizeKeyword(String keyword) {
// https://tools.ietf.org/html/rfc3501 // https://tools.ietf.org/html/rfc3501
StringBuilder sb = new StringBuilder(); StringBuilder sb = new StringBuilder();

@ -430,14 +430,14 @@ public class ServiceSend extends ServiceBase {
long id = message.id; long id = message.id;
imessage.saveChanges(); imessage.saveChanges();
MessageHelper helper = new MessageHelper(imessage); MessageHelper helper = new MessageHelper(imessage, ServiceSend.this);
if (message.uid != null) { if (message.uid != null) {
Log.e("Outbox id=" + message.id + " uid=" + message.uid); Log.e("Outbox id=" + message.id + " uid=" + message.uid);
message.uid = null; message.uid = null;
} }
MessageHelper.MessageParts parts = helper.getMessageParts(this); MessageHelper.MessageParts parts = helper.getMessageParts();
String body = parts.getHtml(this); String body = parts.getHtml(this);
try { try {

Loading…
Cancel
Save