@triedotdev/mcp 1.0.41 → 1.0.43
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.
- package/README.md +138 -3
- package/dist/{chunk-G2GNVUMP.js → chunk-2I6CFJTR.js} +62 -429
- package/dist/chunk-2I6CFJTR.js.map +1 -0
- package/dist/chunk-52SSNKXS.js +814 -0
- package/dist/chunk-52SSNKXS.js.map +1 -0
- package/dist/{chunk-B3MBKB2U.js → chunk-GBGONSOR.js} +201 -3
- package/dist/chunk-GBGONSOR.js.map +1 -0
- package/dist/cli/main.js +163 -11
- package/dist/cli/main.js.map +1 -1
- package/dist/cli/yolo-daemon.js +3 -3
- package/dist/index.js +47 -10
- package/dist/index.js.map +1 -1
- package/dist/workers/agent-worker.js +2 -1
- package/dist/workers/agent-worker.js.map +1 -1
- package/package.json +1 -1
- package/dist/chunk-B3MBKB2U.js.map +0 -1
- package/dist/chunk-G2GNVUMP.js.map +0 -1
- package/dist/chunk-Q4RVENDE.js +0 -229
- package/dist/chunk-Q4RVENDE.js.map +0 -1
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/utils/project-info.ts","../src/skills/installer.ts","../src/skills/parser.ts","../src/utils/context-state.ts"],"sourcesContent":["/**\n * Project Info Manager\n * \n * Manages the .trie/PROJECT.md file that stores user-defined project context.\n * Unlike AGENTS.md (auto-generated), PROJECT.md is fully user-controlled.\n */\n\nimport { readFile, writeFile, mkdir } from 'fs/promises';\nimport { existsSync } from 'fs';\nimport { join } from 'path';\nimport { getWorkingDirectory } from './workspace.js';\n\nconst PROJECT_MD_PATH = '.trie/PROJECT.md';\n\n/**\n * Default PROJECT.md template\n */\nexport function getProjectTemplate(): string {\n return `# Project Information\n\n> This file stores important project context for AI assistants.\n> Edit freely - this file is yours, not auto-generated.\n> Available via MCP resource: \\`trie://project\\`\n\n---\n\n## Project Overview\n\n<!-- Describe your project's purpose and goals -->\n\n[Add project description here]\n\n---\n\n## Technology Stack\n\n<!-- List frameworks, languages, databases, cloud services, etc. -->\n\n- **Language:** \n- **Framework:** \n- **Database:** \n- **Hosting:** \n\n---\n\n## Architecture\n\n<!-- Key patterns, architectural decisions, and system design -->\n\n[Describe your architecture here]\n\n---\n\n## Coding Conventions\n\n<!-- Style guidelines, naming conventions, patterns to follow -->\n\n- \n- \n- \n\n---\n\n## Environment\n\n<!-- URLs, API endpoints, deployment info -->\n\n| Environment | URL | Notes |\n|-------------|-----|-------|\n| Development | | |\n| Staging | | |\n| Production | | |\n\n---\n\n## Team\n\n<!-- Ownership, contacts, responsibilities -->\n\n- **Owner:** \n- **Team:** \n\n---\n\n## Compliance\n\n<!-- HIPAA, SOC2, GDPR, PCI-DSS requirements if applicable -->\n\n- [ ] GDPR\n- [ ] SOC2\n- [ ] HIPAA\n- [ ] PCI-DSS\n\n---\n\n## AI Instructions\n\n<!-- Special instructions for AI assistants working on this project -->\n\nWhen working on this project, AI assistants should:\n\n1. \n2. \n3. \n\n---\n\n*This file is read by Trie agents and exposed via \\`trie://project\\` MCP resource.*\n*Edit this file to provide context to Claude Code, Cursor, GitHub Actions, and other AI tools.*\n`;\n}\n\n/**\n * Check if PROJECT.md exists\n */\nexport function projectInfoExists(workDir?: string): boolean {\n const dir = workDir || getWorkingDirectory(undefined, true);\n const projectPath = join(dir, PROJECT_MD_PATH);\n return existsSync(projectPath);\n}\n\n/**\n * Load PROJECT.md content\n */\nexport async function loadProjectInfo(workDir?: string): Promise<string | null> {\n const dir = workDir || getWorkingDirectory(undefined, true);\n const projectPath = join(dir, PROJECT_MD_PATH);\n \n try {\n if (!existsSync(projectPath)) {\n return null;\n }\n return await readFile(projectPath, 'utf-8');\n } catch {\n return null;\n }\n}\n\n/**\n * Save PROJECT.md content\n */\nexport async function saveProjectInfo(content: string, workDir?: string): Promise<void> {\n const dir = workDir || getWorkingDirectory(undefined, true);\n const trieDir = join(dir, '.trie');\n const projectPath = join(dir, PROJECT_MD_PATH);\n \n // Ensure .trie directory exists\n await mkdir(trieDir, { recursive: true });\n \n await writeFile(projectPath, content, 'utf-8');\n}\n\n/**\n * Initialize PROJECT.md with template\n */\nexport async function initProjectInfo(workDir?: string): Promise<{ created: boolean; path: string }> {\n const dir = workDir || getWorkingDirectory(undefined, true);\n const projectPath = join(dir, PROJECT_MD_PATH);\n \n if (existsSync(projectPath)) {\n return { created: false, path: projectPath };\n }\n \n await saveProjectInfo(getProjectTemplate(), dir);\n return { created: true, path: projectPath };\n}\n\n/**\n * Get a specific section from PROJECT.md\n */\nexport async function getProjectSection(sectionName: string, workDir?: string): Promise<string | null> {\n const content = await loadProjectInfo(workDir);\n if (!content) return null;\n \n // Find section by header\n const sectionRegex = new RegExp(\n `## ${escapeRegex(sectionName)}\\\\s*\\\\n([\\\\s\\\\S]*?)(?=\\\\n## |\\\\n---\\\\s*$|$)`,\n 'i'\n );\n \n const match = content.match(sectionRegex);\n if (match) {\n return match[1].trim();\n }\n \n return null;\n}\n\n/**\n * Update a specific section in PROJECT.md\n */\nexport async function updateProjectSection(\n sectionName: string, \n newContent: string, \n workDir?: string\n): Promise<boolean> {\n let content = await loadProjectInfo(workDir);\n \n if (!content) {\n // Initialize with template first\n await initProjectInfo(workDir);\n content = await loadProjectInfo(workDir);\n if (!content) return false;\n }\n \n // Find and replace section content\n const sectionRegex = new RegExp(\n `(## ${escapeRegex(sectionName)}\\\\s*\\\\n)([\\\\s\\\\S]*?)((?=\\\\n## )|(?=\\\\n---\\\\s*$)|$)`,\n 'i'\n );\n \n if (content.match(sectionRegex)) {\n const updatedContent = content.replace(sectionRegex, `$1\\n${newContent}\\n\\n$3`);\n await saveProjectInfo(updatedContent, workDir);\n return true;\n }\n \n return false;\n}\n\n/**\n * Append content to a section\n */\nexport async function appendToSection(\n sectionName: string,\n contentToAdd: string,\n workDir?: string\n): Promise<boolean> {\n const currentContent = await getProjectSection(sectionName, workDir);\n if (currentContent === null) return false;\n \n const newContent = currentContent + '\\n' + contentToAdd;\n return updateProjectSection(sectionName, newContent, workDir);\n}\n\n/**\n * Get all section names from PROJECT.md\n */\nexport async function getProjectSections(workDir?: string): Promise<string[]> {\n const content = await loadProjectInfo(workDir);\n if (!content) return [];\n \n const sectionRegex = /^## (.+)$/gm;\n const sections: string[] = [];\n let match;\n \n while ((match = sectionRegex.exec(content)) !== null) {\n sections.push(match[1].trim());\n }\n \n return sections;\n}\n\n/**\n * Get project info as structured data (for JSON responses)\n */\nexport async function getProjectInfoStructured(workDir?: string): Promise<{\n exists: boolean;\n path: string;\n sections: Record<string, string>;\n raw: string | null;\n}> {\n const dir = workDir || getWorkingDirectory(undefined, true);\n const projectPath = join(dir, PROJECT_MD_PATH);\n const content = await loadProjectInfo(dir);\n \n if (!content) {\n return {\n exists: false,\n path: projectPath,\n sections: {},\n raw: null,\n };\n }\n \n const sectionNames = await getProjectSections(dir);\n const sections: Record<string, string> = {};\n \n for (const name of sectionNames) {\n const sectionContent = await getProjectSection(name, dir);\n if (sectionContent) {\n sections[name] = sectionContent;\n }\n }\n \n return {\n exists: true,\n path: projectPath,\n sections,\n raw: content,\n };\n}\n\n/**\n * Escape special regex characters\n */\nfunction escapeRegex(str: string): string {\n return str.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n}\n","/**\n * Skill Installer\n * \n * Installs skills from GitHub repositories into .trie/skills/\n * Skills can come from any repo that follows the SKILL.md format.\n */\n\nimport { mkdir, rm, writeFile, readdir, readFile, access, cp } from 'fs/promises';\nimport { join } from 'path';\nimport { exec } from 'child_process';\nimport { promisify } from 'util';\nimport { parseSkillMd } from './parser.js';\nimport { getWorkingDirectory } from '../utils/workspace.js';\nimport type { InstalledSkill, SkillInstallResult } from '../types/external-skill.js';\n\nconst execAsync = promisify(exec);\n\n/**\n * Install a skill from a GitHub repository\n * \n * @param source - GitHub source in format \"owner/repo\" or \"owner/repo/skill-name\"\n * @param skillName - Optional skill name if repo contains multiple skills\n */\nexport async function installSkill(source: string, skillName?: string): Promise<SkillInstallResult> {\n const parts = source.split('/');\n if (parts.length < 2) {\n return { success: false, name: 'unknown', error: 'Invalid source format. Use owner/repo or owner/repo/skill-name' };\n }\n \n const owner = parts[0];\n const repo = parts[1];\n const specifiedSkill = skillName || parts[2];\n \n const skillsDir = join(getWorkingDirectory(undefined, true), '.trie', 'skills');\n const tempDir = join(skillsDir, `.temp-${Date.now()}`);\n \n try {\n await mkdir(skillsDir, { recursive: true });\n \n // Clone the repository\n const repoUrl = `https://github.com/${owner}/${repo}.git`;\n await execAsync(`git clone --depth 1 \"${repoUrl}\" \"${tempDir}\"`, { timeout: 60000 });\n \n // Find the SKILL.md file\n const sourcePath = await findSkillPath(tempDir, specifiedSkill);\n if (!sourcePath) {\n throw new Error(`SKILL.md not found in repository. Searched in: root, skills/, ${specifiedSkill || 'subdirectories'}`);\n }\n \n // Parse and validate the skill\n const parsed = await parseSkillMd(sourcePath);\n const name = parsed.frontmatter.name;\n \n // Copy to final location\n const targetPath = join(skillsDir, name);\n \n // Remove existing if present\n await rm(targetPath, { recursive: true, force: true });\n \n // Copy skill directory\n await cp(sourcePath, targetPath, { recursive: true });\n \n // Write installation metadata\n await writeFile(join(targetPath, '.installed.json'), JSON.stringify({\n installedFrom: source,\n installedAt: new Date().toISOString(),\n repository: `${owner}/${repo}`,\n }, null, 2));\n \n return { success: true, name, path: targetPath };\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n return { success: false, name: skillName || 'unknown', error: message };\n } finally {\n // Clean up temp directory\n await rm(tempDir, { recursive: true, force: true }).catch(() => {});\n }\n}\n\n/**\n * Find the path to a SKILL.md within a cloned repository\n */\nasync function findSkillPath(repoPath: string, skillName?: string): Promise<string | null> {\n const searchPaths: string[] = [];\n \n if (skillName) {\n // Search for specific skill\n searchPaths.push(\n join(repoPath, 'skills', skillName),\n join(repoPath, skillName),\n );\n }\n \n // Also search common locations\n searchPaths.push(\n repoPath,\n join(repoPath, 'skill'),\n );\n \n // If no specific skill, try to find any skill directory\n if (!skillName) {\n try {\n const skillsSubdir = join(repoPath, 'skills');\n await access(skillsSubdir);\n const entries = await readdir(skillsSubdir, { withFileTypes: true });\n for (const entry of entries) {\n if (entry.isDirectory() && !entry.name.startsWith('.')) {\n searchPaths.push(join(skillsSubdir, entry.name));\n }\n }\n } catch {\n // skills/ directory doesn't exist\n }\n }\n \n // Check each path for SKILL.md\n for (const searchPath of searchPaths) {\n try {\n await parseSkillMd(searchPath);\n return searchPath;\n } catch {\n // Not a valid skill directory\n }\n }\n \n return null;\n}\n\n/**\n * List all installed skills\n */\nexport async function listInstalledSkills(): Promise<InstalledSkill[]> {\n const skillsDir = join(getWorkingDirectory(undefined, true), '.trie', 'skills');\n const skills: InstalledSkill[] = [];\n \n try {\n const entries = await readdir(skillsDir, { withFileTypes: true });\n \n for (const entry of entries) {\n if (!entry.isDirectory() || entry.name.startsWith('.')) continue;\n \n const skillPath = join(skillsDir, entry.name);\n \n try {\n const parsed = await parseSkillMd(skillPath);\n const metaPath = join(skillPath, '.installed.json');\n \n let meta = { installedFrom: 'unknown', installedAt: new Date().toISOString() };\n try {\n meta = JSON.parse(await readFile(metaPath, 'utf-8'));\n } catch {\n // No metadata file\n }\n \n skills.push({\n name: parsed.frontmatter.name,\n description: parsed.frontmatter.description,\n path: skillPath,\n installedFrom: meta.installedFrom,\n installedAt: meta.installedAt,\n });\n } catch {\n // Invalid skill directory, skip\n }\n }\n } catch {\n // Skills directory doesn't exist\n }\n \n return skills;\n}\n\n/**\n * Remove an installed skill\n */\nexport async function removeSkill(skillName: string): Promise<boolean> {\n const skillsDir = join(getWorkingDirectory(undefined, true), '.trie', 'skills');\n const skillPath = join(skillsDir, skillName);\n \n try {\n await rm(skillPath, { recursive: true });\n return true;\n } catch {\n return false;\n }\n}\n\n/**\n * Get a specific installed skill by name\n */\nexport async function getInstalledSkill(skillName: string): Promise<InstalledSkill | null> {\n const skills = await listInstalledSkills();\n return skills.find(s => s.name === skillName) || null;\n}\n","/**\n * Skill Parser\n * \n * Parses SKILL.md files to extract frontmatter and content.\n * Skills follow the Agent Skills specification from skills.sh.\n */\n\nimport { readFile } from 'fs/promises';\nimport { join } from 'path';\nimport type { ParsedSkill, SkillFrontmatter } from '../types/external-skill.js';\n\n/**\n * Parse a SKILL.md file from a skill directory\n */\nexport async function parseSkillMd(skillPath: string): Promise<ParsedSkill> {\n const skillMdPath = join(skillPath, 'SKILL.md');\n const rawContent = await readFile(skillMdPath, 'utf-8');\n \n const frontmatterMatch = rawContent.match(/^---\\n([\\s\\S]*?)\\n---/);\n if (!frontmatterMatch || !frontmatterMatch[1]) {\n throw new Error('Invalid SKILL.md: missing YAML frontmatter');\n }\n \n const frontmatter = parseYamlFrontmatter(frontmatterMatch[1]);\n \n if (!frontmatter.name || !frontmatter.description) {\n throw new Error('Invalid SKILL.md: missing required name or description in frontmatter');\n }\n \n const content = rawContent.slice(frontmatterMatch[0].length).trim();\n \n return {\n frontmatter,\n content,\n rawContent,\n };\n}\n\n/**\n * Simple YAML frontmatter parser\n * Handles basic key: value pairs without external dependencies\n */\nfunction parseYamlFrontmatter(yaml: string): SkillFrontmatter {\n const result: Record<string, unknown> = {};\n const lines = yaml.split('\\n');\n \n for (const line of lines) {\n const trimmed = line.trim();\n if (!trimmed || trimmed.startsWith('#')) continue;\n \n const colonIndex = trimmed.indexOf(':');\n if (colonIndex === -1) continue;\n \n const key = trimmed.slice(0, colonIndex).trim();\n let value: unknown = trimmed.slice(colonIndex + 1).trim();\n \n if (value === '') continue;\n \n // Remove quotes if present\n if (typeof value === 'string') {\n if ((value.startsWith('\"') && value.endsWith('\"')) ||\n (value.startsWith(\"'\") && value.endsWith(\"'\"))) {\n value = value.slice(1, -1);\n }\n \n // Handle arrays (simple format: [item1, item2])\n if (typeof value === 'string' && value.startsWith('[') && value.endsWith(']')) {\n value = value.slice(1, -1).split(',').map(s => s.trim().replace(/^[\"']|[\"']$/g, ''));\n }\n }\n \n result[key] = value;\n }\n \n return result as unknown as SkillFrontmatter;\n}\n\n/**\n * Extract just the name from a SKILL.md without full parsing\n */\nexport async function extractSkillName(skillPath: string): Promise<string | null> {\n try {\n const parsed = await parseSkillMd(skillPath);\n return parsed.frontmatter.name;\n } catch {\n return null;\n }\n}\n","/**\n * Context State Manager\n * \n * Manages the .trie/AGENTS.md file that provides context to AI coding assistants.\n * Updates automatically when agents run to keep context current.\n */\n\nimport { readFile, writeFile, mkdir } from 'fs/promises';\nimport { existsSync } from 'fs';\nimport { join } from 'path';\nimport { getWorkingDirectory } from './workspace.js';\nimport type { AgentResult } from '../types/index.js';\nimport type { SkillRecord } from '../types/external-skill.js';\nimport { loadProjectInfo, projectInfoExists } from './project-info.js';\n\nexport interface ScanSummary {\n timestamp: string;\n agents: string[];\n filesScanned: number;\n issues: {\n critical: number;\n serious: number;\n moderate: number;\n low: number;\n total: number;\n };\n duration: number;\n hotFiles: Array<{ file: string; issueCount: number }>;\n}\n\nexport interface ContextState {\n lastScan: ScanSummary | null;\n healthScore: number;\n activePriorities: string[];\n contextSignals: Record<string, boolean>;\n agentStatus: Record<string, {\n lastRun: string | null;\n issuesFound: number;\n skillsApplied?: string[];\n }>;\n scanHistory: ScanSummary[];\n customAgents: string[];\n skills: Record<string, SkillRecord>;\n environment?: string;\n}\n\nconst AGENTS_MD_PATH = '.trie/AGENTS.md';\nconst STATE_JSON_PATH = '.trie/state.json';\n\n/**\n * Load current context state\n */\nexport async function loadContextState(): Promise<ContextState> {\n const workDir = getWorkingDirectory(undefined, true);\n const statePath = join(workDir, STATE_JSON_PATH);\n const defaults = getDefaultState();\n \n try {\n if (existsSync(statePath)) {\n const content = await readFile(statePath, 'utf-8');\n const loaded = JSON.parse(content);\n // Merge with defaults to ensure new fields exist\n return {\n ...defaults,\n ...loaded,\n skills: loaded.skills || defaults.skills,\n };\n }\n } catch {\n // Return default state\n }\n \n return defaults;\n}\n\n/**\n * Save context state\n */\nexport async function saveContextState(state: ContextState): Promise<void> {\n const workDir = getWorkingDirectory(undefined, true);\n const trieDir = join(workDir, '.trie');\n const statePath = join(workDir, STATE_JSON_PATH);\n \n // Ensure directory exists\n await mkdir(trieDir, { recursive: true });\n \n await writeFile(statePath, JSON.stringify(state, null, 2));\n}\n\n/**\n * Update context state after a scan\n */\nexport async function updateContextAfterScan(\n results: AgentResult[],\n filesScanned: number,\n contextSignals: Record<string, boolean>,\n duration: number\n): Promise<void> {\n const state = await loadContextState();\n const now = new Date().toISOString();\n \n // Calculate issue counts\n const allIssues = results.flatMap(r => r.issues);\n const issueCounts = {\n critical: allIssues.filter(i => i.severity === 'critical').length,\n serious: allIssues.filter(i => i.severity === 'serious').length,\n moderate: allIssues.filter(i => i.severity === 'moderate').length,\n low: allIssues.filter(i => i.severity === 'low').length,\n total: allIssues.length,\n };\n \n // Find hot files\n const fileIssueMap = new Map<string, number>();\n for (const issue of allIssues) {\n const count = fileIssueMap.get(issue.file) || 0;\n fileIssueMap.set(issue.file, count + 1);\n }\n const hotFiles = Array.from(fileIssueMap.entries())\n .sort((a, b) => b[1] - a[1])\n .slice(0, 10)\n .map(([file, issueCount]) => ({ file, issueCount }));\n \n // Create scan summary\n const scanSummary: ScanSummary = {\n timestamp: now,\n agents: results.map(r => r.agent),\n filesScanned,\n issues: issueCounts,\n duration,\n hotFiles,\n };\n \n // Update agent status\n for (const result of results) {\n state.agentStatus[result.agent] = {\n lastRun: now,\n issuesFound: result.issues.length,\n };\n }\n \n // Calculate health score (0-100)\n // Penalize critical issues heavily, serious issues moderately\n const criticalPenalty = issueCounts.critical * 25;\n const seriousPenalty = issueCounts.serious * 10;\n const moderatePenalty = issueCounts.moderate * 3;\n const lowPenalty = issueCounts.low * 1;\n const totalPenalty = Math.min(100, criticalPenalty + seriousPenalty + moderatePenalty + lowPenalty);\n state.healthScore = Math.max(0, 100 - totalPenalty);\n \n // Update priorities based on issues\n state.activePriorities = generatePriorities(issueCounts, contextSignals);\n \n // Update context signals\n state.contextSignals = { ...state.contextSignals, ...contextSignals };\n \n // Update scan history (keep last 20)\n state.scanHistory = [scanSummary, ...state.scanHistory.slice(0, 19)];\n state.lastScan = scanSummary;\n \n // Save state\n await saveContextState(state);\n \n // Update AGENTS.md\n await updateAgentsMd(state);\n}\n\n/**\n * Update the AGENTS.md file with current state\n */\nasync function updateAgentsMd(state: ContextState): Promise<void> {\n const workDir = getWorkingDirectory(undefined, true);\n const mdPath = join(workDir, AGENTS_MD_PATH);\n \n let content: string;\n try {\n content = await readFile(mdPath, 'utf-8');\n } catch {\n // Use template if file doesn't exist\n content = getAgentsMdTemplate();\n }\n \n // Update sections\n content = updateSection(content, 'Project State', generateProjectStateTable(state));\n content = updateSection(content, 'Active Priorities', generatePrioritiesList(state));\n content = updateSection(content, 'Agent Status', generateAgentStatusTable(state));\n content = updateSection(content, 'Recent Scan History', generateScanHistoryTable(state));\n content = updateSection(content, 'Context Signals Detected', generateContextSignals(state));\n content = updateSection(content, 'Risk Assessment', generateRiskAssessment(state));\n content = updateSection(content, 'Hot Files', generateHotFilesSection(state));\n \n // Update last updated timestamp\n content = content.replace(\n /Last updated:.*$/m,\n `Last updated: ${new Date().toISOString()}`\n );\n \n await writeFile(mdPath, content);\n}\n\n/**\n * Update a specific section in the markdown\n */\nfunction updateSection(content: string, sectionName: string, newContent: string): string {\n // Look for the section header and replace content until the next section\n const sectionRegex = new RegExp(\n `(### ${sectionName}[\\\\s\\\\S]*?)(?=###|---|\\n## |$)`,\n 'g'\n );\n \n const replacement = `### ${sectionName}\\n${newContent}\\n\\n`;\n \n if (content.match(sectionRegex)) {\n return content.replace(sectionRegex, replacement);\n }\n \n // Section doesn't exist, don't add it\n return content;\n}\n\n/**\n * Generate the project state table\n */\nfunction generateProjectStateTable(state: ContextState): string {\n const lastScan = state.lastScan;\n const lastScanDate = lastScan ? new Date(lastScan.timestamp).toLocaleString() : 'Never';\n const criticalCount = lastScan?.issues.critical ?? 0;\n const totalTasks = lastScan?.issues.total ?? 0;\n \n return `| Metric | Value | Updated |\n|--------|-------|---------|\n| Last Scan | ${lastScanDate} | ${lastScan ? 'Auto' : '-'} |\n| Critical Issues | ${criticalCount} | ${lastScan ? 'Auto' : '-'} |\n| Open Tasks | ${totalTasks} | ${lastScan ? 'Auto' : '-'} |\n| Health Score | ${state.healthScore}% | ${lastScan ? 'Auto' : '-'} |`;\n}\n\n/**\n * Generate priorities list\n */\nfunction generatePrioritiesList(state: ContextState): string {\n if (state.activePriorities.length === 0) {\n return '_No active priorities. Run a scan to identify issues._';\n }\n \n return state.activePriorities\n .map((p, i) => `${i + 1}. ${p}`)\n .join('\\n');\n}\n\n/**\n * Generate priorities based on scan results\n */\nfunction generatePriorities(\n issues: { critical: number; serious: number; moderate: number; low: number },\n contextSignals: Record<string, boolean>\n): string[] {\n const priorities: string[] = [];\n \n if (issues.critical > 0) {\n priorities.push(`🚨 Fix ${issues.critical} critical security issue${issues.critical > 1 ? 's' : ''} immediately`);\n }\n \n if (issues.serious > 0) {\n priorities.push(`⚠️ Address ${issues.serious} serious issue${issues.serious > 1 ? 's' : ''} before deployment`);\n }\n \n if (contextSignals.touchesAuth && issues.critical === 0) {\n priorities.push('✅ Auth code reviewed - continue monitoring');\n }\n \n if (contextSignals.touchesPayments) {\n priorities.push('💳 Payment code detected - ensure PCI compliance');\n }\n \n if (contextSignals.touchesUserData) {\n priorities.push('🔐 User data handling detected - verify privacy compliance');\n }\n \n if (issues.moderate > 5) {\n priorities.push(`📋 Schedule time to address ${issues.moderate} moderate issues`);\n }\n \n if (priorities.length === 0) {\n priorities.push('✨ No critical issues - focus on feature development');\n }\n \n return priorities.slice(0, 5);\n}\n\n/**\n * Generate agent status table\n */\nfunction generateAgentStatusTable(state: ContextState): string {\n const builtInAgents = [\n 'security', 'privacy', 'legal', 'accessibility', 'bugs',\n 'design', 'architecture', 'performance', 'devops', 'soc2',\n 'e2e', 'typecheck', 'visual-qa', 'data-flow'\n ];\n \n let table = `| Agent | Status | Last Run | Issues Found |\n|-------|--------|----------|--------------|`;\n \n for (const agent of builtInAgents) {\n const status = state.agentStatus[agent];\n const lastRun = status?.lastRun \n ? new Date(status.lastRun).toLocaleDateString()\n : 'Never';\n const issues = status?.issuesFound ?? '-';\n const statusEmoji = status ? '✅' : '⏸️';\n \n table += `\\n| ${agent} | ${statusEmoji} Ready | ${lastRun} | ${issues} |`;\n }\n \n return table;\n}\n\n/**\n * Generate scan history table\n */\nfunction generateScanHistoryTable(state: ContextState): string {\n if (state.scanHistory.length === 0) {\n return `| Date | Agents | Files | Issues | Duration |\n|------|--------|-------|--------|----------|\n| - | - | - | - | - |`;\n }\n \n let table = `| Date | Agents | Files | Issues | Duration |\n|------|--------|-------|--------|----------|`;\n \n for (const scan of state.scanHistory.slice(0, 10)) {\n const date = new Date(scan.timestamp).toLocaleDateString();\n const agents = scan.agents.slice(0, 3).join(', ') + (scan.agents.length > 3 ? '...' : '');\n const duration = `${(scan.duration / 1000).toFixed(1)}s`;\n \n table += `\\n| ${date} | ${agents} | ${scan.filesScanned} | ${scan.issues.total} | ${duration} |`;\n }\n \n return table;\n}\n\n/**\n * Generate context signals section\n */\nfunction generateContextSignals(state: ContextState): string {\n const signals = [\n 'touchesAuth',\n 'touchesPayments', \n 'touchesUserData',\n 'touchesAPI',\n 'touchesDatabase',\n 'touchesCrypto'\n ];\n \n return signals\n .map(s => {\n const value = state.contextSignals[s];\n const emoji = value === true ? '✅' : value === false ? '❌' : '❓';\n return `- \\`${s}\\`: ${emoji} ${value === undefined ? 'Unknown' : value ? 'Yes' : 'No'}`;\n })\n .join('\\n');\n}\n\n/**\n * Generate risk assessment section\n */\nfunction generateRiskAssessment(state: ContextState): string {\n const score = state.healthScore;\n let riskLevel: string;\n let confidence: number;\n \n if (state.lastScan === null) {\n return `- Overall Risk: Unknown\n- Confidence: 0%`;\n }\n \n if (score >= 90) {\n riskLevel = '🟢 Low';\n confidence = 95;\n } else if (score >= 70) {\n riskLevel = '🟡 Medium';\n confidence = 85;\n } else if (score >= 50) {\n riskLevel = '🟠 High';\n confidence = 80;\n } else {\n riskLevel = '🔴 Critical';\n confidence = 90;\n }\n \n return `- Overall Risk: ${riskLevel}\n- Health Score: ${score}%\n- Confidence: ${confidence}%`;\n}\n\n/**\n * Generate hot files section\n */\nfunction generateHotFilesSection(state: ContextState): string {\n if (!state.lastScan || state.lastScan.hotFiles.length === 0) {\n return '_Run a scan to identify hot files._';\n }\n \n return state.lastScan.hotFiles\n .map(f => `- \\`${f.file}\\` - ${f.issueCount} issue${f.issueCount > 1 ? 's' : ''}`)\n .join('\\n');\n}\n\n/**\n * Get default state\n */\nfunction getDefaultState(): ContextState {\n return {\n lastScan: null,\n healthScore: 0,\n activePriorities: [\n 'Initial setup required - run first scan with `trie scan`',\n 'Configure agents in `.trie/config.json`',\n 'Set up CI/CD integration'\n ],\n contextSignals: {},\n agentStatus: {},\n scanHistory: [],\n customAgents: [],\n skills: {},\n environment: detectEnvironment(),\n };\n}\n\n/**\n * Detect the current execution environment\n */\nexport function detectEnvironment(): string {\n if (process.env.GITHUB_ACTIONS) return 'github-actions';\n if (process.env.GITLAB_CI) return 'gitlab-ci';\n if (process.env.CI) return 'ci';\n const parent = process.env._ || '';\n if (parent.includes('cursor')) return 'cursor';\n if (parent.includes('claude')) return 'claude-code';\n return 'cli';\n}\n\n/**\n * Record skill installation in context state\n */\nexport async function recordSkillInstalled(params: {\n name: string;\n source: string;\n}): Promise<void> {\n const state = await loadContextState();\n state.skills[params.name] = {\n source: params.source,\n installedAt: new Date().toISOString(),\n timesApplied: 0,\n appliedBy: [],\n };\n await saveContextState(state);\n}\n\n/**\n * Record skill usage by an agent\n */\nexport async function recordSkillUsage(params: {\n skillNames: string[];\n agentName: string;\n}): Promise<void> {\n const state = await loadContextState();\n const now = new Date().toISOString();\n \n for (const skillName of params.skillNames) {\n const skillRecord = state.skills[skillName];\n if (skillRecord) {\n skillRecord.timesApplied++;\n skillRecord.lastApplied = now;\n if (!skillRecord.appliedBy.includes(params.agentName)) {\n skillRecord.appliedBy.push(params.agentName);\n }\n }\n }\n \n const agentStatus = state.agentStatus[params.agentName];\n if (agentStatus) {\n agentStatus.skillsApplied = params.skillNames;\n }\n \n await saveContextState(state);\n}\n\n/**\n * Get the AGENTS.md template\n */\nfunction getAgentsMdTemplate(): string {\n return `# Trie Agent Context\n\n> **Auto-generated file** - Updated automatically when agents run.\n> Last updated: Never (initial state)\n\nThis file provides prioritized context for all AI coding assistants working with this codebase.\nAgents should read this file first and update it after completing scans.\n\n---\n\n## Quick Context (Read First)\n\n### Project State\n| Metric | Value | Updated |\n|--------|-------|---------|\n| Last Scan | Never | - |\n| Critical Issues | 0 | - |\n| Open Tasks | 0 | - |\n| Health Score | Unknown | - |\n\n### Active Priorities\n1. Initial setup required - run first scan with \\`trie scan\\`\n2. Configure agents in \\`.trie/config.json\\`\n3. Set up CI/CD integration\n\n### Hot Files\n_Run a scan to identify hot files._\n\n---\n\n## Agent Status\n\n### Agent Status\n| Agent | Status | Last Run | Issues Found |\n|-------|--------|----------|--------------|\n| security | Ready | Never | - |\n| privacy | Ready | Never | - |\n| bugs | Ready | Never | - |\n\n### Recent Scan History\n| Date | Agents | Files | Issues | Duration |\n|------|--------|-------|--------|----------|\n| - | - | - | - | - |\n\n---\n\n## Context Analysis\n\n### Context Signals Detected\n- \\`touchesAuth\\`: Unknown\n- \\`touchesPayments\\`: Unknown\n- \\`touchesUserData\\`: Unknown\n- \\`touchesAPI\\`: Unknown\n- \\`touchesDatabase\\`: Unknown\n- \\`touchesCrypto\\`: Unknown\n\n### Risk Assessment\n- Overall Risk: Unknown\n- Confidence: 0%\n\n---\n\n*This file is maintained by Trie agents. Manual edits will be preserved in non-auto sections.*\n`;\n}\n\n/**\n * Add custom agent to context state\n */\nexport async function addCustomAgentToContext(agentName: string): Promise<void> {\n const state = await loadContextState();\n \n if (!state.customAgents.includes(agentName)) {\n state.customAgents.push(agentName);\n await saveContextState(state);\n }\n}\n\n/**\n * Get context state for AI assistants (formatted summary)\n */\nexport async function getContextForAI(): Promise<string> {\n const state = await loadContextState();\n const workDir = getWorkingDirectory(undefined, true);\n \n const lines: string[] = [];\n \n // Include PROJECT.md content first (user-defined context takes priority)\n if (projectInfoExists(workDir)) {\n const projectInfo = await loadProjectInfo(workDir);\n if (projectInfo) {\n lines.push(projectInfo);\n lines.push('');\n lines.push('---');\n lines.push('');\n }\n }\n \n // Add Trie scan context\n lines.push(\n '## Trie Scan Context',\n '',\n `**Health Score:** ${state.healthScore}%`,\n `**Last Scan:** ${state.lastScan ? new Date(state.lastScan.timestamp).toLocaleString() : 'Never'}`,\n '',\n '**Active Priorities:**',\n ...state.activePriorities.map(p => `- ${p}`),\n ''\n );\n \n if (state.lastScan) {\n lines.push(\n '**Recent Issues:**',\n `- Critical: ${state.lastScan.issues.critical}`,\n `- Serious: ${state.lastScan.issues.serious}`,\n `- Moderate: ${state.lastScan.issues.moderate}`,\n `- Low: ${state.lastScan.issues.low}`,\n ''\n );\n \n if (state.lastScan.hotFiles.length > 0) {\n lines.push(\n '**Hot Files (most issues):**',\n ...state.lastScan.hotFiles.slice(0, 5).map(f => `- ${f.file}: ${f.issueCount} issues`),\n ''\n );\n }\n }\n \n return lines.join('\\n');\n}\n"],"mappings":"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","names":["mkdir","writeFile","readFile","join","readFile","join","join","mkdir","writeFile","readFile","readFile","writeFile","mkdir","existsSync","join","join","existsSync","readFile","mkdir","writeFile"]}
|
|
@@ -1,3 +1,8 @@
|
|
|
1
|
+
import {
|
|
2
|
+
listInstalledSkills,
|
|
3
|
+
parseSkillMd,
|
|
4
|
+
recordSkillUsage
|
|
5
|
+
} from "./chunk-52SSNKXS.js";
|
|
1
6
|
import {
|
|
2
7
|
checkFileLevelIssues,
|
|
3
8
|
getVibeCodeTrie,
|
|
@@ -5,7 +10,9 @@ import {
|
|
|
5
10
|
} from "./chunk-3CS6Z2SL.js";
|
|
6
11
|
import {
|
|
7
12
|
AgentSmithAgent,
|
|
8
|
-
BaseAgent
|
|
13
|
+
BaseAgent,
|
|
14
|
+
isAIAvailable,
|
|
15
|
+
runAIAnalysis
|
|
9
16
|
} from "./chunk-OB45V2QC.js";
|
|
10
17
|
import {
|
|
11
18
|
getWorkingDirectory
|
|
@@ -8484,6 +8491,182 @@ Output STRICT JSON:
|
|
|
8484
8491
|
}
|
|
8485
8492
|
};
|
|
8486
8493
|
|
|
8494
|
+
// src/agents/skill-review.ts
|
|
8495
|
+
var SkillReviewAgent = class extends BaseAgent {
|
|
8496
|
+
name = "skill-review";
|
|
8497
|
+
description = "Applies installed skills from GitHub repositories to code review";
|
|
8498
|
+
version = "1.0.0";
|
|
8499
|
+
skills = [];
|
|
8500
|
+
loadedSkillContent = /* @__PURE__ */ new Map();
|
|
8501
|
+
get priority() {
|
|
8502
|
+
return {
|
|
8503
|
+
name: this.name,
|
|
8504
|
+
tier: 2,
|
|
8505
|
+
estimatedTimeMs: 500,
|
|
8506
|
+
dependencies: []
|
|
8507
|
+
};
|
|
8508
|
+
}
|
|
8509
|
+
/**
|
|
8510
|
+
* Agent decides when to activate based on context and installed skills
|
|
8511
|
+
*/
|
|
8512
|
+
shouldActivate(context) {
|
|
8513
|
+
if (this.skills.length === 0) return false;
|
|
8514
|
+
return context.touchesUI || context.touchesAPI || context.isNewFeature;
|
|
8515
|
+
}
|
|
8516
|
+
getActivationConfidence(context) {
|
|
8517
|
+
if (!this.shouldActivate(context)) return 0;
|
|
8518
|
+
let confidence = 0.5;
|
|
8519
|
+
if (context.touchesUI) confidence += 0.2;
|
|
8520
|
+
if (context.isNewFeature) confidence += 0.15;
|
|
8521
|
+
if (context.framework) confidence += 0.1;
|
|
8522
|
+
if (this.skills.length > 1) confidence += 0.1;
|
|
8523
|
+
return Math.min(1, confidence);
|
|
8524
|
+
}
|
|
8525
|
+
/**
|
|
8526
|
+
* Load installed skills before scanning
|
|
8527
|
+
*/
|
|
8528
|
+
async loadSkills() {
|
|
8529
|
+
this.skills = await listInstalledSkills();
|
|
8530
|
+
this.loadedSkillContent.clear();
|
|
8531
|
+
for (const skill of this.skills) {
|
|
8532
|
+
try {
|
|
8533
|
+
const parsed = await parseSkillMd(skill.path);
|
|
8534
|
+
this.loadedSkillContent.set(skill.name, parsed);
|
|
8535
|
+
} catch {
|
|
8536
|
+
}
|
|
8537
|
+
}
|
|
8538
|
+
}
|
|
8539
|
+
/**
|
|
8540
|
+
* Get names of skills this agent will apply
|
|
8541
|
+
*/
|
|
8542
|
+
getAppliedSkillNames() {
|
|
8543
|
+
return this.skills.map((s) => s.name);
|
|
8544
|
+
}
|
|
8545
|
+
/**
|
|
8546
|
+
* Check if any skills are loaded
|
|
8547
|
+
*/
|
|
8548
|
+
hasSkills() {
|
|
8549
|
+
return this.skills.length > 0;
|
|
8550
|
+
}
|
|
8551
|
+
/**
|
|
8552
|
+
* Override to always use AI since skills require AI interpretation
|
|
8553
|
+
*/
|
|
8554
|
+
shouldAlwaysUseAI() {
|
|
8555
|
+
return true;
|
|
8556
|
+
}
|
|
8557
|
+
/**
|
|
8558
|
+
* Override scan to track skill usage after completion
|
|
8559
|
+
*/
|
|
8560
|
+
async scan(files, context) {
|
|
8561
|
+
const result = await super.scan(files, context);
|
|
8562
|
+
if (this.skills.length > 0 && result.issues.length > 0) {
|
|
8563
|
+
await recordSkillUsage({
|
|
8564
|
+
skillNames: this.getAppliedSkillNames(),
|
|
8565
|
+
agentName: this.name
|
|
8566
|
+
}).catch(() => {
|
|
8567
|
+
});
|
|
8568
|
+
}
|
|
8569
|
+
return result;
|
|
8570
|
+
}
|
|
8571
|
+
/**
|
|
8572
|
+
* Agent orchestrates analysis - skills provide the knowledge
|
|
8573
|
+
*/
|
|
8574
|
+
async analyzeFiles(files, context) {
|
|
8575
|
+
if (!isAIAvailable()) return [];
|
|
8576
|
+
if (this.loadedSkillContent.size === 0) return [];
|
|
8577
|
+
const snippets = [];
|
|
8578
|
+
const filesToAnalyze = files.slice(0, 10);
|
|
8579
|
+
for (const file of filesToAnalyze) {
|
|
8580
|
+
try {
|
|
8581
|
+
const content = await this.readFile(file);
|
|
8582
|
+
const rel = file.replace(context.workingDir, "").replace(/^\//, "");
|
|
8583
|
+
const truncated = content.slice(0, 3e3);
|
|
8584
|
+
snippets.push(`### ${rel}
|
|
8585
|
+
\`\`\`
|
|
8586
|
+
${truncated}
|
|
8587
|
+
\`\`\``);
|
|
8588
|
+
} catch {
|
|
8589
|
+
}
|
|
8590
|
+
}
|
|
8591
|
+
if (snippets.length === 0) return [];
|
|
8592
|
+
const skillContexts = this.buildSkillContext();
|
|
8593
|
+
const systemPrompt = `You are the Skill Review Agent. You apply specialized skills to code review.
|
|
8594
|
+
|
|
8595
|
+
Apply these skills to the code:
|
|
8596
|
+
|
|
8597
|
+
${skillContexts}
|
|
8598
|
+
|
|
8599
|
+
---
|
|
8600
|
+
|
|
8601
|
+
For each issue found, specify which skill's guideline was violated.
|
|
8602
|
+
Output ONLY valid JSON in this format:
|
|
8603
|
+
{
|
|
8604
|
+
"issues": [
|
|
8605
|
+
{
|
|
8606
|
+
"severity": "critical|serious|moderate|low",
|
|
8607
|
+
"issue": "Description of the issue",
|
|
8608
|
+
"skill": "skill-name-that-found-this",
|
|
8609
|
+
"file": "path/to/file.ts",
|
|
8610
|
+
"line": 123,
|
|
8611
|
+
"fix": "How to fix it"
|
|
8612
|
+
}
|
|
8613
|
+
]
|
|
8614
|
+
}`;
|
|
8615
|
+
const result = await runAIAnalysis({
|
|
8616
|
+
systemPrompt,
|
|
8617
|
+
userPrompt: `Review this code:
|
|
8618
|
+
|
|
8619
|
+
${snippets.join("\n\n")}`,
|
|
8620
|
+
maxTokens: 4096,
|
|
8621
|
+
temperature: 0.2
|
|
8622
|
+
});
|
|
8623
|
+
if (!result.success) return [];
|
|
8624
|
+
return this.parseSkillIssues(result.content, files);
|
|
8625
|
+
}
|
|
8626
|
+
buildSkillContext() {
|
|
8627
|
+
const contexts = [];
|
|
8628
|
+
for (const [name, parsed] of this.loadedSkillContent) {
|
|
8629
|
+
contexts.push(`## Skill: ${name}
|
|
8630
|
+
${parsed.rawContent}`);
|
|
8631
|
+
}
|
|
8632
|
+
return contexts.join("\n\n---\n\n");
|
|
8633
|
+
}
|
|
8634
|
+
parseSkillIssues(response, files) {
|
|
8635
|
+
const issues = [];
|
|
8636
|
+
try {
|
|
8637
|
+
const jsonMatch = response.match(/\{[\s\S]*\}/);
|
|
8638
|
+
if (!jsonMatch) return issues;
|
|
8639
|
+
const parsed = JSON.parse(jsonMatch[0]);
|
|
8640
|
+
if (!parsed.issues || !Array.isArray(parsed.issues)) return issues;
|
|
8641
|
+
for (const item of parsed.issues) {
|
|
8642
|
+
const severity = this.normalizeSeverity(item.severity);
|
|
8643
|
+
const skillName = item.skill || "unknown-skill";
|
|
8644
|
+
issues.push(this.createIssue(
|
|
8645
|
+
this.generateIssueId(),
|
|
8646
|
+
severity,
|
|
8647
|
+
`[${skillName}] ${item.issue || "Issue detected"}`,
|
|
8648
|
+
item.fix || "See skill guidelines",
|
|
8649
|
+
item.file || files[0] || "unknown",
|
|
8650
|
+
item.line,
|
|
8651
|
+
0.8,
|
|
8652
|
+
void 0,
|
|
8653
|
+
false,
|
|
8654
|
+
{ category: skillName }
|
|
8655
|
+
));
|
|
8656
|
+
}
|
|
8657
|
+
} catch {
|
|
8658
|
+
}
|
|
8659
|
+
return issues;
|
|
8660
|
+
}
|
|
8661
|
+
normalizeSeverity(severity) {
|
|
8662
|
+
const normalized = (severity || "").toLowerCase();
|
|
8663
|
+
if (normalized === "critical") return "critical";
|
|
8664
|
+
if (normalized === "serious" || normalized === "high") return "serious";
|
|
8665
|
+
if (normalized === "moderate" || normalized === "medium") return "moderate";
|
|
8666
|
+
return "low";
|
|
8667
|
+
}
|
|
8668
|
+
};
|
|
8669
|
+
|
|
8487
8670
|
// src/agents/custom-agent.ts
|
|
8488
8671
|
var CustomAgent = class extends BaseAgent {
|
|
8489
8672
|
name;
|
|
@@ -8731,7 +8914,9 @@ var AgentRegistryImpl = class {
|
|
|
8731
8914
|
// Cost analysis agent
|
|
8732
8915
|
new MoneybagAgent(),
|
|
8733
8916
|
// Production readiness gate
|
|
8734
|
-
new ProductionReadyAgent()
|
|
8917
|
+
new ProductionReadyAgent(),
|
|
8918
|
+
// Skill review agent (applies external skills)
|
|
8919
|
+
new SkillReviewAgent()
|
|
8735
8920
|
];
|
|
8736
8921
|
console.error(`Loaded config for ${builtinAgents.length} built-in agents`);
|
|
8737
8922
|
for (const agent of builtinAgents) {
|
|
@@ -8767,6 +8952,19 @@ var AgentRegistryImpl = class {
|
|
|
8767
8952
|
} catch (error) {
|
|
8768
8953
|
this.customAgentsLoaded = true;
|
|
8769
8954
|
}
|
|
8955
|
+
await this.loadSkillsForAgent();
|
|
8956
|
+
}
|
|
8957
|
+
/**
|
|
8958
|
+
* Load installed skills for the SkillReviewAgent
|
|
8959
|
+
*/
|
|
8960
|
+
async loadSkillsForAgent() {
|
|
8961
|
+
const skillReviewAgent = this.agents.get("skill-review");
|
|
8962
|
+
if (skillReviewAgent && skillReviewAgent instanceof SkillReviewAgent) {
|
|
8963
|
+
await skillReviewAgent.loadSkills();
|
|
8964
|
+
if (skillReviewAgent.hasSkills()) {
|
|
8965
|
+
console.error(`Loaded ${skillReviewAgent.getAppliedSkillNames().length} skill(s) for skill-review agent`);
|
|
8966
|
+
}
|
|
8967
|
+
}
|
|
8770
8968
|
}
|
|
8771
8969
|
/**
|
|
8772
8970
|
* Reload custom agents (useful after creating new ones)
|
|
@@ -8858,4 +9056,4 @@ export {
|
|
|
8858
9056
|
CustomAgent,
|
|
8859
9057
|
getAgentRegistry
|
|
8860
9058
|
};
|
|
8861
|
-
//# sourceMappingURL=chunk-
|
|
9059
|
+
//# sourceMappingURL=chunk-GBGONSOR.js.map
|