From 80dbad66bb9d3b239844afdfb70eeeb3f423e7ba Mon Sep 17 00:00:00 2001 From: David Morrison Date: Thu, 14 Nov 2024 23:09:14 -0700 Subject: [PATCH] fix: pod lifecycle data using gvks --- sk-core/src/k8s/util.rs | 5 ++ sk-driver/src/mutation.rs | 6 +- sk-driver/src/tests/mutation_test.rs | 4 +- sk-store/src/index.rs | 7 ++- sk-store/src/lib.rs | 6 +- sk-store/src/pod_owners_map.rs | 73 +++++++++++++--------- sk-store/src/store.rs | 22 ++++--- sk-store/src/tests/pod_owners_map_test.rs | 46 +++++++++----- sk-store/src/tests/trace_store_test.rs | 75 +++++++++++++++-------- 9 files changed, 156 insertions(+), 88 deletions(-) diff --git a/sk-core/src/k8s/util.rs b/sk-core/src/k8s/util.rs index b07d8965..4d352039 100644 --- a/sk-core/src/k8s/util.rs +++ b/sk-core/src/k8s/util.rs @@ -76,6 +76,11 @@ where build_object_meta_helper(Some(namespace.into()), name, sim_name, owner) } +pub fn format_gvk_name(gvk: &GVK, ns_name: &str) -> String { + format!("{gvk}:{ns_name}") +} + + pub fn sanitize_obj(obj: &mut DynamicObject, api_version: &str, kind: &str) { obj.metadata.creation_timestamp = None; obj.metadata.deletion_timestamp = None; diff --git a/sk-driver/src/mutation.rs b/sk-driver/src/mutation.rs index cd8d5df8..5706097b 100644 --- a/sk-driver/src/mutation.rs +++ b/sk-driver/src/mutation.rs @@ -119,16 +119,16 @@ fn add_lifecycle_annotation( ) -> EmptyResult { if let Some(orig_ns) = pod.annotations().get(ORIG_NAMESPACE_ANNOTATION_KEY) { for owner in owners { - let gvk = GVK::from_owner_ref(owner)?; + let owner_gvk = GVK::from_owner_ref(owner)?; let owner_ns_name = format!("{}/{}", orig_ns, owner.name); - if !ctx.store.has_obj(&gvk, &owner_ns_name) { + if !ctx.store.has_obj(&owner_gvk, &owner_ns_name) { continue; } let hash = jsonutils::hash(&serde_json::to_value(&pod.stable_spec()?)?); let seq = mut_data.count(hash); - let lifecycle = ctx.store.lookup_pod_lifecycle(&owner_ns_name, hash, seq); + let lifecycle = ctx.store.lookup_pod_lifecycle(&owner_gvk, &owner_ns_name, hash, seq); if let Some(patch) = to_annotation_patch(&lifecycle) { info!("applying lifecycle annotations (hash={hash}, seq={seq})"); if pod.metadata.annotations.is_none() { diff --git a/sk-driver/src/tests/mutation_test.rs b/sk-driver/src/tests/mutation_test.rs index fdb431fb..3ab345dd 100644 --- a/sk-driver/src/tests/mutation_test.rs +++ b/sk-driver/src/tests/mutation_test.rs @@ -130,8 +130,8 @@ async fn test_mutate_pod(mut test_pod: corev1::Pod, mut adm_resp: AdmissionRespo let mut store = MockTraceStore::new(); let _ = store .expect_lookup_pod_lifecycle() - .with(predicate::always(), predicate::eq(EMPTY_POD_SPEC_HASH), predicate::eq(0)) - .returning(|_, _, _| PodLifecycleData::Finished(1, 2)) + .with(predicate::always(), predicate::always(), predicate::eq(EMPTY_POD_SPEC_HASH), predicate::eq(0)) + .returning(|_, _, _, _| PodLifecycleData::Finished(1, 2)) .once(); let _ = store.expect_has_obj().returning(move |_gvk, o| o == owner_ns_name); diff --git a/sk-store/src/index.rs b/sk-store/src/index.rs index 386347a2..8dfd893a 100644 --- a/sk-store/src/index.rs +++ b/sk-store/src/index.rs @@ -5,7 +5,10 @@ use serde::{ Deserialize, Serialize, }; -use sk_core::k8s::GVK; +use sk_core::k8s::{ + format_gvk_name, + GVK, +}; #[derive(Default, Deserialize, Serialize)] pub struct TraceIndex { @@ -25,7 +28,7 @@ impl TraceIndex { pub fn flattened_keys(&self) -> Vec { self.index .iter() - .flat_map(|(gvk, gvk_hash)| gvk_hash.keys().map(move |k| format!("{gvk}:{k}"))) + .flat_map(|(gvk, gvk_hash)| gvk_hash.keys().map(move |ns_name| format_gvk_name(gvk, ns_name))) .collect() } diff --git a/sk-store/src/lib.rs b/sk-store/src/lib.rs index cdcb1712..b513273c 100644 --- a/sk-store/src/lib.rs +++ b/sk-store/src/lib.rs @@ -53,14 +53,14 @@ pub struct ExportedTrace { config: TracerConfig, events: Vec, index: TraceIndex, - pod_lifecycles: HashMap, + pod_lifecycles: HashMap<(GVK, String), PodLifecyclesMap>, } pub trait TraceStorable { fn create_or_update_obj(&mut self, obj: &DynamicObject, ts: i64, maybe_old_hash: Option) -> EmptyResult; fn delete_obj(&mut self, obj: &DynamicObject, ts: i64) -> EmptyResult; fn update_all_objs_for_gvk(&mut self, gvk: &GVK, objs: &[DynamicObject], ts: i64) -> EmptyResult; - fn lookup_pod_lifecycle(&self, owner_ns_name: &str, pod_hash: u64, seq: usize) -> PodLifecycleData; + fn lookup_pod_lifecycle(&self, gvk: &GVK, owner_ns_name: &str, pod_hash: u64, seq: usize) -> PodLifecycleData; fn record_pod_lifecycle( &mut self, ns_name: &str, @@ -91,7 +91,7 @@ pub mod mock { fn create_or_update_obj(&mut self, obj: &DynamicObject, ts: i64, maybe_old_hash: Option) -> EmptyResult; fn delete_obj(&mut self, obj: &DynamicObject, ts: i64) -> EmptyResult; fn update_all_objs_for_gvk(&mut self, gvk: &GVK, objs: &[DynamicObject], ts: i64) -> EmptyResult; - fn lookup_pod_lifecycle(&self, owner_ns_name: &str, pod_hash: u64, seq: usize) -> PodLifecycleData; + fn lookup_pod_lifecycle(&self, owner_gvk: &GVK, owner_ns_name: &str, pod_hash: u64, seq: usize) -> PodLifecycleData; fn record_pod_lifecycle( &mut self, ns_name: &str, diff --git a/sk-store/src/pod_owners_map.rs b/sk-store/src/pod_owners_map.rs index 35e00d07..b78b7975 100644 --- a/sk-store/src/pod_owners_map.rs +++ b/sk-store/src/pod_owners_map.rs @@ -2,9 +2,15 @@ use std::collections::hash_map::Entry; use std::collections::HashMap; use sk_core::errors::*; -use sk_core::k8s::PodLifecycleData; +use sk_core::k8s::{ + format_gvk_name, + PodLifecycleData, + GVK, +}; use tracing::*; +use crate::TraceIndex; + // The PodOwnersMap tracks lifecycle data for all pods that are owned by some object that we care // about (e.g., if we are tracking Deployments, the owners map will track the lifecycle data for // all pods that are owned by a Deployment). @@ -46,14 +52,14 @@ pub type PodLifecyclesMap = HashMap>; #[derive(Default)] pub(crate) struct PodOwnersMap { - m: HashMap, - index: HashMap, + m: HashMap<(GVK, String), PodLifecyclesMap>, + index: HashMap, } impl PodOwnersMap { pub(crate) fn new_from_parts( - m: HashMap, - index: HashMap, + m: HashMap<(GVK, String), PodLifecyclesMap>, + index: HashMap, ) -> PodOwnersMap { PodOwnersMap { m, index } } @@ -64,20 +70,23 @@ impl PodOwnersMap { pub(crate) fn lifecycle_data_for<'a>( &'a self, + owner_gvk: &GVK, owner_ns_name: &str, pod_hash: u64, ) -> Option<&'a Vec> { - self.m.get(owner_ns_name)?.get(&pod_hash) + self.m.get(&(owner_gvk.clone(), owner_ns_name.into()))?.get(&pod_hash) } pub(crate) fn store_new_pod_lifecycle( &mut self, - ns_name: &str, + pod_ns_name: &str, + owner_gvk: &GVK, owner_ns_name: &str, hash: u64, lifecycle_data: &PodLifecycleData, ) { - let idx = match self.m.entry(owner_ns_name.into()) { + let owner_gvk_name = format_gvk_name(owner_gvk, owner_ns_name); + let idx = match self.m.entry((owner_gvk.clone(), owner_ns_name.into())) { Entry::Vacant(e) => { e.insert([(hash, vec![lifecycle_data.clone()])].into()); 0 @@ -89,30 +98,33 @@ impl PodOwnersMap { }, }; - info!("inserting pod {ns_name} owned by {owner_ns_name} with hash {hash}: {lifecycle_data:?}"); - self.index.insert(ns_name.into(), (owner_ns_name.into(), hash, idx)); + info!("inserting pod {pod_ns_name} owned by {owner_gvk_name} with hash {hash}: {lifecycle_data:?}"); + self.index + .insert(pod_ns_name.into(), ((owner_gvk.clone(), owner_ns_name.into()), hash, idx)); } - pub(crate) fn update_pod_lifecycle(&mut self, ns_name: &str, lifecycle_data: &PodLifecycleData) -> EmptyResult { - match self.index.get(ns_name) { - None => bail!("pod {} not present in index", ns_name), - Some((owner_ns_name, hash, sequence_idx)) => { + pub(crate) fn update_pod_lifecycle(&mut self, pod_ns_name: &str, lifecycle_data: &PodLifecycleData) -> EmptyResult { + match self.index.get(pod_ns_name) { + None => bail!("pod {} not present in index", pod_ns_name), + Some(((owner_gvk, owner_ns_name), hash, sequence_idx)) => { let owner_entry = self .m - .get_mut(owner_ns_name) - .ok_or(anyhow!("no owner entry for pod {}", ns_name))?; - let pods = - owner_entry - .get_mut(hash) - .ok_or(anyhow!("no entry for pod {} matching hash {}", ns_name, hash))?; + .get_mut(&(owner_gvk.clone(), owner_ns_name.into())) + .ok_or(anyhow!("no owner entry for pod {}", pod_ns_name))?; + let pods = owner_entry.get_mut(hash).ok_or(anyhow!( + "no entry for pod {} matching hash {}", + pod_ns_name, + hash + ))?; let pod_entry = pods.get_mut(*sequence_idx).ok_or(anyhow!( "no sequence index {} for pod {} matching hash {}", sequence_idx, - ns_name, + pod_ns_name, hash ))?; - info!("updating pod {ns_name} owned by {owner_ns_name} with hash {hash}: {lifecycle_data:?}"); + let owner_gvk_name = format_gvk_name(owner_gvk, owner_ns_name); + info!("updating pod {pod_ns_name} owned by {owner_gvk_name} with hash {hash}: {lifecycle_data:?}"); *pod_entry = lifecycle_data.clone(); Ok(()) }, @@ -126,21 +138,24 @@ impl PodOwnersMap { &self, start_ts: i64, end_ts: i64, - index: &HashMap, - ) -> HashMap { + index: &TraceIndex, + ) -> HashMap<(GVK, String), PodLifecyclesMap> { self.m .iter() // The filtering is a little complicated here; if the owning object isn't in the index, // we discard it. Also, if none of the pods belonging to the owning object land // within the given time window, we want to discard it. Otherwise, we want to filter // down the list of pods to the ones that fall between the given time window. - .filter_map(|(owner, lifecycles_map)| { - if !index.contains_key(owner) { + .filter_map(|((owner_gvk, owner_ns_name), lifecycles_map)| { + if !index.contains(owner_gvk, owner_ns_name) { return None; } // Note the question mark here, doing a bunch of heavy lifting - Some((owner.clone(), filter_lifecycles_map(start_ts, end_ts, lifecycles_map)?)) + Some(( + (owner_gvk.clone(), owner_ns_name.clone()), + filter_lifecycles_map(start_ts, end_ts, lifecycles_map)?, + )) }) .collect() } @@ -170,7 +185,7 @@ pub(crate) fn filter_lifecycles_map( #[cfg(test)] impl PodOwnersMap { - pub(crate) fn pod_owner_meta(&self, ns_name: &str) -> Option<&(String, u64, usize)> { - self.index.get(ns_name) + pub(crate) fn pod_owner_meta(&self, pod_ns_name: &str) -> Option<&((GVK, String), u64, usize)> { + self.index.get(pod_ns_name) } } diff --git a/sk-store/src/store.rs b/sk-store/src/store.rs index da6cf3ea..2e46db8d 100644 --- a/sk-store/src/store.rs +++ b/sk-store/src/store.rs @@ -75,13 +75,13 @@ impl TraceStore { // Collect all pod lifecycle data that is a) between the start and end times, and b) is // owned by some object contained in the trace - // let pod_lifecycles = self.pod_owners.filter(start_ts, end_ts, &index); + let pod_lifecycles = self.pod_owners.filter(start_ts, end_ts, &index); let data = rmp_serde::to_vec_named(&ExportedTrace { version: CURRENT_TRACE_VERSION, config: self.config.clone(), events, index, - pod_lifecycles: HashMap::new(), // TODO pod lifecycles don't work in SK2.0 + pod_lifecycles, })?; info!("Exported {} events", num_events); @@ -230,8 +230,14 @@ impl TraceStorable for TraceStore { Ok(()) } - fn lookup_pod_lifecycle(&self, owner_ns_name: &str, pod_hash: u64, seq: usize) -> PodLifecycleData { - let maybe_lifecycle_data = self.pod_owners.lifecycle_data_for(owner_ns_name, pod_hash); + fn lookup_pod_lifecycle( + &self, + owner_gvk: &GVK, + owner_ns_name: &str, + pod_hash: u64, + seq: usize, + ) -> PodLifecycleData { + let maybe_lifecycle_data = self.pod_owners.lifecycle_data_for(owner_gvk, owner_ns_name, pod_hash); match maybe_lifecycle_data { Some(data) => data[seq % data.len()].clone(), _ => PodLifecycleData::Empty, @@ -260,12 +266,12 @@ impl TraceStorable for TraceStore { for owner in &owners { // Pods are guaranteed to have namespaces, so the unwrap is fine let owner_ns_name = format!("{}/{}", pod.namespace().unwrap(), owner.name); - let gvk = GVK::from_owner_ref(owner)?; - if !self.has_obj(&gvk, &owner_ns_name) { + let owner_gvk = GVK::from_owner_ref(owner)?; + if !self.has_obj(&owner_gvk, &owner_ns_name) { continue; } - if !self.config.track_lifecycle_for(&gvk) { + if !self.config.track_lifecycle_for(&owner_gvk) { continue; } @@ -280,7 +286,7 @@ impl TraceStorable for TraceStore { // more things out from this and/or allow users to specify what is filtered out. let hash = jsonutils::hash(&serde_json::to_value(&pod.stable_spec()?)?); self.pod_owners - .store_new_pod_lifecycle(ns_name, &owner_ns_name, hash, lifecycle_data); + .store_new_pod_lifecycle(ns_name, &owner_gvk, &owner_ns_name, hash, lifecycle_data); break; } } else { diff --git a/sk-store/src/tests/pod_owners_map_test.rs b/sk-store/src/tests/pod_owners_map_test.rs index ca8b16f2..88bdb8a4 100644 --- a/sk-store/src/tests/pod_owners_map_test.rs +++ b/sk-store/src/tests/pod_owners_map_test.rs @@ -17,34 +17,48 @@ fn owners_map() -> PodOwnersMap { #[rstest] fn test_store_new_pod_lifecycle(mut owners_map: PodOwnersMap) { - owners_map.store_new_pod_lifecycle("podA", "deployment1", 1234, &PodLifecycleData::Running(5)); - owners_map.store_new_pod_lifecycle("podB", "deployment1", 1234, &PodLifecycleData::Running(7)); - owners_map.store_new_pod_lifecycle("podC", "deployment1", 5678, &PodLifecycleData::Running(9)); - owners_map.store_new_pod_lifecycle("podD", "deployment2", 5678, &PodLifecycleData::Running(13)); + owners_map.store_new_pod_lifecycle("podA", &DEPL_GVK, "deployment1", 1234, &PodLifecycleData::Running(5)); + owners_map.store_new_pod_lifecycle("podB", &DEPL_GVK, "deployment1", 1234, &PodLifecycleData::Running(7)); + owners_map.store_new_pod_lifecycle("podC", &DEPL_GVK, "deployment1", 5678, &PodLifecycleData::Running(9)); + owners_map.store_new_pod_lifecycle("podD", &DEPL_GVK, "deployment2", 5678, &PodLifecycleData::Running(13)); assert_eq!( - owners_map.lifecycle_data_for("deployment1", 1234).unwrap(), + owners_map.lifecycle_data_for(&DEPL_GVK, "deployment1", 1234).unwrap(), &vec![PodLifecycleData::Running(5), PodLifecycleData::Running(7)] ); - assert_eq!(owners_map.lifecycle_data_for("deployment1", 5678).unwrap(), &vec![PodLifecycleData::Running(9)]); - assert_eq!(owners_map.lifecycle_data_for("deployment2", 5678).unwrap(), &vec![PodLifecycleData::Running(13)]); + assert_eq!( + owners_map.lifecycle_data_for(&DEPL_GVK, "deployment1", 5678).unwrap(), + &vec![PodLifecycleData::Running(9)] + ); + assert_eq!( + owners_map.lifecycle_data_for(&DEPL_GVK, "deployment2", 5678).unwrap(), + &vec![PodLifecycleData::Running(13)] + ); - assert_eq!(*owners_map.pod_owner_meta("podA").unwrap(), ("deployment1".to_string(), 1234, 0)); - assert_eq!(*owners_map.pod_owner_meta("podB").unwrap(), ("deployment1".to_string(), 1234, 1)); - assert_eq!(*owners_map.pod_owner_meta("podC").unwrap(), ("deployment1".to_string(), 5678, 0)); - assert_eq!(*owners_map.pod_owner_meta("podD").unwrap(), ("deployment2".to_string(), 5678, 0)); + assert_eq!(*owners_map.pod_owner_meta("podA").unwrap(), ((DEPL_GVK.clone(), "deployment1".into()), 1234, 0)); + assert_eq!(*owners_map.pod_owner_meta("podB").unwrap(), ((DEPL_GVK.clone(), "deployment1".into()), 1234, 1)); + assert_eq!(*owners_map.pod_owner_meta("podC").unwrap(), ((DEPL_GVK.clone(), "deployment1".into()), 5678, 0)); + assert_eq!(*owners_map.pod_owner_meta("podD").unwrap(), ((DEPL_GVK.clone(), "deployment2".into()), 5678, 0)); } #[rstest] fn test_filter_owners_map() { - let index = HashMap::from([("test/deployment1".into(), 9876), ("test/deployment2".into(), 5432)]); + let mut index = TraceIndex::new(); + index.insert(DEPL_GVK.clone(), "test/deployment1".into(), 9876); + index.insert(DEPL_GVK.clone(), "test/deployment2".into(), 5432); let owners_map = PodOwnersMap::new_from_parts( HashMap::from([ - ("test/deployment1".into(), PodLifecyclesMap::from([(1234, vec![PodLifecycleData::Finished(1, 2)])])), ( - "test/deployment2".into(), + (DEPL_GVK.clone(), "test/deployment1".into()), + PodLifecyclesMap::from([(1234, vec![PodLifecycleData::Finished(1, 2)])]), + ), + ( + (DEPL_GVK.clone(), "test/deployment2".into()), PodLifecyclesMap::from([(5678, vec![PodLifecycleData::Running(6), PodLifecycleData::Running(11)])]), ), - ("test/deployment3".into(), PodLifecyclesMap::from([(9999, vec![PodLifecycleData::Finished(1, 2)])])), + ( + (DEPL_GVK.clone(), "test/deployment3".into()), + PodLifecyclesMap::from([(9999, vec![PodLifecycleData::Finished(1, 2)])]), + ), ]), HashMap::new(), ); @@ -53,7 +67,7 @@ fn test_filter_owners_map() { assert_eq!( res, HashMap::from([( - "test/deployment2".into(), + (DEPL_GVK.clone(), "test/deployment2".into()), PodLifecyclesMap::from([(5678, vec![PodLifecycleData::Running(6)])]), )]) ); diff --git a/sk-store/src/tests/trace_store_test.rs b/sk-store/src/tests/trace_store_test.rs index 2dc2562b..36eaf43f 100644 --- a/sk-store/src/tests/trace_store_test.rs +++ b/sk-store/src/tests/trace_store_test.rs @@ -33,14 +33,14 @@ fn owner_ref() -> metav1::OwnerReference { #[rstest] fn test_lookup_pod_lifecycle_no_owner(tracer: TraceStore) { - let res = tracer.lookup_pod_lifecycle(TEST_DEPLOYMENT, EMPTY_POD_SPEC_HASH, 0); + let res = tracer.lookup_pod_lifecycle(&DEPL_GVK, TEST_DEPLOYMENT, EMPTY_POD_SPEC_HASH, 0); assert_eq!(res, PodLifecycleData::Empty); } #[rstest] fn test_lookup_pod_lifecycle_no_hash(mut tracer: TraceStore) { tracer.index.insert(DEPL_GVK.clone(), TEST_DEPLOYMENT.into(), 1234); - let res = tracer.lookup_pod_lifecycle(TEST_DEPLOYMENT, EMPTY_POD_SPEC_HASH, 0); + let res = tracer.lookup_pod_lifecycle(&DEPL_GVK, TEST_DEPLOYMENT, EMPTY_POD_SPEC_HASH, 0); assert_eq!(res, PodLifecycleData::Empty); } @@ -51,11 +51,14 @@ fn test_lookup_pod_lifecycle(mut tracer: TraceStore) { tracer.index.insert(DEPL_GVK.clone(), owner_ns_name.clone(), 1234); tracer.pod_owners = PodOwnersMap::new_from_parts( - HashMap::from([(owner_ns_name.clone(), HashMap::from([(EMPTY_POD_SPEC_HASH, vec![pod_lifecycle.clone()])]))]), + HashMap::from([( + (DEPL_GVK.clone(), owner_ns_name.clone()), + HashMap::from([(EMPTY_POD_SPEC_HASH, vec![pod_lifecycle.clone()])]), + )]), HashMap::new(), ); - let res = tracer.lookup_pod_lifecycle(&owner_ns_name, EMPTY_POD_SPEC_HASH, 0); + let res = tracer.lookup_pod_lifecycle(&DEPL_GVK, &owner_ns_name, EMPTY_POD_SPEC_HASH, 0); assert_eq!(res, pod_lifecycle); } @@ -290,6 +293,22 @@ fn test_record_pod_lifecycle_already_stored_no_data(mut tracer: TraceStore, owne )); } +fn mock_pod_owners_map( + pod_ns_name: &str, + pod_spec_hash: u64, + owner_ns_name: &str, + init_lifecycle_data: Vec, + pod_seq_idx: usize, +) -> PodOwnersMap { + PodOwnersMap::new_from_parts( + HashMap::from([( + (DEPL_GVK.clone(), owner_ns_name.into()), + HashMap::from([(EMPTY_POD_SPEC_HASH, init_lifecycle_data)]), + )]), + HashMap::from([(pod_ns_name.into(), ((DEPL_GVK.clone(), owner_ns_name.into()), pod_spec_hash, pod_seq_idx))]), + ) +} + #[rstest] fn test_record_pod_lifecycle_already_stored_no_pod(mut tracer: TraceStore, owner_ref: metav1::OwnerReference) { let new_lifecycle_data = PodLifecycleData::Finished(5, 45); @@ -303,18 +322,18 @@ fn test_record_pod_lifecycle_already_stored_no_pod(mut tracer: TraceStore, owner let mut expected_lifecycle_data = init_lifecycle_data.clone(); expected_lifecycle_data[pod_seq_idx] = new_lifecycle_data.clone(); - let ns_name = format!("{}/{}", TEST_NAMESPACE, "the-pod"); + let pod_ns_name = format!("{}/{}", TEST_NAMESPACE, "the-pod"); let owner_ns_name = format!("{}/{}", TEST_NAMESPACE, owner_ref.name); - tracer.pod_owners = PodOwnersMap::new_from_parts( - HashMap::from([(owner_ns_name.clone(), HashMap::from([(EMPTY_POD_SPEC_HASH, init_lifecycle_data)]))]), - HashMap::from([(ns_name.clone(), (owner_ns_name.clone(), EMPTY_POD_SPEC_HASH, pod_seq_idx))]), - ); + tracer.pod_owners = + mock_pod_owners_map(&pod_ns_name, EMPTY_POD_SPEC_HASH, &owner_ns_name, init_lifecycle_data, pod_seq_idx); tracer - .record_pod_lifecycle(&ns_name, None, vec![owner_ref], &new_lifecycle_data) + .record_pod_lifecycle(&pod_ns_name, None, vec![owner_ref], &new_lifecycle_data) .unwrap(); assert_eq!( - tracer.pod_owners.lifecycle_data_for(&owner_ns_name, EMPTY_POD_SPEC_HASH), + tracer + .pod_owners + .lifecycle_data_for(&DEPL_GVK, &owner_ns_name, EMPTY_POD_SPEC_HASH), Some(&expected_lifecycle_data) ); } @@ -333,7 +352,7 @@ fn test_record_pod_lifecycle_with_new_pod_no_tracked_owner( .unwrap(); let unused_hash = 0; - assert_eq!(tracer.pod_owners.lifecycle_data_for(&owner_ns_name, unused_hash), None); + assert_eq!(tracer.pod_owners.lifecycle_data_for(&DEPL_GVK, &owner_ns_name, unused_hash), None); } #[rstest] @@ -355,7 +374,9 @@ fn test_record_pod_lifecycle_with_new_pod_hash( .record_pod_lifecycle(&ns_name, Some(test_pod), vec![owner_ref], &new_lifecycle_data.clone()) .unwrap(); - let lifecycle_data = tracer.pod_owners.lifecycle_data_for(&owner_ns_name, EMPTY_POD_SPEC_HASH); + let lifecycle_data = tracer + .pod_owners + .lifecycle_data_for(&DEPL_GVK, &owner_ns_name, EMPTY_POD_SPEC_HASH); if track_lifecycle { assert_eq!(lifecycle_data, Some(&vec![new_lifecycle_data])); } else { @@ -374,21 +395,26 @@ fn test_record_pod_lifecycle_with_new_pod_existing_hash( let mut expected_lifecycle_data = init_lifecycle_data.clone(); expected_lifecycle_data.push(new_lifecycle_data.clone()); - let ns_name = test_pod.namespaced_name(); + let pod_ns_name = test_pod.namespaced_name(); let owner_ns_name = format!("{}/{}", TEST_NAMESPACE, owner_ref.name); tracer.index.insert(DEPL_GVK.clone(), owner_ns_name.clone(), TEST_DEPL_HASH); tracer.pod_owners = PodOwnersMap::new_from_parts( - HashMap::from([(owner_ns_name.clone(), HashMap::from([(EMPTY_POD_SPEC_HASH, init_lifecycle_data)]))]), - HashMap::from([("asdf".into(), (owner_ns_name.clone(), 1234, 0))]), + HashMap::from([( + (DEPL_GVK.clone(), owner_ns_name.clone()), + HashMap::from([(EMPTY_POD_SPEC_HASH, init_lifecycle_data)]), + )]), + HashMap::from([("asdf".into(), ((DEPL_GVK.clone(), owner_ns_name.clone()), 1234, 0))]), ); tracer - .record_pod_lifecycle(&ns_name, Some(test_pod), vec![owner_ref], &new_lifecycle_data) + .record_pod_lifecycle(&pod_ns_name, Some(test_pod), vec![owner_ref], &new_lifecycle_data) .unwrap(); assert_eq!( - tracer.pod_owners.lifecycle_data_for(&owner_ns_name, EMPTY_POD_SPEC_HASH), + tracer + .pod_owners + .lifecycle_data_for(&DEPL_GVK, &owner_ns_name, EMPTY_POD_SPEC_HASH), Some(&expected_lifecycle_data) ); } @@ -403,21 +429,20 @@ fn test_record_pod_lifecycle_with_existing_pod( let init_lifecycle_data = vec![PodLifecycleData::Running(5)]; let expected_lifecycle_data = vec![new_lifecycle_data.clone()]; - let ns_name = test_pod.namespaced_name(); + let pod_ns_name = test_pod.namespaced_name(); let owner_ns_name = format!("{}/{}", TEST_NAMESPACE, owner_ref.name); tracer.index.insert(DEPL_GVK.clone(), owner_ns_name.clone(), TEST_DEPL_HASH); - tracer.pod_owners = PodOwnersMap::new_from_parts( - HashMap::from([(owner_ns_name.clone(), HashMap::from([(EMPTY_POD_SPEC_HASH, init_lifecycle_data)]))]), - HashMap::from([(ns_name.clone(), (owner_ns_name.clone(), EMPTY_POD_SPEC_HASH, 0))]), - ); + tracer.pod_owners = mock_pod_owners_map(&pod_ns_name, EMPTY_POD_SPEC_HASH, &owner_ns_name, init_lifecycle_data, 0); tracer - .record_pod_lifecycle(&ns_name, Some(test_pod), vec![owner_ref], &new_lifecycle_data) + .record_pod_lifecycle(&pod_ns_name, Some(test_pod), vec![owner_ref], &new_lifecycle_data) .unwrap(); assert_eq!( - tracer.pod_owners.lifecycle_data_for(&owner_ns_name, EMPTY_POD_SPEC_HASH), + tracer + .pod_owners + .lifecycle_data_for(&DEPL_GVK, &owner_ns_name, EMPTY_POD_SPEC_HASH), Some(&expected_lifecycle_data) ); }