FairEmail/app/src/main/java/eu/faircode/email/MailService.java

377 lines
15 KiB
Java
Raw Normal View History

2019-07-29 09:17:12 +00:00
package eu.faircode.email;
2019-09-18 14:34:07 +00:00
import android.accounts.Account;
import android.accounts.AccountManager;
2019-07-29 09:17:12 +00:00
import android.content.Context;
2019-09-18 14:34:07 +00:00
import android.text.TextUtils;
2019-07-29 09:17:12 +00:00
2019-09-18 14:34:07 +00:00
import com.sun.mail.imap.IMAPFolder;
2019-07-29 09:17:12 +00:00
import com.sun.mail.imap.IMAPStore;
import com.sun.mail.smtp.SMTPTransport;
import com.sun.mail.util.MailConnectException;
import java.net.Inet4Address;
import java.net.Inet6Address;
import java.net.InetAddress;
import java.net.UnknownHostException;
2019-09-18 14:34:07 +00:00
import java.util.ArrayList;
2019-07-29 09:17:12 +00:00
import java.util.HashMap;
import java.util.LinkedHashMap;
2019-09-18 14:34:07 +00:00
import java.util.List;
2019-09-23 20:07:22 +00:00
import java.util.Locale;
2019-07-29 09:17:12 +00:00
import java.util.Map;
import java.util.Properties;
2019-07-30 18:53:37 +00:00
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
2019-07-29 09:17:12 +00:00
2019-09-18 14:34:07 +00:00
import javax.mail.AuthenticationFailedException;
import javax.mail.Folder;
2019-07-29 09:17:12 +00:00
import javax.mail.MessagingException;
import javax.mail.NoSuchProviderException;
import javax.mail.Service;
import javax.mail.Session;
2019-09-19 15:41:26 +00:00
import javax.mail.Store;
2019-07-29 09:17:12 +00:00
public class MailService implements AutoCloseable {
private Context context;
private String protocol;
private boolean useip;
2019-07-29 09:17:12 +00:00
private boolean debug;
private Properties properties;
private Session isession;
private Service iservice;
2019-07-30 18:53:37 +00:00
private ExecutorService executor = Executors.newCachedThreadPool(Helper.backgroundThreadFactory);
2019-09-18 14:34:07 +00:00
static final int AUTH_TYPE_PASSWORD = 1;
static final int AUTH_TYPE_GMAIL = 2;
2019-07-29 14:42:17 +00:00
private final static int CONNECT_TIMEOUT = 20 * 1000; // milliseconds
private final static int WRITE_TIMEOUT = 60 * 1000; // milliseconds
private final static int READ_TIMEOUT = 60 * 1000; // milliseconds
private final static int FETCH_SIZE = 256 * 1024; // bytes, default 16K
private final static int POOL_TIMEOUT = 45 * 1000; // milliseconds, default 45 sec
private static final int APPEND_BUFFER_SIZE = 4 * 1024 * 1024; // bytes
2019-07-29 09:17:12 +00:00
private MailService() {
}
2019-07-29 14:42:17 +00:00
MailService(Context context, String protocol, String realm, boolean insecure, boolean debug) throws NoSuchProviderException {
this.context = context.getApplicationContext();
2019-07-29 09:17:12 +00:00
this.protocol = protocol;
this.debug = debug;
2019-07-30 16:53:18 +00:00
properties = MessageHelper.getSessionProperties();
2019-07-29 14:42:17 +00:00
2019-07-30 16:53:18 +00:00
properties.put("mail.event.scope", "folder");
2019-07-30 18:53:37 +00:00
properties.put("mail.event.executor", executor);
2019-07-29 14:42:17 +00:00
2019-09-23 20:07:22 +00:00
String checkserveridentity = Boolean.toString(!insecure).toLowerCase(Locale.ROOT);
2019-07-29 14:42:17 +00:00
2019-09-19 15:41:26 +00:00
if ("pop3".equals(protocol) || "pop3s".equals(protocol)) {
2019-09-20 07:45:36 +00:00
this.debug = true;
2019-09-19 15:41:26 +00:00
// https://javaee.github.io/javamail/docs/api/com/sun/mail/pop3/package-summary.html#properties
properties.put("mail." + protocol + ".ssl.checkserveridentity", checkserveridentity);
properties.put("mail." + protocol + ".ssl.trust", "*");
properties.put("mail.pop3s.starttls.enable", "false");
properties.put("mail.pop3.starttls.enable", "true");
properties.put("mail.pop3.starttls.required", "true");
// TODO: make timeouts configurable?
properties.put("mail." + protocol + ".connectiontimeout", Integer.toString(CONNECT_TIMEOUT));
properties.put("mail." + protocol + ".writetimeout", Integer.toString(WRITE_TIMEOUT)); // one thread overhead
properties.put("mail." + protocol + ".timeout", Integer.toString(READ_TIMEOUT));
} else if ("imap".equals(protocol) || "imaps".equals(protocol)) {
2019-07-29 14:42:17 +00:00
// https://javaee.github.io/javamail/docs/api/com/sun/mail/imap/package-summary.html#properties
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".ssl.checkserveridentity", checkserveridentity);
properties.put("mail." + protocol + ".ssl.trust", "*");
2019-07-29 14:42:17 +00:00
2019-07-30 16:53:18 +00:00
properties.put("mail.imaps.starttls.enable", "false");
2019-07-29 14:42:17 +00:00
2019-07-30 16:53:18 +00:00
properties.put("mail.imap.starttls.enable", "true");
properties.put("mail.imap.starttls.required", "true");
2019-07-29 14:42:17 +00:00
if (realm != null)
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".auth.ntlm.domain", realm);
2019-07-29 14:42:17 +00:00
// TODO: make timeouts configurable?
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".connectiontimeout", Integer.toString(CONNECT_TIMEOUT));
properties.put("mail." + protocol + ".writetimeout", Integer.toString(WRITE_TIMEOUT)); // one thread overhead
properties.put("mail." + protocol + ".timeout", Integer.toString(READ_TIMEOUT));
2019-07-29 14:42:17 +00:00
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".connectionpool.debug", "true");
properties.put("mail." + protocol + ".connectionpoolsize", "2");
properties.put("mail." + protocol + ".connectionpooltimeout", Integer.toString(POOL_TIMEOUT));
2019-07-29 14:42:17 +00:00
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".finalizecleanclose", "false");
2019-07-29 14:42:17 +00:00
// https://tools.ietf.org/html/rfc4978
// https://docs.oracle.com/javase/8/docs/api/java/util/zip/Deflater.html
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".compress.enable", "true");
//properties.put("mail.imaps.compress.level", "-1");
//properties.put("mail.imaps.compress.strategy", "0");
2019-07-29 14:42:17 +00:00
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".throwsearchexception", "true");
properties.put("mail." + protocol + ".fetchsize", Integer.toString(FETCH_SIZE));
properties.put("mail." + protocol + ".peek", "true");
properties.put("mail." + protocol + ".appendbuffersize", Integer.toString(APPEND_BUFFER_SIZE));
2019-07-29 14:42:17 +00:00
} else if ("smtp".equals(protocol) || "smtps".equals(protocol)) {
// https://javaee.github.io/javamail/docs/api/com/sun/mail/smtp/package-summary.html#properties
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".ssl.checkserveridentity", checkserveridentity);
properties.put("mail." + protocol + ".ssl.trust", "*");
2019-07-29 14:42:17 +00:00
2019-07-30 16:53:18 +00:00
properties.put("mail.smtps.starttls.enable", "false");
2019-07-29 14:42:17 +00:00
2019-07-30 16:53:18 +00:00
properties.put("mail.smtp.starttls.enable", "true");
properties.put("mail.smtp.starttls.required", "true");
2019-07-29 14:42:17 +00:00
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".auth", "true");
2019-07-29 14:42:17 +00:00
if (realm != null)
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".auth.ntlm.domain", realm);
2019-07-29 14:42:17 +00:00
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".connectiontimeout", Integer.toString(CONNECT_TIMEOUT));
properties.put("mail." + protocol + ".writetimeout", Integer.toString(WRITE_TIMEOUT)); // one thread overhead
properties.put("mail." + protocol + ".timeout", Integer.toString(READ_TIMEOUT));
2019-07-29 14:42:17 +00:00
} else
throw new NoSuchProviderException(protocol);
2019-07-29 09:17:12 +00:00
}
void setPartialFetch(boolean enabled) {
if (!enabled)
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".partialfetch", "false");
2019-07-29 09:17:12 +00:00
}
void setUseIp(boolean enabled) {
2019-07-30 16:53:18 +00:00
useip = enabled;
2019-07-29 09:17:12 +00:00
}
void setSeparateStoreConnection() {
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".separatestoreconnection", "true");
2019-07-29 09:17:12 +00:00
}
void setLeaveOnServer(boolean keep) {
properties.put("mail." + protocol + ".rsetbeforequit", Boolean.toString(keep));
}
2019-07-29 09:17:12 +00:00
public void connect(EntityAccount account) throws MessagingException {
2019-09-19 11:21:37 +00:00
String password = connect(account.host, account.port, account.auth_type, account.user, account.password);
if (password != null) {
DB db = DB.getInstance(context);
int count = db.account().setAccountPassword(account.id, account.password);
Log.i(account.name + " token refreshed=" + count);
}
2019-07-29 09:17:12 +00:00
}
public void connect(EntityIdentity identity) throws MessagingException {
2019-09-19 11:21:37 +00:00
String password = connect(identity.host, identity.port, identity.auth_type, identity.user, identity.password);
if (password != null) {
DB db = DB.getInstance(context);
int count = db.identity().setIdentityPassword(identity.id, identity.password);
Log.i(identity.email + " token refreshed=" + count);
}
2019-07-29 09:17:12 +00:00
}
2019-09-19 11:21:37 +00:00
public String connect(String host, int port, int auth, String user, String password) throws MessagingException {
2019-07-29 09:17:12 +00:00
try {
2019-09-18 14:34:07 +00:00
if (auth == AUTH_TYPE_GMAIL)
properties.put("mail." + protocol + ".auth.mechanisms", "XOAUTH2");
//if (BuildConfig.DEBUG)
// throw new MailConnectException(new SocketConnectException("Debug", new Exception(), host, port, 0));
2019-09-18 14:34:07 +00:00
2019-07-29 09:17:12 +00:00
_connect(context, host, port, user, password);
2019-09-19 11:21:37 +00:00
return null;
2019-09-18 14:34:07 +00:00
} catch (AuthenticationFailedException ex) {
// Refresh token
if (auth == AUTH_TYPE_GMAIL)
try {
String type = "com.google";
AccountManager am = AccountManager.get(context);
Account[] accounts = am.getAccountsByType(type);
for (Account account : accounts)
if (user.equals(account.name)) {
Log.i("Refreshing token user=" + user);
am.invalidateAuthToken(type, password);
2019-09-18 20:11:48 +00:00
String token = am.blockingGetAuthToken(account, getAuthTokenType(type), true);
if (token == null)
2019-09-24 12:52:46 +00:00
throw new IllegalArgumentException("No token on refresh");
2019-09-18 14:34:07 +00:00
2019-09-18 20:11:48 +00:00
_connect(context, host, port, user, token);
2019-09-19 11:21:37 +00:00
return token;
2019-09-18 14:34:07 +00:00
}
2019-09-19 11:21:37 +00:00
2019-09-24 12:52:46 +00:00
throw new IllegalArgumentException("Account not found");
2019-09-24 13:34:39 +00:00
} catch (Exception ex1) {
2019-09-18 14:34:07 +00:00
Log.e(ex1);
2019-09-24 13:34:39 +00:00
throw new AuthenticationFailedException(ex.getMessage(), ex1);
2019-09-18 14:34:07 +00:00
}
else
throw ex;
2019-07-29 09:17:12 +00:00
} catch (MailConnectException ex) {
try {
// Some devices resolve IPv6 addresses while not having IPv6 connectivity
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".ssl.checkserveridentity", "false");
InetAddress[] iaddrs = InetAddress.getAllByName(host);
if (iaddrs.length > 1)
for (InetAddress iaddr : iaddrs)
try {
Log.i("Falling back to " + iaddr.getHostAddress());
_connect(context, iaddr.getHostAddress(), port, user, password);
2019-09-19 11:21:37 +00:00
return null;
} catch (MessagingException ex1) {
Log.w(ex1);
}
} catch (Throwable ex1) {
Log.w(ex1);
}
throw ex;
2019-07-29 09:17:12 +00:00
}
}
private void _connect(Context context, String host, int port, String user, String password) throws MessagingException {
isession = Session.getInstance(properties, null);
isession.setDebug(debug);
//System.setProperty("mail.socket.debug", Boolean.toString(debug));
2019-09-19 15:41:26 +00:00
if ("pop3".equals(protocol) || "pop3s".equals(protocol)) {
iservice = isession.getStore(protocol);
iservice.connect(host, port, user, password);
} else if ("imap".equals(protocol) || "imaps".equals(protocol)) {
2019-07-29 09:17:12 +00:00
iservice = isession.getStore(protocol);
iservice.connect(host, port, user, password);
// https://www.ietf.org/rfc/rfc2971.txt
2019-09-19 15:41:26 +00:00
IMAPStore istore = (IMAPStore) getStore();
if (istore.hasCapability("ID"))
2019-07-29 09:17:12 +00:00
try {
Map<String, String> id = new LinkedHashMap<>();
id.put("name", context.getString(R.string.app_name));
id.put("version", BuildConfig.VERSION_NAME);
2019-09-19 15:41:26 +00:00
Map<String, String> sid = istore.id(id);
2019-07-29 09:17:12 +00:00
if (sid != null) {
Map<String, String> crumb = new HashMap<>();
for (String key : sid.keySet()) {
crumb.put(key, sid.get(key));
EntityLog.log(context, "Server " + key + "=" + sid.get(key));
}
2019-08-12 11:07:14 +00:00
Log.breadcrumb("server", crumb);
2019-07-29 09:17:12 +00:00
}
} catch (MessagingException ex) {
Log.w(ex);
}
} else if ("smtp".equals(protocol) || "smtps".equals(protocol)) {
String haddr = host;
if (useip)
try {
// This assumes getByName always returns the same address (type)
InetAddress addr = InetAddress.getByName(host);
if (addr instanceof Inet4Address)
haddr = "[" + Inet4Address.getLocalHost().getHostAddress() + "]";
else
haddr = "[IPv6:" + Inet6Address.getLocalHost().getHostAddress() + "]";
} catch (UnknownHostException ex) {
Log.w(ex);
}
Log.i("Using localhost=" + haddr);
2019-07-30 16:53:18 +00:00
properties.put("mail." + protocol + ".localhost", haddr);
2019-07-29 09:17:12 +00:00
iservice = isession.getTransport(protocol);
iservice.connect(host, port, user, password);
} else
throw new NoSuchProviderException(protocol);
}
2019-09-18 14:34:07 +00:00
static String getAuthTokenType(String type) {
// https://developers.google.com/gmail/imap/xoauth2-protocol
if ("com.google".equals(type))
return "oauth2:https://mail.google.com/";
return null;
}
List<EntityFolder> getFolders() throws MessagingException {
List<EntityFolder> folders = new ArrayList<>();
List<EntityFolder> guesses = new ArrayList<>();
for (Folder ifolder : getStore().getDefaultFolder().list("*")) {
String fullName = ifolder.getFullName();
String[] attrs = ((IMAPFolder) ifolder).getAttributes();
String type = EntityFolder.getType(attrs, fullName, true);
Log.i(fullName + " attrs=" + TextUtils.join(" ", attrs) + " type=" + type);
if (type != null) {
EntityFolder folder = new EntityFolder(fullName, type);
folders.add(folder);
if (EntityFolder.USER.equals(type)) {
String guess = EntityFolder.guessType(fullName);
if (guess != null)
guesses.add(folder);
}
}
}
for (EntityFolder guess : guesses) {
boolean has = false;
String gtype = EntityFolder.guessType(guess.name);
for (EntityFolder folder : folders)
if (folder.type.equals(gtype)) {
has = true;
break;
}
if (!has) {
guess.type = gtype;
Log.i(guess.name + " guessed type=" + gtype);
}
}
boolean inbox = false;
boolean drafts = false;
for (EntityFolder folder : folders)
if (EntityFolder.INBOX.equals(folder.type))
inbox = true;
else if (EntityFolder.DRAFTS.equals(folder.type))
drafts = true;
if (!inbox || !drafts)
return null;
return folders;
}
2019-09-19 15:41:26 +00:00
Store getStore() {
return (Store) iservice;
2019-07-29 09:17:12 +00:00
}
SMTPTransport getTransport() {
2019-07-30 16:53:18 +00:00
return (SMTPTransport) iservice;
2019-07-29 09:17:12 +00:00
}
2019-09-19 15:41:26 +00:00
boolean hasCapability(String capability) throws MessagingException {
Store store = getStore();
if (store instanceof IMAPStore)
return ((IMAPStore) getStore()).hasCapability(capability);
else
return false;
}
2019-07-29 09:17:12 +00:00
public void close() throws MessagingException {
try {
if (iservice != null)
iservice.close();
} finally {
2019-07-30 16:53:18 +00:00
context = null;
2019-07-29 09:17:12 +00:00
}
}
}