use clap::Args;
use tracing::info;
use openstack_sdk::AsyncOpenStack;
use crate::Cli;
use crate::OpenStackCliError;
use crate::output::OutputProcessor;
use crate::common::parse_key_val;
use clap::ValueEnum;
use openstack_sdk::api::QueryAsync;
use openstack_sdk::api::container_infrastructure_management::v1::clustertemplate::create;
use openstack_types::container_infrastructure_management::v1::clustertemplate::response::create::ClustertemplateResponse;
use serde_json::Value;
#[derive(Args)]
#[command(about = "Create new cluster template")]
pub struct ClustertemplateCommand {
#[command(flatten)]
query: QueryParameters,
#[command(flatten)]
path: PathParameters,
#[arg(help_heading = "Body parameters", long)]
apiserver_port: Option<i32>,
#[arg(help_heading = "Body parameters", long)]
cluster_distro: Option<String>,
#[arg(help_heading = "Body parameters", long)]
coe: Option<Coe>,
#[arg(help_heading = "Body parameters", long)]
created_at: Option<String>,
#[arg(help_heading = "Body parameters", long)]
dns_nameserver: Option<String>,
#[arg(help_heading = "Body parameters", long)]
docker_storage_driver: Option<String>,
#[arg(help_heading = "Body parameters", long)]
docker_volume_size: Option<i32>,
#[arg(help_heading = "Body parameters", long)]
driver: Option<String>,
#[arg(help_heading = "Body parameters", long)]
external_network_id: Option<String>,
#[arg(help_heading = "Body parameters", long)]
fixed_network: Option<String>,
#[arg(help_heading = "Body parameters", long)]
fixed_subnet: Option<String>,
#[arg(help_heading = "Body parameters", long)]
flavor_id: Option<String>,
#[arg(help_heading = "Body parameters", long)]
floating_ip_enabled: Option<String>,
#[arg(help_heading = "Body parameters", long)]
hidden: Option<String>,
#[arg(help_heading = "Body parameters", long)]
http_proxy: Option<String>,
#[arg(help_heading = "Body parameters", long)]
https_proxy: Option<String>,
#[arg(help_heading = "Body parameters", long)]
image_id: String,
#[arg(help_heading = "Body parameters", long)]
insecure_registry: Option<String>,
#[arg(help_heading = "Body parameters", long)]
keypair_id: Option<String>,
#[arg(help_heading = "Body parameters", long, value_name="key=value", value_parser=parse_key_val::<String, String>)]
labels: Option<Vec<(String, String)>>,
#[arg(action=clap::ArgAction::Append, help_heading = "Body parameters", long, value_name="JSON", value_parser=crate::common::parse_json)]
links: Option<Vec<Value>>,
#[arg(help_heading = "Body parameters", long)]
master_flavor_id: Option<String>,
#[arg(help_heading = "Body parameters", long)]
master_lb_enabled: Option<String>,
#[arg(help_heading = "Body parameters", long)]
name: Option<String>,
#[arg(help_heading = "Body parameters", long)]
network_driver: Option<String>,
#[arg(help_heading = "Body parameters", long)]
no_proxy: Option<String>,
#[arg(help_heading = "Body parameters", long)]
project_id: Option<String>,
#[arg(help_heading = "Body parameters", long)]
public: Option<String>,
#[arg(help_heading = "Body parameters", long)]
registry_enabled: Option<String>,
#[arg(help_heading = "Body parameters", long)]
server_type: Option<ServerType>,
#[arg(help_heading = "Body parameters", long)]
tags: Option<String>,
#[arg(help_heading = "Body parameters", long)]
tls_disabled: Option<String>,
#[arg(help_heading = "Body parameters", long)]
updated_at: Option<String>,
#[arg(help_heading = "Body parameters", long)]
user_id: Option<String>,
#[arg(help_heading = "Body parameters", long)]
uuid: Option<String>,
#[arg(help_heading = "Body parameters", long)]
volume_driver: Option<String>,
}
#[derive(Args)]
struct QueryParameters {}
#[derive(Args)]
struct PathParameters {}
#[derive(Clone, Eq, Ord, PartialEq, PartialOrd, ValueEnum)]
enum Coe {
Kubernetes,
}
#[derive(Clone, Eq, Ord, PartialEq, PartialOrd, ValueEnum)]
enum ServerType {
Bm,
Vm,
}
impl ClustertemplateCommand {
pub async fn take_action(
&self,
parsed_args: &Cli,
client: &mut AsyncOpenStack,
) -> Result<(), OpenStackCliError> {
info!("Create Clustertemplate");
let op = OutputProcessor::from_args(
parsed_args,
Some("container-infrastructure-management.clustertemplate"),
Some("create"),
);
op.validate_args(parsed_args)?;
let mut ep_builder = create::Request::builder();
if let Some(arg) = &self.apiserver_port {
ep_builder.apiserver_port(*arg);
}
if let Some(arg) = &self.cluster_distro {
ep_builder.cluster_distro(arg);
}
if let Some(arg) = &self.coe {
let tmp = match arg {
Coe::Kubernetes => create::Coe::Kubernetes,
};
ep_builder.coe(tmp);
}
if let Some(arg) = &self.created_at {
ep_builder.created_at(arg);
}
if let Some(arg) = &self.dns_nameserver {
ep_builder.dns_nameserver(arg);
}
if let Some(arg) = &self.docker_storage_driver {
ep_builder.docker_storage_driver(arg);
}
if let Some(arg) = &self.docker_volume_size {
ep_builder.docker_volume_size(*arg);
}
if let Some(arg) = &self.driver {
ep_builder.driver(arg);
}
if let Some(arg) = &self.external_network_id {
ep_builder.external_network_id(arg);
}
if let Some(arg) = &self.fixed_network {
ep_builder.fixed_network(arg);
}
if let Some(arg) = &self.fixed_subnet {
ep_builder.fixed_subnet(arg);
}
if let Some(arg) = &self.flavor_id {
ep_builder.flavor_id(arg);
}
if let Some(arg) = &self.floating_ip_enabled {
ep_builder.floating_ip_enabled(arg);
}
if let Some(arg) = &self.hidden {
ep_builder.hidden(arg);
}
if let Some(arg) = &self.http_proxy {
ep_builder.http_proxy(arg);
}
if let Some(arg) = &self.https_proxy {
ep_builder.https_proxy(arg);
}
ep_builder.image_id(&self.image_id);
if let Some(arg) = &self.insecure_registry {
ep_builder.insecure_registry(arg);
}
if let Some(arg) = &self.keypair_id {
ep_builder.keypair_id(arg);
}
if let Some(arg) = &self.labels {
ep_builder.labels(arg.iter().cloned());
}
if let Some(arg) = &self.links {
let links_builder: Vec<create::Links> = arg
.iter()
.flat_map(|v| serde_json::from_value::<create::Links>(v.to_owned()))
.collect::<Vec<create::Links>>();
ep_builder.links(links_builder);
}
if let Some(arg) = &self.master_flavor_id {
ep_builder.master_flavor_id(arg);
}
if let Some(arg) = &self.master_lb_enabled {
ep_builder.master_lb_enabled(arg);
}
if let Some(arg) = &self.name {
ep_builder.name(arg);
}
if let Some(arg) = &self.network_driver {
ep_builder.network_driver(arg);
}
if let Some(arg) = &self.no_proxy {
ep_builder.no_proxy(arg);
}
if let Some(arg) = &self.project_id {
ep_builder.project_id(arg);
}
if let Some(arg) = &self.public {
ep_builder.public(arg);
}
if let Some(arg) = &self.registry_enabled {
ep_builder.registry_enabled(arg);
}
if let Some(arg) = &self.server_type {
let tmp = match arg {
ServerType::Bm => create::ServerType::Bm,
ServerType::Vm => create::ServerType::Vm,
};
ep_builder.server_type(tmp);
}
if let Some(arg) = &self.tags {
ep_builder.tags(arg);
}
if let Some(arg) = &self.tls_disabled {
ep_builder.tls_disabled(arg);
}
if let Some(arg) = &self.updated_at {
ep_builder.updated_at(arg);
}
if let Some(arg) = &self.user_id {
ep_builder.user_id(arg);
}
if let Some(arg) = &self.uuid {
ep_builder.uuid(arg);
}
if let Some(arg) = &self.volume_driver {
ep_builder.volume_driver(arg);
}
let ep = ep_builder
.build()
.map_err(|x| OpenStackCliError::EndpointBuild(x.to_string()))?;
let data = ep.query_async(client).await?;
op.output_single::<ClustertemplateResponse>(data)?;
op.show_command_hint()?;
Ok(())
}
}