MCP Outline Server
by Vortiago
Verified
- mcp-outline
- tests
- features
- documents
"""
Tests for document collaboration tools.
"""
from unittest.mock import MagicMock, patch
import pytest
from mcp_outline.features.documents.common import OutlineClientError
from mcp_outline.features.documents.document_collaboration import (
_format_comments,
)
# Mock FastMCP for registering tools
class MockMCP:
def __init__(self):
self.tools = {}
def tool(self):
def decorator(func):
self.tools[func.__name__] = func
return func
return decorator
# Sample comment data
SAMPLE_COMMENTS = [
{
"id": "comment1",
"data": {"content": "This is a test comment"},
"createdAt": "2023-01-01T12:00:00Z",
"createdBy": {
"id": "user1",
"name": "Test User"
}
},
{
"id": "comment2",
"data": {"content": "Another comment"},
"createdAt": "2023-01-02T12:00:00Z",
"createdBy": {
"id": "user2",
"name": "Another User"
}
}
]
# Sample documents for backlinks
SAMPLE_BACKLINK_DOCUMENTS = [
{
"id": "doc1",
"title": "Referencing Document 1",
"updatedAt": "2023-01-01T12:00:00Z"
},
{
"id": "doc2",
"title": "Referencing Document 2",
"updatedAt": "2023-01-02T12:00:00Z"
}
]
@pytest.fixture
def mcp():
"""Fixture to provide mock MCP instance."""
return MockMCP()
@pytest.fixture
def register_collaboration_tools(mcp):
"""Fixture to register document collaboration tools."""
from mcp_outline.features.documents.document_collaboration import (
register_tools,
)
register_tools(mcp)
return mcp
class TestDocumentCollaborationFormatters:
"""Tests for document collaboration formatting functions."""
def test_format_comments_with_data(self):
"""Test formatting comments with valid data."""
result = _format_comments(SAMPLE_COMMENTS)
# Verify the result contains the expected information
assert "# Document Comments" in result
assert "Comment by Test User" in result
assert "This is a test comment" in result
assert "2023-01-01" in result
assert "Comment by Another User" in result
assert "Another comment" in result
def test_format_comments_empty(self):
"""Test formatting empty comments list."""
result = _format_comments([])
assert "No comments found for this document" in result
def test_format_comments_missing_fields(self):
"""Test formatting comments with missing fields."""
# Comments with missing fields
incomplete_comments = [
# Missing user name
{
"id": "comment1",
"data": {"content": "Comment with missing user"},
"createdAt": "2023-01-01T12:00:00Z",
"createdBy": {}
},
# Missing created date
{
"id": "comment2",
"data": {"content": "Comment with missing date"},
"createdBy": {
"name": "Test User"
}
},
# Missing text
{
"id": "comment3",
"createdAt": "2023-01-03T12:00:00Z",
"createdBy": {
"name": "Another User"
}
}
]
result = _format_comments(incomplete_comments)
# Verify the result handles missing fields gracefully
assert "Unknown User" in result
assert "Comment with missing user" in result
assert "Test User" in result
assert "Comment with missing date" in result
assert "Another User" in result
class TestDocumentCollaborationTools:
"""Tests for document collaboration tools."""
@patch("mcp_outline.features.documents.document_collaboration.get_outline_client")
def test_list_document_comments_success(
self, mock_get_client, register_collaboration_tools
):
"""Test list_document_comments tool success case."""
# Set up mock client
mock_client = MagicMock()
mock_client.post.return_value = {"data": SAMPLE_COMMENTS}
mock_get_client.return_value = mock_client
# Call the tool
result = register_collaboration_tools.tools[
"list_document_comments"]("doc123")
# Verify client was called correctly
mock_client.post.assert_called_once_with(
"comments.list", {
"documentId": "doc123",
"includeAnchorText": False,
"limit": 25,
"offset": 0
}
)
# Verify result contains expected information
assert "# Document Comments" in result
assert "Comment by Test User" in result
assert "This is a test comment" in result
@patch("mcp_outline.features.documents.document_collaboration.get_outline_client")
def test_list_document_comments_empty(
self, mock_get_client, register_collaboration_tools
):
"""Test list_document_comments tool with no comments."""
# Set up mock client with empty response
mock_client = MagicMock()
mock_client.post.return_value = {"data": []}
mock_get_client.return_value = mock_client
# Call the tool
result = register_collaboration_tools.tools[
"list_document_comments"]("doc123")
# Verify result contains expected message
assert "No comments found" in result
@patch("mcp_outline.features.documents.document_collaboration.get_outline_client")
def test_get_comment_success(
self, mock_get_client, register_collaboration_tools
):
"""Test get_comment tool success case."""
# Set up mock client
mock_client = MagicMock()
mock_client.post.return_value = {"data": SAMPLE_COMMENTS[0]}
mock_get_client.return_value = mock_client
# Call the tool
result = register_collaboration_tools.tools[
"get_comment"]("comment1")
# Verify client was called correctly
mock_client.post.assert_called_once_with(
"comments.info", {
"id": "comment1",
"includeAnchorText": False
}
)
# Verify result contains expected information
assert "# Comment by Test User" in result
assert "This is a test comment" in result
assert "2023-01-01" in result
@patch("mcp_outline.features.documents.document_collaboration.get_outline_client")
def test_get_comment_not_found(
self, mock_get_client, register_collaboration_tools
):
"""Test get_comment tool with comment not found."""
# Set up mock client with empty response
mock_client = MagicMock()
mock_client.post.return_value = {"data": {}}
mock_get_client.return_value = mock_client
# Call the tool
result = register_collaboration_tools.tools[
"get_comment"]("comment999")
# Verify result contains expected message
assert "Comment not found" in result
@patch("mcp_outline.features.documents.document_collaboration.get_outline_client")
def test_get_document_backlinks_success(
self, mock_get_client, register_collaboration_tools
):
"""Test get_document_backlinks tool success case."""
# Set up mock client
mock_client = MagicMock()
mock_client.post.return_value = {"data": SAMPLE_BACKLINK_DOCUMENTS}
mock_get_client.return_value = mock_client
# Call the tool
result = register_collaboration_tools.tools[
"get_document_backlinks"]("doc123")
# Verify client was called correctly
mock_client.post.assert_called_once_with(
"documents.list", {"backlinkDocumentId": "doc123"}
)
# Verify result contains expected information
assert "# Documents Linking to This Document" in result
assert "Referencing Document 1" in result
assert "doc1" in result
assert "Referencing Document 2" in result
assert "doc2" in result
@patch("mcp_outline.features.documents.document_collaboration.get_outline_client")
def test_get_document_backlinks_none(
self, mock_get_client, register_collaboration_tools
):
"""Test get_document_backlinks tool with no backlinks."""
# Set up mock client with empty response
mock_client = MagicMock()
mock_client.post.return_value = {"data": []}
mock_get_client.return_value = mock_client
# Call the tool
result = register_collaboration_tools.tools[
"get_document_backlinks"]("doc123")
# Verify result contains expected message
assert "No documents link to this document" in result
@patch("mcp_outline.features.documents.document_collaboration.get_outline_client")
def test_get_document_backlinks_client_error(
self, mock_get_client, register_collaboration_tools
):
"""Test get_document_backlinks tool with client error."""
# Set up mock client to raise an error
mock_client = MagicMock()
mock_client.post.side_effect = OutlineClientError("API error")
mock_get_client.return_value = mock_client
# Call the tool
result = register_collaboration_tools.tools[
"get_document_backlinks"]("doc123")
# Verify error is handled and returned
assert "Error retrieving backlinks" in result
assert "API error" in result