mirror of
https://github.com/langgenius/dify.git
synced 2026-03-26 00:38:03 +08:00
Made-with: Cursor # Conflicts: # api/controllers/console/app/workflow_draft_variable.py # api/core/agent/cot_agent_runner.py # api/core/agent/cot_chat_agent_runner.py # api/core/agent/cot_completion_agent_runner.py # api/core/agent/fc_agent_runner.py # api/core/app/apps/advanced_chat/app_generator.py # api/core/app/apps/advanced_chat/app_runner.py # api/core/app/apps/agent_chat/app_runner.py # api/core/app/apps/workflow/app_generator.py # api/core/app/apps/workflow/app_runner.py # api/core/app/entities/app_invoke_entities.py # api/core/app/entities/queue_entities.py # api/core/llm_generator/output_parser/structured_output.py # api/core/workflow/workflow_entry.py # api/dify_graph/context/__init__.py # api/dify_graph/entities/tool_entities.py # api/dify_graph/file/file_manager.py # api/dify_graph/graph_engine/response_coordinator/coordinator.py # api/dify_graph/graph_events/node.py # api/dify_graph/node_events/node.py # api/dify_graph/nodes/agent/agent_node.py # api/dify_graph/nodes/llm/entities.py # api/dify_graph/nodes/llm/llm_utils.py # api/dify_graph/nodes/llm/node.py # api/dify_graph/nodes/question_classifier/question_classifier_node.py # api/dify_graph/runtime/graph_runtime_state.py # api/dify_graph/variables/segments.py # api/factories/variable_factory.py # api/services/variable_truncator.py # api/tests/unit_tests/utils/structured_output_parser/test_structured_output_parser.py # api/uv.lock # web/app/components/app-sidebar/app-info.tsx # web/app/components/app-sidebar/app-sidebar-dropdown.tsx # web/app/components/app/create-app-modal/index.spec.tsx # web/app/components/apps/__tests__/list.spec.tsx # web/app/components/apps/app-card.tsx # web/app/components/apps/list.tsx # web/app/components/header/account-dropdown/compliance.tsx # web/app/components/header/account-dropdown/index.tsx # web/app/components/header/account-dropdown/support.tsx # web/app/components/workflow-app/components/workflow-onboarding-modal/index.tsx # web/app/components/workflow/panel/debug-and-preview/hooks.ts # web/contract/console/apps.ts # web/contract/router.ts # web/eslint-suppressions.json # web/next.config.ts # web/pnpm-lock.yaml
223 lines
6.2 KiB
Python
223 lines
6.2 KiB
Python
from __future__ import annotations
|
|
|
|
import json
|
|
from abc import ABC
|
|
from builtins import type as type_
|
|
from collections.abc import Sequence
|
|
from enum import StrEnum
|
|
from typing import Any, Union
|
|
|
|
from pydantic import BaseModel, field_validator, model_validator
|
|
|
|
from dify_graph.enums import ErrorStrategy
|
|
|
|
from .exc import DefaultValueTypeError
|
|
|
|
_NumberType = Union[int, float]
|
|
|
|
|
|
class RetryConfig(BaseModel):
|
|
"""node retry config"""
|
|
|
|
max_retries: int = 0 # max retry times
|
|
retry_interval: int = 0 # retry interval in milliseconds
|
|
retry_enabled: bool = False # whether retry is enabled
|
|
|
|
@property
|
|
def retry_interval_seconds(self) -> float:
|
|
return self.retry_interval / 1000
|
|
|
|
|
|
class VariableSelector(BaseModel):
|
|
"""
|
|
Variable Selector.
|
|
"""
|
|
|
|
variable: str
|
|
value_selector: Sequence[str]
|
|
|
|
|
|
class OutputVariableType(StrEnum):
|
|
STRING = "string"
|
|
NUMBER = "number"
|
|
INTEGER = "integer"
|
|
SECRET = "secret"
|
|
BOOLEAN = "boolean"
|
|
OBJECT = "object"
|
|
FILE = "file"
|
|
ARRAY = "array"
|
|
ARRAY_STRING = "array[string]"
|
|
ARRAY_NUMBER = "array[number]"
|
|
ARRAY_OBJECT = "array[object]"
|
|
ARRAY_BOOLEAN = "array[boolean]"
|
|
ARRAY_FILE = "array[file]"
|
|
ANY = "any"
|
|
ARRAY_ANY = "array[any]"
|
|
|
|
|
|
class OutputVariableEntity(BaseModel):
|
|
"""
|
|
Output Variable Entity.
|
|
"""
|
|
|
|
variable: str
|
|
value_type: OutputVariableType = OutputVariableType.ANY
|
|
value_selector: Sequence[str]
|
|
|
|
@field_validator("value_type", mode="before")
|
|
@classmethod
|
|
def normalize_value_type(cls, v: Any) -> Any:
|
|
"""
|
|
Normalize value_type to handle case-insensitive array types.
|
|
Converts 'Array[...]' to 'array[...]' for backward compatibility.
|
|
"""
|
|
if isinstance(v, str) and v.startswith("Array["):
|
|
return v.lower()
|
|
return v
|
|
|
|
|
|
class DefaultValueType(StrEnum):
|
|
STRING = "string"
|
|
NUMBER = "number"
|
|
OBJECT = "object"
|
|
ARRAY_NUMBER = "array[number]"
|
|
ARRAY_STRING = "array[string]"
|
|
ARRAY_OBJECT = "array[object]"
|
|
ARRAY_FILES = "array[file]"
|
|
|
|
|
|
class DefaultValue(BaseModel):
|
|
value: Any = None
|
|
type: DefaultValueType
|
|
key: str
|
|
|
|
@staticmethod
|
|
def _parse_json(value: str):
|
|
"""Unified JSON parsing handler"""
|
|
try:
|
|
return json.loads(value)
|
|
except json.JSONDecodeError:
|
|
raise DefaultValueTypeError(f"Invalid JSON format for value: {value}")
|
|
|
|
@staticmethod
|
|
def _validate_array(value: Any, element_type: type_ | tuple[type_, ...]) -> bool:
|
|
"""Unified array type validation"""
|
|
return isinstance(value, list) and all(isinstance(x, element_type) for x in value)
|
|
|
|
@staticmethod
|
|
def _convert_number(value: str) -> float:
|
|
"""Unified number conversion handler"""
|
|
try:
|
|
return float(value)
|
|
except ValueError:
|
|
raise DefaultValueTypeError(f"Cannot convert to number: {value}")
|
|
|
|
@model_validator(mode="after")
|
|
def validate_value_type(self) -> DefaultValue:
|
|
# Type validation configuration
|
|
type_validators: dict[DefaultValueType, dict[str, Any]] = {
|
|
DefaultValueType.STRING: {
|
|
"type": str,
|
|
"converter": lambda x: x,
|
|
},
|
|
DefaultValueType.NUMBER: {
|
|
"type": _NumberType,
|
|
"converter": self._convert_number,
|
|
},
|
|
DefaultValueType.OBJECT: {
|
|
"type": dict,
|
|
"converter": self._parse_json,
|
|
},
|
|
DefaultValueType.ARRAY_NUMBER: {
|
|
"type": list,
|
|
"element_type": _NumberType,
|
|
"converter": self._parse_json,
|
|
},
|
|
DefaultValueType.ARRAY_STRING: {
|
|
"type": list,
|
|
"element_type": str,
|
|
"converter": self._parse_json,
|
|
},
|
|
DefaultValueType.ARRAY_OBJECT: {
|
|
"type": list,
|
|
"element_type": dict,
|
|
"converter": self._parse_json,
|
|
},
|
|
}
|
|
|
|
validator: dict[str, Any] = type_validators.get(self.type, {})
|
|
if not validator:
|
|
if self.type == DefaultValueType.ARRAY_FILES:
|
|
# Handle files type
|
|
return self
|
|
raise DefaultValueTypeError(f"Unsupported type: {self.type}")
|
|
|
|
# Handle string input cases
|
|
if isinstance(self.value, str) and self.type != DefaultValueType.STRING:
|
|
self.value = validator["converter"](self.value)
|
|
|
|
# Validate base type
|
|
if not isinstance(self.value, validator["type"]):
|
|
raise DefaultValueTypeError(f"Value must be {validator['type'].__name__} type for {self.value}")
|
|
|
|
# Validate array element types
|
|
if validator["type"] == list and not self._validate_array(self.value, validator["element_type"]):
|
|
raise DefaultValueTypeError(f"All elements must be {validator['element_type'].__name__} for {self.value}")
|
|
|
|
return self
|
|
|
|
|
|
class BaseNodeData(ABC, BaseModel):
|
|
title: str
|
|
desc: str | None = None
|
|
version: str = "1"
|
|
error_strategy: ErrorStrategy | None = None
|
|
default_value: list[DefaultValue] | None = None
|
|
retry_config: RetryConfig = RetryConfig()
|
|
|
|
# Parent node ID when this node is used as an extractor.
|
|
# If set, this node is an "attached" extractor node that extracts values
|
|
# from list[PromptMessage] for the parent node's parameters.
|
|
parent_node_id: str | None = None
|
|
|
|
@property
|
|
def is_extractor_node(self) -> bool:
|
|
"""Check if this node is an extractor node (has parent_node_id)."""
|
|
return self.parent_node_id is not None
|
|
|
|
@property
|
|
def default_value_dict(self) -> dict[str, Any]:
|
|
if self.default_value:
|
|
return {item.key: item.value for item in self.default_value}
|
|
return {}
|
|
|
|
|
|
class BaseIterationNodeData(BaseNodeData):
|
|
start_node_id: str | None = None
|
|
|
|
|
|
class BaseIterationState(BaseModel):
|
|
iteration_node_id: str
|
|
index: int
|
|
inputs: dict
|
|
|
|
class MetaData(BaseModel):
|
|
pass
|
|
|
|
metadata: MetaData
|
|
|
|
|
|
class BaseLoopNodeData(BaseNodeData):
|
|
start_node_id: str | None = None
|
|
|
|
|
|
class BaseLoopState(BaseModel):
|
|
loop_node_id: str
|
|
index: int
|
|
inputs: dict
|
|
|
|
class MetaData(BaseModel):
|
|
pass
|
|
|
|
metadata: MetaData
|