init commit
This commit is contained in:
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
@@ -0,0 +1 @@
|
||||
/target
|
||||
1181
Cargo.lock
generated
Normal file
1181
Cargo.lock
generated
Normal file
File diff suppressed because it is too large
Load Diff
14
Cargo.toml
Normal file
14
Cargo.toml
Normal file
@@ -0,0 +1,14 @@
|
||||
[package]
|
||||
name = "skyforge"
|
||||
version = "0.1.0"
|
||||
edition = "2021"
|
||||
|
||||
[dependencies]
|
||||
clap = { version = "4.5.11" }
|
||||
regex = "1.10.5"
|
||||
serde = { version = "1.0", features = ["derive"] }
|
||||
serde_yml = "0.0.10"
|
||||
serde_json = "1.0.112"
|
||||
tera = "1.20.0"
|
||||
thiserror = "1.0"
|
||||
walkdir = "2.3"
|
||||
28
README.md
Normal file
28
README.md
Normal file
@@ -0,0 +1,28 @@
|
||||
# Skyforge
|
||||
|
||||
## Brief
|
||||
|
||||
Skyforge was designed to assist in render thousands of device configurations across the globe.
|
||||
|
||||
## Assumptions
|
||||
|
||||
- Partitions are groups of regions
|
||||
- Regions are groups of zones
|
||||
- Zones are groups of devices
|
||||
- Layers are groups of common devices and facilitate template mapping
|
||||
|
||||
## Functionality
|
||||
|
||||
Skyforge takes a user provide regex pattern, performs a walk on a `./spec` dir, and matches a list of devices specifications that do not have the word "common" in their path.
|
||||
All group files are labeled with common and mappable from the file itself.
|
||||
|
||||
For each file_path matched, Skyforge then maps to all consituent files:
|
||||
|
||||
- Layer - from by the common file in parent dir and maps the region
|
||||
- Zonal - from the first group of chars in filename up to a `-` (region_id + zone_id)
|
||||
- Regional - from the region_id of containing folder
|
||||
- Partitional - from either layer (common.yaml) or regional yaml
|
||||
|
||||
Once all files are found, a compiled_specifcation is built.
|
||||
This spec is then passed to Tera as context.
|
||||
Tera then loads the template files for that layer and renders the configuration file.
|
||||
2
demo/spec/common/eu.yaml
Normal file
2
demo/spec/common/eu.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
partitional:
|
||||
location: european union
|
||||
2
demo/spec/common/us.yaml
Normal file
2
demo/spec/common/us.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
partitional:
|
||||
location: united states
|
||||
2
demo/spec/xyz/common/ex.yaml
Normal file
2
demo/spec/xyz/common/ex.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
regional:
|
||||
partition: us
|
||||
2
demo/spec/xyz/ex-edge-r1/common.yaml
Normal file
2
demo/spec/xyz/ex-edge-r1/common.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
common:
|
||||
layer: ex-edge-r
|
||||
2
demo/spec/xyz/ex-edge-r1/xyz1-ex-edge-r101.yaml
Normal file
2
demo/spec/xyz/ex-edge-r1/xyz1-ex-edge-r101.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
device:
|
||||
hostname: xyz1-ex-edge-r101
|
||||
2
demo/spec/xyz/ex-edge-r1/xyz1-ex-edge-r102.yaml
Normal file
2
demo/spec/xyz/ex-edge-r1/xyz1-ex-edge-r102.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
device:
|
||||
hostname: xyz1-ex-edge-r102
|
||||
2
demo/spec/xyz/ex-edge-r1/xyz1.common.yaml
Normal file
2
demo/spec/xyz/ex-edge-r1/xyz1.common.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
zonal:
|
||||
az: 1
|
||||
2
demo/spec/xyz/ex-edge-r1/xyz2-ex-edge-r101.yaml
Normal file
2
demo/spec/xyz/ex-edge-r1/xyz2-ex-edge-r101.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
device:
|
||||
hostname: xyz2-ex-edge-r101
|
||||
2
demo/spec/xyz/ex-edge-r1/xyz2-ex-edge-r102.yaml
Normal file
2
demo/spec/xyz/ex-edge-r1/xyz2-ex-edge-r102.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
device:
|
||||
hostname: xyz2-ex-edge-r102
|
||||
2
demo/spec/xyz/ex-edge-r1/xyz2.common.yaml
Normal file
2
demo/spec/xyz/ex-edge-r1/xyz2.common.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
zonal:
|
||||
az: 2
|
||||
3
demo/spec/xyz/ex-edge-r2/common.yaml
Normal file
3
demo/spec/xyz/ex-edge-r2/common.yaml
Normal file
@@ -0,0 +1,3 @@
|
||||
common:
|
||||
partition: eu
|
||||
layer: ex-edge-r
|
||||
2
demo/spec/xyz/ex-edge-r2/xyz1-ex-edge-r201.yaml
Normal file
2
demo/spec/xyz/ex-edge-r2/xyz1-ex-edge-r201.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
device:
|
||||
hostname: xyz1-ex-edge-r201
|
||||
2
demo/spec/xyz/ex-edge-r2/xyz1-ex-edge-r202.yaml
Normal file
2
demo/spec/xyz/ex-edge-r2/xyz1-ex-edge-r202.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
device:
|
||||
hostname: xyz1-ex-edge-r202
|
||||
2
demo/spec/xyz/ex-edge-r2/xyz1.common.yaml
Normal file
2
demo/spec/xyz/ex-edge-r2/xyz1.common.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
zonal:
|
||||
az: 1
|
||||
2
demo/spec/xyz/ex-edge-r2/xyz2-ex-edge-r201.yaml
Normal file
2
demo/spec/xyz/ex-edge-r2/xyz2-ex-edge-r201.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
device:
|
||||
hostname: xyz2-ex-edge-r201
|
||||
2
demo/spec/xyz/ex-edge-r2/xyz2-ex-edge-r202.yaml
Normal file
2
demo/spec/xyz/ex-edge-r2/xyz2-ex-edge-r202.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
device:
|
||||
hostname: xyz2-ex-edge-r202
|
||||
2
demo/spec/xyz/ex-edge-r2/xyz2.common.yaml
Normal file
2
demo/spec/xyz/ex-edge-r2/xyz2.common.yaml
Normal file
@@ -0,0 +1,2 @@
|
||||
zonal:
|
||||
az: 2
|
||||
4
demo/tmpl/common/ex/system.tmpl
Normal file
4
demo/tmpl/common/ex/system.tmpl
Normal file
@@ -0,0 +1,4 @@
|
||||
system {
|
||||
hostname {{ hostname }};
|
||||
location "{{ location }}";
|
||||
}
|
||||
2
demo/tmpl/ex-edge-r/chassis.tmpl
Normal file
2
demo/tmpl/ex-edge-r/chassis.tmpl
Normal file
@@ -0,0 +1,2 @@
|
||||
chassis {
|
||||
}
|
||||
2
demo/tmpl/ex-edge-r/interfaces.tmpl
Normal file
2
demo/tmpl/ex-edge-r/interfaces.tmpl
Normal file
@@ -0,0 +1,2 @@
|
||||
interfaces {
|
||||
}
|
||||
2
demo/tmpl/ex-edge-r/protocols.tmpl
Normal file
2
demo/tmpl/ex-edge-r/protocols.tmpl
Normal file
@@ -0,0 +1,2 @@
|
||||
protocols {
|
||||
}
|
||||
5
demo/tmpl/ex-edge-r/structure.yaml
Normal file
5
demo/tmpl/ex-edge-r/structure.yaml
Normal file
@@ -0,0 +1,5 @@
|
||||
files:
|
||||
- system
|
||||
- chassis
|
||||
- interfaces
|
||||
- protocols
|
||||
1
demo/tmpl/ex-edge-r/system.tmpl
Symbolic link
1
demo/tmpl/ex-edge-r/system.tmpl
Symbolic link
@@ -0,0 +1 @@
|
||||
../common/ex/system.tmpl
|
||||
130
src/cli.rs
Normal file
130
src/cli.rs
Normal file
@@ -0,0 +1,130 @@
|
||||
use crate::log::LogLevel;
|
||||
use clap::{Arg, ArgAction, ArgGroup, Command};
|
||||
use regex::Regex;
|
||||
use std::fmt;
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct Args {
|
||||
pub devices: Regex,
|
||||
pub loglevel: LogLevel,
|
||||
pub env: EnvVars,
|
||||
}
|
||||
|
||||
impl fmt::Display for Args {
|
||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||
write!(
|
||||
f,
|
||||
"devices: {}, loglevel: {}, env: {}",
|
||||
self.devices, self.loglevel, self.env
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct EnvVars {
|
||||
pub spec_path: String,
|
||||
pub tmpl_path: String,
|
||||
pub out_path: String,
|
||||
pub log_path: String,
|
||||
}
|
||||
|
||||
impl fmt::Display for EnvVars {
|
||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||
write!(
|
||||
f,
|
||||
"spec_path: {}, tmpl_path: {}, out_path: {}, log_path: {}",
|
||||
self.spec_path, self.tmpl_path, self.out_path, self.log_path
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
/// loads `command line arguments` and `environment variables` using custom `clap`
|
||||
pub fn parse_args() -> Args {
|
||||
let matches: clap::ArgMatches = build().get_matches();
|
||||
|
||||
let raw_devices = matches.get_one::<String>("devices").unwrap().to_string();
|
||||
let devices: Regex =
|
||||
Regex::new(&raw_devices).expect("Invalid regex pattern provided for `devices`");
|
||||
|
||||
let loglevel: LogLevel = match matches.get_flag("debug") {
|
||||
true => LogLevel::Debug,
|
||||
false => match matches.get_flag("verbose") {
|
||||
true => LogLevel::Verbose,
|
||||
false => LogLevel::Info,
|
||||
},
|
||||
};
|
||||
|
||||
let env: EnvVars = parse_env();
|
||||
|
||||
Args {
|
||||
devices,
|
||||
loglevel,
|
||||
env,
|
||||
}
|
||||
}
|
||||
|
||||
/// loads only environment variables
|
||||
pub fn parse_env() -> EnvVars {
|
||||
EnvVars {
|
||||
spec_path: match std::env::var("SF_SPEC_PATH") {
|
||||
Ok(path) => path.trim_end_matches('/').to_string(),
|
||||
Err(_) => String::from("./spec"),
|
||||
},
|
||||
tmpl_path: match std::env::var("SF_TMPL_PATH") {
|
||||
Ok(path) => path.trim_end_matches('/').to_string(),
|
||||
Err(_) => String::from("./tmpl"),
|
||||
},
|
||||
out_path: match std::env::var("SF_OUT_PATH") {
|
||||
Ok(path) => path.trim_end_matches('/').to_string(),
|
||||
Err(_) => String::from("./out"),
|
||||
},
|
||||
log_path: match std::env::var("SF_LOG_PATH") {
|
||||
Ok(path) => path.trim_end_matches('/').to_string(),
|
||||
Err(_) => String::from("./log"),
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
const ABOUT_MSG: &str = r#"Skyforge Config Generation Engine"#;
|
||||
const ENV_MSG: &str = r#"Environment:
|
||||
SF_SPEC_PATH Path to the directory containing templates. Defaults to "./spec".
|
||||
SF_TMPL_PATH Path to the directory containing specifications. Defaults to "./tmpl".
|
||||
SF_OUT_PATH Path to the directory for command output. Defaults to "./out".
|
||||
SF_LOG_PATH Path to the directory for log output. Defaults to "./log".
|
||||
"#;
|
||||
|
||||
/// builds a custom command line argument parser
|
||||
fn build() -> Command {
|
||||
Command::new("app")
|
||||
.about(ABOUT_MSG)
|
||||
.version(env!("CARGO_PKG_VERSION"))
|
||||
.author("rskntroot")
|
||||
.arg(
|
||||
Arg::new("devices")
|
||||
.long("devices")
|
||||
.short('d')
|
||||
.help("A regular expression pattern")
|
||||
.required(true),
|
||||
)
|
||||
.arg(
|
||||
Arg::new("debug")
|
||||
.long("debug")
|
||||
.help("Print debug information")
|
||||
.action(ArgAction::SetTrue)
|
||||
.required(false),
|
||||
)
|
||||
.arg(
|
||||
Arg::new("verbose")
|
||||
.short('v')
|
||||
.long("verbose")
|
||||
.help("Print verbose information")
|
||||
.action(ArgAction::SetTrue)
|
||||
.required(false),
|
||||
)
|
||||
.group(
|
||||
ArgGroup::new("loglevel")
|
||||
.args(&["debug", "verbose"])
|
||||
.required(false),
|
||||
)
|
||||
.after_help(ENV_MSG)
|
||||
}
|
||||
78
src/log.rs
Normal file
78
src/log.rs
Normal file
@@ -0,0 +1,78 @@
|
||||
#![allow(dead_code)]
|
||||
|
||||
#[derive(Debug, PartialEq, Clone, Copy)]
|
||||
pub enum LogLevel {
|
||||
Debug,
|
||||
Verbose,
|
||||
Info,
|
||||
Warning,
|
||||
Critical,
|
||||
None,
|
||||
}
|
||||
|
||||
impl LogLevel {
|
||||
pub fn value(&self) -> u8 {
|
||||
match self {
|
||||
LogLevel::Debug => u8::MIN,
|
||||
LogLevel::Verbose => u8::from(30),
|
||||
LogLevel::Info => u8::from(60),
|
||||
LogLevel::Warning => u8::from(90),
|
||||
LogLevel::Critical => u8::from(120),
|
||||
LogLevel::None => u8::MAX,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl std::fmt::Display for LogLevel {
|
||||
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
|
||||
write!(f, "{:?}", self)
|
||||
}
|
||||
}
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! info {
|
||||
($current_level:expr, $($msg:expr),*) => {
|
||||
if LogLevel::Info.value() >= $current_level.value() {
|
||||
println!($($msg),*);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! verb {
|
||||
($current_level:expr, $($msg:expr),*) => {
|
||||
if LogLevel::Verbose.value() >= $current_level.value() {
|
||||
println!($($msg),*);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! dbug {
|
||||
($current_level:expr, $($msg:expr),*) => {
|
||||
if LogLevel::Debug.value() >= $current_level.value() {
|
||||
let message = format!($($msg),*);
|
||||
for line in message.lines() {
|
||||
println!(" |D| {}", line);
|
||||
}
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! warn {
|
||||
($current_level:expr, $($msg:expr),*) => {
|
||||
if LogLevel::Warning.value() >= $current_level.value() {
|
||||
eprintln!($($msg),*);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
#[macro_export]
|
||||
macro_rules! crit {
|
||||
($current_level:expr, $($msg:expr),*) => {
|
||||
if LogLevel::Critical.value() >= $current_level.value() {
|
||||
eprintln!($($msg),*);
|
||||
}
|
||||
};
|
||||
}
|
||||
31
src/main.rs
Normal file
31
src/main.rs
Normal file
@@ -0,0 +1,31 @@
|
||||
mod cli;
|
||||
mod log;
|
||||
mod specs;
|
||||
mod tmpls;
|
||||
|
||||
use log::LogLevel;
|
||||
|
||||
fn main() {
|
||||
let args: cli::Args = cli::parse_args();
|
||||
let dbg: LogLevel = args.loglevel;
|
||||
|
||||
dbug!(dbg, "{}", &args);
|
||||
|
||||
let specifications: Vec<specs::Specification> =
|
||||
specs::compile(&args.devices, &args.env.spec_path, dbg);
|
||||
|
||||
for spec in specifications {
|
||||
let result = tmpls::process_templates(&spec, dbg).ok().unwrap();
|
||||
verb!(
|
||||
dbg,
|
||||
"Compiled Spec:\n{}",
|
||||
serde_json::to_string_pretty(&spec.compiled).unwrap()
|
||||
);
|
||||
info!(dbg, "Rendered Config:");
|
||||
for line in result {
|
||||
if line != "\n" {
|
||||
print!("{}", line)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
232
src/specs.rs
Normal file
232
src/specs.rs
Normal file
@@ -0,0 +1,232 @@
|
||||
use crate::{info, verb, LogLevel};
|
||||
use regex::Regex;
|
||||
use serde_json::{json, Map, Value};
|
||||
use serde_yml;
|
||||
use std::path::{Path, PathBuf};
|
||||
use std::str::FromStr;
|
||||
use std::{env, fmt, fs};
|
||||
use walkdir::WalkDir;
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct Specification {
|
||||
pub partitional: String,
|
||||
pub regional: String,
|
||||
pub common: String,
|
||||
pub zonal: String,
|
||||
pub device: String,
|
||||
pub compiled: Value,
|
||||
}
|
||||
|
||||
impl Specification {
|
||||
pub fn build(
|
||||
partitional: &str,
|
||||
regional: &str,
|
||||
common: &str,
|
||||
zonal: &str,
|
||||
device: &str,
|
||||
) -> Result<Self, Box<dyn std::error::Error>> {
|
||||
// Read all YAML files
|
||||
let yaml_contents: Vec<String> = [partitional, regional, common, zonal, device]
|
||||
.iter()
|
||||
.filter_map(|path| fs::read_to_string(path).ok())
|
||||
.collect();
|
||||
|
||||
// Convert each YAML to JSON Value and collect them
|
||||
let json_values: Vec<Value> = yaml_contents
|
||||
.iter()
|
||||
.map(|content| serde_yml::from_str(content))
|
||||
.collect::<Result<Vec<Value>, _>>()?;
|
||||
|
||||
// Merge all objects under a single key
|
||||
let mut merged_map = Map::new();
|
||||
for value in json_values {
|
||||
if let Some(obj) = value.as_object() {
|
||||
for (_key, value) in obj {
|
||||
if let Some(inner_obj) = value.as_object() {
|
||||
for (k, v) in inner_obj {
|
||||
merged_map.insert(k.clone(), v.clone());
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Create the final merged object
|
||||
let compiled = json!({"data": merged_map});
|
||||
|
||||
Ok(Self {
|
||||
partitional: String::from(partitional),
|
||||
regional: String::from(regional),
|
||||
common: String::from(common),
|
||||
zonal: String::from(zonal),
|
||||
device: String::from(device),
|
||||
compiled,
|
||||
})
|
||||
}
|
||||
|
||||
pub fn get_layer(&self) -> String {
|
||||
PathBuf::from(&self.device)
|
||||
.parent()
|
||||
.unwrap()
|
||||
.file_name()
|
||||
.unwrap()
|
||||
.to_string_lossy()
|
||||
.chars()
|
||||
.filter(|c| !c.is_numeric())
|
||||
.collect::<String>()
|
||||
}
|
||||
|
||||
pub fn get_hostname(&self) -> String {
|
||||
PathBuf::from(&self.device)
|
||||
.file_name()
|
||||
.unwrap()
|
||||
.to_string_lossy()
|
||||
.into_owned()
|
||||
}
|
||||
}
|
||||
|
||||
impl fmt::Display for Specification {
|
||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||
write!(
|
||||
f,
|
||||
"Specification:\n\
|
||||
Partitional: {}\n\
|
||||
Regional: {}\n\
|
||||
Common: {}\n\
|
||||
Zonal: {}\n\
|
||||
Device: {}\n\
|
||||
Compiled:\n{}",
|
||||
self.partitional, self.regional, self.common, self.zonal, self.device, self.compiled
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
pub fn compile(pattern: &Regex, spec_path: &String, dbg: LogLevel) -> Vec<Specification> {
|
||||
let spec_list: Vec<String> = filter_specs(pattern, get_specs(spec_path, dbg));
|
||||
info!(
|
||||
dbg,
|
||||
"Matched {} devices against '{}'",
|
||||
&spec_list.len(),
|
||||
&pattern
|
||||
);
|
||||
for spec in &spec_list {
|
||||
verb!(dbg, " | {}", spec);
|
||||
}
|
||||
let mut specifications: Vec<Specification> = Vec::new();
|
||||
for spec in spec_list {
|
||||
let zonal: String = get_zonal(&spec);
|
||||
let common: String = get_common(&spec);
|
||||
let regional: String = get_regional(&spec);
|
||||
let partitional: String = get_partional(&common, ®ional);
|
||||
specifications.push(
|
||||
match Specification::build(&partitional, ®ional, &common, &zonal, &spec) {
|
||||
Ok(compiled_spec) => {
|
||||
verb!(
|
||||
dbg,
|
||||
"Compiled Spec for '{}'\n | {}\n | {}\n | {}\n | {}\n | {}",
|
||||
compiled_spec.get_hostname(),
|
||||
compiled_spec.partitional,
|
||||
compiled_spec.regional,
|
||||
compiled_spec.common,
|
||||
compiled_spec.zonal,
|
||||
compiled_spec.device
|
||||
);
|
||||
compiled_spec
|
||||
}
|
||||
Err(e) => panic!("failed to build Specification: {}", e),
|
||||
},
|
||||
)
|
||||
}
|
||||
specifications
|
||||
}
|
||||
|
||||
fn get_partional(common: &str, regional: &str) -> String {
|
||||
let re = Regex::new(r"partition: ([^\n\r$]+)").expect("failed");
|
||||
|
||||
[common, regional]
|
||||
.iter()
|
||||
.find_map(|&path| {
|
||||
fs::read_to_string(path).ok().and_then(|contents| {
|
||||
re.captures(&contents).and_then(|captures| {
|
||||
captures
|
||||
.get(1)
|
||||
.map(|matched| format!("./spec/common/{}.yaml", matched.as_str().trim()))
|
||||
})
|
||||
})
|
||||
})
|
||||
.unwrap_or_else(|| format!("./spec/common/default.yaml"))
|
||||
}
|
||||
|
||||
fn get_regional(path: &str) -> String {
|
||||
let dir_name = PathBuf::from(path)
|
||||
.parent()
|
||||
.unwrap()
|
||||
.file_name()
|
||||
.unwrap()
|
||||
.to_string_lossy()
|
||||
.into_owned();
|
||||
|
||||
let new_path = PathBuf::from(path)
|
||||
.parent()
|
||||
.unwrap()
|
||||
.parent()
|
||||
.unwrap()
|
||||
.join("common")
|
||||
.join(format!("{}.yaml", &dir_name[0..2]));
|
||||
|
||||
String::from(new_path.to_str().unwrap())
|
||||
}
|
||||
|
||||
fn get_common(spec: &str) -> String {
|
||||
let path: PathBuf = PathBuf::from_str(&spec).expect("Path was invalid");
|
||||
String::from(path.with_file_name("common.yaml").to_str().unwrap())
|
||||
}
|
||||
|
||||
fn get_zonal(spec: &str) -> String {
|
||||
let path: PathBuf = PathBuf::from_str(&spec).expect("Path was invalid");
|
||||
|
||||
String::from(format!(
|
||||
"{}.common.yaml",
|
||||
path.file_name()
|
||||
.and_then(|filename| filename.to_str())
|
||||
.and_then(|filename| filename.split_once('-'))
|
||||
.map(|(zone, _)| path.with_file_name(zone))
|
||||
.unwrap_or_else(|| path.with_file_name("common.yaml"))
|
||||
.to_str()
|
||||
.unwrap()
|
||||
.to_string(),
|
||||
))
|
||||
}
|
||||
|
||||
fn filter_specs(pattern: &Regex, spec_list: Vec<String>) -> Vec<String> {
|
||||
spec_list
|
||||
.into_iter()
|
||||
.filter(|spec| {
|
||||
Path::new(spec)
|
||||
.file_name()
|
||||
.and_then(|name| name.to_str())
|
||||
.map_or(false, |name| pattern.is_match(name))
|
||||
})
|
||||
.collect()
|
||||
}
|
||||
|
||||
fn get_specs(spec_path: &String, dbg: LogLevel) -> Vec<String> {
|
||||
let mut result = Vec::new();
|
||||
let root = Path::new(&spec_path);
|
||||
for entry in WalkDir::new(root).into_iter().filter_map(|e| e.ok()) {
|
||||
if let Ok(path) = entry.path().strip_prefix(root) {
|
||||
let path_str = path.to_string_lossy().to_string();
|
||||
|
||||
if !path_str.contains("common") && path_str.ends_with("yaml") {
|
||||
result.push(format!("{}/{}", spec_path, path_str));
|
||||
}
|
||||
}
|
||||
}
|
||||
info!(
|
||||
dbg,
|
||||
"Skyforge found {} renderable devices in {}",
|
||||
result.len(),
|
||||
env::current_dir().unwrap().display()
|
||||
);
|
||||
result
|
||||
}
|
||||
56
src/tmpls.rs
Normal file
56
src/tmpls.rs
Normal file
@@ -0,0 +1,56 @@
|
||||
use crate::specs::Specification;
|
||||
use crate::{verb, LogLevel};
|
||||
use serde_json::Value;
|
||||
use serde_yml;
|
||||
use std::fs;
|
||||
use tera::{Context, Tera};
|
||||
|
||||
#[derive(serde::Deserialize)]
|
||||
struct TemplateConfig {
|
||||
files: Vec<String>,
|
||||
}
|
||||
|
||||
pub fn process_templates(
|
||||
spec: &Specification,
|
||||
dbg: LogLevel,
|
||||
) -> Result<Vec<String>, Box<dyn std::error::Error>> {
|
||||
// Create Tera context from spec.compiled
|
||||
let mut context = Context::new();
|
||||
if let Value::Object(map) = &spec.compiled {
|
||||
if let Some(Value::Object(data_map)) = map.get("data") {
|
||||
for (key, value) in data_map {
|
||||
context.insert(key, value);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Get the base directory path
|
||||
let base_dir = format!("./tmpl/{}", spec.get_layer());
|
||||
|
||||
// Read structure.yaml
|
||||
let structure_path = format!("{}/structure.yaml", base_dir);
|
||||
let structure_content = fs::read_to_string(&structure_path)?;
|
||||
let config: TemplateConfig = serde_yml::from_str(&structure_content)?;
|
||||
|
||||
// Initialize Tera with the specific directory
|
||||
let mut tera = Tera::default();
|
||||
|
||||
// Process each template
|
||||
verb!(dbg, "Rendering Templates");
|
||||
let mut rendered_templates = Vec::new();
|
||||
for template_name in config.files {
|
||||
// Read the template file directly
|
||||
let template_path = format!("{}/{}.tmpl", base_dir, template_name);
|
||||
verb!(dbg, " | {}", &template_path);
|
||||
let template_content = fs::read_to_string(&template_path)?;
|
||||
|
||||
// Add this specific template to Tera
|
||||
tera.add_raw_template(&template_name, &template_content)?;
|
||||
|
||||
// Render the template
|
||||
let rendered = tera.render(&template_name, &context)?;
|
||||
rendered_templates.push(rendered);
|
||||
}
|
||||
|
||||
Ok(rendered_templates)
|
||||
}
|
||||
Reference in New Issue
Block a user