fix: handle EACCES on data/settings.json in Docker containers

- settings-store: auto-detect writable path at startup — tries
  <cwd>/data/settings.json first, falls back to /tmp/lingvai-settings.json
  if the directory is not writable. Logs a warning when fallback is used.
  Also supports SETTINGS_PATH env var for explicit override.

- Dockerfile: switch from yarn to npm, explicitly create /app/data with
  chown nextjs:nodejs so the directory is writable at runtime without
  needing a privileged volume mount.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-03-10 08:23:29 +01:00
parent 0799101da3
commit e034771087
2 changed files with 45 additions and 17 deletions

View File

@@ -17,18 +17,43 @@ const DEFAULT_SETTINGS: Settings = {
adminPasswordHash: process.env["ADMIN_PASSWORD"] ?? "admin"
};
const SETTINGS_PATH = path.join(process.cwd(), "data", "settings.json");
function ensureDataDir() {
const dir = path.dirname(SETTINGS_PATH);
if (!fs.existsSync(dir)) {
fs.mkdirSync(dir, { recursive: true });
/**
* Resolve a writable path for settings.json.
* Priority:
* 1. SETTINGS_PATH env var (explicit override)
* 2. <cwd>/data/settings.json (default, works when data/ is writable)
* 3. /tmp/lingvai-settings.json (fallback for read-only containers)
*/
function resolveSettingsPath(): string {
if (process.env["SETTINGS_PATH"]) {
return process.env["SETTINGS_PATH"];
}
const primary = path.join(process.cwd(), "data", "settings.json");
try {
const dir = path.dirname(primary);
fs.mkdirSync(dir, { recursive: true });
// Test write access by opening with 'a' (append/create without truncating)
const fd = fs.openSync(primary, "a");
fs.closeSync(fd);
return primary;
} catch {
return path.join("/tmp", "lingvai-settings.json");
}
}
// Resolve once at module load so every call uses the same path
const SETTINGS_PATH = resolveSettingsPath();
if (SETTINGS_PATH.startsWith("/tmp")) {
console.warn(
`[lingvai] data/settings.json is not writable. ` +
`Settings will be stored at ${SETTINGS_PATH}. ` +
`Mount a writable volume at /app/data or set SETTINGS_PATH to persist across restarts.`
);
}
export function readSettings(): Settings {
try {
ensureDataDir();
if (!fs.existsSync(SETTINGS_PATH)) {
return { ...DEFAULT_SETTINGS };
}
@@ -40,9 +65,10 @@ export function readSettings(): Settings {
}
export function writeSettings(updates: Partial<Settings>): Settings {
ensureDataDir();
const current = readSettings();
const next = { ...current, ...updates };
const dir = path.dirname(SETTINGS_PATH);
fs.mkdirSync(dir, { recursive: true });
fs.writeFileSync(SETTINGS_PATH, JSON.stringify(next, null, 2), "utf-8");
return next;
}