90 lines
3.4 KiB
Python
90 lines
3.4 KiB
Python
# standard imports
|
||
import logging
|
||
|
||
# third-party imports
|
||
import moolb
|
||
|
||
# local imports
|
||
from cic_cache.db import list_transactions_mined
|
||
from cic_cache.db import list_transactions_account_mined
|
||
|
||
logg = logging.getLogger()
|
||
|
||
|
||
class BloomCache:
|
||
|
||
|
||
def __init__(self, session):
|
||
self.session = session
|
||
|
||
@staticmethod
|
||
def __get_filter_size(n):
|
||
n = 8192 * 8
|
||
logg.warning('filter size hardcoded to {}'.format(n))
|
||
return n
|
||
|
||
|
||
def load_transactions(self, offset, limit):
|
||
"""Retrieves a list of transactions from cache and creates a bloom filter pointing to blocks and transactions.
|
||
|
||
Block and transaction numbers are serialized as 32-bit big-endian numbers. The input to the second bloom filter is the concatenation of the serialized block number and transaction index.
|
||
|
||
For example, if the block number is 13 and the transaction index is 42, the input are:
|
||
|
||
block filter: 0x0d000000
|
||
block+tx filter: 0x0d0000002a0000000
|
||
|
||
:param offset: Offset in data set to return transactions from
|
||
:type offset: int
|
||
:param limit: Max number of transactions to retrieve
|
||
:type limit: int
|
||
:return: Lowest block, bloom filter for blocks, bloom filter for blocks|tx
|
||
:rtype: tuple
|
||
"""
|
||
rows = list_transactions_mined(self.session, offset, limit)
|
||
|
||
f_block = moolb.Bloom(BloomCache.__get_filter_size(limit), 3)
|
||
f_blocktx = moolb.Bloom(BloomCache.__get_filter_size(limit), 3)
|
||
highest_block = -1
|
||
lowest_block = -1
|
||
for r in rows:
|
||
if highest_block == -1:
|
||
highest_block = r[0]
|
||
lowest_block = r[0]
|
||
block = r[0].to_bytes(4, byteorder='big')
|
||
tx = r[1].to_bytes(4, byteorder='big')
|
||
f_block.add(block)
|
||
f_blocktx.add(block + tx)
|
||
logg.debug('added block {} tx {} lo {} hi {}'.format(r[0], r[1], lowest_block, highest_block))
|
||
return (lowest_block, highest_block, f_block.to_bytes(), f_blocktx.to_bytes(),)
|
||
|
||
|
||
def load_transactions_account(self, address, offset, limit):
|
||
"""Same as load_transactions(...), but only retrieves transactions where the specified account address is sender or recipient.
|
||
|
||
:param address: Address to retrieve transactions for.
|
||
:type address: str, 0x-hex
|
||
:param offset: Offset in data set to return transactions from
|
||
:type offset: int
|
||
:param limit: Max number of transactions to retrieve
|
||
:type limit: int
|
||
:return: Lowest block, bloom filter for blocks, bloom filter for blocks|tx
|
||
:rtype: tuple
|
||
"""
|
||
rows = list_transactions_account_mined(self.session, address, offset, limit)
|
||
|
||
f_block = moolb.Bloom(BloomCache.__get_filter_size(limit), 3)
|
||
f_blocktx = moolb.Bloom(BloomCache.__get_filter_size(limit), 3)
|
||
highest_block = -1;
|
||
lowest_block = -1;
|
||
for r in rows:
|
||
if highest_block == -1:
|
||
highest_block = r[0]
|
||
lowest_block = r[0]
|
||
block = r[0].to_bytes(4, byteorder='big')
|
||
tx = r[1].to_bytes(4, byteorder='big')
|
||
f_block.add(block)
|
||
f_blocktx.add(block + tx)
|
||
logg.debug('added block {} tx {} lo {} hi {}'.format(r[0], r[1], lowest_block, highest_block))
|
||
return (lowest_block, highest_block, f_block.to_bytes(), f_blocktx.to_bytes(),)
|