

# Use elicitation with your AgentCore gateway
<a name="gateway-mcp-elicitation"></a>

Elicitation is an MCP feature that allows an MCP server to request additional information from the client during a tool call. When a tool needs user confirmation, authentication, or additional input to proceed, the server sends an elicitation request back to the client. AgentCore Gateway forwards elicitation requests from MCP server targets to your clients, replacing the request `id` with a gateway-generated identifier.

## Prerequisites
<a name="gateway-mcp-elicitation-prereqs"></a>

To use elicitation with your gateway, you must have:
+  **Sessions enabled** — Elicitation requires session support. See [Use MCP sessions with your gateway](gateway-sessions.md).
+  **Response streaming enabled** — Elicitation requests are sent as Server-Sent Events (SSE) chunks during an open connection. Set `streamingConfiguration.enableResponseStreaming` to `true` in your gateway’s `protocolConfiguration.mcp`.
+  **MCP server target type** — Elicitation is only supported for MCP server targets. The elicitation originates from the MCP server and is forwarded through the gateway to the client.
+  **Client declares elicitation capability** — The client must declare support for elicitation during the `initialize` request for the gateway to forward elicitation requests.

## Supported elicitation modes
<a name="gateway-mcp-elicitation-modes"></a>

AgentCore Gateway supports three elicitation modes defined by the MCP specification:


| Mode | Description | 
| --- | --- | 
|  **Form mode**  | The server sends a structured form with fields for the client to fill out. Used for collecting user confirmations, preferences, or input data. The request remains open while waiting for the response. | 
|  **URL mode (request-based)**  | The server sends a URL that the user must visit to complete an action (typically authentication). The request remains open while waiting for the action to complete. | 
|  **URL mode (exception-based)**  | The server throws a `URLElicitationRequiredError` containing a URL. The request closes, the user completes the action at the URL, and the client retries the original tool call. | 

## Capability negotiation
<a name="gateway-mcp-elicitation-capability"></a>

The gateway only declares elicitation support to an MCP server target if:

1. The **client** declared elicitation support during `initialize`.

1. The **MCP protocol version** supports the elicitation mode — `form` mode requires version `2025-03-26` or later, `url` modes require version `2025-11-25` or later.

1. The gateway matches the specific elicitation capabilities declared by the client (form, url, or both).

## Form mode elicitation flow
<a name="gateway-mcp-elicitation-form-flow"></a>

1. Client sends a `tools/call` request with the `Mcp-Session-Id` header.

1. Gateway forwards the tool call to the MCP server target.

1. The target opens an SSE stream and sends an `elicitation/create` request as the first event.

1. Gateway forwards the `elicitation/create` request to the client on the SSE stream, replacing the request `id`.

1. The client presents the form to the user and collects the response.

1. The client sends a new request with the elicitation response (action: `accept` or `decline`) using the same `Mcp-Session-Id`.

1. Gateway forwards the response to the MCP server target.

1. The target acknowledges with HTTP 202 Accepted.

1. The target completes the tool call and sends the final result on the original SSE stream.

1. Gateway forwards the final result to the client and closes the stream.

## URL mode (exception-based) elicitation flow
<a name="gateway-mcp-elicitation-url-exception-flow"></a>

1. Client sends a `tools/call` request with the `Mcp-Session-Id` header.

1. Gateway forwards the tool call to the MCP server target.

1. The target throws a `URLElicitationRequiredError` as a JSON-RPC error, containing the URL and an elicitation ID.

1. Gateway forwards the `URLElicitationRequiredError` to the client, replacing the request `id`.

1. The client redirects the user to the provided URL to complete the action (typically OAuth authentication).

1. After the user completes the action, the client retries the original `tools/call` request.

1. Gateway forwards the retry to the target. The target completes the tool call since the URL elicitation was fulfilled.

1. Gateway forwards the final tool result to the client.

## Parallel tool calls with elicitations
<a name="gateway-mcp-elicitation-parallel"></a>

A client can initiate multiple `tools/call` requests within the same session, even while an elicitation is pending. Each elicitation is tracked independently by its `id`. When sending an elicitation response, the client must include the same `id` that was sent by the gateway in the `elicitation/create` request.

## Guidance for MCP server target developers
<a name="gateway-mcp-elicitation-server-guidance"></a>

**Important**  
MCP server targets that send elicitation requests **should** wrap elicitation calls in try-catch blocks and handle the case where the client does not support elicitation. If the gateway’s client did not declare elicitation capability, the gateway does not declare it to the target. If the target sends an elicitation anyway, the gateway returns a `-32601` (Method not found) error to the target.  
Servers should implement a fallback path (such as using default values or skipping the operation) when elicitation is not available.

## Error handling
<a name="gateway-mcp-elicitation-errors"></a>


| Scenario | Error | Description | 
| --- | --- | --- | 
| Client sends an elicitation response when no elicitation is pending | JSON-RPC `-32600` (Invalid Request) | No matching elicitation found for this session. | 
| Client sends elicitation response with an `id` that doesn’t match a pending elicitation | JSON-RPC `-32600` (Invalid Request) | The `id` must match the one sent by the gateway in the `elicitation/create` request. | 
| Connection breaks between gateway and MCP server target | JSON-RPC error with DependencyFailedException | Client should retry the original tool call request. | 
| Connection breaks between client and gateway | N/A | The pending elicitation is cleaned up. Client should retry the tool call. | 
| MCP server sends elicitation but gateway did not declare support | JSON-RPC `-32601` (Method not found) | Returned to the MCP server target. See [Troubleshooting](#gateway-mcp-elicitation-troubleshooting). | 

## Troubleshooting
<a name="gateway-mcp-elicitation-troubleshooting"></a>

 **Error: "Error calling tool 'sample\_tool': Method not found: elicitation/create"** 

This error occurs when an MCP server target sends an elicitation request but the gateway’s client did not declare elicitation capability during `initialize`. The gateway returns a `-32601` (Method not found) error to the target, and the target may return this as a tool execution error to the client.

To resolve:
+  **If you are the MCP server developer**: Add error handling around your elicitation calls. Implement a fallback path when elicitation is not supported:

  ```
  try:
      result = await context.session.create_elicitation(
          message="Confirm this action?",
          requested_schema={"type": "object", "properties": {"confirm": {"type": "boolean"}}}
      )
  except Exception as e:
      # Fallback when client doesn't support elicitation
      logger.warning(f"Elicitation not supported: {e}")
      result = default_action()
  ```
+  **If you are the gateway client developer**: Ensure your client declares elicitation capability during `initialize`:

  ```
  {
    "capabilities": {
      "elicitation": {
        "form": {},
        "url": {}
      }
    }
  }
  ```

## Code samples
<a name="gateway-mcp-elicitation-examples"></a>

### Form mode example
<a name="gateway-mcp-elicitation-examples-form"></a>

In form mode, the server sends a structured schema for the client to fill out. The request remains open while waiting for the response.

**Example**  

1. 

   ```
   import requests
   import json
   import sseclient
   
   gateway_url = "https://mygateway-abcdefghij.gateway.bedrock-agentcore.us-west-2.amazonaws.com/mcp"
   headers = {
       "Content-Type": "application/json",
       "Accept": "text/event-stream",
       "Authorization": "Bearer YOUR_ACCESS_TOKEN"
   }
   
   # Step 1: Initialize with elicitation capability
   init_response = requests.post(gateway_url, headers=headers, json={
       "jsonrpc": "2.0",
       "id": "init-request",
       "method": "initialize",
       "params": {
           "protocolVersion": "2025-06-18",
           "capabilities": {"elicitation": {"form": {}, "url": {}}},
           "clientInfo": {"name": "my-agent", "version": "1.0.0"}
       }
   })
   session_id = init_response.headers["Mcp-Session-Id"]
   headers["Mcp-Session-Id"] = session_id
   
   # Step 2: Call tool (streaming response)
   response = requests.post(gateway_url, headers=headers, json={
       "jsonrpc": "2.0",
       "id": "tool-call-1",
       "method": "tools/call",
       "params": {
           "name": "use_aws",
           "arguments": {"command": "aws s3 rm s3://my-bucket/important-file"}
       }
   }, stream=True)
   
   # Step 3: Process SSE events
   client = sseclient.SSEClient(response)
   for event in client.events():
       data = json.loads(event.data)
       if data.get("method") == "elicitation/create":
           elicitation_id = data["id"]
           print(f"Form elicitation received: {data['params']['message']}")
   
           # Step 4: Send elicitation response
           requests.post(gateway_url, headers=headers, json={
               "jsonrpc": "2.0",
               "id": elicitation_id,
               "result": {"action": "accept", "content": {"confirm": True}}
           })
       elif "result" in data:
           print(f"Tool result: {data['result']}")
           break
   ```

1. 

   ```
   from mcp import ClientSession
   from mcp.client.streamable_http import streamablehttp_client
   import asyncio
   
   async def elicitation_handler(request):
       """Handle form elicitation requests from the server."""
       print(f"Elicitation: {request.params.message}")
       return {"action": "accept", "content": {"confirm": True}}
   
   async def use_elicitation(url, token):
       headers = {"Authorization": f"Bearer {token}"}
   
       async with streamablehttp_client(url=url, headers=headers) as (
           read_stream, write_stream, _
       ):
           async with ClientSession(
               read_stream, write_stream,
               elicitation_handler=elicitation_handler
           ) as session:
               await session.initialize()
               result = await session.call_tool(
                   name="use_aws",
                   arguments={"command": "aws s3 rm s3://my-bucket/important-file"}
               )
               print(f"Tool result: {result}")
               return result
   
   asyncio.run(use_elicitation(
       url="https://mygateway-abcdefghij.gateway.bedrock-agentcore.us-west-2.amazonaws.com/mcp",
       token="YOUR_ACCESS_TOKEN"
   ))
   ```

1. 

   ```
   from mcp.client.streamable_http import streamablehttp_client
   from mcp.types import ElicitResult
   from strands import Agent
   from strands.tools.mcp import MCPClient
   
   mcp_url = "https://mygateway-abcdefghij.gateway.bedrock-agentcore.us-west-2.amazonaws.com/mcp"
   access_token = "YOUR_ACCESS_TOKEN"
   
   async def elicitation_callback(context, params):
       """Handle form elicitation requests from the MCP server target."""
       print(f"Elicitation: {params.message}")
       user_response = get_user_input(params)  # Your UI logic
       return ElicitResult(action="accept", content=user_response)
   
   mcp_client = MCPClient(
       lambda: streamablehttp_client(
           mcp_url, headers={"Authorization": f"Bearer {access_token}"}
       ),
       elicitation_callback=elicitation_callback,
   )
   
   with mcp_client:
       agent = Agent(tools=mcp_client.list_tools_sync())
       response = agent("Delete the file s3://my-bucket/important-file using AWS CLI")
       print(response)
   ```

1. 

   ```
   from langchain_mcp_adapters.client import MultiServerMCPClient
   from langchain_mcp_adapters.callbacks import Callbacks, CallbackContext
   from langchain.agents import create_agent
   from mcp.shared.context import RequestContext
   from mcp.types import ElicitRequestParams, ElicitResult
   
   async def on_elicitation(
       mcp_context: RequestContext,
       params: ElicitRequestParams,
       context: CallbackContext,
   ) -> ElicitResult:
       """Handle elicitation requests from MCP servers."""
       print(f"[{context.server_name}] Elicitation: {params.message}")
       # Prompt user for input based on params.requestedSchema
       return ElicitResult(
           action="accept",
           content={"confirm": True},
       )
   
   client = MultiServerMCPClient(
       {
           "gateway": {
               "url": "https://mygateway-abcdefghij.gateway.bedrock-agentcore.us-west-2.amazonaws.com/mcp",
               "transport": "http",
               "headers": {"Authorization": "Bearer YOUR_ACCESS_TOKEN"},
           }
       },
       callbacks=Callbacks(on_elicitation=on_elicitation),
   )
   
   tools = await client.get_tools()
   agent = create_agent("claude-sonnet-4-20250514", tools)
   
   result = await agent.ainvoke(
       {"messages": [{"role": "user", "content": "Delete s3://my-bucket/important-file"}]}
   )
   ```

### URL mode example
<a name="gateway-mcp-elicitation-examples-url"></a>

In URL mode, the server sends a URL that the user must visit to complete an action (typically OAuth authentication). The request remains open while waiting for the user to complete the action at the URL.

**Example**  

1. 

   ```
   import requests
   import json
   import sseclient
   import webbrowser
   
   gateway_url = "https://mygateway-abcdefghij.gateway.bedrock-agentcore.us-west-2.amazonaws.com/mcp"
   headers = {
       "Content-Type": "application/json",
       "Accept": "text/event-stream",
       "Authorization": "Bearer YOUR_ACCESS_TOKEN",
       "Mcp-Session-Id": "session-abc123def456"
   }
   
   # Call tool that triggers URL elicitation
   response = requests.post(gateway_url, headers=headers, json={
       "jsonrpc": "2.0",
       "id": "tool-call-2",
       "method": "tools/call",
       "params": {
           "name": "access_github_repo",
           "arguments": {"repo": "my-org/my-repo"}
       }
   }, stream=True)
   
   # Process SSE events
   client = sseclient.SSEClient(response)
   for event in client.events():
       data = json.loads(event.data)
       if data.get("method") == "elicitation/create":
           elicitation_id = data["id"]
           url = data["params"]["url"]
           print(f"URL elicitation: {data['params']['message']}")
   
           # Open browser for user to complete authentication
           webbrowser.open(url)
           input("Press Enter after completing authentication...")
   
           # Send elicitation response
           requests.post(gateway_url, headers=headers, json={
               "jsonrpc": "2.0",
               "id": elicitation_id,
               "result": {"action": "accept"}
           })
       elif "result" in data:
           print(f"Tool result: {data['result']}")
           break
   ```

1. 

   ```
   from mcp import ClientSession
   from mcp.client.streamable_http import streamablehttp_client
   import asyncio
   import webbrowser
   
   async def elicitation_handler(request):
       """Handle URL elicitation by opening the browser."""
       if hasattr(request.params, 'url') and request.params.url:
           print(f"Opening URL for authentication: {request.params.url}")
           webbrowser.open(request.params.url)
           input("Press Enter after completing authentication...")
           return {"action": "accept"}
       # Form mode fallback
       return {"action": "accept", "content": {}}
   
   async def use_url_elicitation(url, token):
       headers = {"Authorization": f"Bearer {token}"}
   
       async with streamablehttp_client(url=url, headers=headers) as (
           read_stream, write_stream, _
       ):
           async with ClientSession(
               read_stream, write_stream,
               elicitation_handler=elicitation_handler
           ) as session:
               await session.initialize()
               result = await session.call_tool(
                   name="access_github_repo",
                   arguments={"repo": "my-org/my-repo"}
               )
               print(f"Tool result: {result}")
               return result
   
   asyncio.run(use_url_elicitation(
       url="https://mygateway-abcdefghij.gateway.bedrock-agentcore.us-west-2.amazonaws.com/mcp",
       token="YOUR_ACCESS_TOKEN"
   ))
   ```

1. 

   ```
   from mcp.client.streamable_http import streamablehttp_client
   from mcp.types import ElicitResult
   from strands import Agent
   from strands.tools.mcp import MCPClient
   import webbrowser
   
   mcp_url = "https://mygateway-abcdefghij.gateway.bedrock-agentcore.us-west-2.amazonaws.com/mcp"
   access_token = "YOUR_ACCESS_TOKEN"
   
   async def elicitation_callback(context, params):
       """Handle URL elicitation by opening the browser."""
       if hasattr(params, 'url') and params.url:
           print(f"Opening URL: {params.url}")
           webbrowser.open(params.url)
           input("Press Enter after completing authentication...")
           return ElicitResult(action="accept")
       return ElicitResult(action="accept", content={})
   
   mcp_client = MCPClient(
       lambda: streamablehttp_client(
           mcp_url, headers={"Authorization": f"Bearer {access_token}"}
       ),
       elicitation_callback=elicitation_callback,
   )
   
   with mcp_client:
       agent = Agent(tools=mcp_client.list_tools_sync())
       response = agent("Access the my-org/my-repo GitHub repository")
       print(response)
   ```

1. 

   ```
   from langchain_mcp_adapters.client import MultiServerMCPClient
   from langchain_mcp_adapters.callbacks import Callbacks, CallbackContext
   from langchain.agents import create_agent
   from mcp.shared.context import RequestContext
   from mcp.types import ElicitRequestParams, ElicitResult
   import webbrowser
   
   async def on_elicitation(
       mcp_context: RequestContext,
       params: ElicitRequestParams,
       context: CallbackContext,
   ) -> ElicitResult:
       """Handle URL elicitation by opening the browser."""
       if params.url:
           print(f"[{context.server_name}] Opening URL: {params.url}")
           webbrowser.open(params.url)
           input("Press Enter after completing authentication...")
           return ElicitResult(action="accept")
       return ElicitResult(action="accept", content={})
   
   client = MultiServerMCPClient(
       {
           "gateway": {
               "url": "https://mygateway-abcdefghij.gateway.bedrock-agentcore.us-west-2.amazonaws.com/mcp",
               "transport": "http",
               "headers": {"Authorization": "Bearer YOUR_ACCESS_TOKEN"},
           }
       },
       callbacks=Callbacks(on_elicitation=on_elicitation),
   )
   
   tools = await client.get_tools()
   agent = create_agent("claude-sonnet-4-20250514", tools)
   
   result = await agent.ainvoke(
       {"messages": [{"role": "user", "content": "Access the my-org/my-repo GitHub repository"}]}
   )
   ```