use diagnostics_data::{InspectData, InspectHandleName, InspectMetadata};
use diagnostics_hierarchy::{
ArrayContent, DiagnosticsHierarchy, ExponentialHistogram, LinearHistogram, Property,
};
use either::Either;
use nom::HexDisplay;
use num_traits::{Bounded, Zero};
use std::fmt;
use std::ops::{Add, AddAssign, Mul, MulAssign};
const INDENT: usize = 2;
const HEX_DISPLAY_CHUNK_SIZE: usize = 16;
pub fn format<W>(w: &mut W, path: &str, diagnostics_hierarchy: DiagnosticsHierarchy) -> fmt::Result
where
W: fmt::Write,
{
writeln!(w, "{}:", path)?;
output_hierarchy(w, &diagnostics_hierarchy, 1)
}
pub fn output_schema<W>(w: &mut W, schema: &InspectData) -> fmt::Result
where
W: fmt::Write,
{
let InspectData {
moniker,
metadata: InspectMetadata { errors, name, component_url, timestamp, escrowed },
payload,
data_source: _,
version: _,
} = schema;
writeln!(w, "{moniker}:")?;
writeln!(w, " metadata:")?;
if let Some(errors) = &errors {
let errors = errors.join(", ");
writeln!(w, " errors = {errors}")?;
}
match &name {
InspectHandleName::Filename(f) => {
writeln!(w, " filename = {f}")?;
}
InspectHandleName::Name(n) => {
writeln!(w, " name = {n}")?;
}
}
writeln!(w, " component_url = {component_url}")?;
writeln!(w, " timestamp = {}", timestamp.into_nanos())?;
if *escrowed {
writeln!(w, " escrowed = true")?;
}
match &payload {
Some(hierarchy) => {
writeln!(w, " payload:")?;
output_hierarchy(w, hierarchy, 2)
}
None => writeln!(w, " payload: null"),
}
}
fn output_hierarchy<W>(
w: &mut W,
diagnostics_hierarchy: &DiagnosticsHierarchy,
indent: usize,
) -> fmt::Result
where
W: fmt::Write,
{
let name_indent = " ".repeat(INDENT * indent);
let value_indent = " ".repeat(INDENT * (indent + 1));
let array_broken_line_indent = " ".repeat(INDENT * (indent + 2));
writeln!(w, "{}{}:", name_indent, diagnostics_hierarchy.name)?;
for property in &diagnostics_hierarchy.properties {
match property {
Property::String(name, value) => writeln!(w, "{}{} = {}", value_indent, name, value)?,
Property::Int(name, value) => writeln!(w, "{}{} = {}", value_indent, name, value)?,
Property::Uint(name, value) => writeln!(w, "{}{} = {}", value_indent, name, value)?,
Property::Double(name, value) => {
writeln!(w, "{}{} = {:.6}", value_indent, name, value)?
}
Property::Bytes(name, array) => {
let byte_str = array.to_hex(HEX_DISPLAY_CHUNK_SIZE);
writeln!(w, "{}{} = Binary:\n{}", value_indent, name, byte_str.trim())?;
}
Property::Bool(name, value) => writeln!(w, "{}{} = {}", value_indent, name, value)?,
Property::IntArray(name, array) => output_array(w, &value_indent, name, array)?,
Property::UintArray(name, array) => output_array(w, &value_indent, name, array)?,
Property::DoubleArray(name, array) => output_array(w, &value_indent, name, array)?,
Property::StringList(name, list) => {
let max_line_length = 100;
let length_of_brackets = 2;
let length_of_comma_space = 2;
let total_len = list
.iter()
.fold(length_of_brackets, |acc, v| acc + v.len() + length_of_comma_space);
if total_len < max_line_length {
writeln!(w, "{}{} = {:?}", value_indent, name, list)?;
} else {
writeln!(w, "{}{} = [", value_indent, name)?;
for v in list {
writeln!(w, r#"{}"{}","#, array_broken_line_indent, v)?;
}
writeln!(w, "{}]", value_indent)?;
}
}
}
}
for child in &diagnostics_hierarchy.children {
output_hierarchy(w, child, indent + 1)?;
}
Ok(())
}
trait FromUsize {
fn from_usize(n: usize) -> Self;
}
impl FromUsize for i64 {
fn from_usize(n: usize) -> Self {
i64::try_from(n).unwrap()
}
}
impl FromUsize for u64 {
fn from_usize(n: usize) -> Self {
u64::try_from(n).unwrap()
}
}
impl FromUsize for f64 {
fn from_usize(n: usize) -> Self {
u64::try_from(n).unwrap() as f64
}
}
trait ExponentialBucketBound {
fn bound(floor: Self, initial_step: Self, step_multiplier: Self, index: usize) -> Self;
}
impl ExponentialBucketBound for i64 {
fn bound(floor: Self, initial_step: Self, step_multiplier: Self, index: usize) -> Self {
floor + initial_step * i64::pow(step_multiplier, index as u32)
}
}
impl ExponentialBucketBound for u64 {
fn bound(floor: Self, initial_step: Self, step_multiplier: Self, index: usize) -> Self {
floor + initial_step * u64::pow(step_multiplier, index as u32)
}
}
impl ExponentialBucketBound for f64 {
fn bound(floor: Self, initial_step: Self, step_multiplier: Self, index: usize) -> Self {
floor + initial_step * f64::powi(step_multiplier, (index as u64) as i32)
}
}
struct ExponentialBucketBoundsArgs<T> {
floor: T,
initial_step: T,
step_multiplier: T,
index: usize,
size: usize,
}
fn exponential_bucket_bounds<T>(args: ExponentialBucketBoundsArgs<T>) -> (T, T)
where
T: Bounded + Add<Output = T> + ExponentialBucketBound + Copy,
{
let ExponentialBucketBoundsArgs { floor, initial_step, step_multiplier, index, size } = args;
match index {
0 => (T::min_value(), floor),
1 if size == 2 => (floor, T::max_value()),
1 => (floor, floor + initial_step),
index if index == size - 1 => {
(T::bound(floor, initial_step, step_multiplier, index - 2), T::max_value())
}
_ => (
T::bound(floor, initial_step, step_multiplier, index - 2),
T::bound(floor, initial_step, step_multiplier, index - 1),
),
}
}
struct LinearBucketBoundsArgs<T> {
floor: T,
step: T,
index: usize,
size: usize,
}
fn linear_bucket_bounds<T>(args: LinearBucketBoundsArgs<T>) -> (T, T)
where
T: Bounded + Add<Output = T> + Mul<Output = T> + FromUsize + Copy,
{
let LinearBucketBoundsArgs { floor, step, index, size } = args;
match index {
0 => (T::min_value(), floor),
index if index == size - 1 => (floor + step * T::from_usize(index - 1), T::max_value()),
_ => (floor + step * T::from_usize(index - 1), floor + step * T::from_usize(index)),
}
}
struct OutputHistogramArgs<'a, T, F> {
indent: &'a str,
name: &'a str,
histogram_type: &'a str,
counts: &'a [T],
indexes: Option<&'a [usize]>,
size: usize,
bound_calculator: F,
}
fn output_histogram<T, F, W>(w: &mut W, args: OutputHistogramArgs<'_, T, F>) -> fmt::Result
where
W: fmt::Write,
T: NumberFormat + fmt::Display + PartialOrd + Zero,
F: Fn(usize) -> (T, T),
{
let OutputHistogramArgs {
indent,
name,
histogram_type,
counts,
indexes,
size,
bound_calculator,
} = args;
let value_indent = format!("{}{}", indent, " ".repeat(INDENT));
write!(
w,
"{indent}{name}:\n\
{value_indent}type = {histogram_type}\n\
{value_indent}size = {size}\n\
{value_indent}buckets = ["
)?;
let mut nonzero_counts = match indexes {
None => Either::Left(counts.iter().enumerate().filter(|(_, count)| **count > T::zero())),
Some(indexes) => Either::Right(
indexes
.iter()
.zip(counts)
.filter(|(_, count)| **count > T::zero())
.map(|(u, t)| (*u, t)),
),
}
.peekable();
while let Some((index, count)) = nonzero_counts.next() {
let (low_bound, high_bound) = bound_calculator(index);
write!(w, "[{},{})={}", low_bound.format(), high_bound.format(), count)?;
if nonzero_counts.peek().is_some() {
write!(w, ", ")?;
}
}
writeln!(w, "]")
}
fn output_array<T, W>(
w: &mut W,
value_indent: &str,
name: &str,
array: &ArrayContent<T>,
) -> fmt::Result
where
W: fmt::Write,
T: AddAssign
+ MulAssign
+ Copy
+ Add<Output = T>
+ fmt::Display
+ NumberFormat
+ Bounded
+ Mul<Output = T>
+ ExponentialBucketBound
+ FromUsize
+ PartialOrd
+ Zero,
{
match array {
ArrayContent::Values(values) => {
write!(w, "{value_indent}{name} = [")?;
for (i, value) in values.iter().enumerate() {
write!(w, "{value}")?;
if i < values.len() - 1 {
w.write_str(", ")?;
}
}
writeln!(w, "]")
}
ArrayContent::LinearHistogram(LinearHistogram { floor, step, counts, indexes, size }) => {
let bucket_bounder = |index| {
linear_bucket_bounds::<T>(LinearBucketBoundsArgs {
floor: *floor,
step: *step,
index,
size: *size,
})
};
output_histogram(
w,
OutputHistogramArgs {
indent: value_indent,
name,
histogram_type: "linear",
counts,
indexes: indexes.as_deref(),
size: *size,
bound_calculator: bucket_bounder,
},
)
}
ArrayContent::ExponentialHistogram(ExponentialHistogram {
floor,
initial_step,
step_multiplier,
counts,
indexes,
size,
}) => {
let bucket_bounder = |index| {
exponential_bucket_bounds::<T>(ExponentialBucketBoundsArgs {
floor: *floor,
initial_step: *initial_step,
step_multiplier: *step_multiplier,
index,
size: *size,
})
};
output_histogram(
w,
OutputHistogramArgs {
indent: value_indent,
name,
histogram_type: "exponential",
counts,
indexes: indexes.as_deref(),
size: *size,
bound_calculator: bucket_bounder,
},
)
}
}
}
trait NumberFormat {
fn format(&self) -> String;
}
impl NumberFormat for i64 {
fn format(&self) -> String {
match *self {
i64::MAX => "<max>".to_string(),
std::i64::MIN => "<min>".to_string(),
x => format!("{}", x),
}
}
}
impl NumberFormat for u64 {
fn format(&self) -> String {
match *self {
u64::MAX => "<max>".to_string(),
x => format!("{}", x),
}
}
}
impl NumberFormat for usize {
fn format(&self) -> String {
match *self {
std::usize::MAX => "<max>".to_string(),
x => format!("{}", x),
}
}
}
impl NumberFormat for f64 {
fn format(&self) -> String {
if *self == f64::MAX || *self == f64::INFINITY {
"inf".to_string()
} else if *self == f64::MIN || *self == f64::NEG_INFINITY {
"-inf".to_string()
} else {
format!("{}", self)
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use diagnostics_data::{InspectDataBuilder, Timestamp};
use test_case::test_case;
#[fuchsia::test]
fn test_array_line_breaks() {
let h = DiagnosticsHierarchy {
name: "test".into(),
properties: vec![Property::StringList(
"short_array".to_owned(),
vec!["short".into(), "array".into()],
)],
children: vec![],
missing: vec![],
};
let mut buf = String::new();
output_hierarchy(&mut buf, &h, 2).unwrap();
let expected = r#" test:
short_array = ["short", "array"]
"#;
assert_eq!(expected, buf);
let long_array = vec![
"12345678".into(),
"12345678".into(),
"12345678".into(),
"12345678".into(),
"12345678".into(),
"12345678".into(),
"12345678".into(),
"12345678".into(),
"12345678".into(),
"12345678".into(),
"12345678".into(),
"12345678".into(),
"12345678".into(),
];
let h = DiagnosticsHierarchy {
name: "test".into(),
properties: vec![Property::StringList("long_array".to_owned(), long_array)],
children: vec![],
missing: vec![],
};
let mut buf = String::new();
output_hierarchy(&mut buf, &h, 2).unwrap();
let expected = r#" test:
long_array = [
"12345678",
"12345678",
"12345678",
"12345678",
"12345678",
"12345678",
"12345678",
"12345678",
"12345678",
"12345678",
"12345678",
"12345678",
"12345678",
]
"#;
assert_eq!(expected, buf);
}
#[test_case(LinearBucketBoundsArgs { floor: -10, step: 2, index: 0, size: 4 }, (i64::MIN, -10))]
#[test_case(LinearBucketBoundsArgs { floor: -10, step: 2, index: 1, size: 4 }, (-10, -8))]
#[test_case(LinearBucketBoundsArgs { floor: -10, step: 2, index: 2, size: 4 }, (-8, -6))]
#[test_case(LinearBucketBoundsArgs { floor: -10, step: 2, index: 3, size: 4 }, (-6, i64::MAX))]
#[fuchsia::test]
fn test_linear_bucket_bounds_i64(args: LinearBucketBoundsArgs<i64>, bounds: (i64, i64)) {
assert_eq!(linear_bucket_bounds(args), bounds);
}
#[test_case(LinearBucketBoundsArgs { floor: 0, step: 2, index: 0, size: 4 }, (0, 0))]
#[test_case(LinearBucketBoundsArgs { floor: 0, step: 2, index: 1, size: 4 }, (0, 2))]
#[test_case(LinearBucketBoundsArgs { floor: 0, step: 2, index: 2, size: 4 }, (2, 4))]
#[test_case(LinearBucketBoundsArgs { floor: 0, step: 2, index: 3, size: 4 }, (4, u64::MAX))]
#[test_case(LinearBucketBoundsArgs { floor: 1, step: 2, index: 0, size: 4 }, (0, 1))]
#[test_case(LinearBucketBoundsArgs { floor: 1, step: 2, index: 1, size: 4 }, (1, 3))]
#[test_case(LinearBucketBoundsArgs { floor: 1, step: 2, index: 2, size: 4 }, (3, 5))]
#[test_case(LinearBucketBoundsArgs { floor: 1, step: 2, index: 3, size: 4 }, (5, u64::MAX))]
#[fuchsia::test]
fn test_linear_bucket_bounds_u64(args: LinearBucketBoundsArgs<u64>, bounds: (u64, u64)) {
assert_eq!(linear_bucket_bounds(args), bounds);
}
#[test_case(
LinearBucketBoundsArgs { floor: -0.5, step: 0.5, index: 0, size: 4 },
(f64::MIN, -0.5)
)]
#[test_case(LinearBucketBoundsArgs { floor: -0.5, step: 0.5, index: 1, size: 4 }, (-0.5, 0.0))]
#[test_case(LinearBucketBoundsArgs { floor: -0.5, step: 0.5, index: 2, size: 4 }, (0.0, 0.5))]
#[test_case(
LinearBucketBoundsArgs { floor: -0.5, step: 0.5, index: 3, size: 4 },
(0.5, f64::MAX)
)]
#[fuchsia::test]
fn test_linear_bucket_bounds_f64(args: LinearBucketBoundsArgs<f64>, bounds: (f64, f64)) {
assert_eq!(linear_bucket_bounds(args), bounds);
}
#[test_case(LinearBucketBoundsArgs { floor: -10, step: 2, index: 0, size: 3 }, (i64::MIN, -10))]
#[test_case(LinearBucketBoundsArgs { floor: -10, step: 2, index: 1, size: 3 }, (-10, -8))]
#[test_case(LinearBucketBoundsArgs { floor: -10, step: 2, index: 2, size: 3 }, (-8, i64::MAX))]
#[test_case(LinearBucketBoundsArgs { floor: -10, step: 2, index: 0, size: 2 }, (i64::MIN, -10))]
#[test_case(LinearBucketBoundsArgs { floor: -10, step: 2, index: 1, size: 2 }, (-10, i64::MAX))]
#[fuchsia::test]
fn test_linear_small_bucket_bounds(args: LinearBucketBoundsArgs<i64>, bounds: (i64, i64)) {
assert_eq!(linear_bucket_bounds(args), bounds);
}
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 0,
size: 5 },
(i64::MIN, -10)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 1,
size: 5 },
(-10, -8)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 2,
size: 5
},
(-8, -4)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 3,
size: 5
},
(-4, 8)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 4,
size: 5
},
(8, i64::MAX)
)]
#[fuchsia::test]
fn test_exponential_bucket_bounds_i64(
args: ExponentialBucketBoundsArgs<i64>,
bounds: (i64, i64),
) {
assert_eq!(exponential_bucket_bounds(args), bounds);
}
#[test_case(
ExponentialBucketBoundsArgs {
floor: 0,
initial_step: 2,
step_multiplier: 3,
index: 0,
size: 5
},
(0, 0)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: 0,
initial_step: 2,
step_multiplier: 3,
index: 1,
size: 5
},
(0, 2)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: 0,
initial_step: 2,
step_multiplier: 3,
index: 2,
size: 5
},
(2, 6)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: 0,
initial_step: 2,
step_multiplier: 3,
index: 3,
size: 5
},
(6, 18)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: 0,
initial_step: 2,
step_multiplier: 3,
index: 4,
size: 5 },
(18, u64::MAX)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: 1,
initial_step: 2,
step_multiplier: 3,
index: 0,
size: 5
},
(0, 1)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: 1,
initial_step: 2,
step_multiplier: 3,
index: 1,
size: 5
},
(1, 3)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: 1,
initial_step: 2,
step_multiplier: 3,
index: 2,
size: 5
},
(3, 7)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: 1,
initial_step: 2,
step_multiplier: 3,
index: 3,
size: 5
},
(7, 19)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: 1,
initial_step: 2,
step_multiplier: 3,
index: 4,
size: 5
},
(19, u64::MAX)
)]
#[fuchsia::test]
fn test_exponential_bucket_bounds_u64(
args: ExponentialBucketBoundsArgs<u64>,
bounds: (u64, u64),
) {
assert_eq!(exponential_bucket_bounds(args), bounds);
}
#[test_case(
ExponentialBucketBoundsArgs {
floor: -0.5,
initial_step: 0.5,
step_multiplier: 3.0,
index: 0,
size: 5 },
(f64::MIN, -0.5)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -0.5,
initial_step: 0.5,
step_multiplier: 3.0,
index: 1,
size: 5 },
(-0.5, 0.0)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -0.5,
initial_step: 0.5,
step_multiplier: 3.0,
index: 2,
size: 5 },
(0.0, 1.0)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -0.5,
initial_step: 0.5,
step_multiplier: 3.0,
index: 3,
size: 5 },
(1.0, 4.0)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -0.5,
initial_step: 0.5,
step_multiplier: 3.0,
index: 4,
size: 5 },
(4.0, f64::MAX)
)]
#[fuchsia::test]
fn test_exponential_bucket_bounds_f64(
args: ExponentialBucketBoundsArgs<f64>,
bounds: (f64, f64),
) {
assert_eq!(exponential_bucket_bounds(args), bounds);
}
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 0,
size: 4
},
(i64::MIN, -10)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 1,
size: 4
},
(-10, -8)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 2,
size: 4
},
(-8, -4)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 3,
size: 4
},
(-4, i64::MAX)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 0,
size: 3
},
(i64::MIN, -10)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 1,
size: 3
},
(-10, -8)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 2,
size: 3
},
(-8, i64::MAX)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 0,
size: 2
},
(i64::MIN, -10)
)]
#[test_case(
ExponentialBucketBoundsArgs {
floor: -10,
initial_step: 2,
step_multiplier: 3,
index: 1,
size: 2
},
(-10, i64::MAX)
)]
#[fuchsia::test]
fn test_exponential_small_bucket_bounds(
args: ExponentialBucketBoundsArgs<i64>,
bounds: (i64, i64),
) {
assert_eq!(exponential_bucket_bounds(args), bounds);
}
#[fuchsia::test]
fn render_escrowed_data() {
let data = InspectDataBuilder::new(
"a/b/c/d".try_into().unwrap(),
"test-url",
Timestamp::from_nanos(123456i64),
)
.escrowed(true)
.build();
let mut buf = String::new();
output_schema(&mut buf, &data).unwrap();
let expected = r#"a/b/c/d:
metadata:
name = root
component_url = test-url
timestamp = 123456
escrowed = true
payload: null
"#;
assert_eq!(expected, buf);
}
}