forked from lix-project/lix-installer
8bb37f1bcf
* Uninstall shouldn't fail fast * wip * wip * No longer fails fast * Tidy up error handling * Touchup post merge * Refactor nix tests * Some minor fixes * Uninstall fail tests * Fiddle with messaging * nixfmt * Tweak display a bit * fix docs * Fix Mac * Revert setting I was testing * Reflect feedback about a log level
139 lines
4.1 KiB
Rust
139 lines
4.1 KiB
Rust
/*! The Determinate [Nix](https://github.com/NixOS/nix) Installer
|
|
|
|
`nix-installer` breaks down into three main concepts:
|
|
|
|
* [`Action`]: An executable or revertable step, possibly orchestrating sub-[`Action`]s using things
|
|
like [`JoinSet`](tokio::task::JoinSet)s.
|
|
* [`InstallPlan`]: A set of [`Action`]s, along with some metadata, which can be carried out to
|
|
drive an install or revert.
|
|
* [`Planner`](planner::Planner): Something which can be used to plan out an [`InstallPlan`].
|
|
|
|
It is possible to create custom [`Action`]s and [`Planner`](planner::Planner)s to suit the needs of your project, team, or organization.
|
|
|
|
In the simplest case, `nix-installer` can be asked to determine a default plan for the platform and install
|
|
it, uninstalling if anything goes wrong:
|
|
|
|
```rust,no_run
|
|
use std::error::Error;
|
|
use nix_installer::InstallPlan;
|
|
|
|
# async fn default_install() -> color_eyre::Result<()> {
|
|
let mut plan = InstallPlan::default().await?;
|
|
match plan.install(None).await {
|
|
Ok(()) => tracing::info!("Done"),
|
|
Err(e) => {
|
|
match e.source() {
|
|
Some(source) => tracing::error!("{e}: {}", source),
|
|
None => tracing::error!("{e}"),
|
|
};
|
|
plan.uninstall(None).await?;
|
|
},
|
|
};
|
|
#
|
|
# Ok(())
|
|
# }
|
|
```
|
|
|
|
Sometimes choosing a specific planner is desired:
|
|
|
|
```rust,no_run
|
|
use std::error::Error;
|
|
use nix_installer::{InstallPlan, planner::Planner};
|
|
|
|
# async fn chosen_planner_install() -> color_eyre::Result<()> {
|
|
#[cfg(target_os = "linux")]
|
|
let planner = nix_installer::planner::steam_deck::SteamDeck::default().await?;
|
|
#[cfg(target_os = "macos")]
|
|
let planner = nix_installer::planner::macos::Macos::default().await?;
|
|
|
|
// Or call `crate::planner::BuiltinPlanner::default()`
|
|
// Match on the result to customize.
|
|
|
|
// Customize any settings...
|
|
|
|
let mut plan = InstallPlan::plan(planner).await?;
|
|
match plan.install(None).await {
|
|
Ok(()) => tracing::info!("Done"),
|
|
Err(e) => {
|
|
match e.source() {
|
|
Some(source) => tracing::error!("{e}: {}", source),
|
|
None => tracing::error!("{e}"),
|
|
};
|
|
plan.uninstall(None).await?;
|
|
},
|
|
};
|
|
#
|
|
# Ok(())
|
|
# }
|
|
```
|
|
|
|
*/
|
|
|
|
pub mod action;
|
|
#[cfg(feature = "cli")]
|
|
pub mod cli;
|
|
#[cfg(feature = "diagnostics")]
|
|
pub mod diagnostics;
|
|
mod error;
|
|
mod os;
|
|
mod plan;
|
|
pub mod planner;
|
|
pub mod settings;
|
|
|
|
use std::{ffi::OsStr, path::Path, process::Output};
|
|
|
|
pub use error::NixInstallerError;
|
|
pub use plan::InstallPlan;
|
|
use planner::BuiltinPlanner;
|
|
|
|
use reqwest::Certificate;
|
|
use tokio::process::Command;
|
|
|
|
use crate::action::{Action, ActionErrorKind};
|
|
|
|
#[tracing::instrument(level = "debug", skip_all, fields(command = %format!("{:?}", command.as_std())))]
|
|
async fn execute_command(command: &mut Command) -> Result<Output, ActionErrorKind> {
|
|
tracing::trace!("Executing");
|
|
let output = command
|
|
.output()
|
|
.await
|
|
.map_err(|e| ActionErrorKind::command(command, e))?;
|
|
match output.status.success() {
|
|
true => Ok(output),
|
|
false => Err(ActionErrorKind::command_output(command, output)),
|
|
}
|
|
}
|
|
|
|
#[tracing::instrument(level = "debug", skip_all, fields(
|
|
k = %k.as_ref().to_string_lossy(),
|
|
v = %v.as_ref().to_string_lossy(),
|
|
))]
|
|
fn set_env(k: impl AsRef<OsStr>, v: impl AsRef<OsStr>) {
|
|
tracing::trace!("Setting env");
|
|
std::env::set_var(k.as_ref(), v.as_ref());
|
|
}
|
|
|
|
async fn parse_ssl_cert(ssl_cert_file: &Path) -> Result<Certificate, CertificateError> {
|
|
let cert_buf = tokio::fs::read(ssl_cert_file)
|
|
.await
|
|
.map_err(|e| CertificateError::Read(ssl_cert_file.to_path_buf(), e))?;
|
|
// We actually try them since things could be `.crt` and `pem` format or `der` format
|
|
let cert = if let Ok(cert) = Certificate::from_pem(cert_buf.as_slice()) {
|
|
cert
|
|
} else if let Ok(cert) = Certificate::from_der(cert_buf.as_slice()) {
|
|
cert
|
|
} else {
|
|
return Err(CertificateError::UnknownCertFormat);
|
|
};
|
|
Ok(cert)
|
|
}
|
|
|
|
#[derive(Debug, thiserror::Error)]
|
|
pub enum CertificateError {
|
|
#[error(transparent)]
|
|
Reqwest(reqwest::Error),
|
|
#[error("Read path `{0}`")]
|
|
Read(std::path::PathBuf, #[source] std::io::Error),
|
|
#[error("Unknown certificate format, `der` and `pem` supported")]
|
|
UnknownCertFormat,
|
|
}
|