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 {
#[serde(default, skip_serializing_if = "Option::is_none")]
pub unit: Option<Unit>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub flavor: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub kind: Option<String>,
#[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, Copy, Debug, Default, PartialEq, Eq, Serialize, Deserialize)]
#[serde(rename_all = "lowercase")]
pub enum SupersedeScope {
#[default]
Full,
Partial,
}
#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)]
#[serde(untagged)]
pub enum SupersedeItem {
Full(String),
Scoped(SupersedeScoped),
}
#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct SupersedeScoped {
pub spec: String,
#[serde(default)]
pub scope: SupersedeScope,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub unit: Option<Unit>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub note: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub rationale: Option<String>,
}
impl SupersedeItem {
pub fn spec(&self) -> &str {
match self {
SupersedeItem::Full(id) => id,
SupersedeItem::Scoped(s) => &s.spec,
}
}
pub fn is_full(&self) -> bool {
match self {
SupersedeItem::Full(_) => true,
SupersedeItem::Scoped(s) => s.scope == SupersedeScope::Full,
}
}
pub fn partial_unit(&self) -> Option<&Unit> {
match self {
SupersedeItem::Scoped(s) if s.scope == SupersedeScope::Partial => s.unit.as_ref(),
_ => None,
}
}
pub fn set_spec(&mut self, id: String) {
match self {
SupersedeItem::Full(s) => *s = id,
SupersedeItem::Scoped(s) => s.spec = id,
}
}
}
pub(crate) fn normalize_supersedes(items: Vec<SupersedeItem>) -> Vec<SupersedeItem> {
items
.into_iter()
.map(|item| match item {
SupersedeItem::Scoped(s) if s.scope == SupersedeScope::Full => {
SupersedeItem::Full(s.spec)
}
other => other,
})
.collect()
}
#[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>,
}