- Fix unstable-agent-babysitter: add promptAsync to test mock - Fix claude-code-mcp-loader: isolate tests from user home configs - Fix npm-dist-tags: encode packageName for scoped packages - Fix agent-builder: clone source to prevent shared object mutation - Fix add-plugin-to-opencode-config: handle JSONC with leading comments - Fix auth-plugins/add-provider-config: error on parse failures - Fix bun-install: clear timeout on completion - Fix git-diff-stats: include untracked files in diff summary
233 lines
6.1 KiB
TypeScript
233 lines
6.1 KiB
TypeScript
import { describe, it, expect, beforeEach, afterEach, mock } from "bun:test"
|
|
import { mkdirSync, writeFileSync, rmSync } from "fs"
|
|
import { join } from "path"
|
|
import { tmpdir } from "os"
|
|
|
|
const TEST_DIR = join(tmpdir(), "mcp-loader-test-" + Date.now())
|
|
const TEST_HOME = join(TEST_DIR, "home")
|
|
|
|
describe("getSystemMcpServerNames", () => {
|
|
beforeEach(() => {
|
|
mkdirSync(TEST_DIR, { recursive: true })
|
|
mkdirSync(TEST_HOME, { recursive: true })
|
|
mock.module("os", () => ({
|
|
homedir: () => TEST_HOME,
|
|
tmpdir,
|
|
}))
|
|
mock.module("../../shared", () => ({
|
|
getClaudeConfigDir: () => join(TEST_HOME, ".claude"),
|
|
}))
|
|
})
|
|
|
|
afterEach(() => {
|
|
rmSync(TEST_DIR, { recursive: true, force: true })
|
|
})
|
|
|
|
it("returns empty set when no .mcp.json files exist", async () => {
|
|
// given
|
|
const originalCwd = process.cwd()
|
|
process.chdir(TEST_DIR)
|
|
|
|
try {
|
|
// when
|
|
const { getSystemMcpServerNames } = await import("./loader")
|
|
const names = getSystemMcpServerNames()
|
|
|
|
// then
|
|
expect(names).toBeInstanceOf(Set)
|
|
expect(names.size).toBe(0)
|
|
} finally {
|
|
process.chdir(originalCwd)
|
|
}
|
|
})
|
|
|
|
it("returns server names from project .mcp.json", async () => {
|
|
// given
|
|
const mcpConfig = {
|
|
mcpServers: {
|
|
playwright: {
|
|
command: "npx",
|
|
args: ["@playwright/mcp@latest"],
|
|
},
|
|
sqlite: {
|
|
command: "uvx",
|
|
args: ["mcp-server-sqlite"],
|
|
},
|
|
},
|
|
}
|
|
writeFileSync(join(TEST_DIR, ".mcp.json"), JSON.stringify(mcpConfig))
|
|
|
|
const originalCwd = process.cwd()
|
|
process.chdir(TEST_DIR)
|
|
|
|
try {
|
|
// when
|
|
const { getSystemMcpServerNames } = await import("./loader")
|
|
const names = getSystemMcpServerNames()
|
|
|
|
// then
|
|
expect(names.has("playwright")).toBe(true)
|
|
expect(names.has("sqlite")).toBe(true)
|
|
expect(names.size).toBe(2)
|
|
} finally {
|
|
process.chdir(originalCwd)
|
|
}
|
|
})
|
|
|
|
it("returns server names from .claude/.mcp.json", async () => {
|
|
// given
|
|
mkdirSync(join(TEST_DIR, ".claude"), { recursive: true })
|
|
const mcpConfig = {
|
|
mcpServers: {
|
|
memory: {
|
|
command: "npx",
|
|
args: ["-y", "@anthropic-ai/mcp-server-memory"],
|
|
},
|
|
},
|
|
}
|
|
writeFileSync(join(TEST_DIR, ".claude", ".mcp.json"), JSON.stringify(mcpConfig))
|
|
|
|
const originalCwd = process.cwd()
|
|
process.chdir(TEST_DIR)
|
|
|
|
try {
|
|
// when
|
|
const { getSystemMcpServerNames } = await import("./loader")
|
|
const names = getSystemMcpServerNames()
|
|
|
|
// then
|
|
expect(names.has("memory")).toBe(true)
|
|
} finally {
|
|
process.chdir(originalCwd)
|
|
}
|
|
})
|
|
|
|
it("excludes disabled MCP servers", async () => {
|
|
// given
|
|
const mcpConfig = {
|
|
mcpServers: {
|
|
playwright: {
|
|
command: "npx",
|
|
args: ["@playwright/mcp@latest"],
|
|
disabled: true,
|
|
},
|
|
active: {
|
|
command: "npx",
|
|
args: ["some-mcp"],
|
|
},
|
|
},
|
|
}
|
|
writeFileSync(join(TEST_DIR, ".mcp.json"), JSON.stringify(mcpConfig))
|
|
|
|
const originalCwd = process.cwd()
|
|
process.chdir(TEST_DIR)
|
|
|
|
try {
|
|
// when
|
|
const { getSystemMcpServerNames } = await import("./loader")
|
|
const names = getSystemMcpServerNames()
|
|
|
|
// then
|
|
expect(names.has("playwright")).toBe(false)
|
|
expect(names.has("active")).toBe(true)
|
|
} finally {
|
|
process.chdir(originalCwd)
|
|
}
|
|
})
|
|
|
|
it("merges server names from multiple .mcp.json files", async () => {
|
|
// given
|
|
mkdirSync(join(TEST_DIR, ".claude"), { recursive: true })
|
|
|
|
const projectMcp = {
|
|
mcpServers: {
|
|
playwright: { command: "npx", args: ["@playwright/mcp@latest"] },
|
|
},
|
|
}
|
|
const localMcp = {
|
|
mcpServers: {
|
|
memory: { command: "npx", args: ["-y", "@anthropic-ai/mcp-server-memory"] },
|
|
},
|
|
}
|
|
|
|
writeFileSync(join(TEST_DIR, ".mcp.json"), JSON.stringify(projectMcp))
|
|
writeFileSync(join(TEST_DIR, ".claude", ".mcp.json"), JSON.stringify(localMcp))
|
|
|
|
const originalCwd = process.cwd()
|
|
process.chdir(TEST_DIR)
|
|
|
|
try {
|
|
// when
|
|
const { getSystemMcpServerNames } = await import("./loader")
|
|
const names = getSystemMcpServerNames()
|
|
|
|
// then
|
|
expect(names.has("playwright")).toBe(true)
|
|
expect(names.has("memory")).toBe(true)
|
|
} finally {
|
|
process.chdir(originalCwd)
|
|
}
|
|
})
|
|
|
|
it("reads user-level MCP config from ~/.claude.json", async () => {
|
|
// given
|
|
const userConfigPath = join(TEST_HOME, ".claude.json")
|
|
const userMcpConfig = {
|
|
mcpServers: {
|
|
"user-server": {
|
|
command: "npx",
|
|
args: ["user-mcp-server"],
|
|
},
|
|
},
|
|
}
|
|
writeFileSync(userConfigPath, JSON.stringify(userMcpConfig))
|
|
|
|
const originalCwd = process.cwd()
|
|
process.chdir(TEST_DIR)
|
|
|
|
try {
|
|
// when
|
|
const { getSystemMcpServerNames } = await import("./loader")
|
|
const names = getSystemMcpServerNames()
|
|
|
|
// then
|
|
expect(names.has("user-server")).toBe(true)
|
|
} finally {
|
|
process.chdir(originalCwd)
|
|
}
|
|
})
|
|
|
|
it("reads both ~/.claude.json and ~/.claude/.mcp.json for user scope", async () => {
|
|
// given
|
|
const claudeDir = join(TEST_HOME, ".claude")
|
|
mkdirSync(claudeDir, { recursive: true })
|
|
|
|
writeFileSync(join(TEST_HOME, ".claude.json"), JSON.stringify({
|
|
mcpServers: {
|
|
"server-from-claude-json": { command: "npx", args: ["server-a"] },
|
|
},
|
|
}))
|
|
|
|
writeFileSync(join(claudeDir, ".mcp.json"), JSON.stringify({
|
|
mcpServers: {
|
|
"server-from-mcp-json": { command: "npx", args: ["server-b"] },
|
|
},
|
|
}))
|
|
|
|
const originalCwd = process.cwd()
|
|
process.chdir(TEST_DIR)
|
|
|
|
try {
|
|
// when
|
|
const { getSystemMcpServerNames } = await import("./loader")
|
|
const names = getSystemMcpServerNames()
|
|
|
|
// then
|
|
expect(names.has("server-from-claude-json")).toBe(true)
|
|
expect(names.has("server-from-mcp-json")).toBe(true)
|
|
} finally {
|
|
process.chdir(originalCwd)
|
|
}
|
|
})
|
|
})
|