Files
pi/packages/coding-agent/test/restore-sandbox-env.test.ts
mdsjip 5b8deef2f9 fix(coding-agent,ai): fallback to /proc/self/environ in Bun sandbox (#3801)
Bun compiled binaries have an empty process.env when running inside
sandbox environments (e.g. nono on Linux/macOS). This broke API key
detection and model discovery because all process.env.* lookups returned
undefined.

- Add restoreSandboxEnv() helper that reads /proc/self/environ when Bun
  is detected and process.env is empty, populating process.env before
  any other code runs (coding-agent/src/bun/cli.ts entry point)
- Add getProcEnv() fallback in env-api-keys.ts for direct @mariozechner/pi-ai
  consumers that may not go through the coding-agent entry point
- Add unit tests for restoreSandboxEnv
2026-04-27 18:36:53 +02:00

78 lines
2.1 KiB
TypeScript

import { describe, expect, it, vi } from "vitest";
const readFileSync = vi.fn();
vi.mock("node:fs", () => ({
readFileSync,
}));
const { restoreSandboxEnv } = await import("../src/bun/restore-sandbox-env.js");
describe("restoreSandboxEnv", () => {
it("does nothing when not running under bun", () => {
const originalVersions = Object.getOwnPropertyDescriptor(process, "versions");
Object.defineProperty(process, "versions", {
value: { node: "20.0.0" },
});
const envBefore = { ...process.env };
restoreSandboxEnv();
expect(process.env).toEqual(envBefore);
if (originalVersions) {
Object.defineProperty(process, "versions", originalVersions);
}
});
it("does nothing when process.env already has entries", () => {
const originalVersions = Object.getOwnPropertyDescriptor(process, "versions");
Object.defineProperty(process, "versions", {
value: { bun: "1.2.0", node: "20.0.0" },
});
process.env.RESTORE_SANDBOX_ENV_TEST = "1";
const envBefore = { ...process.env };
restoreSandboxEnv();
expect(process.env).toEqual(envBefore);
delete process.env.RESTORE_SANDBOX_ENV_TEST;
if (originalVersions) {
Object.defineProperty(process, "versions", originalVersions);
}
});
it("restores environment from /proc/self/environ when bun env is empty", () => {
const originalVersions = Object.getOwnPropertyDescriptor(process, "versions");
Object.defineProperty(process, "versions", {
value: { bun: "1.2.0", node: "20.0.0" },
});
// Clear env to simulate the bun sandbox bug.
const envBackup = { ...process.env };
for (const key of Object.keys(process.env)) {
delete process.env[key];
}
readFileSync.mockReturnValue("FOO=bar\0BAZ=qux\0");
restoreSandboxEnv();
expect(readFileSync).toHaveBeenCalledWith("/proc/self/environ", "utf-8");
expect(process.env.FOO).toBe("bar");
expect(process.env.BAZ).toBe("qux");
// Restore.
for (const key of Object.keys(process.env)) {
delete process.env[key];
}
Object.assign(process.env, envBackup);
if (originalVersions) {
Object.defineProperty(process, "versions", originalVersions);
}
readFileSync.mockReset();
});
});