1use std::{2 cell::OnceCell,3 collections::BTreeSet,4 ffi::{OsStr, OsString},5 fmt::Display,6 io::Write,7 ops::Deref,8 path::PathBuf,9 str::FromStr,10 sync::{Arc, Mutex, MutexGuard, OnceLock},11};1213use anyhow::{anyhow, bail, ensure, Context, Result};14use fleet_shared::SecretData;15use nix_eval::{nix_go, nix_go_json, util::assert_warn, NixSession, Value};16use openssh::SessionBuilder;17use serde::de::DeserializeOwned;18use tempfile::NamedTempFile;1920use crate::{21 command::MyCommand,22 fleetdata::{FleetData, FleetSecret, FleetSharedSecret},23};2425pub struct FleetConfigInternals {26 pub local_system: String,27 pub directory: PathBuf,28 pub data: Mutex<FleetData>,29 pub nix_args: Vec<OsString>,30 31 pub config_field: Value,32 33 pub localhost: String,3435 36 pub default_pkgs: Value,37 pub nixpkgs: Value,3839 pub nix_session: NixSession,40}414243#[derive(Clone)]44pub struct Config(pub Arc<FleetConfigInternals>);4546impl Deref for Config {47 type Target = FleetConfigInternals;4849 fn deref(&self) -> &Self::Target {50 &self.051 }52}5354#[derive(Clone, Copy, Debug)]55pub enum EscalationStrategy {56 Sudo,57 Run0,58 Su,59}6061#[derive(Clone, PartialEq, Copy)]62pub enum DeployKind {63 64 UpgradeToFleet,65 66 Fleet,67 68 69 NixosInstall,70}7172impl FromStr for DeployKind {73 type Err = anyhow::Error;74 fn from_str(s: &str) -> std::result::Result<Self, Self::Err> {75 match s {76 "upgrade-to-fleet" => Ok(Self::UpgradeToFleet),77 "fleet" => Ok(Self::Fleet),78 "nixos-install" => Ok(Self::NixosInstall),79 v => bail!("unknown deploy_kind: {v}; expected on of \"upgrade-to-fleet\", \"fleet\", \"nixos-install\""),80 }81 }82}83pub struct ConfigHost {84 config: Config,85 pub name: String,86 groups: OnceCell<Vec<String>>,8788 deploy_kind: OnceCell<DeployKind>,8990 pub host_config: Option<Value>,91 pub nixos_config: OnceCell<Value>,92 pub pkgs_override: Option<Value>,9394 95 pub local: bool,96 pub session: OnceLock<Arc<openssh::Session>>,97}9899impl ConfigHost {100 pub fn set_deploy_kind(&self, kind: DeployKind) {101 self.deploy_kind102 .set(kind)103 .ok()104 .expect("deploy kind is already set");105 }106 pub async fn deploy_kind(&self) -> Result<DeployKind> {107 if let Some(kind) = self.deploy_kind.get() {108 return Ok(kind.clone());109 }110 let is_fleet_managed = match self.file_exists("/etc/FLEET_HOST").await {111 Ok(v) => v,112 Err(e) => {113 bail!("failed to query remote system kind: {}", e);114 }115 };116 if !is_fleet_managed {117 bail!(indoc::indoc! {"118 host is not marked as managed by fleet119 if you're not trying to lustrate/install system from scratch,120 you should either121 1. manually create /etc/FLEET_HOST file on the target host,122 2. use ?deploy_kind=fleet host argument if you're upgrading from older version of fleet123 3. use ?deploy_kind=upgrade_to_fleet if you're upgrading from plain nixos to fleet-managed nixos124 "});125 }126 127 let _ = self.deploy_kind.set(DeployKind::Fleet);128 Ok(self129 .deploy_kind130 .get()131 .expect("deploy kind is just set")132 .clone())133 }134 pub async fn escalation_strategy(&self) -> Result<EscalationStrategy> {135 136 137 if (self.find_in_path("sudo").await).is_ok() {138 return Ok(EscalationStrategy::Sudo);139 }140 if (self.find_in_path("run0").await).is_ok() {141 return Ok(EscalationStrategy::Run0);142 }143 Ok(EscalationStrategy::Su)144 }145 async fn open_session(&self) -> Result<Arc<openssh::Session>> {146 assert!(!self.local, "do not open ssh connection to local session");147 148 if let Some(session) = &self.session.get() {149 return Ok((*session).clone());150 };151 let session = SessionBuilder::default();152 let session = session153 .connect(&self.name)154 .await155 .map_err(|e| anyhow!("ssh error while connecting to {}: {e}", self.name))?;156 let session = Arc::new(session);157 self.session.set(session.clone()).expect("TOCTOU happened");158 Ok(session)159 }160 pub async fn mktemp_dir(&self) -> Result<String> {161 let mut cmd = self.cmd("mktemp").await?;162 cmd.arg("-d");163 let path = cmd.run_string().await?;164 Ok(path.trim_end().to_owned())165 }166 pub async fn file_exists(&self, path: impl AsRef<OsStr>) -> Result<bool> {167 let mut cmd = self.cmd("sh").await?;168 cmd.arg("-c")169 .arg("test -e \"$1\" && echo true || echo false")170 .arg("_")171 .arg(path);172 Ok(cmd.run_value().await?)173 }174 pub async fn read_file_bin(&self, path: impl AsRef<OsStr>) -> Result<Vec<u8>> {175 let mut cmd = self.cmd("cat").await?;176 cmd.arg(path);177 cmd.run_bytes().await178 }179 pub async fn read_file_text(&self, path: impl AsRef<OsStr>) -> Result<String> {180 let mut cmd = self.cmd("cat").await?;181 cmd.arg(path);182 cmd.run_string().await183 }184 pub async fn read_dir(&self, path: impl AsRef<OsStr>) -> Result<Vec<String>> {185 let mut cmd = self.cmd("ls").await?;186 cmd.arg(path);187 let out = cmd.run_string().await?;188 let mut lines = out.split('\n');189 if let Some(last) = lines.next_back() {190 ensure!(last.is_empty(), "output of ls should end with newline");191 }192 Ok(lines.map(ToOwned::to_owned).collect())193 }194 #[allow(dead_code)]195 pub async fn read_file_json<D: DeserializeOwned>(&self, path: impl AsRef<OsStr>) -> Result<D> {196 let text = self.read_file_text(path).await?;197 Ok(serde_json::from_str(&text)?)198 }199 pub async fn read_env(&self, env: &str) -> Result<String> {200 let mut cmd = self.cmd("printenv").await?;201 cmd.arg(env);202 cmd.run_string().await203 }204 pub async fn find_in_path(&self, command: &str) -> Result<String> {205 206 207 208 209 210 211 212 213 214 215 216 217 let mut cmd = self218 .cmd_escalation(219 220 EscalationStrategy::Su,221 "which",222 )223 .await?;224 cmd.arg(command);225 cmd.run_string().await226 }227 pub async fn read_file_value<D: FromStr>(&self, path: impl AsRef<OsStr>) -> Result<D>228 where229 <D as FromStr>::Err: Display,230 {231 let text = self.read_file_text(path).await?;232 D::from_str(&text).map_err(|e| anyhow!("failed to parse value: {e}"))233 }234 pub async fn cmd(&self, cmd: impl AsRef<OsStr>) -> Result<MyCommand> {235 self.cmd_escalation(self.escalation_strategy().await?, cmd)236 .await237 }238 pub async fn cmd_escalation(239 &self,240 escalation: EscalationStrategy,241 cmd: impl AsRef<OsStr>,242 ) -> Result<MyCommand> {243 if self.local {244 Ok(MyCommand::new(escalation, cmd))245 } else {246 let session = self.open_session().await?;247 Ok(MyCommand::new_on(escalation, cmd, session))248 }249 }250 pub async fn nix_cmd(&self) -> Result<MyCommand> {251 let mut nix = self.cmd("nix").await?;252 nix.args([253 "--extra-experimental-features",254 "nix-command",255 "--extra-experimental-features",256 "flakes",257 ]);258 Ok(nix)259 }260261 pub async fn decrypt(&self, data: SecretData) -> Result<Vec<u8>> {262 ensure!(data.encrypted, "secret is not encrypted");263 let mut cmd = self.cmd("fleet-install-secrets").await?;264 cmd.arg("decrypt").eqarg("--secret", data.to_string());265 let encoded = cmd266 .sudo()267 .run_string()268 .await269 .context("failed to call remote host for decrypt")?;270 let data: SecretData = encoded.parse().map_err(|e| anyhow!("{e}"))?;271 ensure!(!data.encrypted, "secret came out encrypted");272 Ok(data.data)273 }274 pub async fn reencrypt(&self, data: SecretData, targets: Vec<String>) -> Result<SecretData> {275 ensure!(data.encrypted, "secret is not encrypted");276 let mut cmd = self.cmd("fleet-install-secrets").await?;277 cmd.arg("reencrypt").eqarg("--secret", data.to_string());278 for target in targets {279 let key = self.config.key(&target).await?;280 cmd.eqarg("--targets", key);281 }282 let encoded = cmd283 .sudo()284 .run_string()285 .await286 .context("failed to call remote host for decrypt")?;287 let data: SecretData = encoded.parse().map_err(|e| anyhow!("{e}"))?;288 ensure!(data.encrypted, "secret came out not encrypted");289 Ok(data)290 }291 292 pub async fn remote_derivation(&self, path: &PathBuf) -> Result<PathBuf> {293 if self.local {294 295 return Ok(path.to_owned());296 }297 let mut nix = MyCommand::new(298 299 EscalationStrategy::Su,300 "nix",301 );302 nix.arg("copy").arg("--substitute-on-destination");303304 match self.deploy_kind().await? {305 DeployKind::Fleet | DeployKind::UpgradeToFleet => {306 nix.comparg("--to", format!("ssh-ng://{}", self.name));307 }308 DeployKind::NixosInstall => {309 nix310 311 .arg("--no-check-sigs")312 .comparg(313 "--to",314 format!("ssh-ng://root@{}-install?remote-store=/mnt", self.name),315 );316 }317 }318 nix.arg(path);319 nix.run_nix().await.context("nix copy")?;320 Ok(path.to_owned())321 }322 pub async fn systemctl_stop(&self, name: &str) -> Result<()> {323 let mut cmd = self.cmd("systemctl").await?;324 cmd.arg("stop").arg(name);325 cmd.sudo().run().await326 }327 pub async fn systemctl_start(&self, name: &str) -> Result<()> {328 let mut cmd = self.cmd("systemctl").await?;329 cmd.arg("start").arg(name);330 cmd.sudo().run().await331 }332333 pub async fn rm_file(&self, path: impl AsRef<OsStr>, sudo: bool) -> Result<()> {334 let mut cmd = self.cmd("rm").await?;335 cmd.arg("-f").arg(path);336 if sudo {337 cmd = cmd.sudo()338 }339 cmd.run().await340 }341}342impl ConfigHost {343 344 345 pub async fn tags(&self) -> Result<Vec<String>> {346 if let Some(v) = self.groups.get() {347 return Ok(v.clone());348 }349 let Some(host_config) = &self.host_config else {350 return Ok(vec![]);351 };352 let tags: Vec<String> = nix_go_json!(host_config.tags);353354 let _ = self.groups.set(tags.clone());355356 Ok(tags)357 }358 pub async fn nixos_config(&self) -> Result<Value> {359 if let Some(v) = self.nixos_config.get() {360 return Ok(v.clone());361 }362 let Some(host_config) = &self.host_config else {363 bail!("local host has no nixos_config");364 };365 let nixos_config = nix_go!(host_config.nixos.config);366 assert_warn("nixos config evaluation", &nixos_config).await?;367368 let _ = self.nixos_config.set(nixos_config.clone());369370 Ok(nixos_config)371 }372373 pub async fn list_configured_secrets(&self) -> Result<Vec<String>> {374 let nixos = self.nixos_config().await?;375 let secrets = nix_go!(nixos.secrets);376 let mut out = Vec::new();377 for name in secrets.list_fields().await? {378 let secret = nix_go!(secrets[{ name }]);379 let is_shared: bool = nix_go_json!(secret.shared);380 if is_shared {381 continue;382 }383 out.push(name);384 }385 Ok(out)386 }387 pub async fn secret_field(&self, name: &str) -> Result<Value> {388 let nixos = self.nixos_config().await?;389 Ok(nix_go!(nixos.secrets[{ name }]))390 }391392 393 pub async fn pkgs(&self) -> Result<Value> {394 if let Some(value) = &self.pkgs_override {395 return Ok(value.clone());396 }397 let Some(host_config) = &self.host_config else {398 bail!("local host has no host_config");399 };400 401 Ok(nix_go!(host_config.nixos.options._module.args.value.pkgs))402 }403}404405impl Config {406 pub async fn tagged_hostnames(&self, tag: &str) -> Result<Vec<String>> {407 let config = &self.config_field;408 let tagged: Vec<String> = nix_go_json!(config.taggedWith[{ tag }]);409 Ok(tagged)410 }411 pub async fn expand_owner_set(&self, owners: Vec<String>) -> Result<BTreeSet<String>> {412 let mut out = BTreeSet::new();413 for owner in owners {414 if let Some(tag) = owner.strip_prefix('@') {415 let hosts = self.tagged_hostnames(tag).await?;416 out.extend(hosts);417 } else {418 out.insert(owner);419 }420 }421 Ok(out)422 }423 pub fn local_host(&self) -> ConfigHost {424 ConfigHost {425 config: self.clone(),426 name: "<virtual localhost>".to_owned(),427 host_config: None,428 nixos_config: OnceCell::new(),429 groups: {430 let cell = OnceCell::new();431 let _ = cell.set(vec![]);432 cell433 },434 pkgs_override: Some(self.default_pkgs.clone()),435436 local: true,437 session: OnceLock::new(),438 deploy_kind: OnceCell::new(),439 }440 }441442 pub async fn host(&self, name: &str) -> Result<ConfigHost> {443 let config = &self.config_field;444 let host_config = nix_go!(config.hosts[{ name }]);445446 Ok(ConfigHost {447 config: self.clone(),448 name: name.to_owned(),449 host_config: Some(host_config),450 nixos_config: OnceCell::new(),451 groups: OnceCell::new(),452 pkgs_override: None,453454 455 local: self.localhost == name,456 session: OnceLock::new(),457 deploy_kind: OnceCell::new(),458 })459 }460 pub async fn list_hosts(&self) -> Result<Vec<ConfigHost>> {461 let config = &self.config_field;462 let names = nix_go!(config.hosts).list_fields().await?;463 let mut out = vec![];464 for name in names {465 out.push(self.host(&name).await?);466 }467 Ok(out)468 }469 470 pub async fn system_config(&self, host: &str) -> Result<Value> {471 let fleet_field = &self.config_field;472 Ok(nix_go!(fleet_field.hosts[{ host }].nixos.config))473 }474475 476 pub async fn list_configured_shared(&self) -> Result<Vec<String>> {477 let config_field = &self.config_field;478 Ok(nix_go!(config_field.sharedSecrets).list_fields().await?)479 }480 481 pub fn list_shared(&self) -> Vec<String> {482 let data = self.data();483 data.shared_secrets.keys().cloned().collect()484 }485 pub fn has_shared(&self, name: &str) -> bool {486 let data = self.data();487 data.shared_secrets.contains_key(name)488 }489 pub fn replace_shared(&self, name: String, shared: FleetSharedSecret) {490 let mut data = self.data_mut();491 data.shared_secrets.insert(name.to_owned(), shared);492 }493 pub fn remove_shared(&self, secret: &str) {494 let mut data = self.data_mut();495 data.shared_secrets.remove(secret);496 }497498 pub fn list_secrets(&self, host: &str) -> Vec<String> {499 let data = self.data();500 let Some(secrets) = data.host_secrets.get(host) else {501 return Vec::new();502 };503 secrets.keys().cloned().collect()504 }505506 pub fn has_secret(&self, host: &str, secret: &str) -> bool {507 let data = self.data();508 let Some(host_secrets) = data.host_secrets.get(host) else {509 return false;510 };511 host_secrets.contains_key(secret)512 }513 pub fn insert_secret(&self, host: &str, secret: String, value: FleetSecret) {514 let mut data = self.data_mut();515 let host_secrets = data.host_secrets.entry(host.to_owned()).or_default();516 host_secrets.insert(secret, value);517 }518519 pub fn host_secret(&self, host: &str, secret: &str) -> Result<FleetSecret> {520 let data = self.data();521 let Some(host_secrets) = data.host_secrets.get(host) else {522 bail!("no secrets for machine {host}");523 };524 let Some(secret) = host_secrets.get(secret) else {525 bail!("machine {host} has no secret {secret}");526 };527 Ok(secret.clone())528 }529 pub fn shared_secret(&self, secret: &str) -> Result<FleetSharedSecret> {530 let data = self.data();531 let Some(secret) = data.shared_secrets.get(secret) else {532 bail!("no shared secret {secret}");533 };534 Ok(secret.clone())535 }536 pub async fn shared_secret_expected_owners(&self, secret: &str) -> Result<Vec<String>> {537 let config_field = &self.config_field;538 Ok(nix_go_json!(539 config_field.sharedSecrets[{ secret }].expectedOwners540 ))541 }542543 544 545 546 547 548 549 550 pub fn data(&self) -> MutexGuard<FleetData> {551 self.data.lock().unwrap()552 }553 pub fn data_mut(&self) -> MutexGuard<FleetData> {554 self.data.lock().unwrap()555 }556 pub fn save(&self) -> Result<()> {557 let mut tempfile = NamedTempFile::new_in(self.directory.clone()).context("failed to create updated version of fleet.nix in the same directory as original.\nDo you have write access to it? Access only to the fleet.nix won't be enough, the directory is used for atomic overwrite operation.\nIt is not recommended to use fleet by root anyway, move fleet project to your home directory.")?;558 let data = nixlike::serialize(&self.data() as &FleetData)?;559 tempfile.write_all(560 format!(561 "# This file contains fleet state and shouldn't be edited by hand\n\n{}\n\n# vim: ts=2 et nowrap\n",562 data563 )564 .as_bytes(),565 )?;566 let mut fleet_data_path = self.directory.clone();567 fleet_data_path.push("fleet.nix");568 tempfile.persist(fleet_data_path)?;569 Ok(())570 }571}