openclaw/src/config/paths.ts

285 lines
8.7 KiB
TypeScript
Raw Normal View History

2026-01-28 00:15:54 +00:00
import fs from "node:fs";
2026-01-04 07:05:04 +01:00
import os from "node:os";
import path from "node:path";
2026-02-18 01:29:02 +00:00
import { expandHomePrefix, resolveRequiredHomeDir } from "../infra/home-dir.js";
import type { OpenClawConfig } from "./types.js";
2026-01-04 07:05:04 +01:00
/**
2026-01-30 03:15:10 +01:00
* Nix mode detection: When OPENCLAW_NIX_MODE=1, the gateway is running under Nix.
2026-01-04 07:05:04 +01:00
* In this mode:
* - No auto-install flows should be attempted
* - Missing dependencies should produce actionable Nix-specific error messages
* - Config is managed externally (read-only from Nix perspective)
*/
export function resolveIsNixMode(env: NodeJS.ProcessEnv = process.env): boolean {
2026-01-30 03:15:10 +01:00
return env.OPENCLAW_NIX_MODE === "1";
2026-01-04 07:05:04 +01:00
}
export const isNixMode = resolveIsNixMode();
2026-02-14 17:14:15 +00:00
// Support historical (and occasionally misspelled) legacy state dirs.
const LEGACY_STATE_DIRNAMES = [".clawdbot", ".moldbot", ".moltbot"] as const;
2026-01-30 03:15:10 +01:00
const NEW_STATE_DIRNAME = ".openclaw";
const CONFIG_FILENAME = "openclaw.json";
2026-02-14 17:14:15 +00:00
const LEGACY_CONFIG_FILENAMES = ["clawdbot.json", "moldbot.json", "moltbot.json"] as const;
function resolveDefaultHomeDir(): string {
return resolveRequiredHomeDir(process.env, os.homedir);
}
/** Build a homedir thunk that respects OPENCLAW_HOME for the given env. */
function envHomedir(env: NodeJS.ProcessEnv): () => string {
return () => resolveRequiredHomeDir(env, os.homedir);
}
function legacyStateDirs(homedir: () => string = resolveDefaultHomeDir): string[] {
2026-01-30 03:15:10 +01:00
return LEGACY_STATE_DIRNAMES.map((dir) => path.join(homedir(), dir));
}
function newStateDir(homedir: () => string = resolveDefaultHomeDir): string {
return path.join(homedir(), NEW_STATE_DIRNAME);
}
export function resolveLegacyStateDir(homedir: () => string = resolveDefaultHomeDir): string {
2026-01-30 03:15:10 +01:00
return legacyStateDirs(homedir)[0] ?? newStateDir(homedir);
}
export function resolveLegacyStateDirs(homedir: () => string = resolveDefaultHomeDir): string[] {
2026-01-30 03:15:10 +01:00
return legacyStateDirs(homedir);
2026-01-28 00:15:54 +00:00
}
export function resolveNewStateDir(homedir: () => string = resolveDefaultHomeDir): string {
2026-01-28 00:15:54 +00:00
return newStateDir(homedir);
}
2026-01-04 07:05:04 +01:00
/**
* State directory for mutable data (sessions, logs, caches).
2026-01-30 03:15:10 +01:00
* Can be overridden via OPENCLAW_STATE_DIR.
* Default: ~/.openclaw
2026-01-04 07:05:04 +01:00
*/
export function resolveStateDir(
env: NodeJS.ProcessEnv = process.env,
homedir: () => string = envHomedir(env),
2026-01-04 07:05:04 +01:00
): string {
const effectiveHomedir = () => resolveRequiredHomeDir(env, homedir);
2026-01-30 03:15:10 +01:00
const override = env.OPENCLAW_STATE_DIR?.trim() || env.CLAWDBOT_STATE_DIR?.trim();
if (override) {
return resolveUserPath(override, env, effectiveHomedir);
}
const newDir = newStateDir(effectiveHomedir);
if (env.OPENCLAW_TEST_FAST === "1") {
return newDir;
}
const legacyDirs = legacyStateDirs(effectiveHomedir);
2026-01-28 00:15:54 +00:00
const hasNew = fs.existsSync(newDir);
if (hasNew) {
return newDir;
}
2026-01-30 03:15:10 +01:00
const existingLegacy = legacyDirs.find((dir) => {
try {
return fs.existsSync(dir);
} catch {
return false;
}
});
if (existingLegacy) {
return existingLegacy;
}
2026-01-30 03:15:10 +01:00
return newDir;
2026-01-04 07:05:04 +01:00
}
function resolveUserPath(
input: string,
env: NodeJS.ProcessEnv = process.env,
homedir: () => string = envHomedir(env),
): string {
const trimmed = input.trim();
if (!trimmed) {
return trimmed;
}
if (trimmed.startsWith("~")) {
const expanded = expandHomePrefix(trimmed, {
home: resolveRequiredHomeDir(env, homedir),
env,
homedir,
});
2026-01-08 03:02:46 +00:00
return path.resolve(expanded);
}
return path.resolve(trimmed);
}
export const STATE_DIR = resolveStateDir();
2026-01-04 07:05:04 +01:00
/**
* Config file path (JSON5).
2026-01-30 03:15:10 +01:00
* Can be overridden via OPENCLAW_CONFIG_PATH.
* Default: ~/.openclaw/openclaw.json (or $OPENCLAW_STATE_DIR/openclaw.json)
2026-01-04 07:05:04 +01:00
*/
2026-01-28 00:15:54 +00:00
export function resolveCanonicalConfigPath(
env: NodeJS.ProcessEnv = process.env,
stateDir: string = resolveStateDir(env, envHomedir(env)),
2026-01-28 00:15:54 +00:00
): string {
2026-01-30 03:15:10 +01:00
const override = env.OPENCLAW_CONFIG_PATH?.trim() || env.CLAWDBOT_CONFIG_PATH?.trim();
if (override) {
return resolveUserPath(override, env, envHomedir(env));
}
2026-01-28 00:15:54 +00:00
return path.join(stateDir, CONFIG_FILENAME);
}
/**
* Resolve the active config path by preferring existing config candidates
2026-01-30 03:15:10 +01:00
* before falling back to the canonical path.
2026-01-28 00:15:54 +00:00
*/
export function resolveConfigPathCandidate(
env: NodeJS.ProcessEnv = process.env,
homedir: () => string = envHomedir(env),
2026-01-28 00:15:54 +00:00
): string {
if (env.OPENCLAW_TEST_FAST === "1") {
return resolveCanonicalConfigPath(env, resolveStateDir(env, homedir));
}
2026-01-28 00:15:54 +00:00
const candidates = resolveDefaultConfigCandidates(env, homedir);
const existing = candidates.find((candidate) => {
try {
return fs.existsSync(candidate);
} catch {
return false;
}
});
if (existing) {
return existing;
}
2026-01-28 00:15:54 +00:00
return resolveCanonicalConfigPath(env, resolveStateDir(env, homedir));
}
/**
2026-01-30 03:15:10 +01:00
* Active config path (prefers existing config files).
2026-01-28 00:15:54 +00:00
*/
2026-01-04 07:05:04 +01:00
export function resolveConfigPath(
env: NodeJS.ProcessEnv = process.env,
stateDir: string = resolveStateDir(env, envHomedir(env)),
homedir: () => string = envHomedir(env),
2026-01-04 07:05:04 +01:00
): string {
2026-01-30 03:15:10 +01:00
const override = env.OPENCLAW_CONFIG_PATH?.trim();
if (override) {
return resolveUserPath(override, env, homedir);
}
if (env.OPENCLAW_TEST_FAST === "1") {
return path.join(stateDir, CONFIG_FILENAME);
}
2026-01-30 03:15:10 +01:00
const stateOverride = env.OPENCLAW_STATE_DIR?.trim();
2026-01-28 00:15:54 +00:00
const candidates = [
path.join(stateDir, CONFIG_FILENAME),
2026-01-30 03:15:10 +01:00
...LEGACY_CONFIG_FILENAMES.map((name) => path.join(stateDir, name)),
2026-01-28 00:15:54 +00:00
];
const existing = candidates.find((candidate) => {
try {
return fs.existsSync(candidate);
} catch {
return false;
}
});
if (existing) {
return existing;
}
if (stateOverride) {
return path.join(stateDir, CONFIG_FILENAME);
}
2026-01-28 00:15:54 +00:00
const defaultStateDir = resolveStateDir(env, homedir);
if (path.resolve(stateDir) === path.resolve(defaultStateDir)) {
return resolveConfigPathCandidate(env, homedir);
}
return path.join(stateDir, CONFIG_FILENAME);
2026-01-04 07:05:04 +01:00
}
2026-01-28 00:15:54 +00:00
export const CONFIG_PATH = resolveConfigPathCandidate();
2026-01-04 07:05:04 +01:00
/**
2026-01-30 03:15:10 +01:00
* Resolve default config path candidates across default locations.
* Order: explicit config path state-dir-derived paths new default.
*/
export function resolveDefaultConfigCandidates(
env: NodeJS.ProcessEnv = process.env,
homedir: () => string = envHomedir(env),
): string[] {
const effectiveHomedir = () => resolveRequiredHomeDir(env, homedir);
2026-01-30 03:15:10 +01:00
const explicit = env.OPENCLAW_CONFIG_PATH?.trim() || env.CLAWDBOT_CONFIG_PATH?.trim();
if (explicit) {
return [resolveUserPath(explicit, env, effectiveHomedir)];
}
const candidates: string[] = [];
2026-01-30 03:15:10 +01:00
const openclawStateDir = env.OPENCLAW_STATE_DIR?.trim() || env.CLAWDBOT_STATE_DIR?.trim();
if (openclawStateDir) {
const resolved = resolveUserPath(openclawStateDir, env, effectiveHomedir);
2026-01-30 03:15:10 +01:00
candidates.push(path.join(resolved, CONFIG_FILENAME));
candidates.push(...LEGACY_CONFIG_FILENAMES.map((name) => path.join(resolved, name)));
}
const defaultDirs = [newStateDir(effectiveHomedir), ...legacyStateDirs(effectiveHomedir)];
2026-01-30 03:15:10 +01:00
for (const dir of defaultDirs) {
candidates.push(path.join(dir, CONFIG_FILENAME));
candidates.push(...LEGACY_CONFIG_FILENAMES.map((name) => path.join(dir, name)));
}
return candidates;
}
2026-01-04 07:05:04 +01:00
export const DEFAULT_GATEWAY_PORT = 18789;
2026-01-25 09:21:39 +00:00
/**
* Gateway lock directory (ephemeral).
2026-01-30 03:15:10 +01:00
* Default: os.tmpdir()/openclaw-<uid> (uid suffix when available).
2026-01-25 09:21:39 +00:00
*/
export function resolveGatewayLockDir(tmpdir: () => string = os.tmpdir): string {
const base = tmpdir();
const uid = typeof process.getuid === "function" ? process.getuid() : undefined;
2026-01-30 03:15:10 +01:00
const suffix = uid != null ? `openclaw-${uid}` : "openclaw";
2026-01-25 09:21:39 +00:00
return path.join(base, suffix);
}
const OAUTH_FILENAME = "oauth.json";
/**
* OAuth credentials storage directory.
*
* Precedence:
2026-01-30 03:15:10 +01:00
* - `OPENCLAW_OAUTH_DIR` (explicit override)
* - `$*_STATE_DIR/credentials` (canonical server/default)
*/
export function resolveOAuthDir(
env: NodeJS.ProcessEnv = process.env,
stateDir: string = resolveStateDir(env, envHomedir(env)),
): string {
2026-01-30 03:15:10 +01:00
const override = env.OPENCLAW_OAUTH_DIR?.trim();
if (override) {
return resolveUserPath(override, env, envHomedir(env));
}
return path.join(stateDir, "credentials");
}
export function resolveOAuthPath(
env: NodeJS.ProcessEnv = process.env,
stateDir: string = resolveStateDir(env, envHomedir(env)),
): string {
return path.join(resolveOAuthDir(env, stateDir), OAUTH_FILENAME);
}
2026-01-04 07:05:04 +01:00
export function resolveGatewayPort(
2026-01-30 03:15:10 +01:00
cfg?: OpenClawConfig,
2026-01-04 07:05:04 +01:00
env: NodeJS.ProcessEnv = process.env,
): number {
2026-01-30 03:15:10 +01:00
const envRaw = env.OPENCLAW_GATEWAY_PORT?.trim() || env.CLAWDBOT_GATEWAY_PORT?.trim();
2026-01-04 07:05:04 +01:00
if (envRaw) {
const parsed = Number.parseInt(envRaw, 10);
if (Number.isFinite(parsed) && parsed > 0) {
return parsed;
}
2026-01-04 07:05:04 +01:00
}
const configPort = cfg?.gateway?.port;
if (typeof configPort === "number" && Number.isFinite(configPort)) {
if (configPort > 0) {
return configPort;
}
2026-01-04 07:05:04 +01:00
}
return DEFAULT_GATEWAY_PORT;
}