Merge pull request #452 from sileht/master

Fix imap re-connection
This commit is contained in:
enkore 2016-09-20 19:19:26 +02:00 committed by GitHub
commit e0bd9385af

View File

@ -1,60 +1,18 @@
try: try:
from imaplib2 import IMAP4, IMAP4_SSL from imaplib2.imaplib2 import IMAP4, IMAP4_SSL
use_idle = True use_idle = True
except: except ImportError:
from imaplib import IMAP4, IMAP4_SSL from imaplib import IMAP4, IMAP4_SSL
use_idle = False use_idle = False
import contextlib
import time
import socket import socket
from threading import * from threading import Thread
from i3pystatus.mail import Backend from i3pystatus.mail import Backend
# This is the threading object that does all the waiting on IMAP_EXCEPTIONS = (socket.error, socket.gaierror, IMAP4.abort, IMAP4.error)
# the event
class Idler(object):
def __init__(self, conn, callback, callback_reconnect):
self.thread = Thread(target=self.idle)
self.M = conn
self.callback_reconnect = callback_reconnect
self.event = Event()
self.callback = callback
def start(self):
self.thread.start()
def stop(self):
self.event.set()
def join(self):
self.thread.join()
def idle(self):
while True:
if self.event.isSet():
return
self.needsync = False
def callback(args):
if not self.event.isSet():
self.needsync = True
self.event.set()
try:
self.M.idle(callback=callback)
self.event.wait()
if self.needsync:
self.event.clear()
self.callback()
except:
break
self.M = self.callback_reconnect()
self.stop()
self.start()
class IMAP(Backend): class IMAP(Backend):
@ -84,46 +42,55 @@ class IMAP(Backend):
if self.ssl: if self.ssl:
self.imap_class = IMAP4_SSL self.imap_class = IMAP4_SSL
self.conn = self.get_connection()
if use_idle: if use_idle:
idler = Idler(self.conn, self.count_new_mail, self.get_connection) self.thread = Thread(target=self._idle_thread)
idler.start() self.daemon = True
self.thread.start()
def get_connection(self): @contextlib.contextmanager
def ensure_connection(self):
try:
if self.connection: if self.connection:
try:
self.connection.select(self.mailbox) self.connection.select(self.mailbox)
except socket.error:
# NOTE(sileht): retry just once if the connection have been
# broken to ensure this is not a sporadic connection lost.
# Like wifi reconnect, sleep wake up
try:
self.connection.logout()
except socket.error:
pass
self.connection = None
if not self.connection: if not self.connection:
self.connection = self.imap_class(self.host, self.port) self.connection = self.imap_class(self.host, self.port)
self.connection.login(self.username, self.password) self.connection.login(self.username, self.password)
self.connection.select(self.mailbox) self.connection.select(self.mailbox)
yield
except IMAP_EXCEPTIONS:
# NOTE(sileht): retry just once if the connection have been
# broken to ensure this is not a sporadic connection lost.
# Like wifi reconnect, sleep wake up
try:
self.connection.close()
except IMAP_EXCEPTIONS:
pass
try:
self.connection.logout()
except IMAP_EXCEPTIONS:
pass
# Wait a bit when disconnection occurs to not hog the cpu
time.sleep(1)
self.connection = None
return self.connection def _idle_thread(self):
# update mail count on startup
with self.ensure_connection():
self.count_new_mail()
while True:
with self.ensure_connection():
# Block until new mails
self.connection.idle()
# Read how many
self.count_new_mail()
def count_new_mail(self): def count_new_mail(self):
self.last = len(self.conn.search(None, "UnSeen")[1][0].split()) self.last = len(self.connection.search(None, "UnSeen")[1][0].split())
@property @property
def unread(self): def unread(self):
try:
conn = self.get_connection()
except socket.gaierror:
pass
else:
self.last = len(conn.search(None, "UnSeen")[1][0].split())
finally:
if not use_idle: if not use_idle:
with self.ensure_connection():
self.count_new_mail() self.count_new_mail()
return self.last return self.last