from typing import TYPE_CHECKING from typing import List import secrets from asyncio import wait_for from onionrblocks import Block import logger from ..dandelion import StemAcceptResult from ..constants import BLOCK_ID_SIZE, BLOCK_MAX_SIZE from ..constants import MAX_INBOUND_DANDELION_EDGE, MAX_STEM_BLOCKS_PER_STREAM from ..blockqueues import gossip_block_queues block_size_digits = len(str(BLOCK_MAX_SIZE)) base_wait_timeout = 120 if TYPE_CHECKING: from asyncio import StreamWriter, StreamReader async def accept_stem_blocks( reader: 'StreamReader', writer: 'StreamWriter', inbound_edge_count: List[int]): if inbound_edge_count[0] >= MAX_INBOUND_DANDELION_EDGE: writer.write(StemAcceptResult.DENY) return writer.write(StemAcceptResult.ALLOW) inbound_edge_count[0] += 1 # Start getting the first block read_routine = reader.readexactly(BLOCK_ID_SIZE) block_queue_to_use = secrets.choice(gossip_block_queues) for _ in range(MAX_STEM_BLOCKS_PER_STREAM): block_id = ( await wait_for(read_routine, base_wait_timeout)).decode('utf-8') if not block_id: break block_size = (await wait_for( reader.readexactly(block_size_digits), base_wait_timeout)).decode('utf-8') if not block_size: break if not all(c in "0123456789" for c in block_size): raise ValueError("Invalid block size data (non 0-9 char)") block_size = int(block_size) if block_size > BLOCK_MAX_SIZE: raise ValueError("Max block size") raw_block: bytes = await wait_for( reader.readexactly(block_size), base_wait_timeout * 6) if not raw_block: break logger.debug("Got a stem block, put into queue", terminal=True) block_queue_to_use.put( Block(block_id, raw_block, auto_verify=True) ) # Regardless of stem phase, we add to queue # Client will decide if they are to be stemmed read_routine = reader.readexactly(BLOCK_ID_SIZE)