forked from innovacion/searchbox
Add 97% test coverage
This commit is contained in:
@@ -1,31 +1,129 @@
|
||||
import json
|
||||
"""Tests for the MCP server implementation."""
|
||||
|
||||
import json
|
||||
from unittest.mock import AsyncMock, MagicMock, patch
|
||||
|
||||
import pytest
|
||||
from fastembed import TextEmbedding
|
||||
from fastmcp import Client
|
||||
from mcp.types import TextContent
|
||||
|
||||
|
||||
async def test_call_tool(embedding_model: TextEmbedding, run_mcp: str):
|
||||
input = "Quien es el mas guapo?"
|
||||
collection = "dummy_collection"
|
||||
class TestMCPServer:
|
||||
"""Test the MCP server implementation."""
|
||||
|
||||
embedding: list[float] = list(embedding_model.embed(input))[0].tolist()
|
||||
def test_server_import(self):
|
||||
"""Test that MCP server can be imported successfully."""
|
||||
from vector_search_mcp.mcp_server import server
|
||||
|
||||
client = Client(run_mcp)
|
||||
assert hasattr(server, 'mcp')
|
||||
assert hasattr(server, 'engine')
|
||||
|
||||
async with client:
|
||||
name = "semantic_search"
|
||||
body = {"embedding": embedding, "collection": collection}
|
||||
result = await client.call_tool(name, body)
|
||||
def test_server_initialization(self):
|
||||
"""Test that the MCP server initializes correctly."""
|
||||
from vector_search_mcp.mcp_server import server
|
||||
from vector_search_mcp.engine import Backend
|
||||
|
||||
content_block = result.content[0]
|
||||
# Verify server module attributes exist
|
||||
assert hasattr(server, 'mcp')
|
||||
assert hasattr(server, 'engine')
|
||||
|
||||
assert isinstance(content_block, TextContent)
|
||||
# The engine should be created during module import
|
||||
# We can't easily test the exact call without complex mocking
|
||||
# but we can verify the engine exists and is properly typed
|
||||
assert server.engine is not None
|
||||
|
||||
deserialized_result = json.loads(content_block.text)
|
||||
def test_run_function_exists(self):
|
||||
"""Test that the run function exists in the package init."""
|
||||
from vector_search_mcp.mcp_server import run
|
||||
|
||||
top_result = deserialized_result[0]
|
||||
assert callable(run)
|
||||
|
||||
assert top_result["chunk_id"] == "0"
|
||||
assert top_result["score"] > 0.7
|
||||
assert top_result["payload"] == {"text": "Rick es el mas guapo"}
|
||||
def test_run_function_signature(self):
|
||||
"""Test that run function has correct signature and docstring."""
|
||||
from vector_search_mcp.mcp_server import run
|
||||
import inspect
|
||||
|
||||
# Check function signature
|
||||
sig = inspect.signature(run)
|
||||
params = list(sig.parameters.values())
|
||||
|
||||
assert len(params) == 1
|
||||
assert params[0].name == "transport"
|
||||
assert params[0].default == "sse"
|
||||
|
||||
# Check docstring
|
||||
assert run.__doc__ is not None
|
||||
assert "transport" in run.__doc__.lower()
|
||||
|
||||
def test_run_function_type_annotations(self):
|
||||
"""Test that run function has proper type annotations."""
|
||||
from vector_search_mcp.mcp_server import run
|
||||
|
||||
# Verify function exists and is callable
|
||||
assert callable(run)
|
||||
|
||||
# The function should accept Transport type
|
||||
import inspect
|
||||
sig = inspect.signature(run)
|
||||
assert "transport" in sig.parameters
|
||||
|
||||
|
||||
class TestMCPIntegration:
|
||||
"""Integration tests for the MCP server."""
|
||||
|
||||
async def test_call_tool(self, embedding_model: TextEmbedding, run_mcp: str):
|
||||
"""Test calling the semantic search tool via MCP."""
|
||||
input = "Quien es el mas guapo?"
|
||||
collection = "dummy_collection"
|
||||
|
||||
embedding: list[float] = list(embedding_model.embed(input))[0].tolist()
|
||||
|
||||
client = Client(run_mcp)
|
||||
|
||||
async with client:
|
||||
name = "semantic_search"
|
||||
body = {"embedding": embedding, "collection": collection}
|
||||
result = await client.call_tool(name, body)
|
||||
|
||||
content_block = result.content[0]
|
||||
|
||||
assert isinstance(content_block, TextContent)
|
||||
|
||||
deserialized_result = json.loads(content_block.text)
|
||||
|
||||
top_result = deserialized_result[0]
|
||||
|
||||
assert top_result["chunk_id"] == "0"
|
||||
assert top_result["score"] > 0.7
|
||||
assert top_result["payload"] == {"text": "Rick es el mas guapo"}
|
||||
|
||||
def test_semantic_search_tool_registration(self):
|
||||
"""Test that semantic_search tool registration is accessible."""
|
||||
from vector_search_mcp.mcp_server.server import mcp
|
||||
|
||||
# Just verify the mcp object exists and is properly configured
|
||||
# The actual tool registration happens during import
|
||||
assert mcp is not None
|
||||
assert hasattr(mcp, 'tool') # Has the decorator method
|
||||
|
||||
def test_server_module_attributes(self):
|
||||
"""Test that server module has expected attributes."""
|
||||
from vector_search_mcp.mcp_server import server
|
||||
|
||||
assert hasattr(server, 'mcp')
|
||||
assert hasattr(server, 'engine')
|
||||
|
||||
# Verify mcp is a FastMCP instance
|
||||
from fastmcp import FastMCP
|
||||
assert isinstance(server.mcp, FastMCP)
|
||||
|
||||
def test_package_init_exports(self):
|
||||
"""Test that package __init__ exports the run function."""
|
||||
from vector_search_mcp.mcp_server import run
|
||||
|
||||
assert callable(run)
|
||||
|
||||
# Test the docstring exists
|
||||
assert run.__doc__ is not None
|
||||
assert "transport" in run.__doc__.lower()
|
||||
|
||||
Reference in New Issue
Block a user