2026-05-03 11:34:38 +02:00
|
|
|
use eyre::{Result, format_err};
|
2025-07-17 15:42:08 +02:00
|
|
|
use log::info;
|
|
|
|
|
use std::path::Path;
|
|
|
|
|
use tokio::fs;
|
|
|
|
|
|
2026-05-21 10:54:49 +02:00
|
|
|
use crate::File;
|
2026-02-10 21:23:11 +01:00
|
|
|
|
2026-05-03 11:34:38 +02:00
|
|
|
pub async fn files(files: &[File], root: &str, dry_run: bool) -> Result<()> {
|
|
|
|
|
for f in files {
|
|
|
|
|
if let Err(e) = file(f, root, dry_run).await {
|
2026-05-20 12:23:47 +02:00
|
|
|
return Err(format_err!("{}: {e}", f.path));
|
2026-05-03 11:34:38 +02:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub async fn file(file: &File, root: &str, dry_run: bool) -> Result<()> {
|
2026-05-20 12:23:47 +02:00
|
|
|
let path = chroot(root, &file.path);
|
|
|
|
|
let path = Path::new(&path);
|
2025-07-17 15:42:08 +02:00
|
|
|
|
2026-05-20 12:23:47 +02:00
|
|
|
if !dry_run && let Some(parent) = path.parent() {
|
|
|
|
|
fs::create_dir_all(parent).await?;
|
|
|
|
|
}
|
2025-07-17 15:42:08 +02:00
|
|
|
|
2026-05-20 12:23:47 +02:00
|
|
|
let kind = file.kind();
|
|
|
|
|
let content = kind.content()?;
|
|
|
|
|
|
|
|
|
|
use crate::{FileKind as K};
|
|
|
|
|
match kind.as_ref() {
|
|
|
|
|
K::Skip => {
|
|
|
|
|
info!("{}: kind is skip", file.path);
|
|
|
|
|
return Ok(());
|
|
|
|
|
}
|
|
|
|
|
K::Content(_) | K::Content64(_) | K::Parts(_) => {
|
|
|
|
|
let content = content.expect("this file kind should have content");
|
|
|
|
|
if dry_run {
|
|
|
|
|
info!(
|
|
|
|
|
"would create {} ({} bytes)",
|
|
|
|
|
file.path,
|
|
|
|
|
content.len()
|
|
|
|
|
);
|
|
|
|
|
} else {
|
|
|
|
|
fs::write(path, &content).await?;
|
2026-01-25 20:01:50 +01:00
|
|
|
}
|
2026-05-20 12:23:47 +02:00
|
|
|
}
|
|
|
|
|
K::Dir => {
|
|
|
|
|
if dry_run {
|
|
|
|
|
info!("would create {} (directory)", file.path);
|
|
|
|
|
} else {
|
|
|
|
|
fs::create_dir(path).await?;
|
2026-02-21 18:15:39 +01:00
|
|
|
}
|
|
|
|
|
}
|
2026-05-20 12:23:47 +02:00
|
|
|
K::Symlink(tgt) => {
|
|
|
|
|
if dry_run {
|
|
|
|
|
info!("would create {} (symlink to {})", file.path, tgt);
|
|
|
|
|
} else {
|
|
|
|
|
let _ = fs::remove_file(path).await; // we're ln --force
|
|
|
|
|
fs::symlink(tgt, path).await?;
|
|
|
|
|
}
|
2025-07-17 15:42:08 +02:00
|
|
|
}
|
2026-05-20 12:23:47 +02:00
|
|
|
}
|
2025-07-17 15:42:08 +02:00
|
|
|
|
2026-05-20 12:23:47 +02:00
|
|
|
if dry_run {
|
|
|
|
|
return Ok(());
|
|
|
|
|
}
|
2025-07-17 15:42:08 +02:00
|
|
|
|
2026-05-20 12:23:47 +02:00
|
|
|
if !file.is_symlink() {
|
|
|
|
|
set_perms(path, file.mode).await?;
|
2025-07-17 15:42:08 +02:00
|
|
|
}
|
|
|
|
|
|
2026-05-20 12:23:47 +02:00
|
|
|
info!("created {}", file.path);
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-17 15:42:08 +02:00
|
|
|
pub async fn set_perms(path: impl AsRef<Path>, mode: Option<u32>) -> std::io::Result<()> {
|
|
|
|
|
if let Some(mode) = mode.filter(|m| *m != 0) {
|
|
|
|
|
use std::os::unix::fs::PermissionsExt;
|
|
|
|
|
let mode = std::fs::Permissions::from_mode(mode);
|
|
|
|
|
fs::set_permissions(path, mode).await?;
|
|
|
|
|
}
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn chroot(root: &str, path: &str) -> String {
|
|
|
|
|
format!("{root}/{}", path.trim_start_matches(|c| c == '/'))
|
|
|
|
|
}
|