| | import time |
| | import uuid |
| | from os import getenv |
| | from typing import cast |
| |
|
| | import pytest |
| |
|
| | from core.app.entities.app_invoke_entities import InvokeFrom |
| | from core.workflow.entities.node_entities import NodeRunResult |
| | from core.workflow.entities.variable_pool import VariablePool |
| | from core.workflow.enums import SystemVariableKey |
| | from core.workflow.graph_engine.entities.graph import Graph |
| | from core.workflow.graph_engine.entities.graph_init_params import GraphInitParams |
| | from core.workflow.graph_engine.entities.graph_runtime_state import GraphRuntimeState |
| | from core.workflow.nodes.code.code_node import CodeNode |
| | from core.workflow.nodes.code.entities import CodeNodeData |
| | from models.enums import UserFrom |
| | from models.workflow import WorkflowNodeExecutionStatus, WorkflowType |
| | from tests.integration_tests.workflow.nodes.__mock.code_executor import setup_code_executor_mock |
| |
|
| | CODE_MAX_STRING_LENGTH = int(getenv("CODE_MAX_STRING_LENGTH", "10000")) |
| |
|
| |
|
| | def init_code_node(code_config: dict): |
| | graph_config = { |
| | "edges": [ |
| | { |
| | "id": "start-source-code-target", |
| | "source": "start", |
| | "target": "code", |
| | }, |
| | ], |
| | "nodes": [{"data": {"type": "start"}, "id": "start"}, code_config], |
| | } |
| |
|
| | graph = Graph.init(graph_config=graph_config) |
| |
|
| | init_params = GraphInitParams( |
| | tenant_id="1", |
| | app_id="1", |
| | workflow_type=WorkflowType.WORKFLOW, |
| | workflow_id="1", |
| | graph_config=graph_config, |
| | user_id="1", |
| | user_from=UserFrom.ACCOUNT, |
| | invoke_from=InvokeFrom.DEBUGGER, |
| | call_depth=0, |
| | ) |
| |
|
| | |
| | variable_pool = VariablePool( |
| | system_variables={SystemVariableKey.FILES: [], SystemVariableKey.USER_ID: "aaa"}, |
| | user_inputs={}, |
| | environment_variables=[], |
| | conversation_variables=[], |
| | ) |
| | variable_pool.add(["code", "123", "args1"], 1) |
| | variable_pool.add(["code", "123", "args2"], 2) |
| |
|
| | node = CodeNode( |
| | id=str(uuid.uuid4()), |
| | graph_init_params=init_params, |
| | graph=graph, |
| | graph_runtime_state=GraphRuntimeState(variable_pool=variable_pool, start_at=time.perf_counter()), |
| | config=code_config, |
| | ) |
| |
|
| | return node |
| |
|
| |
|
| | @pytest.mark.parametrize("setup_code_executor_mock", [["none"]], indirect=True) |
| | def test_execute_code(setup_code_executor_mock): |
| | code = """ |
| | def main(args1: int, args2: int) -> dict: |
| | return { |
| | "result": args1 + args2, |
| | } |
| | """ |
| | |
| | code = "\n".join([line[4:] for line in code.split("\n")]) |
| |
|
| | code_config = { |
| | "id": "code", |
| | "data": { |
| | "outputs": { |
| | "result": { |
| | "type": "number", |
| | }, |
| | }, |
| | "title": "123", |
| | "variables": [ |
| | { |
| | "variable": "args1", |
| | "value_selector": ["1", "123", "args1"], |
| | }, |
| | {"variable": "args2", "value_selector": ["1", "123", "args2"]}, |
| | ], |
| | "answer": "123", |
| | "code_language": "python3", |
| | "code": code, |
| | }, |
| | } |
| |
|
| | node = init_code_node(code_config) |
| | node.graph_runtime_state.variable_pool.add(["1", "123", "args1"], 1) |
| | node.graph_runtime_state.variable_pool.add(["1", "123", "args2"], 2) |
| |
|
| | |
| | result = node._run() |
| | assert isinstance(result, NodeRunResult) |
| | assert result.status == WorkflowNodeExecutionStatus.SUCCEEDED |
| | assert result.outputs is not None |
| | assert result.outputs["result"] == 3 |
| | assert result.error is None |
| |
|
| |
|
| | @pytest.mark.parametrize("setup_code_executor_mock", [["none"]], indirect=True) |
| | def test_execute_code_output_validator(setup_code_executor_mock): |
| | code = """ |
| | def main(args1: int, args2: int) -> dict: |
| | return { |
| | "result": args1 + args2, |
| | } |
| | """ |
| | |
| | code = "\n".join([line[4:] for line in code.split("\n")]) |
| |
|
| | code_config = { |
| | "id": "code", |
| | "data": { |
| | "outputs": { |
| | "result": { |
| | "type": "string", |
| | }, |
| | }, |
| | "title": "123", |
| | "variables": [ |
| | { |
| | "variable": "args1", |
| | "value_selector": ["1", "123", "args1"], |
| | }, |
| | {"variable": "args2", "value_selector": ["1", "123", "args2"]}, |
| | ], |
| | "answer": "123", |
| | "code_language": "python3", |
| | "code": code, |
| | }, |
| | } |
| |
|
| | node = init_code_node(code_config) |
| | node.graph_runtime_state.variable_pool.add(["1", "123", "args1"], 1) |
| | node.graph_runtime_state.variable_pool.add(["1", "123", "args2"], 2) |
| |
|
| | |
| | result = node._run() |
| | assert isinstance(result, NodeRunResult) |
| | assert result.status == WorkflowNodeExecutionStatus.FAILED |
| | assert result.error == "Output variable `result` must be a string" |
| |
|
| |
|
| | def test_execute_code_output_validator_depth(): |
| | code = """ |
| | def main(args1: int, args2: int) -> dict: |
| | return { |
| | "result": { |
| | "result": args1 + args2, |
| | } |
| | } |
| | """ |
| | |
| | code = "\n".join([line[4:] for line in code.split("\n")]) |
| |
|
| | code_config = { |
| | "id": "code", |
| | "data": { |
| | "outputs": { |
| | "string_validator": { |
| | "type": "string", |
| | }, |
| | "number_validator": { |
| | "type": "number", |
| | }, |
| | "number_array_validator": { |
| | "type": "array[number]", |
| | }, |
| | "string_array_validator": { |
| | "type": "array[string]", |
| | }, |
| | "object_validator": { |
| | "type": "object", |
| | "children": { |
| | "result": { |
| | "type": "number", |
| | }, |
| | "depth": { |
| | "type": "object", |
| | "children": { |
| | "depth": { |
| | "type": "object", |
| | "children": { |
| | "depth": { |
| | "type": "number", |
| | } |
| | }, |
| | } |
| | }, |
| | }, |
| | }, |
| | }, |
| | }, |
| | "title": "123", |
| | "variables": [ |
| | { |
| | "variable": "args1", |
| | "value_selector": ["1", "123", "args1"], |
| | }, |
| | {"variable": "args2", "value_selector": ["1", "123", "args2"]}, |
| | ], |
| | "answer": "123", |
| | "code_language": "python3", |
| | "code": code, |
| | }, |
| | } |
| |
|
| | node = init_code_node(code_config) |
| |
|
| | |
| | result = { |
| | "number_validator": 1, |
| | "string_validator": "1", |
| | "number_array_validator": [1, 2, 3, 3.333], |
| | "string_array_validator": ["1", "2", "3"], |
| | "object_validator": {"result": 1, "depth": {"depth": {"depth": 1}}}, |
| | } |
| |
|
| | node.node_data = cast(CodeNodeData, node.node_data) |
| |
|
| | |
| | node._transform_result(result, node.node_data.outputs) |
| |
|
| | |
| | result = { |
| | "number_validator": "1", |
| | "string_validator": 1, |
| | "number_array_validator": ["1", "2", "3", "3.333"], |
| | "string_array_validator": [1, 2, 3], |
| | "object_validator": {"result": "1", "depth": {"depth": {"depth": "1"}}}, |
| | } |
| |
|
| | |
| | with pytest.raises(ValueError): |
| | node._transform_result(result, node.node_data.outputs) |
| |
|
| | |
| | result = { |
| | "number_validator": 1, |
| | "string_validator": (CODE_MAX_STRING_LENGTH + 1) * "1", |
| | "number_array_validator": [1, 2, 3, 3.333], |
| | "string_array_validator": ["1", "2", "3"], |
| | "object_validator": {"result": 1, "depth": {"depth": {"depth": 1}}}, |
| | } |
| |
|
| | |
| | with pytest.raises(ValueError): |
| | node._transform_result(result, node.node_data.outputs) |
| |
|
| | |
| | result = { |
| | "number_validator": 1, |
| | "string_validator": "1", |
| | "number_array_validator": [1, 2, 3, 3.333] * 2000, |
| | "string_array_validator": ["1", "2", "3"], |
| | "object_validator": {"result": 1, "depth": {"depth": {"depth": 1}}}, |
| | } |
| |
|
| | |
| | with pytest.raises(ValueError): |
| | node._transform_result(result, node.node_data.outputs) |
| |
|
| |
|
| | def test_execute_code_output_object_list(): |
| | code = """ |
| | def main(args1: int, args2: int) -> dict: |
| | return { |
| | "result": { |
| | "result": args1 + args2, |
| | } |
| | } |
| | """ |
| | |
| | code = "\n".join([line[4:] for line in code.split("\n")]) |
| |
|
| | code_config = { |
| | "id": "code", |
| | "data": { |
| | "outputs": { |
| | "object_list": { |
| | "type": "array[object]", |
| | }, |
| | }, |
| | "title": "123", |
| | "variables": [ |
| | { |
| | "variable": "args1", |
| | "value_selector": ["1", "123", "args1"], |
| | }, |
| | {"variable": "args2", "value_selector": ["1", "123", "args2"]}, |
| | ], |
| | "answer": "123", |
| | "code_language": "python3", |
| | "code": code, |
| | }, |
| | } |
| |
|
| | node = init_code_node(code_config) |
| |
|
| | |
| | result = { |
| | "object_list": [ |
| | { |
| | "result": 1, |
| | }, |
| | { |
| | "result": 2, |
| | }, |
| | { |
| | "result": [1, 2, 3], |
| | }, |
| | ] |
| | } |
| |
|
| | node.node_data = cast(CodeNodeData, node.node_data) |
| |
|
| | |
| | node._transform_result(result, node.node_data.outputs) |
| |
|
| | |
| | result = { |
| | "object_list": [ |
| | { |
| | "result": 1, |
| | }, |
| | { |
| | "result": 2, |
| | }, |
| | { |
| | "result": [1, 2, 3], |
| | }, |
| | 1, |
| | ] |
| | } |
| |
|
| | |
| | with pytest.raises(ValueError): |
| | node._transform_result(result, node.node_data.outputs) |
| |
|