vellum-ai 0.9.16rc2__py3-none-any.whl → 0.9.16rc4__py3-none-any.whl

Sign up to get free protection for your applications and to get access to all the features.
Files changed (245) hide show
  1. vellum/plugins/__init__.py +0 -0
  2. vellum/plugins/pydantic.py +74 -0
  3. vellum/plugins/utils.py +19 -0
  4. vellum/plugins/vellum_mypy.py +639 -3
  5. vellum/workflows/README.md +90 -0
  6. vellum/workflows/__init__.py +5 -0
  7. vellum/workflows/constants.py +43 -0
  8. vellum/workflows/descriptors/__init__.py +0 -0
  9. vellum/workflows/descriptors/base.py +339 -0
  10. vellum/workflows/descriptors/tests/test_utils.py +83 -0
  11. vellum/workflows/descriptors/utils.py +90 -0
  12. vellum/workflows/edges/__init__.py +5 -0
  13. vellum/workflows/edges/edge.py +23 -0
  14. vellum/workflows/emitters/__init__.py +5 -0
  15. vellum/workflows/emitters/base.py +14 -0
  16. vellum/workflows/environment/__init__.py +5 -0
  17. vellum/workflows/environment/environment.py +7 -0
  18. vellum/workflows/errors/__init__.py +6 -0
  19. vellum/workflows/errors/types.py +20 -0
  20. vellum/workflows/events/__init__.py +31 -0
  21. vellum/workflows/events/node.py +125 -0
  22. vellum/workflows/events/tests/__init__.py +0 -0
  23. vellum/workflows/events/tests/test_event.py +216 -0
  24. vellum/workflows/events/types.py +52 -0
  25. vellum/workflows/events/utils.py +5 -0
  26. vellum/workflows/events/workflow.py +139 -0
  27. vellum/workflows/exceptions.py +15 -0
  28. vellum/workflows/expressions/__init__.py +0 -0
  29. vellum/workflows/expressions/accessor.py +52 -0
  30. vellum/workflows/expressions/and_.py +32 -0
  31. vellum/workflows/expressions/begins_with.py +31 -0
  32. vellum/workflows/expressions/between.py +38 -0
  33. vellum/workflows/expressions/coalesce_expression.py +41 -0
  34. vellum/workflows/expressions/contains.py +30 -0
  35. vellum/workflows/expressions/does_not_begin_with.py +31 -0
  36. vellum/workflows/expressions/does_not_contain.py +30 -0
  37. vellum/workflows/expressions/does_not_end_with.py +31 -0
  38. vellum/workflows/expressions/does_not_equal.py +25 -0
  39. vellum/workflows/expressions/ends_with.py +31 -0
  40. vellum/workflows/expressions/equals.py +25 -0
  41. vellum/workflows/expressions/greater_than.py +33 -0
  42. vellum/workflows/expressions/greater_than_or_equal_to.py +33 -0
  43. vellum/workflows/expressions/in_.py +31 -0
  44. vellum/workflows/expressions/is_blank.py +24 -0
  45. vellum/workflows/expressions/is_not_blank.py +24 -0
  46. vellum/workflows/expressions/is_not_null.py +21 -0
  47. vellum/workflows/expressions/is_not_undefined.py +22 -0
  48. vellum/workflows/expressions/is_null.py +21 -0
  49. vellum/workflows/expressions/is_undefined.py +22 -0
  50. vellum/workflows/expressions/less_than.py +33 -0
  51. vellum/workflows/expressions/less_than_or_equal_to.py +33 -0
  52. vellum/workflows/expressions/not_between.py +38 -0
  53. vellum/workflows/expressions/not_in.py +31 -0
  54. vellum/workflows/expressions/or_.py +32 -0
  55. vellum/workflows/graph/__init__.py +3 -0
  56. vellum/workflows/graph/graph.py +131 -0
  57. vellum/workflows/graph/tests/__init__.py +0 -0
  58. vellum/workflows/graph/tests/test_graph.py +437 -0
  59. vellum/workflows/inputs/__init__.py +5 -0
  60. vellum/workflows/inputs/base.py +55 -0
  61. vellum/workflows/logging.py +14 -0
  62. vellum/workflows/nodes/__init__.py +46 -0
  63. vellum/workflows/nodes/bases/__init__.py +7 -0
  64. vellum/workflows/nodes/bases/base.py +332 -0
  65. vellum/workflows/nodes/bases/base_subworkflow_node/__init__.py +5 -0
  66. vellum/workflows/nodes/bases/base_subworkflow_node/node.py +10 -0
  67. vellum/workflows/nodes/bases/tests/__init__.py +0 -0
  68. vellum/workflows/nodes/bases/tests/test_base_node.py +125 -0
  69. vellum/workflows/nodes/core/__init__.py +16 -0
  70. vellum/workflows/nodes/core/error_node/__init__.py +5 -0
  71. vellum/workflows/nodes/core/error_node/node.py +26 -0
  72. vellum/workflows/nodes/core/inline_subworkflow_node/__init__.py +5 -0
  73. vellum/workflows/nodes/core/inline_subworkflow_node/node.py +73 -0
  74. vellum/workflows/nodes/core/map_node/__init__.py +5 -0
  75. vellum/workflows/nodes/core/map_node/node.py +147 -0
  76. vellum/workflows/nodes/core/map_node/tests/__init__.py +0 -0
  77. vellum/workflows/nodes/core/map_node/tests/test_node.py +65 -0
  78. vellum/workflows/nodes/core/retry_node/__init__.py +5 -0
  79. vellum/workflows/nodes/core/retry_node/node.py +106 -0
  80. vellum/workflows/nodes/core/retry_node/tests/__init__.py +0 -0
  81. vellum/workflows/nodes/core/retry_node/tests/test_node.py +93 -0
  82. vellum/workflows/nodes/core/templating_node/__init__.py +5 -0
  83. vellum/workflows/nodes/core/templating_node/custom_filters.py +12 -0
  84. vellum/workflows/nodes/core/templating_node/exceptions.py +2 -0
  85. vellum/workflows/nodes/core/templating_node/node.py +123 -0
  86. vellum/workflows/nodes/core/templating_node/render.py +55 -0
  87. vellum/workflows/nodes/core/templating_node/tests/test_templating_node.py +21 -0
  88. vellum/workflows/nodes/core/try_node/__init__.py +5 -0
  89. vellum/workflows/nodes/core/try_node/node.py +110 -0
  90. vellum/workflows/nodes/core/try_node/tests/__init__.py +0 -0
  91. vellum/workflows/nodes/core/try_node/tests/test_node.py +82 -0
  92. vellum/workflows/nodes/displayable/__init__.py +31 -0
  93. vellum/workflows/nodes/displayable/api_node/__init__.py +5 -0
  94. vellum/workflows/nodes/displayable/api_node/node.py +44 -0
  95. vellum/workflows/nodes/displayable/bases/__init__.py +11 -0
  96. vellum/workflows/nodes/displayable/bases/api_node/__init__.py +5 -0
  97. vellum/workflows/nodes/displayable/bases/api_node/node.py +70 -0
  98. vellum/workflows/nodes/displayable/bases/base_prompt_node/__init__.py +5 -0
  99. vellum/workflows/nodes/displayable/bases/base_prompt_node/node.py +60 -0
  100. vellum/workflows/nodes/displayable/bases/inline_prompt_node/__init__.py +5 -0
  101. vellum/workflows/nodes/displayable/bases/inline_prompt_node/constants.py +13 -0
  102. vellum/workflows/nodes/displayable/bases/inline_prompt_node/node.py +118 -0
  103. vellum/workflows/nodes/displayable/bases/prompt_deployment_node.py +98 -0
  104. vellum/workflows/nodes/displayable/bases/search_node.py +90 -0
  105. vellum/workflows/nodes/displayable/code_execution_node/__init__.py +5 -0
  106. vellum/workflows/nodes/displayable/code_execution_node/node.py +197 -0
  107. vellum/workflows/nodes/displayable/code_execution_node/tests/__init__.py +0 -0
  108. vellum/workflows/nodes/displayable/code_execution_node/tests/fixtures/__init__.py +0 -0
  109. vellum/workflows/nodes/displayable/code_execution_node/tests/fixtures/main.py +3 -0
  110. vellum/workflows/nodes/displayable/code_execution_node/tests/test_code_execution_node.py +111 -0
  111. vellum/workflows/nodes/displayable/code_execution_node/utils.py +10 -0
  112. vellum/workflows/nodes/displayable/conditional_node/__init__.py +5 -0
  113. vellum/workflows/nodes/displayable/conditional_node/node.py +25 -0
  114. vellum/workflows/nodes/displayable/final_output_node/__init__.py +5 -0
  115. vellum/workflows/nodes/displayable/final_output_node/node.py +43 -0
  116. vellum/workflows/nodes/displayable/guardrail_node/__init__.py +5 -0
  117. vellum/workflows/nodes/displayable/guardrail_node/node.py +97 -0
  118. vellum/workflows/nodes/displayable/inline_prompt_node/__init__.py +5 -0
  119. vellum/workflows/nodes/displayable/inline_prompt_node/node.py +41 -0
  120. vellum/workflows/nodes/displayable/merge_node/__init__.py +5 -0
  121. vellum/workflows/nodes/displayable/merge_node/node.py +10 -0
  122. vellum/workflows/nodes/displayable/prompt_deployment_node/__init__.py +5 -0
  123. vellum/workflows/nodes/displayable/prompt_deployment_node/node.py +45 -0
  124. vellum/workflows/nodes/displayable/search_node/__init__.py +5 -0
  125. vellum/workflows/nodes/displayable/search_node/node.py +26 -0
  126. vellum/workflows/nodes/displayable/subworkflow_deployment_node/__init__.py +5 -0
  127. vellum/workflows/nodes/displayable/subworkflow_deployment_node/node.py +156 -0
  128. vellum/workflows/nodes/displayable/tests/__init__.py +0 -0
  129. vellum/workflows/nodes/displayable/tests/test_inline_text_prompt_node.py +148 -0
  130. vellum/workflows/nodes/displayable/tests/test_search_node_wth_text_output.py +134 -0
  131. vellum/workflows/nodes/displayable/tests/test_text_prompt_deployment_node.py +80 -0
  132. vellum/workflows/nodes/utils.py +27 -0
  133. vellum/workflows/outputs/__init__.py +6 -0
  134. vellum/workflows/outputs/base.py +196 -0
  135. vellum/workflows/ports/__init__.py +7 -0
  136. vellum/workflows/ports/node_ports.py +75 -0
  137. vellum/workflows/ports/port.py +75 -0
  138. vellum/workflows/ports/utils.py +40 -0
  139. vellum/workflows/references/__init__.py +17 -0
  140. vellum/workflows/references/environment_variable.py +20 -0
  141. vellum/workflows/references/execution_count.py +20 -0
  142. vellum/workflows/references/external_input.py +49 -0
  143. vellum/workflows/references/input.py +7 -0
  144. vellum/workflows/references/lazy.py +55 -0
  145. vellum/workflows/references/node.py +43 -0
  146. vellum/workflows/references/output.py +78 -0
  147. vellum/workflows/references/state_value.py +23 -0
  148. vellum/workflows/references/vellum_secret.py +15 -0
  149. vellum/workflows/references/workflow_input.py +41 -0
  150. vellum/workflows/resolvers/__init__.py +5 -0
  151. vellum/workflows/resolvers/base.py +15 -0
  152. vellum/workflows/runner/__init__.py +5 -0
  153. vellum/workflows/runner/runner.py +588 -0
  154. vellum/workflows/runner/types.py +18 -0
  155. vellum/workflows/state/__init__.py +5 -0
  156. vellum/workflows/state/base.py +327 -0
  157. vellum/workflows/state/context.py +18 -0
  158. vellum/workflows/state/encoder.py +57 -0
  159. vellum/workflows/state/store.py +28 -0
  160. vellum/workflows/state/tests/__init__.py +0 -0
  161. vellum/workflows/state/tests/test_state.py +113 -0
  162. vellum/workflows/types/__init__.py +0 -0
  163. vellum/workflows/types/core.py +91 -0
  164. vellum/workflows/types/generics.py +14 -0
  165. vellum/workflows/types/stack.py +39 -0
  166. vellum/workflows/types/tests/__init__.py +0 -0
  167. vellum/workflows/types/tests/test_utils.py +76 -0
  168. vellum/workflows/types/utils.py +164 -0
  169. vellum/workflows/utils/__init__.py +0 -0
  170. vellum/workflows/utils/names.py +13 -0
  171. vellum/workflows/utils/tests/__init__.py +0 -0
  172. vellum/workflows/utils/tests/test_names.py +15 -0
  173. vellum/workflows/utils/tests/test_vellum_variables.py +25 -0
  174. vellum/workflows/utils/vellum_variables.py +81 -0
  175. vellum/workflows/vellum_client.py +18 -0
  176. vellum/workflows/workflows/__init__.py +5 -0
  177. vellum/workflows/workflows/base.py +365 -0
  178. {vellum_ai-0.9.16rc2.dist-info → vellum_ai-0.9.16rc4.dist-info}/METADATA +2 -1
  179. {vellum_ai-0.9.16rc2.dist-info → vellum_ai-0.9.16rc4.dist-info}/RECORD +245 -7
  180. vellum_cli/__init__.py +72 -0
  181. vellum_cli/aliased_group.py +103 -0
  182. vellum_cli/config.py +96 -0
  183. vellum_cli/image_push.py +112 -0
  184. vellum_cli/logger.py +36 -0
  185. vellum_cli/pull.py +73 -0
  186. vellum_cli/push.py +121 -0
  187. vellum_cli/tests/test_config.py +100 -0
  188. vellum_cli/tests/test_pull.py +152 -0
  189. vellum_ee/workflows/__init__.py +0 -0
  190. vellum_ee/workflows/display/__init__.py +0 -0
  191. vellum_ee/workflows/display/base.py +73 -0
  192. vellum_ee/workflows/display/nodes/__init__.py +4 -0
  193. vellum_ee/workflows/display/nodes/base_node_display.py +116 -0
  194. vellum_ee/workflows/display/nodes/base_node_vellum_display.py +36 -0
  195. vellum_ee/workflows/display/nodes/get_node_display_class.py +25 -0
  196. vellum_ee/workflows/display/nodes/tests/__init__.py +0 -0
  197. vellum_ee/workflows/display/nodes/tests/test_base_node_display.py +47 -0
  198. vellum_ee/workflows/display/nodes/types.py +18 -0
  199. vellum_ee/workflows/display/nodes/utils.py +33 -0
  200. vellum_ee/workflows/display/nodes/vellum/__init__.py +32 -0
  201. vellum_ee/workflows/display/nodes/vellum/api_node.py +205 -0
  202. vellum_ee/workflows/display/nodes/vellum/code_execution_node.py +71 -0
  203. vellum_ee/workflows/display/nodes/vellum/conditional_node.py +217 -0
  204. vellum_ee/workflows/display/nodes/vellum/final_output_node.py +61 -0
  205. vellum_ee/workflows/display/nodes/vellum/guardrail_node.py +49 -0
  206. vellum_ee/workflows/display/nodes/vellum/inline_prompt_node.py +170 -0
  207. vellum_ee/workflows/display/nodes/vellum/inline_subworkflow_node.py +99 -0
  208. vellum_ee/workflows/display/nodes/vellum/map_node.py +100 -0
  209. vellum_ee/workflows/display/nodes/vellum/merge_node.py +48 -0
  210. vellum_ee/workflows/display/nodes/vellum/prompt_deployment_node.py +68 -0
  211. vellum_ee/workflows/display/nodes/vellum/search_node.py +193 -0
  212. vellum_ee/workflows/display/nodes/vellum/subworkflow_deployment_node.py +58 -0
  213. vellum_ee/workflows/display/nodes/vellum/templating_node.py +67 -0
  214. vellum_ee/workflows/display/nodes/vellum/tests/__init__.py +0 -0
  215. vellum_ee/workflows/display/nodes/vellum/tests/test_utils.py +106 -0
  216. vellum_ee/workflows/display/nodes/vellum/try_node.py +38 -0
  217. vellum_ee/workflows/display/nodes/vellum/utils.py +76 -0
  218. vellum_ee/workflows/display/tests/__init__.py +0 -0
  219. vellum_ee/workflows/display/tests/workflow_serialization/__init__.py +0 -0
  220. vellum_ee/workflows/display/tests/workflow_serialization/test_basic_api_node_serialization.py +426 -0
  221. vellum_ee/workflows/display/tests/workflow_serialization/test_basic_code_execution_node_serialization.py +607 -0
  222. vellum_ee/workflows/display/tests/workflow_serialization/test_basic_conditional_node_serialization.py +1175 -0
  223. vellum_ee/workflows/display/tests/workflow_serialization/test_basic_guardrail_node_serialization.py +235 -0
  224. vellum_ee/workflows/display/tests/workflow_serialization/test_basic_inline_subworkflow_serialization.py +511 -0
  225. vellum_ee/workflows/display/tests/workflow_serialization/test_basic_map_node_serialization.py +372 -0
  226. vellum_ee/workflows/display/tests/workflow_serialization/test_basic_merge_node_serialization.py +272 -0
  227. vellum_ee/workflows/display/tests/workflow_serialization/test_basic_prompt_deployment_serialization.py +289 -0
  228. vellum_ee/workflows/display/tests/workflow_serialization/test_basic_subworkflow_deployment_serialization.py +354 -0
  229. vellum_ee/workflows/display/tests/workflow_serialization/test_basic_terminal_node_serialization.py +123 -0
  230. vellum_ee/workflows/display/tests/workflow_serialization/test_basic_try_node_serialization.py +84 -0
  231. vellum_ee/workflows/display/tests/workflow_serialization/test_complex_terminal_node_serialization.py +233 -0
  232. vellum_ee/workflows/display/types.py +46 -0
  233. vellum_ee/workflows/display/utils/__init__.py +0 -0
  234. vellum_ee/workflows/display/utils/tests/__init__.py +0 -0
  235. vellum_ee/workflows/display/utils/tests/test_uuids.py +16 -0
  236. vellum_ee/workflows/display/utils/uuids.py +24 -0
  237. vellum_ee/workflows/display/utils/vellum.py +121 -0
  238. vellum_ee/workflows/display/vellum.py +357 -0
  239. vellum_ee/workflows/display/workflows/__init__.py +5 -0
  240. vellum_ee/workflows/display/workflows/base_workflow_display.py +302 -0
  241. vellum_ee/workflows/display/workflows/get_vellum_workflow_display_class.py +32 -0
  242. vellum_ee/workflows/display/workflows/vellum_workflow_display.py +386 -0
  243. {vellum_ai-0.9.16rc2.dist-info → vellum_ai-0.9.16rc4.dist-info}/LICENSE +0 -0
  244. {vellum_ai-0.9.16rc2.dist-info → vellum_ai-0.9.16rc4.dist-info}/WHEEL +0 -0
  245. {vellum_ai-0.9.16rc2.dist-info → vellum_ai-0.9.16rc4.dist-info}/entry_points.txt +0 -0
@@ -0,0 +1,110 @@
1
+ from typing import TYPE_CHECKING, Any, Callable, Dict, Generic, Optional, Tuple, Type, TypeVar
2
+
3
+ from vellum.workflows.errors.types import VellumError, VellumErrorCode
4
+ from vellum.workflows.exceptions import NodeException
5
+ from vellum.workflows.nodes.bases import BaseNode
6
+ from vellum.workflows.nodes.bases.base import BaseNodeMeta
7
+ from vellum.workflows.outputs.base import BaseOutputs
8
+ from vellum.workflows.types.generics import StateType
9
+
10
+ if TYPE_CHECKING:
11
+ from vellum.workflows import BaseWorkflow
12
+
13
+ Subworkflow = Type["BaseWorkflow"]
14
+ _T = TypeVar("_T", bound=BaseOutputs)
15
+
16
+
17
+ class _TryNodeMeta(BaseNodeMeta):
18
+ def __new__(cls, name: str, bases: Tuple[Type, ...], dct: Dict[str, Any]) -> Any:
19
+ node_class = super().__new__(cls, name, bases, dct)
20
+
21
+ subworkflow_attribute = dct.get("subworkflow")
22
+ if not subworkflow_attribute:
23
+ return node_class
24
+
25
+ subworkflow_outputs = getattr(subworkflow_attribute, "Outputs")
26
+ if not issubclass(subworkflow_outputs, BaseOutputs):
27
+ raise ValueError("subworkflow.Outputs must be a subclass of BaseOutputs")
28
+
29
+ outputs_class = dct.get("Outputs")
30
+ if not outputs_class:
31
+ raise ValueError("Outputs class not found in base classes")
32
+
33
+ if not issubclass(outputs_class, BaseNode.Outputs):
34
+ raise ValueError("Outputs class must be a subclass of BaseNode.Outputs")
35
+
36
+ for descriptor in subworkflow_outputs:
37
+ if descriptor.name == "error":
38
+ raise ValueError("`error` is a reserved name for TryNode.Outputs")
39
+
40
+ setattr(outputs_class, descriptor.name, descriptor)
41
+
42
+ return node_class
43
+
44
+
45
+ class TryNode(BaseNode[StateType], Generic[StateType], metaclass=_TryNodeMeta):
46
+ """
47
+ Used to execute a Subworkflow and handle errors.
48
+
49
+ on_error_code: Optional[VellumErrorCode] = None - The error code to handle
50
+ subworkflow: Type["BaseWorkflow"] - The Subworkflow to execute
51
+ """
52
+
53
+ on_error_code: Optional[VellumErrorCode] = None
54
+ subworkflow: Type["BaseWorkflow"]
55
+
56
+ class Outputs(BaseNode.Outputs):
57
+ error: Optional[VellumError] = None
58
+
59
+ def run(self) -> Outputs:
60
+ subworkflow = self.subworkflow(
61
+ parent_state=self.state,
62
+ )
63
+ terminal_event = subworkflow.run()
64
+
65
+ if terminal_event.name == "workflow.execution.fulfilled":
66
+ outputs = self.Outputs()
67
+ for descriptor, value in terminal_event.outputs:
68
+ setattr(outputs, descriptor.name, value)
69
+ return outputs
70
+ elif terminal_event.name == "workflow.execution.paused":
71
+ raise NodeException(
72
+ code=VellumErrorCode.INVALID_OUTPUTS,
73
+ message="Subworkflow unexpectedly paused within Try Node",
74
+ )
75
+ elif self.on_error_code and self.on_error_code != terminal_event.error.code:
76
+ raise NodeException(
77
+ code=VellumErrorCode.INVALID_OUTPUTS,
78
+ message=f"""Unexpected rejection: {terminal_event.error.code.value}.
79
+ Message: {terminal_event.error.message}""",
80
+ )
81
+ else:
82
+ return self.Outputs(
83
+ error=terminal_event.error,
84
+ )
85
+
86
+ @classmethod
87
+ def wrap(cls, on_error_code: Optional[VellumErrorCode] = None) -> Callable[..., Type["TryNode"]]:
88
+ _on_error_code = on_error_code
89
+
90
+ def decorator(inner_cls: Type[BaseNode]) -> Type["TryNode"]:
91
+ # Investigate how to use dependency injection to avoid circular imports
92
+ # https://app.shortcut.com/vellum/story/4116
93
+ from vellum.workflows import BaseWorkflow
94
+
95
+ class Subworkflow(BaseWorkflow):
96
+ inner_cls._is_wrapped_node = True
97
+ graph = inner_cls
98
+
99
+ # mypy is wrong here, this works and is defined
100
+ class Outputs(inner_cls.Outputs): # type: ignore[name-defined]
101
+ pass
102
+
103
+ class WrappedNode(TryNode[StateType]):
104
+ on_error_code = _on_error_code
105
+
106
+ subworkflow = Subworkflow
107
+
108
+ return WrappedNode
109
+
110
+ return decorator
File without changes
@@ -0,0 +1,82 @@
1
+ import pytest
2
+
3
+ from vellum.workflows.errors.types import VellumError, VellumErrorCode
4
+ from vellum.workflows.exceptions import NodeException
5
+ from vellum.workflows.inputs.base import BaseInputs
6
+ from vellum.workflows.nodes.bases import BaseNode
7
+ from vellum.workflows.nodes.core.try_node.node import TryNode
8
+ from vellum.workflows.outputs import BaseOutputs
9
+ from vellum.workflows.state.base import BaseState, StateMeta
10
+
11
+
12
+ def test_try_node__on_error_code__successfully_caught():
13
+ # GIVEN a try node that is configured to catch PROVIDER_ERROR
14
+ @TryNode.wrap(on_error_code=VellumErrorCode.PROVIDER_ERROR)
15
+ class TestNode(BaseNode):
16
+ class Outputs(BaseOutputs):
17
+ value: int
18
+
19
+ def run(self) -> Outputs:
20
+ raise NodeException(message="This will be caught", code=VellumErrorCode.PROVIDER_ERROR)
21
+
22
+ # WHEN the node is run and throws a PROVIDER_ERROR
23
+ node = TestNode(state=BaseState())
24
+ outputs = node.run()
25
+
26
+ # THEN the exception is retried
27
+ assert outputs == {
28
+ "error": VellumError(message="This will be caught", code=VellumErrorCode.PROVIDER_ERROR),
29
+ }
30
+
31
+
32
+ def test_try_node__retry_on_error_code__missed():
33
+ # GIVEN a try node that is configured to catch PROVIDER_ERROR
34
+ @TryNode.wrap(on_error_code=VellumErrorCode.PROVIDER_ERROR)
35
+ class TestNode(BaseNode):
36
+ class Outputs(BaseOutputs):
37
+ value: int
38
+
39
+ def run(self) -> Outputs:
40
+ raise NodeException(message="This will be missed", code=VellumErrorCode.INTERNAL_ERROR)
41
+
42
+ # WHEN the node is run and throws a different exception
43
+ node = TestNode(state=BaseState())
44
+ with pytest.raises(NodeException) as exc_info:
45
+ node.run()
46
+
47
+ # THEN the exception is not caught
48
+ assert exc_info.value.message == "Unexpected rejection: INTERNAL_ERROR.\nMessage: This will be missed"
49
+ assert exc_info.value.code == VellumErrorCode.INVALID_OUTPUTS
50
+
51
+
52
+ def test_try_node__use_parent_inputs_and_state():
53
+ # GIVEN a parent workflow Inputs and State
54
+ class Inputs(BaseInputs):
55
+ foo: str
56
+
57
+ class State(BaseState):
58
+ bar: str
59
+
60
+ # AND a try node that uses the parent's inputs and state
61
+ @TryNode.wrap()
62
+ class TestNode(BaseNode):
63
+ foo = Inputs.foo
64
+ bar = State.bar
65
+
66
+ class Outputs(BaseOutputs):
67
+ value: str
68
+
69
+ def run(self) -> Outputs:
70
+ return self.Outputs(value=f"{self.foo} {self.bar}")
71
+
72
+ # WHEN the node is run
73
+ node = TestNode(
74
+ state=State(
75
+ bar="bar",
76
+ meta=StateMeta(workflow_inputs=Inputs(foo="foo")),
77
+ ),
78
+ )
79
+ outputs = node.run()
80
+
81
+ # THEN the data is used successfully
82
+ assert outputs == {"value": "foo bar"}
@@ -0,0 +1,31 @@
1
+ from ..core.error_node import ErrorNode
2
+ from ..core.inline_subworkflow_node import InlineSubworkflowNode
3
+ from ..core.map_node import MapNode
4
+ from ..core.templating_node import TemplatingNode
5
+ from .api_node import APINode
6
+ from .code_execution_node import CodeExecutionNode
7
+ from .conditional_node import ConditionalNode
8
+ from .final_output_node import FinalOutputNode
9
+ from .guardrail_node import GuardrailNode
10
+ from .inline_prompt_node import InlinePromptNode
11
+ from .merge_node import MergeNode
12
+ from .prompt_deployment_node import PromptDeploymentNode
13
+ from .search_node import SearchNode
14
+ from .subworkflow_deployment_node import SubworkflowDeploymentNode
15
+
16
+ __all__ = [
17
+ "APINode",
18
+ "CodeExecutionNode",
19
+ "ConditionalNode",
20
+ "ErrorNode",
21
+ "InlinePromptNode",
22
+ "InlineSubworkflowNode",
23
+ "GuardrailNode",
24
+ "MapNode",
25
+ "MergeNode",
26
+ "SubworkflowDeploymentNode",
27
+ "PromptDeploymentNode",
28
+ "SearchNode",
29
+ "TemplatingNode",
30
+ "FinalOutputNode",
31
+ ]
@@ -0,0 +1,5 @@
1
+ from .node import APINode
2
+
3
+ __all__ = [
4
+ "APINode",
5
+ ]
@@ -0,0 +1,44 @@
1
+ from typing import Optional, Union
2
+
3
+ from vellum.workflows.constants import AuthorizationType
4
+ from vellum.workflows.nodes.displayable.bases.api_node import BaseAPINode
5
+ from vellum.workflows.references.vellum_secret import VellumSecretReference
6
+
7
+
8
+ class APINode(BaseAPINode):
9
+ """
10
+ Used to execute an API call. This node exists to be backwards compatible with Vellum's API Node, and for most cases,
11
+ you should extend from `APINode` directly.
12
+
13
+ url: str - The URL to send the request to.
14
+ method: APIRequestMethod - The HTTP method to use for the request.
15
+ data: Optional[str] - The data to send in the request body.
16
+ json: Optional["JsonObject"] - The JSON data to send in the request body.
17
+ headers: Optional[Dict[str, Union[str, VellumSecret]]] - The headers to send in the request.
18
+
19
+ authorization_type: Optional[AuthorizationType] = None - The type of authorization to use for the API call.
20
+ api_key_header_key: Optional[str] = None - The header key to use for the API key authorization.
21
+ bearer_token_value: Optional[str] = None - The bearer token value to use for the bearer token authorization.
22
+ """
23
+
24
+ authorization_type: Optional[AuthorizationType] = None
25
+ api_key_header_key: Optional[str] = None
26
+ api_key_header_value: Optional[Union[str, VellumSecretReference]] = None
27
+ bearer_token_value: Optional[str] = None
28
+
29
+ def run(self) -> BaseAPINode.Outputs:
30
+ headers = self.headers or {}
31
+ header_overrides = {}
32
+
33
+ if (
34
+ self.authorization_type == AuthorizationType.API_KEY
35
+ and self.api_key_header_key
36
+ and self.api_key_header_value
37
+ ):
38
+ header_overrides[self.api_key_header_key] = self.api_key_header_value
39
+ elif self.authorization_type == AuthorizationType.BEARER_TOKEN:
40
+ header_overrides["Authorization"] = f"Bearer {self.bearer_token_value}"
41
+
42
+ return self._run(
43
+ method=self.method, url=self.url, data=self.data, json=self.json, headers={**headers, **header_overrides}
44
+ )
@@ -0,0 +1,11 @@
1
+ from .api_node import BaseAPINode
2
+ from .inline_prompt_node import BaseInlinePromptNode
3
+ from .prompt_deployment_node import BasePromptDeploymentNode
4
+ from .search_node import BaseSearchNode
5
+
6
+ __all__ = [
7
+ "BaseAPINode",
8
+ "BaseInlinePromptNode",
9
+ "BasePromptDeploymentNode",
10
+ "BaseSearchNode",
11
+ ]
@@ -0,0 +1,5 @@
1
+ from .node import BaseAPINode
2
+
3
+ __all__ = [
4
+ "BaseAPINode",
5
+ ]
@@ -0,0 +1,70 @@
1
+ from typing import Any, Dict, Generic, Optional, Union
2
+
3
+ from requests import Request, RequestException, Session
4
+ from requests.exceptions import JSONDecodeError
5
+
6
+ from vellum.workflows.constants import APIRequestMethod
7
+ from vellum.workflows.errors.types import VellumErrorCode
8
+ from vellum.workflows.exceptions import NodeException
9
+ from vellum.workflows.nodes.bases import BaseNode
10
+ from vellum.workflows.outputs import BaseOutputs
11
+ from vellum.workflows.types.core import JsonObject, VellumSecret
12
+ from vellum.workflows.types.generics import StateType
13
+
14
+
15
+ class BaseAPINode(BaseNode, Generic[StateType]):
16
+ """
17
+ Used to execute an API call.
18
+
19
+ url: str - The URL to send the request to.
20
+ method: APIRequestMethod - The HTTP method to use for the request.
21
+ data: Optional[str] - The data to send in the request body.
22
+ json: Optional["JsonObject"] - The JSON data to send in the request body.
23
+ headers: Optional[Dict[str, Union[str, VellumSecret]]] - The headers to send in the request.
24
+ """
25
+
26
+ url: str
27
+ method: APIRequestMethod
28
+ data: Optional[str] = None
29
+ json: Optional["JsonObject"] = None
30
+ headers: Optional[Dict[str, Union[str, VellumSecret]]] = None
31
+
32
+ class Outputs(BaseOutputs):
33
+ json: Optional["JsonObject"]
34
+ headers: Dict[str, str]
35
+ status_code: int
36
+ text: str
37
+
38
+ def run(self) -> Outputs:
39
+ return self._run(method=self.method, url=self.url, data=self.data, json=self.json, headers=self.headers)
40
+
41
+ def _run(
42
+ self,
43
+ method: APIRequestMethod,
44
+ url: str,
45
+ data: Optional[str] = None,
46
+ json: Any = None,
47
+ headers: Any = None,
48
+ ) -> Outputs:
49
+ try:
50
+ prepped = Request(method=method.value, url=url, data=data, json=json, headers=headers).prepare()
51
+ except Exception as e:
52
+ raise NodeException(f"Failed to prepare HTTP request: {e}", code=VellumErrorCode.PROVIDER_ERROR)
53
+
54
+ try:
55
+ with Session() as session:
56
+ response = session.send(prepped)
57
+ except RequestException as e:
58
+ raise NodeException(f"HTTP request failed: {e}", code=VellumErrorCode.PROVIDER_ERROR)
59
+
60
+ try:
61
+ json = response.json()
62
+ except JSONDecodeError:
63
+ json = None
64
+
65
+ return self.Outputs(
66
+ json=json,
67
+ headers={header: value for header, value in response.headers.items()},
68
+ status_code=response.status_code,
69
+ text=response.text,
70
+ )
@@ -0,0 +1,5 @@
1
+ from .node import BasePromptNode
2
+
3
+ __all__ = [
4
+ "BasePromptNode",
5
+ ]
@@ -0,0 +1,60 @@
1
+ from abc import abstractmethod
2
+ from typing import ClassVar, Generator, Generic, Iterator, List, Optional, Union
3
+
4
+ from vellum import AdHocExecutePromptEvent, ExecutePromptEvent, PromptOutput
5
+ from vellum.core import RequestOptions
6
+
7
+ from vellum.workflows.errors.types import VellumErrorCode
8
+ from vellum.workflows.exceptions import NodeException
9
+ from vellum.workflows.nodes.bases import BaseNode
10
+ from vellum.workflows.outputs.base import BaseOutput, BaseOutputs
11
+ from vellum.workflows.types.core import EntityInputsInterface
12
+ from vellum.workflows.types.generics import StateType
13
+
14
+
15
+ class BasePromptNode(BaseNode, Generic[StateType]):
16
+ # Inputs that are passed to the Prompt
17
+ prompt_inputs: ClassVar[EntityInputsInterface]
18
+
19
+ request_options: Optional[RequestOptions] = None
20
+
21
+ class Outputs(BaseOutputs):
22
+ results: List[PromptOutput]
23
+
24
+ @abstractmethod
25
+ def _get_prompt_event_stream(self) -> Union[Iterator[AdHocExecutePromptEvent], Iterator[ExecutePromptEvent]]:
26
+ pass
27
+
28
+ def run(self) -> Iterator[BaseOutput]:
29
+ outputs = yield from self._process_prompt_event_stream()
30
+ if outputs is None:
31
+ raise NodeException(
32
+ message="Expected to receive outputs from Prompt",
33
+ code=VellumErrorCode.INTERNAL_ERROR,
34
+ )
35
+
36
+ def _process_prompt_event_stream(self) -> Generator[BaseOutput, None, Optional[List[PromptOutput]]]:
37
+ prompt_event_stream = self._get_prompt_event_stream()
38
+
39
+ outputs: Optional[List[PromptOutput]] = None
40
+ for event in prompt_event_stream:
41
+ if event.state == "INITIATED":
42
+ continue
43
+ elif event.state == "STREAMING":
44
+ yield BaseOutput(name="results", delta=event.output.value)
45
+ elif event.state == "FULFILLED":
46
+ outputs = event.outputs
47
+ yield BaseOutput(name="results", value=event.outputs)
48
+ elif event.state == "REJECTED":
49
+ if event.error.code in VellumErrorCode._value2member_map_:
50
+ raise NodeException(
51
+ message=event.error.message,
52
+ code=VellumErrorCode(event.error.code),
53
+ )
54
+ else:
55
+ raise NodeException(
56
+ message=event.error.message,
57
+ code=VellumErrorCode.INTERNAL_ERROR,
58
+ )
59
+
60
+ return outputs
@@ -0,0 +1,5 @@
1
+ from .node import BaseInlinePromptNode
2
+
3
+ __all__ = [
4
+ "BaseInlinePromptNode",
5
+ ]
@@ -0,0 +1,13 @@
1
+ from vellum import PromptParameters
2
+
3
+ DEFAULT_PROMPT_PARAMETERS = PromptParameters(
4
+ stop=[],
5
+ temperature=0.0,
6
+ max_tokens=4096,
7
+ top_p=1.0,
8
+ top_k=0,
9
+ frequency_penalty=0.0,
10
+ presence_penalty=0.0,
11
+ logit_bias=None,
12
+ custom_parameters=None,
13
+ )
@@ -0,0 +1,118 @@
1
+ from uuid import uuid4
2
+ from typing import ClassVar, Generic, Iterator, List, Optional, Tuple, cast
3
+
4
+ from vellum import (
5
+ AdHocExecutePromptEvent,
6
+ AdHocExpandMeta,
7
+ ChatMessage,
8
+ FunctionDefinition,
9
+ PromptBlock,
10
+ PromptParameters,
11
+ PromptRequestChatHistoryInput,
12
+ PromptRequestInput,
13
+ PromptRequestJsonInput,
14
+ PromptRequestStringInput,
15
+ VellumVariable,
16
+ )
17
+
18
+ from vellum.workflows.constants import OMIT
19
+ from vellum.workflows.errors import VellumErrorCode
20
+ from vellum.workflows.exceptions import NodeException
21
+ from vellum.workflows.nodes.displayable.bases.base_prompt_node import BasePromptNode
22
+ from vellum.workflows.nodes.displayable.bases.inline_prompt_node.constants import DEFAULT_PROMPT_PARAMETERS
23
+ from vellum.workflows.types.generics import StateType
24
+
25
+
26
+ class BaseInlinePromptNode(BasePromptNode, Generic[StateType]):
27
+ """
28
+ Used to execute a Prompt defined inline.
29
+
30
+ prompt_inputs: EntityInputsInterface - The inputs for the Prompt
31
+ ml_model: str - Either the ML Model's UUID or its name.
32
+ blocks: List[PromptBlock] - The blocks that make up the Prompt
33
+ functions: Optional[List[FunctionDefinition]] - The functions to include in the Prompt
34
+ parameters: PromptParameters - The parameters for the Prompt
35
+ expand_meta: Optional[AdHocExpandMeta] - Expandable execution fields to include in the response
36
+ request_options: Optional[RequestOptions] - The request options to use for the Prompt Execution
37
+ """
38
+
39
+ ml_model: ClassVar[str]
40
+
41
+ # The blocks that make up the Prompt
42
+ blocks: ClassVar[List[PromptBlock]]
43
+
44
+ # The functions/tools that a Prompt has access to
45
+ functions: Optional[List[FunctionDefinition]] = OMIT
46
+
47
+ parameters: PromptParameters = DEFAULT_PROMPT_PARAMETERS
48
+ expand_meta: Optional[AdHocExpandMeta] = OMIT
49
+
50
+ def _get_prompt_event_stream(self) -> Iterator[AdHocExecutePromptEvent]:
51
+ input_variables, input_values = self._compile_prompt_inputs()
52
+
53
+ return self._context.vellum_client.ad_hoc.adhoc_execute_prompt_stream(
54
+ ml_model=self.ml_model,
55
+ input_values=input_values,
56
+ input_variables=input_variables,
57
+ parameters=self.parameters,
58
+ blocks=self.blocks,
59
+ functions=self.functions,
60
+ expand_meta=self.expand_meta,
61
+ request_options=self.request_options,
62
+ )
63
+
64
+ def _compile_prompt_inputs(self) -> Tuple[List[VellumVariable], List[PromptRequestInput]]:
65
+ input_variables: List[VellumVariable] = []
66
+ input_values: List[PromptRequestInput] = []
67
+
68
+ for input_name, input_value in self.prompt_inputs.items():
69
+ if isinstance(input_value, str):
70
+ input_variables.append(
71
+ VellumVariable(
72
+ # TODO: Determine whether or not we actually need an id here and if we do,
73
+ # figure out how to maintain stable id references.
74
+ # https://app.shortcut.com/vellum/story/4080
75
+ id=str(uuid4()),
76
+ key=input_name,
77
+ type="STRING",
78
+ )
79
+ )
80
+ input_values.append(
81
+ PromptRequestStringInput(
82
+ key=input_name,
83
+ value=input_value,
84
+ )
85
+ )
86
+ elif isinstance(input_value, list) and all(isinstance(message, ChatMessage) for message in input_value):
87
+ input_variables.append(
88
+ VellumVariable(
89
+ # TODO: Determine whether or not we actually need an id here and if we do,
90
+ # figure out how to maintain stable id references.
91
+ # https://app.shortcut.com/vellum/story/4080
92
+ id=str(uuid4()),
93
+ key=input_name,
94
+ type="CHAT_HISTORY",
95
+ )
96
+ )
97
+ input_values.append(
98
+ PromptRequestChatHistoryInput(
99
+ key=input_name,
100
+ value=cast(List[ChatMessage], input_value),
101
+ )
102
+ )
103
+ elif isinstance(input_value, dict):
104
+ # Note: We may want to fail early here if we know that input_value is not
105
+ # JSON serializable.
106
+ input_values.append(
107
+ PromptRequestJsonInput(
108
+ key=input_name,
109
+ value=input_value,
110
+ )
111
+ )
112
+ else:
113
+ raise NodeException(
114
+ message=f"Unrecognized input type for input '{input_name}': {input_value.__class__}",
115
+ code=VellumErrorCode.INVALID_INPUTS,
116
+ )
117
+
118
+ return input_variables, input_values
@@ -0,0 +1,98 @@
1
+ from uuid import UUID
2
+ from typing import Any, ClassVar, Dict, Generic, Iterator, List, Optional, Sequence, Union, cast
3
+
4
+ from vellum import (
5
+ ChatHistoryInputRequest,
6
+ ChatMessage,
7
+ ExecutePromptEvent,
8
+ JsonInputRequest,
9
+ PromptDeploymentExpandMetaRequest,
10
+ PromptDeploymentInputRequest,
11
+ RawPromptExecutionOverridesRequest,
12
+ StringInputRequest,
13
+ )
14
+
15
+ from vellum.workflows.constants import LATEST_RELEASE_TAG, OMIT
16
+ from vellum.workflows.errors import VellumErrorCode
17
+ from vellum.workflows.exceptions import NodeException
18
+ from vellum.workflows.nodes.displayable.bases.base_prompt_node import BasePromptNode
19
+ from vellum.workflows.types.generics import StateType
20
+
21
+
22
+ class BasePromptDeploymentNode(BasePromptNode, Generic[StateType]):
23
+ """
24
+ Used to execute a Prompt Deployment.
25
+
26
+ prompt_inputs: EntityInputsInterface - The inputs for the Prompt
27
+ deployment: Union[UUID, str] - Either the Prompt Deployment's UUID or its name.
28
+ release_tag: str - The release tag to use for the Prompt Execution
29
+ external_id: Optional[str] - The external ID to use for the Prompt Execution
30
+ expand_meta: Optional[PromptDeploymentExpandMetaRequest] - Expandable execution fields to include in the response
31
+ raw_overrides: Optional[RawPromptExecutionOverridesRequest] - The raw overrides to use for the Prompt Execution
32
+ expand_raw: Optional[Sequence[str]] - Expandable raw fields to include in the response
33
+ metadata: Optional[Dict[str, Optional[Any]]] - The metadata to use for the Prompt Execution
34
+ request_options: Optional[RequestOptions] - The request options to use for the Prompt Execution
35
+ """
36
+
37
+ # Either the Prompt Deployment's UUID or its name.
38
+ deployment: ClassVar[Union[UUID, str]]
39
+
40
+ release_tag: str = LATEST_RELEASE_TAG
41
+ external_id: Optional[str] = OMIT
42
+
43
+ expand_meta: Optional[PromptDeploymentExpandMetaRequest] = OMIT
44
+ raw_overrides: Optional[RawPromptExecutionOverridesRequest] = OMIT
45
+ expand_raw: Optional[Sequence[str]] = OMIT
46
+ metadata: Optional[Dict[str, Optional[Any]]] = OMIT
47
+
48
+ def _get_prompt_event_stream(self) -> Iterator[ExecutePromptEvent]:
49
+ return self._context.vellum_client.execute_prompt_stream(
50
+ inputs=self._compile_prompt_inputs(),
51
+ prompt_deployment_id=str(self.deployment) if isinstance(self.deployment, UUID) else None,
52
+ prompt_deployment_name=self.deployment if isinstance(self.deployment, str) else None,
53
+ release_tag=self.release_tag,
54
+ external_id=self.external_id,
55
+ expand_meta=self.expand_meta,
56
+ raw_overrides=self.raw_overrides,
57
+ expand_raw=self.expand_raw,
58
+ metadata=self.metadata,
59
+ request_options=self.request_options,
60
+ )
61
+
62
+ def _compile_prompt_inputs(self) -> List[PromptDeploymentInputRequest]:
63
+ # TODO: We may want to consolidate with subworkflow deployment input compilation
64
+ # https://app.shortcut.com/vellum/story/4117
65
+
66
+ compiled_inputs: List[PromptDeploymentInputRequest] = []
67
+
68
+ for input_name, input_value in self.prompt_inputs.items():
69
+ if isinstance(input_value, str):
70
+ compiled_inputs.append(
71
+ StringInputRequest(
72
+ name=input_name,
73
+ value=input_value,
74
+ )
75
+ )
76
+ elif isinstance(input_value, list) and all(isinstance(message, ChatMessage) for message in input_value):
77
+ compiled_inputs.append(
78
+ ChatHistoryInputRequest(
79
+ name=input_name,
80
+ value=cast(List[ChatMessage], input_value),
81
+ )
82
+ )
83
+ elif isinstance(input_value, dict):
84
+ # Note: We may want to fail early here if we know that input_value is not
85
+ # JSON serializable.
86
+ compiled_inputs.append(
87
+ JsonInputRequest(
88
+ name=input_name,
89
+ value=cast(Dict[str, Any], input_value),
90
+ )
91
+ )
92
+ else:
93
+ raise NodeException(
94
+ message=f"Unrecognized input type for input '{input_name}': {input_value.__class__}",
95
+ code=VellumErrorCode.INVALID_INPUTS,
96
+ )
97
+
98
+ return compiled_inputs