moai-adk 0.15.1__py3-none-any.whl → 0.25.4__py3-none-any.whl
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.
Potentially problematic release.
This version of moai-adk might be problematic. Click here for more details.
- moai_adk/__init__.py +1 -2
- moai_adk/__main__.py +85 -2
- moai_adk/cli/__init__.py +0 -1
- moai_adk/cli/commands/__init__.py +0 -1
- moai_adk/cli/commands/analyze.py +127 -0
- moai_adk/cli/commands/backup.py +5 -3
- moai_adk/cli/commands/doctor.py +35 -11
- moai_adk/cli/commands/improve_user_experience.py +348 -0
- moai_adk/cli/commands/init.py +150 -23
- moai_adk/cli/commands/language.py +269 -0
- moai_adk/cli/commands/migrate.py +158 -0
- moai_adk/cli/commands/status.py +13 -12
- moai_adk/cli/commands/update.py +364 -60
- moai_adk/cli/commands/validate_links.py +118 -0
- moai_adk/cli/main.py +3 -2
- moai_adk/cli/prompts/init_prompts.py +79 -82
- moai_adk/core/__init__.py +0 -1
- moai_adk/core/analysis/__init__.py +9 -0
- moai_adk/core/analysis/session_analyzer.py +439 -0
- moai_adk/core/claude_integration.py +421 -0
- moai_adk/core/command_helpers.py +270 -0
- moai_adk/core/config/__init__.py +6 -0
- moai_adk/core/config/auto_spec_config.py +346 -0
- moai_adk/core/config/migration.py +133 -12
- moai_adk/core/context_manager.py +279 -0
- moai_adk/core/diagnostics/slash_commands.py +0 -1
- moai_adk/core/error_recovery_system.py +1289 -0
- moai_adk/core/git/__init__.py +0 -1
- moai_adk/core/git/branch.py +0 -1
- moai_adk/core/git/branch_manager.py +4 -4
- moai_adk/core/git/checkpoint.py +1 -5
- moai_adk/core/git/commit.py +0 -1
- moai_adk/core/git/event_detector.py +3 -5
- moai_adk/core/git/manager.py +0 -1
- moai_adk/core/hooks/post_tool_auto_spec_completion.py +925 -0
- moai_adk/core/integration/__init__.py +22 -0
- moai_adk/core/integration/engine.py +169 -0
- moai_adk/core/integration/integration_tester.py +225 -0
- moai_adk/core/integration/models.py +88 -0
- moai_adk/core/integration/utils.py +211 -0
- moai_adk/core/issue_creator.py +28 -18
- moai_adk/core/language_config.py +202 -0
- moai_adk/core/language_validator.py +556 -0
- moai_adk/core/mcp/setup.py +113 -0
- moai_adk/core/migration/__init__.py +18 -0
- moai_adk/core/migration/backup_manager.py +208 -0
- moai_adk/core/migration/file_migrator.py +218 -0
- moai_adk/core/migration/version_detector.py +143 -0
- moai_adk/core/migration/version_migrator.py +228 -0
- moai_adk/core/performance/__init__.py +6 -0
- moai_adk/core/performance/cache_system.py +318 -0
- moai_adk/core/performance/parallel_processor.py +116 -0
- moai_adk/core/project/__init__.py +0 -1
- moai_adk/core/project/backup_utils.py +2 -7
- moai_adk/core/project/checker.py +3 -3
- moai_adk/core/project/detector.py +20 -40
- moai_adk/core/project/initializer.py +42 -17
- moai_adk/core/project/phase_executor.py +415 -58
- moai_adk/core/project/validator.py +6 -25
- moai_adk/core/quality/__init__.py +1 -1
- moai_adk/core/quality/trust_checker.py +64 -110
- moai_adk/core/quality/validators/__init__.py +1 -1
- moai_adk/core/quality/validators/base_validator.py +1 -1
- moai_adk/core/rollback_manager.py +993 -0
- moai_adk/core/session_manager.py +667 -0
- moai_adk/core/spec/confidence_scoring.py +749 -0
- moai_adk/core/spec/ears_template_engine.py +1182 -0
- moai_adk/core/spec/quality_validator.py +721 -0
- moai_adk/core/spec_status_manager.py +488 -0
- moai_adk/core/template/__init__.py +0 -1
- moai_adk/core/template/backup.py +41 -1
- moai_adk/core/template/config.py +11 -12
- moai_adk/core/template/languages.py +0 -1
- moai_adk/core/template/merger.py +79 -22
- moai_adk/core/template/processor.py +614 -40
- moai_adk/core/template_engine.py +36 -27
- moai_adk/foundation/git/commit_templates.py +565 -0
- moai_adk/foundation/trust/trust_principles.py +725 -0
- moai_adk/foundation/trust/validation_checklist.py +1678 -0
- moai_adk/statusline/__init__.py +38 -0
- moai_adk/statusline/alfred_detector.py +107 -0
- moai_adk/statusline/config.py +364 -0
- moai_adk/statusline/enhanced_output_style_detector.py +364 -0
- moai_adk/statusline/git_collector.py +190 -0
- moai_adk/statusline/main.py +228 -0
- moai_adk/statusline/metrics_tracker.py +78 -0
- moai_adk/statusline/renderer.py +327 -0
- moai_adk/statusline/update_checker.py +135 -0
- moai_adk/statusline/version_reader.py +647 -0
- moai_adk/templates/.git-hooks/pre-commit +66 -0
- moai_adk/templates/.git-hooks/pre-push +116 -4
- moai_adk/templates/.github/workflows/moai-gitflow.yml +1 -7
- moai_adk/templates/.github/workflows/spec-issue-sync.yml +0 -1
- moai_adk/templates/.gitignore +44 -0
- moai_adk/templates/.mcp.json +22 -0
- moai_adk/templates/CLAUDE.md +450 -1071
- moai_adk/utils/__init__.py +0 -1
- moai_adk/utils/banner.py +0 -1
- moai_adk/utils/common.py +308 -0
- moai_adk/utils/link_validator.py +249 -0
- moai_adk/utils/logger.py +4 -9
- moai_adk/utils/safe_file_reader.py +210 -0
- moai_adk/utils/user_experience.py +531 -0
- moai_adk-0.25.4.dist-info/METADATA +2279 -0
- moai_adk-0.25.4.dist-info/RECORD +112 -0
- moai_adk/core/tags/__init__.py +0 -86
- moai_adk/core/tags/ci_validator.py +0 -463
- moai_adk/core/tags/cli.py +0 -283
- moai_adk/core/tags/generator.py +0 -109
- moai_adk/core/tags/inserter.py +0 -99
- moai_adk/core/tags/mapper.py +0 -126
- moai_adk/core/tags/parser.py +0 -76
- moai_adk/core/tags/pre_commit_validator.py +0 -393
- moai_adk/core/tags/reporter.py +0 -956
- moai_adk/core/tags/tags.py +0 -149
- moai_adk/core/tags/validator.py +0 -897
- moai_adk/templates/.claude/agents/alfred/backend-expert.md +0 -319
- moai_adk/templates/.claude/agents/alfred/cc-manager.md +0 -316
- moai_adk/templates/.claude/agents/alfred/debug-helper.md +0 -208
- moai_adk/templates/.claude/agents/alfred/devops-expert.md +0 -464
- moai_adk/templates/.claude/agents/alfred/doc-syncer.md +0 -214
- moai_adk/templates/.claude/agents/alfred/frontend-expert.md +0 -357
- moai_adk/templates/.claude/agents/alfred/git-manager.md +0 -406
- moai_adk/templates/.claude/agents/alfred/implementation-planner.md +0 -423
- moai_adk/templates/.claude/agents/alfred/project-manager.md +0 -312
- moai_adk/templates/.claude/agents/alfred/quality-gate.md +0 -343
- moai_adk/templates/.claude/agents/alfred/skill-factory.md +0 -865
- moai_adk/templates/.claude/agents/alfred/spec-builder.md +0 -426
- moai_adk/templates/.claude/agents/alfred/tag-agent.md +0 -361
- moai_adk/templates/.claude/agents/alfred/tdd-implementer.md +0 -428
- moai_adk/templates/.claude/agents/alfred/trust-checker.md +0 -375
- moai_adk/templates/.claude/agents/alfred/ui-ux-expert.md +0 -571
- moai_adk/templates/.claude/commands/alfred/0-project.md +0 -1854
- moai_adk/templates/.claude/commands/alfred/1-plan.md +0 -880
- moai_adk/templates/.claude/commands/alfred/2-run.md +0 -793
- moai_adk/templates/.claude/commands/alfred/3-sync.md +0 -1084
- moai_adk/templates/.claude/commands/alfred/9-feedback.md +0 -149
- moai_adk/templates/.claude/hooks/alfred/core/project.py +0 -748
- moai_adk/templates/.claude/hooks/alfred/core/timeout.py +0 -136
- moai_adk/templates/.claude/hooks/alfred/core/ttl_cache.py +0 -108
- moai_adk/templates/.claude/hooks/alfred/core/version_cache.py +0 -198
- moai_adk/templates/.claude/hooks/alfred/handlers/__init__.py +0 -29
- moai_adk/templates/.claude/hooks/alfred/post_tool__log_changes.py +0 -94
- moai_adk/templates/.claude/hooks/alfred/pre_tool__auto_checkpoint.py +0 -100
- moai_adk/templates/.claude/hooks/alfred/session_end__cleanup.py +0 -94
- moai_adk/templates/.claude/hooks/alfred/session_start__show_project_info.py +0 -94
- moai_adk/templates/.claude/hooks/alfred/shared/core/__init__.py +0 -170
- moai_adk/templates/.claude/hooks/alfred/shared/core/checkpoint.py +0 -271
- moai_adk/templates/.claude/hooks/alfred/shared/core/context.py +0 -67
- moai_adk/templates/.claude/hooks/alfred/shared/core/project.py +0 -749
- moai_adk/templates/.claude/hooks/alfred/shared/core/tags.py +0 -230
- moai_adk/templates/.claude/hooks/alfred/shared/core/version_cache.py +0 -198
- moai_adk/templates/.claude/hooks/alfred/shared/handlers/__init__.py +0 -21
- moai_adk/templates/.claude/hooks/alfred/shared/handlers/notification.py +0 -154
- moai_adk/templates/.claude/hooks/alfred/shared/handlers/session.py +0 -174
- moai_adk/templates/.claude/hooks/alfred/shared/handlers/tool.py +0 -87
- moai_adk/templates/.claude/hooks/alfred/shared/handlers/user.py +0 -61
- moai_adk/templates/.claude/hooks/alfred/user_prompt__jit_load_docs.py +0 -112
- moai_adk/templates/.claude/hooks/alfred/utils/__init__.py +0 -1
- moai_adk/templates/.claude/hooks/alfred/utils/timeout.py +0 -161
- moai_adk/templates/.claude/settings.json +0 -144
- moai_adk/templates/.claude/skills/moai-alfred-agent-guide/SKILL.md +0 -70
- moai_adk/templates/.claude/skills/moai-alfred-agent-guide/examples.md +0 -62
- moai_adk/templates/.claude/skills/moai-alfred-agent-guide/reference.md +0 -242
- moai_adk/templates/.claude/skills/moai-alfred-config-schema/SKILL.md +0 -56
- moai_adk/templates/.claude/skills/moai-alfred-config-schema/examples.md +0 -28
- moai_adk/templates/.claude/skills/moai-alfred-config-schema/reference.md +0 -444
- moai_adk/templates/.claude/skills/moai-alfred-context-budget/SKILL.md +0 -62
- moai_adk/templates/.claude/skills/moai-alfred-context-budget/examples.md +0 -28
- moai_adk/templates/.claude/skills/moai-alfred-context-budget/reference.md +0 -405
- moai_adk/templates/.claude/skills/moai-alfred-dev-guide/SKILL.md +0 -51
- moai_adk/templates/.claude/skills/moai-alfred-dev-guide/examples.md +0 -355
- moai_adk/templates/.claude/skills/moai-alfred-dev-guide/reference.md +0 -239
- moai_adk/templates/.claude/skills/moai-alfred-ears-authoring/SKILL.md +0 -113
- moai_adk/templates/.claude/skills/moai-alfred-ears-authoring/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-alfred-ears-authoring/reference.md +0 -28
- moai_adk/templates/.claude/skills/moai-alfred-expertise-detection/SKILL.md +0 -323
- moai_adk/templates/.claude/skills/moai-alfred-expertise-detection/examples.md +0 -286
- moai_adk/templates/.claude/skills/moai-alfred-expertise-detection/reference.md +0 -126
- moai_adk/templates/.claude/skills/moai-alfred-git-workflow/SKILL.md +0 -122
- moai_adk/templates/.claude/skills/moai-alfred-git-workflow/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-alfred-git-workflow/reference.md +0 -29
- moai_adk/templates/.claude/skills/moai-alfred-gitflow-policy/SKILL.md +0 -74
- moai_adk/templates/.claude/skills/moai-alfred-gitflow-policy/examples.md +0 -4
- moai_adk/templates/.claude/skills/moai-alfred-gitflow-policy/reference.md +0 -269
- moai_adk/templates/.claude/skills/moai-alfred-interactive-questions/SKILL.md +0 -237
- moai_adk/templates/.claude/skills/moai-alfred-interactive-questions/examples.md +0 -615
- moai_adk/templates/.claude/skills/moai-alfred-interactive-questions/reference.md +0 -653
- moai_adk/templates/.claude/skills/moai-alfred-issue-labels/SKILL.md +0 -19
- moai_adk/templates/.claude/skills/moai-alfred-issue-labels/examples.md +0 -4
- moai_adk/templates/.claude/skills/moai-alfred-issue-labels/reference.md +0 -150
- moai_adk/templates/.claude/skills/moai-alfred-language-detection/SKILL.md +0 -113
- moai_adk/templates/.claude/skills/moai-alfred-language-detection/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-alfred-language-detection/reference.md +0 -28
- moai_adk/templates/.claude/skills/moai-alfred-persona-roles/SKILL.md +0 -198
- moai_adk/templates/.claude/skills/moai-alfred-persona-roles/examples.md +0 -431
- moai_adk/templates/.claude/skills/moai-alfred-persona-roles/reference.md +0 -141
- moai_adk/templates/.claude/skills/moai-alfred-practices/SKILL.md +0 -89
- moai_adk/templates/.claude/skills/moai-alfred-practices/examples.md +0 -122
- moai_adk/templates/.claude/skills/moai-alfred-practices/reference.md +0 -369
- moai_adk/templates/.claude/skills/moai-alfred-proactive-suggestions/SKILL.md +0 -508
- moai_adk/templates/.claude/skills/moai-alfred-proactive-suggestions/examples.md +0 -481
- moai_adk/templates/.claude/skills/moai-alfred-proactive-suggestions/reference.md +0 -100
- moai_adk/templates/.claude/skills/moai-alfred-reporting/SKILL.md +0 -273
- moai_adk/templates/.claude/skills/moai-alfred-rules/SKILL.md +0 -77
- moai_adk/templates/.claude/skills/moai-alfred-rules/examples.md +0 -265
- moai_adk/templates/.claude/skills/moai-alfred-rules/reference.md +0 -539
- moai_adk/templates/.claude/skills/moai-alfred-session-state/SKILL.md +0 -19
- moai_adk/templates/.claude/skills/moai-alfred-session-state/examples.md +0 -4
- moai_adk/templates/.claude/skills/moai-alfred-session-state/reference.md +0 -84
- moai_adk/templates/.claude/skills/moai-alfred-spec-authoring/README.md +0 -137
- moai_adk/templates/.claude/skills/moai-alfred-spec-authoring/SKILL.md +0 -219
- moai_adk/templates/.claude/skills/moai-alfred-spec-authoring/examples/validate-spec.sh +0 -161
- moai_adk/templates/.claude/skills/moai-alfred-spec-authoring/examples.md +0 -541
- moai_adk/templates/.claude/skills/moai-alfred-spec-authoring/reference.md +0 -622
- moai_adk/templates/.claude/skills/moai-alfred-spec-metadata-extended/SKILL.md +0 -115
- moai_adk/templates/.claude/skills/moai-alfred-spec-metadata-extended/examples.md +0 -4
- moai_adk/templates/.claude/skills/moai-alfred-spec-metadata-extended/reference.md +0 -348
- moai_adk/templates/.claude/skills/moai-alfred-spec-metadata-validation/SKILL.md +0 -113
- moai_adk/templates/.claude/skills/moai-alfred-spec-metadata-validation/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-alfred-spec-metadata-validation/reference.md +0 -28
- moai_adk/templates/.claude/skills/moai-alfred-tag-scanning/SKILL.md +0 -113
- moai_adk/templates/.claude/skills/moai-alfred-tag-scanning/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-alfred-tag-scanning/reference.md +0 -28
- moai_adk/templates/.claude/skills/moai-alfred-todowrite-pattern/SKILL.md +0 -19
- moai_adk/templates/.claude/skills/moai-alfred-todowrite-pattern/examples.md +0 -4
- moai_adk/templates/.claude/skills/moai-alfred-todowrite-pattern/reference.md +0 -211
- moai_adk/templates/.claude/skills/moai-alfred-trust-validation/SKILL.md +0 -113
- moai_adk/templates/.claude/skills/moai-alfred-trust-validation/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-alfred-trust-validation/reference.md +0 -28
- moai_adk/templates/.claude/skills/moai-alfred-workflow/SKILL.md +0 -288
- moai_adk/templates/.claude/skills/moai-cc-agents/SKILL.md +0 -269
- moai_adk/templates/.claude/skills/moai-cc-agents/templates/agent-template.md +0 -32
- moai_adk/templates/.claude/skills/moai-cc-claude-md/SKILL.md +0 -298
- moai_adk/templates/.claude/skills/moai-cc-claude-md/templates/CLAUDE-template.md +0 -26
- moai_adk/templates/.claude/skills/moai-cc-commands/SKILL.md +0 -307
- moai_adk/templates/.claude/skills/moai-cc-commands/templates/command-template.md +0 -21
- moai_adk/templates/.claude/skills/moai-cc-hooks/SKILL.md +0 -252
- moai_adk/templates/.claude/skills/moai-cc-hooks/scripts/pre-bash-check.sh +0 -19
- moai_adk/templates/.claude/skills/moai-cc-hooks/scripts/preserve-permissions.sh +0 -19
- moai_adk/templates/.claude/skills/moai-cc-hooks/scripts/validate-bash-command.py +0 -24
- moai_adk/templates/.claude/skills/moai-cc-mcp-plugins/SKILL.md +0 -199
- moai_adk/templates/.claude/skills/moai-cc-mcp-plugins/templates/settings-mcp-template.json +0 -39
- moai_adk/templates/.claude/skills/moai-cc-memory/SKILL.md +0 -316
- moai_adk/templates/.claude/skills/moai-cc-memory/templates/session-summary-template.md +0 -18
- moai_adk/templates/.claude/skills/moai-cc-settings/SKILL.md +0 -263
- moai_adk/templates/.claude/skills/moai-cc-settings/templates/settings-complete-template.json +0 -30
- moai_adk/templates/.claude/skills/moai-cc-skill-descriptions/SKILL.md +0 -19
- moai_adk/templates/.claude/skills/moai-cc-skill-descriptions/examples.md +0 -4
- moai_adk/templates/.claude/skills/moai-cc-skill-descriptions/reference.md +0 -218
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/CHECKLIST.md +0 -482
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/EXAMPLES.md +0 -278
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/INTERACTIVE-DISCOVERY.md +0 -524
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/METADATA.md +0 -477
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/PARALLEL-ANALYSIS-REPORT.md +0 -429
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/PYTHON-VERSION-MATRIX.md +0 -391
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/SKILL-FACTORY-WORKFLOW.md +0 -431
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/SKILL-UPDATE-ADVISOR.md +0 -577
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/SKILL.md +0 -271
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/STEP-BY-STEP-GUIDE.md +0 -466
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/STRUCTURE.md +0 -583
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/WEB-RESEARCH.md +0 -526
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/reference.md +0 -465
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/scripts/generate-structure.sh +0 -328
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/scripts/validate-skill.sh +0 -312
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/templates/SKILL_TEMPLATE.md +0 -245
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/templates/examples-template.md +0 -285
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/templates/reference-template.md +0 -278
- moai_adk/templates/.claude/skills/moai-cc-skill-factory/templates/scripts-template.sh +0 -303
- moai_adk/templates/.claude/skills/moai-cc-skills/SKILL.md +0 -291
- moai_adk/templates/.claude/skills/moai-cc-skills/templates/SKILL-template.md +0 -15
- moai_adk/templates/.claude/skills/moai-design-systems/SKILL.md +0 -802
- moai_adk/templates/.claude/skills/moai-design-systems/examples.md +0 -1238
- moai_adk/templates/.claude/skills/moai-design-systems/reference.md +0 -673
- moai_adk/templates/.claude/skills/moai-domain-backend/SKILL.md +0 -290
- moai_adk/templates/.claude/skills/moai-domain-backend/examples.md +0 -1633
- moai_adk/templates/.claude/skills/moai-domain-backend/reference.md +0 -660
- moai_adk/templates/.claude/skills/moai-domain-cli-tool/SKILL.md +0 -123
- moai_adk/templates/.claude/skills/moai-domain-cli-tool/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-domain-cli-tool/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-domain-data-science/SKILL.md +0 -123
- moai_adk/templates/.claude/skills/moai-domain-data-science/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-domain-data-science/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-domain-database/SKILL.md +0 -123
- moai_adk/templates/.claude/skills/moai-domain-database/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-domain-database/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-domain-devops/SKILL.md +0 -124
- moai_adk/templates/.claude/skills/moai-domain-devops/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-domain-devops/reference.md +0 -31
- moai_adk/templates/.claude/skills/moai-domain-frontend/SKILL.md +0 -128
- moai_adk/templates/.claude/skills/moai-domain-frontend/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-domain-frontend/reference.md +0 -31
- moai_adk/templates/.claude/skills/moai-domain-ml/SKILL.md +0 -123
- moai_adk/templates/.claude/skills/moai-domain-ml/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-domain-ml/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-domain-mobile-app/SKILL.md +0 -123
- moai_adk/templates/.claude/skills/moai-domain-mobile-app/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-domain-mobile-app/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-domain-security/SKILL.md +0 -123
- moai_adk/templates/.claude/skills/moai-domain-security/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-domain-security/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-domain-web-api/SKILL.md +0 -123
- moai_adk/templates/.claude/skills/moai-domain-web-api/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-domain-web-api/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-essentials-debug/SKILL.md +0 -303
- moai_adk/templates/.claude/skills/moai-essentials-debug/examples.md +0 -1064
- moai_adk/templates/.claude/skills/moai-essentials-debug/reference.md +0 -1047
- moai_adk/templates/.claude/skills/moai-essentials-perf/SKILL.md +0 -113
- moai_adk/templates/.claude/skills/moai-essentials-perf/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-essentials-perf/reference.md +0 -28
- moai_adk/templates/.claude/skills/moai-essentials-refactor/SKILL.md +0 -113
- moai_adk/templates/.claude/skills/moai-essentials-refactor/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-essentials-refactor/reference.md +0 -28
- moai_adk/templates/.claude/skills/moai-essentials-review/SKILL.md +0 -113
- moai_adk/templates/.claude/skills/moai-essentials-review/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-essentials-review/reference.md +0 -28
- moai_adk/templates/.claude/skills/moai-foundation-ears/SKILL.md +0 -116
- moai_adk/templates/.claude/skills/moai-foundation-ears/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-foundation-ears/reference.md +0 -28
- moai_adk/templates/.claude/skills/moai-foundation-git/SKILL.md +0 -122
- moai_adk/templates/.claude/skills/moai-foundation-git/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-foundation-git/reference.md +0 -29
- moai_adk/templates/.claude/skills/moai-foundation-langs/SKILL.md +0 -113
- moai_adk/templates/.claude/skills/moai-foundation-langs/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-foundation-langs/reference.md +0 -28
- moai_adk/templates/.claude/skills/moai-foundation-specs/SKILL.md +0 -113
- moai_adk/templates/.claude/skills/moai-foundation-specs/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-foundation-specs/reference.md +0 -28
- moai_adk/templates/.claude/skills/moai-foundation-tags/SKILL.md +0 -113
- moai_adk/templates/.claude/skills/moai-foundation-tags/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-foundation-tags/reference.md +0 -28
- moai_adk/templates/.claude/skills/moai-foundation-trust/SKILL.md +0 -307
- moai_adk/templates/.claude/skills/moai-foundation-trust/examples.md +0 -0
- moai_adk/templates/.claude/skills/moai-foundation-trust/reference.md +0 -1099
- moai_adk/templates/.claude/skills/moai-lang-c/SKILL.md +0 -124
- moai_adk/templates/.claude/skills/moai-lang-c/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-c/reference.md +0 -31
- moai_adk/templates/.claude/skills/moai-lang-cpp/SKILL.md +0 -124
- moai_adk/templates/.claude/skills/moai-lang-cpp/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-cpp/reference.md +0 -31
- moai_adk/templates/.claude/skills/moai-lang-csharp/SKILL.md +0 -123
- moai_adk/templates/.claude/skills/moai-lang-csharp/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-csharp/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-lang-dart/SKILL.md +0 -123
- moai_adk/templates/.claude/skills/moai-lang-dart/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-dart/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-lang-go/SKILL.md +0 -127
- moai_adk/templates/.claude/skills/moai-lang-go/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-go/reference.md +0 -31
- moai_adk/templates/.claude/skills/moai-lang-java/SKILL.md +0 -126
- moai_adk/templates/.claude/skills/moai-lang-java/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-java/reference.md +0 -31
- moai_adk/templates/.claude/skills/moai-lang-javascript/SKILL.md +0 -125
- moai_adk/templates/.claude/skills/moai-lang-javascript/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-javascript/reference.md +0 -32
- moai_adk/templates/.claude/skills/moai-lang-kotlin/SKILL.md +0 -124
- moai_adk/templates/.claude/skills/moai-lang-kotlin/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-kotlin/reference.md +0 -31
- moai_adk/templates/.claude/skills/moai-lang-php/SKILL.md +0 -126
- moai_adk/templates/.claude/skills/moai-lang-php/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-php/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-lang-python/SKILL.md +0 -433
- moai_adk/templates/.claude/skills/moai-lang-python/examples.md +0 -624
- moai_adk/templates/.claude/skills/moai-lang-python/reference.md +0 -316
- moai_adk/templates/.claude/skills/moai-lang-r/SKILL.md +0 -123
- moai_adk/templates/.claude/skills/moai-lang-r/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-r/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-lang-ruby/SKILL.md +0 -124
- moai_adk/templates/.claude/skills/moai-lang-ruby/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-ruby/reference.md +0 -31
- moai_adk/templates/.claude/skills/moai-lang-rust/SKILL.md +0 -127
- moai_adk/templates/.claude/skills/moai-lang-rust/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-rust/reference.md +0 -31
- moai_adk/templates/.claude/skills/moai-lang-scala/SKILL.md +0 -125
- moai_adk/templates/.claude/skills/moai-lang-scala/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-scala/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-lang-shell/SKILL.md +0 -123
- moai_adk/templates/.claude/skills/moai-lang-shell/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-shell/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-lang-sql/SKILL.md +0 -124
- moai_adk/templates/.claude/skills/moai-lang-sql/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-sql/reference.md +0 -31
- moai_adk/templates/.claude/skills/moai-lang-swift/SKILL.md +0 -123
- moai_adk/templates/.claude/skills/moai-lang-swift/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-swift/reference.md +0 -30
- moai_adk/templates/.claude/skills/moai-lang-typescript/SKILL.md +0 -133
- moai_adk/templates/.claude/skills/moai-lang-typescript/examples.md +0 -29
- moai_adk/templates/.claude/skills/moai-lang-typescript/reference.md +0 -34
- moai_adk/templates/.claude/skills/moai-project-documentation.md +0 -622
- moai_adk/templates/.github/workflows/c-tag-validation.yml +0 -11
- moai_adk/templates/.github/workflows/cpp-tag-validation.yml +0 -11
- moai_adk/templates/.github/workflows/csharp-tag-validation.yml +0 -11
- moai_adk/templates/.github/workflows/dart-tag-validation.yml +0 -11
- moai_adk/templates/.github/workflows/go-tag-validation.yml +0 -130
- moai_adk/templates/.github/workflows/java-tag-validation.yml +0 -11
- moai_adk/templates/.github/workflows/javascript-tag-validation.yml +0 -135
- moai_adk/templates/.github/workflows/kotlin-tag-validation.yml +0 -11
- moai_adk/templates/.github/workflows/php-tag-validation.yml +0 -11
- moai_adk/templates/.github/workflows/python-tag-validation.yml +0 -118
- moai_adk/templates/.github/workflows/release.yml +0 -118
- moai_adk/templates/.github/workflows/ruby-tag-validation.yml +0 -11
- moai_adk/templates/.github/workflows/rust-tag-validation.yml +0 -11
- moai_adk/templates/.github/workflows/shell-tag-validation.yml +0 -11
- moai_adk/templates/.github/workflows/swift-tag-validation.yml +0 -11
- moai_adk/templates/.github/workflows/tag-report.yml +0 -269
- moai_adk/templates/.github/workflows/tag-validation.yml +0 -186
- moai_adk/templates/.github/workflows/typescript-tag-validation.yml +0 -154
- moai_adk/templates/.moai/config.json +0 -115
- moai_adk/templates/workflows/go-tag-validation.yml +0 -30
- moai_adk/templates/workflows/javascript-tag-validation.yml +0 -41
- moai_adk/templates/workflows/python-tag-validation.yml +0 -42
- moai_adk/templates/workflows/typescript-tag-validation.yml +0 -31
- moai_adk-0.15.1.dist-info/METADATA +0 -3094
- moai_adk-0.15.1.dist-info/RECORD +0 -365
- {moai_adk-0.15.1.dist-info → moai_adk-0.25.4.dist-info}/WHEEL +0 -0
- {moai_adk-0.15.1.dist-info → moai_adk-0.25.4.dist-info}/entry_points.txt +0 -0
- {moai_adk-0.15.1.dist-info → moai_adk-0.25.4.dist-info}/licenses/LICENSE +0 -0
|
@@ -1,230 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env python3
|
|
2
|
-
# @CODE:HOOK-TAG-001 | SPEC: TBD | TEST: tests/hooks/test_tag_validation.py
|
|
3
|
-
"""TAG validation helpers for MoAI-ADK hooks
|
|
4
|
-
|
|
5
|
-
Fast checks used by PreToolUse/PostToolUse to nudge users when
|
|
6
|
-
new or modified files are missing required @TAG annotations.
|
|
7
|
-
|
|
8
|
-
Configurable rules with sensible defaults:
|
|
9
|
-
- Load patterns from .moai/tag-rules.json if present.
|
|
10
|
-
- Otherwise, apply default glob patterns (folder names are not hard-coded only).
|
|
11
|
-
|
|
12
|
-
Defaults (order matters; first match wins):
|
|
13
|
-
1) SPEC
|
|
14
|
-
- .moai/specs/**
|
|
15
|
-
- **/SPEC-*/spec.md
|
|
16
|
-
2) TEST
|
|
17
|
-
- **/*_test.py, **/test_*.py, **/*.test.* (ts,tsx,js,jsx,go,rs)
|
|
18
|
-
- **/*.spec.* (ts,tsx,js,jsx)
|
|
19
|
-
- tests/**
|
|
20
|
-
3) DOC
|
|
21
|
-
- docs/**/*.md, **/README.md, **/*.api.md
|
|
22
|
-
4) CODE
|
|
23
|
-
- Source extensions: .py,.ts,.tsx,.js,.jsx,.go,.rs,.java,.kt,.rb,.php,.c,.cpp,.cs,.swift,.scala
|
|
24
|
-
- Excluding TEST patterns
|
|
25
|
-
|
|
26
|
-
Notes:
|
|
27
|
-
- Best-effort: skip binary/large files and non-target paths
|
|
28
|
-
- Do not block execution; return a list of issues for messaging
|
|
29
|
-
"""
|
|
30
|
-
|
|
31
|
-
from __future__ import annotations
|
|
32
|
-
|
|
33
|
-
import fnmatch
|
|
34
|
-
import json
|
|
35
|
-
import subprocess
|
|
36
|
-
from dataclasses import dataclass
|
|
37
|
-
from pathlib import Path
|
|
38
|
-
from typing import Iterable, List, Optional
|
|
39
|
-
|
|
40
|
-
DEFAULT_CODE_EXTS = (
|
|
41
|
-
".py",
|
|
42
|
-
".ts",
|
|
43
|
-
".tsx",
|
|
44
|
-
".js",
|
|
45
|
-
".jsx",
|
|
46
|
-
".go",
|
|
47
|
-
".rs",
|
|
48
|
-
".java",
|
|
49
|
-
".kt",
|
|
50
|
-
".rb",
|
|
51
|
-
".php",
|
|
52
|
-
".c",
|
|
53
|
-
".cpp",
|
|
54
|
-
".cs",
|
|
55
|
-
".swift",
|
|
56
|
-
".scala",
|
|
57
|
-
)
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
@dataclass
|
|
61
|
-
class TagIssue:
|
|
62
|
-
path: str
|
|
63
|
-
expected: str # one of @SPEC, @TEST, @CODE, @DOC
|
|
64
|
-
reason: str
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
@dataclass
|
|
68
|
-
class Rule:
|
|
69
|
-
include: List[str]
|
|
70
|
-
expect: str # '@SPEC:' | '@TEST:' | '@CODE:' | '@DOC:'
|
|
71
|
-
exclude: List[str]
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
def _load_rules(cwd: str) -> List[Rule]:
|
|
75
|
-
"""Load tag rules from .moai/tag-rules.json or return defaults.
|
|
76
|
-
|
|
77
|
-
Schema example:
|
|
78
|
-
{
|
|
79
|
-
"rules": [
|
|
80
|
-
{"include": ["**/*_test.py", "**/*.test.ts"], "expect": "@TEST:", "exclude": []},
|
|
81
|
-
{"include": ["docs/**/*.md", "**/README.md"], "expect": "@DOC:", "exclude": []}
|
|
82
|
-
]
|
|
83
|
-
}
|
|
84
|
-
"""
|
|
85
|
-
cfg = Path(cwd) / ".moai" / "tag-rules.json"
|
|
86
|
-
if cfg.exists():
|
|
87
|
-
try:
|
|
88
|
-
data = json.loads(cfg.read_text(encoding="utf-8"))
|
|
89
|
-
items = data.get("rules", [])
|
|
90
|
-
rules: List[Rule] = []
|
|
91
|
-
for it in items:
|
|
92
|
-
include = list(it.get("include", []))
|
|
93
|
-
expect = str(it.get("expect", ""))
|
|
94
|
-
exclude = list(it.get("exclude", []))
|
|
95
|
-
if include and expect in ("@SPEC:", "@TEST:", "@CODE:", "@DOC:"):
|
|
96
|
-
rules.append(Rule(include=include, expect=expect, exclude=exclude))
|
|
97
|
-
if rules:
|
|
98
|
-
return rules
|
|
99
|
-
except Exception:
|
|
100
|
-
pass
|
|
101
|
-
|
|
102
|
-
# Defaults (ordered)
|
|
103
|
-
return [
|
|
104
|
-
Rule(include=[".moai/specs/**", "**/SPEC-*/spec.md"], expect="@SPEC:", exclude=[]),
|
|
105
|
-
Rule(
|
|
106
|
-
include=[
|
|
107
|
-
"**/*_test.py",
|
|
108
|
-
"**/test_*.py",
|
|
109
|
-
"**/*.test.ts",
|
|
110
|
-
"**/*.test.tsx",
|
|
111
|
-
"**/*.test.js",
|
|
112
|
-
"**/*.test.jsx",
|
|
113
|
-
"**/*.test.go",
|
|
114
|
-
"**/*.test.rs",
|
|
115
|
-
"**/*.spec.ts",
|
|
116
|
-
"**/*.spec.tsx",
|
|
117
|
-
"tests/**",
|
|
118
|
-
],
|
|
119
|
-
expect="@TEST:",
|
|
120
|
-
exclude=[".claude/**"],
|
|
121
|
-
),
|
|
122
|
-
Rule(
|
|
123
|
-
include=["docs/**/*.md", "**/README.md", "**/*.api.md"],
|
|
124
|
-
expect="@DOC:",
|
|
125
|
-
exclude=[".claude/**"],
|
|
126
|
-
),
|
|
127
|
-
Rule(
|
|
128
|
-
include=["**/*"],
|
|
129
|
-
expect="@CODE:",
|
|
130
|
-
exclude=[
|
|
131
|
-
"tests/**",
|
|
132
|
-
"docs/**",
|
|
133
|
-
".moai/**",
|
|
134
|
-
".claude/**",
|
|
135
|
-
"**/*.md",
|
|
136
|
-
"**/*.json",
|
|
137
|
-
"**/*.yml",
|
|
138
|
-
"**/*.yaml",
|
|
139
|
-
"**/*.toml",
|
|
140
|
-
"**/*.lock",
|
|
141
|
-
"**/*.svg",
|
|
142
|
-
"**/*.png",
|
|
143
|
-
"**/*.jpg",
|
|
144
|
-
"**/*.jpeg",
|
|
145
|
-
"**/*.gif",
|
|
146
|
-
],
|
|
147
|
-
),
|
|
148
|
-
]
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
def _match_any(path: str, patterns: List[str]) -> bool:
|
|
152
|
-
return any(fnmatch.fnmatch(path, pat) for pat in patterns)
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
def _needs_tag_str(path_str: str, rules: List[Rule]) -> Optional[str]:
|
|
156
|
-
p = path_str
|
|
157
|
-
for rule in rules:
|
|
158
|
-
if _match_any(p, rule.include) and not _match_any(p, rule.exclude):
|
|
159
|
-
if rule.expect == "@CODE:":
|
|
160
|
-
# CODE: limit to source-like extensions to reduce noise
|
|
161
|
-
if not any(p.endswith(ext) for ext in DEFAULT_CODE_EXTS):
|
|
162
|
-
continue
|
|
163
|
-
return rule.expect
|
|
164
|
-
return None
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
def _has_tag(content: str, expected: str) -> bool:
|
|
168
|
-
return expected in content
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
def _iter_recent_changes(cwd: str) -> Iterable[Path]:
|
|
172
|
-
root = Path(cwd)
|
|
173
|
-
try:
|
|
174
|
-
# Staged files
|
|
175
|
-
r1 = subprocess.run(
|
|
176
|
-
["git", "diff", "--name-only", "--cached"],
|
|
177
|
-
cwd=cwd,
|
|
178
|
-
capture_output=True,
|
|
179
|
-
text=True,
|
|
180
|
-
timeout=1,
|
|
181
|
-
)
|
|
182
|
-
# Modified (unstaged) tracked files
|
|
183
|
-
r2 = subprocess.run(
|
|
184
|
-
["git", "ls-files", "-m"], cwd=cwd, capture_output=True, text=True, timeout=1
|
|
185
|
-
)
|
|
186
|
-
# Untracked (other) files respecting .gitignore
|
|
187
|
-
r3 = subprocess.run(
|
|
188
|
-
["git", "ls-files", "-o", "--exclude-standard"],
|
|
189
|
-
cwd=cwd,
|
|
190
|
-
capture_output=True,
|
|
191
|
-
text=True,
|
|
192
|
-
timeout=1,
|
|
193
|
-
)
|
|
194
|
-
names = set()
|
|
195
|
-
if r1.returncode == 0:
|
|
196
|
-
names.update([line.strip() for line in r1.stdout.splitlines() if line.strip()])
|
|
197
|
-
if r2.returncode == 0:
|
|
198
|
-
names.update([line.strip() for line in r2.stdout.splitlines() if line.strip()])
|
|
199
|
-
if r3.returncode == 0:
|
|
200
|
-
names.update([line.strip() for line in r3.stdout.splitlines() if line.strip()])
|
|
201
|
-
for n in names:
|
|
202
|
-
p = (root / n).resolve()
|
|
203
|
-
if p.is_file():
|
|
204
|
-
yield p
|
|
205
|
-
except Exception:
|
|
206
|
-
return []
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
def scan_recent_changes_for_missing_tags(cwd: str) -> list[TagIssue]:
|
|
210
|
-
issues: list[TagIssue] = []
|
|
211
|
-
rules = _load_rules(cwd)
|
|
212
|
-
root = Path(cwd).resolve()
|
|
213
|
-
for path in _iter_recent_changes(cwd):
|
|
214
|
-
try:
|
|
215
|
-
content = path.read_text(encoding="utf-8", errors="ignore")
|
|
216
|
-
except Exception:
|
|
217
|
-
continue
|
|
218
|
-
# compute relative path once and use for matching/excluding
|
|
219
|
-
try:
|
|
220
|
-
rel = path.resolve().relative_to(root)
|
|
221
|
-
rel_s = rel.as_posix()
|
|
222
|
-
except Exception:
|
|
223
|
-
rel_s = path.name
|
|
224
|
-
|
|
225
|
-
expected = _needs_tag_str(rel_s, rules)
|
|
226
|
-
if not expected:
|
|
227
|
-
continue
|
|
228
|
-
if not _has_tag(content, expected):
|
|
229
|
-
issues.append(TagIssue(path=rel_s, expected=expected, reason="missing tag"))
|
|
230
|
-
return issues
|
|
@@ -1,198 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env python3
|
|
2
|
-
# @CODE:VERSION-CACHE-001
|
|
3
|
-
"""Version information cache with TTL support
|
|
4
|
-
|
|
5
|
-
TTL-based caching system for version check results to minimize network calls
|
|
6
|
-
during SessionStart hook execution.
|
|
7
|
-
|
|
8
|
-
SPEC: SPEC-UPDATE-ENHANCE-001 - SessionStart 버전 체크 시스템 강화
|
|
9
|
-
Phase 1: Cache System Implementation
|
|
10
|
-
"""
|
|
11
|
-
|
|
12
|
-
import json
|
|
13
|
-
from datetime import datetime, timezone
|
|
14
|
-
from pathlib import Path
|
|
15
|
-
from typing import Any
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
class VersionCache:
|
|
19
|
-
"""TTL-based version information cache
|
|
20
|
-
|
|
21
|
-
Caches version check results with configurable Time-To-Live (TTL)
|
|
22
|
-
to avoid excessive network calls to PyPI during SessionStart events.
|
|
23
|
-
|
|
24
|
-
Attributes:
|
|
25
|
-
cache_dir: Directory to store cache file
|
|
26
|
-
ttl_hours: Time-to-live in hours (default 24)
|
|
27
|
-
cache_file: Path to the cache JSON file
|
|
28
|
-
|
|
29
|
-
Examples:
|
|
30
|
-
>>> cache = VersionCache(Path(".moai/cache"), ttl_hours=24)
|
|
31
|
-
>>> cache.save({"current_version": "0.8.1", "latest_version": "0.9.0"})
|
|
32
|
-
True
|
|
33
|
-
>>> cache.is_valid()
|
|
34
|
-
True
|
|
35
|
-
>>> data = cache.load()
|
|
36
|
-
>>> data["current_version"]
|
|
37
|
-
'0.8.1'
|
|
38
|
-
"""
|
|
39
|
-
|
|
40
|
-
def __init__(self, cache_dir: Path, ttl_hours: int = 4):
|
|
41
|
-
"""Initialize cache with TTL in hours
|
|
42
|
-
|
|
43
|
-
Args:
|
|
44
|
-
cache_dir: Directory where cache file will be stored
|
|
45
|
-
ttl_hours: Time-to-live in hours (default 4)
|
|
46
|
-
"""
|
|
47
|
-
self.cache_dir = Path(cache_dir)
|
|
48
|
-
self.ttl_hours = ttl_hours
|
|
49
|
-
self.cache_file = self.cache_dir / "version-check.json"
|
|
50
|
-
|
|
51
|
-
def _calculate_age_hours(self, last_check_iso: str) -> float:
|
|
52
|
-
"""Calculate age in hours from ISO timestamp (internal helper)
|
|
53
|
-
|
|
54
|
-
Normalizes timezone-aware and naive datetimes for consistent comparison.
|
|
55
|
-
|
|
56
|
-
Args:
|
|
57
|
-
last_check_iso: ISO format timestamp string
|
|
58
|
-
|
|
59
|
-
Returns:
|
|
60
|
-
Age in hours
|
|
61
|
-
|
|
62
|
-
Raises:
|
|
63
|
-
ValueError: If timestamp parsing fails
|
|
64
|
-
"""
|
|
65
|
-
last_check = datetime.fromisoformat(last_check_iso)
|
|
66
|
-
|
|
67
|
-
# Normalize to naive datetime (remove timezone for comparison)
|
|
68
|
-
if last_check.tzinfo is not None:
|
|
69
|
-
last_check = last_check.replace(tzinfo=None)
|
|
70
|
-
|
|
71
|
-
now = datetime.now()
|
|
72
|
-
return (now - last_check).total_seconds() / 3600
|
|
73
|
-
|
|
74
|
-
def is_valid(self) -> bool:
|
|
75
|
-
"""Check if cache exists and is not expired
|
|
76
|
-
|
|
77
|
-
Returns:
|
|
78
|
-
True if cache file exists and is within TTL, False otherwise
|
|
79
|
-
|
|
80
|
-
Examples:
|
|
81
|
-
>>> cache = VersionCache(Path(".moai/cache"))
|
|
82
|
-
>>> cache.is_valid()
|
|
83
|
-
False # No cache file exists yet
|
|
84
|
-
"""
|
|
85
|
-
if not self.cache_file.exists():
|
|
86
|
-
return False
|
|
87
|
-
|
|
88
|
-
try:
|
|
89
|
-
with open(self.cache_file, "r") as f:
|
|
90
|
-
data = json.load(f)
|
|
91
|
-
|
|
92
|
-
age_hours = self._calculate_age_hours(data["last_check"])
|
|
93
|
-
return age_hours < self.ttl_hours
|
|
94
|
-
|
|
95
|
-
except (json.JSONDecodeError, KeyError, ValueError, OSError):
|
|
96
|
-
# Corrupted or invalid cache file
|
|
97
|
-
return False
|
|
98
|
-
|
|
99
|
-
def load(self) -> dict[str, Any] | None:
|
|
100
|
-
"""Load cached version info if valid
|
|
101
|
-
|
|
102
|
-
Returns:
|
|
103
|
-
Cached version info dictionary if valid, None otherwise
|
|
104
|
-
|
|
105
|
-
Examples:
|
|
106
|
-
>>> cache = VersionCache(Path(".moai/cache"))
|
|
107
|
-
>>> data = cache.load()
|
|
108
|
-
>>> data is None
|
|
109
|
-
True # No valid cache exists
|
|
110
|
-
"""
|
|
111
|
-
if not self.is_valid():
|
|
112
|
-
return None
|
|
113
|
-
|
|
114
|
-
try:
|
|
115
|
-
with open(self.cache_file, "r") as f:
|
|
116
|
-
return json.load(f)
|
|
117
|
-
except (json.JSONDecodeError, OSError):
|
|
118
|
-
# Graceful degradation on read errors
|
|
119
|
-
return None
|
|
120
|
-
|
|
121
|
-
def save(self, version_info: dict[str, Any]) -> bool:
|
|
122
|
-
"""Save version info to cache file
|
|
123
|
-
|
|
124
|
-
Creates cache directory if it doesn't exist.
|
|
125
|
-
Updates last_check timestamp to current time if not provided.
|
|
126
|
-
|
|
127
|
-
Args:
|
|
128
|
-
version_info: Version information dictionary to cache
|
|
129
|
-
|
|
130
|
-
Returns:
|
|
131
|
-
True on successful save, False on error
|
|
132
|
-
|
|
133
|
-
Examples:
|
|
134
|
-
>>> cache = VersionCache(Path(".moai/cache"))
|
|
135
|
-
>>> cache.save({"current_version": "0.8.1"})
|
|
136
|
-
True
|
|
137
|
-
"""
|
|
138
|
-
try:
|
|
139
|
-
# Create cache directory if it doesn't exist
|
|
140
|
-
self.cache_dir.mkdir(parents=True, exist_ok=True)
|
|
141
|
-
|
|
142
|
-
# Update last_check timestamp only if not provided (for testing)
|
|
143
|
-
if "last_check" not in version_info:
|
|
144
|
-
version_info["last_check"] = datetime.now(timezone.utc).isoformat()
|
|
145
|
-
|
|
146
|
-
# Write to cache file
|
|
147
|
-
with open(self.cache_file, "w") as f:
|
|
148
|
-
json.dump(version_info, f, indent=2)
|
|
149
|
-
|
|
150
|
-
return True
|
|
151
|
-
|
|
152
|
-
except (OSError, TypeError):
|
|
153
|
-
# Graceful degradation on write errors
|
|
154
|
-
return False
|
|
155
|
-
|
|
156
|
-
def clear(self) -> bool:
|
|
157
|
-
"""Clear/remove cache file
|
|
158
|
-
|
|
159
|
-
Returns:
|
|
160
|
-
True if cache file was removed or didn't exist, False on error
|
|
161
|
-
|
|
162
|
-
Examples:
|
|
163
|
-
>>> cache = VersionCache(Path(".moai/cache"))
|
|
164
|
-
>>> cache.clear()
|
|
165
|
-
True
|
|
166
|
-
"""
|
|
167
|
-
try:
|
|
168
|
-
if self.cache_file.exists():
|
|
169
|
-
self.cache_file.unlink()
|
|
170
|
-
return True
|
|
171
|
-
except OSError:
|
|
172
|
-
return False
|
|
173
|
-
|
|
174
|
-
def get_age_hours(self) -> float:
|
|
175
|
-
"""Get age of cache in hours
|
|
176
|
-
|
|
177
|
-
Returns:
|
|
178
|
-
Age in hours, or 0.0 if cache doesn't exist or is invalid
|
|
179
|
-
|
|
180
|
-
Examples:
|
|
181
|
-
>>> cache = VersionCache(Path(".moai/cache"))
|
|
182
|
-
>>> cache.get_age_hours()
|
|
183
|
-
0.0 # No cache exists
|
|
184
|
-
"""
|
|
185
|
-
if not self.cache_file.exists():
|
|
186
|
-
return 0.0
|
|
187
|
-
|
|
188
|
-
try:
|
|
189
|
-
with open(self.cache_file, "r") as f:
|
|
190
|
-
data = json.load(f)
|
|
191
|
-
|
|
192
|
-
return self._calculate_age_hours(data["last_check"])
|
|
193
|
-
|
|
194
|
-
except (json.JSONDecodeError, KeyError, ValueError, OSError):
|
|
195
|
-
return 0.0
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
__all__ = ["VersionCache"]
|
|
@@ -1,21 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env python3
|
|
2
|
-
"""Event handlers for Alfred Hooks
|
|
3
|
-
|
|
4
|
-
Claude Code Event Handlers
|
|
5
|
-
"""
|
|
6
|
-
|
|
7
|
-
from .notification import handle_notification, handle_stop, handle_subagent_stop
|
|
8
|
-
from .session import handle_session_end, handle_session_start
|
|
9
|
-
from .tool import handle_post_tool_use, handle_pre_tool_use
|
|
10
|
-
from .user import handle_user_prompt_submit
|
|
11
|
-
|
|
12
|
-
__all__ = [
|
|
13
|
-
"handle_session_start",
|
|
14
|
-
"handle_session_end",
|
|
15
|
-
"handle_user_prompt_submit",
|
|
16
|
-
"handle_pre_tool_use",
|
|
17
|
-
"handle_post_tool_use",
|
|
18
|
-
"handle_notification",
|
|
19
|
-
"handle_stop",
|
|
20
|
-
"handle_subagent_stop",
|
|
21
|
-
]
|
|
@@ -1,154 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env python3
|
|
2
|
-
"""Notification and control handlers
|
|
3
|
-
|
|
4
|
-
Notification, Stop, SubagentStop event handling
|
|
5
|
-
"""
|
|
6
|
-
|
|
7
|
-
import json
|
|
8
|
-
from datetime import datetime
|
|
9
|
-
from pathlib import Path
|
|
10
|
-
|
|
11
|
-
from core import HookPayload, HookResult
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
def _get_command_state_file(cwd: str) -> Path:
|
|
15
|
-
"""Get the path to command state tracking file"""
|
|
16
|
-
state_dir = Path(cwd) / ".moai" / "memory"
|
|
17
|
-
state_dir.mkdir(parents=True, exist_ok=True)
|
|
18
|
-
return state_dir / "command-execution-state.json"
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
def _load_command_state(cwd: str) -> dict:
|
|
22
|
-
"""Load current command execution state"""
|
|
23
|
-
try:
|
|
24
|
-
state_file = _get_command_state_file(cwd)
|
|
25
|
-
if state_file.exists():
|
|
26
|
-
with open(state_file, "r", encoding="utf-8") as f:
|
|
27
|
-
return json.load(f)
|
|
28
|
-
except Exception:
|
|
29
|
-
pass
|
|
30
|
-
return {"last_command": None, "last_timestamp": None, "is_running": False}
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
def _save_command_state(cwd: str, state: dict) -> None:
|
|
34
|
-
"""Save command execution state"""
|
|
35
|
-
try:
|
|
36
|
-
state_file = _get_command_state_file(cwd)
|
|
37
|
-
with open(state_file, "w", encoding="utf-8") as f:
|
|
38
|
-
json.dump(state, f, indent=2)
|
|
39
|
-
except Exception:
|
|
40
|
-
pass
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
def _is_duplicate_command(current_cmd: str, last_cmd: str, last_timestamp: str) -> bool:
|
|
44
|
-
"""Check if current command is a duplicate of the last one within 3 seconds"""
|
|
45
|
-
if not last_cmd or not last_timestamp or current_cmd != last_cmd:
|
|
46
|
-
return False
|
|
47
|
-
|
|
48
|
-
try:
|
|
49
|
-
last_time = datetime.fromisoformat(last_timestamp)
|
|
50
|
-
current_time = datetime.now()
|
|
51
|
-
time_diff = (current_time - last_time).total_seconds()
|
|
52
|
-
# Consider it a duplicate if same command within 3 seconds
|
|
53
|
-
return time_diff < 3
|
|
54
|
-
except Exception:
|
|
55
|
-
return False
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
def handle_notification(payload: HookPayload) -> HookResult:
|
|
59
|
-
"""Notification event handler
|
|
60
|
-
|
|
61
|
-
Detects and warns about duplicate command executions
|
|
62
|
-
(When the same /alfred: command is triggered multiple times within 3 seconds)
|
|
63
|
-
"""
|
|
64
|
-
cwd = payload.get("cwd", ".")
|
|
65
|
-
notification = payload.get("notification", {})
|
|
66
|
-
|
|
67
|
-
# Extract command information from notification
|
|
68
|
-
current_cmd = None
|
|
69
|
-
if isinstance(notification, dict):
|
|
70
|
-
# Check if notification contains command information
|
|
71
|
-
text = notification.get("text", "") or str(notification)
|
|
72
|
-
if "/alfred:" in text:
|
|
73
|
-
# Extract /alfred: command
|
|
74
|
-
import re
|
|
75
|
-
|
|
76
|
-
match = re.search(r"/alfred:\S+", text)
|
|
77
|
-
if match:
|
|
78
|
-
current_cmd = match.group()
|
|
79
|
-
|
|
80
|
-
if not current_cmd:
|
|
81
|
-
return HookResult()
|
|
82
|
-
|
|
83
|
-
# Load current state
|
|
84
|
-
state = _load_command_state(cwd)
|
|
85
|
-
last_cmd = state.get("last_command")
|
|
86
|
-
last_timestamp = state.get("last_timestamp")
|
|
87
|
-
|
|
88
|
-
# Check for duplicate
|
|
89
|
-
if _is_duplicate_command(current_cmd, last_cmd, last_timestamp):
|
|
90
|
-
warning_msg = (
|
|
91
|
-
f"⚠️ Duplicate command detected: '{current_cmd}' "
|
|
92
|
-
f"is running multiple times within 3 seconds.\n"
|
|
93
|
-
f"This may indicate a system issue. Check logs in `.moai/logs/command-invocations.log`"
|
|
94
|
-
)
|
|
95
|
-
|
|
96
|
-
# Update state - mark as duplicate detected
|
|
97
|
-
state["duplicate_detected"] = True
|
|
98
|
-
state["duplicate_command"] = current_cmd
|
|
99
|
-
state["duplicate_timestamp"] = datetime.now().isoformat()
|
|
100
|
-
_save_command_state(cwd, state)
|
|
101
|
-
|
|
102
|
-
return HookResult(system_message=warning_msg, continue_execution=True)
|
|
103
|
-
|
|
104
|
-
# Update state with current command
|
|
105
|
-
state["last_command"] = current_cmd
|
|
106
|
-
state["last_timestamp"] = datetime.now().isoformat()
|
|
107
|
-
state["is_running"] = True
|
|
108
|
-
state["duplicate_detected"] = False
|
|
109
|
-
_save_command_state(cwd, state)
|
|
110
|
-
|
|
111
|
-
return HookResult()
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
def handle_stop(payload: HookPayload) -> HookResult:
|
|
115
|
-
"""Stop event handler
|
|
116
|
-
|
|
117
|
-
Marks command execution as complete
|
|
118
|
-
"""
|
|
119
|
-
cwd = payload.get("cwd", ".")
|
|
120
|
-
state = _load_command_state(cwd)
|
|
121
|
-
state["is_running"] = False
|
|
122
|
-
state["last_timestamp"] = datetime.now().isoformat()
|
|
123
|
-
_save_command_state(cwd, state)
|
|
124
|
-
|
|
125
|
-
return HookResult()
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
def handle_subagent_stop(payload: HookPayload) -> HookResult:
|
|
129
|
-
"""SubagentStop event handler
|
|
130
|
-
|
|
131
|
-
Records when a sub-agent finishes execution
|
|
132
|
-
"""
|
|
133
|
-
cwd = payload.get("cwd", ".")
|
|
134
|
-
|
|
135
|
-
# Extract subagent name if available
|
|
136
|
-
subagent_name = (
|
|
137
|
-
payload.get("subagent", {}).get("name")
|
|
138
|
-
if isinstance(payload.get("subagent"), dict)
|
|
139
|
-
else None
|
|
140
|
-
)
|
|
141
|
-
|
|
142
|
-
try:
|
|
143
|
-
state_file = _get_command_state_file(cwd).parent / "subagent-execution.log"
|
|
144
|
-
timestamp = datetime.now().isoformat()
|
|
145
|
-
|
|
146
|
-
with open(state_file, "a", encoding="utf-8") as f:
|
|
147
|
-
f.write(f"{timestamp} | Subagent Stop | {subagent_name}\n")
|
|
148
|
-
except Exception:
|
|
149
|
-
pass
|
|
150
|
-
|
|
151
|
-
return HookResult()
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
__all__ = ["handle_notification", "handle_stop", "handle_subagent_stop"]
|