UNPKG

@nanocollective/nanocoder

Version:

A local-first CLI coding agent that brings the power of agentic coding tools like Claude Code and Gemini CLI to local models or controlled APIs like OpenRouter

132 lines 5.24 kB
import { existsSync } from 'fs'; import { dirname, join, sep } from 'path'; import { fileURLToPath } from 'url'; import { logWarning } from './message-queue.js'; const safeProcess = typeof process !== 'undefined' ? process : {}; /** * Detects installation method from a given module path. * Exported for testing purposes. * @param modulePath The path to check * @returns The detected installation method or null if not detected from path */ export function detectFromPath(modulePath) { // Strategy 1: Check for Nix installation (most specific) // Nix store has `/nix/store/` path with store hashes - this is very reliable if (modulePath.includes('/nix/store/')) { return 'nix'; } // Strategy 2: Check for Homebrew installation via path // Homebrew puts packages under Cellar directory in standard locations // Common paths: /opt/homebrew, /usr/local, /home/linuxbrew/.linuxbrew if (modulePath.includes(`${sep}Cellar${sep}`) || modulePath.includes(`${sep}homebrew${sep}`)) { return 'homebrew'; } // Strategy 3: Check for npm/pnpm/yarn installation using multiple signals if (isNpmBasedInstallation(modulePath)) { return 'npm'; } return null; } /** * Detects installation method from environment variables. * Exported for testing purposes. * @returns The detected installation method or null if not detected from env */ export function detectFromEnv() { // Check npm-specific env vars first (more specific to npm context) if (safeProcess.env?.npm_config_prefix || safeProcess.env?.npm_config_global || safeProcess.env?.PNPM_HOME || safeProcess.env?.npm_execpath) { return 'npm'; } // Homebrew env vars are a weak signal — HOMEBREW_PREFIX is set system-wide // on any macOS with Homebrew installed, not just for Homebrew-installed packages. // Only use as a last resort fallback. if (safeProcess.env?.HOMEBREW_PREFIX || safeProcess.env?.HOMEBREW_CELLAR) { return 'homebrew'; } return null; } /** * Detects how Nanocoder was installed by using multiple detection strategies. * Uses a combination of path inspection, environment variables, and file system markers. * An environment variable `NANOCODER_INSTALL_METHOD` can be used to override detection for testing. * @returns {InstallationMethod} The detected installation method. */ export function detectInstallationMethod() { // Env var override has highest priority for testing / debugging const envOverride = safeProcess.env?.NANOCODER_INSTALL_METHOD; if (envOverride) { const validMethods = [ 'npm', 'homebrew', 'nix', 'unknown', ]; if (validMethods.includes(envOverride)) { return envOverride; } // Warn about invalid value but continue with normal detection logWarning(`Invalid NANOCODER_INSTALL_METHOD: "${envOverride}". Valid values: ${validMethods.join(', ')}`); } // Strategy 1: Path-based detection (most reliable — checks actual install location) const modulePath = dirname(fileURLToPath(import.meta.url)); const pathResult = detectFromPath(modulePath); if (pathResult) { return pathResult; } // Strategy 2: Environment variables as fallback const envResult = detectFromEnv(); if (envResult) { return envResult; } return 'unknown'; } /** * Checks if this is an npm-based installation (npm, pnpm, or yarn) based on the module path. * Environment variable checks are handled separately by detectFromEnv(). */ function isNpmBasedInstallation(modulePath) { // Check 1: Standard node_modules path (npm, yarn v1) if (modulePath.includes('node_modules')) { return true; } // Check 2: pnpm store structure (.pnpm directory) if (modulePath.includes(`.pnpm${sep}`)) { return true; } // Check 3: Look for .bin directory in parent paths (all package managers use this) // This handles symlinked executables const binDirPattern = `${sep}.bin${sep}`; if (modulePath.includes(binDirPattern)) { return true; } // Check 4: Look for package.json in expected locations relative to the module // For global installs, package.json should be in parent directories // This handles edge cases like custom install locations return hasPackageJsonMarker(modulePath); } /** * Walks up the directory tree looking for package.json as a marker of npm installation. * Only checks a few levels to avoid excessive file system operations. */ function hasPackageJsonMarker(startPath) { let currentPath = startPath; const maxLevelsToCheck = 4; // Limit to prevent excessive traversal for (let i = 0; i < maxLevelsToCheck; i++) { const packageJsonPath = join(currentPath, 'package.json'); // nosemgrep if (existsSync(packageJsonPath)) { return true; } const parentPath = dirname(currentPath); // Stop if we've reached the root if (parentPath === currentPath) { break; } currentPath = parentPath; } return false; } //# sourceMappingURL=installation-detector.js.map