mcp-dice

MIT License
1
  • Apple
import json import logging import random import re from collections.abc import Sequence from datetime import datetime from typing import Any, Tuple from mcp.server import Server from mcp.types import Resource, Tool, TextContent, ImageContent, EmbeddedResource from pydantic import AnyUrl # Configure logging logging.basicConfig(level=logging.INFO) logger = logging.getLogger("mcp-dice") # Default dice configuration DEFAULT_ROLL = "2d6" def parse_dice_notation(notation: str) -> Tuple[int, int, int]: """ Parse dice notation (e.g., '2d6+3') into number of dice, sides, and modifier. Args: notation: Dice notation string (e.g., '2d6+3', '1d20-2') Returns: Tuple of (number of dice, number of sides, modifier) """ # Updated pattern to support optional modifier pattern = re.compile(r"^(\d+)d(\d+)(?:([+-]\d+))?$") match = pattern.match(notation) if not match: raise ValueError(f"Invalid dice notation: {notation}") n_dice, n_sides = map(int, match.groups()[:2]) # Parse modifier if present, default to 0 if not modifier_str = match.group(3) modifier = int(modifier_str) if modifier_str else 0 return n_dice, n_sides, modifier def roll_dice(n_dice: int, n_sides: int, modifier: int = 0) -> dict[str, Any]: """ Roll the specified number of dice with given sides and add modifier. Args: n_dice: Number of dice to roll n_sides: Number of sides on each die modifier: Numeric modifier to add to the total (can be negative) Returns: Dictionary containing roll results and metadata """ if n_dice < 1: raise ValueError("Number of dice must be positive") if n_sides < 2: raise ValueError("Number of sides must be at least 2") rolls = [random.randint(1, n_sides) for _ in range(n_dice)] roll_sum = sum(rolls) total = roll_sum + modifier # Format the notation with modifier notation = f"{n_dice}d{n_sides}" if modifier > 0: notation += f"+{modifier}" elif modifier < 0: notation += f"{modifier}" return { "rolls": rolls, "sum": roll_sum, "modifier": modifier, "total": total, "notation": notation, "timestamp": datetime.now().isoformat(), } app = Server("mcp-dice") @app.list_resources() async def list_resources() -> list[Resource]: """List available dice rolling resources.""" uri = AnyUrl(f"dice://{DEFAULT_ROLL}") return [ Resource( uri=uri, name=f"Random {DEFAULT_ROLL} roll", mimeType="application/json", description="Roll two six-sided dice", ) ] @app.read_resource() async def read_resource(uri: AnyUrl) -> str: """Read dice roll results.""" if not str(uri).startswith("dice://"): raise ValueError(f"Unknown resource: {uri}") try: # Extract dice notation from URI notation = str(uri).split("//")[-1] n_dice, n_sides, modifier = parse_dice_notation(notation) # Perform the roll roll_result = roll_dice(n_dice, n_sides, modifier) return json.dumps(roll_result, indent=2) except Exception as e: raise RuntimeError(f"Dice rolling error: {str(e)}") @app.list_tools() async def list_tools() -> list[Tool]: """List available dice rolling tools.""" return [ Tool( name="roll_dice", description="Roll dice using standard notation (e.g., '2d6+3', '1d20-2')", inputSchema={ "type": "object", "properties": { "notation": { "type": "string", "description": "Dice notation (e.g., '2d6+3', '1d20-2')", "pattern": r"^\d+d\d+([+-]\d+)?$", } }, "required": ["notation"], }, ) ] @app.call_tool() async def call_tool( name: str, arguments: Any ) -> Sequence[TextContent | ImageContent | EmbeddedResource]: """Handle tool calls for dice rolling.""" if name != "roll_dice": raise ValueError(f"Unknown tool: {name}") if not isinstance(arguments, dict) or "notation" not in arguments: raise ValueError("Invalid dice roll arguments") try: notation = arguments["notation"] n_dice, n_sides, modifier = parse_dice_notation(notation) roll_result = roll_dice(n_dice, n_sides, modifier) return [TextContent(type="text", text=json.dumps(roll_result, indent=2))] except Exception as e: logger.error(f"Dice rolling error: {str(e)}") raise RuntimeError(f"Dice rolling error: {str(e)}") async def main(): # Import here to avoid issues with event loops from mcp.server.stdio import stdio_server async with stdio_server() as (read_stream, write_stream): await app.run(read_stream, write_stream, app.create_initialization_options()) if __name__ == "__main__": import asyncio asyncio.run(main())