qubit-spi 0.2.2

Typed service provider registry infrastructure for Rust
Documentation
/*******************************************************************************
 *
 *    Copyright (c) 2026 Haixing Hu.
 *
 *    SPDX-License-Identifier: Apache-2.0
 *
 *    Licensed under the Apache License, Version 2.0.
 *
 ******************************************************************************/
//! Provider metadata captured by registries.

use std::collections::HashSet;

use crate::{
    ProviderName,
    ProviderRegistryError,
};

/// Stable provider metadata used for registration and selection.
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct ProviderDescriptor {
    /// Canonical provider id.
    id: ProviderName,
    /// Additional accepted provider aliases.
    aliases: Vec<ProviderName>,
    /// Priority used by automatic selection.
    priority: i32,
}

impl ProviderDescriptor {
    /// Creates a descriptor with no aliases and zero priority.
    ///
    /// # Parameters
    /// - `id`: Provider id.
    ///
    /// # Returns
    /// Provider descriptor.
    ///
    /// # Errors
    /// Returns [`ProviderRegistryError`] when `id` is not a valid provider name.
    #[inline]
    pub fn new(id: &str) -> Result<Self, ProviderRegistryError> {
        Ok(Self {
            id: ProviderName::new(id)?,
            aliases: Vec::new(),
            priority: 0,
        })
    }

    /// Sets provider aliases.
    ///
    /// # Parameters
    /// - `aliases`: Provider aliases.
    ///
    /// # Returns
    /// Updated provider descriptor.
    ///
    /// # Errors
    /// Returns [`ProviderRegistryError`] when any alias is not a valid provider
    /// name, or when aliases duplicate the descriptor id or each other.
    #[inline]
    pub fn with_aliases(mut self, aliases: &[&str]) -> Result<Self, ProviderRegistryError> {
        let aliases = aliases
            .iter()
            .map(|alias| ProviderName::new(alias))
            .collect::<Result<Vec<_>, _>>()?;
        validate_unique_names(&self.id, &aliases)?;
        self.aliases = aliases;
        Ok(self)
    }

    /// Sets provider priority.
    ///
    /// # Parameters
    /// - `priority`: Provider priority. Larger values are preferred.
    ///
    /// # Returns
    /// Updated provider descriptor.
    #[inline]
    pub fn with_priority(mut self, priority: i32) -> Self {
        self.priority = priority;
        self
    }

    /// Gets the canonical provider id.
    ///
    /// # Returns
    /// Provider id.
    #[inline]
    pub fn id(&self) -> &ProviderName {
        &self.id
    }

    /// Gets provider aliases.
    ///
    /// # Returns
    /// Provider aliases.
    #[inline]
    pub fn aliases(&self) -> &[ProviderName] {
        &self.aliases
    }

    /// Gets provider aliases as string slices.
    ///
    /// # Returns
    /// Provider aliases as normalized string slices.
    #[inline]
    pub fn aliases_as_str(&self) -> Vec<&str> {
        self.aliases.iter().map(ProviderName::as_str).collect()
    }

    /// Gets provider priority.
    ///
    /// # Returns
    /// Provider priority.
    #[inline]
    pub fn priority(&self) -> i32 {
        self.priority
    }

    /// Iterates over the provider id followed by aliases.
    ///
    /// # Returns
    /// Iterator over all normalized names exposed by this descriptor.
    #[inline]
    pub(crate) fn names(&self) -> impl Iterator<Item = &ProviderName> {
        std::iter::once(&self.id).chain(self.aliases.iter())
    }
}

/// Validates that descriptor names are internally unique.
///
/// # Parameters
/// - `id`: Canonical provider id.
/// - `aliases`: Normalized provider aliases.
///
/// # Errors
/// Returns [`ProviderRegistryError::DuplicateProviderName`] when an alias
/// duplicates the id or another alias.
fn validate_unique_names(
    id: &ProviderName,
    aliases: &[ProviderName],
) -> Result<(), ProviderRegistryError> {
    let mut names = HashSet::with_capacity(aliases.len() + 1);
    names.insert(id.clone());
    for alias in aliases {
        if !names.insert(alias.clone()) {
            return Err(ProviderRegistryError::DuplicateProviderName {
                name: alias.clone(),
            });
        }
    }
    Ok(())
}