feat(homelab): setup route & chain generation
This commit is contained in:
@@ -1,19 +1,31 @@
|
||||
use std::collections::BTreeSet;
|
||||
|
||||
use serde::{Deserialize, Serialize};
|
||||
|
||||
use crate::{Config, HelperError};
|
||||
|
||||
#[derive(Serialize, Deserialize, Default)]
|
||||
pub struct Route {
|
||||
#[serde(default)]
|
||||
kind: RouteKind,
|
||||
name: String,
|
||||
hostname: String,
|
||||
hostname: Option<String>,
|
||||
entrypoint: Option<String>,
|
||||
namespace: String,
|
||||
service: Option<String>,
|
||||
port: i16,
|
||||
private: bool,
|
||||
}
|
||||
|
||||
impl Route {
|
||||
fn hostname(&self) -> &str {
|
||||
self.hostname.as_ref().unwrap_or(&self.name)
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Serialize, Deserialize, Default)]
|
||||
enum RouteKind {
|
||||
#[default]
|
||||
HTTP,
|
||||
TCP,
|
||||
}
|
||||
@@ -30,17 +42,62 @@ pub fn generate_routes(config: &Config) -> Result<(), HelperError> {
|
||||
acc.push_str(&generate_route(r));
|
||||
acc
|
||||
});
|
||||
let chains = generate_chains(&config.routes);
|
||||
std::fs::write("kustomize/routes.yaml", &routes)?;
|
||||
std::fs::write("kustomize/traefik/chains.yaml", &chains)?;
|
||||
println!("Wrote: {}", routes);
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn generate_route(route: &Route) -> String {
|
||||
match route.kind {
|
||||
RouteKind::HTTP => generate_http_route(route),
|
||||
RouteKind::TCP => generate_tcp_route(route),
|
||||
}
|
||||
}
|
||||
|
||||
fn generate_chains(routes: &[Route]) -> String {
|
||||
let namespaces = routes
|
||||
.iter()
|
||||
.filter_map(|r| {
|
||||
if !r.private {
|
||||
return None;
|
||||
}
|
||||
Some(r.namespace.as_str())
|
||||
})
|
||||
.collect::<BTreeSet<_>>();
|
||||
|
||||
namespaces
|
||||
.iter()
|
||||
.enumerate()
|
||||
.fold(String::new(), |mut acc, (i, n)| {
|
||||
if i > 0 {
|
||||
acc.push_str("\n---\n");
|
||||
}
|
||||
acc.push_str(&format!(
|
||||
r#"apiVersion: traefik.io/v1alpha1
|
||||
kind: Middleware
|
||||
metadata:
|
||||
name: private-networks
|
||||
namespace: {}
|
||||
spec:
|
||||
chain:
|
||||
middlewares:
|
||||
- name: private-networks
|
||||
namespace: kube-system"#,
|
||||
n
|
||||
));
|
||||
acc
|
||||
})
|
||||
}
|
||||
|
||||
fn generate_http_route(route: &Route) -> String {
|
||||
let mut filters_section = String::new();
|
||||
if route.private {
|
||||
filters_section = format!(
|
||||
r#"filters:
|
||||
r#"
|
||||
filters:
|
||||
- type: ExtensionRef
|
||||
extensionRef:
|
||||
group: traefik.io
|
||||
@@ -51,7 +108,7 @@ fn generate_route(route: &Route) -> String {
|
||||
|
||||
format!(
|
||||
r#"apiVersion: gateway.networking.k8s.io/v1
|
||||
kind: {}
|
||||
kind: HTTPRoute
|
||||
metadata:
|
||||
name: {}
|
||||
namespace: {}
|
||||
@@ -64,15 +121,49 @@ spec:
|
||||
rules:
|
||||
- backendRefs:
|
||||
- name: {}
|
||||
port: {}
|
||||
{}"#,
|
||||
port: {}{}"#,
|
||||
route.name,
|
||||
route.namespace,
|
||||
route.hostname,
|
||||
route.service.clone().unwrap_or_else(|| route.name.clone()),
|
||||
route.hostname(),
|
||||
route.service.as_ref().unwrap_or_else(|| &route.name),
|
||||
route.port,
|
||||
filters_section
|
||||
)
|
||||
.trim_end()
|
||||
.to_string()
|
||||
}
|
||||
|
||||
fn generate_tcp_route(route: &Route) -> String {
|
||||
let mut middlewares_section = String::new();
|
||||
if route.private {
|
||||
middlewares_section = format!(
|
||||
r#"
|
||||
middlewares:
|
||||
- name: private-networks"#
|
||||
);
|
||||
}
|
||||
|
||||
format!(
|
||||
r#"apiVersion: traefik.io/v1alpha1
|
||||
kind: IngressRouteTCP
|
||||
metadata:
|
||||
name: {}
|
||||
namespace: {}
|
||||
spec:
|
||||
entryPoints:
|
||||
- {}
|
||||
routes:
|
||||
- match: HostSNI(`*`){}
|
||||
services:
|
||||
- name: {}
|
||||
port: {}"#,
|
||||
route.name,
|
||||
route.namespace,
|
||||
route.entrypoint,
|
||||
middlewares_section,
|
||||
route.service.as_ref().unwrap_or_else(|| &route.name),
|
||||
route.port
|
||||
)
|
||||
.trim_end()
|
||||
.to_string()
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user