diff options
| author | stuebinm | 2022-12-01 00:11:56 +0100 | 
|---|---|---|
| committer | stuebinm | 2022-12-01 00:11:56 +0100 | 
| commit | 23c6195d83d4cc09545468982dda92c549d4a877 (patch) | |
| tree | 651ef8763c6c7bf2c18c539c7313e250de827a83 | |
| parent | 955434a8c86cf8a2ae7e4d62e67441e281398d46 (diff) | |
fancy colours
Diffstat (limited to '')
| -rw-r--r-- | Cargo.lock | 2 | ||||
| -rw-r--r-- | Cargo.toml | 2 | ||||
| -rw-r--r-- | src/fancy.rs | 200 | ||||
| -rw-r--r-- | src/main.rs | 4 | 
4 files changed, 207 insertions, 1 deletions
| @@ -857,9 +857,11 @@ version = "0.1.0"  dependencies = [   "clap",   "miette", + "owo-colors",   "protobuf",   "protobuf-codegen",   "protobuf-json-mapping", + "protobuf-support",   "reqwest",   "tokio",  ] @@ -11,8 +11,10 @@ protobuf-codegen = "3.2"  [dependencies]  protobuf = "3.2"  protobuf-json-mapping = "3.2" +protobuf-support = "3.2"  reqwest = { version = "0.11", features = ["json", "rustls-tls"], default-features = false }  tokio = { version = "1", features = ["full"] }  clap = { version = "4", features = ["derive"] }  miette = { version = "5", features = ["fancy"] } +owo-colors = "3.5" diff --git a/src/fancy.rs b/src/fancy.rs new file mode 100644 index 0000000..477018f --- /dev/null +++ b/src/fancy.rs @@ -0,0 +1,200 @@ + +use owo_colors::Style; +use protobuf::reflect::*; +use protobuf::MessageDyn; +use protobuf::UnknownValueRef; +use std::fmt; +use std::fmt::Display; +use std::fmt::Write; +use owo_colors::OwoColorize; + +use protobuf_support::text_format::quote_bytes_to; + + +pub fn print_to_string_fancy(m: &dyn MessageDyn) -> String { +    let mut r = String::new(); +    print_to_internal(&MessageRef::from(m), &mut r, true, 0); +    r.to_string() +} + +fn print_str_to(s: &str, buf: &mut String) { +    // TODO: keep printable Unicode +    quote_bytes_to(s.as_bytes(), buf); +} + +fn do_indent(buf: &mut String, pretty: bool, indent: usize) { +    if pretty && indent > 0 { +        for _ in 0..indent { +            buf.push_str("  "); +        } +    } +} + +trait FieldName: fmt::Display {} +impl<'a> FieldName for &'a str {} +impl FieldName for u32 {} + +fn print_start_field<F: FieldName>( +    buf: &mut String, +    pretty: bool, +    indent: usize, +    first: &mut bool, +    field_name: F, +) { +    if !*first && !pretty { +        buf.push_str(" "); +    } +    do_indent(buf, pretty, indent); +    *first = false; +    if indent == 0 { +        write!(buf, "{}", field_name.bold().bright_red()).unwrap(); +    } else { +        write!(buf, "{}", field_name.bright_blue()).unwrap(); +    } +} + +fn print_end_field(buf: &mut String, pretty: bool) { +    if pretty { +        buf.push_str("\n"); +    } +} + + +struct Prefix (Style); +struct Suffix (Style); + +impl Display for Prefix { +    fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result { +        self.0.fmt_prefix(fmt) +    } +} +impl Display for Suffix { +    fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result { +        self.0.fmt_suffix(fmt) +    } +} + +fn print_field<F: FieldName>( +    buf: &mut String, +    pretty: bool, +    indent: usize, +    first: &mut bool, +    field_name: F, +    value: ReflectValueRef, +) { +    print_start_field(buf, pretty, indent, first, field_name); + +    match value { +        ReflectValueRef::Message(m) => { +            buf.push_str(" {"); +            if pretty { +                buf.push_str("\n"); +            } +            print_to_internal(&m, buf, pretty, indent + 1); +            do_indent(buf, pretty, indent); +            buf.push_str("}"); +        } +        ReflectValueRef::Enum(d, v) => { +            buf.push_str(": "); +            match d.value_by_number(v) { +                Some(e) => write!(buf, "{}", e.name().bright_yellow()).unwrap(), +                None => write!(buf, ": {}", v.bright_red()).unwrap(), +            } +        } +        ReflectValueRef::String(s) => { +            let style = Style::new().bright_cyan(); +            buf.push_str(": "); +            write!(buf, "{}", Prefix(style)).unwrap(); +            print_str_to(s, buf); +            write!(buf, "{}", Suffix(style)).unwrap(); +        } +        ReflectValueRef::Bytes(b) => { +            buf.push_str(": "); +            quote_bytes_to(b, buf); +        } +        ReflectValueRef::I32(v) => { +            write!(buf, ": {}", v.bright_yellow()).unwrap(); +        } +        ReflectValueRef::I64(v) => { +            write!(buf, ": {}", v.bright_yellow()).unwrap(); +        } +        ReflectValueRef::U32(v) => { +            write!(buf, ": {}", v.bright_yellow()).unwrap(); +        } +        ReflectValueRef::U64(v) => { +            write!(buf, ": {}", v.bright_yellow()).unwrap(); +        } +        ReflectValueRef::Bool(v) => { +            write!(buf, ": {}", v.bright_yellow()).unwrap(); +        } +        ReflectValueRef::F32(v) => { +            write!(buf, ": {}", v.bright_yellow()).unwrap(); +        } +        ReflectValueRef::F64(v) => { +            write!(buf, ": {}", v.bright_yellow()).unwrap(); +        } +    } + +    print_end_field(buf, pretty); +} + + +pub fn print_to_internal(m: &MessageRef, buf: &mut String, pretty: bool, indent: usize) { +    let d = m.descriptor_dyn(); +    let mut first = true; +    for f in d.fields() { +        match f.get_reflect(&**m) { +            ReflectFieldRef::Map(map) => { +                for (k, v) in &map { +                    print_start_field(buf, pretty, indent, &mut first, f.name()); +                    buf.push_str(" {"); +                    if pretty { +                        buf.push_str("\n"); +                    } + +                    let mut entry_first = true; + +                    print_field(buf, pretty, indent + 1, &mut entry_first, "key", k); +                    print_field(buf, pretty, indent + 1, &mut entry_first, "value", v); +                    do_indent(buf, pretty, indent); +                    buf.push_str("}"); +                    print_end_field(buf, pretty); +                } +            } +            ReflectFieldRef::Repeated(repeated) => { +                for v in repeated { +                    print_field(buf, pretty, indent, &mut first, f.name(), v); +                } +            } +            ReflectFieldRef::Optional(optional) => { +                if let Some(v) = optional.value() { +                    print_field(buf, pretty, indent, &mut first, f.name(), v); +                } +            } +        } +    } + +    let mut fields: Vec<(u32, UnknownValueRef)> = m.unknown_fields_dyn().iter().collect(); +    // Sort for stable output +    fields.sort_by_key(|(field_number, _)| *field_number); +    for (field_number, value) in fields { +        // TODO: try decode nested message for length-delimited +        print_field( +            buf, +            pretty, +            indent, +            &mut first, +            field_number, +            to_reflect_value_ref(&value), +        ); +    } +} + +fn to_reflect_value_ref<'o>(r: &'o UnknownValueRef<'o>) -> ReflectValueRef<'o> { +    match r { +        UnknownValueRef::Fixed32(v) => ReflectValueRef::U32(*v), +        UnknownValueRef::Fixed64(v) => ReflectValueRef::U64(*v), +        UnknownValueRef::Varint(v) => ReflectValueRef::U64(*v), +        UnknownValueRef::LengthDelimited(v) => ReflectValueRef::Bytes(v), +    } +} diff --git a/src/main.rs b/src/main.rs index feb6e9c..59e7e51 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,5 +1,6 @@  mod protos; +mod fancy;  use protos::protos::gtfs_realtime::FeedMessage;  use protobuf::Message; @@ -52,8 +53,9 @@ async fn main() -> miette::Result<()> {          true =>               println!("{}", protobuf_json_mapping::print_to_string(&proto).into_diagnostic()?),          false =>  -            println!("{}", protobuf::text_format::print_to_string_pretty(&proto)) +            println!("{}", fancy::print_to_string_fancy(&proto))      } +      Ok(())  } | 
