moon_config 2.0.13

Core workspace, project, and moon configuration.
Documentation
use crate::shapes::GlobPath;
use crate::{config_enum, config_struct, is_false};
use indexmap::IndexMap;
use rustc_hash::FxHashMap;
use schematic::{Config, PathSegment, ValidateError};

config_enum!(
    /// A mapping of file paths and file globs to owners.
    #[derive(Config)]
    #[serde(untagged)]
    pub enum OwnersPaths {
        /// A list of file paths and glob patterns. The owner of these
        /// is the project-level `defaultOwner`.
        #[setting(default)]
        List(Vec<GlobPath>),

        /// A map of file paths and glob patterns to owners.
        Map(IndexMap<GlobPath, Vec<String>>),
    }
);

impl OwnersPaths {
    pub fn is_empty(&self) -> bool {
        match self {
            OwnersPaths::List(list) => list.is_empty(),
            OwnersPaths::Map(map) => map.is_empty(),
        }
    }
}

fn validate_paths<C>(
    value: &PartialOwnersPaths,
    data: &PartialOwnersConfig,
    _context: &C,
    _finalize: bool,
) -> Result<(), ValidateError> {
    match value {
        PartialOwnersPaths::List(list) => {
            if !list.is_empty() && data.default_owner.is_none() {
                return Err(ValidateError::new(
                    "a default owner is required when defining a list of paths",
                ));
            }
        }
        PartialOwnersPaths::Map(map) => {
            for (key, value) in map {
                if value.is_empty() && data.default_owner.is_none() {
                    return Err(ValidateError::with_segment(
                        "a default owner is required when defining an empty list of owners",
                        PathSegment::Key(key.to_string()),
                    ));
                }
            }
        }
    };

    Ok(())
}

config_struct!(
    /// Defines ownership of source code within the current project, by mapping
    /// file paths and glob patterns to owners. An owner is either a user, team, or group.
    #[derive(Config)]
    pub struct OwnersConfig {
        /// Bitbucket only. A map of custom groups (prefixed with `@@@`),
        /// to a list of user and normal groups.
        #[serde(default, skip_serializing_if = "FxHashMap::is_empty")]
        pub custom_groups: FxHashMap<String, Vec<String>>,

        /// The default owner for `paths`.
        #[serde(default, skip_serializing_if = "Option::is_none")]
        pub default_owner: Option<String>,

        /// GitLab only. Marks the code owners section as optional.
        #[serde(default, skip_serializing_if = "is_false")]
        pub optional: bool,

        /// A list or map of file paths and glob patterns to owners.
        /// When a list, the `defaultOwner` is the owner, and each item is a path.
        /// When a map, the key is a path, and the value is a list of owners.
        #[setting(nested, validate = validate_paths)]
        pub paths: OwnersPaths,

        /// Bitbucket and GitLab only. The number of approvals required for the
        /// request to be satisfied. For Bitbucket, utilizes the `Check()` condition.
        /// For GitLab, marks the code owners section as required.
        #[serde(default, skip_serializing_if = "Option::is_none")]
        pub required_approvals: Option<u8>,
    }
);