1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
use clap::{ArgMatches, Command};
use seaplane::api::{
metadata::v1::Key,
shared::v1::{Directory, RangeQueryContext},
};
use crate::{
api::MetadataReq,
cli::{cmds::metadata::common, CliCommand},
context::{Ctx, MetadataCtx},
error::{CliError, CliErrorKind, Result},
ops::metadata::KeyValues,
printer::{Output, OutputFormat},
};
static LONG_ABOUT: &str = "List one or more metadata key-value pairs
Keys and values will be displayed in base64 encoded format by default because they may contain
arbitrary binary data. Using --decode allows one to decode them and display the unencoded
values.";
#[derive(Copy, Clone, Debug)]
pub struct SeaplaneMetadataList;
impl SeaplaneMetadataList {
pub fn command() -> Command {
Command::new("list")
.visible_alias("ls")
.about("List one or more metadata key-value pairs")
.long_about(LONG_ABOUT)
.arg(
arg!(dir =["DIR"])
.help("The root directory of the metadata key-value pairs to list"),
)
.arg(common::base64())
.args(common::display_args())
.group(common::keys_or_values())
.arg(arg!(--from - ('f') =["KEY"]).help("Only print metadata key-value pairs after this key (note: if this key has a value it will be included in the results)"))
}
}
impl CliCommand for SeaplaneMetadataList {
fn run(&self, ctx: &mut Ctx) -> Result<()> {
let kvs = {
let mdctx = ctx.md_ctx.get_or_init();
let mut range = RangeQueryContext::new();
if let Some(dir) = &mdctx.directory {
range.set_directory(dir.clone());
}
if let Some(from) = &mdctx.from {
range.set_from(from.clone());
}
let mut req = MetadataReq::new(ctx)?;
req.set_dir(range)?;
KeyValues::from_model(req.get_all_pages()?)
};
match ctx.args.out_format {
OutputFormat::Json => kvs.print_json(ctx)?,
OutputFormat::Table => kvs.print_table(ctx)?,
}
Ok(())
}
fn update_ctx(&self, matches: &ArgMatches, ctx: &mut Ctx) -> Result<()> {
ctx.md_ctx.init(MetadataCtx::default());
ctx.args.out_format = matches.get_one("format").copied().unwrap_or_default();
let mut mdctx = ctx.md_ctx.get_mut().unwrap();
mdctx.base64 = matches.get_flag("base64");
mdctx.decode = matches.get_flag("decode");
mdctx.decode_safe = matches.get_flag("decode-safe");
mdctx.no_decode = matches.get_flag("no-decode");
mdctx.no_keys = matches.get_flag("only-values");
mdctx.no_values = matches.get_flag("only-keys");
mdctx.no_header = matches.get_flag("no-header");
mdctx.keys_width_limit = matches
.get_one::<usize>("keys-width-limit")
.copied()
.unwrap_or_default();
mdctx.values_width_limit = matches
.get_one::<usize>("values-width-limit")
.copied()
.unwrap_or_default();
mdctx.from =
maybe_base64_arg!(matches, "from", matches.get_flag("base64")).map(Key::from_encoded);
mdctx.directory = maybe_base64_arg!(matches, "dir", matches.get_flag("base64"))
.map(Directory::from_encoded);
if matches.get_flag("human-readable") && !(mdctx.decode || mdctx.no_decode) {
mdctx.decode_safe = true
};
if mdctx.decode && ctx.args.out_format != OutputFormat::Table {
let format_arg = format!("--format {}", ctx.args.out_format);
return Err(CliError::from(CliErrorKind::ConflictingArguments(
"--decode".to_owned(),
format_arg,
)));
}
Ok(())
}
}