// SPDX-FileCopyrightText: 2020 Serokell // // SPDX-License-Identifier: MPL-2.0 use std::borrow::Cow; use std::path::PathBuf; use merge::Merge; use thiserror::Error; #[macro_export] macro_rules! good_panic { ($($tts:tt)*) => {{ error!($($tts)*); std::process::exit(1); }} } pub mod data; pub mod deploy; pub mod push; #[derive(Debug)] pub struct CmdOverrides { pub ssh_user: Option, pub profile_user: Option, pub ssh_opts: Option, pub fast_connection: Option, pub auto_rollback: Option, pub hostname: Option, pub magic_rollback: Option, pub temp_path: Option, pub confirm_timeout: Option, } #[derive(PartialEq, Debug)] pub struct DeployFlake<'a> { pub repo: &'a str, pub node: Option<&'a str>, pub profile: Option<&'a str>, } pub fn parse_flake(flake: &str) -> DeployFlake { let flake_fragment_start = flake.find('#'); let (repo, maybe_fragment) = match flake_fragment_start { Some(s) => (&flake[..s], Some(&flake[s + 1..])), None => (flake, None), }; let (node, profile) = match maybe_fragment { Some(fragment) => { let fragment_profile_start = fragment.find('.'); match fragment_profile_start { Some(s) => (Some(&fragment[..s]), Some(&fragment[s + 1..])), None => (Some(fragment), None), } } None => (None, None), }; DeployFlake { repo, node, profile, } } #[test] fn test_parse_flake() { assert_eq!( parse_flake("../deploy/examples/system#example"), DeployFlake { repo: "../deploy/examples/system", node: Some("example"), profile: None } ); assert_eq!( parse_flake("../deploy/examples/system#example.system"), DeployFlake { repo: "../deploy/examples/system", node: Some("example"), profile: Some("system") } ); assert_eq!( parse_flake("../deploy/examples/system"), DeployFlake { repo: "../deploy/examples/system", node: None, profile: None, } ); } #[derive(Debug)] pub struct DeployData<'a> { pub node_name: &'a str, pub node: &'a data::Node, pub profile_name: &'a str, pub profile: &'a data::Profile, pub cmd_overrides: &'a CmdOverrides, pub merged_settings: data::GenericSettings, } #[derive(Debug)] pub struct DeployDefs<'a> { pub ssh_user: Cow<'a, str>, pub profile_user: Cow<'a, str>, pub profile_path: Cow<'a, str>, pub current_exe: PathBuf, pub sudo: Option, } #[derive(Error, Debug)] pub enum DeployDataDefsError { #[error("Neither `user` nor `sshUser` are set for profile {0} of node {1}")] NoProfileUser(String, String), #[error("Error reading current executable path: {0}")] ExecutablePathNotFound(std::io::Error), #[error("Executable was not in the Nix store")] NotNixStored, } impl<'a> DeployData<'a> { pub fn defs(&'a self) -> Result, DeployDataDefsError> { let ssh_user: Cow = match self.merged_settings.ssh_user { Some(ref u) => u.into(), None => whoami::username().into(), }; let profile_user: Cow = match self.merged_settings.user { Some(ref x) => x.into(), None => match self.merged_settings.ssh_user { Some(ref x) => x.into(), None => { return Err(DeployDataDefsError::NoProfileUser( self.profile_name.to_owned(), self.node_name.to_owned(), )) } }, }; let profile_path: Cow = match self.profile.profile_settings.profile_path { None => match &profile_user[..] { "root" => format!("/nix/var/nix/profiles/{}", self.profile_name).into(), _ => format!( "/nix/var/nix/profiles/per-user/{}/{}", profile_user, self.profile_name ) .into(), }, Some(ref x) => x.into(), }; let sudo: Option = match self.merged_settings.user { Some(ref user) if user != &ssh_user => Some(format!("sudo -u {}", user)), _ => None, }; let current_exe = std::env::current_exe().map_err(DeployDataDefsError::ExecutablePathNotFound)?; if !current_exe.starts_with("/nix/store/") { return Err(DeployDataDefsError::NotNixStored); } Ok(DeployDefs { ssh_user, profile_user, profile_path, current_exe, sudo, }) } } pub fn make_deploy_data<'a, 's>( top_settings: &'s data::GenericSettings, node: &'a data::Node, node_name: &'a str, profile: &'a data::Profile, profile_name: &'a str, cmd_overrides: &'a CmdOverrides, ) -> DeployData<'a> { let mut merged_settings = top_settings.clone(); merged_settings.merge(node.generic_settings.clone()); merged_settings.merge(profile.generic_settings.clone()); if cmd_overrides.ssh_user.is_some() { merged_settings.ssh_user = cmd_overrides.ssh_user.clone(); } if cmd_overrides.profile_user.is_some() { merged_settings.user = cmd_overrides.profile_user.clone(); } if let Some(ref ssh_opts) = cmd_overrides.ssh_opts { merged_settings.ssh_opts = ssh_opts.split(' ').map(|x| x.to_owned()).collect(); } if let Some(fast_connection) = cmd_overrides.fast_connection { merged_settings.fast_connection = Some(fast_connection); } if let Some(auto_rollback) = cmd_overrides.auto_rollback { merged_settings.auto_rollback = Some(auto_rollback); } if let Some(magic_rollback) = cmd_overrides.magic_rollback { merged_settings.magic_rollback = Some(magic_rollback); } DeployData { profile, profile_name, node, node_name, cmd_overrides, merged_settings, } } #[derive(Error, Debug)] pub enum DeployPathToActivatePathError { #[error("Deploy path did not have a parent directory")] PathTooShort, #[error("Deploy path was not valid utf8")] InvalidUtf8, } pub fn deploy_path_to_activate_path_str( deploy_path: &std::path::Path, ) -> Result { Ok(format!( "{}/activate", deploy_path .parent() .ok_or(DeployPathToActivatePathError::PathTooShort)? .to_str() .ok_or(DeployPathToActivatePathError::InvalidUtf8)? .to_owned() )) } #[test] fn test_activate_path_generation() { match deploy_path_to_activate_path_str(&std::path::PathBuf::from( "/blah/blah/deploy-rs/bin/deploy", )) { Err(_) => panic!(""), Ok(x) => assert_eq!(x, "/blah/blah/deploy-rs/bin/activate".to_string()), } }