Skip to content

Latest commit

 

History

History
217 lines (173 loc) · 7.38 KB

File metadata and controls

217 lines (173 loc) · 7.38 KB

Development Guidelines

This document contains critical information about working with this codebase. Follow these guidelines precisely.

Core Development Rules

  1. Package Management

    • ONLY use uv, NEVER pip
    • Installation: uv add <package>
    • Running tools: uv run <tool>
    • Upgrading: uv lock --upgrade-package <package>
    • FORBIDDEN: uv pip install, @latest syntax
  2. Code Quality

    • Type hints required for all code
    • Public APIs must have docstrings
    • Functions must be focused and small
    • Follow existing patterns exactly
    • Line length: 120 chars maximum
    • FORBIDDEN: imports inside functions. THEY SHOULD BE AT THE TOP OF THE FILE.
  3. Testing Requirements

    • Framework: uv run --frozen pytest
    • Async testing: use anyio, not asyncio
    • Do not use Test prefixed classes, use functions
    • Coverage: test edge cases and errors
    • New features require tests
    • Bug fixes require regression tests
    • IMPORTANT: The tests/client/test_client.py is the most well designed test file. Follow its patterns.
    • IMPORTANT: Be minimal, and focus on E2E tests: Use the mcp.client.Client whenever possible.
    • IMPORTANT: Before pushing, verify 100% branch coverage on changed files by running uv run --frozen pytest -x (coverage is configured in pyproject.toml with fail_under = 100 and branch = true). If any branch is uncovered, add a test for it before pushing.
    • Avoid anyio.sleep() with a fixed duration to wait for async operations. Instead:
      • Use anyio.Event — set it in the callback/handler, await event.wait() in the test
      • For stream messages, use await stream.receive() instead of sleep() + receive_nowait()
      • Exception: sleep() is appropriate when testing time-based features (e.g., timeouts)
    • Wrap indefinite waits (event.wait(), stream.receive()) in anyio.fail_after(5) to prevent hangs

Test files mirror the source tree: src/mcp/client/streamable_http.pytests/client/test_streamable_http.py Add tests to the existing file for that module.

  • For commits fixing bugs or adding features based on user reports add:

    git commit --trailer "Reported-by:<name>"

    Where <name> is the name of the user.

  • For commits related to a Github issue, add

    git commit --trailer "Github-Issue:#<number>"
  • NEVER ever mention a co-authored-by or similar aspects. In particular, never mention the tool used to create the commit message or PR.

Pull Requests

  • Create a detailed message of what changed. Focus on the high level description of the problem it tries to solve, and how it is solved. Don't go into the specifics of the code unless it adds clarity.

  • NEVER ever mention a co-authored-by or similar aspects. In particular, never mention the tool used to create the commit message or PR.

Breaking Changes

When making breaking changes, document them in docs/migration.md. Include:

  • What changed
  • Why it changed
  • How to migrate existing code

Search for related sections in the migration guide and group related changes together rather than adding new standalone sections.

Python Tools

Code Formatting

  1. Ruff

    • Format: uv run --frozen ruff format .
    • Check: uv run --frozen ruff check .
    • Fix: uv run --frozen ruff check . --fix
    • Critical issues:
      • Line length (88 chars)
      • Import sorting (I001)
      • Unused imports
    • Line wrapping:
      • Strings: use parentheses
      • Function calls: multi-line with proper indent
      • Imports: try to use a single line
  2. Type Checking

    • Tool: uv run --frozen pyright
    • Requirements:
      • Type narrowing for strings
      • Version warnings can be ignored if checks pass
  3. Pre-commit

    • Config: .pre-commit-config.yaml
    • Runs: on git commit
    • Tools: Prettier (YAML/JSON), Ruff (Python)
    • Ruff updates:
      • Check PyPI versions
      • Update config rev
      • Commit config first

Code Snippet System

scripts/sync_snippets.py replaces the content between <!-- snippet-source ... --> / <!-- /snippet-source --> markers with code from the referenced source file. The source file is the source of truth — never edit synced content directly in the target.

To sync only part of a file, append #RegionName to the path. Regions are delimited in source files by # region Name / # endregion Name markers. Each example lives in a named function (returning -> None) wrapping a region. Names follow ClassName_methodName_variant for methods, functionName_variant for standalone functions, or module_overview for module docstrings. Pick a descriptive variant suffix (_basic, _sync/_async, _with_context, etc.):

def MyClass_do_thing_basic(obj: MyClass) -> None:
    # region MyClass_do_thing_basic
    result = obj.do_thing("arg")
    print(result)
    # endregion MyClass_do_thing_basic

Function parameters supply typed dependencies the example needs but does not create (e.g., server: MCPServer); module-level stubs are only for truly undefined references (e.g., async def fetch_data() -> str: ...).

NEVER put # type: ignore, # pyright: ignore, or # noqa inside a region — these sync verbatim into the target. Restructure the code to address the errors instead.

After editing an example file, run uv run --frozen pyright to verify types, then uv run python scripts/sync_snippets.py to sync. Use --check to verify without modifying files.

Markdown Code Examples

Code examples in README.v2.md and docs/**/*.md use explicit paths relative to the repo root:

<!-- snippet-source examples/snippets/servers/basic_tool.py -->
```python
# replaced by sync script
```
<!-- /snippet-source -->

Docstring Code Examples

Code examples in src/ docstrings use companion files in examples/snippets/docstrings/, mirroring the source tree (src/mcp/foo/bar.pyexamples/snippets/docstrings/mcp/foo/bar.py). Companion files are standalone scripts (not packages) starting with from __future__ import annotations.

Docstrings use path-less #Region markers (only supported in src/ files):

    Example:
        <!-- snippet-source #MyClass_do_thing_basic -->
        ```python
        result = obj.do_thing("arg")
        print(result)
        ```
        <!-- /snippet-source -->

Error Resolution

  1. CI Failures

    • Fix order:
      1. Formatting
      2. Type errors
      3. Linting
    • Type errors:
      • Get full line context
      • Check Optional types
      • Add type narrowing
      • Verify function signatures
  2. Common Issues

    • Line length:
      • Break strings with parentheses
      • Multi-line function calls
      • Split imports
    • Types:
      • Add None checks
      • Narrow string types
      • Match existing patterns
  3. Best Practices

    • Check git status before commits
    • Run formatters before type checks
    • Keep changes minimal
    • Follow existing patterns
    • Document public APIs
    • Test thoroughly

Exception Handling

  • Always use logger.exception() instead of logger.error() when catching exceptions
    • Don't include the exception in the message: logger.exception("Failed") not logger.exception(f"Failed: {e}")
  • Catch specific exceptions where possible:
    • File ops: except (OSError, PermissionError):
    • JSON: except json.JSONDecodeError:
    • Network: except (ConnectionError, TimeoutError):
  • FORBIDDEN except Exception: - unless in top-level handlers