sui_node/handle.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141
// Copyright (c) Mysten Labs, Inc.
// SPDX-License-Identifier: Apache-2.0
//! SuiNodeHandle wraps SuiNode in a way suitable for access by test code.
//!
//! When starting a SuiNode directly, in a test (as opposed to using Swarm), the node may be
//! running inside of a simulator node. It is therefore a mistake to do something like:
//!
//! ```ignore
//! use test_utils::authority::{start_node, spawn_checkpoint_processes};
//!
//! let node = start_node(config, registry).await;
//! spawn_checkpoint_processes(config, &[node]).await;
//! ```
//!
//! Because this would cause the checkpointing processes to be running inside the current
//! simulator node rather than the node in which the SuiNode is running.
//!
//! SuiNodeHandle provides an easy way to do the right thing here:
//!
//! ```ignore
//! let node_handle = start_node(config, registry).await;
//! node_handle.with_async(|sui_node| async move {
//! spawn_checkpoint_processes(config, &[sui_node]).await;
//! });
//! ```
//!
//! Code executed inside of with or with_async will run in the context of the simulator node.
//! This allows tests to break the simulator abstraction and magically mutate or inspect state that
//! is conceptually running on a different "machine", but without producing extremely confusing
//! behavior that might result otherwise. (For instance, any network connection that is initiated
//! from a task spawned from within a with or with_async will appear to originate from the correct
//! simulator node.
//!
//! It is possible to exfiltrate state:
//!
//! ```ignore
//! let state = node_handle.with(|sui_node| sui_node.state);
//! // DO NOT DO THIS!
//! do_stuff_with_state(state)
//! ```
//!
//! We can't prevent this completely, but we can at least make the right way the easy way.
use super::SuiNode;
use std::future::Future;
use std::sync::Arc;
use sui_core::authority::AuthorityState;
/// Wrap SuiNode to allow correct access to SuiNode in simulator tests.
pub struct SuiNodeHandle {
node: Option<Arc<SuiNode>>,
shutdown_on_drop: bool,
}
impl SuiNodeHandle {
pub fn new(node: Arc<SuiNode>) -> Self {
Self {
node: Some(node),
shutdown_on_drop: false,
}
}
pub fn inner(&self) -> &Arc<SuiNode> {
self.node.as_ref().unwrap()
}
pub fn with<T>(&self, cb: impl FnOnce(&SuiNode) -> T) -> T {
let _guard = self.guard();
cb(self.inner())
}
pub fn state(&self) -> Arc<AuthorityState> {
self.with(|sui_node| sui_node.state())
}
pub fn shutdown_on_drop(&mut self) {
self.shutdown_on_drop = true;
}
}
impl Clone for SuiNodeHandle {
fn clone(&self) -> Self {
Self {
node: self.node.clone(),
shutdown_on_drop: false,
}
}
}
#[cfg(not(msim))]
impl SuiNodeHandle {
// Must return something to silence lints above at `let _guard = ...`
fn guard(&self) -> u32 {
0
}
pub async fn with_async<'a, F, R, T>(&'a self, cb: F) -> T
where
F: FnOnce(&'a SuiNode) -> R,
R: Future<Output = T>,
{
cb(self.inner()).await
}
}
#[cfg(msim)]
impl SuiNodeHandle {
fn guard(&self) -> sui_simulator::runtime::NodeEnterGuard {
self.inner().sim_state.sim_node.enter_node()
}
pub async fn with_async<'a, F, R, T>(&'a self, cb: F) -> T
where
F: FnOnce(&'a SuiNode) -> R,
R: Future<Output = T>,
{
let fut = cb(self.node.as_ref().unwrap());
self.inner()
.sim_state
.sim_node
.await_future_in_node(fut)
.await
}
}
#[cfg(msim)]
impl Drop for SuiNodeHandle {
fn drop(&mut self) {
if self.shutdown_on_drop {
let node_id = self.inner().sim_state.sim_node.id();
sui_simulator::runtime::Handle::try_current().map(|h| h.delete_node(node_id));
}
}
}
impl From<Arc<SuiNode>> for SuiNodeHandle {
fn from(node: Arc<SuiNode>) -> Self {
SuiNodeHandle::new(node)
}
}