Skip to content

Commit 81c5065

Browse files
authored
Merge branch 'main' into improve-claude-md
2 parents 2466010 + 7080fcf commit 81c5065

File tree

3 files changed

+87
-7
lines changed

3 files changed

+87
-7
lines changed

src/mcp/server/auth/routes.py

Lines changed: 2 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -10,15 +10,15 @@
1010
from starlette.types import ASGIApp
1111

1212
from mcp.server.auth.handlers.authorize import AuthorizationHandler
13-
from mcp.server.auth.handlers.metadata import MetadataHandler
13+
from mcp.server.auth.handlers.metadata import MetadataHandler, ProtectedResourceMetadataHandler
1414
from mcp.server.auth.handlers.register import RegistrationHandler
1515
from mcp.server.auth.handlers.revoke import RevocationHandler
1616
from mcp.server.auth.handlers.token import TokenHandler
1717
from mcp.server.auth.middleware.client_auth import ClientAuthenticator
1818
from mcp.server.auth.provider import OAuthAuthorizationServerProvider
1919
from mcp.server.auth.settings import ClientRegistrationOptions, RevocationOptions
2020
from mcp.server.streamable_http import MCP_PROTOCOL_VERSION_HEADER
21-
from mcp.shared.auth import OAuthMetadata
21+
from mcp.shared.auth import OAuthMetadata, ProtectedResourceMetadata
2222

2323

2424
def validate_issuer_url(url: AnyHttpUrl):
@@ -224,9 +224,6 @@ def create_protected_resource_routes(
224224
Returns:
225225
List of Starlette routes for protected resource metadata
226226
"""
227-
from mcp.server.auth.handlers.metadata import ProtectedResourceMetadataHandler
228-
from mcp.shared.auth import ProtectedResourceMetadata
229-
230227
metadata = ProtectedResourceMetadata(
231228
resource=resource_url,
232229
authorization_servers=authorization_servers,

src/mcp/server/fastmcp/resources/templates.py

Lines changed: 7 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66
import re
77
from collections.abc import Callable
88
from typing import TYPE_CHECKING, Any
9+
from urllib.parse import unquote
910

1011
from pydantic import BaseModel, Field, validate_call
1112

@@ -83,12 +84,16 @@ def from_function(
8384
)
8485

8586
def matches(self, uri: str) -> dict[str, Any] | None:
86-
"""Check if URI matches template and extract parameters."""
87+
"""Check if URI matches template and extract parameters.
88+
89+
Extracted parameters are URL-decoded to handle percent-encoded characters.
90+
"""
8791
# Convert template to regex pattern
8892
pattern = self.uri_template.replace("{", "(?P<").replace("}", ">[^/]+)")
8993
match = re.match(f"^{pattern}$", uri)
9094
if match:
91-
return match.groupdict()
95+
# URL-decode all extracted parameter values
96+
return {key: unquote(value) for key, value in match.groupdict().items()}
9297
return None
9398

9499
async def create_resource(
Lines changed: 78 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,78 @@
1+
"""Test that URL-encoded parameters are decoded in resource templates.
2+
3+
Regression test for https://github.com/modelcontextprotocol/python-sdk/issues/973
4+
"""
5+
6+
from mcp.server.fastmcp.resources import ResourceTemplate
7+
8+
9+
def test_template_matches_decodes_space():
10+
"""Test that %20 is decoded to space."""
11+
12+
def search(query: str) -> str: # pragma: no cover
13+
return f"Results for: {query}"
14+
15+
template = ResourceTemplate.from_function(
16+
fn=search,
17+
uri_template="search://{query}",
18+
name="search",
19+
)
20+
21+
params = template.matches("search://hello%20world")
22+
assert params is not None
23+
assert params["query"] == "hello world"
24+
25+
26+
def test_template_matches_decodes_accented_characters():
27+
"""Test that %C3%A9 is decoded to e with accent."""
28+
29+
def search(query: str) -> str: # pragma: no cover
30+
return f"Results for: {query}"
31+
32+
template = ResourceTemplate.from_function(
33+
fn=search,
34+
uri_template="search://{query}",
35+
name="search",
36+
)
37+
38+
params = template.matches("search://caf%C3%A9")
39+
assert params is not None
40+
assert params["query"] == "café"
41+
42+
43+
def test_template_matches_decodes_complex_phrase():
44+
"""Test complex French phrase from the original issue."""
45+
46+
def search(query: str) -> str: # pragma: no cover
47+
return f"Results for: {query}"
48+
49+
template = ResourceTemplate.from_function(
50+
fn=search,
51+
uri_template="search://{query}",
52+
name="search",
53+
)
54+
55+
params = template.matches("search://stick%20correcteur%20teint%C3%A9%20anti-imperfections")
56+
assert params is not None
57+
assert params["query"] == "stick correcteur teinté anti-imperfections"
58+
59+
60+
def test_template_matches_preserves_plus_sign():
61+
"""Test that plus sign remains as plus (not converted to space).
62+
63+
In URI encoding, %20 is space. Plus-as-space is only for
64+
application/x-www-form-urlencoded (HTML forms).
65+
"""
66+
67+
def search(query: str) -> str: # pragma: no cover
68+
return f"Results for: {query}"
69+
70+
template = ResourceTemplate.from_function(
71+
fn=search,
72+
uri_template="search://{query}",
73+
name="search",
74+
)
75+
76+
params = template.matches("search://hello+world")
77+
assert params is not None
78+
assert params["query"] == "hello+world"

0 commit comments

Comments
 (0)