From ae0aa14d5fd94d302f0fa353447eaa40214f72b8 Mon Sep 17 00:00:00 2001 From: Nichol Yip Date: Fri, 6 Dec 2024 13:58:07 -0800 Subject: [PATCH] Fix issue where EnvOpList was not properly deserialized and bypassed the multiple priority config check. Signed-off-by: Nichol Yip --- crates/spk-schema/src/environ.rs | 39 +++++++++++++++++- crates/spk-schema/src/install_spec.rs | 58 +++++---------------------- crates/spk-schema/src/lib.rs | 2 +- crates/spk-schema/src/v0/spec.rs | 7 ++-- 4 files changed, 54 insertions(+), 52 deletions(-) diff --git a/crates/spk-schema/src/environ.rs b/crates/spk-schema/src/environ.rs index d01933352..cc8a98935 100644 --- a/crates/spk-schema/src/environ.rs +++ b/crates/spk-schema/src/environ.rs @@ -188,9 +188,46 @@ impl EnvOp { } } -#[derive(Clone, Debug, Default, Deserialize, Eq, Hash, Ord, PartialEq, PartialOrd, Serialize)] +#[derive(Clone, Debug, Default, Eq, Hash, Ord, PartialEq, PartialOrd, Serialize)] pub struct EnvOpList(Vec); +impl<'de> Deserialize<'de> for EnvOpList { + fn deserialize(deserializer: D) -> std::result::Result + where + D: serde::Deserializer<'de>, + { + struct EnvConfVisitor; + + impl<'de> serde::de::Visitor<'de> for EnvConfVisitor { + type Value = EnvOpList; + + fn expecting(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result { + f.write_str("an environment configuration") + } + + fn visit_seq(self, mut seq: A) -> std::result::Result + where + A: serde::de::SeqAccess<'de>, + { + let mut vec = EnvOpList::default(); + + while let Some(elem) = seq.next_element::()? { + if vec.iter().any(|x: &EnvOp| x.kind() == OpKind::Priority) + && elem.kind() == OpKind::Priority + { + return Err(serde::de::Error::custom( + "Multiple priority config cannot be set.", + )); + }; + vec.push(elem); + } + Ok(vec) + } + } + deserializer.deserialize_seq(EnvConfVisitor) + } +} + impl IsDefault for EnvOpList { fn is_default(&self) -> bool { self.0.is_empty() diff --git a/crates/spk-schema/src/install_spec.rs b/crates/spk-schema/src/install_spec.rs index 37eda5588..ef77bec51 100644 --- a/crates/spk-schema/src/install_spec.rs +++ b/crates/spk-schema/src/install_spec.rs @@ -17,12 +17,12 @@ use crate::foundation::option_map::OptionMap; use crate::{ ComponentSpecList, EmbeddedPackagesList, - EnvOp, + // EnvOp, EnvOpList, Lint, LintedItem, Lints, - OpKind, + // OpKind, Package, RequirementsList, Result, @@ -46,7 +46,6 @@ mod install_spec_test; PartialOrd, Serialize, )] -#[serde(from = "RawInstallSpec")] pub struct InstallSpec { #[serde(default, skip_serializing_if = "Vec::is_empty")] pub requirements: RequirementsList, @@ -63,7 +62,6 @@ where D: Default, { fn lints(&mut self) -> Vec { - // self.lints.extend(std::mem::take(&mut self.environment.lints)); std::mem::take(&mut self.lints) } } @@ -81,7 +79,7 @@ where _phantom: PhantomData, } -impl From> for InstallSpec +impl From> for RawInstallSpec where D: Default, { @@ -206,69 +204,35 @@ impl From for InstallSpec { } } -#[derive(Deserialize)] -struct RawInstallSpec { +/// A raw, unvalidated install spec. +#[derive(Serialize, Default)] +pub struct RawInstallSpec { #[serde(default)] requirements: RequirementsList, #[serde(default)] embedded: EmbeddedPackagesList, #[serde(default)] components: ComponentSpecList, - #[serde(default, deserialize_with = "deserialize_env_conf")] + #[serde(default)] environment: EnvOpList, } -impl<'de> Deserialize<'de> for InstallSpec { +impl<'de> Deserialize<'de> for RawInstallSpec { fn deserialize(deserializer: D) -> std::result::Result where D: serde::de::Deserializer<'de>, { - deserializer.deserialize_map(InstallSpecVisitor::::default()) + deserializer.deserialize_map(InstallSpecVisitor::::default()) } } -impl<'de> Deserialize<'de> for LintedItem { +impl<'de> Deserialize<'de> for LintedItem { fn deserialize(deserializer: D) -> std::result::Result where D: serde::de::Deserializer<'de>, { - deserializer.deserialize_map(InstallSpecVisitor::>::default()) - } -} - -fn deserialize_env_conf<'de, D>(deserializer: D) -> std::result::Result -where - D: serde::Deserializer<'de>, -{ - struct EnvConfVisitor; - - impl<'de> serde::de::Visitor<'de> for EnvConfVisitor { - type Value = EnvOpList; - - fn expecting(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result { - f.write_str("an environment configuration") - } - - fn visit_seq(self, mut seq: A) -> std::result::Result - where - A: serde::de::SeqAccess<'de>, - { - let mut vec = EnvOpList::default(); - - while let Some(elem) = seq.next_element::()? { - if vec.iter().any(|x: &EnvOp| x.kind() == OpKind::Priority) - && elem.kind() == OpKind::Priority - { - return Err(serde::de::Error::custom( - "Multiple priority config cannot be set.", - )); - }; - vec.push(elem); - } - Ok(vec) - } + deserializer.deserialize_map(InstallSpecVisitor::>::default()) } - deserializer.deserialize_seq(EnvConfVisitor) } impl<'de, D> serde::de::Visitor<'de> for InstallSpecVisitor diff --git a/crates/spk-schema/src/lib.rs b/crates/spk-schema/src/lib.rs index fd410ea83..872688d3a 100644 --- a/crates/spk-schema/src/lib.rs +++ b/crates/spk-schema/src/lib.rs @@ -46,7 +46,7 @@ pub use environ::{ }; pub use error::{Error, Result}; pub use input_variant::InputVariant; -pub use install_spec::InstallSpec; +pub use install_spec::{InstallSpec, RawInstallSpec}; pub use lints::{Lint, LintedItem, Lints, UnknownKey}; pub use option::{Inheritance, Opt}; pub use package::{Package, PackageMut}; diff --git a/crates/spk-schema/src/v0/spec.rs b/crates/spk-schema/src/v0/spec.rs index dc3731d5e..b1fb35188 100644 --- a/crates/spk-schema/src/v0/spec.rs +++ b/crates/spk-schema/src/v0/spec.rs @@ -66,6 +66,7 @@ use crate::{ Opt, Package, PackageMut, + RawInstallSpec, Recipe, RequirementsList, Result, @@ -990,7 +991,7 @@ struct SpecVisitor { sources: Option>>, build: Option>, tests: Option>>, - install: Option>, + install: Option>, #[field_names_as_array(skip)] check_build_spec: bool, #[field_names_as_array(skip)] @@ -1047,7 +1048,7 @@ where .iter() .map(|l| l.item.clone()) .collect_vec(), - install: value.install.take().unwrap_or_default().item, + install: value.install.take().unwrap_or_default().item.into(), } } } @@ -1143,7 +1144,7 @@ where } } "tests" => self.tests = Some(map.next_value::>>()?), - "install" => self.install = Some(map.next_value::>()?), + "install" => self.install = Some(map.next_value::>()?), "api" => { map.next_value::()?; }