// 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()),
}
}