1
0
mirror of https://github.com/moparisthebest/k-9 synced 2025-01-03 17:58:10 -05:00

Merge imap-parser branch.

Fixes issue 1547.
This commit is contained in:
cketti 2010-05-19 13:31:48 +00:00
parent 0e3f9a9db4
commit 22ce159fe6
5 changed files with 432 additions and 207 deletions

View File

@ -1727,11 +1727,7 @@ public class MessagingController implements Runnable
*/
for (Part part : viewables)
{
fp.clear();
fp.add(part);
// TODO what happens if the network connection dies? We've got partial
// messages with incorrect status stored.
remoteFolder.fetch(new Message[] { message }, fp, null);
remoteFolder.fetchPart(message, part, null);
}
// Store the updated message locally
localFolder.appendMessages(new Message[] { message });
@ -3175,9 +3171,11 @@ public class MessagingController implements Runnable
remoteFolder = remoteStore.getFolder(message.getFolder().getName());
remoteFolder.open(OpenMode.READ_WRITE);
FetchProfile fp = new FetchProfile();
fp.add(part);
remoteFolder.fetch(new Message[] { message }, fp, null);
//FIXME: This is an ugly hack that won't be needed once the Message objects have been united.
Message remoteMessage = remoteFolder.getMessage(message.getUid());
remoteMessage.setBody(message.getBody());
remoteFolder.fetchPart(remoteMessage, part, null);
localFolder.updateMessage((LocalMessage)message);
for (MessagingListener l : getListeners())
{

View File

@ -15,7 +15,7 @@ import java.util.ArrayList;
* any information it needs to download the content.
* </pre>
*/
public class FetchProfile extends ArrayList<Object>
public class FetchProfile extends ArrayList<FetchProfile.Item>
{
/**
* Default items available for pre-fetching. It should be expected that any

View File

@ -132,6 +132,12 @@ public abstract class Folder
public abstract void fetch(Message[] messages, FetchProfile fp,
MessageRetrievalListener listener) throws MessagingException;
public void fetchPart(Message message, Part part,
MessageRetrievalListener listener) throws MessagingException
{
throw new RuntimeException("fetchPart() not implemented.");
}
public abstract void delete(boolean recurse) throws MessagingException;
public abstract String getName();

View File

@ -1,7 +1,3 @@
/**
*
*/
package com.fsck.k9.mail.store;
import android.util.Log;
@ -9,7 +5,6 @@ import com.fsck.k9.K9;
import com.fsck.k9.FixedLengthInputStream;
import com.fsck.k9.PeekableInputStream;
import com.fsck.k9.mail.MessagingException;
import java.io.IOException;
import java.io.InputStream;
import java.text.ParseException;
@ -20,19 +15,24 @@ import java.util.Locale;
public class ImapResponseParser
{
SimpleDateFormat mDateTimeFormat = new SimpleDateFormat("dd-MMM-yyyy HH:mm:ss Z", Locale.US);
private static final SimpleDateFormat mDateTimeFormat = new SimpleDateFormat("dd-MMM-yyyy HH:mm:ss Z", Locale.US);
private static final SimpleDateFormat badDateTimeFormat = new SimpleDateFormat("dd MMM yyyy HH:mm:ss Z", Locale.US);
private static final SimpleDateFormat badDateTimeFormat2 = new SimpleDateFormat("E, dd MMM yyyy HH:mm:ss Z", Locale.US);
SimpleDateFormat badDateTimeFormat = new SimpleDateFormat("dd MMM yyyy HH:mm:ss Z", Locale.US);
SimpleDateFormat badDateTimeFormat2 = new SimpleDateFormat("E, dd MMM yyyy HH:mm:ss Z", Locale.US);
PeekableInputStream mIn;
InputStream mActiveLiteral;
private PeekableInputStream mIn;
private ImapResponse mResponse;
private Exception mException;
public ImapResponseParser(PeekableInputStream in)
{
this.mIn = in;
}
public ImapResponse readResponse() throws IOException
{
return readResponse(null);
}
/**
* Reads the next response available on the stream and returns an
* ImapResponse object that represents it.
@ -40,49 +40,60 @@ public class ImapResponseParser
* @return
* @throws IOException
*/
public ImapResponse readResponse() throws IOException
public ImapResponse readResponse(IImapResponseCallback callback) throws IOException
{
ImapResponse response = new ImapResponse();
if (mActiveLiteral != null)
try
{
while (mActiveLiteral.read() != -1)
;
mActiveLiteral = null;
ImapResponse response = new ImapResponse();
mResponse = response;
mResponse.mCallback = callback;
int ch = mIn.peek();
if (ch == '*')
{
parseUntaggedResponse();
readTokens(response);
}
else if (ch == '+')
{
response.mCommandContinuationRequested =
parseCommandContinuationRequest();
readTokens(response);
}
else
{
response.mTag = parseTaggedResponse();
readTokens(response);
}
if (K9.DEBUG)
{
Log.v(K9.LOG_TAG, "<<< " + response.toString());
}
if (mException != null)
{
throw new RuntimeException("readResponse(): Exception in callback method", mException);
}
return response;
}
int ch = mIn.peek();
if (ch == '*')
finally
{
parseUntaggedResponse();
readTokens(response);
mResponse.mCallback = null;
mResponse = null;
mException = null;
}
else if (ch == '+')
{
response.mCommandContinuationRequested =
parseCommandContinuationRequest();
readTokens(response);
}
else
{
response.mTag = parseTaggedResponse();
readTokens(response);
}
if (K9.DEBUG)
{
Log.v(K9.LOG_TAG, "<<< " + response.toString());
}
return response;
}
private void readTokens(ImapResponse response) throws IOException
{
response.clear();
Object token;
while ((token = readToken()) != null)
while ((token = readToken(response)) != null)
{
response.add(token);
if (mActiveLiteral != null)
if (!(token instanceof ImapList))
{
break;
response.add(token);
}
}
response.mCompleted = token == null;
@ -99,36 +110,30 @@ public class ImapResponseParser
* tokens.
* @throws IOException
*/
public Object readToken() throws IOException
private Object readToken(ImapResponse response) throws IOException
{
while (true)
{
Object token = parseToken();
if (token == null || !token.equals(")") || !token.equals("]"))
Object token = parseToken(response);
if (token == null || !(token.equals(")") || token.equals("]")))
{
return token;
}
}
}
private Object parseToken() throws IOException
private Object parseToken(ImapList parent) throws IOException
{
if (mActiveLiteral != null)
{
while (mActiveLiteral.read() != -1)
;
mActiveLiteral = null;
}
while (true)
{
int ch = mIn.peek();
if (ch == '(')
{
return parseList();
return parseList(parent);
}
else if (ch == '[')
{
return parseSequence();
return parseSequence(parent);
}
else if (ch == ')')
{
@ -146,8 +151,7 @@ public class ImapResponseParser
}
else if (ch == '{')
{
mActiveLiteral = parseLiteral();
return mActiveLiteral;
return parseLiteral();
}
else if (ch == ' ')
{
@ -196,27 +200,27 @@ public class ImapResponseParser
return tag;
}
private ImapList parseList() throws IOException
private ImapList parseList(ImapList parent) throws IOException
{
expect('(');
ImapList list = new ImapList();
parent.add(list);
Object token;
while (true)
{
token = parseToken();
token = parseToken(list);
if (token == null)
{
break;
}
else if (token instanceof InputStream)
{
list.add(token);
break;
}
else if (token.equals(")"))
{
break;
}
else if (token instanceof ImapList)
{
// Do nothing
}
else
{
list.add(token);
@ -225,27 +229,27 @@ public class ImapResponseParser
return list;
}
private ImapList parseSequence() throws IOException
private ImapList parseSequence(ImapList parent) throws IOException
{
expect('[');
ImapList list = new ImapList();
parent.add(list);
Object token;
while (true)
{
token = parseToken();
token = parseToken(list);
if (token == null)
{
break;
}
else if (token instanceof InputStream)
{
list.add(token);
break;
}
else if (token.equals("]"))
{
break;
}
else if (token instanceof ImapList)
{
// Do nothing
}
else
{
list.add(token);
@ -297,14 +301,66 @@ public class ImapResponseParser
* @param mListener
* @throws IOException
*/
private InputStream parseLiteral() throws IOException
private Object parseLiteral() throws IOException
{
expect('{');
int size = Integer.parseInt(readStringUntil('}'));
expect('\r');
expect('\n');
FixedLengthInputStream fixed = new FixedLengthInputStream(mIn, size);
return fixed;
if (size == 0)
{
return "";
}
if (mResponse.mCallback != null)
{
FixedLengthInputStream fixed = new FixedLengthInputStream(mIn, size);
Object result = null;
try
{
result = mResponse.mCallback.foundLiteral(mResponse, fixed);
}
catch (IOException e)
{
// Pass IOExceptions through
throw e;
}
catch (Exception e)
{
// Catch everything else and save it for later.
mException = e;
//Log.e(K9.LOG_TAG, "parseLiteral(): Exception in callback method", e);
}
// Check if only some of the literal data was read
int available = fixed.available();
if ((available > 0) && (available != size))
{
// If so, skip the rest
fixed.skip(fixed.available());
}
if (result != null)
{
return result;
}
}
byte[] data = new byte[size];
int read = 0;
while (read != size)
{
int count = mIn.read(data, read, size - read);
if (count == -1)
{
throw new IOException("parseLiteral(): end of stream reached");
}
read += count;
}
return new String(data, "US-ASCII");
}
/**
@ -536,6 +592,7 @@ public class ImapResponseParser
public class ImapResponse extends ImapList
{
private boolean mCompleted;
private IImapResponseCallback mCallback;
boolean mCommandContinuationRequested;
String mTag;
@ -595,4 +652,27 @@ public class ImapResponseParser
return o1 == o2;
}
}
public interface IImapResponseCallback
{
/**
* Callback method that is called by the parser when a literal string
* is found in an IMAP response.
*
* @param response ImapResponse object with the fields that have been
* parsed up until now (excluding the literal string).
* @param literal FixedLengthInputStream that can be used to access
* the literal string.
*
* @return an Object that will be put in the ImapResponse object at the
* place of the literal string.
*
* @throws IOException passed-through if thrown by FixedLengthInputStream
* @throws Exception if something goes wrong. Parsing will be resumed
* and the exception will be thrown after the
* complete IMAP response has been parsed.
*/
public Object foundLiteral(ImapResponse response, FixedLengthInputStream literal)
throws IOException, Exception;
}
}

View File

@ -7,6 +7,7 @@ import android.net.NetworkInfo;
import android.os.PowerManager;
import android.util.Log;
import com.fsck.k9.Account;
import com.fsck.k9.FixedLengthInputStream;
import com.fsck.k9.K9;
import com.fsck.k9.PeekableInputStream;
import com.fsck.k9.Utility;
@ -50,8 +51,6 @@ import java.util.concurrent.atomic.AtomicInteger;
* <pre>
* TODO Need to start keeping track of UIDVALIDITY
* TODO Need a default response handler for things like folder updates
* TODO In fetch(), if we need a ImapMessage and were given
* something else we can try to do a pre-fetch first.
* </pre>
*/
public class ImapStore extends Store
@ -1133,26 +1132,6 @@ public class ImapStore extends Store
{
fetchFields.add("BODY.PEEK[]");
}
for (Object o : fp)
{
if (o != null && o instanceof Part)
{
Part part = (Part) o;
String[] parts = part.getHeader(MimeHeader.HEADER_ANDROID_ATTACHMENT_STORE_DATA);
if (parts != null)
{
String partId = parts[0];
if ("TEXT".equalsIgnoreCase(partId))
{
fetchFields.add(String.format("BODY.PEEK[TEXT]<0.%d>", FETCH_BODY_SANE_SUGGESTED_SIZE));
}
else
{
fetchFields.add("BODY.PEEK[" + partId + "]");
}
}
}
}
try
{
@ -1162,11 +1141,16 @@ public class ImapStore extends Store
), false);
ImapResponse response;
int messageNumber = 0;
ImapResponseParser.IImapResponseCallback callback = null;
if (fp.contains(FetchProfile.Item.BODY) || fp.contains(FetchProfile.Item.BODY_SANE) || fp.contains(FetchProfile.Item.ENVELOPE))
{
callback = new FetchBodyCallback(messageMap);
}
do
{
response = mConnection.readResponse();
if (K9.DEBUG)
Log.v(K9.LOG_TAG, "response for fetch: " + response + " for " + getLogId());
response = mConnection.readResponse(callback);
if (response.mTag == null && ImapResponseParser.equalsIgnoreCase(response.get(1), "FETCH"))
{
@ -1205,117 +1189,25 @@ public class ImapStore extends Store
ImapMessage imapMessage = (ImapMessage) message;
if (fetchList.containsKey("FLAGS"))
{
ImapList flags = fetchList.getKeyedList("FLAGS");
if (flags != null)
{
for (int i = 0, count = flags.size(); i < count; i++)
{
String flag = flags.getString(i);
if (flag.equalsIgnoreCase("\\Deleted"))
{
imapMessage.setFlagInternal(Flag.DELETED, true);
}
else if (flag.equalsIgnoreCase("\\Answered"))
{
imapMessage.setFlagInternal(Flag.ANSWERED, true);
}
else if (flag.equalsIgnoreCase("\\Seen"))
{
imapMessage.setFlagInternal(Flag.SEEN, true);
}
else if (flag.equalsIgnoreCase("\\Flagged"))
{
imapMessage.setFlagInternal(Flag.FLAGGED, true);
}
}
}
}
Object literal = handleFetchResponse(imapMessage, fetchList);
if (fetchList.containsKey("INTERNALDATE"))
if (literal != null)
{
Date internalDate = fetchList.getKeyedDate("INTERNALDATE");
message.setInternalDate(internalDate);
}
if (fetchList.containsKey("RFC822.SIZE"))
{
int size = fetchList.getKeyedNumber("RFC822.SIZE");
imapMessage.setSize(size);
}
if (fetchList.containsKey("BODYSTRUCTURE"))
{
ImapList bs = fetchList.getKeyedList("BODYSTRUCTURE");
if (bs != null)
{
try
{
parseBodyStructure(bs, message, "TEXT");
}
catch (MessagingException e)
{
if (K9.DEBUG)
Log.d(K9.LOG_TAG, "Error handling message for " + getLogId(), e);
message.setBody(null);
}
}
}
if (fetchList.containsKey("BODY"))
{
Part part = null;
for (Object o : fp)
{
if (o instanceof Part)
{
part = (Part) o;
break;
}
}
int index = fetchList.getKeyIndex("BODY") + 2;
Object literal = fetchList.getObject(index);
// Check if there's an origin octet
if (literal instanceof String)
{
String originOctet = (String)literal;
if (originOctet.startsWith("<"))
{
literal = fetchList.getObject(index + 1);
}
}
InputStream bodyStream;
if (literal instanceof InputStream)
{
bodyStream = (InputStream)literal;
}
else if (literal instanceof String)
{
String bodyString = (String)literal;
if (K9.DEBUG)
Log.v(K9.LOG_TAG, "Part is a String: '" + bodyString + "' for " + getLogId());
bodyStream = new ByteArrayInputStream(bodyString.getBytes());
InputStream bodyStream = new ByteArrayInputStream(bodyString.getBytes());
imapMessage.parse(bodyStream);
}
else if (literal instanceof Integer)
{
// All the work was done in FetchBodyCallback.foundLiteral()
}
else
{
// This shouldn't happen
throw new MessagingException("Got FETCH response with bogus parameters");
}
if (part != null)
{
String contentTransferEncoding = part.getHeader(
MimeHeader.HEADER_CONTENT_TRANSFER_ENCODING)[0];
part.setBody(MimeUtility.decodeBody(bodyStream, contentTransferEncoding));
}
else
{
imapMessage.parse(bodyStream);
}
}
if (listener != null)
@ -1339,6 +1231,193 @@ public class ImapStore extends Store
}
}
@Override
public void fetchPart(Message message, Part part, MessageRetrievalListener listener)
throws MessagingException
{
checkOpen();
String[] parts = part.getHeader(MimeHeader.HEADER_ANDROID_ATTACHMENT_STORE_DATA);
if (parts == null)
{
return;
}
String fetch;
String partId = parts[0];
if ("TEXT".equalsIgnoreCase(partId))
{
fetch = String.format("BODY.PEEK[TEXT]<0.%d>", FETCH_BODY_SANE_SUGGESTED_SIZE);
}
else
{
fetch = String.format("BODY.PEEK[%s]", partId);
}
try
{
mConnection.sendCommand(
String.format("UID FETCH %s (UID %s)", message.getUid(), fetch),
false);
ImapResponse response;
int messageNumber = 0;
ImapResponseParser.IImapResponseCallback callback = new FetchPartCallback(part);
do
{
response = mConnection.readResponse(callback);
if ((response.mTag == null) &&
(ImapResponseParser.equalsIgnoreCase(response.get(1), "FETCH")))
{
ImapList fetchList = (ImapList)response.getKeyedValue("FETCH");
String uid = fetchList.getKeyedString("UID");
if (!message.getUid().equals(uid))
{
if (K9.DEBUG)
Log.d(K9.LOG_TAG, "Did not ask for UID " + uid + " for " + getLogId());
handleUntaggedResponse(response);
continue;
}
if (listener != null)
{
listener.messageStarted(uid, messageNumber++, 1);
}
ImapMessage imapMessage = (ImapMessage) message;
Object literal = handleFetchResponse(imapMessage, fetchList);
if (literal != null)
{
if (literal instanceof Body)
{
// Most of the work was done in FetchAttchmentCallback.foundLiteral()
part.setBody((Body)literal);
}
else if (literal instanceof String)
{
String bodyString = (String)literal;
InputStream bodyStream = new ByteArrayInputStream(bodyString.getBytes());
String contentTransferEncoding = part.getHeader(
MimeHeader.HEADER_CONTENT_TRANSFER_ENCODING)[0];
part.setBody(MimeUtility.decodeBody(bodyStream, contentTransferEncoding));
}
else
{
// This shouldn't happen
throw new MessagingException("Got FETCH response with bogus parameters");
}
}
if (listener != null)
{
listener.messageFinished(message, messageNumber, 1);
}
}
else
{
handleUntaggedResponse(response);
}
while (response.more());
}
while (response.mTag == null);
}
catch (IOException ioe)
{
throw ioExceptionHandler(mConnection, ioe);
}
}
// Returns value of body field
private Object handleFetchResponse(ImapMessage message, ImapList fetchList) throws MessagingException
{
Object result = null;
if (fetchList.containsKey("FLAGS"))
{
ImapList flags = fetchList.getKeyedList("FLAGS");
if (flags != null)
{
for (int i = 0, count = flags.size(); i < count; i++)
{
String flag = flags.getString(i);
if (flag.equalsIgnoreCase("\\Deleted"))
{
message.setFlagInternal(Flag.DELETED, true);
}
else if (flag.equalsIgnoreCase("\\Answered"))
{
message.setFlagInternal(Flag.ANSWERED, true);
}
else if (flag.equalsIgnoreCase("\\Seen"))
{
message.setFlagInternal(Flag.SEEN, true);
}
else if (flag.equalsIgnoreCase("\\Flagged"))
{
message.setFlagInternal(Flag.FLAGGED, true);
}
}
}
}
if (fetchList.containsKey("INTERNALDATE"))
{
Date internalDate = fetchList.getKeyedDate("INTERNALDATE");
message.setInternalDate(internalDate);
}
if (fetchList.containsKey("RFC822.SIZE"))
{
int size = fetchList.getKeyedNumber("RFC822.SIZE");
message.setSize(size);
}
if (fetchList.containsKey("BODYSTRUCTURE"))
{
ImapList bs = fetchList.getKeyedList("BODYSTRUCTURE");
if (bs != null)
{
try
{
parseBodyStructure(bs, message, "TEXT");
}
catch (MessagingException e)
{
if (K9.DEBUG)
Log.d(K9.LOG_TAG, "Error handling message for " + getLogId(), e);
message.setBody(null);
}
}
}
if (fetchList.containsKey("BODY"))
{
int index = fetchList.getKeyIndex("BODY") + 2;
result = fetchList.getObject(index);
// Check if there's an origin octet
if (result instanceof String)
{
String originOctet = (String)result;
if (originOctet.startsWith("<"))
{
result = fetchList.getObject(index + 1);
}
}
}
return result;
}
@Override
public Flag[] getPermanentFlags() throws MessagingException
{
@ -2384,10 +2463,15 @@ public class ImapStore extends Store
}
private ImapResponse readResponse() throws IOException, MessagingException
{
return readResponse(null);
}
private ImapResponse readResponse(ImapResponseParser.IImapResponseCallback callback) throws IOException, MessagingException
{
try
{
ImapResponse response = mParser.readResponse();
ImapResponse response = mParser.readResponse(callback);
if (K9.DEBUG)
Log.v(K9.LOG_TAG, getLogId() + "<<<" + response);
@ -3369,4 +3453,61 @@ public class ImapStore extends Store
{
List<ImapResponse> search() throws IOException, MessagingException;
}
private class FetchBodyCallback implements ImapResponseParser.IImapResponseCallback
{
private HashMap<String, Message> mMessageMap;
FetchBodyCallback(HashMap<String, Message> mesageMap)
{
mMessageMap = mesageMap;
}
@Override
public Object foundLiteral(ImapResponse response,
FixedLengthInputStream literal) throws IOException, Exception
{
if (response.mTag == null &&
ImapResponseParser.equalsIgnoreCase(response.get(1), "FETCH"))
{
ImapList fetchList = (ImapList)response.getKeyedValue("FETCH");
String uid = fetchList.getKeyedString("UID");
ImapMessage message = (ImapMessage) mMessageMap.get(uid);
message.parse(literal);
// Return placeholder object
return new Integer(1);
}
return null;
}
}
private class FetchPartCallback implements ImapResponseParser.IImapResponseCallback
{
private Part mPart;
FetchPartCallback(Part part)
{
mPart = part;
}
@Override
public Object foundLiteral(ImapResponse response,
FixedLengthInputStream literal) throws IOException, Exception
{
if (response.mTag == null &&
ImapResponseParser.equalsIgnoreCase(response.get(1), "FETCH"))
{
//TODO: check for correct UID
String contentTransferEncoding = mPart.getHeader(
MimeHeader.HEADER_CONTENT_TRANSFER_ENCODING)[0];
return MimeUtility.decodeBody(literal, contentTransferEncoding);
}
return null;
}
}
}