Compare commits
2 commits
master
...
fix_order3
Author | SHA1 | Date | |
---|---|---|---|
|
c6a4d5602e | ||
|
c73f64c424 |
2 changed files with 29 additions and 49 deletions
|
@ -513,10 +513,8 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
)
|
)
|
||||||
return True
|
return True
|
||||||
|
|
||||||
synced_txs = []
|
|
||||||
to_request = {}
|
to_request = {}
|
||||||
pending_synced_history = {}
|
pending_synced_history = {}
|
||||||
updated_cached_items = {}
|
|
||||||
already_synced = set()
|
already_synced = set()
|
||||||
|
|
||||||
already_synced_offset = 0
|
already_synced_offset = 0
|
||||||
|
@ -526,40 +524,22 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
already_synced.add((txid, remote_height))
|
already_synced.add((txid, remote_height))
|
||||||
already_synced_offset += 1
|
already_synced_offset += 1
|
||||||
continue
|
continue
|
||||||
cache_item = self._tx_cache.get(txid)
|
|
||||||
if cache_item is None:
|
|
||||||
cache_item = TransactionCacheItem()
|
|
||||||
self._tx_cache[txid] = cache_item
|
|
||||||
|
|
||||||
tx_indexes = {}
|
tx_indexes = {}
|
||||||
|
|
||||||
for i, (txid, remote_height) in enumerate(remote_history):
|
for i, (txid, remote_height) in enumerate(remote_history):
|
||||||
tx_indexes[txid] = i
|
tx_indexes[txid] = i
|
||||||
if (txid, remote_height) in already_synced:
|
if (txid, remote_height) in already_synced:
|
||||||
continue
|
continue
|
||||||
cache_item = self._tx_cache.get(txid)
|
|
||||||
cache_item.pending_verifications += 1
|
|
||||||
updated_cached_items[txid] = cache_item
|
|
||||||
|
|
||||||
assert cache_item is not None, 'cache item is none'
|
|
||||||
# tx = cache_item.tx
|
|
||||||
# if cache_item.tx is not None and \
|
|
||||||
# cache_item.tx.height >= remote_height and \
|
|
||||||
# (cache_item.tx.is_verified or remote_height < 1):
|
|
||||||
# synced_txs.append(cache_item.tx) # cached tx is already up-to-date
|
|
||||||
# pending_synced_history[i] = f'{tx.id}:{tx.height}:'
|
|
||||||
# continue
|
|
||||||
to_request[i] = (txid, remote_height)
|
to_request[i] = (txid, remote_height)
|
||||||
|
|
||||||
log.debug(
|
log.debug(
|
||||||
"request %i transactions, %i/%i for %s are already synced", len(to_request), len(synced_txs),
|
"request %i transactions, %i/%i for %s are already synced", len(to_request), len(already_synced),
|
||||||
len(remote_history), address
|
len(remote_history), address
|
||||||
)
|
)
|
||||||
remote_history_txids = set(txid for txid, _ in remote_history)
|
remote_history_txids = set(txid for txid, _ in remote_history)
|
||||||
async for tx in self.request_synced_transactions(to_request, remote_history_txids, address):
|
async for tx in self.request_synced_transactions(to_request, remote_history_txids, address):
|
||||||
pending_synced_history[tx_indexes[tx.id]] = f"{tx.id}:{tx.height}:"
|
pending_synced_history[tx_indexes[tx.id]] = f"{tx.id}:{tx.height}:"
|
||||||
synced_txs.append(tx)
|
if len(pending_synced_history) % 100 == 0:
|
||||||
if len(synced_txs) >= 100:
|
|
||||||
log.info("Syncing address %s: %d/%d", address, len(pending_synced_history), len(to_request))
|
log.info("Syncing address %s: %d/%d", address, len(pending_synced_history), len(to_request))
|
||||||
log.info("Sync finished for address %s: %d/%d", address, len(pending_synced_history), len(to_request))
|
log.info("Sync finished for address %s: %d/%d", address, len(pending_synced_history), len(to_request))
|
||||||
|
|
||||||
|
@ -572,19 +552,7 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
if f"{txid}:{height}:" != pending_synced_history[i]:
|
if f"{txid}:{height}:" != pending_synced_history[i]:
|
||||||
log.warning("history mismatch: %s vs %s", remote_history[remote_i], pending_synced_history[i])
|
log.warning("history mismatch: %s vs %s", remote_history[remote_i], pending_synced_history[i])
|
||||||
synced_history += pending_synced_history[i]
|
synced_history += pending_synced_history[i]
|
||||||
await self.db.save_transaction_io_batch(
|
await self.db.set_address_history(address, synced_history)
|
||||||
synced_txs, address, self.address_to_hash160(address), synced_history
|
|
||||||
)
|
|
||||||
while synced_txs:
|
|
||||||
self._on_transaction_controller.add(TransactionEvent(address, synced_txs.pop()))
|
|
||||||
|
|
||||||
cache_size = self.config.get("tx_cache_size", 10_000)
|
|
||||||
for txid, cache_item in updated_cached_items.items():
|
|
||||||
cache_item.pending_verifications -= 1
|
|
||||||
if cache_item.pending_verifications < 0:
|
|
||||||
log.warning("config value tx cache size %i needs to be increased", cache_size)
|
|
||||||
cache_item.pending_verifications = 0
|
|
||||||
|
|
||||||
|
|
||||||
if address_manager is None:
|
if address_manager is None:
|
||||||
address_manager = await self.get_address_manager_for_address(address)
|
address_manager = await self.get_address_manager_for_address(address)
|
||||||
|
@ -592,13 +560,6 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
if address_manager is not None:
|
if address_manager is not None:
|
||||||
await address_manager.ensure_address_gap()
|
await address_manager.ensure_address_gap()
|
||||||
|
|
||||||
for txid, cache_item in updated_cached_items.items():
|
|
||||||
if self._tx_cache.get(txid) is not cache_item:
|
|
||||||
log.warning("tx cache corrupted while syncing %s, reattempt sync=%s", address, reattempt_update)
|
|
||||||
if reattempt_update:
|
|
||||||
return await self.update_history(address, remote_status, address_manager, False)
|
|
||||||
return False
|
|
||||||
|
|
||||||
local_status, local_history = \
|
local_status, local_history = \
|
||||||
await self.get_local_status_and_history(address, synced_history)
|
await self.get_local_status_and_history(address, synced_history)
|
||||||
|
|
||||||
|
@ -654,6 +615,13 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
remote_heights = {}
|
remote_heights = {}
|
||||||
|
|
||||||
for txid, height in sorted(to_request, key=lambda x: x[1]):
|
for txid, height in sorted(to_request, key=lambda x: x[1]):
|
||||||
|
if txid in self._tx_cache:
|
||||||
|
tx = self._tx_cache[txid].tx
|
||||||
|
if tx is not None and tx.is_verified:
|
||||||
|
yield tx
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
self._tx_cache[txid] = TransactionCacheItem()
|
||||||
remote_heights[txid] = height
|
remote_heights[txid] = height
|
||||||
if len(batches[-1]) == 100:
|
if len(batches[-1]) == 100:
|
||||||
batches.append([])
|
batches.append([])
|
||||||
|
@ -661,24 +629,36 @@ class Ledger(metaclass=LedgerRegistry):
|
||||||
if not batches[-1]:
|
if not batches[-1]:
|
||||||
batches.pop()
|
batches.pop()
|
||||||
|
|
||||||
for batch in batches:
|
for routine in asyncio.as_completed([self._single_batch(batch, remote_heights) for batch in batches]):
|
||||||
async for tx in self._single_batch(batch, remote_heights):
|
for tx in await routine:
|
||||||
yield tx
|
yield tx
|
||||||
|
|
||||||
async def request_synced_transactions(self, to_request, remote_history, address):
|
async def request_synced_transactions(self, to_request, remote_history, address):
|
||||||
pending_sync = []
|
pending_sync = []
|
||||||
async for tx in self.request_transactions(((txid, height) for txid, height in to_request.values())):
|
async for tx in self.request_transactions(((txid, height) for txid, height in to_request.values())):
|
||||||
pending_sync.append(asyncio.ensure_future(self._sync(tx, remote_history)))
|
pending_sync.append(asyncio.ensure_future(self._sync(tx, remote_history)))
|
||||||
|
log.info("%d/%d", len(pending_sync), len(to_request))
|
||||||
yield tx
|
yield tx
|
||||||
await asyncio.gather(*pending_sync)
|
pending_sync = await asyncio.gather(*pending_sync)
|
||||||
|
await self.db.save_transaction_io_batch(
|
||||||
|
pending_sync, address, self.address_to_hash160(address), ""
|
||||||
|
)
|
||||||
|
while pending_sync:
|
||||||
|
self._on_transaction_controller.add(TransactionEvent(address, pending_sync.pop()))
|
||||||
|
|
||||||
async def _single_batch(self, batch, remote_heights):
|
async def _single_batch(self, batch, remote_heights):
|
||||||
batch_result = await self.network.retriable_call(self.network.get_transaction_batch, batch)
|
heights = {remote_heights[txid] for txid in batch}
|
||||||
|
unrestriced = 0 < min(heights) < max(heights) < max(self.headers.checkpoints or [0])
|
||||||
|
batch_result = await self.network.retriable_call(self.network.get_transaction_batch, batch, not unrestriced)
|
||||||
|
txs = []
|
||||||
|
pending_verification = []
|
||||||
for txid, (raw, merkle) in batch_result.items():
|
for txid, (raw, merkle) in batch_result.items():
|
||||||
remote_height = remote_heights[txid]
|
remote_height = remote_heights[txid]
|
||||||
tx = Transaction(unhexlify(raw), height=remote_height)
|
tx = Transaction(unhexlify(raw), height=remote_height)
|
||||||
await self.maybe_verify_transaction(tx, remote_height, merkle)
|
pending_verification.append(self.maybe_verify_transaction(tx, remote_height, merkle))
|
||||||
yield tx
|
txs.append(tx)
|
||||||
|
await asyncio.gather(*pending_verification)
|
||||||
|
return txs
|
||||||
|
|
||||||
async def _sync(self, tx, remote_history):
|
async def _sync(self, tx, remote_history):
|
||||||
check_db_for_txos = {}
|
check_db_for_txos = {}
|
||||||
|
|
|
@ -40,7 +40,7 @@ class MockNetwork:
|
||||||
merkle = await self.get_merkle(tx_hash, known_height)
|
merkle = await self.get_merkle(tx_hash, known_height)
|
||||||
return tx, merkle
|
return tx, merkle
|
||||||
|
|
||||||
async def get_transaction_batch(self, txids):
|
async def get_transaction_batch(self, txids, restricted):
|
||||||
return {
|
return {
|
||||||
txid: await self.get_transaction_and_merkle(txid)
|
txid: await self.get_transaction_and_merkle(txid)
|
||||||
for txid in txids
|
for txid in txids
|
||||||
|
|
Loading…
Add table
Reference in a new issue