1use sentry_core::protocol::{Event, Value};
2#[cfg(feature = "logs")]
3use sentry_core::protocol::{Log, LogAttribute, LogLevel};
4use sentry_core::{Breadcrumb, Level};
5use std::collections::BTreeMap;
6#[cfg(feature = "logs")]
7use std::time::SystemTime;
8
9pub fn convert_log_level(level: log::Level) -> Level {
11 match level {
12 log::Level::Error => Level::Error,
13 log::Level::Warn => Level::Warning,
14 log::Level::Info => Level::Info,
15 log::Level::Debug | log::Level::Trace => Level::Debug,
16 }
17}
18
19#[cfg(feature = "logs")]
21pub fn convert_log_level_to_sentry_log_level(level: log::Level) -> LogLevel {
22 match level {
23 log::Level::Error => LogLevel::Error,
24 log::Level::Warn => LogLevel::Warn,
25 log::Level::Info => LogLevel::Info,
26 log::Level::Debug => LogLevel::Debug,
27 log::Level::Trace => LogLevel::Trace,
28 }
29}
30
31#[derive(Default)]
33struct AttributeVisitor {
34 json_values: BTreeMap<String, Value>,
35}
36
37impl AttributeVisitor {
38 fn record<T: Into<Value>>(&mut self, key: &str, value: T) {
39 self.json_values.insert(key.to_owned(), value.into());
40 }
41}
42
43impl log::kv::VisitSource<'_> for AttributeVisitor {
44 fn visit_pair(
45 &mut self,
46 key: log::kv::Key,
47 value: log::kv::Value,
48 ) -> Result<(), log::kv::Error> {
49 let key = key.as_str();
50
51 if let Some(value) = value.to_borrowed_str() {
52 self.record(key, value);
53 } else if let Some(value) = value.to_u64() {
54 self.record(key, value);
55 } else if let Some(value) = value.to_f64() {
56 self.record(key, value);
57 } else if let Some(value) = value.to_bool() {
58 self.record(key, value);
59 } else {
60 self.record(key, format!("{value:?}"));
61 };
62
63 Ok(())
64 }
65}
66
67fn extract_record_attributes(record: &log::Record<'_>) -> AttributeVisitor {
68 let mut visitor = AttributeVisitor::default();
69 let _ = record.key_values().visit(&mut visitor);
70 visitor
71}
72
73pub fn breadcrumb_from_record(record: &log::Record<'_>) -> Breadcrumb {
75 let visitor = extract_record_attributes(record);
76
77 Breadcrumb {
78 ty: "log".into(),
79 level: convert_log_level(record.level()),
80 category: Some(record.target().into()),
81 message: Some(record.args().to_string()),
82 data: visitor.json_values,
83 ..Default::default()
84 }
85}
86
87pub fn event_from_record(record: &log::Record<'_>) -> Event<'static> {
89 let visitor = extract_record_attributes(record);
90 let attributes = visitor.json_values;
91
92 let mut contexts = BTreeMap::new();
93
94 let mut metadata_map = BTreeMap::new();
95 metadata_map.insert("logger.target".into(), record.target().into());
96 if let Some(module_path) = record.module_path() {
97 metadata_map.insert("logger.module_path".into(), module_path.into());
98 }
99 if let Some(file) = record.file() {
100 metadata_map.insert("logger.file".into(), file.into());
101 }
102 if let Some(line) = record.line() {
103 metadata_map.insert("logger.line".into(), line.into());
104 }
105 contexts.insert(
106 "Rust Log Metadata".to_string(),
107 sentry_core::protocol::Context::Other(metadata_map),
108 );
109
110 if !attributes.is_empty() {
111 contexts.insert(
112 "Rust Log Attributes".to_string(),
113 sentry_core::protocol::Context::Other(attributes),
114 );
115 }
116
117 Event {
118 logger: Some(record.target().into()),
119 level: convert_log_level(record.level()),
120 message: Some(record.args().to_string()),
121 contexts,
122 ..Default::default()
123 }
124}
125
126pub fn exception_from_record(record: &log::Record<'_>) -> Event<'static> {
128 event_from_record(record)
133}
134
135#[cfg(feature = "logs")]
137pub fn log_from_record(record: &log::Record<'_>) -> Log {
138 let visitor = extract_record_attributes(record);
139
140 let mut attributes: BTreeMap<String, LogAttribute> = visitor
141 .json_values
142 .into_iter()
143 .map(|(key, val)| (key, val.into()))
144 .collect();
145
146 attributes.insert("logger.target".into(), record.target().into());
147 if let Some(module_path) = record.module_path() {
148 attributes.insert("logger.module_path".into(), module_path.into());
149 }
150 if let Some(file) = record.file() {
151 attributes.insert("logger.file".into(), file.into());
152 }
153 if let Some(line) = record.line() {
154 attributes.insert("logger.line".into(), line.into());
155 }
156
157 attributes.insert("sentry.origin".into(), "auto.logger.log".into());
158
159 Log {
160 level: convert_log_level_to_sentry_log_level(record.level()),
161 body: format!("{}", record.args()),
162 trace_id: None,
163 timestamp: SystemTime::now(),
164 severity_number: None,
165 attributes,
166 }
167}