2020-04-11 20:15:04 -04:00
|
|
|
import os.path
|
2020-05-01 09:28:51 -04:00
|
|
|
import asyncio
|
2020-04-11 20:15:04 -04:00
|
|
|
import sqlite3
|
|
|
|
from typing import Optional
|
2020-05-01 09:28:51 -04:00
|
|
|
from concurrent.futures import ThreadPoolExecutor
|
2020-04-11 20:15:04 -04:00
|
|
|
|
|
|
|
|
2020-05-01 09:28:51 -04:00
|
|
|
FILES = [
|
|
|
|
'block_index',
|
|
|
|
'claims'
|
|
|
|
]
|
|
|
|
|
2020-04-11 20:15:04 -04:00
|
|
|
|
2020-05-01 09:28:51 -04:00
|
|
|
class BlockchainDB:
|
2020-04-11 20:15:04 -04:00
|
|
|
|
|
|
|
def __init__(self, directory: str):
|
2020-05-01 09:28:51 -04:00
|
|
|
self.directory = directory
|
|
|
|
self.connection: Optional[sqlite3.Connection] = None
|
|
|
|
self.executor: Optional[ThreadPoolExecutor] = None
|
|
|
|
|
|
|
|
async def run_in_executor(self, *args):
|
|
|
|
return await asyncio.get_event_loop().run_in_executor(self.executor, *args)
|
|
|
|
|
|
|
|
def sync_open(self):
|
|
|
|
self.connection = sqlite3.connect(
|
|
|
|
os.path.join(self.directory, FILES[0]+'.sqlite'),
|
|
|
|
timeout=60.0 * 5
|
|
|
|
)
|
|
|
|
for file in FILES[1:]:
|
|
|
|
self.connection.execute(
|
|
|
|
f"ATTACH DATABASE '{os.path.join(self.directory, file+'.sqlite')}' AS {file}"
|
|
|
|
)
|
|
|
|
self.connection.row_factory = sqlite3.Row
|
2020-04-11 20:15:04 -04:00
|
|
|
|
2020-05-01 09:28:51 -04:00
|
|
|
async def open(self):
|
|
|
|
assert self.executor is None, "Database is already open."
|
|
|
|
self.executor = ThreadPoolExecutor(max_workers=1)
|
|
|
|
return await self.run_in_executor(self.sync_open)
|
2020-04-11 20:15:04 -04:00
|
|
|
|
2020-05-01 09:28:51 -04:00
|
|
|
def sync_close(self):
|
|
|
|
self.connection.close()
|
|
|
|
self.connection = None
|
2020-04-11 20:15:04 -04:00
|
|
|
|
2020-05-01 09:28:51 -04:00
|
|
|
async def close(self):
|
|
|
|
if self.executor is not None:
|
|
|
|
if self.connection is not None:
|
|
|
|
await self.run_in_executor(self.sync_close)
|
|
|
|
self.executor.shutdown()
|
|
|
|
self.executor = None
|
|
|
|
|
|
|
|
def sync_execute(self, sql: str, *args):
|
|
|
|
return self.connection.execute(sql, *args)
|
|
|
|
|
|
|
|
async def execute(self, sql, *args):
|
|
|
|
return await self.run_in_executor(self.sync_execute, sql, *args)
|
|
|
|
|
|
|
|
def sync_execute_fetchall(self, sql: str, *args):
|
|
|
|
return list(self.connection.execute(sql, *args).fetchall())
|
|
|
|
|
|
|
|
async def execute_fetchall(self, sql: str, *args):
|
|
|
|
return await self.run_in_executor(self.sync_execute_fetchall, sql, *args)
|
|
|
|
|
2020-06-05 00:35:22 -04:00
|
|
|
def sync_get_block_files(self, file_number=None, above_height=None):
|
|
|
|
sql = """
|
2020-05-20 17:54:38 -04:00
|
|
|
SELECT
|
|
|
|
file as file_number,
|
|
|
|
COUNT(hash) as blocks,
|
|
|
|
SUM(txcount) as txs,
|
|
|
|
MAX(height) as max_height
|
2020-06-05 00:35:22 -04:00
|
|
|
FROM block_info
|
|
|
|
WHERE status&1 AND status&4
|
|
|
|
"""
|
|
|
|
args = ()
|
|
|
|
if file_number is not None and above_height is not None:
|
|
|
|
sql += "AND file = ? AND height > ?"
|
|
|
|
args = (file_number, above_height)
|
|
|
|
return self.sync_execute_fetchall(sql + " GROUP BY file ORDER BY file ASC;", args)
|
|
|
|
|
|
|
|
async def get_block_files(self, file_number=None, above_height=None):
|
|
|
|
return await self.run_in_executor(
|
|
|
|
self.sync_get_block_files, file_number, above_height
|
2020-04-11 20:15:04 -04:00
|
|
|
)
|
|
|
|
|
2020-05-20 17:54:38 -04:00
|
|
|
def sync_get_blocks_in_file(self, block_file, above_height=-1):
|
2020-05-01 09:28:51 -04:00
|
|
|
return self.sync_execute_fetchall(
|
2020-04-11 20:15:04 -04:00
|
|
|
"""
|
|
|
|
SELECT datapos as data_offset, height, hash as block_hash, txCount as txs
|
2020-05-18 08:22:23 -04:00
|
|
|
FROM block_info
|
2020-06-05 00:35:22 -04:00
|
|
|
WHERE file = ? AND height > ? AND status&1 AND status&4
|
2020-05-18 08:22:23 -04:00
|
|
|
ORDER BY datapos ASC;
|
2020-05-20 17:54:38 -04:00
|
|
|
""", (block_file, above_height)
|
2020-04-11 20:15:04 -04:00
|
|
|
)
|
2020-05-01 09:28:51 -04:00
|
|
|
|
2020-05-20 17:54:38 -04:00
|
|
|
async def get_blocks_in_file(self, block_file, above_height=-1):
|
|
|
|
return await self.run_in_executor(self.sync_get_blocks_in_file, block_file, above_height)
|
2020-05-01 09:28:51 -04:00
|
|
|
|
|
|
|
def sync_get_claimtrie(self):
|
|
|
|
return self.sync_execute_fetchall(
|
|
|
|
"""
|
|
|
|
SELECT
|
|
|
|
takeover.name AS normalized,
|
|
|
|
takeover.claimID AS claim_hash,
|
|
|
|
takeover.height AS last_take_over_height,
|
|
|
|
originalHeight AS original_height,
|
|
|
|
updateHeight AS update_height,
|
|
|
|
validHeight AS valid_height,
|
|
|
|
activationHeight AS activation_height,
|
|
|
|
expirationHeight AS expiration_height
|
|
|
|
FROM takeover JOIN claim USING (claimID)
|
|
|
|
GROUP BY takeover.name HAVING MAX(height);
|
|
|
|
"""
|
|
|
|
)
|
|
|
|
|
|
|
|
async def get_claimtrie(self):
|
|
|
|
return await self.run_in_executor(self.sync_get_claimtrie)
|
|
|
|
|
|
|
|
def sync_get_claims(self):
|
|
|
|
return self.sync_execute_fetchall(
|
|
|
|
"""
|
|
|
|
SELECT
|
|
|
|
claimID AS claim_hash,
|
|
|
|
txID AS tx_hash,
|
|
|
|
txN AS position,
|
|
|
|
amount,
|
|
|
|
originalHeight AS original_height,
|
|
|
|
updateHeight AS update_height,
|
|
|
|
validHeight AS valid_height,
|
|
|
|
activationHeight AS activation_height,
|
|
|
|
expirationHeight AS expiration_height
|
|
|
|
FROM claims.claim
|
|
|
|
"""
|
|
|
|
)
|
|
|
|
|
|
|
|
async def get_claims(self):
|
|
|
|
return await self.run_in_executor(self.sync_get_claims)
|