Onionr/src/gossip/server/__init__.py

123 lines
4.4 KiB
Python
Raw Normal View History

import asyncio
2022-04-20 05:28:29 +00:00
from audioop import add
import traceback
2022-02-11 06:56:19 +00:00
from typing import TYPE_CHECKING
from typing import Set, Tuple
2022-02-11 06:56:19 +00:00
from queue import Queue
2022-03-02 13:29:59 +00:00
from gossip import constants
from ..connectpeer import connect_peer
2022-02-11 06:56:19 +00:00
2022-02-25 07:02:04 +00:00
from onionrplugins import onionrevents
import logger
2022-02-25 07:02:04 +00:00
2022-02-11 06:56:19 +00:00
if TYPE_CHECKING:
from onionrblocks import Block
from peer import Peer
from ordered_set import OrderedSet
2022-03-02 13:29:59 +00:00
from asyncio import StreamReader, StreamWriter
2022-02-11 06:56:19 +00:00
from filepaths import gossip_server_socket_file
2022-03-02 13:29:59 +00:00
from ..commands import GossipCommands
from ..peerset import gossip_peer_set
2022-03-02 13:29:59 +00:00
from .acceptstem import accept_stem_blocks
2022-04-04 05:48:30 +00:00
from .diffuseblocks import diffuse_blocks
"""
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.
2022-02-11 06:56:19 +00:00
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 <https://www.gnu.org/licenses/>.
"""
inbound_dandelion_edge_count = [0]
def gossip_server():
2022-03-02 13:29:59 +00:00
async def peer_connected(
reader: 'StreamReader', writer: 'StreamWriter'):
while True:
2022-02-22 20:34:19 +00:00
try:
2022-03-26 23:24:40 +00:00
cmd = await asyncio.wait_for(reader.readexactly(1), 60)
2022-02-22 20:34:19 +00:00
except asyncio.exceptions.CancelledError:
break
2022-03-26 23:24:40 +00:00
except asyncio.IncompleteReadError:
break
2022-02-22 20:34:19 +00:00
cmd = int.from_bytes(cmd, 'big')
if cmd == b'' or cmd == 0:
continue
match GossipCommands(cmd):
case GossipCommands.PING:
writer.write(b'PONG')
2022-02-25 07:02:04 +00:00
case GossipCommands.ANNOUNCE:
async def _read_announce():
2022-04-20 05:28:29 +00:00
address = await reader.readuntil(b'\n')
2022-03-26 23:24:40 +00:00
if address:
onionrevents.event(
'announce_rec',
data={'address': address,
2022-04-05 06:17:40 +00:00
'callback': connect_peer},
2022-03-26 23:24:40 +00:00
threaded=True)
writer.write(int(1).to_bytes(1, 'big'))
2022-02-25 07:02:04 +00:00
await asyncio.wait_for(_read_announce(), 10)
2022-03-02 13:29:59 +00:00
case GossipCommands.PEER_EXCHANGE:
2022-04-20 05:28:29 +00:00
for peer in gossip_peer_set:
2022-04-20 05:28:29 +00:00
writer.write(peer.transport_address.encode('utf-8') + b'\n')
await writer.drain()
2022-03-26 23:24:40 +00:00
case GossipCommands.STREAM_BLOCKS:
try:
2022-04-04 05:48:30 +00:00
await diffuse_blocks(reader, writer)
2022-03-26 23:24:40 +00:00
except Exception:
logger.warn(
f"Err streaming blocks\n{traceback.format_exc()}",
terminal=True)
2022-03-02 13:29:59 +00:00
case GossipCommands.PUT_BLOCKS:
2022-03-26 23:24:40 +00:00
# Pick block queue & append stemmed blocks to it
try:
await accept_stem_blocks(
reader, writer,
inbound_dandelion_edge_count)
2022-03-20 23:05:44 +00:00
except asyncio.exceptions.TimeoutError:
logger.debug(
"Inbound edge timed out when steming blocks to us",
terminal=True)
except asyncio.exceptions.IncompleteReadError:
logger.debug(
"Inbound edge timed out (Incomplete Read) when steming blocks to us",
terminal=True)
except Exception:
logger.warn(
f"Err acceptind stem blocks\n{traceback.format_exc()}",
terminal=True)
# Subtract dandelion edge, make sure >=0
inbound_dandelion_edge_count[0] = \
max(inbound_dandelion_edge_count[0] - 1, 0)
break
await writer.drain()
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())