attune-ai 2.0.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.
- attune/__init__.py +358 -0
- attune/adaptive/__init__.py +13 -0
- attune/adaptive/task_complexity.py +127 -0
- attune/agent_monitoring.py +414 -0
- attune/cache/__init__.py +117 -0
- attune/cache/base.py +166 -0
- attune/cache/dependency_manager.py +256 -0
- attune/cache/hash_only.py +251 -0
- attune/cache/hybrid.py +457 -0
- attune/cache/storage.py +285 -0
- attune/cache_monitor.py +356 -0
- attune/cache_stats.py +298 -0
- attune/cli/__init__.py +152 -0
- attune/cli/__main__.py +12 -0
- attune/cli/commands/__init__.py +1 -0
- attune/cli/commands/batch.py +264 -0
- attune/cli/commands/cache.py +248 -0
- attune/cli/commands/help.py +331 -0
- attune/cli/commands/info.py +140 -0
- attune/cli/commands/inspect.py +436 -0
- attune/cli/commands/inspection.py +57 -0
- attune/cli/commands/memory.py +48 -0
- attune/cli/commands/metrics.py +92 -0
- attune/cli/commands/orchestrate.py +184 -0
- attune/cli/commands/patterns.py +207 -0
- attune/cli/commands/profiling.py +202 -0
- attune/cli/commands/provider.py +98 -0
- attune/cli/commands/routing.py +285 -0
- attune/cli/commands/setup.py +96 -0
- attune/cli/commands/status.py +235 -0
- attune/cli/commands/sync.py +166 -0
- attune/cli/commands/tier.py +121 -0
- attune/cli/commands/utilities.py +114 -0
- attune/cli/commands/workflow.py +579 -0
- attune/cli/core.py +32 -0
- attune/cli/parsers/__init__.py +68 -0
- attune/cli/parsers/batch.py +118 -0
- attune/cli/parsers/cache.py +65 -0
- attune/cli/parsers/help.py +41 -0
- attune/cli/parsers/info.py +26 -0
- attune/cli/parsers/inspect.py +66 -0
- attune/cli/parsers/metrics.py +42 -0
- attune/cli/parsers/orchestrate.py +61 -0
- attune/cli/parsers/patterns.py +54 -0
- attune/cli/parsers/provider.py +40 -0
- attune/cli/parsers/routing.py +110 -0
- attune/cli/parsers/setup.py +42 -0
- attune/cli/parsers/status.py +47 -0
- attune/cli/parsers/sync.py +31 -0
- attune/cli/parsers/tier.py +33 -0
- attune/cli/parsers/workflow.py +77 -0
- attune/cli/utils/__init__.py +1 -0
- attune/cli/utils/data.py +242 -0
- attune/cli/utils/helpers.py +68 -0
- attune/cli_legacy.py +3957 -0
- attune/cli_minimal.py +1159 -0
- attune/cli_router.py +437 -0
- attune/cli_unified.py +814 -0
- attune/config/__init__.py +66 -0
- attune/config/xml_config.py +286 -0
- attune/config.py +545 -0
- attune/coordination.py +870 -0
- attune/core.py +1511 -0
- attune/core_modules/__init__.py +15 -0
- attune/cost_tracker.py +626 -0
- attune/dashboard/__init__.py +41 -0
- attune/dashboard/app.py +512 -0
- attune/dashboard/simple_server.py +435 -0
- attune/dashboard/standalone_server.py +547 -0
- attune/discovery.py +306 -0
- attune/emergence.py +306 -0
- attune/exceptions.py +123 -0
- attune/feedback_loops.py +373 -0
- attune/hot_reload/README.md +473 -0
- attune/hot_reload/__init__.py +62 -0
- attune/hot_reload/config.py +83 -0
- attune/hot_reload/integration.py +229 -0
- attune/hot_reload/reloader.py +298 -0
- attune/hot_reload/watcher.py +183 -0
- attune/hot_reload/websocket.py +177 -0
- attune/levels.py +577 -0
- attune/leverage_points.py +441 -0
- attune/logging_config.py +261 -0
- attune/mcp/__init__.py +10 -0
- attune/mcp/server.py +506 -0
- attune/memory/__init__.py +237 -0
- attune/memory/claude_memory.py +469 -0
- attune/memory/config.py +224 -0
- attune/memory/control_panel.py +1290 -0
- attune/memory/control_panel_support.py +145 -0
- attune/memory/cross_session.py +845 -0
- attune/memory/edges.py +179 -0
- attune/memory/encryption.py +159 -0
- attune/memory/file_session.py +770 -0
- attune/memory/graph.py +570 -0
- attune/memory/long_term.py +913 -0
- attune/memory/long_term_types.py +99 -0
- attune/memory/mixins/__init__.py +25 -0
- attune/memory/mixins/backend_init_mixin.py +249 -0
- attune/memory/mixins/capabilities_mixin.py +208 -0
- attune/memory/mixins/handoff_mixin.py +208 -0
- attune/memory/mixins/lifecycle_mixin.py +49 -0
- attune/memory/mixins/long_term_mixin.py +352 -0
- attune/memory/mixins/promotion_mixin.py +109 -0
- attune/memory/mixins/short_term_mixin.py +182 -0
- attune/memory/nodes.py +179 -0
- attune/memory/redis_bootstrap.py +540 -0
- attune/memory/security/__init__.py +31 -0
- attune/memory/security/audit_logger.py +932 -0
- attune/memory/security/pii_scrubber.py +640 -0
- attune/memory/security/secrets_detector.py +678 -0
- attune/memory/short_term.py +2192 -0
- attune/memory/simple_storage.py +302 -0
- attune/memory/storage/__init__.py +15 -0
- attune/memory/storage_backend.py +167 -0
- attune/memory/summary_index.py +583 -0
- attune/memory/types.py +446 -0
- attune/memory/unified.py +182 -0
- attune/meta_workflows/__init__.py +74 -0
- attune/meta_workflows/agent_creator.py +248 -0
- attune/meta_workflows/builtin_templates.py +567 -0
- attune/meta_workflows/cli_commands/__init__.py +56 -0
- attune/meta_workflows/cli_commands/agent_commands.py +321 -0
- attune/meta_workflows/cli_commands/analytics_commands.py +442 -0
- attune/meta_workflows/cli_commands/config_commands.py +232 -0
- attune/meta_workflows/cli_commands/memory_commands.py +182 -0
- attune/meta_workflows/cli_commands/template_commands.py +354 -0
- attune/meta_workflows/cli_commands/workflow_commands.py +382 -0
- attune/meta_workflows/cli_meta_workflows.py +59 -0
- attune/meta_workflows/form_engine.py +292 -0
- attune/meta_workflows/intent_detector.py +409 -0
- attune/meta_workflows/models.py +569 -0
- attune/meta_workflows/pattern_learner.py +738 -0
- attune/meta_workflows/plan_generator.py +384 -0
- attune/meta_workflows/session_context.py +397 -0
- attune/meta_workflows/template_registry.py +229 -0
- attune/meta_workflows/workflow.py +984 -0
- attune/metrics/__init__.py +12 -0
- attune/metrics/collector.py +31 -0
- attune/metrics/prompt_metrics.py +194 -0
- attune/models/__init__.py +172 -0
- attune/models/__main__.py +13 -0
- attune/models/adaptive_routing.py +437 -0
- attune/models/auth_cli.py +444 -0
- attune/models/auth_strategy.py +450 -0
- attune/models/cli.py +655 -0
- attune/models/empathy_executor.py +354 -0
- attune/models/executor.py +257 -0
- attune/models/fallback.py +762 -0
- attune/models/provider_config.py +282 -0
- attune/models/registry.py +472 -0
- attune/models/tasks.py +359 -0
- attune/models/telemetry/__init__.py +71 -0
- attune/models/telemetry/analytics.py +594 -0
- attune/models/telemetry/backend.py +196 -0
- attune/models/telemetry/data_models.py +431 -0
- attune/models/telemetry/storage.py +489 -0
- attune/models/token_estimator.py +420 -0
- attune/models/validation.py +280 -0
- attune/monitoring/__init__.py +52 -0
- attune/monitoring/alerts.py +946 -0
- attune/monitoring/alerts_cli.py +448 -0
- attune/monitoring/multi_backend.py +271 -0
- attune/monitoring/otel_backend.py +362 -0
- attune/optimization/__init__.py +19 -0
- attune/optimization/context_optimizer.py +272 -0
- attune/orchestration/__init__.py +67 -0
- attune/orchestration/agent_templates.py +707 -0
- attune/orchestration/config_store.py +499 -0
- attune/orchestration/execution_strategies.py +2111 -0
- attune/orchestration/meta_orchestrator.py +1168 -0
- attune/orchestration/pattern_learner.py +696 -0
- attune/orchestration/real_tools.py +931 -0
- attune/pattern_cache.py +187 -0
- attune/pattern_library.py +542 -0
- attune/patterns/debugging/all_patterns.json +81 -0
- attune/patterns/debugging/workflow_20260107_1770825e.json +77 -0
- attune/patterns/refactoring_memory.json +89 -0
- attune/persistence.py +564 -0
- attune/platform_utils.py +265 -0
- attune/plugins/__init__.py +28 -0
- attune/plugins/base.py +361 -0
- attune/plugins/registry.py +268 -0
- attune/project_index/__init__.py +32 -0
- attune/project_index/cli.py +335 -0
- attune/project_index/index.py +667 -0
- attune/project_index/models.py +504 -0
- attune/project_index/reports.py +474 -0
- attune/project_index/scanner.py +777 -0
- attune/project_index/scanner_parallel.py +291 -0
- attune/prompts/__init__.py +61 -0
- attune/prompts/config.py +77 -0
- attune/prompts/context.py +177 -0
- attune/prompts/parser.py +285 -0
- attune/prompts/registry.py +313 -0
- attune/prompts/templates.py +208 -0
- attune/redis_config.py +302 -0
- attune/redis_memory.py +799 -0
- attune/resilience/__init__.py +56 -0
- attune/resilience/circuit_breaker.py +256 -0
- attune/resilience/fallback.py +179 -0
- attune/resilience/health.py +300 -0
- attune/resilience/retry.py +209 -0
- attune/resilience/timeout.py +135 -0
- attune/routing/__init__.py +43 -0
- attune/routing/chain_executor.py +433 -0
- attune/routing/classifier.py +217 -0
- attune/routing/smart_router.py +234 -0
- attune/routing/workflow_registry.py +343 -0
- attune/scaffolding/README.md +589 -0
- attune/scaffolding/__init__.py +35 -0
- attune/scaffolding/__main__.py +14 -0
- attune/scaffolding/cli.py +240 -0
- attune/scaffolding/templates/base_wizard.py.jinja2 +121 -0
- attune/scaffolding/templates/coach_wizard.py.jinja2 +321 -0
- attune/scaffolding/templates/domain_wizard.py.jinja2 +408 -0
- attune/scaffolding/templates/linear_flow_wizard.py.jinja2 +203 -0
- attune/socratic/__init__.py +256 -0
- attune/socratic/ab_testing.py +958 -0
- attune/socratic/blueprint.py +533 -0
- attune/socratic/cli.py +703 -0
- attune/socratic/collaboration.py +1114 -0
- attune/socratic/domain_templates.py +924 -0
- attune/socratic/embeddings.py +738 -0
- attune/socratic/engine.py +794 -0
- attune/socratic/explainer.py +682 -0
- attune/socratic/feedback.py +772 -0
- attune/socratic/forms.py +629 -0
- attune/socratic/generator.py +732 -0
- attune/socratic/llm_analyzer.py +637 -0
- attune/socratic/mcp_server.py +702 -0
- attune/socratic/session.py +312 -0
- attune/socratic/storage.py +667 -0
- attune/socratic/success.py +730 -0
- attune/socratic/visual_editor.py +860 -0
- attune/socratic/web_ui.py +958 -0
- attune/telemetry/__init__.py +39 -0
- attune/telemetry/agent_coordination.py +475 -0
- attune/telemetry/agent_tracking.py +367 -0
- attune/telemetry/approval_gates.py +545 -0
- attune/telemetry/cli.py +1231 -0
- attune/telemetry/commands/__init__.py +14 -0
- attune/telemetry/commands/dashboard_commands.py +696 -0
- attune/telemetry/event_streaming.py +409 -0
- attune/telemetry/feedback_loop.py +567 -0
- attune/telemetry/usage_tracker.py +591 -0
- attune/templates.py +754 -0
- attune/test_generator/__init__.py +38 -0
- attune/test_generator/__main__.py +14 -0
- attune/test_generator/cli.py +234 -0
- attune/test_generator/generator.py +355 -0
- attune/test_generator/risk_analyzer.py +216 -0
- attune/test_generator/templates/unit_test.py.jinja2 +272 -0
- attune/tier_recommender.py +384 -0
- attune/tools.py +183 -0
- attune/trust/__init__.py +28 -0
- attune/trust/circuit_breaker.py +579 -0
- attune/trust_building.py +527 -0
- attune/validation/__init__.py +19 -0
- attune/validation/xml_validator.py +281 -0
- attune/vscode_bridge.py +173 -0
- attune/workflow_commands.py +780 -0
- attune/workflow_patterns/__init__.py +33 -0
- attune/workflow_patterns/behavior.py +249 -0
- attune/workflow_patterns/core.py +76 -0
- attune/workflow_patterns/output.py +99 -0
- attune/workflow_patterns/registry.py +255 -0
- attune/workflow_patterns/structural.py +288 -0
- attune/workflows/__init__.py +539 -0
- attune/workflows/autonomous_test_gen.py +1268 -0
- attune/workflows/base.py +2667 -0
- attune/workflows/batch_processing.py +342 -0
- attune/workflows/bug_predict.py +1084 -0
- attune/workflows/builder.py +273 -0
- attune/workflows/caching.py +253 -0
- attune/workflows/code_review.py +1048 -0
- attune/workflows/code_review_adapters.py +312 -0
- attune/workflows/code_review_pipeline.py +722 -0
- attune/workflows/config.py +645 -0
- attune/workflows/dependency_check.py +644 -0
- attune/workflows/document_gen/__init__.py +25 -0
- attune/workflows/document_gen/config.py +30 -0
- attune/workflows/document_gen/report_formatter.py +162 -0
- attune/workflows/document_gen/workflow.py +1426 -0
- attune/workflows/document_manager.py +216 -0
- attune/workflows/document_manager_README.md +134 -0
- attune/workflows/documentation_orchestrator.py +1205 -0
- attune/workflows/history.py +510 -0
- attune/workflows/keyboard_shortcuts/__init__.py +39 -0
- attune/workflows/keyboard_shortcuts/generators.py +391 -0
- attune/workflows/keyboard_shortcuts/parsers.py +416 -0
- attune/workflows/keyboard_shortcuts/prompts.py +295 -0
- attune/workflows/keyboard_shortcuts/schema.py +193 -0
- attune/workflows/keyboard_shortcuts/workflow.py +509 -0
- attune/workflows/llm_base.py +363 -0
- attune/workflows/manage_docs.py +87 -0
- attune/workflows/manage_docs_README.md +134 -0
- attune/workflows/manage_documentation.py +821 -0
- attune/workflows/new_sample_workflow1.py +149 -0
- attune/workflows/new_sample_workflow1_README.md +150 -0
- attune/workflows/orchestrated_health_check.py +849 -0
- attune/workflows/orchestrated_release_prep.py +600 -0
- attune/workflows/output.py +413 -0
- attune/workflows/perf_audit.py +863 -0
- attune/workflows/pr_review.py +762 -0
- attune/workflows/progress.py +785 -0
- attune/workflows/progress_server.py +322 -0
- attune/workflows/progressive/README 2.md +454 -0
- attune/workflows/progressive/README.md +454 -0
- attune/workflows/progressive/__init__.py +82 -0
- attune/workflows/progressive/cli.py +219 -0
- attune/workflows/progressive/core.py +488 -0
- attune/workflows/progressive/orchestrator.py +723 -0
- attune/workflows/progressive/reports.py +520 -0
- attune/workflows/progressive/telemetry.py +274 -0
- attune/workflows/progressive/test_gen.py +495 -0
- attune/workflows/progressive/workflow.py +589 -0
- attune/workflows/refactor_plan.py +694 -0
- attune/workflows/release_prep.py +895 -0
- attune/workflows/release_prep_crew.py +969 -0
- attune/workflows/research_synthesis.py +404 -0
- attune/workflows/routing.py +168 -0
- attune/workflows/secure_release.py +593 -0
- attune/workflows/security_adapters.py +297 -0
- attune/workflows/security_audit.py +1329 -0
- attune/workflows/security_audit_phase3.py +355 -0
- attune/workflows/seo_optimization.py +633 -0
- attune/workflows/step_config.py +234 -0
- attune/workflows/telemetry_mixin.py +269 -0
- attune/workflows/test5.py +125 -0
- attune/workflows/test5_README.md +158 -0
- attune/workflows/test_coverage_boost_crew.py +849 -0
- attune/workflows/test_gen/__init__.py +52 -0
- attune/workflows/test_gen/ast_analyzer.py +249 -0
- attune/workflows/test_gen/config.py +88 -0
- attune/workflows/test_gen/data_models.py +38 -0
- attune/workflows/test_gen/report_formatter.py +289 -0
- attune/workflows/test_gen/test_templates.py +381 -0
- attune/workflows/test_gen/workflow.py +655 -0
- attune/workflows/test_gen.py +54 -0
- attune/workflows/test_gen_behavioral.py +477 -0
- attune/workflows/test_gen_parallel.py +341 -0
- attune/workflows/test_lifecycle.py +526 -0
- attune/workflows/test_maintenance.py +627 -0
- attune/workflows/test_maintenance_cli.py +590 -0
- attune/workflows/test_maintenance_crew.py +840 -0
- attune/workflows/test_runner.py +622 -0
- attune/workflows/tier_tracking.py +531 -0
- attune/workflows/xml_enhanced_crew.py +285 -0
- attune_ai-2.0.0.dist-info/METADATA +1026 -0
- attune_ai-2.0.0.dist-info/RECORD +457 -0
- attune_ai-2.0.0.dist-info/WHEEL +5 -0
- attune_ai-2.0.0.dist-info/entry_points.txt +26 -0
- attune_ai-2.0.0.dist-info/licenses/LICENSE +201 -0
- attune_ai-2.0.0.dist-info/licenses/LICENSE_CHANGE_ANNOUNCEMENT.md +101 -0
- attune_ai-2.0.0.dist-info/top_level.txt +5 -0
- attune_healthcare/__init__.py +13 -0
- attune_healthcare/monitors/__init__.py +9 -0
- attune_healthcare/monitors/clinical_protocol_monitor.py +315 -0
- attune_healthcare/monitors/monitoring/__init__.py +44 -0
- attune_healthcare/monitors/monitoring/protocol_checker.py +300 -0
- attune_healthcare/monitors/monitoring/protocol_loader.py +214 -0
- attune_healthcare/monitors/monitoring/sensor_parsers.py +306 -0
- attune_healthcare/monitors/monitoring/trajectory_analyzer.py +389 -0
- attune_llm/README.md +553 -0
- attune_llm/__init__.py +28 -0
- attune_llm/agent_factory/__init__.py +53 -0
- attune_llm/agent_factory/adapters/__init__.py +85 -0
- attune_llm/agent_factory/adapters/autogen_adapter.py +312 -0
- attune_llm/agent_factory/adapters/crewai_adapter.py +483 -0
- attune_llm/agent_factory/adapters/haystack_adapter.py +298 -0
- attune_llm/agent_factory/adapters/langchain_adapter.py +362 -0
- attune_llm/agent_factory/adapters/langgraph_adapter.py +333 -0
- attune_llm/agent_factory/adapters/native.py +228 -0
- attune_llm/agent_factory/adapters/wizard_adapter.py +423 -0
- attune_llm/agent_factory/base.py +305 -0
- attune_llm/agent_factory/crews/__init__.py +67 -0
- attune_llm/agent_factory/crews/code_review.py +1113 -0
- attune_llm/agent_factory/crews/health_check.py +1262 -0
- attune_llm/agent_factory/crews/refactoring.py +1128 -0
- attune_llm/agent_factory/crews/security_audit.py +1018 -0
- attune_llm/agent_factory/decorators.py +287 -0
- attune_llm/agent_factory/factory.py +558 -0
- attune_llm/agent_factory/framework.py +193 -0
- attune_llm/agent_factory/memory_integration.py +328 -0
- attune_llm/agent_factory/resilient.py +320 -0
- attune_llm/agents_md/__init__.py +22 -0
- attune_llm/agents_md/loader.py +218 -0
- attune_llm/agents_md/parser.py +271 -0
- attune_llm/agents_md/registry.py +307 -0
- attune_llm/claude_memory.py +466 -0
- attune_llm/cli/__init__.py +8 -0
- attune_llm/cli/sync_claude.py +487 -0
- attune_llm/code_health.py +1313 -0
- attune_llm/commands/__init__.py +51 -0
- attune_llm/commands/context.py +375 -0
- attune_llm/commands/loader.py +301 -0
- attune_llm/commands/models.py +231 -0
- attune_llm/commands/parser.py +371 -0
- attune_llm/commands/registry.py +429 -0
- attune_llm/config/__init__.py +29 -0
- attune_llm/config/unified.py +291 -0
- attune_llm/context/__init__.py +22 -0
- attune_llm/context/compaction.py +455 -0
- attune_llm/context/manager.py +434 -0
- attune_llm/contextual_patterns.py +361 -0
- attune_llm/core.py +907 -0
- attune_llm/git_pattern_extractor.py +435 -0
- attune_llm/hooks/__init__.py +24 -0
- attune_llm/hooks/config.py +306 -0
- attune_llm/hooks/executor.py +289 -0
- attune_llm/hooks/registry.py +302 -0
- attune_llm/hooks/scripts/__init__.py +39 -0
- attune_llm/hooks/scripts/evaluate_session.py +201 -0
- attune_llm/hooks/scripts/first_time_init.py +285 -0
- attune_llm/hooks/scripts/pre_compact.py +207 -0
- attune_llm/hooks/scripts/session_end.py +183 -0
- attune_llm/hooks/scripts/session_start.py +163 -0
- attune_llm/hooks/scripts/suggest_compact.py +225 -0
- attune_llm/learning/__init__.py +30 -0
- attune_llm/learning/evaluator.py +438 -0
- attune_llm/learning/extractor.py +514 -0
- attune_llm/learning/storage.py +560 -0
- attune_llm/levels.py +227 -0
- attune_llm/pattern_confidence.py +414 -0
- attune_llm/pattern_resolver.py +272 -0
- attune_llm/pattern_summary.py +350 -0
- attune_llm/providers.py +967 -0
- attune_llm/routing/__init__.py +32 -0
- attune_llm/routing/model_router.py +362 -0
- attune_llm/security/IMPLEMENTATION_SUMMARY.md +413 -0
- attune_llm/security/PHASE2_COMPLETE.md +384 -0
- attune_llm/security/PHASE2_SECRETS_DETECTOR_COMPLETE.md +271 -0
- attune_llm/security/QUICK_REFERENCE.md +316 -0
- attune_llm/security/README.md +262 -0
- attune_llm/security/__init__.py +62 -0
- attune_llm/security/audit_logger.py +929 -0
- attune_llm/security/audit_logger_example.py +152 -0
- attune_llm/security/pii_scrubber.py +640 -0
- attune_llm/security/secrets_detector.py +678 -0
- attune_llm/security/secrets_detector_example.py +304 -0
- attune_llm/security/secure_memdocs.py +1192 -0
- attune_llm/security/secure_memdocs_example.py +278 -0
- attune_llm/session_status.py +745 -0
- attune_llm/state.py +246 -0
- attune_llm/utils/__init__.py +5 -0
- attune_llm/utils/tokens.py +349 -0
- attune_software/SOFTWARE_PLUGIN_README.md +57 -0
- attune_software/__init__.py +13 -0
- attune_software/cli/__init__.py +120 -0
- attune_software/cli/inspect.py +362 -0
- attune_software/cli.py +574 -0
- attune_software/plugin.py +188 -0
- workflow_scaffolding/__init__.py +11 -0
- workflow_scaffolding/__main__.py +12 -0
- workflow_scaffolding/cli.py +206 -0
- workflow_scaffolding/generator.py +265 -0
|
@@ -0,0 +1,237 @@
|
|
|
1
|
+
"""Empathy Framework Memory Module
|
|
2
|
+
|
|
3
|
+
Unified two-tier memory system for AI agent collaboration:
|
|
4
|
+
|
|
5
|
+
SHORT-TERM MEMORY (Redis):
|
|
6
|
+
- Agent coordination and working memory
|
|
7
|
+
- TTL-based automatic expiration (5 min - 7 days)
|
|
8
|
+
- Role-based access control (Observer → Steward)
|
|
9
|
+
- Pattern staging before validation
|
|
10
|
+
|
|
11
|
+
LONG-TERM MEMORY (Persistent):
|
|
12
|
+
- Cross-session pattern storage
|
|
13
|
+
- Classification-based access (PUBLIC/INTERNAL/SENSITIVE)
|
|
14
|
+
- PII scrubbing and secrets detection
|
|
15
|
+
- AES-256-GCM encryption for SENSITIVE patterns
|
|
16
|
+
- Compliance: GDPR, HIPAA, SOC2
|
|
17
|
+
|
|
18
|
+
RECOMMENDED USAGE (Unified API):
|
|
19
|
+
from attune.memory import UnifiedMemory
|
|
20
|
+
|
|
21
|
+
# Initialize with environment auto-detection
|
|
22
|
+
memory = UnifiedMemory(user_id="agent@company.com")
|
|
23
|
+
|
|
24
|
+
# Short-term operations
|
|
25
|
+
memory.stash("working_data", {"key": "value"})
|
|
26
|
+
data = memory.retrieve("working_data")
|
|
27
|
+
|
|
28
|
+
# Long-term operations
|
|
29
|
+
result = memory.persist_pattern(
|
|
30
|
+
content="Algorithm for X",
|
|
31
|
+
pattern_type="algorithm",
|
|
32
|
+
)
|
|
33
|
+
pattern = memory.recall_pattern(result["pattern_id"])
|
|
34
|
+
|
|
35
|
+
# Pattern promotion (short-term → long-term)
|
|
36
|
+
staged_id = memory.stage_pattern({"content": "..."})
|
|
37
|
+
memory.promote_pattern(staged_id)
|
|
38
|
+
|
|
39
|
+
ADVANCED USAGE (Direct Access):
|
|
40
|
+
from attune.memory import (
|
|
41
|
+
# Short-term (Redis)
|
|
42
|
+
RedisShortTermMemory,
|
|
43
|
+
AccessTier,
|
|
44
|
+
get_redis_memory,
|
|
45
|
+
|
|
46
|
+
# Long-term (Persistent)
|
|
47
|
+
SecureMemDocsIntegration,
|
|
48
|
+
Classification,
|
|
49
|
+
|
|
50
|
+
# Security
|
|
51
|
+
PIIScrubber,
|
|
52
|
+
SecretsDetector,
|
|
53
|
+
AuditLogger,
|
|
54
|
+
)
|
|
55
|
+
|
|
56
|
+
Copyright 2025 Smart AI Memory, LLC
|
|
57
|
+
Licensed under Fair Source 0.9
|
|
58
|
+
"""
|
|
59
|
+
|
|
60
|
+
# Short-term memory (Redis)
|
|
61
|
+
# Claude Memory integration
|
|
62
|
+
from .claude_memory import ClaudeMemoryConfig, ClaudeMemoryLoader
|
|
63
|
+
|
|
64
|
+
# Memory configuration
|
|
65
|
+
from .config import check_redis_connection, get_railway_redis, get_redis_config, get_redis_memory
|
|
66
|
+
|
|
67
|
+
# Control Panel
|
|
68
|
+
from .control_panel import ControlPanelConfig, MemoryControlPanel, MemoryStats
|
|
69
|
+
|
|
70
|
+
# Cross-session communication
|
|
71
|
+
from .cross_session import (
|
|
72
|
+
BackgroundService,
|
|
73
|
+
ConflictResult,
|
|
74
|
+
ConflictStrategy,
|
|
75
|
+
CrossSessionCoordinator,
|
|
76
|
+
SessionInfo,
|
|
77
|
+
SessionType,
|
|
78
|
+
check_redis_cross_session_support,
|
|
79
|
+
generate_agent_id,
|
|
80
|
+
)
|
|
81
|
+
|
|
82
|
+
# Memory Graph (Cross-Workflow Intelligence)
|
|
83
|
+
from .edges import REVERSE_EDGE_TYPES, WORKFLOW_EDGE_PATTERNS, Edge, EdgeType
|
|
84
|
+
|
|
85
|
+
# File-based session memory (always available, no Redis required)
|
|
86
|
+
from .file_session import FileSessionConfig, FileSessionMemory, get_file_session_memory
|
|
87
|
+
from .graph import MemoryGraph
|
|
88
|
+
|
|
89
|
+
# Long-term memory (Persistent patterns)
|
|
90
|
+
from .long_term import (
|
|
91
|
+
Classification,
|
|
92
|
+
ClassificationRules,
|
|
93
|
+
EncryptionManager,
|
|
94
|
+
MemDocsStorage,
|
|
95
|
+
PatternMetadata,
|
|
96
|
+
SecureMemDocsIntegration,
|
|
97
|
+
SecurePattern,
|
|
98
|
+
SecurityError,
|
|
99
|
+
)
|
|
100
|
+
from .long_term import PermissionError as MemoryPermissionError
|
|
101
|
+
from .nodes import BugNode, Node, NodeType, PatternNode, PerformanceNode, VulnerabilityNode
|
|
102
|
+
|
|
103
|
+
# Redis Bootstrap
|
|
104
|
+
from .redis_bootstrap import (
|
|
105
|
+
RedisStartMethod,
|
|
106
|
+
RedisStatus,
|
|
107
|
+
ensure_redis,
|
|
108
|
+
get_redis_or_mock,
|
|
109
|
+
stop_redis,
|
|
110
|
+
)
|
|
111
|
+
|
|
112
|
+
# Security components
|
|
113
|
+
from .security import ( # Audit Logging; PII Scrubbing; Secrets Detection
|
|
114
|
+
AuditEvent,
|
|
115
|
+
AuditLogger,
|
|
116
|
+
PIIDetection,
|
|
117
|
+
PIIPattern,
|
|
118
|
+
PIIScrubber,
|
|
119
|
+
SecretDetection,
|
|
120
|
+
SecretsDetector,
|
|
121
|
+
SecretType,
|
|
122
|
+
SecurityViolation,
|
|
123
|
+
Severity,
|
|
124
|
+
detect_secrets,
|
|
125
|
+
)
|
|
126
|
+
from .short_term import RedisShortTermMemory
|
|
127
|
+
|
|
128
|
+
# Conversation Summary Index
|
|
129
|
+
from .summary_index import AgentContext, ConversationSummaryIndex
|
|
130
|
+
|
|
131
|
+
# Types (extracted to types.py for cleaner separation)
|
|
132
|
+
from .types import (
|
|
133
|
+
AccessTier,
|
|
134
|
+
AgentCredentials,
|
|
135
|
+
ConflictContext,
|
|
136
|
+
PaginatedResult,
|
|
137
|
+
RedisConfig,
|
|
138
|
+
RedisMetrics,
|
|
139
|
+
StagedPattern,
|
|
140
|
+
TimeWindowQuery,
|
|
141
|
+
TTLStrategy,
|
|
142
|
+
)
|
|
143
|
+
from .types import SecurityError as ShortTermSecurityError
|
|
144
|
+
|
|
145
|
+
# Unified memory interface
|
|
146
|
+
from .unified import Environment, MemoryConfig, UnifiedMemory
|
|
147
|
+
|
|
148
|
+
__all__ = [
|
|
149
|
+
"REVERSE_EDGE_TYPES",
|
|
150
|
+
"WORKFLOW_EDGE_PATTERNS",
|
|
151
|
+
"AccessTier",
|
|
152
|
+
"AgentContext",
|
|
153
|
+
"AgentCredentials",
|
|
154
|
+
"AuditEvent",
|
|
155
|
+
# Security - Audit
|
|
156
|
+
"AuditLogger",
|
|
157
|
+
# Cross-session communication
|
|
158
|
+
"BackgroundService",
|
|
159
|
+
"BugNode",
|
|
160
|
+
"Classification",
|
|
161
|
+
"ClassificationRules",
|
|
162
|
+
# Claude Memory
|
|
163
|
+
"ClaudeMemoryConfig",
|
|
164
|
+
"ClaudeMemoryLoader",
|
|
165
|
+
# File Session Memory (always available)
|
|
166
|
+
"FileSessionConfig",
|
|
167
|
+
"FileSessionMemory",
|
|
168
|
+
"get_file_session_memory",
|
|
169
|
+
"ConflictContext",
|
|
170
|
+
"ConflictResult",
|
|
171
|
+
"ConflictStrategy",
|
|
172
|
+
"ControlPanelConfig",
|
|
173
|
+
# Conversation Summary Index
|
|
174
|
+
"ConversationSummaryIndex",
|
|
175
|
+
"CrossSessionCoordinator",
|
|
176
|
+
"Edge",
|
|
177
|
+
"EdgeType",
|
|
178
|
+
"EncryptionManager",
|
|
179
|
+
"Environment",
|
|
180
|
+
"MemDocsStorage",
|
|
181
|
+
"MemoryConfig",
|
|
182
|
+
# Control Panel
|
|
183
|
+
"MemoryControlPanel",
|
|
184
|
+
# Memory Graph (Cross-Workflow Intelligence)
|
|
185
|
+
"MemoryGraph",
|
|
186
|
+
"MemoryPermissionError",
|
|
187
|
+
"MemoryStats",
|
|
188
|
+
"Node",
|
|
189
|
+
"NodeType",
|
|
190
|
+
# Pagination and Query Types
|
|
191
|
+
"PaginatedResult",
|
|
192
|
+
"PIIDetection",
|
|
193
|
+
"PIIPattern",
|
|
194
|
+
# Security - PII
|
|
195
|
+
"PIIScrubber",
|
|
196
|
+
"PatternMetadata",
|
|
197
|
+
"PatternNode",
|
|
198
|
+
"PerformanceNode",
|
|
199
|
+
# Redis Configuration and Metrics
|
|
200
|
+
"RedisConfig",
|
|
201
|
+
"RedisMetrics",
|
|
202
|
+
# Short-term Memory
|
|
203
|
+
"RedisShortTermMemory",
|
|
204
|
+
"RedisStartMethod",
|
|
205
|
+
"RedisStatus",
|
|
206
|
+
"SecretDetection",
|
|
207
|
+
"SecretType",
|
|
208
|
+
# Security - Secrets
|
|
209
|
+
"SecretsDetector",
|
|
210
|
+
# Long-term Memory
|
|
211
|
+
"SecureMemDocsIntegration",
|
|
212
|
+
"SecurePattern",
|
|
213
|
+
"SecurityError",
|
|
214
|
+
"SecurityViolation",
|
|
215
|
+
"SessionInfo",
|
|
216
|
+
"SessionType",
|
|
217
|
+
"Severity",
|
|
218
|
+
"ShortTermSecurityError",
|
|
219
|
+
"StagedPattern",
|
|
220
|
+
"TTLStrategy",
|
|
221
|
+
"TimeWindowQuery",
|
|
222
|
+
# Unified Memory Interface (recommended)
|
|
223
|
+
"UnifiedMemory",
|
|
224
|
+
"VulnerabilityNode",
|
|
225
|
+
"check_redis_connection",
|
|
226
|
+
"check_redis_cross_session_support",
|
|
227
|
+
"detect_secrets",
|
|
228
|
+
# Redis Bootstrap
|
|
229
|
+
"ensure_redis",
|
|
230
|
+
"generate_agent_id",
|
|
231
|
+
"get_railway_redis",
|
|
232
|
+
"get_redis_config",
|
|
233
|
+
# Configuration
|
|
234
|
+
"get_redis_memory",
|
|
235
|
+
"get_redis_or_mock",
|
|
236
|
+
"stop_redis",
|
|
237
|
+
]
|
|
@@ -0,0 +1,469 @@
|
|
|
1
|
+
"""Claude Memory Integration Module
|
|
2
|
+
|
|
3
|
+
Reads and integrates Claude Code's CLAUDE.md memory files with the Empathy Framework.
|
|
4
|
+
Supports hierarchical memory loading (Enterprise → Project → User) and @import directives.
|
|
5
|
+
|
|
6
|
+
Example CLAUDE.md structure:
|
|
7
|
+
Enterprise: /etc/claude/CLAUDE.md or env var CLAUDE_ENTERPRISE_MEMORY
|
|
8
|
+
User: ~/.claude/CLAUDE.md
|
|
9
|
+
Project: ./.claude/CLAUDE.md or ./CLAUDE.md
|
|
10
|
+
|
|
11
|
+
Author: Empathy Framework Team
|
|
12
|
+
Version: 1.8.0-alpha
|
|
13
|
+
License: Fair Source 0.9
|
|
14
|
+
"""
|
|
15
|
+
|
|
16
|
+
import os
|
|
17
|
+
import re
|
|
18
|
+
from dataclasses import dataclass, field
|
|
19
|
+
from pathlib import Path
|
|
20
|
+
|
|
21
|
+
import structlog
|
|
22
|
+
|
|
23
|
+
from attune.config import _validate_file_path
|
|
24
|
+
|
|
25
|
+
logger = structlog.get_logger(__name__)
|
|
26
|
+
|
|
27
|
+
|
|
28
|
+
@dataclass
|
|
29
|
+
class ClaudeMemoryConfig:
|
|
30
|
+
"""Configuration for Claude memory integration"""
|
|
31
|
+
|
|
32
|
+
# Enable/disable Claude memory integration
|
|
33
|
+
enabled: bool = False
|
|
34
|
+
|
|
35
|
+
# Memory scope (which levels to load)
|
|
36
|
+
load_enterprise: bool = True
|
|
37
|
+
load_user: bool = True
|
|
38
|
+
load_project: bool = True
|
|
39
|
+
|
|
40
|
+
# Enterprise memory location (optional, from env var or config)
|
|
41
|
+
enterprise_memory_path: str | None = None
|
|
42
|
+
|
|
43
|
+
# Project root (defaults to current working directory)
|
|
44
|
+
project_root: str | None = None
|
|
45
|
+
|
|
46
|
+
# Maximum import depth (prevent infinite loops)
|
|
47
|
+
max_import_depth: int = 5
|
|
48
|
+
|
|
49
|
+
# Maximum file size to read (prevent memory issues)
|
|
50
|
+
max_file_size_bytes: int = 1_000_000 # 1 MB
|
|
51
|
+
|
|
52
|
+
# Validate memory files before using
|
|
53
|
+
validate_files: bool = True
|
|
54
|
+
|
|
55
|
+
|
|
56
|
+
@dataclass
|
|
57
|
+
class MemoryFile:
|
|
58
|
+
"""Represents a loaded CLAUDE.md memory file"""
|
|
59
|
+
|
|
60
|
+
level: str # "enterprise", "user", "project"
|
|
61
|
+
path: str
|
|
62
|
+
content: str
|
|
63
|
+
imports: list[str] = field(default_factory=list)
|
|
64
|
+
load_order: int = 0 # Lower numbers load first
|
|
65
|
+
|
|
66
|
+
|
|
67
|
+
class ClaudeMemoryLoader:
|
|
68
|
+
"""Loads and manages Claude Code memory files (CLAUDE.md).
|
|
69
|
+
|
|
70
|
+
Follows Claude Code's hierarchical memory system:
|
|
71
|
+
1. Enterprise memory (organization-wide)
|
|
72
|
+
2. User memory (personal preferences)
|
|
73
|
+
3. Project memory (team/project specific)
|
|
74
|
+
|
|
75
|
+
Supports @import directives for modular memory organization.
|
|
76
|
+
"""
|
|
77
|
+
|
|
78
|
+
def __init__(self, config: ClaudeMemoryConfig | None = None):
|
|
79
|
+
self.config = config or ClaudeMemoryConfig()
|
|
80
|
+
self._memory_cache: dict[str, MemoryFile] = {}
|
|
81
|
+
self._import_stack: list[str] = [] # Track imports to detect cycles
|
|
82
|
+
|
|
83
|
+
def load_all_memory(self, project_root: str | None = None) -> str:
|
|
84
|
+
"""Load all Claude memory files and return combined content.
|
|
85
|
+
|
|
86
|
+
Args:
|
|
87
|
+
project_root: Project root directory (defaults to cwd)
|
|
88
|
+
|
|
89
|
+
Returns:
|
|
90
|
+
Combined memory content from all levels
|
|
91
|
+
|
|
92
|
+
Example:
|
|
93
|
+
loader = ClaudeMemoryLoader(ClaudeMemoryConfig(enabled=True))
|
|
94
|
+
memory = loader.load_all_memory("/path/to/project")
|
|
95
|
+
# Use memory in LLM system prompt
|
|
96
|
+
|
|
97
|
+
"""
|
|
98
|
+
if not self.config.enabled:
|
|
99
|
+
logger.debug("claude_memory_disabled")
|
|
100
|
+
return ""
|
|
101
|
+
|
|
102
|
+
project_root = project_root or self.config.project_root or os.getcwd()
|
|
103
|
+
memory_files: list[MemoryFile] = []
|
|
104
|
+
|
|
105
|
+
try:
|
|
106
|
+
# Load in hierarchical order (lower precedence first)
|
|
107
|
+
if self.config.load_enterprise:
|
|
108
|
+
if enterprise_memory := self._load_enterprise_memory():
|
|
109
|
+
enterprise_memory.load_order = 1
|
|
110
|
+
memory_files.append(enterprise_memory)
|
|
111
|
+
|
|
112
|
+
if self.config.load_user:
|
|
113
|
+
if user_memory := self._load_user_memory():
|
|
114
|
+
user_memory.load_order = 2
|
|
115
|
+
memory_files.append(user_memory)
|
|
116
|
+
|
|
117
|
+
if self.config.load_project:
|
|
118
|
+
if project_memory := self._load_project_memory(project_root):
|
|
119
|
+
project_memory.load_order = 3
|
|
120
|
+
memory_files.append(project_memory)
|
|
121
|
+
|
|
122
|
+
# Sort by load order
|
|
123
|
+
memory_files.sort(key=lambda m: m.load_order)
|
|
124
|
+
|
|
125
|
+
# Combine content
|
|
126
|
+
combined = self._combine_memory_files(memory_files)
|
|
127
|
+
|
|
128
|
+
logger.info(
|
|
129
|
+
"claude_memory_loaded",
|
|
130
|
+
files_count=len(memory_files),
|
|
131
|
+
total_chars=len(combined),
|
|
132
|
+
)
|
|
133
|
+
|
|
134
|
+
return combined
|
|
135
|
+
|
|
136
|
+
except Exception as e:
|
|
137
|
+
logger.error("claude_memory_load_failed", error=str(e))
|
|
138
|
+
if self.config.validate_files:
|
|
139
|
+
raise
|
|
140
|
+
return ""
|
|
141
|
+
|
|
142
|
+
def _load_enterprise_memory(self) -> MemoryFile | None:
|
|
143
|
+
"""Load enterprise-level memory"""
|
|
144
|
+
# Check environment variable first
|
|
145
|
+
if env_path := os.getenv("CLAUDE_ENTERPRISE_MEMORY"):
|
|
146
|
+
return self._load_memory_file(env_path, "enterprise")
|
|
147
|
+
|
|
148
|
+
# Check config
|
|
149
|
+
if self.config.enterprise_memory_path:
|
|
150
|
+
return self._load_memory_file(self.config.enterprise_memory_path, "enterprise")
|
|
151
|
+
|
|
152
|
+
# Check standard location (Unix-like systems)
|
|
153
|
+
standard_path = Path("/etc/claude/CLAUDE.md")
|
|
154
|
+
if standard_path.exists():
|
|
155
|
+
return self._load_memory_file(str(standard_path), "enterprise")
|
|
156
|
+
|
|
157
|
+
logger.debug("enterprise_memory_not_found")
|
|
158
|
+
return None
|
|
159
|
+
|
|
160
|
+
def _load_user_memory(self) -> MemoryFile | None:
|
|
161
|
+
"""Load user-level memory"""
|
|
162
|
+
user_memory_path = Path.home() / ".claude" / "CLAUDE.md"
|
|
163
|
+
|
|
164
|
+
if user_memory_path.exists():
|
|
165
|
+
return self._load_memory_file(str(user_memory_path), "user")
|
|
166
|
+
|
|
167
|
+
logger.debug("user_memory_not_found")
|
|
168
|
+
return None
|
|
169
|
+
|
|
170
|
+
def _load_project_memory(self, project_root: str) -> MemoryFile | None:
|
|
171
|
+
"""Load project-level memory"""
|
|
172
|
+
project_path = Path(project_root)
|
|
173
|
+
|
|
174
|
+
# Try .claude/CLAUDE.md first (recommended)
|
|
175
|
+
claude_dir_memory = project_path / ".claude" / "CLAUDE.md"
|
|
176
|
+
if claude_dir_memory.exists():
|
|
177
|
+
return self._load_memory_file(str(claude_dir_memory), "project")
|
|
178
|
+
|
|
179
|
+
# Fallback to CLAUDE.md in project root
|
|
180
|
+
root_memory = project_path / "CLAUDE.md"
|
|
181
|
+
if root_memory.exists():
|
|
182
|
+
return self._load_memory_file(str(root_memory), "project")
|
|
183
|
+
|
|
184
|
+
logger.debug("project_memory_not_found", project_root=project_root)
|
|
185
|
+
return None
|
|
186
|
+
|
|
187
|
+
def _load_memory_file(self, file_path: str, level: str, depth: int = 0) -> MemoryFile | None:
|
|
188
|
+
"""Load a single memory file and process imports.
|
|
189
|
+
|
|
190
|
+
Args:
|
|
191
|
+
file_path: Path to CLAUDE.md file
|
|
192
|
+
level: Memory level (enterprise/user/project)
|
|
193
|
+
depth: Current import depth (for recursion control)
|
|
194
|
+
|
|
195
|
+
Returns:
|
|
196
|
+
MemoryFile object or None if failed
|
|
197
|
+
|
|
198
|
+
"""
|
|
199
|
+
# Check depth limit
|
|
200
|
+
if depth > self.config.max_import_depth:
|
|
201
|
+
logger.warning(
|
|
202
|
+
"import_depth_exceeded",
|
|
203
|
+
file_path=file_path,
|
|
204
|
+
depth=depth,
|
|
205
|
+
max_depth=self.config.max_import_depth,
|
|
206
|
+
)
|
|
207
|
+
return None
|
|
208
|
+
|
|
209
|
+
# Check for circular imports
|
|
210
|
+
if file_path in self._import_stack:
|
|
211
|
+
logger.warning("circular_import_detected", file_path=file_path)
|
|
212
|
+
return None
|
|
213
|
+
|
|
214
|
+
# Check cache
|
|
215
|
+
if file_path in self._memory_cache:
|
|
216
|
+
return self._memory_cache[file_path]
|
|
217
|
+
|
|
218
|
+
try:
|
|
219
|
+
path = Path(file_path)
|
|
220
|
+
|
|
221
|
+
# Validate file exists
|
|
222
|
+
if not path.exists():
|
|
223
|
+
logger.debug("memory_file_not_found", file_path=file_path)
|
|
224
|
+
return None
|
|
225
|
+
|
|
226
|
+
# Check file size
|
|
227
|
+
file_size = path.stat().st_size
|
|
228
|
+
if file_size > self.config.max_file_size_bytes:
|
|
229
|
+
logger.warning(
|
|
230
|
+
"memory_file_too_large",
|
|
231
|
+
file_path=file_path,
|
|
232
|
+
size_bytes=file_size,
|
|
233
|
+
max_bytes=self.config.max_file_size_bytes,
|
|
234
|
+
)
|
|
235
|
+
return None
|
|
236
|
+
|
|
237
|
+
# Read file
|
|
238
|
+
content = path.read_text(encoding="utf-8")
|
|
239
|
+
|
|
240
|
+
# Process imports
|
|
241
|
+
self._import_stack.append(file_path)
|
|
242
|
+
processed_content, imports = self._process_imports(content, path.parent, depth)
|
|
243
|
+
self._import_stack.pop()
|
|
244
|
+
|
|
245
|
+
# Create memory file object
|
|
246
|
+
memory_file = MemoryFile(
|
|
247
|
+
level=level,
|
|
248
|
+
path=file_path,
|
|
249
|
+
content=processed_content,
|
|
250
|
+
imports=imports,
|
|
251
|
+
)
|
|
252
|
+
|
|
253
|
+
# Cache
|
|
254
|
+
self._memory_cache[file_path] = memory_file
|
|
255
|
+
|
|
256
|
+
logger.debug(
|
|
257
|
+
"memory_file_loaded",
|
|
258
|
+
level=level,
|
|
259
|
+
path=file_path,
|
|
260
|
+
size_chars=len(content),
|
|
261
|
+
imports_count=len(imports),
|
|
262
|
+
)
|
|
263
|
+
|
|
264
|
+
return memory_file
|
|
265
|
+
|
|
266
|
+
except Exception as e:
|
|
267
|
+
logger.error("memory_file_load_error", file_path=file_path, error=str(e))
|
|
268
|
+
if self.config.validate_files:
|
|
269
|
+
raise
|
|
270
|
+
return None
|
|
271
|
+
|
|
272
|
+
def _process_imports(self, content: str, base_dir: Path, depth: int) -> tuple[str, list[str]]:
|
|
273
|
+
"""Process @import directives in memory content.
|
|
274
|
+
|
|
275
|
+
Supports syntax: @path/to/file.md
|
|
276
|
+
|
|
277
|
+
Args:
|
|
278
|
+
content: Raw memory file content
|
|
279
|
+
base_dir: Base directory for resolving relative paths
|
|
280
|
+
depth: Current import depth
|
|
281
|
+
|
|
282
|
+
Returns:
|
|
283
|
+
(processed_content, list_of_imported_paths)
|
|
284
|
+
|
|
285
|
+
"""
|
|
286
|
+
# Match @import syntax: @path/to/file
|
|
287
|
+
import_pattern = re.compile(r"^@([^\s]+)$", re.MULTILINE)
|
|
288
|
+
|
|
289
|
+
imports: list[str] = []
|
|
290
|
+
processed_lines: list[str] = []
|
|
291
|
+
|
|
292
|
+
for line in content.split("\n"):
|
|
293
|
+
match = import_pattern.match(line.strip())
|
|
294
|
+
|
|
295
|
+
if match:
|
|
296
|
+
import_path = match.group(1)
|
|
297
|
+
|
|
298
|
+
# Resolve path (relative to base_dir)
|
|
299
|
+
resolved_path = (base_dir / import_path).resolve()
|
|
300
|
+
|
|
301
|
+
if resolved_path.exists():
|
|
302
|
+
# Recursively load imported file
|
|
303
|
+
imported_file = self._load_memory_file(
|
|
304
|
+
str(resolved_path),
|
|
305
|
+
level="import",
|
|
306
|
+
depth=depth + 1,
|
|
307
|
+
)
|
|
308
|
+
|
|
309
|
+
if imported_file:
|
|
310
|
+
# Include imported content
|
|
311
|
+
processed_lines.append(f"# Imported from: {import_path}")
|
|
312
|
+
processed_lines.append(imported_file.content)
|
|
313
|
+
imports.append(str(resolved_path))
|
|
314
|
+
else:
|
|
315
|
+
processed_lines.append(f"# Import failed: {import_path}")
|
|
316
|
+
logger.warning("import_failed", import_path=str(resolved_path))
|
|
317
|
+
else:
|
|
318
|
+
processed_lines.append(f"# Import not found: {import_path}")
|
|
319
|
+
logger.warning("import_not_found", import_path=import_path)
|
|
320
|
+
else:
|
|
321
|
+
# Regular line, keep as-is
|
|
322
|
+
processed_lines.append(line)
|
|
323
|
+
|
|
324
|
+
return "\n".join(processed_lines), imports
|
|
325
|
+
|
|
326
|
+
def _combine_memory_files(self, memory_files: list[MemoryFile]) -> str:
|
|
327
|
+
"""Combine multiple memory files into a single memory string.
|
|
328
|
+
|
|
329
|
+
Files are combined in load order with clear section markers.
|
|
330
|
+
"""
|
|
331
|
+
if not memory_files:
|
|
332
|
+
return ""
|
|
333
|
+
|
|
334
|
+
sections: list[str] = []
|
|
335
|
+
|
|
336
|
+
for memory_file in memory_files:
|
|
337
|
+
section_header = f"""
|
|
338
|
+
---
|
|
339
|
+
# Claude Memory: {memory_file.level.upper()} Level
|
|
340
|
+
# Source: {memory_file.path}
|
|
341
|
+
---
|
|
342
|
+
"""
|
|
343
|
+
sections.append(section_header)
|
|
344
|
+
sections.append(memory_file.content)
|
|
345
|
+
|
|
346
|
+
return "\n".join(sections)
|
|
347
|
+
|
|
348
|
+
def clear_cache(self):
|
|
349
|
+
"""Clear the memory file cache"""
|
|
350
|
+
self._memory_cache.clear()
|
|
351
|
+
logger.debug("memory_cache_cleared")
|
|
352
|
+
|
|
353
|
+
def get_loaded_files(self) -> list[str]:
|
|
354
|
+
"""Get list of all loaded memory file paths"""
|
|
355
|
+
return list(self._memory_cache.keys())
|
|
356
|
+
|
|
357
|
+
|
|
358
|
+
def create_default_project_memory(project_root: str, framework: str = "empathy"):
|
|
359
|
+
"""Create a default .claude/CLAUDE.md file for a project.
|
|
360
|
+
|
|
361
|
+
Args:
|
|
362
|
+
project_root: Project root directory
|
|
363
|
+
framework: Framework name (e.g., "empathy", "langchain")
|
|
364
|
+
|
|
365
|
+
Example:
|
|
366
|
+
create_default_project_memory("/path/to/project")
|
|
367
|
+
|
|
368
|
+
"""
|
|
369
|
+
project_path = Path(project_root)
|
|
370
|
+
claude_dir = project_path / ".claude"
|
|
371
|
+
memory_file = claude_dir / "CLAUDE.md"
|
|
372
|
+
|
|
373
|
+
# Create .claude directory if it doesn't exist
|
|
374
|
+
claude_dir.mkdir(exist_ok=True)
|
|
375
|
+
|
|
376
|
+
# Don't overwrite existing file
|
|
377
|
+
if memory_file.exists():
|
|
378
|
+
logger.info("project_memory_exists", path=str(memory_file))
|
|
379
|
+
return
|
|
380
|
+
|
|
381
|
+
# Create default content
|
|
382
|
+
if framework.lower() == "empathy":
|
|
383
|
+
default_content = """# Project Memory: Empathy Framework
|
|
384
|
+
|
|
385
|
+
## Project Context
|
|
386
|
+
This project uses the Empathy Framework for AI-powered software development assistance.
|
|
387
|
+
|
|
388
|
+
## Code Style Preferences
|
|
389
|
+
- Follow PEP 8 for Python code
|
|
390
|
+
- Use type hints for all function signatures
|
|
391
|
+
- Prefer descriptive variable names over comments
|
|
392
|
+
- Maximum line length: 100 characters
|
|
393
|
+
|
|
394
|
+
## Architecture Patterns
|
|
395
|
+
- Use async/await for I/O operations
|
|
396
|
+
- Implement Level 4 Anticipatory predictions where applicable
|
|
397
|
+
- Store patterns in MemDocs for cross-domain learning
|
|
398
|
+
- Follow the 5-level empathy maturity model
|
|
399
|
+
|
|
400
|
+
## Security Requirements
|
|
401
|
+
- Never commit API keys or secrets
|
|
402
|
+
- Always validate user input
|
|
403
|
+
- Use parameterized queries (no SQL injection)
|
|
404
|
+
- Implement proper error handling
|
|
405
|
+
|
|
406
|
+
## Testing Standards
|
|
407
|
+
- Maintain 90%+ test coverage
|
|
408
|
+
- Write tests for all workflows
|
|
409
|
+
- Use pytest fixtures for common setup
|
|
410
|
+
- Mock external API calls
|
|
411
|
+
|
|
412
|
+
## Documentation
|
|
413
|
+
- Document all public APIs
|
|
414
|
+
- Include usage examples
|
|
415
|
+
- Keep README.md up to date
|
|
416
|
+
- Use docstrings for all classes and functions
|
|
417
|
+
|
|
418
|
+
## Privacy & Compliance
|
|
419
|
+
- PII must be scrubbed before external API calls
|
|
420
|
+
- Audit log all external requests
|
|
421
|
+
- Follow GDPR/HIPAA requirements
|
|
422
|
+
- Use local-only mode for sensitive code
|
|
423
|
+
|
|
424
|
+
---
|
|
425
|
+
Generated by Empathy Framework v1.8.0
|
|
426
|
+
"""
|
|
427
|
+
else:
|
|
428
|
+
default_content = """# Project Memory
|
|
429
|
+
|
|
430
|
+
## Project Context
|
|
431
|
+
Add project-specific instructions here.
|
|
432
|
+
|
|
433
|
+
## Code Preferences
|
|
434
|
+
- Add your coding style preferences
|
|
435
|
+
|
|
436
|
+
## Architecture
|
|
437
|
+
- Document key architectural decisions
|
|
438
|
+
|
|
439
|
+
## Security
|
|
440
|
+
- List security requirements
|
|
441
|
+
|
|
442
|
+
## Testing
|
|
443
|
+
- Define testing standards
|
|
444
|
+
|
|
445
|
+
---
|
|
446
|
+
Add @imports to include shared memory files:
|
|
447
|
+
@../shared-standards.md
|
|
448
|
+
"""
|
|
449
|
+
|
|
450
|
+
# Write file
|
|
451
|
+
validated_path = _validate_file_path(str(memory_file))
|
|
452
|
+
validated_path.write_text(default_content, encoding="utf-8")
|
|
453
|
+
logger.info("project_memory_created", path=str(validated_path))
|
|
454
|
+
|
|
455
|
+
|
|
456
|
+
# Example usage:
|
|
457
|
+
if __name__ == "__main__":
|
|
458
|
+
# Example 1: Basic usage
|
|
459
|
+
config = ClaudeMemoryConfig(enabled=True)
|
|
460
|
+
loader = ClaudeMemoryLoader(config)
|
|
461
|
+
memory = loader.load_all_memory()
|
|
462
|
+
print(f"Loaded memory ({len(memory)} chars):")
|
|
463
|
+
print(memory[:500] if memory else "No memory found")
|
|
464
|
+
|
|
465
|
+
# Example 2: Create default project memory
|
|
466
|
+
# create_default_project_memory(".", framework="empathy")
|
|
467
|
+
|
|
468
|
+
# Example 3: Get loaded files
|
|
469
|
+
print(f"\nLoaded files: {loader.get_loaded_files()}")
|