2025-06-12 20:43:32 +08:00
|
|
|
# Copyright (c) 2025 Bytedance Ltd. and/or its affiliates
|
|
|
|
|
# SPDX-License-Identifier: MIT
|
|
|
|
|
|
|
|
|
|
import pytest
|
2025-07-12 14:43:25 +08:00
|
|
|
from unittest.mock import patch
|
2025-06-12 20:43:32 +08:00
|
|
|
from src.tools.python_repl import python_repl_tool
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class TestPythonReplTool:
|
|
|
|
|
|
|
|
|
|
@patch("src.tools.python_repl.repl")
|
|
|
|
|
@patch("src.tools.python_repl.logger")
|
|
|
|
|
def test_successful_code_execution(self, mock_logger, mock_repl):
|
|
|
|
|
# Arrange
|
|
|
|
|
code = "print('Hello, World!')"
|
|
|
|
|
expected_output = "Hello, World!\n"
|
|
|
|
|
mock_repl.run.return_value = expected_output
|
|
|
|
|
|
|
|
|
|
# Act
|
|
|
|
|
result = python_repl_tool(code)
|
|
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
|
mock_repl.run.assert_called_once_with(code)
|
|
|
|
|
mock_logger.info.assert_called_with("Code execution successful")
|
|
|
|
|
assert "Successfully executed:" in result
|
|
|
|
|
assert code in result
|
|
|
|
|
assert expected_output in result
|
|
|
|
|
|
|
|
|
|
@patch("src.tools.python_repl.repl")
|
|
|
|
|
@patch("src.tools.python_repl.logger")
|
|
|
|
|
def test_invalid_input_type(self, mock_logger, mock_repl):
|
|
|
|
|
# Arrange
|
|
|
|
|
invalid_code = 123
|
|
|
|
|
|
|
|
|
|
# Act & Assert - expect ValidationError from LangChain
|
|
|
|
|
with pytest.raises(Exception) as exc_info:
|
|
|
|
|
python_repl_tool(invalid_code)
|
|
|
|
|
|
|
|
|
|
# Verify that it's a validation error
|
|
|
|
|
assert "ValidationError" in str(
|
|
|
|
|
type(exc_info.value)
|
|
|
|
|
) or "validation error" in str(exc_info.value)
|
|
|
|
|
|
|
|
|
|
# The REPL should not be called since validation fails first
|
|
|
|
|
mock_repl.run.assert_not_called()
|
|
|
|
|
|
|
|
|
|
@patch("src.tools.python_repl.repl")
|
|
|
|
|
@patch("src.tools.python_repl.logger")
|
|
|
|
|
def test_code_execution_with_error_in_result(self, mock_logger, mock_repl):
|
|
|
|
|
# Arrange
|
|
|
|
|
code = "invalid_function()"
|
|
|
|
|
error_result = "NameError: name 'invalid_function' is not defined"
|
|
|
|
|
mock_repl.run.return_value = error_result
|
|
|
|
|
|
|
|
|
|
# Act
|
|
|
|
|
result = python_repl_tool(code)
|
|
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
|
mock_repl.run.assert_called_once_with(code)
|
|
|
|
|
mock_logger.error.assert_called_with(error_result)
|
|
|
|
|
assert "Error executing code:" in result
|
|
|
|
|
assert code in result
|
|
|
|
|
assert error_result in result
|
|
|
|
|
|
|
|
|
|
@patch("src.tools.python_repl.repl")
|
|
|
|
|
@patch("src.tools.python_repl.logger")
|
|
|
|
|
def test_code_execution_with_exception_in_result(self, mock_logger, mock_repl):
|
|
|
|
|
# Arrange
|
|
|
|
|
code = "1/0"
|
|
|
|
|
exception_result = "ZeroDivisionError: division by zero"
|
|
|
|
|
mock_repl.run.return_value = exception_result
|
|
|
|
|
|
|
|
|
|
# Act
|
|
|
|
|
result = python_repl_tool(code)
|
|
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
|
mock_repl.run.assert_called_once_with(code)
|
|
|
|
|
mock_logger.error.assert_called_with(exception_result)
|
|
|
|
|
assert "Error executing code:" in result
|
|
|
|
|
assert code in result
|
|
|
|
|
assert exception_result in result
|
|
|
|
|
|
|
|
|
|
@patch("src.tools.python_repl.repl")
|
|
|
|
|
@patch("src.tools.python_repl.logger")
|
|
|
|
|
def test_code_execution_raises_exception(self, mock_logger, mock_repl):
|
|
|
|
|
# Arrange
|
|
|
|
|
code = "print('test')"
|
|
|
|
|
exception = RuntimeError("REPL failed")
|
|
|
|
|
mock_repl.run.side_effect = exception
|
|
|
|
|
|
|
|
|
|
# Act
|
|
|
|
|
result = python_repl_tool(code)
|
|
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
|
mock_repl.run.assert_called_once_with(code)
|
|
|
|
|
mock_logger.error.assert_called_with(repr(exception))
|
|
|
|
|
assert "Error executing code:" in result
|
|
|
|
|
assert code in result
|
|
|
|
|
assert repr(exception) in result
|
|
|
|
|
|
|
|
|
|
@patch("src.tools.python_repl.repl")
|
|
|
|
|
@patch("src.tools.python_repl.logger")
|
|
|
|
|
def test_successful_execution_with_calculation(self, mock_logger, mock_repl):
|
|
|
|
|
# Arrange
|
|
|
|
|
code = "result = 2 + 3\nprint(result)"
|
|
|
|
|
expected_output = "5\n"
|
|
|
|
|
mock_repl.run.return_value = expected_output
|
|
|
|
|
|
|
|
|
|
# Act
|
|
|
|
|
result = python_repl_tool(code)
|
|
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
|
mock_repl.run.assert_called_once_with(code)
|
|
|
|
|
mock_logger.info.assert_any_call("Executing Python code")
|
|
|
|
|
mock_logger.info.assert_any_call("Code execution successful")
|
|
|
|
|
assert "Successfully executed:" in result
|
|
|
|
|
assert code in result
|
|
|
|
|
assert expected_output in result
|
|
|
|
|
|
|
|
|
|
@patch("src.tools.python_repl.repl")
|
|
|
|
|
@patch("src.tools.python_repl.logger")
|
|
|
|
|
def test_empty_string_code(self, mock_logger, mock_repl):
|
|
|
|
|
# Arrange
|
|
|
|
|
code = ""
|
|
|
|
|
mock_repl.run.return_value = ""
|
|
|
|
|
|
|
|
|
|
# Act
|
|
|
|
|
result = python_repl_tool(code)
|
|
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
|
mock_repl.run.assert_called_once_with(code)
|
|
|
|
|
mock_logger.info.assert_called_with("Code execution successful")
|
|
|
|
|
assert "Successfully executed:" in result
|
|
|
|
|
|
|
|
|
|
@patch("src.tools.python_repl.repl")
|
|
|
|
|
@patch("src.tools.python_repl.logger")
|
|
|
|
|
def test_logging_calls(self, mock_logger, mock_repl):
|
|
|
|
|
# Arrange
|
|
|
|
|
code = "x = 1"
|
|
|
|
|
mock_repl.run.return_value = ""
|
|
|
|
|
|
|
|
|
|
# Act
|
|
|
|
|
python_repl_tool(code)
|
|
|
|
|
|
|
|
|
|
# Assert
|
|
|
|
|
mock_logger.info.assert_any_call("Executing Python code")
|
|
|
|
|
mock_logger.info.assert_any_call("Code execution successful")
|