fix & fmt
This commit is contained in:
parent
48646c7cad
commit
f6d90695f6
26 changed files with 660 additions and 323 deletions
|
@ -4,9 +4,9 @@ use serde::Serialize;
|
||||||
use tokio::fs::remove_file;
|
use tokio::fs::remove_file;
|
||||||
use tokio::process::Command;
|
use tokio::process::Command;
|
||||||
|
|
||||||
use crate::{execute_command};
|
use crate::execute_command;
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable, ActionState, Action};
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable};
|
||||||
|
|
||||||
const SERVICE_SRC: &str = "/nix/var/nix/profiles/default/lib/systemd/system/nix-daemon.service";
|
const SERVICE_SRC: &str = "/nix/var/nix/profiles/default/lib/systemd/system/nix-daemon.service";
|
||||||
const SOCKET_SRC: &str = "/nix/var/nix/profiles/default/lib/systemd/system/nix-daemon.socket";
|
const SOCKET_SRC: &str = "/nix/var/nix/profiles/default/lib/systemd/system/nix-daemon.socket";
|
||||||
|
@ -62,49 +62,53 @@ impl Actionable for ConfigureNixDaemonService {
|
||||||
.arg("--create")
|
.arg("--create")
|
||||||
.arg("--prefix=/nix/var/nix"),
|
.arg("--prefix=/nix/var/nix"),
|
||||||
)
|
)
|
||||||
.await.map_err(Self::Error::CommandFailed)?;
|
.await
|
||||||
|
|
||||||
execute_command(
|
|
||||||
Command::new("systemctl").arg("link").arg(SERVICE_SRC),
|
|
||||||
)
|
|
||||||
.await.map_err(Self::Error::CommandFailed)?;
|
|
||||||
|
|
||||||
execute_command(Command::new("systemctl").arg("link").arg(SOCKET_SRC)).await
|
|
||||||
.map_err(Self::Error::CommandFailed)?;
|
.map_err(Self::Error::CommandFailed)?;
|
||||||
|
|
||||||
execute_command(Command::new("systemctl").arg("daemon-reload")).await
|
execute_command(Command::new("systemctl").arg("link").arg(SERVICE_SRC))
|
||||||
|
.await
|
||||||
|
.map_err(Self::Error::CommandFailed)?;
|
||||||
|
|
||||||
|
execute_command(Command::new("systemctl").arg("link").arg(SOCKET_SRC))
|
||||||
|
.await
|
||||||
|
.map_err(Self::Error::CommandFailed)?;
|
||||||
|
|
||||||
|
execute_command(Command::new("systemctl").arg("daemon-reload"))
|
||||||
|
.await
|
||||||
.map_err(Self::Error::CommandFailed)?;
|
.map_err(Self::Error::CommandFailed)?;
|
||||||
|
|
||||||
*action_state = ActionState::Completed;
|
*action_state = ActionState::Completed;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { action_state } = self;
|
let Self { action_state } = self;
|
||||||
tracing::info!("Unconfiguring nix daemon service");
|
tracing::info!("Unconfiguring nix daemon service");
|
||||||
|
|
||||||
// We don't need to do this! Systemd does it for us! (In fact, it's an error if we try to do this...)
|
// We don't need to do this! Systemd does it for us! (In fact, it's an error if we try to do this...)
|
||||||
execute_command(Command::new("systemctl").args(["disable", SOCKET_SRC])).await
|
execute_command(Command::new("systemctl").args(["disable", SOCKET_SRC]))
|
||||||
|
.await
|
||||||
.map_err(Self::Error::CommandFailed)?;
|
.map_err(Self::Error::CommandFailed)?;
|
||||||
|
|
||||||
execute_command(
|
execute_command(Command::new("systemctl").args(["disable", SERVICE_SRC]))
|
||||||
Command::new("systemctl").args(["disable", SERVICE_SRC]),
|
.await
|
||||||
)
|
.map_err(Self::Error::CommandFailed)?;
|
||||||
.await.map_err(Self::Error::CommandFailed)?;
|
|
||||||
|
|
||||||
execute_command(
|
execute_command(
|
||||||
Command::new("systemd-tmpfiles")
|
Command::new("systemd-tmpfiles")
|
||||||
.arg("--remove")
|
.arg("--remove")
|
||||||
.arg("--prefix=/nix/var/nix"),
|
.arg("--prefix=/nix/var/nix"),
|
||||||
)
|
)
|
||||||
.await.map_err(Self::Error::CommandFailed)?;
|
.await
|
||||||
|
.map_err(Self::Error::CommandFailed)?;
|
||||||
|
|
||||||
remove_file(TMPFILES_DEST).await
|
remove_file(TMPFILES_DEST)
|
||||||
|
.await
|
||||||
.map_err(|e| Self::Error::RemoveFile(PathBuf::from(TMPFILES_DEST), e))?;
|
.map_err(|e| Self::Error::RemoveFile(PathBuf::from(TMPFILES_DEST), e))?;
|
||||||
|
|
||||||
execute_command(Command::new("systemctl").arg("daemon-reload")).await
|
execute_command(Command::new("systemctl").arg("daemon-reload"))
|
||||||
|
.await
|
||||||
.map_err(Self::Error::CommandFailed)?;
|
.map_err(Self::Error::CommandFailed)?;
|
||||||
|
|
||||||
*action_state = ActionState::Reverted;
|
*action_state = ActionState::Reverted;
|
||||||
|
@ -112,8 +116,6 @@ impl Actionable for ConfigureNixDaemonService {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
impl From<ConfigureNixDaemonService> for Action {
|
impl From<ConfigureNixDaemonService> for Action {
|
||||||
fn from(v: ConfigureNixDaemonService) -> Self {
|
fn from(v: ConfigureNixDaemonService) -> Self {
|
||||||
Action::ConfigureNixDaemonService(v)
|
Action::ConfigureNixDaemonService(v)
|
||||||
|
@ -128,16 +130,21 @@ pub enum ConfigureNixDaemonServiceError {
|
||||||
std::path::PathBuf,
|
std::path::PathBuf,
|
||||||
#[source]
|
#[source]
|
||||||
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
std::io::Error
|
std::io::Error,
|
||||||
),
|
),
|
||||||
#[error("Command failed to execute")]
|
#[error("Command failed to execute")]
|
||||||
CommandFailed(
|
CommandFailed(
|
||||||
#[source]
|
#[source]
|
||||||
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
std::io::Error
|
std::io::Error,
|
||||||
),
|
),
|
||||||
#[error("Remove file `{0}`")]
|
#[error("Remove file `{0}`")]
|
||||||
RemoveFile(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
RemoveFile(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("No supported init system found")]
|
#[error("No supported init system found")]
|
||||||
InitNotSupported,
|
InitNotSupported,
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
|
|
||||||
use std::os::unix::prelude::PermissionsExt;
|
use std::os::unix::prelude::PermissionsExt;
|
||||||
use std::path::{Path, PathBuf};
|
use std::path::{Path, PathBuf};
|
||||||
|
|
||||||
|
@ -6,9 +5,7 @@ use nix::unistd::{chown, Group, User};
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use tokio::fs::{create_dir, remove_dir_all};
|
use tokio::fs::{create_dir, remove_dir_all};
|
||||||
|
|
||||||
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable};
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable, ActionState, Action};
|
|
||||||
|
|
||||||
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
||||||
pub struct CreateDirectory {
|
pub struct CreateDirectory {
|
||||||
|
@ -100,12 +97,10 @@ impl Actionable for CreateDirectory {
|
||||||
.await
|
.await
|
||||||
.map_err(|e| Self::Error::SetPermissions(*mode, path.to_owned(), e))?;
|
.map_err(|e| Self::Error::SetPermissions(*mode, path.to_owned(), e))?;
|
||||||
|
|
||||||
|
|
||||||
*action_state = ActionState::Completed;
|
*action_state = ActionState::Completed;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self {
|
let Self {
|
||||||
|
@ -126,32 +121,66 @@ impl Actionable for CreateDirectory {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
impl From<CreateDirectory> for Action {
|
impl From<CreateDirectory> for Action {
|
||||||
fn from(v: CreateDirectory) -> Self {
|
fn from(v: CreateDirectory) -> Self {
|
||||||
Action::CreateDirectory(v)
|
Action::CreateDirectory(v)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug, thiserror::Error, Serialize)]
|
#[derive(Debug, thiserror::Error, Serialize)]
|
||||||
pub enum CreateDirectoryError {
|
pub enum CreateDirectoryError {
|
||||||
#[error("Directory exists `{0}`")]
|
#[error("Directory exists `{0}`")]
|
||||||
Exists(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
Exists(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("Creating directory `{0}`")]
|
#[error("Creating directory `{0}`")]
|
||||||
Creating(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
Creating(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("Removing directory `{0}`")]
|
#[error("Removing directory `{0}`")]
|
||||||
Removing(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
Removing(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("Set mode `{0}` on `{1}`")]
|
#[error("Set mode `{0}` on `{1}`")]
|
||||||
SetPermissions(u32, std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
SetPermissions(
|
||||||
|
u32,
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("Chowning directory `{0}`")]
|
#[error("Chowning directory `{0}`")]
|
||||||
Chown(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] nix::errno::Errno),
|
Chown(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
nix::errno::Errno,
|
||||||
|
),
|
||||||
#[error("Getting uid for user `{0}`")]
|
#[error("Getting uid for user `{0}`")]
|
||||||
UserId(String, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] nix::errno::Errno),
|
UserId(
|
||||||
|
String,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
nix::errno::Errno,
|
||||||
|
),
|
||||||
#[error("Getting user `{0}`")]
|
#[error("Getting user `{0}`")]
|
||||||
NoUser(String),
|
NoUser(String),
|
||||||
#[error("Getting gid for group `{0}`")]
|
#[error("Getting gid for group `{0}`")]
|
||||||
GroupId(String, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] nix::errno::Errno),
|
GroupId(
|
||||||
|
String,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
nix::errno::Errno,
|
||||||
|
),
|
||||||
#[error("Getting group `{0}`")]
|
#[error("Getting group `{0}`")]
|
||||||
NoGroup(String),
|
NoGroup(String),
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,11 +2,11 @@ use nix::unistd::{chown, Group, User};
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use std::path::{Path, PathBuf};
|
use std::path::{Path, PathBuf};
|
||||||
use tokio::{
|
use tokio::{
|
||||||
fs::{OpenOptions, remove_file},
|
fs::{remove_file, OpenOptions},
|
||||||
io::AsyncWriteExt,
|
io::AsyncWriteExt,
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::{actions::{ActionState, Action, ActionError}};
|
use crate::actions::{Action, ActionState};
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable};
|
use crate::actions::{ActionDescription, Actionable};
|
||||||
|
|
||||||
|
@ -111,7 +111,6 @@ impl Actionable for CreateFile {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self {
|
let Self {
|
||||||
|
@ -126,7 +125,8 @@ impl Actionable for CreateFile {
|
||||||
|
|
||||||
tracing::trace!(path = %path.display(), "Deleting file");
|
tracing::trace!(path = %path.display(), "Deleting file");
|
||||||
|
|
||||||
remove_file(&path).await
|
remove_file(&path)
|
||||||
|
.await
|
||||||
.map_err(|e| Self::Error::RemoveFile(path.to_owned(), e))?;
|
.map_err(|e| Self::Error::RemoveFile(path.to_owned(), e))?;
|
||||||
|
|
||||||
*action_state = ActionState::Reverted;
|
*action_state = ActionState::Reverted;
|
||||||
|
@ -145,19 +145,49 @@ pub enum CreateFileError {
|
||||||
#[error("File exists `{0}`")]
|
#[error("File exists `{0}`")]
|
||||||
Exists(std::path::PathBuf),
|
Exists(std::path::PathBuf),
|
||||||
#[error("Remove file `{0}`")]
|
#[error("Remove file `{0}`")]
|
||||||
RemoveFile(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
RemoveFile(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("Open file `{0}`")]
|
#[error("Open file `{0}`")]
|
||||||
OpenFile(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
OpenFile(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("Write file `{0}`")]
|
#[error("Write file `{0}`")]
|
||||||
WriteFile(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
WriteFile(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("Getting uid for user `{0}`")]
|
#[error("Getting uid for user `{0}`")]
|
||||||
UserId(String, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] nix::errno::Errno),
|
UserId(
|
||||||
|
String,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
nix::errno::Errno,
|
||||||
|
),
|
||||||
#[error("Getting user `{0}`")]
|
#[error("Getting user `{0}`")]
|
||||||
NoUser(String),
|
NoUser(String),
|
||||||
#[error("Getting gid for group `{0}`")]
|
#[error("Getting gid for group `{0}`")]
|
||||||
GroupId(String, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] nix::errno::Errno),
|
GroupId(
|
||||||
|
String,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
nix::errno::Errno,
|
||||||
|
),
|
||||||
#[error("Getting group `{0}`")]
|
#[error("Getting group `{0}`")]
|
||||||
NoGroup(String),
|
NoGroup(String),
|
||||||
#[error("Chowning directory `{0}`")]
|
#[error("Chowning directory `{0}`")]
|
||||||
Chown(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] nix::errno::Errno),
|
Chown(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
nix::errno::Errno,
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use tokio::process::Command;
|
use tokio::process::Command;
|
||||||
|
|
||||||
use crate::{HarmonicError, execute_command};
|
use crate::execute_command;
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable, ActionState, Action, ActionError};
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable};
|
||||||
|
|
||||||
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
||||||
pub struct CreateGroup {
|
pub struct CreateGroup {
|
||||||
|
@ -15,7 +15,11 @@ pub struct CreateGroup {
|
||||||
impl CreateGroup {
|
impl CreateGroup {
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
pub fn plan(name: String, gid: usize) -> Self {
|
pub fn plan(name: String, gid: usize) -> Self {
|
||||||
Self { name, gid, action_state: ActionState::Planned }
|
Self {
|
||||||
|
name,
|
||||||
|
gid,
|
||||||
|
action_state: ActionState::Planned,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -23,7 +27,11 @@ impl CreateGroup {
|
||||||
impl Actionable for CreateGroup {
|
impl Actionable for CreateGroup {
|
||||||
type Error = CreateGroupError;
|
type Error = CreateGroupError;
|
||||||
fn description(&self) -> Vec<ActionDescription> {
|
fn description(&self) -> Vec<ActionDescription> {
|
||||||
let Self { name, gid, action_state: _ } = &self;
|
let Self {
|
||||||
|
name,
|
||||||
|
gid,
|
||||||
|
action_state: _,
|
||||||
|
} = &self;
|
||||||
vec![ActionDescription::new(
|
vec![ActionDescription::new(
|
||||||
format!("Create group {name} with GID {gid}"),
|
format!("Create group {name} with GID {gid}"),
|
||||||
vec![format!(
|
vec![format!(
|
||||||
|
@ -34,12 +42,15 @@ impl Actionable for CreateGroup {
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn execute(&mut self) -> Result<(), Self::Error> {
|
async fn execute(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { name, gid, action_state } = self;
|
let Self {
|
||||||
|
name,
|
||||||
execute_command(
|
gid,
|
||||||
Command::new("groupadd").args(["-g", &gid.to_string(), "--system", &name]),
|
action_state,
|
||||||
).await.map_err(CreateGroupError::Command)?;
|
} = self;
|
||||||
|
|
||||||
|
execute_command(Command::new("groupadd").args(["-g", &gid.to_string(), "--system", &name]))
|
||||||
|
.await
|
||||||
|
.map_err(CreateGroupError::Command)?;
|
||||||
|
|
||||||
*action_state = ActionState::Completed;
|
*action_state = ActionState::Completed;
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -47,11 +58,15 @@ impl Actionable for CreateGroup {
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { name, gid: _, action_state } = self;
|
let Self {
|
||||||
|
name,
|
||||||
|
gid: _,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
|
|
||||||
execute_command(
|
execute_command(Command::new("groupdel").arg(&name))
|
||||||
Command::new("groupdel").arg(&name),
|
.await
|
||||||
).await.map_err(CreateGroupError::Command)?;
|
.map_err(CreateGroupError::Command)?;
|
||||||
|
|
||||||
*action_state = ActionState::Reverted;
|
*action_state = ActionState::Reverted;
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -67,5 +82,9 @@ impl From<CreateGroup> for Action {
|
||||||
#[derive(Debug, thiserror::Error, Serialize)]
|
#[derive(Debug, thiserror::Error, Serialize)]
|
||||||
pub enum CreateGroupError {
|
pub enum CreateGroupError {
|
||||||
#[error("Failed to execute command")]
|
#[error("Failed to execute command")]
|
||||||
Command(#[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error)
|
Command(
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
}
|
}
|
|
@ -2,14 +2,15 @@ use nix::unistd::{chown, Group, User};
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use std::{
|
use std::{
|
||||||
io::SeekFrom,
|
io::SeekFrom,
|
||||||
path::{Path, PathBuf}, os::unix::prelude::PermissionsExt, f32::consts::E,
|
os::unix::prelude::PermissionsExt,
|
||||||
|
path::{Path, PathBuf},
|
||||||
};
|
};
|
||||||
use tokio::{
|
use tokio::{
|
||||||
fs::{create_dir_all, OpenOptions, remove_file},
|
fs::{remove_file, OpenOptions},
|
||||||
io::{AsyncSeekExt, AsyncWriteExt, AsyncReadExt},
|
io::{AsyncReadExt, AsyncSeekExt, AsyncWriteExt},
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::{HarmonicError, actions::{ActionState, Action, ActionError}};
|
use crate::actions::{Action, ActionState};
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable};
|
use crate::actions::{ActionDescription, Actionable};
|
||||||
|
|
||||||
|
@ -110,12 +111,10 @@ impl Actionable for CreateOrAppendFile {
|
||||||
tracing::trace!(path = %path.display(), "Chowning");
|
tracing::trace!(path = %path.display(), "Chowning");
|
||||||
chown(path, Some(uid), Some(gid)).map_err(|e| Self::Error::Chown(path.clone(), e))?;
|
chown(path, Some(uid), Some(gid)).map_err(|e| Self::Error::Chown(path.clone(), e))?;
|
||||||
|
|
||||||
|
|
||||||
*action_state = ActionState::Completed;
|
*action_state = ActionState::Completed;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self {
|
let Self {
|
||||||
|
@ -137,7 +136,8 @@ impl Actionable for CreateOrAppendFile {
|
||||||
.map_err(|e| Self::Error::ReadFile(path.to_owned(), e))?;
|
.map_err(|e| Self::Error::ReadFile(path.to_owned(), e))?;
|
||||||
|
|
||||||
let mut file_contents = String::default();
|
let mut file_contents = String::default();
|
||||||
file.read_to_string(&mut file_contents).await
|
file.read_to_string(&mut file_contents)
|
||||||
|
.await
|
||||||
.map_err(|e| Self::Error::SeekFile(path.to_owned(), e))?;
|
.map_err(|e| Self::Error::SeekFile(path.to_owned(), e))?;
|
||||||
|
|
||||||
if let Some(start) = file_contents.rfind(buf.as_str()) {
|
if let Some(start) = file_contents.rfind(buf.as_str()) {
|
||||||
|
@ -146,7 +146,9 @@ impl Actionable for CreateOrAppendFile {
|
||||||
}
|
}
|
||||||
|
|
||||||
if buf.is_empty() {
|
if buf.is_empty() {
|
||||||
remove_file(&path).await.map_err(|e| Self::Error::RemoveFile(path.to_owned(), e))?;
|
remove_file(&path)
|
||||||
|
.await
|
||||||
|
.map_err(|e| Self::Error::RemoveFile(path.to_owned(), e))?;
|
||||||
} else {
|
} else {
|
||||||
file.seek(SeekFrom::Start(0))
|
file.seek(SeekFrom::Start(0))
|
||||||
.await
|
.await
|
||||||
|
@ -161,7 +163,6 @@ impl Actionable for CreateOrAppendFile {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
impl From<CreateOrAppendFile> for Action {
|
impl From<CreateOrAppendFile> for Action {
|
||||||
fn from(v: CreateOrAppendFile) -> Self {
|
fn from(v: CreateOrAppendFile) -> Self {
|
||||||
Action::CreateOrAppendFile(v)
|
Action::CreateOrAppendFile(v)
|
||||||
|
@ -171,25 +172,71 @@ impl From<CreateOrAppendFile> for Action {
|
||||||
#[derive(Debug, thiserror::Error, Serialize)]
|
#[derive(Debug, thiserror::Error, Serialize)]
|
||||||
pub enum CreateOrAppendFileError {
|
pub enum CreateOrAppendFileError {
|
||||||
#[error("Remove file `{0}`")]
|
#[error("Remove file `{0}`")]
|
||||||
RemoveFile(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
RemoveFile(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("Remove file `{0}`")]
|
#[error("Remove file `{0}`")]
|
||||||
ReadFile(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
ReadFile(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("Open file `{0}`")]
|
#[error("Open file `{0}`")]
|
||||||
OpenFile(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
OpenFile(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("Write file `{0}`")]
|
#[error("Write file `{0}`")]
|
||||||
WriteFile(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
WriteFile(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("Seek file `{0}`")]
|
#[error("Seek file `{0}`")]
|
||||||
SeekFile(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
SeekFile(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("Getting uid for user `{0}`")]
|
#[error("Getting uid for user `{0}`")]
|
||||||
UserId(String, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] nix::errno::Errno),
|
UserId(
|
||||||
|
String,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
nix::errno::Errno,
|
||||||
|
),
|
||||||
#[error("Getting user `{0}`")]
|
#[error("Getting user `{0}`")]
|
||||||
NoUser(String),
|
NoUser(String),
|
||||||
#[error("Getting gid for group `{0}`")]
|
#[error("Getting gid for group `{0}`")]
|
||||||
GroupId(String, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] nix::errno::Errno),
|
GroupId(
|
||||||
|
String,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
nix::errno::Errno,
|
||||||
|
),
|
||||||
#[error("Getting group `{0}`")]
|
#[error("Getting group `{0}`")]
|
||||||
NoGroup(String),
|
NoGroup(String),
|
||||||
#[error("Set mode `{0}` on `{1}`")]
|
#[error("Set mode `{0}` on `{1}`")]
|
||||||
SetPermissions(u32, std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
SetPermissions(
|
||||||
|
u32,
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
#[error("Chowning directory `{0}`")]
|
#[error("Chowning directory `{0}`")]
|
||||||
Chown(std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] nix::errno::Errno),
|
Chown(
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
nix::errno::Errno,
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use tokio::process::Command;
|
use tokio::process::Command;
|
||||||
|
|
||||||
use crate::{HarmonicError, execute_command};
|
use crate::execute_command;
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable, ActionState, Action, ActionError};
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable};
|
||||||
|
|
||||||
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
||||||
pub struct CreateUser {
|
pub struct CreateUser {
|
||||||
|
@ -16,7 +16,12 @@ pub struct CreateUser {
|
||||||
impl CreateUser {
|
impl CreateUser {
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
pub fn plan(name: String, uid: usize, gid: usize) -> Self {
|
pub fn plan(name: String, uid: usize, gid: usize) -> Self {
|
||||||
Self { name, uid, gid, action_state: ActionState::Planned }
|
Self {
|
||||||
|
name,
|
||||||
|
uid,
|
||||||
|
gid,
|
||||||
|
action_state: ActionState::Planned,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -36,7 +41,12 @@ impl Actionable for CreateUser {
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn execute(&mut self) -> Result<(), Self::Error> {
|
async fn execute(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { name, uid, gid, action_state } = self;
|
let Self {
|
||||||
|
name,
|
||||||
|
uid,
|
||||||
|
gid,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
|
|
||||||
execute_command(Command::new("useradd").args([
|
execute_command(Command::new("useradd").args([
|
||||||
"--home-dir",
|
"--home-dir",
|
||||||
|
@ -56,20 +66,26 @@ impl Actionable for CreateUser {
|
||||||
"--password",
|
"--password",
|
||||||
"\"!\"",
|
"\"!\"",
|
||||||
&name.to_string(),
|
&name.to_string(),
|
||||||
])).await.map_err(Self::Error::Command)?;
|
]))
|
||||||
|
.await
|
||||||
|
.map_err(Self::Error::Command)?;
|
||||||
|
|
||||||
*action_state = ActionState::Completed;
|
*action_state = ActionState::Completed;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { name, uid: _, gid: _, action_state } = self;
|
let Self {
|
||||||
|
name,
|
||||||
|
uid: _,
|
||||||
|
gid: _,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
|
|
||||||
execute_command(Command::new("userdel").args([
|
execute_command(Command::new("userdel").args([&name.to_string()]))
|
||||||
&name.to_string(),
|
.await
|
||||||
])).await.map_err(Self::Error::Command)?;
|
.map_err(Self::Error::Command)?;
|
||||||
|
|
||||||
*action_state = ActionState::Completed;
|
*action_state = ActionState::Completed;
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -85,5 +101,9 @@ impl From<CreateUser> for Action {
|
||||||
#[derive(Debug, thiserror::Error, Serialize)]
|
#[derive(Debug, thiserror::Error, Serialize)]
|
||||||
pub enum CreateUserError {
|
pub enum CreateUserError {
|
||||||
#[error("Failed to execute command")]
|
#[error("Failed to execute command")]
|
||||||
Command(#[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error)
|
Command(
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,13 +1,11 @@
|
||||||
use std::path::{PathBuf};
|
use std::path::PathBuf;
|
||||||
|
|
||||||
use bytes::Buf;
|
use bytes::Buf;
|
||||||
use reqwest::Url;
|
use reqwest::Url;
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use tokio::task::{spawn_blocking, JoinError};
|
use tokio::task::{spawn_blocking, JoinError};
|
||||||
|
|
||||||
use crate::HarmonicError;
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable};
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable, ActionState, Action, ActionError};
|
|
||||||
|
|
||||||
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
||||||
pub struct FetchNix {
|
pub struct FetchNix {
|
||||||
|
@ -22,7 +20,11 @@ impl FetchNix {
|
||||||
// TODO(@hoverbear): Check URL exists?
|
// TODO(@hoverbear): Check URL exists?
|
||||||
// TODO(@hoverbear): Check tempdir exists
|
// TODO(@hoverbear): Check tempdir exists
|
||||||
|
|
||||||
Ok(Self { url, destination, action_state: ActionState::Planned })
|
Ok(Self {
|
||||||
|
url,
|
||||||
|
destination,
|
||||||
|
action_state: ActionState::Planned,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -30,7 +32,11 @@ impl FetchNix {
|
||||||
impl Actionable for FetchNix {
|
impl Actionable for FetchNix {
|
||||||
type Error = FetchNixError;
|
type Error = FetchNixError;
|
||||||
fn description(&self) -> Vec<ActionDescription> {
|
fn description(&self) -> Vec<ActionDescription> {
|
||||||
let Self { url, destination, action_state: _ } = &self;
|
let Self {
|
||||||
|
url,
|
||||||
|
destination,
|
||||||
|
action_state: _,
|
||||||
|
} = &self;
|
||||||
vec![ActionDescription::new(
|
vec![ActionDescription::new(
|
||||||
format!("Fetch Nix from `{url}`"),
|
format!("Fetch Nix from `{url}`"),
|
||||||
vec![format!(
|
vec![format!(
|
||||||
|
@ -42,7 +48,11 @@ impl Actionable for FetchNix {
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn execute(&mut self) -> Result<(), Self::Error> {
|
async fn execute(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { url, destination, action_state } = self;
|
let Self {
|
||||||
|
url,
|
||||||
|
destination,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
|
|
||||||
tracing::trace!(%url, "Fetching url");
|
tracing::trace!(%url, "Fetching url");
|
||||||
let res = reqwest::get(url.clone())
|
let res = reqwest::get(url.clone())
|
||||||
|
@ -67,10 +77,13 @@ impl Actionable for FetchNix {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { url: _, destination: _, action_state } = self;
|
let Self {
|
||||||
|
url: _,
|
||||||
|
destination: _,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
|
|
||||||
tracing::trace!("Nothing to do for `FetchNix` revert");
|
tracing::trace!("Nothing to do for `FetchNix` revert");
|
||||||
|
|
||||||
|
@ -91,10 +104,19 @@ pub enum FetchNixError {
|
||||||
Join(
|
Join(
|
||||||
#[from]
|
#[from]
|
||||||
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
JoinError
|
JoinError,
|
||||||
),
|
),
|
||||||
#[error("Request error")]
|
#[error("Request error")]
|
||||||
Reqwest(#[from] #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] reqwest::Error),
|
Reqwest(
|
||||||
|
#[from]
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
reqwest::Error,
|
||||||
|
),
|
||||||
#[error("Unarchiving error")]
|
#[error("Unarchiving error")]
|
||||||
Unarchive(#[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
Unarchive(
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,9 +13,7 @@ mod place_nix_configuration;
|
||||||
mod setup_default_profile;
|
mod setup_default_profile;
|
||||||
mod start_systemd_unit;
|
mod start_systemd_unit;
|
||||||
|
|
||||||
pub use configure_nix_daemon_service::{
|
pub use configure_nix_daemon_service::{ConfigureNixDaemonService, ConfigureNixDaemonServiceError};
|
||||||
ConfigureNixDaemonService, ConfigureNixDaemonServiceError,
|
|
||||||
};
|
|
||||||
pub use create_directory::{CreateDirectory, CreateDirectoryError};
|
pub use create_directory::{CreateDirectory, CreateDirectoryError};
|
||||||
pub use create_file::{CreateFile, CreateFileError};
|
pub use create_file::{CreateFile, CreateFileError};
|
||||||
pub use create_group::{CreateGroup, CreateGroupError};
|
pub use create_group::{CreateGroup, CreateGroupError};
|
||||||
|
@ -23,9 +21,7 @@ pub use create_or_append_file::{CreateOrAppendFile, CreateOrAppendFileError};
|
||||||
pub use create_user::{CreateUser, CreateUserError};
|
pub use create_user::{CreateUser, CreateUserError};
|
||||||
pub use fetch_nix::{FetchNix, FetchNixError};
|
pub use fetch_nix::{FetchNix, FetchNixError};
|
||||||
pub use move_unpacked_nix::{MoveUnpackedNix, MoveUnpackedNixError};
|
pub use move_unpacked_nix::{MoveUnpackedNix, MoveUnpackedNixError};
|
||||||
pub use place_channel_configuration::{
|
pub use place_channel_configuration::{PlaceChannelConfiguration, PlaceChannelConfigurationError};
|
||||||
PlaceChannelConfiguration, PlaceChannelConfigurationError,
|
|
||||||
};
|
|
||||||
pub use place_nix_configuration::{PlaceNixConfiguration, PlaceNixConfigurationError};
|
pub use place_nix_configuration::{PlaceNixConfiguration, PlaceNixConfigurationError};
|
||||||
pub use setup_default_profile::{SetupDefaultProfile, SetupDefaultProfileError};
|
pub use setup_default_profile::{SetupDefaultProfile, SetupDefaultProfileError};
|
||||||
pub use start_systemd_unit::{StartSystemdUnit, StartSystemdUnitError};
|
pub use start_systemd_unit::{StartSystemdUnit, StartSystemdUnitError};
|
||||||
|
|
|
@ -2,9 +2,7 @@ use std::path::{Path, PathBuf};
|
||||||
|
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
|
|
||||||
use crate::HarmonicError;
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable};
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable, ActionState, Action, ActionError};
|
|
||||||
|
|
||||||
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
||||||
pub struct MoveUnpackedNix {
|
pub struct MoveUnpackedNix {
|
||||||
|
@ -16,7 +14,10 @@ impl MoveUnpackedNix {
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
pub async fn plan(source: PathBuf) -> Result<Self, MoveUnpackedNixError> {
|
pub async fn plan(source: PathBuf) -> Result<Self, MoveUnpackedNixError> {
|
||||||
// Note: Do NOT try to check for the source/dest since the installer creates those
|
// Note: Do NOT try to check for the source/dest since the installer creates those
|
||||||
Ok(Self { source, action_state: ActionState::Planned })
|
Ok(Self {
|
||||||
|
source,
|
||||||
|
action_state: ActionState::Planned,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -24,7 +25,10 @@ impl MoveUnpackedNix {
|
||||||
impl Actionable for MoveUnpackedNix {
|
impl Actionable for MoveUnpackedNix {
|
||||||
type Error = MoveUnpackedNixError;
|
type Error = MoveUnpackedNixError;
|
||||||
fn description(&self) -> Vec<ActionDescription> {
|
fn description(&self) -> Vec<ActionDescription> {
|
||||||
let Self { source, action_state: _ } = &self;
|
let Self {
|
||||||
|
source,
|
||||||
|
action_state: _,
|
||||||
|
} = &self;
|
||||||
vec![ActionDescription::new(
|
vec![ActionDescription::new(
|
||||||
format!("Move the downloaded Nix into `/nix`"),
|
format!("Move the downloaded Nix into `/nix`"),
|
||||||
vec![format!(
|
vec![format!(
|
||||||
|
@ -36,7 +40,10 @@ impl Actionable for MoveUnpackedNix {
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn execute(&mut self) -> Result<(), Self::Error> {
|
async fn execute(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { source, action_state } = self;
|
let Self {
|
||||||
|
source,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
|
|
||||||
// TODO(@Hoverbear): I would like to make this less awful
|
// TODO(@Hoverbear): I would like to make this less awful
|
||||||
let found_nix_paths =
|
let found_nix_paths =
|
||||||
|
@ -58,10 +65,12 @@ impl Actionable for MoveUnpackedNix {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { source: _, action_state } = self;
|
let Self {
|
||||||
|
source: _,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
|
|
||||||
tracing::trace!("Nothing to do for `MoveUnpackedNix` revert");
|
tracing::trace!("Nothing to do for `MoveUnpackedNix` revert");
|
||||||
|
|
||||||
|
@ -79,9 +88,25 @@ impl From<MoveUnpackedNix> for Action {
|
||||||
#[derive(Debug, thiserror::Error, Serialize)]
|
#[derive(Debug, thiserror::Error, Serialize)]
|
||||||
pub enum MoveUnpackedNixError {
|
pub enum MoveUnpackedNixError {
|
||||||
#[error("Glob pattern error")]
|
#[error("Glob pattern error")]
|
||||||
GlobPatternError(#[from] #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] glob::PatternError),
|
GlobPatternError(
|
||||||
|
#[from]
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
glob::PatternError,
|
||||||
|
),
|
||||||
#[error("Glob globbing error")]
|
#[error("Glob globbing error")]
|
||||||
GlobGlobError(#[from] #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] glob::GlobError),
|
GlobGlobError(
|
||||||
|
#[from]
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
glob::GlobError,
|
||||||
|
),
|
||||||
#[error("Rename `{0}` to `{1}`")]
|
#[error("Rename `{0}` to `{1}`")]
|
||||||
Rename(std::path::PathBuf, std::path::PathBuf, #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error),
|
Rename(
|
||||||
|
std::path::PathBuf,
|
||||||
|
std::path::PathBuf,
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,7 @@
|
||||||
use reqwest::Url;
|
use reqwest::Url;
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
|
|
||||||
use crate::HarmonicError;
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable};
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable, ActionState, Action, ActionError};
|
|
||||||
|
|
||||||
use super::{CreateFile, CreateFileError};
|
use super::{CreateFile, CreateFileError};
|
||||||
|
|
||||||
|
@ -18,14 +16,24 @@ pub struct PlaceChannelConfiguration {
|
||||||
|
|
||||||
impl PlaceChannelConfiguration {
|
impl PlaceChannelConfiguration {
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
pub async fn plan(channels: Vec<(String, Url)>, force: bool) -> Result<Self, PlaceChannelConfigurationError> {
|
pub async fn plan(
|
||||||
|
channels: Vec<(String, Url)>,
|
||||||
|
force: bool,
|
||||||
|
) -> Result<Self, PlaceChannelConfigurationError> {
|
||||||
let buf = channels
|
let buf = channels
|
||||||
.iter()
|
.iter()
|
||||||
.map(|(name, url)| format!("{} {}", url, name))
|
.map(|(name, url)| format!("{} {}", url, name))
|
||||||
.collect::<Vec<_>>()
|
.collect::<Vec<_>>()
|
||||||
.join("\n");
|
.join("\n");
|
||||||
let create_file =
|
let create_file = CreateFile::plan(
|
||||||
CreateFile::plan(NIX_CHANNELS_PATH, "root".into(), "root".into(), 0o0664, buf, force).await?;
|
NIX_CHANNELS_PATH,
|
||||||
|
"root".into(),
|
||||||
|
"root".into(),
|
||||||
|
0o0664,
|
||||||
|
buf,
|
||||||
|
force,
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
create_file,
|
create_file,
|
||||||
channels,
|
channels,
|
||||||
|
@ -63,7 +71,6 @@ impl Actionable for PlaceChannelConfiguration {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self {
|
let Self {
|
||||||
|
@ -85,7 +92,6 @@ impl From<PlaceChannelConfiguration> for Action {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug, thiserror::Error, Serialize)]
|
#[derive(Debug, thiserror::Error, Serialize)]
|
||||||
pub enum PlaceChannelConfigurationError {
|
pub enum PlaceChannelConfigurationError {
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
|
|
|
@ -1,10 +1,8 @@
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
|
|
||||||
use crate::HarmonicError;
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable};
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable, ActionState, Action, ActionError};
|
use super::{CreateDirectory, CreateDirectoryError, CreateFile, CreateFileError};
|
||||||
|
|
||||||
use super::{CreateFile, CreateFileError, CreateDirectory, CreateDirectoryError};
|
|
||||||
|
|
||||||
const NIX_CONF_FOLDER: &str = "/etc/nix";
|
const NIX_CONF_FOLDER: &str = "/etc/nix";
|
||||||
const NIX_CONF: &str = "/etc/nix/nix.conf";
|
const NIX_CONF: &str = "/etc/nix/nix.conf";
|
||||||
|
@ -34,10 +32,16 @@ impl PlaceNixConfiguration {
|
||||||
",
|
",
|
||||||
extra_conf = extra_conf.unwrap_or_else(|| "".into()),
|
extra_conf = extra_conf.unwrap_or_else(|| "".into()),
|
||||||
);
|
);
|
||||||
let create_directory = CreateDirectory::plan(NIX_CONF_FOLDER, "root".into(), "root".into(), 0o0755, force).await?;
|
let create_directory =
|
||||||
|
CreateDirectory::plan(NIX_CONF_FOLDER, "root".into(), "root".into(), 0o0755, force)
|
||||||
|
.await?;
|
||||||
let create_file =
|
let create_file =
|
||||||
CreateFile::plan(NIX_CONF, "root".into(), "root".into(), 0o0664, buf, force).await?;
|
CreateFile::plan(NIX_CONF, "root".into(), "root".into(), 0o0664, buf, force).await?;
|
||||||
Ok(Self { create_directory, create_file, action_state: ActionState::Planned })
|
Ok(Self {
|
||||||
|
create_directory,
|
||||||
|
create_file,
|
||||||
|
action_state: ActionState::Planned,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -48,13 +52,20 @@ impl Actionable for PlaceNixConfiguration {
|
||||||
fn description(&self) -> Vec<ActionDescription> {
|
fn description(&self) -> Vec<ActionDescription> {
|
||||||
vec![ActionDescription::new(
|
vec![ActionDescription::new(
|
||||||
format!("Place the nix configuration in `{NIX_CONF}`"),
|
format!("Place the nix configuration in `{NIX_CONF}`"),
|
||||||
vec!["This file is read by the Nix daemon to set its configuration options at runtime.".to_string()],
|
vec![
|
||||||
|
"This file is read by the Nix daemon to set its configuration options at runtime."
|
||||||
|
.to_string(),
|
||||||
|
],
|
||||||
)]
|
)]
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn execute(&mut self) -> Result<(), Self::Error> {
|
async fn execute(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { create_file, create_directory, action_state } = self;
|
let Self {
|
||||||
|
create_file,
|
||||||
|
create_directory,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
|
|
||||||
create_directory.execute().await?;
|
create_directory.execute().await?;
|
||||||
create_file.execute().await?;
|
create_file.execute().await?;
|
||||||
|
@ -63,10 +74,13 @@ impl Actionable for PlaceNixConfiguration {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { create_file, create_directory, action_state } = self;
|
let Self {
|
||||||
|
create_file,
|
||||||
|
create_directory,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
|
|
||||||
create_file.revert().await?;
|
create_file.revert().await?;
|
||||||
create_directory.revert().await?;
|
create_directory.revert().await?;
|
||||||
|
@ -82,7 +96,6 @@ impl From<PlaceNixConfiguration> for Action {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug, thiserror::Error, Serialize)]
|
#[derive(Debug, thiserror::Error, Serialize)]
|
||||||
pub enum PlaceNixConfigurationError {
|
pub enum PlaceNixConfigurationError {
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
|
|
|
@ -1,4 +1,7 @@
|
||||||
use crate::{execute_command, actions::{ActionState, Action, ActionError}, set_env};
|
use crate::{
|
||||||
|
actions::{Action, ActionState},
|
||||||
|
execute_command, set_env,
|
||||||
|
};
|
||||||
|
|
||||||
use glob::glob;
|
use glob::glob;
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
|
@ -15,7 +18,10 @@ pub struct SetupDefaultProfile {
|
||||||
impl SetupDefaultProfile {
|
impl SetupDefaultProfile {
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
pub async fn plan(channels: Vec<String>) -> Result<Self, SetupDefaultProfileError> {
|
pub async fn plan(channels: Vec<String>) -> Result<Self, SetupDefaultProfileError> {
|
||||||
Ok(Self { channels, action_state: ActionState::Planned })
|
Ok(Self {
|
||||||
|
channels,
|
||||||
|
action_state: ActionState::Planned,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -31,7 +37,10 @@ impl Actionable for SetupDefaultProfile {
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn execute(&mut self) -> Result<(), Self::Error> {
|
async fn execute(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { channels, action_state } = self;
|
let Self {
|
||||||
|
channels,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
tracing::info!("Setting up default profile");
|
tracing::info!("Setting up default profile");
|
||||||
|
|
||||||
// Find an `nix` package
|
// Find an `nix` package
|
||||||
|
@ -59,7 +68,8 @@ impl Actionable for SetupDefaultProfile {
|
||||||
.arg("-i")
|
.arg("-i")
|
||||||
.arg(&nix_pkg),
|
.arg(&nix_pkg),
|
||||||
)
|
)
|
||||||
.await.map_err(SetupDefaultProfileError::Command)?;
|
.await
|
||||||
|
.map_err(SetupDefaultProfileError::Command)?;
|
||||||
|
|
||||||
// Find an `nss-cacert` package, add it too.
|
// Find an `nss-cacert` package, add it too.
|
||||||
let nss_ca_cert_pkg_glob = "/nix/store/*-nss-cacert-*";
|
let nss_ca_cert_pkg_glob = "/nix/store/*-nss-cacert-*";
|
||||||
|
@ -86,7 +96,8 @@ impl Actionable for SetupDefaultProfile {
|
||||||
.arg("-i")
|
.arg("-i")
|
||||||
.arg(&nss_ca_cert_pkg),
|
.arg(&nss_ca_cert_pkg),
|
||||||
)
|
)
|
||||||
.await.map_err(SetupDefaultProfileError::Command)?;
|
.await
|
||||||
|
.map_err(SetupDefaultProfileError::Command)?;
|
||||||
|
|
||||||
set_env(
|
set_env(
|
||||||
"NIX_SSL_CERT_FILE",
|
"NIX_SSL_CERT_FILE",
|
||||||
|
@ -104,16 +115,20 @@ impl Actionable for SetupDefaultProfile {
|
||||||
"/nix/var/nix/profiles/default/etc/ssl/certs/ca-bundle.crt",
|
"/nix/var/nix/profiles/default/etc/ssl/certs/ca-bundle.crt",
|
||||||
);
|
);
|
||||||
|
|
||||||
execute_command(&mut command).await.map_err(SetupDefaultProfileError::Command)?;
|
execute_command(&mut command)
|
||||||
|
.await
|
||||||
|
.map_err(SetupDefaultProfileError::Command)?;
|
||||||
}
|
}
|
||||||
*action_state = ActionState::Completed;
|
*action_state = ActionState::Completed;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { channels: _, action_state } = self;
|
let Self {
|
||||||
|
channels: _,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
|
|
||||||
std::env::remove_var("NIX_SSL_CERT_FILE");
|
std::env::remove_var("NIX_SSL_CERT_FILE");
|
||||||
|
|
||||||
|
@ -128,15 +143,28 @@ impl From<SetupDefaultProfile> for Action {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug, thiserror::Error, Serialize)]
|
#[derive(Debug, thiserror::Error, Serialize)]
|
||||||
pub enum SetupDefaultProfileError {
|
pub enum SetupDefaultProfileError {
|
||||||
#[error("Glob pattern error")]
|
#[error("Glob pattern error")]
|
||||||
GlobPatternError(#[from] #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] glob::PatternError),
|
GlobPatternError(
|
||||||
|
#[from]
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
glob::PatternError,
|
||||||
|
),
|
||||||
#[error("Glob globbing error")]
|
#[error("Glob globbing error")]
|
||||||
GlobGlobError(#[from] #[source] #[serde(serialize_with = "crate::serialize_error_to_display")] glob::GlobError),
|
GlobGlobError(
|
||||||
|
#[from]
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
glob::GlobError,
|
||||||
|
),
|
||||||
#[error("Unarchived Nix store did not appear to include a `nss-cacert` location")]
|
#[error("Unarchived Nix store did not appear to include a `nss-cacert` location")]
|
||||||
NoNssCacert,
|
NoNssCacert,
|
||||||
#[error("Failed to execute command")]
|
#[error("Failed to execute command")]
|
||||||
Command(#[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error)
|
Command(
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,10 +1,9 @@
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use tokio::process::Command;
|
use tokio::process::Command;
|
||||||
|
|
||||||
use crate::actions::meta::StartNixDaemon;
|
use crate::execute_command;
|
||||||
use crate::{execute_command, HarmonicError};
|
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable, ActionState, Action, ActionError};
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable};
|
||||||
|
|
||||||
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
||||||
pub struct StartSystemdUnit {
|
pub struct StartSystemdUnit {
|
||||||
|
@ -15,7 +14,10 @@ pub struct StartSystemdUnit {
|
||||||
impl StartSystemdUnit {
|
impl StartSystemdUnit {
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
pub async fn plan(unit: String) -> Result<Self, StartSystemdUnitError> {
|
pub async fn plan(unit: String) -> Result<Self, StartSystemdUnitError> {
|
||||||
Ok(Self { unit, action_state: ActionState::Planned })
|
Ok(Self {
|
||||||
|
unit,
|
||||||
|
action_state: ActionState::Planned,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -62,7 +64,8 @@ impl Actionable for StartSystemdUnit {
|
||||||
.arg("--now")
|
.arg("--now")
|
||||||
.arg(format!("{unit}")),
|
.arg(format!("{unit}")),
|
||||||
)
|
)
|
||||||
.await.map_err(StartSystemdUnitError::Command)?;
|
.await
|
||||||
|
.map_err(StartSystemdUnitError::Command)?;
|
||||||
|
|
||||||
*action_state = ActionState::Completed;
|
*action_state = ActionState::Completed;
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -73,12 +76,9 @@ impl Actionable for StartSystemdUnit {
|
||||||
let Self { unit, action_state } = self;
|
let Self { unit, action_state } = self;
|
||||||
|
|
||||||
// TODO(@Hoverbear): Handle proxy vars
|
// TODO(@Hoverbear): Handle proxy vars
|
||||||
execute_command(
|
execute_command(Command::new("systemctl").arg("stop").arg(format!("{unit}")))
|
||||||
Command::new("systemctl")
|
.await
|
||||||
.arg("stop")
|
.map_err(StartSystemdUnitError::Command)?;
|
||||||
.arg(format!("{unit}")),
|
|
||||||
)
|
|
||||||
.await.map_err(StartSystemdUnitError::Command)?;
|
|
||||||
|
|
||||||
*action_state = ActionState::Reverted;
|
*action_state = ActionState::Reverted;
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -94,5 +94,9 @@ impl From<StartSystemdUnit> for Action {
|
||||||
#[derive(Debug, thiserror::Error, Serialize)]
|
#[derive(Debug, thiserror::Error, Serialize)]
|
||||||
pub enum StartSystemdUnitError {
|
pub enum StartSystemdUnitError {
|
||||||
#[error("Failed to execute command")]
|
#[error("Failed to execute command")]
|
||||||
Command(#[source] #[serde(serialize_with = "crate::serialize_error_to_display")] std::io::Error)
|
Command(
|
||||||
|
#[source]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
std::io::Error,
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,12 +2,14 @@ use serde::Serialize;
|
||||||
|
|
||||||
use crate::actions::{
|
use crate::actions::{
|
||||||
base::{
|
base::{
|
||||||
ConfigureNixDaemonService, ConfigureNixDaemonServiceError, PlaceNixConfiguration,
|
ConfigureNixDaemonService, ConfigureNixDaemonServiceError, PlaceChannelConfiguration,
|
||||||
PlaceNixConfigurationError, SetupDefaultProfile, SetupDefaultProfileError, PlaceChannelConfiguration, PlaceChannelConfigurationError,
|
PlaceChannelConfigurationError, PlaceNixConfiguration, PlaceNixConfigurationError,
|
||||||
|
SetupDefaultProfile, SetupDefaultProfileError,
|
||||||
},
|
},
|
||||||
meta::{ConfigureShellProfile, ConfigureShellProfileError}, ActionState, Action, ActionError,
|
meta::{ConfigureShellProfile, ConfigureShellProfileError},
|
||||||
|
Action, ActionState,
|
||||||
};
|
};
|
||||||
use crate::{HarmonicError, InstallSettings};
|
use crate::InstallSettings;
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable};
|
use crate::actions::{ActionDescription, Actionable};
|
||||||
|
|
||||||
|
@ -39,8 +41,12 @@ impl ConfigureNix {
|
||||||
};
|
};
|
||||||
let place_channel_configuration =
|
let place_channel_configuration =
|
||||||
PlaceChannelConfiguration::plan(settings.channels, settings.force).await?;
|
PlaceChannelConfiguration::plan(settings.channels, settings.force).await?;
|
||||||
let place_nix_configuration =
|
let place_nix_configuration = PlaceNixConfiguration::plan(
|
||||||
PlaceNixConfiguration::plan(settings.nix_build_group_name, settings.extra_conf, settings.force).await?;
|
settings.nix_build_group_name,
|
||||||
|
settings.extra_conf,
|
||||||
|
settings.force,
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
let configure_nix_daemon_service = ConfigureNixDaemonService::plan().await?;
|
let configure_nix_daemon_service = ConfigureNixDaemonService::plan().await?;
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
|
@ -91,16 +97,51 @@ impl Actionable for ConfigureNix {
|
||||||
|
|
||||||
if let Some(configure_shell_profile) = configure_shell_profile {
|
if let Some(configure_shell_profile) = configure_shell_profile {
|
||||||
tokio::try_join!(
|
tokio::try_join!(
|
||||||
async move { setup_default_profile.execute().await.map_err(|e| ConfigureNixError::from(e)) },
|
async move {
|
||||||
async move { place_nix_configuration.execute().await.map_err(|e| ConfigureNixError::from(e)) },
|
setup_default_profile
|
||||||
async move { place_channel_configuration.execute().await.map_err(|e| ConfigureNixError::from(e)) },
|
.execute()
|
||||||
async move { configure_shell_profile.execute().await.map_err(|e| ConfigureNixError::from(e)) },
|
.await
|
||||||
|
.map_err(|e| ConfigureNixError::from(e))
|
||||||
|
},
|
||||||
|
async move {
|
||||||
|
place_nix_configuration
|
||||||
|
.execute()
|
||||||
|
.await
|
||||||
|
.map_err(|e| ConfigureNixError::from(e))
|
||||||
|
},
|
||||||
|
async move {
|
||||||
|
place_channel_configuration
|
||||||
|
.execute()
|
||||||
|
.await
|
||||||
|
.map_err(|e| ConfigureNixError::from(e))
|
||||||
|
},
|
||||||
|
async move {
|
||||||
|
configure_shell_profile
|
||||||
|
.execute()
|
||||||
|
.await
|
||||||
|
.map_err(|e| ConfigureNixError::from(e))
|
||||||
|
},
|
||||||
)?;
|
)?;
|
||||||
} else {
|
} else {
|
||||||
tokio::try_join!(
|
tokio::try_join!(
|
||||||
async move { setup_default_profile.execute().await.map_err(|e| ConfigureNixError::from(e)) },
|
async move {
|
||||||
async move { place_nix_configuration.execute().await.map_err(|e| ConfigureNixError::from(e)) },
|
setup_default_profile
|
||||||
async move { place_channel_configuration.execute().await.map_err(|e| ConfigureNixError::from(e)) },
|
.execute()
|
||||||
|
.await
|
||||||
|
.map_err(|e| ConfigureNixError::from(e))
|
||||||
|
},
|
||||||
|
async move {
|
||||||
|
place_nix_configuration
|
||||||
|
.execute()
|
||||||
|
.await
|
||||||
|
.map_err(|e| ConfigureNixError::from(e))
|
||||||
|
},
|
||||||
|
async move {
|
||||||
|
place_channel_configuration
|
||||||
|
.execute()
|
||||||
|
.await
|
||||||
|
.map_err(|e| ConfigureNixError::from(e))
|
||||||
|
},
|
||||||
)?;
|
)?;
|
||||||
};
|
};
|
||||||
configure_nix_daemon_service.execute().await?;
|
configure_nix_daemon_service.execute().await?;
|
||||||
|
@ -109,7 +150,6 @@ impl Actionable for ConfigureNix {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self {
|
let Self {
|
||||||
|
|
|
@ -1,12 +1,10 @@
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
|
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use tokio::task::{JoinSet, JoinError};
|
use tokio::task::{JoinError, JoinSet};
|
||||||
|
|
||||||
use crate::HarmonicError;
|
|
||||||
|
|
||||||
use crate::actions::base::{CreateOrAppendFile, CreateOrAppendFileError};
|
use crate::actions::base::{CreateOrAppendFile, CreateOrAppendFileError};
|
||||||
use crate::actions::{ActionDescription, Actionable, ActionState, Action, ActionError};
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable};
|
||||||
|
|
||||||
const PROFILE_TARGETS: &[&str] = &[
|
const PROFILE_TARGETS: &[&str] = &[
|
||||||
"/etc/bashrc",
|
"/etc/bashrc",
|
||||||
|
@ -79,12 +77,17 @@ impl Actionable for ConfigureShellProfile {
|
||||||
|
|
||||||
for (idx, create_or_append_file) in create_or_append_files.iter().enumerate() {
|
for (idx, create_or_append_file) in create_or_append_files.iter().enumerate() {
|
||||||
let mut create_or_append_file_clone = create_or_append_file.clone();
|
let mut create_or_append_file_clone = create_or_append_file.clone();
|
||||||
let _abort_handle = set.spawn(async move { create_or_append_file_clone.execute().await?; Result::<_, CreateOrAppendFileError>::Ok((idx, create_or_append_file_clone)) });
|
let _abort_handle = set.spawn(async move {
|
||||||
|
create_or_append_file_clone.execute().await?;
|
||||||
|
Result::<_, CreateOrAppendFileError>::Ok((idx, create_or_append_file_clone))
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
while let Some(result) = set.join_next().await {
|
while let Some(result) = set.join_next().await {
|
||||||
match result {
|
match result {
|
||||||
Ok(Ok((idx, create_or_append_file))) => create_or_append_files[idx] = create_or_append_file,
|
Ok(Ok((idx, create_or_append_file))) => {
|
||||||
|
create_or_append_files[idx] = create_or_append_file
|
||||||
|
},
|
||||||
Ok(Err(e)) => errors.push(e),
|
Ok(Err(e)) => errors.push(e),
|
||||||
Err(e) => return Err(e.into()),
|
Err(e) => return Err(e.into()),
|
||||||
};
|
};
|
||||||
|
@ -94,7 +97,9 @@ impl Actionable for ConfigureShellProfile {
|
||||||
if errors.len() == 1 {
|
if errors.len() == 1 {
|
||||||
return Err(errors.into_iter().next().unwrap().into());
|
return Err(errors.into_iter().next().unwrap().into());
|
||||||
} else {
|
} else {
|
||||||
return Err(ConfigureShellProfileError::MultipleCreateOrAppendFile(errors));
|
return Err(ConfigureShellProfileError::MultipleCreateOrAppendFile(
|
||||||
|
errors,
|
||||||
|
));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -102,7 +107,6 @@ impl Actionable for ConfigureShellProfile {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self {
|
let Self {
|
||||||
|
@ -116,12 +120,17 @@ impl Actionable for ConfigureShellProfile {
|
||||||
|
|
||||||
for (idx, create_or_append_file) in create_or_append_files.iter().enumerate() {
|
for (idx, create_or_append_file) in create_or_append_files.iter().enumerate() {
|
||||||
let mut create_or_append_file_clone = create_or_append_file.clone();
|
let mut create_or_append_file_clone = create_or_append_file.clone();
|
||||||
let _abort_handle = set.spawn(async move { create_or_append_file_clone.revert().await?; Result::<_, CreateOrAppendFileError>::Ok((idx, create_or_append_file_clone)) });
|
let _abort_handle = set.spawn(async move {
|
||||||
|
create_or_append_file_clone.revert().await?;
|
||||||
|
Result::<_, CreateOrAppendFileError>::Ok((idx, create_or_append_file_clone))
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
while let Some(result) = set.join_next().await {
|
while let Some(result) = set.join_next().await {
|
||||||
match result {
|
match result {
|
||||||
Ok(Ok((idx, create_or_append_file))) => create_or_append_files[idx] = create_or_append_file,
|
Ok(Ok((idx, create_or_append_file))) => {
|
||||||
|
create_or_append_files[idx] = create_or_append_file
|
||||||
|
},
|
||||||
Ok(Err(e)) => errors.push(e),
|
Ok(Err(e)) => errors.push(e),
|
||||||
Err(e) => return Err(e.into()),
|
Err(e) => return Err(e.into()),
|
||||||
};
|
};
|
||||||
|
@ -131,7 +140,9 @@ impl Actionable for ConfigureShellProfile {
|
||||||
if errors.len() == 1 {
|
if errors.len() == 1 {
|
||||||
return Err(errors.into_iter().next().unwrap().into());
|
return Err(errors.into_iter().next().unwrap().into());
|
||||||
} else {
|
} else {
|
||||||
return Err(ConfigureShellProfileError::MultipleCreateOrAppendFile(errors));
|
return Err(ConfigureShellProfileError::MultipleCreateOrAppendFile(
|
||||||
|
errors,
|
||||||
|
));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -153,5 +164,9 @@ pub enum ConfigureShellProfileError {
|
||||||
#[error("Multiple errors: {}", .0.iter().map(|v| format!("{v}")).collect::<Vec<_>>().join(" & "))]
|
#[error("Multiple errors: {}", .0.iter().map(|v| format!("{v}")).collect::<Vec<_>>().join(" & "))]
|
||||||
MultipleCreateOrAppendFile(Vec<CreateOrAppendFileError>),
|
MultipleCreateOrAppendFile(Vec<CreateOrAppendFileError>),
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
Join(#[from] #[serde(serialize_with = "crate::serialize_error_to_display")] JoinError),
|
Join(
|
||||||
|
#[from]
|
||||||
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
|
JoinError,
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,9 +1,7 @@
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
|
|
||||||
use crate::HarmonicError;
|
|
||||||
|
|
||||||
use crate::actions::base::{CreateDirectory, CreateDirectoryError};
|
use crate::actions::base::{CreateDirectory, CreateDirectoryError};
|
||||||
use crate::actions::{ActionDescription, Actionable, ActionState, Action, ActionError};
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable};
|
||||||
|
|
||||||
const PATHS: &[&str] = &[
|
const PATHS: &[&str] = &[
|
||||||
"/nix",
|
"/nix",
|
||||||
|
@ -39,7 +37,10 @@ impl CreateNixTree {
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(Self { create_directories, action_state: ActionState::Planned })
|
Ok(Self {
|
||||||
|
create_directories,
|
||||||
|
action_state: ActionState::Planned,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -50,15 +51,27 @@ impl Actionable for CreateNixTree {
|
||||||
vec![ActionDescription::new(
|
vec![ActionDescription::new(
|
||||||
format!("Create a directory tree in `/nix`"),
|
format!("Create a directory tree in `/nix`"),
|
||||||
vec![
|
vec![
|
||||||
format!("Nix and the Nix daemon require a Nix Store, which will be stored at `/nix`"),
|
format!(
|
||||||
format!("Creates: {}", PATHS.iter().map(|v| format!("`{v}`")).collect::<Vec<_>>().join(", ")),
|
"Nix and the Nix daemon require a Nix Store, which will be stored at `/nix`"
|
||||||
|
),
|
||||||
|
format!(
|
||||||
|
"Creates: {}",
|
||||||
|
PATHS
|
||||||
|
.iter()
|
||||||
|
.map(|v| format!("`{v}`"))
|
||||||
|
.collect::<Vec<_>>()
|
||||||
|
.join(", ")
|
||||||
|
),
|
||||||
],
|
],
|
||||||
)]
|
)]
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn execute(&mut self) -> Result<(), Self::Error> {
|
async fn execute(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { create_directories, action_state } = self;
|
let Self {
|
||||||
|
create_directories,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
|
|
||||||
// Just do sequential since parallizing this will have little benefit
|
// Just do sequential since parallizing this will have little benefit
|
||||||
for create_directory in create_directories {
|
for create_directory in create_directories {
|
||||||
|
@ -69,10 +82,12 @@ impl Actionable for CreateNixTree {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { create_directories, action_state } = self;
|
let Self {
|
||||||
|
create_directories,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
|
|
||||||
// Just do sequential since parallizing this will have little benefit
|
// Just do sequential since parallizing this will have little benefit
|
||||||
for create_directory in create_directories.iter_mut().rev() {
|
for create_directory in create_directories.iter_mut().rev() {
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
use tokio::task::{JoinSet, JoinError};
|
use tokio::task::{JoinError, JoinSet};
|
||||||
|
|
||||||
use crate::{HarmonicError, InstallSettings};
|
use crate::InstallSettings;
|
||||||
|
|
||||||
use crate::actions::base::{CreateGroup, CreateGroupError, CreateUserError};
|
use crate::actions::base::{CreateGroup, CreateGroupError, CreateUserError};
|
||||||
use crate::actions::{ActionDescription, Actionable, CreateUser, ActionState, Action};
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable, CreateUser};
|
||||||
|
|
||||||
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
||||||
pub struct CreateUsersAndGroup {
|
pub struct CreateUsersAndGroup {
|
||||||
|
@ -89,7 +89,6 @@ impl Actionable for CreateUsersAndGroup {
|
||||||
action_state,
|
action_state,
|
||||||
} = self;
|
} = self;
|
||||||
|
|
||||||
|
|
||||||
// Create group
|
// Create group
|
||||||
create_group.execute().await?;
|
create_group.execute().await?;
|
||||||
|
|
||||||
|
@ -101,7 +100,10 @@ impl Actionable for CreateUsersAndGroup {
|
||||||
|
|
||||||
for (idx, create_user) in create_users.iter().enumerate() {
|
for (idx, create_user) in create_users.iter().enumerate() {
|
||||||
let mut create_user_clone = create_user.clone();
|
let mut create_user_clone = create_user.clone();
|
||||||
let _abort_handle = set.spawn(async move { create_user_clone.execute().await?; Result::<_, CreateUserError>::Ok((idx, create_user_clone)) });
|
let _abort_handle = set.spawn(async move {
|
||||||
|
create_user_clone.execute().await?;
|
||||||
|
Result::<_, CreateUserError>::Ok((idx, create_user_clone))
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
while let Some(result) = set.join_next().await {
|
while let Some(result) = set.join_next().await {
|
||||||
|
@ -120,7 +122,6 @@ impl Actionable for CreateUsersAndGroup {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
*action_state = ActionState::Completed;
|
*action_state = ActionState::Completed;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -146,7 +147,10 @@ impl Actionable for CreateUsersAndGroup {
|
||||||
|
|
||||||
for (idx, create_user) in create_users.iter().enumerate() {
|
for (idx, create_user) in create_users.iter().enumerate() {
|
||||||
let mut create_user_clone = create_user.clone();
|
let mut create_user_clone = create_user.clone();
|
||||||
let _abort_handle = set.spawn(async move { create_user_clone.revert().await?; Result::<_, CreateUserError>::Ok((idx, create_user_clone)) });
|
let _abort_handle = set.spawn(async move {
|
||||||
|
create_user_clone.revert().await?;
|
||||||
|
Result::<_, CreateUserError>::Ok((idx, create_user_clone))
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
while let Some(result) = set.join_next().await {
|
while let Some(result) = set.join_next().await {
|
||||||
|
@ -173,14 +177,12 @@ impl Actionable for CreateUsersAndGroup {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
impl From<CreateUsersAndGroup> for Action {
|
impl From<CreateUsersAndGroup> for Action {
|
||||||
fn from(v: CreateUsersAndGroup) -> Self {
|
fn from(v: CreateUsersAndGroup) -> Self {
|
||||||
Action::CreateUsersAndGroup(v)
|
Action::CreateUsersAndGroup(v)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug, thiserror::Error, Serialize)]
|
#[derive(Debug, thiserror::Error, Serialize)]
|
||||||
pub enum CreateUsersAndGroupError {
|
pub enum CreateUsersAndGroupError {
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
|
@ -193,7 +195,6 @@ pub enum CreateUsersAndGroupError {
|
||||||
Join(
|
Join(
|
||||||
#[from]
|
#[from]
|
||||||
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
JoinError
|
JoinError,
|
||||||
),
|
),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -3,14 +3,11 @@ use tempdir::TempDir;
|
||||||
use tokio::task::JoinError;
|
use tokio::task::JoinError;
|
||||||
|
|
||||||
use crate::actions::base::{FetchNix, FetchNixError, MoveUnpackedNix, MoveUnpackedNixError};
|
use crate::actions::base::{FetchNix, FetchNixError, MoveUnpackedNix, MoveUnpackedNixError};
|
||||||
use crate::{HarmonicError, InstallSettings};
|
use crate::InstallSettings;
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable, ActionState, Action, ActionError};
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable};
|
||||||
|
|
||||||
use super::{
|
use super::{CreateNixTree, CreateNixTreeError, CreateUsersAndGroup, CreateUsersAndGroupError};
|
||||||
CreateNixTree, CreateNixTreeError,
|
|
||||||
CreateUsersAndGroup, CreateUsersAndGroupError,
|
|
||||||
};
|
|
||||||
|
|
||||||
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
||||||
pub struct ProvisionNix {
|
pub struct ProvisionNix {
|
||||||
|
@ -76,10 +73,16 @@ impl Actionable for ProvisionNix {
|
||||||
|
|
||||||
// We fetch nix while doing the rest, then move it over.
|
// We fetch nix while doing the rest, then move it over.
|
||||||
let mut fetch_nix_clone = fetch_nix.clone();
|
let mut fetch_nix_clone = fetch_nix.clone();
|
||||||
let fetch_nix_handle = tokio::task::spawn(async { fetch_nix_clone.execute().await?; Result::<_, Self::Error>::Ok(fetch_nix_clone) });
|
let fetch_nix_handle = tokio::task::spawn(async {
|
||||||
|
fetch_nix_clone.execute().await?;
|
||||||
|
Result::<_, Self::Error>::Ok(fetch_nix_clone)
|
||||||
|
});
|
||||||
|
|
||||||
create_users_and_group.execute().await?;
|
create_users_and_group.execute().await?;
|
||||||
create_nix_tree.execute().await.map_err(ProvisionNixError::from)?;
|
create_nix_tree
|
||||||
|
.execute()
|
||||||
|
.await
|
||||||
|
.map_err(ProvisionNixError::from)?;
|
||||||
|
|
||||||
*fetch_nix = fetch_nix_handle.await.map_err(ProvisionNixError::from)??;
|
*fetch_nix = fetch_nix_handle.await.map_err(ProvisionNixError::from)??;
|
||||||
move_unpacked_nix.execute().await?;
|
move_unpacked_nix.execute().await?;
|
||||||
|
@ -100,10 +103,16 @@ impl Actionable for ProvisionNix {
|
||||||
|
|
||||||
// We fetch nix while doing the rest, then move it over.
|
// We fetch nix while doing the rest, then move it over.
|
||||||
let mut fetch_nix_clone = fetch_nix.clone();
|
let mut fetch_nix_clone = fetch_nix.clone();
|
||||||
let fetch_nix_handle = tokio::task::spawn(async { fetch_nix_clone.revert().await?; Result::<_, Self::Error>::Ok(fetch_nix_clone) });
|
let fetch_nix_handle = tokio::task::spawn(async {
|
||||||
|
fetch_nix_clone.revert().await?;
|
||||||
|
Result::<_, Self::Error>::Ok(fetch_nix_clone)
|
||||||
|
});
|
||||||
|
|
||||||
create_users_and_group.revert().await?;
|
create_users_and_group.revert().await?;
|
||||||
create_nix_tree.revert().await.map_err(ProvisionNixError::from)?;
|
create_nix_tree
|
||||||
|
.revert()
|
||||||
|
.await
|
||||||
|
.map_err(ProvisionNixError::from)?;
|
||||||
|
|
||||||
*fetch_nix = fetch_nix_handle.await.map_err(ProvisionNixError::from)??;
|
*fetch_nix = fetch_nix_handle.await.map_err(ProvisionNixError::from)??;
|
||||||
move_unpacked_nix.revert().await?;
|
move_unpacked_nix.revert().await?;
|
||||||
|
@ -119,14 +128,13 @@ impl From<ProvisionNix> for Action {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#[derive(Debug, thiserror::Error, Serialize)]
|
#[derive(Debug, thiserror::Error, Serialize)]
|
||||||
pub enum ProvisionNixError {
|
pub enum ProvisionNixError {
|
||||||
#[error("Failed create tempdir")]
|
#[error("Failed create tempdir")]
|
||||||
TempDir(
|
TempDir(
|
||||||
#[source]
|
#[source]
|
||||||
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
std::io::Error
|
std::io::Error,
|
||||||
),
|
),
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
FetchNix(#[from] FetchNixError),
|
FetchNix(#[from] FetchNixError),
|
||||||
|
@ -134,7 +142,7 @@ pub enum ProvisionNixError {
|
||||||
Join(
|
Join(
|
||||||
#[from]
|
#[from]
|
||||||
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
#[serde(serialize_with = "crate::serialize_error_to_display")]
|
||||||
JoinError
|
JoinError,
|
||||||
),
|
),
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
CreateUsersAndGroup(#[from] CreateUsersAndGroupError),
|
CreateUsersAndGroup(#[from] CreateUsersAndGroupError),
|
||||||
|
|
|
@ -1,9 +1,8 @@
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
|
|
||||||
use crate::actions::base::{StartSystemdUnit, StartSystemdUnitError};
|
use crate::actions::base::{StartSystemdUnit, StartSystemdUnitError};
|
||||||
use crate::HarmonicError;
|
|
||||||
|
|
||||||
use crate::actions::{ActionDescription, Actionable, ActionState, Action, ActionError};
|
use crate::actions::{Action, ActionDescription, ActionState, Actionable};
|
||||||
|
|
||||||
/// This is mostly indirection for supporting non-systemd
|
/// This is mostly indirection for supporting non-systemd
|
||||||
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
#[derive(Debug, serde::Deserialize, serde::Serialize, Clone)]
|
||||||
|
@ -33,7 +32,10 @@ impl Actionable for StartNixDaemon {
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn execute(&mut self) -> Result<(), Self::Error> {
|
async fn execute(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { start_systemd_socket, action_state } = self;
|
let Self {
|
||||||
|
start_systemd_socket,
|
||||||
|
action_state,
|
||||||
|
} = self;
|
||||||
|
|
||||||
start_systemd_socket.execute().await?;
|
start_systemd_socket.execute().await?;
|
||||||
|
|
||||||
|
@ -43,7 +45,11 @@ impl Actionable for StartNixDaemon {
|
||||||
|
|
||||||
#[tracing::instrument(skip_all)]
|
#[tracing::instrument(skip_all)]
|
||||||
async fn revert(&mut self) -> Result<(), Self::Error> {
|
async fn revert(&mut self) -> Result<(), Self::Error> {
|
||||||
let Self { start_systemd_socket, action_state, .. } = self;
|
let Self {
|
||||||
|
start_systemd_socket,
|
||||||
|
action_state,
|
||||||
|
..
|
||||||
|
} = self;
|
||||||
|
|
||||||
start_systemd_socket.revert().await?;
|
start_systemd_socket.revert().await?;
|
||||||
|
|
||||||
|
@ -61,5 +67,5 @@ impl From<StartNixDaemon> for Action {
|
||||||
#[derive(Debug, thiserror::Error, Serialize)]
|
#[derive(Debug, thiserror::Error, Serialize)]
|
||||||
pub enum StartNixDaemonError {
|
pub enum StartNixDaemonError {
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
StartSystemdUnit(#[from] StartSystemdUnitError)
|
StartSystemdUnit(#[from] StartSystemdUnitError),
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
pub mod base;
|
pub mod base;
|
||||||
pub mod meta;
|
pub mod meta;
|
||||||
|
|
||||||
use std::{error::Error, fmt::Display};
|
use std::error::Error;
|
||||||
|
|
||||||
use base::{
|
use base::{
|
||||||
ConfigureNixDaemonService, ConfigureNixDaemonServiceError, CreateDirectory,
|
ConfigureNixDaemonService, ConfigureNixDaemonServiceError, CreateDirectory,
|
||||||
|
@ -13,11 +13,10 @@ use base::{
|
||||||
};
|
};
|
||||||
use meta::{
|
use meta::{
|
||||||
ConfigureNix, ConfigureNixError, ConfigureShellProfile, ConfigureShellProfileError,
|
ConfigureNix, ConfigureNixError, ConfigureShellProfile, ConfigureShellProfileError,
|
||||||
CreateNixTree, CreateNixTreeError, CreateUsersAndGroup, CreateUsersAndGroupError,
|
CreateNixTree, CreateNixTreeError, CreateUsersAndGroup, CreateUsersAndGroupError, ProvisionNix,
|
||||||
ProvisionNix, ProvisionNixError, StartNixDaemon, StartNixDaemonError,
|
ProvisionNixError, StartNixDaemon, StartNixDaemonError,
|
||||||
};
|
};
|
||||||
use serde::{Deserialize, de::DeserializeOwned, Serialize};
|
use serde::{de::DeserializeOwned, Deserialize, Serialize};
|
||||||
|
|
||||||
|
|
||||||
use self::base::{StartSystemdUnit, StartSystemdUnitError};
|
use self::base::{StartSystemdUnit, StartSystemdUnitError};
|
||||||
|
|
||||||
|
|
|
@ -1,13 +1,10 @@
|
||||||
use std::{process::ExitCode, path::PathBuf};
|
use std::{path::PathBuf, process::ExitCode};
|
||||||
|
|
||||||
use clap::{ArgAction, Parser};
|
use clap::{ArgAction, Parser};
|
||||||
use harmonic::InstallPlan;
|
|
||||||
use eyre::WrapErr;
|
use eyre::WrapErr;
|
||||||
|
use harmonic::InstallPlan;
|
||||||
|
|
||||||
use crate::{
|
use crate::{cli::CommandExecute, interaction};
|
||||||
cli::CommandExecute,
|
|
||||||
interaction,
|
|
||||||
};
|
|
||||||
|
|
||||||
/// An opinionated, experimental Nix installer
|
/// An opinionated, experimental Nix installer
|
||||||
#[derive(Debug, Parser)]
|
#[derive(Debug, Parser)]
|
||||||
|
@ -29,7 +26,9 @@ impl CommandExecute for Execute {
|
||||||
async fn execute(self) -> eyre::Result<ExitCode> {
|
async fn execute(self) -> eyre::Result<ExitCode> {
|
||||||
let Self { no_confirm, plan } = self;
|
let Self { no_confirm, plan } = self;
|
||||||
|
|
||||||
let install_plan_string = tokio::fs::read_to_string(plan).await.wrap_err("Reading plan")?;
|
let install_plan_string = tokio::fs::read_to_string(plan)
|
||||||
|
.await
|
||||||
|
.wrap_err("Reading plan")?;
|
||||||
let mut plan: InstallPlan = serde_json::from_str(&install_plan_string)?;
|
let mut plan: InstallPlan = serde_json::from_str(&install_plan_string)?;
|
||||||
|
|
||||||
if !no_confirm {
|
if !no_confirm {
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
use std::{process::ExitCode, path::PathBuf};
|
use std::{path::PathBuf, process::ExitCode};
|
||||||
|
|
||||||
use clap::{ArgAction, Parser};
|
use clap::{ArgAction, Parser};
|
||||||
use harmonic::{InstallPlan, InstallSettings};
|
use harmonic::{InstallPlan, InstallSettings};
|
||||||
use tokio::io::AsyncWriteExt;
|
|
||||||
use eyre::WrapErr;
|
use eyre::WrapErr;
|
||||||
|
|
||||||
use crate::cli::{arg::ChannelValue, CommandExecute};
|
use crate::cli::{arg::ChannelValue, CommandExecute};
|
||||||
|
@ -80,7 +80,9 @@ impl CommandExecute for Plan {
|
||||||
let install_plan = InstallPlan::new(settings).await?;
|
let install_plan = InstallPlan::new(settings).await?;
|
||||||
|
|
||||||
let json = serde_json::to_string_pretty(&install_plan)?;
|
let json = serde_json::to_string_pretty(&install_plan)?;
|
||||||
tokio::fs::write(plan, json).await.wrap_err("Writing plan")?;
|
tokio::fs::write(plan, json)
|
||||||
|
.await
|
||||||
|
.wrap_err("Writing plan")?;
|
||||||
|
|
||||||
Ok(ExitCode::SUCCESS)
|
Ok(ExitCode::SUCCESS)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,13 +1,10 @@
|
||||||
use std::{process::ExitCode, path::PathBuf};
|
use std::{path::PathBuf, process::ExitCode};
|
||||||
|
|
||||||
use clap::{ArgAction, Parser};
|
use clap::{ArgAction, Parser};
|
||||||
use harmonic::InstallPlan;
|
|
||||||
use eyre::WrapErr;
|
use eyre::WrapErr;
|
||||||
|
use harmonic::InstallPlan;
|
||||||
|
|
||||||
use crate::{
|
use crate::{cli::CommandExecute, interaction};
|
||||||
cli::CommandExecute,
|
|
||||||
interaction,
|
|
||||||
};
|
|
||||||
|
|
||||||
/// An opinionated, experimental Nix installer
|
/// An opinionated, experimental Nix installer
|
||||||
#[derive(Debug, Parser)]
|
#[derive(Debug, Parser)]
|
||||||
|
@ -27,9 +24,14 @@ pub(crate) struct Revert {
|
||||||
impl CommandExecute for Revert {
|
impl CommandExecute for Revert {
|
||||||
#[tracing::instrument(skip_all, fields())]
|
#[tracing::instrument(skip_all, fields())]
|
||||||
async fn execute(self) -> eyre::Result<ExitCode> {
|
async fn execute(self) -> eyre::Result<ExitCode> {
|
||||||
let Self { no_confirm, receipt } = self;
|
let Self {
|
||||||
|
no_confirm,
|
||||||
|
receipt,
|
||||||
|
} = self;
|
||||||
|
|
||||||
let install_receipt_string = tokio::fs::read_to_string(receipt).await.wrap_err("Reading receipt")?;
|
let install_receipt_string = tokio::fs::read_to_string(receipt)
|
||||||
|
.await
|
||||||
|
.wrap_err("Reading receipt")?;
|
||||||
let mut plan: InstallPlan = serde_json::from_str(&install_receipt_string)?;
|
let mut plan: InstallPlan = serde_json::from_str(&install_receipt_string)?;
|
||||||
|
|
||||||
if !no_confirm {
|
if !no_confirm {
|
||||||
|
|
|
@ -5,7 +5,11 @@ use crate::actions::ActionError;
|
||||||
#[derive(thiserror::Error, Debug)]
|
#[derive(thiserror::Error, Debug)]
|
||||||
pub enum HarmonicError {
|
pub enum HarmonicError {
|
||||||
#[error("Error executing action")]
|
#[error("Error executing action")]
|
||||||
ActionError(#[source] #[from] ActionError),
|
ActionError(
|
||||||
|
#[source]
|
||||||
|
#[from]
|
||||||
|
ActionError,
|
||||||
|
),
|
||||||
#[error("Recording install receipt")]
|
#[error("Recording install receipt")]
|
||||||
RecordingReceipt(PathBuf, #[source] std::io::Error),
|
RecordingReceipt(PathBuf, #[source] std::io::Error),
|
||||||
#[error(transparent)]
|
#[error(transparent)]
|
||||||
|
|
39
src/lib.rs
39
src/lib.rs
|
@ -3,43 +3,26 @@ mod error;
|
||||||
mod plan;
|
mod plan;
|
||||||
mod settings;
|
mod settings;
|
||||||
|
|
||||||
use std::{
|
use std::{ffi::OsStr, fmt::Display, process::ExitStatus};
|
||||||
ffi::OsStr,
|
|
||||||
fs::Permissions,
|
|
||||||
io::SeekFrom,
|
|
||||||
os::unix::fs::PermissionsExt,
|
|
||||||
path::{Path, PathBuf},
|
|
||||||
process::ExitStatus, fmt::Display,
|
|
||||||
};
|
|
||||||
|
|
||||||
pub use error::HarmonicError;
|
pub use error::HarmonicError;
|
||||||
pub use plan::InstallPlan;
|
pub use plan::InstallPlan;
|
||||||
use serde::Serializer;
|
use serde::Serializer;
|
||||||
pub use settings::InstallSettings;
|
pub use settings::InstallSettings;
|
||||||
|
|
||||||
use bytes::Buf;
|
use tokio::process::Command;
|
||||||
use glob::glob;
|
|
||||||
use reqwest::Url;
|
|
||||||
use tempdir::TempDir;
|
|
||||||
use tokio::{
|
|
||||||
io::{AsyncSeekExt, AsyncWriteExt},
|
|
||||||
process::Command,
|
|
||||||
task::spawn_blocking,
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
#[tracing::instrument(skip_all, fields(command = %format!("{:?}", command.as_std())))]
|
#[tracing::instrument(skip_all, fields(command = %format!("{:?}", command.as_std())))]
|
||||||
async fn execute_command(
|
async fn execute_command(command: &mut Command) -> Result<ExitStatus, std::io::Error> {
|
||||||
command: &mut Command,
|
|
||||||
) -> Result<ExitStatus, std::io::Error> {
|
|
||||||
tracing::trace!("Executing");
|
tracing::trace!("Executing");
|
||||||
let command_str = format!("{:?}", command.as_std());
|
let command_str = format!("{:?}", command.as_std());
|
||||||
let status = command
|
let status = command.status().await?;
|
||||||
.status()
|
|
||||||
.await?;
|
|
||||||
match status.success() {
|
match status.success() {
|
||||||
true => Ok(status),
|
true => Ok(status),
|
||||||
false => Err(std::io::Error::new(std::io::ErrorKind::Other, format!("Command `{command_str}` failed status"))),
|
false => Err(std::io::Error::new(
|
||||||
|
std::io::ErrorKind::Other,
|
||||||
|
format!("Command `{command_str}` failed status"),
|
||||||
|
)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -52,6 +35,10 @@ fn set_env(k: impl AsRef<OsStr>, v: impl AsRef<OsStr>) {
|
||||||
std::env::set_var(k.as_ref(), v.as_ref());
|
std::env::set_var(k.as_ref(), v.as_ref());
|
||||||
}
|
}
|
||||||
|
|
||||||
fn serialize_error_to_display<E, S>(err: &E, ser: S) -> Result<S::Ok, S::Error> where E: Display, S: Serializer {
|
fn serialize_error_to_display<E, S>(err: &E, ser: S) -> Result<S::Ok, S::Error>
|
||||||
|
where
|
||||||
|
E: Display,
|
||||||
|
S: Serializer,
|
||||||
|
{
|
||||||
ser.serialize_str(&format!("{err:#}"))
|
ser.serialize_str(&format!("{err:#}"))
|
||||||
}
|
}
|
45
src/plan.rs
45
src/plan.rs
|
@ -1,12 +1,9 @@
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
|
|
||||||
use serde::{Deserialize, Serialize};
|
|
||||||
use tokio::fs::File;
|
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
actions::{
|
actions::{
|
||||||
meta::{ConfigureNix, ProvisionNix, StartNixDaemon},
|
meta::{ConfigureNix, ProvisionNix, StartNixDaemon},
|
||||||
Action, ActionDescription, Actionable, ActionState, ActionError,
|
ActionDescription, ActionError, Actionable,
|
||||||
},
|
},
|
||||||
settings::InstallSettings,
|
settings::InstallSettings,
|
||||||
HarmonicError,
|
HarmonicError,
|
||||||
|
@ -88,11 +85,14 @@ impl InstallPlan {
|
||||||
pub async fn new(settings: InstallSettings) -> Result<Self, HarmonicError> {
|
pub async fn new(settings: InstallSettings) -> Result<Self, HarmonicError> {
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
settings: settings.clone(),
|
settings: settings.clone(),
|
||||||
provision_nix: ProvisionNix::plan(settings.clone()).await
|
provision_nix: ProvisionNix::plan(settings.clone())
|
||||||
|
.await
|
||||||
.map_err(|e| ActionError::from(e))?,
|
.map_err(|e| ActionError::from(e))?,
|
||||||
configure_nix: ConfigureNix::plan(settings).await
|
configure_nix: ConfigureNix::plan(settings)
|
||||||
|
.await
|
||||||
.map_err(|e| ActionError::from(e))?,
|
.map_err(|e| ActionError::from(e))?,
|
||||||
start_nix_daemon: StartNixDaemon::plan().await
|
start_nix_daemon: StartNixDaemon::plan()
|
||||||
|
.await
|
||||||
.map_err(|e| ActionError::from(e))?,
|
.map_err(|e| ActionError::from(e))?,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -102,17 +102,24 @@ impl InstallPlan {
|
||||||
// This is **deliberately sequential**.
|
// This is **deliberately sequential**.
|
||||||
// Actions which are parallelizable are represented by "group actions" like CreateUsers
|
// Actions which are parallelizable are represented by "group actions" like CreateUsers
|
||||||
// The plan itself represents the concept of the sequence of stages.
|
// The plan itself represents the concept of the sequence of stages.
|
||||||
self.provision_nix.execute().await
|
self.provision_nix
|
||||||
|
.execute()
|
||||||
|
.await
|
||||||
.map_err(|e| ActionError::from(e))?;
|
.map_err(|e| ActionError::from(e))?;
|
||||||
self.configure_nix.execute().await
|
self.configure_nix
|
||||||
|
.execute()
|
||||||
|
.await
|
||||||
.map_err(|e| ActionError::from(e))?;
|
.map_err(|e| ActionError::from(e))?;
|
||||||
self.start_nix_daemon.execute().await
|
self.start_nix_daemon
|
||||||
|
.execute()
|
||||||
|
.await
|
||||||
.map_err(|e| ActionError::from(e))?;
|
.map_err(|e| ActionError::from(e))?;
|
||||||
|
|
||||||
let install_receipt_path = PathBuf::from("/nix/receipt.json");
|
let install_receipt_path = PathBuf::from("/nix/receipt.json");
|
||||||
let self_json = serde_json::to_string_pretty(&self)
|
let self_json =
|
||||||
.map_err(HarmonicError::SerializingReceipt)?;
|
serde_json::to_string_pretty(&self).map_err(HarmonicError::SerializingReceipt)?;
|
||||||
tokio::fs::write(&install_receipt_path, self_json).await
|
tokio::fs::write(&install_receipt_path, self_json)
|
||||||
|
.await
|
||||||
.map_err(|e| HarmonicError::RecordingReceipt(install_receipt_path, e))?;
|
.map_err(|e| HarmonicError::RecordingReceipt(install_receipt_path, e))?;
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -123,11 +130,17 @@ impl InstallPlan {
|
||||||
// This is **deliberately sequential**.
|
// This is **deliberately sequential**.
|
||||||
// Actions which are parallelizable are represented by "group actions" like CreateUsers
|
// Actions which are parallelizable are represented by "group actions" like CreateUsers
|
||||||
// The plan itself represents the concept of the sequence of stages.
|
// The plan itself represents the concept of the sequence of stages.
|
||||||
self.start_nix_daemon.revert().await
|
self.start_nix_daemon
|
||||||
|
.revert()
|
||||||
|
.await
|
||||||
.map_err(|e| ActionError::from(e))?;
|
.map_err(|e| ActionError::from(e))?;
|
||||||
self.configure_nix.revert().await
|
self.configure_nix
|
||||||
|
.revert()
|
||||||
|
.await
|
||||||
.map_err(|e| ActionError::from(e))?;
|
.map_err(|e| ActionError::from(e))?;
|
||||||
self.provision_nix.revert().await
|
self.provision_nix
|
||||||
|
.revert()
|
||||||
|
.await
|
||||||
.map_err(|e| ActionError::from(e))?;
|
.map_err(|e| ActionError::from(e))?;
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
|
Loading…
Reference in a new issue