tracing_forest/
lib.rs

1//! [![github-img]][github-url] [![crates-img]][crates-url] [![docs-img]][docs-url]
2//!
3//! [github-url]: https://github.com/QnnOkabayashi/tracing-forest
4//! [crates-url]: https://crates.io/crates/tracing-forest
5//! [docs-url]: crate
6//! [github-img]: https://img.shields.io/badge/github-8da0cb?style=for-the-badge&labelColor=555555&logo=github
7//! [crates-img]: https://img.shields.io/badge/crates.io-fc8d62?style=for-the-badge&labelColor=555555&logo=rust
8//! [docs-img]: https://img.shields.io/badge/docs.rs-66c2a5?style=for-the-badge&labelColor=555555&logoColor=white&logo=
9//!
10//! Preserve contextual coherence among trace data from concurrent tasks.
11//!
12//! # Overview
13//!
14//! [`tracing`] is a framework for instrumenting programs to collect structured
15//! and async-aware diagnostics via the [`Subscriber`] trait. The
16//! [`tracing-subscriber`] crate provides tools for composing [`Subscriber`]s
17//! from smaller units. This crate extends [`tracing-subscriber`] by providing
18//! [`ForestLayer`], a [`Layer`] that preserves contextual coherence of trace
19//! data from concurrent tasks when logging.
20//!
21//! This crate is intended for programs running many nontrivial and disjoint
22//! tasks concurrently, like server backends. Unlike [other `Subscriber`s][crate#contextual-coherence-in-action]
23//! which simply keep track of the context of an event, `tracing-forest` preserves
24//! the contextual coherence when writing logs even in parallel contexts, allowing
25//! readers to easily trace a sequence of events from the same task.
26//!
27//! `tracing-forest` is intended for authoring applications.
28//!
29//! [`tracing-subscriber`]: tracing_subscriber
30//! [`Layer`]: tracing_subscriber::layer::Layer
31//! [`Subscriber`]: tracing::subscriber::Subscriber
32//!
33//! # Getting started
34//!
35//! The easiest way to get started is to enable all features. Do this by
36//! adding the following to your `Cargo.toml` file:
37//! ```toml
38//! tracing-forest = { version = "0.2", features = ["full"] }
39//! ```
40//! Then, add [`tracing_forest::init`](crate::init) to your main function:
41//! ```
42//! tracing_forest::init();
43//! ```
44//! This crate also provides tools for much more advanced configurations:
45//! ```
46//! use tracing_forest::{traits::*, util::*};
47//!
48//! #[tokio::main]
49//! async fn main() {
50//!     tracing_forest::worker_task()
51//!         .set_global(true)
52//!         .map_sender(|sender| sender.or_stderr())
53//!         .build_on(|subscriber| subscriber
54//!             .with(EnvFilter::from_default_env())
55//!             .with(LevelFilter::INFO)
56//!         )
57//!         .on(async {
58//!             // -- snip --
59//!         })
60//!         .await;
61//! }
62//! ```
63//! For useful configuration abstractions, see the [`runtime` module documentation][runtime].
64//!
65//! # Contextual coherence in action
66//!
67//! Similar to this crate, the [`tracing-tree`] crate collects and writes trace
68//! data as a tree. Unlike this crate, it doesn't maintain contextual coherence
69//! in parallel contexts.
70//!
71//! Observe the below program, which simulates serving multiple clients concurrently.
72//! ```
73//! # async fn some_expensive_operation() {}
74//! # mod tracing_tree {
75//! #     #[derive(Default)]
76//! #     pub struct HierarchicalLayer;
77//! #     impl<S: tracing::Subscriber> tracing_subscriber::Layer<S> for HierarchicalLayer {}
78//! # }
79//! use tracing::info;
80//! use tracing_subscriber::{layer::SubscriberExt, util::SubscriberInitExt, Registry};
81//! use tracing_tree::HierarchicalLayer;
82//!
83//! #[tracing::instrument]
84//! async fn conn(id: u32) {
85//!     for i in 0..3 {
86//!         some_expensive_operation().await;
87//!         info!(id, "step {}", i);
88//!     }
89//! }
90//!
91//! #[tokio::main(flavor = "multi_thread")]
92//! async fn main() {
93//!     // Use a `tracing-tree` subscriber
94//!     Registry::default()
95//!         .with(HierarchicalLayer::default())
96//!         .init();
97//!
98//!     let connections: Vec<_> = (0..3)
99//!         .map(|id| tokio::spawn(conn(id)))
100//!         .collect();
101//!
102//!     for conn in connections {
103//!         conn.await.unwrap();
104//!     }
105//! }
106//! ```
107//! `tracing-tree` isn't intended for concurrent use, and this is demonstrated
108//! by the output of the program:
109//! ```log
110//! conn id=2
111//! conn id=0
112//! conn id=1
113//!   23ms  INFO step 0, id=2
114//!   84ms  INFO step 0, id=1
115//!   94ms  INFO step 1, id=2
116//!   118ms  INFO step 0, id=0
117//!   130ms  INFO step 1, id=1
118//!   193ms  INFO step 2, id=2
119//!
120//!   217ms  INFO step 1, id=0
121//!   301ms  INFO step 2, id=1
122//!
123//!   326ms  INFO step 2, id=0
124//!
125//! ```
126//! We can instead use `tracing-forest` as a drop-in replacement for `tracing-tree`.
127//! ```
128//! use tracing::info;
129//! use tracing_subscriber::{layer::SubscriberExt, util::SubscriberInitExt, Registry};
130//! use tracing_forest::ForestLayer;
131//!
132//! #[tracing::instrument]
133//! async fn conn(id: u32) {
134//!     // -- snip --
135//! }
136//!
137//! #[tokio::main(flavor = "multi_thread")]
138//! async fn main() {
139//!     // Use a `tracing-forest` subscriber
140//!     Registry::default()
141//!         .with(ForestLayer::default())
142//!         .init();
143//!
144//!     // -- snip --
145//! }
146//! ```
147//! Now we can easily trace what happened:
148//! ```log
149//! INFO     conn [ 150µs | 100.00% ] id: 1
150//! INFO     ┝━ i [info]: step 0 | id: 1
151//! INFO     ┝━ i [info]: step 1 | id: 1
152//! INFO     ┕━ i [info]: step 2 | id: 1
153//! INFO     conn [ 343µs | 100.00% ] id: 0
154//! INFO     ┝━ i [info]: step 0 | id: 0
155//! INFO     ┝━ i [info]: step 1 | id: 0
156//! INFO     ┕━ i [info]: step 2 | id: 0
157//! INFO     conn [ 233µs | 100.00% ] id: 2
158//! INFO     ┝━ i [info]: step 0 | id: 2
159//! INFO     ┝━ i [info]: step 1 | id: 2
160//! INFO     ┕━ i [info]: step 2 | id: 2
161//! ```
162//!
163//! [`tracing-tree`]: https://crates.io/crates/tracing-tree
164//!
165//! # Categorizing events with tags
166//!
167//! This crate allows attaching supplemental categorical information to events with [`Tag`]s.
168//!
169//! Without tags, it's difficult to distinguish where events are occurring in a system.
170//! ```log
171//! INFO     i [info]: some info for the admin
172//! ERROR    🚨 [error]: the request timed out
173//! ERROR    🚨 [error]: the db has been breached
174//! ```
175//!
176//! Tags help make this distinction more visible.
177//! ```log
178//! INFO     i [admin.info]: some info for the admin
179//! ERROR    🚨 [request.error]: the request timed out
180//! ERROR    🔐 [security.critical]: the db has been breached
181//! ```
182//!
183//! See the [`tag` module-level documentation](mod@crate::tag) for details.
184//!
185//! # Attaching `Uuid`s to trace data
186//!
187//! When the `uuid` feature is enabled, the `ForestLayer` will automatically attach
188//! [`Uuid`]s to trace data. Events will adopt the UUID of their span, or the "nil"
189//! UUID at the root level. Spans will adopt the UUID of parent spans, or generate
190//! a new UUID at the root level.
191//!
192//! A span's `Uuid` can also be passed in manually to override adopting the parent's
193//! `Uuid` by passing it in as a field named `uuid`:
194//! ```
195//! # use tracing::info_span;
196//! # use uuid::Uuid;
197//! let id = Uuid::new_v4();
198//!
199//! let span = info_span!("my_span", uuid = %id);
200//! ```
201//!
202//! It can also be retreived from the most recently entered span with
203//! [`tracing_forest::id`](crate::id):
204//! ```
205//! # use tracing::info_span;
206//! # use uuid::Uuid;
207//! # tracing_forest::init();
208//! let id = Uuid::new_v4();
209//!
210//! info_span!("my_span", uuid = %id).in_scope(|| {
211//!     let current_id = tracing_forest::id();
212//!
213//!     assert!(id == current_id);
214//! });
215//! ```
216//!
217//! # Immediate logs
218//!
219//! Since `tracing-forest` stores trace data in memory until the root span finishes,
220//! it can be a long time until a log is written. This may not be acceptable in
221//! certain use cases.
222//!
223//! To resolve this, the `immediate` field can be used on an event to print the
224//! event and its parent spans to stderr. Unlike `eprintln!`, the event will
225//! still appear in the trace tree written once the root span closes.
226//!
227//! ## Example
228//!
229//! ```
230//! use tracing::{info, trace_span};
231//!
232//! tracing_forest::init();
233//!
234//! trace_span!("my_span").in_scope(|| {
235//!     info!("first");
236//!     info!("second");
237//!     info!(immediate = true, "third, but immediately");
238//! });
239//! ```
240//! ```log
241//! INFO     i IMMEDIATE i my_span > third, but immediately
242//! TRACE    my_span [ 125µs | 100.000% ]
243//! INFO     ┝━ i [info]: first
244//! INFO     ┝━ i [info]: second
245//! INFO     ┕━ i [info]: third, but immediately
246//! ```
247//!
248//! # Feature flags
249//!
250//! This crate uses feature flags to reduce dependency bloat.
251//!
252//! * `full`: Enables all features listed below.
253//! * `uuid`: Enables spans to carry operation IDs.
254//! * `chrono`: Enables timestamps on trace data.
255//! * `ansi`: Enables ANSI terminal colors.
256//! * `smallvec`: Enables some performance optimizations.
257//! * `tokio`: Enables [`worker_task`] and [`capture`].
258//! * `serde`: Enables log trees to be serialized, which is [useful for formatting][serde_fmt].
259//! * `env-filter`: Re-exports [`EnvFilter`] from the [`util`] module.
260//! * `defer`: Allows marking a span with `defer = true` so that it will not be displayed unless it has child nodes.
261//!
262//! By default, only `smallvec` in enabled.
263//!
264//! [`Uuid`]: uuid::Uuid
265//! [serde_fmt]: crate::printer::Formatter#examples
266//! [`EnvFilter`]: tracing_subscriber::EnvFilter
267
268#![doc(issue_tracker_base_url = "https://github.com/QnnOkabayashi/tracing-forest/issues")]
269#![cfg_attr(
270    docsrs,
271    // Allows displaying cfgs/feature flags in the documentation.
272    feature(doc_cfg),
273    // Allows adding traits to RustDoc's list of "notable traits"
274    // feature(doc_notable_trait),
275    // Fail the docs build if any intra-docs links are broken
276    deny(rustdoc::broken_intra_doc_links),
277)]
278#![deny(warnings)]
279#![warn(unused_extern_crates)]
280#![warn(missing_docs)]
281
282pub mod printer;
283pub mod processor;
284pub mod tag;
285pub mod tree;
286#[macro_use]
287mod cfg;
288mod fail;
289mod layer;
290
291pub use layer::{init, test_init, ForestLayer};
292pub use printer::{Formatter, PrettyPrinter, Printer};
293pub use processor::Processor;
294pub use tag::Tag;
295
296cfg_tokio! {
297    pub mod runtime;
298    pub use runtime::{capture, worker_task};
299}
300
301cfg_uuid! {
302    pub use layer::id::id;
303}
304
305/// Bring traits from this crate, `tracing`, and `tracing_subscriber` into scope
306/// anonymously.
307pub mod traits {
308    pub use crate::Processor as _;
309    pub use tracing::Instrument as _;
310    pub use tracing_subscriber::{layer::SubscriberExt as _, util::SubscriberInitExt as _};
311}
312
313/// Bring Tracing's event and span macros into scope, along with other sensible defaults.
314pub mod util {
315    #[doc(no_inline)]
316    pub use crate::ForestLayer;
317    #[doc(no_inline)]
318    pub use tracing::metadata::LevelFilter;
319    #[doc(no_inline)]
320    pub use tracing::{
321        debug, debug_span, error, error_span, info, info_span, trace, trace_span, warn, warn_span,
322        Event, Level,
323    };
324    #[cfg(feature = "env-filter")]
325    #[doc(no_inline)]
326    pub use tracing_subscriber::EnvFilter;
327}