Shared FastMCP infrastructure: auth, middleware, logging, server-factory helpers
Project description
fastmcp-pvl-core
The opinionated shared implementation for the pvliesdonk/*-mcp
server family. fastmcp-pvl-core owns the shape of cross-cutting
concerns — auth, middleware, logging, config, and server-factory
builders — and exposes narrow hooks to downstream servers for
domain-specific behaviour. Downstream conforms
to the shape; pvl-core does not adapt to downstream preferences. See
Design principles for the rationale and the
classification test that follows from it.
Ecosystem
fastmcp-server-template— copier template that scaffolds new FastMCP servers on top of this library.- Active consumers:
markdown-vault-mcp,scholar-mcp,image-generation-mcp. - Public API changes here propagate to consumers via periodic
copier updateruns against the template. - See the template's README for the update flow and the expected project shape.
Design principles
fastmcp-pvl-core is not a buffet of helpers downstream picks from
à la carte. It is the load-bearing layer that fixes the shape of
cross-cutting concerns across the server family so the family stays
coherent as it grows. Five principles follow from that role.
Shape decisions live in pvl-core
Tool names, parameter shapes, route structures, capability declarations, error envelopes, environment-variable contracts — pvl-core picks one shape and downstream conforms. If two downstream servers would each prefer a different shape, the resolution is for pvl-core to pick one and migrate the others to it, not for pvl-core to grow an override kwarg.
Hooks expose domain-specific behaviour only
A hook like "where in my storage model do these bytes go?" is appropriate — pvl-core cannot know the answer for a particular downstream. A hook like "what should this tool be called?" or "what HTTP status code should an oversize body return?" is not — those are shape decisions pvl-core owns, and downstream accepts them.
The test for any proposed kwarg on a register_* helper, Build*
factory, or middleware constructor: would pvl-core be wrong to
make this decision itself? If pvl-core could pick a sensible value
and downstream has no domain-specific basis to disagree, pvl-core
picks it — no kwarg. If pvl-core literally cannot answer because
the answer is about the downstream's domain, the kwarg exists and is
not optional unless the entire feature is opt-in. There is no third
bucket of "pvl-core has a default but downstream can override."
Operator-side configuration (TTL ceilings, max body sizes, listening ports, debug flags) is a separate axis — environment variables, not kwargs. The kwarg surface is purely domain hooks.
If a proposed kwarg mixes the two — a legitimate hook bundled with an override of shape — split it: keep the hook, drop the override. PRs that grow override kwargs disguised as hooks are rejected.
Spec docs are protocol extensions, not design docs
Files under docs/specs/ describe the wire format and behaviour
requirements between independently developed servers — what bytes
move between systems and under what rules. Implementation choices
that pvl-core happens to make (lazy materialisation strategies, route
mechanics, framework-specific helpers, downstream tool naming and
registration mechanics) do not belong in a spec doc; they belong in
pvl-core's own implementor docs and code comments. Real spec gaps are
resolved through a proper spec evolution — a new release with the
version field bumped — not through inline amendments to a published
version.
Pre-existing downstream conflicts resolve by migration
If a downstream server has already shipped a different shape (a
differently named tool, a divergent parameter, a custom error
envelope), the resolution is for the downstream to migrate.
pvl-core does not grow a compatibility shim to spare downstream the
migration cost, even when the migration is large. If the migration
cannot land immediately, file a tracked downstream issue and ship
the breaking change in pvl-core anyway — the umbrella tracker
coordinates the cutover and the
fastmcp-server-template
scaffold updates carry the new shape forward to fresh consumers.
This applies to shape divergence (the things owned by pvl-core). Domain-specific divergence between downstreams is expected and does not require any migration — downstreams are supposed to differ in domain logic.
Downstream reuses pvl-core; it does not reimplement the protocol
Downstream servers reuse pvl-core's implementation of the shared
cross-cutting protocols — auth, logging, and the rest. They do not
reimplement a wire protocol independently. The specs under
docs/specs/ are the wire authority; pvl-core is their single shared
implementation. No implementation is "the reference" — not pvl-core's
either; the spec is.
If pvl-core's implementation is wrong, or diverges from a spec, the fix is to correct pvl-core centrally — one change, every downstream follows — or to evolve the spec. A downstream that believes pvl-core is wrong files the issue against pvl-core; it does not fork the behaviour and reimplement it locally.
API stability
This package is stable at 2.x and follows
semantic versioning: breaking changes bump the
major version, new features bump the minor, bugfixes bump the patch.
"Public API" means symbols re-exported from the top-level
fastmcp_pvl_core package (see __all__), which intentionally
covers both the runtime surface (auth, middleware, factory builders,
env/config helpers) and the CLI parser helpers consumed by downstream
server.py entrypoints. Modules prefixed with _ are internal and
may change without a major-version bump.
Install
uv add fastmcp-pvl-core
# If you use RemoteAuthProvider mode:
uv add "fastmcp-pvl-core[remote-auth]"
# For attaching a remote Python debugger inside a container image:
uv add "fastmcp-pvl-core[debug]"
Usage
See src/fastmcp_pvl_core/ for the full surface. Typical usage:
from fastmcp import FastMCP
from fastmcp_pvl_core import (
ServerConfig, build_auth, build_instructions,
wire_middleware_stack, env,
)
config = ServerConfig.from_env("MY_APP")
mcp = FastMCP(
name="my-app",
instructions=build_instructions(read_only=False, env_prefix="MY_APP", domain_line="…"),
auth=build_auth(config),
)
wire_middleware_stack(mcp)
Logging
configure_logging_from_env resolves the log level from the -v CLI flag
(forces DEBUG), then FASTMCP_LOG_LEVEL, then defaults to INFO.
At INFO and above, two noisy third-party loggers are demoted to WARNING
so they do not flood the operator log stream:
uvicorn.access— theINFO: <ip> - "POST /mcp ..."HTTP access log.mcp.server.lowlevel.server— the MCP SDK'sProcessing request of type ...line.
Both reappear at DEBUG (-v or FASTMCP_LOG_LEVEL=DEBUG). uvicorn.error
is never demoted — it carries genuine bind / startup failures.
wire_middleware_stack installs a single conforming request-logging
middleware. Every line it emits starts with a bare snake_case event name,
followed by key=value pairs, with request timing carried inline:
tool_call_started tool=read method=tools/call source=client
tool_call_completed tool=read duration_ms=68.57
tool_call_failed tool=read duration_ms=109.84 error_type=ValueError error="Section '1.3' not found"
Non-tool messages use a generic request_* / notification_* vocabulary
keyed by method=. Set FASTMCP_ENABLE_RICH_LOGGING=false to emit one JSON
object per record instead of key=value text — for log aggregators such as
the ELK stack or Splunk.
Per-user subject mapping (bearer auth)
Bearer auth has two modes:
-
Single token —
MY_APP_BEARER_TOKEN=<token>accepts one shared token. Authenticated callers all share the same subject (default"bearer-anon"; override withMY_APP_BEARER_DEFAULT_SUBJECT=<value>). -
Mapped tokens —
MY_APP_BEARER_TOKENS_FILE=/path/to/tokens.tomlloads a token→subject map at startup. Each token resolves to a distinct subject string for downstream attribution (audit logs, ACLs, request metadata).
# tokens.toml
[tokens]
"ghp_alice_xxxxxxxx" = "user:alice@example.com"
"sk_ci_yyyyyyyy" = "service:ci-bot"
If both MY_APP_BEARER_TOKEN and MY_APP_BEARER_TOKENS_FILE are set,
the file wins and a WARNING is logged. Subject strings are opaque to
the library; the <kind>:<id> convention (user:, service:,
token:) is documentation only.
If MY_APP_BEARER_TOKENS_FILE is set but the file is missing,
unparseable, or schema-invalid, the loader raises
fastmcp_pvl_core.ConfigurationError at startup — the server fails
fast rather than silently denying every request. The exception type
is part of the public API; downstream code can import and except
it as a stable contract.
MY_APP_BEARER_DEFAULT_SUBJECT only applies when bearer auth runs in
single-token mode (either standalone or as the bearer side of multi
mode alongside OIDC). It is ignored when MY_APP_BEARER_TOKENS_FILE
is set, including in multi mode — mapped mode uses the per-token
subjects from the TOML file.
Identifying the caller — get_subject
Tools, middleware, and resource handlers can call
fastmcp_pvl_core.get_subject() to retrieve the subject of the current
request without knowing which auth mode is active:
from fastmcp_pvl_core import get_subject
@mcp.tool
def whoami() -> str:
subject = get_subject()
return subject or "anonymous"
Resolution order:
- Token present: prefer
claims["sub"](OIDC's standard subject claim); fall back toclient_idifsubis absent. The auth builders normaliseclient_idper mode:bearer-single→bearer_default_subject(default"bearer-anon").bearer-mapped→ the per-token subject from the TOML map.- OIDC modes (
oidc-proxy,remote) → typicallyclaims["sub"]wins (a real OIDC token always carriessub); theclient_idfallback is defensive. multi→ bearer-validated requests follow the bearer path, OIDC-validated requests follow the OIDC path.
- No token,
auth_mode == "none": returns the literal"local". - No token, auth required: returns
None— caller decides whether to fall back or error.
Authorization (opt-in) — AuthorizationMiddleware
Tools, resources, and prompts can opt into per-subject access control by
setting meta={"required_scope": "<scope>"} at registration. A
configured AuthorizationMiddleware enforces the static check and
filters list_* responses to what the caller can use:
from pathlib import Path
from fastmcp_pvl_core import (
AuthorizationMiddleware, load_acl, make_acl_authorizer, check_authorization,
)
authorizer = make_acl_authorizer(load_acl(Path("/etc/my-app/acl.toml")))
mcp.add_middleware(AuthorizationMiddleware(authorizer=authorizer))
@mcp.tool(meta={"required_scope": "write"})
async def edit_document(project_id: str, doc_id: str, body: str) -> None:
# Coarse "write" gate already passed at middleware. Per-project gate here:
check_authorization(f"write:{project_id}")
...
ACL TOML schema (loaded by load_acl):
[subjects]
"user:alice@example.com" = ["read", "write"]
"user:admin@example.com" = ["*"] # wildcard scope
"service:ci-bot" = ["read"]
"local" = ["*"] # stdio mode
Key properties:
- Opt-in per component. Tools / resources / prompts without
meta["required_scope"]are unrestricted regardless of caller. *is the only library-treated special scope ("any required scope passes"). All other scopes are opaque strings; downstream chooses the vocabulary.- Subject-side wildcards (
*as an ACL key) are rejected at load time. load_aclfails fast withConfigurationErroron every malformed condition — never silent denial.- ACL is loaded once at startup. Restart to pick up changes.
- Authorization scopes are application-level and distinct from the OAuth scopes carried in tokens.
- Subject is logged on every deny at WARNING. The wire-side payload
omits the subject by default to limit cross-user info disclosure;
pass
AuthorizationMiddleware(..., expose_subject_in_error=True)to include it (e.g. for internal-only servers).
For the full design rationale and deviations from the originating
issue, see
docs/specs/authorization-submodule.md.
Remote debugging in containers
Containerised consumers can opt into a remote Python debugger by calling
maybe_start_debugpy(env_prefix) early in their CLI entrypoint, passing
the same per-app prefix the server uses for the rest of its config:
from fastmcp_pvl_core import configure_logging_from_env, maybe_start_debugpy
def main() -> None:
configure_logging_from_env()
maybe_start_debugpy("MY_APP") # no-op unless MY_APP_DEBUG_PORT is set
...
Environment contract ({PREFIX} matches the argument):
{PREFIX}_DEBUG_PORT— TCP port to listen on. Unset, blank, or any value that parses to0is a silent no-op. Non-numeric or out-of-1..65535values log aWARNINGand the helper returns without raising.{PREFIX}_DEBUG_WAIT— when truthy (1/true/yes/on, case-insensitive), block startup until the IDE attaches. Default is non-blocking.- If
debugpy.listen()itself fails (port in use, permission denied, debugpy-internal error), the helper logs aWARNINGand continues — a debug-port problem must never crash the server.
Install the optional debug extra on images that need the listener:
uv add "fastmcp-pvl-core[debug]" # quote brackets in zsh
# or, equivalently:
uv add debugpy
The helper logs a WARNING and continues if debugpy is unavailable,
so it is safe to ship in default scaffolds.
⚠️ Security: the listener binds
0.0.0.0and debugpy's DAP protocol is unauthenticated — any peer that can reach the port has arbitrary code execution as the server process. Only enable{PREFIX}_DEBUG_PORTin environments where the port is reachable solely from a trusted developer workstation, e.g.kubectl port-forward,docker run -p 127.0.0.1:5678:5678(loopback bind), or an SSH tunnel. Never publish the debug port on a public network.
License
MIT
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 fastmcp_pvl_core-3.0.0.tar.gz.
File metadata
- Download URL: fastmcp_pvl_core-3.0.0.tar.gz
- Upload date:
- Size: 317.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
f570c2a8befb1a7ea6424fda3c9dfc910b0b446e16959d9799197125e3049df0
|
|
| MD5 |
6e3a956a555f973a67b5c8077e5fde90
|
|
| BLAKE2b-256 |
e79a65e113da78159b75e7968b457ca423082803beafcee0b6f65b79666aa9c5
|
Provenance
The following attestation bundles were made for fastmcp_pvl_core-3.0.0.tar.gz:
Publisher:
release.yml on pvliesdonk/fastmcp-pvl-core
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
fastmcp_pvl_core-3.0.0.tar.gz -
Subject digest:
f570c2a8befb1a7ea6424fda3c9dfc910b0b446e16959d9799197125e3049df0 - Sigstore transparency entry: 1670431223
- Sigstore integration time:
-
Permalink:
pvliesdonk/fastmcp-pvl-core@08449a9fc7433e3b52224481fc9fc6b176eff03d -
Branch / Tag:
refs/heads/main - Owner: https://github.com/pvliesdonk
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@08449a9fc7433e3b52224481fc9fc6b176eff03d -
Trigger Event:
workflow_dispatch
-
Statement type:
File details
Details for the file fastmcp_pvl_core-3.0.0-py3-none-any.whl.
File metadata
- Download URL: fastmcp_pvl_core-3.0.0-py3-none-any.whl
- Upload date:
- Size: 49.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
1b7d25a1c29958212d29aaa0aeebee9b3d06644e480f386828e4458855aba455
|
|
| MD5 |
824b3b9afaea025fd4aba65f8bb7ef74
|
|
| BLAKE2b-256 |
20914e8f01bdf3f0a224d89a2d056c59894c6f9a43dd03d36d6a863e6c81978c
|
Provenance
The following attestation bundles were made for fastmcp_pvl_core-3.0.0-py3-none-any.whl:
Publisher:
release.yml on pvliesdonk/fastmcp-pvl-core
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
fastmcp_pvl_core-3.0.0-py3-none-any.whl -
Subject digest:
1b7d25a1c29958212d29aaa0aeebee9b3d06644e480f386828e4458855aba455 - Sigstore transparency entry: 1670431524
- Sigstore integration time:
-
Permalink:
pvliesdonk/fastmcp-pvl-core@08449a9fc7433e3b52224481fc9fc6b176eff03d -
Branch / Tag:
refs/heads/main - Owner: https://github.com/pvliesdonk
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@08449a9fc7433e3b52224481fc9fc6b176eff03d -
Trigger Event:
workflow_dispatch
-
Statement type: