diff options
author | Robin Krahl <robin.krahl@ireas.org> | 2020-09-05 13:18:50 +0200 |
---|---|---|
committer | Robin Krahl <robin.krahl@ireas.org> | 2020-09-05 13:34:27 +0200 |
commit | bb809992ad543ea4c0c31897fbf2d130394dd80e (patch) | |
tree | e22b26d7eb2236efa83bbc9de50065263a16c1c3 /src/output.rs | |
parent | 04b4262cdf4bbb4e2698d8ce51a261bf294a2da3 (diff) | |
download | nitrocli-bb809992ad543ea4c0c31897fbf2d130394dd80e.tar.gz nitrocli-bb809992ad543ea4c0c31897fbf2d130394dd80e.tar.bz2 |
Add json output format
This patch adds a new output format, JSON. It uses serde to serialize
the output types.
todo: man page, changelog
Diffstat (limited to 'src/output.rs')
-rw-r--r-- | src/output.rs | 35 |
1 files changed, 27 insertions, 8 deletions
diff --git a/src/output.rs b/src/output.rs index 7c85a82..37daf92 100644 --- a/src/output.rs +++ b/src/output.rs @@ -5,8 +5,11 @@ //! Defines data types that can be formatted in different output formats. +use std::collections; use std::fmt; +use anyhow::Context as _; + use crate::args; use crate::Context; @@ -33,17 +36,21 @@ pub trait Output { } /// A single object. -pub struct Value<T: fmt::Display>(T); +pub struct Value<T: fmt::Display + serde::Serialize> { + key: String, + value: T, +} /// A list of objects of the same type that is displayed as a table with a fallback message for an /// empty list. pub struct Table<T: TableItem> { + key: String, items: Vec<T>, empty_message: String, } /// A trait for objects that can be displayed in a table. -pub trait TableItem { +pub trait TableItem: serde::Serialize { /// Returns the column headers for this type of table items. fn headers() -> Vec<&'static str>; /// Returns the values of the column for this table item. @@ -56,23 +63,28 @@ pub struct TextObject { items: Vec<(usize, String, String)>, } -impl<T: fmt::Display> Value<T> { - pub fn new(value: T) -> Value<T> { - Value(value) +impl<T: fmt::Display + serde::Serialize> Value<T> { + pub fn new(key: impl Into<String>, value: T) -> Value<T> { + Value { + key: key.into(), + value, + } } } -impl<T: fmt::Display> Output for Value<T> { +impl<T: fmt::Display + serde::Serialize> Output for Value<T> { fn format(&self, format: args::OutputFormat) -> anyhow::Result<String> { match format { - args::OutputFormat::Text => Ok(self.0.to_string()), + args::OutputFormat::Json => get_json(&self.key, &self.value), + args::OutputFormat::Text => Ok(self.value.to_string()), } } } impl<T: TableItem> Table<T> { - pub fn new(empty_message: impl Into<String>) -> Table<T> { + pub fn new(key: impl Into<String>, empty_message: impl Into<String>) -> Table<T> { Table { + key: key.into(), items: Vec::new(), empty_message: empty_message.into(), } @@ -90,6 +102,7 @@ impl<T: TableItem> Table<T> { impl<T: TableItem> Output for Table<T> { fn format(&self, format: args::OutputFormat) -> anyhow::Result<String> { match format { + args::OutputFormat::Json => get_json(&self.key, &self.items), args::OutputFormat::Text => { if self.items.is_empty() { Ok(self.empty_message.clone()) @@ -173,3 +186,9 @@ impl fmt::Display for TextObject { Ok(()) } } + +fn get_json<T: serde::Serialize + ?Sized>(key: &str, value: &T) -> anyhow::Result<String> { + let mut map = collections::HashMap::new(); + let _ = map.insert(key, value); + serde_json::to_string_pretty(&map).context("Could not serialize output to JSON") +} |