Abylaikhan Zulbukharov
1d80bf5e65
feat(mcp): add Streamable HTTP transport with OAuth 2.0 ( #1033 )
...
* feat(mcp): add Streamable HTTP transport with OAuth 2.0
Odysseus could only reach MCP servers over stdio and SSE, so modern
remote servers like https://mcp.higgsfield.ai/mcp (Streamable HTTP,
gated behind OAuth) could not be connected.
Add an `http` transport that connects via the SDK's
streamablehttp_client and authenticates with the SDK's
OAuthClientProvider: RFC 9728 protected-resource discovery, RFC 8414
authorization-server metadata, Dynamic Client Registration,
authorization-code + PKCE, and token refresh. A small bridge
(src/mcp_oauth.py) connects the SDK's blocking callback to the existing
web callback route via an asyncio.Future keyed by the OAuth `state`,
and the dynamic client registration plus tokens persist per-server in a
new encrypted `oauth_tokens` column.
The connect runs as a bounded background task so the "Add server"
request returns immediately; redirect_handler publishes needs_auth +
auth_url to connection state as soon as discovery/DCR completes (which
can exceed the bounded wait), and the UI polls until connected. Remote
users finish via the existing paste-back flow. The Google OAuth path is
left unchanged.
- core/database.py: encrypted oauth_tokens column + migration
- src/mcp_oauth.py: OAuth provider, DB-backed TokenStorage, state registry
- src/mcp_manager.py: http dispatch, background connect, _connect_http
- routes/mcp_routes.py: http validation, needs_auth/auth_url, callback bridge
- static/js/settings.js: Streamable HTTP option + OAuth flow with polling
- tests: 5 new unit tests (transport dispatch, registry, token storage)
Verified against the live Higgsfield server: discovery, DCR (client_id
issued), loopback redirect accepted, and a PKCE authorization URL with
needs_auth status. No regressions (full suite delta is only the 5 added
passing tests).
* fix(mcp): address PR #1033 review feedback
- mcp_oauth: derive redirect URI from OAUTH_REDIRECT_BASE_URL/APP_PUBLIC_URL
(default http://localhost:7000 ) instead of hardcoding the port
- mcp_oauth: leave OAuth scope unset so the SDK derives it from the server's
WWW-Authenticate/protected-resource metadata; hardcoding an OIDC scope broke
non-OpenID MCP servers (verified: Higgsfield still gets its server-derived
scope)
- mcp_oauth: prune abandoned OAuth flows (_prune_stale + _pending_ts) so the
module-level registries can't grow unbounded
- mcp_oauth: persist tokens/client-info in a single DB session/commit
(_update) instead of a load+save double round-trip
- mcp_manager: cancel and drop the background connect task in
disconnect_server so a deleted server stops publishing status
- database: document why the oauth_tokens migration uses TEXT while the model
declares EncryptedText (encryption is applied at the Python layer)
- settings.js: surface persistent OAuth-poll failures and an explicit timeout
message instead of silently swallowing errors
- tests: cover the stale-flow pruning
* static/js/settings.js now shows an in-flight loading state on the buttons that fire requests:
2026-06-05 02:40:52 +02:00
..
2026-06-04 15:44:25 +01:00
2026-05-31 23:58:26 +09:00
2026-06-04 11:39:49 +01:00
2026-06-05 02:28:42 +02:00
2026-06-03 14:23:10 +09:00
2026-06-04 13:20:04 +01:00
2026-06-03 01:47:13 +09:00
2026-06-02 20:35:57 +09:00
2026-06-04 13:00:17 +00:00
2026-06-04 22:36:05 +02:00
2026-06-03 14:06:39 +09:00
2026-06-04 00:37:28 +01:00
2026-06-03 08:56:48 +09:00
2026-06-03 00:57:20 +09:00
2026-06-03 22:47:00 +09:00
2026-06-03 08:11:37 +09:00
2026-06-02 20:28:26 +09:00
2026-06-02 20:42:32 +09:00
2026-06-02 01:32:00 +02:00
2026-06-01 02:22:17 +00:00
2026-06-04 02:32:54 +01:00
2026-06-03 14:14:16 +09:00
2026-06-03 14:23:40 +09:00
2026-06-04 11:51:47 +01:00
2026-06-03 13:37:54 +09:00
2026-06-02 06:02:49 +09:00
2026-06-03 08:38:27 +09:00
2026-06-03 13:29:14 +09:00
2026-06-03 14:07:14 +09:00
2026-06-03 14:11:45 +09:00
2026-06-04 21:05:52 +01:00
2026-06-04 18:20:41 +01:00
2026-06-03 08:59:16 +09:00
2026-06-02 12:40:28 +09:00
2026-06-04 04:01:21 +01:00
2026-06-03 02:50:02 +09:00
2026-06-03 08:35:16 +09:00
2026-06-03 01:44:02 +09:00
2026-06-03 08:57:15 +09:00
2026-06-04 15:44:25 +01:00
2026-06-02 23:14:52 +09:00
2026-06-02 20:42:37 +09:00
2026-06-03 14:14:41 +09:00
2026-06-01 13:42:44 +09:00
2026-06-03 14:24:14 +09:00
2026-06-04 00:27:29 +01:00
2026-06-03 13:37:39 +09:00
2026-06-03 14:16:58 +09:00
2026-06-03 14:16:55 +09:00
2026-06-02 20:55:30 +09:00
2026-06-02 21:00:58 +09:00
2026-06-04 19:26:58 +02:00
2026-06-04 00:27:29 +01:00
2026-06-02 20:52:08 +09:00
2026-06-01 22:33:35 +09:00
2026-06-04 20:49:37 +02:00
2026-06-03 08:59:49 +09:00
2026-06-01 22:22:41 +09:00
2026-06-03 14:06:57 +09:00
2026-06-02 20:42:21 +09:00
2026-06-03 14:14:27 +09:00
2026-06-04 18:37:32 +02:00
2026-06-04 00:27:29 +01:00
2026-06-02 13:15:22 +09:00
2026-06-04 19:17:15 +01:00
2026-06-01 22:24:27 +09:00
2026-06-03 14:23:01 +09:00
2026-06-03 14:23:34 +09:00
2026-06-03 14:11:21 +09:00
2026-06-03 14:24:04 +09:00
2026-06-03 01:36:57 +09:00
2026-06-03 14:12:14 +09:00
2026-06-03 13:25:33 +09:00
2026-06-03 14:11:56 +09:00
2026-06-03 04:26:15 +09:00
2026-06-04 12:55:06 +01:00
2026-06-03 03:48:25 +09:00
2026-06-03 00:09:48 +09:00
2026-06-04 23:35:34 +01:00
2026-06-02 22:52:37 +09:00
2026-06-03 13:23:35 +09:00
2026-06-04 21:13:14 +02:00
2026-06-04 21:13:14 +02:00
2026-06-04 02:50:19 +01:00
2026-06-02 12:25:56 +09:00
2026-06-03 03:00:52 +09:00
2026-06-04 20:23:17 +02:00
2026-06-03 14:23:07 +09:00
2026-06-02 20:34:25 +09:00
2026-06-03 08:11:44 +09:00
2026-06-03 04:20:48 +09:00
2026-06-04 04:44:34 +01:00
2026-06-04 04:44:34 +01:00
2026-06-02 23:52:22 +09:00
2026-06-03 13:25:41 +09:00
2026-06-02 12:41:25 +09:00
2026-06-03 01:03:58 +09:00
2026-06-04 00:32:50 +01:00
2026-06-03 13:28:31 +09:00
2026-06-04 15:44:25 +01:00
2026-06-03 03:31:09 +09:00
2026-06-03 13:35:01 +09:00
2026-06-03 08:59:55 +09:00
2026-06-04 00:27:29 +01:00
2026-06-02 11:48:54 +09:00
2026-06-03 13:40:19 +09:00
2026-06-04 04:42:23 +01:00
2026-06-03 13:28:38 +09:00
2026-06-02 20:35:27 +09:00
2026-06-03 14:23:43 +09:00
2026-06-03 13:23:01 +09:00
2026-06-02 06:00:02 +09:00
2026-06-04 23:22:02 +01:00
2026-06-03 14:07:03 +09:00
2026-06-03 14:24:20 +09:00
2026-06-03 14:23:58 +09:00
2026-06-03 13:28:53 +09:00
2026-06-02 23:11:04 +09:00
2026-06-02 20:28:01 +09:00
2026-06-04 20:47:47 +02:00
2026-06-05 02:21:50 +02:00
2026-06-04 02:56:05 +01:00
2026-06-02 13:15:06 +09:00
2026-06-02 23:28:24 +09:00
2026-06-03 14:06:31 +09:00
2026-06-03 14:12:41 +09:00
2026-06-05 02:28:42 +02:00
2026-06-04 17:53:18 +01:00
2026-06-02 11:10:43 +09:00
2026-06-01 14:23:22 -04:00
2026-06-02 22:34:52 +09:00
2026-06-03 08:59:36 +09:00
2026-06-02 22:35:30 +09:00
2026-06-03 14:24:30 +09:00
2026-06-03 14:24:07 +09:00
2026-06-02 20:32:56 +09:00
2026-06-03 14:14:37 +09:00
2026-06-04 15:44:25 +01:00
2026-06-04 15:44:25 +01:00
2026-06-02 20:34:05 +09:00
2026-06-03 13:34:17 +09:00
2026-06-03 14:23:04 +09:00
2026-06-01 22:35:24 +09:00
2026-06-03 13:23:08 +09:00
2026-06-03 13:45:56 +09:00
2026-06-03 22:47:00 +09:00
2026-06-03 04:05:40 +09:00
2026-06-03 00:30:03 +09:00
2026-06-04 21:50:16 +02:00
2026-06-03 13:30:11 +09:00
2026-06-03 14:22:51 +09:00
2026-06-02 11:36:01 +09:00
2026-06-02 21:01:42 +09:00
2026-06-03 14:11:10 +09:00
2026-06-01 22:47:47 +09:00
2026-06-02 23:12:34 +09:00
2026-06-03 14:22:42 +09:00
2026-06-03 14:23:30 +09:00
2026-06-02 13:07:20 +09:00
2026-06-03 03:19:39 +09:00
2026-06-04 20:02:13 +02:00
2026-06-04 03:53:14 +01:00
2026-06-02 22:36:12 +09:00
2026-06-02 23:02:28 +09:00
2026-06-03 14:22:49 +09:00
2026-06-04 03:23:59 +01:00
2026-06-04 03:26:35 +01:00
2026-06-03 09:00:04 +09:00
2026-06-03 14:07:11 +09:00
2026-06-03 13:24:24 +09:00
2026-06-03 13:24:33 +09:00
2026-06-02 11:12:54 +09:00
2026-06-03 13:29:21 +09:00
2026-06-02 11:14:31 +09:00
2026-06-03 13:29:36 +09:00
2026-06-02 05:54:23 +09:00
2026-06-03 13:33:50 +09:00
2026-06-02 20:27:24 +09:00
2026-06-03 01:41:24 +09:00
2026-06-04 20:18:19 +02:00
2026-06-04 00:27:29 +01:00
2026-06-03 13:37:14 +09:00
2026-06-02 11:39:40 +09:00
2026-06-05 00:10:11 +02:00
2026-06-03 22:47:00 +09:00
2026-06-04 13:53:10 +01:00
2026-06-02 23:04:58 +09:00
2026-06-02 23:01:04 +09:00
2026-06-02 20:36:54 +09:00
2026-06-02 11:47:58 +09:00
2026-06-03 08:59:30 +09:00
2026-06-03 13:31:10 +09:00
2026-06-03 08:57:23 +09:00
2026-06-03 01:53:47 +09:00
2026-06-04 20:46:10 +02:00
2026-06-05 02:28:42 +02:00
2026-06-03 14:17:02 +09:00
2026-06-03 09:00:10 +09:00
2026-06-02 11:28:52 +09:00
2026-06-04 03:05:37 +01:00
2026-06-03 00:49:29 +09:00
2026-06-04 15:44:25 +01:00
2026-06-04 15:44:25 +01:00
2026-06-03 08:35:54 +09:00
2026-06-03 13:45:33 +09:00
2026-06-05 02:40:52 +02:00
2026-06-05 02:40:52 +02:00
2026-06-03 03:46:07 +09:00
2026-06-04 12:51:31 +00:00
2026-06-03 13:41:46 +09:00
2026-06-03 14:11:42 +09:00
2026-06-03 13:25:48 +09:00
2026-06-03 14:07:00 +09:00
2026-06-02 11:27:31 +09:00
2026-06-04 05:31:15 +01:00
2026-06-04 16:26:11 +01:00
2026-06-03 13:35:09 +09:00
2026-06-03 23:13:56 +01:00
2026-06-03 13:38:02 +09:00
2026-06-04 19:47:08 +02:00
2026-06-03 14:11:49 +09:00
2026-06-02 23:31:09 +09:00
2026-06-04 04:56:11 +01:00
2026-06-03 14:11:31 +09:00
2026-06-04 17:53:18 +01:00
2026-06-03 14:16:52 +09:00
2026-06-03 04:07:31 +09:00
2026-06-02 21:00:50 +09:00
2026-06-04 12:06:24 +02:00
2026-06-04 15:44:25 +01:00
2026-06-04 20:51:23 +02:00
2026-06-02 20:51:16 +09:00
2026-06-04 18:39:45 +01:00
2026-06-03 08:56:56 +09:00
2026-06-02 11:41:33 +09:00
2026-06-02 12:24:18 +09:00
2026-06-03 08:57:47 +09:00
2026-06-03 01:20:43 +09:00
2026-06-01 18:27:17 +09:00
2026-06-03 14:06:42 +09:00
2026-06-03 13:29:57 +09:00
2026-06-02 20:35:44 +09:00
2026-06-03 13:42:05 +09:00
2026-06-03 08:37:29 +09:00
2026-06-02 11:28:52 +09:00
2026-06-01 22:33:35 +09:00
2026-06-03 04:02:16 +09:00
2026-06-01 22:36:53 +09:00
2026-06-04 04:30:01 +01:00
2026-06-04 20:52:41 +02:00
2026-06-04 12:51:48 +01:00
2026-06-04 03:55:22 +01:00
2026-06-03 14:12:45 +09:00
2026-06-03 13:23:50 +09:00
2026-06-03 08:57:35 +09:00
2026-06-03 14:10:58 +09:00
2026-06-03 03:59:05 +09:00
2026-06-02 20:32:08 +09:00
2026-06-03 04:09:28 +09:00
2026-06-03 14:12:31 +09:00
2026-06-02 20:42:43 +09:00
2026-06-02 11:11:17 +09:00
2026-06-02 12:53:50 +09:00
2026-06-03 14:23:21 +09:00
2026-06-03 14:11:14 +09:00
2026-06-03 01:10:06 +09:00
2026-06-03 13:31:33 +09:00
2026-06-03 13:29:51 +09:00
2026-06-03 08:36:45 +09:00
2026-06-04 03:49:59 +01:00
2026-05-31 23:58:26 +09:00
2026-06-02 23:33:22 +09:00
2026-06-03 03:42:01 +09:00
2026-06-02 23:18:15 +09:00
2026-06-04 20:37:59 +02:00
2026-06-04 03:14:43 +01:00
2026-06-03 13:37:22 +09:00
2026-06-02 11:42:20 +09:00
2026-06-03 02:32:38 +09:00
2026-06-03 08:36:57 +09:00
2026-06-03 14:12:38 +09:00
2026-06-03 23:19:28 +01:00
2026-06-03 13:29:29 +09:00
2026-06-03 13:34:40 +09:00
2026-06-02 20:42:15 +09:00
2026-06-02 22:51:25 +09:00
2026-06-02 11:30:53 +09:00
2026-06-03 03:24:09 +09:00
2026-06-03 08:57:09 +09:00
2026-06-01 23:25:38 +01:00
2026-06-04 20:41:35 +02:00
2026-06-03 13:42:24 +09:00
2026-06-01 22:26:37 +09:00
2026-06-02 05:58:58 +09:00
2026-06-02 12:24:50 +09:00
2026-06-03 13:34:33 +09:00
2026-06-04 22:28:00 +01:00
2026-06-04 03:14:43 +01:00
2026-06-03 01:21:57 +09:00
2026-06-03 13:44:18 +09:00
2026-06-03 14:14:30 +09:00
2026-06-03 14:12:07 +09:00
2026-06-04 18:57:24 +02:00
2026-06-02 10:53:33 +09:00
2026-06-03 13:24:17 +09:00
2026-06-02 23:37:15 +09:00
2026-06-03 14:22:55 +09:00
2026-06-04 18:10:55 +02:00
2026-06-04 18:10:55 +02:00
2026-06-04 18:10:55 +02:00
2026-06-03 14:24:23 +09:00
2026-06-03 13:30:28 +09:00
2026-06-03 08:36:01 +09:00
2026-06-01 23:09:41 +09:00
2026-06-03 22:47:00 +09:00
2026-06-03 14:23:52 +09:00
2026-06-03 22:47:00 +09:00
2026-06-04 14:19:51 +01:00
2026-06-03 13:33:56 +09:00
2026-06-03 03:56:54 +09:00
2026-06-04 21:05:52 +01:00
2026-06-02 23:09:15 +09:00
2026-06-03 13:22:52 +09:00
2026-06-02 12:34:42 +09:00
2026-06-02 21:02:26 +09:00
2026-06-03 14:22:58 +09:00
2026-06-03 13:45:07 +09:00
2026-06-04 21:42:23 +02:00
2026-06-03 02:40:22 +09:00
2026-06-03 08:35:47 +09:00
2026-06-04 12:53:44 +01:00
2026-06-04 21:05:52 +01:00
2026-06-03 14:24:01 +09:00
2026-06-01 16:55:09 +09:00
2026-06-04 21:43:25 +01:00
2026-06-03 08:57:41 +09:00
2026-06-03 14:23:27 +09:00
2026-06-03 14:11:05 +09:00
2026-06-03 03:53:05 +09:00
2026-06-01 22:38:56 +09:00
2026-06-02 23:20:00 +09:00
2026-06-03 13:37:46 +09:00
2026-06-03 14:10:54 +09:00
2026-06-03 14:16:48 +09:00
2026-06-03 13:22:46 +09:00
2026-06-04 20:51:23 +02:00
2026-06-03 14:06:53 +09:00
2026-06-02 11:15:45 +09:00
2026-06-03 03:16:11 +09:00
2026-06-03 08:37:05 +09:00
2026-06-03 14:11:38 +09:00
2026-06-02 20:28:36 +09:00
2026-06-03 02:27:43 +09:00
2026-06-03 08:59:24 +09:00
2026-06-03 00:42:56 +09:00
2026-06-03 14:24:11 +09:00
2026-06-03 13:34:25 +09:00
2026-06-02 10:44:39 +09:00
2026-06-04 18:11:42 +01:00
2026-06-04 00:27:29 +01:00
2026-06-04 18:10:55 +02:00
2026-06-03 14:23:15 +09:00
2026-06-04 19:26:58 +02:00
2026-06-02 20:43:24 +09:00
2026-06-02 20:51:21 +09:00
2026-06-04 00:27:29 +01:00
2026-06-04 15:44:25 +01:00
2026-06-03 13:31:26 +09:00
2026-06-03 14:12:35 +09:00
2026-06-04 13:18:50 +01:00
2026-06-02 20:32:20 +09:00
2026-06-03 08:59:42 +09:00
2026-06-02 23:13:30 +09:00
2026-06-03 13:33:43 +09:00
2026-06-04 11:45:10 +01:00
2026-06-04 00:27:29 +01:00
2026-06-03 01:26:47 +09:00
2026-06-04 21:19:16 +02:00
2026-06-02 20:43:29 +09:00
2026-06-03 14:12:03 +09:00
2026-06-03 13:24:00 +09:00
2026-06-02 12:57:45 +09:00
2026-06-03 08:57:03 +09:00
2026-06-03 04:12:23 +09:00
2026-06-03 00:59:01 +09:00
2026-06-03 13:28:45 +09:00
2026-06-03 01:16:30 +09:00
2026-06-03 04:04:19 +09:00
2026-06-02 20:42:26 +09:00
2026-06-02 23:46:33 +09:00
2026-06-04 13:20:04 +01:00
2026-06-03 02:13:51 +09:00
2026-06-02 23:03:46 +09:00
2026-06-03 04:17:40 +09:00
2026-06-03 14:22:45 +09:00
2026-06-03 14:06:35 +09:00
2026-06-01 22:26:13 +09:00
2026-06-04 15:44:25 +01:00
2026-06-04 21:21:51 +01:00
2026-06-02 11:23:40 +09:00
2026-06-02 20:45:32 +09:00
2026-06-05 00:06:37 +02:00
2026-06-02 20:44:24 +09:00
2026-06-03 13:38:11 +09:00
2026-06-03 13:29:01 +09:00
2026-06-03 13:29:08 +09:00