Browse Source

Refactored account/folder monitoring

main
M66B 6 years ago
parent
commit
c04ce42eff
44 changed files with 348 additions and 377 deletions
  1. +2
    -0
      app/src/main/java/eu/faircode/email/AdapterAccount.java
  2. +2
    -1
      app/src/main/java/eu/faircode/email/EntityFolder.java
  3. +334
    -376
      app/src/main/java/eu/faircode/email/ServiceSynchronize.java
  4. BIN
      app/src/main/res/drawable-hdpi/baseline_close_black_18.png
  5. BIN
      app/src/main/res/drawable-hdpi/baseline_close_black_24.png
  6. BIN
      app/src/main/res/drawable-hdpi/baseline_close_black_36.png
  7. BIN
      app/src/main/res/drawable-hdpi/baseline_close_black_48.png
  8. BIN
      app/src/main/res/drawable-hdpi/baseline_close_white_18.png
  9. BIN
      app/src/main/res/drawable-hdpi/baseline_close_white_24.png
  10. BIN
      app/src/main/res/drawable-hdpi/baseline_close_white_36.png
  11. BIN
      app/src/main/res/drawable-hdpi/baseline_close_white_48.png
  12. BIN
      app/src/main/res/drawable-mdpi/baseline_close_black_18.png
  13. BIN
      app/src/main/res/drawable-mdpi/baseline_close_black_24.png
  14. BIN
      app/src/main/res/drawable-mdpi/baseline_close_black_36.png
  15. BIN
      app/src/main/res/drawable-mdpi/baseline_close_black_48.png
  16. BIN
      app/src/main/res/drawable-mdpi/baseline_close_white_18.png
  17. BIN
      app/src/main/res/drawable-mdpi/baseline_close_white_24.png
  18. BIN
      app/src/main/res/drawable-mdpi/baseline_close_white_36.png
  19. BIN
      app/src/main/res/drawable-mdpi/baseline_close_white_48.png
  20. BIN
      app/src/main/res/drawable-xhdpi/baseline_close_black_18.png
  21. BIN
      app/src/main/res/drawable-xhdpi/baseline_close_black_24.png
  22. BIN
      app/src/main/res/drawable-xhdpi/baseline_close_black_36.png
  23. BIN
      app/src/main/res/drawable-xhdpi/baseline_close_black_48.png
  24. BIN
      app/src/main/res/drawable-xhdpi/baseline_close_white_18.png
  25. BIN
      app/src/main/res/drawable-xhdpi/baseline_close_white_24.png
  26. BIN
      app/src/main/res/drawable-xhdpi/baseline_close_white_36.png
  27. BIN
      app/src/main/res/drawable-xhdpi/baseline_close_white_48.png
  28. BIN
      app/src/main/res/drawable-xxhdpi/baseline_close_black_18.png
  29. BIN
      app/src/main/res/drawable-xxhdpi/baseline_close_black_24.png
  30. BIN
      app/src/main/res/drawable-xxhdpi/baseline_close_black_36.png
  31. BIN
      app/src/main/res/drawable-xxhdpi/baseline_close_black_48.png
  32. BIN
      app/src/main/res/drawable-xxhdpi/baseline_close_white_18.png
  33. BIN
      app/src/main/res/drawable-xxhdpi/baseline_close_white_24.png
  34. BIN
      app/src/main/res/drawable-xxhdpi/baseline_close_white_36.png
  35. BIN
      app/src/main/res/drawable-xxhdpi/baseline_close_white_48.png
  36. BIN
      app/src/main/res/drawable-xxxhdpi/baseline_close_black_18.png
  37. BIN
      app/src/main/res/drawable-xxxhdpi/baseline_close_black_24.png
  38. BIN
      app/src/main/res/drawable-xxxhdpi/baseline_close_black_36.png
  39. BIN
      app/src/main/res/drawable-xxxhdpi/baseline_close_black_48.png
  40. BIN
      app/src/main/res/drawable-xxxhdpi/baseline_close_white_18.png
  41. BIN
      app/src/main/res/drawable-xxxhdpi/baseline_close_white_24.png
  42. BIN
      app/src/main/res/drawable-xxxhdpi/baseline_close_white_36.png
  43. BIN
      app/src/main/res/drawable-xxxhdpi/baseline_close_white_48.png
  44. +10
    -0
      app/src/main/res/drawable/baseline_close_24.xml

+ 2
- 0
app/src/main/java/eu/faircode/email/AdapterAccount.java View File

@ -89,6 +89,8 @@ public class AdapterAccount extends RecyclerView.Adapter<AdapterAccount.ViewHold
ivState.setImageResource(R.drawable.baseline_cloud_24);
else if ("connecting".equals(account.state))
ivState.setImageResource(R.drawable.baseline_cloud_queue_24);
else if ("closing".equals(account.state))
ivState.setImageResource(R.drawable.baseline_close_24);
else
ivState.setImageResource(R.drawable.baseline_cloud_off_24);
ivState.setVisibility(account.synchronize ? View.VISIBLE : View.INVISIBLE);


+ 2
- 1
app/src/main/java/eu/faircode/email/EntityFolder.java View File

@ -115,7 +115,8 @@ public class EntityFolder implements Parcelable {
public boolean equals(Object obj) {
if (obj instanceof EntityFolder) {
EntityFolder other = (EntityFolder) obj;
return ((this.account == null ? other.account == null : this.account.equals(other.account)) &&
return (this.id.equals(other.id) &&
(this.account == null ? other.account == null : this.account.equals(other.account)) &&
this.name.equals(other.name) &&
this.type.equals(other.type) &&
this.synchronize.equals(other.synchronize) &&


+ 334
- 376
app/src/main/java/eu/faircode/email/ServiceSynchronize.java View File

@ -82,6 +82,7 @@ import javax.mail.FolderNotFoundException;
import javax.mail.Message;
import javax.mail.MessageRemovedException;
import javax.mail.MessagingException;
import javax.mail.NoSuchProviderException;
import javax.mail.SendFailedException;
import javax.mail.Session;
import javax.mail.Transport;
@ -116,7 +117,8 @@ public class ServiceSynchronize extends LifecycleService {
private static final int CONNECT_BACKOFF_START = 2; // seconds
private static final int CONNECT_BACKOFF_MAX = 128; // seconds
private static final long NOOP_INTERVAL = 9 * 60 * 1000L; // ms
private static final long STORE_NOOP_INTERVAL = 9 * 60 * 1000L; // ms
private static final long FOLDER_NOOP_INTERVAL = 9 * 60 * 1000L; // ms
private static final int ATTACHMENT_BUFFER_SIZE = 8192; // bytes
static final String ACTION_PROCESS_OPERATIONS = BuildConfig.APPLICATION_ID + ".PROCESS_OPERATIONS";
@ -171,7 +173,7 @@ public class ServiceSynchronize extends LifecycleService {
ConnectivityManager cm = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
cm.unregisterNetworkCallback(serviceManager);
serviceManager.stop();
serviceManager.stop(false);
stopForeground(true);
@ -331,40 +333,48 @@ public class ServiceSynchronize extends LifecycleService {
}
}
private void monitorAccount(final EntityAccount account, final ServiceState state) {
final List<Thread> threads = new ArrayList<>();
private void monitorAccount(final EntityAccount account, final ServiceState state) throws NoSuchProviderException {
Log.i(Helper.TAG, account.name + " start");
final DB db = DB.getInstance(this);
final ExecutorService executor = Executors.newSingleThreadExecutor();
Log.i(Helper.TAG, account.name + " start");
Properties props = MessageHelper.getSessionProperties();
props.setProperty("mail.imaps.peek", "true");
props.setProperty("mail.mime.address.strict", "false");
props.setProperty("mail.mime.decodetext.strict", "false");
//props.put("mail.imaps.minidletime", "5000");
boolean debug = PreferenceManager.getDefaultSharedPreferences(this).getBoolean("debug", false);
if (debug)
System.setProperty("mail.socket.debug", "true");
final DB db = DB.getInstance(ServiceSynchronize.this);
final Session isession = Session.getInstance(props, null);
isession.setDebug(debug);
// adb -t 1 logcat | grep "fairemail\|System.out"
int backoff = CONNECT_BACKOFF_START;
while (state.running) {
IMAPStore istore = null;
final IMAPStore istore = (IMAPStore) isession.getStore("imaps");
final Map<EntityFolder, IMAPFolder> folders = new HashMap<>();
List<Thread> noops = new ArrayList<>();
List<Thread> idlers = new ArrayList<>();
try {
Properties props = MessageHelper.getSessionProperties();
props.setProperty("mail.imaps.peek", "true");
props.setProperty("mail.mime.address.strict", "false");
props.setProperty("mail.mime.decodetext.strict", "false");
//props.put("mail.imaps.minidletime", "5000");
boolean debug = PreferenceManager.getDefaultSharedPreferences(this).getBoolean("debug", false);
final Session isession = Session.getInstance(props, null);
isession.setDebug(debug);
// adb -t 1 logcat | grep "eu.faircode.email\|System.out"
istore = (IMAPStore) isession.getStore("imaps");
final IMAPStore fstore = istore;
// Listen for events
// Listen for store events
istore.addStoreListener(new StoreListener() {
@Override
public void notification(StoreEvent e) {
Log.i(Helper.TAG, account.name + " event: " + e.getMessage());
db.account().setAccountError(account.id, e.getMessage());
synchronized (state) {
state.notifyAll();
}
}
});
// Listen for folder events
istore.addFolderListener(new FolderAdapter() {
@Override
public void folderCreated(FolderEvent e) {
@ -382,243 +392,344 @@ public class ServiceSynchronize extends LifecycleService {
}
});
// Listen for connection changes
// Listen for connection events
istore.addConnectionListener(new ConnectionAdapter() {
Map<Long, IMAPFolder> mapFolder = new HashMap<>();
@Override
public void opened(ConnectionEvent e) {
Log.i(Helper.TAG, account.name + " opened");
}
db.account().setAccountState(account.id, "connected");
db.account().setAccountError(account.id, null);
@Override
public void disconnected(ConnectionEvent e) {
Log.e(Helper.TAG, account.name + " disconnected event");
}
try {
synchronizeFolders(account, fstore);
@Override
public void closed(ConnectionEvent e) {
Log.e(Helper.TAG, account.name + " closed event");
}
});
// Initiate connection
Log.i(Helper.TAG, account.name + " connect");
db.account().setAccountState(account.id, "connecting");
istore.connect(account.host, account.port, account.user, account.password);
for (final EntityFolder folder : db.folder().getFolders(account.id, true)) {
Log.i(Helper.TAG, account.name + " sync folder " + folder.name);
backoff = CONNECT_BACKOFF_START;
db.account().setAccountState(account.id, "connected");
db.account().setAccountError(account.id, null);
// Monitor folders
Thread t = new Thread(new Runnable() {
@Override
public void run() {
IMAPFolder ifolder = null;
try {
Log.i(Helper.TAG, folder.name + " start");
// Update folder list
try {
synchronizeFolders(account, istore);
} catch (MessagingException ex) {
// Don't show to user
throw new IllegalStateException("synchronize folders", ex);
}
db.folder().setFolderState(folder.id, "connecting");
// Synchronize folders
for (final EntityFolder folder : db.folder().getFolders(account.id, true))
try {
Log.i(Helper.TAG, account.name + " sync folder " + folder.name);
ifolder = (IMAPFolder) fstore.getFolder(folder.name);
ifolder.open(Folder.READ_WRITE);
db.folder().setFolderState(folder.id, "connecting");
db.folder().setFolderState(folder.id, "connected");
db.folder().setFolderError(folder.id, null);
final IMAPFolder ifolder = (IMAPFolder) istore.getFolder(folder.name);
ifolder.open(Folder.READ_WRITE);
folders.put(folder, ifolder);
db.folder().setFolderState(folder.id, "connected");
db.folder().setFolderError(folder.id, null);
// Listen for new and deleted messages
ifolder.addMessageCountListener(new MessageCountAdapter() {
@Override
public void messagesAdded(MessageCountEvent e) {
synchronized (lock) {
try {
Log.i(Helper.TAG, folder.name + " messages added");
for (Message imessage : e.getMessages())
try {
synchronizeMessage(folder, ifolder, (IMAPMessage) imessage);
} catch (MessageRemovedException ex) {
Log.w(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
synchronized (mapFolder) {
mapFolder.put(folder.id, ifolder);
}
} catch (Throwable ex) {
Log.e(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, folder.name, ex);
monitorFolder(account, folder, fstore, ifolder, state);
db.folder().setFolderError(folder.id, Helper.formatThrowable(ex));
} catch (Throwable ex) {
// MessagingException
// - message: connection failure
// - event: Too many simultaneous connections. (Failure)
// TODO: retry?
synchronized (state) {
state.notifyAll();
}
}
}
}
Log.e(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, folder.name, ex);
@Override
public void messagesRemoved(MessageCountEvent e) {
synchronized (lock) {
try {
Log.i(Helper.TAG, folder.name + " messages removed");
for (Message imessage : e.getMessages())
try {
long uid = ifolder.getUID(imessage);
db.folder().setFolderError(folder.id, Helper.formatThrowable(ex));
DB db = DB.getInstance(ServiceSynchronize.this);
int count = db.message().deleteMessage(folder.id, uid);
// Check connection
synchronized (state) {
state.notifyAll();
}
} finally {
if (ifolder != null && ifolder.isOpen()) {
try {
ifolder.close(false);
} catch (MessagingException ex) {
Log.w(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
}
Log.i(Helper.TAG, "Deleted uid=" + uid + " count=" + count);
} catch (MessageRemovedException ex) {
Log.w(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
}
} catch (Throwable ex) {
Log.e(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, folder.name, ex);
db.folder().setFolderState(folder.id, null);
db.folder().setFolderError(folder.id, Helper.formatThrowable(ex));
Log.i(Helper.TAG, folder.name + " stopped");
synchronized (state) {
state.notifyAll();
}
}
}, "sync.folder." + folder.id);
t.start();
threads.add(t);
}
}
});
// Listen for folder operations
IntentFilter f = new IntentFilter(ACTION_PROCESS_OPERATIONS);
f.addDataType("account/" + account.id);
LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(ServiceSynchronize.this);
lbm.registerReceiver(processReceiver, f);
// Fetch e-mail
synchronizeMessages(folder, ifolder);
// Run folder operations
Log.i(Helper.TAG, "listen process folder");
for (final EntityFolder folder : db.folder().getFolders(account.id))
if (!EntityFolder.OUTBOX.equals(folder.type))
lbm.sendBroadcast(new Intent(ACTION_PROCESS_OPERATIONS)
.setType("account/" + account.id)
.putExtra("folder", folder.id));
// Flags (like "seen") at the remote could be changed while synchronizing
} catch (Throwable ex) {
Log.e(Helper.TAG, account.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, null, ex);
// Listen for changed messages
ifolder.addMessageChangedListener(new MessageChangedListener() {
@Override
public void messageChanged(MessageChangedEvent e) {
synchronized (lock) {
try {
try {
Log.i(Helper.TAG, folder.name + " message changed");
synchronizeMessage(folder, ifolder, (IMAPMessage) e.getMessage());
} catch (MessageRemovedException ex) {
Log.w(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
}
} catch (Throwable ex) {
Log.e(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, folder.name, ex);
db.account().setAccountError(account.id, Helper.formatThrowable(ex));
db.folder().setFolderError(folder.id, Helper.formatThrowable(ex));
// Check connection
synchronized (state) {
state.notifyAll();
synchronized (state) {
state.notifyAll();
}
}
}
}
}
}
});
// Keep folder connection alive
Thread noop = new Thread(new Runnable() {
@Override
public void run() {
try {
Log.i(Helper.TAG, folder.name + " start noop");
while (state.running && ifolder.isOpen()) {
Log.i(Helper.TAG, folder.name + " request NOOP");
ifolder.doCommand(new IMAPFolder.ProtocolCommand() {
public Object doCommand(IMAPProtocol p) throws ProtocolException {
Log.i(Helper.TAG, ifolder.getName() + " start NOOP");
p.simpleCommand("NOOP", null);
Log.i(Helper.TAG, ifolder.getName() + " end NOOP");
return null;
}
});
@Override
public void disconnected(ConnectionEvent e) {
Log.e(Helper.TAG, account.name + " disconnected event");
try {
Thread.sleep(FOLDER_NOOP_INTERVAL);
} catch (InterruptedException ex) {
Log.w(Helper.TAG, folder.name + " noop " + ex.getMessage());
}
}
} catch (Throwable ex) {
Log.e(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, folder.name, ex);
db.account().setAccountState(account.id, null);
db.folder().setFolderError(folder.id, Helper.formatThrowable(ex));
LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(ServiceSynchronize.this);
lbm.unregisterReceiver(processReceiver);
synchronized (state) {
state.notifyAll();
}
} finally {
Log.i(Helper.TAG, folder.name + " end noop");
}
}
}, "sync.noop." + folder.id);
noop.start();
noops.add(noop);
// Receive folder events
Thread idle = new Thread(new Runnable() {
@Override
public void run() {
try {
Log.i(Helper.TAG, folder.name + " start idle");
while (state.running && ifolder.isOpen()) {
Log.i(Helper.TAG, folder.name + " do idle");
ifolder.idle(false);
Log.i(Helper.TAG, folder.name + " done idle");
}
} catch (Throwable ex) {
Log.e(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, folder.name, ex);
synchronized (mapFolder) {
mapFolder.clear();
}
db.folder().setFolderError(folder.id, Helper.formatThrowable(ex));
// Check connection
synchronized (state) {
state.notifyAll();
}
synchronized (state) {
state.notifyAll();
}
} finally {
Log.i(Helper.TAG, folder.name + " end idle");
}
}
}, "sync.idle." + folder.id);
idle.start();
idlers.add(idle);
} catch (MessagingException ex) {
// Don't show to user
throw new FolderClosedException(folders.get(folder), "start folder", ex);
} catch (IOException ex) {
// Don't show to user
throw new FolderClosedException(folders.get(folder), "start folder", ex);
}
BroadcastReceiver processReceiver = new BroadcastReceiver() {
@Override
public void closed(ConnectionEvent e) {
Log.e(Helper.TAG, account.name + " closed event");
db.account().setAccountState(account.id, null);
public void onReceive(Context context, Intent intent) {
final long fid = intent.getLongExtra("folder", -1);
//Log.v(Helper.TAG, "run operations folder=" + fid);
LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(ServiceSynchronize.this);
lbm.unregisterReceiver(processReceiver);
synchronized (mapFolder) {
mapFolder.clear();
}
// Check connection
synchronized (state) {
state.notifyAll();
}
}
try {
executor.submit(new Runnable() {
@Override
public void run() {
// Get folder
EntityFolder folder = null;
IMAPFolder ifolder = null;
for (EntityFolder f : folders.keySet())
if (f.id == fid) {
folder = f;
ifolder = folders.get(f);
break;
}
private BroadcastReceiver processReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
final long fid = intent.getLongExtra("folder", -1);
final boolean shouldClose = (ifolder == null);
IMAPFolder ifolder;
synchronized (mapFolder) {
ifolder = mapFolder.get(fid);
}
final boolean shouldClose = (ifolder == null);
final IMAPFolder ffolder = ifolder;
try {
if (folder == null)
throw new IllegalArgumentException("Unknown folder=" + fid);
Log.v(Helper.TAG, "run operations folder=" + fid + " offline=" + shouldClose);
try {
executor.submit(new Runnable() {
@Override
public void run() {
DB db = DB.getInstance(ServiceSynchronize.this);
EntityFolder folder = db.folder().getFolder(fid);
IMAPFolder ifolder = ffolder;
try {
Log.v(Helper.TAG, folder.name + " start operations");
if (shouldClose)
Log.v(Helper.TAG, folder.name + " start operations offline=" + shouldClose);
if (ifolder == null) {
// Prevent unnecessary folder connections
if (db.operation().getOperationCount(fid) == 0)
return;
if (ifolder == null) {
// Prevent unnecessary folder connections
if (db.operation().getOperationCount(fid) == 0)
return;
ifolder = (IMAPFolder) fstore.getFolder(folder.name);
ifolder.open(Folder.READ_WRITE);
}
ifolder = (IMAPFolder) istore.getFolder(folder.name);
ifolder.open(Folder.READ_WRITE);
}
processOperations(folder, isession, fstore, ifolder);
} catch (Throwable ex) {
Log.e(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, folder.name, ex);
} finally {
if (shouldClose)
if (ifolder != null && ifolder.isOpen()) {
try {
ifolder.close(false);
} catch (MessagingException ex) {
Log.w(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
}
processOperations(folder, isession, istore, ifolder);
} catch (Throwable ex) {
Log.e(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, folder.name, ex);
} finally {
if (shouldClose)
if (ifolder != null && ifolder.isOpen()) {
try {
ifolder.close(false);
} catch (MessagingException ex) {
Log.w(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
}
Log.v(Helper.TAG, folder.name + " stop operations");
}
}
//Log.v(Helper.TAG, folder.name + " stop operations");
}
});
} catch (RejectedExecutionException ex) {
Log.w(Helper.TAG, ex + "\n" + Log.getStackTraceString(ex));
}
}
});
} catch (RejectedExecutionException ex) {
Log.w(Helper.TAG, ex + "\n" + Log.getStackTraceString(ex));
}
};
});
// Initiate connection
Log.i(Helper.TAG, account.name + " connect");
db.account().setAccountState(account.id, "connecting");
istore.connect(account.host, account.port, account.user, account.password);
backoff = CONNECT_BACKOFF_START;
}
};
// Keep alive
boolean connected = false;
do {
try {
// Listen for folder operations
IntentFilter f = new IntentFilter(ACTION_PROCESS_OPERATIONS);
f.addDataType("account/" + account.id);
LocalBroadcastManager lbm = LocalBroadcastManager.getInstance(ServiceSynchronize.this);
lbm.registerReceiver(processReceiver, f);
try {
// Process pending folder operations
Log.i(Helper.TAG, "listen process folder");
for (final EntityFolder folder : folders.keySet())
if (!EntityFolder.OUTBOX.equals(folder.type))
lbm.sendBroadcast(new Intent(ACTION_PROCESS_OPERATIONS)
.setType("account/" + account.id)
.putExtra("folder", folder.id));
// Keep store alive
while (state.running && istore.isConnected()) {
Log.i(Helper.TAG, "Checking folders");
for (EntityFolder folder : folders.keySet())
if (!folders.get(folder).isOpen())
throw new FolderClosedException(folders.get(folder));
// Wait for stop or folder error
Log.i(Helper.TAG, account.name + " wait");
synchronized (state) {
state.wait();
state.wait(STORE_NOOP_INTERVAL);
}
Log.i(Helper.TAG, account.name + " waited");
} catch (InterruptedException ex) {
Log.w(Helper.TAG, account.name + " wait " + ex.toString());
}
if (state.running) {
Log.i(Helper.TAG, account.name + " NOOP");
connected = istore.isConnected();
}
} while (state.running && connected);
if (state.running)
Log.w(Helper.TAG, account.name + " not connected anymore");
else
Log.i(Helper.TAG, account.name + " not running anymore");
Log.i(Helper.TAG, account.name + " done running=" + state.running);
} finally {
lbm.unregisterReceiver(processReceiver);
}
} catch (Throwable ex) {
Log.e(Helper.TAG, account.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, null, ex);
db.account().setAccountError(account.id, Helper.formatThrowable(ex));
} finally {
// Close store
Log.i(Helper.TAG, account.name + " closing");
db.account().setAccountState(account.id, "closing");
try {
// This can take 20 seconds
// This can take some time
istore.close();
} catch (MessagingException ex) {
Log.w(Helper.TAG, account.name + " " + ex + "\n" + Log.getStackTraceString(ex));
} finally {
Log.i(Helper.TAG, account.name + " closed");
db.account().setAccountState(account.id, null);
for (EntityFolder folder : folders.keySet())
db.folder().setFolderState(folder.id, null);
}
// Stop noop
for (Thread noop : noops) {
noop.interrupt();
join(noop);
}
// Stop idle
for (Thread idle : idlers) {
idle.interrupt();
join(idle);
}
Log.i(Helper.TAG, account.name + " closed");
}
if (state.running) {
@ -634,158 +745,9 @@ public class ServiceSynchronize extends LifecycleService {
}
}
db.account().setAccountState(account.id, null);
for (Thread t : threads)
join(t);
threads.clear();
executor.shutdown();
Log.i(Helper.TAG, account.name + " stopped");
}
private void monitorFolder(
final EntityAccount account, final EntityFolder folder,
final IMAPStore istore, final IMAPFolder ifolder,
final ServiceState state) throws MessagingException, JSONException, IOException {
final DB db = DB.getInstance(ServiceSynchronize.this);
// Listen for new and deleted messages
ifolder.addMessageCountListener(new MessageCountAdapter() {
@Override
public void messagesAdded(MessageCountEvent e) {
synchronized (lock) {
try {
Log.i(Helper.TAG, folder.name + " messages added");
for (Message imessage : e.getMessages())
try {
synchronizeMessage(folder, ifolder, (IMAPMessage) imessage);
} catch (MessageRemovedException ex) {
Log.w(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
}
} catch (Throwable ex) {
Log.e(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, folder.name, ex);
db.folder().setFolderError(folder.id, Helper.formatThrowable(ex));
// Check connection
synchronized (state) {
state.notifyAll();
}
}
}
}
@Override
public void messagesRemoved(MessageCountEvent e) {
synchronized (lock) {
try {
Log.i(Helper.TAG, folder.name + " messages removed");
for (Message imessage : e.getMessages())
try {
long uid = ifolder.getUID(imessage);
DB db = DB.getInstance(ServiceSynchronize.this);
int count = db.message().deleteMessage(folder.id, uid);
Log.i(Helper.TAG, "Deleted uid=" + uid + " count=" + count);
} catch (MessageRemovedException ex) {
Log.w(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
}
} catch (Throwable ex) {
Log.e(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, folder.name, ex);
db.folder().setFolderError(folder.id, Helper.formatThrowable(ex));
// Check connection
synchronized (state) {
state.notifyAll();
}
}
}
}
});
// Fetch e-mail
synchronizeMessages(folder, ifolder);
// Flags (like "seen") at the remote could be changed while synchronizing
// Listen for changed messages
ifolder.addMessageChangedListener(new MessageChangedListener() {
@Override
public void messageChanged(MessageChangedEvent e) {
synchronized (lock) {
try {
try {
Log.i(Helper.TAG, folder.name + " message changed");
synchronizeMessage(folder, ifolder, (IMAPMessage) e.getMessage());
} catch (MessageRemovedException ex) {
Log.w(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
}
} catch (Throwable ex) {
Log.e(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, folder.name, ex);
db.folder().setFolderError(folder.id, Helper.formatThrowable(ex));
// Check connection
synchronized (state) {
state.notifyAll();
}
}
}
}
});
// Keep alive
Log.i(Helper.TAG, folder.name + " start");
try {
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
try {
boolean open;
do {
try {
Thread.sleep(NOOP_INTERVAL);
} catch (InterruptedException ex) {
Log.w(Helper.TAG, folder.name + " noop " + ex.getMessage());
}
open = ifolder.isOpen();
if (open)
noop(folder, ifolder);
} while (open);
} catch (Throwable ex) {
Log.e(Helper.TAG, folder.name + " " + ex + "\n" + Log.getStackTraceString(ex));
reportError(account.name, folder.name, ex);
db.folder().setFolderError(folder.id, Helper.formatThrowable(ex));
} finally {
// Check connection
synchronized (state) {
state.notifyAll();
}
}
}
}, "sync.noop." + folder.id);
thread.start();
// Idle
while (state.running) {
Log.i(Helper.TAG, folder.name + " start idle");
ifolder.idle(false);
Log.i(Helper.TAG, folder.name + " end idle");
}
} finally {
Log.i(Helper.TAG, folder.name + " end");
}
}
private void processOperations(EntityFolder folder, Session isession, IMAPStore istore, IMAPFolder ifolder) throws MessagingException, JSONException, IOException {
synchronized (lock) {
try {
@ -1141,7 +1103,7 @@ public class ServiceSynchronize extends LifecycleService {
}
}
private void synchronizeMessages(EntityFolder folder, IMAPFolder ifolder) throws MessagingException, JSONException, IOException {
private void synchronizeMessages(EntityFolder folder, IMAPFolder ifolder) throws MessagingException, IOException {
try {
Log.v(Helper.TAG, folder.name + " start sync after=" + folder.after);
@ -1218,8 +1180,8 @@ public class ServiceSynchronize extends LifecycleService {
}
}
private int synchronizeMessage(EntityFolder folder, IMAPFolder ifolder, IMAPMessage imessage) throws MessagingException, JSONException, IOException {
long uid = -1;
private int synchronizeMessage(EntityFolder folder, IMAPFolder ifolder, IMAPMessage imessage) throws MessagingException, IOException {
long uid;
try {
FetchProfile fp = new FetchProfile();
fp.add(UIDFolder.FetchProfileItem.UID);
@ -1227,7 +1189,7 @@ public class ServiceSynchronize extends LifecycleService {
ifolder.fetch(new Message[]{imessage}, fp);
uid = ifolder.getUID(imessage);
Log.v(Helper.TAG, folder.name + " start sync uid=" + uid);
//Log.v(Helper.TAG, folder.name + " start sync uid=" + uid);
if (imessage.isExpunged()) {
Log.i(Helper.TAG, folder.name + " expunged uid=" + uid);
@ -1277,10 +1239,10 @@ public class ServiceSynchronize extends LifecycleService {
message.seen = seen;
message.ui_seen = seen;
db.message().updateMessage(message);
Log.v(Helper.TAG, folder.name + " updated id=" + message.id + " uid=" + message.uid + " seen=" + seen);
Log.i(Helper.TAG, folder.name + " updated id=" + message.id + " uid=" + message.uid + " seen=" + seen);
result = -1;
} else
Log.v(Helper.TAG, folder.name + " unchanged id=" + message.id + " uid=" + message.uid);
; //Log.v(Helper.TAG, folder.name + " unchanged id=" + message.id + " uid=" + message.uid);
}
db.setTransactionSuccessful();
@ -1327,7 +1289,7 @@ public class ServiceSynchronize extends LifecycleService {
message.ui_hide = false;
message.id = db.message().insertMessage(message);
Log.v(Helper.TAG, folder.name + " added id=" + message.id + " uid=" + message.uid);
Log.i(Helper.TAG, folder.name + " added id=" + message.id + " uid=" + message.uid);
int sequence = 0;
for (EntityAttachment attachment : helper.getAttachments()) {
@ -1342,25 +1304,13 @@ public class ServiceSynchronize extends LifecycleService {
return 1;
} finally {
Log.v(Helper.TAG, folder.name + " end sync uid=" + uid);
//Log.v(Helper.TAG, folder.name + " end sync uid=" + uid);
}
}
private void noop(EntityFolder folder, final IMAPFolder ifolder) throws MessagingException {
Log.i(Helper.TAG, folder.name + " request NOOP");
ifolder.doCommand(new IMAPFolder.ProtocolCommand() {
public Object doCommand(IMAPProtocol p) throws ProtocolException {
Log.i(Helper.TAG, ifolder.getName() + " start NOOP");
p.simpleCommand("NOOP", null);
Log.i(Helper.TAG, ifolder.getName() + " end NOOP");
return null;
}
});
}
private class ServiceManager extends ConnectivityManager.NetworkCallback {
private ServiceState state = new ServiceState();
private boolean connected = false;
private boolean running = false;
private Thread main;
private EntityFolder outbox = null;
private ExecutorService lifecycle = Executors.newSingleThreadExecutor();
@ -1370,12 +1320,15 @@ public class ServiceSynchronize extends LifecycleService {
public void onAvailable(Network network) {
Log.i(Helper.TAG, "Network available " + network);
if (!connected) {
Log.i(Helper.TAG, "Network not connected");
connected = true;
if (running)
Log.i(Helper.TAG, "Service already running");
else {
Log.i(Helper.TAG, "Service not running");
running = true;
lifecycle.submit(new Runnable() {
@Override
public void run() {
Log.i(Helper.TAG, "Starting service");
start();
}
});
@ -1386,28 +1339,30 @@ public class ServiceSynchronize extends LifecycleService {
public void onLost(Network network) {
Log.i(Helper.TAG, "Network lost " + network);
if (connected) {
Log.i(Helper.TAG, "Network connected");
if (running) {
Log.i(Helper.TAG, "Service running");
ConnectivityManager cm = getSystemService(ConnectivityManager.class);
NetworkInfo ni = cm.getActiveNetworkInfo();
if (ni != null)
Log.i(Helper.TAG, "Network active=" + ni);
Log.i(Helper.TAG, "Network active=" + (ni == null ? null : ni.toString()));
if (ni == null || !ni.isConnected()) {
Log.i(Helper.TAG, "Network disconnected=" + ni);
connected = false;
running = false;
lifecycle.submit(new Runnable() {
@Override
public void run() {
stop();
Log.i(Helper.TAG, "Stopping service");
stop(true);
}
});
}
}
} else
Log.i(Helper.TAG, "Service not running");
}
public void start() {
private void start() {
synchronized (state) {
state.running = true;
state.disconnected = false;
}
main = new Thread(new Runnable() {
@ -1480,14 +1435,16 @@ public class ServiceSynchronize extends LifecycleService {
main.start();
}
public void stop() {
private void stop(boolean disconnected) {
if (main != null) {
synchronized (state) {
state.running = false;
state.disconnected = disconnected;
state.notifyAll();
}
main.interrupt(); // stop backoff
// stop wait or backoff
main.interrupt();
join(main);
main = null;
@ -1564,7 +1521,7 @@ public class ServiceSynchronize extends LifecycleService {
public void quit() {
Log.i(Helper.TAG, "Service quit");
serviceManager.stop();
serviceManager.stop(false);
Log.i(Helper.TAG, "Service quited");
stopSelf();
}
@ -1613,5 +1570,6 @@ public class ServiceSynchronize extends LifecycleService {
private class ServiceState {
boolean running = false;
boolean disconnected = false;
}
}

BIN
app/src/main/res/drawable-hdpi/baseline_close_black_18.png View File

Before After
Width: 27  |  Height: 27  |  Size: 220 B

BIN
app/src/main/res/drawable-hdpi/baseline_close_black_24.png View File

Before After
Width: 36  |  Height: 36  |  Size: 219 B

BIN
app/src/main/res/drawable-hdpi/baseline_close_black_36.png View File

Before After
Width: 54  |  Height: 54  |  Size: 319 B

BIN
app/src/main/res/drawable-hdpi/baseline_close_black_48.png View File

Before After
Width: 72  |  Height: 72  |  Size: 311 B

BIN
app/src/main/res/drawable-hdpi/baseline_close_white_18.png View File

Before After
Width: 27  |  Height: 27  |  Size: 228 B

BIN
app/src/main/res/drawable-hdpi/baseline_close_white_24.png View File

Before After
Width: 36  |  Height: 36  |  Size: 225 B

BIN
app/src/main/res/drawable-hdpi/baseline_close_white_36.png View File

Before After
Width: 54  |  Height: 54  |  Size: 312 B

BIN
app/src/main/res/drawable-hdpi/baseline_close_white_48.png View File

Before After
Width: 72  |  Height: 72  |  Size: 312 B

BIN
app/src/main/res/drawable-mdpi/baseline_close_black_18.png View File

Before After
Width: 18  |  Height: 18  |  Size: 143 B

BIN
app/src/main/res/drawable-mdpi/baseline_close_black_24.png View File

Before After
Width: 24  |  Height: 24  |  Size: 176 B

BIN
app/src/main/res/drawable-mdpi/baseline_close_black_36.png View File

Before After
Width: 36  |  Height: 36  |  Size: 219 B

BIN
app/src/main/res/drawable-mdpi/baseline_close_black_48.png View File

Before After
Width: 48  |  Height: 48  |  Size: 253 B

BIN
app/src/main/res/drawable-mdpi/baseline_close_white_18.png View File

Before After
Width: 18  |  Height: 18  |  Size: 146 B

BIN
app/src/main/res/drawable-mdpi/baseline_close_white_24.png View File

Before After
Width: 24  |  Height: 24  |  Size: 180 B

BIN
app/src/main/res/drawable-mdpi/baseline_close_white_36.png View File

Before After
Width: 36  |  Height: 36  |  Size: 225 B

BIN
app/src/main/res/drawable-mdpi/baseline_close_white_48.png View File

Before After
Width: 48  |  Height: 48  |  Size: 251 B

BIN
app/src/main/res/drawable-xhdpi/baseline_close_black_18.png View File

Before After
Width: 36  |  Height: 36  |  Size: 219 B

BIN
app/src/main/res/drawable-xhdpi/baseline_close_black_24.png View File

Before After
Width: 48  |  Height: 48  |  Size: 253 B

BIN
app/src/main/res/drawable-xhdpi/baseline_close_black_36.png View File

Before After
Width: 72  |  Height: 72  |  Size: 311 B

BIN
app/src/main/res/drawable-xhdpi/baseline_close_black_48.png View File

Before After
Width: 96  |  Height: 96  |  Size: 389 B

BIN
app/src/main/res/drawable-xhdpi/baseline_close_white_18.png View File

Before After
Width: 36  |  Height: 36  |  Size: 225 B

BIN
app/src/main/res/drawable-xhdpi/baseline_close_white_24.png View File

Before After
Width: 48  |  Height: 48  |  Size: 251 B

BIN
app/src/main/res/drawable-xhdpi/baseline_close_white_36.png View File

Before After
Width: 72  |  Height: 72  |  Size: 312 B

BIN
app/src/main/res/drawable-xhdpi/baseline_close_white_48.png View File

Before After
Width: 96  |  Height: 96  |  Size: 389 B

BIN
app/src/main/res/drawable-xxhdpi/baseline_close_black_18.png View File

Before After
Width: 54  |  Height: 54  |  Size: 319 B

BIN
app/src/main/res/drawable-xxhdpi/baseline_close_black_24.png View File

Before After
Width: 72  |  Height: 72  |  Size: 311 B

BIN
app/src/main/res/drawable-xxhdpi/baseline_close_black_36.png View File

Before After
Width: 108  |  Height: 108  |  Size: 449 B

BIN
app/src/main/res/drawable-xxhdpi/baseline_close_black_48.png View File

Before After
Width: 144  |  Height: 144  |  Size: 544 B

BIN
app/src/main/res/drawable-xxhdpi/baseline_close_white_18.png View File

Before After
Width: 54  |  Height: 54  |  Size: 312 B

BIN
app/src/main/res/drawable-xxhdpi/baseline_close_white_24.png View File

Before After
Width: 72  |  Height: 72  |  Size: 312 B

BIN
app/src/main/res/drawable-xxhdpi/baseline_close_white_36.png View File

Before After
Width: 108  |  Height: 108  |  Size: 449 B

BIN
app/src/main/res/drawable-xxhdpi/baseline_close_white_48.png View File

Before After
Width: 144  |  Height: 144  |  Size: 544 B

BIN
app/src/main/res/drawable-xxxhdpi/baseline_close_black_18.png View File

Before After
Width: 72  |  Height: 72  |  Size: 311 B

BIN
app/src/main/res/drawable-xxxhdpi/baseline_close_black_24.png View File

Before After
Width: 96  |  Height: 96  |  Size: 389 B

BIN
app/src/main/res/drawable-xxxhdpi/baseline_close_black_36.png View File

Before After
Width: 144  |  Height: 144  |  Size: 544 B

BIN
app/src/main/res/drawable-xxxhdpi/baseline_close_black_48.png View File

Before After
Width: 192  |  Height: 192  |  Size: 738 B

BIN
app/src/main/res/drawable-xxxhdpi/baseline_close_white_18.png View File

Before After
Width: 72  |  Height: 72  |  Size: 312 B

BIN
app/src/main/res/drawable-xxxhdpi/baseline_close_white_24.png View File

Before After
Width: 96  |  Height: 96  |  Size: 389 B

BIN
app/src/main/res/drawable-xxxhdpi/baseline_close_white_36.png View File

Before After
Width: 144  |  Height: 144  |  Size: 544 B

BIN
app/src/main/res/drawable-xxxhdpi/baseline_close_white_48.png View File

Before After
Width: 192  |  Height: 192  |  Size: 747 B

+ 10
- 0
app/src/main/res/drawable/baseline_close_24.xml View File

@ -0,0 +1,10 @@
<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:width="24dp"
android:height="24dp"
android:viewportWidth="24.0"
android:viewportHeight="24.0"
android:tint="?attr/colorControlNormal">
<path
android:fillColor="@android:color/white"
android:pathData="M19,6.41L17.59,5 12,10.59 6.41,5 5,6.41 10.59,12 5,17.59 6.41,19 12,13.41 17.59,19 19,17.59 13.41,12z"/>
</vector>

Loading…
Cancel
Save