mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-06-16 21:21:03 +00:00
Add logger configuration hook (#10440)
* Initial poc Signed-off-by: Andrei Sandu <andrei-mihail@parity.io> * Make config available to logger hook Signed-off-by: Andrei Sandu <andrei-mihail@parity.io> * fmt Signed-off-by: Andrei Sandu <andrei-mihail@parity.io> * Fix tests Signed-off-by: Andrei Sandu <andrei-mihail@parity.io> * fmt Signed-off-by: Andrei Sandu <andrei-mihail@parity.io> * Add metric prefix option in sc_cli::RunCmd Signed-off-by: Andrei Sandu <andrei-mihail@parity.io> * Remove print Signed-off-by: Andrei Sandu <andrei-mihail@parity.io> * Review fixes Signed-off-by: Andrei Sandu <andrei-mihail@parity.io> * fmt Signed-off-by: Andrei Sandu <andrei-mihail@parity.io> * fix docs Signed-off-by: Andrei Sandu <andrei-mihail@parity.io>
This commit is contained in:
@@ -581,10 +581,40 @@ pub trait CliConfiguration<DCV: DefaultConfigurationValues = ()>: Sized {
|
|||||||
/// This method:
|
/// This method:
|
||||||
///
|
///
|
||||||
/// 1. Sets the panic handler
|
/// 1. Sets the panic handler
|
||||||
|
/// 2. Optionally customize logger/profiling
|
||||||
/// 2. Initializes the logger
|
/// 2. Initializes the logger
|
||||||
/// 3. Raises the FD limit
|
/// 3. Raises the FD limit
|
||||||
fn init<C: SubstrateCli>(&self) -> Result<()> {
|
///
|
||||||
sp_panic_handler::set(&C::support_url(), &C::impl_version());
|
/// The `logger_hook` closure is executed before the logger is constructed
|
||||||
|
/// and initialized. It is useful for setting up a custom profiler.
|
||||||
|
///
|
||||||
|
/// Example:
|
||||||
|
/// ```
|
||||||
|
/// use sc_tracing::{SpanDatum, TraceEvent};
|
||||||
|
/// struct TestProfiler;
|
||||||
|
///
|
||||||
|
/// impl sc_tracing::TraceHandler for TestProfiler {
|
||||||
|
/// fn handle_span(&self, sd: &SpanDatum) {}
|
||||||
|
/// fn handle_event(&self, _event: &TraceEvent) {}
|
||||||
|
/// };
|
||||||
|
///
|
||||||
|
/// fn logger_hook() -> impl FnOnce(&mut sc_cli::LoggerBuilder, &sc_service::Configuration) -> () {
|
||||||
|
/// |logger_builder, config| {
|
||||||
|
/// logger_builder.with_custom_profiling(Box::new(TestProfiler{}));
|
||||||
|
/// }
|
||||||
|
/// }
|
||||||
|
/// ```
|
||||||
|
fn init<F>(
|
||||||
|
&self,
|
||||||
|
support_url: &String,
|
||||||
|
impl_version: &String,
|
||||||
|
logger_hook: F,
|
||||||
|
config: &Configuration,
|
||||||
|
) -> Result<()>
|
||||||
|
where
|
||||||
|
F: FnOnce(&mut LoggerBuilder, &Configuration),
|
||||||
|
{
|
||||||
|
sp_panic_handler::set(support_url, impl_version);
|
||||||
|
|
||||||
let mut logger = LoggerBuilder::new(self.log_filters()?);
|
let mut logger = LoggerBuilder::new(self.log_filters()?);
|
||||||
logger
|
logger
|
||||||
@@ -600,6 +630,9 @@ pub trait CliConfiguration<DCV: DefaultConfigurationValues = ()>: Sized {
|
|||||||
logger.with_colors(false);
|
logger.with_colors(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Call hook for custom profiling setup.
|
||||||
|
logger_hook(&mut logger, &config);
|
||||||
|
|
||||||
logger.init()?;
|
logger.init()?;
|
||||||
|
|
||||||
if let Some(new_limit) = fdlimit::raise_fd_limit() {
|
if let Some(new_limit) = fdlimit::raise_fd_limit() {
|
||||||
|
|||||||
@@ -224,10 +224,46 @@ pub trait SubstrateCli: Sized {
|
|||||||
/// Create a runner for the command provided in argument. This will create a Configuration and
|
/// Create a runner for the command provided in argument. This will create a Configuration and
|
||||||
/// a tokio runtime
|
/// a tokio runtime
|
||||||
fn create_runner<T: CliConfiguration>(&self, command: &T) -> error::Result<Runner<Self>> {
|
fn create_runner<T: CliConfiguration>(&self, command: &T) -> error::Result<Runner<Self>> {
|
||||||
command.init::<Self>()?;
|
let tokio_runtime = build_runtime()?;
|
||||||
Runner::new(self, command)
|
let config = command.create_configuration(self, tokio_runtime.handle().clone())?;
|
||||||
|
|
||||||
|
command.init(&Self::support_url(), &Self::impl_version(), |_, _| {}, &config)?;
|
||||||
|
Runner::new(config, tokio_runtime)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Create a runner for the command provided in argument. The `logger_hook` can be used to setup
|
||||||
|
/// a custom profiler or update the logger configuration before it is initialized.
|
||||||
|
///
|
||||||
|
/// Example:
|
||||||
|
/// ```
|
||||||
|
/// use sc_tracing::{SpanDatum, TraceEvent};
|
||||||
|
/// struct TestProfiler;
|
||||||
|
///
|
||||||
|
/// impl sc_tracing::TraceHandler for TestProfiler {
|
||||||
|
/// fn handle_span(&self, sd: &SpanDatum) {}
|
||||||
|
/// fn handle_event(&self, _event: &TraceEvent) {}
|
||||||
|
/// };
|
||||||
|
///
|
||||||
|
/// fn logger_hook() -> impl FnOnce(&mut sc_cli::LoggerBuilder, &sc_service::Configuration) -> () {
|
||||||
|
/// |logger_builder, config| {
|
||||||
|
/// logger_builder.with_custom_profiling(Box::new(TestProfiler{}));
|
||||||
|
/// }
|
||||||
|
/// }
|
||||||
|
/// ```
|
||||||
|
fn create_runner_with_logger_hook<T: CliConfiguration, F>(
|
||||||
|
&self,
|
||||||
|
command: &T,
|
||||||
|
logger_hook: F,
|
||||||
|
) -> error::Result<Runner<Self>>
|
||||||
|
where
|
||||||
|
F: FnOnce(&mut LoggerBuilder, &Configuration),
|
||||||
|
{
|
||||||
|
let tokio_runtime = build_runtime()?;
|
||||||
|
let config = command.create_configuration(self, tokio_runtime.handle().clone())?;
|
||||||
|
|
||||||
|
command.init(&Self::support_url(), &Self::impl_version(), logger_hook, &config)?;
|
||||||
|
Runner::new(config, tokio_runtime)
|
||||||
|
}
|
||||||
/// Native runtime version.
|
/// Native runtime version.
|
||||||
fn native_runtime_version(chain_spec: &Box<dyn ChainSpec>) -> &'static RuntimeVersion;
|
fn native_runtime_version(chain_spec: &Box<dyn ChainSpec>) -> &'static RuntimeVersion;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,7 +16,7 @@
|
|||||||
// You should have received a copy of the GNU General Public License
|
// You should have received a copy of the GNU General Public License
|
||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
use crate::{error::Error as CliError, CliConfiguration, Result, SubstrateCli};
|
use crate::{error::Error as CliError, Result, SubstrateCli};
|
||||||
use chrono::prelude::*;
|
use chrono::prelude::*;
|
||||||
use futures::{future, future::FutureExt, pin_mut, select, Future};
|
use futures::{future, future::FutureExt, pin_mut, select, Future};
|
||||||
use log::info;
|
use log::info;
|
||||||
@@ -112,15 +112,8 @@ pub struct Runner<C: SubstrateCli> {
|
|||||||
|
|
||||||
impl<C: SubstrateCli> Runner<C> {
|
impl<C: SubstrateCli> Runner<C> {
|
||||||
/// Create a new runtime with the command provided in argument
|
/// Create a new runtime with the command provided in argument
|
||||||
pub fn new<T: CliConfiguration>(cli: &C, command: &T) -> Result<Runner<C>> {
|
pub fn new(config: Configuration, tokio_runtime: tokio::runtime::Runtime) -> Result<Runner<C>> {
|
||||||
let tokio_runtime = build_runtime()?;
|
Ok(Runner { config, tokio_runtime, phantom: PhantomData })
|
||||||
let runtime_handle = tokio_runtime.handle().clone();
|
|
||||||
|
|
||||||
Ok(Runner {
|
|
||||||
config: command.create_configuration(cli, runtime_handle)?,
|
|
||||||
tokio_runtime,
|
|
||||||
phantom: PhantomData,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Log information about the node itself.
|
/// Log information about the node itself.
|
||||||
|
|||||||
@@ -622,11 +622,11 @@ fn wasm_tracing_should_work(wasm_method: WasmExecutionMethod) {
|
|||||||
struct TestTraceHandler(Arc<Mutex<Vec<SpanDatum>>>);
|
struct TestTraceHandler(Arc<Mutex<Vec<SpanDatum>>>);
|
||||||
|
|
||||||
impl sc_tracing::TraceHandler for TestTraceHandler {
|
impl sc_tracing::TraceHandler for TestTraceHandler {
|
||||||
fn handle_span(&self, sd: SpanDatum) {
|
fn handle_span(&self, sd: &SpanDatum) {
|
||||||
self.0.lock().unwrap().push(sd);
|
self.0.lock().unwrap().push(sd.clone());
|
||||||
}
|
}
|
||||||
|
|
||||||
fn handle_event(&self, _event: TraceEvent) {}
|
fn handle_event(&self, _event: &TraceEvent) {}
|
||||||
}
|
}
|
||||||
|
|
||||||
let traces = Arc::new(Mutex::new(Vec::new()));
|
let traces = Arc::new(Mutex::new(Vec::new()));
|
||||||
|
|||||||
@@ -188,8 +188,6 @@ pub struct PrometheusConfig {
|
|||||||
|
|
||||||
impl PrometheusConfig {
|
impl PrometheusConfig {
|
||||||
/// Create a new config using the default registry.
|
/// Create a new config using the default registry.
|
||||||
///
|
|
||||||
/// The default registry prefixes metrics with `substrate`.
|
|
||||||
pub fn new_with_default_registry(port: SocketAddr, chain_id: String) -> Self {
|
pub fn new_with_default_registry(port: SocketAddr, chain_id: String) -> Self {
|
||||||
let param = iter::once((String::from("chain"), chain_id)).collect();
|
let param = iter::once((String::from("chain"), chain_id)).collect();
|
||||||
Self {
|
Self {
|
||||||
|
|||||||
@@ -58,7 +58,7 @@ const ZERO_DURATION: Duration = Duration::from_nanos(0);
|
|||||||
/// Responsible for assigning ids to new spans, which are not re-used.
|
/// Responsible for assigning ids to new spans, which are not re-used.
|
||||||
pub struct ProfilingLayer {
|
pub struct ProfilingLayer {
|
||||||
targets: Vec<(String, Level)>,
|
targets: Vec<(String, Level)>,
|
||||||
trace_handler: Box<dyn TraceHandler>,
|
trace_handlers: Vec<Box<dyn TraceHandler>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Used to configure how to receive the metrics
|
/// Used to configure how to receive the metrics
|
||||||
@@ -76,14 +76,14 @@ impl Default for TracingReceiver {
|
|||||||
|
|
||||||
/// A handler for tracing `SpanDatum`
|
/// A handler for tracing `SpanDatum`
|
||||||
pub trait TraceHandler: Send + Sync {
|
pub trait TraceHandler: Send + Sync {
|
||||||
/// Process a `SpanDatum`
|
/// Process a `SpanDatum`.
|
||||||
fn handle_span(&self, span: SpanDatum);
|
fn handle_span(&self, span: &SpanDatum);
|
||||||
/// Process a `TraceEvent`
|
/// Process a `TraceEvent`.
|
||||||
fn handle_event(&self, event: TraceEvent);
|
fn handle_event(&self, event: &TraceEvent);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Represents a tracing event, complete with values
|
/// Represents a tracing event, complete with values
|
||||||
#[derive(Debug)]
|
#[derive(Debug, Clone)]
|
||||||
pub struct TraceEvent {
|
pub struct TraceEvent {
|
||||||
/// Name of the event.
|
/// Name of the event.
|
||||||
pub name: String,
|
pub name: String,
|
||||||
@@ -98,7 +98,7 @@ pub struct TraceEvent {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Represents a single instance of a tracing span
|
/// Represents a single instance of a tracing span
|
||||||
#[derive(Debug)]
|
#[derive(Debug, Clone)]
|
||||||
pub struct SpanDatum {
|
pub struct SpanDatum {
|
||||||
/// id for this span
|
/// id for this span
|
||||||
pub id: Id,
|
pub id: Id,
|
||||||
@@ -213,6 +213,15 @@ impl fmt::Display for Values {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Trace handler event types.
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub enum TraceHandlerEvents {
|
||||||
|
/// An event.
|
||||||
|
Event(TraceEvent),
|
||||||
|
/// A span.
|
||||||
|
Span(SpanDatum),
|
||||||
|
}
|
||||||
|
|
||||||
impl ProfilingLayer {
|
impl ProfilingLayer {
|
||||||
/// Takes a `TracingReceiver` and a comma separated list of targets,
|
/// Takes a `TracingReceiver` and a comma separated list of targets,
|
||||||
/// either with a level: "pallet=trace,frame=debug"
|
/// either with a level: "pallet=trace,frame=debug"
|
||||||
@@ -231,7 +240,12 @@ impl ProfilingLayer {
|
|||||||
/// wasm_tracing indicates whether to enable wasm traces
|
/// wasm_tracing indicates whether to enable wasm traces
|
||||||
pub fn new_with_handler(trace_handler: Box<dyn TraceHandler>, targets: &str) -> Self {
|
pub fn new_with_handler(trace_handler: Box<dyn TraceHandler>, targets: &str) -> Self {
|
||||||
let targets: Vec<_> = targets.split(',').map(|s| parse_target(s)).collect();
|
let targets: Vec<_> = targets.split(',').map(|s| parse_target(s)).collect();
|
||||||
Self { targets, trace_handler }
|
Self { targets, trace_handlers: vec![trace_handler] }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Attach additional handlers to allow handling of custom events/spans.
|
||||||
|
pub fn add_handler(&mut self, trace_handler: Box<dyn TraceHandler>) {
|
||||||
|
self.trace_handlers.push(trace_handler);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn check_target(&self, target: &str, level: &Level) -> bool {
|
fn check_target(&self, target: &str, level: &Level) -> bool {
|
||||||
@@ -242,6 +256,18 @@ impl ProfilingLayer {
|
|||||||
}
|
}
|
||||||
false
|
false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Sequentially dispatch a trace event to all handlers.
|
||||||
|
fn dispatch_event(&self, event: TraceHandlerEvents) {
|
||||||
|
match &event {
|
||||||
|
TraceHandlerEvents::Span(span_datum) => {
|
||||||
|
self.trace_handlers.iter().for_each(|handler| handler.handle_span(span_datum));
|
||||||
|
},
|
||||||
|
TraceHandlerEvents::Event(event) => {
|
||||||
|
self.trace_handlers.iter().for_each(|handler| handler.handle_event(event));
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Default to TRACE if no level given or unable to parse Level
|
// Default to TRACE if no level given or unable to parse Level
|
||||||
@@ -320,7 +346,7 @@ where
|
|||||||
values,
|
values,
|
||||||
parent_id,
|
parent_id,
|
||||||
};
|
};
|
||||||
self.trace_handler.handle_event(trace_event);
|
self.dispatch_event(TraceHandlerEvents::Event(trace_event));
|
||||||
}
|
}
|
||||||
|
|
||||||
fn on_enter(&self, span: &Id, ctx: Context<S>) {
|
fn on_enter(&self, span: &Id, ctx: Context<S>) {
|
||||||
@@ -348,10 +374,10 @@ where
|
|||||||
span_datum.target = t;
|
span_datum.target = t;
|
||||||
}
|
}
|
||||||
if self.check_target(&span_datum.target, &span_datum.level) {
|
if self.check_target(&span_datum.target, &span_datum.level) {
|
||||||
self.trace_handler.handle_span(span_datum);
|
self.dispatch_event(TraceHandlerEvents::Span(span_datum));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
self.trace_handler.handle_span(span_datum);
|
self.dispatch_event(TraceHandlerEvents::Span(span_datum));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -374,7 +400,7 @@ fn log_level(level: Level) -> log::Level {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl TraceHandler for LogTraceHandler {
|
impl TraceHandler for LogTraceHandler {
|
||||||
fn handle_span(&self, span_datum: SpanDatum) {
|
fn handle_span(&self, span_datum: &SpanDatum) {
|
||||||
if span_datum.values.is_empty() {
|
if span_datum.values.is_empty() {
|
||||||
log::log!(
|
log::log!(
|
||||||
log_level(span_datum.level),
|
log_level(span_datum.level),
|
||||||
@@ -383,7 +409,7 @@ impl TraceHandler for LogTraceHandler {
|
|||||||
span_datum.name,
|
span_datum.name,
|
||||||
span_datum.overall_time.as_nanos(),
|
span_datum.overall_time.as_nanos(),
|
||||||
span_datum.id.into_u64(),
|
span_datum.id.into_u64(),
|
||||||
span_datum.parent_id.map(|s| s.into_u64()),
|
span_datum.parent_id.as_ref().map(|s| s.into_u64()),
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
log::log!(
|
log::log!(
|
||||||
@@ -393,18 +419,18 @@ impl TraceHandler for LogTraceHandler {
|
|||||||
span_datum.name,
|
span_datum.name,
|
||||||
span_datum.overall_time.as_nanos(),
|
span_datum.overall_time.as_nanos(),
|
||||||
span_datum.id.into_u64(),
|
span_datum.id.into_u64(),
|
||||||
span_datum.parent_id.map(|s| s.into_u64()),
|
span_datum.parent_id.as_ref().map(|s| s.into_u64()),
|
||||||
span_datum.values,
|
span_datum.values,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn handle_event(&self, event: TraceEvent) {
|
fn handle_event(&self, event: &TraceEvent) {
|
||||||
log::log!(
|
log::log!(
|
||||||
log_level(event.level),
|
log_level(event.level),
|
||||||
"{}, parent_id: {:?}, {}",
|
"{}, parent_id: {:?}, {}",
|
||||||
event.target,
|
event.target,
|
||||||
event.parent_id.map(|s| s.into_u64()),
|
event.parent_id.as_ref().map(|s| s.into_u64()),
|
||||||
event.values,
|
event.values,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -447,12 +473,12 @@ mod tests {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl TraceHandler for TestTraceHandler {
|
impl TraceHandler for TestTraceHandler {
|
||||||
fn handle_span(&self, sd: SpanDatum) {
|
fn handle_span(&self, sd: &SpanDatum) {
|
||||||
self.spans.lock().push(sd);
|
self.spans.lock().push(sd.clone());
|
||||||
}
|
}
|
||||||
|
|
||||||
fn handle_event(&self, event: TraceEvent) {
|
fn handle_event(&self, event: &TraceEvent) {
|
||||||
self.events.lock().push(event);
|
self.events.lock().push(event.clone());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -193,6 +193,7 @@ where
|
|||||||
pub struct LoggerBuilder {
|
pub struct LoggerBuilder {
|
||||||
directives: String,
|
directives: String,
|
||||||
profiling: Option<(crate::TracingReceiver, String)>,
|
profiling: Option<(crate::TracingReceiver, String)>,
|
||||||
|
custom_profiler: Option<Box<dyn crate::TraceHandler>>,
|
||||||
log_reloading: bool,
|
log_reloading: bool,
|
||||||
force_colors: Option<bool>,
|
force_colors: Option<bool>,
|
||||||
detailed_output: bool,
|
detailed_output: bool,
|
||||||
@@ -204,6 +205,7 @@ impl LoggerBuilder {
|
|||||||
Self {
|
Self {
|
||||||
directives: directives.into(),
|
directives: directives.into(),
|
||||||
profiling: None,
|
profiling: None,
|
||||||
|
custom_profiler: None,
|
||||||
log_reloading: false,
|
log_reloading: false,
|
||||||
force_colors: None,
|
force_colors: None,
|
||||||
detailed_output: false,
|
detailed_output: false,
|
||||||
@@ -220,6 +222,15 @@ impl LoggerBuilder {
|
|||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Add a custom profiler.
|
||||||
|
pub fn with_custom_profiling(
|
||||||
|
&mut self,
|
||||||
|
custom_profiler: Box<dyn crate::TraceHandler>,
|
||||||
|
) -> &mut Self {
|
||||||
|
self.custom_profiler = Some(custom_profiler);
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
/// Wether or not to disable log reloading.
|
/// Wether or not to disable log reloading.
|
||||||
pub fn with_log_reloading(&mut self, enabled: bool) -> &mut Self {
|
pub fn with_log_reloading(&mut self, enabled: bool) -> &mut Self {
|
||||||
self.log_reloading = enabled;
|
self.log_reloading = enabled;
|
||||||
@@ -256,7 +267,12 @@ impl LoggerBuilder {
|
|||||||
self.detailed_output,
|
self.detailed_output,
|
||||||
|builder| enable_log_reloading!(builder),
|
|builder| enable_log_reloading!(builder),
|
||||||
)?;
|
)?;
|
||||||
let profiling = crate::ProfilingLayer::new(tracing_receiver, &profiling_targets);
|
let mut profiling =
|
||||||
|
crate::ProfilingLayer::new(tracing_receiver, &profiling_targets);
|
||||||
|
|
||||||
|
self.custom_profiler
|
||||||
|
.into_iter()
|
||||||
|
.for_each(|profiler| profiling.add_handler(profiler));
|
||||||
|
|
||||||
tracing::subscriber::set_global_default(subscriber.with(profiling))?;
|
tracing::subscriber::set_global_default(subscriber.with(profiling))?;
|
||||||
|
|
||||||
@@ -269,7 +285,12 @@ impl LoggerBuilder {
|
|||||||
self.detailed_output,
|
self.detailed_output,
|
||||||
|builder| builder,
|
|builder| builder,
|
||||||
)?;
|
)?;
|
||||||
let profiling = crate::ProfilingLayer::new(tracing_receiver, &profiling_targets);
|
let mut profiling =
|
||||||
|
crate::ProfilingLayer::new(tracing_receiver, &profiling_targets);
|
||||||
|
|
||||||
|
self.custom_profiler
|
||||||
|
.into_iter()
|
||||||
|
.for_each(|profiler| profiling.add_handler(profiler));
|
||||||
|
|
||||||
tracing::subscriber::set_global_default(subscriber.with(profiling))?;
|
tracing::subscriber::set_global_default(subscriber.with(profiling))?;
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user