191 lines
6.4 KiB
Rust
191 lines
6.4 KiB
Rust
#![deny(warnings)]
|
|
#![allow(dead_code)]
|
|
//! [TSM.ID].[11031972] -- Platform X Ecosystem
|
|
//! xcu-labyrinth -- Multi-hop Obfuscated Routing
|
|
//! Traffic path randomization so no single node knows full route
|
|
|
|
use std::collections::HashMap;
|
|
|
|
#[derive(Debug)]
|
|
pub enum LabyrinthError {
|
|
NoRoute(String),
|
|
NodeFailed(String),
|
|
EncryptionFailed(String),
|
|
}
|
|
impl std::fmt::Display for LabyrinthError {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
match self { Self::NoRoute(e) => write!(f, "No route: {e}"),
|
|
Self::NodeFailed(e) => write!(f, "Node: {e}"),
|
|
Self::EncryptionFailed(e) => write!(f, "Encrypt: {e}"), }
|
|
}
|
|
}
|
|
impl std::error::Error for LabyrinthError {}
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub struct LabyrinthNode {
|
|
pub id: String,
|
|
pub latency_ms: u32,
|
|
pub bandwidth_mbps: u32,
|
|
pub trust_score: f64,
|
|
pub country: String,
|
|
pub is_alive: bool,
|
|
}
|
|
|
|
/// Onion-layered routing envelope
|
|
#[derive(Debug, Clone)]
|
|
pub struct OnionEnvelope {
|
|
pub layers: Vec<EncryptedLayer>,
|
|
pub total_hops: usize,
|
|
}
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub struct EncryptedLayer {
|
|
pub next_hop: String,
|
|
pub encrypted_payload: Vec<u8>,
|
|
pub layer_key_hash: u64,
|
|
}
|
|
|
|
pub struct Labyrinth {
|
|
nodes: HashMap<String, LabyrinthNode>,
|
|
min_hops: usize,
|
|
max_hops: usize,
|
|
avoid_countries: Vec<String>,
|
|
entropy_state: u64,
|
|
}
|
|
|
|
impl Labyrinth {
|
|
pub fn new(min_hops: usize, max_hops: usize, avoid: Vec<String>) -> Self {
|
|
Self {
|
|
nodes: HashMap::new(), min_hops, max_hops,
|
|
avoid_countries: avoid,
|
|
entropy_state: 0xa5a5a5a5deadbeef,
|
|
}
|
|
}
|
|
|
|
pub fn add_node(&mut self, node: LabyrinthNode) {
|
|
self.nodes.insert(node.id.clone(), node);
|
|
}
|
|
|
|
fn next_random(&mut self) -> u64 {
|
|
self.entropy_state ^= self.entropy_state << 13;
|
|
self.entropy_state ^= self.entropy_state >> 7;
|
|
self.entropy_state ^= self.entropy_state << 17;
|
|
self.entropy_state
|
|
}
|
|
|
|
/// Select route through the labyrinth
|
|
pub fn build_route(&mut self, source: &str, destination: &str) -> Result<Vec<String>, LabyrinthError> {
|
|
// Pre-compute random values before borrowing self.nodes
|
|
let hop_rand = self.next_random();
|
|
let node_count = self.nodes.len();
|
|
let rand_scores: Vec<f64> = (0..node_count)
|
|
.map(|_| (self.next_random() % 100) as f64 * 0.3)
|
|
.collect();
|
|
|
|
let eligible: Vec<&LabyrinthNode> = self.nodes.values()
|
|
.filter(|n| n.is_alive)
|
|
.filter(|n| !self.avoid_countries.contains(&n.country))
|
|
.filter(|n| n.id != source && n.id != destination)
|
|
.collect();
|
|
|
|
if eligible.len() < self.min_hops {
|
|
return Err(LabyrinthError::NoRoute(format!("Need {} hops, only {} nodes", self.min_hops, eligible.len())));
|
|
}
|
|
|
|
let hop_count = self.min_hops + (hop_rand as usize % (self.max_hops - self.min_hops + 1));
|
|
let hop_count = hop_count.min(eligible.len());
|
|
|
|
// Score nodes: prefer high trust, low latency, diverse countries
|
|
let mut scored: Vec<(&LabyrinthNode, f64)> = eligible.iter().enumerate().map(|(i, n)| {
|
|
let rand_component = rand_scores.get(i).copied().unwrap_or(0.0);
|
|
let score = n.trust_score * 50.0
|
|
+ (1000.0 / (n.latency_ms as f64 + 1.0))
|
|
+ n.bandwidth_mbps as f64 * 0.1
|
|
+ rand_component;
|
|
(*n, score)
|
|
}).collect();
|
|
|
|
scored.sort_by(|a, b| b.1.partial_cmp(&a.1).unwrap_or(std::cmp::Ordering::Equal));
|
|
|
|
// Pick top nodes but ensure country diversity
|
|
let mut route = vec![source.to_string()];
|
|
let mut used_countries = std::collections::HashSet::new();
|
|
|
|
for (node, _) in &scored {
|
|
if route.len() - 1 >= hop_count { break; }
|
|
if !used_countries.contains(&node.country) || route.len() > 3 {
|
|
route.push(node.id.clone());
|
|
used_countries.insert(node.country.clone());
|
|
}
|
|
}
|
|
|
|
route.push(destination.to_string());
|
|
Ok(route)
|
|
}
|
|
|
|
/// Build onion-encrypted envelope for the route
|
|
pub fn build_onion(&mut self, route: &[String], payload: &[u8]) -> Result<OnionEnvelope, LabyrinthError> {
|
|
let mut layers = Vec::new();
|
|
let mut current_payload = payload.to_vec();
|
|
|
|
// Build layers from destination back to source (onion wrapping)
|
|
for i in (1..route.len()).rev() {
|
|
let next_hop = &route[i];
|
|
let layer_key = self.next_random();
|
|
|
|
// XOR encrypt each layer
|
|
let encrypted: Vec<u8> = current_payload.iter().enumerate()
|
|
.map(|(j, &b)| b ^ ((layer_key >> ((j % 8) * 8)) & 0xFF) as u8)
|
|
.collect();
|
|
|
|
layers.push(EncryptedLayer {
|
|
next_hop: next_hop.clone(),
|
|
encrypted_payload: encrypted.clone(),
|
|
layer_key_hash: layer_key & 0xFFFFFFFF,
|
|
});
|
|
|
|
current_payload = encrypted;
|
|
}
|
|
|
|
layers.reverse();
|
|
Ok(OnionEnvelope { layers, total_hops: route.len() - 2 })
|
|
}
|
|
|
|
/// Peel one layer of the onion (at each relay node)
|
|
pub fn peel_layer(&self, layer: &EncryptedLayer, key: u64) -> Vec<u8> {
|
|
layer.encrypted_payload.iter().enumerate()
|
|
.map(|(j, &b)| b ^ ((key >> ((j % 8) * 8)) & 0xFF) as u8)
|
|
.collect()
|
|
}
|
|
|
|
pub fn node_count(&self) -> usize { self.nodes.len() }
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
fn make_nodes(lab: &mut Labyrinth) {
|
|
for (id, country) in [("node-de","DE"),("node-jp","JP"),("node-br","BR"),("node-sg","SG"),("node-ch","CH")] {
|
|
lab.add_node(LabyrinthNode { id: id.into(), latency_ms: 50, bandwidth_mbps: 100, trust_score: 0.9, country: country.into(), is_alive: true });
|
|
}
|
|
}
|
|
#[test]
|
|
fn test_route_building() {
|
|
let mut lab = Labyrinth::new(2, 4, vec!["CN".into()]);
|
|
make_nodes(&mut lab);
|
|
let route = lab.build_route("source", "dest").unwrap();
|
|
assert!(route.len() >= 4);
|
|
assert_eq!(route[0], "source");
|
|
assert_eq!(route.last().unwrap(), "dest");
|
|
}
|
|
#[test]
|
|
fn test_onion_wrap() {
|
|
let mut lab = Labyrinth::new(2, 3, vec![]);
|
|
make_nodes(&mut lab);
|
|
let route = lab.build_route("src", "dst").unwrap();
|
|
let envelope = lab.build_onion(&route, b"secret").unwrap();
|
|
assert!(envelope.total_hops >= 2);
|
|
assert!(!envelope.layers.is_empty());
|
|
}
|
|
}
|