-
Notifications
You must be signed in to change notification settings - Fork 23
/
Copy pathmain.py
119 lines (91 loc) · 3.59 KB
/
main.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
import asyncio
import logging
import os
import threading
import time
from BlocksProcessor import BlocksProcessor
from TxAddrMappingUpdater import TxAddrMappingUpdater
from VirtualChainProcessor import VirtualChainProcessor
from dbsession import create_all
from helper import KeyValueStore
from kaspad.KaspadMultiClient import KaspadMultiClient
logging.basicConfig(format="%(asctime)s::%(levelname)s::%(name)s::%(message)s",
level=logging.DEBUG if os.getenv("DEBUG", False) else logging.INFO,
handlers=[
logging.StreamHandler()
]
)
# disable sqlalchemy notifications
logging.getLogger('sqlalchemy').setLevel(logging.ERROR)
# get file logger
_logger = logging.getLogger(__name__)
# create tables in database
_logger.info('Creating DBs if not exist.')
create_all(drop=False)
kaspad_hosts = []
for i in range(100):
try:
kaspad_hosts.append(os.environ[f"KASPAD_HOST_{i + 1}"].strip())
except KeyError:
break
if not kaspad_hosts:
raise Exception('Please set at least KASPAD_HOST_1 environment variable.')
# create Kaspad client
client = KaspadMultiClient(kaspad_hosts)
task_runner = None
async def main():
# initialize kaspads
await client.initialize_all()
# wait for client to be synced
while client.kaspads[0].is_synced == False:
_logger.info('Client not synced yet. Waiting...')
time.sleep(60)
# find last acceptedTx's block hash, when restarting this tool
start_hash = KeyValueStore.get("vspc_last_start_hash")
# if there is nothing in the db, just get latest block.
if not start_hash:
daginfo = await client.request("getBlockDagInfoRequest", {})
start_hash = daginfo["getBlockDagInfoResponse"]["virtualParentHashes"][0]
_logger.info(f"Start hash: {start_hash}")
# create instances of blocksprocessor and virtualchainprocessor
bp = BlocksProcessor(client)
vcp = VirtualChainProcessor(client, start_hash)
async def handle_blocks_commited(e):
"""
this function is executed, when a new cluster of blocks were added to the database
"""
global task_runner
if task_runner and not task_runner.done():
return
_logger.debug('Update is_accepted for TXs.')
task_runner = asyncio.create_task(vcp.yield_to_database())
# set up event to fire after adding new blocks
bp.on_commited += handle_blocks_commited
# start blocks processor working concurrent
while True:
try:
await bp.loop(start_hash)
except Exception:
_logger.exception('Exception occured and script crashed..')
raise
if __name__ == '__main__':
tx_addr_mapping_updater = TxAddrMappingUpdater()
# custom exception hook for thread
def custom_hook(args):
global tx_addr_mapping_updater
# report the failure
_logger.error(f'Thread failed: {args.exc_value}')
thread = args[3]
# check if TxAddrMappingUpdater
if thread.name == 'TxAddrMappingUpdater':
p = threading.Thread(target=tx_addr_mapping_updater.loop, daemon=True, name="TxAddrMappingUpdater")
p.start()
raise Exception("TxAddrMappingUpdater thread crashed.")
# set the exception hook
threading.excepthook = custom_hook
# run TxAddrMappingUpdater
# will be rerun
_logger.info('Starting updater thread now.')
threading.Thread(target=tx_addr_mapping_updater.loop, daemon=True, name="TxAddrMappingUpdater").start()
_logger.info('Starting main thread now.')
asyncio.run(main())