rasa-pro 3.14.0.dev20250922__py3-none-any.whl → 3.14.0rc2__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 rasa-pro might be problematic. Click here for more details.
- rasa/__main__.py +15 -3
- rasa/agents/__init__.py +0 -0
- rasa/agents/agent_factory.py +122 -0
- rasa/agents/agent_manager.py +211 -0
- rasa/agents/constants.py +43 -0
- rasa/agents/core/__init__.py +0 -0
- rasa/agents/core/agent_protocol.py +107 -0
- rasa/agents/core/types.py +81 -0
- rasa/agents/exceptions.py +38 -0
- rasa/agents/protocol/__init__.py +5 -0
- rasa/agents/protocol/a2a/__init__.py +0 -0
- rasa/agents/protocol/a2a/a2a_agent.py +879 -0
- rasa/agents/protocol/mcp/__init__.py +0 -0
- rasa/agents/protocol/mcp/mcp_base_agent.py +726 -0
- rasa/agents/protocol/mcp/mcp_open_agent.py +327 -0
- rasa/agents/protocol/mcp/mcp_task_agent.py +522 -0
- rasa/agents/schemas/__init__.py +13 -0
- rasa/agents/schemas/agent_input.py +38 -0
- rasa/agents/schemas/agent_output.py +26 -0
- rasa/agents/schemas/agent_tool_result.py +65 -0
- rasa/agents/schemas/agent_tool_schema.py +186 -0
- rasa/agents/templates/__init__.py +0 -0
- rasa/agents/templates/mcp_open_agent_prompt_template.jinja2 +20 -0
- rasa/agents/templates/mcp_task_agent_prompt_template.jinja2 +22 -0
- rasa/agents/utils.py +206 -0
- rasa/agents/validation.py +485 -0
- rasa/api.py +24 -9
- rasa/builder/config.py +6 -2
- rasa/builder/copilot/constants.py +4 -1
- rasa/builder/copilot/copilot.py +155 -79
- rasa/builder/copilot/models.py +304 -108
- rasa/builder/copilot/prompts/copilot_training_error_handler_prompt.jinja2 +53 -0
- rasa/builder/guardrails/{lakera.py → clients.py} +55 -5
- rasa/builder/guardrails/constants.py +3 -0
- rasa/builder/guardrails/models.py +45 -10
- rasa/builder/guardrails/policy_checker.py +324 -0
- rasa/builder/guardrails/utils.py +42 -276
- rasa/builder/jobs.py +182 -12
- rasa/builder/llm_service.py +32 -5
- rasa/builder/models.py +13 -3
- rasa/builder/project_generator.py +6 -1
- rasa/builder/service.py +31 -15
- rasa/builder/training_service.py +18 -24
- rasa/builder/validation_service.py +1 -1
- rasa/cli/arguments/default_arguments.py +12 -0
- rasa/cli/arguments/run.py +2 -0
- rasa/cli/arguments/train.py +2 -0
- rasa/cli/data.py +10 -8
- rasa/cli/dialogue_understanding_test.py +10 -7
- rasa/cli/e2e_test.py +9 -6
- rasa/cli/evaluate.py +4 -2
- rasa/cli/export.py +5 -2
- rasa/cli/inspect.py +8 -4
- rasa/cli/interactive.py +5 -4
- rasa/cli/llm_fine_tuning.py +11 -6
- rasa/cli/project_templates/finance/domain/general/help.yml +0 -0
- rasa/cli/project_templates/tutorial/credentials.yml +10 -0
- rasa/cli/run.py +12 -10
- rasa/cli/scaffold.py +4 -4
- rasa/cli/shell.py +9 -5
- rasa/cli/studio/studio.py +1 -1
- rasa/cli/test.py +34 -14
- rasa/cli/train.py +41 -28
- rasa/cli/utils.py +1 -393
- rasa/cli/validation/__init__.py +0 -0
- rasa/cli/validation/bot_config.py +223 -0
- rasa/cli/validation/config_path_validation.py +257 -0
- rasa/cli/x.py +8 -4
- rasa/constants.py +7 -1
- rasa/core/actions/action.py +51 -10
- rasa/core/actions/grpc_custom_action_executor.py +1 -1
- rasa/core/agent.py +19 -2
- rasa/core/available_agents.py +229 -0
- rasa/core/brokers/kafka.py +5 -1
- rasa/core/channels/__init__.py +82 -35
- rasa/core/channels/development_inspector.py +3 -3
- rasa/core/channels/inspector/README.md +25 -13
- rasa/core/channels/inspector/dist/assets/{arc-35222594.js → arc-6177260a.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{blockDiagram-38ab4fdb-a0efbfd3.js → blockDiagram-38ab4fdb-b054f038.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{c4Diagram-3d4e48cf-0584c0f2.js → c4Diagram-3d4e48cf-f25427d5.js} +1 -1
- rasa/core/channels/inspector/dist/assets/channel-bf9cbb34.js +1 -0
- rasa/core/channels/inspector/dist/assets/{classDiagram-70f12bd4-39f40dbe.js → classDiagram-70f12bd4-c7a2af53.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{classDiagram-v2-f2320105-1ad755f3.js → classDiagram-v2-f2320105-58db65c0.js} +1 -1
- rasa/core/channels/inspector/dist/assets/clone-8f9083bb.js +1 -0
- rasa/core/channels/inspector/dist/assets/{createText-2e5e7dd3-b0f4f0fe.js → createText-2e5e7dd3-088372e2.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{edges-e0da2a9e-9039bff9.js → edges-e0da2a9e-58676240.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{erDiagram-9861fffd-65c9b127.js → erDiagram-9861fffd-0c14d7c6.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{flowDb-956e92f1-4f08b38e.js → flowDb-956e92f1-ea63f85c.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{flowDiagram-66a62f08-e95c362a.js → flowDiagram-66a62f08-a2af48cd.js} +1 -1
- rasa/core/channels/inspector/dist/assets/flowDiagram-v2-96b9c2cf-9ecd5b59.js +1 -0
- rasa/core/channels/inspector/dist/assets/{flowchart-elk-definition-4a651766-703c3015.js → flowchart-elk-definition-4a651766-6937abe7.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{ganttDiagram-c361ad54-699328ea.js → ganttDiagram-c361ad54-7473f357.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{gitGraphDiagram-72cf32ee-04cf4b05.js → gitGraphDiagram-72cf32ee-d0c9405e.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{graph-ee94449e.js → graph-0a6f8466.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{index-3862675e-940162b4.js → index-3862675e-7610671a.js} +1 -1
- rasa/core/channels/inspector/dist/assets/index-74e01d94.js +1354 -0
- rasa/core/channels/inspector/dist/assets/{infoDiagram-f8f76790-c79c2866.js → infoDiagram-f8f76790-be397dc7.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{journeyDiagram-49397b02-84489d30.js → journeyDiagram-49397b02-4cefbf62.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{layout-a9aa9858.js → layout-e7fbc2bf.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{line-eb73cf26.js → line-a8aa457c.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{linear-b3399f9a.js → linear-3351e0d2.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{mindmap-definition-fc14e90a-b095bf1a.js → mindmap-definition-fc14e90a-b8cbf605.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{pieDiagram-8a3498a8-07644b66.js → pieDiagram-8a3498a8-f327f774.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{quadrantDiagram-120e2f19-573a3f9c.js → quadrantDiagram-120e2f19-2854c591.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{requirementDiagram-deff3bca-d457e1e1.js → requirementDiagram-deff3bca-964985d5.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{sankeyDiagram-04a897e0-9d26e1a2.js → sankeyDiagram-04a897e0-edeb4f33.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{sequenceDiagram-704730f1-3a9cde10.js → sequenceDiagram-704730f1-fcf70125.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{stateDiagram-587899a1-4f3e8cec.js → stateDiagram-587899a1-0e770395.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{stateDiagram-v2-d93cdb3a-e617e5bf.js → stateDiagram-v2-d93cdb3a-af8dcd22.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{styles-6aaf32cf-eab30d2f.js → styles-6aaf32cf-36a9e70d.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{styles-9a916d00-09994be2.js → styles-9a916d00-884a8b5b.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{styles-c10674c1-b7110364.js → styles-c10674c1-dc097813.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{svgDrawCommon-08f97a94-3ebc92ad.js → svgDrawCommon-08f97a94-5a2c7eed.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{timeline-definition-85554ec2-7d13d2f2.js → timeline-definition-85554ec2-e89c4f6e.js} +1 -1
- rasa/core/channels/inspector/dist/assets/{xychartDiagram-e933f94c-488385e1.js → xychartDiagram-e933f94c-afb6fe56.js} +1 -1
- rasa/core/channels/inspector/dist/index.html +1 -1
- rasa/core/channels/inspector/package.json +18 -18
- rasa/core/channels/inspector/src/App.tsx +29 -4
- rasa/core/channels/inspector/src/components/DialogueAgentStack.tsx +108 -0
- rasa/core/channels/inspector/src/components/{DialogueStack.tsx → DialogueHistoryStack.tsx} +4 -2
- rasa/core/channels/inspector/src/helpers/audio/audiostream.ts +7 -4
- rasa/core/channels/inspector/src/helpers/formatters.test.ts +4 -0
- rasa/core/channels/inspector/src/helpers/formatters.ts +24 -3
- rasa/core/channels/inspector/src/helpers/utils.test.ts +127 -0
- rasa/core/channels/inspector/src/helpers/utils.ts +66 -1
- rasa/core/channels/inspector/src/theme/base/styles.ts +19 -1
- rasa/core/channels/inspector/src/types.ts +21 -0
- rasa/core/channels/inspector/yarn.lock +336 -189
- rasa/core/channels/studio_chat.py +6 -6
- rasa/core/channels/telegram.py +4 -9
- rasa/core/channels/voice_stream/genesys.py +1 -1
- rasa/core/channels/voice_stream/tts/deepgram.py +140 -0
- rasa/core/channels/voice_stream/twilio_media_streams.py +5 -1
- rasa/core/channels/voice_stream/voice_channel.py +3 -0
- rasa/core/concurrent_lock_store.py +38 -21
- rasa/core/config/__init__.py +0 -0
- rasa/core/{available_endpoints.py → config/available_endpoints.py} +51 -16
- rasa/core/config/configuration.py +260 -0
- rasa/core/config/credentials.py +19 -0
- rasa/core/config/message_procesing_config.py +34 -0
- rasa/core/constants.py +10 -0
- rasa/core/iam_credentials_providers/aws_iam_credentials_providers.py +69 -4
- rasa/core/iam_credentials_providers/credentials_provider_protocol.py +2 -1
- rasa/core/lock_store.py +4 -0
- rasa/core/policies/enterprise_search_policy.py +5 -3
- rasa/core/policies/flow_policy.py +4 -4
- rasa/core/policies/flows/agent_executor.py +632 -0
- rasa/core/policies/flows/flow_executor.py +136 -75
- rasa/core/policies/flows/mcp_tool_executor.py +298 -0
- rasa/core/policies/intentless_policy.py +1 -1
- rasa/core/policies/ted_policy.py +20 -12
- rasa/core/policies/unexpected_intent_policy.py +6 -0
- rasa/core/processor.py +68 -44
- rasa/core/redis_connection_factory.py +7 -2
- rasa/core/run.py +37 -8
- rasa/core/test.py +4 -0
- rasa/core/tracker_stores/redis_tracker_store.py +4 -0
- rasa/core/tracker_stores/sql_tracker_store.py +3 -1
- rasa/core/tracker_stores/tracker_store.py +3 -7
- rasa/core/train.py +1 -1
- rasa/core/training/interactive.py +20 -18
- rasa/core/training/story_conflict.py +5 -5
- rasa/core/utils.py +22 -23
- rasa/dialogue_understanding/commands/__init__.py +8 -0
- rasa/dialogue_understanding/commands/cancel_flow_command.py +19 -5
- rasa/dialogue_understanding/commands/chit_chat_answer_command.py +21 -2
- rasa/dialogue_understanding/commands/clarify_command.py +20 -2
- rasa/dialogue_understanding/commands/continue_agent_command.py +91 -0
- rasa/dialogue_understanding/commands/knowledge_answer_command.py +21 -2
- rasa/dialogue_understanding/commands/restart_agent_command.py +162 -0
- rasa/dialogue_understanding/commands/start_flow_command.py +68 -7
- rasa/dialogue_understanding/commands/utils.py +124 -2
- rasa/dialogue_understanding/generator/command_parser.py +4 -0
- rasa/dialogue_understanding/generator/llm_based_command_generator.py +50 -12
- rasa/dialogue_understanding/generator/llm_command_generator.py +1 -1
- rasa/dialogue_understanding/generator/multi_step/multi_step_llm_command_generator.py +1 -1
- rasa/dialogue_understanding/generator/prompt_templates/agent_command_prompt_v2_claude_3_5_sonnet_20240620_template.jinja2 +66 -0
- rasa/dialogue_understanding/generator/prompt_templates/agent_command_prompt_v2_gpt_4o_2024_11_20_template.jinja2 +66 -0
- rasa/dialogue_understanding/generator/prompt_templates/agent_command_prompt_v3_claude_3_5_sonnet_20240620_template.jinja2 +89 -0
- rasa/dialogue_understanding/generator/prompt_templates/agent_command_prompt_v3_gpt_4o_2024_11_20_template.jinja2 +88 -0
- rasa/dialogue_understanding/generator/single_step/compact_llm_command_generator.py +42 -7
- rasa/dialogue_understanding/generator/single_step/search_ready_llm_command_generator.py +40 -3
- rasa/dialogue_understanding/generator/single_step/single_step_based_llm_command_generator.py +20 -3
- rasa/dialogue_understanding/patterns/cancel.py +27 -6
- rasa/dialogue_understanding/patterns/clarify.py +3 -14
- rasa/dialogue_understanding/patterns/continue_interrupted.py +239 -6
- rasa/dialogue_understanding/patterns/default_flows_for_patterns.yml +46 -8
- rasa/dialogue_understanding/processor/command_processor.py +136 -15
- rasa/dialogue_understanding/stack/dialogue_stack.py +98 -2
- rasa/dialogue_understanding/stack/frames/flow_stack_frame.py +57 -0
- rasa/dialogue_understanding/stack/utils.py +57 -3
- rasa/dialogue_understanding/utils.py +24 -4
- rasa/dialogue_understanding_test/du_test_runner.py +8 -3
- rasa/e2e_test/e2e_test_runner.py +13 -3
- rasa/engine/caching.py +2 -2
- rasa/engine/constants.py +1 -1
- rasa/engine/recipes/default_components.py +138 -49
- rasa/engine/recipes/default_recipe.py +108 -11
- rasa/engine/runner/dask.py +8 -5
- rasa/engine/validation.py +19 -6
- rasa/graph_components/validators/default_recipe_validator.py +86 -28
- rasa/hooks.py +5 -5
- rasa/llm_fine_tuning/utils.py +2 -2
- rasa/model_training.py +60 -47
- rasa/nlu/classifiers/diet_classifier.py +198 -98
- rasa/nlu/classifiers/logistic_regression_classifier.py +1 -4
- rasa/nlu/classifiers/mitie_intent_classifier.py +3 -0
- rasa/nlu/classifiers/sklearn_intent_classifier.py +1 -3
- rasa/nlu/extractors/crf_entity_extractor.py +9 -10
- rasa/nlu/extractors/mitie_entity_extractor.py +3 -0
- rasa/nlu/extractors/spacy_entity_extractor.py +3 -0
- rasa/nlu/featurizers/dense_featurizer/convert_featurizer.py +4 -0
- rasa/nlu/featurizers/dense_featurizer/lm_featurizer.py +5 -0
- rasa/nlu/featurizers/dense_featurizer/mitie_featurizer.py +2 -0
- rasa/nlu/featurizers/dense_featurizer/spacy_featurizer.py +3 -0
- rasa/nlu/featurizers/sparse_featurizer/count_vectors_featurizer.py +4 -2
- rasa/nlu/featurizers/sparse_featurizer/lexical_syntactic_featurizer.py +4 -0
- rasa/nlu/selectors/response_selector.py +10 -2
- rasa/nlu/tokenizers/jieba_tokenizer.py +3 -4
- rasa/nlu/tokenizers/mitie_tokenizer.py +3 -2
- rasa/nlu/tokenizers/spacy_tokenizer.py +3 -2
- rasa/nlu/utils/mitie_utils.py +3 -0
- rasa/nlu/utils/spacy_utils.py +3 -2
- rasa/plugin.py +8 -8
- rasa/privacy/privacy_manager.py +12 -3
- rasa/server.py +15 -3
- rasa/shared/agents/__init__.py +0 -0
- rasa/shared/agents/auth/__init__.py +0 -0
- rasa/shared/agents/auth/agent_auth_factory.py +105 -0
- rasa/shared/agents/auth/agent_auth_manager.py +92 -0
- rasa/shared/agents/auth/auth_strategy/__init__.py +19 -0
- rasa/shared/agents/auth/auth_strategy/agent_auth_strategy.py +52 -0
- rasa/shared/agents/auth/auth_strategy/api_key_auth_strategy.py +42 -0
- rasa/shared/agents/auth/auth_strategy/bearer_token_auth_strategy.py +28 -0
- rasa/shared/agents/auth/auth_strategy/oauth2_auth_strategy.py +167 -0
- rasa/shared/agents/auth/constants.py +12 -0
- rasa/shared/agents/auth/types.py +12 -0
- rasa/shared/agents/utils.py +35 -0
- rasa/shared/constants.py +8 -0
- rasa/shared/core/constants.py +16 -1
- rasa/shared/core/domain.py +0 -7
- rasa/shared/core/events.py +327 -0
- rasa/shared/core/flows/constants.py +5 -0
- rasa/shared/core/flows/flows_list.py +21 -5
- rasa/shared/core/flows/flows_yaml_schema.json +119 -184
- rasa/shared/core/flows/steps/call.py +49 -5
- rasa/shared/core/flows/steps/collect.py +98 -13
- rasa/shared/core/flows/validation.py +372 -8
- rasa/shared/core/flows/yaml_flows_io.py +3 -2
- rasa/shared/core/slots.py +2 -2
- rasa/shared/core/trackers.py +5 -2
- rasa/shared/exceptions.py +16 -0
- rasa/shared/importers/rasa.py +1 -1
- rasa/shared/importers/utils.py +9 -3
- rasa/shared/providers/llm/_base_litellm_client.py +41 -9
- rasa/shared/providers/llm/litellm_router_llm_client.py +8 -4
- rasa/shared/providers/llm/llm_client.py +7 -3
- rasa/shared/providers/llm/llm_response.py +66 -0
- rasa/shared/providers/llm/self_hosted_llm_client.py +8 -4
- rasa/shared/utils/common.py +24 -0
- rasa/shared/utils/health_check/health_check.py +7 -3
- rasa/shared/utils/llm.py +39 -16
- rasa/shared/utils/mcp/__init__.py +0 -0
- rasa/shared/utils/mcp/server_connection.py +247 -0
- rasa/shared/utils/mcp/utils.py +20 -0
- rasa/shared/utils/schemas/events.py +42 -0
- rasa/shared/utils/yaml.py +3 -1
- rasa/studio/pull/pull.py +3 -2
- rasa/studio/train.py +8 -7
- rasa/studio/upload.py +3 -6
- rasa/telemetry.py +69 -5
- rasa/tracing/config.py +45 -12
- rasa/tracing/constants.py +14 -0
- rasa/tracing/instrumentation/attribute_extractors.py +142 -9
- rasa/tracing/instrumentation/instrumentation.py +626 -21
- rasa/tracing/instrumentation/intentless_policy_instrumentation.py +4 -4
- rasa/tracing/instrumentation/metrics.py +32 -0
- rasa/tracing/metric_instrument_provider.py +68 -0
- rasa/utils/common.py +92 -1
- rasa/utils/endpoints.py +11 -2
- rasa/utils/log_utils.py +96 -5
- rasa/utils/ml_utils.py +1 -1
- rasa/utils/tensorflow/__init__.py +7 -0
- rasa/utils/tensorflow/callback.py +136 -101
- rasa/utils/tensorflow/crf.py +1 -1
- rasa/utils/tensorflow/data_generator.py +21 -8
- rasa/utils/tensorflow/layers.py +21 -11
- rasa/utils/tensorflow/metrics.py +7 -3
- rasa/utils/tensorflow/models.py +56 -8
- rasa/utils/tensorflow/rasa_layers.py +8 -6
- rasa/utils/tensorflow/transformer.py +2 -3
- rasa/utils/train_utils.py +54 -24
- rasa/validator.py +5 -5
- rasa/version.py +1 -1
- {rasa_pro-3.14.0.dev20250922.dist-info → rasa_pro-3.14.0rc2.dist-info}/METADATA +47 -41
- {rasa_pro-3.14.0.dev20250922.dist-info → rasa_pro-3.14.0rc2.dist-info}/RECORD +299 -238
- rasa/builder/scrape_rasa_docs.py +0 -97
- rasa/core/channels/inspector/dist/assets/channel-8e08bed9.js +0 -1
- rasa/core/channels/inspector/dist/assets/clone-78c82dea.js +0 -1
- rasa/core/channels/inspector/dist/assets/flowDiagram-v2-96b9c2cf-2b08f601.js +0 -1
- rasa/core/channels/inspector/dist/assets/index-c941dcb3.js +0 -1336
- {rasa_pro-3.14.0.dev20250922.dist-info → rasa_pro-3.14.0rc2.dist-info}/NOTICE +0 -0
- {rasa_pro-3.14.0.dev20250922.dist-info → rasa_pro-3.14.0rc2.dist-info}/WHEEL +0 -0
- {rasa_pro-3.14.0.dev20250922.dist-info → rasa_pro-3.14.0rc2.dist-info}/entry_points.txt +0 -0
rasa/cli/utils.py
CHANGED
|
@@ -1,415 +1,23 @@
|
|
|
1
|
-
import argparse
|
|
2
1
|
import importlib
|
|
3
2
|
import json
|
|
4
3
|
import os
|
|
5
4
|
import sys
|
|
6
|
-
import time
|
|
7
5
|
from pathlib import Path
|
|
8
6
|
from types import FrameType
|
|
9
|
-
from typing import TYPE_CHECKING, Any, Dict, List,
|
|
7
|
+
from typing import TYPE_CHECKING, Any, Dict, List, Text
|
|
10
8
|
|
|
11
|
-
import randomname
|
|
12
9
|
import structlog
|
|
13
10
|
|
|
14
|
-
import rasa.shared.utils.cli
|
|
15
11
|
import rasa.shared.utils.io
|
|
16
|
-
from rasa import telemetry
|
|
17
|
-
from rasa.exceptions import ModelNotFound, ValidationError
|
|
18
|
-
from rasa.shared.constants import (
|
|
19
|
-
ASSISTANT_ID_DEFAULT_VALUE,
|
|
20
|
-
ASSISTANT_ID_KEY,
|
|
21
|
-
DEFAULT_CONFIG_PATH,
|
|
22
|
-
)
|
|
23
|
-
from rasa.shared.importers.importer import TrainingDataImporter
|
|
24
|
-
from rasa.shared.utils.common import display_research_study_prompt
|
|
25
|
-
from rasa.shared.utils.yaml import read_config_file
|
|
26
|
-
from rasa.utils.io import write_yaml
|
|
27
12
|
|
|
28
13
|
if TYPE_CHECKING:
|
|
29
14
|
from questionary import Question
|
|
30
|
-
from typing_extensions import Literal
|
|
31
|
-
|
|
32
|
-
from rasa.validator import Validator
|
|
33
15
|
|
|
34
16
|
structlogger = structlog.get_logger()
|
|
35
17
|
|
|
36
18
|
FREE_TEXT_INPUT_PROMPT = "Type out your own message..."
|
|
37
19
|
|
|
38
20
|
|
|
39
|
-
@overload
|
|
40
|
-
def get_validated_path(
|
|
41
|
-
current: Optional[Union[Path, Text]],
|
|
42
|
-
parameter: Text,
|
|
43
|
-
default: Optional[Union[Path, Text, List[Text]]] = ...,
|
|
44
|
-
none_is_valid: "Literal[False]" = ...,
|
|
45
|
-
) -> Union[Path, Text]: ...
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
@overload
|
|
49
|
-
def get_validated_path(
|
|
50
|
-
current: Optional[Union[Path, Text]],
|
|
51
|
-
parameter: Text,
|
|
52
|
-
default: Optional[Union[Path, Text, List[Text]]] = ...,
|
|
53
|
-
none_is_valid: "Literal[True]" = ...,
|
|
54
|
-
) -> Optional[Union[Path, Text]]: ...
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
def get_validated_path(
|
|
58
|
-
current: Optional[Union[Path, Text]],
|
|
59
|
-
parameter: Text,
|
|
60
|
-
default: Optional[Union[Path, Text, List[Text]]] = None,
|
|
61
|
-
none_is_valid: bool = False,
|
|
62
|
-
) -> Optional[Union[Path, Text]]:
|
|
63
|
-
"""Checks whether a file path or its default value is valid and returns it.
|
|
64
|
-
|
|
65
|
-
Args:
|
|
66
|
-
current: The parsed value.
|
|
67
|
-
parameter: The name of the parameter.
|
|
68
|
-
default: one or multiple default values of the parameter.
|
|
69
|
-
none_is_valid: `True` if `None` is valid value for the path,
|
|
70
|
-
else `False``
|
|
71
|
-
|
|
72
|
-
Returns:
|
|
73
|
-
The current value if valid,
|
|
74
|
-
otherwise one of the default values of the argument if valid,
|
|
75
|
-
otherwise `None` if allowed,
|
|
76
|
-
otherwise raises an error and exits.
|
|
77
|
-
"""
|
|
78
|
-
if current and os.path.exists(current):
|
|
79
|
-
return current
|
|
80
|
-
|
|
81
|
-
if parameter == "model":
|
|
82
|
-
raise ModelNotFound(
|
|
83
|
-
f"The provided model path '{current}' could not be found. "
|
|
84
|
-
"Provide an existing model path."
|
|
85
|
-
)
|
|
86
|
-
|
|
87
|
-
# try to find a valid option among the defaults
|
|
88
|
-
if isinstance(default, str) or isinstance(default, Path):
|
|
89
|
-
default_options = [str(default)]
|
|
90
|
-
elif isinstance(default, list):
|
|
91
|
-
default_options = default
|
|
92
|
-
else:
|
|
93
|
-
default_options = []
|
|
94
|
-
|
|
95
|
-
valid_options = (option for option in default_options if os.path.exists(option))
|
|
96
|
-
chosen_option = next(valid_options, None)
|
|
97
|
-
|
|
98
|
-
# warn and log if user-chosen parameter wasn't found and thus overwritten
|
|
99
|
-
if chosen_option:
|
|
100
|
-
shared_info = f"Using default location '{chosen_option}' instead."
|
|
101
|
-
if current is None:
|
|
102
|
-
structlogger.debug(
|
|
103
|
-
"cli.get_validated_path.parameter_not_set",
|
|
104
|
-
parameter=parameter,
|
|
105
|
-
event_info=(f"Parameter '{parameter}' was not set. {shared_info}"),
|
|
106
|
-
)
|
|
107
|
-
elif current not in default_options:
|
|
108
|
-
structlogger.warn(
|
|
109
|
-
"cli.get_validated_path.path_does_not_exists",
|
|
110
|
-
path=current,
|
|
111
|
-
event_info=(
|
|
112
|
-
f"The path '{current}' does not seem to exist. {shared_info}"
|
|
113
|
-
),
|
|
114
|
-
)
|
|
115
|
-
|
|
116
|
-
if chosen_option is None and not none_is_valid:
|
|
117
|
-
cancel_cause_not_found(current, parameter, default)
|
|
118
|
-
|
|
119
|
-
return chosen_option
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
def missing_config_keys(
|
|
123
|
-
path: Union["Path", Text], mandatory_keys: List[Text]
|
|
124
|
-
) -> List[Text]:
|
|
125
|
-
"""Checks whether the config file at `path` contains the `mandatory_keys`.
|
|
126
|
-
|
|
127
|
-
Args:
|
|
128
|
-
path: The path to the config file.
|
|
129
|
-
mandatory_keys: A list of mandatory config keys.
|
|
130
|
-
|
|
131
|
-
Returns:
|
|
132
|
-
The list of missing config keys.
|
|
133
|
-
"""
|
|
134
|
-
if not os.path.exists(path):
|
|
135
|
-
return mandatory_keys
|
|
136
|
-
|
|
137
|
-
config_data = read_config_file(path)
|
|
138
|
-
|
|
139
|
-
return [k for k in mandatory_keys if k not in config_data or config_data[k] is None]
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
def validate_assistant_id_in_config(config_file: Union["Path", Text]) -> None:
|
|
143
|
-
"""Verifies that the assistant_id key exists and has a unique value in config.
|
|
144
|
-
|
|
145
|
-
Issues a warning if the key does not exist or has the default value and replaces it
|
|
146
|
-
with a pseudo-random string value.
|
|
147
|
-
"""
|
|
148
|
-
config_data = read_config_file(config_file, reader_type=["safe", "rt"])
|
|
149
|
-
assistant_id = config_data.get(ASSISTANT_ID_KEY)
|
|
150
|
-
|
|
151
|
-
if assistant_id is None or assistant_id == ASSISTANT_ID_DEFAULT_VALUE:
|
|
152
|
-
structlogger.warn(
|
|
153
|
-
"cli.validate_assistant_id_in_config.missing_unique_assistant_id_key",
|
|
154
|
-
config=config_file,
|
|
155
|
-
missing_key=ASSISTANT_ID_KEY,
|
|
156
|
-
event_info=(
|
|
157
|
-
f"The config file '{config_file!s}' is "
|
|
158
|
-
f"missing a unique value for the "
|
|
159
|
-
f"'{ASSISTANT_ID_KEY}' mandatory key. "
|
|
160
|
-
f"Proceeding with generating a random "
|
|
161
|
-
f"value and overwriting the '{ASSISTANT_ID_KEY}'"
|
|
162
|
-
f" in the config file."
|
|
163
|
-
),
|
|
164
|
-
)
|
|
165
|
-
|
|
166
|
-
# add random value for assistant id, overwrite config file
|
|
167
|
-
time_format = "%Y%m%d-%H%M%S"
|
|
168
|
-
config_data[ASSISTANT_ID_KEY] = (
|
|
169
|
-
f"{time.strftime(time_format)}-{randomname.get_name()}"
|
|
170
|
-
)
|
|
171
|
-
|
|
172
|
-
write_yaml(data=config_data, target=config_file, should_preserve_key_order=True)
|
|
173
|
-
|
|
174
|
-
return
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
def validate_config_path(
|
|
178
|
-
config: Optional[Union[Text, "Path"]],
|
|
179
|
-
default_config: Text = DEFAULT_CONFIG_PATH,
|
|
180
|
-
) -> Text:
|
|
181
|
-
"""Verifies that the config path exists.
|
|
182
|
-
|
|
183
|
-
Exit if the config file does not exist.
|
|
184
|
-
|
|
185
|
-
Args:
|
|
186
|
-
config: Path to the config file.
|
|
187
|
-
default_config: default config to use if the file at `config` doesn't exist.
|
|
188
|
-
|
|
189
|
-
Returns: The path to the config file.
|
|
190
|
-
"""
|
|
191
|
-
config = rasa.cli.utils.get_validated_path(config, "config", default_config)
|
|
192
|
-
|
|
193
|
-
if not config or not os.path.exists(config):
|
|
194
|
-
display_research_study_prompt()
|
|
195
|
-
raise ValidationError(
|
|
196
|
-
code="cli.validate_config_path.does_not_exists",
|
|
197
|
-
config=config,
|
|
198
|
-
event_info=(
|
|
199
|
-
f"The config file '{config}' does not exist. "
|
|
200
|
-
f"Use '--config' to specify a valid config file."
|
|
201
|
-
),
|
|
202
|
-
)
|
|
203
|
-
|
|
204
|
-
return str(config)
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
def validate_mandatory_config_keys(
|
|
208
|
-
config: Union[Text, "Path"],
|
|
209
|
-
mandatory_keys: List[Text],
|
|
210
|
-
) -> Text:
|
|
211
|
-
"""Get a config from a config file and check if it is valid.
|
|
212
|
-
|
|
213
|
-
Exit if the config isn't valid.
|
|
214
|
-
|
|
215
|
-
Args:
|
|
216
|
-
config: Path to the config file.
|
|
217
|
-
mandatory_keys: The keys that have to be specified in the config file.
|
|
218
|
-
|
|
219
|
-
Returns: The path to the config file if the config is valid.
|
|
220
|
-
"""
|
|
221
|
-
missing_keys = set(rasa.cli.utils.missing_config_keys(config, mandatory_keys))
|
|
222
|
-
if missing_keys:
|
|
223
|
-
display_research_study_prompt()
|
|
224
|
-
raise ValidationError(
|
|
225
|
-
code="cli.validate_mandatory_config_keys.missing_keys",
|
|
226
|
-
config=config,
|
|
227
|
-
missing_keys=missing_keys,
|
|
228
|
-
event_info=(
|
|
229
|
-
"The config file '{}' is missing mandatory parameters: "
|
|
230
|
-
"'{}'. Add missing parameters to config file and try again.".format(
|
|
231
|
-
config, "', '".join(missing_keys)
|
|
232
|
-
)
|
|
233
|
-
),
|
|
234
|
-
)
|
|
235
|
-
|
|
236
|
-
return str(config)
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
def get_validated_config(
|
|
240
|
-
config: Optional[Union[Text, "Path"]],
|
|
241
|
-
mandatory_keys: List[Text],
|
|
242
|
-
default_config: Text = DEFAULT_CONFIG_PATH,
|
|
243
|
-
) -> Text:
|
|
244
|
-
"""Validates config and returns path to validated config file."""
|
|
245
|
-
config = validate_config_path(config, default_config)
|
|
246
|
-
validate_assistant_id_in_config(config)
|
|
247
|
-
|
|
248
|
-
config = validate_mandatory_config_keys(config, mandatory_keys)
|
|
249
|
-
|
|
250
|
-
return config
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
def validate_files(
|
|
254
|
-
fail_on_warnings: bool,
|
|
255
|
-
max_history: Optional[int],
|
|
256
|
-
importer: TrainingDataImporter,
|
|
257
|
-
stories_only: bool = False,
|
|
258
|
-
flows_only: bool = False,
|
|
259
|
-
translations_only: bool = False,
|
|
260
|
-
) -> None:
|
|
261
|
-
"""Validates either the story structure or the entire project.
|
|
262
|
-
|
|
263
|
-
Args:
|
|
264
|
-
fail_on_warnings: `True` if the process should exit with a non-zero status
|
|
265
|
-
max_history: The max history to use when validating the story structure.
|
|
266
|
-
importer: The `TrainingDataImporter` to use to load the training data.
|
|
267
|
-
stories_only: If `True`, only the story structure is validated.
|
|
268
|
-
flows_only: If `True`, only the flows are validated.
|
|
269
|
-
translations_only: If `True`, only the translations data is validated.
|
|
270
|
-
"""
|
|
271
|
-
from rasa.validator import Validator
|
|
272
|
-
|
|
273
|
-
validator = Validator.from_importer(importer)
|
|
274
|
-
|
|
275
|
-
if stories_only:
|
|
276
|
-
all_good = _validate_story_structure(validator, max_history, fail_on_warnings)
|
|
277
|
-
elif flows_only:
|
|
278
|
-
all_good = validator.verify_flows()
|
|
279
|
-
elif translations_only:
|
|
280
|
-
all_good = validator.verify_translations()
|
|
281
|
-
else:
|
|
282
|
-
if importer.get_domain().is_empty():
|
|
283
|
-
structlogger.error(
|
|
284
|
-
"cli.validate_files.empty_domain",
|
|
285
|
-
event_info="Encountered empty domain during validation.",
|
|
286
|
-
)
|
|
287
|
-
display_research_study_prompt()
|
|
288
|
-
raise ValidationError(
|
|
289
|
-
code="cli.validate_files.empty_domain",
|
|
290
|
-
event_info="Encountered empty domain during validation.",
|
|
291
|
-
)
|
|
292
|
-
|
|
293
|
-
valid_domain = _validate_domain(validator)
|
|
294
|
-
valid_nlu = _validate_nlu(validator, fail_on_warnings)
|
|
295
|
-
valid_stories = _validate_story_structure(
|
|
296
|
-
validator, max_history, fail_on_warnings
|
|
297
|
-
)
|
|
298
|
-
valid_flows = validator.verify_flows()
|
|
299
|
-
if validator.config:
|
|
300
|
-
valid_translations = validator.verify_translations(summary_mode=True)
|
|
301
|
-
else:
|
|
302
|
-
valid_translations = True
|
|
303
|
-
valid_CALM_slot_mappings = validator.validate_CALM_slot_mappings()
|
|
304
|
-
|
|
305
|
-
all_good = (
|
|
306
|
-
valid_domain
|
|
307
|
-
and valid_nlu
|
|
308
|
-
and valid_stories
|
|
309
|
-
and valid_flows
|
|
310
|
-
and valid_translations
|
|
311
|
-
and valid_CALM_slot_mappings
|
|
312
|
-
)
|
|
313
|
-
|
|
314
|
-
if validator.config:
|
|
315
|
-
validator.warn_if_config_mandatory_keys_are_not_set()
|
|
316
|
-
|
|
317
|
-
telemetry.track_validate_files(all_good)
|
|
318
|
-
if not all_good:
|
|
319
|
-
structlogger.error(
|
|
320
|
-
"cli.validate_files.project_validation_error",
|
|
321
|
-
event_info="Project validation completed with errors.",
|
|
322
|
-
)
|
|
323
|
-
display_research_study_prompt()
|
|
324
|
-
raise ValidationError(
|
|
325
|
-
code="cli.validate_files.project_validation_error",
|
|
326
|
-
event_info="Project validation completed with errors.",
|
|
327
|
-
)
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
def _validate_domain(validator: "Validator") -> bool:
|
|
331
|
-
valid_domain_validity = validator.verify_domain_validity()
|
|
332
|
-
valid_actions_in_stories_rules = validator.verify_actions_in_stories_rules()
|
|
333
|
-
valid_forms_in_stories_rules = validator.verify_forms_in_stories_rules()
|
|
334
|
-
valid_form_slots = validator.verify_form_slots()
|
|
335
|
-
valid_slot_mappings = validator.verify_slot_mappings()
|
|
336
|
-
valid_responses = validator.check_for_no_empty_parenthesis_in_responses()
|
|
337
|
-
valid_buttons = validator.validate_button_payloads()
|
|
338
|
-
valid_slot_validation = validator.verify_slot_validation()
|
|
339
|
-
valid_conditional_responses = (
|
|
340
|
-
validator.validate_conditional_response_variation_predicates()
|
|
341
|
-
)
|
|
342
|
-
return (
|
|
343
|
-
valid_domain_validity
|
|
344
|
-
and valid_actions_in_stories_rules
|
|
345
|
-
and valid_forms_in_stories_rules
|
|
346
|
-
and valid_form_slots
|
|
347
|
-
and valid_slot_mappings
|
|
348
|
-
and valid_responses
|
|
349
|
-
and valid_buttons
|
|
350
|
-
and valid_slot_validation
|
|
351
|
-
and valid_conditional_responses
|
|
352
|
-
)
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
def _validate_nlu(validator: "Validator", fail_on_warnings: bool) -> bool:
|
|
356
|
-
return validator.verify_nlu(not fail_on_warnings)
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
def _validate_story_structure(
|
|
360
|
-
validator: "Validator", max_history: Optional[int], fail_on_warnings: bool
|
|
361
|
-
) -> bool:
|
|
362
|
-
# Check if a valid setting for `max_history` was given
|
|
363
|
-
if isinstance(max_history, int) and max_history < 1:
|
|
364
|
-
raise argparse.ArgumentTypeError(
|
|
365
|
-
f"The value of `--max-history {max_history}` is not a positive integer."
|
|
366
|
-
)
|
|
367
|
-
|
|
368
|
-
return validator.verify_story_structure(
|
|
369
|
-
not fail_on_warnings, max_history=max_history
|
|
370
|
-
)
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
def cancel_cause_not_found(
|
|
374
|
-
current: Optional[Union["Path", Text]],
|
|
375
|
-
parameter: Text,
|
|
376
|
-
default: Optional[Union["Path", Text, List[Text]]],
|
|
377
|
-
) -> None:
|
|
378
|
-
"""Exits with an error because the given path was not valid.
|
|
379
|
-
|
|
380
|
-
Args:
|
|
381
|
-
current: The path given by the user.
|
|
382
|
-
parameter: The name of the parameter.
|
|
383
|
-
default: The default value of the parameter.
|
|
384
|
-
|
|
385
|
-
"""
|
|
386
|
-
default_clause = ""
|
|
387
|
-
if default and isinstance(default, str):
|
|
388
|
-
default_clause = f"use the default location ('{default}') or"
|
|
389
|
-
elif default and isinstance(default, list):
|
|
390
|
-
default_clause = f"use one of the default locations ({', '.join(default)}) or"
|
|
391
|
-
|
|
392
|
-
structlogger.error(
|
|
393
|
-
"cli.path_does_not_exist",
|
|
394
|
-
path=current,
|
|
395
|
-
event_info=(
|
|
396
|
-
f"The path '{current}' does not exist. "
|
|
397
|
-
f"Please make sure to {default_clause} specify it "
|
|
398
|
-
f"with '--{parameter}'."
|
|
399
|
-
),
|
|
400
|
-
)
|
|
401
|
-
display_research_study_prompt()
|
|
402
|
-
raise ValidationError(
|
|
403
|
-
code="cli.path_does_not_exist",
|
|
404
|
-
path=current,
|
|
405
|
-
event_info=(
|
|
406
|
-
f"The path '{current}' does not exist. "
|
|
407
|
-
f"Please make sure to {default_clause} specify it "
|
|
408
|
-
f"with '--{parameter}'."
|
|
409
|
-
),
|
|
410
|
-
)
|
|
411
|
-
|
|
412
|
-
|
|
413
21
|
def parse_last_positional_argument_as_model_path() -> None:
|
|
414
22
|
"""Fixes the parsing of a potential positional model path argument."""
|
|
415
23
|
if (
|
|
File without changes
|
|
@@ -0,0 +1,223 @@
|
|
|
1
|
+
import argparse
|
|
2
|
+
import os
|
|
3
|
+
from typing import TYPE_CHECKING, Optional
|
|
4
|
+
|
|
5
|
+
import structlog
|
|
6
|
+
|
|
7
|
+
from rasa import telemetry
|
|
8
|
+
from rasa.exceptions import ValidationError
|
|
9
|
+
from rasa.shared.importers.importer import TrainingDataImporter
|
|
10
|
+
from rasa.shared.utils.common import display_research_study_prompt
|
|
11
|
+
|
|
12
|
+
if TYPE_CHECKING:
|
|
13
|
+
from rasa.validator import Validator
|
|
14
|
+
|
|
15
|
+
structlogger = structlog.get_logger()
|
|
16
|
+
|
|
17
|
+
FREE_TEXT_INPUT_PROMPT = "Type out your own message..."
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
def _validate_domain(validator: "Validator") -> bool:
|
|
21
|
+
valid_domain_validity = validator.verify_domain_validity()
|
|
22
|
+
valid_actions_in_stories_rules = validator.verify_actions_in_stories_rules()
|
|
23
|
+
valid_forms_in_stories_rules = validator.verify_forms_in_stories_rules()
|
|
24
|
+
valid_form_slots = validator.verify_form_slots()
|
|
25
|
+
valid_slot_mappings = validator.verify_slot_mappings()
|
|
26
|
+
valid_responses = validator.check_for_no_empty_parenthesis_in_responses()
|
|
27
|
+
valid_buttons = validator.validate_button_payloads()
|
|
28
|
+
valid_slot_validation = validator.verify_slot_validation()
|
|
29
|
+
valid_conditional_responses = (
|
|
30
|
+
validator.validate_conditional_response_variation_predicates()
|
|
31
|
+
)
|
|
32
|
+
return (
|
|
33
|
+
valid_domain_validity
|
|
34
|
+
and valid_actions_in_stories_rules
|
|
35
|
+
and valid_forms_in_stories_rules
|
|
36
|
+
and valid_form_slots
|
|
37
|
+
and valid_slot_mappings
|
|
38
|
+
and valid_responses
|
|
39
|
+
and valid_buttons
|
|
40
|
+
and valid_slot_validation
|
|
41
|
+
and valid_conditional_responses
|
|
42
|
+
)
|
|
43
|
+
|
|
44
|
+
|
|
45
|
+
def _validate_nlu(validator: "Validator", fail_on_warnings: bool) -> bool:
|
|
46
|
+
return validator.verify_nlu(not fail_on_warnings)
|
|
47
|
+
|
|
48
|
+
|
|
49
|
+
def _validate_story_structure(
|
|
50
|
+
validator: "Validator", max_history: Optional[int], fail_on_warnings: bool
|
|
51
|
+
) -> bool:
|
|
52
|
+
# Check if a valid setting for `max_history` was given
|
|
53
|
+
if isinstance(max_history, int) and max_history < 1:
|
|
54
|
+
raise argparse.ArgumentTypeError(
|
|
55
|
+
f"The value of `--max-history {max_history}` is not a positive integer."
|
|
56
|
+
)
|
|
57
|
+
|
|
58
|
+
return validator.verify_story_structure(
|
|
59
|
+
not fail_on_warnings, max_history=max_history
|
|
60
|
+
)
|
|
61
|
+
|
|
62
|
+
|
|
63
|
+
def _validate_sub_agents(sub_agents_path: str) -> bool:
|
|
64
|
+
"""Validates sub-agents configuration.
|
|
65
|
+
|
|
66
|
+
Args:
|
|
67
|
+
sub_agents_path: Path to the sub-agents directory.
|
|
68
|
+
|
|
69
|
+
Returns:
|
|
70
|
+
True if validation passes, False otherwise.
|
|
71
|
+
"""
|
|
72
|
+
from rasa.agents.validation import validate_agent_folder
|
|
73
|
+
from rasa.core.constants import DEFAULT_SUB_AGENTS
|
|
74
|
+
|
|
75
|
+
try:
|
|
76
|
+
# Check if the sub-agents directory exists
|
|
77
|
+
if not os.path.isdir(sub_agents_path):
|
|
78
|
+
# If the sub-agents-path points to the default folder and it doesn't exist,
|
|
79
|
+
# no agents are available.
|
|
80
|
+
# AvailableAgents will handle the non-existing folder gracefully.
|
|
81
|
+
|
|
82
|
+
if sub_agents_path == DEFAULT_SUB_AGENTS:
|
|
83
|
+
structlogger.info(
|
|
84
|
+
"cli.validate_files.sub_agents_validation",
|
|
85
|
+
sub_agents_path=sub_agents_path,
|
|
86
|
+
event_info="Default sub-agents directory does not exist, "
|
|
87
|
+
"no sub-agents will be available.",
|
|
88
|
+
)
|
|
89
|
+
return True
|
|
90
|
+
else:
|
|
91
|
+
# For user-specified paths, the directory must exist
|
|
92
|
+
structlogger.error(
|
|
93
|
+
"cli.validate_files.sub_agents_validation_error",
|
|
94
|
+
sub_agents_path=sub_agents_path,
|
|
95
|
+
event_info=f"Sub-agents directory '{sub_agents_path}' "
|
|
96
|
+
"does not exist.",
|
|
97
|
+
)
|
|
98
|
+
return False
|
|
99
|
+
|
|
100
|
+
# Validate the actual config content using AvailableAgents
|
|
101
|
+
# This will validate file existence, structure, mandatory keys, etc.
|
|
102
|
+
try:
|
|
103
|
+
validate_agent_folder(sub_agents_path)
|
|
104
|
+
|
|
105
|
+
except ValidationError as e:
|
|
106
|
+
# This is a validation error - log it and return False
|
|
107
|
+
structlogger.error(
|
|
108
|
+
"cli.validate_files.sub_agents_validation_error",
|
|
109
|
+
sub_agents_path=sub_agents_path,
|
|
110
|
+
validation_error=str(e),
|
|
111
|
+
event_info=f"Sub-agents configuration validation failed: {e}",
|
|
112
|
+
)
|
|
113
|
+
return False
|
|
114
|
+
|
|
115
|
+
except Exception as e:
|
|
116
|
+
# This is an unexpected error
|
|
117
|
+
structlogger.error(
|
|
118
|
+
"cli.validate_files.sub_agents_validation_error",
|
|
119
|
+
sub_agents_path=sub_agents_path,
|
|
120
|
+
error=str(e),
|
|
121
|
+
event_info=f"Unexpected error during sub-agents validation: {e}",
|
|
122
|
+
)
|
|
123
|
+
return False
|
|
124
|
+
|
|
125
|
+
structlogger.info(
|
|
126
|
+
"cli.validate_files.sub_agents_validation_success",
|
|
127
|
+
sub_agents_path=sub_agents_path,
|
|
128
|
+
event_info="Sub-agents validation passed successfully.",
|
|
129
|
+
)
|
|
130
|
+
return True
|
|
131
|
+
|
|
132
|
+
except Exception as e:
|
|
133
|
+
# This is an unexpected error
|
|
134
|
+
structlogger.error(
|
|
135
|
+
"cli.validate_files.sub_agents_validation_error",
|
|
136
|
+
sub_agents_path=sub_agents_path,
|
|
137
|
+
error=str(e),
|
|
138
|
+
event_info="Sub-agents validation failed.",
|
|
139
|
+
)
|
|
140
|
+
return False
|
|
141
|
+
|
|
142
|
+
|
|
143
|
+
def validate_files(
|
|
144
|
+
fail_on_warnings: bool,
|
|
145
|
+
max_history: Optional[int],
|
|
146
|
+
importer: TrainingDataImporter,
|
|
147
|
+
stories_only: bool = False,
|
|
148
|
+
flows_only: bool = False,
|
|
149
|
+
translations_only: bool = False,
|
|
150
|
+
sub_agents: Optional[str] = None,
|
|
151
|
+
) -> None:
|
|
152
|
+
"""Validates either the story structure or the entire project.
|
|
153
|
+
|
|
154
|
+
Args:
|
|
155
|
+
fail_on_warnings: `True` if the process should exit with a non-zero status
|
|
156
|
+
max_history: The max history to use when validating the story structure.
|
|
157
|
+
importer: The `TrainingDataImporter` to use to load the training data.
|
|
158
|
+
stories_only: If `True`, only the story structure is validated.
|
|
159
|
+
flows_only: If `True`, only the flows are validated.
|
|
160
|
+
translations_only: If `True`, only the translations data is validated.
|
|
161
|
+
sub_agents: Path to sub-agents directory for validation.
|
|
162
|
+
"""
|
|
163
|
+
from rasa.validator import Validator
|
|
164
|
+
|
|
165
|
+
validator = Validator.from_importer(importer)
|
|
166
|
+
|
|
167
|
+
if stories_only:
|
|
168
|
+
all_good = _validate_story_structure(validator, max_history, fail_on_warnings)
|
|
169
|
+
elif flows_only:
|
|
170
|
+
all_good = validator.verify_flows()
|
|
171
|
+
elif translations_only:
|
|
172
|
+
all_good = validator.verify_translations()
|
|
173
|
+
else:
|
|
174
|
+
if importer.get_domain().is_empty():
|
|
175
|
+
structlogger.error(
|
|
176
|
+
"cli.validate_files.empty_domain",
|
|
177
|
+
event_info="Encountered empty domain during validation.",
|
|
178
|
+
)
|
|
179
|
+
display_research_study_prompt()
|
|
180
|
+
raise ValidationError(
|
|
181
|
+
code="cli.validate_files.empty_domain",
|
|
182
|
+
event_info="Encountered empty domain during validation.",
|
|
183
|
+
)
|
|
184
|
+
|
|
185
|
+
valid_domain = _validate_domain(validator)
|
|
186
|
+
valid_nlu = _validate_nlu(validator, fail_on_warnings)
|
|
187
|
+
valid_stories = _validate_story_structure(
|
|
188
|
+
validator, max_history, fail_on_warnings
|
|
189
|
+
)
|
|
190
|
+
valid_flows = validator.verify_flows()
|
|
191
|
+
if validator.config:
|
|
192
|
+
valid_translations = validator.verify_translations(summary_mode=True)
|
|
193
|
+
else:
|
|
194
|
+
valid_translations = True
|
|
195
|
+
valid_CALM_slot_mappings = validator.validate_CALM_slot_mappings()
|
|
196
|
+
|
|
197
|
+
# Validate sub-agents if specified
|
|
198
|
+
valid_sub_agents = _validate_sub_agents(sub_agents) if sub_agents else True
|
|
199
|
+
|
|
200
|
+
all_good = (
|
|
201
|
+
valid_domain
|
|
202
|
+
and valid_nlu
|
|
203
|
+
and valid_stories
|
|
204
|
+
and valid_flows
|
|
205
|
+
and valid_translations
|
|
206
|
+
and valid_CALM_slot_mappings
|
|
207
|
+
and valid_sub_agents
|
|
208
|
+
)
|
|
209
|
+
|
|
210
|
+
if validator.config:
|
|
211
|
+
validator.warn_if_config_mandatory_keys_are_not_set()
|
|
212
|
+
|
|
213
|
+
telemetry.track_validate_files(all_good)
|
|
214
|
+
if not all_good:
|
|
215
|
+
structlogger.error(
|
|
216
|
+
"cli.validate_files.project_validation_error",
|
|
217
|
+
event_info="Project validation completed with errors.",
|
|
218
|
+
)
|
|
219
|
+
display_research_study_prompt()
|
|
220
|
+
raise ValidationError(
|
|
221
|
+
code="cli.validate_files.project_validation_error",
|
|
222
|
+
event_info="Project validation completed with errors.",
|
|
223
|
+
)
|