@ -13,6 +13,7 @@ import ssl
import time
import time
from bisect import bisect_left
from bisect import bisect_left
from collections import defaultdict
from collections import defaultdict
from concurrent . futures import ThreadPoolExecutor
from functools import partial
from functools import partial
import pylru
import pylru
@ -53,6 +54,8 @@ class Controller(util.LoggedClass):
# Set this event to cleanly shutdown
# Set this event to cleanly shutdown
self . shutdown_event = asyncio . Event ( )
self . shutdown_event = asyncio . Event ( )
self . loop = asyncio . get_event_loop ( )
self . loop = asyncio . get_event_loop ( )
self . executor = ThreadPoolExecutor ( )
self . loop . set_default_executor ( self . executor )
self . start = time . time ( )
self . start = time . time ( )
self . coin = env . coin
self . coin = env . coin
self . daemon = Daemon ( env . coin . daemon_urls ( env . daemon_url ) )
self . daemon = Daemon ( env . coin . daemon_urls ( env . daemon_url ) )
@ -215,8 +218,8 @@ class Controller(util.LoggedClass):
for n in range ( 4 ) :
for n in range ( 4 ) :
add_future ( self . serve_requests ( ) )
add_future ( self . serve_requests ( ) )
bp_future = asyncio . ensure_future ( self . bp . main_loop ( ) )
futures = [ ]
futures = [ ]
add_future ( self . bp . main_loop ( ) )
add_future ( self . bp . prefetcher . main_loop ( ) )
add_future ( self . bp . prefetcher . main_loop ( ) )
add_future ( await_bp_catchup ( ) )
add_future ( await_bp_catchup ( ) )
@ -225,35 +228,36 @@ class Controller(util.LoggedClass):
self . logger . info ( ' shutting down gracefully ' )
self . logger . info ( ' shutting down gracefully ' )
self . state = self . SHUTTING_DOWN
self . state = self . SHUTTING_DOWN
# First tell the block processor to shut down, it may need to
# Close servers and sessions
# perform a lengthy flush. Then shut down the rest.
self . bp . on_shutdown ( )
self . close_servers ( list ( self . servers . keys ( ) ) )
self . close_servers ( list ( self . servers . keys ( ) ) )
for session in self . sessions :
self . close_session ( session )
# Cancel the futures
for future in futures :
for future in futures :
future . cancel ( )
future . cancel ( )
# Now wait for the cleanup to complete
await asyncio . wait ( futures )
await self . close_sessions ( )
if not bp_future . done ( ) :
# Wait for the executor to finish anything it's doing
self . logger . info ( ' waiting for block processor ' )
self . executor . shutdown ( )
await bp_future
self . bp . shutdown ( )
def close_servers ( self , kinds ) :
def close_servers ( self , kinds ) :
''' Close the servers of the given kinds (TCP etc.). '''
''' Close the servers of the given kinds (TCP etc.). '''
self . logger . info ( ' closing down {} listening servers '
if kinds :
. format ( ' , ' . join ( kinds ) ) )
self . logger . info ( ' closing down {} listening servers '
. format ( ' , ' . join ( kinds ) ) )
for kind in kinds :
for kind in kinds :
server = self . servers . pop ( kind , None )
server = self . servers . pop ( kind , None )
if server :
if server :
server . close ( )
server . close ( )
async def close _sessions( self , secs = 30 ) :
async def wait_for _sessions( self , secs = 30 ) :
if not self . sessions :
if not self . sessions :
return
return
self . logger . info ( ' waiting up to {:d} seconds for socket cleanup '
self . logger . info ( ' waiting up to {:d} seconds for socket cleanup '
. format ( secs ) )
. format ( secs ) )
for session in self . sessions :
self . close_session ( session )
limit = time . time ( ) + secs
limit = time . time ( ) + secs
while self . sessions and time . time ( ) < limit :
while self . sessions and time . time ( ) < limit :
self . clear_stale_sessions ( grace = secs / / 2 )
self . clear_stale_sessions ( grace = secs / / 2 )