Onionr/src/gossip/server/acceptstem.py

71 lines
2.1 KiB
Python

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)