bug fix: properly stop ConnectionManager

It is possible (likely) that a manage call is in progress when
`stop` is called. When that happens, _manage will continue to
run, and schedule another call - and the manager won't actually stop,
and will likely cause an error as other components have been torn down.

This fix adds a deferred that gets created when a manage call starts
and is fired when its done.  At this points its safe to start the
stopping process.  Also add a check to not schedule another manage
call if we're stopped

This fixes https://app.asana.com/0/142330900434470/239832897034382
This commit is contained in:
Job Evers-Meltzer 2017-01-11 11:45:37 -06:00
parent 0bb62515a8
commit cb2bb6ee6b

View file

@ -29,6 +29,8 @@ class ConnectionManager(object):
self._peer_connections = {} # {Peer: PeerConnectionHandler} self._peer_connections = {} # {Peer: PeerConnectionHandler}
self._connections_closing = {} # {Peer: deferred (fired when the connection is closed)} self._connections_closing = {} # {Peer: deferred (fired when the connection is closed)}
self._next_manage_call = None self._next_manage_call = None
# a deferred that gets fired when a _manage call is set
self._manage_deferred = None
self.stopped = True self.stopped = True
def start(self): def start(self):
@ -44,6 +46,10 @@ class ConnectionManager(object):
@defer.inlineCallbacks @defer.inlineCallbacks
def stop(self): def stop(self):
self.stopped = True self.stopped = True
# wait for the current manage call to finish
if self._manage_deferred:
yield self._manage_deferred
# in case we stopped between manage calls, cancel the next one
if self._next_manage_call and self._next_manage_call.active(): if self._next_manage_call and self._next_manage_call.active():
self._next_manage_call.cancel() self._next_manage_call.cancel()
self._next_manage_call = None self._next_manage_call = None
@ -127,6 +133,8 @@ class ConnectionManager(object):
@defer.inlineCallbacks @defer.inlineCallbacks
def _manage(self): def _manage(self):
self._manage_deferred = defer.Deferred()
from twisted.internet import reactor from twisted.internet import reactor
if len(self._peer_connections) < conf.settings.max_connections_per_stream: if len(self._peer_connections) < conf.settings.max_connections_per_stream:
try: try:
@ -137,6 +145,9 @@ class ConnectionManager(object):
except Exception: except Exception:
# log this otherwise it will just end up as an unhandled error in deferred # log this otherwise it will just end up as an unhandled error in deferred
log.exception('Something bad happened picking a peer') log.exception('Something bad happened picking a peer')
self._manage_deferred.callback(None)
self._manage_deferred = None
if not self.stopped:
self._next_manage_call = reactor.callLater(1, self._manage) self._next_manage_call = reactor.callLater(1, self._manage)
def _rank_request_creator_connections(self): def _rank_request_creator_connections(self):