use bytes::Bytes;
use clap::Args;
use eyre::{WrapErr, eyre};
use http::Response;
use http::{HeaderName, HeaderValue};
use regex::Regex;
use serde::{Deserialize, Serialize};
use std::collections::HashMap;
use tracing::info;
use crate::Cli;
use crate::OpenStackCliError;
use crate::output::OutputProcessor;
use structable::{StructTable, StructTableOptions};
use openstack_sdk::{
AsyncOpenStack,
api::RestClient,
types::{ApiVersion, ServiceType},
};
use crate::common::HashMapStringString;
use crate::common::parse_key_val;
use openstack_sdk::api::RawQueryAsync;
use openstack_sdk::api::object_store::v1::account::head::Request as GetRequest;
use openstack_sdk::api::object_store::v1::account::set::Request;
#[derive(Args, Clone, Debug)]
pub struct AccountCommand {
#[arg(long, value_name="key=value", value_parser = parse_key_val::<String, String>)]
property: Vec<(String, String)>,
}
#[derive(Deserialize, Debug, Clone, Serialize, StructTable)]
pub struct Account {}
impl AccountCommand {
pub async fn take_action(
&self,
parsed_args: &Cli,
client: &mut AsyncOpenStack,
) -> Result<(), OpenStackCliError> {
info!("Post Account with {:?}", self);
let op = OutputProcessor::from_args(parsed_args, Some("object-store.account"), Some("set"));
op.validate_args(parsed_args)?;
let mut ep_builder = Request::builder();
let ep = client.get_service_endpoint(
&ServiceType::ObjectStore,
Some(ApiVersion::new(1, 0)).as_ref(),
)?;
let account = ep
.url()
.path_segments()
.ok_or_else(|| eyre!("Object Store endpoint must not point to a bare domain"))?
.rfind(|x| !x.is_empty());
if let Some(account) = account {
ep_builder.account(account);
}
ep_builder.headers(
self.property
.iter()
.map(|(k, v)| {
Ok::<(Option<HeaderName>, HeaderValue), eyre::Report>((
Some(
HeaderName::from_bytes(k.as_bytes())
.wrap_err_with(|| format!("{} cannot be used as header name", k))?,
),
HeaderValue::from_str(v.as_str()).wrap_err_with(|| {
format!("{} cannot be used as the header value", v)
})?,
))
})
.collect::<Result<Vec<(_, _)>, _>>()?
.into_iter(),
);
let ep = ep_builder
.build()
.map_err(|x| OpenStackCliError::EndpointBuild(x.to_string()))?;
let _rsp: Response<Bytes> = ep.raw_query_async(client).await?;
let mut ep_builder = GetRequest::builder();
if let Some(account) = account {
ep_builder.account(account);
}
let ep = ep_builder
.build()
.map_err(|x| OpenStackCliError::EndpointBuild(x.to_string()))?;
let rsp: Response<Bytes> = ep.raw_query_async(client).await?;
let mut metadata: HashMap<String, String> = HashMap::new();
let headers = rsp.headers();
let regexes: Vec<Regex> = vec![
Regex::new(r"(?i)X-Account-Meta-\.*").wrap_err("failed to compile the regex")?,
Regex::new(r"(?i)X-Account-Storage-Policy\.*Bytes-Used")
.wrap_err("failed to compile the regex")?,
Regex::new(r"(?i)X-Account-Storage-Policy\.*Container-Count")
.wrap_err("failed to compile the regex")?,
Regex::new(r"(?i)X-Account-Storage-Policy\.*Object-Count")
.wrap_err("failed to compile the regex")?,
];
for (hdr, val) in headers.iter() {
if [
"x-account-meta-temp-url-key",
"x-account-meta-temp-url-key-2",
"x-timestamp",
"x-account-bytes-used",
"x-account-container-count",
"x-account-object-count",
"x-account-meta-quota-bytes",
"x-account-access-control",
]
.contains(&hdr.as_str())
{
metadata.insert(
hdr.to_string(),
val.to_str().unwrap_or_default().to_string(),
);
} else if !regexes.is_empty() {
for rex in regexes.iter() {
if rex.is_match(hdr.as_str()) {
metadata.insert(
hdr.to_string(),
val.to_str().unwrap_or_default().to_string(),
);
}
}
}
}
let data = HashMapStringString(metadata);
op.output_single::<HashMapStringString>(serde_json::to_value(&data)?)?;
op.show_command_hint()?;
Ok(())
}
}