moai-adk 0.10.1__py3-none-any.whl → 0.11.0__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/core/project/phase_executor.py +4 -0
- moai_adk/core/tags/ci_validator.py +33 -3
- moai_adk/core/template_engine.py +6 -2
- moai_adk/templates/.github/workflows/moai-gitflow.yml +6 -1
- moai_adk/templates/.github/workflows/release.yml +6 -2
- moai_adk/templates/.github/workflows/tag-validation.yml +53 -8
- moai_adk/templates/CLAUDE.md +458 -67
- {moai_adk-0.10.1.dist-info → moai_adk-0.11.0.dist-info}/METADATA +28 -13
- moai_adk-0.11.0.dist-info/RECORD +77 -0
- 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/doc-syncer.md +0 -214
- moai_adk/templates/.claude/agents/alfred/git-manager.md +0 -406
- moai_adk/templates/.claude/agents/alfred/implementation-planner.md +0 -350
- moai_adk/templates/.claude/agents/alfred/project-manager.md +0 -273
- 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 -287
- moai_adk/templates/.claude/agents/alfred/tag-agent.md +0 -287
- moai_adk/templates/.claude/agents/alfred/tdd-implementer.md +0 -326
- moai_adk/templates/.claude/agents/alfred/trust-checker.md +0 -375
- moai_adk/templates/.claude/commands/alfred/0-project.md +0 -1189
- moai_adk/templates/.claude/commands/alfred/1-plan.md +0 -728
- moai_adk/templates/.claude/commands/alfred/2-run.md +0 -545
- moai_adk/templates/.claude/commands/alfred/3-sync.md +0 -683
- moai_adk/templates/.claude/commands/alfred/9-feedback.md +0 -149
- moai_adk/templates/.claude/hooks/alfred/alfred_hooks.py +0 -209
- moai_adk/templates/.claude/hooks/alfred/core/project.py +0 -750
- moai_adk/templates/.claude/hooks/alfred/core/version_cache.py +0 -198
- moai_adk/templates/.claude/hooks/alfred/notification__handle_events.py +0 -102
- moai_adk/templates/.claude/hooks/alfred/post_tool__log_changes.py +0 -102
- moai_adk/templates/.claude/hooks/alfred/pre_tool__auto_checkpoint.py +0 -108
- moai_adk/templates/.claude/hooks/alfred/session_end__cleanup.py +0 -102
- moai_adk/templates/.claude/hooks/alfred/session_start__show_project_info.py +0 -102
- 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 -756
- moai_adk/templates/.claude/hooks/alfred/shared/core/tags.py +0 -198
- 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 -25
- moai_adk/templates/.claude/hooks/alfred/shared/handlers/session.py +0 -175
- moai_adk/templates/.claude/hooks/alfred/shared/handlers/tool.py +0 -90
- moai_adk/templates/.claude/hooks/alfred/shared/handlers/user.py +0 -61
- moai_adk/templates/.claude/hooks/alfred/stop__handle_interrupt.py +0 -102
- moai_adk/templates/.claude/hooks/alfred/subagent_stop__handle_subagent_end.py +0 -102
- moai_adk/templates/.claude/hooks/alfred/user_prompt__jit_load_docs.py +0 -120
- moai_adk/templates/.claude/output-styles/alfred/agentic-coding.md +0 -640
- moai_adk/templates/.claude/output-styles/alfred/moai-adk-learning.md +0 -696
- moai_adk/templates/.claude/output-styles/alfred/study-with-alfred.md +0 -474
- moai_adk/templates/.claude/settings.json +0 -144
- 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-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-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-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-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-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-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-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-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 -124
- 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 -124
- 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 -124
- 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 -123
- 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 -431
- 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 -124
- 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 -123
- 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 -127
- 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-skill-factory/CHECKLIST.md +0 -482
- moai_adk/templates/.claude/skills/moai-skill-factory/EXAMPLES.md +0 -278
- moai_adk/templates/.claude/skills/moai-skill-factory/INTERACTIVE-DISCOVERY.md +0 -524
- moai_adk/templates/.claude/skills/moai-skill-factory/METADATA.md +0 -477
- moai_adk/templates/.claude/skills/moai-skill-factory/PARALLEL-ANALYSIS-REPORT.md +0 -429
- moai_adk/templates/.claude/skills/moai-skill-factory/PYTHON-VERSION-MATRIX.md +0 -391
- moai_adk/templates/.claude/skills/moai-skill-factory/SKILL-FACTORY-WORKFLOW.md +0 -431
- moai_adk/templates/.claude/skills/moai-skill-factory/SKILL-UPDATE-ADVISOR.md +0 -577
- moai_adk/templates/.claude/skills/moai-skill-factory/SKILL.md +0 -271
- moai_adk/templates/.claude/skills/moai-skill-factory/STEP-BY-STEP-GUIDE.md +0 -466
- moai_adk/templates/.claude/skills/moai-skill-factory/STRUCTURE.md +0 -583
- moai_adk/templates/.claude/skills/moai-skill-factory/WEB-RESEARCH.md +0 -526
- moai_adk/templates/.claude/skills/moai-skill-factory/reference.md +0 -465
- moai_adk/templates/.claude/skills/moai-skill-factory/scripts/generate-structure.sh +0 -328
- moai_adk/templates/.claude/skills/moai-skill-factory/scripts/validate-skill.sh +0 -312
- moai_adk/templates/.claude/skills/moai-skill-factory/templates/SKILL_TEMPLATE.md +0 -245
- moai_adk/templates/.claude/skills/moai-skill-factory/templates/examples-template.md +0 -285
- moai_adk/templates/.claude/skills/moai-skill-factory/templates/reference-template.md +0 -278
- moai_adk/templates/.claude/skills/moai-skill-factory/templates/scripts-template.sh +0 -303
- moai_adk/templates/.claude/skills/moai-spec-authoring/README.md +0 -137
- moai_adk/templates/.claude/skills/moai-spec-authoring/SKILL.md +0 -219
- moai_adk/templates/.claude/skills/moai-spec-authoring/examples/validate-spec.sh +0 -161
- moai_adk/templates/.claude/skills/moai-spec-authoring/examples.md +0 -541
- moai_adk/templates/.claude/skills/moai-spec-authoring/reference.md +0 -622
- moai_adk/templates/.moai/config.json +0 -113
- moai_adk/templates/.moai/memory/CLAUDE-AGENTS-GUIDE.md +0 -208
- moai_adk/templates/.moai/memory/CLAUDE-PRACTICES.md +0 -369
- moai_adk/templates/.moai/memory/CLAUDE-RULES.md +0 -539
- moai_adk/templates/.moai/memory/DEVELOPMENT-GUIDE.md +0 -344
- moai_adk/templates/.moai/memory/ISSUE-LABEL-MAPPING.md +0 -150
- moai_adk/templates/.moai/memory/SKILLS-DESCRIPTION-POLICY.md +0 -218
- moai_adk/templates/.moai/memory/gitflow-protection-policy.md +0 -330
- moai_adk/templates/.moai/memory/spec-metadata.md +0 -356
- moai_adk/templates/.moai/project/product.md +0 -161
- moai_adk/templates/.moai/project/structure.md +0 -156
- moai_adk/templates/.moai/project/tech.md +0 -227
- moai_adk-0.10.1.dist-info/RECORD +0 -309
- {moai_adk-0.10.1.dist-info → moai_adk-0.11.0.dist-info}/WHEEL +0 -0
- {moai_adk-0.10.1.dist-info → moai_adk-0.11.0.dist-info}/entry_points.txt +0 -0
- {moai_adk-0.10.1.dist-info → moai_adk-0.11.0.dist-info}/licenses/LICENSE +0 -0
|
@@ -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 = 24):
|
|
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 24)
|
|
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,102 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env python3
|
|
2
|
-
# @CODE:HOOKS-CLARITY-001 | SPEC: Individual hook files for better UX
|
|
3
|
-
"""Notification Hook: Handle System Notifications
|
|
4
|
-
|
|
5
|
-
Claude Code Event: Notification
|
|
6
|
-
Purpose: Process system notifications and alerts from Claude Code
|
|
7
|
-
Execution: Triggered when Claude Code sends notification events
|
|
8
|
-
|
|
9
|
-
Output: Continue execution (currently a stub for future enhancements)
|
|
10
|
-
"""
|
|
11
|
-
|
|
12
|
-
import json
|
|
13
|
-
import signal
|
|
14
|
-
import sys
|
|
15
|
-
from pathlib import Path
|
|
16
|
-
from typing import Any
|
|
17
|
-
|
|
18
|
-
# Setup import path for shared modules
|
|
19
|
-
HOOKS_DIR = Path(__file__).parent
|
|
20
|
-
SHARED_DIR = HOOKS_DIR / "shared"
|
|
21
|
-
if str(SHARED_DIR) not in sys.path:
|
|
22
|
-
sys.path.insert(0, str(SHARED_DIR))
|
|
23
|
-
|
|
24
|
-
from handlers import handle_notification
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
class HookTimeoutError(Exception):
|
|
28
|
-
"""Hook execution timeout exception"""
|
|
29
|
-
pass
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
def _timeout_handler(signum, frame):
|
|
33
|
-
"""Signal handler for 5-second timeout"""
|
|
34
|
-
raise HookTimeoutError("Hook execution exceeded 5-second timeout")
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
def main() -> None:
|
|
38
|
-
"""Main entry point for Notification hook
|
|
39
|
-
|
|
40
|
-
Currently a stub for future functionality:
|
|
41
|
-
- Filter and categorize notifications
|
|
42
|
-
- Send alerts to external systems (Slack, email)
|
|
43
|
-
- Log important events
|
|
44
|
-
- Trigger automated responses
|
|
45
|
-
|
|
46
|
-
Exit Codes:
|
|
47
|
-
0: Success
|
|
48
|
-
1: Error (timeout, JSON parse failure, handler exception)
|
|
49
|
-
"""
|
|
50
|
-
# Set 5-second timeout
|
|
51
|
-
signal.signal(signal.SIGALRM, _timeout_handler)
|
|
52
|
-
signal.alarm(5)
|
|
53
|
-
|
|
54
|
-
try:
|
|
55
|
-
# Read JSON payload from stdin
|
|
56
|
-
input_data = sys.stdin.read()
|
|
57
|
-
data = json.loads(input_data) if input_data.strip() else {}
|
|
58
|
-
|
|
59
|
-
# Call handler
|
|
60
|
-
result = handle_notification(data)
|
|
61
|
-
|
|
62
|
-
# Output result as JSON
|
|
63
|
-
print(json.dumps(result.to_dict()))
|
|
64
|
-
sys.exit(0)
|
|
65
|
-
|
|
66
|
-
except HookTimeoutError:
|
|
67
|
-
# Timeout - return minimal valid response
|
|
68
|
-
timeout_response: dict[str, Any] = {
|
|
69
|
-
"continue": True,
|
|
70
|
-
"systemMessage": "⚠️ Notification handler timeout"
|
|
71
|
-
}
|
|
72
|
-
print(json.dumps(timeout_response))
|
|
73
|
-
print("Notification hook timeout after 5 seconds", file=sys.stderr)
|
|
74
|
-
sys.exit(1)
|
|
75
|
-
|
|
76
|
-
except json.JSONDecodeError as e:
|
|
77
|
-
# JSON parse error
|
|
78
|
-
error_response: dict[str, Any] = {
|
|
79
|
-
"continue": True,
|
|
80
|
-
"hookSpecificOutput": {"error": f"JSON parse error: {e}"}
|
|
81
|
-
}
|
|
82
|
-
print(json.dumps(error_response))
|
|
83
|
-
print(f"Notification JSON parse error: {e}", file=sys.stderr)
|
|
84
|
-
sys.exit(1)
|
|
85
|
-
|
|
86
|
-
except Exception as e:
|
|
87
|
-
# Unexpected error
|
|
88
|
-
error_response: dict[str, Any] = {
|
|
89
|
-
"continue": True,
|
|
90
|
-
"hookSpecificOutput": {"error": f"Notification error: {e}"}
|
|
91
|
-
}
|
|
92
|
-
print(json.dumps(error_response))
|
|
93
|
-
print(f"Notification unexpected error: {e}", file=sys.stderr)
|
|
94
|
-
sys.exit(1)
|
|
95
|
-
|
|
96
|
-
finally:
|
|
97
|
-
# Always cancel alarm
|
|
98
|
-
signal.alarm(0)
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
if __name__ == "__main__":
|
|
102
|
-
main()
|
|
@@ -1,102 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env python3
|
|
2
|
-
# @CODE:HOOKS-CLARITY-001 | SPEC: Individual hook files for better UX
|
|
3
|
-
"""PostToolUse Hook: Log Tool Usage and Changes
|
|
4
|
-
|
|
5
|
-
Claude Code Event: PostToolUse
|
|
6
|
-
Purpose: Log tool execution results and track changes for audit trail
|
|
7
|
-
Execution: Triggered after Edit, Write, or MultiEdit tools are used
|
|
8
|
-
Matcher: Edit|Write|MultiEdit
|
|
9
|
-
|
|
10
|
-
Output: Continue execution (currently a stub for future enhancements)
|
|
11
|
-
"""
|
|
12
|
-
|
|
13
|
-
import json
|
|
14
|
-
import signal
|
|
15
|
-
import sys
|
|
16
|
-
from pathlib import Path
|
|
17
|
-
from typing import Any
|
|
18
|
-
|
|
19
|
-
# Setup import path for shared modules
|
|
20
|
-
HOOKS_DIR = Path(__file__).parent
|
|
21
|
-
SHARED_DIR = HOOKS_DIR / "shared"
|
|
22
|
-
if str(SHARED_DIR) not in sys.path:
|
|
23
|
-
sys.path.insert(0, str(SHARED_DIR))
|
|
24
|
-
|
|
25
|
-
from handlers import handle_post_tool_use
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
class HookTimeoutError(Exception):
|
|
29
|
-
"""Hook execution timeout exception"""
|
|
30
|
-
pass
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
def _timeout_handler(signum, frame):
|
|
34
|
-
"""Signal handler for 5-second timeout"""
|
|
35
|
-
raise HookTimeoutError("Hook execution exceeded 5-second timeout")
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
def main() -> None:
|
|
39
|
-
"""Main entry point for PostToolUse hook
|
|
40
|
-
|
|
41
|
-
Currently a stub for future functionality:
|
|
42
|
-
- Change tracking and audit logging
|
|
43
|
-
- Metrics collection (files modified, lines changed)
|
|
44
|
-
- Integration with external monitoring systems
|
|
45
|
-
|
|
46
|
-
Exit Codes:
|
|
47
|
-
0: Success
|
|
48
|
-
1: Error (timeout, JSON parse failure, handler exception)
|
|
49
|
-
"""
|
|
50
|
-
# Set 5-second timeout
|
|
51
|
-
signal.signal(signal.SIGALRM, _timeout_handler)
|
|
52
|
-
signal.alarm(5)
|
|
53
|
-
|
|
54
|
-
try:
|
|
55
|
-
# Read JSON payload from stdin
|
|
56
|
-
input_data = sys.stdin.read()
|
|
57
|
-
data = json.loads(input_data) if input_data.strip() else {}
|
|
58
|
-
|
|
59
|
-
# Call handler
|
|
60
|
-
result = handle_post_tool_use(data)
|
|
61
|
-
|
|
62
|
-
# Output result as JSON
|
|
63
|
-
print(json.dumps(result.to_dict()))
|
|
64
|
-
sys.exit(0)
|
|
65
|
-
|
|
66
|
-
except HookTimeoutError:
|
|
67
|
-
# Timeout - return minimal valid response
|
|
68
|
-
timeout_response: dict[str, Any] = {
|
|
69
|
-
"continue": True,
|
|
70
|
-
"systemMessage": "⚠️ PostToolUse timeout - continuing"
|
|
71
|
-
}
|
|
72
|
-
print(json.dumps(timeout_response))
|
|
73
|
-
print("PostToolUse hook timeout after 5 seconds", file=sys.stderr)
|
|
74
|
-
sys.exit(1)
|
|
75
|
-
|
|
76
|
-
except json.JSONDecodeError as e:
|
|
77
|
-
# JSON parse error
|
|
78
|
-
error_response: dict[str, Any] = {
|
|
79
|
-
"continue": True,
|
|
80
|
-
"hookSpecificOutput": {"error": f"JSON parse error: {e}"}
|
|
81
|
-
}
|
|
82
|
-
print(json.dumps(error_response))
|
|
83
|
-
print(f"PostToolUse JSON parse error: {e}", file=sys.stderr)
|
|
84
|
-
sys.exit(1)
|
|
85
|
-
|
|
86
|
-
except Exception as e:
|
|
87
|
-
# Unexpected error
|
|
88
|
-
error_response: dict[str, Any] = {
|
|
89
|
-
"continue": True,
|
|
90
|
-
"hookSpecificOutput": {"error": f"PostToolUse error: {e}"}
|
|
91
|
-
}
|
|
92
|
-
print(json.dumps(error_response))
|
|
93
|
-
print(f"PostToolUse unexpected error: {e}", file=sys.stderr)
|
|
94
|
-
sys.exit(1)
|
|
95
|
-
|
|
96
|
-
finally:
|
|
97
|
-
# Always cancel alarm
|
|
98
|
-
signal.alarm(0)
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
if __name__ == "__main__":
|
|
102
|
-
main()
|
|
@@ -1,108 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env python3
|
|
2
|
-
# @CODE:HOOKS-CLARITY-001 | SPEC: Individual hook files for better UX
|
|
3
|
-
"""PreToolUse Hook: Automatic Safety Checkpoint Creation
|
|
4
|
-
|
|
5
|
-
Claude Code Event: PreToolUse
|
|
6
|
-
Purpose: Detect risky operations and automatically create Git checkpoints before execution
|
|
7
|
-
Execution: Triggered before Edit, Write, or MultiEdit tools are used
|
|
8
|
-
Matcher: Edit|Write|MultiEdit
|
|
9
|
-
|
|
10
|
-
Output: System message with checkpoint information (if created)
|
|
11
|
-
|
|
12
|
-
Risky Operations Detected:
|
|
13
|
-
- Bash: rm -rf, git merge, git reset --hard
|
|
14
|
-
- Edit/Write: CLAUDE.md, config.json, critical files
|
|
15
|
-
- MultiEdit: Operations affecting ≥10 files
|
|
16
|
-
"""
|
|
17
|
-
|
|
18
|
-
import json
|
|
19
|
-
import signal
|
|
20
|
-
import sys
|
|
21
|
-
from pathlib import Path
|
|
22
|
-
from typing import Any
|
|
23
|
-
|
|
24
|
-
# Setup import path for shared modules
|
|
25
|
-
HOOKS_DIR = Path(__file__).parent
|
|
26
|
-
SHARED_DIR = HOOKS_DIR / "shared"
|
|
27
|
-
if str(SHARED_DIR) not in sys.path:
|
|
28
|
-
sys.path.insert(0, str(SHARED_DIR))
|
|
29
|
-
|
|
30
|
-
from handlers import handle_pre_tool_use
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
class HookTimeoutError(Exception):
|
|
34
|
-
"""Hook execution timeout exception"""
|
|
35
|
-
pass
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
def _timeout_handler(signum, frame):
|
|
39
|
-
"""Signal handler for 5-second timeout"""
|
|
40
|
-
raise HookTimeoutError("Hook execution exceeded 5-second timeout")
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
def main() -> None:
|
|
44
|
-
"""Main entry point for PreToolUse hook
|
|
45
|
-
|
|
46
|
-
Analyzes tool usage and creates checkpoints for risky operations:
|
|
47
|
-
1. Detects dangerous patterns (rm -rf, git reset, etc.)
|
|
48
|
-
2. Creates Git checkpoint: checkpoint/before-{operation}-{timestamp}
|
|
49
|
-
3. Logs checkpoint to .moai/checkpoints.log
|
|
50
|
-
4. Returns guidance message to user
|
|
51
|
-
|
|
52
|
-
Exit Codes:
|
|
53
|
-
0: Success (checkpoint created or not needed)
|
|
54
|
-
1: Error (timeout, JSON parse failure, handler exception)
|
|
55
|
-
"""
|
|
56
|
-
# Set 5-second timeout
|
|
57
|
-
signal.signal(signal.SIGALRM, _timeout_handler)
|
|
58
|
-
signal.alarm(5)
|
|
59
|
-
|
|
60
|
-
try:
|
|
61
|
-
# Read JSON payload from stdin
|
|
62
|
-
input_data = sys.stdin.read()
|
|
63
|
-
data = json.loads(input_data) if input_data.strip() else {}
|
|
64
|
-
|
|
65
|
-
# Call handler
|
|
66
|
-
result = handle_pre_tool_use(data)
|
|
67
|
-
|
|
68
|
-
# Output result as JSON
|
|
69
|
-
print(json.dumps(result.to_dict()))
|
|
70
|
-
sys.exit(0)
|
|
71
|
-
|
|
72
|
-
except HookTimeoutError:
|
|
73
|
-
# Timeout - return minimal valid response (allow operation to continue)
|
|
74
|
-
timeout_response: dict[str, Any] = {
|
|
75
|
-
"continue": True,
|
|
76
|
-
"systemMessage": "⚠️ Checkpoint creation timeout - operation proceeding without checkpoint"
|
|
77
|
-
}
|
|
78
|
-
print(json.dumps(timeout_response))
|
|
79
|
-
print("PreToolUse hook timeout after 5 seconds", file=sys.stderr)
|
|
80
|
-
sys.exit(1)
|
|
81
|
-
|
|
82
|
-
except json.JSONDecodeError as e:
|
|
83
|
-
# JSON parse error - allow operation to continue
|
|
84
|
-
error_response: dict[str, Any] = {
|
|
85
|
-
"continue": True,
|
|
86
|
-
"hookSpecificOutput": {"error": f"JSON parse error: {e}"}
|
|
87
|
-
}
|
|
88
|
-
print(json.dumps(error_response))
|
|
89
|
-
print(f"PreToolUse JSON parse error: {e}", file=sys.stderr)
|
|
90
|
-
sys.exit(1)
|
|
91
|
-
|
|
92
|
-
except Exception as e:
|
|
93
|
-
# Unexpected error - allow operation to continue
|
|
94
|
-
error_response: dict[str, Any] = {
|
|
95
|
-
"continue": True,
|
|
96
|
-
"hookSpecificOutput": {"error": f"PreToolUse error: {e}"}
|
|
97
|
-
}
|
|
98
|
-
print(json.dumps(error_response))
|
|
99
|
-
print(f"PreToolUse unexpected error: {e}", file=sys.stderr)
|
|
100
|
-
sys.exit(1)
|
|
101
|
-
|
|
102
|
-
finally:
|
|
103
|
-
# Always cancel alarm
|
|
104
|
-
signal.alarm(0)
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
if __name__ == "__main__":
|
|
108
|
-
main()
|
|
@@ -1,102 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env python3
|
|
2
|
-
# @CODE:HOOKS-CLARITY-001 | SPEC: Individual hook files for better UX
|
|
3
|
-
"""SessionEnd Hook: Session Cleanup and Finalization
|
|
4
|
-
|
|
5
|
-
Claude Code Event: SessionEnd
|
|
6
|
-
Purpose: Clean up resources and finalize session when Claude Code exits
|
|
7
|
-
Execution: Triggered when Claude Code session ends
|
|
8
|
-
|
|
9
|
-
Output: Continue execution (currently a stub for future enhancements)
|
|
10
|
-
"""
|
|
11
|
-
|
|
12
|
-
import json
|
|
13
|
-
import signal
|
|
14
|
-
import sys
|
|
15
|
-
from pathlib import Path
|
|
16
|
-
from typing import Any
|
|
17
|
-
|
|
18
|
-
# Setup import path for shared modules
|
|
19
|
-
HOOKS_DIR = Path(__file__).parent
|
|
20
|
-
SHARED_DIR = HOOKS_DIR / "shared"
|
|
21
|
-
if str(SHARED_DIR) not in sys.path:
|
|
22
|
-
sys.path.insert(0, str(SHARED_DIR))
|
|
23
|
-
|
|
24
|
-
from handlers import handle_session_end
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
class HookTimeoutError(Exception):
|
|
28
|
-
"""Hook execution timeout exception"""
|
|
29
|
-
pass
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
def _timeout_handler(signum, frame):
|
|
33
|
-
"""Signal handler for 5-second timeout"""
|
|
34
|
-
raise HookTimeoutError("Hook execution exceeded 5-second timeout")
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
def main() -> None:
|
|
38
|
-
"""Main entry point for SessionEnd hook
|
|
39
|
-
|
|
40
|
-
Currently a stub for future functionality:
|
|
41
|
-
- Clear temporary caches
|
|
42
|
-
- Save session metrics
|
|
43
|
-
- Upload analytics (if enabled)
|
|
44
|
-
- Cleanup background processes
|
|
45
|
-
|
|
46
|
-
Exit Codes:
|
|
47
|
-
0: Success
|
|
48
|
-
1: Error (timeout, JSON parse failure, handler exception)
|
|
49
|
-
"""
|
|
50
|
-
# Set 5-second timeout
|
|
51
|
-
signal.signal(signal.SIGALRM, _timeout_handler)
|
|
52
|
-
signal.alarm(5)
|
|
53
|
-
|
|
54
|
-
try:
|
|
55
|
-
# Read JSON payload from stdin
|
|
56
|
-
input_data = sys.stdin.read()
|
|
57
|
-
data = json.loads(input_data) if input_data.strip() else {}
|
|
58
|
-
|
|
59
|
-
# Call handler
|
|
60
|
-
result = handle_session_end(data)
|
|
61
|
-
|
|
62
|
-
# Output result as JSON
|
|
63
|
-
print(json.dumps(result.to_dict()))
|
|
64
|
-
sys.exit(0)
|
|
65
|
-
|
|
66
|
-
except HookTimeoutError:
|
|
67
|
-
# Timeout - return minimal valid response
|
|
68
|
-
timeout_response: dict[str, Any] = {
|
|
69
|
-
"continue": True,
|
|
70
|
-
"systemMessage": "⚠️ SessionEnd cleanup timeout - session ending anyway"
|
|
71
|
-
}
|
|
72
|
-
print(json.dumps(timeout_response))
|
|
73
|
-
print("SessionEnd hook timeout after 5 seconds", file=sys.stderr)
|
|
74
|
-
sys.exit(1)
|
|
75
|
-
|
|
76
|
-
except json.JSONDecodeError as e:
|
|
77
|
-
# JSON parse error
|
|
78
|
-
error_response: dict[str, Any] = {
|
|
79
|
-
"continue": True,
|
|
80
|
-
"hookSpecificOutput": {"error": f"JSON parse error: {e}"}
|
|
81
|
-
}
|
|
82
|
-
print(json.dumps(error_response))
|
|
83
|
-
print(f"SessionEnd JSON parse error: {e}", file=sys.stderr)
|
|
84
|
-
sys.exit(1)
|
|
85
|
-
|
|
86
|
-
except Exception as e:
|
|
87
|
-
# Unexpected error
|
|
88
|
-
error_response: dict[str, Any] = {
|
|
89
|
-
"continue": True,
|
|
90
|
-
"hookSpecificOutput": {"error": f"SessionEnd error: {e}"}
|
|
91
|
-
}
|
|
92
|
-
print(json.dumps(error_response))
|
|
93
|
-
print(f"SessionEnd unexpected error: {e}", file=sys.stderr)
|
|
94
|
-
sys.exit(1)
|
|
95
|
-
|
|
96
|
-
finally:
|
|
97
|
-
# Always cancel alarm
|
|
98
|
-
signal.alarm(0)
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
if __name__ == "__main__":
|
|
102
|
-
main()
|