[pbs-devel] [REBASED backup 03/14] add node config

Dietmar Maurer dietmar at proxmox.com
Fri Apr 30 08:26:33 CEST 2021


Again, please can you include the suggested changes?

On 4/29/21 3:13 PM, Wolfgang Bumiller wrote:
> Signed-off-by: Wolfgang Bumiller <w.bumiller at proxmox.com>
> ---
>   src/config.rs      |   1 +
>   src/config/node.rs | 190 +++++++++++++++++++++++++++++++++++++++++++++
>   2 files changed, 191 insertions(+)
>   create mode 100644 src/config/node.rs
>
> diff --git a/src/config.rs b/src/config.rs
> index 83ea0461..94b7fb6c 100644
> --- a/src/config.rs
> +++ b/src/config.rs
> @@ -20,6 +20,7 @@ pub mod acme;
>   pub mod cached_user_info;
>   pub mod datastore;
>   pub mod network;
> +pub mod node;
>   pub mod remote;
>   pub mod sync;
>   pub mod tfa;
> diff --git a/src/config/node.rs b/src/config/node.rs
> new file mode 100644
> index 00000000..b6abeef3
> --- /dev/null
> +++ b/src/config/node.rs
> @@ -0,0 +1,190 @@
> +use std::fs::File;
> +use std::time::Duration;
> +
> +use anyhow::{format_err, Error};
> +use nix::sys::stat::Mode;
> +use serde::{Deserialize, Serialize};
> +
> +use proxmox::api::api;
> +use proxmox::api::schema::{self, Updater};
> +use proxmox::tools::fs::{replace_file, CreateOptions};
> +
> +use crate::acme::AcmeClient;
> +use crate::config::acme::{AccountName, AcmeDomain};
> +
> +const CONF_FILE: &str = configdir!("/node.cfg");
> +const LOCK_FILE: &str = configdir!("/.node.cfg.lock");
> +const LOCK_TIMEOUT: Duration = Duration::from_secs(5);
> +
> +pub fn read_lock() -> Result<File, Error> {
> +    proxmox::tools::fs::open_file_locked(LOCK_FILE, LOCK_TIMEOUT, false)
> +}
> +
> +pub fn write_lock() -> Result<File, Error> {
> +    proxmox::tools::fs::open_file_locked(LOCK_FILE, LOCK_TIMEOUT, true)
> +}
> +
> +/// Read the Node Config.
> +pub fn config() -> Result<(NodeConfig, [u8; 32]), Error> {
> +    let content =
> +        proxmox::tools::fs::file_read_optional_string(CONF_FILE)?.unwrap_or_else(|| "".to_string());
> +
> +    let digest = openssl::sha::sha256(content.as_bytes());
> +    let data: NodeConfig = crate::tools::config::from_str(&content, &NodeConfig::API_SCHEMA)?;
> +
> +    Ok((data, digest))
> +}
> +
> +/// Write the Node Config, requires the write lock to be held.
> +pub fn save_config(config: &NodeConfig) -> Result<(), Error> {
> +    let raw = crate::tools::config::to_bytes(config, &NodeConfig::API_SCHEMA)?;
> +
> +    let backup_user = crate::backup::backup_user()?;
> +    let options = CreateOptions::new()
> +        .perm(Mode::from_bits_truncate(0o0640))
> +        .owner(nix::unistd::ROOT)
> +        .group(backup_user.gid);
> +
> +    replace_file(CONF_FILE, &raw, options)
> +}
> +
> +#[api(
> +    properties: {
> +        account: { type: AccountName },
> +    }
> +)]
> +#[derive(Deserialize, Serialize)]
> +/// The ACME configuration.
> +///
> +/// Currently only contains the name of the account use.
> +pub struct AcmeConfig {
> +    /// Account to use to acquire ACME certificates.
> +    account: AccountName,
> +}
> +
> +#[api(
> +    properties: {
> +        acme: {
> +            optional: true,
> +            type: String,
> +            format: &schema::ApiStringFormat::PropertyString(&AcmeConfig::API_SCHEMA),
> +        },
> +        acmedomain0: {
> +            type: String,
> +            optional: true,
> +            format: &schema::ApiStringFormat::PropertyString(&AcmeDomain::API_SCHEMA),
> +        },
> +        acmedomain1: {
> +            type: String,
> +            optional: true,
> +            format: &schema::ApiStringFormat::PropertyString(&AcmeDomain::API_SCHEMA),
> +        },
> +        acmedomain2: {
> +            type: String,
> +            optional: true,
> +            format: &schema::ApiStringFormat::PropertyString(&AcmeDomain::API_SCHEMA),
> +        },
> +        acmedomain3: {
> +            type: String,
> +            optional: true,
> +            format: &schema::ApiStringFormat::PropertyString(&AcmeDomain::API_SCHEMA),
> +        },
> +        acmedomain4: {
> +            type: String,
> +            optional: true,
> +            format: &schema::ApiStringFormat::PropertyString(&AcmeDomain::API_SCHEMA),
> +        },
> +    },
> +)]
> +#[derive(Deserialize, Serialize, Updater)]
> +/// Node specific configuration.
> +pub struct NodeConfig {
> +    /// The acme account to use on this node.
> +    #[serde(skip_serializing_if = "Updater::is_empty")]
> +    acme: Option<String>,
> +
> +    /// ACME domain to get a certificate for for this node.
> +    #[serde(skip_serializing_if = "Updater::is_empty")]
> +    acmedomain0: Option<String>,
> +
> +    /// ACME domain to get a certificate for for this node.
> +    #[serde(skip_serializing_if = "Updater::is_empty")]
> +    acmedomain1: Option<String>,
> +
> +    /// ACME domain to get a certificate for for this node.
> +    #[serde(skip_serializing_if = "Updater::is_empty")]
> +    acmedomain2: Option<String>,
> +
> +    /// ACME domain to get a certificate for for this node.
> +    #[serde(skip_serializing_if = "Updater::is_empty")]
> +    acmedomain3: Option<String>,
> +
> +    /// ACME domain to get a certificate for for this node.
> +    #[serde(skip_serializing_if = "Updater::is_empty")]
> +    acmedomain4: Option<String>,
> +}
> +
> +impl NodeConfig {
> +    pub fn acme_config(&self) -> Option<Result<AcmeConfig, Error>> {
> +        self.acme.as_deref().map(|config| -> Result<_, Error> {
> +            Ok(crate::tools::config::from_property_string(
> +                config,
> +                &AcmeConfig::API_SCHEMA,
> +            )?)
> +        })
> +    }
> +
> +    pub async fn acme_client(&self) -> Result<AcmeClient, Error> {
> +        AcmeClient::load(
> +            &self
> +                .acme_config()
> +                .ok_or_else(|| format_err!("no acme client configured"))??
> +                .account,
> +        )
> +        .await
> +    }
> +
> +    pub fn acme_domains(&self) -> AcmeDomainIter {
> +        AcmeDomainIter::new(self)
> +    }
> +}
> +
> +pub struct AcmeDomainIter<'a> {
> +    config: &'a NodeConfig,
> +    index: usize,
> +}
> +
> +impl<'a> AcmeDomainIter<'a> {
> +    fn new(config: &'a NodeConfig) -> Self {
> +        Self { config, index: 0 }
> +    }
> +}
> +
> +impl<'a> Iterator for AcmeDomainIter<'a> {
> +    type Item = Result<AcmeDomain, Error>;
> +
> +    fn next(&mut self) -> Option<Self::Item> {
> +        let domain = loop {
> +            let index = self.index;
> +            self.index += 1;
> +
> +            let domain = match index {
> +                0 => self.config.acmedomain0.as_deref(),
> +                1 => self.config.acmedomain1.as_deref(),
> +                2 => self.config.acmedomain2.as_deref(),
> +                3 => self.config.acmedomain3.as_deref(),
> +                4 => self.config.acmedomain4.as_deref(),
> +                _ => return None,
> +            };
> +
> +            if let Some(domain) = domain {
> +                break domain;
> +            }
> +        };
> +
> +        Some(crate::tools::config::from_property_string(
> +            domain,
> +            &AcmeDomain::API_SCHEMA,
> +        ))
> +    }
> +}





More information about the pbs-devel mailing list