prjct-cli 0.59.0 → 0.60.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,443 @@
1
+ /**
2
+ * Tests for NestedContextResolver - AGENTS.md discovery and resolution
3
+ * PRJ-101: Hierarchical scope system
4
+ */
5
+
6
+ import { afterEach, beforeEach, describe, expect, test } from 'bun:test'
7
+ import fs from 'node:fs/promises'
8
+ import os from 'node:os'
9
+ import path from 'node:path'
10
+ import NestedContextResolver from '../../services/nested-context-resolver'
11
+
12
+ // Test directory setup
13
+ let testDir: string
14
+ let resolver: NestedContextResolver
15
+
16
+ beforeEach(async () => {
17
+ // Create temp directory for tests
18
+ testDir = path.join(os.tmpdir(), `prjct-test-${Date.now()}`)
19
+ await fs.mkdir(testDir, { recursive: true })
20
+ resolver = new NestedContextResolver(testDir)
21
+ })
22
+
23
+ afterEach(async () => {
24
+ // Cleanup temp directory
25
+ try {
26
+ await fs.rm(testDir, { recursive: true, force: true })
27
+ } catch {
28
+ // Ignore cleanup errors
29
+ }
30
+ })
31
+
32
+ // =============================================================================
33
+ // PRJCT.md Discovery Tests (existing functionality)
34
+ // =============================================================================
35
+
36
+ describe('NestedContextResolver - PRJCT.md', () => {
37
+ test('discovers root PRJCT.md', async () => {
38
+ await fs.writeFile(path.join(testDir, 'PRJCT.md'), '## Rules\n\n- Rule 1\n- Rule 2')
39
+
40
+ const contexts = await resolver.discoverContextFiles()
41
+
42
+ expect(contexts).toHaveLength(1)
43
+ expect(contexts[0].depth).toBe(0)
44
+ expect(contexts[0].sections).toHaveLength(1)
45
+ expect(contexts[0].sections[0].name).toBe('Rules')
46
+ })
47
+
48
+ test('parses sections with override marker', async () => {
49
+ await fs.writeFile(path.join(testDir, 'PRJCT.md'), '## Rules @override\n\n- Override rule')
50
+
51
+ const contexts = await resolver.discoverContextFiles()
52
+
53
+ expect(contexts[0].sections[0].override).toBe(true)
54
+ expect(contexts[0].sections[0].name).toBe('Rules')
55
+ })
56
+
57
+ test('handles empty project (no PRJCT.md)', async () => {
58
+ const contexts = await resolver.discoverContextFiles()
59
+ expect(contexts).toHaveLength(0)
60
+ })
61
+ })
62
+
63
+ // =============================================================================
64
+ // AGENTS.md Discovery Tests (new functionality)
65
+ // =============================================================================
66
+
67
+ describe('NestedContextResolver - AGENTS.md Discovery', () => {
68
+ test('discovers root AGENTS.md', async () => {
69
+ await fs.writeFile(
70
+ path.join(testDir, 'AGENTS.md'),
71
+ `## Backend
72
+
73
+ Handles backend development.
74
+
75
+ ### Triggers
76
+ - api
77
+ - endpoint
78
+
79
+ ### Rules
80
+ - Use async/await
81
+ `
82
+ )
83
+
84
+ const agentFiles = await resolver.discoverAgentFiles()
85
+
86
+ expect(agentFiles).toHaveLength(1)
87
+ expect(agentFiles[0].depth).toBe(0)
88
+ expect(agentFiles[0].agents).toHaveLength(1)
89
+ expect(agentFiles[0].agents[0].name).toBe('Backend')
90
+ })
91
+
92
+ test('parses multiple agents from single file', async () => {
93
+ await fs.writeFile(
94
+ path.join(testDir, 'AGENTS.md'),
95
+ `## Frontend
96
+
97
+ Frontend specialist.
98
+
99
+ ### Triggers
100
+ - component
101
+ - ui
102
+
103
+ ## Backend
104
+
105
+ Backend specialist.
106
+
107
+ ### Triggers
108
+ - api
109
+ - database
110
+ `
111
+ )
112
+
113
+ const agentFiles = await resolver.discoverAgentFiles()
114
+
115
+ expect(agentFiles[0].agents).toHaveLength(2)
116
+ expect(agentFiles[0].agents[0].name).toBe('Frontend')
117
+ expect(agentFiles[0].agents[1].name).toBe('Backend')
118
+ })
119
+
120
+ test('parses agent triggers as array', async () => {
121
+ await fs.writeFile(
122
+ path.join(testDir, 'AGENTS.md'),
123
+ `## Testing
124
+
125
+ Testing specialist.
126
+
127
+ ### Triggers
128
+ - write test
129
+ - add test
130
+ - unit test
131
+ `
132
+ )
133
+
134
+ const agentFiles = await resolver.discoverAgentFiles()
135
+ const agent = agentFiles[0].agents[0]
136
+
137
+ expect(agent.triggers).toEqual(['write test', 'add test', 'unit test'])
138
+ })
139
+
140
+ test('parses agent rules as array', async () => {
141
+ await fs.writeFile(
142
+ path.join(testDir, 'AGENTS.md'),
143
+ `## Backend
144
+
145
+ Backend specialist.
146
+
147
+ ### Rules
148
+ - Use TypeScript
149
+ - Validate inputs
150
+ - Log errors
151
+ `
152
+ )
153
+
154
+ const agentFiles = await resolver.discoverAgentFiles()
155
+ const agent = agentFiles[0].agents[0]
156
+
157
+ expect(agent.rules).toEqual(['Use TypeScript', 'Validate inputs', 'Log errors'])
158
+ })
159
+
160
+ test('parses code patterns from code blocks', async () => {
161
+ await fs.writeFile(
162
+ path.join(testDir, 'AGENTS.md'),
163
+ `## Backend
164
+
165
+ Backend specialist.
166
+
167
+ ### Patterns
168
+ \`\`\`typescript
169
+ async function handler(req: Request) {
170
+ return new Response('ok')
171
+ }
172
+ \`\`\`
173
+ `
174
+ )
175
+
176
+ const agentFiles = await resolver.discoverAgentFiles()
177
+ const agent = agentFiles[0].agents[0]
178
+
179
+ expect(agent.patterns).toHaveLength(1)
180
+ expect(agent.patterns![0]).toContain('async function handler')
181
+ })
182
+
183
+ test('detects @override marker on agent', async () => {
184
+ await fs.writeFile(
185
+ path.join(testDir, 'AGENTS.md'),
186
+ `## Frontend @override
187
+
188
+ Overrides parent frontend agent.
189
+ `
190
+ )
191
+
192
+ const agentFiles = await resolver.discoverAgentFiles()
193
+ const agent = agentFiles[0].agents[0]
194
+
195
+ expect(agent.name).toBe('Frontend')
196
+ expect(agent.override).toBe(true)
197
+ })
198
+
199
+ test('handles empty AGENTS.md', async () => {
200
+ await fs.writeFile(path.join(testDir, 'AGENTS.md'), '# AGENTS.md\n\nNo agents defined.')
201
+
202
+ const agentFiles = await resolver.discoverAgentFiles()
203
+
204
+ expect(agentFiles).toHaveLength(1)
205
+ expect(agentFiles[0].agents).toHaveLength(0)
206
+ })
207
+
208
+ test('handles missing AGENTS.md', async () => {
209
+ const agentFiles = await resolver.discoverAgentFiles()
210
+ expect(agentFiles).toHaveLength(0)
211
+ })
212
+ })
213
+
214
+ // =============================================================================
215
+ // AGENTS.md Hierarchy Tests
216
+ // =============================================================================
217
+
218
+ describe('NestedContextResolver - AGENTS.md Hierarchy', () => {
219
+ test('discovers nested AGENTS.md in subdirectories', async () => {
220
+ // Create root AGENTS.md
221
+ await fs.writeFile(
222
+ path.join(testDir, 'AGENTS.md'),
223
+ `## GlobalAgent
224
+
225
+ Global agent for all.
226
+ `
227
+ )
228
+
229
+ // Create subdirectory with AGENTS.md
230
+ const subDir = path.join(testDir, 'packages', 'web')
231
+ await fs.mkdir(subDir, { recursive: true })
232
+ await fs.writeFile(
233
+ path.join(subDir, 'AGENTS.md'),
234
+ `## WebAgent
235
+
236
+ Web-specific agent.
237
+ `
238
+ )
239
+
240
+ const agentFiles = await resolver.discoverAgentFiles()
241
+
242
+ expect(agentFiles).toHaveLength(2)
243
+ expect(agentFiles.find((af) => af.depth === 0)?.agents[0].name).toBe('GlobalAgent')
244
+ })
245
+
246
+ test('builds parent-child relationships', async () => {
247
+ // Create root AGENTS.md
248
+ await fs.writeFile(path.join(testDir, 'AGENTS.md'), '## Root\n\nRoot agent.')
249
+
250
+ // Create child AGENTS.md
251
+ const childDir = path.join(testDir, 'src')
252
+ await fs.mkdir(childDir, { recursive: true })
253
+ await fs.writeFile(path.join(childDir, 'AGENTS.md'), '## Child\n\nChild agent.')
254
+
255
+ const agentFiles = await resolver.discoverAgentFiles()
256
+
257
+ const root = agentFiles.find((af) => af.depth === 0)
258
+ const child = agentFiles.find((af) => af.depth > 0)
259
+
260
+ expect(root).toBeDefined()
261
+ expect(child).toBeDefined()
262
+ expect(child?.parent).toBe(root)
263
+ expect(root?.children).toContain(child)
264
+ })
265
+
266
+ test('resolves agents for path with inheritance', async () => {
267
+ // Root defines base agents
268
+ await fs.writeFile(
269
+ path.join(testDir, 'AGENTS.md'),
270
+ `## Shared
271
+
272
+ Shared rules for all.
273
+
274
+ ### Rules
275
+ - Rule from root
276
+ `
277
+ )
278
+
279
+ // Child adds more rules
280
+ const childDir = path.join(testDir, 'packages', 'api')
281
+ await fs.mkdir(childDir, { recursive: true })
282
+ await fs.writeFile(
283
+ path.join(childDir, 'AGENTS.md'),
284
+ `## Shared
285
+
286
+ Extended in child.
287
+
288
+ ### Rules
289
+ - Rule from child
290
+ `
291
+ )
292
+
293
+ const resolved = await resolver.resolveAgentsForPath(childDir)
294
+
295
+ expect(resolved.agents).toHaveLength(1)
296
+ expect(resolved.agents[0].name).toBe('Shared')
297
+ // Rules should be merged
298
+ expect(resolved.agents[0].rules).toContain('Rule from root')
299
+ expect(resolved.agents[0].rules).toContain('Rule from child')
300
+ })
301
+
302
+ test('override replaces parent agent entirely', async () => {
303
+ // Root defines agent
304
+ await fs.writeFile(
305
+ path.join(testDir, 'AGENTS.md'),
306
+ `## Frontend
307
+
308
+ Root frontend agent.
309
+
310
+ ### Rules
311
+ - Root rule 1
312
+ - Root rule 2
313
+ `
314
+ )
315
+
316
+ // Child overrides
317
+ const childDir = path.join(testDir, 'web')
318
+ await fs.mkdir(childDir, { recursive: true })
319
+ await fs.writeFile(
320
+ path.join(childDir, 'AGENTS.md'),
321
+ `## Frontend @override
322
+
323
+ Completely new frontend agent.
324
+
325
+ ### Rules
326
+ - Child rule only
327
+ `
328
+ )
329
+
330
+ const resolved = await resolver.resolveAgentsForPath(childDir)
331
+
332
+ expect(resolved.agents).toHaveLength(1)
333
+ expect(resolved.agents[0].rules).toHaveLength(1)
334
+ expect(resolved.agents[0].rules).toContain('Child rule only')
335
+ expect(resolved.overrides).toContain(`web/AGENTS.md:Frontend`)
336
+ })
337
+
338
+ test('adds new agents from child without affecting parent agents', async () => {
339
+ // Root has one agent
340
+ await fs.writeFile(path.join(testDir, 'AGENTS.md'), '## Backend\n\nBackend agent.')
341
+
342
+ // Child adds different agent
343
+ const childDir = path.join(testDir, 'mobile')
344
+ await fs.mkdir(childDir, { recursive: true })
345
+ await fs.writeFile(path.join(childDir, 'AGENTS.md'), '## Mobile\n\nMobile agent.')
346
+
347
+ const resolved = await resolver.resolveAgentsForPath(childDir)
348
+
349
+ expect(resolved.agents).toHaveLength(2)
350
+ expect(resolved.agents.map((a) => a.name).sort()).toEqual(['Backend', 'Mobile'])
351
+ })
352
+
353
+ test('tracks sources in resolution', async () => {
354
+ await fs.writeFile(path.join(testDir, 'AGENTS.md'), '## Agent\n\nRoot.')
355
+
356
+ const level1 = path.join(testDir, 'level1')
357
+ await fs.mkdir(level1)
358
+ await fs.writeFile(path.join(level1, 'AGENTS.md'), '## Agent\n\nLevel 1.')
359
+
360
+ const level2 = path.join(level1, 'level2')
361
+ await fs.mkdir(level2)
362
+ await fs.writeFile(path.join(level2, 'AGENTS.md'), '## Agent\n\nLevel 2.')
363
+
364
+ const resolved = await resolver.resolveAgentsForPath(level2)
365
+
366
+ expect(resolved.sources).toHaveLength(3)
367
+ expect(resolved.sources[0]).toBe('AGENTS.md')
368
+ expect(resolved.sources[1]).toBe('level1/AGENTS.md')
369
+ expect(resolved.sources[2]).toBe('level1/level2/AGENTS.md')
370
+ })
371
+ })
372
+
373
+ // =============================================================================
374
+ // Edge Cases
375
+ // =============================================================================
376
+
377
+ describe('NestedContextResolver - Edge Cases', () => {
378
+ test('ignores node_modules directories', async () => {
379
+ await fs.writeFile(path.join(testDir, 'AGENTS.md'), '## Root\n\nRoot.')
380
+
381
+ // Create AGENTS.md in node_modules (should be ignored)
382
+ const nmDir = path.join(testDir, 'node_modules', 'some-package')
383
+ await fs.mkdir(nmDir, { recursive: true })
384
+ await fs.writeFile(path.join(nmDir, 'AGENTS.md'), '## ShouldIgnore\n\nIgnored.')
385
+
386
+ const agentFiles = await resolver.discoverAgentFiles()
387
+
388
+ expect(agentFiles).toHaveLength(1)
389
+ expect(agentFiles[0].agents[0].name).toBe('Root')
390
+ })
391
+
392
+ test('ignores dot directories', async () => {
393
+ await fs.writeFile(path.join(testDir, 'AGENTS.md'), '## Root\n\nRoot.')
394
+
395
+ // Create AGENTS.md in .git (should be ignored)
396
+ const gitDir = path.join(testDir, '.git', 'hooks')
397
+ await fs.mkdir(gitDir, { recursive: true })
398
+ await fs.writeFile(path.join(gitDir, 'AGENTS.md'), '## ShouldIgnore\n\nIgnored.')
399
+
400
+ const agentFiles = await resolver.discoverAgentFiles()
401
+
402
+ expect(agentFiles).toHaveLength(1)
403
+ })
404
+
405
+ test('handles malformed AGENTS.md gracefully', async () => {
406
+ await fs.writeFile(
407
+ path.join(testDir, 'AGENTS.md'),
408
+ `# Not a proper agent header
409
+
410
+ Some text here
411
+
412
+ ### Rules without agent
413
+ - orphan rule
414
+
415
+ ## Proper Agent
416
+
417
+ This one is valid.
418
+ `
419
+ )
420
+
421
+ const agentFiles = await resolver.discoverAgentFiles()
422
+
423
+ // Should only find the valid agent
424
+ expect(agentFiles[0].agents).toHaveLength(1)
425
+ expect(agentFiles[0].agents[0].name).toBe('Proper Agent')
426
+ })
427
+
428
+ test('limits scanning depth to prevent infinite recursion', async () => {
429
+ // Create deeply nested directory structure
430
+ let currentDir = testDir
431
+ for (let i = 0; i < 10; i++) {
432
+ currentDir = path.join(currentDir, `level${i}`)
433
+ await fs.mkdir(currentDir, { recursive: true })
434
+ await fs.writeFile(path.join(currentDir, 'AGENTS.md'), `## Level${i}\n\nLevel ${i} agent.`)
435
+ }
436
+
437
+ // Should complete without hanging (depth limit is 5)
438
+ const agentFiles = await resolver.discoverAgentFiles()
439
+
440
+ // Should find root + up to 5 levels deep
441
+ expect(agentFiles.length).toBeLessThanOrEqual(6)
442
+ })
443
+ })
@@ -455,89 +455,97 @@ export class AnalysisCommands extends PrjctCommandsBase {
455
455
 
456
456
  /**
457
457
  * Display sync results (extracted to avoid duplication)
458
+ *
459
+ * UX Design (PRJ-100):
460
+ * - Summary first: success + key metrics on first lines
461
+ * - Scannable: single-line metrics, minimal vertical space
462
+ * - Changes focused: show what changed, not everything that exists
463
+ * - Next steps prominent: clear call to action at bottom
458
464
  */
459
465
  private async showSyncResult(
460
466
  result: Awaited<ReturnType<typeof syncService.sync>>,
461
467
  startTime: number
462
468
  ): Promise<CommandResult> {
463
- // Update global config
464
- const globalConfigResult = await commandInstaller.installGlobalConfig()
465
- if (globalConfigResult.success) {
466
- console.log(`📝 Updated ${pathManager.getDisplayPath(globalConfigResult.path!)}`)
469
+ const elapsed = Date.now() - startTime
470
+ const contextFilesCount =
471
+ result.contextFiles.length + (result.aiTools?.filter((t) => t.success).length || 0)
472
+ const agentCount = result.agents.length
473
+ const domainAgentCount = result.agents.filter((a) => a.type === 'domain').length
474
+
475
+ // Update global config (silent - don't clutter output)
476
+ await commandInstaller.installGlobalConfig()
477
+
478
+ // ═══════════════════════════════════════════════════════════════════════
479
+ // SUCCESS LINE - Immediate confirmation with timing
480
+ // ═══════════════════════════════════════════════════════════════════════
481
+ console.log(`✅ Synced ${result.stats.name || 'project'} (${(elapsed / 1000).toFixed(1)}s)\n`)
482
+
483
+ // ═══════════════════════════════════════════════════════════════════════
484
+ // KEY METRICS - Single scannable line
485
+ // ═══════════════════════════════════════════════════════════════════════
486
+ // Only show compression rate if meaningful (> 10%)
487
+ const compressionPct = result.syncMetrics?.compressionRate
488
+ ? Math.round(result.syncMetrics.compressionRate * 100)
489
+ : 0
490
+ const metricsLine = [
491
+ `${result.stats.fileCount} files → ${contextFilesCount} context`,
492
+ `${agentCount} agents`,
493
+ compressionPct > 10 ? `${compressionPct}% reduction` : null,
494
+ ]
495
+ .filter(Boolean)
496
+ .join(' | ')
497
+ console.log(metricsLine)
498
+
499
+ // Stack and branch info
500
+ const framework = result.stats.frameworks.length > 0 ? ` (${result.stats.frameworks[0]})` : ''
501
+ console.log(`Stack: ${result.stats.ecosystem}${framework} | Branch: ${result.git.branch}\n`)
502
+
503
+ // ═══════════════════════════════════════════════════════════════════════
504
+ // CHANGES SECTION - What was generated/updated
505
+ // ═══════════════════════════════════════════════════════════════════════
506
+ console.log('Generated:')
507
+
508
+ // Context files (condensed)
509
+ if (result.contextFiles.length > 0) {
510
+ console.log(` ✓ ${result.contextFiles.length} context files`)
467
511
  }
468
512
 
469
- // Format output
470
- console.log(`🔄 Project synced to prjct v${result.cliVersion}\n`)
471
-
472
- console.log('📊 Project Stats')
473
- console.log(`├── Files: ~${result.stats.fileCount}`)
474
- console.log(`├── Commits: ${result.git.commits}`)
475
- console.log(`├── Version: ${result.stats.version}`)
476
- console.log(`└── Stack: ${result.stats.ecosystem}\n`)
477
-
478
- console.log('🌿 Git Status')
479
- console.log(`├── Branch: ${result.git.branch}`)
480
- console.log(`├── Uncommitted: ${result.git.hasChanges ? 'Yes' : 'Clean'}`)
481
- console.log(`└── Recent: ${result.git.weeklyCommits} commits this week\n`)
482
-
483
- console.log('📁 Context Updated')
484
- for (const file of result.contextFiles) {
485
- console.log(`├── ${file}`)
513
+ // AI tools
514
+ const successTools = result.aiTools?.filter((t) => t.success) || []
515
+ if (successTools.length > 0) {
516
+ const toolNames = successTools.map((t) => t.toolId).join(', ')
517
+ console.log(` ✓ AI tools: ${toolNames}`)
486
518
  }
487
- console.log('')
488
519
 
489
- // Show AI Tools generated (multi-agent output)
490
- if (result.aiTools && result.aiTools.length > 0) {
491
- const successTools = result.aiTools.filter((t) => t.success)
492
- console.log(`🤖 AI Tools Context (${successTools.length})`)
493
- for (const tool of result.aiTools) {
494
- const status = tool.success ? '✓' : '✗'
495
- console.log(`├── ${status} ${tool.outputFile} (${tool.toolId})`)
496
- }
497
- console.log('')
520
+ // Agents (show count with breakdown)
521
+ if (agentCount > 0) {
522
+ const agentSummary =
523
+ domainAgentCount > 0
524
+ ? `${agentCount} agents (${domainAgentCount} domain)`
525
+ : `${agentCount} agents`
526
+ console.log(` ✓ ${agentSummary}`)
498
527
  }
499
528
 
500
- const workflowAgents = result.agents.filter((a) => a.type === 'workflow').map((a) => a.name)
501
- const domainAgents = result.agents.filter((a) => a.type === 'domain').map((a) => a.name)
502
-
503
- console.log(`🤖 Agents Regenerated (${result.agents.length})`)
504
- console.log(`├── Workflow: ${workflowAgents.join(', ')}`)
505
- console.log(`└── Domain: ${domainAgents.join(', ') || 'none'}\n`)
506
-
529
+ // Skills
507
530
  if (result.skills.length > 0) {
508
- console.log('📦 Skills Configured')
509
- for (const skill of result.skills) {
510
- console.log(`├── ${skill.agent}.md → ${skill.skill}`)
511
- }
512
- console.log('')
531
+ const skillWord = result.skills.length === 1 ? 'skill' : 'skills'
532
+ console.log(` ✓ ${result.skills.length} ${skillWord}`)
513
533
  }
514
534
 
535
+ console.log('')
536
+
537
+ // ═══════════════════════════════════════════════════════════════════════
538
+ // STATUS INDICATOR - Repository state
539
+ // ═══════════════════════════════════════════════════════════════════════
515
540
  if (result.git.hasChanges) {
516
- console.log('⚠️ You have uncommitted changes\n')
517
- } else {
518
- console.log('✨ Repository is clean!\n')
541
+ console.log('⚠️ Uncommitted changes detected\n')
519
542
  }
520
543
 
544
+ // ═══════════════════════════════════════════════════════════════════════
545
+ // NEXT STEPS - Clear call to action
546
+ // ═══════════════════════════════════════════════════════════════════════
521
547
  showNextSteps('sync')
522
548
 
523
- // Summary metrics
524
- const elapsed = Date.now() - startTime
525
- const contextFilesCount =
526
- result.contextFiles.length + (result.aiTools?.filter((t) => t.success).length || 0)
527
- const agentCount = result.agents.length
528
-
529
- console.log('─'.repeat(45))
530
- console.log('📊 Sync Summary')
531
- console.log(
532
- ` Stack: ${result.stats.ecosystem} (${result.stats.frameworks.join(', ') || 'no frameworks'})`
533
- )
534
- console.log(` Files: ${result.stats.fileCount} analyzed → ${contextFilesCount} context files`)
535
- console.log(
536
- ` Agents: ${agentCount} (${result.agents.filter((a) => a.type === 'domain').length} domain)`
537
- )
538
- console.log(` Time: ${(elapsed / 1000).toFixed(1)}s`)
539
- console.log('')
540
-
541
549
  return {
542
550
  success: true,
543
551
  data: result,