layer.rs 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. use serde::ser::{SerializeMap, Serializer};
  2. use serde_json::Value;
  3. use std::{fmt, io::Write};
  4. use tracing::{Event, Id, Subscriber};
  5. use tracing_bunyan_formatter::JsonStorage;
  6. use tracing_core::{metadata::Level, span::Attributes};
  7. use tracing_subscriber::{fmt::MakeWriter, layer::Context, registry::SpanRef, Layer};
  8. const LEVEL: &str = "level";
  9. const TIME: &str = "time";
  10. const MESSAGE: &str = "msg";
  11. const LOG_MODULE_PATH: &str = "log.module_path";
  12. const LOG_TARGET_PATH: &str = "log.target";
  13. const FLOWY_RESERVED_FIELDS: [&str; 3] = [LEVEL, TIME, MESSAGE];
  14. const IGNORE_FIELDS: [&str; 2] = [LOG_MODULE_PATH, LOG_TARGET_PATH];
  15. pub struct FlowyFormattingLayer<W: MakeWriter + 'static> {
  16. make_writer: W,
  17. with_target: bool,
  18. }
  19. impl<W: MakeWriter + 'static> FlowyFormattingLayer<W> {
  20. pub fn new(make_writer: W) -> Self {
  21. Self {
  22. make_writer,
  23. with_target: false,
  24. }
  25. }
  26. fn serialize_flowy_folder_fields(
  27. &self,
  28. map_serializer: &mut impl SerializeMap<Error = serde_json::Error>,
  29. message: &str,
  30. _level: &Level,
  31. ) -> Result<(), std::io::Error> {
  32. map_serializer.serialize_entry(MESSAGE, &message)?;
  33. // map_serializer.serialize_entry(LEVEL, &format!("{}", level))?;
  34. // map_serializer.serialize_entry(TIME, &chrono::Utc::now().timestamp())?;
  35. Ok(())
  36. }
  37. fn serialize_span<S: Subscriber + for<'a> tracing_subscriber::registry::LookupSpan<'a>>(
  38. &self,
  39. span: &SpanRef<S>,
  40. ty: Type,
  41. ) -> Result<Vec<u8>, std::io::Error> {
  42. let mut buffer = Vec::new();
  43. let mut serializer = serde_json::Serializer::new(&mut buffer);
  44. let mut map_serializer = serializer.serialize_map(None)?;
  45. let message = format_span_context(span, ty);
  46. self.serialize_flowy_folder_fields(&mut map_serializer, &message, span.metadata().level())?;
  47. if self.with_target {
  48. map_serializer.serialize_entry("target", &span.metadata().target())?;
  49. }
  50. map_serializer.serialize_entry("line", &span.metadata().line())?;
  51. map_serializer.serialize_entry("file", &span.metadata().file())?;
  52. let extensions = span.extensions();
  53. if let Some(visitor) = extensions.get::<JsonStorage>() {
  54. for (key, value) in visitor.values() {
  55. if !FLOWY_RESERVED_FIELDS.contains(key) && !IGNORE_FIELDS.contains(key) {
  56. map_serializer.serialize_entry(key, value)?;
  57. } else {
  58. tracing::debug!(
  59. "{} is a reserved field in the bunyan log format. Skipping it.",
  60. key
  61. );
  62. }
  63. }
  64. }
  65. map_serializer.end()?;
  66. Ok(buffer)
  67. }
  68. fn emit(&self, mut buffer: Vec<u8>) -> Result<(), std::io::Error> {
  69. buffer.write_all(b"\n")?;
  70. self.make_writer.make_writer().write_all(&buffer)
  71. }
  72. }
  73. /// The type of record we are dealing with: entering a span, exiting a span, an
  74. /// event.
  75. #[derive(Clone, Debug)]
  76. pub enum Type {
  77. EnterSpan,
  78. ExitSpan,
  79. Event,
  80. }
  81. impl fmt::Display for Type {
  82. fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
  83. let repr = match self {
  84. Type::EnterSpan => "START",
  85. Type::ExitSpan => "END",
  86. Type::Event => "EVENT",
  87. };
  88. write!(f, "{}", repr)
  89. }
  90. }
  91. fn format_span_context<S: Subscriber + for<'a> tracing_subscriber::registry::LookupSpan<'a>>(
  92. span: &SpanRef<S>,
  93. ty: Type,
  94. ) -> String {
  95. format!("[⛳ {} - {}]", span.metadata().name().to_uppercase(), ty)
  96. }
  97. fn format_event_message<S: Subscriber + for<'a> tracing_subscriber::registry::LookupSpan<'a>>(
  98. current_span: &Option<SpanRef<S>>,
  99. event: &Event,
  100. event_visitor: &JsonStorage<'_>,
  101. ) -> String {
  102. // Extract the "message" field, if provided. Fallback to the target, if missing.
  103. let mut message = event_visitor
  104. .values()
  105. .get("message")
  106. .and_then(|v| match v {
  107. Value::String(s) => Some(s.as_str()),
  108. _ => None,
  109. })
  110. .unwrap_or_else(|| event.metadata().target())
  111. .to_owned();
  112. // If the event is in the context of a span, prepend the span name to the
  113. // message.
  114. if let Some(span) = &current_span {
  115. message = format!("{} {}", format_span_context(span, Type::Event), message);
  116. }
  117. message
  118. }
  119. impl<S, W> Layer<S> for FlowyFormattingLayer<W>
  120. where
  121. S: Subscriber + for<'a> tracing_subscriber::registry::LookupSpan<'a>,
  122. W: MakeWriter + 'static,
  123. {
  124. fn on_event(&self, event: &Event<'_>, ctx: Context<'_, S>) {
  125. // Events do not necessarily happen in the context of a span, hence
  126. // lookup_current returns an `Option<SpanRef<_>>` instead of a
  127. // `SpanRef<_>`.
  128. let current_span = ctx.lookup_current();
  129. let mut event_visitor = JsonStorage::default();
  130. event.record(&mut event_visitor);
  131. // Opting for a closure to use the ? operator and get more linear code.
  132. let format = || {
  133. let mut buffer = Vec::new();
  134. let mut serializer = serde_json::Serializer::new(&mut buffer);
  135. let mut map_serializer = serializer.serialize_map(None)?;
  136. let message = format_event_message(&current_span, event, &event_visitor);
  137. self.serialize_flowy_folder_fields(
  138. &mut map_serializer,
  139. &message,
  140. event.metadata().level(),
  141. )?;
  142. // Additional metadata useful for debugging
  143. // They should be nested under `src` (see https://github.com/trentm/node-bunyan#src )
  144. // but `tracing` does not support nested values yet
  145. if self.with_target {
  146. map_serializer.serialize_entry("target", event.metadata().target())?;
  147. }
  148. // map_serializer.serialize_entry("line", &event.metadata().line())?;
  149. // map_serializer.serialize_entry("file", &event.metadata().file())?;
  150. // Add all the other fields associated with the event, expect the message we
  151. // already used.
  152. for (key, value) in event_visitor.values().iter().filter(|(&key, _)| {
  153. key != "message" && !FLOWY_RESERVED_FIELDS.contains(&key) && !IGNORE_FIELDS.contains(&key)
  154. }) {
  155. map_serializer.serialize_entry(key, value)?;
  156. }
  157. // Add all the fields from the current span, if we have one.
  158. if let Some(span) = &current_span {
  159. let extensions = span.extensions();
  160. if let Some(visitor) = extensions.get::<JsonStorage>() {
  161. for (key, value) in visitor.values() {
  162. if !FLOWY_RESERVED_FIELDS.contains(key) && !IGNORE_FIELDS.contains(key) {
  163. map_serializer.serialize_entry(key, value)?;
  164. } else {
  165. tracing::debug!(
  166. "{} is a reserved field in the flowy log format. Skipping it.",
  167. key
  168. );
  169. }
  170. }
  171. }
  172. }
  173. map_serializer.end()?;
  174. Ok(buffer)
  175. };
  176. let result: std::io::Result<Vec<u8>> = format();
  177. if let Ok(formatted) = result {
  178. let _ = self.emit(formatted);
  179. }
  180. }
  181. fn new_span(&self, _attrs: &Attributes, id: &Id, ctx: Context<'_, S>) {
  182. let span = ctx.span(id).expect("Span not found, this is a bug");
  183. if let Ok(serialized) = self.serialize_span(&span, Type::EnterSpan) {
  184. let _ = self.emit(serialized);
  185. }
  186. }
  187. fn on_close(&self, id: Id, ctx: Context<'_, S>) {
  188. let span = ctx.span(&id).expect("Span not found, this is a bug");
  189. if let Ok(serialized) = self.serialize_span(&span, Type::ExitSpan) {
  190. let _ = self.emit(serialized);
  191. }
  192. }
  193. }