From 944ae9547926eb7e1f0ea9a931eddf4aa11ab646 Mon Sep 17 00:00:00 2001 From: Tobias Richter Date: Wed, 23 Oct 2024 20:46:42 +0200 Subject: [PATCH 1/2] Add validation for pipeline and environment variables, fail on duplicate variable definition --- src/variables.rs | 104 +++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 104 insertions(+) diff --git a/src/variables.rs b/src/variables.rs index fdf1e84..dd009dc 100644 --- a/src/variables.rs +++ b/src/variables.rs @@ -12,6 +12,8 @@ use crate::pipelines::get_pipeline; use crate::HOST_NAME; use colored::*; use reqwest::{Method, StatusCode}; +use std::collections::HashSet; +use std::hash::{Hash, Hasher}; use std::process; use std::thread::sleep; use std::time::Duration; @@ -23,6 +25,37 @@ impl PartialEq for EnvironmentVariable { } } +impl Eq for EnvironmentVariable {} + +impl Hash for EnvironmentVariable { + fn hash(&self, state: &mut H) { + self.name.hash(state); + self.service.hash(state); + } +} + +impl Hash for EnvironmentVariableServiceType { + fn hash(&self, state: &mut H) { + match self { + EnvironmentVariableServiceType::All => { + state.write_u8(1); + } + EnvironmentVariableServiceType::Author => { + state.write_u8(2); + } + EnvironmentVariableServiceType::Publish => { + state.write_u8(3); + } + EnvironmentVariableServiceType::Preview => { + state.write_u8(4); + } + EnvironmentVariableServiceType::Invalid => { + state.write_u8(5); + } + } + } +} + // Make pipeline variables comparable - if they have the same name and same service they are the same. impl PartialEq for PipelineVariable { fn eq(&self, other: &Self) -> bool { @@ -30,6 +63,34 @@ impl PartialEq for PipelineVariable { } } +impl Eq for PipelineVariable {} + +impl Hash for PipelineVariable { + fn hash(&self, state: &mut H) { + self.name.hash(state); + self.service.hash(state); + } +} + +impl Hash for PipelineVariableServiceType { + fn hash(&self, state: &mut H) { + match self { + PipelineVariableServiceType::Build => { + state.write_u8(1); + } + PipelineVariableServiceType::FunctionalTest => { + state.write_u8(2); + } + PipelineVariableServiceType::UiTest => { + state.write_u8(3); + } + PipelineVariableServiceType::Invalid => { + state.write_u8(4); + } + } + } +} + /// Retrieves environment variables for the specified environment. /// /// # Arguments @@ -134,6 +195,21 @@ pub async fn set_env_vars_from_file( println!("{:>4} Environment: {} ({})", "⬛", e.id, env.name); + // ensure there are no duplicate environment variables + let duplicates = find_duplicates(e.variables.clone()); + if duplicates.len() > 0 { + for dv in &duplicates { + eprintln!( + "{:>8} {} name: '{}' service: {}", + "❌".red(), + "ERROR, duplicate variable definition found, please check your file!".red(), + dv.name, + dv.service + ); + } + process::exit(4); + } + // The vector that holds the final variables that will be set or deleted. Will be constructed // by comparing the variables that are currently set in Cloud Manager and those in the local // YAML config file. @@ -375,6 +451,21 @@ pub async fn set_pipeline_vars_from_file( println!("{:>4} Pipeline: {} ({})", "⬛", l.id, pipeline.name); + // ensure there are no duplicate environment variables + let duplicates = find_duplicates(l.variables.clone()); + if duplicates.len() > 0 { + for dv in &duplicates { + eprintln!( + "{:>8} {} name: '{}' service: {}", + "❌".red(), + "ERROR, duplicate variable definition found, please check your file!".red(), + dv.name, + dv.service + ); + } + process::exit(4); + } + // The vector that holds the final variables that will be set or deleted. Will be constructed // by comparing the variables that are currently set in Cloud Manager and those in the local // YAML config file. @@ -515,3 +606,16 @@ pub async fn set_pipeline_vars_from_file( process::exit(2); } } + +fn find_duplicates(vec: Vec) -> Vec { + let mut seen = HashSet::new(); + let mut duplicates = HashSet::new(); + + for item in vec { + if !seen.insert(item.clone()) { + duplicates.insert(item); + } + } + + duplicates.into_iter().collect() +} From 8c096a9daee2cc102e0d26505cade58a26dd6861 Mon Sep 17 00:00:00 2001 From: Benjamin Sommerfeld Date: Thu, 24 Oct 2024 13:25:18 +0200 Subject: [PATCH 2/2] imp: using a vector directly instead of collecting from a HashSet --- src/variables.rs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/variables.rs b/src/variables.rs index dd009dc..dfe7bd8 100644 --- a/src/variables.rs +++ b/src/variables.rs @@ -609,13 +609,13 @@ pub async fn set_pipeline_vars_from_file( fn find_duplicates(vec: Vec) -> Vec { let mut seen = HashSet::new(); - let mut duplicates = HashSet::new(); + let mut duplicates = Vec::new(); for item in vec { if !seen.insert(item.clone()) { - duplicates.insert(item); + duplicates.push(item); } } - duplicates.into_iter().collect() + duplicates }