aiecs 1.0.1__py3-none-any.whl → 1.7.6__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 aiecs might be problematic. Click here for more details.
- aiecs/__init__.py +13 -16
- aiecs/__main__.py +7 -7
- aiecs/aiecs_client.py +269 -75
- aiecs/application/executors/operation_executor.py +79 -54
- aiecs/application/knowledge_graph/__init__.py +7 -0
- aiecs/application/knowledge_graph/builder/__init__.py +37 -0
- aiecs/application/knowledge_graph/builder/data_quality.py +302 -0
- aiecs/application/knowledge_graph/builder/data_reshaping.py +293 -0
- aiecs/application/knowledge_graph/builder/document_builder.py +369 -0
- aiecs/application/knowledge_graph/builder/graph_builder.py +490 -0
- aiecs/application/knowledge_graph/builder/import_optimizer.py +396 -0
- aiecs/application/knowledge_graph/builder/schema_inference.py +462 -0
- aiecs/application/knowledge_graph/builder/schema_mapping.py +563 -0
- aiecs/application/knowledge_graph/builder/structured_pipeline.py +1384 -0
- aiecs/application/knowledge_graph/builder/text_chunker.py +317 -0
- aiecs/application/knowledge_graph/extractors/__init__.py +27 -0
- aiecs/application/knowledge_graph/extractors/base.py +98 -0
- aiecs/application/knowledge_graph/extractors/llm_entity_extractor.py +422 -0
- aiecs/application/knowledge_graph/extractors/llm_relation_extractor.py +347 -0
- aiecs/application/knowledge_graph/extractors/ner_entity_extractor.py +241 -0
- aiecs/application/knowledge_graph/fusion/__init__.py +78 -0
- aiecs/application/knowledge_graph/fusion/ab_testing.py +395 -0
- aiecs/application/knowledge_graph/fusion/abbreviation_expander.py +327 -0
- aiecs/application/knowledge_graph/fusion/alias_index.py +597 -0
- aiecs/application/knowledge_graph/fusion/alias_matcher.py +384 -0
- aiecs/application/knowledge_graph/fusion/cache_coordinator.py +343 -0
- aiecs/application/knowledge_graph/fusion/entity_deduplicator.py +433 -0
- aiecs/application/knowledge_graph/fusion/entity_linker.py +511 -0
- aiecs/application/knowledge_graph/fusion/evaluation_dataset.py +240 -0
- aiecs/application/knowledge_graph/fusion/knowledge_fusion.py +632 -0
- aiecs/application/knowledge_graph/fusion/matching_config.py +489 -0
- aiecs/application/knowledge_graph/fusion/name_normalizer.py +352 -0
- aiecs/application/knowledge_graph/fusion/relation_deduplicator.py +183 -0
- aiecs/application/knowledge_graph/fusion/semantic_name_matcher.py +464 -0
- aiecs/application/knowledge_graph/fusion/similarity_pipeline.py +534 -0
- aiecs/application/knowledge_graph/pattern_matching/__init__.py +21 -0
- aiecs/application/knowledge_graph/pattern_matching/pattern_matcher.py +342 -0
- aiecs/application/knowledge_graph/pattern_matching/query_executor.py +366 -0
- aiecs/application/knowledge_graph/profiling/__init__.py +12 -0
- aiecs/application/knowledge_graph/profiling/query_plan_visualizer.py +195 -0
- aiecs/application/knowledge_graph/profiling/query_profiler.py +223 -0
- aiecs/application/knowledge_graph/reasoning/__init__.py +27 -0
- aiecs/application/knowledge_graph/reasoning/evidence_synthesis.py +341 -0
- aiecs/application/knowledge_graph/reasoning/inference_engine.py +500 -0
- aiecs/application/knowledge_graph/reasoning/logic_form_parser.py +163 -0
- aiecs/application/knowledge_graph/reasoning/logic_parser/__init__.py +79 -0
- aiecs/application/knowledge_graph/reasoning/logic_parser/ast_builder.py +513 -0
- aiecs/application/knowledge_graph/reasoning/logic_parser/ast_nodes.py +913 -0
- aiecs/application/knowledge_graph/reasoning/logic_parser/ast_validator.py +866 -0
- aiecs/application/knowledge_graph/reasoning/logic_parser/error_handler.py +475 -0
- aiecs/application/knowledge_graph/reasoning/logic_parser/parser.py +396 -0
- aiecs/application/knowledge_graph/reasoning/logic_parser/query_context.py +208 -0
- aiecs/application/knowledge_graph/reasoning/logic_query_integration.py +170 -0
- aiecs/application/knowledge_graph/reasoning/query_planner.py +855 -0
- aiecs/application/knowledge_graph/reasoning/reasoning_engine.py +518 -0
- aiecs/application/knowledge_graph/retrieval/__init__.py +27 -0
- aiecs/application/knowledge_graph/retrieval/query_intent_classifier.py +211 -0
- aiecs/application/knowledge_graph/retrieval/retrieval_strategies.py +592 -0
- aiecs/application/knowledge_graph/retrieval/strategy_types.py +23 -0
- aiecs/application/knowledge_graph/search/__init__.py +59 -0
- aiecs/application/knowledge_graph/search/hybrid_search.py +457 -0
- aiecs/application/knowledge_graph/search/reranker.py +293 -0
- aiecs/application/knowledge_graph/search/reranker_strategies.py +535 -0
- aiecs/application/knowledge_graph/search/text_similarity.py +392 -0
- aiecs/application/knowledge_graph/traversal/__init__.py +15 -0
- aiecs/application/knowledge_graph/traversal/enhanced_traversal.py +305 -0
- aiecs/application/knowledge_graph/traversal/path_scorer.py +271 -0
- aiecs/application/knowledge_graph/validators/__init__.py +13 -0
- aiecs/application/knowledge_graph/validators/relation_validator.py +239 -0
- aiecs/application/knowledge_graph/visualization/__init__.py +11 -0
- aiecs/application/knowledge_graph/visualization/graph_visualizer.py +313 -0
- aiecs/common/__init__.py +9 -0
- aiecs/common/knowledge_graph/__init__.py +17 -0
- aiecs/common/knowledge_graph/runnable.py +471 -0
- aiecs/config/__init__.py +20 -5
- aiecs/config/config.py +762 -31
- aiecs/config/graph_config.py +131 -0
- aiecs/config/tool_config.py +399 -0
- aiecs/core/__init__.py +29 -13
- aiecs/core/interface/__init__.py +2 -2
- aiecs/core/interface/execution_interface.py +22 -22
- aiecs/core/interface/storage_interface.py +37 -88
- aiecs/core/registry/__init__.py +31 -0
- aiecs/core/registry/service_registry.py +92 -0
- aiecs/domain/__init__.py +270 -1
- aiecs/domain/agent/__init__.py +191 -0
- aiecs/domain/agent/base_agent.py +3870 -0
- aiecs/domain/agent/exceptions.py +99 -0
- aiecs/domain/agent/graph_aware_mixin.py +569 -0
- aiecs/domain/agent/hybrid_agent.py +1435 -0
- aiecs/domain/agent/integration/__init__.py +29 -0
- aiecs/domain/agent/integration/context_compressor.py +216 -0
- aiecs/domain/agent/integration/context_engine_adapter.py +587 -0
- aiecs/domain/agent/integration/protocols.py +281 -0
- aiecs/domain/agent/integration/retry_policy.py +218 -0
- aiecs/domain/agent/integration/role_config.py +213 -0
- aiecs/domain/agent/knowledge_aware_agent.py +1892 -0
- aiecs/domain/agent/lifecycle.py +291 -0
- aiecs/domain/agent/llm_agent.py +692 -0
- aiecs/domain/agent/memory/__init__.py +12 -0
- aiecs/domain/agent/memory/conversation.py +1124 -0
- aiecs/domain/agent/migration/__init__.py +14 -0
- aiecs/domain/agent/migration/conversion.py +163 -0
- aiecs/domain/agent/migration/legacy_wrapper.py +86 -0
- aiecs/domain/agent/models.py +884 -0
- aiecs/domain/agent/observability.py +479 -0
- aiecs/domain/agent/persistence.py +449 -0
- aiecs/domain/agent/prompts/__init__.py +29 -0
- aiecs/domain/agent/prompts/builder.py +159 -0
- aiecs/domain/agent/prompts/formatters.py +187 -0
- aiecs/domain/agent/prompts/template.py +255 -0
- aiecs/domain/agent/registry.py +253 -0
- aiecs/domain/agent/tool_agent.py +444 -0
- aiecs/domain/agent/tools/__init__.py +15 -0
- aiecs/domain/agent/tools/schema_generator.py +364 -0
- aiecs/domain/community/__init__.py +155 -0
- aiecs/domain/community/agent_adapter.py +469 -0
- aiecs/domain/community/analytics.py +432 -0
- aiecs/domain/community/collaborative_workflow.py +648 -0
- aiecs/domain/community/communication_hub.py +634 -0
- aiecs/domain/community/community_builder.py +320 -0
- aiecs/domain/community/community_integration.py +796 -0
- aiecs/domain/community/community_manager.py +803 -0
- aiecs/domain/community/decision_engine.py +849 -0
- aiecs/domain/community/exceptions.py +231 -0
- aiecs/domain/community/models/__init__.py +33 -0
- aiecs/domain/community/models/community_models.py +234 -0
- aiecs/domain/community/resource_manager.py +461 -0
- aiecs/domain/community/shared_context_manager.py +589 -0
- aiecs/domain/context/__init__.py +40 -10
- aiecs/domain/context/context_engine.py +1910 -0
- aiecs/domain/context/conversation_models.py +87 -53
- aiecs/domain/context/graph_memory.py +582 -0
- aiecs/domain/execution/model.py +12 -4
- aiecs/domain/knowledge_graph/__init__.py +19 -0
- aiecs/domain/knowledge_graph/models/__init__.py +52 -0
- aiecs/domain/knowledge_graph/models/entity.py +148 -0
- aiecs/domain/knowledge_graph/models/evidence.py +178 -0
- aiecs/domain/knowledge_graph/models/inference_rule.py +184 -0
- aiecs/domain/knowledge_graph/models/path.py +171 -0
- aiecs/domain/knowledge_graph/models/path_pattern.py +171 -0
- aiecs/domain/knowledge_graph/models/query.py +261 -0
- aiecs/domain/knowledge_graph/models/query_plan.py +181 -0
- aiecs/domain/knowledge_graph/models/relation.py +202 -0
- aiecs/domain/knowledge_graph/schema/__init__.py +23 -0
- aiecs/domain/knowledge_graph/schema/entity_type.py +131 -0
- aiecs/domain/knowledge_graph/schema/graph_schema.py +253 -0
- aiecs/domain/knowledge_graph/schema/property_schema.py +143 -0
- aiecs/domain/knowledge_graph/schema/relation_type.py +163 -0
- aiecs/domain/knowledge_graph/schema/schema_manager.py +691 -0
- aiecs/domain/knowledge_graph/schema/type_enums.py +209 -0
- aiecs/domain/task/dsl_processor.py +172 -56
- aiecs/domain/task/model.py +20 -8
- aiecs/domain/task/task_context.py +27 -24
- aiecs/infrastructure/__init__.py +0 -2
- aiecs/infrastructure/graph_storage/__init__.py +11 -0
- aiecs/infrastructure/graph_storage/base.py +837 -0
- aiecs/infrastructure/graph_storage/batch_operations.py +458 -0
- aiecs/infrastructure/graph_storage/cache.py +424 -0
- aiecs/infrastructure/graph_storage/distributed.py +223 -0
- aiecs/infrastructure/graph_storage/error_handling.py +380 -0
- aiecs/infrastructure/graph_storage/graceful_degradation.py +294 -0
- aiecs/infrastructure/graph_storage/health_checks.py +378 -0
- aiecs/infrastructure/graph_storage/in_memory.py +1197 -0
- aiecs/infrastructure/graph_storage/index_optimization.py +446 -0
- aiecs/infrastructure/graph_storage/lazy_loading.py +431 -0
- aiecs/infrastructure/graph_storage/metrics.py +344 -0
- aiecs/infrastructure/graph_storage/migration.py +400 -0
- aiecs/infrastructure/graph_storage/pagination.py +483 -0
- aiecs/infrastructure/graph_storage/performance_monitoring.py +456 -0
- aiecs/infrastructure/graph_storage/postgres.py +1563 -0
- aiecs/infrastructure/graph_storage/property_storage.py +353 -0
- aiecs/infrastructure/graph_storage/protocols.py +76 -0
- aiecs/infrastructure/graph_storage/query_optimizer.py +642 -0
- aiecs/infrastructure/graph_storage/schema_cache.py +290 -0
- aiecs/infrastructure/graph_storage/sqlite.py +1373 -0
- aiecs/infrastructure/graph_storage/streaming.py +487 -0
- aiecs/infrastructure/graph_storage/tenant.py +412 -0
- aiecs/infrastructure/messaging/celery_task_manager.py +92 -54
- aiecs/infrastructure/messaging/websocket_manager.py +51 -35
- aiecs/infrastructure/monitoring/__init__.py +22 -0
- aiecs/infrastructure/monitoring/executor_metrics.py +45 -11
- aiecs/infrastructure/monitoring/global_metrics_manager.py +212 -0
- aiecs/infrastructure/monitoring/structured_logger.py +3 -7
- aiecs/infrastructure/monitoring/tracing_manager.py +63 -35
- aiecs/infrastructure/persistence/__init__.py +14 -1
- aiecs/infrastructure/persistence/context_engine_client.py +184 -0
- aiecs/infrastructure/persistence/database_manager.py +67 -43
- aiecs/infrastructure/persistence/file_storage.py +180 -103
- aiecs/infrastructure/persistence/redis_client.py +74 -21
- aiecs/llm/__init__.py +73 -25
- aiecs/llm/callbacks/__init__.py +11 -0
- aiecs/llm/{custom_callbacks.py → callbacks/custom_callbacks.py} +26 -19
- aiecs/llm/client_factory.py +224 -36
- aiecs/llm/client_resolver.py +155 -0
- aiecs/llm/clients/__init__.py +38 -0
- aiecs/llm/clients/base_client.py +324 -0
- aiecs/llm/clients/google_function_calling_mixin.py +457 -0
- aiecs/llm/clients/googleai_client.py +241 -0
- aiecs/llm/clients/openai_client.py +158 -0
- aiecs/llm/clients/openai_compatible_mixin.py +367 -0
- aiecs/llm/clients/vertex_client.py +897 -0
- aiecs/llm/clients/xai_client.py +201 -0
- aiecs/llm/config/__init__.py +51 -0
- aiecs/llm/config/config_loader.py +272 -0
- aiecs/llm/config/config_validator.py +206 -0
- aiecs/llm/config/model_config.py +143 -0
- aiecs/llm/protocols.py +149 -0
- aiecs/llm/utils/__init__.py +10 -0
- aiecs/llm/utils/validate_config.py +89 -0
- aiecs/main.py +140 -121
- aiecs/scripts/aid/VERSION_MANAGEMENT.md +138 -0
- aiecs/scripts/aid/__init__.py +19 -0
- aiecs/scripts/aid/module_checker.py +499 -0
- aiecs/scripts/aid/version_manager.py +235 -0
- aiecs/scripts/{DEPENDENCY_SYSTEM_SUMMARY.md → dependance_check/DEPENDENCY_SYSTEM_SUMMARY.md} +1 -0
- aiecs/scripts/{README_DEPENDENCY_CHECKER.md → dependance_check/README_DEPENDENCY_CHECKER.md} +1 -0
- aiecs/scripts/dependance_check/__init__.py +15 -0
- aiecs/scripts/dependance_check/dependency_checker.py +1835 -0
- aiecs/scripts/{dependency_fixer.py → dependance_check/dependency_fixer.py} +192 -90
- aiecs/scripts/{download_nlp_data.py → dependance_check/download_nlp_data.py} +203 -71
- aiecs/scripts/dependance_patch/__init__.py +7 -0
- aiecs/scripts/dependance_patch/fix_weasel/__init__.py +11 -0
- aiecs/scripts/{fix_weasel_validator.py → dependance_patch/fix_weasel/fix_weasel_validator.py} +21 -14
- aiecs/scripts/{patch_weasel_library.sh → dependance_patch/fix_weasel/patch_weasel_library.sh} +1 -1
- aiecs/scripts/knowledge_graph/__init__.py +3 -0
- aiecs/scripts/knowledge_graph/run_threshold_experiments.py +212 -0
- aiecs/scripts/migrations/multi_tenancy/README.md +142 -0
- aiecs/scripts/tools_develop/README.md +671 -0
- aiecs/scripts/tools_develop/README_CONFIG_CHECKER.md +273 -0
- aiecs/scripts/tools_develop/TOOLS_CONFIG_GUIDE.md +1287 -0
- aiecs/scripts/tools_develop/TOOL_AUTO_DISCOVERY.md +234 -0
- aiecs/scripts/tools_develop/__init__.py +21 -0
- aiecs/scripts/tools_develop/check_all_tools_config.py +548 -0
- aiecs/scripts/tools_develop/check_type_annotations.py +257 -0
- aiecs/scripts/tools_develop/pre-commit-schema-coverage.sh +66 -0
- aiecs/scripts/tools_develop/schema_coverage.py +511 -0
- aiecs/scripts/tools_develop/validate_tool_schemas.py +475 -0
- aiecs/scripts/tools_develop/verify_executor_config_fix.py +98 -0
- aiecs/scripts/tools_develop/verify_tools.py +352 -0
- aiecs/tasks/__init__.py +0 -1
- aiecs/tasks/worker.py +115 -47
- aiecs/tools/__init__.py +194 -72
- aiecs/tools/apisource/__init__.py +99 -0
- aiecs/tools/apisource/intelligence/__init__.py +19 -0
- aiecs/tools/apisource/intelligence/data_fusion.py +632 -0
- aiecs/tools/apisource/intelligence/query_analyzer.py +417 -0
- aiecs/tools/apisource/intelligence/search_enhancer.py +385 -0
- aiecs/tools/apisource/monitoring/__init__.py +9 -0
- aiecs/tools/apisource/monitoring/metrics.py +330 -0
- aiecs/tools/apisource/providers/__init__.py +112 -0
- aiecs/tools/apisource/providers/base.py +671 -0
- aiecs/tools/apisource/providers/census.py +397 -0
- aiecs/tools/apisource/providers/fred.py +535 -0
- aiecs/tools/apisource/providers/newsapi.py +409 -0
- aiecs/tools/apisource/providers/worldbank.py +352 -0
- aiecs/tools/apisource/reliability/__init__.py +12 -0
- aiecs/tools/apisource/reliability/error_handler.py +363 -0
- aiecs/tools/apisource/reliability/fallback_strategy.py +376 -0
- aiecs/tools/apisource/tool.py +832 -0
- aiecs/tools/apisource/utils/__init__.py +9 -0
- aiecs/tools/apisource/utils/validators.py +334 -0
- aiecs/tools/base_tool.py +415 -21
- aiecs/tools/docs/__init__.py +121 -0
- aiecs/tools/docs/ai_document_orchestrator.py +607 -0
- aiecs/tools/docs/ai_document_writer_orchestrator.py +2350 -0
- aiecs/tools/docs/content_insertion_tool.py +1320 -0
- aiecs/tools/docs/document_creator_tool.py +1323 -0
- aiecs/tools/docs/document_layout_tool.py +1160 -0
- aiecs/tools/docs/document_parser_tool.py +1011 -0
- aiecs/tools/docs/document_writer_tool.py +1829 -0
- aiecs/tools/knowledge_graph/__init__.py +17 -0
- aiecs/tools/knowledge_graph/graph_reasoning_tool.py +807 -0
- aiecs/tools/knowledge_graph/graph_search_tool.py +944 -0
- aiecs/tools/knowledge_graph/kg_builder_tool.py +524 -0
- aiecs/tools/langchain_adapter.py +300 -138
- aiecs/tools/schema_generator.py +455 -0
- aiecs/tools/search_tool/__init__.py +100 -0
- aiecs/tools/search_tool/analyzers.py +581 -0
- aiecs/tools/search_tool/cache.py +264 -0
- aiecs/tools/search_tool/constants.py +128 -0
- aiecs/tools/search_tool/context.py +224 -0
- aiecs/tools/search_tool/core.py +778 -0
- aiecs/tools/search_tool/deduplicator.py +119 -0
- aiecs/tools/search_tool/error_handler.py +242 -0
- aiecs/tools/search_tool/metrics.py +343 -0
- aiecs/tools/search_tool/rate_limiter.py +172 -0
- aiecs/tools/search_tool/schemas.py +275 -0
- aiecs/tools/statistics/__init__.py +80 -0
- aiecs/tools/statistics/ai_data_analysis_orchestrator.py +646 -0
- aiecs/tools/statistics/ai_insight_generator_tool.py +508 -0
- aiecs/tools/statistics/ai_report_orchestrator_tool.py +684 -0
- aiecs/tools/statistics/data_loader_tool.py +555 -0
- aiecs/tools/statistics/data_profiler_tool.py +638 -0
- aiecs/tools/statistics/data_transformer_tool.py +580 -0
- aiecs/tools/statistics/data_visualizer_tool.py +498 -0
- aiecs/tools/statistics/model_trainer_tool.py +507 -0
- aiecs/tools/statistics/statistical_analyzer_tool.py +472 -0
- aiecs/tools/task_tools/__init__.py +49 -36
- aiecs/tools/task_tools/chart_tool.py +200 -184
- aiecs/tools/task_tools/classfire_tool.py +268 -267
- aiecs/tools/task_tools/image_tool.py +175 -131
- aiecs/tools/task_tools/office_tool.py +226 -146
- aiecs/tools/task_tools/pandas_tool.py +477 -121
- aiecs/tools/task_tools/report_tool.py +390 -142
- aiecs/tools/task_tools/research_tool.py +149 -79
- aiecs/tools/task_tools/scraper_tool.py +339 -145
- aiecs/tools/task_tools/stats_tool.py +448 -209
- aiecs/tools/temp_file_manager.py +26 -24
- aiecs/tools/tool_executor/__init__.py +18 -16
- aiecs/tools/tool_executor/tool_executor.py +364 -52
- aiecs/utils/LLM_output_structor.py +74 -48
- aiecs/utils/__init__.py +14 -3
- aiecs/utils/base_callback.py +0 -3
- aiecs/utils/cache_provider.py +696 -0
- aiecs/utils/execution_utils.py +50 -31
- aiecs/utils/prompt_loader.py +1 -0
- aiecs/utils/token_usage_repository.py +37 -11
- aiecs/ws/socket_server.py +14 -4
- {aiecs-1.0.1.dist-info → aiecs-1.7.6.dist-info}/METADATA +52 -15
- aiecs-1.7.6.dist-info/RECORD +337 -0
- aiecs-1.7.6.dist-info/entry_points.txt +13 -0
- aiecs/config/registry.py +0 -19
- aiecs/domain/context/content_engine.py +0 -982
- aiecs/llm/base_client.py +0 -99
- aiecs/llm/openai_client.py +0 -125
- aiecs/llm/vertex_client.py +0 -186
- aiecs/llm/xai_client.py +0 -184
- aiecs/scripts/dependency_checker.py +0 -857
- aiecs/scripts/quick_dependency_check.py +0 -269
- aiecs/tools/task_tools/search_api.py +0 -7
- aiecs-1.0.1.dist-info/RECORD +0 -90
- aiecs-1.0.1.dist-info/entry_points.txt +0 -7
- /aiecs/scripts/{setup_nlp_data.sh → dependance_check/setup_nlp_data.sh} +0 -0
- /aiecs/scripts/{README_WEASEL_PATCH.md → dependance_patch/fix_weasel/README_WEASEL_PATCH.md} +0 -0
- /aiecs/scripts/{fix_weasel_validator.sh → dependance_patch/fix_weasel/fix_weasel_validator.sh} +0 -0
- /aiecs/scripts/{run_weasel_patch.sh → dependance_patch/fix_weasel/run_weasel_patch.sh} +0 -0
- {aiecs-1.0.1.dist-info → aiecs-1.7.6.dist-info}/WHEEL +0 -0
- {aiecs-1.0.1.dist-info → aiecs-1.7.6.dist-info}/licenses/LICENSE +0 -0
- {aiecs-1.0.1.dist-info → aiecs-1.7.6.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,563 @@
|
|
|
1
|
+
"""
|
|
2
|
+
Schema Mapping for Structured Data Import
|
|
3
|
+
|
|
4
|
+
Maps structured data (CSV, JSON) columns to knowledge graph entity and relation types
|
|
5
|
+
with support for property transformations.
|
|
6
|
+
"""
|
|
7
|
+
|
|
8
|
+
from typing import Dict, List, Optional, Any, cast, Callable
|
|
9
|
+
from enum import Enum
|
|
10
|
+
from pydantic import BaseModel, Field, field_validator
|
|
11
|
+
from aiecs.domain.knowledge_graph.schema.property_schema import PropertyType
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
class TransformationType(str, Enum):
|
|
15
|
+
"""Types of property transformations"""
|
|
16
|
+
|
|
17
|
+
RENAME = "rename" # Rename column to property
|
|
18
|
+
TYPE_CAST = "type_cast" # Cast value to different type
|
|
19
|
+
COMPUTE = "compute" # Compute value from multiple columns
|
|
20
|
+
CONSTANT = "constant" # Use constant value
|
|
21
|
+
SKIP = "skip" # Skip this column
|
|
22
|
+
|
|
23
|
+
|
|
24
|
+
class PropertyTransformation(BaseModel):
|
|
25
|
+
"""
|
|
26
|
+
Property transformation configuration
|
|
27
|
+
|
|
28
|
+
Defines how a source column/value is transformed into a target property.
|
|
29
|
+
"""
|
|
30
|
+
|
|
31
|
+
transformation_type: TransformationType = Field(..., description="Type of transformation to apply")
|
|
32
|
+
|
|
33
|
+
source_column: Optional[str] = Field(
|
|
34
|
+
default=None,
|
|
35
|
+
description="Source column name (for rename/type_cast/compute)",
|
|
36
|
+
)
|
|
37
|
+
|
|
38
|
+
target_property: str = Field(..., description="Target property name in entity/relation")
|
|
39
|
+
|
|
40
|
+
target_type: Optional[PropertyType] = Field(default=None, description="Target property type (for type_cast)")
|
|
41
|
+
|
|
42
|
+
constant_value: Optional[Any] = Field(
|
|
43
|
+
default=None,
|
|
44
|
+
description="Constant value (for constant transformation)",
|
|
45
|
+
)
|
|
46
|
+
|
|
47
|
+
compute_function: Optional[str] = Field(
|
|
48
|
+
default=None,
|
|
49
|
+
description="Function name for compute transformation (e.g., 'concat', 'sum')",
|
|
50
|
+
)
|
|
51
|
+
|
|
52
|
+
compute_args: Optional[List[str]] = Field(
|
|
53
|
+
default=None,
|
|
54
|
+
description="Additional column names for compute function",
|
|
55
|
+
)
|
|
56
|
+
|
|
57
|
+
@field_validator("transformation_type")
|
|
58
|
+
@classmethod
|
|
59
|
+
def validate_transformation_type(cls, v: TransformationType) -> TransformationType:
|
|
60
|
+
"""Validate transformation type"""
|
|
61
|
+
return v
|
|
62
|
+
|
|
63
|
+
def apply(self, row: Dict[str, Any]) -> Any:
|
|
64
|
+
"""
|
|
65
|
+
Apply transformation to a data row
|
|
66
|
+
|
|
67
|
+
Args:
|
|
68
|
+
row: Dictionary of column name -> value
|
|
69
|
+
|
|
70
|
+
Returns:
|
|
71
|
+
Transformed value for target property
|
|
72
|
+
"""
|
|
73
|
+
if self.transformation_type == TransformationType.RENAME:
|
|
74
|
+
if self.source_column is None:
|
|
75
|
+
raise ValueError("source_column required for rename transformation")
|
|
76
|
+
return row.get(self.source_column)
|
|
77
|
+
|
|
78
|
+
elif self.transformation_type == TransformationType.TYPE_CAST:
|
|
79
|
+
if self.source_column is None:
|
|
80
|
+
raise ValueError("source_column required for type_cast transformation")
|
|
81
|
+
if self.target_type is None:
|
|
82
|
+
raise ValueError("target_type required for type_cast transformation")
|
|
83
|
+
|
|
84
|
+
value = row.get(self.source_column)
|
|
85
|
+
if value is None:
|
|
86
|
+
return None
|
|
87
|
+
|
|
88
|
+
return self._cast_value(value, self.target_type)
|
|
89
|
+
|
|
90
|
+
elif self.transformation_type == TransformationType.COMPUTE:
|
|
91
|
+
if self.compute_function is None:
|
|
92
|
+
raise ValueError("compute_function required for compute transformation")
|
|
93
|
+
|
|
94
|
+
# Get source values
|
|
95
|
+
source_values = []
|
|
96
|
+
if self.source_column:
|
|
97
|
+
source_values.append(row.get(self.source_column))
|
|
98
|
+
if self.compute_args:
|
|
99
|
+
source_values.extend([row.get(col) for col in self.compute_args])
|
|
100
|
+
|
|
101
|
+
return self._compute_value(self.compute_function, source_values)
|
|
102
|
+
|
|
103
|
+
elif self.transformation_type == TransformationType.CONSTANT:
|
|
104
|
+
return self.constant_value
|
|
105
|
+
|
|
106
|
+
elif self.transformation_type == TransformationType.SKIP:
|
|
107
|
+
return None
|
|
108
|
+
|
|
109
|
+
else:
|
|
110
|
+
raise ValueError(f"Unknown transformation type: {self.transformation_type}")
|
|
111
|
+
|
|
112
|
+
def _cast_value(self, value: Any, target_type: PropertyType) -> Any:
|
|
113
|
+
"""Cast value to target type"""
|
|
114
|
+
try:
|
|
115
|
+
if target_type == PropertyType.STRING:
|
|
116
|
+
return str(value)
|
|
117
|
+
elif target_type == PropertyType.INTEGER:
|
|
118
|
+
if isinstance(value, bool):
|
|
119
|
+
raise ValueError(f"Cannot cast boolean {value} to integer")
|
|
120
|
+
return int(float(value)) # Handle "123.0" -> 123
|
|
121
|
+
elif target_type == PropertyType.FLOAT:
|
|
122
|
+
if isinstance(value, bool):
|
|
123
|
+
raise ValueError(f"Cannot cast boolean {value} to float")
|
|
124
|
+
return float(value)
|
|
125
|
+
elif target_type == PropertyType.BOOLEAN:
|
|
126
|
+
if isinstance(value, str):
|
|
127
|
+
return value.lower() in ("true", "1", "yes", "on")
|
|
128
|
+
return bool(value)
|
|
129
|
+
elif target_type == PropertyType.LIST:
|
|
130
|
+
if isinstance(value, list):
|
|
131
|
+
return value
|
|
132
|
+
elif isinstance(value, str):
|
|
133
|
+
# Try to parse as JSON list or comma-separated
|
|
134
|
+
import json
|
|
135
|
+
|
|
136
|
+
try:
|
|
137
|
+
return json.loads(value)
|
|
138
|
+
except json.JSONDecodeError:
|
|
139
|
+
return [v.strip() for v in value.split(",")]
|
|
140
|
+
else:
|
|
141
|
+
return [value]
|
|
142
|
+
elif target_type == PropertyType.DICT:
|
|
143
|
+
if isinstance(value, dict):
|
|
144
|
+
return value
|
|
145
|
+
elif isinstance(value, str):
|
|
146
|
+
import json
|
|
147
|
+
|
|
148
|
+
try:
|
|
149
|
+
return json.loads(value)
|
|
150
|
+
except json.JSONDecodeError:
|
|
151
|
+
# If not valid JSON, wrap as dict with "value" key
|
|
152
|
+
return {"value": value}
|
|
153
|
+
else:
|
|
154
|
+
return {"value": value}
|
|
155
|
+
else:
|
|
156
|
+
return value # ANY type or unknown
|
|
157
|
+
except (ValueError, TypeError) as e:
|
|
158
|
+
raise ValueError(f"Failed to cast {value} to {target_type}: {e}")
|
|
159
|
+
|
|
160
|
+
def _compute_value(self, function_name: str, values: List[Any]) -> Any:
|
|
161
|
+
"""Compute value using function"""
|
|
162
|
+
# Remove None values for most functions
|
|
163
|
+
non_none_values = [v for v in values if v is not None]
|
|
164
|
+
|
|
165
|
+
if function_name == "concat":
|
|
166
|
+
return "".join(str(v) for v in values if v is not None)
|
|
167
|
+
elif function_name == "concat_space":
|
|
168
|
+
return " ".join(str(v) for v in values if v is not None)
|
|
169
|
+
elif function_name == "concat_comma":
|
|
170
|
+
return ", ".join(str(v) for v in values if v is not None)
|
|
171
|
+
elif function_name == "sum":
|
|
172
|
+
return sum(float(v) for v in non_none_values if self._is_numeric(v))
|
|
173
|
+
elif function_name == "avg" or function_name == "average":
|
|
174
|
+
if not non_none_values:
|
|
175
|
+
return None
|
|
176
|
+
numeric_values = [float(v) for v in non_none_values if self._is_numeric(v)]
|
|
177
|
+
if not numeric_values:
|
|
178
|
+
return None
|
|
179
|
+
return sum(numeric_values) / len(numeric_values)
|
|
180
|
+
elif function_name == "max":
|
|
181
|
+
if not non_none_values:
|
|
182
|
+
return None
|
|
183
|
+
numeric_values = [float(v) for v in non_none_values if self._is_numeric(v)]
|
|
184
|
+
if not numeric_values:
|
|
185
|
+
return max(non_none_values)
|
|
186
|
+
return max(numeric_values)
|
|
187
|
+
elif function_name == "min":
|
|
188
|
+
if not non_none_values:
|
|
189
|
+
return None
|
|
190
|
+
numeric_values = [float(v) for v in non_none_values if self._is_numeric(v)]
|
|
191
|
+
if not numeric_values:
|
|
192
|
+
return min(non_none_values)
|
|
193
|
+
return min(numeric_values)
|
|
194
|
+
else:
|
|
195
|
+
raise ValueError(f"Unknown compute function: {function_name}")
|
|
196
|
+
|
|
197
|
+
@staticmethod
|
|
198
|
+
def _is_numeric(value: Any) -> bool:
|
|
199
|
+
"""Check if value is numeric"""
|
|
200
|
+
try:
|
|
201
|
+
float(value)
|
|
202
|
+
return True
|
|
203
|
+
except (ValueError, TypeError):
|
|
204
|
+
return False
|
|
205
|
+
|
|
206
|
+
|
|
207
|
+
class EntityMapping(BaseModel):
|
|
208
|
+
"""
|
|
209
|
+
Entity mapping configuration
|
|
210
|
+
|
|
211
|
+
Maps source data columns to an entity type with property transformations.
|
|
212
|
+
"""
|
|
213
|
+
|
|
214
|
+
source_columns: List[str] = Field(..., description="Source column names to use for this entity")
|
|
215
|
+
|
|
216
|
+
entity_type: str = Field(..., description="Target entity type name")
|
|
217
|
+
|
|
218
|
+
property_mapping: Dict[str, str] = Field(
|
|
219
|
+
default_factory=dict,
|
|
220
|
+
description="Simple column-to-property mapping (column_name -> property_name)",
|
|
221
|
+
)
|
|
222
|
+
|
|
223
|
+
transformations: List[PropertyTransformation] = Field(default_factory=list, description="Property transformations to apply")
|
|
224
|
+
|
|
225
|
+
id_column: Optional[str] = Field(
|
|
226
|
+
default=None,
|
|
227
|
+
description="Column to use as entity ID (default: first column or generated)",
|
|
228
|
+
)
|
|
229
|
+
|
|
230
|
+
@field_validator("source_columns")
|
|
231
|
+
@classmethod
|
|
232
|
+
def validate_source_columns(cls, v: List[str]) -> List[str]:
|
|
233
|
+
"""Validate source columns are not empty"""
|
|
234
|
+
if not v:
|
|
235
|
+
raise ValueError("source_columns cannot be empty")
|
|
236
|
+
return v
|
|
237
|
+
|
|
238
|
+
def map_row_to_entity(self, row: Dict[str, Any], entity_id: Optional[str] = None) -> Dict[str, Any]:
|
|
239
|
+
"""
|
|
240
|
+
Map a data row to entity properties
|
|
241
|
+
|
|
242
|
+
Args:
|
|
243
|
+
row: Dictionary of column name -> value
|
|
244
|
+
entity_id: Optional entity ID (if not provided, will use id_column or generate)
|
|
245
|
+
|
|
246
|
+
Returns:
|
|
247
|
+
Dictionary with entity properties
|
|
248
|
+
"""
|
|
249
|
+
properties = {}
|
|
250
|
+
|
|
251
|
+
# Apply simple property mappings first
|
|
252
|
+
for column, property_name in self.property_mapping.items():
|
|
253
|
+
if column in row:
|
|
254
|
+
properties[property_name] = row[column]
|
|
255
|
+
|
|
256
|
+
# Apply transformations
|
|
257
|
+
for transformation in self.transformations:
|
|
258
|
+
try:
|
|
259
|
+
value = transformation.apply(row)
|
|
260
|
+
if value is not None or transformation.transformation_type != TransformationType.SKIP:
|
|
261
|
+
properties[transformation.target_property] = value
|
|
262
|
+
except Exception as e:
|
|
263
|
+
# Log warning but continue
|
|
264
|
+
import logging
|
|
265
|
+
|
|
266
|
+
logger = logging.getLogger(__name__)
|
|
267
|
+
logger.warning(f"Transformation failed for {transformation.target_property}: {e}")
|
|
268
|
+
|
|
269
|
+
# Determine entity ID
|
|
270
|
+
if entity_id is None:
|
|
271
|
+
if self.id_column and self.id_column in row:
|
|
272
|
+
entity_id = str(row[self.id_column])
|
|
273
|
+
elif self.source_columns:
|
|
274
|
+
# Use first column as ID
|
|
275
|
+
entity_id = str(row.get(self.source_columns[0], ""))
|
|
276
|
+
|
|
277
|
+
return {
|
|
278
|
+
"id": entity_id,
|
|
279
|
+
"type": self.entity_type,
|
|
280
|
+
"properties": properties,
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
|
|
284
|
+
class RelationMapping(BaseModel):
|
|
285
|
+
"""
|
|
286
|
+
Relation mapping configuration
|
|
287
|
+
|
|
288
|
+
Maps source data columns to a relation type between entities.
|
|
289
|
+
"""
|
|
290
|
+
|
|
291
|
+
source_columns: List[str] = Field(..., description="Source column names to use for this relation")
|
|
292
|
+
|
|
293
|
+
relation_type: str = Field(..., description="Target relation type name")
|
|
294
|
+
|
|
295
|
+
source_entity_column: str = Field(..., description="Column name containing source entity ID")
|
|
296
|
+
|
|
297
|
+
target_entity_column: str = Field(..., description="Column name containing target entity ID")
|
|
298
|
+
|
|
299
|
+
property_mapping: Dict[str, str] = Field(default_factory=dict, description="Simple column-to-property mapping")
|
|
300
|
+
|
|
301
|
+
transformations: List[PropertyTransformation] = Field(default_factory=list, description="Property transformations to apply")
|
|
302
|
+
|
|
303
|
+
@field_validator("source_columns")
|
|
304
|
+
@classmethod
|
|
305
|
+
def validate_source_columns(cls, v: List[str]) -> List[str]:
|
|
306
|
+
"""Validate source columns are not empty"""
|
|
307
|
+
if not v:
|
|
308
|
+
raise ValueError("source_columns cannot be empty")
|
|
309
|
+
return v
|
|
310
|
+
|
|
311
|
+
@field_validator("source_entity_column", "target_entity_column")
|
|
312
|
+
@classmethod
|
|
313
|
+
def validate_entity_columns(cls, v: str) -> str:
|
|
314
|
+
"""Validate entity column names are provided"""
|
|
315
|
+
if not v:
|
|
316
|
+
raise ValueError("Entity column names cannot be empty")
|
|
317
|
+
return v
|
|
318
|
+
|
|
319
|
+
def map_row_to_relation(self, row: Dict[str, Any]) -> Dict[str, Any]:
|
|
320
|
+
"""
|
|
321
|
+
Map a data row to relation properties
|
|
322
|
+
|
|
323
|
+
Args:
|
|
324
|
+
row: Dictionary of column name -> value
|
|
325
|
+
|
|
326
|
+
Returns:
|
|
327
|
+
Dictionary with relation properties (source_id, target_id, type, properties)
|
|
328
|
+
"""
|
|
329
|
+
# Get source and target entity IDs
|
|
330
|
+
source_id = str(row.get(self.source_entity_column, ""))
|
|
331
|
+
target_id = str(row.get(self.target_entity_column, ""))
|
|
332
|
+
|
|
333
|
+
if not source_id or not target_id:
|
|
334
|
+
raise ValueError(f"Missing entity IDs: source={source_id}, target={target_id}. " f"Columns: source={self.source_entity_column}, target={self.target_entity_column}")
|
|
335
|
+
|
|
336
|
+
properties = {}
|
|
337
|
+
|
|
338
|
+
# Apply simple property mappings
|
|
339
|
+
for column, property_name in self.property_mapping.items():
|
|
340
|
+
if column in row:
|
|
341
|
+
properties[property_name] = row[column]
|
|
342
|
+
|
|
343
|
+
# Apply transformations
|
|
344
|
+
for transformation in self.transformations:
|
|
345
|
+
try:
|
|
346
|
+
value = transformation.apply(row)
|
|
347
|
+
if value is not None or transformation.transformation_type != TransformationType.SKIP:
|
|
348
|
+
properties[transformation.target_property] = value
|
|
349
|
+
except Exception as e:
|
|
350
|
+
# Log warning but continue
|
|
351
|
+
import logging
|
|
352
|
+
|
|
353
|
+
logger = logging.getLogger(__name__)
|
|
354
|
+
logger.warning(f"Transformation failed for {transformation.target_property}: {e}")
|
|
355
|
+
|
|
356
|
+
return {
|
|
357
|
+
"source_id": source_id,
|
|
358
|
+
"target_id": target_id,
|
|
359
|
+
"type": self.relation_type,
|
|
360
|
+
"properties": properties,
|
|
361
|
+
}
|
|
362
|
+
|
|
363
|
+
|
|
364
|
+
class AggregationFunction(str, Enum):
|
|
365
|
+
"""Statistical aggregation functions"""
|
|
366
|
+
|
|
367
|
+
MEAN = "mean"
|
|
368
|
+
STD = "std"
|
|
369
|
+
MIN = "min"
|
|
370
|
+
MAX = "max"
|
|
371
|
+
COUNT = "count"
|
|
372
|
+
SUM = "sum"
|
|
373
|
+
MEDIAN = "median"
|
|
374
|
+
VARIANCE = "variance"
|
|
375
|
+
|
|
376
|
+
|
|
377
|
+
class AggregationConfig(BaseModel):
|
|
378
|
+
"""
|
|
379
|
+
Configuration for statistical aggregation during import
|
|
380
|
+
|
|
381
|
+
Defines how to compute aggregated statistics from source data
|
|
382
|
+
and store them as entity properties.
|
|
383
|
+
"""
|
|
384
|
+
|
|
385
|
+
source_property: str = Field(..., description="Source property to aggregate")
|
|
386
|
+
|
|
387
|
+
function: AggregationFunction = Field(..., description="Aggregation function to apply")
|
|
388
|
+
|
|
389
|
+
target_property: str = Field(..., description="Target property name for aggregated value")
|
|
390
|
+
|
|
391
|
+
group_by: Optional[List[str]] = Field(
|
|
392
|
+
default=None,
|
|
393
|
+
description="Columns to group by (for grouped aggregations)",
|
|
394
|
+
)
|
|
395
|
+
|
|
396
|
+
filter_condition: Optional[str] = Field(
|
|
397
|
+
default=None,
|
|
398
|
+
description="Optional filter condition (e.g., 'value > 0')",
|
|
399
|
+
)
|
|
400
|
+
|
|
401
|
+
|
|
402
|
+
class EntityAggregation(BaseModel):
|
|
403
|
+
"""
|
|
404
|
+
Aggregation configuration for an entity type
|
|
405
|
+
|
|
406
|
+
Defines aggregations to compute for entities of a specific type.
|
|
407
|
+
"""
|
|
408
|
+
|
|
409
|
+
entity_type: str = Field(..., description="Entity type to aggregate")
|
|
410
|
+
|
|
411
|
+
aggregations: List[AggregationConfig] = Field(
|
|
412
|
+
default_factory=list,
|
|
413
|
+
description="List of aggregations to compute",
|
|
414
|
+
)
|
|
415
|
+
|
|
416
|
+
incremental: bool = Field(
|
|
417
|
+
default=True,
|
|
418
|
+
description="Whether to compute aggregations incrementally during batch processing",
|
|
419
|
+
)
|
|
420
|
+
|
|
421
|
+
|
|
422
|
+
class SchemaMapping(BaseModel):
|
|
423
|
+
"""
|
|
424
|
+
Schema mapping configuration
|
|
425
|
+
|
|
426
|
+
Defines how structured data (CSV, JSON) maps to knowledge graph entities and relations.
|
|
427
|
+
"""
|
|
428
|
+
|
|
429
|
+
entity_mappings: List[EntityMapping] = Field(default_factory=list, description="Entity type mappings")
|
|
430
|
+
|
|
431
|
+
relation_mappings: List[RelationMapping] = Field(default_factory=list, description="Relation type mappings")
|
|
432
|
+
|
|
433
|
+
aggregations: List[EntityAggregation] = Field(
|
|
434
|
+
default_factory=list,
|
|
435
|
+
description="Statistical aggregations to compute during import",
|
|
436
|
+
)
|
|
437
|
+
|
|
438
|
+
validation_config: Optional[Dict[str, Any]] = Field(
|
|
439
|
+
default=None,
|
|
440
|
+
description="Data quality validation configuration",
|
|
441
|
+
)
|
|
442
|
+
|
|
443
|
+
description: Optional[str] = Field(default=None, description="Human-readable description of this mapping")
|
|
444
|
+
|
|
445
|
+
def validate_mapping(self) -> List[str]:
|
|
446
|
+
"""
|
|
447
|
+
Validate mapping consistency
|
|
448
|
+
|
|
449
|
+
Returns:
|
|
450
|
+
List of validation error messages (empty if valid)
|
|
451
|
+
"""
|
|
452
|
+
errors = []
|
|
453
|
+
|
|
454
|
+
# Check entity mappings
|
|
455
|
+
entity_type_names = set()
|
|
456
|
+
for i, mapping in enumerate(self.entity_mappings):
|
|
457
|
+
if not mapping.entity_type:
|
|
458
|
+
errors.append(f"Entity mapping {i}: entity_type is required")
|
|
459
|
+
|
|
460
|
+
if mapping.entity_type in entity_type_names:
|
|
461
|
+
errors.append(f"Entity mapping {i}: duplicate entity_type '{mapping.entity_type}'")
|
|
462
|
+
entity_type_names.add(mapping.entity_type)
|
|
463
|
+
|
|
464
|
+
# Check that source columns are specified
|
|
465
|
+
if not mapping.source_columns:
|
|
466
|
+
errors.append(f"Entity mapping {i}: source_columns cannot be empty")
|
|
467
|
+
|
|
468
|
+
# Check transformations
|
|
469
|
+
for j, trans in enumerate(mapping.transformations):
|
|
470
|
+
if not trans.target_property:
|
|
471
|
+
errors.append(f"Entity mapping {i}, transformation {j}: target_property is required")
|
|
472
|
+
|
|
473
|
+
if trans.transformation_type == TransformationType.RENAME:
|
|
474
|
+
if not trans.source_column:
|
|
475
|
+
errors.append(f"Entity mapping {i}, transformation {j}: " f"source_column required for rename")
|
|
476
|
+
|
|
477
|
+
elif trans.transformation_type == TransformationType.TYPE_CAST:
|
|
478
|
+
if not trans.source_column:
|
|
479
|
+
errors.append(f"Entity mapping {i}, transformation {j}: " f"source_column required for type_cast")
|
|
480
|
+
if not trans.target_type:
|
|
481
|
+
errors.append(f"Entity mapping {i}, transformation {j}: " f"target_type required for type_cast")
|
|
482
|
+
|
|
483
|
+
elif trans.transformation_type == TransformationType.COMPUTE:
|
|
484
|
+
if not trans.compute_function:
|
|
485
|
+
errors.append(f"Entity mapping {i}, transformation {j}: " f"compute_function required for compute")
|
|
486
|
+
|
|
487
|
+
# Check relation mappings
|
|
488
|
+
relation_type_names = set()
|
|
489
|
+
for i, relation_mapping in enumerate(self.relation_mappings):
|
|
490
|
+
if not relation_mapping.relation_type:
|
|
491
|
+
errors.append(f"Relation mapping {i}: relation_type is required")
|
|
492
|
+
|
|
493
|
+
if relation_mapping.relation_type in relation_type_names:
|
|
494
|
+
errors.append(f"Relation mapping {i}: duplicate relation_type '{relation_mapping.relation_type}'")
|
|
495
|
+
relation_type_names.add(relation_mapping.relation_type)
|
|
496
|
+
|
|
497
|
+
# Check entity columns
|
|
498
|
+
if not relation_mapping.source_entity_column:
|
|
499
|
+
errors.append(f"Relation mapping {i}: source_entity_column is required")
|
|
500
|
+
if not relation_mapping.target_entity_column:
|
|
501
|
+
errors.append(f"Relation mapping {i}: target_entity_column is required")
|
|
502
|
+
|
|
503
|
+
# Check that source columns include entity columns
|
|
504
|
+
if relation_mapping.source_entity_column not in relation_mapping.source_columns:
|
|
505
|
+
errors.append(f"Relation mapping {i}: source_entity_column '{relation_mapping.source_entity_column}' " f"must be in source_columns")
|
|
506
|
+
if relation_mapping.target_entity_column not in relation_mapping.source_columns:
|
|
507
|
+
errors.append(f"Relation mapping {i}: target_entity_column '{relation_mapping.target_entity_column}' " f"must be in source_columns")
|
|
508
|
+
|
|
509
|
+
return errors
|
|
510
|
+
|
|
511
|
+
def is_valid(self) -> bool:
|
|
512
|
+
"""
|
|
513
|
+
Check if mapping is valid
|
|
514
|
+
|
|
515
|
+
Returns:
|
|
516
|
+
True if mapping is valid
|
|
517
|
+
"""
|
|
518
|
+
return len(self.validate_mapping()) == 0
|
|
519
|
+
|
|
520
|
+
def get_entity_mapping(self, entity_type: str) -> Optional[EntityMapping]:
|
|
521
|
+
"""
|
|
522
|
+
Get entity mapping by entity type name
|
|
523
|
+
|
|
524
|
+
Args:
|
|
525
|
+
entity_type: Entity type name
|
|
526
|
+
|
|
527
|
+
Returns:
|
|
528
|
+
Entity mapping or None if not found
|
|
529
|
+
"""
|
|
530
|
+
for mapping in self.entity_mappings:
|
|
531
|
+
if mapping.entity_type == entity_type:
|
|
532
|
+
return mapping
|
|
533
|
+
return None
|
|
534
|
+
|
|
535
|
+
def get_relation_mapping(self, relation_type: str) -> Optional[RelationMapping]:
|
|
536
|
+
"""
|
|
537
|
+
Get relation mapping by relation type name
|
|
538
|
+
|
|
539
|
+
Args:
|
|
540
|
+
relation_type: Relation type name
|
|
541
|
+
|
|
542
|
+
Returns:
|
|
543
|
+
Relation mapping or None if not found
|
|
544
|
+
"""
|
|
545
|
+
for mapping in self.relation_mappings:
|
|
546
|
+
if mapping.relation_type == relation_type:
|
|
547
|
+
return mapping
|
|
548
|
+
return None
|
|
549
|
+
|
|
550
|
+
def get_aggregations(self, entity_type: str) -> Optional[EntityAggregation]:
|
|
551
|
+
"""
|
|
552
|
+
Get aggregation configuration for entity type
|
|
553
|
+
|
|
554
|
+
Args:
|
|
555
|
+
entity_type: Entity type name
|
|
556
|
+
|
|
557
|
+
Returns:
|
|
558
|
+
EntityAggregation or None if not found
|
|
559
|
+
"""
|
|
560
|
+
for agg in self.aggregations:
|
|
561
|
+
if agg.entity_type == entity_type:
|
|
562
|
+
return agg
|
|
563
|
+
return None
|