Merge pull request #2454 from philipwhiuk/xoriginalto
Auto select identity based X-Original-To / Delivered-To / X-Envelope-To / TO / CC
This commit is contained in:
commit
f04421f105
5 changed files with 282 additions and 2 deletions
|
@ -17,7 +17,7 @@ import timber.log.Timber;
|
||||||
public abstract class Message implements Part, Body {
|
public abstract class Message implements Part, Body {
|
||||||
|
|
||||||
public enum RecipientType {
|
public enum RecipientType {
|
||||||
TO, CC, BCC,
|
TO, CC, BCC, X_ORIGINAL_TO, DELIVERED_TO, X_ENVELOPE_TO
|
||||||
}
|
}
|
||||||
|
|
||||||
protected String mUid;
|
protected String mUid;
|
||||||
|
|
|
@ -48,6 +48,9 @@ public class MimeMessage extends Message {
|
||||||
protected Address[] mCc;
|
protected Address[] mCc;
|
||||||
protected Address[] mBcc;
|
protected Address[] mBcc;
|
||||||
protected Address[] mReplyTo;
|
protected Address[] mReplyTo;
|
||||||
|
protected Address[] xOriginalTo;
|
||||||
|
protected Address[] deliveredTo;
|
||||||
|
protected Address[] xEnvelopeTo;
|
||||||
|
|
||||||
protected String mMessageId;
|
protected String mMessageId;
|
||||||
private String[] mReferences;
|
private String[] mReferences;
|
||||||
|
@ -85,6 +88,9 @@ public class MimeMessage extends Message {
|
||||||
mCc = null;
|
mCc = null;
|
||||||
mBcc = null;
|
mBcc = null;
|
||||||
mReplyTo = null;
|
mReplyTo = null;
|
||||||
|
xOriginalTo = null;
|
||||||
|
deliveredTo = null;
|
||||||
|
xEnvelopeTo = null;
|
||||||
|
|
||||||
mMessageId = null;
|
mMessageId = null;
|
||||||
mReferences = null;
|
mReferences = null;
|
||||||
|
@ -212,6 +218,24 @@ public class MimeMessage extends Message {
|
||||||
}
|
}
|
||||||
return mBcc;
|
return mBcc;
|
||||||
}
|
}
|
||||||
|
case X_ORIGINAL_TO: {
|
||||||
|
if (xOriginalTo == null) {
|
||||||
|
xOriginalTo = Address.parse(MimeUtility.unfold(getFirstHeader("X-Original-To")));
|
||||||
|
}
|
||||||
|
return xOriginalTo;
|
||||||
|
}
|
||||||
|
case DELIVERED_TO: {
|
||||||
|
if (deliveredTo == null) {
|
||||||
|
deliveredTo = Address.parse(MimeUtility.unfold(getFirstHeader("Delivered-To")));
|
||||||
|
}
|
||||||
|
return deliveredTo;
|
||||||
|
}
|
||||||
|
case X_ENVELOPE_TO: {
|
||||||
|
if (xEnvelopeTo == null) {
|
||||||
|
xEnvelopeTo = Address.parse(MimeUtility.unfold(getFirstHeader("X-Envelope-To")));
|
||||||
|
}
|
||||||
|
return xEnvelopeTo;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
throw new IllegalArgumentException("Unrecognized recipient type.");
|
throw new IllegalArgumentException("Unrecognized recipient type.");
|
||||||
|
@ -243,6 +267,30 @@ public class MimeMessage extends Message {
|
||||||
setHeader("BCC", Address.toEncodedString(addresses));
|
setHeader("BCC", Address.toEncodedString(addresses));
|
||||||
this.mBcc = addresses;
|
this.mBcc = addresses;
|
||||||
}
|
}
|
||||||
|
} else if (type == RecipientType.X_ORIGINAL_TO) {
|
||||||
|
if (addresses == null || addresses.length == 0) {
|
||||||
|
removeHeader("X-Original-To");
|
||||||
|
this.xOriginalTo = null;
|
||||||
|
} else {
|
||||||
|
setHeader("X-Original-To", Address.toEncodedString(addresses));
|
||||||
|
this.xOriginalTo = addresses;
|
||||||
|
}
|
||||||
|
} else if (type == RecipientType.DELIVERED_TO) {
|
||||||
|
if (addresses == null || addresses.length == 0) {
|
||||||
|
removeHeader("Delivered-To");
|
||||||
|
this.deliveredTo = null;
|
||||||
|
} else {
|
||||||
|
setHeader("Delivered-To", Address.toEncodedString(addresses));
|
||||||
|
this.deliveredTo = addresses;
|
||||||
|
}
|
||||||
|
} else if (type == RecipientType.X_ENVELOPE_TO) {
|
||||||
|
if (addresses == null || addresses.length == 0) {
|
||||||
|
removeHeader("X-Envelope-To");
|
||||||
|
this.xEnvelopeTo = null;
|
||||||
|
} else {
|
||||||
|
setHeader("X-Envelope-To", Address.toEncodedString(addresses));
|
||||||
|
this.xEnvelopeTo = addresses;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
throw new IllegalStateException("Unrecognized recipient type.");
|
throw new IllegalStateException("Unrecognized recipient type.");
|
||||||
}
|
}
|
||||||
|
@ -624,6 +672,9 @@ public class MimeMessage extends Message {
|
||||||
destination.mReplyTo = mReplyTo;
|
destination.mReplyTo = mReplyTo;
|
||||||
destination.mReferences = mReferences;
|
destination.mReferences = mReferences;
|
||||||
destination.mInReplyTo = mInReplyTo;
|
destination.mInReplyTo = mInReplyTo;
|
||||||
|
destination.xOriginalTo = xOriginalTo;
|
||||||
|
destination.deliveredTo = deliveredTo;
|
||||||
|
destination.xEnvelopeTo = xEnvelopeTo;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -19,9 +19,11 @@ import com.fsck.k9.mail.BodyPart;
|
||||||
import com.fsck.k9.mail.K9LibRobolectricTestRunner;
|
import com.fsck.k9.mail.K9LibRobolectricTestRunner;
|
||||||
import com.fsck.k9.mail.Message.RecipientType;
|
import com.fsck.k9.mail.Message.RecipientType;
|
||||||
import com.fsck.k9.mail.Multipart;
|
import com.fsck.k9.mail.Multipart;
|
||||||
|
|
||||||
import org.junit.runner.RunWith;
|
import org.junit.runner.RunWith;
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
|
|
||||||
@RunWith(K9LibRobolectricTestRunner.class)
|
@RunWith(K9LibRobolectricTestRunner.class)
|
||||||
|
@ -272,4 +274,63 @@ public class MimeMessageParseTest {
|
||||||
}
|
}
|
||||||
assertEquals(Arrays.asList(expectedParts), actual);
|
assertEquals(Arrays.asList(expectedParts), actual);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void getRecipients_withXOriginalTo() throws Exception {
|
||||||
|
MimeMessage msg = parseWithoutRecurse(toStream(
|
||||||
|
"From: <adam@example.org>\r\n" +
|
||||||
|
"To: <eva@example.org>\r\n" +
|
||||||
|
"X-Original-To: <test@mail.com>\r\n" +
|
||||||
|
"Subject: Testmail\r\n" +
|
||||||
|
"MIME-Version: 1.0\r\n" +
|
||||||
|
"Content-type: text/plain\r\n" +
|
||||||
|
"Content-Transfer-Encoding: 7bit\r\n" +
|
||||||
|
"\r\n" +
|
||||||
|
"this is some test text."));
|
||||||
|
|
||||||
|
Address[] xOriginalAddresses = msg.getRecipients(RecipientType.X_ORIGINAL_TO);
|
||||||
|
|
||||||
|
assertEquals(1, xOriginalAddresses.length);
|
||||||
|
assertEquals(new Address("<test@mail.com>"), xOriginalAddresses[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void getRecipients_withDeliveredTo() throws Exception {
|
||||||
|
MimeMessage msg = parseWithoutRecurse(toStream(
|
||||||
|
"From: <adam@example.org>\r\n" +
|
||||||
|
"To: <eva@example.org>\r\n" +
|
||||||
|
"Delivered-To: <test@mail.com>\r\n" +
|
||||||
|
"Subject: Testmail\r\n" +
|
||||||
|
"MIME-Version: 1.0\r\n" +
|
||||||
|
"Content-type: text/plain\r\n" +
|
||||||
|
"Content-Transfer-Encoding: 7bit\r\n" +
|
||||||
|
"\r\n" +
|
||||||
|
"this is some test text."));
|
||||||
|
|
||||||
|
Address[] deliveredToAddresses = msg.getRecipients(RecipientType.DELIVERED_TO);
|
||||||
|
|
||||||
|
assertEquals(1, deliveredToAddresses.length);
|
||||||
|
assertEquals(new Address("<test@mail.com>"), deliveredToAddresses[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void getRecipients_withXEnvelopeTo() throws Exception {
|
||||||
|
MimeMessage msg = parseWithoutRecurse(toStream(
|
||||||
|
"From: <adam@example.org>\r\n" +
|
||||||
|
"To: <eva@example.org>\r\n" +
|
||||||
|
"X-Envelope-To: <test@mail.com>\r\n" +
|
||||||
|
"Subject: Testmail\r\n" +
|
||||||
|
"MIME-Version: 1.0\r\n" +
|
||||||
|
"Content-type: text/plain\r\n" +
|
||||||
|
"Content-Transfer-Encoding: 7bit\r\n" +
|
||||||
|
"\r\n" +
|
||||||
|
"this is some test text."));
|
||||||
|
|
||||||
|
Address[] xEnvelopeToAddresses = msg.getRecipients(RecipientType.X_ENVELOPE_TO);
|
||||||
|
|
||||||
|
assertEquals(1, xEnvelopeToAddresses.length);
|
||||||
|
assertEquals(new Address("<test@mail.com>"), xEnvelopeToAddresses[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,6 +6,7 @@ import com.fsck.k9.Identity;
|
||||||
import com.fsck.k9.mail.Address;
|
import com.fsck.k9.mail.Address;
|
||||||
import com.fsck.k9.mail.Message;
|
import com.fsck.k9.mail.Message;
|
||||||
|
|
||||||
|
|
||||||
public class IdentityHelper {
|
public class IdentityHelper {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -24,6 +25,35 @@ public class IdentityHelper {
|
||||||
public static Identity getRecipientIdentityFromMessage(Account account, Message message) {
|
public static Identity getRecipientIdentityFromMessage(Account account, Message message) {
|
||||||
Identity recipient = null;
|
Identity recipient = null;
|
||||||
|
|
||||||
|
for (Address address : message.getRecipients(Message.RecipientType.X_ORIGINAL_TO)) {
|
||||||
|
Identity identity = account.findIdentity(address);
|
||||||
|
if (identity != null) {
|
||||||
|
recipient = identity;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (recipient == null) {
|
||||||
|
for (Address address : message.getRecipients(Message.RecipientType.DELIVERED_TO)) {
|
||||||
|
Identity identity = account.findIdentity(address);
|
||||||
|
if (identity != null) {
|
||||||
|
recipient = identity;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (recipient == null) {
|
||||||
|
for (Address address : message.getRecipients(Message.RecipientType.X_ENVELOPE_TO)) {
|
||||||
|
Identity identity = account.findIdentity(address);
|
||||||
|
if (identity != null) {
|
||||||
|
recipient = identity;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (recipient == null) {
|
||||||
for (Address address : message.getRecipients(Message.RecipientType.TO)) {
|
for (Address address : message.getRecipients(Message.RecipientType.TO)) {
|
||||||
Identity identity = account.findIdentity(address);
|
Identity identity = account.findIdentity(address);
|
||||||
if (identity != null) {
|
if (identity != null) {
|
||||||
|
@ -31,6 +61,8 @@ public class IdentityHelper {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (recipient == null) {
|
if (recipient == null) {
|
||||||
Address[] ccAddresses = message.getRecipients(Message.RecipientType.CC);
|
Address[] ccAddresses = message.getRecipients(Message.RecipientType.CC);
|
||||||
if (ccAddresses.length > 0) {
|
if (ccAddresses.length > 0) {
|
||||||
|
|
136
k9mail/src/test/java/com/fsck/k9/helper/IdentityHelperTest.java
Normal file
136
k9mail/src/test/java/com/fsck/k9/helper/IdentityHelperTest.java
Normal file
|
@ -0,0 +1,136 @@
|
||||||
|
package com.fsck.k9.helper;
|
||||||
|
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
|
||||||
|
import com.fsck.k9.Account;
|
||||||
|
import com.fsck.k9.Identity;
|
||||||
|
import com.fsck.k9.K9RobolectricTestRunner;
|
||||||
|
import com.fsck.k9.mail.Message;
|
||||||
|
import com.fsck.k9.mail.Address;
|
||||||
|
import com.fsck.k9.mail.internet.MimeMessage;
|
||||||
|
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.robolectric.RuntimeEnvironment;
|
||||||
|
import org.robolectric.annotation.Config;
|
||||||
|
|
||||||
|
import java.io.ByteArrayInputStream;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
|
@RunWith(K9RobolectricTestRunner.class)
|
||||||
|
public class IdentityHelperTest {
|
||||||
|
|
||||||
|
private Account account;
|
||||||
|
private MimeMessage msg;
|
||||||
|
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() throws Exception {
|
||||||
|
Context context = RuntimeEnvironment.application;
|
||||||
|
createDummyAccount(context);
|
||||||
|
msg = parseWithoutRecurse(toStream(
|
||||||
|
"From: <adam@example.org>\r\n" +
|
||||||
|
"To: <eva@example.org>\r\n" +
|
||||||
|
"Subject: Testmail\r\n" +
|
||||||
|
"MIME-Version: 1.0\r\n" +
|
||||||
|
"Content-type: text/plain\r\n" +
|
||||||
|
"Content-Transfer-Encoding: 7bit\r\n" +
|
||||||
|
"\r\n" +
|
||||||
|
"this is some test text."));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private static MimeMessage parseWithoutRecurse(InputStream data) throws Exception {
|
||||||
|
return MimeMessage.parseMimeMessage(data, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static ByteArrayInputStream toStream(String rawMailData) throws Exception {
|
||||||
|
return new ByteArrayInputStream(rawMailData.getBytes("ISO-8859-1"));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void createDummyAccount(Context context) {
|
||||||
|
account = new DummyAccount(context);
|
||||||
|
setIdentity();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setIdentity() {
|
||||||
|
Identity identity = new Identity();
|
||||||
|
identity.setEmail("test@mail.com");
|
||||||
|
identity.setName("test");
|
||||||
|
Identity identity2 = new Identity();
|
||||||
|
identity2.setEmail("test2@mail.com");
|
||||||
|
identity2.setName("test2");
|
||||||
|
Identity eva = new Identity();
|
||||||
|
eva.setEmail("eva@example.org");
|
||||||
|
eva.setName("Eva");
|
||||||
|
|
||||||
|
List<Identity> identityList = new ArrayList<>();
|
||||||
|
identityList.add(identity);
|
||||||
|
identityList.add(identity2);
|
||||||
|
identityList.add(eva);
|
||||||
|
account.setIdentities(identityList);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testXOriginalTo() throws Exception {
|
||||||
|
Address[] addresses = {new Address("test2@mail.com")};
|
||||||
|
msg.setRecipients(Message.RecipientType.X_ORIGINAL_TO, addresses);
|
||||||
|
|
||||||
|
Identity identity = IdentityHelper.getRecipientIdentityFromMessage(account, msg);
|
||||||
|
assertTrue(identity.getEmail().equalsIgnoreCase("test2@mail.com"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testTo_withoutXOriginalTo() throws Exception {
|
||||||
|
Identity eva = IdentityHelper.getRecipientIdentityFromMessage(account, msg);
|
||||||
|
assertTrue(eva.getEmail().equalsIgnoreCase("eva@example.org"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDeliveredTo() throws Exception {
|
||||||
|
Address[] addresses = {new Address("test2@mail.com")};
|
||||||
|
msg.setRecipients(Message.RecipientType.DELIVERED_TO, addresses);
|
||||||
|
msg.removeHeader("X-Original-To");
|
||||||
|
|
||||||
|
Identity identity = IdentityHelper.getRecipientIdentityFromMessage(account, msg);
|
||||||
|
assertTrue(identity.getEmail().equalsIgnoreCase("test2@mail.com"));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testXEnvelopeTo() throws Exception {
|
||||||
|
Address[] addresses = {new Address("test@mail.com")};
|
||||||
|
msg.setRecipients(Message.RecipientType.X_ENVELOPE_TO, addresses);
|
||||||
|
msg.removeHeader("X-Original-To");
|
||||||
|
msg.removeHeader("Delivered-To");
|
||||||
|
|
||||||
|
Identity identity = IdentityHelper.getRecipientIdentityFromMessage(account, msg);
|
||||||
|
assertTrue(identity.getEmail().equalsIgnoreCase("test@mail.com"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testXEnvelopeTo_withXOriginalTo() throws Exception {
|
||||||
|
Address[] addresses = {new Address("test@mail.com")};
|
||||||
|
Address[] xoriginaltoaddresses = {new Address("test2@mail.com")};
|
||||||
|
msg.setRecipients(Message.RecipientType.X_ENVELOPE_TO, addresses);
|
||||||
|
msg.setRecipients(Message.RecipientType.X_ORIGINAL_TO, xoriginaltoaddresses);
|
||||||
|
|
||||||
|
Identity identity = IdentityHelper.getRecipientIdentityFromMessage(account, msg);
|
||||||
|
assertTrue(identity.getEmail().equalsIgnoreCase("test2@mail.com"));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static class DummyAccount extends Account {
|
||||||
|
|
||||||
|
protected DummyAccount(Context context) {
|
||||||
|
super(context);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue