-
Notifications
You must be signed in to change notification settings - Fork 3.4k
Expand file tree
/
Copy pathdependencies.py
More file actions
68 lines (50 loc) · 1.68 KB
/
dependencies.py
File metadata and controls
68 lines (50 loc) · 1.68 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
"""Dependency injection system for MCPServer."""
from __future__ import annotations
import inspect
from collections.abc import Callable
from typing import Any, Generic, TypeVar
T = TypeVar("T")
class Depends(Generic[T]):
"""Marker class for dependency injection.
Usage:
def get_db() -> Database:
return Database()
@server.tool()
def my_tool(db: Database = Depends(get_db)):
return db.query(...)
Args:
dependency: A callable that provides the dependency
scope: The scope of the dependency (for future use)
use_cache: Whether to cache the dependency result
"""
def __init__(
self,
dependency: Callable[..., T],
*,
use_cache: bool = True,
) -> None:
self.dependency = dependency
self.use_cache = use_cache
def __repr__(self) -> str:
return f"Depends({self.dependency.__name__})"
def find_dependency_parameters(
fn: Callable[..., Any],
) -> dict[str, Depends[Any]]:
"""Find all parameters with Depends() default values.
Args:
fn: Function to inspect
Returns:
Dict mapping parameter names to Depends instances
"""
deps: dict[str, Depends[Any]] = {}
try:
sig = inspect.signature(fn, eval_str=True)
except (ValueError, TypeError): # pragma: no cover (defensive)
return deps
for param_name, param in sig.parameters.items():
if param.default is inspect.Parameter.empty:
continue
# Check if default is Depends instance
if isinstance(param.default, Depends):
deps[param_name] = param.default # type: ignore[assignment]
return deps