moai-adk 0.8.1__py3-none-any.whl → 0.8.2__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.

Files changed (87) hide show
  1. moai_adk/cli/commands/update.py +15 -4
  2. moai_adk/core/tags/__init__.py +87 -0
  3. moai_adk/core/tags/ci_validator.py +435 -0
  4. moai_adk/core/tags/cli.py +283 -0
  5. moai_adk/core/tags/generator.py +109 -0
  6. moai_adk/core/tags/inserter.py +99 -0
  7. moai_adk/core/tags/mapper.py +126 -0
  8. moai_adk/core/tags/parser.py +76 -0
  9. moai_adk/core/tags/pre_commit_validator.py +355 -0
  10. moai_adk/core/tags/reporter.py +959 -0
  11. moai_adk/core/tags/tags.py +149 -0
  12. moai_adk/core/tags/validator.py +897 -0
  13. moai_adk/templates/.claude/agents/alfred/cc-manager.md +25 -2
  14. moai_adk/templates/.claude/agents/alfred/debug-helper.md +24 -12
  15. moai_adk/templates/.claude/agents/alfred/doc-syncer.md +19 -12
  16. moai_adk/templates/.claude/agents/alfred/git-manager.md +20 -12
  17. moai_adk/templates/.claude/agents/alfred/implementation-planner.md +19 -12
  18. moai_adk/templates/.claude/agents/alfred/project-manager.md +29 -2
  19. moai_adk/templates/.claude/agents/alfred/quality-gate.md +25 -2
  20. moai_adk/templates/.claude/agents/alfred/skill-factory.md +30 -2
  21. moai_adk/templates/.claude/agents/alfred/spec-builder.md +26 -11
  22. moai_adk/templates/.claude/agents/alfred/tag-agent.md +30 -8
  23. moai_adk/templates/.claude/agents/alfred/tdd-implementer.md +27 -12
  24. moai_adk/templates/.claude/agents/alfred/trust-checker.md +25 -2
  25. moai_adk/templates/.claude/commands/alfred/0-project.md +5 -0
  26. moai_adk/templates/.claude/commands/alfred/1-plan.md +17 -4
  27. moai_adk/templates/.claude/commands/alfred/2-run.md +7 -0
  28. moai_adk/templates/.claude/commands/alfred/3-sync.md +6 -0
  29. moai_adk/templates/.claude/hooks/alfred/.moai/cache/version-check.json +9 -0
  30. moai_adk/templates/.claude/hooks/alfred/README.md +258 -145
  31. moai_adk/templates/.claude/hooks/alfred/TROUBLESHOOTING.md +471 -0
  32. moai_adk/templates/.claude/hooks/alfred/alfred_hooks.py +92 -57
  33. moai_adk/templates/.claude/hooks/alfred/core/version_cache.py +198 -0
  34. moai_adk/templates/.claude/hooks/alfred/notification__handle_events.py +102 -0
  35. moai_adk/templates/.claude/hooks/alfred/post_tool__log_changes.py +102 -0
  36. moai_adk/templates/.claude/hooks/alfred/pre_tool__auto_checkpoint.py +108 -0
  37. moai_adk/templates/.claude/hooks/alfred/session_end__cleanup.py +102 -0
  38. moai_adk/templates/.claude/hooks/alfred/session_start__show_project_info.py +102 -0
  39. moai_adk/templates/.claude/hooks/alfred/{core → shared/core}/project.py +269 -13
  40. moai_adk/templates/.claude/hooks/alfred/shared/core/version_cache.py +198 -0
  41. moai_adk/templates/.claude/hooks/alfred/{handlers → shared/handlers}/session.py +21 -7
  42. moai_adk/templates/.claude/hooks/alfred/stop__handle_interrupt.py +102 -0
  43. moai_adk/templates/.claude/hooks/alfred/subagent_stop__handle_subagent_end.py +102 -0
  44. moai_adk/templates/.claude/hooks/alfred/user_prompt__jit_load_docs.py +120 -0
  45. moai_adk/templates/.claude/settings.json +5 -5
  46. moai_adk/templates/.claude/skills/moai-foundation-ears/SKILL.md +9 -6
  47. moai_adk/templates/.claude/skills/moai-spec-authoring/README.md +56 -56
  48. moai_adk/templates/.claude/skills/moai-spec-authoring/SKILL.md +101 -100
  49. moai_adk/templates/.claude/skills/moai-spec-authoring/examples/validate-spec.sh +3 -3
  50. moai_adk/templates/.claude/skills/moai-spec-authoring/examples.md +219 -219
  51. moai_adk/templates/.claude/skills/moai-spec-authoring/reference.md +287 -287
  52. moai_adk/templates/.github/ISSUE_TEMPLATE/spec.yml +9 -11
  53. moai_adk/templates/.github/PULL_REQUEST_TEMPLATE.md +9 -21
  54. moai_adk/templates/.github/workflows/moai-release-create.yml +100 -0
  55. moai_adk/templates/.github/workflows/moai-release-pipeline.yml +182 -0
  56. moai_adk/templates/.github/workflows/release.yml +49 -0
  57. moai_adk/templates/.github/workflows/tag-report.yml +261 -0
  58. moai_adk/templates/.github/workflows/tag-validation.yml +176 -0
  59. moai_adk/templates/.moai/config.json +6 -1
  60. moai_adk/templates/.moai/hooks/install.sh +79 -0
  61. moai_adk/templates/.moai/hooks/pre-commit.sh +66 -0
  62. moai_adk/templates/CLAUDE.md +39 -40
  63. moai_adk/templates/src/moai_adk/core/__init__.py +5 -0
  64. moai_adk/templates/src/moai_adk/core/tags/__init__.py +87 -0
  65. moai_adk/templates/src/moai_adk/core/tags/ci_validator.py +435 -0
  66. moai_adk/templates/src/moai_adk/core/tags/cli.py +283 -0
  67. moai_adk/templates/src/moai_adk/core/tags/pre_commit_validator.py +355 -0
  68. moai_adk/templates/src/moai_adk/core/tags/reporter.py +959 -0
  69. moai_adk/templates/src/moai_adk/core/tags/validator.py +897 -0
  70. {moai_adk-0.8.1.dist-info → moai_adk-0.8.2.dist-info}/METADATA +226 -1
  71. {moai_adk-0.8.1.dist-info → moai_adk-0.8.2.dist-info}/RECORD +83 -50
  72. moai_adk/templates/.claude/hooks/alfred/HOOK_SCHEMA_VALIDATION.md +0 -313
  73. moai_adk/templates/.moai/memory/config-schema.md +0 -444
  74. moai_adk/templates/.moai/memory/gitflow-protection-policy.md +0 -220
  75. moai_adk/templates/.moai/memory/spec-metadata.md +0 -356
  76. /moai_adk/templates/.claude/hooks/alfred/{core → shared/core}/__init__.py +0 -0
  77. /moai_adk/templates/.claude/hooks/alfred/{core → shared/core}/checkpoint.py +0 -0
  78. /moai_adk/templates/.claude/hooks/alfred/{core → shared/core}/context.py +0 -0
  79. /moai_adk/templates/.claude/hooks/alfred/{core → shared/core}/tags.py +0 -0
  80. /moai_adk/templates/.claude/hooks/alfred/{handlers → shared/handlers}/__init__.py +0 -0
  81. /moai_adk/templates/.claude/hooks/alfred/{handlers → shared/handlers}/notification.py +0 -0
  82. /moai_adk/templates/.claude/hooks/alfred/{handlers → shared/handlers}/tool.py +0 -0
  83. /moai_adk/templates/.claude/hooks/alfred/{handlers → shared/handlers}/user.py +0 -0
  84. /moai_adk/templates/.moai/memory/{issue-label-mapping.md → ISSUE-LABEL-MAPPING.md} +0 -0
  85. {moai_adk-0.8.1.dist-info → moai_adk-0.8.2.dist-info}/WHEEL +0 -0
  86. {moai_adk-0.8.1.dist-info → moai_adk-0.8.2.dist-info}/entry_points.txt +0 -0
  87. {moai_adk-0.8.1.dist-info → moai_adk-0.8.2.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,198 @@
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) as e:
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"]
@@ -0,0 +1,102 @@
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()
@@ -0,0 +1,102 @@
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()
@@ -0,0 +1,108 @@
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()
@@ -0,0 +1,102 @@
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()