A2A
Since Camel 4.21
Both producer and consumer are supported
The A2A component implements the Agent-to-Agent (A2A) v1.0 protocol, enabling Camel routes to both expose and call A2A-compliant agents.
As a consumer, it turns a Camel route into an A2A agent — automatically serving an agent card at /.well-known/agent-card.json, exposing all A2A operations via HTTP, and managing task state. As a producer, it calls remote A2A agents with automatic agent card discovery, protocol wrapping, and credential management.
The component follows the same design philosophy as REST OpenAPI: it handles protocol plumbing and delegates business logic entirely to the route.
| A2A Protocol The A2A protocol is an open standard for communication between AI agents. It defines how agents discover each other (via agent cards), exchange messages, manage long-running tasks, stream progress updates, and deliver push notifications. The protocol supports both REST (HTTP+JSON) and JSON-RPC 2.0 bindings over HTTP. |
Preview Limitations
The A2A component is Preview in Camel 4.21. It targets the A2A v1.0 protocol, but endpoint options, model classes, generated metadata, and the task-store SPI may still change before the component reaches stable support.
Current Preview limitations:
-
Extended Agent Card /
GetExtendedAgentCardis not implemented in this first release. -
Consumers validate operation requests by default. Local-only examples in this page set
validateAuth=false; network-exposed agents should use an agent card withsecuritySchemesandsecurityRequirementsplusapiKey,bearerToken, oroauthProfileendpoint configuration. -
The REST binding uses A2A custom-method paths containing colons, such as
/message:send. These paths collide on Vert.x/platform-http. UseprotocolBinding=JSONRPCwith platform-http, or usehttpServerComponent=undertoworhttpServerComponent=jettyfor REST custom-method routes. -
Real-time SSE streaming is verified with Undertow and Jetty. Vert.x/platform-http buffers
InputStreamresponses and does not deliver events as they are emitted. -
The default task store is in-memory and single-JVM. Register a custom
A2ATaskStorefor durable or cross-node task state.
Maven users will need to add the following dependency to their pom.xml:
<dependency>
<groupId>org.apache.camel</groupId>
<artifactId>camel-a2a</artifactId>
<version>x.x.x</version>
<!-- use the same version as your Camel core version -->
</dependency> Consumer routes also need an HTTP server component at runtime. For example, add camel-platform-http-vertx for JSON-RPC agents, or camel-undertow / camel-jetty for REST custom-method routes and real-time SSE streaming. See Dependencies and HTTP Server Component Discovery.
URI Format
a2a:agentCardSource[?options]
How agentCardSource determines where the agent card is loaded from:
| Source | Example |
|---|---|
Remote URL (partial) |
|
Remote URL (full) |
|
Classpath |
|
File |
|
Plain name |
|
The same URI works in both from() (consumer: "I am this agent") and to() (producer: "I am calling this agent").
Configuring Options
Camel components are configured on two separate levels:
-
component level
-
endpoint level
Configuring Component Options
At the component level, you set general and shared configurations that are, then, inherited by the endpoints. It is the highest configuration level.
For example, a component may have security settings, credentials for authentication, urls for network connection and so forth.
Some components only have a few options, and others may have many. Because components typically have pre-configured defaults that are commonly used, then you may often only need to configure a few options on a component; or none at all.
You can configure components using:
-
the Component DSL.
-
in a configuration file (
application.properties,*.yamlfiles, etc). -
directly in the Java code.
Configuring Endpoint Options
You usually spend more time setting up endpoints because they have many options. These options help you customize what you want the endpoint to do. The options are also categorized into whether the endpoint is used as a consumer (from), as a producer (to), or both.
Configuring endpoints is most often done directly in the endpoint URI as path and query parameters. You can also use the Endpoint DSL and DataFormat DSL as a type safe way of configuring endpoints and data formats in Java.
A good practice when configuring options is to use Property Placeholders.
Property placeholders provide a few benefits:
-
They help prevent using hardcoded urls, port numbers, sensitive information, and other settings.
-
They allow externalizing the configuration from the code.
-
They help the code to become more flexible and reusable.
The following two sections list all the options, firstly for the component followed by the endpoint.
Component Options
The A2A component supports 3 options, which are listed below.
| Name | Description | Default | Type |
|---|---|---|---|
Allows for bridging the consumer to the Camel routing Error Handler, which mean any exceptions (if possible) occurred while the Camel consumer is trying to pickup incoming messages, or the likes, will now be processed as a message and handled by the routing Error Handler. Important: This is only possible if the 3rd party component allows Camel to be alerted if an exception was thrown. Some components handle this internally only, and therefore bridgeErrorHandler is not possible. In other situations we may improve the Camel component to hook into the 3rd party component and make this possible for future releases. By default the consumer will use the org.apache.camel.spi.ExceptionHandler to deal with exceptions, that will be logged at WARN or ERROR level and ignored. | false | boolean | |
Whether the producer should be started lazy (on the first message). By starting lazy you can use this to allow CamelContext and routes to startup in situations where a producer may otherwise fail during starting and cause the route to fail being started. By deferring this startup to be lazy then the startup failure can be handled during routing messages via Camel’s routing error handlers. Beware that when the first message is processed then creating and starting the producer may take a little time and prolong the total processing time of the processing. | false | boolean | |
Whether autowiring is enabled. This is used for automatic autowiring options (the option must be marked as autowired) by looking up in the registry to find if there is a single instance of matching type, which then gets configured on the component. This can be used for automatic configuring JDBC data sources, JMS connection factories, AWS Clients, etc. | true | boolean |
Endpoint Options
The A2A endpoint is configured using URI syntax:
a2a:agentCardSource
With the following path and query parameters:
Query Parameters (35 parameters)
| Name | Description | Default | Type |
|---|---|---|---|
Agent card bean reference or inline configuration. | AgentCard | ||
The base path for HTTP requests. | String | ||
The data format for the exchange body, following the CXF DataFormat convention. PAYLOAD (default) extracts text content from message parts as a String backward compatible, simple for chatbot routes. POJO sets the body to the full Java model object (Message on consumer, Task or Message on producer) preserving all parts, metadata, and file content. RAW passes the raw JSON string without deserialization useful for forwarding, logging, or compliance. Enum values:
| PAYLOAD | A2ADataFormat | |
The agent description (overrides agent card). | String | ||
Maximum number of history messages to include in the context. | Integer | ||
The host to connect to for producers. | String | ||
The agent name (overrides agent card). | String | ||
The port to connect to for producers. | Integer | ||
The protocol binding to use for communication. Legacy aliases rest and jsonrpc are also accepted. Enum values:
| HTTP+JSON | String | |
Whether to return immediately without waiting for task completion. | false | boolean | |
Whether to validate authentication on incoming consumer operation requests. Disable explicitly only for unauthenticated A2A operation serving. | true | boolean | |
The agent version (overrides agent card). | String | ||
Allows for bridging the consumer to the Camel routing Error Handler, which mean any exceptions (if possible) occurred while the Camel consumer is trying to pickup incoming messages, or the likes, will now be processed as a message and handled by the routing Error Handler. Important: This is only possible if the 3rd party component allows Camel to be alerted if an exception was thrown. Some components handle this internally only, and therefore bridgeErrorHandler is not possible. In other situations we may improve the Camel component to hook into the 3rd party component and make this possible for future releases. By default the consumer will use the org.apache.camel.spi.ExceptionHandler to deal with exceptions, that will be logged at WARN or ERROR level and ignored. | false | boolean | |
To let the consumer use a custom ExceptionHandler. Notice if the option bridgeErrorHandler is enabled then this option is not in use. By default the consumer will deal with exceptions, that will be logged at WARN or ERROR level and ignored. | ExceptionHandler | ||
Sets the exchange pattern when the consumer creates an exchange. Enum values:
| ExchangePattern | ||
The Camel HTTP component to use for serving incoming A2A requests (consumer side). Must implement RestConsumerFactory (e.g., platform-http, jetty, netty-http, undertow, servlet). When not set, the component is auto-discovered from the classpath or falls back to the global camel.rest.component setting. | String | ||
Maximum number of tasks the agent can process concurrently. When the limit is reached, new requests are rejected with ServerBusyError (HTTP 429). Set to 0 (default) for unlimited concurrency. | 0 | int | |
Interval in milliseconds for SSE keep-alive heartbeat comments. Sent as ':' comment lines to prevent proxies from closing idle connections. Independent from asyncTimeout which controls task processing timeout. | 15000 | long | |
Maximum number of SSE events that can be buffered per streaming connection. When the queue is full, new events are dropped with a warning log. Prevents unbounded memory growth from slow clients. | 1000 | int | |
Maximum number of tasks that can wait in the pending queue when all concurrent slots are occupied. Only applies to async requests (returnImmediately=true). Queued tasks receive SUBMITTED status and are processed as capacity becomes available. Set to 0 (default) for no queueing requests are rejected immediately when at capacity. | 0 | int | |
The A2A operation to perform. Enum values:
| MESSAGE_SEND | A2AOperations | |
Connect timeout in milliseconds for the HTTP client used by the producer. | 30000 | long | |
Whether the producer should be started lazy (on the first message). By starting lazy you can use this to allow CamelContext and routes to startup in situations where a producer may otherwise fail during starting and cause the route to fail being started. By deferring this startup to be lazy then the startup failure can be handled during routing messages via Camel’s routing error handlers. Beware that when the first message is processed then creating and starting the producer may take a little time and prolong the total processing time of the processing. | false | boolean | |
Read timeout in milliseconds for the producer’s SSE streaming connection. If no SSE event arrives within this period, the stream is closed with an error. Prevents indefinite blocking when a remote agent stops sending events. | 300000 | long | |
Timeout in milliseconds for asynchronous task operations. | 300000 | long | |
Time-to-live in milliseconds for completed tasks before cleanup. | 3600000 | long | |
Whether the HTTP client should follow redirects. Disabled by default to prevent credential leakage on cross-origin redirects. Enable only when the remote agent is known to issue redirects (e.g., behind a load balancer). | false | boolean | |
Maximum payload size in bytes (default 6MB). | 6291456 | long | |
Maximum number of retry attempts for push notification webhook delivery. | 3 | int | |
Initial backoff in milliseconds for push notification retry. Retries use exponential backoff with this delay multiplied by 2 to the attempt number. | 1000 | long | |
Whether to allow webhook URLs pointing to localhost/loopback addresses. When false (default), push notification webhook URLs targeting 127.0.0.0/8, ::1, or localhost are rejected as SSRF protection. Enable for local development only. | false | boolean | |
API key for authentication. | String | ||
HTTP header name for API key authentication (e.g., X-API-Key, Authorization). | Authorization | String | |
Bearer token for authentication. | String | ||
OAuth profile name for obtaining an access token via the OAuth 2.0 Client Credentials grant. When set, the token is acquired from the configured identity provider and used for authentication. Requires camel-oauth on the classpath. The profile properties are resolved from camel.oauth.profile-name.client-id, camel.oauth.profile-name.client-secret, and camel.oauth.profile-name.token-endpoint. | String |
Message Headers
The A2A component supports 19 message header(s), which is/are listed below:
| Name | Description | Default | Type |
|---|---|---|---|
| Constant: | A2A operation to invoke. | String | |
| Constant: | Task ID. | String | |
| Constant: | Push notification config ID. | String | |
| Constant: | Context ID for multi-turn conversations. | String | |
| Constant: | Message ID. | String | |
| Constant: | Task state. | String | |
| Constant: | A2A method name invoked. | String | |
| Constant: | Response type: task or message. | String | |
CamelA2AReturnImmediately (common) Constant: | Return immediately flag. | Boolean | |
CamelA2AHistoryLength (common) Constant: | Max history messages. | Integer | |
CamelA2AStreamEmitter (consumer) Constant: | SSE stream emitter for route processors. | A2AStreamEmitter | |
CamelA2AListContextId (common) Constant: | Context ID filter for task listing. | String | |
| Constant: | Page size for task listing. | Integer | |
CamelA2AListPageToken (common) Constant: | Page token for task listing pagination. | String | |
CamelA2AListIncludeArtifacts (common) Constant: | Whether to include artifacts in task listing. | Boolean | |
CamelA2AListHistoryLength (common) Constant: | History length for task listing. | Integer | |
CamelA2AListStatusTimestampAfter (common) Constant: | Filter tasks by status timestamp after this value. | String | |
| Constant: | Comma-separated status filter for task listing. | String | |
| Constant: | Negotiated A2A extension URIs requested by the client. | List |
Consumer — Exposing an A2A Agent
The consumer turns a Camel route into an A2A agent. It automatically:
-
Serves the agent card at
GET /.well-known/agent-card.json -
Registers HTTP endpoints for A2A operations (REST routes, or a JSON-RPC dispatcher for supported methods)
-
Validates authentication on incoming requests (when configured)
-
Manages task state via an internal
A2ATaskStore -
Dispatches push notification webhooks when task state changes
-
Filters inbound
Camel*andorg.apache.camel.*headers (case-insensitive) from untrusted requests -
Enforces
maxPayloadSizeon incoming request bodies (returns 413 when exceeded)
The route’s setBody output becomes the agent’s response — the consumer handles all A2A protocol wrapping. The dataFormat parameter controls what the route receives as body: extracted text (PAYLOAD, default), the full Message object (POJO), or raw JSON (RAW).
Basic Agent (Local Only)
The simplest possible local A2A agent. Define your agent’s capabilities in agent-card.json, disable operation auth for local-only use, and let the route handle business logic:
-
YAML
-
Java
- route:
from:
uri: a2a:classpath:agent-card.json
parameters:
validateAuth: false
steps:
- log:
message: "Received: ${body}"
- setBody:
constant: "Hello from my agent!" from("a2a:classpath:agent-card.json?validateAuth=false")
.log("Received: ${body}")
.setBody(constant("Hello from my agent!")); Do not use validateAuth=false for agents exposed to an untrusted network. See Authentication for authenticated operation serving.
The agent card (agent-card.json) declares your agent’s identity and capabilities:
{
"protocolVersion": "1.0",
"name": "my-agent",
"description": "A simple Camel A2A agent",
"url": "http://localhost:8080",
"version": "1.0.0",
"provider": {
"name": "Example",
"url": "https://example.com"
},
"capabilities": {
"streaming": false,
"pushNotifications": false,
"extendedAgentCard": false
},
"supportedInterfaces": [
{
"url": "http://localhost:8080",
"protocolBinding": "HTTP+JSON",
"protocolVersion": "1.0"
}
],
"defaultInputModes": ["text/plain"],
"defaultOutputModes": ["text/plain"],
"skills": [
{
"id": "greet",
"name": "Greeting",
"description": "Returns a greeting message",
"tags": ["greeting"],
"examples": ["Say hello to Ada"],
"inputModes": ["text/plain"],
"outputModes": ["text/plain"]
}
]
} The component supports the HTTP+JSON and JSONRPC protocol bindings. The legacy aliases rest and jsonrpc are also accepted and normalized to the v1.0 binding names. The protocolVersion, defaultInputModes, and defaultOutputModes fields are part of the A2A v1.0 card shape.
The AgentCard Java model can deserialize A2A v1.0 fields such as defaultInputModes and defaultOutputModes, and it also stores unknown JSON properties. URI parameter overrides are limited to name, description, and version; the endpoint resolver preserves card metadata and unknown extension properties from file and bean cards, supplies preview defaults for plain-name routes, and rejects unsupported protocol bindings. |
Card from Parameters (Local Only)
For simple or test agents, the card identity fields can be built from URI parameters - no JSON file needed. Because parameter-built cards do not include security schemes, these local-only examples set validateAuth=false:
-
YAML
-
Java
- route:
from:
uri: >-
a2a:weather-agent
?name=Weather Agent
&description=Provides weather forecasts
&version=1.0.0
&validateAuth=false
steps:
- bean:
ref: weatherService from("a2a:my-agent?name=My Agent&description=A simple agent&version=1.0.0&validateAuth=false")
.setBody(constant("Response from parameter-configured agent")); The component generates the /.well-known/agent-card.json response dynamically from the parameters. URI parameters only cover the agent identity fields above; use an agent card file or an agentCard bean for capabilities, skills, security schemes, supported interfaces, and other card metadata.
JSON-RPC Protocol Binding
By default, the consumer uses the REST binding (HTTP+JSON with colon-notation paths like /message:send). To use JSON-RPC 2.0 (single POST / endpoint), set protocolBinding to JSONRPC. The legacy aliases rest and jsonrpc are also accepted.
- route:
from:
uri: a2a:classpath:agent-card.json
parameters:
protocolBinding: JSONRPC
validateAuth: false
steps:
- setBody:
constant: "JSON-RPC response" This example is local-only because it disables operation auth. Keep validateAuth=true and configure a card security scheme for network-exposed agents.
The REST binding exposes separate HTTP routes for the operation-serving methods listed in Consumer REST API Paths. The JSON-RPC binding exposes a single dispatcher and handles SendMessage, SendStreamingMessage, GetTask, ListTasks, CancelTask, SubscribeToTask, and the push notification config operations. Extended agent cards are outside the Preview scope for this first release.
REST requests and responses use application/a2a+json. JSON-RPC requests and responses use application/json. Streaming operations produce text/event-stream.
Vert.x Colon-in-Path Issue
The A2A REST protocol uses Google’s Custom Method convention with colons in paths (e.g., /message:send, /message:stream, /tasks/{id}:cancel). Vert.x’s router interprets the colon (:) as a path parameter delimiter, causing route collisions:
-
/message:sendand/message:streamboth match the pattern/message{param}— whichever is registered first captures ALL requests -
/tasks/{taskId}:canceland/tasks/{taskId}:subscribecan collide with/tasks/{taskId}
This means operation serving is not reliable with REST binding on Vert.x/platform-http. For example, POST /message:stream can be handled by the SendMessage route instead of the streaming route.
Workaround: Use protocolBinding=JSONRPC for agents running on Vert.x (the default platform-http). Alternatively, use httpServerComponent=jetty or httpServerComponent=undertow; their routers treat colons as literal characters. For real-time SSE streaming, use Jetty or Undertow because Vert.x/platform-http buffers InputStream responses.
Authentication
The component supports OAuth 2.0, OpenID Connect, API key, and HTTP bearer security schemes. OAuth 2.0 and OpenID Connect use the oauthProfile option and the camel-oauth SPI; HTTP bearer can use either an OAuth profile or the static bearerToken option.
| When |
|
|
| The component scopes built-in task and push-notification operations to the authenticated user that created the task. Tenant routing and tenant-aware authorization policy are outside the component scope; implement those rules in the Camel route or in deployment-specific infrastructure. |
securityRequirements follows A2A v1.0 semantics: multiple requirement objects are alternatives (OR), while multiple schemes inside one requirement must all be satisfied together (AND). The list array contains required scopes. The legacy A2A draft field security is accepted on input and converted to securityRequirements; new cards should use securityRequirements.
OAuth / OIDC Authentication
Validate incoming bearer tokens via the camel-oauth SPI. When an oauthProfile is configured, the consumer validates JWT signatures (via JWKS), expiry, issuer, and audience — or uses RFC 7662 token introspection for opaque tokens. Requires camel-oauth on the classpath.
- route:
from:
uri: a2a:classpath:agent-card.json
parameters:
oauthProfile: my-agent
validateAuth: true
steps:
- setBody:
constant: "Authenticated response" Configure the OIDC provider in application properties (these properties are owned by the camel-oauth component — see its documentation for the canonical format):
camel.oauth.my-agent.client-id=my-agent-client
camel.oauth.my-agent.client-secret=my-secret
camel.oauth.my-agent.token-endpoint=http://keycloak:8180/realms/my-realm/protocol/openid-connect/token The agent card must declare the security scheme:
{
"securitySchemes": {
"oidc": {
"openIdConnectSecurityScheme": {
"openIdConnectUrl": "http://keycloak:8180/realms/my-realm/.well-known/openid-configuration"
}
}
},
"securityRequirements": [
{
"schemes": {
"oidc": {
"list": []
}
}
}
]
} When authentication succeeds, the CamelA2AUserProfile exchange property is populated with a Map<String, Object> containing the authenticated user’s profile (subject, issuer, scopes, claims). When using OAuth/OIDC, the full OAuthTokenValidationResult is also available as the CamelOAuthTokenValidationResult exchange property.
API Key Authentication
The apiKeyHeader parameter controls which HTTP header carries the API key. It defaults to Authorization. A common override is X-API-Key:
- route:
from:
uri: a2a:classpath:agent-card.json
parameters:
validateAuth: true
apiKey: "{{my.api.key}}"
apiKeyHeader: X-API-Key # default is Authorization
steps:
- setBody:
constant: "API key protected response" The agent card must declare an apiKey security scheme:
{
"securitySchemes": {
"apikey": {
"apiKeySecurityScheme": {
"location": "header",
"name": "X-API-Key"
}
}
},
"securityRequirements": [
{
"schemes": {
"apikey": {
"list": []
}
}
}
]
} When the scheme declares location=header and name, that header name takes precedence over apiKeyHeader. The component also supports location=query and location=cookie for API key schemes.
Bearer Token
- route:
from:
uri: a2a:classpath:agent-card.json
parameters:
bearerToken: "{{my.token}}"
validateAuth: true
steps:
- setBody:
constant: "Bearer token protected response" The agent card must declare an http bearer security scheme:
{
"securitySchemes": {
"bearer": {
"httpAuthSecurityScheme": {
"scheme": "bearer"
}
}
},
"securityRequirements": [
{
"schemes": {
"bearer": {
"list": []
}
}
}
]
} The agent card endpoint (GET /.well-known/agent-card.json) is always public — clients need it for discovery before they can authenticate. |
SSE Streaming with ${a2a:emit()}
Agents can emit progressive status updates during processing using the ${a2a:emit()} Simple language function. Clients receive these as Server-Sent Events (SSE).
For SendStreamingMessage and SubscribeToTask, the agent card must explicitly set capabilities.streaming=true. If the capabilities object is missing, or streaming is omitted or false, the consumer returns UnsupportedOperationError (HTTP 405 for REST, JSON-RPC -32004 for JSON-RPC).
-
YAML
-
Java
- route:
from:
uri: a2a:classpath:agent-card.json
parameters:
protocolBinding: JSONRPC
httpServerComponent: undertow
steps:
- script:
simple: "${a2a:emit('Connecting to database...')}"
- delay:
constant: 2000
- script:
simple: "${a2a:emit('Processing 1000 records...')}"
- delay:
constant: 3000
- script:
simple: "${a2a:emit('Analysis complete!')}"
- setBody:
constant: "Final analysis results: ..." import org.apache.camel.component.a2a.A2AProgress;
from("a2a:classpath:agent-card.json?httpServerComponent=undertow")
.process(exchange -> {
A2AProgress.emit(exchange, "Connecting...");
//connect to the external service
A2AProgress.emit(exchange, "Processing...");
//process data
exchange.getMessage().setBody("Here is the final answer.");
}); Each ${a2a:emit('message')} emits an SSE status event with TASK_STATE_WORKING state. The final setBody is automatically delivered as the completed response.
You can also emit with an explicit state:
- script:
simple: "${a2a:emit(INPUT_REQUIRED, 'Please provide your address')}" Use script EIP (not setBody) for ${a2a:emit()} calls — script evaluates the expression for its side effect without changing the message body. Use setBody only for the final response. |
For advanced use cases, A2AProgress also supports emitting structured artifacts and intermediate messages:
// Emit a structured artifact (e.g., a generated file)
Artifact artifact = Artifact.builder()
.artifactId("report-1")
.name("Analysis Report")
.parts(List.of(new TextPart("Report content...")))
.build();
A2AProgress.emitArtifact(exchange, artifact, false, true); // append=false, lastChunk=true
// Emit an intermediate agent message (not a status update)
Message msg = Message.builder()
.role(Message.Role.ROLE_AGENT)
.parts(List.of(new TextPart("Intermediate finding...")))
.build();
A2AProgress.emitMessage(exchange, msg); The consumer supports two SSE streaming patterns:
-
SendStreamingMessage— The stream starts with the submittedTask, followed by progress events fromA2AProgressor${a2a:emit()}. When the route completes, the body is emitted as the final message. -
SubscribeToTask— Clients subscribe viaPOST /tasks/{id}:subscribeto receive real-time SSE updates for an existing non-terminal task. The component uses an Event-to-Stream Bridge (QueueStreamEmitter+SseQueueInputStream) with heartbeat comments to keep the connection alive. The stream sends the currentTaskfirst and ends when the task later reaches a terminal state. Subscribing to an already-terminal task returnsUnsupportedOperationError.
| HTTP Server Component for Streaming For real-time SSE streaming, use This requires |
SSE streaming parameters:
-
sseHeartbeatInterval(default 15,000 ms / 15 seconds) — interval for SSE keep-alive heartbeat comments (:lines) sent to prevent proxies from closing idle connections. Independent fromasyncTimeout. -
sseQueueCapacity(default 1,000) — maximum SSE events buffered per streaming connection. When the queue is full, new events are dropped with a warning log. Prevents unbounded memory growth from slow clients.
Async Tasks with returnImmediately
For long-running operations, enable returnImmediately to return a SUBMITTED task immediately. The route processes in the background, and clients poll with GetTask:
- route:
from:
uri: a2a:classpath:agent-card.json
parameters:
returnImmediately: true
asyncTimeout: 30000 # default is 300000 (5 minutes)
steps:
- delay:
constant: 10000
- setBody:
constant: "Long computation result" The asyncTimeout parameter (default: 300,000 ms / 5 minutes) controls how long the background processing can run before the task is marked FAILED.
The consumer manages the task lifecycle automatically: SUBMITTED → WORKING → COMPLETED (or FAILED on error / timeout).
returnImmediately resolution priority (highest wins):
-
Per-request:
configuration.returnImmediatelyin theSendMessageRequestbody -
Per-request:
configuration.blocking=falsein theSendMessageRequestbody (typed inverse of immediate return) -
Per-exchange:
CamelA2AReturnImmediatelyheader -
Endpoint config:
returnImmediatelyURI parameter
SendMessageRequest.configuration is modeled as SendMessageConfiguration. The consumer interprets returnImmediately, blocking, and historyLength; unknown fields are retained for protocol extensions and future A2A options. configuration.historyLength limits the task history returned by synchronous SendMessage task responses.
The REST consumer rejects returnImmediately for streaming operations (SendStreamingMessage) with 400 Bad Request. |
Push Notifications
Agents that use returnImmediately can deliver updates via webhooks. Clients register a webhook URL via CreateTaskPushNotificationConfig, and the component’s built-in PushNotificationDispatcher sends status updates to registered URLs.
The agent card should declare push notification support so clients can discover the feature:
{
"capabilities": {
"pushNotifications": true
}
} Progress updates emitted via ${a2a:emit()} or A2AProgress.emit() are automatically dispatched to all registered webhooks.
Push notification dispatch features:
-
Parallel dispatch — multiple webhooks per task are notified concurrently via
CompletableFuture -
Auth headers —
AuthenticationInfofrom the push config is sent asAuthorization: <scheme> <credentials> -
Retry with exponential backoff — configurable via
pushRetryAttempts(default 3) andpushRetryBackoffMs(default 1s). Backoff formula:delay × 2^attempt. Only retries on 5xx/IOException — client errors (4xx) are not retried -
410 Gone auto-cleanup — webhooks returning HTTP 410 are automatically deregistered
-
SSRF protection — webhook URLs are validated at registration time (blocks private IPs, loopback, link-local, wildcard, and cloud metadata addresses). Non-loopback webhook URLs must use HTTPS. Set
allowLocalWebhookUrls=trueto permit loopback URLs (plain HTTP allowed) for local development only
Capacity Limiting
Control concurrent task processing and queue overflow:
- route:
from:
uri: a2a:classpath:agent-card.json
parameters:
maxConcurrentTasks: 10
taskQueueSize: 50
steps:
- setBody:
constant: "Processing..." | Parameter | Default | Description |
|---|---|---|
|
| Maximum concurrent tasks. A shared semaphore governs all processing paths (sync, async, streaming). |
|
| Pending queue capacity for async requests when all slots are occupied. Synchronous and streaming requests cannot be queued — they are rejected immediately. |
Behavior by processing path:
-
Synchronous (
returnImmediately=false): if no permit, returnsServerBusyError(HTTP 429) -
Asynchronous (
returnImmediately=true): if no permit and queue has space, task queues asSUBMITTED; otherwise HTTP 429 -
Streaming (
SendStreamingMessage): same as synchronous — rejects immediately
The error response follows the A2A error model: REST returns HTTP 429 with ServerBusyError; JSON-RPC returns code -32000 in the JSON-RPC error envelope.
On shutdown, queued tasks are marked FAILED and subscribers are notified.
Payload Size Limits
The consumer enforces maxPayloadSize (default 6,291,456 bytes / 6 MiB) on incoming request bodies. Requests exceeding this limit receive HTTP 413 (REST) or a JSON-RPC error.
- route:
from:
uri: a2a:classpath:agent-card.json
parameters:
maxPayloadSize: 10485760 CORS Support
To enable CORS (required when browsers call your agent directly), set the REST configuration:
camel.rest.enableCors=true The consumer automatically adds Access-Control-* headers (including A2A-specific headers like A2A-Version and A2A-Extensions) and registers OPTIONS preflight handlers for all A2A paths.
A2A Extensions
The agent card can advertise protocol extensions in capabilities.extensions. For protected operation routes, clients can request extensions with the A2A-Extensions HTTP header as a comma-separated list of extension URIs.
The consumer validates requested extensions against the resolved agent card. If any requested URI is not advertised, or if a card extension marked required=true is not requested, the request fails before route processing with UnsupportedExtensionError. Accepted extension URIs are exposed to the route as the CamelA2AExtensions header and exchange property, and are echoed in the A2A-Extensions response header.
Routes can branch directly on CamelA2AExtensions. For reusable behavior, register one or more org.apache.camel.component.a2a.extension.A2AExtensionHandler beans in the Camel registry. A handler is matched by extension URI and is invoked before and after route processing when that extension is negotiated. The handler receives the full AgentExtension declaration from the agent card, including params.
Consumer REST API Paths
When using the default REST protocol binding, the consumer registers these HTTP endpoints:
| Method | Path | Operation |
|---|---|---|
|
| Agent card discovery (always public) |
|
| SendMessage |
|
| SendStreamingMessage (SSE response) |
|
| ListTasks |
|
| GetTask |
|
| CancelTask |
|
| SubscribeToTask (SSE response) |
|
| CreateTaskPushNotificationConfig |
|
| ListTaskPushNotificationConfigs |
|
| GetTaskPushNotificationConfig |
|
| DeleteTaskPushNotificationConfig |
All paths are prefixed with basePath if configured. When using JSON-RPC binding, a single POST / endpoint dispatches based on the JSON-RPC method field.
The built-in ListTasks consumer applies contextId, status, statusTimestampAfter, pageSize, pageToken, includeArtifacts, and historyLength. pageSize defaults to 50 and is capped at 100. pageToken values returned by the consumer are opaque cursor tokens containing a snapshot of the filtered task IDs for that listing. Clients must send the token back unchanged with the same filters; numeric offsets are rejected.
Consumer Error Responses
REST binding responses use the A2A v1.0 error wrapper with an error object. The ErrorInfo.reason detail maps back to these component error names:
| Condition | HTTP status | Error code |
|---|---|---|
Unsupported |
|
|
Authentication validation failure |
|
|
Authenticated caller is not authorized for the task |
|
|
Unsupported requested extension, or required extension not requested |
|
|
Empty |
|
|
Request body larger than |
|
|
Streaming or task subscription requested but |
|
|
Push notification config requested but |
|
|
|
|
|
Invalid parameters (for example negative |
|
|
Agent capacity reached |
|
|
Task or push notification config not found |
|
|
Cancel requested for a terminal task |
|
|
Unhandled consumer exception |
|
|
REST binding returns these error responses as application/a2a+json with an error object containing the HTTP status, status name, message, and a google.rpc.ErrorInfo detail reason such as TASK_NOT_FOUND or VERSION_NOT_SUPPORTED.
JSON-RPC binding returns JSON-RPC error envelopes as application/json. Parse errors use -32700, invalid requests use -32600, unknown methods use -32601, invalid parameters use -32602, internal errors use -32603, capacity and authorization errors use -32000, missing tasks use -32001, non-cancelable terminal tasks use -32002, push notification support errors use -32003, unsupported operations use -32004, unsupported content types use -32005, invalid agent responses use -32006, extension negotiation failures use -32008, and unsupported A2A versions use -32009.
Producer — Calling a Remote Agent
The producer calls a remote A2A agent. It automatically discovers the agent card, selects the protocol binding, acquires credentials, and wraps/unwraps messages.
Basic Call
-
YAML
-
Java
- route:
from:
uri: direct:call-agent
steps:
- setBody:
constant: "What is the weather?"
- to:
uri: a2a:http://remote-agent:8080
- log:
message: "Agent replied: ${a2a:text}" from("direct:call-agent")
.setBody(constant("What is the weather?"))
.to("a2a:http://remote-agent:8080")
.log("Agent replied: ${a2a:text}"); The producer:
-
Fetches the agent card from
http://remote-agent:8080/.well-known/agent-card.json -
Wraps the body as an A2A
SendMessagerequest -
Sends the request using the card’s declared protocol binding
-
Returns the response as the exchange body — by default (
dataFormat=PAYLOAD) as extracted text; withdataFormat=POJOas the fullTaskorMessageJava object
Authenticated Calls
OIDC (Client Credentials)
- to:
uri: a2a:http://remote-agent:8080
parameters:
oauthProfile: my-profile Acquires a token via client-credentials grant, caches it, and refreshes on expiry.
Producer Streaming
The producer supports two streaming modes, controlled by the dataFormat parameter.
Default (PAYLOAD/POJO): Lazy Iterator
The exchange body is a SseEventIterator (implements Iterator<StreamResponse> + Closeable) that parses SSE events lazily from the remote agent’s response stream. Each call to next() blocks until the next event arrives. Use with the Split EIP for event-by-event processing:
-
YAML
-
Java
- setHeader:
name: CamelA2AOperation
constant: MESSAGE_STREAM
- to: a2a:https://agent.example.com
- split:
simple: "${body}"
streaming: true
steps:
- choice:
when:
- simple: "${body.statusUpdate} != null"
steps:
- log:
message: "Progress: ${body.statusUpdate.status.state}"
- simple: "${body.message} != null"
steps:
- log:
message: "Final message received" from("direct:stream")
.setHeader("CamelA2AOperation", constant("MESSAGE_STREAM"))
.to("a2a:https://agent.example.com")
.split(body()).streaming()
.log("Event: ${body}")
.end(); The iterator implements Closeable, so Camel’s Split EIP automatically releases the underlying HTTP connection after iteration.
To buffer all events into a List<StreamResponse>:
- to: a2a:https://agent.example.com
- convertBodyTo:
type: java.util.List Raw Passthrough (RAW mode)
With dataFormat=RAW, the exchange body is the raw InputStream from the remote agent’s SSE response. No parsing or buffering — bytes flow through untouched. Ideal for proxying SSE streams to a browser:
-
YAML
-
Java
- to:
uri: a2a:https://agent.example.com?dataFormat=RAW
parameters:
operation: MESSAGE_STREAM
- setHeader:
name: Content-Type
constant: text/event-stream from("platform-http:/stream")
.to("a2a:https://agent.example.com?dataFormat=RAW&operation=MESSAGE_STREAM")
.setHeader("Content-Type", constant("text/event-stream")); Subscribe to Task Updates
Subscribe to ongoing task updates from a remote agent:
- setHeader:
name: CamelA2AOperation
constant: TASK_SUBSCRIBE
- setHeader:
name: CamelA2ATaskId
simple: "${exchangeProperty.taskId}"
- to: a2a:https://agent.example.com
- split:
simple: "${body}"
streaming: true
steps:
- log:
message: "Task update: ${body.statusUpdate.status.state}" Streaming requests use asyncTimeout (default 5 minutes) instead of the standard 60-second request timeout. SSE heartbeat comments from the remote agent are automatically filtered during parsing.
Async Task Lifecycle (returnImmediately + GetTask Polling)
# Step 1: Submit the task
- setBody:
constant: "Start long operation"
- to:
uri: a2a:http://remote-agent:8083
parameters:
oauthProfile: assistant
- setVariable:
name: taskId
simple: "${header.CamelA2ATaskId}"
# Step 2: Poll for completion
- removeHeaders:
pattern: "*"
- setHeader:
name: CamelA2AOperation
constant: TASK_GET
- setHeader:
name: CamelA2ATaskId
simple: "${variable.taskId}"
- to:
uri: a2a:http://remote-agent:8083
parameters:
oauthProfile: assistant Push Notification Registration
# Step 1: Submit task
- setBody:
constant: "Track my package"
- to: a2a:http://remote-agent:8085
- setVariable:
name: taskId
simple: "${header.CamelA2ATaskId}"
# Step 2: Register push notification webhook
- removeHeaders:
pattern: "*"
- setHeader:
name: CamelA2AOperation
constant: PUSH_CONFIG_CREATE
- setHeader:
name: CamelA2ATaskId
simple: "${variable.taskId}"
- setBody:
simple: "${ref:pushConfig}"
- to: a2a:http://remote-agent:8085 Configure the webhook URL as a bean:
camel.beans.pushConfig = #class:org.apache.camel.component.a2a.model.TaskPushNotificationConfig
camel.beans.pushConfig.url = https://my-server:8090/webhook The full push notification config CRUD is also available. PUSH_CONFIG_GET and PUSH_CONFIG_DELETE require both CamelA2ATaskId and CamelA2APushConfigId. Use JSON-RPC binding for these two producer operations when addressing another camel-a2a agent with separate task and config IDs.
# List push configs for a task
- setHeader:
name: CamelA2AOperation
constant: PUSH_CONFIG_LIST
- setHeader:
name: CamelA2ATaskId
simple: "${variable.taskId}"
- to: a2a:https://agent.example.com
# Get a specific push config
- setHeader:
name: CamelA2AOperation
constant: PUSH_CONFIG_GET
- setHeader:
name: CamelA2ATaskId
simple: "${variable.taskId}"
- setHeader:
name: CamelA2APushConfigId
simple: "${variable.configId}"
- to: "a2a:https://agent.example.com?protocolBinding=JSONRPC"
# Delete a push config
- setHeader:
name: CamelA2AOperation
constant: PUSH_CONFIG_DELETE
- setHeader:
name: CamelA2ATaskId
simple: "${variable.taskId}"
- setHeader:
name: CamelA2APushConfigId
simple: "${variable.configId}"
- to: "a2a:https://agent.example.com?protocolBinding=JSONRPC" Parallel Multicast
Call multiple agents concurrently using Camel’s multicast EIP:
- multicast:
parallelProcessing: true
aggregationStrategy: "#class:MyAggregator"
steps:
- to: direct:call-weather
- to: direct:call-news
- to: direct:call-fortune
- route:
id: call-weather
from:
uri: direct:call-weather
steps:
- removeHeaders:
pattern: "*"
- setBody:
constant: "What is the weather?"
- to:
uri: a2a:http://weather-agent:8080
parameters:
oauthProfile: my-profile | Use |
Address Override
Override the remote agent’s URL from the card using host/port/basePath config:
- to: a2a:https://agent.example.com?host=http://localhost&port=8080 Priority without producer credentials: host config > card’s supportedInterfaces URL > agentCardSource URL. When producer credentials are configured (apiKey, bearerToken, or oauthProfile) and host is not set, the producer sends credentialed requests only to the HTTP(S) agentCardSource origin. This avoids sending credentials to a URL supplied by the remote card’s supportedInterfaces field. The port and basePath producer overrides are applied when host is configured. A host value without a scheme is treated as HTTPS.
The producer does not forward arbitrary inbound Authorization or Cookie headers. Only credentials resolved from the A2A endpoint configuration are applied to outgoing requests. |
Redirect Handling
By default, the producer does not follow HTTP redirects to prevent credential leakage on cross-origin redirects. Enable only when the remote agent is known to issue redirects:
- to: a2a:https://agent.example.com?followRedirects=true followRedirects applies to operation requests only. Agent card discovery never follows redirects — a card URL that returns a redirect fails with an error. |
Producer operation responses must be 2xx. Redirect and other non-2xx responses fail before response deserialization, so a 3xx response cannot be accidentally parsed as an A2A payload.
Request Timeouts
The producer uses different timeouts depending on the operation:
-
Non-streaming requests: hard-coded at 60 seconds. There is currently no configuration parameter to override this.
-
Streaming requests (
SendStreamingMessage,SubscribeToTask): usesasyncTimeout(default 300,000 ms / 5 minutes). -
Streaming read timeout:
streamingReadTimeout(default 300,000 ms / 5 minutes) — if no SSE event arrives from the remote agent within this period, the stream is closed with an error. Prevents indefinite blocking when a remote agent stops sending events. -
Streaming event size: each decoded SSE event is limited by
maxPayloadSize(default 6 MiB). Oversized events fail the producer stream instead of accumulating unbounded memory. -
TCP connection:
connectTimeout(default 30,000 ms / 30 seconds).
Simple Language Functions
The component registers custom Simple language functions under the a2a: namespace:
| Function | Description |
|---|---|
| Emit a status update with |
| Emit a status update with an explicit |
| Extract |
| Extract |
| Extract |
| Extract |
| Extract |
| Extract |
| Full agent card as JSON string. Resolves the card from the A2A endpoint (consumer’s own card, or the cached remote card on the producer). |
| Agent card name. |
| Agent card description. |
| Agent card URL. |
| Agent card version. |
| Skills formatted as human-readable text, one per line: |
| Skills as a JSON array string. |
| Agent card icon URL. |
| Agent card documentation URL. |
| Agent provider as JSON string. |
| Agent capabilities as JSON string. |
| Supported interfaces as JSON array string. |
| Security schemes as JSON object string. |
| A2A v1.0 security requirements as JSON array string. |
| Legacy draft |
The ${a2a:card} functions resolve the agent card from the exchange context. On a consumer route, exchange.getFromEndpoint() provides the agent’s own card. On a producer route, the component scans registered endpoints for a cached remote card. Returns an empty string if no card is available.
The card functions support flat field access only — nested property access like ${a2a:card.skills[0].name} or ${a2a:card.provider.organization} is not supported. For structured data, use ${a2a:card.skills.json} or ${a2a:card} (full JSON) and parse downstream. |
Example — including remote agent skills in an LLM prompt:
- setBody:
simple: |
Generate a plan using these available skills:
${a2a:card.skills} For the full text of all parts combined, use the TypeConverter:
- convertBodyTo:
type: String Exchange Headers
| Header | Type | Description |
|---|---|---|
|
| A2A operation to invoke. Accepts both enum names ( |
|
| Task ID for |
|
| Push notification config ID for get/delete operations. |
|
| Context ID for multi-turn conversations. Set automatically on consumer inbound and producer response. |
|
| Message ID from the request/response. |
|
| Task state from the response (e.g., |
|
| A2A method name invoked (set on producer response). |
|
| Response type after a producer call ( |
|
| Override |
|
| Producer request field for |
|
| Authenticated user profile (set as exchange property, not a header). Populated when |
|
| Context ID filter for |
|
| Page size for |
|
| Pagination token for |
|
| Comma-separated status filter for |
|
| Whether |
|
| Maximum history messages to include per task in |
|
| Status timestamp filter for |
|
| Consumer-side negotiated A2A extension URIs from the |
|
| Consumer-side: the stream emitter injected for streaming routes. Available for advanced use cases — prefer |
CamelA2AResponseTask and CamelA2AUserProfile are exchange properties, not headers. Access them via $\{exchangeProperty.CamelA2AResponseTask} and $\{exchangeProperty.CamelA2AUserProfile} in Simple expressions. CamelA2AResponseTask contains the full Task object from the producer response, useful in polling workflows. CamelA2AUserProfile contains the authenticated user’s profile map, available when validateAuth=true. |
Supported Operations
| Operation (enum) | Method Name | Producer | Consumer |
|---|---|---|---|
|
| Send message, receive Task or Message | Route processes message |
|
| Lazy | Route emits via |
|
| Retrieve task from remote | Read from task store |
|
| List tasks from remote | Query task store |
|
| Cancel remote task | Update store + cancel in-flight async |
|
| Lazy | Event-to-Stream Bridge (real-time SSE) for REST and JSON-RPC bindings |
|
| Register webhook | Store config + SSRF validation |
|
| Retrieve config | Read from store |
|
| List configs | Read from store |
|
| Delete config | Remove from store |
The CamelA2AOperation header accepts both the enum name (left column) and the PascalCase method name (second column).
The built-in JSON-RPC consumer dispatches SendMessage, SendStreamingMessage, GetTask, ListTasks, CancelTask, SubscribeToTask, and the push notification config operations.
Pluggable Authentication
Authentication uses a strategy-per-scheme pattern via the A2ASecuritySchemeHandler SPI. One handler covers both producer (apply credentials) and consumer (validate credentials) for a given security scheme type.
Default handlers:
| Scheme Type | Handler | Description |
|---|---|---|
|
| OAuth profile or static bearer token |
|
| API key in |
|
| Token via |
|
| Same as OAuth2, stores OIDC discovery URL |
Override or extend by registering a custom handler bean:
@BindToRegistry("myCustomAuth")
public class MutualTlsHandler implements A2ASecuritySchemeHandler {
public String schemeType() { return "mutualTls"; }
public void applyCredentials(Exchange exchange, SecurityScheme scheme,
A2AConfiguration config, CamelContext context) {
// producer: configure mTLS
}
public A2AUserProfile validateCredentials(Exchange exchange,
SecurityScheme scheme, A2AConfiguration config) {
// consumer: extract client certificate
return A2AUserProfile.fromMap(Map.of(
"scheme", "mutualTls",
"subject", "CN=client"));
}
} Security Scheme Priority
When the agent card declares multiple security schemes, selection is based on configuration hints:
-
oauthProfileconfigured → OAuth2 / OpenID Connect schemes tried first -
bearerTokenconfigured → HTTP bearer schemes tried first -
apiKeyconfigured → API key schemes tried first -
No hint → schemes tried in agent card declaration order
On the producer, the first matching handler applies credentials. On the consumer, all declared schemes are tried until one succeeds (or all fail with 401).
Task State Machine
A2A tasks follow this state machine:
+--> COMPLETED
|
SUBMITTED --> WORKING +--> FAILED
|
+--> CANCELED
|
+--> INPUT_REQUIRED --> WORKING (resumed)
|
+--> AUTH_REQUIRED --> WORKING (after re-auth)
|
+--> REJECTED All valid task states (wire values use the A2A v1.0 proto names shown below; the $\{a2a:emit()} function uses the short Java enum names — e.g., INPUT_REQUIRED, not TASK_STATE_INPUT_REQUIRED):
| State | Description |
|---|---|
| Unknown or missing task state fallback |
| Task accepted, not yet processing |
| Task is actively being processed |
| Task completed successfully (terminal) |
| Task failed due to error or timeout (terminal) |
| Task was canceled by the client (terminal) |
| Agent needs additional input from the client before proceeding |
| Agent needs the client to re-authenticate before proceeding |
| Agent rejected the task (terminal) |
The consumer manages transitions automatically:
-
SUBMITTED— whenreturnImmediately=trueand the task is accepted -
WORKING— when the route starts processing (async tasks) -
COMPLETED— whensetBodyproduces a non-null response -
FAILED— when the route throws an exception orasyncTimeoutexpires
Routes can emit any state explicitly via ${a2a:emit(INPUT_REQUIRED, 'Please provide your address')} or A2AProgress.emit(exchange, TaskState.AUTH_REQUIRED, "Re-authentication needed").
Unified Body-as-Response
Both streaming and async agents use the same route pattern — setBody is always the final response:
steps:
- script:
simple: "${a2a:emit('progress...')}" # optional status events
- setBody:
constant: "final response" # consumer handles delivery The consumer handles delivery regardless of mechanism:
-
Synchronous: returned directly in the HTTP response
-
Streaming: emitted as an SSE message event
-
Async + polling: stored in the task’s history and status message, returned via
GetTask -
Push notifications: included in the
COMPLETEDstatus webhook payload
Agent Card Resolution
Card fields are resolved with layered precedence (lowest to highest):
-
Agent card JSON file — loaded from the URI source (base)
-
Bean reference —
agentCard=#myBeanfills/overrides fields from a programmatic object -
URI parameters —
name,description,versionoverride everything
Bean reference example — useful when the card needs programmatic construction (e.g., dynamic skills):
- route:
from:
uri: a2a:classpath:agent-card.json
parameters:
agentCard: "#myCardBean" URI parameter overrides — customize base cards per environment via properties:
- route:
from:
uri: a2a:classpath:agent-card.json
parameters:
name: "{{agent.name}}"
version: "{{agent.version}}" Security schemes from the card drive auth handler selection. Config parameters (oauthProfile, bearerToken, apiKey) provide the runtime credentials and influence scheme priority.
The resolver currently merges these card fields from file and bean cards: name, description, url, version, provider, capabilities, skills, supportedInterfaces, securitySchemes, securityRequirements, iconUrl, documentationUrl, defaultInputModes, defaultOutputModes, supportsAuthenticatedExtendedCard, and unknown extension properties. Legacy input cards using security are converted to securityRequirements. URI parameters can override only name, description, and version.
| The card is loaded once at endpoint startup and cached. Periodic card refresh is not yet implemented. |
HTTP Server Component Discovery
The consumer needs an HTTP server to register A2A endpoints. The discovery order is:
-
Explicit
httpServerComponentparameter (e.g.,undertow,jetty) -
Global
camel.rest.component/ REST DSL component configuration -
Already registered
platform-httpcomponent -
Exactly one already registered Camel component implementing
RestConsumerFactory -
Exactly one registry bean implementing
RestConsumerFactory
The preview component does not auto-create arbitrary HTTP server components and does not choose between multiple discovered server factories. Add and register one of the tested server components, or set httpServerComponent explicitly.
Tested behavior in this Preview release:
-
camel-platform-http/ Vert.x - suitable for JSON-RPC agents and serving the public agent card. REST custom-method paths containing colons can collide, and SSEInputStreamresponses are buffered. -
camel-undertowandcamel-jetty- suitable for REST custom-method paths and real-time SSE streaming.
For SSE streaming, specify httpServerComponent=undertow or httpServerComponent=jetty and add the matching component dependency.
Data Format
The dataFormat parameter controls what the exchange body contains, following the same convention as the CXF component. It applies to both consumer (inbound request) and producer (response) sides.
| Value | Default | Description |
|---|---|---|
| Yes | Extracts text content from message parts as a |
| No | Full Java model objects. On the consumer, the body is the |
| No | Raw JSON string with no deserialization. On the consumer, the body is the JSON representation of the incoming |
Example — consumer in POJO mode for multimodal processing:
- route:
from:
uri: a2a:classpath:agent-card.json?dataFormat=POJO
steps:
- log:
message: "Received ${body.parts().size()} parts"
- setBody:
simple: "Processed ${body.parts()[0]}" Example — producer in RAW mode for proxying:
- route:
from:
uri: direct:proxy
steps:
- to: a2a:http://remote-agent:8080?dataFormat=RAW
- log:
message: "Raw JSON: ${body}" Task Store
The default InMemoryTaskStore manages task state with lazy cleanup. It is thread-safe, using concurrent maps for storage and lock-protected eviction for task-associated data such as message-id mappings, subscribers, and push configs.
The completedTaskTtl URI parameter (default 3,600,000 ms / 1 hour) controls the TTL for terminal tasks. Expired tasks are evicted on read or during periodic cleanup.
The following settings are configurable programmatically on the InMemoryTaskStore bean, not via URI parameters:
| Property | Default | Description |
|---|---|---|
|
| Maximum tasks in the store. When exceeded, the oldest terminal task is evicted; if no terminal task exists, no task is evicted. Set to 0 for unlimited. |
|
| Number of |
|
| TTL for non-terminal stuck tasks ( |
To customize, register a bean implementing A2ATaskStore in the Camel registry. The preview SPI is also split into narrower parent contracts (A2ATaskRepository, A2ATaskSubscriptions, A2APushConfigStore, and A2ATaskCleanup) so custom stores can keep storage, subscription, push-config, and cleanup responsibilities separate. Registry-provided stores are wrapped by a guard that enforces terminal-state protection, delegates subscriber registration and notification to the custom store, tracks endpoint subscribers for terminal-event cleanup, validates push webhook URLs, and normalizes push config IDs/task IDs on the endpoint path. Custom stores still need to preserve durable storage and any cross-node notification semantics inside their own implementation; the dispatcher revalidates webhook URLs before delivery.
Custom subscribers can be registered on a task-store bean for audit logging, metrics, or custom delivery:
@BindToRegistry
public A2ATaskStore taskStore() {
MyAuditingTaskStore store = new MyAuditingTaskStore();
store.addGlobalSubscriber((taskId, event) -> auditLog.record(taskId, event));
return store;
} Model Classes
The component uses Java model classes aligned with A2A v1.0:
| Class | Description |
|---|---|
| Task with |
| Agent or user message with |
| Operation request wrapper with |
| Send-message runtime options with |
| Response wrapper containing exactly one of |
| List-tasks request with |
| List-tasks response with |
| SSE event wrapper containing exactly one of |
| Text content part implementing |
| Structured data part implementing |
| File content with |
| Status snapshot with |
| Named output with |
| Push notification webhook config with |
| Push notification config list response with |
Core immutable model types such as Task, Message, Artifact, and AgentCard provide builders, for example: Task.builder().id("t1").contextId("ctx-1").status(new TaskStatus(TaskState.WORKING)).build(). Mutable request/configuration classes use standard getters and setters.
Preview API and SPI Boundaries
The stable user-facing surface for this Preview release is the endpoint URI/options, exchange headers, documented model classes, type converters, and these registry SPIs: A2ATaskStore, A2AExtensionHandler, and A2ASecuritySchemeHandler.
The protocol and operation implementation classes under org.apache.camel.component.a2a.protocol and org.apache.camel.component.a2a.operation are public so the component can share implementation code across packages and tests, but they are not extension SPIs. A2AProtocol is sealed deliberately to the built-in REST and JSON-RPC bindings; select a binding with protocolBinding rather than implementing a custom protocol class.
The JSON mapper helper returns configured mapper copies. Customizing the returned ObjectMapper does not change component-wide serialization behavior.
Dependencies
The component has zero compile-time coupling to any HTTP library. Transport is discovered at runtime:
| Role | SPI | Examples |
|---|---|---|
Consumer HTTP server |
|
|
Producer HTTP client | Built-in | No additional dependency |
OAuth tokens |
|
|
Task state |
| Built-in |
Add the HTTP server component you want as a runtime dependency. Consumer routes require one of these dependencies; producer-only routes do not.
<!-- Consumer: JSON-RPC or non-streaming HTTP endpoint serving -->
<dependency>
<groupId>org.apache.camel</groupId>
<artifactId>camel-platform-http-vertx</artifactId>
<version>x.x.x</version>
<!-- use the same version as your Camel core version -->
</dependency>
<!-- Consumer: REST custom-method routes and real-time SSE streaming -->
<dependency>
<groupId>org.apache.camel</groupId>
<artifactId>camel-undertow</artifactId>
<version>x.x.x</version>
<!-- use the same version as your Camel core version -->
</dependency>
<!-- Alternative consumer transport for REST custom-method routes and real-time SSE streaming -->
<dependency>
<groupId>org.apache.camel</groupId>
<artifactId>camel-jetty</artifactId>
<version>x.x.x</version>
<!-- use the same version as your Camel core version -->
</dependency>
<!-- Optional: OAuth authentication -->
<dependency>
<groupId>org.apache.camel</groupId>
<artifactId>camel-oauth</artifactId>
<version>x.x.x</version>
<!-- use the same version as your Camel core version -->
</dependency>