Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit c3bf010

Browse files
GitGab19plebhash
authored andcommittedNov 22, 2024··
selectors docs
1 parent 2c7904f commit c3bf010

File tree

1 file changed

+244
-39
lines changed

1 file changed

+244
-39
lines changed
 
+244-39
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,37 @@
1-
//! Selectors are used from the routing logic in order to chose to which remote or set of remotes
2-
//! a message should be ralyied, or to which remote or set of remotes a message should be sent.
1+
//! This module provides selectors and routing logic for managing downstream and upstream nodes
2+
//! in a mining proxy environment. Selectors help determine the appropriate remote(s) to relay or send messages to.
3+
//!
4+
//! ## Components
5+
//!
6+
//! - **`ProxyDownstreamMiningSelector`**: A selector for managing downstream nodes in a mining proxy,
7+
//! mapping requests and channel IDs to specific downstream nodes or groups.
8+
//! - **`NullDownstreamMiningSelector`**: A no-op selector for cases where routing logic is unnecessary,
9+
//! commonly used in test scenarios.
10+
//! - **`GeneralMiningSelector`**: A flexible upstream selector that matches downstream nodes with
11+
//! compatible upstream nodes based on pairing settings and flags.
12+
//!
13+
//! ## Traits
14+
//!
15+
//! - **`DownstreamSelector`**: Base trait for all downstream selectors.
16+
//! - **`DownstreamMiningSelector`**: Specialized trait for selectors managing mining-specific downstream nodes.
17+
//! - **`UpstreamSelector`**: Base trait for upstream node selectors.
18+
//! - **`UpstreamMiningSelctor`**: Specialized trait for selectors managing upstream mining nodes.
19+
//!
20+
//! ## Details
21+
//!
22+
//! ### ProxyDownstreamMiningSelector
23+
//! - Manages mappings for request IDs, channel IDs, and downstream groups.
24+
//! - Provides methods to handle standard channel operations, such as opening channels and
25+
//! retrieving or removing downstream nodes associated with a channel.
26+
//!
27+
//! ### NullDownstreamMiningSelector
28+
//! - Implements all required traits but panics if called.
29+
//! - Useful for minimal setups or as a placeholder in tests.
30+
//!
31+
//! ### GeneralMiningSelector
32+
//! - Matches downstream nodes to upstream nodes based on pairing compatibility.
33+
//! - Tracks upstream nodes and their IDs for efficient lookups.
34+
335
use crate::{
436
common_properties::{IsDownstream, IsMiningDownstream, IsMiningUpstream, PairSettings},
537
utils::Mutex,
@@ -8,22 +40,35 @@
840
use nohash_hasher::BuildNoHashHasher;
941
use std::{collections::HashMap, fmt::Debug as D, sync::Arc};
1042

11-
/// A DownstreamMiningSelector useful for routing messages in a mining proxy
43+
/// A selector used for routing messages to specific downstream mining nodes.
44+
///
45+
/// This structure maintains mappings for request IDs, channel IDs, and downstream nodes
46+
/// to facilitate efficient message routing.
1247
#[derive(Debug, Clone, Default)]
1348
pub struct ProxyDownstreamMiningSelector<Down: IsDownstream> {
49+
/// Maps request IDs to their corresponding downstream nodes.
1450
request_id_to_remotes: HashMap<u32, Arc<Mutex<Down>>, BuildNoHashHasher<u32>>,
51+
/// Maps group channel IDs to a list of downstream nodes.
1552
channel_id_to_downstreams: HashMap<u32, Vec<Arc<Mutex<Down>>>, BuildNoHashHasher<u32>>,
53+
/// Maps standard channel IDs to a single downstream node.
1654
channel_id_to_downstream: HashMap<u32, Arc<Mutex<Down>>, BuildNoHashHasher<u32>>,
1755
}
1856

1957
impl<Down: IsDownstream> ProxyDownstreamMiningSelector<Down> {
58+
/// Creates a new `ProxyDownstreamMiningSelector`.
59+
///
60+
/// This initializes the internal mappings with `nohash` hashers for performance.
2061
pub fn new() -> Self {
2162
Self {
2263
request_id_to_remotes: HashMap::with_hasher(BuildNoHashHasher::default()),
2364
channel_id_to_downstreams: HashMap::with_hasher(BuildNoHashHasher::default()),
2465
channel_id_to_downstream: HashMap::with_hasher(BuildNoHashHasher::default()),
2566
}
2667
}
68+
69+
/// Creates a new `ProxyDownstreamMiningSelector` wrapped in a mutex and an `Arc`.
70+
///
71+
/// This is useful for concurrent environments where shared ownership is needed.
2772
pub fn new_as_mutex() -> Arc<Mutex<Self>>
2873
where
2974
Self: Sized,
@@ -33,6 +78,10 @@
3378
}
3479

3580
impl<Down: IsMiningDownstream> ProxyDownstreamMiningSelector<Down> {
81+
/// Removes a downstream node from all mappings.
82+
///
83+
/// # Arguments
84+
/// - `d`: The downstream node to be removed.
3685
fn _remove_downstream(&mut self, d: &Arc<Mutex<Down>>) {
3786
self.request_id_to_remotes.retain(|_, v| !Arc::ptr_eq(v, d));
3887
self.channel_id_to_downstream
@@ -41,12 +90,27 @@
4190
}
4291

4392
impl<Down: IsMiningDownstream> DownstreamMiningSelector<Down>
44-
for ProxyDownstreamMiningSelector<Down>
93+
for ProxyDownstreamMiningSelector<Down>
4594
{
95+
/// Registers a request ID and its associated downstream node.
96+
///
97+
/// # Arguments
98+
/// - `request_id`: The unique request ID.
99+
/// - `downstream`: The downstream node associated with the request.
46100
fn on_open_standard_channel_request(&mut self, request_id: u32, downstream: Arc<Mutex<Down>>) {
47101
self.request_id_to_remotes.insert(request_id, downstream);
48102
}
49103

104+
/// Finalizes the mapping of a standard channel to its downstream node.
105+
///
106+
/// # Arguments
107+
/// - `request_id`: The request ID used during the channel opening.
108+
/// - `g_channel_id`: The group channel ID.
109+
/// - `channel_id`: The specific standard channel ID.
110+
///
111+
/// # Returns
112+
/// - `Ok`: The downstream node associated with the request.
113+
/// - `Err`: If the request ID is unknown.
50114
fn on_open_standard_channel_success(
51115
&mut self,
52116
request_id: u32,
@@ -69,10 +133,25 @@
69133
Ok(downstream)
70134
}
71135

136+
/// Retrieves all downstream nodes associated with a standard/group channel ID.
137+
///
138+
/// # Arguments
139+
/// - `channel_id`: The standard/group channel ID.
140+
///
141+
/// # Returns
142+
/// - `Some`: A reference to the vector of downstream nodes.
143+
/// - `None`: If no nodes are associated with the channel.
72144
fn get_downstreams_in_channel(&self, channel_id: u32) -> Option<&Vec<Arc<Mutex<Down>>>> {
73145
self.channel_id_to_downstreams.get(&channel_id)
74146
}
75147

148+
/// Removes all downstream nodes associated with a standard/group channel ID.
149+
///
150+
/// # Arguments
151+
/// - `channel_id`: The standard/group channel ID.
152+
///
153+
/// # Returns
154+
/// A vector of the removed downstream nodes.
76155
fn remove_downstreams_in_channel(&mut self, channel_id: u32) -> Vec<Arc<Mutex<Down>>> {
77156
let downs = self
78157
.channel_id_to_downstreams
@@ -84,17 +163,34 @@
84163
downs
85164
}
86165

166+
/// Removes a specific downstream node from all mappings.
167+
///
168+
/// # Arguments
169+
/// - `d`: The downstream node to be removed.
87170
fn remove_downstream(&mut self, d: &Arc<Mutex<Down>>) {
88171
for dws in self.channel_id_to_downstreams.values_mut() {
89-
dws.retain(|d| !Arc::ptr_eq(d, d));
172+
dws.retain(|node| !Arc::ptr_eq(node, d));
90173
}
91174

92175
self._remove_downstream(d);
93176
}
94177

178+
/// Retrieves the downstream node associated with a specific standard channel ID.
179+
///
180+
/// # Arguments
181+
/// - `channel_id`: The standard channel ID.
182+
///
183+
/// # Returns
184+
/// - `Some`: The downstream node.
185+
/// - `None`: If no node is associated with the channel.
95186
fn downstream_from_channel_id(&self, channel_id: u32) -> Option<Arc<Mutex<Down>>> {
96187
self.channel_id_to_downstream.get(&channel_id).cloned()
97188
}
189+
190+
/// Retrieves all downstream nodes currently managed by this selector.
191+
///
192+
/// # Returns
193+
/// A vector of downstream nodes.
98194
fn get_all_downstreams(&self) -> Vec<Arc<Mutex<Down>>> {
99195
self.channel_id_to_downstream.values().cloned().collect()
100196
}
@@ -103,49 +199,93 @@
103199
impl<Down: IsMiningDownstream> DownstreamSelector<Down> for ProxyDownstreamMiningSelector<Down> {}
104200

105201
/// Implemented by a selector used by an upstream mining node or and upstream mining node
106-
/// abstraction in order to find the right downstream to which a message should be sent or relayied
202+
/// abstraction in order to find the right downstream to which a message should be sent or relayed.
107203
pub trait DownstreamMiningSelector<Downstream: IsMiningDownstream>:
108-
DownstreamSelector<Downstream>
204+
DownstreamSelector<Downstream>
109205
{
206+
/// Handles a request to open a standard channel.
207+
///
208+
/// # Arguments
209+
/// - `request_id`: The ID of the request.
210+
/// - `downstream`: A reference to the downstream requesting the channel.
110211
fn on_open_standard_channel_request(
111212
&mut self,
112213
request_id: u32,
113214
downstream: Arc<Mutex<Downstream>>,
114215
);
115216

217+
/// Handles a successful response to opening a standard channel.
218+
///
219+
/// # Arguments
220+
/// - `request_id`: The ID of the request.
221+
/// - `g_channel_id`: The global channel ID.
222+
/// - `channel_id`: The local channel ID.
223+
///
224+
/// # Returns
225+
/// - `Result<Arc<Mutex<Downstream>>, Error>`: The downstream associated with the channel or an error.
116226
fn on_open_standard_channel_success(
117227
&mut self,
118228
request_id: u32,
119229
g_channel_id: u32,
120230
channel_id: u32,
121231
) -> Result<Arc<Mutex<Downstream>>, Error>;
122232

123-
// group / standard naming is terrible channel_id in this case can be either the channel_id
124-
// or the group_channel_id
233+
/// Retrieves all downstreams associated with a channel ID.
234+
///
235+
/// # Arguments
236+
/// - `channel_id`: The channel ID to query.
237+
///
238+
/// # Returns
239+
/// - `Option<&Vec<Arc<Mutex<Downstream>>>>`: The list of downstreams or `None`.
125240
fn get_downstreams_in_channel(&self, channel_id: u32) -> Option<&Vec<Arc<Mutex<Downstream>>>>;
126241

242+
/// Removes all downstreams associated with a channel ID.
243+
///
244+
/// # Arguments
245+
/// - `channel_id`: The channel ID to remove downstreams from.
246+
///
247+
/// # Returns
248+
/// - `Vec<Arc<Mutex<Downstream>>>`: The removed downstreams.
127249
fn remove_downstreams_in_channel(&mut self, channel_id: u32) -> Vec<Arc<Mutex<Downstream>>>;
128250

251+
/// Removes a specific downstream.
252+
///
253+
/// # Arguments
254+
/// - `d`: A reference to the downstream to remove.
129255
fn remove_downstream(&mut self, d: &Arc<Mutex<Downstream>>);
130256

131-
// only for standard
257+
/// Retrieves a downstream by channel ID (only for standard channels).
258+
///
259+
/// # Arguments
260+
/// - `channel_id`: The channel ID to query.
261+
///
262+
/// # Returns
263+
/// - `Option<Arc<Mutex<Downstream>>>`: The downstream or `None`.
132264
fn downstream_from_channel_id(&self, channel_id: u32) -> Option<Arc<Mutex<Downstream>>>;
133265

266+
/// Retrieves all downstreams.
267+
///
268+
/// # Returns
269+
/// - `Vec<Arc<Mutex<Downstream>>>`: All downstreams.
134270
fn get_all_downstreams(&self) -> Vec<Arc<Mutex<Downstream>>>;
135271
}
136272

273+
/// A generic downstream selector.
137274
pub trait DownstreamSelector<D: IsDownstream> {}
138275

139-
/// A DownstreamMiningSelector that do nothing. Useful when ParseDownstreamCommonMessages or
140-
/// ParseUpstreamCommonMessages must be implemented in very simple application (eg for test
141-
/// puorposes)
276+
/// A no-op implementation of `DownstreamMiningSelector`.
277+
///
278+
/// This selector is primarily used for testing or minimal setups where routing logic is not needed.
142279
#[derive(Debug, Clone, Copy, Default)]
143280
pub struct NullDownstreamMiningSelector();
144281

145282
impl NullDownstreamMiningSelector {
283+
/// Creates a new `NullDownstreamMiningSelector`.
146284
pub fn new() -> Self {
147285
NullDownstreamMiningSelector()
148286
}
287+
288+
/// Creates a new `NullDownstreamMiningSelector` wrapped in a mutex and an `Arc`.
149289
pub fn new_as_mutex() -> Arc<Mutex<Self>>
150290
where
151291
Self: Sized,
@@ -155,6 +295,9 @@
155295
}
156296

157297
impl<Down: IsMiningDownstream + D> DownstreamMiningSelector<Down> for NullDownstreamMiningSelector {
298+
/// Called when a standard channel open request is received.
299+
///
300+
/// This method is unreachable in `NullDownstreamMiningSelector` since it is a no-op implementation.
158301
fn on_open_standard_channel_request(
159302
&mut self,
160303
_request_id: u32,
@@ -163,6 +306,9 @@
163306
unreachable!("on_open_standard_channel_request")
164307
}
165308

309+
/// Called when a standard channel open request is successful.
310+
///
311+
/// This method is unreachable in `NullDownstreamMiningSelector`.
166312
fn on_open_standard_channel_success(
167313
&mut self,
168314
_request_id: u32,
@@ -172,67 +318,109 @@
172318
unreachable!("on_open_standard_channel_success")
173319
}
174320

321+
/// Retrieves the downstreams in a specific channel.
322+
///
323+
/// This method is unreachable in `NullDownstreamMiningSelector`.
175324
fn get_downstreams_in_channel(&self, _channel_id: u32) -> Option<&Vec<Arc<Mutex<Down>>>> {
176325
unreachable!("get_downstreams_in_channel")
177326
}
327+
328+
/// Removes downstreams in a specific channel.
329+
///
330+
/// This method is unreachable in `NullDownstreamMiningSelector`.
178331
fn remove_downstreams_in_channel(&mut self, _channel_id: u32) -> Vec<Arc<Mutex<Down>>> {
179332
unreachable!("remove_downstreams_in_channel")
180333
}
181334

335+
/// Removes a specific downstream node.
336+
///
337+
/// This method is unreachable in `NullDownstreamMiningSelector`.
338+
fn remove_downstream(&mut self, _d: &Arc<Mutex<Down>>) {
339+
unreachable!("remove_downstream")
340+
}
341+
342+
/// Retrieves the downstream associated with a specific channel ID.
343+
///
344+
/// This method is unreachable in `NullDownstreamMiningSelector`.
182345
fn downstream_from_channel_id(&self, _channel_id: u32) -> Option<Arc<Mutex<Down>>> {
183346
unreachable!("downstream_from_channel_id")
184347
}
348+
349+
/// Retrieves all downstream nodes managed by this selector.
350+
///
351+
/// This method is unreachable in `NullDownstreamMiningSelector`.
185352
fn get_all_downstreams(&self) -> Vec<Arc<Mutex<Down>>> {
186353
unreachable!("get_all_downstreams")
187354
}
188-
189-
fn remove_downstream(&mut self, _d: &Arc<Mutex<Down>>) {
190-
unreachable!("remove_downstream")
191-
}
192355
}
193356

194357
impl<Down: IsDownstream + D> DownstreamSelector<Down> for NullDownstreamMiningSelector {}
195358

359+
/// Trait for selecting upstream nodes in a mining context.
196360
pub trait UpstreamSelector {}
197361

362+
/// Trait for selecting upstream mining nodes.
363+
///
364+
/// This trait allows pairing downstream mining nodes with upstream nodes
365+
/// based on their settings and capabilities.
198366
pub trait UpstreamMiningSelctor<
199367
Down: IsMiningDownstream,
200368
Up: IsMiningUpstream<Down, Sel>,
201369
Sel: DownstreamMiningSelector<Down>,
202370
>: UpstreamSelector
203371
{
204-
#[allow(clippy::type_complexity)]
372+
/// Handles the `SetupConnection` process.
373+
///
374+
/// # Arguments
375+
/// - `pair_settings`: The settings for pairing downstream and upstream nodes.
376+
///
377+
/// # Returns
378+
/// - `Ok((Vec<Arc<Mutex<Up>>>, u32))`: A vector of upstream nodes and their combined flags.
379+
/// - `Err`: If no upstreams are pairable.
205380
fn on_setup_connection(
206381
&mut self,
207382
pair_settings: &PairSettings,
208383
) -> Result<(Vec<Arc<Mutex<Up>>>, u32), Error>;
384+
385+
/// Retrieves an upstream node by its ID.
386+
///
387+
/// # Arguments
388+
/// - `upstream_id`: The unique ID of the upstream node.
389+
///
390+
/// # Returns
391+
/// - `Some`: The upstream node.
392+
/// - `None`: If no upstream is found.
209393
fn get_upstream(&self, upstream_id: u32) -> Option<Arc<Mutex<Up>>>;
210394
}
211395

212-
/// Upstream selector is used to chose between a set of known mining upstream nodes which one/ones
213-
/// can accept messages from a specific mining downstream node
396+
/// General implementation of an upstream mining selector.
214397
#[derive(Debug)]
215398
pub struct GeneralMiningSelector<
216399
Sel: DownstreamMiningSelector<Down>,
217400
Down: IsMiningDownstream,
218401
Up: IsMiningUpstream<Down, Sel>,
219402
> {
403+
/// List of upstream nodes.
220404
pub upstreams: Vec<Arc<Mutex<Up>>>,
405+
/// Mapping of upstream IDs to their respective nodes.
221406
pub id_to_upstream: HashMap<u32, Arc<Mutex<Up>>, BuildNoHashHasher<u32>>,
222407
sel: std::marker::PhantomData<Sel>,
223408
down: std::marker::PhantomData<Down>,
224409
}
225410

226411
impl<
227-
Sel: DownstreamMiningSelector<Down>,
228-
Up: IsMiningUpstream<Down, Sel>,
229-
Down: IsMiningDownstream,
230-
> GeneralMiningSelector<Sel, Down, Up>
412+
Sel: DownstreamMiningSelector<Down>,
413+
Up: IsMiningUpstream<Down, Sel>,
414+
Down: IsMiningDownstream,
415+
> GeneralMiningSelector<Sel, Down, Up>
231416
{
417+
/// Creates a new `GeneralMiningSelector`.
418+
///
419+
/// # Arguments
420+
/// - `upstreams`: A vector of upstream nodes.
232421
pub fn new(upstreams: Vec<Arc<Mutex<Up>>>) -> Self {
233422
let mut id_to_upstream = HashMap::with_hasher(BuildNoHashHasher::default());
234423
for up in &upstreams {
235-
// Is ok to unwrap safe_lock result
236424
id_to_upstream.insert(up.safe_lock(|u| u.get_id()).unwrap(), up.clone());
237425
}
238426
Self {
@@ -242,27 +430,38 @@
242430
down: std::marker::PhantomData,
243431
}
244432
}
433+
434+
/// Updates the list of upstream nodes.
435+
///
436+
/// # Arguments
437+
/// - `upstreams`: The new list of upstream nodes.
245438
pub fn update_upstreams(&mut self, upstreams: Vec<Arc<Mutex<Up>>>) {
246439
self.upstreams = upstreams;
247440
}
248441
}
442+
249443
impl<
250-
Sel: DownstreamMiningSelector<Down>,
251-
Down: IsMiningDownstream,
252-
Up: IsMiningUpstream<Down, Sel>,
253-
> UpstreamSelector for GeneralMiningSelector<Sel, Down, Up>
444+
Sel: DownstreamMiningSelector<Down>,
445+
Down: IsMiningDownstream,
446+
Up: IsMiningUpstream<Down, Sel>,
447+
> UpstreamSelector for GeneralMiningSelector<Sel, Down, Up>
254448
{
255449
}
256450

257451
impl<
258-
Sel: DownstreamMiningSelector<Down>,
259-
Down: IsMiningDownstream,
260-
Up: IsMiningUpstream<Down, Sel>,
261-
> UpstreamMiningSelctor<Down, Up, Sel> for GeneralMiningSelector<Sel, Down, Up>
452+
Sel: DownstreamMiningSelector<Down>,
453+
Down: IsMiningDownstream,
454+
Up: IsMiningUpstream<Down, Sel>,
455+
> UpstreamMiningSelctor<Down, Up, Sel> for GeneralMiningSelector<Sel, Down, Up>
262456
{
263-
/// Return the set of mining upstream nodes that can accept messages from a downstream with
264-
/// the passed PairSettings and the sum of all the accepted flags
265-
#[allow(clippy::type_complexity)]
457+
/// Handles the `SetupConnection` process and determines the pairable upstream nodes.
458+
///
459+
/// # Arguments
460+
/// - `pair_settings`: The settings for pairing downstream and upstream nodes.
461+
///
462+
/// # Returns
463+
/// - `Ok((Vec<Arc<Mutex<Up>>>, u32))`: Pairable upstream nodes and their combined flags.
464+
/// - `Err`: If no upstreams are pairable.
266465
fn on_setup_connection(
267466
&mut self,
268467
pair_settings: &PairSettings,
@@ -272,10 +471,8 @@
272471
for node in &self.upstreams {
273472
let is_pairable = node
274473
.safe_lock(|node| node.is_pairable(pair_settings))
275-
// Is ok to unwrap safe_lock result
276474
.unwrap();
277475
if is_pairable {
278-
// Is ok to unwrap safe_lock result
279476
supported_flags |= node.safe_lock(|n| n.get_flags()).unwrap();
280477
supported_upstreams.push(node.clone());
281478
}
@@ -287,7 +484,15 @@
287484
Err(Error::NoPairableUpstream((2, 2, 0)))
288485
}
289486

487+
/// Retrieves an upstream node by its ID.
488+
///
489+
/// # Arguments
490+
/// - `upstream_id`: The unique ID of the upstream node.
491+
///
492+
/// # Returns
493+
/// - `Some`: The upstream node.
494+
/// - `None`: If no upstream is found.
290495
fn get_upstream(&self, upstream_id: u32) -> Option<Arc<Mutex<Up>>> {
291496
self.id_to_upstream.get(&upstream_id).cloned()
292497
}
293-
}
498+
}

0 commit comments

Comments
 (0)
Please sign in to comment.