mirror of
https://github.com/0glabs/0g-storage-node.git
synced 2025-01-14 09:05:18 +00:00
370 lines
13 KiB
Rust
370 lines
13 KiB
Rust
#[macro_use]
|
|
extern crate tracing;
|
|
|
|
mod metrics;
|
|
pub mod test_utils;
|
|
|
|
use futures::channel::mpsc::Sender;
|
|
use futures::prelude::*;
|
|
use std::sync::Weak;
|
|
use tokio::runtime::{Handle, Runtime};
|
|
|
|
pub use tokio::task::JoinHandle;
|
|
|
|
/// Provides a reason when node is shut down.
|
|
#[derive(Copy, Clone, Debug, Eq, PartialEq)]
|
|
pub enum ShutdownReason {
|
|
/// The node shut down successfully.
|
|
Success(&'static str),
|
|
/// The node shut down due to an error condition.
|
|
Failure(&'static str),
|
|
}
|
|
|
|
impl ShutdownReason {
|
|
pub fn message(&self) -> &'static str {
|
|
match self {
|
|
ShutdownReason::Success(msg) => msg,
|
|
ShutdownReason::Failure(msg) => msg,
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Provides a `Handle` by either:
|
|
///
|
|
/// 1. Holding a `Weak<Runtime>` and calling `Runtime::handle`.
|
|
/// 2. Directly holding a `Handle` and cloning it.
|
|
///
|
|
/// This enum allows the `TaskExecutor` to work in production where a `Weak<Runtime>` is directly
|
|
/// accessible and in testing where the `Runtime` is hidden outside our scope.
|
|
#[derive(Clone)]
|
|
pub enum HandleProvider {
|
|
Runtime(Weak<Runtime>),
|
|
Handle(Handle),
|
|
}
|
|
|
|
impl From<Handle> for HandleProvider {
|
|
fn from(handle: Handle) -> Self {
|
|
HandleProvider::Handle(handle)
|
|
}
|
|
}
|
|
|
|
impl From<Weak<Runtime>> for HandleProvider {
|
|
fn from(weak_runtime: Weak<Runtime>) -> Self {
|
|
HandleProvider::Runtime(weak_runtime)
|
|
}
|
|
}
|
|
|
|
impl HandleProvider {
|
|
/// Returns a `Handle` to a `Runtime`.
|
|
///
|
|
/// May return `None` if the weak reference to the `Runtime` has been dropped (this generally
|
|
/// means Lighthouse is shutting down).
|
|
pub fn handle(&self) -> Option<Handle> {
|
|
match self {
|
|
HandleProvider::Runtime(weak_runtime) => weak_runtime
|
|
.upgrade()
|
|
.map(|runtime| runtime.handle().clone()),
|
|
HandleProvider::Handle(handle) => Some(handle.clone()),
|
|
}
|
|
}
|
|
}
|
|
|
|
/// A wrapper over a runtime handle which can spawn async and blocking tasks.
|
|
#[derive(Clone)]
|
|
pub struct TaskExecutor {
|
|
/// The handle to the runtime on which tasks are spawned
|
|
handle_provider: HandleProvider,
|
|
/// The receiver exit future which on receiving shuts down the task
|
|
exit: exit_future::Exit,
|
|
/// Sender given to tasks, so that if they encounter a state in which execution cannot
|
|
/// continue they can request that everything shuts down.
|
|
///
|
|
/// The task must provide a reason for shutting down.
|
|
signal_tx: Sender<ShutdownReason>,
|
|
}
|
|
|
|
impl TaskExecutor {
|
|
/// Create a new task executor.
|
|
///
|
|
/// ## Note
|
|
///
|
|
/// This function should only be used during testing. In production, prefer to obtain an
|
|
/// instance of `Self` via a `environment::RuntimeContext` (see the `lighthouse/environment`
|
|
/// crate).
|
|
pub fn new<T: Into<HandleProvider>>(
|
|
handle: T,
|
|
exit: exit_future::Exit,
|
|
signal_tx: Sender<ShutdownReason>,
|
|
) -> Self {
|
|
Self {
|
|
handle_provider: handle.into(),
|
|
exit,
|
|
signal_tx,
|
|
}
|
|
}
|
|
|
|
/// Clones the task executor adding a service name.
|
|
pub fn clone_with_name(&self) -> Self {
|
|
TaskExecutor {
|
|
handle_provider: self.handle_provider.clone(),
|
|
exit: self.exit.clone(),
|
|
signal_tx: self.signal_tx.clone(),
|
|
}
|
|
}
|
|
|
|
/// A convenience wrapper for `Self::spawn` which ignores a `Result` as long as both `Ok`/`Err`
|
|
/// are of type `()`.
|
|
///
|
|
/// The purpose of this function is to create a compile error if some function which previously
|
|
/// returned `()` starts returning something else. Such a case may otherwise result in
|
|
/// accidental error suppression.
|
|
pub fn spawn_ignoring_error(
|
|
&self,
|
|
task: impl Future<Output = Result<(), ()>> + Send + 'static,
|
|
name: &'static str,
|
|
) {
|
|
self.spawn(task.map(|_| ()), name)
|
|
}
|
|
|
|
/// Spawn a task to monitor the completion of another task.
|
|
///
|
|
/// If the other task exits by panicking, then the monitor task will shut down the executor.
|
|
fn spawn_monitor<R: Send>(
|
|
&self,
|
|
task_handle: impl Future<Output = Result<R, tokio::task::JoinError>> + Send + 'static,
|
|
name: &'static str,
|
|
) {
|
|
let mut shutdown_sender = self.shutdown_sender();
|
|
|
|
if let Some(handle) = self.handle() {
|
|
handle.spawn(async move {
|
|
let timer = metrics::start_timer_vec(&metrics::TASKS_HISTOGRAM, &[name]);
|
|
if let Err(join_error) = task_handle.await {
|
|
if let Ok(panic) = join_error.try_into_panic() {
|
|
let message = panic.downcast_ref::<&str>().unwrap_or(&"<none>");
|
|
|
|
error!(
|
|
task_name = name,
|
|
message = message,
|
|
advice = "Please check above for a backtrace and notify \
|
|
the developers",
|
|
"Task panic. This is a bug!",
|
|
);
|
|
let _ = shutdown_sender
|
|
.try_send(ShutdownReason::Failure("Panic (fatal error)"));
|
|
}
|
|
}
|
|
drop(timer);
|
|
});
|
|
} else {
|
|
debug!("Couldn't spawn monitor task. Runtime shutting down")
|
|
}
|
|
}
|
|
|
|
/// Spawn a future on the tokio runtime.
|
|
///
|
|
/// The future is wrapped in an `exit_future::Exit`. The task is cancelled when the corresponding
|
|
/// exit_future `Signal` is fired/dropped.
|
|
///
|
|
/// The future is monitored via another spawned future to ensure that it doesn't panic. In case
|
|
/// of a panic, the executor will be shut down via `self.signal_tx`.
|
|
///
|
|
/// This function generates prometheus metrics on number of tasks and task duration.
|
|
pub fn spawn(&self, task: impl Future<Output = ()> + Send + 'static, name: &'static str) {
|
|
if let Some(task_handle) = self.spawn_handle(task, name) {
|
|
self.spawn_monitor(task_handle, name)
|
|
}
|
|
}
|
|
|
|
/// Spawn a future on the tokio runtime. This function does not wrap the task in an `exit_future::Exit`
|
|
/// like [spawn](#method.spawn).
|
|
/// The caller of this function is responsible for wrapping up the task with an `exit_future::Exit` to
|
|
/// ensure that the task gets canceled appropriately.
|
|
/// This function generates prometheus metrics on number of tasks and task duration.
|
|
///
|
|
/// This is useful in cases where the future to be spawned needs to do additional cleanup work when
|
|
/// the task is completed/canceled (e.g. writing local variables to disk) or the task is created from
|
|
/// some framework which does its own cleanup (e.g. a hyper server).
|
|
pub fn spawn_without_exit(
|
|
&self,
|
|
task: impl Future<Output = ()> + Send + 'static,
|
|
name: &'static str,
|
|
) {
|
|
if let Some(int_gauge) = metrics::get_int_gauge(&metrics::ASYNC_TASKS_COUNT, &[name]) {
|
|
let int_gauge_1 = int_gauge.clone();
|
|
let future = task.then(move |_| {
|
|
int_gauge_1.dec();
|
|
futures::future::ready(())
|
|
});
|
|
|
|
int_gauge.inc();
|
|
if let Some(handle) = self.handle() {
|
|
handle.spawn(future);
|
|
} else {
|
|
debug!("Couldn't spawn task. Runtime shutting down");
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Spawn a blocking task on a dedicated tokio thread pool wrapped in an exit future.
|
|
/// This function generates prometheus metrics on number of tasks and task duration.
|
|
pub fn spawn_blocking<F>(&self, task: F, name: &'static str)
|
|
where
|
|
F: FnOnce() + Send + 'static,
|
|
{
|
|
if let Some(task_handle) = self.spawn_blocking_handle(task, name) {
|
|
self.spawn_monitor(task_handle, name)
|
|
}
|
|
}
|
|
|
|
/// Spawn a future on the tokio runtime wrapped in an `exit_future::Exit` returning an optional
|
|
/// join handle to the future.
|
|
/// The task is canceled when the corresponding exit_future `Signal` is fired/dropped.
|
|
///
|
|
/// This function generates prometheus metrics on number of tasks and task duration.
|
|
pub fn spawn_handle<R: Send + 'static>(
|
|
&self,
|
|
task: impl Future<Output = R> + Send + 'static,
|
|
name: &'static str,
|
|
) -> Option<tokio::task::JoinHandle<Option<R>>> {
|
|
let exit = self.exit.clone();
|
|
|
|
if let Some(int_gauge) = metrics::get_int_gauge(&metrics::ASYNC_TASKS_COUNT, &[name]) {
|
|
// Task is shutdown before it completes if `exit` receives
|
|
let int_gauge_1 = int_gauge.clone();
|
|
let future = future::select(Box::pin(task), exit).then(move |either| {
|
|
let result = match either {
|
|
future::Either::Left((value, _)) => {
|
|
trace!(task = name, "Async task completed");
|
|
Some(value)
|
|
}
|
|
future::Either::Right(_) => {
|
|
debug!(task = name, "Async task shutdown, exit received");
|
|
None
|
|
}
|
|
};
|
|
int_gauge_1.dec();
|
|
futures::future::ready(result)
|
|
});
|
|
|
|
int_gauge.inc();
|
|
if let Some(handle) = self.handle() {
|
|
Some(handle.spawn(future))
|
|
} else {
|
|
debug!("Couldn't spawn task. Runtime shutting down");
|
|
None
|
|
}
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
/// Spawn a blocking task on a dedicated tokio thread pool wrapped in an exit future returning
|
|
/// a join handle to the future.
|
|
/// If the runtime doesn't exist, this will return None.
|
|
/// The Future returned behaves like the standard JoinHandle which can return an error if the
|
|
/// task failed.
|
|
/// This function generates prometheus metrics on number of tasks and task duration.
|
|
pub fn spawn_blocking_handle<F, R>(
|
|
&self,
|
|
task: F,
|
|
name: &'static str,
|
|
) -> Option<impl Future<Output = Result<R, tokio::task::JoinError>>>
|
|
where
|
|
F: FnOnce() -> R + Send + 'static,
|
|
R: Send + 'static,
|
|
{
|
|
let timer = metrics::start_timer_vec(&metrics::BLOCKING_TASKS_HISTOGRAM, &[name]);
|
|
metrics::inc_gauge_vec(&metrics::BLOCKING_TASKS_COUNT, &[name]);
|
|
|
|
let join_handle = if let Some(handle) = self.handle() {
|
|
handle.spawn_blocking(task)
|
|
} else {
|
|
debug!("Couldn't spawn task. Runtime shutting down");
|
|
return None;
|
|
};
|
|
|
|
let future = async move {
|
|
let result = match join_handle.await {
|
|
Ok(result) => {
|
|
trace!(task = name, "Blocking task completed");
|
|
Ok(result)
|
|
}
|
|
Err(e) => {
|
|
debug!(error = %e, "Blocking task ended unexpectedly");
|
|
Err(e)
|
|
}
|
|
};
|
|
drop(timer);
|
|
metrics::dec_gauge_vec(&metrics::BLOCKING_TASKS_COUNT, &[name]);
|
|
result
|
|
};
|
|
|
|
Some(future)
|
|
}
|
|
|
|
/// Block the current (non-async) thread on the completion of some future.
|
|
///
|
|
/// ## Warning
|
|
///
|
|
/// This method is "dangerous" since calling it from an async thread will result in a panic! Any
|
|
/// use of this outside of testing should be very deeply considered as Lighthouse has been
|
|
/// burned by this function in the past.
|
|
///
|
|
/// Determining what is an "async thread" is rather challenging; just because a function isn't
|
|
/// marked as `async` doesn't mean it's not being called from an `async` function or there isn't
|
|
/// a `tokio` context present in the thread-local storage due to some `rayon` funkiness. Talk to
|
|
/// @paulhauner if you plan to use this function in production. He has put metrics in here to
|
|
/// track any use of it, so don't think you can pull a sneaky one on him.
|
|
pub fn block_on_dangerous<F: Future>(
|
|
&self,
|
|
future: F,
|
|
name: &'static str,
|
|
) -> Option<F::Output> {
|
|
let timer = metrics::start_timer_vec(&metrics::BLOCK_ON_TASKS_HISTOGRAM, &[name]);
|
|
metrics::inc_gauge_vec(&metrics::BLOCK_ON_TASKS_COUNT, &[name]);
|
|
let handle = self.handle()?;
|
|
let exit = self.exit.clone();
|
|
|
|
debug!(name = "name", "Starting block_on task");
|
|
|
|
handle.block_on(async {
|
|
let output = tokio::select! {
|
|
output = future => {
|
|
debug!(
|
|
name = "name",
|
|
"Completed block_on task"
|
|
);
|
|
Some(output)
|
|
},
|
|
_ = exit => {
|
|
debug!(
|
|
name = "name",
|
|
"Cancelled block_on task"
|
|
|
|
);
|
|
None
|
|
}
|
|
};
|
|
metrics::dec_gauge_vec(&metrics::BLOCK_ON_TASKS_COUNT, &[name]);
|
|
drop(timer);
|
|
output
|
|
})
|
|
}
|
|
|
|
/// Returns a `Handle` to the current runtime.
|
|
pub fn handle(&self) -> Option<Handle> {
|
|
self.handle_provider.handle()
|
|
}
|
|
|
|
/// Returns a copy of the `exit_future::Exit`.
|
|
pub fn exit(&self) -> exit_future::Exit {
|
|
self.exit.clone()
|
|
}
|
|
|
|
/// Get a channel to request shutting down.
|
|
pub fn shutdown_sender(&self) -> Sender<ShutdownReason> {
|
|
self.signal_tx.clone()
|
|
}
|
|
}
|