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}