use crate::config::FnmConfig; use crate::default_version; use crate::package_json::PackageJson; use crate::user_version::UserVersion; use crate::version_file_strategy::VersionFileStrategy; use encoding_rs_io::DecodeReaderBytes; use log::info; use std::io::Read; use std::path::Path; use std::str::FromStr; const PATH_PARTS: [&str; 3] = [".nvmrc", ".node-version", "package.json"]; pub fn get_user_version_for_directory( path: impl AsRef, config: &FnmConfig, ) -> Option { match config.version_file_strategy() { VersionFileStrategy::Local => get_user_version_for_single_directory(path, config), VersionFileStrategy::Recursive => get_user_version_for_directory_recursive(path, config) .or_else(|| { info!("Did not find anything recursively. Falling back to default alias."); default_version::find_default_version(config).map(UserVersion::Full) }), } } fn get_user_version_for_directory_recursive( path: impl AsRef, config: &FnmConfig, ) -> Option { let mut current_path = Some(path.as_ref()); while let Some(child_path) = current_path { if let Some(version) = get_user_version_for_single_directory(child_path, config) { return Some(version); } current_path = child_path.parent(); } None } fn get_user_version_for_single_directory( path: impl AsRef, config: &FnmConfig, ) -> Option { let path = path.as_ref(); for path_part in &PATH_PARTS { let new_path = path.join(path_part); info!( "Looking for version file in {}. exists? {}", new_path.display(), new_path.exists() ); if let Some(version) = get_user_version_for_file(&new_path, config) { return Some(version); } } None } pub fn get_user_version_for_file( path: impl AsRef, config: &FnmConfig, ) -> Option { let is_pkg_json = match path.as_ref().file_name() { Some(name) => name == "package.json", None => false, }; let file = std::fs::File::open(path).ok()?; let file = { let mut reader = DecodeReaderBytes::new(file); let mut version = String::new(); reader.read_to_string(&mut version).map(|_| version) }; match (file, is_pkg_json, config.resolve_engines()) { (_, true, false) => None, (Err(err), _, _) => { info!("Can't read file: {}", err); None } (Ok(version), false, _) => { info!("Found string {:?} in version file", version); UserVersion::from_str(version.trim()).ok() } (Ok(pkg_json), true, true) => { let pkg_json = serde_json::from_str::(&pkg_json).ok(); let range: Option = pkg_json.as_ref().and_then(PackageJson::node_range).cloned(); if let Some(range) = range { info!("Found package.json with {:?} in engines.node field", range); Some(UserVersion::SemverRange(range)) } else { info!("No engines.node range found in package.json"); None } } } }