use anyhow::{Result, bail};
use dynamo_runtime::config::parse_bool;
use std::collections::HashSet;
#[derive(Debug, Clone, Default)]
pub struct NixlBackendConfig {
backends: HashSet<String>,
}
impl NixlBackendConfig {
pub fn new() -> Self {
Self::default()
}
pub fn from_env() -> Result<Self> {
let mut backends = HashSet::new();
for (key, value) in std::env::vars() {
if let Some(remainder) = key.strip_prefix("DYN_KVBM_NIXL_BACKEND_") {
if remainder.contains('_') {
bail!(
"Custom NIXL backend parameters are not yet supported. \
Found: {}. Please use only DYN_KVBM_NIXL_BACKEND_<backend>=true \
to enable backends with default parameters.",
key
);
}
let backend_name = remainder.to_uppercase();
match parse_bool(&value) {
Ok(true) => {
backends.insert(backend_name);
}
Ok(false) => {
continue;
}
Err(e) => bail!("Invalid value for {}: {}", key, e),
}
}
}
if backends.is_empty() {
backends.insert("UCX".to_string());
}
Ok(Self { backends })
}
pub fn with_backend(mut self, backend: impl Into<String>) -> Self {
self.backends.insert(backend.into().to_uppercase());
self
}
pub fn backends(&self) -> &HashSet<String> {
&self.backends
}
pub fn has_backend(&self, backend: &str) -> bool {
self.backends.contains(&backend.to_uppercase())
}
pub fn merge(mut self, other: NixlBackendConfig) -> Self {
self.backends.extend(other.backends);
self
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_new_config_is_empty() {
let config = NixlBackendConfig::new();
assert!(config.backends().is_empty());
}
#[test]
fn test_with_backend() {
let config = NixlBackendConfig::new()
.with_backend("ucx")
.with_backend("gds_mt");
assert!(config.has_backend("ucx"));
assert!(config.has_backend("UCX"));
assert!(config.has_backend("gds_mt"));
assert!(config.has_backend("GDS_MT"));
assert!(!config.has_backend("other"));
}
#[test]
fn test_merge_configs() {
let config1 = NixlBackendConfig::new().with_backend("ucx");
let config2 = NixlBackendConfig::new().with_backend("gds");
let merged = config1.merge(config2);
assert!(merged.has_backend("ucx"));
assert!(merged.has_backend("gds"));
}
#[test]
fn test_backend_name_case_insensitive() {
let config = NixlBackendConfig::new()
.with_backend("ucx")
.with_backend("Gds_mt")
.with_backend("OTHER");
assert!(config.has_backend("UCX"));
assert!(config.has_backend("ucx"));
assert!(config.has_backend("GDS_MT"));
assert!(config.has_backend("gds_mt"));
assert!(config.has_backend("OTHER"));
assert!(config.has_backend("other"));
}
}