Teknium
38d8446011
feat: implement MCP OAuth 2.1 PKCE client support (#5420)
Implement tools/mcp_oauth.py — the OAuth adapter that mcp_tool.py's
existing auth: oauth hook has been waiting for.
Components:
- HermesTokenStorage: persists tokens + client registration to
HERMES_HOME/mcp-tokens/<server>.json with 0o600 permissions
- Callback handler factory: per-flow isolated HTTP handlers (safe for
concurrent OAuth flows across multiple MCP servers)
- OAuthClientProvider integration: wraps the MCP SDK's httpx.Auth
subclass which handles discovery, DCR, PKCE, token exchange,
refresh, and step-up auth (403 insufficient_scope) automatically
- Non-interactive detection: warns when gateway/cron environments
try to OAuth without cached tokens
- Pre-registered client support: injects client_id/secret from config
for servers that don't support Dynamic Client Registration (e.g. Slack)
- Path traversal protection on server names
- remove_oauth_tokens() for cleanup
Config format:
mcp_servers:
sentry:
url: 'https://mcp.sentry.dev/mcp'
auth: oauth
oauth: # all optional
client_id: '...' # skip DCR
client_secret: '...' # confidential client
scope: 'read write' # server-provided by default
Also passes oauth config dict through from mcp_tool.py (was passing
only server_name and url before).
E2E verified: full OAuth flow (401 → discovery → DCR → authorize →
token exchange → authenticated request → tokens persisted) against
local test servers. 23 unit tests + 186 MCP suite tests pass.
2026-04-05 22:08:00 -07:00
..
2026-03-30 13:28:10 +09:00
2026-04-04 12:57:49 -07:00
2026-03-17 02:33:12 -07:00
2026-03-31 08:48:54 +09:00
2026-03-23 07:43:12 -07:00
2026-04-05 12:31:27 -07:00
2026-04-01 04:18:50 -07:00
2026-04-01 12:03:56 -07:00
2026-04-05 12:42:52 -07:00
2026-03-25 19:47:58 -07:00
2026-03-25 15:02:03 -07:00
2026-04-04 12:57:49 -07:00
2026-04-03 21:14:42 -07:00
2026-04-04 10:43:39 -07:00
2026-04-05 12:48:50 -07:00
2026-03-24 08:19:34 -07:00
2026-04-04 10:18:57 -07:00
2026-04-03 22:40:37 -07:00
2026-03-25 19:47:58 -07:00
2026-03-15 20:21:21 -07:00
2026-03-30 13:28:10 +09:00
2026-04-02 12:40:03 +11:00
2026-04-05 22:08:00 -07:00
2026-04-05 22:08:00 -07:00
2026-04-03 13:10:11 -07:00
2026-03-27 15:28:19 -07:00
2026-03-18 02:55:30 -07:00
2026-03-11 20:02:36 -07:00
2026-04-05 12:46:07 -07:00
2026-03-26 19:38:04 -07:00
2026-03-25 15:02:03 -07:00
2026-03-29 15:52:54 -07:00
2026-03-25 19:47:58 -07:00
2026-04-05 11:07:47 -07:00
2026-03-27 21:27:51 -07:00
2026-04-03 21:14:42 -07:00
2026-03-27 15:28:19 -07:00
2026-04-01 04:19:19 -07:00
2026-03-25 15:54:28 -07:00
2026-04-03 21:14:42 -07:00
2026-04-04 16:57:24 -07:00
2026-03-25 19:47:58 -07:00
2026-03-15 20:21:21 -07:00
2026-03-31 08:48:54 +09:00
2026-04-03 00:50:17 -07:00
2026-04-03 22:42:14 -07:00
2026-03-23 15:40:42 -07:00
2026-03-30 02:59:39 -07:00
2026-03-31 12:13:33 -07:00
2026-04-05 11:16:45 -07:00
2026-03-25 15:54:28 -07:00