Misc bug fixes for gossip
This commit is contained in:
parent
5b5e5ef764
commit
9c44069248
@ -6,6 +6,7 @@ import traceback
|
|||||||
from typing import TYPE_CHECKING
|
from typing import TYPE_CHECKING
|
||||||
from typing import Set, Tuple
|
from typing import Set, Tuple
|
||||||
from time import sleep
|
from time import sleep
|
||||||
|
import asyncio
|
||||||
|
|
||||||
from queue import Queue
|
from queue import Queue
|
||||||
|
|
||||||
@ -79,13 +80,17 @@ def gossip_client(
|
|||||||
if dandelion_phase.remaining_time() <= 10:
|
if dandelion_phase.remaining_time() <= 10:
|
||||||
sleep(dandelion_phase.remaining_time())
|
sleep(dandelion_phase.remaining_time())
|
||||||
if dandelion_phase.is_stem_phase():
|
if dandelion_phase.is_stem_phase():
|
||||||
|
logger.debug("Entering stem phase", terminal=True)
|
||||||
try:
|
try:
|
||||||
# Stem out blocks for (roughly) remaining epoch time
|
# Stem out blocks for (roughly) remaining epoch time
|
||||||
await stem_out(
|
asyncio.run(stem_out(
|
||||||
block_queues, peer_set, dandelion_phase)
|
block_queues, peer_set, dandelion_phase))
|
||||||
except TimeoutError:
|
except TimeoutError:
|
||||||
continue
|
continue
|
||||||
|
except Exception:
|
||||||
|
logger.error(traceback.format_exc(), terminal=True)
|
||||||
continue
|
continue
|
||||||
else:
|
else:
|
||||||
|
logger.debug("Entering fluff phase", terminal=True)
|
||||||
# Add block to primary block db, where the diffuser can read it
|
# Add block to primary block db, where the diffuser can read it
|
||||||
store_blocks(block_queues, dandelion_phase)
|
store_blocks(block_queues, dandelion_phase)
|
||||||
|
@ -5,6 +5,8 @@ import traceback
|
|||||||
|
|
||||||
from typing import TYPE_CHECKING, Coroutine, Tuple, List
|
from typing import TYPE_CHECKING, Coroutine, Tuple, List
|
||||||
|
|
||||||
|
from ordered_set import OrderedSet
|
||||||
|
|
||||||
from onionrthreads import add_delayed_thread
|
from onionrthreads import add_delayed_thread
|
||||||
from blockdb import add_block_to_db
|
from blockdb import add_block_to_db
|
||||||
import logger
|
import logger
|
||||||
@ -16,7 +18,7 @@ from ... import dandelion
|
|||||||
from .stemstream import do_stem_stream
|
from .stemstream import do_stem_stream
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
from ordered_set import OrderedSet
|
|
||||||
from onionrblocks import Block
|
from onionrblocks import Block
|
||||||
from ...peer import Peer
|
from ...peer import Peer
|
||||||
from ...dandelion.phase import DandelionPhase
|
from ...dandelion.phase import DandelionPhase
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
from typing import TYPE_CHECKING
|
from typing import TYPE_CHECKING
|
||||||
|
|
||||||
|
from ...constants import BLOCK_MAX_SIZE
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
from queue import Queue
|
from queue import Queue
|
||||||
import socket
|
import socket
|
||||||
@ -20,6 +22,9 @@ async def do_stem_stream(
|
|||||||
# Primary client component that communicate's with gossip.server.acceptstem
|
# Primary client component that communicate's with gossip.server.acceptstem
|
||||||
remaining_time = d_phase.remaining_time()
|
remaining_time = d_phase.remaining_time()
|
||||||
bl: 'Block' = block_queue.get(block=True, timeout=remaining_time)
|
bl: 'Block' = block_queue.get(block=True, timeout=remaining_time)
|
||||||
|
|
||||||
|
block_size = str(len(bl.raw)).zfill(BLOCK_MAX_SIZE)
|
||||||
|
|
||||||
peer_socket.sendall(bl.id)
|
peer_socket.sendall(bl.id)
|
||||||
peer_socket.sendall(len(bl.raw))
|
peer_socket.sendall(block_size.encode('utf-8'))
|
||||||
peer_socket.sendall(bl.raw)
|
peer_socket.sendall(bl.raw)
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
from typing import TYPE_CHECKING, Tuple
|
from typing import TYPE_CHECKING, Tuple
|
||||||
from threading import Thread
|
from threading import Thread
|
||||||
from queue import Queue
|
from queue import Queue
|
||||||
|
from queue import Empty
|
||||||
|
|
||||||
import blockdb
|
import blockdb
|
||||||
|
|
||||||
@ -22,11 +23,11 @@ def store_blocks(
|
|||||||
try:
|
try:
|
||||||
new_queue.put(
|
new_queue.put(
|
||||||
block_queue.get(timeout=dandelion_phase.remaining_time()))
|
block_queue.get(timeout=dandelion_phase.remaining_time()))
|
||||||
except TimeoutError:
|
except Empty:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
for block_queue in block_queues:
|
for block_queue in block_queues:
|
||||||
Thread(target=_watch_queue, args=block_queue, daemon=True).start()
|
Thread(target=_watch_queue, args=[block_queue], daemon=True).start()
|
||||||
|
|
||||||
while not dandelion_phase.is_stem_phase() \
|
while not dandelion_phase.is_stem_phase() \
|
||||||
and dandelion_phase.remaining_time() > 1:
|
and dandelion_phase.remaining_time() > 1:
|
||||||
@ -34,6 +35,6 @@ def store_blocks(
|
|||||||
blockdb.add_block_to_db(
|
blockdb.add_block_to_db(
|
||||||
new_queue.get(timeout=dandelion_phase.remaining_time())
|
new_queue.get(timeout=dandelion_phase.remaining_time())
|
||||||
)
|
)
|
||||||
except TimeoutError:
|
except Empty:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
from time import time
|
from time import time
|
||||||
from hashlib import shake_128
|
from hashlib import shake_128
|
||||||
|
from secrets import randbits
|
||||||
|
|
||||||
|
|
||||||
class DandelionPhase:
|
class DandelionPhase:
|
||||||
@ -8,7 +9,7 @@ class DandelionPhase:
|
|||||||
assert len(self.seed) == 32
|
assert len(self.seed) == 32
|
||||||
self.epoch = int(time())
|
self.epoch = int(time())
|
||||||
self.epoch_interval = epoch_interval_secs
|
self.epoch_interval = epoch_interval_secs
|
||||||
self._is_stem = True
|
self._is_stem = bool(randbits(1))
|
||||||
self.phase_id = b''
|
self.phase_id = b''
|
||||||
|
|
||||||
|
|
||||||
@ -21,7 +22,7 @@ class DandelionPhase:
|
|||||||
int.to_bytes(cur_time, 8, 'big')).digest(8)
|
int.to_bytes(cur_time, 8, 'big')).digest(8)
|
||||||
|
|
||||||
# Use first byte of phase id as random source for stem phase picking
|
# Use first byte of phase id as random source for stem phase picking
|
||||||
if int.from_bytes(self.phase_id[0], 'big') % 2:
|
if self.phase_id[0] % 2:
|
||||||
self._is_stem = True
|
self._is_stem = True
|
||||||
else:
|
else:
|
||||||
self._is_stem = False
|
self._is_stem = False
|
||||||
@ -29,7 +30,8 @@ class DandelionPhase:
|
|||||||
|
|
||||||
def remaining_time(self) -> int:
|
def remaining_time(self) -> int:
|
||||||
current_time = int(time())
|
current_time = int(time())
|
||||||
return self.epoch_interval - (current_time - self.epoch)
|
|
||||||
|
return max(0, self.epoch_interval - (current_time - self.epoch))
|
||||||
|
|
||||||
|
|
||||||
def is_stem_phase(self) -> bool:
|
def is_stem_phase(self) -> bool:
|
||||||
|
@ -85,6 +85,10 @@ def gossip_server(
|
|||||||
block_queues,
|
block_queues,
|
||||||
reader, writer,
|
reader, writer,
|
||||||
inbound_dandelion_edge_count)
|
inbound_dandelion_edge_count)
|
||||||
|
except asyncio.exceptions.TimeoutError:
|
||||||
|
logger.debug(
|
||||||
|
"Inbound edge timed out when steming blocks to us",
|
||||||
|
terminal=True)
|
||||||
except Exception:
|
except Exception:
|
||||||
logger.warn(
|
logger.warn(
|
||||||
f"Err getting\n{traceback.format_exc()}",
|
f"Err getting\n{traceback.format_exc()}",
|
||||||
|
@ -12,7 +12,7 @@ from ..constants import MAX_INBOUND_DANDELION_EDGE, MAX_STEM_BLOCKS_PER_STREAM
|
|||||||
|
|
||||||
|
|
||||||
block_size_digits = len(str(BLOCK_MAX_SIZE))
|
block_size_digits = len(str(BLOCK_MAX_SIZE))
|
||||||
base_wait_timeout = 10
|
base_wait_timeout = 30
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
from queue import Queue
|
from queue import Queue
|
||||||
@ -44,6 +44,9 @@ async def accept_stem_blocks(
|
|||||||
reader.read(block_size_digits),
|
reader.read(block_size_digits),
|
||||||
base_wait_timeout)).decode('utf-8')
|
base_wait_timeout)).decode('utf-8')
|
||||||
|
|
||||||
|
if not block_size:
|
||||||
|
break
|
||||||
|
|
||||||
if not all(c in "0123456789" for c in block_size):
|
if not all(c in "0123456789" for c in block_size):
|
||||||
raise ValueError("Invalid block size data (non 0-9 char)")
|
raise ValueError("Invalid block size data (non 0-9 char)")
|
||||||
block_size = int(block_size)
|
block_size = int(block_size)
|
||||||
@ -53,6 +56,9 @@ async def accept_stem_blocks(
|
|||||||
raw_block: bytes = await wait_for(
|
raw_block: bytes = await wait_for(
|
||||||
reader.read(block_size), base_wait_timeout * 6)
|
reader.read(block_size), base_wait_timeout * 6)
|
||||||
|
|
||||||
|
if not raw_block:
|
||||||
|
break
|
||||||
|
|
||||||
block_queue_to_use.put(
|
block_queue_to_use.put(
|
||||||
Block(block_id, raw_block, auto_verify=True)
|
Block(block_id, raw_block, auto_verify=True)
|
||||||
)
|
)
|
||||||
|
Loading…
Reference in New Issue
Block a user