from math import log10 from binascii import hexlify from sqlalchemy import text, between from sqlalchemy.future import select from ..query_context import context from ..tables import ( SCHEMA_VERSION, metadata, Version, Claim, Support, Block, BlockFilter, BlockGroupFilter, TX, TXFilter, pg_add_account_address_constraints_and_indexes ) def execute(sql): return context().execute(text(sql)) def execute_sql_object(sql): return context().execute(sql) def execute_fetchall(sql): return context().fetchall(text(sql)) def has_filters(): return context().has_records(BlockFilter) def has_claims(): return context().has_records(Claim) def has_supports(): return context().has_records(Support) def get_best_block_height(): return context().fetchmax(Block.c.height, -1) def insert_block(block): context().get_bulk_loader().add_block(block).flush() def get_block_headers(first, last=None): if last is not None: query = ( select('*').select_from(Block) .where(between(Block.c.height, first, last)) .order_by(Block.c.height) ) else: query = select('*').select_from(Block).where(Block.c.height == first) return context().fetchall(query) def get_filters(start_height, end_height=None, granularity=0): assert granularity >= 0, "filter granularity must be 0 or positive number" if granularity == 0: query = ( select('*').select_from(TXFilter) .where(between(TXFilter.c.height, start_height, end_height)) .order_by(TXFilter.c.height) ) elif granularity == 1: query = ( select('*').select_from(BlockFilter) .where(between(BlockFilter.c.height, start_height, end_height)) .order_by(BlockFilter.c.height) ) else: query = ( select('*').select_from(BlockGroupFilter) .where( (BlockGroupFilter.c.height == start_height) & (BlockGroupFilter.c.factor == log10(granularity)) ) .order_by(BlockGroupFilter.c.height) ) result = [] for row in context().fetchall(query): record = { "height": row["height"], "filter": hexlify(row["address_filter"]).decode(), } if granularity == 0: record["txid"] = hexlify(row["tx_hash"][::-1]).decode() result.append(record) return result def insert_transaction(block_hash, tx): context().get_bulk_loader().add_transaction(block_hash, tx).flush(TX) def check_version_and_create_tables(): with context("db.connecting") as ctx: if ctx.has_table('version'): version = ctx.fetchone(select(Version.c.version).limit(1)) if version and version['version'] == SCHEMA_VERSION: return metadata.drop_all(ctx.engine) metadata.create_all(ctx.engine) ctx.execute(Version.insert().values(version=SCHEMA_VERSION)) for table in metadata.sorted_tables: disable_trigger_and_constraints(table.name) if ctx.is_postgres: for statement in pg_add_account_address_constraints_and_indexes: ctx.execute(text(statement)) def disable_trigger_and_constraints(table_name): ctx = context() if ctx.is_postgres: ctx.execute(text(f"ALTER TABLE {table_name} DISABLE TRIGGER ALL;")) if table_name in ('tag', 'stake', 'block_group_filter', 'mempool_filter'): return if ctx.is_postgres: ctx.execute(text( f"ALTER TABLE {table_name} DROP CONSTRAINT {table_name}_pkey CASCADE;" ))