Browse Source

Merge branch 'utxo_optimised' into develop

master
Neil Booth 8 years ago
parent
commit
15e052c728
  1. 2
      query.py
  2. 283
      server/block_processor.py
  3. 100
      server/db.py

2
query.py

@ -16,7 +16,7 @@ Not currently documented; might become easier to use in future.
import sys import sys
from server.env import Env from server.env import Env
from server.DB import DB from server.db import DB
from lib.hash import hash_to_str from lib.hash import hash_to_str

283
server/block_processor.py

@ -12,7 +12,7 @@ import array
import asyncio import asyncio
import itertools import itertools
import os import os
import struct from struct import pack, unpack
import time import time
from bisect import bisect_left from bisect import bisect_left
from collections import defaultdict from collections import defaultdict
@ -28,11 +28,10 @@ from server.storage import open_db
# Limits single address history to ~ 65536 * HIST_ENTRIES_PER_KEY entries # Limits single address history to ~ 65536 * HIST_ENTRIES_PER_KEY entries
HIST_ENTRIES_PER_KEY = 1024 HIST_ENTRIES_PER_KEY = 1024
HIST_VALUE_BYTES = HIST_ENTRIES_PER_KEY * 4 HIST_VALUE_BYTES = HIST_ENTRIES_PER_KEY * 4
ADDR_TX_HASH_LEN = 4
UTXO_TX_HASH_LEN = 4
NO_HASH_168 = bytes([255]) * 21 NO_HASH_168 = bytes([255]) * 21
NO_CACHE_ENTRY = NO_HASH_168 + bytes(12) NO_CACHE_ENTRY = NO_HASH_168 + bytes(12)
def formatted_time(t): def formatted_time(t):
'''Return a number of seconds as a string in days, hours, mins and '''Return a number of seconds as a string in days, hours, mins and
secs.''' secs.'''
@ -143,10 +142,6 @@ class Prefetcher(LoggedClass):
return blocks, size return blocks, size
class MissingUTXOError(Exception):
'''Raised if a mempool tx input UTXO couldn't be found.'''
class ChainReorg(Exception): class ChainReorg(Exception):
'''Raised on a blockchain reorganisation.''' '''Raised on a blockchain reorganisation.'''
@ -214,7 +209,7 @@ class MemPool(LoggedClass):
# The mempool is unordered, so process all outputs first so # The mempool is unordered, so process all outputs first so
# that looking for inputs has full info. # that looking for inputs has full info.
script_hash168 = self.bp.coin.hash168_from_script script_hash168 = self.bp.coin.hash168_from_script
utxo_lookup = self.bp.utxo_lookup db_utxo_lookup = self.bp.db_utxo_lookup
def txout_pair(txout): def txout_pair(txout):
return (script_hash168(txout.pk_script), txout.value) return (script_hash168(txout.pk_script), txout.value)
@ -231,14 +226,8 @@ class MemPool(LoggedClass):
mempool_entry = self.txs.get(hex_hash) mempool_entry = self.txs.get(hex_hash)
if mempool_entry: if mempool_entry:
return mempool_entry[1][txin.prev_idx], True return mempool_entry[1][txin.prev_idx], True
entry = utxo_lookup(txin.prev_hash, txin.prev_idx) pair = db_utxo_lookup(txin.prev_hash, txin.prev_idx)
if entry == NO_CACHE_ENTRY: return pair, False
# This happens when the daemon is a block ahead of us
# and has mempool txs spending new txs in that block
missing_utxos.append(txin)
raise MissingUTXOError
value, = struct.unpack('<Q', entry[-8:])
return (entry[:21], value), False
if initial: if initial:
next_log = time.time() next_log = time.time()
@ -260,7 +249,7 @@ class MemPool(LoggedClass):
try: try:
infos = (txin_info(txin) for txin in tx.inputs) infos = (txin_info(txin) for txin in tx.inputs)
txin_pairs, unconfs = zip(*infos) txin_pairs, unconfs = zip(*infos)
except MissingUTXOError: except self.bp.MissingUTXOError:
# Drop this TX. If other mempool txs depend on it # Drop this TX. If other mempool txs depend on it
# it's harmless - next time the mempool is refreshed # it's harmless - next time the mempool is refreshed
# they'll either be cleaned up or the UTXOs will no # they'll either be cleaned up or the UTXOs will no
@ -499,7 +488,6 @@ class BlockProcessor(server.db.DB):
def remove_excess_history(self, batch): def remove_excess_history(self, batch):
prefix = b'H' prefix = b'H'
unpack = struct.unpack
keys = [] keys = []
for key, hist in self.db.iterator(prefix=prefix): for key, hist in self.db.iterator(prefix=prefix):
flush_id, = unpack('>H', key[-2:]) flush_id, = unpack('>H', key[-2:])
@ -513,7 +501,6 @@ class BlockProcessor(server.db.DB):
def remove_stale_undo_items(self, batch): def remove_stale_undo_items(self, batch):
prefix = b'U' prefix = b'U'
unpack = struct.unpack
cutoff = self.db_height - self.reorg_limit cutoff = self.db_height - self.reorg_limit
keys = [] keys = []
for key, hist in self.db.iterator(prefix=prefix): for key, hist in self.db.iterator(prefix=prefix):
@ -614,7 +601,7 @@ class BlockProcessor(server.db.DB):
def flush_history(self, batch): def flush_history(self, batch):
flush_start = time.time() flush_start = time.time()
flush_id = struct.pack('>H', self.flush_count) flush_id = pack('>H', self.flush_count)
for hash168, hist in self.history.items(): for hash168, hist in self.history.items():
key = b'H' + hash168 + flush_id key = b'H' + hash168 + flush_id
@ -736,7 +723,7 @@ class BlockProcessor(server.db.DB):
def undo_key(self, height): def undo_key(self, height):
'''DB key for undo information at the given height.''' '''DB key for undo information at the given height.'''
return b'U' + struct.pack('>I', height) return b'U' + pack('>I', height)
def write_undo_info(self, height, undo_info): def write_undo_info(self, height, undo_info):
'''Write out undo information for the current height.''' '''Write out undo information for the current height.'''
@ -792,11 +779,11 @@ class BlockProcessor(server.db.DB):
history = self.history history = self.history
tx_num = self.tx_count tx_num = self.tx_count
script_hash168 = self.coin.hash168_from_script script_hash168 = self.coin.hash168_from_script
pack = struct.pack s_pack = pack
for tx, tx_hash in zip(txs, tx_hashes): for tx, tx_hash in zip(txs, tx_hashes):
hash168s = set() hash168s = set()
tx_numb = pack('<I', tx_num) tx_numb = s_pack('<I', tx_num)
# Spend the inputs # Spend the inputs
if not tx.is_coinbase: if not tx.is_coinbase:
@ -811,8 +798,8 @@ class BlockProcessor(server.db.DB):
hash168 = script_hash168(txout.pk_script) hash168 = script_hash168(txout.pk_script)
if hash168: if hash168:
hash168s.add(hash168) hash168s.add(hash168)
put_utxo(tx_hash + pack('<H', idx), put_utxo(tx_hash + s_pack('<H', idx),
hash168 + tx_numb + pack('<Q', txout.value)) hash168 + tx_numb + s_pack('<Q', txout.value))
# Drop any NO_CACHE entry # Drop any NO_CACHE entry
hash168s.discard(NO_CACHE_ENTRY) hash168s.discard(NO_CACHE_ENTRY)
@ -865,7 +852,7 @@ class BlockProcessor(server.db.DB):
n = len(undo_info) n = len(undo_info)
# Use local vars for speed in the loops # Use local vars for speed in the loops
pack = struct.pack s_pack = pack
put_utxo = self.utxo_cache.__setitem__ put_utxo = self.utxo_cache.__setitem__
spend_utxo = self.spend_utxo spend_utxo = self.spend_utxo
@ -883,7 +870,7 @@ class BlockProcessor(server.db.DB):
for txin in reversed(tx.inputs): for txin in reversed(tx.inputs):
n -= 33 n -= 33
undo_item = undo_info[n:n + 33] undo_item = undo_info[n:n + 33]
put_utxo(txin.prev_hash + pack('<H', txin.prev_idx), put_utxo(txin.prev_hash + s_pack('<H', txin.prev_idx),
undo_item) undo_item)
touched.add(undo_item[:21]) touched.add(undo_item[:21])
@ -893,12 +880,12 @@ class BlockProcessor(server.db.DB):
'''An in-memory UTXO cache, representing all changes to UTXO state '''An in-memory UTXO cache, representing all changes to UTXO state
since the last DB flush. since the last DB flush.
We want to store millions, perhaps 10s of millions of these in We want to store millions of these in memory for optimal
memory for optimal performance during initial sync, because then performance during initial sync, because then it is possible to
it is possible to spend UTXOs without ever going to the database spend UTXOs without ever going to the database (other than as an
(other than as an entry in the address history, and there is only entry in the address history, and there is only one such entry per
one such entry per TX not per UTXO). So store them in a Python TX not per UTXO). So store them in a Python dictionary with
dictionary with binary keys and values. binary keys and values.
Key: TX_HASH + TX_IDX (32 + 2 = 34 bytes) Key: TX_HASH + TX_IDX (32 + 2 = 34 bytes)
Value: HASH168 + TX_NUM + VALUE (21 + 4 + 8 = 33 bytes) Value: HASH168 + TX_NUM + VALUE (21 + 4 + 8 = 33 bytes)
@ -911,144 +898,99 @@ class BlockProcessor(server.db.DB):
Semantics: Semantics:
add: Add it to the cache dictionary. add: Add it to the cache dictionary.
spend: Remove it if in the cache dictionary.
Otherwise it's been flushed to the DB. Each UTXO
is responsible for two entries in the DB stored using
compressed keys. Mark both for deletion in the next
flush of the in-memory UTXO cache.
A UTXO is stored in the DB in 2 "tables":
1. The output value and tx number. Must be keyed with a
hash168 prefix so the unspent outputs and balance of an
arbitrary address can be looked up with a simple key
traversal.
Key: b'u' + hash168 + compressed_tx_hash + tx_idx
Value: a (tx_num, value) pair
2. Given a prevout, we need to be able to look up the UTXO key
to remove it. As is keyed by hash168 and that is not part
of the prevout, we need a hash168 lookup.
Key: b'h' + compressed tx_hash + tx_idx
Value: (hash168, tx_num) pair
The compressed TX hash is just the first few bytes of the hash of
the TX the UTXO is in (and needn't be the same number of bytes in
each table). As this is not unique there will be collisions;
tx_num is stored to resolve them. The collision rate is around
0.02% for the hash168 table, and almost zero for the UTXO table
(there are around 100 collisions in the whole bitcoin blockchain).
'''
def utxo_lookup(self, prev_hash, prev_idx): spend: Remove it if in the cache dictionary. Otherwise it's
'''Given a prevout, return a pair (hash168, value). been flushed to the DB. Each UTXO is responsible for two
entries in the DB. Mark them for deletion in the next
cache flush.
If the UTXO is not found, returns (None, None).''' The UTXO database format has to be able to do two things efficiently:
# Fast track is it being in the cache
idx_packed = struct.pack('<H', prev_idx)
value = self.utxo_cache.get(prev_hash + idx_packed, None)
if value:
return value
return self.db_lookup(prev_hash, idx_packed, False)
def db_lookup(self, tx_hash, idx_packed, delete=True): 1. Given an address be able to list its UTXOs and their values
'''Return a UTXO from the DB. Remove it if delete is True. so its balance can be efficiently computed.
Return NO_CACHE_ENTRY if it is not in the DB.''' 2. When processing transactions, for each prevout spent - a (tx_hash,
hash168 = self.hash168(tx_hash, idx_packed, delete) idx) pair - we have to be able to remove it from the DB. To send
if not hash168: notifications to clients we also need to know any address it paid
return NO_CACHE_ENTRY to.
# Read the UTXO through the cache from the disk. We have to To this end we maintain two "tables", one for each point above:
# go through the cache because compressed keys can collide.
key = b'u' + hash168 + tx_hash[:UTXO_TX_HASH_LEN] + idx_packed
data = self.db_cache_get(key)
if data is None:
# Uh-oh, this should not happen...
self.logger.error('found no UTXO for {} / {:d} key {}'
.format(hash_to_str(tx_hash),
struct.unpack('<H', idx_packed),
bytes(key).hex()))
return NO_CACHE_ENTRY
if len(data) == 12: 1. Key: b'u' + address_hash168 + tx_num + tx_idx
if delete: Value: the UTXO value as a 64-bit unsigned integer
self.db_deletes += 1
self.db_cache_delete(key)
return hash168 + data
# Resolve the compressed key collison. These should be
# extremely rare.
assert len(data) % 12 == 0
for n in range(0, len(data), 12):
(tx_num, ) = struct.unpack('<I', data[n:n+4])
this_tx_hash, height = self.get_tx_hash(tx_num)
if tx_hash == this_tx_hash:
result = hash168 + data[n:n+12]
if delete:
self.db_deletes += 1
self.db_cache_write(key, data[:n] + data[n+12:])
return result
raise Exception('could not resolve UTXO key collision') 2. Key: b'h' + compressed_tx_hash + tx_idx
Value: [address_hash168 + tx_num]
def spend_utxo(self, prev_hash, prev_idx): The compressed tx hash is just the first few bytes of the hash of
'''Spend a UTXO and return the cache's value. the tx in which the UTXO was created. As this is not unique there
will are potential collisions when saving and looking up UTXOs;
hence why the second table has a list as its value. The collision
can be resolved with the tx_num. The collision rate is almost
zero (I believe there are around 100 collisions in the whole
bitcoin blockchain).
'''
def spend_utxo(self, tx_hash, tx_idx):
'''Spend a UTXO and return the 33-byte value.
If the UTXO is not in the cache it must be on disk. If the UTXO is not in the cache it may be on disk.
''' '''
# Fast track is it being in the cache # Fast track is it being in the cache
idx_packed = struct.pack('<H', prev_idx) idx_packed = pack('<H', tx_idx)
value = self.utxo_cache.pop(prev_hash + idx_packed, None) cache_value = self.utxo_cache.pop(tx_hash + idx_packed, None)
if value: if cache_value:
self.utxo_cache_spends += 1 self.utxo_cache_spends += 1
return value return cache_value
# Spend it from the DB. Read the UTXO through the cache
# because compressed keys can collide.
# The 4 is the COMPRESSED_TX_HASH_LEN
db_key = b'h' + tx_hash[:4] + idx_packed
db_value = self.db_cache_get(db_key)
if db_value is None:
# Probably a strange UTXO
return NO_CACHE_ENTRY
return self.db_lookup(prev_hash, idx_packed) # FIXME: this matches what we did previously but until we store
# all UTXOs isn't safe
if len(db_value) == 25:
udb_key = b'u' + db_value + idx_packed
utxo_value_packed = self.db.get(udb_key)
if utxo_value_packed:
# Remove the UTXO from both tables
self.db_deletes += 1
self.db_cache[db_key] = None
self.db_cache[udb_key] = None
return db_value + utxo_value_packed
# Fall through to below
assert len(db_value) % 25 == 0
# Find which entry, if any, the TX_HASH matches.
for n in range(0, len(db_value), 25):
tx_num, = unpack('<I', db_value[n+21:n+25])
hash, height = self.get_tx_hash(tx_num)
if hash == tx_hash:
match = db_value[n:n+25]
udb_key = b'u' + match + idx_packed
utxo_value_packed = self.db.get(udb_key)
if utxo_value_packed:
# Remove the UTXO from both tables
self.db_deletes += 1
self.db_cache[db_key] = db_value[:n] + db_value[n + 25:]
self.db_cache[udb_key] = None
return match + utxo_value_packed
def hash168(self, tx_hash, idx_packed, delete=True): # Uh-oh, this should not happen...
'''Return the hash168 paid to by the given TXO. raise self.DBError('UTXO {} / {:,d} not found, key {}'
.format(hash_to_str(tx_hash), tx_idx,
bytes(key).hex()))
Look it up in the DB and removes it if delete is True. Return return NO_CACHE_ENTRY
None if not found.
'''
key = b'h' + tx_hash[:ADDR_TX_HASH_LEN] + idx_packed
data = self.db_cache_get(key)
if data is None:
# Assuming the DB is not corrupt, if delete is True this
# indicates a successful spend of a non-standard script
# as we don't currently record those
return None
if len(data) == 25:
if delete:
self.db_cache_delete(key)
return data[:21]
assert len(data) % 25 == 0
# Resolve the compressed key collision using the TX number
for n in range(0, len(data), 25):
(tx_num, ) = struct.unpack('<I', data[n+21:n+25])
my_hash, height = self.get_tx_hash(tx_num)
if my_hash == tx_hash:
if delete:
self.db_cache_write(key, data[:n] + data[n+25:])
return data[n:n+21]
raise Exception('could not resolve hash168 collision')
def db_cache_write(self, key, value):
'''Cache write of a (key, value) pair to the DB.'''
assert(bool(value))
self.db_cache[key] = value
def db_cache_delete(self, key):
'''Cache deletion of a key from the DB.'''
self.db_cache[key] = None
def db_cache_get(self, key): def db_cache_get(self, key):
'''Fetch a value from the DB through our write cache.''' '''Fetch a 'h' value from the DB through our write cache.'''
value = self.db_cache.get(key) value = self.db_cache.get(key)
if value: if value:
return value return value
@ -1062,28 +1004,23 @@ class BlockProcessor(server.db.DB):
self.logger.info('flushing UTXOs: {:,d} txs and {:,d} blocks' self.logger.info('flushing UTXOs: {:,d} txs and {:,d} blocks'
.format(self.tx_count - self.db_tx_count, .format(self.tx_count - self.db_tx_count,
self.height - self.db_height)) self.height - self.db_height))
hcolls = ucolls = 0 collisions = 0
new_utxos = len(self.utxo_cache) new_utxos = len(self.utxo_cache)
for cache_key, cache_value in self.utxo_cache.items(): for cache_key, cache_value in self.utxo_cache.items():
# Frist write to the hash168 lookup table # Frist write to the hash168 lookup table
key = b'h' + cache_key[:ADDR_TX_HASH_LEN] + cache_key[-2:] # The 4 is the COMPRESSED_TX_HASH_LEN
value = cache_value[:25] db_key = b'h' + cache_key[:4] + cache_key[-2:]
prior_value = self.db_cache_get(key) prior_value = self.db_cache_get(db_key)
if prior_value: # Should rarely happen if prior_value: # Should rarely happen
hcolls += 1 collisions += 1
value += prior_value self.db_cache[db_key] = prior_value + cache_value[:25]
self.db_cache_write(key, value) else:
self.db_cache[db_key] = cache_value[:25]
# Next write the UTXO table # Next write the UTXO table
key = (b'u' + cache_value[:21] + cache_key[:UTXO_TX_HASH_LEN] db_key = b'u' + cache_value[:25] + cache_key[-2:]
+ cache_key[-2:]) self.db_cache[db_key] = cache_value[-8:]
value = cache_value[-12:]
prior_value = self.db_cache_get(key)
if prior_value: # Should almost never happen
ucolls += 1
value += prior_value
self.db_cache_write(key, value)
# GC-ing this now can only help the levelDB write. # GC-ing this now can only help the levelDB write.
self.utxo_cache = {} self.utxo_cache = {}
@ -1092,17 +1029,15 @@ class BlockProcessor(server.db.DB):
for key, value in self.db_cache.items(): for key, value in self.db_cache.items():
if value: if value:
batch.put(key, value) batch.put(key, value)
else: else: # b'' or None
batch.delete(key) batch.delete(key)
adds = new_utxos + self.utxo_cache_spends adds = new_utxos + self.utxo_cache_spends
self.logger.info('UTXO cache adds: {:,d} spends: {:,d} ' self.logger.info('UTXO cache adds: {:,d} spends: {:,d} '
.format(adds, self.utxo_cache_spends)) .format(adds, self.utxo_cache_spends))
self.logger.info('UTXO DB adds: {:,d} spends: {:,d}. ' self.logger.info('DB adds: {:,d} spends: {:,d}, collisions: {:,d}'
'Collisions: hash168: {:,d} UTXO: {:,d}' .format(new_utxos, self.db_deletes, collisions))
.format(new_utxos, self.db_deletes,
hcolls, ucolls))
self.db_cache = {} self.db_cache = {}
self.utxo_cache_spends = self.db_deletes = 0 self.utxo_cache_spends = self.db_deletes = 0

100
server/db.py

@ -11,14 +11,15 @@
import array import array
import ast import ast
import os import os
import struct from struct import pack, unpack
from bisect import bisect_right from bisect import bisect_right
from collections import namedtuple from collections import namedtuple
from lib.util import chunks, LoggedClass from lib.util import chunks, LoggedClass
from lib.hash import double_sha256 from lib.hash import double_sha256, hash_to_str
from server.storage import open_db from server.storage import open_db
UTXO = namedtuple("UTXO", "tx_num tx_pos tx_hash height value") UTXO = namedtuple("UTXO", "tx_num tx_pos tx_hash height value")
class DB(LoggedClass): class DB(LoggedClass):
@ -28,8 +29,11 @@ class DB(LoggedClass):
it was shutdown uncleanly. it was shutdown uncleanly.
''' '''
class MissingUTXOError(Exception):
'''Raised if a mempool tx input UTXO couldn't be found.'''
class DBError(Exception): class DBError(Exception):
pass '''Raised on general DB errors generally indicating corruption.'''
def __init__(self, env): def __init__(self, env):
super().__init__() super().__init__()
@ -168,19 +172,16 @@ class DB(LoggedClass):
Set limit to None to get them all. Set limit to None to get them all.
''' '''
limit = self._resolve_limit(limit) limit = self._resolve_limit(limit)
unpack = struct.unpack s_unpack = unpack
prefix = b'u' + hash168 prefix = b'u' + hash168
for k, v in self.db.iterator(prefix=prefix): for db_key, db_value in self.db.iterator(prefix=prefix):
(tx_pos,) = unpack('<H', k[-2:]) if limit == 0:
return
for n in range(0, len(v), 12): limit -= 1
if limit == 0: tx_num, tx_pos = s_unpack('<IH', db_key[-6:])
return value, = unpack('<Q', db_value)
(tx_num,) = unpack('<I', v[n:n + 4]) tx_hash, height = self.fs_tx_hash(tx_num)
(value,) = unpack('<Q', v[n + 4:n + 12]) yield UTXO(tx_num, tx_pos, tx_hash, height, value)
tx_hash, height = self.fs_tx_hash(tx_num)
yield UTXO(tx_num, tx_pos, tx_hash, height, value)
limit -= 1
def get_utxos_sorted(self, hash168): def get_utxos_sorted(self, hash168):
'''Returns all the UTXOs for an address sorted by height and '''Returns all the UTXOs for an address sorted by height and
@ -188,32 +189,53 @@ class DB(LoggedClass):
return sorted(self.get_utxos(hash168, limit=None)) return sorted(self.get_utxos(hash168, limit=None))
def get_utxo_hash168(self, tx_hash, index): def get_utxo_hash168(self, tx_hash, index):
'''Returns the hash168 for a UTXO.''' '''Returns the hash168 for a UTXO.
Used only for electrum client command-line requests.
'''
hash168 = None hash168 = None
if 0 <= index <= 65535: if 0 <= index <= 65535:
idx_packed = struct.pack('<H', index) idx_packed = pack('<H', index)
hash168 = self.db_hash168(tx_hash, idx_packed) hash168, tx_num_packed = self.db_hash168(tx_hash, idx_packed)
return hash168 return hash168
def db_hash168(self, tx_hash, idx_packed): def db_hash168(self, tx_hash, idx_packed):
'''Return the hash168 paid to by the given TXO. '''Return (hash168, tx_num_packed) for the given TXO.
Return None if not found.''' Both are None if not found.'''
key = b'h' + tx_hash[:ADDR_TX_HASH_LEN] + idx_packed # The 4 is the COMPRESSED_TX_HASH_LEN
data = self.db.get(key) key = b'h' + tx_hash[:4] + idx_packed
if data is None: db_value = self.db.get(key)
return None if db_value:
assert len(db_value) % 25 == 0
if len(data) == 25:
return data[:21] # Find which entry, if any, the TX_HASH matches.
for n in range(0, len(db_value), 25):
assert len(data) % 25 == 0 tx_num_packed = db_value[n + 21: n + 25]
tx_num, = unpack('<I', tx_num_packed)
# Resolve the compressed key collision using the TX number hash, height = self.fs_tx_hash(tx_num)
for n in range(0, len(data), 25): if hash == tx_hash:
tx_num, = struct.unpack('<I', data[n+21:n+25]) return db_value[n:n+21], tx_num_packed
my_hash, height = self.fs_tx_hash(tx_num)
if my_hash == tx_hash: return None, None
return data[n:n+21]
def db_utxo_lookup(self, tx_hash, tx_idx):
raise self.DBError('could not resolve hash168 collision') '''Given a prevout return a (hash168, value) pair.
Raises MissingUTXOError if the UTXO is not found. Used by the
mempool code.
'''
idx_packed = pack('<H', tx_idx)
hash168, tx_num_packed = self.db_hash168(tx_hash, idx_packed)
if not hash168:
# This can happen when the daemon is a block ahead of us
# and has mempool txs spending new txs in that block
raise self.MissingUTXOError
key = b'u' + hash168 + tx_num_packed + idx_packed
db_value = self.db.get(key)
if not db_value:
raise self.DBError('UTXO {} / {:,d} in one table only'
.format(hash_to_str(tx_hash), tx_idx))
value, = unpack('<Q', db_value)
return hash168, value

Loading…
Cancel
Save