vellum-ai 0.9.16rc2__py3-none-any.whl → 0.10.0__py3-none-any.whl
Sign up to get free protection for your applications and to get access to all the features.
- vellum/plugins/__init__.py +0 -0
- vellum/plugins/pydantic.py +74 -0
- vellum/plugins/utils.py +19 -0
- vellum/plugins/vellum_mypy.py +639 -3
- vellum/workflows/README.md +90 -0
- vellum/workflows/__init__.py +5 -0
- vellum/workflows/constants.py +43 -0
- vellum/workflows/descriptors/__init__.py +0 -0
- vellum/workflows/descriptors/base.py +339 -0
- vellum/workflows/descriptors/tests/test_utils.py +83 -0
- vellum/workflows/descriptors/utils.py +90 -0
- vellum/workflows/edges/__init__.py +5 -0
- vellum/workflows/edges/edge.py +23 -0
- vellum/workflows/emitters/__init__.py +5 -0
- vellum/workflows/emitters/base.py +14 -0
- vellum/workflows/environment/__init__.py +5 -0
- vellum/workflows/environment/environment.py +7 -0
- vellum/workflows/errors/__init__.py +6 -0
- vellum/workflows/errors/types.py +20 -0
- vellum/workflows/events/__init__.py +31 -0
- vellum/workflows/events/node.py +125 -0
- vellum/workflows/events/tests/__init__.py +0 -0
- vellum/workflows/events/tests/test_event.py +216 -0
- vellum/workflows/events/types.py +52 -0
- vellum/workflows/events/utils.py +5 -0
- vellum/workflows/events/workflow.py +139 -0
- vellum/workflows/exceptions.py +15 -0
- vellum/workflows/expressions/__init__.py +0 -0
- vellum/workflows/expressions/accessor.py +52 -0
- vellum/workflows/expressions/and_.py +32 -0
- vellum/workflows/expressions/begins_with.py +31 -0
- vellum/workflows/expressions/between.py +38 -0
- vellum/workflows/expressions/coalesce_expression.py +41 -0
- vellum/workflows/expressions/contains.py +30 -0
- vellum/workflows/expressions/does_not_begin_with.py +31 -0
- vellum/workflows/expressions/does_not_contain.py +30 -0
- vellum/workflows/expressions/does_not_end_with.py +31 -0
- vellum/workflows/expressions/does_not_equal.py +25 -0
- vellum/workflows/expressions/ends_with.py +31 -0
- vellum/workflows/expressions/equals.py +25 -0
- vellum/workflows/expressions/greater_than.py +33 -0
- vellum/workflows/expressions/greater_than_or_equal_to.py +33 -0
- vellum/workflows/expressions/in_.py +31 -0
- vellum/workflows/expressions/is_blank.py +24 -0
- vellum/workflows/expressions/is_not_blank.py +24 -0
- vellum/workflows/expressions/is_not_null.py +21 -0
- vellum/workflows/expressions/is_not_undefined.py +22 -0
- vellum/workflows/expressions/is_null.py +21 -0
- vellum/workflows/expressions/is_undefined.py +22 -0
- vellum/workflows/expressions/less_than.py +33 -0
- vellum/workflows/expressions/less_than_or_equal_to.py +33 -0
- vellum/workflows/expressions/not_between.py +38 -0
- vellum/workflows/expressions/not_in.py +31 -0
- vellum/workflows/expressions/or_.py +32 -0
- vellum/workflows/graph/__init__.py +3 -0
- vellum/workflows/graph/graph.py +131 -0
- vellum/workflows/graph/tests/__init__.py +0 -0
- vellum/workflows/graph/tests/test_graph.py +437 -0
- vellum/workflows/inputs/__init__.py +5 -0
- vellum/workflows/inputs/base.py +55 -0
- vellum/workflows/logging.py +14 -0
- vellum/workflows/nodes/__init__.py +46 -0
- vellum/workflows/nodes/bases/__init__.py +7 -0
- vellum/workflows/nodes/bases/base.py +332 -0
- vellum/workflows/nodes/bases/base_subworkflow_node/__init__.py +5 -0
- vellum/workflows/nodes/bases/base_subworkflow_node/node.py +10 -0
- vellum/workflows/nodes/bases/tests/__init__.py +0 -0
- vellum/workflows/nodes/bases/tests/test_base_node.py +125 -0
- vellum/workflows/nodes/core/__init__.py +16 -0
- vellum/workflows/nodes/core/error_node/__init__.py +5 -0
- vellum/workflows/nodes/core/error_node/node.py +26 -0
- vellum/workflows/nodes/core/inline_subworkflow_node/__init__.py +5 -0
- vellum/workflows/nodes/core/inline_subworkflow_node/node.py +73 -0
- vellum/workflows/nodes/core/map_node/__init__.py +5 -0
- vellum/workflows/nodes/core/map_node/node.py +147 -0
- vellum/workflows/nodes/core/map_node/tests/__init__.py +0 -0
- vellum/workflows/nodes/core/map_node/tests/test_node.py +65 -0
- vellum/workflows/nodes/core/retry_node/__init__.py +5 -0
- vellum/workflows/nodes/core/retry_node/node.py +106 -0
- vellum/workflows/nodes/core/retry_node/tests/__init__.py +0 -0
- vellum/workflows/nodes/core/retry_node/tests/test_node.py +93 -0
- vellum/workflows/nodes/core/templating_node/__init__.py +5 -0
- vellum/workflows/nodes/core/templating_node/custom_filters.py +12 -0
- vellum/workflows/nodes/core/templating_node/exceptions.py +2 -0
- vellum/workflows/nodes/core/templating_node/node.py +123 -0
- vellum/workflows/nodes/core/templating_node/render.py +55 -0
- vellum/workflows/nodes/core/templating_node/tests/test_templating_node.py +21 -0
- vellum/workflows/nodes/core/try_node/__init__.py +5 -0
- vellum/workflows/nodes/core/try_node/node.py +110 -0
- vellum/workflows/nodes/core/try_node/tests/__init__.py +0 -0
- vellum/workflows/nodes/core/try_node/tests/test_node.py +82 -0
- vellum/workflows/nodes/displayable/__init__.py +31 -0
- vellum/workflows/nodes/displayable/api_node/__init__.py +5 -0
- vellum/workflows/nodes/displayable/api_node/node.py +44 -0
- vellum/workflows/nodes/displayable/bases/__init__.py +11 -0
- vellum/workflows/nodes/displayable/bases/api_node/__init__.py +5 -0
- vellum/workflows/nodes/displayable/bases/api_node/node.py +70 -0
- vellum/workflows/nodes/displayable/bases/base_prompt_node/__init__.py +5 -0
- vellum/workflows/nodes/displayable/bases/base_prompt_node/node.py +60 -0
- vellum/workflows/nodes/displayable/bases/inline_prompt_node/__init__.py +5 -0
- vellum/workflows/nodes/displayable/bases/inline_prompt_node/constants.py +13 -0
- vellum/workflows/nodes/displayable/bases/inline_prompt_node/node.py +118 -0
- vellum/workflows/nodes/displayable/bases/prompt_deployment_node.py +98 -0
- vellum/workflows/nodes/displayable/bases/search_node.py +90 -0
- vellum/workflows/nodes/displayable/code_execution_node/__init__.py +5 -0
- vellum/workflows/nodes/displayable/code_execution_node/node.py +197 -0
- vellum/workflows/nodes/displayable/code_execution_node/tests/__init__.py +0 -0
- vellum/workflows/nodes/displayable/code_execution_node/tests/fixtures/__init__.py +0 -0
- vellum/workflows/nodes/displayable/code_execution_node/tests/fixtures/main.py +3 -0
- vellum/workflows/nodes/displayable/code_execution_node/tests/test_code_execution_node.py +111 -0
- vellum/workflows/nodes/displayable/code_execution_node/utils.py +10 -0
- vellum/workflows/nodes/displayable/conditional_node/__init__.py +5 -0
- vellum/workflows/nodes/displayable/conditional_node/node.py +25 -0
- vellum/workflows/nodes/displayable/final_output_node/__init__.py +5 -0
- vellum/workflows/nodes/displayable/final_output_node/node.py +43 -0
- vellum/workflows/nodes/displayable/guardrail_node/__init__.py +5 -0
- vellum/workflows/nodes/displayable/guardrail_node/node.py +97 -0
- vellum/workflows/nodes/displayable/inline_prompt_node/__init__.py +5 -0
- vellum/workflows/nodes/displayable/inline_prompt_node/node.py +41 -0
- vellum/workflows/nodes/displayable/merge_node/__init__.py +5 -0
- vellum/workflows/nodes/displayable/merge_node/node.py +10 -0
- vellum/workflows/nodes/displayable/prompt_deployment_node/__init__.py +5 -0
- vellum/workflows/nodes/displayable/prompt_deployment_node/node.py +45 -0
- vellum/workflows/nodes/displayable/search_node/__init__.py +5 -0
- vellum/workflows/nodes/displayable/search_node/node.py +26 -0
- vellum/workflows/nodes/displayable/subworkflow_deployment_node/__init__.py +5 -0
- vellum/workflows/nodes/displayable/subworkflow_deployment_node/node.py +156 -0
- vellum/workflows/nodes/displayable/tests/__init__.py +0 -0
- vellum/workflows/nodes/displayable/tests/test_inline_text_prompt_node.py +148 -0
- vellum/workflows/nodes/displayable/tests/test_search_node_wth_text_output.py +134 -0
- vellum/workflows/nodes/displayable/tests/test_text_prompt_deployment_node.py +80 -0
- vellum/workflows/nodes/utils.py +27 -0
- vellum/workflows/outputs/__init__.py +6 -0
- vellum/workflows/outputs/base.py +196 -0
- vellum/workflows/ports/__init__.py +7 -0
- vellum/workflows/ports/node_ports.py +75 -0
- vellum/workflows/ports/port.py +75 -0
- vellum/workflows/ports/utils.py +40 -0
- vellum/workflows/references/__init__.py +17 -0
- vellum/workflows/references/environment_variable.py +20 -0
- vellum/workflows/references/execution_count.py +20 -0
- vellum/workflows/references/external_input.py +49 -0
- vellum/workflows/references/input.py +7 -0
- vellum/workflows/references/lazy.py +55 -0
- vellum/workflows/references/node.py +43 -0
- vellum/workflows/references/output.py +78 -0
- vellum/workflows/references/state_value.py +23 -0
- vellum/workflows/references/vellum_secret.py +15 -0
- vellum/workflows/references/workflow_input.py +41 -0
- vellum/workflows/resolvers/__init__.py +5 -0
- vellum/workflows/resolvers/base.py +15 -0
- vellum/workflows/runner/__init__.py +5 -0
- vellum/workflows/runner/runner.py +588 -0
- vellum/workflows/runner/types.py +18 -0
- vellum/workflows/state/__init__.py +5 -0
- vellum/workflows/state/base.py +327 -0
- vellum/workflows/state/context.py +18 -0
- vellum/workflows/state/encoder.py +57 -0
- vellum/workflows/state/store.py +28 -0
- vellum/workflows/state/tests/__init__.py +0 -0
- vellum/workflows/state/tests/test_state.py +113 -0
- vellum/workflows/types/__init__.py +0 -0
- vellum/workflows/types/core.py +91 -0
- vellum/workflows/types/generics.py +14 -0
- vellum/workflows/types/stack.py +39 -0
- vellum/workflows/types/tests/__init__.py +0 -0
- vellum/workflows/types/tests/test_utils.py +76 -0
- vellum/workflows/types/utils.py +164 -0
- vellum/workflows/utils/__init__.py +0 -0
- vellum/workflows/utils/names.py +13 -0
- vellum/workflows/utils/tests/__init__.py +0 -0
- vellum/workflows/utils/tests/test_names.py +15 -0
- vellum/workflows/utils/tests/test_vellum_variables.py +25 -0
- vellum/workflows/utils/vellum_variables.py +81 -0
- vellum/workflows/vellum_client.py +18 -0
- vellum/workflows/workflows/__init__.py +5 -0
- vellum/workflows/workflows/base.py +365 -0
- {vellum_ai-0.9.16rc2.dist-info → vellum_ai-0.10.0.dist-info}/METADATA +2 -1
- {vellum_ai-0.9.16rc2.dist-info → vellum_ai-0.10.0.dist-info}/RECORD +245 -7
- vellum_cli/__init__.py +72 -0
- vellum_cli/aliased_group.py +103 -0
- vellum_cli/config.py +96 -0
- vellum_cli/image_push.py +112 -0
- vellum_cli/logger.py +36 -0
- vellum_cli/pull.py +73 -0
- vellum_cli/push.py +121 -0
- vellum_cli/tests/test_config.py +100 -0
- vellum_cli/tests/test_pull.py +152 -0
- vellum_ee/workflows/__init__.py +0 -0
- vellum_ee/workflows/display/__init__.py +0 -0
- vellum_ee/workflows/display/base.py +73 -0
- vellum_ee/workflows/display/nodes/__init__.py +4 -0
- vellum_ee/workflows/display/nodes/base_node_display.py +116 -0
- vellum_ee/workflows/display/nodes/base_node_vellum_display.py +36 -0
- vellum_ee/workflows/display/nodes/get_node_display_class.py +25 -0
- vellum_ee/workflows/display/nodes/tests/__init__.py +0 -0
- vellum_ee/workflows/display/nodes/tests/test_base_node_display.py +47 -0
- vellum_ee/workflows/display/nodes/types.py +18 -0
- vellum_ee/workflows/display/nodes/utils.py +33 -0
- vellum_ee/workflows/display/nodes/vellum/__init__.py +32 -0
- vellum_ee/workflows/display/nodes/vellum/api_node.py +205 -0
- vellum_ee/workflows/display/nodes/vellum/code_execution_node.py +71 -0
- vellum_ee/workflows/display/nodes/vellum/conditional_node.py +217 -0
- vellum_ee/workflows/display/nodes/vellum/final_output_node.py +61 -0
- vellum_ee/workflows/display/nodes/vellum/guardrail_node.py +49 -0
- vellum_ee/workflows/display/nodes/vellum/inline_prompt_node.py +170 -0
- vellum_ee/workflows/display/nodes/vellum/inline_subworkflow_node.py +99 -0
- vellum_ee/workflows/display/nodes/vellum/map_node.py +100 -0
- vellum_ee/workflows/display/nodes/vellum/merge_node.py +48 -0
- vellum_ee/workflows/display/nodes/vellum/prompt_deployment_node.py +68 -0
- vellum_ee/workflows/display/nodes/vellum/search_node.py +193 -0
- vellum_ee/workflows/display/nodes/vellum/subworkflow_deployment_node.py +58 -0
- vellum_ee/workflows/display/nodes/vellum/templating_node.py +67 -0
- vellum_ee/workflows/display/nodes/vellum/tests/__init__.py +0 -0
- vellum_ee/workflows/display/nodes/vellum/tests/test_utils.py +106 -0
- vellum_ee/workflows/display/nodes/vellum/try_node.py +38 -0
- vellum_ee/workflows/display/nodes/vellum/utils.py +76 -0
- vellum_ee/workflows/display/tests/__init__.py +0 -0
- vellum_ee/workflows/display/tests/workflow_serialization/__init__.py +0 -0
- vellum_ee/workflows/display/tests/workflow_serialization/test_basic_api_node_serialization.py +426 -0
- vellum_ee/workflows/display/tests/workflow_serialization/test_basic_code_execution_node_serialization.py +607 -0
- vellum_ee/workflows/display/tests/workflow_serialization/test_basic_conditional_node_serialization.py +1175 -0
- vellum_ee/workflows/display/tests/workflow_serialization/test_basic_guardrail_node_serialization.py +235 -0
- vellum_ee/workflows/display/tests/workflow_serialization/test_basic_inline_subworkflow_serialization.py +511 -0
- vellum_ee/workflows/display/tests/workflow_serialization/test_basic_map_node_serialization.py +372 -0
- vellum_ee/workflows/display/tests/workflow_serialization/test_basic_merge_node_serialization.py +272 -0
- vellum_ee/workflows/display/tests/workflow_serialization/test_basic_prompt_deployment_serialization.py +289 -0
- vellum_ee/workflows/display/tests/workflow_serialization/test_basic_subworkflow_deployment_serialization.py +354 -0
- vellum_ee/workflows/display/tests/workflow_serialization/test_basic_terminal_node_serialization.py +123 -0
- vellum_ee/workflows/display/tests/workflow_serialization/test_basic_try_node_serialization.py +84 -0
- vellum_ee/workflows/display/tests/workflow_serialization/test_complex_terminal_node_serialization.py +233 -0
- vellum_ee/workflows/display/types.py +46 -0
- vellum_ee/workflows/display/utils/__init__.py +0 -0
- vellum_ee/workflows/display/utils/tests/__init__.py +0 -0
- vellum_ee/workflows/display/utils/tests/test_uuids.py +16 -0
- vellum_ee/workflows/display/utils/uuids.py +24 -0
- vellum_ee/workflows/display/utils/vellum.py +121 -0
- vellum_ee/workflows/display/vellum.py +357 -0
- vellum_ee/workflows/display/workflows/__init__.py +5 -0
- vellum_ee/workflows/display/workflows/base_workflow_display.py +302 -0
- vellum_ee/workflows/display/workflows/get_vellum_workflow_display_class.py +32 -0
- vellum_ee/workflows/display/workflows/vellum_workflow_display.py +386 -0
- {vellum_ai-0.9.16rc2.dist-info → vellum_ai-0.10.0.dist-info}/LICENSE +0 -0
- {vellum_ai-0.9.16rc2.dist-info → vellum_ai-0.10.0.dist-info}/WHEEL +0 -0
- {vellum_ai-0.9.16rc2.dist-info → vellum_ai-0.10.0.dist-info}/entry_points.txt +0 -0
@@ -0,0 +1,327 @@
|
|
1
|
+
from collections import defaultdict
|
2
|
+
from copy import deepcopy
|
3
|
+
from dataclasses import field
|
4
|
+
from datetime import datetime
|
5
|
+
from queue import Queue
|
6
|
+
from threading import Lock
|
7
|
+
from uuid import UUID, uuid4
|
8
|
+
from typing import TYPE_CHECKING, Any, Callable, Dict, Iterator, Optional, Sequence, Set, Tuple, Type, cast
|
9
|
+
from typing_extensions import dataclass_transform
|
10
|
+
|
11
|
+
from pydantic import GetCoreSchemaHandler, field_serializer
|
12
|
+
from pydantic_core import core_schema
|
13
|
+
|
14
|
+
from vellum.core.pydantic_utilities import UniversalBaseModel
|
15
|
+
|
16
|
+
from vellum.workflows.constants import UNDEF
|
17
|
+
from vellum.workflows.inputs.base import BaseInputs
|
18
|
+
from vellum.workflows.references import ExternalInputReference, OutputReference, StateValueReference
|
19
|
+
from vellum.workflows.types.generics import StateType
|
20
|
+
from vellum.workflows.types.stack import Stack
|
21
|
+
from vellum.workflows.types.utils import datetime_now, deepcopy_with_exclusions, get_class_by_qualname, infer_types
|
22
|
+
|
23
|
+
if TYPE_CHECKING:
|
24
|
+
from vellum.workflows.nodes.bases import BaseNode
|
25
|
+
|
26
|
+
|
27
|
+
class _Snapshottable:
|
28
|
+
_snapshot_callback: Callable[[], None]
|
29
|
+
|
30
|
+
def __deepcopy__(self, memo: Any) -> "_Snapshottable":
|
31
|
+
return deepcopy_with_exclusions(
|
32
|
+
self,
|
33
|
+
memo=memo,
|
34
|
+
exclusions={
|
35
|
+
"_snapshot_callback": self._snapshot_callback,
|
36
|
+
},
|
37
|
+
)
|
38
|
+
|
39
|
+
|
40
|
+
@dataclass_transform(kw_only_default=True)
|
41
|
+
class _BaseStateMeta(type):
|
42
|
+
def __getattribute__(cls, name: str) -> Any:
|
43
|
+
if not name.startswith("_"):
|
44
|
+
instance = vars(cls).get(name)
|
45
|
+
types = infer_types(cls, name)
|
46
|
+
return StateValueReference(name=name, types=types, instance=instance)
|
47
|
+
|
48
|
+
return super().__getattribute__(name)
|
49
|
+
|
50
|
+
|
51
|
+
class _SnapshottableDict(dict, _Snapshottable):
|
52
|
+
def __setitem__(self, key: Any, value: Any) -> None:
|
53
|
+
super().__setitem__(key, value)
|
54
|
+
self._snapshot_callback()
|
55
|
+
|
56
|
+
|
57
|
+
def _make_snapshottable(value: Any, snapshot_callback: Callable[[], None]) -> Any:
|
58
|
+
"""
|
59
|
+
Edits any value to make it snapshottable on edit. Made as a separate function from `BaseState` to
|
60
|
+
avoid namespace conflicts with subclasses.
|
61
|
+
"""
|
62
|
+
if isinstance(value, _Snapshottable):
|
63
|
+
return value
|
64
|
+
|
65
|
+
if isinstance(value, dict):
|
66
|
+
snapshottable_dict = _SnapshottableDict(value)
|
67
|
+
snapshottable_dict._snapshot_callback = snapshot_callback
|
68
|
+
return snapshottable_dict
|
69
|
+
|
70
|
+
return value
|
71
|
+
|
72
|
+
|
73
|
+
class NodeExecutionCache:
|
74
|
+
_node_execution_ids: Dict[Type["BaseNode"], Stack[UUID]]
|
75
|
+
_node_executions_initiated: Dict[Type["BaseNode"], Set[UUID]]
|
76
|
+
_dependencies_invoked: Dict[str, Set[str]]
|
77
|
+
|
78
|
+
def __init__(
|
79
|
+
self,
|
80
|
+
dependencies_invoked: Optional[Dict[str, Sequence[str]]] = None,
|
81
|
+
node_execution_ids: Optional[Dict[str, Sequence[str]]] = None,
|
82
|
+
node_executions_initiated: Optional[Dict[str, Sequence[str]]] = None,
|
83
|
+
) -> None:
|
84
|
+
self._dependencies_invoked = defaultdict(set)
|
85
|
+
self._node_execution_ids = defaultdict(Stack[UUID])
|
86
|
+
self._node_executions_initiated = defaultdict(set)
|
87
|
+
|
88
|
+
for node, dependencies in (dependencies_invoked or {}).items():
|
89
|
+
self._dependencies_invoked[node].update(dependencies)
|
90
|
+
|
91
|
+
for node, execution_ids in (node_execution_ids or {}).items():
|
92
|
+
node_class = get_class_by_qualname(node)
|
93
|
+
self._node_execution_ids[node_class].extend(UUID(execution_id) for execution_id in execution_ids)
|
94
|
+
|
95
|
+
for node, execution_ids in (node_executions_initiated or {}).items():
|
96
|
+
node_class = get_class_by_qualname(node)
|
97
|
+
self._node_executions_initiated[node_class].update({UUID(execution_id) for execution_id in execution_ids})
|
98
|
+
|
99
|
+
@property
|
100
|
+
def dependencies_invoked(self) -> Dict[str, Set[str]]:
|
101
|
+
return self._dependencies_invoked
|
102
|
+
|
103
|
+
def is_node_initiated(self, node: Type["BaseNode"]) -> bool:
|
104
|
+
return node in self._node_executions_initiated and len(self._node_executions_initiated[node]) > 0
|
105
|
+
|
106
|
+
def initiate_node_execution(self, node: Type["BaseNode"], execution_id: UUID) -> None:
|
107
|
+
self._node_executions_initiated[node].add(execution_id)
|
108
|
+
|
109
|
+
def fulfill_node_execution(self, node: Type["BaseNode"], execution_id: UUID) -> None:
|
110
|
+
self._node_executions_initiated[node].remove(execution_id)
|
111
|
+
self._node_execution_ids[node].push(execution_id)
|
112
|
+
|
113
|
+
def get_execution_count(self, node: Type["BaseNode"]) -> int:
|
114
|
+
return self._node_execution_ids[node].size()
|
115
|
+
|
116
|
+
def dump(self) -> Dict[str, Any]:
|
117
|
+
return {
|
118
|
+
"dependencies_invoked": {
|
119
|
+
node: list(dependencies) for node, dependencies in self._dependencies_invoked.items()
|
120
|
+
},
|
121
|
+
"node_executions_initiated": {
|
122
|
+
str(node): list(execution_ids) for node, execution_ids in self._node_executions_initiated.items()
|
123
|
+
},
|
124
|
+
"node_execution_ids": {
|
125
|
+
str(node): execution_ids.dump() for node, execution_ids in self._node_execution_ids.items()
|
126
|
+
},
|
127
|
+
}
|
128
|
+
|
129
|
+
@classmethod
|
130
|
+
def __get_pydantic_core_schema__(
|
131
|
+
cls, source_type: Type[Any], handler: GetCoreSchemaHandler
|
132
|
+
) -> core_schema.CoreSchema:
|
133
|
+
return core_schema.is_instance_schema(cls)
|
134
|
+
|
135
|
+
|
136
|
+
def uuid4_default_factory() -> UUID:
|
137
|
+
"""
|
138
|
+
Allows us to mock the uuid4 for testing.
|
139
|
+
"""
|
140
|
+
return uuid4()
|
141
|
+
|
142
|
+
|
143
|
+
def default_datetime_factory() -> datetime:
|
144
|
+
"""
|
145
|
+
Makes it possible to mock the datetime factory for testing.
|
146
|
+
"""
|
147
|
+
|
148
|
+
return datetime_now()
|
149
|
+
|
150
|
+
|
151
|
+
class StateMeta(UniversalBaseModel):
|
152
|
+
id: UUID = field(default_factory=uuid4_default_factory)
|
153
|
+
trace_id: UUID = field(default_factory=uuid4_default_factory)
|
154
|
+
span_id: UUID = field(default_factory=uuid4_default_factory)
|
155
|
+
updated_ts: datetime = field(default_factory=default_datetime_factory)
|
156
|
+
workflow_inputs: BaseInputs = field(default_factory=BaseInputs)
|
157
|
+
external_inputs: Dict[ExternalInputReference, Any] = field(default_factory=dict)
|
158
|
+
node_outputs: Dict[OutputReference, Any] = field(default_factory=dict)
|
159
|
+
node_execution_cache: NodeExecutionCache = field(default_factory=NodeExecutionCache)
|
160
|
+
parent: Optional["BaseState"] = None
|
161
|
+
is_terminated: Optional[bool] = None
|
162
|
+
__snapshot_callback__: Optional[Callable[[], None]] = field(init=False, default=None)
|
163
|
+
|
164
|
+
def model_post_init(self, context: Any) -> None:
|
165
|
+
if self.parent:
|
166
|
+
self.trace_id = self.parent.meta.trace_id
|
167
|
+
self.__snapshot_callback__ = None
|
168
|
+
|
169
|
+
def add_snapshot_callback(self, callback: Callable[[], None]) -> None:
|
170
|
+
self.node_outputs = _make_snapshottable(self.node_outputs, callback)
|
171
|
+
self.__snapshot_callback__ = callback
|
172
|
+
|
173
|
+
def __setattr__(self, name: str, value: Any) -> None:
|
174
|
+
if name.startswith("__") or name == "updated_ts":
|
175
|
+
super().__setattr__(name, value)
|
176
|
+
return
|
177
|
+
|
178
|
+
super().__setattr__(name, value)
|
179
|
+
if callable(self.__snapshot_callback__):
|
180
|
+
self.__snapshot_callback__()
|
181
|
+
|
182
|
+
@field_serializer("node_outputs")
|
183
|
+
def serialize_node_outputs(self, node_outputs: Dict[OutputReference, Any], _info: Any) -> Dict[str, Any]:
|
184
|
+
return {str(descriptor): value for descriptor, value in node_outputs.items()}
|
185
|
+
|
186
|
+
@field_serializer("external_inputs")
|
187
|
+
def serialize_external_inputs(
|
188
|
+
self, external_inputs: Dict[ExternalInputReference, Any], _info: Any
|
189
|
+
) -> Dict[str, Any]:
|
190
|
+
return {str(descriptor): value for descriptor, value in external_inputs.items()}
|
191
|
+
|
192
|
+
def __deepcopy__(self, memo: Optional[Dict[int, Any]] = None) -> "StateMeta":
|
193
|
+
if not memo:
|
194
|
+
memo = {}
|
195
|
+
|
196
|
+
new_node_outputs = {
|
197
|
+
descriptor: value if isinstance(value, Queue) else deepcopy(value, memo)
|
198
|
+
for descriptor, value in self.node_outputs.items()
|
199
|
+
}
|
200
|
+
|
201
|
+
memo[id(self.node_outputs)] = new_node_outputs
|
202
|
+
memo[id(self.__snapshot_callback__)] = None
|
203
|
+
|
204
|
+
return super().__deepcopy__(memo)
|
205
|
+
|
206
|
+
|
207
|
+
class BaseState(metaclass=_BaseStateMeta):
|
208
|
+
meta: StateMeta = field(init=False)
|
209
|
+
|
210
|
+
__lock__: Lock = field(init=False)
|
211
|
+
__is_initializing__: bool = field(init=False)
|
212
|
+
__snapshot_callback__: Callable[["BaseState"], None] = field(init=False)
|
213
|
+
|
214
|
+
def __init__(self, meta: Optional[StateMeta] = None, **kwargs: Any) -> None:
|
215
|
+
self.__is_initializing__ = True
|
216
|
+
self.__snapshot_callback__ = lambda state: None
|
217
|
+
self.__lock__ = Lock()
|
218
|
+
|
219
|
+
self.meta = meta or StateMeta()
|
220
|
+
self.meta.add_snapshot_callback(self.__snapshot__)
|
221
|
+
|
222
|
+
# Make all class attribute values snapshottable
|
223
|
+
for name, value in self.__class__.__dict__.items():
|
224
|
+
if not name.startswith("_") and name != "meta":
|
225
|
+
# Bypass __is_initializing__ instead of `setattr`
|
226
|
+
snapshottable_value = _make_snapshottable(value, self.__snapshot__)
|
227
|
+
super().__setattr__(name, snapshottable_value)
|
228
|
+
|
229
|
+
for name, value in kwargs.items():
|
230
|
+
setattr(self, name, value)
|
231
|
+
|
232
|
+
self.__is_initializing__ = False
|
233
|
+
|
234
|
+
def __deepcopy__(self, memo: Any) -> "BaseState":
|
235
|
+
new_state = deepcopy_with_exclusions(
|
236
|
+
self,
|
237
|
+
exclusions={
|
238
|
+
"__lock__": Lock(),
|
239
|
+
},
|
240
|
+
memo=memo,
|
241
|
+
)
|
242
|
+
new_state.meta.add_snapshot_callback(new_state.__snapshot__)
|
243
|
+
return new_state
|
244
|
+
|
245
|
+
def __repr__(self) -> str:
|
246
|
+
values = "\n".join(
|
247
|
+
[f" {key}={value}" for key, value in vars(self).items() if not key.startswith("_") and key != "meta"]
|
248
|
+
)
|
249
|
+
node_outputs = "\n".join([f" {key}={value}" for key, value in self.meta.node_outputs.items()])
|
250
|
+
return f"""\
|
251
|
+
{self.__class__.__name__}:
|
252
|
+
{values}
|
253
|
+
meta:
|
254
|
+
id={self.meta.id}
|
255
|
+
is_terminated={self.meta.is_terminated}
|
256
|
+
updated_ts={self.meta.updated_ts}
|
257
|
+
node_outputs:{' Empty' if not node_outputs else ''}
|
258
|
+
{node_outputs}
|
259
|
+
"""
|
260
|
+
|
261
|
+
def __iter__(self) -> Iterator[Tuple[Any, Any]]:
|
262
|
+
"""
|
263
|
+
Returns an iterator treating all state keys as (key, value) items, allowing consumers to call `dict()`
|
264
|
+
on an instance of this class.
|
265
|
+
"""
|
266
|
+
|
267
|
+
# If the user sets a default value on state (e.g. something = "foo"), it's not on `instance_attributes` below.
|
268
|
+
# So we need to include class_attributes here just in case
|
269
|
+
class_attributes = {key: value for key, value in self.__class__.__dict__.items() if not key.startswith("_")}
|
270
|
+
instance_attributes = {key: value for key, value in self.__dict__.items() if not key.startswith("__")}
|
271
|
+
|
272
|
+
all_attributes = {**class_attributes, **instance_attributes}
|
273
|
+
items = [(key, value) for key, value in all_attributes.items() if key not in ["_lock"]]
|
274
|
+
return iter(items)
|
275
|
+
|
276
|
+
def __getitem__(self, key: str) -> Any:
|
277
|
+
return self.__dict__[key]
|
278
|
+
|
279
|
+
def __setattr__(self, name: str, value: Any) -> None:
|
280
|
+
if name.startswith("_") or self.__is_initializing__:
|
281
|
+
super().__setattr__(name, value)
|
282
|
+
return
|
283
|
+
|
284
|
+
snapshottable_value = _make_snapshottable(value, self.__snapshot__)
|
285
|
+
super().__setattr__(name, snapshottable_value)
|
286
|
+
self.meta.updated_ts = datetime_now()
|
287
|
+
self.__snapshot__()
|
288
|
+
|
289
|
+
def __add__(self, other: StateType) -> StateType:
|
290
|
+
"""
|
291
|
+
Handles merging two states together, preferring the latest state by updated_ts for any given node output.
|
292
|
+
"""
|
293
|
+
|
294
|
+
if not isinstance(other, type(self)):
|
295
|
+
raise TypeError(f"Cannot add {type(other).__name__} to {type(self).__name__}]")
|
296
|
+
|
297
|
+
latest_state = self if self.meta.updated_ts >= other.meta.updated_ts else other
|
298
|
+
oldest_state = other if latest_state == self else self
|
299
|
+
|
300
|
+
for descriptor, value in oldest_state.meta.node_outputs.items():
|
301
|
+
if descriptor not in latest_state.meta.node_outputs:
|
302
|
+
latest_state.meta.node_outputs[descriptor] = value
|
303
|
+
|
304
|
+
for key, value in oldest_state:
|
305
|
+
if not isinstance(key, str):
|
306
|
+
continue
|
307
|
+
|
308
|
+
if key.startswith("_"):
|
309
|
+
continue
|
310
|
+
|
311
|
+
if getattr(latest_state, key, UNDEF) == UNDEF:
|
312
|
+
setattr(latest_state, key, value)
|
313
|
+
|
314
|
+
return cast(StateType, latest_state)
|
315
|
+
|
316
|
+
def __snapshot__(self) -> None:
|
317
|
+
"""
|
318
|
+
Snapshots the current state to the workflow emitter. The invoked callback is overridden by the
|
319
|
+
workflow runner.
|
320
|
+
"""
|
321
|
+
self.__snapshot_callback__(deepcopy(self))
|
322
|
+
|
323
|
+
@classmethod
|
324
|
+
def __get_pydantic_core_schema__(
|
325
|
+
cls, source_type: Type[Any], handler: GetCoreSchemaHandler
|
326
|
+
) -> core_schema.CoreSchema:
|
327
|
+
return core_schema.is_instance_schema(cls)
|
@@ -0,0 +1,18 @@
|
|
1
|
+
from functools import cached_property
|
2
|
+
from typing import Optional
|
3
|
+
|
4
|
+
from vellum import Vellum
|
5
|
+
|
6
|
+
from vellum.workflows.vellum_client import create_vellum_client
|
7
|
+
|
8
|
+
|
9
|
+
class WorkflowContext:
|
10
|
+
def __init__(self, _vellum_client: Optional[Vellum] = None):
|
11
|
+
self._vellum_client = _vellum_client
|
12
|
+
|
13
|
+
@cached_property
|
14
|
+
def vellum_client(self) -> Vellum:
|
15
|
+
if self._vellum_client:
|
16
|
+
return self._vellum_client
|
17
|
+
|
18
|
+
return create_vellum_client()
|
@@ -0,0 +1,57 @@
|
|
1
|
+
from dataclasses import asdict, is_dataclass
|
2
|
+
from datetime import datetime
|
3
|
+
import enum
|
4
|
+
from json import JSONEncoder
|
5
|
+
from uuid import UUID
|
6
|
+
from typing import Any, Callable, Dict, Type
|
7
|
+
|
8
|
+
from pydantic import BaseModel
|
9
|
+
|
10
|
+
from vellum.workflows.inputs.base import BaseInputs
|
11
|
+
from vellum.workflows.outputs.base import BaseOutput, BaseOutputs
|
12
|
+
from vellum.workflows.state.base import BaseState, NodeExecutionCache
|
13
|
+
|
14
|
+
|
15
|
+
class DefaultStateEncoder(JSONEncoder):
|
16
|
+
encoders: Dict[Type, Callable] = {}
|
17
|
+
|
18
|
+
def default(self, obj: Any) -> Any:
|
19
|
+
if isinstance(obj, BaseState):
|
20
|
+
return dict(obj)
|
21
|
+
|
22
|
+
if isinstance(obj, (BaseInputs, BaseOutputs)):
|
23
|
+
return {descriptor.name: value for descriptor, value in obj}
|
24
|
+
|
25
|
+
if isinstance(obj, BaseOutput):
|
26
|
+
return obj.serialize()
|
27
|
+
|
28
|
+
if isinstance(obj, NodeExecutionCache):
|
29
|
+
return obj.dump()
|
30
|
+
|
31
|
+
if isinstance(obj, UUID):
|
32
|
+
return str(obj)
|
33
|
+
|
34
|
+
if isinstance(obj, set):
|
35
|
+
return list(obj)
|
36
|
+
|
37
|
+
if isinstance(obj, BaseModel):
|
38
|
+
return obj.model_dump()
|
39
|
+
|
40
|
+
if isinstance(obj, datetime):
|
41
|
+
return obj.isoformat()
|
42
|
+
|
43
|
+
if isinstance(obj, enum.Enum):
|
44
|
+
return obj.value
|
45
|
+
|
46
|
+
if is_dataclass(obj):
|
47
|
+
# Technically, obj is DataclassInstance | type[DataclassInstance], but asdict expects a DataclassInstance
|
48
|
+
# in practice, we only ever pass the former
|
49
|
+
return asdict(obj) # type: ignore[call-overload]
|
50
|
+
|
51
|
+
if isinstance(obj, type):
|
52
|
+
return str(obj)
|
53
|
+
|
54
|
+
if obj.__class__ in self.encoders:
|
55
|
+
return self.encoders[obj.__class__](obj)
|
56
|
+
|
57
|
+
return super().default(obj)
|
@@ -0,0 +1,28 @@
|
|
1
|
+
from typing import Iterator, List
|
2
|
+
|
3
|
+
from vellum.workflows.events.workflow import WorkflowEvent
|
4
|
+
from vellum.workflows.state.base import BaseState
|
5
|
+
|
6
|
+
|
7
|
+
class Store:
|
8
|
+
def __init__(self) -> None:
|
9
|
+
self._events: List[WorkflowEvent] = []
|
10
|
+
self._state_snapshots: List[BaseState] = []
|
11
|
+
|
12
|
+
def append_event(self, event: WorkflowEvent) -> None:
|
13
|
+
self._events.append(event)
|
14
|
+
|
15
|
+
def append_state_snapshot(self, state: BaseState) -> None:
|
16
|
+
self._state_snapshots.append(state)
|
17
|
+
|
18
|
+
def clear(self) -> None:
|
19
|
+
self._events = []
|
20
|
+
self._state_snapshots = []
|
21
|
+
|
22
|
+
@property
|
23
|
+
def events(self) -> Iterator[WorkflowEvent]:
|
24
|
+
return iter(self._events)
|
25
|
+
|
26
|
+
@property
|
27
|
+
def state_snapshots(self) -> Iterator[BaseState]:
|
28
|
+
return iter(self._state_snapshots)
|
File without changes
|
@@ -0,0 +1,113 @@
|
|
1
|
+
from collections import defaultdict
|
2
|
+
from copy import deepcopy
|
3
|
+
import json
|
4
|
+
from typing import Dict
|
5
|
+
|
6
|
+
from vellum.workflows.nodes.bases import BaseNode
|
7
|
+
from vellum.workflows.outputs.base import BaseOutputs
|
8
|
+
from vellum.workflows.state.base import BaseState
|
9
|
+
from vellum.workflows.state.encoder import DefaultStateEncoder
|
10
|
+
|
11
|
+
snapshot_count: Dict[int, int] = defaultdict(int)
|
12
|
+
|
13
|
+
|
14
|
+
class MockState(BaseState):
|
15
|
+
foo: str
|
16
|
+
nested_dict: Dict[str, int] = {}
|
17
|
+
|
18
|
+
def __snapshot__(self) -> None:
|
19
|
+
global snapshot_count
|
20
|
+
snapshot_count[id(self)] += 1
|
21
|
+
|
22
|
+
|
23
|
+
class MockNode(BaseNode):
|
24
|
+
class Outputs(BaseOutputs):
|
25
|
+
baz: str
|
26
|
+
|
27
|
+
|
28
|
+
def test_state_snapshot__node_attribute_edit():
|
29
|
+
# GIVEN an initial state instance
|
30
|
+
state = MockState(foo="bar")
|
31
|
+
assert snapshot_count[id(state)] == 0
|
32
|
+
|
33
|
+
# WHEN we edit an attribute
|
34
|
+
state.foo = "baz"
|
35
|
+
|
36
|
+
# THEN the snapshot is emitted
|
37
|
+
assert snapshot_count[id(state)] == 1
|
38
|
+
|
39
|
+
|
40
|
+
def test_state_snapshot__node_output_edit():
|
41
|
+
# GIVEN an initial state instance
|
42
|
+
state = MockState(foo="bar")
|
43
|
+
assert snapshot_count[id(state)] == 0
|
44
|
+
|
45
|
+
# WHEN we add a Node Output to state
|
46
|
+
for output in MockNode.Outputs:
|
47
|
+
state.meta.node_outputs[output] = "hello"
|
48
|
+
|
49
|
+
# THEN the snapshot is emitted
|
50
|
+
assert snapshot_count[id(state)] == 1
|
51
|
+
|
52
|
+
|
53
|
+
def test_state_snapshot__nested_dictionary_edit():
|
54
|
+
# GIVEN an initial state instance
|
55
|
+
state = MockState(foo="bar")
|
56
|
+
assert snapshot_count[id(state)] == 0
|
57
|
+
|
58
|
+
# WHEN we edit a nested dictionary
|
59
|
+
state.nested_dict["hello"] = 1
|
60
|
+
|
61
|
+
# THEN the snapshot is emitted
|
62
|
+
assert snapshot_count[id(state)] == 1
|
63
|
+
|
64
|
+
|
65
|
+
def test_state_deepcopy():
|
66
|
+
# GIVEN an initial state instance
|
67
|
+
state = MockState(foo="bar")
|
68
|
+
|
69
|
+
# AND we add a Node Output to state
|
70
|
+
state.meta.node_outputs[MockNode.Outputs.baz] = "hello"
|
71
|
+
|
72
|
+
# WHEN we deepcopy the state
|
73
|
+
deepcopied_state = deepcopy(state)
|
74
|
+
|
75
|
+
# THEN node outputs are deepcopied
|
76
|
+
assert deepcopied_state.meta.node_outputs == state.meta.node_outputs
|
77
|
+
|
78
|
+
|
79
|
+
def test_state_deepcopy__with_node_output_updates():
|
80
|
+
# GIVEN an initial state instance
|
81
|
+
state = MockState(foo="bar")
|
82
|
+
|
83
|
+
# AND we add a Node Output to state
|
84
|
+
state.meta.node_outputs[MockNode.Outputs.baz] = "hello"
|
85
|
+
|
86
|
+
# AND we deepcopy the state
|
87
|
+
deepcopied_state = deepcopy(state)
|
88
|
+
|
89
|
+
# AND we update the original state
|
90
|
+
state.meta.node_outputs[MockNode.Outputs.baz] = "world"
|
91
|
+
|
92
|
+
# THEN the copied state is not updated
|
93
|
+
assert deepcopied_state.meta.node_outputs[MockNode.Outputs.baz] == "hello"
|
94
|
+
|
95
|
+
# AND the original state has had the correct number of snapshots
|
96
|
+
assert snapshot_count[id(state)] == 2
|
97
|
+
|
98
|
+
# AND the copied state has had the correct number of snapshots
|
99
|
+
assert snapshot_count[id(deepcopied_state)] == 0
|
100
|
+
|
101
|
+
|
102
|
+
def test_state_json_serialization__with_node_output_updates():
|
103
|
+
# GIVEN an initial state instance
|
104
|
+
state = MockState(foo="bar")
|
105
|
+
|
106
|
+
# AND we add a Node Output to state
|
107
|
+
state.meta.node_outputs[MockNode.Outputs.baz] = "hello"
|
108
|
+
|
109
|
+
# WHEN we serialize the state
|
110
|
+
json_state = json.loads(json.dumps(state, cls=DefaultStateEncoder))
|
111
|
+
|
112
|
+
# THEN the state is serialized correctly
|
113
|
+
assert json_state["meta"]["node_outputs"] == {"MockNode.Outputs.baz": "hello"}
|
File without changes
|
@@ -0,0 +1,91 @@
|
|
1
|
+
from enum import Enum
|
2
|
+
from typing import ( # type: ignore[attr-defined]
|
3
|
+
Dict,
|
4
|
+
List,
|
5
|
+
Union,
|
6
|
+
_GenericAlias,
|
7
|
+
_SpecialGenericAlias,
|
8
|
+
_UnionGenericAlias,
|
9
|
+
)
|
10
|
+
|
11
|
+
from vellum import (
|
12
|
+
ChatMessage,
|
13
|
+
FunctionCall,
|
14
|
+
FunctionCallRequest,
|
15
|
+
SearchResult,
|
16
|
+
SearchResultRequest,
|
17
|
+
VellumAudio,
|
18
|
+
VellumAudioRequest,
|
19
|
+
VellumError,
|
20
|
+
VellumErrorRequest,
|
21
|
+
VellumImage,
|
22
|
+
VellumImageRequest,
|
23
|
+
VellumValue,
|
24
|
+
VellumValueRequest,
|
25
|
+
)
|
26
|
+
|
27
|
+
JsonArray = List["Json"]
|
28
|
+
JsonObject = Dict[str, "Json"]
|
29
|
+
Json = Union[None, bool, int, float, str, JsonArray, JsonObject]
|
30
|
+
|
31
|
+
# Unions and Generics inherit from `_GenericAlias` instead of `type`
|
32
|
+
# In future versions of python, we'll see `_UnionGenericAlias`
|
33
|
+
UnderGenericAlias = _GenericAlias
|
34
|
+
SpecialGenericAlias = _SpecialGenericAlias
|
35
|
+
UnionGenericAlias = _UnionGenericAlias
|
36
|
+
|
37
|
+
|
38
|
+
class VellumSecret:
|
39
|
+
name: str
|
40
|
+
|
41
|
+
def __init__(self, name: str):
|
42
|
+
self.name = name
|
43
|
+
|
44
|
+
|
45
|
+
VellumValuePrimitive = Union[
|
46
|
+
# String inputs
|
47
|
+
str,
|
48
|
+
# Chat history inputs
|
49
|
+
List[ChatMessage],
|
50
|
+
List[ChatMessage],
|
51
|
+
# Search results inputs
|
52
|
+
List[SearchResultRequest],
|
53
|
+
List[SearchResult],
|
54
|
+
# JSON inputs
|
55
|
+
Json,
|
56
|
+
# Number inputs
|
57
|
+
float,
|
58
|
+
# Function Call Inputs
|
59
|
+
FunctionCall,
|
60
|
+
FunctionCallRequest,
|
61
|
+
# Error Inputs
|
62
|
+
VellumError,
|
63
|
+
VellumErrorRequest,
|
64
|
+
# Array Inputs
|
65
|
+
List[VellumValueRequest],
|
66
|
+
List[VellumValue],
|
67
|
+
# Image Inputs
|
68
|
+
VellumImage,
|
69
|
+
VellumImageRequest,
|
70
|
+
# Audio Inputs
|
71
|
+
VellumAudio,
|
72
|
+
VellumAudioRequest,
|
73
|
+
# Vellum Secrets
|
74
|
+
VellumSecret,
|
75
|
+
]
|
76
|
+
|
77
|
+
EntityInputsInterface = Dict[
|
78
|
+
str,
|
79
|
+
VellumValuePrimitive,
|
80
|
+
]
|
81
|
+
|
82
|
+
|
83
|
+
class MergeBehavior(Enum):
|
84
|
+
AWAIT_ALL = "AWAIT_ALL"
|
85
|
+
AWAIT_ANY = "AWAIT_ANY"
|
86
|
+
|
87
|
+
|
88
|
+
class ConditionType(Enum):
|
89
|
+
IF = "IF"
|
90
|
+
ELIF = "ELIF"
|
91
|
+
ELSE = "ELSE"
|
@@ -0,0 +1,14 @@
|
|
1
|
+
from typing import TYPE_CHECKING, TypeVar
|
2
|
+
|
3
|
+
if TYPE_CHECKING:
|
4
|
+
from vellum.workflows import BaseWorkflow
|
5
|
+
from vellum.workflows.inputs import BaseInputs
|
6
|
+
from vellum.workflows.nodes import BaseNode
|
7
|
+
from vellum.workflows.outputs import BaseOutputs
|
8
|
+
from vellum.workflows.state import BaseState
|
9
|
+
|
10
|
+
NodeType = TypeVar("NodeType", bound="BaseNode")
|
11
|
+
StateType = TypeVar("StateType", bound="BaseState")
|
12
|
+
WorkflowType = TypeVar("WorkflowType", bound="BaseWorkflow")
|
13
|
+
WorkflowInputsType = TypeVar("WorkflowInputsType", bound="BaseInputs")
|
14
|
+
OutputsType = TypeVar("OutputsType", bound="BaseOutputs")
|
@@ -0,0 +1,39 @@
|
|
1
|
+
from collections import deque
|
2
|
+
from typing import Deque, Generic, Iterable, List, TypeVar
|
3
|
+
|
4
|
+
_T = TypeVar("_T")
|
5
|
+
|
6
|
+
|
7
|
+
class Stack(Generic[_T]):
|
8
|
+
def __init__(self) -> None:
|
9
|
+
self._items: Deque[_T] = deque()
|
10
|
+
|
11
|
+
def push(self, item: _T) -> None:
|
12
|
+
self._items.append(item)
|
13
|
+
|
14
|
+
def extend(self, items: Iterable[_T]) -> None:
|
15
|
+
item_list = list(items)
|
16
|
+
for item in item_list[::-1]:
|
17
|
+
self._items.append(item)
|
18
|
+
|
19
|
+
def pop(self) -> _T:
|
20
|
+
if not self.is_empty():
|
21
|
+
return self._items.pop()
|
22
|
+
raise IndexError("pop from empty stack")
|
23
|
+
|
24
|
+
def peek(self) -> _T:
|
25
|
+
if not self.is_empty():
|
26
|
+
return self._items[-1]
|
27
|
+
raise IndexError("peek from empty stack")
|
28
|
+
|
29
|
+
def is_empty(self) -> bool:
|
30
|
+
return len(self._items) == 0
|
31
|
+
|
32
|
+
def size(self) -> int:
|
33
|
+
return len(self._items)
|
34
|
+
|
35
|
+
def __repr__(self) -> str:
|
36
|
+
return f"Stack({self.dump()})"
|
37
|
+
|
38
|
+
def dump(self) -> List[_T]:
|
39
|
+
return [item for item in self._items][::-1]
|
File without changes
|