openstack_cli/image/v2/metadef/namespace/object/
set.rsuse clap::Args;
use serde::{Deserialize, Serialize};
use tracing::info;
use openstack_sdk::AsyncOpenStack;
use crate::output::OutputProcessor;
use crate::Cli;
use crate::OpenStackCliError;
use crate::OutputConfig;
use crate::StructTable;
use crate::common::parse_key_val;
use openstack_sdk::api::image::v2::metadef::namespace::object::set;
use openstack_sdk::api::QueryAsync;
use serde_json::Value;
use std::fmt;
use structable_derive::StructTable;
#[derive(Args)]
pub struct ObjectCommand {
#[command(flatten)]
query: QueryParameters,
#[command(flatten)]
path: PathParameters,
#[arg(help_heading = "Body parameters", long)]
description: Option<String>,
#[arg(help_heading = "Body parameters", long)]
name: String,
#[arg(help_heading = "Body parameters", long, value_name="key=value", value_parser=parse_key_val::<String, Value>)]
properties: Option<Vec<(String, Value)>>,
#[arg(action=clap::ArgAction::Append, help_heading = "Body parameters", long)]
required: Option<Vec<String>>,
}
#[derive(Args)]
struct QueryParameters {}
#[derive(Args)]
struct PathParameters {
#[arg(
help_heading = "Path parameters",
id = "path_param_namespace_name",
value_name = "NAMESPACE_NAME"
)]
namespace_name: String,
#[arg(
help_heading = "Path parameters",
id = "path_param_object_name",
value_name = "OBJECT_NAME"
)]
object_name: String,
}
#[derive(Deserialize, Serialize, Clone, StructTable)]
struct ResponseData {
#[serde()]
#[structable(optional)]
created_at: Option<String>,
#[serde()]
#[structable(optional)]
description: Option<String>,
#[serde()]
#[structable()]
name: String,
#[serde()]
#[structable(optional, pretty)]
properties: Option<Value>,
#[serde()]
#[structable(optional, pretty)]
required: Option<Value>,
#[serde()]
#[structable(optional)]
schema: Option<String>,
#[serde(rename = "self")]
#[structable(optional, title = "self")]
_self: Option<String>,
#[serde()]
#[structable(optional)]
updated_at: Option<String>,
}
#[derive(Default, Clone, Deserialize, Serialize)]
struct ResponseProperties {
additional_items: Option<bool>,
_default: Option<Value>,
description: Option<String>,
_enum: Option<Value>,
items: Option<Value>,
maximum: Option<f32>,
max_items: Option<i32>,
max_length: Option<i32>,
minimum: Option<f32>,
min_items: Option<i32>,
min_length: Option<i32>,
name: Option<String>,
operators: Option<Value>,
pattern: Option<String>,
readonly: Option<bool>,
required: Option<Value>,
title: String,
_type: String,
unique_items: Option<bool>,
}
impl fmt::Display for ResponseProperties {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let data = Vec::from([
format!(
"additional_items={}",
self.additional_items
.map_or(String::new(), |v| v.to_string())
),
format!(
"_default={}",
self._default
.clone()
.map_or(String::new(), |v| v.to_string())
),
format!(
"description={}",
self.description
.clone()
.map_or(String::new(), |v| v.to_string())
),
format!(
"_enum={}",
self._enum.clone().map_or(String::new(), |v| v.to_string())
),
format!(
"items={}",
self.items.clone().map_or(String::new(), |v| v.to_string())
),
format!(
"maximum={}",
self.maximum.map_or(String::new(), |v| v.to_string())
),
format!(
"max_items={}",
self.max_items.map_or(String::new(), |v| v.to_string())
),
format!(
"max_length={}",
self.max_length.map_or(String::new(), |v| v.to_string())
),
format!(
"minimum={}",
self.minimum.map_or(String::new(), |v| v.to_string())
),
format!(
"min_items={}",
self.min_items.map_or(String::new(), |v| v.to_string())
),
format!(
"min_length={}",
self.min_length.map_or(String::new(), |v| v.to_string())
),
format!(
"name={}",
self.name.clone().map_or(String::new(), |v| v.to_string())
),
format!(
"operators={}",
self.operators
.clone()
.map_or(String::new(), |v| v.to_string())
),
format!(
"pattern={}",
self.pattern
.clone()
.map_or(String::new(), |v| v.to_string())
),
format!(
"readonly={}",
self.readonly.map_or(String::new(), |v| v.to_string())
),
format!(
"required={}",
self.required
.clone()
.map_or(String::new(), |v| v.to_string())
),
format!("title={}", self.title),
format!("_type={}", self._type),
format!(
"unique_items={}",
self.unique_items.map_or(String::new(), |v| v.to_string())
),
]);
write!(f, "{}", data.join(";"))
}
}
impl ObjectCommand {
pub async fn take_action(
&self,
parsed_args: &Cli,
client: &mut AsyncOpenStack,
) -> Result<(), OpenStackCliError> {
info!("Set Object");
let op = OutputProcessor::from_args(parsed_args);
op.validate_args(parsed_args)?;
let mut ep_builder = set::Request::builder();
ep_builder.namespace_name(&self.path.namespace_name);
ep_builder.object_name(&self.path.object_name);
if let Some(arg) = &self.description {
ep_builder.description(arg);
}
ep_builder.name(&self.name);
if let Some(arg) = &self.properties {
ep_builder.properties(
arg.iter()
.map(|(k, v)| {
serde_json::from_value(v.to_owned()).map(|v: set::Properties| (k, v))
})
.collect::<Result<Vec<_>, _>>()?
.into_iter(),
);
}
if let Some(arg) = &self.required {
ep_builder.required(arg.iter().map(Into::into).collect::<Vec<_>>());
}
let ep = ep_builder
.build()
.map_err(|x| OpenStackCliError::EndpointBuild(x.to_string()))?;
let data = ep.query_async(client).await?;
op.output_single::<ResponseData>(data)?;
Ok(())
}
}