1
0
Fork 0

Begin implement symlink system

This commit is contained in:
Florian RICHER (MrDev023) 2021-08-09 22:14:23 +02:00
parent e91115cbe4
commit ec5d28731c
6 changed files with 84 additions and 12 deletions

View file

@ -20,7 +20,7 @@ pub struct GitConfig {
pub install_type: file_utils::InstallType, pub install_type: file_utils::InstallType,
} }
fn _get_file_url(package: &str, version: &str, git_identifiers: Vec<GitFileIdentifier>) -> Result<String, String> { fn get_file_url(package: &str, version: &str, git_identifiers: Vec<GitFileIdentifier>) -> Result<String, String> {
let git_response = git::get_git_release_by_version(package, version) let git_response = git::get_git_release_by_version(package, version)
.ok_or(format!("Failed to get git release"))?; .ok_or(format!("Failed to get git release"))?;
@ -39,8 +39,8 @@ fn _get_file_url(package: &str, version: &str, git_identifiers: Vec<GitFileIdent
Err(format!("{} not support current OS or CPU ARCH", package)) Err(format!("{} not support current OS or CPU ARCH", package))
} }
fn _install(config: GitConfig) -> Result<(), String> { fn install(config: GitConfig) -> Result<(), String> {
let url = _get_file_url(config.package.as_str(), config.version.as_str(), config.git_identifiers)?; let url = get_file_url(config.package.as_str(), config.version.as_str(), config.git_identifiers)?;
let filename = url.split('/').last() let filename = url.split('/').last()
.ok_or(format!("Failed to get name of file"))?; .ok_or(format!("Failed to get name of file"))?;
@ -57,14 +57,14 @@ fn _install(config: GitConfig) -> Result<(), String> {
.to_str() .to_str()
.ok_or(format!("Failed to get extracted folder"))?; .ok_or(format!("Failed to get extracted folder"))?;
file_utils::install(extractor_package_temp_folder_str, config.install_type).ok_or(format!("Failed to install"))?; file_utils::install(extractor_package_temp_folder_str, config.package.as_str(), config.install_type).ok_or(format!("Failed to install"))?;
Ok(()) Ok(())
} }
pub fn process(config: GitConfig, config_mode: configure::ConfigMode) -> Result<(), String> { pub fn process(config: GitConfig, config_mode: configure::ConfigMode) -> Result<(), String> {
match config_mode { match config_mode {
configure::ConfigMode::INSTALL => _install(config), configure::ConfigMode::INSTALL => install(config),
configure::ConfigMode::UNINSTALL => Err(format!("not yet implemented")), configure::ConfigMode::UNINSTALL => Err(format!("not yet implemented")),
} }
} }

View file

@ -9,7 +9,7 @@ fn folder_exist(folder: &str) -> bool {
} }
pub fn configure_folder(mode: &ConfigMode) -> Option<()> { pub fn configure_folder(mode: &ConfigMode) -> Option<()> {
let binary_folder_path = file_utils::get_install_dir(file_utils::InstallType::Root).ok()?; let binary_folder_path = file_utils::get_install_dir(&file_utils::InstallType::Root).ok()?;
let binary_folder_str = binary_folder_path.to_str()?; let binary_folder_str = binary_folder_path.to_str()?;
match mode { match mode {

View file

@ -17,7 +17,7 @@ pub fn configure_env(mode: &ConfigMode) -> Option<()> {
let hklm = RegKey::predef(HKEY_CURRENT_USER); let hklm = RegKey::predef(HKEY_CURRENT_USER);
let environment = hklm.open_subkey_with_flags("Environment", KEY_READ | KEY_WRITE).ok()?; let environment = hklm.open_subkey_with_flags("Environment", KEY_READ | KEY_WRITE).ok()?;
let mut reg_value : String = environment.get_value("PATH").ok()?; let mut reg_value : String = environment.get_value("PATH").ok()?;
let binary_folder_path = file_utils::get_install_dir(file_utils::InstallType::Command).ok()?; let binary_folder_path = file_utils::get_install_dir(&file_utils::InstallType::Command).ok()?;
let binary_folder_str = binary_folder_path.to_str()?; let binary_folder_str = binary_folder_path.to_str()?;
match mode { match mode {

View file

@ -0,0 +1,56 @@
use std::env::consts;
use std::path::PathBuf;
use std::fs;
#[cfg(target_os = "windows")]
use std::os::windows::fs::symlink_file;
fn file_is_executable(path: &PathBuf) -> bool {
if consts::FAMILY == "windows" {
match path.extension() {
Some(ext) => {
let extension = match ext.to_str() {
Some(r) => r,
None => ""
};
match extension {
"bat" | "exe" | "ps1" => true,
_ => false,
}
},
None => false
}
} else {
todo!()
}
}
fn get_binary_files(install_package_folder: &PathBuf) -> Result<Vec<PathBuf>, String> {
let mut files: Vec<PathBuf> = vec![];
let paths = fs::read_dir(install_package_folder).map_err(|_| format!("Failed to get binary files"))?;
for path_result in paths {
let path = path_result.map_err(|_| format!("Failed to get file"))?;
let metadata = path.metadata().map_err(|_| format!("Failed to get metadata"))?;
if metadata.is_file() && file_is_executable(&path.path()) {
files.push(path.path().clone());
}
}
Ok(files)
}
pub fn create_symlink_of_binary_files(install_package_folder: &PathBuf, binary_package_folder: &PathBuf) -> Result<(), String> {
let files = get_binary_files(install_package_folder)?;
fs::create_dir_all(&binary_package_folder).map_err(|_| format!("Failed to create bin folder"))?;
for file in files {
let output_path = binary_package_folder.join(file.file_name().ok_or(format!("Failed to get filename"))?);
#[cfg(target_os = "windows")]
symlink_file(file.as_os_str(), output_path.as_os_str()).map_err(|_| format!("Failed to create symlink"))?;
}
Ok(())
}

View file

@ -1,3 +1,5 @@
mod binary_utils;
use std::path::{PathBuf, Path}; use std::path::{PathBuf, Path};
use std::{io, fs}; use std::{io, fs};
@ -7,7 +9,7 @@ pub enum InstallType {
Root Root
} }
pub fn get_install_dir(install_type: InstallType) -> Result<PathBuf, String> { pub fn get_install_dir(install_type: &InstallType) -> Result<PathBuf, String> {
let home_dir = dirs::home_dir() let home_dir = dirs::home_dir()
.ok_or(format!("Failed to get home_directory"))?; .ok_or(format!("Failed to get home_directory"))?;
@ -20,6 +22,12 @@ pub fn get_install_dir(install_type: InstallType) -> Result<PathBuf, String> {
Ok(home_dir.join(super::INSTALL_FOLDER).join(subfolder)) Ok(home_dir.join(super::INSTALL_FOLDER).join(subfolder))
} }
pub fn get_packages_dir(package: &str) -> Result<PathBuf, String> {
let home_dir = dirs::home_dir()
.ok_or(format!("Failed to get home_directory"))?;
Ok(home_dir.join(super::INSTALL_FOLDER).join(super::PACKAGE_FOLDER).join(package))
}
fn copy_dir_all(src: impl AsRef<Path>, dst: impl AsRef<Path>) -> io::Result<()> { fn copy_dir_all(src: impl AsRef<Path>, dst: impl AsRef<Path>) -> io::Result<()> {
fs::create_dir_all(&dst)?; fs::create_dir_all(&dst)?;
for entry in fs::read_dir(src)? { for entry in fs::read_dir(src)? {
@ -34,10 +42,17 @@ fn copy_dir_all(src: impl AsRef<Path>, dst: impl AsRef<Path>) -> io::Result<()>
Ok(()) Ok(())
} }
pub fn install(source_path: &str, install_type: InstallType) -> Option <()> { pub fn install(source_path: &str, package: &str, install_type: InstallType) -> Option <()> {
let source_folder = std::path::Path::new(super::TEMP_FOLDER).join(source_path); let source_folder = std::path::Path::new(super::TEMP_FOLDER).join(source_path);
let package_folder = get_packages_dir(package).ok()?;
let install_folder = get_install_dir(install_type).ok()?; copy_dir_all(source_folder, package_folder.clone()).ok()?;
copy_dir_all(source_folder, install_folder).ok()?;
Some(()) let install_folder = get_install_dir(&install_type).ok()?;
match install_type {
InstallType::Command => binary_utils::create_symlink_of_binary_files(&package_folder, &install_folder).ok(),
InstallType::Config => todo!(),
InstallType::Root => todo!(),
}
} }

View file

@ -1,5 +1,6 @@
pub(self) static TEMP_FOLDER: &str = "temp"; pub(self) static TEMP_FOLDER: &str = "temp";
pub(self) static INSTALL_FOLDER: &str = ".autoconfig"; pub(self) static INSTALL_FOLDER: &str = ".autoconfig";
pub(self) static PACKAGE_FOLDER : &str = "packages";
pub mod downloader; pub mod downloader;
pub mod extractor; pub mod extractor;