mod dbt;
mod error;
mod jinja;
pub use error::TemplateError;
use schemars::JsonSchema;
use serde::{Deserialize, Serialize};
use std::collections::HashMap;
#[derive(Debug, Clone, Serialize, Deserialize, JsonSchema, Default)]
#[serde(rename_all = "camelCase")]
pub struct TemplateConfig {
#[serde(default)]
pub mode: TemplateMode,
#[serde(default)]
pub context: HashMap<String, serde_json::Value>,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize, JsonSchema, Default)]
#[serde(rename_all = "lowercase")]
pub enum TemplateMode {
#[default]
Raw,
Jinja,
Dbt,
}
pub fn template_sql(sql: &str, config: &TemplateConfig) -> Result<String, TemplateError> {
match config.mode {
TemplateMode::Raw => Ok(sql.to_string()),
TemplateMode::Jinja => jinja::render_jinja(sql, &config.context),
TemplateMode::Dbt => jinja::render_dbt(sql, &config.context),
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn raw_mode_passes_through() {
let sql = "SELECT * FROM {{ not_a_template }}";
let config = TemplateConfig::default();
let result = template_sql(sql, &config).unwrap();
assert_eq!(result, sql);
}
#[test]
fn jinja_mode_renders_variables() {
let sql = "SELECT * FROM {{ table }}";
let mut context = HashMap::new();
context.insert("table".to_string(), serde_json::json!("users"));
let config = TemplateConfig {
mode: TemplateMode::Jinja,
context,
};
let result = template_sql(sql, &config).unwrap();
assert_eq!(result, "SELECT * FROM users");
}
#[test]
fn dbt_mode_renders_ref() {
let sql = "SELECT * FROM {{ ref('users') }}";
let config = TemplateConfig {
mode: TemplateMode::Dbt,
context: HashMap::new(),
};
let result = template_sql(sql, &config).unwrap();
assert_eq!(result, "SELECT * FROM users");
}
#[test]
fn config_serialization() {
let config = TemplateConfig {
mode: TemplateMode::Dbt,
context: HashMap::new(),
};
let json = serde_json::to_string(&config).unwrap();
assert!(json.contains("\"mode\":\"dbt\""));
let parsed: TemplateConfig = serde_json::from_str(&json).unwrap();
assert_eq!(parsed.mode, TemplateMode::Dbt);
}
#[test]
fn config_deserialization_with_defaults() {
let json = r#"{ "mode": "jinja" }"#;
let config: TemplateConfig = serde_json::from_str(json).unwrap();
assert_eq!(config.mode, TemplateMode::Jinja);
assert!(config.context.is_empty());
}
}