Files
claude-mem/tests/utils/project-name.test.ts
T
Alex Newman 040729beef fix(project-name): use parent/worktree composite so observations don't cross worktrees
Revert of #1820 behavior. Each worktree now gets its own bucket:
- In a worktree, primary = `parent/worktree` (e.g. `claude-mem/dar-es-salaam`)
- In a main repo, primary = basename (unchanged)
- allProjects is always `[primary]` — strict isolation at query time

Includes a one-off maintenance script (scripts/worktree-remap.ts) that
retroactively reattributes past sessions to their worktree using path
signals in observations and user prompts. Two-rule inference keeps the
remap high-confidence:
  1. The worktree basename in the path matches the session's current
     plain project name (pre-#1820 era; trusted).
  2. Or all worktree path signals converge on a single (parent, worktree)
     across the session.
Ambiguous sessions are skipped.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-04-16 15:40:44 -07:00

144 lines
4.6 KiB
TypeScript

/**
* Project Name Tests
*
* Tests tilde expansion and project name extraction.
* Source: src/utils/project-name.ts
*/
import { describe, it, expect, beforeAll, afterAll } from 'bun:test';
import { homedir } from 'os';
import { getProjectName, getProjectContext } from '../../src/utils/project-name.js';
describe('getProjectName', () => {
describe('tilde expansion', () => {
it('resolves bare ~ to home directory basename', () => {
const home = homedir();
const expected = home.split('/').pop() || home.split('\\').pop() || '';
expect(getProjectName('~')).toBe(expected);
});
it('resolves ~/subpath to subpath', () => {
expect(getProjectName('~/projects/my-app')).toBe('my-app');
});
it('resolves ~/ to home directory basename', () => {
const home = homedir();
const expected = home.split('/').pop() || home.split('\\').pop() || '';
expect(getProjectName('~/')).toBe(expected);
});
});
describe('normal paths', () => {
it('extracts basename from absolute path', () => {
expect(getProjectName('/home/user/my-project')).toBe('my-project');
});
it('extracts basename from nested path', () => {
expect(getProjectName('/Users/test/work/deep/nested/project')).toBe('project');
});
it('handles trailing slash', () => {
expect(getProjectName('/home/user/my-project/')).toBe('my-project');
});
});
describe('edge cases', () => {
it('returns unknown-project for null', () => {
expect(getProjectName(null)).toBe('unknown-project');
});
it('returns unknown-project for undefined', () => {
expect(getProjectName(undefined)).toBe('unknown-project');
});
it('returns unknown-project for empty string', () => {
expect(getProjectName('')).toBe('unknown-project');
});
it('returns unknown-project for whitespace', () => {
expect(getProjectName(' ')).toBe('unknown-project');
});
});
describe('realistic scenarios from #1478', () => {
it('handles ~ the same as full home path', () => {
const home = homedir();
expect(getProjectName('~')).toBe(getProjectName(home));
});
it('handles ~/projects/app the same as /full/path/projects/app', () => {
const home = homedir();
expect(getProjectName('~/projects/app')).toBe(
getProjectName(`${home}/projects/app`)
);
});
});
});
describe('getProjectContext', () => {
it('returns primary project name for normal path', () => {
const ctx = getProjectContext('/home/user/my-project');
expect(ctx.primary).toBe('my-project');
expect(ctx.parent).toBeNull();
expect(ctx.isWorktree).toBe(false);
expect(ctx.allProjects).toEqual(['my-project']);
});
it('resolves ~ path correctly', () => {
const home = homedir();
const ctx = getProjectContext('~');
const ctxHome = getProjectContext(home);
expect(ctx.primary).toBe(ctxHome.primary);
});
it('returns unknown-project context for null', () => {
const ctx = getProjectContext(null);
expect(ctx.primary).toBe('unknown-project');
expect(ctx.parent).toBeNull();
});
describe('worktree isolation', () => {
let tmp: string;
let mainRepo: string;
let worktreeCheckout: string;
beforeAll(async () => {
const { mkdtempSync, mkdirSync, writeFileSync } = await import('fs');
const { join } = await import('path');
const { tmpdir } = await import('os');
tmp = mkdtempSync(join(tmpdir(), 'cm-wt-'));
mainRepo = join(tmp, 'main-repo');
const worktreeGitDir = join(mainRepo, '.git', 'worktrees', 'my-worktree');
worktreeCheckout = join(tmp, 'my-worktree');
mkdirSync(worktreeGitDir, { recursive: true });
mkdirSync(worktreeCheckout, { recursive: true });
writeFileSync(
join(worktreeCheckout, '.git'),
`gitdir: ${worktreeGitDir}\n`
);
});
afterAll(async () => {
const { rmSync } = await import('fs');
rmSync(tmp, { recursive: true, force: true });
});
it('uses parent/worktree composite as primary when in a worktree', () => {
const ctx = getProjectContext(worktreeCheckout);
expect(ctx.isWorktree).toBe(true);
expect(ctx.primary).toBe('main-repo/my-worktree');
expect(ctx.parent).toBe('main-repo');
expect(ctx.allProjects).toEqual(['main-repo/my-worktree']);
});
it('write-path call sites resolve to composite name in worktrees', () => {
const project = getProjectContext(worktreeCheckout).primary;
expect(project).toBe('main-repo/my-worktree');
expect(project).not.toBe('main-repo');
expect(project).not.toBe('my-worktree');
});
});
});