Files
bun.sh/test/cli/run/require-cache.test.ts
2024-12-16 17:04:33 -08:00

309 lines
9.9 KiB
TypeScript

import { describe, expect, test } from "bun:test";
import { bunEnv, bunExe, isBroken, isIntelMacOS, isWindows, tempDirWithFiles } from "harness";
import { join } from "path";
test("require.cache is not an empty object literal when inspected", () => {
const inspected = Bun.inspect(require.cache);
expect(inspected).not.toBe("{}");
expect(inspected).toContain("Module {");
});
// This also tests __dirname and __filename
test("require.cache", () => {
const { stdout, exitCode } = Bun.spawnSync({
cmd: [bunExe(), "run", join(import.meta.dir, "require-cache-fixture.cjs")],
env: bunEnv,
stderr: "inherit",
});
expect(stdout.toString().trim()).toEndWith("--pass--");
expect(exitCode).toBe(0);
});
// https://github.com/oven-sh/bun/issues/5188
test("require.cache does not include unevaluated modules", () => {
const { stdout, exitCode } = Bun.spawnSync({
cmd: [bunExe(), "run", join(import.meta.dir, "require-cache-bug-5188.js")],
env: bunEnv,
stderr: "inherit",
});
expect(stdout.toString().trim()).toEndWith("--pass--");
expect(exitCode).toBe(0);
});
describe.skipIf(isBroken && isIntelMacOS)("files transpiled and loaded don't leak the output source code", () => {
test("via require() with a lot of long export names", () => {
let text = "";
for (let i = 0; i < 10000; i++) {
text += `exports.superDuperExtraCrazyLongNameWowSuchNameLongYouveNeverSeenANameThisLongForACommonJSModuleExport${i} = 1;\n`;
}
console.log("Text length:", text.length);
const dir = tempDirWithFiles("require-cache-bug-leak-1", {
"index.js": text,
"require-cache-bug-leak-fixture.js": `
const path = require.resolve("./index.js");
const gc = global.gc || globalThis?.Bun?.gc || (() => {});
function bust() {
const mod = require.cache[path];
if (mod) {
mod.parent = null;
mod.children = [];
delete require.cache[path];
}
}
for (let i = 0; i < 50; i++) {
require(path);
bust();
}
gc(true);
const baseline = process.memoryUsage.rss();
for (let i = 0; i < 500; i++) {
require(path);
bust(path);
console.log("RSS", (process.memoryUsage.rss() / 1024 / 1024) | 0, "MB");
}
gc(true);
const rss = process.memoryUsage.rss();
const diff = rss - baseline;
console.log("RSS diff", (diff / 1024 / 1024) | 0, "MB");
console.log("RSS", (diff / 1024 / 1024) | 0, "MB");
if (diff > 100 * 1024 * 1024) {
// Bun v1.1.21 reported 844 MB here on macoS arm64.
throw new Error("Memory leak detected");
}
exports.abc = 123;
`,
});
console.log({ dir });
const { exitCode, resourceUsage } = Bun.spawnSync({
cmd: [bunExe(), "run", "--smol", join(dir, "require-cache-bug-leak-fixture.js")],
env: bunEnv,
stdio: ["inherit", "inherit", "inherit"],
});
console.log(resourceUsage);
expect(exitCode).toBe(0);
}, 60000);
test("via await import() with a lot of function calls", () => {
let text = "function i() { return 1; }\n";
for (let i = 0; i < 20000; i++) {
text += `i();\n`;
}
text += "exports.forceCommonJS = true;\n";
console.log("Text length:", text.length);
const dir = tempDirWithFiles("require-cache-bug-leak-3", {
"index.js": text,
"require-cache-bug-leak-fixture.js": `
const path = require.resolve("./index.js");
const gc = global.gc || globalThis?.Bun?.gc || (() => {});
function bust() {
delete require.cache[path];
}
for (let i = 0; i < 100; i++) {
await import(path);
bust();
}
gc(true);
const baseline = process.memoryUsage.rss();
for (let i = 0; i < 400; i++) {
await import(path);
bust(path);
console.log("RSS", (process.memoryUsage.rss() / 1024 / 1024) | 0, "MB");
}
gc(true);
const rss = process.memoryUsage.rss();
const diff = rss - baseline;
console.log("RSS diff", (diff / 1024 / 1024) | 0, "MB");
console.log("RSS", (diff / 1024 / 1024) | 0, "MB");
if (diff > 64 * 1024 * 1024) {
// Bun v1.1.22 reported 1 MB here on macoS arm64.
// Bun v1.1.21 reported 257 MB here on macoS arm64.
throw new Error("Memory leak detected");
}
export default 123;
`,
});
const { exitCode, resourceUsage } = Bun.spawnSync({
cmd: [bunExe(), "run", "--smol", join(dir, "require-cache-bug-leak-fixture.js")],
env: bunEnv,
stdio: ["inherit", "inherit", "inherit"],
});
console.log(resourceUsage);
expect(exitCode).toBe(0);
}, 60000); // takes 4s on an M1 in release build
test("via import() with a lot of long export names", () => {
let text = "";
for (let i = 0; i < 10000; i++) {
text += `export const superDuperExtraCrazyLongNameWowSuchNameLongYouveNeverSeenANameThisLongForACommonJSModuleExport${i} = 1;\n`;
}
const dir = tempDirWithFiles("require-cache-bug-leak-4", {
"index.js": text,
"require-cache-bug-leak-fixture.js": `
const path = require.resolve("./index.js");
const gc = global.gc || globalThis?.Bun?.gc || (() => {});
function bust() {
delete require.cache[path];
}
for (let i = 0; i < 50; i++) {
await import(path);
bust();
}
gc(true);
const baseline = process.memoryUsage.rss();
for (let i = 0; i < 250; i++) {
await import(path);
bust(path);
console.log("RSS", (process.memoryUsage.rss() / 1024 / 1024) | 0, "MB");
}
gc(true);
const rss = process.memoryUsage.rss();
const diff = rss - baseline;
console.log("RSS diff", (diff / 1024 / 1024) | 0, "MB");
console.log("RSS", (diff / 1024 / 1024) | 0, "MB");
if (diff > 64 * 1024 * 1024) {
// Bun v1.1.21 reported 423 MB here on macoS arm64.
// Bun v1.1.22 reported 4 MB here on macoS arm64.
throw new Error("Memory leak detected");
}
export default 124;
`,
});
console.log({ dir });
const { exitCode, resourceUsage } = Bun.spawnSync({
cmd: [bunExe(), "run", "--smol", join(dir, "require-cache-bug-leak-fixture.js")],
env: bunEnv,
stdio: ["inherit", "inherit", "inherit"],
});
console.log(resourceUsage);
expect(exitCode).toBe(0);
}, 60000);
test("via require() with a lot of function calls", () => {
let text = "function i() { return 1; }\n";
for (let i = 0; i < 20000; i++) {
text += `i();\n`;
}
text += "exports.forceCommonJS = true;\n";
console.log("Text length:", text.length);
const dir = tempDirWithFiles("require-cache-bug-leak-2", {
"index.js": text,
"require-cache-bug-leak-fixture.js": `
const path = require.resolve("./index.js");
const gc = global.gc || globalThis?.Bun?.gc || (() => {});
function bust() {
const mod = require.cache[path];
if (mod) {
mod.parent = null;
mod.children = [];
delete require.cache[path];
}
}
for (let i = 0; i < 100; i++) {
require(path);
bust();
}
gc(true);
const baseline = process.memoryUsage.rss();
for (let i = 0; i < 400; i++) {
require(path);
bust(path);
console.log("RSS", (process.memoryUsage.rss() / 1024 / 1024) | 0, "MB");
}
gc(true);
const rss = process.memoryUsage.rss();
const diff = rss - baseline;
console.log("RSS diff", (diff / 1024 / 1024) | 0, "MB");
console.log("RSS", (diff / 1024 / 1024) | 0, "MB");
if (diff > 64 * 1024 * 1024) {
// Bun v1.1.22 reported 4 MB here on macoS arm64.
// Bun v1.1.21 reported 248 MB here on macoS arm64.
throw new Error("Memory leak detected");
}
exports.abc = 123;
`,
});
const { exitCode, resourceUsage } = Bun.spawnSync({
cmd: [bunExe(), "run", "--smol", join(dir, "require-cache-bug-leak-fixture.js")],
env: bunEnv,
stdio: ["inherit", "inherit", "inherit"],
});
console.log(resourceUsage);
expect(exitCode).toBe(0);
}, 60000); // takes 4s on an M1 in release build
});
describe("files transpiled and loaded don't leak the AST", () => {
test("via require()", () => {
const { stdout, exitCode } = Bun.spawnSync({
cmd: [bunExe(), "run", join(import.meta.dir, "require-cache-bug-leak-fixture.js")],
env: bunEnv,
stderr: "inherit",
});
expect(stdout.toString().trim()).toEndWith("--pass--");
expect(exitCode).toBe(0);
}, 20000);
test("via import()", () => {
const { stdout, exitCode } = Bun.spawnSync({
cmd: [bunExe(), "run", join(import.meta.dir, "esm-bug-leak-fixture.mjs")],
env: bunEnv,
stderr: "inherit",
});
expect(stdout.toString().trim()).toEndWith("--pass--");
expect(exitCode).toBe(0);
}, 20000);
});
// These tests are extra slow in debug builds
describe("files transpiled and loaded don't leak file paths", () => {
test("via require()", () => {
const { stdout, exitCode } = Bun.spawnSync({
cmd: [bunExe(), "--smol", "run", join(import.meta.dir, "cjs-fixture-leak-small.js")],
env: bunEnv,
stderr: "inherit",
});
expect(stdout.toString().trim()).toEndWith("--pass--");
expect(exitCode).toBe(0);
}, 30000);
test(
"via import()",
() => {
const { stdout, exitCode } = Bun.spawnSync({
cmd: [bunExe(), "--smol", "run", join(import.meta.dir, "esm-fixture-leak-small.mjs")],
env: bunEnv,
stderr: "inherit",
});
expect(stdout.toString().trim()).toEndWith("--pass--");
expect(exitCode).toBe(0);
},
// TODO: Investigate why this is so slow on Windows
isWindows ? 60000 : 30000,
);
});