import asyncio import traceback from time import time from typing import TYPE_CHECKING from typing import Set, Tuple from threading import Thread from onionrblocks import Block from gossip import constants from ..connectpeer import connect_peer from onionrplugins import onionrevents from logger import log as logging if TYPE_CHECKING: from onionrblocks import Block from peer import Peer from ordered_set import OrderedSet from asyncio import StreamReader, StreamWriter from filepaths import gossip_server_socket_file import blockdb from blockdb import add_block_to_db from ..commands import GossipCommands from ..peerset import gossip_peer_set from .acceptstem import accept_stem_blocks from .diffuseblocks import diffuse_blocks from .import lastincoming """ This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . """ inbound_dandelion_edge_count = [0] def gossip_server(): async def peer_connected( reader: 'StreamReader', writer: 'StreamWriter'): while True: try: cmd = await asyncio.wait_for(reader.readexactly(1), 60) except asyncio.exceptions.CancelledError: break except asyncio.IncompleteReadError: break lastincoming.last_incoming_timestamp = int(time()) cmd = int.from_bytes(cmd, 'big') if cmd == b'' or cmd == 0: continue match GossipCommands(cmd): case GossipCommands.PING: writer.write(b'PONG') case GossipCommands.ANNOUNCE: async def _read_announce(): address = await reader.readuntil(b'\n') if address: onionrevents.event( 'announce_rec', data={'address': address, 'callback': connect_peer}, threaded=True) writer.write(int(1).to_bytes(1, 'big')) await writer.drain() await asyncio.wait_for(_read_announce(), 30) case GossipCommands.PEER_EXCHANGE: for peer in gossip_peer_set: writer.write(peer.transport_address.encode('utf-8') + b'\n') await writer.drain() case GossipCommands.STREAM_BLOCKS: try: await diffuse_blocks(reader, writer) except Exception: logging.debug( f"Err streaming blocks\n{traceback.format_exc()}", ) case GossipCommands.PUT_BLOCKS: # Pick block queue & append stemmed blocks to it try: await accept_stem_blocks( reader, writer, inbound_dandelion_edge_count) except asyncio.exceptions.TimeoutError: pass logging.debug( "Inbound edge timed out when steming blocks to us", ) except asyncio.exceptions.IncompleteReadError: pass logging.debug( "Inbound edge timed out (Incomplete Read) when steming blocks to us", ) except Exception: logging.warn( f"Err accepting stem blocks\n{traceback.format_exc()}", ) # Subtract dandelion edge, make sure >=0 inbound_dandelion_edge_count[0] = \ max(inbound_dandelion_edge_count[0] - 1, 0) case GossipCommands.PUT_BLOCK_DIFFUSE: async def _get_block_diffused(): block_id = await reader.readexactly(constants.BLOCK_ID_SIZE) if blockdb.has_block(block_id): writer.write(int(0).to_bytes(1, 'big')) else: writer.write(int(1).to_bytes(1, 'big')) await writer.drain() block_size = int(await asyncio.wait_for(reader.readexactly(constants.BLOCK_SIZE_LEN), 30)) block_data = await reader.readexactly(block_size) Thread( target=add_block_to_db, args=[ Block(block_id, block_data, auto_verify=True)] ).start() await _get_block_diffused() break try: await writer.drain() except BrokenPipeError: pass writer.close() async def main(): server = await asyncio.start_unix_server( peer_connected, gossip_server_socket_file ) async with server: await server.serve_forever() asyncio.run(main())