You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
nenv/src/mapper/mapped_dir.rs

112 lines
3.0 KiB
Rust

use std::{collections::HashSet, io, path::Path};
use tokio::fs::{self, DirEntry};
use crate::{consts::BIN_DIR, error::MapDirError, repository::node_path::NodePath};
use miette::{Context, IntoDiagnostic, Result};
struct NodeApp {
info: DirEntry,
name: String,
}
impl NodeApp {
pub fn new(info: DirEntry) -> Self {
let path = info.path();
let name = path.file_stem().unwrap();
let name = name.to_string_lossy().into_owned();
Self { info, name }
}
pub fn name(&self) -> &String {
&self.name
}
/// creates wrappers to map this application
pub async fn map_executable(&self) -> Result<()> {
let src_path = BIN_DIR.join(self.info.file_name());
self.write_wrapper_script(&src_path)
.await
.into_diagnostic()
.context("Creating executable wrapper script")
}
#[cfg(not(target_os = "windows"))]
async fn write_wrapper_script(&self, path: &Path) -> Result<(), io::Error> {
fs::write(path, format!("#!/bin/sh\nnenv exec {} \"$@\"", self.name)).await?;
let src_metadata = self.info.metadata().await?;
fs::set_permissions(&path, src_metadata.permissions()).await?;
Ok(())
}
#[cfg(target_os = "windows")]
async fn write_wrapper_script(&self, path: &Path) -> Result<(), io::Error> {
fs::write(
path.with_extension("bat"),
format!("@echo off\nnenv exec {} %*", self.name),
)
.await?;
let src_metadata = self.info.metadata().await?;
fs::set_permissions(&path, src_metadata.permissions()).await?;
Ok(())
}
}
pub async fn map_node_bin(node_path: &NodePath) -> Result<()> {
let mapped_app_names = get_applications(&BIN_DIR)
.await?
.iter()
.map(NodeApp::name)
.cloned()
.collect::<HashSet<_>>();
let mut applications = get_applications(&node_path.bin()).await?;
applications.retain(|app| !mapped_app_names.contains(app.name()));
futures::future::join_all(applications.iter().map(NodeApp::map_executable)).await;
Ok(())
}
async fn get_applications(path: &Path) -> Result<Vec<NodeApp>> {
let mut files = Vec::new();
let mut iter = fs::read_dir(path).await.map_err(|err| MapDirError {
dir: path.to_owned(),
caused_by: err,
})?;
while let Some(entry) = iter
.next_entry()
.await
.into_diagnostic()
.context("Reading directory entries")?
{
let entry_path = entry.path();
if entry_path.is_file() && !exclude_path(&entry_path) {
files.push(NodeApp::new(entry));
}
}
Ok(files)
}
#[cfg(not(target_os = "windows"))]
fn exclude_path(_path: &Path) -> bool {
false
}
#[cfg(target_os = "windows")]
fn exclude_path(path: &Path) -> bool {
let Some(extension) = path.extension() else {
return true;
};
let extension = extension.to_string_lossy();
let allowed_extensions = ["exe", "bat", "cmd", "ps1"];
!allowed_extensions.contains(&extension.as_ref())
}