-
Notifications
You must be signed in to change notification settings - Fork 3.5k
Expand file tree
/
Copy pathmessage.py
More file actions
75 lines (52 loc) · 2.56 KB
/
Copy pathmessage.py
File metadata and controls
75 lines (52 loc) · 2.56 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
69
70
71
72
73
74
75
"""Message wrapper with metadata support.
This module defines a wrapper type that combines JSONRPCMessage with metadata
to support transport-specific features like resumability.
"""
from collections.abc import Awaitable, Callable
from dataclasses import dataclass
from typing import Any, cast
from mcp.types import JSONRPCMessage, RequestId
ResumptionToken = str
ResumptionTokenUpdateCallback = Callable[[ResumptionToken], Awaitable[None]]
def extract_raw_request_id(raw_message: Any) -> RequestId | None:
"""Best-effort extraction of a JSON-RPC request id from an unvalidated payload.
Used to correlate error responses with the originating request when an incoming
message fails JSON-RPC envelope validation: even though the envelope is invalid,
the ``id`` member is often still present in the raw parsed JSON.
Args:
raw_message: The parsed JSON payload, before any envelope validation.
Returns:
The request id when it is a valid JSON-RPC id type (a string, or an integer
that is not a bool — ``bool`` subclasses ``int`` but is not a valid id),
otherwise ``None``.
"""
if isinstance(raw_message, dict):
raw_id = cast("dict[Any, Any]", raw_message).get("id")
if isinstance(raw_id, str) or (isinstance(raw_id, int) and not isinstance(raw_id, bool)):
return raw_id
return None
# Callback type for closing SSE streams without terminating
CloseSSEStreamCallback = Callable[[], Awaitable[None]]
@dataclass
class ClientMessageMetadata:
"""Metadata specific to client messages."""
resumption_token: ResumptionToken | None = None
on_resumption_token_update: Callable[[ResumptionToken], Awaitable[None]] | None = None
@dataclass
class ServerMessageMetadata:
"""Metadata specific to server messages."""
related_request_id: RequestId | None = None
# Transport-specific request context (e.g. starlette Request for HTTP
# transports, None for stdio). Typed as Any because the server layer is
# transport-agnostic.
request_context: Any = None
# Callback to close SSE stream for the current request without terminating
close_sse_stream: CloseSSEStreamCallback | None = None
# Callback to close the standalone GET SSE stream (for unsolicited notifications)
close_standalone_sse_stream: CloseSSEStreamCallback | None = None
MessageMetadata = ClientMessageMetadata | ServerMessageMetadata | None
@dataclass
class SessionMessage:
"""A message with specific metadata for transport-specific features."""
message: JSONRPCMessage
metadata: MessageMetadata = None