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::cluster::create;
use openstack_types::container_infrastructure_management::v1::cluster::response::create::ClusterResponse;
use serde_json::Value;
#[derive(Args)]
#[command(about = "Create new cluster")]
pub struct ClusterCommand {
#[command(flatten)]
query: QueryParameters,
#[command(flatten)]
path: PathParameters,
#[arg(help_heading = "Body parameters", long)]
api_address: Option<String>,
#[arg(help_heading = "Body parameters", long)]
cluster_template_id: String,
#[arg(help_heading = "Body parameters", long)]
coe_version: Option<String>,
#[arg(help_heading = "Body parameters", long)]
container_version: Option<String>,
#[arg(help_heading = "Body parameters", long)]
create_timeout: Option<i32>,
#[arg(help_heading = "Body parameters", long)]
created_at: Option<String>,
#[arg(help_heading = "Body parameters", long)]
discovery_url: Option<String>,
#[arg(help_heading = "Body parameters", long)]
docker_volume_size: Option<i32>,
#[arg(help_heading = "Body parameters", long, value_name="key=value", value_parser=parse_key_val::<String, String>)]
faults: Option<Vec<(String, 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)]
health_status: Option<HealthStatus>,
#[arg(help_heading = "Body parameters", long, value_name="key=value", value_parser=parse_key_val::<String, String>)]
health_status_reason: Option<Vec<(String, String)>>,
#[arg(help_heading = "Body parameters", long)]
keypair: 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(help_heading = "Body parameters", long, value_name="key=value", value_parser=parse_key_val::<String, String>)]
labels_added: Option<Vec<(String, String)>>,
#[arg(help_heading = "Body parameters", long, value_name="key=value", value_parser=parse_key_val::<String, String>)]
labels_overridden: Option<Vec<(String, String)>>,
#[arg(help_heading = "Body parameters", long, value_name="key=value", value_parser=parse_key_val::<String, String>)]
labels_skipped: 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(action=clap::ArgAction::Append, help_heading = "Body parameters", long)]
master_addresses: Option<Vec<String>>,
#[arg(help_heading = "Body parameters", long)]
master_count: Option<i32>,
#[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)]
merge_labels: Option<String>,
#[arg(help_heading = "Body parameters", long)]
name: Option<String>,
#[arg(action=clap::ArgAction::Append, help_heading = "Body parameters", long)]
node_addresses: Option<Vec<String>>,
#[arg(help_heading = "Body parameters", long)]
node_count: Option<i32>,
#[arg(help_heading = "Body parameters", long)]
project_id: Option<String>,
#[arg(help_heading = "Body parameters", long)]
stack_id: Option<String>,
#[arg(help_heading = "Body parameters", long)]
status: Option<Status>,
#[arg(help_heading = "Body parameters", long)]
status_reason: 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>,
}
#[derive(Args)]
struct QueryParameters {}
#[derive(Args)]
struct PathParameters {}
#[derive(Clone, Eq, Ord, PartialEq, PartialOrd, ValueEnum)]
enum HealthStatus {
Healthy,
Unhealthy,
Unknown,
}
#[derive(Clone, Eq, Ord, PartialEq, PartialOrd, ValueEnum)]
enum Status {
AdoptComplete,
CheckComplete,
CreateComplete,
CreateFailed,
CreateInProgress,
DeleteComplete,
DeleteFailed,
DeleteInProgress,
RestoreComplete,
ResumeComplete,
ResumeFailed,
RollbackComplete,
RollbackFailed,
RollbackInProgress,
SnapshotComplete,
UpdateComplete,
UpdateFailed,
UpdateInProgress,
}
impl ClusterCommand {
pub async fn take_action(
&self,
parsed_args: &Cli,
client: &mut AsyncOpenStack,
) -> Result<(), OpenStackCliError> {
info!("Create Cluster");
let op = OutputProcessor::from_args(
parsed_args,
Some("container-infrastructure-management.cluster"),
Some("create"),
);
op.validate_args(parsed_args)?;
let mut ep_builder = create::Request::builder();
if let Some(arg) = &self.api_address {
ep_builder.api_address(arg);
}
ep_builder.cluster_template_id(&self.cluster_template_id);
if let Some(arg) = &self.coe_version {
ep_builder.coe_version(arg);
}
if let Some(arg) = &self.container_version {
ep_builder.container_version(arg);
}
if let Some(arg) = &self.create_timeout {
ep_builder.create_timeout(*arg);
}
if let Some(arg) = &self.created_at {
ep_builder.created_at(arg);
}
if let Some(arg) = &self.discovery_url {
ep_builder.discovery_url(arg);
}
if let Some(arg) = &self.docker_volume_size {
ep_builder.docker_volume_size(*arg);
}
if let Some(arg) = &self.faults {
ep_builder.faults(arg.iter().cloned());
}
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.health_status {
let tmp = match arg {
HealthStatus::Healthy => create::HealthStatus::Healthy,
HealthStatus::Unhealthy => create::HealthStatus::Unhealthy,
HealthStatus::Unknown => create::HealthStatus::Unknown,
};
ep_builder.health_status(tmp);
}
if let Some(arg) = &self.health_status_reason {
ep_builder.health_status_reason(arg.iter().cloned());
}
if let Some(arg) = &self.keypair {
ep_builder.keypair(arg);
}
if let Some(arg) = &self.labels {
ep_builder.labels(arg.iter().cloned());
}
if let Some(arg) = &self.labels_added {
ep_builder.labels_added(arg.iter().cloned());
}
if let Some(arg) = &self.labels_overridden {
ep_builder.labels_overridden(arg.iter().cloned());
}
if let Some(arg) = &self.labels_skipped {
ep_builder.labels_skipped(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_addresses {
ep_builder.master_addresses(arg.iter().map(Into::into).collect::<Vec<_>>());
}
if let Some(arg) = &self.master_count {
ep_builder.master_count(*arg);
}
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.merge_labels {
ep_builder.merge_labels(arg);
}
if let Some(arg) = &self.name {
ep_builder.name(arg);
}
if let Some(arg) = &self.node_addresses {
ep_builder.node_addresses(arg.iter().map(Into::into).collect::<Vec<_>>());
}
if let Some(arg) = &self.node_count {
ep_builder.node_count(*arg);
}
if let Some(arg) = &self.project_id {
ep_builder.project_id(arg);
}
if let Some(arg) = &self.stack_id {
ep_builder.stack_id(arg);
}
if let Some(arg) = &self.status {
let tmp = match arg {
Status::AdoptComplete => create::Status::AdoptComplete,
Status::CheckComplete => create::Status::CheckComplete,
Status::CreateComplete => create::Status::CreateComplete,
Status::CreateFailed => create::Status::CreateFailed,
Status::CreateInProgress => create::Status::CreateInProgress,
Status::DeleteComplete => create::Status::DeleteComplete,
Status::DeleteFailed => create::Status::DeleteFailed,
Status::DeleteInProgress => create::Status::DeleteInProgress,
Status::RestoreComplete => create::Status::RestoreComplete,
Status::ResumeComplete => create::Status::ResumeComplete,
Status::ResumeFailed => create::Status::ResumeFailed,
Status::RollbackComplete => create::Status::RollbackComplete,
Status::RollbackFailed => create::Status::RollbackFailed,
Status::RollbackInProgress => create::Status::RollbackInProgress,
Status::SnapshotComplete => create::Status::SnapshotComplete,
Status::UpdateComplete => create::Status::UpdateComplete,
Status::UpdateFailed => create::Status::UpdateFailed,
Status::UpdateInProgress => create::Status::UpdateInProgress,
};
ep_builder.status(tmp);
}
if let Some(arg) = &self.status_reason {
ep_builder.status_reason(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);
}
let ep = ep_builder
.build()
.map_err(|x| OpenStackCliError::EndpointBuild(x.to_string()))?;
let data = ep.query_async(client).await?;
op.output_single::<ClusterResponse>(data)?;
op.show_command_hint()?;
Ok(())
}
}