fastmcp 2.14.4__py3-none-any.whl → 3.0.0b1__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- fastmcp/_vendor/__init__.py +1 -0
- fastmcp/_vendor/docket_di/README.md +7 -0
- fastmcp/_vendor/docket_di/__init__.py +163 -0
- fastmcp/cli/cli.py +112 -28
- fastmcp/cli/install/claude_code.py +1 -5
- fastmcp/cli/install/claude_desktop.py +1 -5
- fastmcp/cli/install/cursor.py +1 -5
- fastmcp/cli/install/gemini_cli.py +1 -5
- fastmcp/cli/install/mcp_json.py +1 -6
- fastmcp/cli/run.py +146 -5
- fastmcp/client/__init__.py +7 -9
- fastmcp/client/auth/oauth.py +18 -17
- fastmcp/client/client.py +100 -870
- fastmcp/client/elicitation.py +1 -1
- fastmcp/client/mixins/__init__.py +13 -0
- fastmcp/client/mixins/prompts.py +295 -0
- fastmcp/client/mixins/resources.py +325 -0
- fastmcp/client/mixins/task_management.py +157 -0
- fastmcp/client/mixins/tools.py +397 -0
- fastmcp/client/sampling/handlers/anthropic.py +2 -2
- fastmcp/client/sampling/handlers/openai.py +1 -1
- fastmcp/client/tasks.py +3 -3
- fastmcp/client/telemetry.py +47 -0
- fastmcp/client/transports/__init__.py +38 -0
- fastmcp/client/transports/base.py +82 -0
- fastmcp/client/transports/config.py +170 -0
- fastmcp/client/transports/http.py +145 -0
- fastmcp/client/transports/inference.py +154 -0
- fastmcp/client/transports/memory.py +90 -0
- fastmcp/client/transports/sse.py +89 -0
- fastmcp/client/transports/stdio.py +543 -0
- fastmcp/contrib/component_manager/README.md +4 -10
- fastmcp/contrib/component_manager/__init__.py +1 -2
- fastmcp/contrib/component_manager/component_manager.py +95 -160
- fastmcp/contrib/component_manager/example.py +1 -1
- fastmcp/contrib/mcp_mixin/example.py +4 -4
- fastmcp/contrib/mcp_mixin/mcp_mixin.py +11 -4
- fastmcp/decorators.py +41 -0
- fastmcp/dependencies.py +12 -1
- fastmcp/exceptions.py +4 -0
- fastmcp/experimental/server/openapi/__init__.py +18 -15
- fastmcp/mcp_config.py +13 -4
- fastmcp/prompts/__init__.py +6 -3
- fastmcp/prompts/function_prompt.py +465 -0
- fastmcp/prompts/prompt.py +321 -271
- fastmcp/resources/__init__.py +5 -3
- fastmcp/resources/function_resource.py +335 -0
- fastmcp/resources/resource.py +325 -115
- fastmcp/resources/template.py +215 -43
- fastmcp/resources/types.py +27 -12
- fastmcp/server/__init__.py +2 -2
- fastmcp/server/auth/__init__.py +14 -0
- fastmcp/server/auth/auth.py +30 -10
- fastmcp/server/auth/authorization.py +190 -0
- fastmcp/server/auth/oauth_proxy/__init__.py +14 -0
- fastmcp/server/auth/oauth_proxy/consent.py +361 -0
- fastmcp/server/auth/oauth_proxy/models.py +178 -0
- fastmcp/server/auth/{oauth_proxy.py → oauth_proxy/proxy.py} +24 -778
- fastmcp/server/auth/oauth_proxy/ui.py +277 -0
- fastmcp/server/auth/oidc_proxy.py +2 -2
- fastmcp/server/auth/providers/auth0.py +24 -94
- fastmcp/server/auth/providers/aws.py +26 -95
- fastmcp/server/auth/providers/azure.py +41 -129
- fastmcp/server/auth/providers/descope.py +18 -49
- fastmcp/server/auth/providers/discord.py +25 -86
- fastmcp/server/auth/providers/github.py +23 -87
- fastmcp/server/auth/providers/google.py +24 -87
- fastmcp/server/auth/providers/introspection.py +60 -79
- fastmcp/server/auth/providers/jwt.py +30 -67
- fastmcp/server/auth/providers/oci.py +47 -110
- fastmcp/server/auth/providers/scalekit.py +23 -61
- fastmcp/server/auth/providers/supabase.py +18 -47
- fastmcp/server/auth/providers/workos.py +34 -127
- fastmcp/server/context.py +372 -419
- fastmcp/server/dependencies.py +541 -251
- fastmcp/server/elicitation.py +20 -18
- fastmcp/server/event_store.py +3 -3
- fastmcp/server/http.py +16 -6
- fastmcp/server/lifespan.py +198 -0
- fastmcp/server/low_level.py +92 -2
- fastmcp/server/middleware/__init__.py +5 -1
- fastmcp/server/middleware/authorization.py +312 -0
- fastmcp/server/middleware/caching.py +101 -54
- fastmcp/server/middleware/middleware.py +6 -9
- fastmcp/server/middleware/ping.py +70 -0
- fastmcp/server/middleware/tool_injection.py +2 -2
- fastmcp/server/mixins/__init__.py +7 -0
- fastmcp/server/mixins/lifespan.py +217 -0
- fastmcp/server/mixins/mcp_operations.py +392 -0
- fastmcp/server/mixins/transport.py +342 -0
- fastmcp/server/openapi/__init__.py +41 -21
- fastmcp/server/openapi/components.py +16 -339
- fastmcp/server/openapi/routing.py +34 -118
- fastmcp/server/openapi/server.py +67 -392
- fastmcp/server/providers/__init__.py +71 -0
- fastmcp/server/providers/aggregate.py +261 -0
- fastmcp/server/providers/base.py +578 -0
- fastmcp/server/providers/fastmcp_provider.py +674 -0
- fastmcp/server/providers/filesystem.py +226 -0
- fastmcp/server/providers/filesystem_discovery.py +327 -0
- fastmcp/server/providers/local_provider/__init__.py +11 -0
- fastmcp/server/providers/local_provider/decorators/__init__.py +15 -0
- fastmcp/server/providers/local_provider/decorators/prompts.py +256 -0
- fastmcp/server/providers/local_provider/decorators/resources.py +240 -0
- fastmcp/server/providers/local_provider/decorators/tools.py +315 -0
- fastmcp/server/providers/local_provider/local_provider.py +465 -0
- fastmcp/server/providers/openapi/__init__.py +39 -0
- fastmcp/server/providers/openapi/components.py +332 -0
- fastmcp/server/providers/openapi/provider.py +405 -0
- fastmcp/server/providers/openapi/routing.py +109 -0
- fastmcp/server/providers/proxy.py +867 -0
- fastmcp/server/providers/skills/__init__.py +59 -0
- fastmcp/server/providers/skills/_common.py +101 -0
- fastmcp/server/providers/skills/claude_provider.py +44 -0
- fastmcp/server/providers/skills/directory_provider.py +153 -0
- fastmcp/server/providers/skills/skill_provider.py +432 -0
- fastmcp/server/providers/skills/vendor_providers.py +142 -0
- fastmcp/server/providers/wrapped_provider.py +140 -0
- fastmcp/server/proxy.py +34 -700
- fastmcp/server/sampling/run.py +341 -2
- fastmcp/server/sampling/sampling_tool.py +4 -3
- fastmcp/server/server.py +1214 -2171
- fastmcp/server/tasks/__init__.py +2 -1
- fastmcp/server/tasks/capabilities.py +13 -1
- fastmcp/server/tasks/config.py +66 -3
- fastmcp/server/tasks/handlers.py +65 -273
- fastmcp/server/tasks/keys.py +4 -6
- fastmcp/server/tasks/requests.py +474 -0
- fastmcp/server/tasks/routing.py +76 -0
- fastmcp/server/tasks/subscriptions.py +20 -11
- fastmcp/server/telemetry.py +131 -0
- fastmcp/server/transforms/__init__.py +244 -0
- fastmcp/server/transforms/namespace.py +193 -0
- fastmcp/server/transforms/prompts_as_tools.py +175 -0
- fastmcp/server/transforms/resources_as_tools.py +190 -0
- fastmcp/server/transforms/tool_transform.py +96 -0
- fastmcp/server/transforms/version_filter.py +124 -0
- fastmcp/server/transforms/visibility.py +526 -0
- fastmcp/settings.py +34 -96
- fastmcp/telemetry.py +122 -0
- fastmcp/tools/__init__.py +10 -3
- fastmcp/tools/function_parsing.py +201 -0
- fastmcp/tools/function_tool.py +467 -0
- fastmcp/tools/tool.py +215 -362
- fastmcp/tools/tool_transform.py +38 -21
- fastmcp/utilities/async_utils.py +69 -0
- fastmcp/utilities/components.py +152 -91
- fastmcp/utilities/inspect.py +8 -20
- fastmcp/utilities/json_schema.py +12 -5
- fastmcp/utilities/json_schema_type.py +17 -15
- fastmcp/utilities/lifespan.py +56 -0
- fastmcp/utilities/logging.py +12 -4
- fastmcp/utilities/mcp_server_config/v1/mcp_server_config.py +3 -3
- fastmcp/utilities/openapi/parser.py +3 -3
- fastmcp/utilities/pagination.py +80 -0
- fastmcp/utilities/skills.py +253 -0
- fastmcp/utilities/tests.py +0 -16
- fastmcp/utilities/timeout.py +47 -0
- fastmcp/utilities/types.py +1 -1
- fastmcp/utilities/versions.py +285 -0
- {fastmcp-2.14.4.dist-info → fastmcp-3.0.0b1.dist-info}/METADATA +8 -5
- fastmcp-3.0.0b1.dist-info/RECORD +228 -0
- fastmcp/client/transports.py +0 -1170
- fastmcp/contrib/component_manager/component_service.py +0 -209
- fastmcp/prompts/prompt_manager.py +0 -117
- fastmcp/resources/resource_manager.py +0 -338
- fastmcp/server/tasks/converters.py +0 -206
- fastmcp/server/tasks/protocol.py +0 -359
- fastmcp/tools/tool_manager.py +0 -170
- fastmcp/utilities/mcp_config.py +0 -56
- fastmcp-2.14.4.dist-info/RECORD +0 -161
- /fastmcp/server/{openapi → providers/openapi}/README.md +0 -0
- {fastmcp-2.14.4.dist-info → fastmcp-3.0.0b1.dist-info}/WHEEL +0 -0
- {fastmcp-2.14.4.dist-info → fastmcp-3.0.0b1.dist-info}/entry_points.txt +0 -0
- {fastmcp-2.14.4.dist-info → fastmcp-3.0.0b1.dist-info}/licenses/LICENSE +0 -0
|
@@ -0,0 +1,578 @@
|
|
|
1
|
+
"""Base Provider class for dynamic MCP components.
|
|
2
|
+
|
|
3
|
+
This module provides the `Provider` abstraction for providing tools,
|
|
4
|
+
resources, and prompts dynamically at runtime.
|
|
5
|
+
|
|
6
|
+
Example:
|
|
7
|
+
```python
|
|
8
|
+
from fastmcp import FastMCP
|
|
9
|
+
from fastmcp.server.providers import Provider
|
|
10
|
+
from fastmcp.tools import Tool
|
|
11
|
+
|
|
12
|
+
class DatabaseProvider(Provider):
|
|
13
|
+
def __init__(self, db_url: str):
|
|
14
|
+
super().__init__()
|
|
15
|
+
self.db = Database(db_url)
|
|
16
|
+
|
|
17
|
+
async def _list_tools(self) -> list[Tool]:
|
|
18
|
+
rows = await self.db.fetch("SELECT * FROM tools")
|
|
19
|
+
return [self._make_tool(row) for row in rows]
|
|
20
|
+
|
|
21
|
+
async def _get_tool(self, name: str) -> Tool | None:
|
|
22
|
+
row = await self.db.fetchone("SELECT * FROM tools WHERE name = ?", name)
|
|
23
|
+
return self._make_tool(row) if row else None
|
|
24
|
+
|
|
25
|
+
mcp = FastMCP("Server", providers=[DatabaseProvider(db_url)])
|
|
26
|
+
```
|
|
27
|
+
"""
|
|
28
|
+
|
|
29
|
+
from __future__ import annotations
|
|
30
|
+
|
|
31
|
+
from collections.abc import AsyncIterator, Sequence
|
|
32
|
+
from contextlib import asynccontextmanager
|
|
33
|
+
from functools import partial
|
|
34
|
+
from typing import TYPE_CHECKING, Literal, cast
|
|
35
|
+
|
|
36
|
+
from typing_extensions import Self
|
|
37
|
+
|
|
38
|
+
from fastmcp.prompts.prompt import Prompt
|
|
39
|
+
from fastmcp.resources.resource import Resource
|
|
40
|
+
from fastmcp.resources.template import ResourceTemplate
|
|
41
|
+
from fastmcp.server.transforms.visibility import Visibility
|
|
42
|
+
from fastmcp.tools.tool import Tool
|
|
43
|
+
from fastmcp.utilities.async_utils import gather
|
|
44
|
+
from fastmcp.utilities.components import FastMCPComponent
|
|
45
|
+
from fastmcp.utilities.versions import VersionSpec, version_sort_key
|
|
46
|
+
|
|
47
|
+
if TYPE_CHECKING:
|
|
48
|
+
from fastmcp.server.transforms import Transform
|
|
49
|
+
|
|
50
|
+
|
|
51
|
+
class Provider:
|
|
52
|
+
"""Base class for dynamic component providers.
|
|
53
|
+
|
|
54
|
+
Subclass and override whichever methods you need. Default implementations
|
|
55
|
+
return empty lists / None, so you only need to implement what your provider
|
|
56
|
+
supports.
|
|
57
|
+
|
|
58
|
+
Provider semantics:
|
|
59
|
+
- Return `None` from `get_*` methods to indicate "I don't have it" (search continues)
|
|
60
|
+
- Static components (registered via decorators) always take precedence over providers
|
|
61
|
+
- Providers are queried in registration order; first non-None wins
|
|
62
|
+
- Components execute themselves via run()/read()/render() - providers just source them
|
|
63
|
+
|
|
64
|
+
Error handling:
|
|
65
|
+
- `list_*` methods: Errors are logged and the provider returns empty (graceful degradation).
|
|
66
|
+
This allows other providers to still contribute their components.
|
|
67
|
+
"""
|
|
68
|
+
|
|
69
|
+
def __init__(self) -> None:
|
|
70
|
+
self._transforms: list[Transform] = []
|
|
71
|
+
|
|
72
|
+
def __repr__(self) -> str:
|
|
73
|
+
return f"{self.__class__.__name__}()"
|
|
74
|
+
|
|
75
|
+
@property
|
|
76
|
+
def transforms(self) -> list[Transform]:
|
|
77
|
+
"""All transforms applied to components from this provider."""
|
|
78
|
+
return list(self._transforms)
|
|
79
|
+
|
|
80
|
+
def add_transform(self, transform: Transform) -> None:
|
|
81
|
+
"""Add a transform to this provider.
|
|
82
|
+
|
|
83
|
+
Transforms modify components (tools, resources, prompts) as they flow
|
|
84
|
+
through the provider. They're applied in order - first added is innermost.
|
|
85
|
+
|
|
86
|
+
Args:
|
|
87
|
+
transform: The transform to add.
|
|
88
|
+
|
|
89
|
+
Example:
|
|
90
|
+
```python
|
|
91
|
+
from fastmcp.server.transforms import Namespace
|
|
92
|
+
|
|
93
|
+
provider = MyProvider()
|
|
94
|
+
provider.add_transform(Namespace("api"))
|
|
95
|
+
# Tools become "api_toolname"
|
|
96
|
+
```
|
|
97
|
+
"""
|
|
98
|
+
self._transforms.append(transform)
|
|
99
|
+
|
|
100
|
+
def wrap_transform(self, transform: Transform) -> Provider:
|
|
101
|
+
"""Return a new provider with this transform applied (immutable).
|
|
102
|
+
|
|
103
|
+
Unlike add_transform() which mutates this provider, wrap_transform()
|
|
104
|
+
returns a new provider that wraps this one. The original provider
|
|
105
|
+
is unchanged.
|
|
106
|
+
|
|
107
|
+
This is useful when you want to apply transforms without side effects,
|
|
108
|
+
such as adding the same provider to multiple aggregators with different
|
|
109
|
+
namespaces.
|
|
110
|
+
|
|
111
|
+
Args:
|
|
112
|
+
transform: The transform to apply.
|
|
113
|
+
|
|
114
|
+
Returns:
|
|
115
|
+
A new provider that wraps this one with the transform applied.
|
|
116
|
+
|
|
117
|
+
Example:
|
|
118
|
+
```python
|
|
119
|
+
from fastmcp.server.transforms import Namespace
|
|
120
|
+
|
|
121
|
+
provider = MyProvider()
|
|
122
|
+
namespaced = provider.wrap_transform(Namespace("api"))
|
|
123
|
+
# provider is unchanged
|
|
124
|
+
# namespaced returns tools as "api_toolname"
|
|
125
|
+
```
|
|
126
|
+
"""
|
|
127
|
+
# Import here to avoid circular imports
|
|
128
|
+
from fastmcp.server.providers.wrapped_provider import _WrappedProvider
|
|
129
|
+
|
|
130
|
+
return _WrappedProvider(self, transform)
|
|
131
|
+
|
|
132
|
+
# -------------------------------------------------------------------------
|
|
133
|
+
# Internal transform chain building
|
|
134
|
+
# -------------------------------------------------------------------------
|
|
135
|
+
|
|
136
|
+
async def list_tools(self) -> Sequence[Tool]:
|
|
137
|
+
"""List tools with all transforms applied.
|
|
138
|
+
|
|
139
|
+
Applies transforms sequentially: base → transforms (in order).
|
|
140
|
+
Each transform receives the result from the previous transform.
|
|
141
|
+
Components may be marked as disabled but are NOT filtered here -
|
|
142
|
+
filtering happens at the server level to allow session transforms to override.
|
|
143
|
+
|
|
144
|
+
Returns:
|
|
145
|
+
Transformed sequence of tools (including disabled ones).
|
|
146
|
+
"""
|
|
147
|
+
tools = await self._list_tools()
|
|
148
|
+
for transform in self.transforms:
|
|
149
|
+
tools = await transform.list_tools(tools)
|
|
150
|
+
return tools
|
|
151
|
+
|
|
152
|
+
async def get_tool(
|
|
153
|
+
self, name: str, version: VersionSpec | None = None
|
|
154
|
+
) -> Tool | None:
|
|
155
|
+
"""Get tool by transformed name with all transforms applied.
|
|
156
|
+
|
|
157
|
+
Note: This method does NOT filter disabled components. The Server
|
|
158
|
+
(FastMCP) performs enabled filtering after all transforms complete,
|
|
159
|
+
allowing session-level transforms to override provider-level disables.
|
|
160
|
+
|
|
161
|
+
Args:
|
|
162
|
+
name: The transformed tool name to look up.
|
|
163
|
+
version: Optional version filter. If None, returns highest version.
|
|
164
|
+
|
|
165
|
+
Returns:
|
|
166
|
+
The tool if found (may be marked disabled), None if not found.
|
|
167
|
+
"""
|
|
168
|
+
|
|
169
|
+
async def base(n: str, version: VersionSpec | None = None) -> Tool | None:
|
|
170
|
+
return await self._get_tool(n, version)
|
|
171
|
+
|
|
172
|
+
chain = base
|
|
173
|
+
for transform in self.transforms:
|
|
174
|
+
chain = partial(transform.get_tool, call_next=chain)
|
|
175
|
+
|
|
176
|
+
return await chain(name, version=version)
|
|
177
|
+
|
|
178
|
+
async def list_resources(self) -> Sequence[Resource]:
|
|
179
|
+
"""List resources with all transforms applied.
|
|
180
|
+
|
|
181
|
+
Components may be marked as disabled but are NOT filtered here.
|
|
182
|
+
"""
|
|
183
|
+
resources = await self._list_resources()
|
|
184
|
+
for transform in self.transforms:
|
|
185
|
+
resources = await transform.list_resources(resources)
|
|
186
|
+
return resources
|
|
187
|
+
|
|
188
|
+
async def get_resource(
|
|
189
|
+
self, uri: str, version: VersionSpec | None = None
|
|
190
|
+
) -> Resource | None:
|
|
191
|
+
"""Get resource by transformed URI with all transforms applied.
|
|
192
|
+
|
|
193
|
+
Note: This method does NOT filter disabled components. The Server
|
|
194
|
+
(FastMCP) performs enabled filtering after all transforms complete.
|
|
195
|
+
|
|
196
|
+
Args:
|
|
197
|
+
uri: The transformed resource URI to look up.
|
|
198
|
+
version: Optional version filter. If None, returns highest version.
|
|
199
|
+
|
|
200
|
+
Returns:
|
|
201
|
+
The resource if found (may be marked disabled), None if not found.
|
|
202
|
+
"""
|
|
203
|
+
|
|
204
|
+
async def base(u: str, version: VersionSpec | None = None) -> Resource | None:
|
|
205
|
+
return await self._get_resource(u, version)
|
|
206
|
+
|
|
207
|
+
chain = base
|
|
208
|
+
for transform in self.transforms:
|
|
209
|
+
chain = partial(transform.get_resource, call_next=chain)
|
|
210
|
+
|
|
211
|
+
return await chain(uri, version=version)
|
|
212
|
+
|
|
213
|
+
async def list_resource_templates(self) -> Sequence[ResourceTemplate]:
|
|
214
|
+
"""List resource templates with all transforms applied.
|
|
215
|
+
|
|
216
|
+
Components may be marked as disabled but are NOT filtered here.
|
|
217
|
+
"""
|
|
218
|
+
templates = await self._list_resource_templates()
|
|
219
|
+
for transform in self.transforms:
|
|
220
|
+
templates = await transform.list_resource_templates(templates)
|
|
221
|
+
return templates
|
|
222
|
+
|
|
223
|
+
async def get_resource_template(
|
|
224
|
+
self, uri: str, version: VersionSpec | None = None
|
|
225
|
+
) -> ResourceTemplate | None:
|
|
226
|
+
"""Get resource template by transformed URI with all transforms applied.
|
|
227
|
+
|
|
228
|
+
Note: This method does NOT filter disabled components. The Server
|
|
229
|
+
(FastMCP) performs enabled filtering after all transforms complete.
|
|
230
|
+
|
|
231
|
+
Args:
|
|
232
|
+
uri: The transformed template URI to look up.
|
|
233
|
+
version: Optional version filter. If None, returns highest version.
|
|
234
|
+
|
|
235
|
+
Returns:
|
|
236
|
+
The template if found (may be marked disabled), None if not found.
|
|
237
|
+
"""
|
|
238
|
+
|
|
239
|
+
async def base(
|
|
240
|
+
u: str, version: VersionSpec | None = None
|
|
241
|
+
) -> ResourceTemplate | None:
|
|
242
|
+
return await self._get_resource_template(u, version)
|
|
243
|
+
|
|
244
|
+
chain = base
|
|
245
|
+
for transform in self.transforms:
|
|
246
|
+
chain = partial(transform.get_resource_template, call_next=chain)
|
|
247
|
+
|
|
248
|
+
return await chain(uri, version=version)
|
|
249
|
+
|
|
250
|
+
async def list_prompts(self) -> Sequence[Prompt]:
|
|
251
|
+
"""List prompts with all transforms applied.
|
|
252
|
+
|
|
253
|
+
Components may be marked as disabled but are NOT filtered here.
|
|
254
|
+
"""
|
|
255
|
+
prompts = await self._list_prompts()
|
|
256
|
+
for transform in self.transforms:
|
|
257
|
+
prompts = await transform.list_prompts(prompts)
|
|
258
|
+
return prompts
|
|
259
|
+
|
|
260
|
+
async def get_prompt(
|
|
261
|
+
self, name: str, version: VersionSpec | None = None
|
|
262
|
+
) -> Prompt | None:
|
|
263
|
+
"""Get prompt by transformed name with all transforms applied.
|
|
264
|
+
|
|
265
|
+
Note: This method does NOT filter disabled components. The Server
|
|
266
|
+
(FastMCP) performs enabled filtering after all transforms complete.
|
|
267
|
+
|
|
268
|
+
Args:
|
|
269
|
+
name: The transformed prompt name to look up.
|
|
270
|
+
version: Optional version filter. If None, returns highest version.
|
|
271
|
+
|
|
272
|
+
Returns:
|
|
273
|
+
The prompt if found (may be marked disabled), None if not found.
|
|
274
|
+
"""
|
|
275
|
+
|
|
276
|
+
async def base(n: str, version: VersionSpec | None = None) -> Prompt | None:
|
|
277
|
+
return await self._get_prompt(n, version)
|
|
278
|
+
|
|
279
|
+
chain = base
|
|
280
|
+
for transform in self.transforms:
|
|
281
|
+
chain = partial(transform.get_prompt, call_next=chain)
|
|
282
|
+
|
|
283
|
+
return await chain(name, version=version)
|
|
284
|
+
|
|
285
|
+
# -------------------------------------------------------------------------
|
|
286
|
+
# Private list/get methods (override these to provide components)
|
|
287
|
+
# -------------------------------------------------------------------------
|
|
288
|
+
|
|
289
|
+
async def _list_tools(self) -> Sequence[Tool]:
|
|
290
|
+
"""Return all available tools.
|
|
291
|
+
|
|
292
|
+
Override to provide tools dynamically. Returns ALL versions of all tools.
|
|
293
|
+
The server handles deduplication to show one tool per name.
|
|
294
|
+
"""
|
|
295
|
+
return []
|
|
296
|
+
|
|
297
|
+
async def _get_tool(
|
|
298
|
+
self, name: str, version: VersionSpec | None = None
|
|
299
|
+
) -> Tool | None:
|
|
300
|
+
"""Get a specific tool by name.
|
|
301
|
+
|
|
302
|
+
Default implementation filters _list_tools() and picks the highest version
|
|
303
|
+
that matches the spec.
|
|
304
|
+
|
|
305
|
+
Args:
|
|
306
|
+
name: The tool name.
|
|
307
|
+
version: Optional version filter. If None, returns highest version.
|
|
308
|
+
If specified, returns highest version matching the spec.
|
|
309
|
+
|
|
310
|
+
Returns:
|
|
311
|
+
The Tool if found, or None to continue searching other providers.
|
|
312
|
+
"""
|
|
313
|
+
tools = await self._list_tools()
|
|
314
|
+
matching = [t for t in tools if t.name == name]
|
|
315
|
+
if version:
|
|
316
|
+
matching = [t for t in matching if version.matches(t.version)]
|
|
317
|
+
if not matching:
|
|
318
|
+
return None
|
|
319
|
+
return max(matching, key=version_sort_key) # type: ignore[type-var]
|
|
320
|
+
|
|
321
|
+
async def _list_resources(self) -> Sequence[Resource]:
|
|
322
|
+
"""Return all available resources.
|
|
323
|
+
|
|
324
|
+
Override to provide resources dynamically. Returns ALL versions of all resources.
|
|
325
|
+
The server handles deduplication to show one resource per URI.
|
|
326
|
+
"""
|
|
327
|
+
return []
|
|
328
|
+
|
|
329
|
+
async def _get_resource(
|
|
330
|
+
self, uri: str, version: VersionSpec | None = None
|
|
331
|
+
) -> Resource | None:
|
|
332
|
+
"""Get a specific resource by URI.
|
|
333
|
+
|
|
334
|
+
Default implementation filters _list_resources() and returns highest
|
|
335
|
+
version matching the spec.
|
|
336
|
+
|
|
337
|
+
Args:
|
|
338
|
+
uri: The resource URI.
|
|
339
|
+
version: Optional version filter. If None, returns highest version.
|
|
340
|
+
|
|
341
|
+
Returns:
|
|
342
|
+
The Resource if found, or None to continue searching other providers.
|
|
343
|
+
"""
|
|
344
|
+
resources = await self._list_resources()
|
|
345
|
+
matching = [r for r in resources if str(r.uri) == uri]
|
|
346
|
+
if version:
|
|
347
|
+
matching = [r for r in matching if version.matches(r.version)]
|
|
348
|
+
if not matching:
|
|
349
|
+
return None
|
|
350
|
+
return max(matching, key=version_sort_key) # type: ignore[type-var]
|
|
351
|
+
|
|
352
|
+
async def _list_resource_templates(self) -> Sequence[ResourceTemplate]:
|
|
353
|
+
"""Return all available resource templates.
|
|
354
|
+
|
|
355
|
+
Override to provide resource templates dynamically. Returns ALL versions.
|
|
356
|
+
The server handles deduplication.
|
|
357
|
+
"""
|
|
358
|
+
return []
|
|
359
|
+
|
|
360
|
+
async def _get_resource_template(
|
|
361
|
+
self, uri: str, version: VersionSpec | None = None
|
|
362
|
+
) -> ResourceTemplate | None:
|
|
363
|
+
"""Get a resource template that matches the given URI.
|
|
364
|
+
|
|
365
|
+
Default implementation lists all templates, finds those whose pattern
|
|
366
|
+
matches the URI, and returns the highest version matching the spec.
|
|
367
|
+
|
|
368
|
+
Args:
|
|
369
|
+
uri: The URI to match against templates.
|
|
370
|
+
version: Optional version filter. If None, returns highest version.
|
|
371
|
+
|
|
372
|
+
Returns:
|
|
373
|
+
The ResourceTemplate if a matching one is found, or None to continue searching.
|
|
374
|
+
"""
|
|
375
|
+
templates = await self._list_resource_templates()
|
|
376
|
+
matching = [t for t in templates if t.matches(uri) is not None]
|
|
377
|
+
if version:
|
|
378
|
+
matching = [t for t in matching if version.matches(t.version)]
|
|
379
|
+
if not matching:
|
|
380
|
+
return None
|
|
381
|
+
return max(matching, key=version_sort_key) # type: ignore[type-var]
|
|
382
|
+
|
|
383
|
+
async def _list_prompts(self) -> Sequence[Prompt]:
|
|
384
|
+
"""Return all available prompts.
|
|
385
|
+
|
|
386
|
+
Override to provide prompts dynamically. Returns ALL versions of all prompts.
|
|
387
|
+
The server handles deduplication to show one prompt per name.
|
|
388
|
+
"""
|
|
389
|
+
return []
|
|
390
|
+
|
|
391
|
+
async def _get_prompt(
|
|
392
|
+
self, name: str, version: VersionSpec | None = None
|
|
393
|
+
) -> Prompt | None:
|
|
394
|
+
"""Get a specific prompt by name.
|
|
395
|
+
|
|
396
|
+
Default implementation filters _list_prompts() and picks the highest version
|
|
397
|
+
matching the spec.
|
|
398
|
+
|
|
399
|
+
Args:
|
|
400
|
+
name: The prompt name.
|
|
401
|
+
version: Optional version filter. If None, returns highest version.
|
|
402
|
+
|
|
403
|
+
Returns:
|
|
404
|
+
The Prompt if found, or None to continue searching other providers.
|
|
405
|
+
"""
|
|
406
|
+
prompts = await self._list_prompts()
|
|
407
|
+
matching = [p for p in prompts if p.name == name]
|
|
408
|
+
if version:
|
|
409
|
+
matching = [p for p in matching if version.matches(p.version)]
|
|
410
|
+
if not matching:
|
|
411
|
+
return None
|
|
412
|
+
return max(matching, key=version_sort_key) # type: ignore[type-var]
|
|
413
|
+
|
|
414
|
+
# -------------------------------------------------------------------------
|
|
415
|
+
# Task registration
|
|
416
|
+
# -------------------------------------------------------------------------
|
|
417
|
+
|
|
418
|
+
async def get_tasks(self) -> Sequence[FastMCPComponent]:
|
|
419
|
+
"""Return components that should be registered as background tasks.
|
|
420
|
+
|
|
421
|
+
Override to customize which components are task-eligible.
|
|
422
|
+
Default calls list_* methods, applies provider transforms, and filters
|
|
423
|
+
for components with task_config.mode != 'forbidden'.
|
|
424
|
+
|
|
425
|
+
Used by the server during startup to register functions with Docket.
|
|
426
|
+
"""
|
|
427
|
+
# Fetch all component types in parallel
|
|
428
|
+
results = await gather(
|
|
429
|
+
self._list_tools(),
|
|
430
|
+
self._list_resources(),
|
|
431
|
+
self._list_resource_templates(),
|
|
432
|
+
self._list_prompts(),
|
|
433
|
+
)
|
|
434
|
+
tools = cast(Sequence[Tool], results[0])
|
|
435
|
+
resources = cast(Sequence[Resource], results[1])
|
|
436
|
+
templates = cast(Sequence[ResourceTemplate], results[2])
|
|
437
|
+
prompts = cast(Sequence[Prompt], results[3])
|
|
438
|
+
|
|
439
|
+
# Apply provider's own transforms sequentially
|
|
440
|
+
# For tasks, we need the fully-transformed names
|
|
441
|
+
for transform in self.transforms:
|
|
442
|
+
tools = await transform.list_tools(tools)
|
|
443
|
+
resources = await transform.list_resources(resources)
|
|
444
|
+
templates = await transform.list_resource_templates(templates)
|
|
445
|
+
prompts = await transform.list_prompts(prompts)
|
|
446
|
+
|
|
447
|
+
return [
|
|
448
|
+
c
|
|
449
|
+
for c in [
|
|
450
|
+
*tools,
|
|
451
|
+
*resources,
|
|
452
|
+
*templates,
|
|
453
|
+
*prompts,
|
|
454
|
+
]
|
|
455
|
+
if c.task_config.supports_tasks()
|
|
456
|
+
]
|
|
457
|
+
|
|
458
|
+
# -------------------------------------------------------------------------
|
|
459
|
+
# Lifecycle methods
|
|
460
|
+
# -------------------------------------------------------------------------
|
|
461
|
+
|
|
462
|
+
@asynccontextmanager
|
|
463
|
+
async def lifespan(self) -> AsyncIterator[None]:
|
|
464
|
+
"""User-overridable lifespan for custom setup and teardown.
|
|
465
|
+
|
|
466
|
+
Override this method to perform provider-specific initialization
|
|
467
|
+
like opening database connections, setting up external resources,
|
|
468
|
+
or other state management needed for the provider's lifetime.
|
|
469
|
+
|
|
470
|
+
The lifespan scope matches the server's lifespan - code before yield
|
|
471
|
+
runs at startup, code after yield runs at shutdown.
|
|
472
|
+
|
|
473
|
+
Example:
|
|
474
|
+
```python
|
|
475
|
+
@asynccontextmanager
|
|
476
|
+
async def lifespan(self):
|
|
477
|
+
# Setup
|
|
478
|
+
self.db = await connect_database()
|
|
479
|
+
try:
|
|
480
|
+
yield
|
|
481
|
+
finally:
|
|
482
|
+
# Teardown
|
|
483
|
+
await self.db.close()
|
|
484
|
+
```
|
|
485
|
+
"""
|
|
486
|
+
yield
|
|
487
|
+
|
|
488
|
+
# -------------------------------------------------------------------------
|
|
489
|
+
# Enable/Disable
|
|
490
|
+
# -------------------------------------------------------------------------
|
|
491
|
+
|
|
492
|
+
def enable(
|
|
493
|
+
self,
|
|
494
|
+
*,
|
|
495
|
+
names: set[str] | None = None,
|
|
496
|
+
keys: set[str] | None = None,
|
|
497
|
+
version: VersionSpec | None = None,
|
|
498
|
+
tags: set[str] | None = None,
|
|
499
|
+
components: set[Literal["tool", "resource", "template", "prompt"]]
|
|
500
|
+
| None = None,
|
|
501
|
+
only: bool = False,
|
|
502
|
+
) -> Self:
|
|
503
|
+
"""Enable components matching all specified criteria.
|
|
504
|
+
|
|
505
|
+
Adds a visibility transform that marks matching components as enabled.
|
|
506
|
+
Later transforms override earlier ones, so enable after disable makes
|
|
507
|
+
the component enabled.
|
|
508
|
+
|
|
509
|
+
With only=True, switches to allowlist mode - first disables everything,
|
|
510
|
+
then enables matching components.
|
|
511
|
+
|
|
512
|
+
Args:
|
|
513
|
+
names: Component names or URIs to enable.
|
|
514
|
+
keys: Component keys to enable (e.g., {"tool:my_tool@v1"}).
|
|
515
|
+
version: Component version spec to enable (e.g., VersionSpec(eq="v1") or
|
|
516
|
+
VersionSpec(gte="v2")). Unversioned components will not match.
|
|
517
|
+
tags: Enable components with these tags.
|
|
518
|
+
components: Component types to include (e.g., {"tool", "prompt"}).
|
|
519
|
+
only: If True, ONLY enable matching components (allowlist mode).
|
|
520
|
+
|
|
521
|
+
Returns:
|
|
522
|
+
Self for method chaining.
|
|
523
|
+
"""
|
|
524
|
+
if only:
|
|
525
|
+
# Allowlist: disable everything, then enable matching
|
|
526
|
+
# The enable transform runs later on return path, so it overrides
|
|
527
|
+
self._transforms.append(Visibility(False, match_all=True))
|
|
528
|
+
self._transforms.append(
|
|
529
|
+
Visibility(
|
|
530
|
+
True,
|
|
531
|
+
names=names,
|
|
532
|
+
keys=keys,
|
|
533
|
+
version=version,
|
|
534
|
+
components=set(components) if components else None,
|
|
535
|
+
tags=set(tags) if tags else None,
|
|
536
|
+
)
|
|
537
|
+
)
|
|
538
|
+
|
|
539
|
+
return self
|
|
540
|
+
|
|
541
|
+
def disable(
|
|
542
|
+
self,
|
|
543
|
+
*,
|
|
544
|
+
names: set[str] | None = None,
|
|
545
|
+
keys: set[str] | None = None,
|
|
546
|
+
version: VersionSpec | None = None,
|
|
547
|
+
tags: set[str] | None = None,
|
|
548
|
+
components: set[Literal["tool", "resource", "template", "prompt"]]
|
|
549
|
+
| None = None,
|
|
550
|
+
) -> Self:
|
|
551
|
+
"""Disable components matching all specified criteria.
|
|
552
|
+
|
|
553
|
+
Adds a visibility transform that marks matching components as disabled.
|
|
554
|
+
Components can be re-enabled by calling enable() with matching criteria
|
|
555
|
+
(the later transform wins).
|
|
556
|
+
|
|
557
|
+
Args:
|
|
558
|
+
names: Component names or URIs to disable.
|
|
559
|
+
keys: Component keys to disable (e.g., {"tool:my_tool@v1"}).
|
|
560
|
+
version: Component version spec to disable (e.g., VersionSpec(eq="v1") or
|
|
561
|
+
VersionSpec(gte="v2")). Unversioned components will not match.
|
|
562
|
+
tags: Disable components with these tags.
|
|
563
|
+
components: Component types to include (e.g., {"tool", "prompt"}).
|
|
564
|
+
|
|
565
|
+
Returns:
|
|
566
|
+
Self for method chaining.
|
|
567
|
+
"""
|
|
568
|
+
self._transforms.append(
|
|
569
|
+
Visibility(
|
|
570
|
+
False,
|
|
571
|
+
names=names,
|
|
572
|
+
keys=keys,
|
|
573
|
+
version=version,
|
|
574
|
+
components=set(components) if components else None,
|
|
575
|
+
tags=set(tags) if tags else None,
|
|
576
|
+
)
|
|
577
|
+
)
|
|
578
|
+
return self
|