use serde::{Deserialize, Serialize};
use crate::unit::Unit;
#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct ExtendItem {
pub spec: String,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub unit: Option<Unit>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub paths: Option<Vec<String>>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub nature: Option<String>,
}
#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct RefineItem {
pub aspect: String,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub unit: Option<Unit>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub paths: Option<Vec<String>>,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub refines_specs: Vec<String>,
}
pub(crate) fn expand_extend_paths(items: Vec<ExtendItem>) -> Result<Vec<ExtendItem>, String> {
let mut out = Vec::with_capacity(items.len());
for mut item in items {
let Some(paths) = item.paths.take() else {
out.push(item);
continue;
};
if item.unit.is_some() {
return Err(format!(
"extends item for spec '{}' cannot carry both unit: and paths:",
item.spec
));
}
if paths.is_empty() {
return Err(format!(
"extends item for spec '{}' has an empty paths: list",
item.spec
));
}
for path in paths {
out.push(ExtendItem {
spec: item.spec.clone(),
unit: Some(Unit::File { path }),
paths: None,
nature: item.nature.clone(),
});
}
}
Ok(out)
}
pub(crate) fn expand_refine_paths(items: Vec<RefineItem>) -> Result<Vec<RefineItem>, String> {
let mut out = Vec::with_capacity(items.len());
for mut item in items {
let Some(paths) = item.paths.take() else {
out.push(item);
continue;
};
if item.unit.is_some() {
return Err(format!(
"refines item for aspect '{}' cannot carry both unit: and paths:",
item.aspect
));
}
if paths.is_empty() {
return Err(format!(
"refines item for aspect '{}' has an empty paths: list",
item.aspect
));
}
for path in paths {
out.push(RefineItem {
aspect: item.aspect.clone(),
unit: Some(Unit::File { path }),
paths: None,
refines_specs: item.refines_specs.clone(),
});
}
}
Ok(out)
}
#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct CoAuthorityItem {
pub unit: Unit,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub with_specs: Vec<String>,
}
#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct ConstrainItem {
pub unit: Unit,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub note: Option<String>,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub target_specs: Vec<String>,
}
#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct ReferenceItem {
#[serde(default, skip_serializing_if = "Option::is_none")]
pub unit: Option<Unit>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub provenance: Option<Provenance>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub role: Option<String>,
}
#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct Provenance {
pub kind: String,
#[serde(rename = "ref")]
pub reference: String,
}
#[derive(Clone, Debug, PartialEq, Eq, Default, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct Origin {
#[serde(default)]
pub retroactive: bool,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub paths: Vec<String>,
}