mirror of
https://github.com/mistralai/mistral-vibe
synced 2026-04-25 17:14:55 +02:00
Co-authored-by: Clément Sirieix <clement.sirieix@mistral.ai> Co-authored-by: Kim-Adeline Miguel <kimadeline.miguel@mistral.ai> Co-authored-by: Lucas Marandat <31749711+lucasmrdt@users.noreply.github.com> Co-authored-by: Michel Thomazo <51709227+michelTho@users.noreply.github.com> Co-authored-by: Paul Cacheux <paul.cacheux@mistral.ai> Co-authored-by: Peter Evers <pevers90@gmail.com> Co-authored-by: Pierre Rossinès <pierre.rossines@mistral.ai> Co-authored-by: Pierre Rossinès <pierre.rossines@protonmail.com> Co-authored-by: Quentin <quentin.torroba@mistral.ai> Co-authored-by: Simon Van de Kerckhove <simon.vandekerckhove@mistral.ai> Co-authored-by: Val <102326092+vdeva@users.noreply.github.com> Co-authored-by: Vincent G <10739306+VinceOPS@users.noreply.github.com> Co-authored-by: Mistral Vibe <vibe@mistral.ai>
278 lines
9.7 KiB
Python
278 lines
9.7 KiB
Python
from __future__ import annotations
|
|
|
|
from pathlib import Path
|
|
|
|
import pytest
|
|
|
|
from tests.mock.utils import collect_result
|
|
from vibe.acp.tools.builtins.write_file import AcpWriteFileState, WriteFile
|
|
from vibe.core.tools.base import ToolError
|
|
from vibe.core.tools.builtins.write_file import (
|
|
WriteFileArgs,
|
|
WriteFileConfig,
|
|
WriteFileResult,
|
|
)
|
|
from vibe.core.types import ToolCallEvent, ToolResultEvent
|
|
|
|
|
|
class MockClient:
|
|
def __init__(
|
|
self, write_error: Exception | None = None, file_exists: bool = False
|
|
) -> None:
|
|
self._write_error = write_error
|
|
self._file_exists = file_exists
|
|
self._write_text_file_called = False
|
|
self._session_update_called = False
|
|
self._last_write_params: dict[str, str] = {}
|
|
|
|
async def write_text_file(
|
|
self, content: str, path: str, session_id: str, **kwargs
|
|
) -> None:
|
|
self._write_text_file_called = True
|
|
self._last_write_params = {
|
|
"content": content,
|
|
"path": path,
|
|
"session_id": session_id,
|
|
}
|
|
|
|
if self._write_error:
|
|
raise self._write_error
|
|
|
|
async def session_update(self, session_id: str, update, **kwargs) -> None:
|
|
self._session_update_called = True
|
|
|
|
|
|
@pytest.fixture
|
|
def mock_client() -> MockClient:
|
|
return MockClient()
|
|
|
|
|
|
@pytest.fixture
|
|
def acp_write_file_tool(
|
|
mock_client: MockClient, tmp_path: Path, monkeypatch: pytest.MonkeyPatch
|
|
) -> WriteFile:
|
|
monkeypatch.chdir(tmp_path)
|
|
config = WriteFileConfig()
|
|
state = AcpWriteFileState.model_construct(
|
|
client=mock_client,
|
|
session_id="test_session_123",
|
|
tool_call_id="test_tool_call_456",
|
|
)
|
|
return WriteFile(config_getter=lambda: config, state=state)
|
|
|
|
|
|
class TestAcpWriteFileBasic:
|
|
def test_get_name(self) -> None:
|
|
assert WriteFile.get_name() == "write_file"
|
|
|
|
|
|
class TestAcpWriteFileExecution:
|
|
@pytest.mark.asyncio
|
|
async def test_run_success_new_file(
|
|
self, acp_write_file_tool: WriteFile, mock_client: MockClient, tmp_path: Path
|
|
) -> None:
|
|
test_file = tmp_path / "test_file.txt"
|
|
args = WriteFileArgs(path=str(test_file), content="Hello, world!")
|
|
result = await collect_result(acp_write_file_tool.run(args))
|
|
|
|
assert isinstance(result, WriteFileResult)
|
|
assert result.path == str(test_file)
|
|
assert result.content == "Hello, world!"
|
|
assert result.bytes_written == len(b"Hello, world!")
|
|
assert result.file_existed is False
|
|
assert mock_client._write_text_file_called
|
|
assert mock_client._session_update_called
|
|
|
|
# Verify write_text_file was called correctly
|
|
params = mock_client._last_write_params
|
|
assert params["session_id"] == "test_session_123"
|
|
assert params["path"] == str(test_file)
|
|
assert params["content"] == "Hello, world!"
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_run_success_overwrite(
|
|
self, mock_client: MockClient, tmp_path: Path, monkeypatch: pytest.MonkeyPatch
|
|
) -> None:
|
|
monkeypatch.chdir(tmp_path)
|
|
tool = WriteFile(
|
|
config_getter=lambda: WriteFileConfig(),
|
|
state=AcpWriteFileState.model_construct(
|
|
client=mock_client, session_id="test_session", tool_call_id="test_call"
|
|
),
|
|
)
|
|
|
|
test_file = tmp_path / "existing_file.txt"
|
|
test_file.touch()
|
|
# Simulate existing file by checking in the core tool logic
|
|
# The ACP tool doesn't check existence, it's handled by the core tool
|
|
args = WriteFileArgs(path=str(test_file), content="New content", overwrite=True)
|
|
result = await collect_result(tool.run(args))
|
|
|
|
assert isinstance(result, WriteFileResult)
|
|
assert result.path == str(test_file)
|
|
assert result.content == "New content"
|
|
assert result.bytes_written == len(b"New content")
|
|
assert result.file_existed is True
|
|
assert mock_client._write_text_file_called
|
|
assert mock_client._session_update_called
|
|
|
|
# Verify write_text_file was called correctly
|
|
params = mock_client._last_write_params
|
|
assert params["session_id"] == "test_session"
|
|
assert params["path"] == str(test_file)
|
|
assert params["content"] == "New content"
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_run_write_error(
|
|
self, mock_client: MockClient, tmp_path: Path, monkeypatch: pytest.MonkeyPatch
|
|
) -> None:
|
|
monkeypatch.chdir(tmp_path)
|
|
mock_client._write_error = RuntimeError("Permission denied")
|
|
|
|
tool = WriteFile(
|
|
config_getter=lambda: WriteFileConfig(),
|
|
state=AcpWriteFileState.model_construct(
|
|
client=mock_client, session_id="test_session", tool_call_id="test_call"
|
|
),
|
|
)
|
|
|
|
test_file = tmp_path / "test.txt"
|
|
args = WriteFileArgs(path=str(test_file), content="test")
|
|
with pytest.raises(ToolError) as exc_info:
|
|
await collect_result(tool.run(args))
|
|
|
|
assert str(exc_info.value) == f"Error writing {test_file}: Permission denied"
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_run_without_connection(
|
|
self, tmp_path: Path, monkeypatch: pytest.MonkeyPatch
|
|
) -> None:
|
|
monkeypatch.chdir(tmp_path)
|
|
tool = WriteFile(
|
|
config_getter=lambda: WriteFileConfig(),
|
|
state=AcpWriteFileState.model_construct(
|
|
client=None, session_id="test_session", tool_call_id="test_call"
|
|
),
|
|
)
|
|
|
|
args = WriteFileArgs(path=str(tmp_path / "test.txt"), content="test")
|
|
with pytest.raises(ToolError) as exc_info:
|
|
await collect_result(tool.run(args))
|
|
|
|
assert (
|
|
str(exc_info.value)
|
|
== "Client not available in tool state. This tool can only be used within an ACP session."
|
|
)
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_run_without_session_id(
|
|
self, tmp_path: Path, monkeypatch: pytest.MonkeyPatch
|
|
) -> None:
|
|
monkeypatch.chdir(tmp_path)
|
|
mock_client = MockClient()
|
|
tool = WriteFile(
|
|
config_getter=lambda: WriteFileConfig(),
|
|
state=AcpWriteFileState.model_construct(
|
|
client=mock_client, session_id=None, tool_call_id="test_call"
|
|
),
|
|
)
|
|
|
|
args = WriteFileArgs(path=str(tmp_path / "test.txt"), content="test")
|
|
with pytest.raises(ToolError) as exc_info:
|
|
await collect_result(tool.run(args))
|
|
|
|
assert (
|
|
str(exc_info.value)
|
|
== "Session ID not available in tool state. This tool can only be used within an ACP session."
|
|
)
|
|
|
|
|
|
class TestAcpWriteFileSessionUpdates:
|
|
def test_tool_call_session_update(self) -> None:
|
|
event = ToolCallEvent(
|
|
tool_name="write_file",
|
|
tool_call_id="test_call_123",
|
|
args=WriteFileArgs(path="/tmp/test.txt", content="Hello"),
|
|
tool_class=WriteFile,
|
|
)
|
|
|
|
update = WriteFile.tool_call_session_update(event)
|
|
assert update is not None
|
|
assert update.session_update == "tool_call"
|
|
assert update.tool_call_id == "test_call_123"
|
|
assert update.kind == "edit"
|
|
assert update.title is not None
|
|
assert update.content is not None
|
|
assert isinstance(update.content, list)
|
|
assert len(update.content) == 1
|
|
assert update.content[0].type == "diff"
|
|
assert update.content[0].path == "/tmp/test.txt"
|
|
assert update.content[0].old_text is None
|
|
assert update.content[0].new_text == "Hello"
|
|
assert update.locations is not None
|
|
assert len(update.locations) == 1
|
|
assert update.locations[0].path == "/tmp/test.txt"
|
|
|
|
def test_tool_call_session_update_invalid_args(self) -> None:
|
|
from vibe.core.types import FunctionCall, ToolCall
|
|
|
|
class InvalidArgs:
|
|
pass
|
|
|
|
event = ToolCallEvent.model_construct(
|
|
tool_name="write_file",
|
|
tool_call_id="test_call_123",
|
|
args=InvalidArgs(), # type: ignore[arg-type]
|
|
tool_class=WriteFile,
|
|
llm_tool_call=ToolCall(
|
|
function=FunctionCall(name="write_file", arguments="{}"),
|
|
type="function",
|
|
index=0,
|
|
),
|
|
)
|
|
|
|
update = WriteFile.tool_call_session_update(event)
|
|
assert update is None
|
|
|
|
def test_tool_result_session_update(self) -> None:
|
|
result = WriteFileResult(
|
|
path="/tmp/test.txt", content="Hello", bytes_written=5, file_existed=False
|
|
)
|
|
|
|
event = ToolResultEvent(
|
|
tool_name="write_file",
|
|
tool_call_id="test_call_123",
|
|
result=result,
|
|
tool_class=WriteFile,
|
|
)
|
|
|
|
update = WriteFile.tool_result_session_update(event)
|
|
assert update is not None
|
|
assert update.session_update == "tool_call_update"
|
|
assert update.tool_call_id == "test_call_123"
|
|
assert update.status == "completed"
|
|
assert update.content is not None
|
|
assert isinstance(update.content, list)
|
|
assert len(update.content) == 1
|
|
assert update.content[0].type == "diff"
|
|
assert update.content[0].path == "/tmp/test.txt"
|
|
assert update.content[0].old_text is None
|
|
assert update.content[0].new_text == "Hello"
|
|
assert update.locations is not None
|
|
assert len(update.locations) == 1
|
|
assert update.locations[0].path == "/tmp/test.txt"
|
|
|
|
def test_tool_result_session_update_invalid_result(self) -> None:
|
|
class InvalidResult:
|
|
pass
|
|
|
|
event = ToolResultEvent.model_construct(
|
|
tool_name="write_file",
|
|
tool_call_id="test_call_123",
|
|
result=InvalidResult(), # type: ignore[arg-type]
|
|
tool_class=WriteFile,
|
|
)
|
|
|
|
update = WriteFile.tool_result_session_update(event)
|
|
assert update is None
|