From 24b34f70f76a5795d51987442f392dc374cc71eb Mon Sep 17 00:00:00 2001 From: "Andrew J. Stone" Date: Sat, 27 Jan 2024 00:07:52 +0000 Subject: [PATCH] Use DB backed blueprints from #4899 --- ...an_execution.rs => blueprint_execution.rs} | 21 +++-- ...an_blueprint_load.rs => blueprint_load.rs} | 79 ++++++++----------- nexus/src/app/background/init.rs | 41 +++++----- nexus/src/app/background/mod.rs | 4 +- nexus/src/app/mod.rs | 4 - 5 files changed, 62 insertions(+), 87 deletions(-) rename nexus/src/app/background/{plan_execution.rs => blueprint_execution.rs} (96%) rename nexus/src/app/background/{plan_blueprint_load.rs => blueprint_load.rs} (69%) diff --git a/nexus/src/app/background/plan_execution.rs b/nexus/src/app/background/blueprint_execution.rs similarity index 96% rename from nexus/src/app/background/plan_execution.rs rename to nexus/src/app/background/blueprint_execution.rs index c85d96abbe..c2e0f3dd55 100644 --- a/nexus/src/app/background/plan_execution.rs +++ b/nexus/src/app/background/blueprint_execution.rs @@ -24,19 +24,19 @@ use uuid::Uuid; /// Background task that takes a [`Blueprint`] and realizes the change to /// the state of the system based on the `Blueprint`. -pub struct PlanExecutor { +pub struct BlueprintExecutor { datastore: Arc, rx_blueprint: watch::Receiver>>, } -impl PlanExecutor { +impl BlueprintExecutor { // Temporary until we wire up the background task #[allow(unused)] pub fn new( datastore: Arc, rx_blueprint: watch::Receiver>>, - ) -> PlanExecutor { - PlanExecutor { datastore, rx_blueprint } + ) -> BlueprintExecutor { + BlueprintExecutor { datastore, rx_blueprint } } // This is a modified copy of the functionality from `nexus/src/app/sled.rs`. @@ -92,7 +92,7 @@ impl PlanExecutor { let client = match self.sled_client(&opctx, &sled_id).await { Ok(client) => client, Err(err) => { - warn!(log, "{}", err); + warn!(log, "{err:#}"); return Some(err); } }; @@ -105,7 +105,7 @@ impl PlanExecutor { match result { Err(error) => { - warn!(log, "{}", error); + warn!(log, "{error:#}"); Some(error) } Ok(_) => { @@ -130,7 +130,7 @@ impl PlanExecutor { } } -impl BackgroundTask for PlanExecutor { +impl BackgroundTask for BlueprintExecutor { fn activate<'a>( &'a mut self, opctx: &'a OpContext, @@ -138,7 +138,7 @@ impl BackgroundTask for PlanExecutor { async { // Get the latest blueprint, cloning to prevent holding a read lock // on the watch. - let blueprint = self.rx_blueprint.borrow().clone(); + let blueprint = self.rx_blueprint.borrow_and_update().clone(); let Some(blueprint) = blueprint else { warn!(&opctx.log, @@ -210,7 +210,7 @@ mod test { ); let (blueprint_tx, blueprint_rx) = watch::channel(None); - let mut task = PlanExecutor::new(datastore.clone(), blueprint_rx); + let mut task = BlueprintExecutor::new(datastore.clone(), blueprint_rx); // With no blueprint we should fail with an appropriate message. let value = task.activate(&opctx).await; @@ -267,8 +267,7 @@ mod test { // Zones are updated in a particular order, but each request contains // the full set of zones that must be running. - // See https://github.com/oxidecomputer/omicron/blob/main/sled-agent/src/rack_setup/service.rs#L976-L998 - // for more details. + // See `rack_setup::service::ServiceInner::run` for more details. let mut zones = OmicronZonesConfig { generation, zones: vec![OmicronZoneConfig { diff --git a/nexus/src/app/background/plan_blueprint_load.rs b/nexus/src/app/background/blueprint_load.rs similarity index 69% rename from nexus/src/app/background/plan_blueprint_load.rs rename to nexus/src/app/background/blueprint_load.rs index f6299da8bf..b61a383a86 100644 --- a/nexus/src/app/background/plan_blueprint_load.rs +++ b/nexus/src/app/background/blueprint_load.rs @@ -8,11 +8,8 @@ //! changes. use super::common::BackgroundTask; -use crate::app::deployment; use futures::future::BoxFuture; use futures::FutureExt; -use nexus_db_queries::authz; -use nexus_db_queries::authz::Action; use nexus_db_queries::context::OpContext; use nexus_db_queries::db::DataStore; use nexus_types::deployment::Blueprint; @@ -21,8 +18,6 @@ use std::sync::Arc; use tokio::sync::watch; pub struct TargetBlueprintLoader { - blueprints: Arc>, - #[allow(unused)] datastore: Arc, last: Option>, tx: watch::Sender>>, @@ -30,36 +25,15 @@ pub struct TargetBlueprintLoader { } impl TargetBlueprintLoader { - pub fn new( - blueprints: Arc>, - datastore: Arc, - ) -> TargetBlueprintLoader { + pub fn new(datastore: Arc) -> TargetBlueprintLoader { let (tx, rx) = watch::channel(None); - TargetBlueprintLoader { blueprints, datastore, last: None, tx, rx } + TargetBlueprintLoader { datastore, last: None, tx, rx } } /// Expose the target blueprint pub fn watcher(&self) -> watch::Receiver>> { self.rx.clone() } - - // This function is a modified copy from `nexus/src/app/deployment.rs` for - // use until the types are in the datastore. - // - // This is a stand-in for a datastore function that fetches the current - // target information and the target blueprint's contents. This helper - // exists to combine the authz check with the lookup, which is what the - // datastore function will eventually do. - async fn blueprint_target( - &self, - opctx: &OpContext, - ) -> Result, anyhow::Error> { - opctx.authorize(Action::Read, &authz::BLUEPRINT_CONFIG).await?; - let blueprints = self.blueprints.lock().unwrap(); - Ok(blueprints.target.target_id.and_then(|target_id| { - blueprints.all_blueprints.get(&target_id).cloned() - })) - } } impl BackgroundTask for TargetBlueprintLoader { @@ -79,7 +53,8 @@ impl BackgroundTask for TargetBlueprintLoader { }; // Retrieve the latest target blueprint - let result = self.blueprint_target(opctx).await; + let result = + self.datastore.blueprint_target_get_current_full(opctx).await; // Decide what to do with the result match (&self.last, result) { @@ -101,8 +76,8 @@ impl BackgroundTask for TargetBlueprintLoader { json!({}) } (Some(old), Ok(None)) => { - // We have transitioned from having a blueprint to not having one. - // This should not happen. + // We have transitioned from having a blueprint to not + // having one. This should not happen. let message = format!( "target blueprint with id {} was removed. There is no \ longer any target blueprint", @@ -110,9 +85,8 @@ impl BackgroundTask for TargetBlueprintLoader { ); error!(&log, "{}", message); json!({"error": message}) - } - (None, Ok(Some(new_target))) => { + (None, Ok(Some((_, new_target)))) => { // We've found a new target blueprint for the first time. // Save it and notify any watchers. let target_id = new_target.id.to_string(); @@ -125,9 +99,11 @@ impl BackgroundTask for TargetBlueprintLoader { ); self.last = Some(Arc::new(new_target)); self.tx.send_replace(self.last.clone()); - json!({"target_id": target_id, "time_created": time_created}) + json!({ + "target_id": target_id, "time_created": time_created + }) } - (Some(old), Ok(Some(new))) => { + (Some(old), Ok(Some((_, new)))) => { let target_id = new.id.to_string(); let time_created = new.time_created.to_string(); if old.id != new.id { @@ -140,7 +116,10 @@ impl BackgroundTask for TargetBlueprintLoader { ); self.last = Some(Arc::new(new)); self.tx.send_replace(self.last.clone()); - json!({"target_id": target_id, "time_created": time_created}) + json!({ + "target_id": target_id, + "time_created": time_created + }) } else { // The new target id matches the old target id // @@ -157,20 +136,24 @@ impl BackgroundTask for TargetBlueprintLoader { error!(&log, "{}", message); json!({"error": message}) } else { - // We found a new target blueprint that exactly matches - // the old target blueprint. This is the common case - // when we're activated by a timeout. - debug!( - log, - "found latest target blueprint (unchanged)"; - "target_id" => &target_id, - "time_created" => &time_created.clone() - ); - json!({"target_id": target_id, "time_created": time_created}) - } + // We found a new target blueprint that exactly matches + // the old target blueprint. This is the common case + // when we're activated by a timeout. + debug!( + log, + "found latest target blueprint (unchanged)"; + "target_id" => &target_id, + "time_created" => &time_created.clone() + ); + json!({ + "target_id": target_id, + "time_created": time_created + }) + } } } } - }.boxed() + } + .boxed() } } diff --git a/nexus/src/app/background/init.rs b/nexus/src/app/background/init.rs index a7b7b5c4fe..6dfea99657 100644 --- a/nexus/src/app/background/init.rs +++ b/nexus/src/app/background/init.rs @@ -4,8 +4,8 @@ //! Background task initialization -use crate::app::deployment; - +use super::blueprint_execution; +use super::blueprint_load; use super::common; use super::dns_config; use super::dns_propagation; @@ -14,8 +14,6 @@ use super::external_endpoints; use super::inventory_collection; use super::nat_cleanup; use super::phantom_disks; -use super::plan_blueprint_load; -use super::plan_execution; use nexus_db_model::DnsGroup; use nexus_db_queries::context::OpContext; use nexus_db_queries::db::DataStore; @@ -62,10 +60,10 @@ pub struct BackgroundTasks { pub task_phantom_disks: common::TaskHandle, /// task handle for blueprint target loader - pub task_plan_blueprint_target_loader: common::TaskHandle, + pub task_blueprint_loader: common::TaskHandle, - /// task handle for plan execution background task - pub task_plan_executor: common::TaskHandle, + /// task handle for blueprint execution background task + pub task_blueprint_executor: common::TaskHandle, } impl BackgroundTasks { @@ -77,7 +75,6 @@ impl BackgroundTasks { dpd_clients: &HashMap>, nexus_id: Uuid, resolver: internal_dns::resolver::Resolver, - blueprints: Arc>, ) -> BackgroundTasks { let mut driver = common::Driver::new(); @@ -176,13 +173,11 @@ impl BackgroundTasks { }; // Background task: blueprint loader - let blueprint_loader = plan_blueprint_load::TargetBlueprintLoader::new( - blueprints, - datastore.clone(), - ); + let blueprint_loader = + blueprint_load::TargetBlueprintLoader::new(datastore.clone()); let rx_blueprint = blueprint_loader.watcher(); - let task_plan_blueprint_target_loader = driver.register( - String::from("blueprint_target_loader"), + let task_blueprint_loader = driver.register( + String::from("blueprint_loader"), String::from("Loads the current target blueprint from the DB"), // TODO: Add to `BackgroundTaskConfig`? std::time::Duration::from_secs(5), @@ -191,15 +186,17 @@ impl BackgroundTasks { vec![], ); - // Background task: plan executor - let plan_executor = - plan_execution::PlanExecutor::new(datastore, rx_blueprint.clone()); - let task_plan_executor = driver.register( - String::from("plan_executor"), + // Background task: blueprint executor + let blueprint_executor = blueprint_execution::BlueprintExecutor::new( + datastore, + rx_blueprint.clone(), + ); + let task_blueprint_executor = driver.register( + String::from("blueprint_executor"), String::from("Executes the target blueprint"), // TODO: Add to `BackgroundTaskConfig`? std::time::Duration::from_secs(60), - Box::new(plan_executor), + Box::new(blueprint_executor), opctx.child(BTreeMap::new()), vec![Box::new(rx_blueprint)], ); @@ -215,8 +212,8 @@ impl BackgroundTasks { nat_cleanup, task_inventory_collection, task_phantom_disks, - task_plan_blueprint_target_loader, - task_plan_executor, + task_blueprint_loader, + task_blueprint_executor, } } diff --git a/nexus/src/app/background/mod.rs b/nexus/src/app/background/mod.rs index 3b5a9b3ad9..236881271d 100644 --- a/nexus/src/app/background/mod.rs +++ b/nexus/src/app/background/mod.rs @@ -4,6 +4,8 @@ //! Background tasks +mod blueprint_execution; +mod blueprint_load; mod common; mod dns_config; mod dns_propagation; @@ -13,8 +15,6 @@ mod init; mod inventory_collection; mod nat_cleanup; mod phantom_disks; -mod plan_blueprint_load; -mod plan_execution; mod status; pub use common::Driver; diff --git a/nexus/src/app/mod.rs b/nexus/src/app/mod.rs index 42d031dc9a..5d8c321cd4 100644 --- a/nexus/src/app/mod.rs +++ b/nexus/src/app/mod.rs @@ -357,9 +357,6 @@ impl Nexus { Arc::clone(&db_datastore), ); - let blueprints = - Arc::new(std::sync::Mutex::new(deployment::Blueprints::new())); - let background_tasks = background::BackgroundTasks::start( &background_ctx, Arc::clone(&db_datastore), @@ -367,7 +364,6 @@ impl Nexus { &dpd_clients, config.deployment.id, resolver.clone(), - Arc::clone(&blueprints), ); let external_resolver = {