@su-record/vibe 2.7.12 → 2.7.14
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/.env.example +37 -37
- package/CLAUDE.md +126 -126
- package/LICENSE +21 -21
- package/README.md +449 -580
- package/agents/architect-low.md +41 -41
- package/agents/architect-medium.md +59 -59
- package/agents/architect.md +80 -80
- package/agents/build-error-resolver.md +115 -115
- package/agents/compounder.md +261 -261
- package/agents/diagrammer.md +178 -178
- package/agents/docs/api-documenter.md +99 -99
- package/agents/docs/changelog-writer.md +93 -93
- package/agents/e2e-tester.md +294 -266
- package/agents/explorer-low.md +42 -42
- package/agents/explorer-medium.md +59 -59
- package/agents/explorer.md +48 -48
- package/agents/implementer-low.md +43 -43
- package/agents/implementer-medium.md +52 -52
- package/agents/implementer.md +54 -54
- package/agents/junior-mentor.md +141 -141
- package/agents/planning/requirements-analyst.md +84 -84
- package/agents/planning/ux-advisor.md +83 -83
- package/agents/qa/acceptance-tester.md +86 -86
- package/agents/qa/edge-case-finder.md +93 -93
- package/agents/refactor-cleaner.md +143 -143
- package/agents/research/best-practices-agent.md +199 -199
- package/agents/research/codebase-patterns-agent.md +157 -157
- package/agents/research/framework-docs-agent.md +188 -188
- package/agents/research/security-advisory-agent.md +213 -213
- package/agents/review/architecture-reviewer.md +107 -107
- package/agents/review/complexity-reviewer.md +116 -116
- package/agents/review/data-integrity-reviewer.md +88 -88
- package/agents/review/git-history-reviewer.md +103 -103
- package/agents/review/performance-reviewer.md +86 -86
- package/agents/review/python-reviewer.md +150 -150
- package/agents/review/rails-reviewer.md +139 -139
- package/agents/review/react-reviewer.md +144 -144
- package/agents/review/security-reviewer.md +80 -80
- package/agents/review/simplicity-reviewer.md +140 -140
- package/agents/review/test-coverage-reviewer.md +116 -116
- package/agents/review/typescript-reviewer.md +127 -127
- package/agents/searcher.md +54 -54
- package/agents/simplifier.md +120 -120
- package/agents/tester.md +49 -49
- package/agents/ui/ui-a11y-auditor.md +93 -93
- package/agents/ui/ui-antipattern-detector.md +94 -94
- package/agents/ui/ui-dataviz-advisor.md +69 -69
- package/agents/ui/ui-design-system-gen.md +57 -57
- package/agents/ui/ui-industry-analyzer.md +49 -49
- package/agents/ui/ui-layout-architect.md +65 -65
- package/agents/ui/ui-stack-implementer.md +68 -68
- package/agents/ui/ux-compliance-reviewer.md +81 -81
- package/agents/ui-previewer.md +258 -260
- package/commands/vibe.analyze.md +8 -0
- package/commands/vibe.review.md +10 -3
- package/commands/vibe.run.md +2078 -2022
- package/commands/vibe.spec.md +10 -10
- package/commands/vibe.spec.review.md +565 -558
- package/commands/vibe.utils.md +413 -413
- package/commands/vibe.verify.md +45 -0
- package/dist/cli/auth.d.ts.map +1 -1
- package/dist/cli/auth.js +1 -7
- package/dist/cli/auth.js.map +1 -1
- package/dist/cli/collaborator.js +52 -52
- package/dist/cli/commands/evolution.js +12 -12
- package/dist/cli/commands/info.d.ts.map +1 -1
- package/dist/cli/commands/info.js +55 -70
- package/dist/cli/commands/info.js.map +1 -1
- package/dist/cli/commands/init.d.ts.map +1 -1
- package/dist/cli/commands/init.js +6 -7
- package/dist/cli/commands/init.js.map +1 -1
- package/dist/cli/commands/remove.js +14 -14
- package/dist/cli/commands/sentinel.js +27 -27
- package/dist/cli/commands/setup.js +1 -1
- package/dist/cli/commands/setup.js.map +1 -1
- package/dist/cli/commands/skills.js +5 -5
- package/dist/cli/commands/slack.js +10 -10
- package/dist/cli/commands/telegram.js +12 -12
- package/dist/cli/commands/update.d.ts.map +1 -1
- package/dist/cli/commands/update.js +3 -4
- package/dist/cli/commands/update.js.map +1 -1
- package/dist/cli/detect.js +32 -32
- package/dist/cli/index.js +51 -55
- package/dist/cli/index.js.map +1 -1
- package/dist/cli/llm/claude-commands.js +16 -16
- package/dist/cli/llm/config.js +20 -20
- package/dist/cli/llm/config.js.map +1 -1
- package/dist/cli/llm/gemini-commands.d.ts +4 -6
- package/dist/cli/llm/gemini-commands.d.ts.map +1 -1
- package/dist/cli/llm/gemini-commands.js +52 -322
- package/dist/cli/llm/gemini-commands.js.map +1 -1
- package/dist/cli/llm/gpt-commands.js +21 -21
- package/dist/cli/llm/gpt-commands.js.map +1 -1
- package/dist/cli/llm/help.js +21 -21
- package/dist/cli/postinstall/constants.js +1 -1
- package/dist/cli/postinstall/constants.js.map +1 -1
- package/dist/cli/postinstall/cursor-agents.js +32 -32
- package/dist/cli/postinstall/cursor-rules.js +83 -83
- package/dist/cli/postinstall/cursor-skills.js +743 -743
- package/dist/cli/postinstall/inline-skills.js +1 -1
- package/dist/cli/postinstall/inline-skills.js.map +1 -1
- package/dist/cli/setup/Provisioner.js +42 -42
- package/dist/cli/types.d.ts +2 -18
- package/dist/cli/types.d.ts.map +1 -1
- package/dist/cli/utils.d.ts +0 -9
- package/dist/cli/utils.d.ts.map +1 -1
- package/dist/cli/utils.js +0 -28
- package/dist/cli/utils.js.map +1 -1
- package/dist/infra/lib/DeepInit.js +24 -24
- package/dist/infra/lib/IterationTracker.js +11 -11
- package/dist/infra/lib/PythonParser.js +108 -108
- package/dist/infra/lib/ReviewRace.js +96 -96
- package/dist/infra/lib/SkillFrontmatter.js +28 -28
- package/dist/infra/lib/SkillQualityGate.js +9 -9
- package/dist/infra/lib/SkillRepository.js +159 -159
- package/dist/infra/lib/UltraQA.js +99 -99
- package/dist/infra/lib/autonomy/AuditStore.js +41 -41
- package/dist/infra/lib/autonomy/ConfirmationStore.js +30 -30
- package/dist/infra/lib/autonomy/EventOutbox.js +38 -38
- package/dist/infra/lib/autonomy/PolicyEngine.js +18 -18
- package/dist/infra/lib/autonomy/SecuritySentinel.js +1 -1
- package/dist/infra/lib/autonomy/SuggestionStore.js +33 -33
- package/dist/infra/lib/config/GlobalConfigManager.d.ts +0 -2
- package/dist/infra/lib/config/GlobalConfigManager.d.ts.map +1 -1
- package/dist/infra/lib/config/GlobalConfigManager.js +0 -27
- package/dist/infra/lib/config/GlobalConfigManager.js.map +1 -1
- package/dist/infra/lib/embedding/VectorStore.js +22 -22
- package/dist/infra/lib/evolution/AgentAnalyzer.js +10 -10
- package/dist/infra/lib/evolution/DescriptionOptimizer.js +21 -21
- package/dist/infra/lib/evolution/GenerationRegistry.js +36 -36
- package/dist/infra/lib/evolution/InsightStore.js +90 -90
- package/dist/infra/lib/evolution/RollbackManager.js +5 -5
- package/dist/infra/lib/evolution/SkillBenchmark.js +23 -23
- package/dist/infra/lib/evolution/SkillEvalRunner.js +50 -50
- package/dist/infra/lib/evolution/SkillGapDetector.js +10 -10
- package/dist/infra/lib/evolution/UsageTracker.js +28 -28
- package/dist/infra/lib/gemini/auth.d.ts +4 -16
- package/dist/infra/lib/gemini/auth.d.ts.map +1 -1
- package/dist/infra/lib/gemini/auth.js +10 -405
- package/dist/infra/lib/gemini/auth.js.map +1 -1
- package/dist/infra/lib/gemini/capabilities.d.ts +4 -8
- package/dist/infra/lib/gemini/capabilities.d.ts.map +1 -1
- package/dist/infra/lib/gemini/capabilities.js +8 -166
- package/dist/infra/lib/gemini/capabilities.js.map +1 -1
- package/dist/infra/lib/gemini/chat.d.ts +4 -13
- package/dist/infra/lib/gemini/chat.d.ts.map +1 -1
- package/dist/infra/lib/gemini/chat.js +10 -323
- package/dist/infra/lib/gemini/chat.js.map +1 -1
- package/dist/infra/lib/gemini/completion.d.ts +5 -15
- package/dist/infra/lib/gemini/completion.d.ts.map +1 -1
- package/dist/infra/lib/gemini/completion.js +6 -97
- package/dist/infra/lib/gemini/completion.js.map +1 -1
- package/dist/infra/lib/gemini/constants.d.ts +2 -31
- package/dist/infra/lib/gemini/constants.d.ts.map +1 -1
- package/dist/infra/lib/gemini/constants.js +2 -77
- package/dist/infra/lib/gemini/constants.js.map +1 -1
- package/dist/infra/lib/gemini/index.d.ts +5 -8
- package/dist/infra/lib/gemini/index.d.ts.map +1 -1
- package/dist/infra/lib/gemini/index.js +4 -7
- package/dist/infra/lib/gemini/index.js.map +1 -1
- package/dist/infra/lib/gemini/models.d.ts +3 -4
- package/dist/infra/lib/gemini/models.d.ts.map +1 -1
- package/dist/infra/lib/gemini/models.js +8 -84
- package/dist/infra/lib/gemini/models.js.map +1 -1
- package/dist/infra/lib/gemini/orchestration.js +5 -5
- package/dist/infra/lib/gemini/types.d.ts +16 -44
- package/dist/infra/lib/gemini/types.d.ts.map +1 -1
- package/dist/infra/lib/gemini/types.js +1 -1
- package/dist/infra/lib/gpt/auth.d.ts +2 -5
- package/dist/infra/lib/gpt/auth.d.ts.map +1 -1
- package/dist/infra/lib/gpt/auth.js +8 -38
- package/dist/infra/lib/gpt/auth.js.map +1 -1
- package/dist/infra/lib/gpt/chat.d.ts +3 -3
- package/dist/infra/lib/gpt/chat.d.ts.map +1 -1
- package/dist/infra/lib/gpt/chat.js +37 -53
- package/dist/infra/lib/gpt/chat.js.map +1 -1
- package/dist/infra/lib/gpt/constants.d.ts +2 -5
- package/dist/infra/lib/gpt/constants.d.ts.map +1 -1
- package/dist/infra/lib/gpt/constants.js +4 -9
- package/dist/infra/lib/gpt/constants.js.map +1 -1
- package/dist/infra/lib/gpt/embedding.d.ts +1 -1
- package/dist/infra/lib/gpt/embedding.js +3 -3
- package/dist/infra/lib/gpt/embedding.js.map +1 -1
- package/dist/infra/lib/gpt/oauth.d.ts +6 -39
- package/dist/infra/lib/gpt/oauth.d.ts.map +1 -1
- package/dist/infra/lib/gpt/oauth.js +8 -340
- package/dist/infra/lib/gpt/oauth.js.map +1 -1
- package/dist/infra/lib/gpt/orchestration.js +5 -5
- package/dist/infra/lib/gpt/orchestration.js.map +1 -1
- package/dist/infra/lib/gpt/specializations.d.ts +2 -2
- package/dist/infra/lib/gpt/specializations.js +3 -3
- package/dist/infra/lib/gpt/specializations.js.map +1 -1
- package/dist/infra/lib/gpt/types.d.ts +1 -1
- package/dist/infra/lib/gpt/types.d.ts.map +1 -1
- package/dist/infra/lib/llm/auth/AuthProfileManager.d.ts +2 -2
- package/dist/infra/lib/llm/auth/AuthProfileManager.d.ts.map +1 -1
- package/dist/infra/lib/llm/auth/AuthProfileManager.js.map +1 -1
- package/dist/infra/lib/llm/auth/AuthProfileManager.test.js +1 -1
- package/dist/infra/lib/llm/auth/AuthProfileManager.test.js.map +1 -1
- package/dist/infra/lib/llm/auth/TokenRefresher.d.ts +1 -1
- package/dist/infra/lib/llm/auth/TokenRefresher.js +1 -1
- package/dist/infra/lib/llm/auth/index.d.ts +2 -12
- package/dist/infra/lib/llm/auth/index.d.ts.map +1 -1
- package/dist/infra/lib/llm/auth/index.js +5 -63
- package/dist/infra/lib/llm/auth/index.js.map +1 -1
- package/dist/infra/lib/llm/types.d.ts +1 -1
- package/dist/infra/lib/llm/types.d.ts.map +1 -1
- package/dist/infra/lib/memory/KnowledgeGraph.js +4 -4
- package/dist/infra/lib/memory/MemorySearch.js +57 -57
- package/dist/infra/lib/memory/MemoryStorage.js +181 -181
- package/dist/infra/lib/memory/ObservationStore.js +28 -28
- package/dist/infra/lib/memory/ReflectionStore.js +30 -30
- package/dist/infra/lib/memory/SessionRAGRetriever.js +7 -7
- package/dist/infra/lib/memory/SessionRAGStore.js +225 -225
- package/dist/infra/lib/memory/SessionSummarizer.js +9 -9
- package/dist/infra/orchestrator/AgentManager.js +12 -12
- package/dist/infra/orchestrator/AgentRegistry.js +65 -65
- package/dist/infra/orchestrator/MultiLlmResearch.js +8 -8
- package/dist/infra/orchestrator/SwarmOrchestrator.test.js +16 -16
- package/dist/infra/orchestrator/parallelResearch.js +24 -24
- package/dist/tools/convention/analyzeComplexity.test.js +115 -115
- package/dist/tools/convention/validateCodeQuality.test.js +104 -104
- package/dist/tools/memory/createMemoryTimeline.js +10 -10
- package/dist/tools/memory/getMemoryGraph.js +12 -12
- package/dist/tools/memory/getSessionContext.js +9 -9
- package/dist/tools/memory/linkMemories.js +14 -14
- package/dist/tools/memory/listMemories.js +4 -4
- package/dist/tools/memory/recallMemory.js +4 -4
- package/dist/tools/memory/saveMemory.js +4 -4
- package/dist/tools/memory/searchMemoriesAdvanced.js +23 -23
- package/dist/tools/semantic/analyzeDependencyGraph.js +12 -12
- package/dist/tools/semantic/astGrep.test.js +6 -6
- package/dist/tools/spec/prdParser.test.js +171 -171
- package/dist/tools/spec/specGenerator.js +169 -169
- package/dist/tools/spec/traceabilityMatrix.js +64 -64
- package/dist/tools/spec/traceabilityMatrix.test.js +28 -28
- package/hooks/gemini-hooks.json +73 -73
- package/hooks/hooks.json +137 -137
- package/hooks/scripts/code-check.js +70 -70
- package/hooks/scripts/context-save.js +212 -212
- package/hooks/scripts/hud-status.js +291 -291
- package/hooks/scripts/keyword-detector.js +214 -214
- package/hooks/scripts/llm-orchestrate.js +475 -646
- package/hooks/scripts/post-edit.js +32 -32
- package/hooks/scripts/pre-tool-guard.js +125 -125
- package/hooks/scripts/prompt-dispatcher.js +185 -185
- package/hooks/scripts/sentinel-guard.js +104 -104
- package/hooks/scripts/session-start.js +106 -106
- package/hooks/scripts/stop-notify.js +209 -209
- package/hooks/scripts/utils.js +100 -100
- package/languages/csharp-unity.md +515 -515
- package/languages/gdscript-godot.md +470 -470
- package/languages/ruby-rails.md +489 -489
- package/languages/typescript-angular.md +433 -433
- package/languages/typescript-astro.md +416 -416
- package/languages/typescript-electron.md +406 -406
- package/languages/typescript-nestjs.md +524 -524
- package/languages/typescript-svelte.md +407 -407
- package/languages/typescript-tauri.md +365 -365
- package/package.json +121 -121
- package/skills/agents-md/SKILL.md +120 -120
- package/skills/arch-guard/SKILL.md +180 -180
- package/skills/brand-assets/SKILL.md +146 -146
- package/skills/capability-loop/SKILL.md +167 -167
- package/skills/characterization-test/SKILL.md +206 -206
- package/skills/commerce-patterns/SKILL.md +59 -59
- package/skills/commit-push-pr/SKILL.md +75 -75
- package/skills/context7-usage/SKILL.md +105 -105
- package/skills/core-capabilities/SKILL.md +48 -48
- package/skills/e2e-commerce/SKILL.md +57 -57
- package/skills/exec-plan/SKILL.md +147 -147
- package/skills/frontend-design/SKILL.md +73 -73
- package/skills/git-worktree/SKILL.md +72 -72
- package/skills/handoff/SKILL.md +109 -109
- package/skills/parallel-research/SKILL.md +87 -87
- package/skills/priority-todos/SKILL.md +63 -63
- package/skills/seo-checklist/SKILL.md +57 -57
- package/skills/techdebt/SKILL.md +122 -122
- package/skills/tool-fallback/SKILL.md +103 -103
- package/skills/typescript-advanced-types/SKILL.md +66 -65
- package/skills/ui-ux-pro-max/SKILL.md +206 -206
- package/skills/vercel-react-best-practices/SKILL.md +59 -59
- package/skills/video-production/SKILL.md +51 -51
- package/vibe/config.json +29 -29
- package/vibe/constitution.md +227 -227
- package/vibe/rules/principles/communication-guide.md +98 -98
- package/vibe/rules/principles/development-philosophy.md +52 -52
- package/vibe/rules/principles/quick-start.md +102 -102
- package/vibe/rules/quality/bdd-contract-testing.md +393 -393
- package/vibe/rules/quality/checklist.md +276 -276
- package/vibe/rules/quality/performance.md +236 -236
- package/vibe/rules/quality/testing-strategy.md +440 -440
- package/vibe/rules/standards/anti-patterns.md +541 -541
- package/vibe/rules/standards/code-structure.md +291 -291
- package/vibe/rules/standards/complexity-metrics.md +313 -313
- package/vibe/rules/standards/git-workflow.md +237 -237
- package/vibe/rules/standards/naming-conventions.md +198 -198
- package/vibe/rules/standards/security.md +305 -305
- package/vibe/rules/writing/document-style.md +74 -74
- package/vibe/setup.sh +31 -31
- package/vibe/templates/constitution-template.md +252 -252
- package/vibe/templates/contract-backend-template.md +526 -526
- package/vibe/templates/contract-frontend-template.md +599 -599
- package/vibe/templates/feature-template.md +96 -96
- package/vibe/templates/spec-template.md +221 -221
- package/vibe/ui-ux-data/charts.csv +26 -26
- package/vibe/ui-ux-data/colors.csv +97 -97
- package/vibe/ui-ux-data/icons.csv +101 -101
- package/vibe/ui-ux-data/landing.csv +31 -31
- package/vibe/ui-ux-data/products.csv +96 -96
- package/vibe/ui-ux-data/react-performance.csv +45 -45
- package/vibe/ui-ux-data/stacks/astro.csv +54 -54
- package/vibe/ui-ux-data/stacks/flutter.csv +53 -53
- package/vibe/ui-ux-data/stacks/html-tailwind.csv +56 -56
- package/vibe/ui-ux-data/stacks/jetpack-compose.csv +53 -53
- package/vibe/ui-ux-data/stacks/nextjs.csv +53 -53
- package/vibe/ui-ux-data/stacks/nuxt-ui.csv +51 -51
- package/vibe/ui-ux-data/stacks/nuxtjs.csv +59 -59
- package/vibe/ui-ux-data/stacks/react-native.csv +52 -52
- package/vibe/ui-ux-data/stacks/react.csv +54 -54
- package/vibe/ui-ux-data/stacks/shadcn.csv +61 -61
- package/vibe/ui-ux-data/stacks/svelte.csv +54 -54
- package/vibe/ui-ux-data/stacks/swiftui.csv +51 -51
- package/vibe/ui-ux-data/stacks/vue.csv +50 -50
- package/vibe/ui-ux-data/styles.csv +68 -68
- package/vibe/ui-ux-data/typography.csv +57 -57
- package/vibe/ui-ux-data/ui-reasoning.csv +101 -101
- package/vibe/ui-ux-data/ux-guidelines.csv +99 -99
- package/vibe/ui-ux-data/version.json +31 -31
- package/vibe/ui-ux-data/web-interface.csv +31 -31
- package/commands/vibe.voice.md +0 -79
|
@@ -1,185 +1,185 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
/**
|
|
3
|
-
* UserPromptSubmit 디스패처
|
|
4
|
-
*
|
|
5
|
-
* UserPromptSubmit은 matcher를 지원하지 않아 모든 hook이 매번 실행됨.
|
|
6
|
-
* 이 디스패처가 stdin에서 prompt를 읽고, 패턴 매칭 후 해당 스크립트만 실행.
|
|
7
|
-
*
|
|
8
|
-
* 이점:
|
|
9
|
-
* - 외부 LLM 호출(GPT/Gemini)이 패턴 매칭 없이 발동하지 않음
|
|
10
|
-
* - context window에 불필요한 응답이 주입되지 않음
|
|
11
|
-
* - 단일 프로세스에서 매칭 후 필요한 스크립트만 fork
|
|
12
|
-
*/
|
|
13
|
-
import { execFile } from 'child_process';
|
|
14
|
-
import { fileURLToPath } from 'url';
|
|
15
|
-
import path from 'path';
|
|
16
|
-
|
|
17
|
-
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
|
18
|
-
|
|
19
|
-
// stdin에서 prompt 읽기
|
|
20
|
-
let inputData = '';
|
|
21
|
-
for await (const chunk of process.stdin) {
|
|
22
|
-
inputData += chunk;
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
let prompt = '';
|
|
26
|
-
try {
|
|
27
|
-
const parsed = JSON.parse(inputData);
|
|
28
|
-
prompt = parsed.prompt || '';
|
|
29
|
-
} catch {
|
|
30
|
-
process.exit(0);
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
if (!prompt) process.exit(0);
|
|
34
|
-
|
|
35
|
-
// 패턴 → 실행할 스크립트 매핑
|
|
36
|
-
// 각 항목: { pattern, script, args, label }
|
|
37
|
-
const DISPATCH_RULES = [
|
|
38
|
-
// 항상 실행 (경량 스크립트)
|
|
39
|
-
{
|
|
40
|
-
pattern: null, // always
|
|
41
|
-
script: 'keyword-detector.js',
|
|
42
|
-
args: [prompt],
|
|
43
|
-
label: 'keyword',
|
|
44
|
-
},
|
|
45
|
-
|
|
46
|
-
// 패턴 매칭이 필요한 스크립트
|
|
47
|
-
{
|
|
48
|
-
pattern: /ultrawork|ulw|울트라워크|ralph|ralplan/i,
|
|
49
|
-
script: null, // keyword-detector가 이미 처리
|
|
50
|
-
label: 'skip',
|
|
51
|
-
},
|
|
52
|
-
// echo 전용 (stdout으로 직접 출력)
|
|
53
|
-
{
|
|
54
|
-
pattern: /e2e.*테스트|e2e.*test|playwright|브라우저.*테스트|browser.*test/i,
|
|
55
|
-
script: null,
|
|
56
|
-
echo: '[E2E MODE] Use /vibe.utils --e2e for Playwright-based browser testing. Supports visual regression and video recording.',
|
|
57
|
-
label: 'e2e-echo',
|
|
58
|
-
},
|
|
59
|
-
|
|
60
|
-
// 외부 LLM 호출 (GPT/Gemini) - 패턴 매칭 필수
|
|
61
|
-
{
|
|
62
|
-
pattern: /아키텍처.*(검토|리뷰|분석)|architecture.*(review|analyz)|설계.*검토|구조.*분석.*해/i,
|
|
63
|
-
script: 'llm-orchestrate.js',
|
|
64
|
-
args: ['gpt', 'orchestrate', 'You are a software architect. Analyze and review the architecture.'],
|
|
65
|
-
label: 'gpt-architecture',
|
|
66
|
-
},
|
|
67
|
-
{
|
|
68
|
-
pattern: /(UI|UX).*(리뷰|검토|피드백|개선)|사용자.*경험.*검토|디자인.*리뷰|design.*feedback/i,
|
|
69
|
-
script: 'llm-orchestrate.js',
|
|
70
|
-
args: ['gemini', 'orchestrate', 'You are a UI/UX expert. Analyze and provide feedback.'],
|
|
71
|
-
label: 'gemini-uiux',
|
|
72
|
-
},
|
|
73
|
-
{
|
|
74
|
-
pattern: /디버깅.*해|버그.*찾아|find.*bug|debug.*this.*code/i,
|
|
75
|
-
script: 'llm-orchestrate.js',
|
|
76
|
-
args: ['gpt', 'orchestrate', 'You are a debugging expert. Find bugs and suggest fixes.'],
|
|
77
|
-
label: 'gpt-debug',
|
|
78
|
-
},
|
|
79
|
-
{
|
|
80
|
-
pattern: /코드.*정적.*분석|코드.*분석.*해줘|analyze.*code.*quality/i,
|
|
81
|
-
script: 'llm-orchestrate.js',
|
|
82
|
-
args: ['gemini', 'orchestrate', 'You are a code analysis expert. Review and analyze the code.'],
|
|
83
|
-
label: 'gemini-analysis',
|
|
84
|
-
},
|
|
85
|
-
{
|
|
86
|
-
pattern: /코드.*리뷰|code.*review|PR.*리뷰|리뷰.*해줘.*코드/i,
|
|
87
|
-
script: 'llm-orchestrate.js',
|
|
88
|
-
args: ['gpt', 'orchestrate', 'You are a code review expert. Review the code for best practices, security, and performance.'],
|
|
89
|
-
label: 'gpt-codereview',
|
|
90
|
-
},
|
|
91
|
-
{
|
|
92
|
-
pattern: /추론.*해|reasoning|복잡.*분석|deep.*analysis/i,
|
|
93
|
-
script: 'llm-orchestrate.js',
|
|
94
|
-
args: ['gpt', 'orchestrate', 'You are a reasoning expert. Analyze the problem deeply and provide detailed reasoning.'],
|
|
95
|
-
label: 'gpt-reasoning',
|
|
96
|
-
},
|
|
97
|
-
|
|
98
|
-
// 테스트용
|
|
99
|
-
{
|
|
100
|
-
pattern: /^test-gpt/i,
|
|
101
|
-
script: 'llm-orchestrate.js',
|
|
102
|
-
args: ['gpt', 'orchestrate', 'You are a helpful assistant. Answer the user\'s question clearly and concisely.'],
|
|
103
|
-
label: 'test-gpt',
|
|
104
|
-
},
|
|
105
|
-
{
|
|
106
|
-
pattern: /^test-gemini/i,
|
|
107
|
-
script: 'llm-orchestrate.js',
|
|
108
|
-
args: ['gemini', 'orchestrate', 'You are a helpful assistant. Answer the user\'s question clearly and concisely.'],
|
|
109
|
-
label: 'test-gemini',
|
|
110
|
-
},
|
|
111
|
-
];
|
|
112
|
-
|
|
113
|
-
// 매칭된 스크립트 실행
|
|
114
|
-
const execPromises = [];
|
|
115
|
-
|
|
116
|
-
for (const rule of DISPATCH_RULES) {
|
|
117
|
-
if (rule.label === 'skip') continue;
|
|
118
|
-
|
|
119
|
-
// pattern이 null이면 항상 실행, 아니면 매칭 확인
|
|
120
|
-
if (rule.pattern !== null && !rule.pattern.test(prompt)) continue;
|
|
121
|
-
|
|
122
|
-
// echo 규칙: 직접 stdout 출력
|
|
123
|
-
if (rule.echo) {
|
|
124
|
-
process.stdout.write(rule.echo + '\n');
|
|
125
|
-
continue;
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
if (!rule.script) continue;
|
|
129
|
-
|
|
130
|
-
const scriptPath = path.join(__dirname, rule.script);
|
|
131
|
-
const args = rule.args || [];
|
|
132
|
-
|
|
133
|
-
execPromises.push(
|
|
134
|
-
new Promise((resolve) => {
|
|
135
|
-
execFile('node', [scriptPath, ...args], {
|
|
136
|
-
timeout: 30000,
|
|
137
|
-
env: { ...process.env },
|
|
138
|
-
}, (error, stdout, stderr) => {
|
|
139
|
-
if (stdout?.trim()) {
|
|
140
|
-
process.stdout.write(stdout);
|
|
141
|
-
}
|
|
142
|
-
resolve();
|
|
143
|
-
});
|
|
144
|
-
})
|
|
145
|
-
);
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
await Promise.all(execPromises);
|
|
149
|
-
|
|
150
|
-
// Evolution: Gap detection — log unmatched prompts for skill gap analysis
|
|
151
|
-
const matched = DISPATCH_RULES.some(r =>
|
|
152
|
-
r.label !== 'skip' && r.label !== 'keyword' &&
|
|
153
|
-
r.pattern !== null && r.pattern.test(prompt) &&
|
|
154
|
-
(r.script || r.echo)
|
|
155
|
-
);
|
|
156
|
-
|
|
157
|
-
if (!matched) {
|
|
158
|
-
setImmediate(async () => {
|
|
159
|
-
try {
|
|
160
|
-
const configPath = path.join(process.env.CLAUDE_PROJECT_DIR || '.', '.claude', 'vibe', 'config.json');
|
|
161
|
-
let gapEnabled = true;
|
|
162
|
-
try {
|
|
163
|
-
const fs = await import('fs');
|
|
164
|
-
if (fs.existsSync(configPath)) {
|
|
165
|
-
const config = JSON.parse(fs.readFileSync(configPath, 'utf8'));
|
|
166
|
-
gapEnabled = config.evolution?.gapDetection !== false && config.evolution?.enabled !== false;
|
|
167
|
-
}
|
|
168
|
-
} catch { /* ignore */ }
|
|
169
|
-
|
|
170
|
-
if (gapEnabled) {
|
|
171
|
-
const LIB_BASE = (await import('./utils.js')).getLibBaseUrl();
|
|
172
|
-
const [memMod, gapMod] = await Promise.all([
|
|
173
|
-
import(`${LIB_BASE}memory/MemoryStorage.js`),
|
|
174
|
-
import(`${LIB_BASE}evolution/SkillGapDetector.js`),
|
|
175
|
-
]);
|
|
176
|
-
const storage = new memMod.MemoryStorage(process.env.CLAUDE_PROJECT_DIR || '.');
|
|
177
|
-
const detector = new gapMod.SkillGapDetector(storage);
|
|
178
|
-
detector.logMiss(prompt.slice(0, 200));
|
|
179
|
-
storage.close();
|
|
180
|
-
}
|
|
181
|
-
} catch (e) {
|
|
182
|
-
process.stderr.write(`[Evolution] Gap log error: ${e.message}\n`);
|
|
183
|
-
}
|
|
184
|
-
});
|
|
185
|
-
}
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* UserPromptSubmit 디스패처
|
|
4
|
+
*
|
|
5
|
+
* UserPromptSubmit은 matcher를 지원하지 않아 모든 hook이 매번 실행됨.
|
|
6
|
+
* 이 디스패처가 stdin에서 prompt를 읽고, 패턴 매칭 후 해당 스크립트만 실행.
|
|
7
|
+
*
|
|
8
|
+
* 이점:
|
|
9
|
+
* - 외부 LLM 호출(GPT/Gemini)이 패턴 매칭 없이 발동하지 않음
|
|
10
|
+
* - context window에 불필요한 응답이 주입되지 않음
|
|
11
|
+
* - 단일 프로세스에서 매칭 후 필요한 스크립트만 fork
|
|
12
|
+
*/
|
|
13
|
+
import { execFile } from 'child_process';
|
|
14
|
+
import { fileURLToPath } from 'url';
|
|
15
|
+
import path from 'path';
|
|
16
|
+
|
|
17
|
+
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
|
18
|
+
|
|
19
|
+
// stdin에서 prompt 읽기
|
|
20
|
+
let inputData = '';
|
|
21
|
+
for await (const chunk of process.stdin) {
|
|
22
|
+
inputData += chunk;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
let prompt = '';
|
|
26
|
+
try {
|
|
27
|
+
const parsed = JSON.parse(inputData);
|
|
28
|
+
prompt = parsed.prompt || '';
|
|
29
|
+
} catch {
|
|
30
|
+
process.exit(0);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
if (!prompt) process.exit(0);
|
|
34
|
+
|
|
35
|
+
// 패턴 → 실행할 스크립트 매핑
|
|
36
|
+
// 각 항목: { pattern, script, args, label }
|
|
37
|
+
const DISPATCH_RULES = [
|
|
38
|
+
// 항상 실행 (경량 스크립트)
|
|
39
|
+
{
|
|
40
|
+
pattern: null, // always
|
|
41
|
+
script: 'keyword-detector.js',
|
|
42
|
+
args: [prompt],
|
|
43
|
+
label: 'keyword',
|
|
44
|
+
},
|
|
45
|
+
|
|
46
|
+
// 패턴 매칭이 필요한 스크립트
|
|
47
|
+
{
|
|
48
|
+
pattern: /ultrawork|ulw|울트라워크|ralph|ralplan/i,
|
|
49
|
+
script: null, // keyword-detector가 이미 처리
|
|
50
|
+
label: 'skip',
|
|
51
|
+
},
|
|
52
|
+
// echo 전용 (stdout으로 직접 출력)
|
|
53
|
+
{
|
|
54
|
+
pattern: /e2e.*테스트|e2e.*test|playwright|브라우저.*테스트|browser.*test/i,
|
|
55
|
+
script: null,
|
|
56
|
+
echo: '[E2E MODE] Use /vibe.utils --e2e for Playwright-based browser testing. Supports visual regression and video recording.',
|
|
57
|
+
label: 'e2e-echo',
|
|
58
|
+
},
|
|
59
|
+
|
|
60
|
+
// 외부 LLM 호출 (GPT/Gemini) - 패턴 매칭 필수
|
|
61
|
+
{
|
|
62
|
+
pattern: /아키텍처.*(검토|리뷰|분석)|architecture.*(review|analyz)|설계.*검토|구조.*분석.*해/i,
|
|
63
|
+
script: 'llm-orchestrate.js',
|
|
64
|
+
args: ['gpt', 'orchestrate', 'You are a software architect. Analyze and review the architecture.'],
|
|
65
|
+
label: 'gpt-architecture',
|
|
66
|
+
},
|
|
67
|
+
{
|
|
68
|
+
pattern: /(UI|UX).*(리뷰|검토|피드백|개선)|사용자.*경험.*검토|디자인.*리뷰|design.*feedback/i,
|
|
69
|
+
script: 'llm-orchestrate.js',
|
|
70
|
+
args: ['gemini', 'orchestrate', 'You are a UI/UX expert. Analyze and provide feedback.'],
|
|
71
|
+
label: 'gemini-uiux',
|
|
72
|
+
},
|
|
73
|
+
{
|
|
74
|
+
pattern: /디버깅.*해|버그.*찾아|find.*bug|debug.*this.*code/i,
|
|
75
|
+
script: 'llm-orchestrate.js',
|
|
76
|
+
args: ['gpt', 'orchestrate', 'You are a debugging expert. Find bugs and suggest fixes.'],
|
|
77
|
+
label: 'gpt-debug',
|
|
78
|
+
},
|
|
79
|
+
{
|
|
80
|
+
pattern: /코드.*정적.*분석|코드.*분석.*해줘|analyze.*code.*quality/i,
|
|
81
|
+
script: 'llm-orchestrate.js',
|
|
82
|
+
args: ['gemini', 'orchestrate', 'You are a code analysis expert. Review and analyze the code.'],
|
|
83
|
+
label: 'gemini-analysis',
|
|
84
|
+
},
|
|
85
|
+
{
|
|
86
|
+
pattern: /코드.*리뷰|code.*review|PR.*리뷰|리뷰.*해줘.*코드/i,
|
|
87
|
+
script: 'llm-orchestrate.js',
|
|
88
|
+
args: ['gpt', 'orchestrate', 'You are a code review expert. Review the code for best practices, security, and performance.'],
|
|
89
|
+
label: 'gpt-codereview',
|
|
90
|
+
},
|
|
91
|
+
{
|
|
92
|
+
pattern: /추론.*해|reasoning|복잡.*분석|deep.*analysis/i,
|
|
93
|
+
script: 'llm-orchestrate.js',
|
|
94
|
+
args: ['gpt', 'orchestrate', 'You are a reasoning expert. Analyze the problem deeply and provide detailed reasoning.'],
|
|
95
|
+
label: 'gpt-reasoning',
|
|
96
|
+
},
|
|
97
|
+
|
|
98
|
+
// 테스트용
|
|
99
|
+
{
|
|
100
|
+
pattern: /^test-gpt/i,
|
|
101
|
+
script: 'llm-orchestrate.js',
|
|
102
|
+
args: ['gpt', 'orchestrate', 'You are a helpful assistant. Answer the user\'s question clearly and concisely.'],
|
|
103
|
+
label: 'test-gpt',
|
|
104
|
+
},
|
|
105
|
+
{
|
|
106
|
+
pattern: /^test-gemini/i,
|
|
107
|
+
script: 'llm-orchestrate.js',
|
|
108
|
+
args: ['gemini', 'orchestrate', 'You are a helpful assistant. Answer the user\'s question clearly and concisely.'],
|
|
109
|
+
label: 'test-gemini',
|
|
110
|
+
},
|
|
111
|
+
];
|
|
112
|
+
|
|
113
|
+
// 매칭된 스크립트 실행
|
|
114
|
+
const execPromises = [];
|
|
115
|
+
|
|
116
|
+
for (const rule of DISPATCH_RULES) {
|
|
117
|
+
if (rule.label === 'skip') continue;
|
|
118
|
+
|
|
119
|
+
// pattern이 null이면 항상 실행, 아니면 매칭 확인
|
|
120
|
+
if (rule.pattern !== null && !rule.pattern.test(prompt)) continue;
|
|
121
|
+
|
|
122
|
+
// echo 규칙: 직접 stdout 출력
|
|
123
|
+
if (rule.echo) {
|
|
124
|
+
process.stdout.write(rule.echo + '\n');
|
|
125
|
+
continue;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
if (!rule.script) continue;
|
|
129
|
+
|
|
130
|
+
const scriptPath = path.join(__dirname, rule.script);
|
|
131
|
+
const args = rule.args || [];
|
|
132
|
+
|
|
133
|
+
execPromises.push(
|
|
134
|
+
new Promise((resolve) => {
|
|
135
|
+
execFile('node', [scriptPath, ...args], {
|
|
136
|
+
timeout: 30000,
|
|
137
|
+
env: { ...process.env },
|
|
138
|
+
}, (error, stdout, stderr) => {
|
|
139
|
+
if (stdout?.trim()) {
|
|
140
|
+
process.stdout.write(stdout);
|
|
141
|
+
}
|
|
142
|
+
resolve();
|
|
143
|
+
});
|
|
144
|
+
})
|
|
145
|
+
);
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
await Promise.all(execPromises);
|
|
149
|
+
|
|
150
|
+
// Evolution: Gap detection — log unmatched prompts for skill gap analysis
|
|
151
|
+
const matched = DISPATCH_RULES.some(r =>
|
|
152
|
+
r.label !== 'skip' && r.label !== 'keyword' &&
|
|
153
|
+
r.pattern !== null && r.pattern.test(prompt) &&
|
|
154
|
+
(r.script || r.echo)
|
|
155
|
+
);
|
|
156
|
+
|
|
157
|
+
if (!matched) {
|
|
158
|
+
setImmediate(async () => {
|
|
159
|
+
try {
|
|
160
|
+
const configPath = path.join(process.env.CLAUDE_PROJECT_DIR || '.', '.claude', 'vibe', 'config.json');
|
|
161
|
+
let gapEnabled = true;
|
|
162
|
+
try {
|
|
163
|
+
const fs = await import('fs');
|
|
164
|
+
if (fs.existsSync(configPath)) {
|
|
165
|
+
const config = JSON.parse(fs.readFileSync(configPath, 'utf8'));
|
|
166
|
+
gapEnabled = config.evolution?.gapDetection !== false && config.evolution?.enabled !== false;
|
|
167
|
+
}
|
|
168
|
+
} catch { /* ignore */ }
|
|
169
|
+
|
|
170
|
+
if (gapEnabled) {
|
|
171
|
+
const LIB_BASE = (await import('./utils.js')).getLibBaseUrl();
|
|
172
|
+
const [memMod, gapMod] = await Promise.all([
|
|
173
|
+
import(`${LIB_BASE}memory/MemoryStorage.js`),
|
|
174
|
+
import(`${LIB_BASE}evolution/SkillGapDetector.js`),
|
|
175
|
+
]);
|
|
176
|
+
const storage = new memMod.MemoryStorage(process.env.CLAUDE_PROJECT_DIR || '.');
|
|
177
|
+
const detector = new gapMod.SkillGapDetector(storage);
|
|
178
|
+
detector.logMiss(prompt.slice(0, 200));
|
|
179
|
+
storage.close();
|
|
180
|
+
}
|
|
181
|
+
} catch (e) {
|
|
182
|
+
process.stderr.write(`[Evolution] Gap log error: ${e.message}\n`);
|
|
183
|
+
}
|
|
184
|
+
});
|
|
185
|
+
}
|
|
@@ -1,104 +1,104 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
/**
|
|
3
|
-
* Sentinel Guard — PreToolUse hook
|
|
4
|
-
* Protects sentinel files and blocks dangerous operations.
|
|
5
|
-
* Runs before pre-tool-guard.js.
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
const SENTINEL_PATH_PREFIX = 'src/infra/lib/autonomy/';
|
|
9
|
-
|
|
10
|
-
const SENTINEL_PATH_RE = /^(\.\/|\.\\)?src[\\/]infra[\\/]lib[\\/]autonomy[\\/]/;
|
|
11
|
-
|
|
12
|
-
const DANGEROUS_BASH_RE =
|
|
13
|
-
/\b(rm\s+-rf|kill\s+-9|drop\s+table|truncate|shutdown|reboot|mkfs|dd\s+if=)\b/i;
|
|
14
|
-
|
|
15
|
-
/**
|
|
16
|
-
* Extract file path from tool input
|
|
17
|
-
*/
|
|
18
|
-
function extractFilePath(toolName, input) {
|
|
19
|
-
if (!input) return null;
|
|
20
|
-
try {
|
|
21
|
-
const parsed = typeof input === 'string' ? JSON.parse(input) : input;
|
|
22
|
-
if (toolName === 'Write' || toolName === 'Edit' || toolName === 'Read') {
|
|
23
|
-
return parsed.file_path || parsed.filePath || null;
|
|
24
|
-
}
|
|
25
|
-
} catch {
|
|
26
|
-
// Not JSON, try as plain string
|
|
27
|
-
return typeof input === 'string' ? input : null;
|
|
28
|
-
}
|
|
29
|
-
return null;
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
/**
|
|
33
|
-
* Extract command from Bash tool input
|
|
34
|
-
*/
|
|
35
|
-
function extractBashCommand(input) {
|
|
36
|
-
if (!input) return null;
|
|
37
|
-
try {
|
|
38
|
-
const parsed = typeof input === 'string' ? JSON.parse(input) : input;
|
|
39
|
-
return parsed.command || null;
|
|
40
|
-
} catch {
|
|
41
|
-
return typeof input === 'string' ? input : null;
|
|
42
|
-
}
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
/**
|
|
46
|
-
* Check if path targets sentinel files
|
|
47
|
-
*/
|
|
48
|
-
function isSentinelPath(filePath) {
|
|
49
|
-
if (!filePath) return false;
|
|
50
|
-
const normalized = filePath.replace(/\\/g, '/').replace(/^\.\//, '');
|
|
51
|
-
return normalized.startsWith(SENTINEL_PATH_PREFIX) || SENTINEL_PATH_RE.test(filePath);
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
/**
|
|
55
|
-
* Main guard logic
|
|
56
|
-
*/
|
|
57
|
-
function guard(toolName, toolInput) {
|
|
58
|
-
// Write/Edit targeting sentinel files → block
|
|
59
|
-
if (toolName === 'Write' || toolName === 'Edit') {
|
|
60
|
-
const filePath = extractFilePath(toolName, toolInput);
|
|
61
|
-
if (isSentinelPath(filePath)) {
|
|
62
|
-
return {
|
|
63
|
-
decision: 'block',
|
|
64
|
-
reason: `Sentinel files are protected. Cannot modify: ${filePath}`,
|
|
65
|
-
};
|
|
66
|
-
}
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
// Bash targeting sentinel files or dangerous commands
|
|
70
|
-
if (toolName === 'Bash') {
|
|
71
|
-
const command = extractBashCommand(toolInput);
|
|
72
|
-
if (command && isSentinelPath(command)) {
|
|
73
|
-
return {
|
|
74
|
-
decision: 'block',
|
|
75
|
-
reason: `Sentinel files are protected. Dangerous command targeting sentinel path.`,
|
|
76
|
-
};
|
|
77
|
-
}
|
|
78
|
-
if (command && DANGEROUS_BASH_RE.test(command)) {
|
|
79
|
-
// Check if command targets sentinel files
|
|
80
|
-
if (command.includes(SENTINEL_PATH_PREFIX) || command.includes('src/infra/lib/autonomy')) {
|
|
81
|
-
return {
|
|
82
|
-
decision: 'block',
|
|
83
|
-
reason: `Dangerous command targeting sentinel path: ${command}`,
|
|
84
|
-
};
|
|
85
|
-
}
|
|
86
|
-
}
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
// Allow — return undefined for normal flow
|
|
90
|
-
return undefined;
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
// Main execution
|
|
94
|
-
const toolName = process.argv[2] || '';
|
|
95
|
-
const toolInput = process.argv[3] || process.env.TOOL_INPUT || '';
|
|
96
|
-
|
|
97
|
-
const result = guard(toolName, toolInput);
|
|
98
|
-
|
|
99
|
-
if (result) {
|
|
100
|
-
console.log(JSON.stringify(result));
|
|
101
|
-
process.exit(1);
|
|
102
|
-
}
|
|
103
|
-
|
|
104
|
-
process.exit(0);
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* Sentinel Guard — PreToolUse hook
|
|
4
|
+
* Protects sentinel files and blocks dangerous operations.
|
|
5
|
+
* Runs before pre-tool-guard.js.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
const SENTINEL_PATH_PREFIX = 'src/infra/lib/autonomy/';
|
|
9
|
+
|
|
10
|
+
const SENTINEL_PATH_RE = /^(\.\/|\.\\)?src[\\/]infra[\\/]lib[\\/]autonomy[\\/]/;
|
|
11
|
+
|
|
12
|
+
const DANGEROUS_BASH_RE =
|
|
13
|
+
/\b(rm\s+-rf|kill\s+-9|drop\s+table|truncate|shutdown|reboot|mkfs|dd\s+if=)\b/i;
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Extract file path from tool input
|
|
17
|
+
*/
|
|
18
|
+
function extractFilePath(toolName, input) {
|
|
19
|
+
if (!input) return null;
|
|
20
|
+
try {
|
|
21
|
+
const parsed = typeof input === 'string' ? JSON.parse(input) : input;
|
|
22
|
+
if (toolName === 'Write' || toolName === 'Edit' || toolName === 'Read') {
|
|
23
|
+
return parsed.file_path || parsed.filePath || null;
|
|
24
|
+
}
|
|
25
|
+
} catch {
|
|
26
|
+
// Not JSON, try as plain string
|
|
27
|
+
return typeof input === 'string' ? input : null;
|
|
28
|
+
}
|
|
29
|
+
return null;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* Extract command from Bash tool input
|
|
34
|
+
*/
|
|
35
|
+
function extractBashCommand(input) {
|
|
36
|
+
if (!input) return null;
|
|
37
|
+
try {
|
|
38
|
+
const parsed = typeof input === 'string' ? JSON.parse(input) : input;
|
|
39
|
+
return parsed.command || null;
|
|
40
|
+
} catch {
|
|
41
|
+
return typeof input === 'string' ? input : null;
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Check if path targets sentinel files
|
|
47
|
+
*/
|
|
48
|
+
function isSentinelPath(filePath) {
|
|
49
|
+
if (!filePath) return false;
|
|
50
|
+
const normalized = filePath.replace(/\\/g, '/').replace(/^\.\//, '');
|
|
51
|
+
return normalized.startsWith(SENTINEL_PATH_PREFIX) || SENTINEL_PATH_RE.test(filePath);
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Main guard logic
|
|
56
|
+
*/
|
|
57
|
+
function guard(toolName, toolInput) {
|
|
58
|
+
// Write/Edit targeting sentinel files → block
|
|
59
|
+
if (toolName === 'Write' || toolName === 'Edit') {
|
|
60
|
+
const filePath = extractFilePath(toolName, toolInput);
|
|
61
|
+
if (isSentinelPath(filePath)) {
|
|
62
|
+
return {
|
|
63
|
+
decision: 'block',
|
|
64
|
+
reason: `Sentinel files are protected. Cannot modify: ${filePath}`,
|
|
65
|
+
};
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// Bash targeting sentinel files or dangerous commands
|
|
70
|
+
if (toolName === 'Bash') {
|
|
71
|
+
const command = extractBashCommand(toolInput);
|
|
72
|
+
if (command && isSentinelPath(command)) {
|
|
73
|
+
return {
|
|
74
|
+
decision: 'block',
|
|
75
|
+
reason: `Sentinel files are protected. Dangerous command targeting sentinel path.`,
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
if (command && DANGEROUS_BASH_RE.test(command)) {
|
|
79
|
+
// Check if command targets sentinel files
|
|
80
|
+
if (command.includes(SENTINEL_PATH_PREFIX) || command.includes('src/infra/lib/autonomy')) {
|
|
81
|
+
return {
|
|
82
|
+
decision: 'block',
|
|
83
|
+
reason: `Dangerous command targeting sentinel path: ${command}`,
|
|
84
|
+
};
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// Allow — return undefined for normal flow
|
|
90
|
+
return undefined;
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
// Main execution
|
|
94
|
+
const toolName = process.argv[2] || '';
|
|
95
|
+
const toolInput = process.argv[3] || process.env.TOOL_INPUT || '';
|
|
96
|
+
|
|
97
|
+
const result = guard(toolName, toolInput);
|
|
98
|
+
|
|
99
|
+
if (result) {
|
|
100
|
+
console.log(JSON.stringify(result));
|
|
101
|
+
process.exit(1);
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
process.exit(0);
|