use regex::Regex;
use std::borrow::Cow;
use std::collections::BTreeMap;
use std::env;
use std::fmt;
use std::sync::Arc;
use thiserror::Error;
const ESCAPE_PLACEHOLDER: &str = "\x00ESCAPED_DOLLAR\x00";
type GetEnvVarFn = Arc<dyn Fn(&str) -> Option<String> + Send + Sync>;
#[derive(Clone)]
pub struct Vars {
map: BTreeMap<String, String>,
get_env_var: GetEnvVarFn,
escape_re: Regex,
bracketed_re: Regex,
simple_re: Regex,
}
impl fmt::Debug for Vars {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("Vars").field("map", &self.map).finish_non_exhaustive()
}
}
impl Default for Vars {
fn default() -> Self {
Self::new()
}
}
impl Vars {
pub fn new() -> Self {
Self {
map: BTreeMap::new(),
get_env_var: Arc::new(|k| env::var(k).ok()),
escape_re: Regex::new(r"\$\$").unwrap(),
bracketed_re: Regex::new(r"\$\{([^}]+)\}").unwrap(),
simple_re: Regex::new(r"\$([a-zA-Z_][a-zA-Z0-9_]*)").unwrap(),
}
}
pub fn with_env_lookup(mut self, f: impl Fn(&str) -> Option<String> + Send + Sync + 'static) -> Self {
self.get_env_var = Arc::new(f);
self
}
pub fn with(mut self, key: impl Into<String>, value: impl Into<String>) -> Self {
self.insert(key, value);
self
}
pub fn insert(&mut self, key: impl Into<String>, value: impl Into<String>) -> &mut Self {
self.map.insert(key.into(), value.into());
self
}
pub fn get(&self, key: &str) -> Option<Cow<'_, str>> {
if let Some(value) = self.map.get(key) {
Some(Cow::Borrowed(value.as_str()))
} else {
(self.get_env_var)(key).map(Cow::Owned)
}
}
pub fn expand(&self, template: &str) -> Result<String, VarError> {
let escaped = self.escape_re.replace_all(template, ESCAPE_PLACEHOLDER);
let bracketed = self.substitute(&self.bracketed_re, &escaped)?;
let simple = self.substitute(&self.simple_re, &bracketed)?;
Ok(simple.replace(ESCAPE_PLACEHOLDER, "$"))
}
fn substitute(&self, re: &Regex, text: &str) -> Result<String, VarError> {
let mut missing = None;
let result = re.replace_all(text, |caps: ®ex::Captures| {
let var_name = &caps[1];
if let Some(value) = self.get(var_name) {
value.into_owned()
} else {
missing = Some(var_name.to_string());
caps[0].to_string()
}
});
match missing {
Some(var) => Err(VarError::NotFound(var)),
None => Ok(result.into_owned()),
}
}
pub fn has_reference(&self, s: &str) -> bool {
self.bracketed_re.is_match(s) || self.simple_re.is_match(s)
}
}
#[derive(Debug, Error)]
pub enum VarError {
#[error("Environment variable '{0}' not found")]
NotFound(String),
}
#[cfg(test)]
mod tests {
use super::*;
fn no_env() -> Vars {
Vars::new().with_env_lookup(|_| None)
}
#[test]
fn test_simple_var() {
let vars = no_env().with("TEST_VAR_SIMPLE", "hello");
let result = vars.expand("$TEST_VAR_SIMPLE world").unwrap();
assert_eq!(result, "hello world");
}
#[test]
fn test_bracketed_var() {
let vars = no_env().with("TEST_VAR_BRACKET", "hello");
let result = vars.expand("${TEST_VAR_BRACKET} world").unwrap();
assert_eq!(result, "hello world");
}
#[test]
fn test_escape_sequence() {
let result = no_env().expand("$$VAR").unwrap();
assert_eq!(result, "$VAR");
}
#[test]
fn test_multiple_vars() {
let vars = no_env().with("VAR1", "hello").with("VAR2", "world");
let result = vars.expand("$VAR1 ${VAR2}!").unwrap();
assert_eq!(result, "hello world!");
}
#[test]
fn test_missing_var() {
let result = no_env().expand("$MISSING_VAR");
assert!(matches!(result, Err(VarError::NotFound(ref name)) if name == "MISSING_VAR"));
}
#[test]
fn test_unclosed_brace_left_as_is() {
let result = no_env().expand("${VAR").unwrap();
assert_eq!(result, "${VAR");
}
#[test]
fn test_empty_string() {
let result = no_env().expand("").unwrap();
assert_eq!(result, "");
}
#[test]
fn test_no_vars() {
let result = no_env().expand("plain text").unwrap();
assert_eq!(result, "plain text");
}
#[test]
fn test_dollar_at_end() {
let result = no_env().expand("text$").unwrap();
assert_eq!(result, "text$");
}
#[test]
fn test_var_with_underscore() {
let vars = no_env().with("MY_TEST_VAR", "value");
let result = vars.expand("$MY_TEST_VAR").unwrap();
assert_eq!(result, "value");
}
#[test]
fn test_var_with_numbers() {
let vars = no_env().with("VAR123", "value");
let result = vars.expand("$VAR123").unwrap();
assert_eq!(result, "value");
}
#[test]
fn test_special_char_stops_var_name() {
let vars = no_env().with("VAR", "value");
let result = vars.expand("$VAR-suffix").unwrap();
assert_eq!(result, "value-suffix");
}
#[test]
fn vars_lookup_takes_precedence_over_env() {
let vars = Vars::new()
.with_env_lookup(|k| (k == "WORKSPACE").then(|| "/from-env".into()))
.with("WORKSPACE", "/from-vars");
let result = vars.expand("${WORKSPACE}/foo").unwrap();
assert_eq!(result, "/from-vars/foo");
}
#[test]
fn vars_falls_through_to_env_when_missing() {
let vars = Vars::new().with_env_lookup(|k| (k == "ONLY_IN_ENV").then(|| "from-env".into()));
let result = vars.expand("$ONLY_IN_ENV").unwrap();
assert_eq!(result, "from-env");
}
#[test]
fn has_reference_detects_both_forms() {
let vars = no_env();
assert!(vars.has_reference("${WORKSPACE}/foo"));
assert!(vars.has_reference("$HOME/bar"));
assert!(!vars.has_reference("plain/path"));
assert!(!vars.has_reference("$"));
assert!(!vars.has_reference("$$"));
}
}