246 lines
9.9 KiB
TypeScript
246 lines
9.9 KiB
TypeScript
import { describe, it, expect } from 'bun:test';
|
|
import { readFileSync } from 'fs';
|
|
import { join } from 'path';
|
|
|
|
const installSourcePath = join(
|
|
__dirname,
|
|
'..',
|
|
'src',
|
|
'npx-cli',
|
|
'commands',
|
|
'install.ts',
|
|
);
|
|
const installSource = readFileSync(installSourcePath, 'utf-8');
|
|
const codexInstallerSourcePath = join(
|
|
__dirname,
|
|
'..',
|
|
'src',
|
|
'services',
|
|
'integrations',
|
|
'CodexCliInstaller.ts',
|
|
);
|
|
const codexInstallerSource = readFileSync(codexInstallerSourcePath, 'utf-8');
|
|
const syncMarketplaceSourcePath = join(
|
|
__dirname,
|
|
'..',
|
|
'scripts',
|
|
'sync-marketplace.cjs',
|
|
);
|
|
const syncMarketplaceSource = readFileSync(syncMarketplaceSourcePath, 'utf-8');
|
|
|
|
describe('Install Non-TTY Support', () => {
|
|
describe('isInteractive flag', () => {
|
|
it('defines isInteractive based on process.stdin.isTTY', () => {
|
|
expect(installSource).toContain('const isInteractive = process.stdin.isTTY === true');
|
|
});
|
|
|
|
it('uses strict equality (===) not truthy check for isTTY', () => {
|
|
const match = installSource.match(/const isInteractive = process\.stdin\.isTTY === true/);
|
|
expect(match).not.toBeNull();
|
|
});
|
|
});
|
|
|
|
describe('runTasks helper', () => {
|
|
it('defines a runTasks function', () => {
|
|
expect(installSource).toContain('async function runTasks');
|
|
});
|
|
|
|
it('has interactive branch using p.tasks', () => {
|
|
expect(installSource).toContain('await p.tasks(tasks)');
|
|
});
|
|
|
|
it('has non-interactive fallback using console.log', () => {
|
|
expect(installSource).toContain('console.log(` ${msg}`)');
|
|
});
|
|
|
|
it('branches on isInteractive', () => {
|
|
expect(installSource).toContain('if (isInteractive)');
|
|
});
|
|
});
|
|
|
|
describe('log wrapper', () => {
|
|
it('defines log.info that falls back to console.log', () => {
|
|
expect(installSource).toContain('info: (msg: string) =>');
|
|
expect(installSource).toMatch(/info:.*console\.log/);
|
|
});
|
|
|
|
it('defines log.success that falls back to console.log', () => {
|
|
expect(installSource).toContain('success: (msg: string) =>');
|
|
expect(installSource).toMatch(/success:.*console\.log/);
|
|
});
|
|
|
|
it('defines log.warn that falls back to console.warn', () => {
|
|
expect(installSource).toContain('warn: (msg: string) =>');
|
|
expect(installSource).toMatch(/warn:.*console\.warn/);
|
|
});
|
|
|
|
it('defines log.error that falls back to console.error', () => {
|
|
expect(installSource).toContain('error: (msg: string) =>');
|
|
expect(installSource).toMatch(/error:.*console\.error/);
|
|
});
|
|
});
|
|
|
|
describe('non-interactive install path', () => {
|
|
it('defaults to claude-code when not interactive and no IDE specified', () => {
|
|
expect(installSource).toContain("selectedIDEs = ['claude-code']");
|
|
});
|
|
|
|
it('uses console.log for intro in non-interactive mode', () => {
|
|
expect(installSource).toContain("console.log('claude-mem install')");
|
|
});
|
|
|
|
it('uses console.log for note/summary in non-interactive mode', () => {
|
|
expect(installSource).toContain("console.log(`\\n ${installStatus}`)");
|
|
});
|
|
|
|
it('copies Codex marketplace metadata to the durable marketplace directory', () => {
|
|
const copyRegion = installSource.slice(
|
|
installSource.indexOf('const allowedTopLevelEntries = ['),
|
|
installSource.indexOf('function copyPluginToCache'),
|
|
);
|
|
expect(copyRegion).toContain("'.agents'");
|
|
expect(copyRegion).toContain("'.codex-plugin'");
|
|
expect(copyRegion).toContain("'.mcp.json'");
|
|
});
|
|
|
|
it('validates the bundled plugin as the Codex marketplace source', () => {
|
|
expect(codexInstallerSource).toContain("path.join('plugin', '.codex-plugin', 'plugin.json')");
|
|
expect(codexInstallerSource).toContain("path.join('plugin', '.mcp.json')");
|
|
expect(codexInstallerSource).toContain("path.join('plugin', 'hooks', 'codex-hooks.json')");
|
|
expect(codexInstallerSource).toContain("path.join('plugin', 'skills', 'mem-search', 'SKILL.md')");
|
|
});
|
|
|
|
it('does not exclude MCP manifests during local marketplace sync', () => {
|
|
const gitignoreExcludeRegion = syncMarketplaceSource.slice(
|
|
syncMarketplaceSource.indexOf('function getGitignoreExcludes'),
|
|
syncMarketplaceSource.indexOf('const branch = getCurrentBranch'),
|
|
);
|
|
expect(gitignoreExcludeRegion).toContain("'.mcp.json'");
|
|
expect(gitignoreExcludeRegion).toContain('syncManagedFiles.has(line)');
|
|
});
|
|
|
|
it('registers Codex against the durable marketplace directory', () => {
|
|
expect(installSource).toContain('installCodexCli(marketplaceDirectory())');
|
|
});
|
|
|
|
it('refreshes Codex marketplace cache after registration', () => {
|
|
const installRegion = codexInstallerSource.slice(
|
|
codexInstallerSource.indexOf('export async function installCodexCli'),
|
|
codexInstallerSource.indexOf('export function uninstallCodexCli'),
|
|
);
|
|
expect(installRegion).toContain("['plugin', 'marketplace', 'upgrade', MARKETPLACE_NAME]");
|
|
expect(installRegion).toContain('installed plugin cache');
|
|
});
|
|
|
|
it('enables Codex plugin hooks during install', () => {
|
|
const installRegion = codexInstallerSource.slice(
|
|
codexInstallerSource.indexOf('export async function installCodexCli'),
|
|
codexInstallerSource.indexOf('export function uninstallCodexCli'),
|
|
);
|
|
expect(installRegion).toContain("['features', 'enable', 'plugin_hooks']");
|
|
expect(installRegion).toContain('codex features enable plugin_hooks');
|
|
});
|
|
|
|
it('captures Codex CLI output for install failure reporting', () => {
|
|
const runCodexRegion = codexInstallerSource.slice(
|
|
codexInstallerSource.indexOf('function runCodex'),
|
|
codexInstallerSource.indexOf('function removeCodexAgentsMdContext'),
|
|
);
|
|
expect(runCodexRegion).toContain('spawnSync');
|
|
expect(runCodexRegion).not.toContain("stdio: 'inherit'");
|
|
});
|
|
|
|
it('checks Codex CLI marketplace version before registration', () => {
|
|
const installRegion = codexInstallerSource.slice(
|
|
codexInstallerSource.indexOf('export async function installCodexCli'),
|
|
codexInstallerSource.indexOf('export function uninstallCodexCli'),
|
|
);
|
|
expect(codexInstallerSource).toContain("const MIN_CODEX_MARKETPLACE_VERSION = '0.128.0'");
|
|
expect(codexInstallerSource).toContain("spawnSync('codex', ['--version']");
|
|
expect(installRegion.indexOf('assertCodexMarketplaceSupported()'))
|
|
.toBeLessThan(installRegion.indexOf("runCodex(['plugin', 'marketplace', 'add', marketplaceRoot])"));
|
|
});
|
|
|
|
it('removes legacy Codex AGENTS context only after marketplace registration succeeds', () => {
|
|
const installRegion = codexInstallerSource.slice(
|
|
codexInstallerSource.indexOf('export async function installCodexCli'),
|
|
codexInstallerSource.indexOf('export function uninstallCodexCli'),
|
|
);
|
|
expect(installRegion.indexOf("runCodex(['plugin', 'marketplace', 'add', marketplaceRoot])"))
|
|
.toBeLessThan(installRegion.indexOf('cleanupLegacyCodexAgentsMdContext()'));
|
|
});
|
|
|
|
it('reports legacy Codex AGENTS cleanup failures to callers', () => {
|
|
expect(codexInstallerSource).toContain('function removeCodexAgentsMdContext(): boolean');
|
|
expect(codexInstallerSource).toContain('if (!cleanupLegacyCodexAgentsMdContext())');
|
|
});
|
|
|
|
it('does not fail Codex install after marketplace registration when only AGENTS cleanup fails', () => {
|
|
const installRegion = codexInstallerSource.slice(
|
|
codexInstallerSource.indexOf('export async function installCodexCli'),
|
|
codexInstallerSource.indexOf('export function uninstallCodexCli'),
|
|
);
|
|
const cleanupFailureRegion = installRegion.slice(
|
|
installRegion.indexOf('if (!cleanupLegacyCodexAgentsMdContext())'),
|
|
installRegion.indexOf('Installation complete!'),
|
|
);
|
|
expect(cleanupFailureRegion).toContain('console.warn');
|
|
expect(cleanupFailureRegion).not.toContain('return 1');
|
|
});
|
|
});
|
|
|
|
describe('TaskDescriptor interface', () => {
|
|
it('defines a task interface with title and task function', () => {
|
|
expect(installSource).toContain('interface TaskDescriptor');
|
|
expect(installSource).toContain('title: string');
|
|
expect(installSource).toContain('task: (message: (msg: string) => void) => Promise<string>');
|
|
});
|
|
});
|
|
|
|
describe('InstallOptions interface', () => {
|
|
it('exports InstallOptions with optional ide field', () => {
|
|
expect(installSource).toContain('export interface InstallOptions');
|
|
expect(installSource).toContain('ide?: string');
|
|
});
|
|
});
|
|
|
|
describe('post-install Next Steps copy', () => {
|
|
it('frames the choice as two paths', () => {
|
|
expect(installSource).toContain('Two paths from here:');
|
|
});
|
|
|
|
it('sets timing honesty about second-session memory injection', () => {
|
|
expect(installSource).toContain('Memory injection starts on your second session in a project.');
|
|
});
|
|
|
|
it('addresses privacy: everything stays local', () => {
|
|
expect(installSource).toContain('Everything stays in ');
|
|
expect(installSource).toContain("pc.cyan('~/.claude-mem')");
|
|
});
|
|
|
|
it('keeps /learn-codebase as the optional front-load path', () => {
|
|
expect(installSource).toContain('/learn-codebase');
|
|
});
|
|
|
|
it('demotes the uninstall caveat into a dim footer', () => {
|
|
expect(installSource).toContain('close all Claude Code sessions before uninstalling');
|
|
});
|
|
|
|
it('does not advertise /mem-search in the post-install Next Steps', () => {
|
|
const nextStepsRegion = installSource.slice(
|
|
installSource.indexOf('const nextSteps = '),
|
|
installSource.indexOf("p.note(nextSteps.join"),
|
|
);
|
|
expect(nextStepsRegion).not.toContain('/mem-search');
|
|
});
|
|
|
|
it('does not advertise /knowledge-agent in the post-install Next Steps', () => {
|
|
const nextStepsRegion = installSource.slice(
|
|
installSource.indexOf('const nextSteps = '),
|
|
installSource.indexOf("p.note(nextSteps.join"),
|
|
);
|
|
expect(nextStepsRegion).not.toContain('/knowledge-agent');
|
|
});
|
|
});
|
|
});
|