@ -197,19 +197,35 @@ class BlockProcessor(server.db.DB):
self . logger . info ( ' flushing DB cache at {:,d} MB '
self . logger . info ( ' flushing DB cache at {:,d} MB '
. format ( self . cache_MB ) )
. format ( self . cache_MB ) )
async def executor ( self , func , * args , * * kwargs ) :
''' Run func taking args in the executor. '''
loop = asyncio . get_event_loop ( )
await loop . run_in_executor ( None , partial ( func , * args , * * kwargs ) )
def caught_up ( self ) :
''' Called when first caught up after starting. '''
if not self . caught_up_event . is_set ( ) :
self . first_sync = False
self . flush ( True )
if self . utxo_db . for_sync :
self . logger . info ( ' {} synced to height {:,d} '
. format ( VERSION , self . height ) )
self . open_dbs ( )
self . caught_up_event . set ( )
async def main_loop ( self , touched ) :
async def main_loop ( self , touched ) :
''' Main loop for block processing. '''
''' Main loop for block processing. '''
# Simulate a reorg if requested
if self . env . force_reorg > 0 :
self . logger . info ( ' DEBUG: simulating reorg of {:,d} blocks '
. format ( self . env . force_reorg ) )
await self . handle_chain_reorg ( set ( ) , self . env . force_reorg )
while True :
while True :
blocks = await self . prefetcher . get_blocks ( )
blocks = await self . prefetcher . get_blocks ( )
if blocks :
if blocks :
await self . advance_blocks ( blocks , touched )
start = time . time ( )
await self . check_and_advance_blocks ( blocks , touched )
if not self . first_sync :
s = ' ' if len ( blocks ) == 1 else ' s '
self . logger . info ( ' processed {:,d} block {} in {:.1f} s '
. format ( len ( blocks ) , s ,
time . time ( ) - start ) )
elif blocks is None :
elif blocks is None :
break # Shutdown
break # Shutdown
else :
else :
@ -219,22 +235,50 @@ class BlockProcessor(server.db.DB):
self . flush ( True )
self . flush ( True )
self . logger . info ( ' shutdown complete ' )
self . logger . info ( ' shutdown complete ' )
async def advance_blocks ( self , blocks , touched ) :
async def check_and_ advance_blocks( self , blocks , touched ) :
''' Process the list of blocks passed. Detects and handles reorgs. '''
''' Process the list of blocks passed. Detects and handles reorgs. '''
def job ( ) :
first = self . height + 1
for block in blocks :
headers = [ self . coin . block_header ( block , first + n )
self . advance_block ( block , touched )
for n , block in enumerate ( blocks ) ]
hprevs = [ self . coin . header_prevhash ( h ) for h in headers ]
start = time . time ( )
chain = [ self . tip ] + [ self . coin . header_hash ( h ) for h in headers [ : - 1 ] ]
loop = asyncio . get_event_loop ( )
try :
if hprevs == chain :
await loop . run_in_executor ( None , job )
await self . executor ( self . advance_blocks , blocks , headers , touched )
except ChainReorg :
elif hprevs [ 0 ] != chain [ 0 ] :
await self . handle_chain_reorg ( touched )
await self . handle_chain_reorg ( touched )
else :
# It is probably possible but extremely rare that what
# bitcoind returns doesn't form a chain because it
# reorg-ed the chain as it was processing the batched
# block hash requests. Should this happen it's simplest
# just to reset the prefetcher and try again.
self . logger . warning ( ' daemon blocks do not form a chain; '
' resetting the prefetcher ' )
await self . prefetcher . clear ( self . height )
def advance_blocks ( self , blocks , headers , touched ) :
''' Synchronously advance the blocks.
It is already verified they correctly connect onto our tip .
'''
block_txs = self . coin . block_txs
daemon_height = self . daemon . cached_height ( )
for block in blocks :
self . height + = 1
txs = block_txs ( block , self . height )
self . tx_hashes . append ( b ' ' . join ( tx_hash for tx , tx_hash in txs ) )
undo_info = self . advance_txs ( txs , touched )
if daemon_height - self . height < = self . env . reorg_limit :
self . write_undo_info ( self . height , b ' ' . join ( undo_info ) )
self . headers . extend ( headers )
self . tip = self . coin . header_hash ( headers [ - 1 ] )
# If caught up, flush everything as client queries are
# performed on the DB.
if self . caught_up_event . is_set ( ) :
if self . caught_up_event . is_set ( ) :
# Flush everything as queries are performed on the DB and
# not in-memory.
self . flush ( True )
self . flush ( True )
else :
else :
touched . clear ( )
touched . clear ( )
@ -242,38 +286,20 @@ class BlockProcessor(server.db.DB):
self . check_cache_size ( )
self . check_cache_size ( )
self . next_cache_check = time . time ( ) + 30
self . next_cache_check = time . time ( ) + 30
if not self . first_sync :
s = ' ' if len ( blocks ) == 1 else ' s '
self . logger . info ( ' processed {:,d} block {} in {:.1f} s '
. format ( len ( blocks ) , s ,
time . time ( ) - start ) )
def caught_up ( self ) :
''' Called when first caught up after starting. '''
if not self . caught_up_event . is_set ( ) :
self . first_sync = False
self . flush ( True )
if self . utxo_db . for_sync :
self . logger . info ( ' {} synced to height {:,d} '
. format ( VERSION , self . height ) )
self . open_dbs ( )
self . caught_up_event . set ( )
async def handle_chain_reorg ( self , touched , count = None ) :
async def handle_chain_reorg ( self , touched , count = None ) :
''' Handle a chain reorganisation.
''' Handle a chain reorganisation.
Count is the number of blocks to simulate a reorg , or None for
Count is the number of blocks to simulate a reorg , or None for
a real reorg . '''
a real reorg . '''
self . logger . info ( ' chain reorg detected ' )
self . logger . info ( ' chain reorg detected ' )
self . flush ( True )
await self . executor ( self . flush , True )
hashes = await self . reorg_hashes ( count )
hashes = await self . reorg_hashes ( count )
# Reverse and convert to hex strings.
# Reverse and convert to hex strings.
hashes = [ hash_to_str ( hash ) for hash in reversed ( hashes ) ]
hashes = [ hash_to_str ( hash ) for hash in reversed ( hashes ) ]
for hex_hashes in chunks ( hashes , 50 ) :
for hex_hashes in chunks ( hashes , 50 ) :
blocks = await self . daemon . raw_blocks ( hex_hashes )
blocks = await self . daemon . raw_blocks ( hex_hashes )
self . backup_blocks ( blocks , touched )
await self . executor ( self . backup_blocks , blocks , touched )
await self . prefetcher . clear ( self . height )
await self . prefetcher . clear ( self . height )
async def reorg_hashes ( self , count ) :
async def reorg_hashes ( self , count ) :
@ -306,9 +332,10 @@ class BlockProcessor(server.db.DB):
else :
else :
start = ( self . height - count ) + 1
start = ( self . height - count ) + 1
self . logger . info ( ' chain was reorganised: {:,d} blocks at '
s = ' ' if count == 1 else ' s '
' heights {:,d} - {:,d} were replaced '
self . logger . info ( ' chain was reorganised replacing {:,d} block {} at '
. format ( count , start , start + count - 1 ) )
' heights {:,d} - {:,d} '
. format ( count , s , start , start + count - 1 ) )
return self . fs_block_hashes ( start , count )
return self . fs_block_hashes ( start , count )
@ -462,27 +489,6 @@ class BlockProcessor(server.db.DB):
if utxo_MB + hist_MB > = self . cache_MB or hist_MB > = self . cache_MB / / 5 :
if utxo_MB + hist_MB > = self . cache_MB or hist_MB > = self . cache_MB / / 5 :
self . flush ( utxo_MB > = self . cache_MB * 4 / / 5 )
self . flush ( utxo_MB > = self . cache_MB * 4 / / 5 )
def fs_advance_block ( self , header , txs ) :
''' Update unflushed FS state for a new block. '''
prior_tx_count = self . tx_counts [ - 1 ] if self . tx_counts else 0
# Cache the new header, tx hashes and cumulative tx count
self . headers . append ( header )
self . tx_hashes . append ( b ' ' . join ( tx_hash for tx , tx_hash in txs ) )
self . tx_counts . append ( prior_tx_count + len ( txs ) )
def advance_block ( self , block , touched ) :
header , txs = self . coin . read_block ( block , self . height + 1 )
if self . tip != self . coin . header_prevhash ( header ) :
raise ChainReorg
self . fs_advance_block ( header , txs )
self . tip = self . coin . header_hash ( header )
self . height + = 1
undo_info = self . advance_txs ( txs , touched )
if self . daemon . cached_height ( ) - self . height < = self . env . reorg_limit :
self . write_undo_info ( self . height , b ' ' . join ( undo_info ) )
def advance_txs ( self , txs , touched ) :
def advance_txs ( self , txs , touched ) :
undo_info = [ ]
undo_info = [ ]
@ -524,6 +530,7 @@ class BlockProcessor(server.db.DB):
tx_num + = 1
tx_num + = 1
self . tx_count = tx_num
self . tx_count = tx_num
self . tx_counts . append ( tx_num )
self . history_size = history_size
self . history_size = history_size
return undo_info
return undo_info
@ -537,7 +544,7 @@ class BlockProcessor(server.db.DB):
self . assert_flushed ( )
self . assert_flushed ( )
for block in blocks :
for block in blocks :
header , txs = self . coin . read_ block( block , self . height )
txs = self . coin . block_txs ( block , self . height )
header_hash = self . coin . header_hash ( header )
header_hash = self . coin . header_hash ( header )
if header_hash != self . tip :
if header_hash != self . tip :
raise ChainError ( ' backup block {} is not tip {} at height {:,d} '
raise ChainError ( ' backup block {} is not tip {} at height {:,d} '