Skip to main content

Mountain/ProcessManagement/NodeResolver/
TryNvm.rs

1#![allow(non_snake_case)]
2
3//! nvm lookup. `NVM_BIN` wins (set inside an nvm-sourced shell). Fallback
4//! walks `$NVM_DIR/versions/node` and picks the lexicographically largest
5//! version (rough proxy for "latest installed").
6
7use std::path::PathBuf;
8
9use crate::ProcessManagement::NodeResolver::{NodeExecutableName, NodeSource, ResolvedNode};
10
11pub fn Fn() -> Option<ResolvedNode::Struct> {
12	if let Ok(NvmBin) = std::env::var("NVM_BIN") {
13		let Candidate = PathBuf::from(NvmBin).join(NodeExecutableName::Fn());
14
15		if Candidate.exists() {
16			return Some(ResolvedNode::Struct { Path:Candidate, Source:NodeSource::Enum::Nvm });
17		}
18	}
19
20	let NvmDir = std::env::var("NVM_DIR").ok().or_else(|| {
21		std::env::var("HOME")
22			.ok()
23			.map(|H| PathBuf::from(H).join(".nvm").to_string_lossy().into_owned())
24	})?;
25
26	let VersionsDirectory = PathBuf::from(&NvmDir).join("versions").join("node");
27
28	let Entries = std::fs::read_dir(&VersionsDirectory).ok()?;
29
30	let mut BestCandidate:Option<PathBuf> = None;
31
32	for Entry in Entries.flatten() {
33		let NodePath = Entry.path().join("bin").join(NodeExecutableName::Fn());
34
35		if !NodePath.exists() {
36			continue;
37		}
38
39		BestCandidate = match BestCandidate {
40			Some(Existing) if Existing > NodePath => Some(Existing),
41
42			_ => Some(NodePath),
43		};
44	}
45
46	BestCandidate.map(|Path| ResolvedNode::Struct { Path, Source:NodeSource::Enum::Nvm })
47}