mirror of
https://github.com/zhom/banderole.git
synced 2026-06-06 06:23:53 +02:00
243 lines
7.9 KiB
Rust
243 lines
7.9 KiB
Rust
use crate::platform::Platform;
|
|
use anyhow::{Context, Result};
|
|
use futures_util::StreamExt;
|
|
use lazy_static::lazy_static;
|
|
use std::collections::HashMap;
|
|
use std::path::{Path, PathBuf};
|
|
use std::sync::Mutex;
|
|
use tokio::fs;
|
|
use tokio::io::AsyncWriteExt;
|
|
|
|
lazy_static! {
|
|
static ref NODE_VERSION_CACHE: Mutex<HashMap<String, PathBuf>> = Mutex::new(HashMap::new());
|
|
}
|
|
|
|
pub struct NodeDownloader {
|
|
platform: Platform,
|
|
cache_dir: PathBuf,
|
|
node_version: String,
|
|
}
|
|
|
|
impl NodeDownloader {
|
|
pub fn new_with_persistent_cache(node_version: String) -> Result<Self> {
|
|
let cache_dir = Self::get_persistent_cache_dir()?;
|
|
Ok(Self {
|
|
platform: Platform::current(),
|
|
cache_dir,
|
|
node_version,
|
|
})
|
|
}
|
|
|
|
fn get_persistent_cache_dir() -> Result<PathBuf> {
|
|
let cache_dir = if let Some(cache_home) = std::env::var_os("XDG_CACHE_HOME") {
|
|
PathBuf::from(cache_home).join("banderole")
|
|
} else if let Some(home) = std::env::var_os("HOME") {
|
|
PathBuf::from(home).join(".cache").join("banderole")
|
|
} else if let Some(appdata) = std::env::var_os("APPDATA") {
|
|
PathBuf::from(appdata).join("banderole").join("cache")
|
|
} else {
|
|
std::env::temp_dir().join("banderole-cache")
|
|
};
|
|
|
|
std::fs::create_dir_all(&cache_dir)
|
|
.context("Failed to create persistent cache directory")?;
|
|
|
|
Ok(cache_dir)
|
|
}
|
|
|
|
pub async fn ensure_node_binary(&self) -> Result<PathBuf> {
|
|
// Create cache key for this version and platform
|
|
let cache_key = format!("{}:{}", self.node_version, self.platform);
|
|
|
|
// Check in-memory cache first
|
|
{
|
|
let cache = NODE_VERSION_CACHE
|
|
.lock()
|
|
.map_err(|e| anyhow::anyhow!("Failed to acquire cache lock: {}", e))?;
|
|
if let Some(cached_path) = cache.get(&cache_key) {
|
|
if cached_path.exists() {
|
|
return Ok(cached_path.clone());
|
|
}
|
|
}
|
|
}
|
|
|
|
// Check disk cache
|
|
let node_dir = self
|
|
.cache_dir
|
|
.join("node")
|
|
.join(&self.node_version)
|
|
.join(self.platform.to_string());
|
|
|
|
let node_executable = node_dir.join(self.platform.node_executable_path());
|
|
|
|
if node_executable.exists() {
|
|
// Update in-memory cache
|
|
let mut cache = NODE_VERSION_CACHE
|
|
.lock()
|
|
.map_err(|e| anyhow::anyhow!("Failed to acquire cache lock: {}", e))?;
|
|
cache.insert(cache_key, node_executable.clone());
|
|
return Ok(node_executable);
|
|
}
|
|
|
|
println!(
|
|
"Downloading Node.js {} for {}...",
|
|
self.node_version, self.platform
|
|
);
|
|
|
|
// Create cache directory
|
|
fs::create_dir_all(&node_dir)
|
|
.await
|
|
.context("Failed to create node cache directory")?;
|
|
|
|
// Download and extract Node.js
|
|
self.download_and_extract_node(&node_dir).await?;
|
|
|
|
if !node_executable.exists() {
|
|
anyhow::bail!(
|
|
"Node executable not found after extraction: {}",
|
|
node_executable.display()
|
|
);
|
|
}
|
|
|
|
// Make executable on Unix systems
|
|
#[cfg(unix)]
|
|
{
|
|
use std::os::unix::fs::PermissionsExt;
|
|
let mut perms = fs::metadata(&node_executable).await?.permissions();
|
|
perms.set_mode(0o755);
|
|
fs::set_permissions(&node_executable, perms).await?;
|
|
}
|
|
|
|
Ok(node_executable)
|
|
}
|
|
|
|
async fn download_and_extract_node(&self, target_dir: &Path) -> Result<()> {
|
|
let archive_name = self.platform.node_archive_name(&self.node_version);
|
|
let url = format!(
|
|
"https://nodejs.org/dist/v{}/{}",
|
|
self.node_version, archive_name
|
|
);
|
|
|
|
// Download the archive
|
|
let response = reqwest::get(&url)
|
|
.await
|
|
.context("Failed to download Node.js archive")?;
|
|
|
|
if !response.status().is_success() {
|
|
anyhow::bail!("Failed to download Node.js: HTTP {}", response.status());
|
|
}
|
|
|
|
let archive_path = target_dir.join(&archive_name);
|
|
let mut file = fs::File::create(&archive_path)
|
|
.await
|
|
.context("Failed to create archive file")?;
|
|
|
|
let mut stream = response.bytes_stream();
|
|
while let Some(chunk) = stream.next().await {
|
|
let chunk = chunk.context("Failed to read download chunk")?;
|
|
file.write_all(&chunk)
|
|
.await
|
|
.context("Failed to write archive chunk")?;
|
|
}
|
|
|
|
file.flush().await.context("Failed to flush archive file")?;
|
|
drop(file);
|
|
|
|
// Extract the archive
|
|
if self.platform.is_windows() {
|
|
self.extract_zip(&archive_path, target_dir).await?;
|
|
} else {
|
|
self.extract_tar_gz(&archive_path, target_dir).await?;
|
|
}
|
|
|
|
// Clean up archive
|
|
fs::remove_file(&archive_path)
|
|
.await
|
|
.context("Failed to remove archive file")?;
|
|
|
|
// Update in-memory cache with the path to the node executable
|
|
let node_executable_path = target_dir.join(self.platform.node_executable_path());
|
|
let mut cache = NODE_VERSION_CACHE
|
|
.lock()
|
|
.map_err(|e| anyhow::anyhow!("Failed to acquire cache lock: {}", e))?;
|
|
cache.insert(
|
|
format!("{}:{}", self.node_version, self.platform),
|
|
node_executable_path.clone(),
|
|
);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
async fn extract_zip(&self, archive_path: &Path, target_dir: &Path) -> Result<()> {
|
|
let file = std::fs::File::open(archive_path).context("Failed to open zip archive")?;
|
|
|
|
let mut archive = zip::ZipArchive::new(file).context("Failed to read zip archive")?;
|
|
|
|
for i in 0..archive.len() {
|
|
let mut file = archive.by_index(i).context("Failed to read zip entry")?;
|
|
|
|
let outpath = match file.enclosed_name() {
|
|
Some(path) => {
|
|
// Remove the top-level directory from the path
|
|
let components: Vec<_> = path.components().collect();
|
|
if components.len() > 1 {
|
|
target_dir.join(components[1..].iter().collect::<PathBuf>())
|
|
} else {
|
|
continue;
|
|
}
|
|
}
|
|
None => continue,
|
|
};
|
|
|
|
if file.is_dir() {
|
|
fs::create_dir_all(&outpath)
|
|
.await
|
|
.context("Failed to create directory")?;
|
|
} else {
|
|
if let Some(p) = outpath.parent() {
|
|
fs::create_dir_all(p)
|
|
.await
|
|
.context("Failed to create parent directory")?;
|
|
}
|
|
|
|
let mut outfile = fs::File::create(&outpath)
|
|
.await
|
|
.context("Failed to create output file")?;
|
|
|
|
let mut buffer = Vec::new();
|
|
std::io::copy(&mut file, &mut buffer).context("Failed to read zip entry")?;
|
|
|
|
outfile
|
|
.write_all(&buffer)
|
|
.await
|
|
.context("Failed to write output file")?;
|
|
}
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
async fn extract_tar_gz(&self, archive_path: &Path, target_dir: &Path) -> Result<()> {
|
|
let output = tokio::process::Command::new("tar")
|
|
.args(&[
|
|
"-xzf",
|
|
archive_path.to_str().unwrap(),
|
|
"-C",
|
|
target_dir.to_str().unwrap(),
|
|
"--strip-components=1",
|
|
])
|
|
.output()
|
|
.await
|
|
.context("Failed to execute tar command")?;
|
|
|
|
if !output.status.success() {
|
|
anyhow::bail!(
|
|
"tar extraction failed: {}",
|
|
String::from_utf8_lossy(&output.stderr)
|
|
);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|