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.
Files changed (175) hide show
  1. fastmcp/_vendor/__init__.py +1 -0
  2. fastmcp/_vendor/docket_di/README.md +7 -0
  3. fastmcp/_vendor/docket_di/__init__.py +163 -0
  4. fastmcp/cli/cli.py +112 -28
  5. fastmcp/cli/install/claude_code.py +1 -5
  6. fastmcp/cli/install/claude_desktop.py +1 -5
  7. fastmcp/cli/install/cursor.py +1 -5
  8. fastmcp/cli/install/gemini_cli.py +1 -5
  9. fastmcp/cli/install/mcp_json.py +1 -6
  10. fastmcp/cli/run.py +146 -5
  11. fastmcp/client/__init__.py +7 -9
  12. fastmcp/client/auth/oauth.py +18 -17
  13. fastmcp/client/client.py +100 -870
  14. fastmcp/client/elicitation.py +1 -1
  15. fastmcp/client/mixins/__init__.py +13 -0
  16. fastmcp/client/mixins/prompts.py +295 -0
  17. fastmcp/client/mixins/resources.py +325 -0
  18. fastmcp/client/mixins/task_management.py +157 -0
  19. fastmcp/client/mixins/tools.py +397 -0
  20. fastmcp/client/sampling/handlers/anthropic.py +2 -2
  21. fastmcp/client/sampling/handlers/openai.py +1 -1
  22. fastmcp/client/tasks.py +3 -3
  23. fastmcp/client/telemetry.py +47 -0
  24. fastmcp/client/transports/__init__.py +38 -0
  25. fastmcp/client/transports/base.py +82 -0
  26. fastmcp/client/transports/config.py +170 -0
  27. fastmcp/client/transports/http.py +145 -0
  28. fastmcp/client/transports/inference.py +154 -0
  29. fastmcp/client/transports/memory.py +90 -0
  30. fastmcp/client/transports/sse.py +89 -0
  31. fastmcp/client/transports/stdio.py +543 -0
  32. fastmcp/contrib/component_manager/README.md +4 -10
  33. fastmcp/contrib/component_manager/__init__.py +1 -2
  34. fastmcp/contrib/component_manager/component_manager.py +95 -160
  35. fastmcp/contrib/component_manager/example.py +1 -1
  36. fastmcp/contrib/mcp_mixin/example.py +4 -4
  37. fastmcp/contrib/mcp_mixin/mcp_mixin.py +11 -4
  38. fastmcp/decorators.py +41 -0
  39. fastmcp/dependencies.py +12 -1
  40. fastmcp/exceptions.py +4 -0
  41. fastmcp/experimental/server/openapi/__init__.py +18 -15
  42. fastmcp/mcp_config.py +13 -4
  43. fastmcp/prompts/__init__.py +6 -3
  44. fastmcp/prompts/function_prompt.py +465 -0
  45. fastmcp/prompts/prompt.py +321 -271
  46. fastmcp/resources/__init__.py +5 -3
  47. fastmcp/resources/function_resource.py +335 -0
  48. fastmcp/resources/resource.py +325 -115
  49. fastmcp/resources/template.py +215 -43
  50. fastmcp/resources/types.py +27 -12
  51. fastmcp/server/__init__.py +2 -2
  52. fastmcp/server/auth/__init__.py +14 -0
  53. fastmcp/server/auth/auth.py +30 -10
  54. fastmcp/server/auth/authorization.py +190 -0
  55. fastmcp/server/auth/oauth_proxy/__init__.py +14 -0
  56. fastmcp/server/auth/oauth_proxy/consent.py +361 -0
  57. fastmcp/server/auth/oauth_proxy/models.py +178 -0
  58. fastmcp/server/auth/{oauth_proxy.py → oauth_proxy/proxy.py} +24 -778
  59. fastmcp/server/auth/oauth_proxy/ui.py +277 -0
  60. fastmcp/server/auth/oidc_proxy.py +2 -2
  61. fastmcp/server/auth/providers/auth0.py +24 -94
  62. fastmcp/server/auth/providers/aws.py +26 -95
  63. fastmcp/server/auth/providers/azure.py +41 -129
  64. fastmcp/server/auth/providers/descope.py +18 -49
  65. fastmcp/server/auth/providers/discord.py +25 -86
  66. fastmcp/server/auth/providers/github.py +23 -87
  67. fastmcp/server/auth/providers/google.py +24 -87
  68. fastmcp/server/auth/providers/introspection.py +60 -79
  69. fastmcp/server/auth/providers/jwt.py +30 -67
  70. fastmcp/server/auth/providers/oci.py +47 -110
  71. fastmcp/server/auth/providers/scalekit.py +23 -61
  72. fastmcp/server/auth/providers/supabase.py +18 -47
  73. fastmcp/server/auth/providers/workos.py +34 -127
  74. fastmcp/server/context.py +372 -419
  75. fastmcp/server/dependencies.py +541 -251
  76. fastmcp/server/elicitation.py +20 -18
  77. fastmcp/server/event_store.py +3 -3
  78. fastmcp/server/http.py +16 -6
  79. fastmcp/server/lifespan.py +198 -0
  80. fastmcp/server/low_level.py +92 -2
  81. fastmcp/server/middleware/__init__.py +5 -1
  82. fastmcp/server/middleware/authorization.py +312 -0
  83. fastmcp/server/middleware/caching.py +101 -54
  84. fastmcp/server/middleware/middleware.py +6 -9
  85. fastmcp/server/middleware/ping.py +70 -0
  86. fastmcp/server/middleware/tool_injection.py +2 -2
  87. fastmcp/server/mixins/__init__.py +7 -0
  88. fastmcp/server/mixins/lifespan.py +217 -0
  89. fastmcp/server/mixins/mcp_operations.py +392 -0
  90. fastmcp/server/mixins/transport.py +342 -0
  91. fastmcp/server/openapi/__init__.py +41 -21
  92. fastmcp/server/openapi/components.py +16 -339
  93. fastmcp/server/openapi/routing.py +34 -118
  94. fastmcp/server/openapi/server.py +67 -392
  95. fastmcp/server/providers/__init__.py +71 -0
  96. fastmcp/server/providers/aggregate.py +261 -0
  97. fastmcp/server/providers/base.py +578 -0
  98. fastmcp/server/providers/fastmcp_provider.py +674 -0
  99. fastmcp/server/providers/filesystem.py +226 -0
  100. fastmcp/server/providers/filesystem_discovery.py +327 -0
  101. fastmcp/server/providers/local_provider/__init__.py +11 -0
  102. fastmcp/server/providers/local_provider/decorators/__init__.py +15 -0
  103. fastmcp/server/providers/local_provider/decorators/prompts.py +256 -0
  104. fastmcp/server/providers/local_provider/decorators/resources.py +240 -0
  105. fastmcp/server/providers/local_provider/decorators/tools.py +315 -0
  106. fastmcp/server/providers/local_provider/local_provider.py +465 -0
  107. fastmcp/server/providers/openapi/__init__.py +39 -0
  108. fastmcp/server/providers/openapi/components.py +332 -0
  109. fastmcp/server/providers/openapi/provider.py +405 -0
  110. fastmcp/server/providers/openapi/routing.py +109 -0
  111. fastmcp/server/providers/proxy.py +867 -0
  112. fastmcp/server/providers/skills/__init__.py +59 -0
  113. fastmcp/server/providers/skills/_common.py +101 -0
  114. fastmcp/server/providers/skills/claude_provider.py +44 -0
  115. fastmcp/server/providers/skills/directory_provider.py +153 -0
  116. fastmcp/server/providers/skills/skill_provider.py +432 -0
  117. fastmcp/server/providers/skills/vendor_providers.py +142 -0
  118. fastmcp/server/providers/wrapped_provider.py +140 -0
  119. fastmcp/server/proxy.py +34 -700
  120. fastmcp/server/sampling/run.py +341 -2
  121. fastmcp/server/sampling/sampling_tool.py +4 -3
  122. fastmcp/server/server.py +1214 -2171
  123. fastmcp/server/tasks/__init__.py +2 -1
  124. fastmcp/server/tasks/capabilities.py +13 -1
  125. fastmcp/server/tasks/config.py +66 -3
  126. fastmcp/server/tasks/handlers.py +65 -273
  127. fastmcp/server/tasks/keys.py +4 -6
  128. fastmcp/server/tasks/requests.py +474 -0
  129. fastmcp/server/tasks/routing.py +76 -0
  130. fastmcp/server/tasks/subscriptions.py +20 -11
  131. fastmcp/server/telemetry.py +131 -0
  132. fastmcp/server/transforms/__init__.py +244 -0
  133. fastmcp/server/transforms/namespace.py +193 -0
  134. fastmcp/server/transforms/prompts_as_tools.py +175 -0
  135. fastmcp/server/transforms/resources_as_tools.py +190 -0
  136. fastmcp/server/transforms/tool_transform.py +96 -0
  137. fastmcp/server/transforms/version_filter.py +124 -0
  138. fastmcp/server/transforms/visibility.py +526 -0
  139. fastmcp/settings.py +34 -96
  140. fastmcp/telemetry.py +122 -0
  141. fastmcp/tools/__init__.py +10 -3
  142. fastmcp/tools/function_parsing.py +201 -0
  143. fastmcp/tools/function_tool.py +467 -0
  144. fastmcp/tools/tool.py +215 -362
  145. fastmcp/tools/tool_transform.py +38 -21
  146. fastmcp/utilities/async_utils.py +69 -0
  147. fastmcp/utilities/components.py +152 -91
  148. fastmcp/utilities/inspect.py +8 -20
  149. fastmcp/utilities/json_schema.py +12 -5
  150. fastmcp/utilities/json_schema_type.py +17 -15
  151. fastmcp/utilities/lifespan.py +56 -0
  152. fastmcp/utilities/logging.py +12 -4
  153. fastmcp/utilities/mcp_server_config/v1/mcp_server_config.py +3 -3
  154. fastmcp/utilities/openapi/parser.py +3 -3
  155. fastmcp/utilities/pagination.py +80 -0
  156. fastmcp/utilities/skills.py +253 -0
  157. fastmcp/utilities/tests.py +0 -16
  158. fastmcp/utilities/timeout.py +47 -0
  159. fastmcp/utilities/types.py +1 -1
  160. fastmcp/utilities/versions.py +285 -0
  161. {fastmcp-2.14.4.dist-info → fastmcp-3.0.0b1.dist-info}/METADATA +8 -5
  162. fastmcp-3.0.0b1.dist-info/RECORD +228 -0
  163. fastmcp/client/transports.py +0 -1170
  164. fastmcp/contrib/component_manager/component_service.py +0 -209
  165. fastmcp/prompts/prompt_manager.py +0 -117
  166. fastmcp/resources/resource_manager.py +0 -338
  167. fastmcp/server/tasks/converters.py +0 -206
  168. fastmcp/server/tasks/protocol.py +0 -359
  169. fastmcp/tools/tool_manager.py +0 -170
  170. fastmcp/utilities/mcp_config.py +0 -56
  171. fastmcp-2.14.4.dist-info/RECORD +0 -161
  172. /fastmcp/server/{openapi → providers/openapi}/README.md +0 -0
  173. {fastmcp-2.14.4.dist-info → fastmcp-3.0.0b1.dist-info}/WHEEL +0 -0
  174. {fastmcp-2.14.4.dist-info → fastmcp-3.0.0b1.dist-info}/entry_points.txt +0 -0
  175. {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