empathy-framework 4.6.6__py3-none-any.whl → 4.7.1__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.
Files changed (273) hide show
  1. empathy_framework-4.7.1.dist-info/METADATA +690 -0
  2. empathy_framework-4.7.1.dist-info/RECORD +379 -0
  3. {empathy_framework-4.6.6.dist-info → empathy_framework-4.7.1.dist-info}/top_level.txt +1 -2
  4. empathy_healthcare_plugin/monitors/monitoring/__init__.py +9 -9
  5. empathy_llm_toolkit/agent_factory/__init__.py +6 -6
  6. empathy_llm_toolkit/agent_factory/adapters/wizard_adapter.py +7 -10
  7. empathy_llm_toolkit/agents_md/__init__.py +22 -0
  8. empathy_llm_toolkit/agents_md/loader.py +218 -0
  9. empathy_llm_toolkit/agents_md/parser.py +271 -0
  10. empathy_llm_toolkit/agents_md/registry.py +307 -0
  11. empathy_llm_toolkit/commands/__init__.py +51 -0
  12. empathy_llm_toolkit/commands/context.py +375 -0
  13. empathy_llm_toolkit/commands/loader.py +301 -0
  14. empathy_llm_toolkit/commands/models.py +231 -0
  15. empathy_llm_toolkit/commands/parser.py +371 -0
  16. empathy_llm_toolkit/commands/registry.py +429 -0
  17. empathy_llm_toolkit/config/__init__.py +8 -8
  18. empathy_llm_toolkit/config/unified.py +3 -7
  19. empathy_llm_toolkit/context/__init__.py +22 -0
  20. empathy_llm_toolkit/context/compaction.py +455 -0
  21. empathy_llm_toolkit/context/manager.py +434 -0
  22. empathy_llm_toolkit/hooks/__init__.py +24 -0
  23. empathy_llm_toolkit/hooks/config.py +306 -0
  24. empathy_llm_toolkit/hooks/executor.py +289 -0
  25. empathy_llm_toolkit/hooks/registry.py +302 -0
  26. empathy_llm_toolkit/hooks/scripts/__init__.py +39 -0
  27. empathy_llm_toolkit/hooks/scripts/evaluate_session.py +201 -0
  28. empathy_llm_toolkit/hooks/scripts/first_time_init.py +285 -0
  29. empathy_llm_toolkit/hooks/scripts/pre_compact.py +207 -0
  30. empathy_llm_toolkit/hooks/scripts/session_end.py +183 -0
  31. empathy_llm_toolkit/hooks/scripts/session_start.py +163 -0
  32. empathy_llm_toolkit/hooks/scripts/suggest_compact.py +225 -0
  33. empathy_llm_toolkit/learning/__init__.py +30 -0
  34. empathy_llm_toolkit/learning/evaluator.py +438 -0
  35. empathy_llm_toolkit/learning/extractor.py +514 -0
  36. empathy_llm_toolkit/learning/storage.py +560 -0
  37. empathy_llm_toolkit/providers.py +4 -11
  38. empathy_llm_toolkit/security/__init__.py +17 -17
  39. empathy_llm_toolkit/utils/tokens.py +2 -5
  40. empathy_os/__init__.py +202 -70
  41. empathy_os/cache_monitor.py +5 -3
  42. empathy_os/cli/__init__.py +11 -55
  43. empathy_os/cli/__main__.py +29 -15
  44. empathy_os/cli/commands/inspection.py +21 -12
  45. empathy_os/cli/commands/memory.py +4 -12
  46. empathy_os/cli/commands/profiling.py +198 -0
  47. empathy_os/cli/commands/utilities.py +27 -7
  48. empathy_os/cli.py +28 -57
  49. empathy_os/cli_unified.py +525 -1164
  50. empathy_os/cost_tracker.py +9 -3
  51. empathy_os/dashboard/server.py +200 -2
  52. empathy_os/hot_reload/__init__.py +7 -7
  53. empathy_os/hot_reload/config.py +6 -7
  54. empathy_os/hot_reload/integration.py +35 -35
  55. empathy_os/hot_reload/reloader.py +57 -57
  56. empathy_os/hot_reload/watcher.py +28 -28
  57. empathy_os/hot_reload/websocket.py +2 -2
  58. empathy_os/memory/__init__.py +11 -4
  59. empathy_os/memory/claude_memory.py +1 -1
  60. empathy_os/memory/cross_session.py +8 -12
  61. empathy_os/memory/edges.py +6 -6
  62. empathy_os/memory/file_session.py +770 -0
  63. empathy_os/memory/graph.py +30 -30
  64. empathy_os/memory/nodes.py +6 -6
  65. empathy_os/memory/short_term.py +15 -9
  66. empathy_os/memory/unified.py +606 -140
  67. empathy_os/meta_workflows/agent_creator.py +3 -9
  68. empathy_os/meta_workflows/cli_meta_workflows.py +113 -53
  69. empathy_os/meta_workflows/form_engine.py +6 -18
  70. empathy_os/meta_workflows/intent_detector.py +64 -24
  71. empathy_os/meta_workflows/models.py +3 -1
  72. empathy_os/meta_workflows/pattern_learner.py +13 -31
  73. empathy_os/meta_workflows/plan_generator.py +55 -47
  74. empathy_os/meta_workflows/session_context.py +2 -3
  75. empathy_os/meta_workflows/workflow.py +20 -51
  76. empathy_os/models/cli.py +2 -2
  77. empathy_os/models/tasks.py +1 -2
  78. empathy_os/models/telemetry.py +4 -1
  79. empathy_os/models/token_estimator.py +3 -1
  80. empathy_os/monitoring/alerts.py +938 -9
  81. empathy_os/monitoring/alerts_cli.py +346 -183
  82. empathy_os/orchestration/execution_strategies.py +12 -29
  83. empathy_os/orchestration/pattern_learner.py +20 -26
  84. empathy_os/orchestration/real_tools.py +6 -15
  85. empathy_os/platform_utils.py +2 -1
  86. empathy_os/plugins/__init__.py +2 -2
  87. empathy_os/plugins/base.py +64 -64
  88. empathy_os/plugins/registry.py +32 -32
  89. empathy_os/project_index/index.py +49 -15
  90. empathy_os/project_index/models.py +1 -2
  91. empathy_os/project_index/reports.py +1 -1
  92. empathy_os/project_index/scanner.py +1 -0
  93. empathy_os/redis_memory.py +10 -7
  94. empathy_os/resilience/__init__.py +1 -1
  95. empathy_os/resilience/health.py +10 -10
  96. empathy_os/routing/__init__.py +7 -7
  97. empathy_os/routing/chain_executor.py +37 -37
  98. empathy_os/routing/classifier.py +36 -36
  99. empathy_os/routing/smart_router.py +40 -40
  100. empathy_os/routing/{wizard_registry.py → workflow_registry.py} +47 -47
  101. empathy_os/scaffolding/__init__.py +8 -8
  102. empathy_os/scaffolding/__main__.py +1 -1
  103. empathy_os/scaffolding/cli.py +28 -28
  104. empathy_os/socratic/__init__.py +3 -19
  105. empathy_os/socratic/ab_testing.py +25 -36
  106. empathy_os/socratic/blueprint.py +38 -38
  107. empathy_os/socratic/cli.py +34 -20
  108. empathy_os/socratic/collaboration.py +30 -28
  109. empathy_os/socratic/domain_templates.py +9 -1
  110. empathy_os/socratic/embeddings.py +17 -13
  111. empathy_os/socratic/engine.py +135 -70
  112. empathy_os/socratic/explainer.py +70 -60
  113. empathy_os/socratic/feedback.py +24 -19
  114. empathy_os/socratic/forms.py +15 -10
  115. empathy_os/socratic/generator.py +51 -35
  116. empathy_os/socratic/llm_analyzer.py +25 -23
  117. empathy_os/socratic/mcp_server.py +99 -159
  118. empathy_os/socratic/session.py +19 -13
  119. empathy_os/socratic/storage.py +98 -67
  120. empathy_os/socratic/success.py +38 -27
  121. empathy_os/socratic/visual_editor.py +51 -39
  122. empathy_os/socratic/web_ui.py +99 -66
  123. empathy_os/telemetry/cli.py +3 -1
  124. empathy_os/telemetry/usage_tracker.py +1 -3
  125. empathy_os/test_generator/__init__.py +3 -3
  126. empathy_os/test_generator/cli.py +28 -28
  127. empathy_os/test_generator/generator.py +64 -66
  128. empathy_os/test_generator/risk_analyzer.py +11 -11
  129. empathy_os/vscode_bridge 2.py +173 -0
  130. empathy_os/vscode_bridge.py +173 -0
  131. empathy_os/workflows/__init__.py +212 -120
  132. empathy_os/workflows/batch_processing.py +8 -24
  133. empathy_os/workflows/bug_predict.py +1 -1
  134. empathy_os/workflows/code_review.py +20 -5
  135. empathy_os/workflows/code_review_pipeline.py +13 -8
  136. empathy_os/workflows/keyboard_shortcuts/workflow.py +6 -2
  137. empathy_os/workflows/manage_documentation.py +1 -0
  138. empathy_os/workflows/orchestrated_health_check.py +6 -11
  139. empathy_os/workflows/orchestrated_release_prep.py +3 -3
  140. empathy_os/workflows/pr_review.py +18 -10
  141. empathy_os/workflows/progressive/README 2.md +454 -0
  142. empathy_os/workflows/progressive/__init__ 2.py +92 -0
  143. empathy_os/workflows/progressive/__init__.py +2 -12
  144. empathy_os/workflows/progressive/cli 2.py +242 -0
  145. empathy_os/workflows/progressive/cli.py +14 -37
  146. empathy_os/workflows/progressive/core 2.py +488 -0
  147. empathy_os/workflows/progressive/core.py +12 -12
  148. empathy_os/workflows/progressive/orchestrator 2.py +701 -0
  149. empathy_os/workflows/progressive/orchestrator.py +166 -144
  150. empathy_os/workflows/progressive/reports 2.py +528 -0
  151. empathy_os/workflows/progressive/reports.py +22 -31
  152. empathy_os/workflows/progressive/telemetry 2.py +280 -0
  153. empathy_os/workflows/progressive/telemetry.py +8 -14
  154. empathy_os/workflows/progressive/test_gen 2.py +514 -0
  155. empathy_os/workflows/progressive/test_gen.py +29 -48
  156. empathy_os/workflows/progressive/workflow 2.py +628 -0
  157. empathy_os/workflows/progressive/workflow.py +31 -70
  158. empathy_os/workflows/release_prep.py +21 -6
  159. empathy_os/workflows/release_prep_crew.py +1 -0
  160. empathy_os/workflows/secure_release.py +13 -6
  161. empathy_os/workflows/security_audit.py +8 -3
  162. empathy_os/workflows/test_coverage_boost_crew.py +3 -2
  163. empathy_os/workflows/test_maintenance_crew.py +1 -0
  164. empathy_os/workflows/test_runner.py +16 -12
  165. empathy_software_plugin/SOFTWARE_PLUGIN_README.md +25 -703
  166. empathy_software_plugin/cli.py +0 -122
  167. patterns/README.md +119 -0
  168. patterns/__init__.py +95 -0
  169. patterns/behavior.py +298 -0
  170. patterns/code_review_memory.json +441 -0
  171. patterns/core.py +97 -0
  172. patterns/debugging.json +3763 -0
  173. patterns/empathy.py +268 -0
  174. patterns/health_check_memory.json +505 -0
  175. patterns/input.py +161 -0
  176. patterns/memory_graph.json +8 -0
  177. patterns/refactoring_memory.json +1113 -0
  178. patterns/registry.py +663 -0
  179. patterns/security_memory.json +8 -0
  180. patterns/structural.py +415 -0
  181. patterns/validation.py +194 -0
  182. coach_wizards/__init__.py +0 -45
  183. coach_wizards/accessibility_wizard.py +0 -91
  184. coach_wizards/api_wizard.py +0 -91
  185. coach_wizards/base_wizard.py +0 -209
  186. coach_wizards/cicd_wizard.py +0 -91
  187. coach_wizards/code_reviewer_README.md +0 -60
  188. coach_wizards/code_reviewer_wizard.py +0 -180
  189. coach_wizards/compliance_wizard.py +0 -91
  190. coach_wizards/database_wizard.py +0 -91
  191. coach_wizards/debugging_wizard.py +0 -91
  192. coach_wizards/documentation_wizard.py +0 -91
  193. coach_wizards/generate_wizards.py +0 -347
  194. coach_wizards/localization_wizard.py +0 -173
  195. coach_wizards/migration_wizard.py +0 -91
  196. coach_wizards/monitoring_wizard.py +0 -91
  197. coach_wizards/observability_wizard.py +0 -91
  198. coach_wizards/performance_wizard.py +0 -91
  199. coach_wizards/prompt_engineering_wizard.py +0 -661
  200. coach_wizards/refactoring_wizard.py +0 -91
  201. coach_wizards/scaling_wizard.py +0 -90
  202. coach_wizards/security_wizard.py +0 -92
  203. coach_wizards/testing_wizard.py +0 -91
  204. empathy_framework-4.6.6.dist-info/METADATA +0 -1597
  205. empathy_framework-4.6.6.dist-info/RECORD +0 -410
  206. empathy_llm_toolkit/wizards/__init__.py +0 -43
  207. empathy_llm_toolkit/wizards/base_wizard.py +0 -364
  208. empathy_llm_toolkit/wizards/customer_support_wizard.py +0 -190
  209. empathy_llm_toolkit/wizards/healthcare_wizard.py +0 -378
  210. empathy_llm_toolkit/wizards/patient_assessment_README.md +0 -64
  211. empathy_llm_toolkit/wizards/patient_assessment_wizard.py +0 -193
  212. empathy_llm_toolkit/wizards/technology_wizard.py +0 -209
  213. empathy_os/wizard_factory_cli.py +0 -170
  214. empathy_software_plugin/wizards/__init__.py +0 -42
  215. empathy_software_plugin/wizards/advanced_debugging_wizard.py +0 -395
  216. empathy_software_plugin/wizards/agent_orchestration_wizard.py +0 -511
  217. empathy_software_plugin/wizards/ai_collaboration_wizard.py +0 -503
  218. empathy_software_plugin/wizards/ai_context_wizard.py +0 -441
  219. empathy_software_plugin/wizards/ai_documentation_wizard.py +0 -503
  220. empathy_software_plugin/wizards/base_wizard.py +0 -288
  221. empathy_software_plugin/wizards/book_chapter_wizard.py +0 -519
  222. empathy_software_plugin/wizards/code_review_wizard.py +0 -604
  223. empathy_software_plugin/wizards/debugging/__init__.py +0 -50
  224. empathy_software_plugin/wizards/debugging/bug_risk_analyzer.py +0 -414
  225. empathy_software_plugin/wizards/debugging/config_loaders.py +0 -446
  226. empathy_software_plugin/wizards/debugging/fix_applier.py +0 -469
  227. empathy_software_plugin/wizards/debugging/language_patterns.py +0 -385
  228. empathy_software_plugin/wizards/debugging/linter_parsers.py +0 -470
  229. empathy_software_plugin/wizards/debugging/verification.py +0 -369
  230. empathy_software_plugin/wizards/enhanced_testing_wizard.py +0 -537
  231. empathy_software_plugin/wizards/memory_enhanced_debugging_wizard.py +0 -816
  232. empathy_software_plugin/wizards/multi_model_wizard.py +0 -501
  233. empathy_software_plugin/wizards/pattern_extraction_wizard.py +0 -422
  234. empathy_software_plugin/wizards/pattern_retriever_wizard.py +0 -400
  235. empathy_software_plugin/wizards/performance/__init__.py +0 -9
  236. empathy_software_plugin/wizards/performance/bottleneck_detector.py +0 -221
  237. empathy_software_plugin/wizards/performance/profiler_parsers.py +0 -278
  238. empathy_software_plugin/wizards/performance/trajectory_analyzer.py +0 -429
  239. empathy_software_plugin/wizards/performance_profiling_wizard.py +0 -305
  240. empathy_software_plugin/wizards/prompt_engineering_wizard.py +0 -425
  241. empathy_software_plugin/wizards/rag_pattern_wizard.py +0 -461
  242. empathy_software_plugin/wizards/security/__init__.py +0 -32
  243. empathy_software_plugin/wizards/security/exploit_analyzer.py +0 -290
  244. empathy_software_plugin/wizards/security/owasp_patterns.py +0 -241
  245. empathy_software_plugin/wizards/security/vulnerability_scanner.py +0 -604
  246. empathy_software_plugin/wizards/security_analysis_wizard.py +0 -322
  247. empathy_software_plugin/wizards/security_learning_wizard.py +0 -740
  248. empathy_software_plugin/wizards/tech_debt_wizard.py +0 -726
  249. empathy_software_plugin/wizards/testing/__init__.py +0 -27
  250. empathy_software_plugin/wizards/testing/coverage_analyzer.py +0 -459
  251. empathy_software_plugin/wizards/testing/quality_analyzer.py +0 -525
  252. empathy_software_plugin/wizards/testing/test_suggester.py +0 -533
  253. empathy_software_plugin/wizards/testing_wizard.py +0 -274
  254. wizards/__init__.py +0 -82
  255. wizards/admission_assessment_wizard.py +0 -644
  256. wizards/care_plan.py +0 -321
  257. wizards/clinical_assessment.py +0 -769
  258. wizards/discharge_planning.py +0 -77
  259. wizards/discharge_summary_wizard.py +0 -468
  260. wizards/dosage_calculation.py +0 -497
  261. wizards/incident_report_wizard.py +0 -454
  262. wizards/medication_reconciliation.py +0 -85
  263. wizards/nursing_assessment.py +0 -171
  264. wizards/patient_education.py +0 -654
  265. wizards/quality_improvement.py +0 -705
  266. wizards/sbar_report.py +0 -324
  267. wizards/sbar_wizard.py +0 -608
  268. wizards/shift_handoff_wizard.py +0 -535
  269. wizards/soap_note_wizard.py +0 -679
  270. wizards/treatment_plan.py +0 -15
  271. {empathy_framework-4.6.6.dist-info → empathy_framework-4.7.1.dist-info}/WHEEL +0 -0
  272. {empathy_framework-4.6.6.dist-info → empathy_framework-4.7.1.dist-info}/entry_points.txt +0 -0
  273. {empathy_framework-4.6.6.dist-info → empathy_framework-4.7.1.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,302 @@
1
+ """Hook Registry
2
+
3
+ Central registry for managing and dispatching hooks by event type.
4
+
5
+ Copyright 2025 Smart-AI-Memory
6
+ Licensed under Fair Source License 0.9
7
+ """
8
+
9
+ import logging
10
+ from collections.abc import Callable
11
+ from typing import Any
12
+
13
+ from empathy_llm_toolkit.hooks.config import (
14
+ HookConfig,
15
+ HookDefinition,
16
+ HookEvent,
17
+ HookMatcher,
18
+ HookRule,
19
+ HookType,
20
+ )
21
+
22
+ logger = logging.getLogger(__name__)
23
+
24
+
25
+ class HookRegistry:
26
+ """Central registry for hook management and dispatch.
27
+
28
+ The registry handles:
29
+ - Loading hook configuration
30
+ - Matching events to hooks
31
+ - Dispatching hooks to the executor
32
+ - Tracking hook execution results
33
+
34
+ Example:
35
+ registry = HookRegistry()
36
+ registry.load_config(config)
37
+
38
+ # Register a Python function hook
39
+ registry.register(
40
+ event=HookEvent.SESSION_START,
41
+ handler=my_session_start_handler,
42
+ description="Initialize session state"
43
+ )
44
+
45
+ # Fire hooks for an event
46
+ results = await registry.fire(
47
+ HookEvent.SESSION_START,
48
+ context={"session_id": "abc123"}
49
+ )
50
+ """
51
+
52
+ def __init__(self, config: HookConfig | None = None):
53
+ """Initialize the hook registry.
54
+
55
+ Args:
56
+ config: Optional hook configuration to load
57
+
58
+ """
59
+ self.config = config or HookConfig()
60
+ self._python_handlers: dict[str, Callable] = {}
61
+ self._execution_log: list[dict[str, Any]] = []
62
+
63
+ def load_config(self, config: HookConfig) -> None:
64
+ """Load or replace hook configuration.
65
+
66
+ Args:
67
+ config: New hook configuration
68
+
69
+ """
70
+ self.config = config
71
+ logger.info("Loaded hook configuration with %d event types", len(config.hooks))
72
+
73
+ def register(
74
+ self,
75
+ event: HookEvent,
76
+ handler: Callable[..., Any],
77
+ description: str = "",
78
+ matcher: HookMatcher | None = None,
79
+ priority: int = 0,
80
+ ) -> str:
81
+ """Register a Python function as a hook handler.
82
+
83
+ Args:
84
+ event: Event type to hook
85
+ handler: Python callable to execute
86
+ description: Human-readable description
87
+ matcher: Optional matcher for conditional execution
88
+ priority: Execution priority (higher = earlier)
89
+
90
+ Returns:
91
+ Handler ID for later reference
92
+
93
+ """
94
+ # Generate unique handler ID
95
+ handler_id = f"{event.value}_{handler.__name__}_{id(handler)}"
96
+ self._python_handlers[handler_id] = handler
97
+
98
+ # Create hook definition pointing to the handler
99
+ hook_def = HookDefinition(
100
+ type=HookType.PYTHON,
101
+ command=handler_id,
102
+ description=description,
103
+ )
104
+
105
+ self.config.add_hook(
106
+ event=event,
107
+ hook=hook_def,
108
+ matcher=matcher,
109
+ priority=priority,
110
+ )
111
+
112
+ logger.debug("Registered hook '%s' for event %s", handler_id, event.value)
113
+ return handler_id
114
+
115
+ def unregister(self, handler_id: str) -> bool:
116
+ """Unregister a hook handler by ID.
117
+
118
+ Args:
119
+ handler_id: Handler ID returned from register()
120
+
121
+ Returns:
122
+ True if handler was found and removed
123
+
124
+ """
125
+ if handler_id in self._python_handlers:
126
+ del self._python_handlers[handler_id]
127
+ # Remove from config rules
128
+ for event_rules in self.config.hooks.values():
129
+ for rule in event_rules:
130
+ rule.hooks = [h for h in rule.hooks if h.command != handler_id]
131
+ logger.debug("Unregistered hook '%s'", handler_id)
132
+ return True
133
+ return False
134
+
135
+ def get_matching_hooks(
136
+ self,
137
+ event: HookEvent,
138
+ context: dict[str, Any],
139
+ ) -> list[tuple[HookRule, HookDefinition]]:
140
+ """Get all hooks that match the given event and context.
141
+
142
+ Args:
143
+ event: Event type
144
+ context: Event context for matching
145
+
146
+ Returns:
147
+ List of (rule, hook) tuples that match
148
+
149
+ """
150
+ if not self.config.enabled:
151
+ return []
152
+
153
+ matching = []
154
+ for rule in self.config.get_hooks_for_event(event):
155
+ if rule.matcher.matches(context):
156
+ for hook in rule.hooks:
157
+ matching.append((rule, hook))
158
+
159
+ return matching
160
+
161
+ async def fire(
162
+ self,
163
+ event: HookEvent,
164
+ context: dict[str, Any] | None = None,
165
+ ) -> list[dict[str, Any]]:
166
+ """Fire all matching hooks for an event.
167
+
168
+ Args:
169
+ event: Event type to fire
170
+ context: Event context for matching and execution
171
+
172
+ Returns:
173
+ List of execution results
174
+
175
+ """
176
+ from empathy_llm_toolkit.hooks.executor import HookExecutor
177
+
178
+ context = context or {}
179
+ matching_hooks = self.get_matching_hooks(event, context)
180
+
181
+ if not matching_hooks:
182
+ logger.debug("No hooks matched for event %s", event.value)
183
+ return []
184
+
185
+ logger.info(
186
+ "Firing %d hook(s) for event %s",
187
+ len(matching_hooks),
188
+ event.value,
189
+ )
190
+
191
+ executor = HookExecutor(python_handlers=self._python_handlers)
192
+ results = []
193
+
194
+ for _rule, hook in matching_hooks:
195
+ try:
196
+ result = await executor.execute(hook, context)
197
+ execution_record = {
198
+ "event": event.value,
199
+ "hook": hook.command,
200
+ "description": hook.description,
201
+ "success": result.get("success", False),
202
+ "output": result.get("output"),
203
+ "error": result.get("error"),
204
+ "duration_ms": result.get("duration_ms"),
205
+ }
206
+ results.append(execution_record)
207
+
208
+ if self.config.log_executions:
209
+ self._execution_log.append(execution_record)
210
+
211
+ except Exception as e:
212
+ error_record = {
213
+ "event": event.value,
214
+ "hook": hook.command,
215
+ "success": False,
216
+ "error": str(e),
217
+ }
218
+ results.append(error_record)
219
+ logger.error("Hook execution failed: %s - %s", hook.command, e)
220
+
221
+ if hook.on_error == "raise":
222
+ raise
223
+
224
+ return results
225
+
226
+ def fire_sync(
227
+ self,
228
+ event: HookEvent,
229
+ context: dict[str, Any] | None = None,
230
+ ) -> list[dict[str, Any]]:
231
+ """Synchronous version of fire().
232
+
233
+ Args:
234
+ event: Event type to fire
235
+ context: Event context
236
+
237
+ Returns:
238
+ List of execution results
239
+
240
+ """
241
+ import asyncio
242
+
243
+ try:
244
+ loop = asyncio.get_running_loop()
245
+ # If we're in an async context, use run_coroutine_threadsafe
246
+
247
+ future = asyncio.run_coroutine_threadsafe(
248
+ self.fire(event, context),
249
+ loop,
250
+ )
251
+ return future.result(timeout=self.config.default_timeout)
252
+ except RuntimeError:
253
+ # No running loop, create one
254
+ return asyncio.run(self.fire(event, context))
255
+
256
+ def get_execution_log(
257
+ self,
258
+ limit: int = 100,
259
+ event_filter: HookEvent | None = None,
260
+ ) -> list[dict[str, Any]]:
261
+ """Get recent hook execution log entries.
262
+
263
+ Args:
264
+ limit: Maximum entries to return
265
+ event_filter: Optional event type filter
266
+
267
+ Returns:
268
+ List of execution records
269
+
270
+ """
271
+ log = self._execution_log
272
+ if event_filter:
273
+ log = [e for e in log if e.get("event") == event_filter.value]
274
+ return log[-limit:]
275
+
276
+ def clear_execution_log(self) -> None:
277
+ """Clear the execution log."""
278
+ self._execution_log.clear()
279
+
280
+ def get_stats(self) -> dict[str, Any]:
281
+ """Get hook registry statistics.
282
+
283
+ Returns:
284
+ Dictionary with stats
285
+
286
+ """
287
+ total_hooks = sum(
288
+ sum(len(rule.hooks) for rule in rules) for rules in self.config.hooks.values()
289
+ )
290
+
291
+ executions = len(self._execution_log)
292
+ successes = sum(1 for e in self._execution_log if e.get("success"))
293
+
294
+ return {
295
+ "enabled": self.config.enabled,
296
+ "total_hooks": total_hooks,
297
+ "event_types": len([v for v in self.config.hooks.values() if v]),
298
+ "python_handlers": len(self._python_handlers),
299
+ "total_executions": executions,
300
+ "successful_executions": successes,
301
+ "success_rate": (successes / executions * 100) if executions else 0,
302
+ }
@@ -0,0 +1,39 @@
1
+ """Hook Scripts
2
+
3
+ Pre-built hook scripts for common Empathy Framework events.
4
+
5
+ Architectural patterns inspired by everything-claude-code by Affaan Mustafa.
6
+ See: https://github.com/affaan-m/everything-claude-code (MIT License)
7
+ See: ACKNOWLEDGMENTS.md for full attribution.
8
+
9
+ Copyright 2025 Smart-AI-Memory
10
+ Licensed under Fair Source License 0.9
11
+ """
12
+
13
+ from empathy_llm_toolkit.hooks.scripts.evaluate_session import (
14
+ apply_learned_patterns,
15
+ get_learning_summary,
16
+ run_evaluate_session,
17
+ )
18
+ from empathy_llm_toolkit.hooks.scripts.first_time_init import (
19
+ check_init,
20
+ handle_init_response,
21
+ initialize_project,
22
+ )
23
+ from empathy_llm_toolkit.hooks.scripts.pre_compact import run_pre_compact
24
+ from empathy_llm_toolkit.hooks.scripts.session_end import main as session_end
25
+ from empathy_llm_toolkit.hooks.scripts.session_start import main as session_start
26
+ from empathy_llm_toolkit.hooks.scripts.suggest_compact import main as suggest_compact
27
+
28
+ __all__ = [
29
+ "session_start",
30
+ "session_end",
31
+ "suggest_compact",
32
+ "run_pre_compact",
33
+ "run_evaluate_session",
34
+ "get_learning_summary",
35
+ "apply_learned_patterns",
36
+ "check_init",
37
+ "handle_init_response",
38
+ "initialize_project",
39
+ ]
@@ -0,0 +1,201 @@
1
+ """Evaluate Session Hook Script
2
+
3
+ Evaluates sessions at end for learning potential and extracts patterns.
4
+ Called during SessionEnd hook to enable continuous learning.
5
+
6
+ Architectural patterns inspired by everything-claude-code by Affaan Mustafa.
7
+ See: https://github.com/affaan-m/everything-claude-code (MIT License)
8
+ See: ACKNOWLEDGMENTS.md for full attribution.
9
+
10
+ Copyright 2025 Smart AI Memory, LLC
11
+ Licensed under Fair Source 0.9
12
+ """
13
+
14
+ from __future__ import annotations
15
+
16
+ import logging
17
+ from typing import Any
18
+
19
+ logger = logging.getLogger(__name__)
20
+
21
+
22
+ def run_evaluate_session(context: dict[str, Any]) -> dict[str, Any]:
23
+ """Evaluate a session for learning potential.
24
+
25
+ This hook is called at session end to:
26
+ 1. Evaluate if the session has learning value
27
+ 2. Extract patterns if valuable
28
+ 3. Store patterns for future use
29
+
30
+ Args:
31
+ context: Hook context containing:
32
+ - collaboration_state: CollaborationState
33
+ - user_id: User identifier
34
+ - session_id: Session identifier
35
+ - storage_dir: Optional storage directory
36
+ - min_score: Optional minimum score for extraction
37
+
38
+ Returns:
39
+ Dict with:
40
+ - evaluated: Whether evaluation completed
41
+ - quality: Session quality rating
42
+ - score: Numeric score (0.0-1.0)
43
+ - patterns_extracted: Number of patterns extracted
44
+ - patterns_stored: Number of patterns stored
45
+ - learnable_topics: List of identified topics
46
+ - reasoning: Human-readable evaluation reasoning
47
+ - error: Error message if failed
48
+ """
49
+ try:
50
+ collaboration_state = context.get("collaboration_state")
51
+ user_id = context.get("user_id")
52
+ session_id = context.get("session_id", "")
53
+ storage_dir = context.get("storage_dir", ".empathy/learned_skills")
54
+ min_score = context.get("min_score", 0.4)
55
+
56
+ if not collaboration_state:
57
+ return {
58
+ "evaluated": False,
59
+ "error": "No collaboration state provided",
60
+ }
61
+
62
+ if not user_id:
63
+ user_id = collaboration_state.user_id
64
+
65
+ # Import learning components
66
+ from empathy_llm_toolkit.learning import (
67
+ LearnedSkillsStorage,
68
+ PatternExtractor,
69
+ SessionEvaluator,
70
+ )
71
+
72
+ # Evaluate the session
73
+ evaluator = SessionEvaluator(min_score_for_extraction=min_score)
74
+ evaluation = evaluator.evaluate(collaboration_state)
75
+
76
+ result = {
77
+ "evaluated": True,
78
+ "quality": evaluation.quality.value,
79
+ "score": evaluation.score,
80
+ "learnable_topics": evaluation.learnable_topics,
81
+ "reasoning": evaluation.reasoning,
82
+ "patterns_extracted": 0,
83
+ "patterns_stored": 0,
84
+ }
85
+
86
+ # Extract patterns if recommended
87
+ if evaluation.recommended_extraction:
88
+ extractor = PatternExtractor()
89
+ patterns = extractor.extract_patterns(collaboration_state, session_id)
90
+
91
+ result["patterns_extracted"] = len(patterns)
92
+
93
+ if patterns:
94
+ # Store patterns
95
+ storage = LearnedSkillsStorage(storage_dir=storage_dir)
96
+ stored_ids = storage.save_patterns(user_id, patterns)
97
+ result["patterns_stored"] = len(stored_ids)
98
+
99
+ # Include pattern summaries
100
+ result["pattern_summaries"] = [
101
+ {
102
+ "id": p.pattern_id,
103
+ "category": p.category.value,
104
+ "trigger": p.trigger[:50],
105
+ "confidence": p.confidence,
106
+ }
107
+ for p in patterns[:5] # Top 5
108
+ ]
109
+
110
+ logger.info(
111
+ f"Extracted {len(patterns)} patterns, stored {len(stored_ids)} "
112
+ f"for user {user_id}"
113
+ )
114
+ else:
115
+ logger.debug(f"Session not recommended for extraction (score: {evaluation.score:.2f})")
116
+
117
+ return result
118
+
119
+ except Exception as e:
120
+ logger.exception(f"Session evaluation failed: {e}")
121
+ return {
122
+ "evaluated": False,
123
+ "error": str(e),
124
+ }
125
+
126
+
127
+ def get_learning_summary(context: dict[str, Any]) -> dict[str, Any]:
128
+ """Get learning summary for a user.
129
+
130
+ Args:
131
+ context: Context with user_id and optional storage_dir
132
+
133
+ Returns:
134
+ Learning summary dictionary
135
+ """
136
+ try:
137
+ user_id = context.get("user_id")
138
+ storage_dir = context.get("storage_dir", ".empathy/learned_skills")
139
+
140
+ if not user_id:
141
+ return {"error": "No user_id provided"}
142
+
143
+ from empathy_llm_toolkit.learning import LearnedSkillsStorage
144
+
145
+ storage = LearnedSkillsStorage(storage_dir=storage_dir)
146
+ return storage.get_summary(user_id)
147
+
148
+ except Exception as e:
149
+ logger.exception(f"Failed to get learning summary: {e}")
150
+ return {"error": str(e)}
151
+
152
+
153
+ def apply_learned_patterns(context: dict[str, Any]) -> str:
154
+ """Generate context injection from learned patterns.
155
+
156
+ Args:
157
+ context: Context with user_id, max_patterns, and optional filters
158
+
159
+ Returns:
160
+ Formatted markdown for context injection
161
+ """
162
+ try:
163
+ user_id = context.get("user_id")
164
+ max_patterns = context.get("max_patterns", 5)
165
+ storage_dir = context.get("storage_dir", ".empathy/learned_skills")
166
+
167
+ if not user_id:
168
+ return ""
169
+
170
+ from empathy_llm_toolkit.learning import LearnedSkillsStorage, PatternCategory
171
+
172
+ # Parse category filter if provided
173
+ categories = None
174
+ if cat_filter := context.get("categories"):
175
+ categories = [PatternCategory(c) for c in cat_filter]
176
+
177
+ storage = LearnedSkillsStorage(storage_dir=storage_dir)
178
+ return storage.format_patterns_for_context(
179
+ user_id,
180
+ max_patterns=max_patterns,
181
+ categories=categories,
182
+ )
183
+
184
+ except Exception as e:
185
+ logger.exception(f"Failed to apply learned patterns: {e}")
186
+ return ""
187
+
188
+
189
+ # Entry point for hook execution
190
+ if __name__ == "__main__":
191
+ print("Evaluate Session Hook Script")
192
+ print("This script is called at session end to evaluate learning potential")
193
+ print()
194
+ print("Required context keys:")
195
+ print(" - collaboration_state: CollaborationState object")
196
+ print(" - user_id: User identifier (optional, uses state.user_id)")
197
+ print()
198
+ print("Optional context keys:")
199
+ print(" - session_id: Session identifier")
200
+ print(" - storage_dir: Storage directory path")
201
+ print(" - min_score: Minimum score for extraction (default: 0.4)")