@evomap/evolver's validator sandbox allowlist permits `npm`/`npx`, yielding RCE from Hub-delivered validation tasks via lifecycle scripts
Summary
The validator-mode sandbox executor (src/gep/validator/sandboxExecutor.js) places npm and npx in its hard executable allowlist. Because npm install and npx -y -p execute arbitrary code by design (preinstall/install/postinstall lifecycle scripts and remote-package bin entries), and because validator nodes consume validation_commands strings from unsigned Hub responses with no per-response signature check, an attacker who controls or MITMs the Hub achieves automatic remote code execution on every validator node within one daemon poll (default 60s).
Details
End-to-end chain:
-
src/gep/validator/index.js:71-87—fetchValidationTasks()POSTs to/a2a/fetchand readsvalidation_tasksfrom the JSON response. The outbound request is signed viabuildHubHeaders(), but the Hub's response is parsed directly withawait res.json()and no signature is verified ondata.payload. -
src/gep/validator/index.js:98-108—validateOneTask()extractstask.validation_commands(an array of attacker-controlled strings) and passes it straight torunInSandbox(commands, {}). No call topolicyCheck.isValidationCommandAllowed()happens on this path. The author's own comment atsandboxExecutor.js:41-42acknowledges this gap: "This closes the gap where validation_commands go straight from Hub to runInSandbox without passing through policyCheck.isValidationCommandAllowed()." -
src/gep/validator/sandboxExecutor.js:172-218—runSingleCommandcallsparseCommand(cmd), then checksALLOWED_EXECUTABLES.has(parsed.executable):// sandboxExecutor.js:35 const ALLOWED_EXECUTABLES = new Set(['node', 'npm', 'npx']);parseCommandonly rejects shell metacharacters (| & ; > < \$) and unbalanced quotes. A string likenpm install /tmp/evil-pkg --no-audit --no-fundcontains none of those and parses cleanly into{ executable: 'npm', args: [...] }`. -
sandboxExecutor.js:54-66—assertNodeCommandSafeis a no-op for non-nodeexecutables:function assertNodeCommandSafe(parsed) { if (parsed.executable !== 'node') return; // npm/npx skip every check ... }The
BLOCKED_NODE_FLAGSset (-e,-r,--loader, etc.) therefore never gatesnpmornpxinvocations. -
sandboxExecutor.js:213—spawn('npm', [...], { shell: false, cwd: sandboxDir, env })runsnpm. npm's documented behavior is to execute the package'spreinstall,install, andpostinstallscripts;npxdownloads a remote package and executes itsbinentry. Both yield arbitrary code execution in the validator process's UID/permissions. -
src/gep/validator/index.js:189— the validator daemon polls every 60s by default (EVOLVER_VALIDATOR_DAEMON_INTERVAL_MS), and validator mode is on by default since v1.69.0 (isValidatorEnabled()returnstrueunless explicitly disabled,index.js:25-34).
The "sandbox" is nominal: it sets a fresh cwd and a stripped env (HOME → tmpdir to hide ~/.npmrc/~/.ssh), but PATH is preserved (so npm/npx resolve), there is no container/chroot/seccomp/uid drop, and nothing prevents the spawned process from writing arbitrary files, opening outbound connections, or reading any file readable by the validator process.
The author's documented threat model at sandboxExecutor.js:31-34 explicitly includes Hub compromise:
"Any command whose first token is not in this set is rejected before spawn(). This prevents command injection via Hub-delivered task.command strings even if Hub itself is compromised or mis-signs a task."
Putting npm and npx on that allowlist defeats that stated goal — both are arbitrary-code-execution-by-design tools.
PoC
Reproduced against v1.70.0-beta.4 (HEAD on main):
Step 1 — plant a malicious package locally (the remote-tarball variant works identically; npm fetches and runs lifecycle scripts in both cases):
mkdir -p /tmp/evil-pkg-validator
cat > /tmp/evil-pkg-validator/package.json !a.startsWith('-'));
const FORBIDDEN = new Set(['install', 'i', 'add', 'ci', 'exec', 'x', 'run', 'run-script', 'rebuild', 'pack', 'publish']);
if (FORBIDDEN.has(sub)) {
throw new Error('npm/npx subcommand not allowed in sandbox: ' + sub);
}
// Require --ignore-scripts on every npm invocation as defense-in-depth.
if (parsed.executable === 'npm' && !parsed.args.includes('--ignore-scripts')) {
throw new Error('npm in sandbox requires --ignore-scripts');
}
// npx always fetches+executes — disallow entirely.
if (parsed.executable === 'npx') {
throw new Error('npx is not allowed in sandbox');
}
}
Additionally:
- Sign the Hub's
/a2a/fetchresponse the same way outbound requests are signed (buildHubHeaders). Verify the signature ondata.payloadinfetchValidationTasksbefore handing tasks torunInSandbox. This closes the network-MITM variant that does not require Hub compromise. - Run
runInSandboxunder real isolation — drop privileges, disable network, mount tmpfs, apply seccomp — rather than relying solely on an allowlist. The currentbuildSandboxEnvonly redirectsHOME/TMPDIR; the spawned process otherwise has full host access. - Apply
policyCheck.isValidationCommandAllowed()to Hub-deliveredvalidation_commandsinvalidateOneTask, mirroring the gate that already exists for capsule-derived commands insolidify.js/skill2gep.js.