use super::relations::TopologyRelationIndex;
use super::{NNS_TOPOLOGY_GAPS_REPORT_SCHEMA_VERSION, NnsTopologyGapsReport};
use crate::nns::{
data_center::report::NnsDataCenterListReport,
node::report::{NnsNodeListReport, NnsNodeRow},
node_operator::report::{NnsNodeOperatorListReport, NnsNodeOperatorRow},
node_provider::report::NnsNodeProviderListReport,
topology::report::NnsTopologyGapRow,
};
pub(super) fn topology_gaps_report_from_reports(
network: String,
source_endpoint: String,
node_report: NnsNodeListReport,
node_provider_report: NnsNodeProviderListReport,
node_operator_report: NnsNodeOperatorListReport,
data_center_report: NnsDataCenterListReport,
) -> NnsTopologyGapsReport {
let index = TopologyRelationIndex::from_reports(
&node_provider_report,
&node_operator_report,
&data_center_report,
);
let mut gaps = collect_node_gaps(&node_report.nodes, &index);
gaps.extend(collect_node_operator_gaps(
&node_operator_report.node_operators,
&index,
));
sort_gap_rows(&mut gaps);
let gap_count = gaps.len();
let status = if gap_count == 0 { "ok" } else { "attention" }.to_string();
NnsTopologyGapsReport {
schema_version: NNS_TOPOLOGY_GAPS_REPORT_SCHEMA_VERSION,
network,
source_endpoint,
status,
gap_count,
gaps,
}
}
fn collect_node_gaps(
nodes: &[NnsNodeRow],
index: &TopologyRelationIndex<'_>,
) -> Vec<NnsTopologyGapRow> {
let mut gaps = Vec::new();
for node in nodes {
if !index.has_node_provider(&node.node_provider_principal) {
gaps.push(topology_gap_row(
"node",
&node.node_principal,
"node_provider",
&node.node_provider_principal,
));
}
if !index.has_node_operator(&node.node_operator_principal) {
gaps.push(topology_gap_row(
"node",
&node.node_principal,
"node_operator",
&node.node_operator_principal,
));
}
if !index.has_data_center(&node.data_center_id) {
gaps.push(topology_gap_row(
"node",
&node.node_principal,
"data_center",
&node.data_center_id,
));
}
}
gaps
}
fn collect_node_operator_gaps(
operators: &[NnsNodeOperatorRow],
index: &TopologyRelationIndex<'_>,
) -> Vec<NnsTopologyGapRow> {
let mut gaps = Vec::new();
for operator in operators {
if !index.has_node_provider(&operator.node_provider_principal) {
gaps.push(topology_gap_row(
"node_operator",
&operator.node_operator_principal,
"node_provider",
&operator.node_provider_principal,
));
}
if !index.has_data_center(&operator.data_center_id) {
gaps.push(topology_gap_row(
"node_operator",
&operator.node_operator_principal,
"data_center",
&operator.data_center_id,
));
}
}
gaps
}
fn sort_gap_rows(gaps: &mut [NnsTopologyGapRow]) {
gaps.sort_by(|left, right| {
(
left.subject_kind.as_str(),
left.subject.as_str(),
left.missing_relation.as_str(),
left.referenced_id.as_str(),
)
.cmp(&(
right.subject_kind.as_str(),
right.subject.as_str(),
right.missing_relation.as_str(),
right.referenced_id.as_str(),
))
});
}
fn topology_gap_row(
subject_kind: &str,
subject: &str,
missing_relation: &str,
referenced_id: &str,
) -> NnsTopologyGapRow {
NnsTopologyGapRow {
subject_kind: subject_kind.to_string(),
subject: subject.to_string(),
missing_relation: missing_relation.to_string(),
referenced_id: referenced_id.to_string(),
}
}