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:
parent
0bb62515a8
commit
cb2bb6ee6b
1 changed files with 12 additions and 1 deletions
|
@ -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):
|
||||||
|
|
Loading…
Add table
Reference in a new issue