mirror of
https://github.com/0glabs/0g-storage-node.git
synced 2025-01-13 16:45:18 +00:00
294 lines
10 KiB
Rust
294 lines
10 KiB
Rust
// Copyright 2019 Parity Technologies (UK) Ltd.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a
|
|
// copy of this software and associated documentation files (the "Software"),
|
|
// to deal in the Software without restriction, including without limitation
|
|
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
// and/or sell copies of the Software, and to permit persons to whom the
|
|
// Software is furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in
|
|
// all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
// DEALINGS IN THE SOFTWARE.
|
|
|
|
// This basis of this file has been taken from the rust-libp2p codebase:
|
|
// https://github.com/libp2p/rust-libp2p
|
|
|
|
mod peers;
|
|
|
|
pub use peers::QueryState;
|
|
pub(crate) use peers::{
|
|
closest::{FindNodeQuery, FindNodeQueryConfig},
|
|
predicate::{PredicateQuery, PredicateQueryConfig},
|
|
};
|
|
|
|
use crate::kbucket::{Key, PredicateKey};
|
|
use fnv::FnvHashMap;
|
|
use std::time::{Duration, Instant};
|
|
|
|
pub trait TargetKey<TNodeId> {
|
|
fn key(&self) -> Key<TNodeId>;
|
|
}
|
|
|
|
/// A `QueryPool` provides an aggregate state machine for driving `Query`s to completion.
|
|
///
|
|
/// Internally, a `Query` is in turn driven by an underlying `QueryPeerIter`
|
|
/// that determines the peer selection strategy, i.e. the order in which the
|
|
/// peers involved in the query should be contacted.
|
|
pub struct QueryPool<TTarget, TNodeId, TResult> {
|
|
next_id: usize,
|
|
query_timeout: Duration,
|
|
queries: FnvHashMap<QueryId, Query<TTarget, TNodeId, TResult>>,
|
|
}
|
|
|
|
/// The observable states emitted by [`QueryPool::poll`].
|
|
#[allow(clippy::type_complexity)]
|
|
pub enum QueryPoolState<'a, TTarget, TNodeId, TResult> {
|
|
/// The pool is idle, i.e. there are no queries to process.
|
|
Idle,
|
|
/// At least one query is waiting for results. `Some(request)` indicates
|
|
/// that a new request is now being waited on.
|
|
Waiting(Option<(&'a mut Query<TTarget, TNodeId, TResult>, TNodeId)>),
|
|
/// A query has finished.
|
|
Finished(Query<TTarget, TNodeId, TResult>),
|
|
/// A query has timed out.
|
|
Timeout(Query<TTarget, TNodeId, TResult>),
|
|
}
|
|
|
|
impl<TTarget, TNodeId, TResult> QueryPool<TTarget, TNodeId, TResult>
|
|
where
|
|
TTarget: TargetKey<TNodeId>,
|
|
TNodeId: Into<Key<TNodeId>> + Eq + Clone,
|
|
TResult: Into<TNodeId> + Clone,
|
|
{
|
|
/// Creates a new `QueryPool` with the given configuration.
|
|
pub fn new(query_timeout: Duration) -> Self {
|
|
QueryPool {
|
|
next_id: 0,
|
|
query_timeout,
|
|
queries: Default::default(),
|
|
}
|
|
}
|
|
|
|
/// Returns an iterator over the queries in the pool.
|
|
pub fn iter(&self) -> impl Iterator<Item = &Query<TTarget, TNodeId, TResult>> {
|
|
self.queries.values()
|
|
}
|
|
|
|
/// Adds a query to the pool that iterates towards the closest peers to the target.
|
|
pub fn add_findnode_query<I>(
|
|
&mut self,
|
|
config: FindNodeQueryConfig,
|
|
target: TTarget,
|
|
peers: I,
|
|
) -> QueryId
|
|
where
|
|
I: IntoIterator<Item = Key<TNodeId>>,
|
|
{
|
|
let target_key = target.key();
|
|
let findnode_query = FindNodeQuery::with_config(config, target_key, peers);
|
|
let peer_iter = QueryPeerIter::FindNode(findnode_query);
|
|
self.add(peer_iter, target)
|
|
}
|
|
|
|
/// Adds a query to the pool that returns peers that satisfy a predicate.
|
|
pub(crate) fn add_predicate_query<I>(
|
|
&mut self,
|
|
config: PredicateQueryConfig,
|
|
target: TTarget,
|
|
peers: I,
|
|
predicate: impl Fn(&TResult) -> bool + Send + 'static,
|
|
) -> QueryId
|
|
where
|
|
I: IntoIterator<Item = PredicateKey<TNodeId>>,
|
|
{
|
|
let target_key = target.key();
|
|
let predicate_query = PredicateQuery::with_config(config, target_key, peers, predicate);
|
|
let peer_iter = QueryPeerIter::Predicate(predicate_query);
|
|
self.add(peer_iter, target)
|
|
}
|
|
|
|
fn add(&mut self, peer_iter: QueryPeerIter<TNodeId, TResult>, target: TTarget) -> QueryId {
|
|
let id = QueryId(self.next_id);
|
|
self.next_id = self.next_id.wrapping_add(1);
|
|
let query = Query::new(id, peer_iter, target);
|
|
self.queries.insert(id, query);
|
|
id
|
|
}
|
|
|
|
/// Returns a mutable reference to a query with the given ID, if it is in the pool.
|
|
pub fn get_mut(&mut self, id: QueryId) -> Option<&mut Query<TTarget, TNodeId, TResult>> {
|
|
self.queries.get_mut(&id)
|
|
}
|
|
|
|
/// Polls the pool to advance the queries.
|
|
pub fn poll(&mut self) -> QueryPoolState<'_, TTarget, TNodeId, TResult> {
|
|
let now = Instant::now();
|
|
let mut finished = None;
|
|
let mut waiting = None;
|
|
let mut timeout = None;
|
|
|
|
for (&query_id, query) in self.queries.iter_mut() {
|
|
query.started = query.started.or(Some(now));
|
|
match query.next(now) {
|
|
QueryState::Finished => {
|
|
finished = Some(query_id);
|
|
break;
|
|
}
|
|
QueryState::Waiting(Some(return_peer)) => {
|
|
waiting = Some((query_id, return_peer));
|
|
break;
|
|
}
|
|
QueryState::Waiting(None) | QueryState::WaitingAtCapacity => {
|
|
let elapsed = now - query.started.unwrap_or(now);
|
|
if elapsed >= self.query_timeout {
|
|
timeout = Some(query_id);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if let Some((query_id, return_peer)) = waiting {
|
|
let query = self.queries.get_mut(&query_id).expect("s.a.");
|
|
return QueryPoolState::Waiting(Some((query, return_peer)));
|
|
}
|
|
|
|
if let Some(query_id) = finished {
|
|
let query = self.queries.remove(&query_id).expect("s.a.");
|
|
return QueryPoolState::Finished(query);
|
|
}
|
|
|
|
if let Some(query_id) = timeout {
|
|
let query = self.queries.remove(&query_id).expect("s.a.");
|
|
return QueryPoolState::Timeout(query);
|
|
}
|
|
|
|
if self.queries.is_empty() {
|
|
QueryPoolState::Idle
|
|
} else {
|
|
QueryPoolState::Waiting(None)
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Unique identifier for an active query.
|
|
#[derive(Debug, Copy, Clone, Hash, PartialEq, Eq)]
|
|
pub struct QueryId(pub usize);
|
|
|
|
impl std::ops::Deref for QueryId {
|
|
type Target = usize;
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.0
|
|
}
|
|
}
|
|
|
|
/// A query in a `QueryPool`.
|
|
pub struct Query<TTarget, TNodeId, TResult> {
|
|
/// The unique ID of the query.
|
|
id: QueryId,
|
|
/// The peer iterator that drives the query state.
|
|
peer_iter: QueryPeerIter<TNodeId, TResult>,
|
|
/// The instant when the query started (i.e. began waiting for the first
|
|
/// result from a peer).
|
|
started: Option<Instant>,
|
|
/// Target we are looking for.
|
|
target: TTarget,
|
|
}
|
|
|
|
/// The peer selection strategies that can be used by queries.
|
|
enum QueryPeerIter<TNodeId, TResult> {
|
|
FindNode(FindNodeQuery<TNodeId>),
|
|
Predicate(PredicateQuery<TNodeId, TResult>),
|
|
}
|
|
|
|
impl<TTarget, TNodeId, TResult> Query<TTarget, TNodeId, TResult>
|
|
where
|
|
TTarget: TargetKey<TNodeId>,
|
|
TNodeId: Into<Key<TNodeId>> + Eq + Clone,
|
|
TResult: Into<TNodeId> + Clone,
|
|
{
|
|
/// Creates a new query without starting it.
|
|
fn new(id: QueryId, peer_iter: QueryPeerIter<TNodeId, TResult>, target: TTarget) -> Self {
|
|
Query {
|
|
id,
|
|
peer_iter,
|
|
target,
|
|
started: None,
|
|
}
|
|
}
|
|
|
|
/// Gets the unique ID of the query.
|
|
pub fn id(&self) -> QueryId {
|
|
self.id
|
|
}
|
|
|
|
/// Informs the query that the attempt to contact `peer` failed.
|
|
pub fn on_failure(&mut self, peer: &TNodeId) {
|
|
match &mut self.peer_iter {
|
|
QueryPeerIter::FindNode(iter) => iter.on_failure(peer),
|
|
QueryPeerIter::Predicate(iter) => iter.on_failure(peer),
|
|
}
|
|
}
|
|
|
|
/// Informs the query that the attempt to contact `peer` succeeded,
|
|
/// possibly resulting in new peers that should be incorporated into
|
|
/// the query, if applicable.
|
|
pub fn on_success<'a>(&mut self, peer: &TNodeId, new_peers: &'a [TResult])
|
|
where
|
|
&'a TResult: Into<TNodeId>,
|
|
{
|
|
match &mut self.peer_iter {
|
|
QueryPeerIter::FindNode(iter) => {
|
|
iter.on_success(peer, new_peers.iter().map(|result| result.into()).collect())
|
|
}
|
|
QueryPeerIter::Predicate(iter) => iter.on_success(peer, new_peers),
|
|
}
|
|
}
|
|
|
|
/// Advances the state of the underlying peer iterator.
|
|
fn next(&mut self, now: Instant) -> QueryState<TNodeId> {
|
|
match &mut self.peer_iter {
|
|
QueryPeerIter::FindNode(iter) => iter.next(now),
|
|
QueryPeerIter::Predicate(iter) => iter.next(now),
|
|
}
|
|
}
|
|
|
|
/// Consumes the query, producing the final `QueryResult`.
|
|
pub fn into_result(self) -> QueryResult<TTarget, impl Iterator<Item = TNodeId>> {
|
|
let peers = match self.peer_iter {
|
|
QueryPeerIter::FindNode(iter) => iter.into_result(),
|
|
QueryPeerIter::Predicate(iter) => iter.into_result(),
|
|
};
|
|
QueryResult {
|
|
target: self.target,
|
|
closest_peers: peers.into_iter(),
|
|
}
|
|
}
|
|
|
|
/// Returns a reference to the query `target`.
|
|
pub fn target(&self) -> &TTarget {
|
|
&self.target
|
|
}
|
|
|
|
/// Returns a mutable reference to the query `target`.
|
|
pub fn target_mut(&mut self) -> &mut TTarget {
|
|
&mut self.target
|
|
}
|
|
}
|
|
|
|
/// The result of a `Query`.
|
|
pub struct QueryResult<TTarget, TClosest> {
|
|
/// The target of the query.
|
|
pub target: TTarget,
|
|
/// The closest peers to the target found by the query.
|
|
pub closest_peers: TClosest,
|
|
}
|