Model Context Protocol (MCP) server and client for the Lauren framework
Project description
lauren-mcp
Model Context Protocol server and client for Lauren applications — expose any Lauren service as an MCP tool endpoint, and wire remote MCP tools into a Lauren AI agent in a single line.
Documentation: https://lauren-framework.github.io/lauren-mcp/
Source Code: https://github.com/lauren-framework/lauren-mcp
For AI Agents & Coding Assistants
Install all skills in one command
# Claude Code, Cursor, Copilot, Continue, Codex CLI — auto-detected
npx skills add lauren-framework/lauren-mcp
This copies all SKILL.md context packs into your agent's global skills
directory (~/.claude/skills/, ~/.cursor/skills/, etc.). The next time your
agent opens a Lauren project it has pre-loaded expertise on wiring MCP servers,
consuming remote MCP tools, schema generation, transport configuration, and more.
| Resource | What it contains |
|---|---|
llms.txt |
2 KB package overview — start here |
llms-full.txt |
Complete API reference — all 60+ symbols, signatures, common errors |
AGENTS.md |
Agent rules, by-task lookup, file ownership, common errors, definition of done |
CLAUDE.md |
Conventions, commands, golden rules |
skills/ |
Copy-paste skill guides for common tasks |
Installation
| Command | What you get |
|---|---|
pip install lauren-mcp |
Core: wire types + server decorators + stdio client |
pip install "lauren-mcp[ws]" |
+ WebSocket client (websockets) |
pip install "lauren-mcp[http]" |
+ HTTP+SSE client (httpx + httpx-sse) |
pip install "lauren-mcp[pydantic]" |
+ Pydantic model schemas (pydantic>=2) |
pip install "lauren-mcp[msgspec]" |
+ msgspec.Struct schemas (msgspec) |
pip install "lauren-mcp[cli]" |
+ lmcp CLI (typer + uvicorn) |
pip install "lauren-mcp[otel]" |
+ OpenTelemetry tracing (opentelemetry-api) |
pip install "lauren-mcp[all]" |
Everything |
MCP protocol versions
| Version | Transport | Status |
|---|---|---|
| 2024-11-05 | Legacy SSE | Supported |
| 2025-03-26 | Streamable HTTP | Supported |
| 2025-06-18 | Streamable HTTP | Supported |
| 2025-11-25 | Streamable HTTP | Supported (default) |
Server features
@mcp_server(path, transport="ws")— transport options:"ws","sse","streamable","both","all"@mcp_tool(title=, annotations=ToolAnnotations(...), timeout=30.0, tags={"search"}, output_schema=MyModel, structured_output=True)@mcp_resource(uri_template, mime_type=...)with RFC 6570 extensions ({+path},{?page,size})@mcp_prompt(title=),@mcp_completion(target, argument),@mcp_lifespanMcpToolContextinjection —ctx.report_progress(),ctx.log/debug/info/warning/error(),ctx.sample(),ctx.elicit(),ctx.elicit_url(),ctx.cancel_requested- Rich schema generation: Pydantic, msgspec.Struct,
@dataclass, TypedDict, Literal,Annotated+Field - Binary resources:
bytesreturn → auto base64 blob;BlobResource,ResourceResult - Dynamic catalog:
listChanged: True, autonotifications/tools/list_changed - Server composition:
mounts=[(OtherCls, "prefix_")],proxies=[(client, "prefix_")] - OpenAPI import:
build_openapi_server_class(spec, http_client=...) - Built-in resource types:
FileResource,HttpResource,DirectoryResource - Per-tool
@use_guards(G)for method-level access control (e.g. admin-only tools) - Per-tool
@use_interceptors(I)for cross-cutting concerns (audit, caching, timing) - Per-tool
@use_exception_handlers(H)to map domain exceptions toisError: True @set_metadata(key, value)per tool for guard-readable configuration- Class-level guards, interceptors, and middleware via Lauren pipeline —
@use_guards,@use_interceptors - DNS rebinding protection:
TransportSecuritySettings - SSE event store:
InMemoryEventStorefor resumable connections - OpenTelemetry tracing:
instrument_otel=Trueonfor_root() - CLI:
lmcp run,lmcp dev,lmcp inspect,lmcp call,lmcp install
Client features
McpServer.stdio(),McpServer.ws(),McpServer.http()(legacy SSE),McpServer.streamable_http()(MCP 2025-03-26+)- All factories accept:
protocol_version=,roots=,progress_handler=,log_handler=,list_changed_handler=,sampling_handler=,elicitation_handler=,resource_updated_handler= client.protocol_versionproperty afterconnect()client.on_progress/on_log/on_list_changed/on_resource_updated()return an unsubscribe callableclient.subscribe_resource(uri),client.unsubscribe_resource(uri)client.set_logging_level(level)— 8 severity levelsclient.complete(ref, argument)ClientCredentialsProviderfor OAuth client credentials flow
Per-tool decorators
Lauren's guard, interceptor, and exception-handler decorators can be applied to
individual @mcp_tool / @mcp_resource / @mcp_prompt methods.
@mcp_tool() must be the outermost decorator; Lauren decorators go inside
(closer to async def):
from lauren import use_guards, use_interceptors, use_exception_handlers, set_metadata
from lauren import injectable
from lauren_mcp import mcp_server, mcp_tool, McpToolContext, McpExecutionContext
@injectable()
class AdminGuard:
async def can_activate(self, ctx: McpExecutionContext) -> bool:
return ctx.headers.get("x-role") == "admin" if ctx.headers else False
@mcp_server("/mcp")
class AdminServer:
@set_metadata("required_role", "admin")
@use_guards(AdminGuard)
@mcp_tool()
async def purge_cache(self, ctx: McpToolContext) -> dict:
"""Purge the application cache (admin only)."""
return {"purged": True}
Calling purge_cache without the x-role: admin header returns
INTERNAL_ERROR with data.type="FORBIDDEN". Use McpForbiddenError and
McpExecutionContext (both in lauren_mcp) to inspect guard rejections.
Quick start — Server
from lauren_mcp import (
mcp_server, mcp_tool, McpToolContext, McpToolNameCollision,
McpServerModule, ToolAnnotations, BlobResource,
)
from lauren_mcp.server import mcp_lifespan
from lauren import LaurenFactory, module
@mcp_server("/mcp")
class CatalogueServer:
@mcp_lifespan
async def lifespan(self):
db = await connect_db()
try:
yield {"db": db}
finally:
await db.close()
@mcp_tool(
annotations=ToolAnnotations(readOnlyHint=True),
timeout=30.0,
tags={"search"},
)
async def search(self, query: str, limit: int = 10, ctx: McpToolContext = ...) -> list:
"""Search items.
Args:
query: Search terms.
limit: Max results.
"""
await ctx.report_progress(0, total=100, message="Starting search")
db = ctx.lifespan_context["db"]
results = await db.search(query, limit)
await ctx.info("Search complete", {"count": len(results)})
return results
@mcp_resource("/img/{name}", mime_type="image/png")
async def image(self, name: str) -> bytes:
return open(f"images/{name}.png", "rb").read()
@module(imports=[McpServerModule.for_root(CatalogueServer, transport="all")])
class App:
pass
app = LaurenFactory.create(App)
# WebSocket: ws://localhost:8000/mcp/ws
# Streamable HTTP: http://localhost:8000/mcp/
Quick start — Client
from lauren_mcp import McpServer
client = McpServer.streamable_http(
"http://localhost:8000/mcp",
progress_handler=lambda p: print(f"Progress: {p['progress']} — {p.get('message', '')}"),
log_handler=lambda p: print(f"[{p['level']}] {p['data']['message']}"),
)
await client.connect()
print(f"Protocol: {client.protocol_version}") # "2025-11-25"
tools = await client.list_tools()
result = await client.call_tool("search", {"query": "coffee", "limit": 5})
Stdio subprocess client
from lauren_mcp import McpServer, McpServerConfig
config = McpServerConfig(
alias="fs",
client=McpServer.stdio(
["npx", "-y", "@modelcontextprotocol/server-filesystem", "/tmp"]
),
)
# Tools available as: fs__read_file, fs__write_file, fs__list_directory, ...
Examples
The examples/filesystem/ directory contains a fully-working end-to-end example:
| File | Description |
|---|---|
server.py |
Lauren-based Filesystem MCP server (Streamable HTTP) |
client.py |
Interactive CLI client powered by the Poolside inference backend (OpenAI-compatible), with Rich UI for pretty-printing tool calls and results |
pyproject.toml |
Self-contained project with [client] and [deploy] optional extras |
.env.example |
Environment variable reference (OPENAI_API_KEY, OPENAI_API_BASE_URL, MCP_SERVER_URL, etc.) |
# 1. Start the server
MCP_FS_ROOT=/tmp/sandbox uv run python examples/filesystem/server.py
# 2. Run the interactive client (separate terminal)
OPENAI_API_KEY=<key> uv run --extra client python examples/filesystem/client.py
Documentation
API summary
# Decorators
mcp_server(path, transport="ws", name=, version=, description=)
mcp_tool(title=, annotations=, timeout=, tags=, meta=, output_schema=, structured_output=)
mcp_resource(uri_template, mime_type=, title=, annotations=)
mcp_prompt(title=)
mcp_lifespan
mcp_completion(target, argument)
# Module
McpServerModule.for_root(
server_cls,
transport="ws", # "ws" | "sse" | "streamable" | "both" | "all"
log_level=,
mounts=, # [(OtherServerCls, "prefix_")]
proxies=, # [(McpClientProtocol, "prefix_")]
instrument_otel=False,
event_store=,
stateless_http=False,
)
# Client factories (all accept protocol_version=, roots=, progress_handler=,
# log_handler=, list_changed_handler=, sampling_handler=,
# elicitation_handler=, resource_updated_handler=)
McpServer.stdio(command, env=, max_retries=)
McpServer.ws(url, ...)
McpServer.http(url, ...) # legacy SSE (MCP 2024-11-05)
McpServer.streamable_http(url, ...) # MCP 2025-03-26+
# Client methods
client.connect() / client.close()
client.protocol_version # property, set after connect()
client.list_tools() / list_resources() / list_prompts()
client.call_tool(name, arguments)
client.read_resource(uri)
client.get_prompt(name, arguments)
client.complete(ref, argument)
client.set_logging_level(level)
client.subscribe_resource(uri) / unsubscribe_resource(uri)
client.on_progress(handler) / on_log(handler)
client.on_list_changed(handler) / on_resource_updated(handler)
client.notify_roots_changed()
# Context (injected into @mcp_tool via ctx: McpToolContext parameter)
ctx.report_progress(progress, total=, message=)
ctx.log/debug/info/notice/warning/error/critical(message, data=)
ctx.sample(messages, model_preferences=, tools=, tool_choice=, max_tool_iterations=)
ctx.elicit(schema, message=)
ctx.elicit_url(url, message=)
ctx.cancel_requested # asyncio.Event
ctx.lifespan_context # dict yielded by @mcp_lifespan
# Wire types (selection)
ToolSchema, ToolAnnotations, ToolCallParams, ToolResult, ToolOutput
ResourceSchema, ResourceAnnotations, BlobResource, ResourceResult
PromptSchema, PromptMessage, PromptArgument
TextContent, ImageContent, AudioContent, EmbeddedResource
ResourceLink, ToolUseContent, ToolResultContent
SamplingMessage, CreateMessageParams, CreateMessageResult
ElicitResult, UrlElicitResult
CompletionResult, Root, Role
McpCallError, McpToolNameCollision
McpSamplingNotAvailable, McpElicitationNotAvailable, McpUrlElicitationNotAvailable
LATEST, STABLE, SUPPORTED
# OpenAPI import
build_openapi_server_class(spec, http_client=, route_entries=)
RouteEntry
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file lauren_mcp-1.0.0.tar.gz.
File metadata
- Download URL: lauren_mcp-1.0.0.tar.gz
- Upload date:
- Size: 140.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
4c584dbf923b243cbd3846236c7432518746af07d6d54f66dea26f9fa332e874
|
|
| MD5 |
11eb7cb3bf558cc97190c1b8975eb2be
|
|
| BLAKE2b-256 |
775fa293feb8fd3b98809d3999a66fc11393975ba49ebc5417a92c1ef32ba0ce
|
Provenance
The following attestation bundles were made for lauren_mcp-1.0.0.tar.gz:
Publisher:
release.yml on lauren-framework/lauren-mcp
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
lauren_mcp-1.0.0.tar.gz -
Subject digest:
4c584dbf923b243cbd3846236c7432518746af07d6d54f66dea26f9fa332e874 - Sigstore transparency entry: 1818506097
- Sigstore integration time:
-
Permalink:
lauren-framework/lauren-mcp@a5ab99b124f457f5d8805ff86fd6dfdf2d22471c -
Branch / Tag:
refs/tags/v1.0.0 - Owner: https://github.com/lauren-framework
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@a5ab99b124f457f5d8805ff86fd6dfdf2d22471c -
Trigger Event:
push
-
Statement type:
File details
Details for the file lauren_mcp-1.0.0-py3-none-any.whl.
File metadata
- Download URL: lauren_mcp-1.0.0-py3-none-any.whl
- Upload date:
- Size: 142.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
21553a071d284e144861653e25bf57bbef3ad1f6d9225a796b6dd6e4d41001b9
|
|
| MD5 |
4c4915a523fc1a0f793a3b1a17090d1d
|
|
| BLAKE2b-256 |
8b2fcb82bcb1aa53e26ad40cae82c196bd5faa0b445dd8c5e51440d543f021a9
|
Provenance
The following attestation bundles were made for lauren_mcp-1.0.0-py3-none-any.whl:
Publisher:
release.yml on lauren-framework/lauren-mcp
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
lauren_mcp-1.0.0-py3-none-any.whl -
Subject digest:
21553a071d284e144861653e25bf57bbef3ad1f6d9225a796b6dd6e4d41001b9 - Sigstore transparency entry: 1818506108
- Sigstore integration time:
-
Permalink:
lauren-framework/lauren-mcp@a5ab99b124f457f5d8805ff86fd6dfdf2d22471c -
Branch / Tag:
refs/tags/v1.0.0 - Owner: https://github.com/lauren-framework
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@a5ab99b124f457f5d8805ff86fd6dfdf2d22471c -
Trigger Event:
push
-
Statement type: