mirror of https://github.com/M66B/FairEmail.git
379 lines
16 KiB
Java
379 lines
16 KiB
Java
|
package eu.faircode.email;
|
||
|
|
||
|
/*
|
||
|
This file is part of FairEmail.
|
||
|
|
||
|
FairEmail is free software: you can redistribute it and/or modify
|
||
|
it under the terms of the GNU General Public License as published by
|
||
|
the Free Software Foundation, either version 3 of the License, or
|
||
|
(at your option) any later version.
|
||
|
|
||
|
FairEmail is distributed in the hope that it will be useful,
|
||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
GNU General Public License for more details.
|
||
|
|
||
|
You should have received a copy of the GNU General Public License
|
||
|
along with FairEmail. If not, see <http://www.gnu.org/licenses/>.
|
||
|
|
||
|
Copyright 2018-2019 by Marcel Bokhorst (M66B)
|
||
|
*/
|
||
|
|
||
|
import android.app.Notification;
|
||
|
import android.app.NotificationManager;
|
||
|
import android.content.Context;
|
||
|
import android.content.Intent;
|
||
|
import android.net.ConnectivityManager;
|
||
|
import android.net.Network;
|
||
|
import android.net.NetworkCapabilities;
|
||
|
import android.net.NetworkInfo;
|
||
|
import android.net.NetworkRequest;
|
||
|
import android.os.PowerManager;
|
||
|
import android.text.TextUtils;
|
||
|
|
||
|
import java.io.File;
|
||
|
import java.io.IOException;
|
||
|
import java.net.Inet6Address;
|
||
|
import java.net.InetAddress;
|
||
|
import java.util.ArrayList;
|
||
|
import java.util.Arrays;
|
||
|
import java.util.Date;
|
||
|
import java.util.List;
|
||
|
import java.util.Properties;
|
||
|
|
||
|
import javax.mail.Address;
|
||
|
import javax.mail.AuthenticationFailedException;
|
||
|
import javax.mail.Message;
|
||
|
import javax.mail.MessageRemovedException;
|
||
|
import javax.mail.MessagingException;
|
||
|
import javax.mail.SendFailedException;
|
||
|
import javax.mail.Session;
|
||
|
import javax.mail.Transport;
|
||
|
import javax.mail.internet.InternetAddress;
|
||
|
import javax.mail.internet.MimeMessage;
|
||
|
|
||
|
import androidx.core.app.NotificationCompat;
|
||
|
import androidx.lifecycle.LifecycleService;
|
||
|
|
||
|
public class ServiceSend extends LifecycleService {
|
||
|
private static final int IDENTITY_ERROR_AFTER = 30; // minutes
|
||
|
|
||
|
@Override
|
||
|
public void onCreate() {
|
||
|
Log.i("Service send create");
|
||
|
super.onCreate();
|
||
|
|
||
|
ConnectivityManager cm = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
|
||
|
NetworkRequest.Builder builder = new NetworkRequest.Builder();
|
||
|
builder.addCapability(NetworkCapabilities.NET_CAPABILITY_INTERNET);
|
||
|
cm.registerNetworkCallback(builder.build(), networkCallback);
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public void onDestroy() {
|
||
|
Log.i("Service send destroy");
|
||
|
|
||
|
ConnectivityManager cm = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
|
||
|
cm.unregisterNetworkCallback(networkCallback);
|
||
|
|
||
|
stopForeground(true);
|
||
|
|
||
|
NotificationManager nm = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
|
||
|
nm.cancel(Helper.NOTIFICATION_SEND);
|
||
|
|
||
|
super.onDestroy();
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public int onStartCommand(Intent intent, int flags, int startId) {
|
||
|
// Build notification
|
||
|
NotificationCompat.Builder builder = new NotificationCompat.Builder(this, "service");
|
||
|
|
||
|
builder
|
||
|
.setSmallIcon(R.drawable.baseline_send_24)
|
||
|
.setContentTitle(getString(R.string.title_notification_sending))
|
||
|
.setAutoCancel(false)
|
||
|
.setShowWhen(false)
|
||
|
.setPriority(Notification.PRIORITY_MIN)
|
||
|
.setCategory(Notification.CATEGORY_STATUS)
|
||
|
.setVisibility(NotificationCompat.VISIBILITY_SECRET);
|
||
|
|
||
|
startForeground(Helper.NOTIFICATION_SEND, builder.build());
|
||
|
|
||
|
super.onStartCommand(intent, flags, startId);
|
||
|
|
||
|
return START_STICKY;
|
||
|
}
|
||
|
|
||
|
ConnectivityManager.NetworkCallback networkCallback = new ConnectivityManager.NetworkCallback() {
|
||
|
private Thread thread = null;
|
||
|
|
||
|
@Override
|
||
|
public void onAvailable(Network network) {
|
||
|
Log.i("Service send available=" + network);
|
||
|
|
||
|
if (thread == null || !thread.isAlive()) {
|
||
|
thread = new Thread(new Runnable() {
|
||
|
@Override
|
||
|
public void run() {
|
||
|
PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE);
|
||
|
PowerManager.WakeLock wl = pm.newWakeLock(
|
||
|
PowerManager.PARTIAL_WAKE_LOCK, BuildConfig.APPLICATION_ID + ":send");
|
||
|
try {
|
||
|
wl.acquire();
|
||
|
|
||
|
DB db = DB.getInstance(ServiceSend.this);
|
||
|
EntityFolder outbox = db.folder().getOutbox();
|
||
|
try {
|
||
|
db.folder().setFolderError(outbox.id, null);
|
||
|
db.folder().setFolderSyncState(outbox.id, "syncing");
|
||
|
|
||
|
ConnectivityManager cm = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
|
||
|
|
||
|
List<EntityOperation> ops = db.operation().getOperations(outbox.id);
|
||
|
Log.i(outbox.name + " pending operations=" + ops.size());
|
||
|
for (EntityOperation op : ops) {
|
||
|
EntityMessage message = db.message().getMessage(op.message);
|
||
|
try {
|
||
|
Log.i(outbox.name +
|
||
|
" start op=" + op.id + "/" + op.name +
|
||
|
" msg=" + op.message +
|
||
|
" args=" + op.args);
|
||
|
|
||
|
if (message == null)
|
||
|
throw new MessageRemovedException();
|
||
|
|
||
|
send(message);
|
||
|
|
||
|
db.operation().deleteOperation(op.id);
|
||
|
} catch (Throwable ex) {
|
||
|
Log.e(ex);
|
||
|
|
||
|
if (message != null)
|
||
|
db.message().setMessageError(message.id, Helper.formatThrowable(ex));
|
||
|
|
||
|
if (ex instanceof SendFailedException)
|
||
|
db.operation().deleteOperation(op.id);
|
||
|
else
|
||
|
throw ex;
|
||
|
} finally {
|
||
|
Log.i(outbox.name + " end op=" + op.id + "/" + op.name);
|
||
|
}
|
||
|
|
||
|
NetworkInfo ni = cm.getActiveNetworkInfo();
|
||
|
if (ni == null || !ni.isConnected())
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (db.operation().getOperations(outbox.id).size() == 0)
|
||
|
stopSelf();
|
||
|
} catch (Throwable ex) {
|
||
|
Log.e(outbox.name, ex);
|
||
|
db.folder().setFolderError(outbox.id, Helper.formatThrowable(ex, true));
|
||
|
} finally {
|
||
|
db.folder().setFolderSyncState(outbox.id, null);
|
||
|
}
|
||
|
} finally {
|
||
|
wl.release();
|
||
|
}
|
||
|
}
|
||
|
});
|
||
|
thread.start();
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
private void send(EntityMessage message) throws MessagingException, IOException {
|
||
|
DB db = DB.getInstance(this);
|
||
|
|
||
|
// Mark attempt
|
||
|
if (message.last_attempt == null) {
|
||
|
message.last_attempt = new Date().getTime();
|
||
|
db.message().setMessageLastAttempt(message.id, message.last_attempt);
|
||
|
}
|
||
|
|
||
|
EntityIdentity ident = db.identity().getIdentity(message.identity);
|
||
|
|
||
|
// Get properties
|
||
|
Properties props = MessageHelper.getSessionProperties(ident.auth_type, ident.realm, ident.insecure);
|
||
|
InetAddress ip = (ident.use_ip ? Helper.getLocalIp(this) : null);
|
||
|
if (ip == null) {
|
||
|
EntityLog.log(this, "Send local host=" + ident.host);
|
||
|
if (ident.starttls)
|
||
|
props.put("mail.smtp.localhost", ident.host);
|
||
|
else
|
||
|
props.put("mail.smtps.localhost", ident.host);
|
||
|
} else {
|
||
|
InetAddress localhost = InetAddress.getLocalHost();
|
||
|
String haddr = "[" + (localhost instanceof Inet6Address ? "IPv6:" : "") + localhost.getHostAddress() + "]";
|
||
|
EntityLog.log(this, "Send local address=" + haddr);
|
||
|
if (ident.starttls)
|
||
|
props.put("mail.smtp.localhost", haddr);
|
||
|
else
|
||
|
props.put("mail.smtps.localhost", haddr);
|
||
|
}
|
||
|
|
||
|
// Create session
|
||
|
final Session isession = Session.getInstance(props, null);
|
||
|
isession.setDebug(true);
|
||
|
|
||
|
// Create message
|
||
|
MimeMessage imessage = MessageHelper.from(this, message, isession, ident.plain_only);
|
||
|
|
||
|
// Add reply to
|
||
|
if (ident.replyto != null)
|
||
|
imessage.setReplyTo(new Address[]{new InternetAddress(ident.replyto)});
|
||
|
|
||
|
// Add bcc
|
||
|
if (ident.bcc != null) {
|
||
|
List<Address> bcc = new ArrayList<>();
|
||
|
Address[] existing = imessage.getRecipients(Message.RecipientType.BCC);
|
||
|
if (existing != null)
|
||
|
bcc.addAll(Arrays.asList(existing));
|
||
|
bcc.add(new InternetAddress(ident.bcc));
|
||
|
imessage.setRecipients(Message.RecipientType.BCC, bcc.toArray(new Address[0]));
|
||
|
}
|
||
|
|
||
|
// defacto standard
|
||
|
if (ident.delivery_receipt)
|
||
|
imessage.addHeader("Return-Receipt-To", ident.replyto == null ? ident.email : ident.replyto);
|
||
|
|
||
|
// https://tools.ietf.org/html/rfc3798
|
||
|
if (ident.read_receipt)
|
||
|
imessage.addHeader("Disposition-Notification-To", ident.replyto == null ? ident.email : ident.replyto);
|
||
|
|
||
|
// Create transport
|
||
|
// TODO: cache transport?
|
||
|
try (Transport itransport = isession.getTransport(ident.getProtocol())) {
|
||
|
// Connect transport
|
||
|
db.identity().setIdentityState(ident.id, "connecting");
|
||
|
try {
|
||
|
itransport.connect(ident.host, ident.port, ident.user, ident.password);
|
||
|
} catch (AuthenticationFailedException ex) {
|
||
|
if (ident.auth_type == Helper.AUTH_TYPE_GMAIL) {
|
||
|
EntityAccount account = db.account().getAccount(ident.account);
|
||
|
ident.password = Helper.refreshToken(this, "com.google", ident.user, account.password);
|
||
|
DB.getInstance(this).identity().setIdentityPassword(ident.id, ident.password);
|
||
|
itransport.connect(ident.host, ident.port, ident.user, ident.password);
|
||
|
} else
|
||
|
throw ex;
|
||
|
}
|
||
|
|
||
|
db.identity().setIdentityState(ident.id, "connected");
|
||
|
|
||
|
// Send message
|
||
|
Address[] to = imessage.getAllRecipients();
|
||
|
itransport.sendMessage(imessage, to);
|
||
|
EntityLog.log(this, "Sent via " + ident.host + "/" + ident.user +
|
||
|
" to " + TextUtils.join(", ", to));
|
||
|
|
||
|
// Append replied/forwarded text
|
||
|
StringBuilder sb = new StringBuilder();
|
||
|
sb.append(Helper.readText(EntityMessage.getFile(this, message.id)));
|
||
|
File refFile = EntityMessage.getRefFile(this, message.id);
|
||
|
if (refFile.exists())
|
||
|
sb.append(Helper.readText(refFile));
|
||
|
Helper.writeText(EntityMessage.getFile(this, message.id), sb.toString());
|
||
|
|
||
|
try {
|
||
|
db.beginTransaction();
|
||
|
|
||
|
db.message().setMessageSent(message.id, imessage.getSentDate().getTime());
|
||
|
db.message().setMessageSeen(message.id, true);
|
||
|
db.message().setMessageUiSeen(message.id, true);
|
||
|
db.message().setMessageError(message.id, null);
|
||
|
|
||
|
EntityFolder sent = db.folder().getFolderByType(message.account, EntityFolder.SENT);
|
||
|
if (ident.store_sent && sent != null) {
|
||
|
db.message().setMessageFolder(message.id, sent.id);
|
||
|
message.folder = sent.id;
|
||
|
EntityOperation.queue(this, db, message, EntityOperation.ADD);
|
||
|
} else
|
||
|
db.message().setMessageUiHide(message.id, true);
|
||
|
|
||
|
db.setTransactionSuccessful();
|
||
|
} finally {
|
||
|
db.endTransaction();
|
||
|
}
|
||
|
|
||
|
if (refFile.exists())
|
||
|
refFile.delete();
|
||
|
|
||
|
if (message.inreplyto != null) {
|
||
|
List<EntityMessage> replieds = db.message().getMessageByMsgId(message.account, message.inreplyto);
|
||
|
for (EntityMessage replied : replieds)
|
||
|
EntityOperation.queue(this, db, replied, EntityOperation.ANSWERED, true);
|
||
|
}
|
||
|
|
||
|
db.identity().setIdentityConnected(ident.id, new Date().getTime());
|
||
|
db.identity().setIdentityError(ident.id, null);
|
||
|
|
||
|
NotificationManager nm = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
|
||
|
nm.cancel("send", message.identity.intValue());
|
||
|
|
||
|
if (message.to != null)
|
||
|
for (Address recipient : message.to) {
|
||
|
String email = ((InternetAddress) recipient).getAddress();
|
||
|
String name = ((InternetAddress) recipient).getPersonal();
|
||
|
List<EntityContact> contacts = db.contact().getContacts(EntityContact.TYPE_TO, email);
|
||
|
if (contacts.size() == 0) {
|
||
|
EntityContact contact = new EntityContact();
|
||
|
contact.type = EntityContact.TYPE_TO;
|
||
|
contact.email = email;
|
||
|
contact.name = name;
|
||
|
db.contact().insertContact(contact);
|
||
|
Log.i("Inserted recipient contact=" + contact);
|
||
|
} else {
|
||
|
EntityContact contact = contacts.get(0);
|
||
|
if (name != null && !name.equals(contact.name)) {
|
||
|
contact.name = name;
|
||
|
db.contact().updateContact(contact);
|
||
|
Log.i("Updated recipient contact=" + contact);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
} catch (MessagingException ex) {
|
||
|
if (ex instanceof SendFailedException) {
|
||
|
SendFailedException sfe = (SendFailedException) ex;
|
||
|
|
||
|
StringBuilder sb = new StringBuilder();
|
||
|
|
||
|
sb.append(sfe.getMessage());
|
||
|
|
||
|
sb.append(' ').append(getString(R.string.title_address_sent));
|
||
|
sb.append(' ').append(MessageHelper.formatAddresses(sfe.getValidSentAddresses()));
|
||
|
|
||
|
sb.append(' ').append(getString(R.string.title_address_unsent));
|
||
|
sb.append(' ').append(MessageHelper.formatAddresses(sfe.getValidUnsentAddresses()));
|
||
|
|
||
|
sb.append(' ').append(getString(R.string.title_address_invalid));
|
||
|
sb.append(' ').append(MessageHelper.formatAddresses(sfe.getInvalidAddresses()));
|
||
|
|
||
|
ex = new SendFailedException(
|
||
|
sb.toString(),
|
||
|
sfe.getNextException(),
|
||
|
sfe.getValidSentAddresses(),
|
||
|
sfe.getValidUnsentAddresses(),
|
||
|
sfe.getInvalidAddresses());
|
||
|
}
|
||
|
|
||
|
db.identity().setIdentityError(ident.id, Helper.formatThrowable(ex));
|
||
|
|
||
|
EntityLog.log(this, ident.name + " last attempt: " + new Date(message.last_attempt));
|
||
|
|
||
|
long now = new Date().getTime();
|
||
|
long delayed = now - message.last_attempt;
|
||
|
if (delayed > IDENTITY_ERROR_AFTER * 60 * 1000L) {
|
||
|
Log.i("Reporting send error after=" + delayed);
|
||
|
NotificationManager nm = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
|
||
|
nm.notify("send", message.identity.intValue(),
|
||
|
Helper.getNotificationError(this, ident.name, ex).build());
|
||
|
}
|
||
|
|
||
|
throw ex;
|
||
|
} finally {
|
||
|
db.identity().setIdentityState(ident.id, null);
|
||
|
}
|
||
|
}
|
||
|
}
|