Step 07 - Using Remote Agents (A2A)
New Requirement: Distributing the Pricing Service
In the previous steps, you built a complete disposition system with the Supervisor Pattern (Step 4), Human-in-the-Loop approval (Step 5), and multimodal image analysis that enriches rental feedback with visual observations from car photos (Step 6). The system works well, but the Miles of Smiles management team has a new architectural requirement:
The vehicle pricing logic needs to be maintained by a separate team and run as an independent service.
This is a common real-world scenario where:
- Different teams own different capabilities: The pricing team has specialized expertise in vehicle valuations and wants to maintain their own service
- The service needs to be reusable: Multiple client applications (not just car management) might need pricing estimates
- Independent scaling is required: The pricing service might need different resources than the main application
You’ll learn how to convert the local PricingAgent into a remote service using the Agent-to-Agent (A2A) protocol.
What You’ll Learn
In this step, you will:
- Understand the Agent-to-Agent (A2A) protocol for distributed agent communication
- Convert the local
PricingAgentinto a remote A2A service - Build a client agent that connects to remote A2A agents using
@A2AClientAgent - Create an A2A server that exposes an AI agent as a remote service
- Learn about AgentCard, AgentExecutor, and TaskUpdater components from the A2A SDK
- Understand the difference between Tasks and Messages in A2A protocol
- Run multiple Quarkus applications that communicate via A2A
- See the architectural trade-offs: lose Supervisor Pattern sophistication, gain distribution benefits
Note
At the moment the A2A integration is quite low-level and requires some boilerplate code. The Quarkus LangChain4j team is working on higher-level abstractions to simplify A2A usage in future releases.
Understanding the A2A Protocol
The Agent-to-Agent (A2A) protocol is an open protocol for AI agents to communicate across different systems and platforms.
Why A2A?
- Separation of concerns: Different teams can develop specialized agents independently
- Scalability: Distribute agent workload across multiple systems
- Reusability: One agent can serve multiple client applications
- Technology independence: Agents can be implemented in different languages/frameworks
A2A Architecture
graph LR
subgraph "Quarkus Runtime 1: Car Management System"
W[CarProcessingWorkflow]
PA["PricingAgent<br/>@A2AClientAgent"]
W --> PA
end
subgraph "A2A Protocol Layer"
AP[JSON-RPC over HTTP]
end
subgraph "Quarkus Runtime 2: Pricing Service"
AC[AgentCard<br/>Agent Metadata]
AE[AgentExecutor<br/>Request Handler]
PAI[PricingAgent<br/>AI Service]
AC -.describes.-> PAI
AE --> PAI
end
PA -->|A2A Request| AP
AP -->|A2A Response| PA
AP <-->|JSON-RPC| AE
The Flow:
- Client agent (
PricingAgentwith@A2AClientAgent) sends a request to the remote agent - A2A Protocol Layer (JSON-RPC) transports the request over HTTP
- AgentCard describes the remote agent’s capabilities (skills, inputs, outputs)
- AgentExecutor receives the request and orchestrates the execution
- Remote AI agent (
PricingAgentAI service) processes the request - Response flows back through the same path
Additional A2A Info
For more information about the A2A protocol and the actors involved, see the A2A documentation.
Understanding Tasks vs. Messages
The A2A protocol distinguishes between two types of interactions:
| Concept | Description | Use Case |
|---|---|---|
| Task | A long-running job with a defined goal and tracked state | “Estimate the market value of this vehicle” |
| Message | A single conversational exchange with no tracked state | Chat messages, quick questions |
In this step, we’ll use Tasks because vehicle pricing is a discrete job with a clear objective.
Task Lifecycle:
sequenceDiagram
participant Client as Client Agent
participant Server as A2A Server
participant Executor as AgentExecutor
participant AI as AI Agent
Client->>Server: Create Task (POST /tasks)
Server->>Executor: Initialize TaskUpdater
Executor->>AI: Execute with input
AI->>AI: Process and use tools
AI->>Executor: Return result
Executor->>Server: Update task status
Server->>Client: Task result
What Are We Going to Build?
We’ll convert Step 5’s architecture to use a remote pricing agent:
- Keep all HITL features: DispositionProposalAgent, HumanApprovalAgent, value-based routing, approval workflow — all carried forward from Step 5
- Keep DispositionAgent local: Disposition logic stays in the main application (same as Step 5)
- Convert PricingAgent to A2A Client: Changes from local agent to remote A2A client
- Create Remote A2A Server: A separate Quarkus application exposing the pricing service
The Complete Architecture:
graph TD
subgraph "Main Application (localhost:8080)"
R[Rental/Cleaning/Maintenance Returns]
FW[FeedbackAnalysisWorkflow<br/>Parallel Mapper]
FA[FeedbackAnalysisAgent]
FSA[FleetSupervisorAgent<br/>Supervisor]
PAC["PricingAgent<br/>@A2AClientAgent"]
DA[DispositionAgent<br/>Local]
R --> FW
FW --> FA
FA --> FSA
FSA --> PAC
FSA --> DA
end
subgraph "Remote Pricing Service (localhost:8888)"
AC[AgentCard]
AE[AgentExecutor]
PAI[PricingAgent<br/>AI Service]
AE --> PAI
end
PAC -->|A2A Protocol| AE
Prerequisites
Before starting:
- Completed Step 06 - This step directly builds on Step 6’s architecture
- Application from Step 06 is stopped (Ctrl+C)
- Ports 8080 and 8888 are available (you’ll run two applications simultaneously)
- Understanding of Step 6’s multimodal image analysis, Step 5’s HITL Pattern, and Step 4’s Supervisor Pattern (we keep the same patterns, just make PricingAgent remote)
Understanding the Project Structure
The Step 07 code includes two separate Quarkus applications:
section-2/step-07/
├── multi-agent-system/ # Main car management application (port 8080)
│ ├── src/main/java/com/carmanagement/
│ │ ├── agentic/
│ │ │ ├── agents/
│ │ │ │ ├── PricingAgent.java # A2A client agent
│ │ │ │ ├── DispositionAgent.java # Local agent
│ │ │ │ ├── DispositionProposalAgent.java # Creates proposals
│ │ │ │ ├── HumanApprovalAgent.java # @HumanInTheLoop
│ │ │ │ └── FeedbackAnalysisAgent.java # Parameterized feedback analyzer
│ │ │ └── workflow/
│ │ │ ├── FeedbackAnalysisWorkflow.java # Parallel mapper analysis
│ │ │ └── CarProcessingWorkflow.java # Main orchestrator
│ │ ├── model/
│ │ │ └── ApprovalProposal.java # Approval entity
│ │ ├── resource/
│ │ │ └── ApprovalResource.java # Approval REST endpoints
│ │ └── service/
│ │ └── ApprovalService.java # Manages HITL workflow
│ └── pom.xml
│
└── remote-a2a-agent/ # Remote pricing service (port 8888)
├── src/main/java/com/demo/
│ ├── PricingAgentCard.java # Describes agent capabilities
│ ├── PricingAgentExecutor.java # Handles A2A requests
│ └── PricingAgent.java # AI service for vehicle pricing
└── pom.xml
Why Two Applications?
- Simulates a real-world scenario where different teams maintain different agents
- The pricing service could be reused by multiple client applications
- Demonstrates cross-application agent communication via A2A
Warning: this chapter involves many steps
In order to build out the solution, you will need to go through quite a few steps. While it is entirely possible to make the code changes manually (or via copy/paste), we recommend starting fresh from Step 07 with the changes already applied. You will then be able to walk through this chapter and focus on the examples and suggested experiments at the end of this chapter.
Navigate to the complete section-2/step-07/multi-agent-system directory:
If you want to continue building on your previous code, place yourself at the root of your project and copy the updated files:
cp ../step-07/multi-agent-system/pom.xml ./pom.xml
cp ../step-07/multi-agent-system/src/main/java/com/carmanagement/model/CarInfo.java ./src/main/java/com/carmanagement/model/CarInfo.java
cp ../step-07/multi-agent-system/src/main/java/com/carmanagement/model/CarStatus.java ./src/main/java/com/carmanagement/model/CarStatus.java
cp ../step-07/multi-agent-system/src/main/resources/META-INF/resources/css/styles.css ./src/main/resources/META-INF/resources/css/styles.css
cp ../step-07/multi-agent-system/src/main/resources/META-INF/resources/js/app.js ./src/main/resources/META-INF/resources/js/app.js
cp ../step-07/multi-agent-system/src/main/resources/META-INF/resources/index.html ./src/main/resources/META-INF/resources/index.html
cp ../step-07/multi-agent-system/src/main/resources/import.sql ./src/main/resources/import.sql
copy ..\step-07\multi-agent-system\pom.xml .\pom.xml
copy ..\step-07\multi-agent-system\src\main\java\com\carmanagement\model\CarInfo.java .\src\main\java\com\carmanagement\model\CarInfo.java
copy ..\step-07\multi-agent-system\src\main\java\com\carmanagement\model\CarStatus.java .\src\main\java\com\carmanagement\model\CarStatus.java
copy ..\step-07\multi-agent-system\src\main\resources\META-INF\resources\css\styles.css .\src\main\resources\META-INF\resources\css\styles.css
copy ..\step-07\multi-agent-system\src\main\resources\META-INF\resources\js\app.js .\src\main\resources\META-INF\resources\js\app.js
copy ..\step-07\multi-agent-system\src\main\resources\META-INF\resources\index.html .\src\main\resources\META-INF\resources\index.html
copy ..\step-07\multi-agent-system\src\main\resources\import.sql .\src\main\resources\import.sql
Part 1: Convert PricingAgent to A2A Client
The only change needed in the main application is converting the PricingAgent from a local agent to an A2A client.
Step 1: Update the PricingAgent to A2A Client
This is the key change from Step 5! The PricingAgent was a local agent with detailed pricing guidelines, depreciation tables, and an @Output post-processor. Now it becomes a simple client that delegates to the remote service.
Step 5 Version (Local):
- Had detailed
@SystemMessagewith pricing guidelines and depreciation tables - Had
@Outputmethod to normalize value format - Made decisions locally using AI
Step 7 Version (A2A Client):
- Uses
@A2AClientAgentto connect to remote service - Delegates all pricing logic to the remote service
- No
@Outputmethod needed — the remote service handles formatting
In src/main/java/com/carmanagement/agentic/agents, update PricingAgent.java:
package com.carmanagement.agentic.agents;
import dev.langchain4j.agentic.declarative.A2AClientAgent;
/**
* Agent that estimates the market value of a vehicle.
* Delegates to the remote A2A pricing service.
*/
public interface PricingAgent {
@A2AClientAgent(a2aServerUrl = "http://localhost:8888",
outputKey = "carValue",
description = "Pricing specialist that estimates vehicle market value based on make, model, year, and condition")
String estimateValue(String carMake, String carModel, Integer carYear, String carCondition);
}
Let’s break it down:
@A2AClientAgent Annotation
This annotation transforms the method into an A2A client:
a2aServerUrl: The URL of the remote A2A server
The Method Signature
These parameters are sent to the remote agent as task inputs. The parameters match exactly what the remote PricingAgent expects (same as Step 5’s local version).
How It Works
-
When this method is called, Quarkus LangChain4j:
- Creates an A2A Task with the method parameters as inputs
- Sends the task to the remote server via JSON-RPC
- Waits for the remote agent to complete the task
- Returns the result as a String
-
No manual HTTP requests needed
- Type-safe: compile-time checking of parameters
- Automatic error handling and retries
Part 2: Build the Remote A2A Server
Now let’s build the remote pricing service that will handle A2A requests from the main application.
Navigate to the remote-a2a-agent directory:
Step 2: Create the PricingAgent (AI Service)
The AI agent that estimates vehicle market values — the same logic that was local in Step 5.
In src/main/java/com/demo, create PricingAgent.java:
package com.demo;
import dev.langchain4j.service.SystemMessage;
import dev.langchain4j.service.UserMessage;
import io.quarkiverse.langchain4j.RegisterAiService;
import jakarta.enterprise.context.ApplicationScoped;
/**
* Agent that estimates the market value of a vehicle.
*/
@RegisterAiService
@ApplicationScoped
public interface PricingAgent {
@SystemMessage("""
You are a vehicle pricing specialist with expertise in market valuations.
Today is {current_date}. Use this to calculate the current year and vehicle age.
Use these pricing guidelines:
Brand Base Values (new current-year models):
- Luxury brands (Mercedes-Benz, BMW, Audi): $50,000-$70,000
- Premium trucks (Ford F-150): $45,000-$60,000
- Mainstream brands (Toyota, Honda, Chevrolet): $28,000-$42,000
- Economy brands (Nissan): $22,000-$35,000
Depreciation (calculate age as: current year - vehicle year):
- Age 1 year (nearly new): -12% from base value
- Age 2 years: -15% additional (27% total depreciation)
- Age 3 years: -12% additional (39% total depreciation)
- Age 4 years: -10% additional (49% total depreciation)
- Age 5+ years: -8% per additional year
Condition Adjustments (apply after depreciation):
- Excellent/Like new: +5% to depreciated value
- Good/Recently serviced: No adjustment
- Fair/Minor issues: -10% from depreciated value
- Poor/Needs work: -20% from depreciated value
Provide:
1. Estimated market value (single dollar amount with comma separator)
2. Brief justification (2-3 sentences explaining age, condition, and brand factors)
Format your response as:
Estimated Value: $XX,XXX
Justification: [Your reasoning including vehicle age]
""")
@UserMessage("""
Estimate the current market value of this vehicle:
- Make: {carMake}
- Model: {carModel}
- Year: {carYear}
- Condition: {carCondition}
""")
String estimateValue(String carMake, String carModel, Integer carYear, String carCondition);
}
Key Points:
@RegisterAiService: Registers this as an AI service- System message: Identical to step-05’s local PricingAgent — same pricing guidelines and depreciation tables
- Parameters:
carMake,carModel,carYear,carCondition— exactly matching the client’s method signature - No tools needed: Pricing is purely LLM-based, no tool invocation
AI Service vs. Agentic Agent
Notice this is a traditional AI service (from Section 1), not an agentic workflow. The A2A server can expose both types.
Step 3: Create the AgentCard
The AgentCard describes the agent’s capabilities, skills, and interface.
In src/main/java/com/demo, create PricingAgentCard.java:
package com.demo;
import java.util.List;
import jakarta.enterprise.context.ApplicationScoped;
import jakarta.enterprise.inject.Produces;
import io.a2a.server.PublicAgentCard;
import io.a2a.spec.AgentCapabilities;
import io.a2a.spec.AgentCard;
import io.a2a.spec.AgentInterface;
import io.a2a.spec.AgentSkill;
import io.a2a.spec.TransportProtocol;
@ApplicationScoped
public class PricingAgentCard {
@Produces
@PublicAgentCard
public AgentCard agentCard() {
return new AgentCard.Builder()
.name("Pricing Agent")
.description("Estimates the market value of a vehicle based on make, model, year, and condition.")
.url("http://localhost:8888/")
.version("1.0.0")
.protocolVersion("1.0.0")
.capabilities(new AgentCapabilities.Builder()
.streaming(true)
.pushNotifications(false)
.stateTransitionHistory(false)
.build())
.defaultInputModes(List.of("text"))
.defaultOutputModes(List.of("text"))
.skills(List.of(new AgentSkill.Builder()
.id("pricing")
.name("Vehicle pricing")
.description("Estimates the market value of a vehicle based on make, model, year, and condition")
.tags(List.of("pricing", "valuation"))
.build()))
.preferredTransport(TransportProtocol.JSONRPC.asString())
.additionalInterfaces(List.of(
new AgentInterface(TransportProtocol.JSONRPC.asString(), "http://localhost:8888/")))
.build();
}
}
Let’s break it down:
@PublicAgentCard Annotation
This makes the AgentCard available at the /card endpoint.
Clients can query this endpoint to discover the agent’s capabilities.
AgentCard Components
Basic Information:
.name("Pricing Agent")
.description("Estimates the market value of a vehicle based on make, model, year, and condition.")
.url("http://localhost:8888/")
.version("1.0.0")
Capabilities:
.capabilities(new AgentCapabilities.Builder()
.streaming(true)
.pushNotifications(false)
.stateTransitionHistory(false)
.build())
Skills:
.skills(List.of(new AgentSkill.Builder()
.id("pricing")
.name("Vehicle pricing")
.description("Estimates the market value of a vehicle based on make, model, year, and condition")
.tags(List.of("pricing", "valuation"))
.build()))
Skills describe what the agent can do. This helps clients discover appropriate agents for their needs.
Transport Protocol:
.preferredTransport(TransportProtocol.JSONRPC.asString())
.additionalInterfaces(List.of(
new AgentInterface(TransportProtocol.JSONRPC.asString(), "http://localhost:8888")))
Specifies that this agent communicates via JSON-RPC over HTTP.
Step 4: Create the AgentExecutor
The AgentExecutor handles incoming A2A requests and orchestrates the AI agent.
In src/main/java/com/demo, create PricingAgentExecutor.java:
package com.demo;
import jakarta.enterprise.context.ApplicationScoped;
import jakarta.enterprise.inject.Produces;
import io.a2a.server.agentexecution.AgentExecutor;
import io.a2a.server.agentexecution.RequestContext;
import io.a2a.server.events.EventQueue;
import io.a2a.server.tasks.TaskUpdater;
import io.quarkus.logging.Log;
import java.util.ArrayList;
import java.util.List;
import io.a2a.spec.JSONRPCError;
import io.a2a.spec.Message;
import io.a2a.spec.Part;
import io.a2a.spec.TextPart;
import io.a2a.spec.UnsupportedOperationError;
/**
* Executor for the PricingAgent.
* Handles the integration between the A2A framework and the PricingAgent.
*/
@ApplicationScoped
public class PricingAgentExecutor {
@Produces
public AgentExecutor agentExecutor(PricingAgent pricingAgent) {
return new AgentExecutor() {
@Override
public void execute(RequestContext context, EventQueue eventQueue) throws JSONRPCError {
Log.info("Remote A2A PricingAgent called");
TaskUpdater updater = new TaskUpdater(context, eventQueue);
if (context.getTask() == null) {
updater.submit();
}
updater.startWork();
List<String> inputs = new ArrayList<>();
// Process the request message
Message message = context.getMessage();
if (message.getParts() != null) {
for (Part<?> part : message.getParts()) {
if (part instanceof TextPart textPart) {
inputs.add(textPart.getText());
}
}
}
Log.debugf("Estimating value for %s %s %s",
inputs.get(0), inputs.get(1), inputs.get(2));
// Call the pricing agent with all parameters
String agentResponse = pricingAgent.estimateValue(
inputs.get(0), // carMake
inputs.get(1), // carModel
Integer.parseInt(inputs.get(2)), // carYear
inputs.get(3)); // carCondition
Log.debugf("PricingAgent response: %s", agentResponse);
// Return the result
TextPart responsePart = new TextPart(agentResponse, null);
List<Part<?>> parts = List.of(responsePart);
updater.addArtifact(parts, null, null, null);
updater.complete();
}
@Override
public void cancel(RequestContext context, EventQueue eventQueue) throws JSONRPCError {
throw new UnsupportedOperationError();
}
};
}
}
Let’s break it down:
CDI Bean with AgentExecutor Factory
@ApplicationScoped
public class PricingAgentExecutor {
@Produces
public AgentExecutor agentExecutor(PricingAgent pricingAgent)
Produces an AgentExecutor bean that Quarkus LangChain4j will use to handle A2A task requests.
Task Processing
The executor extracts the input parameters from the incoming message and calls the PricingAgent:
String agentResponse = pricingAgent.estimateValue(
inputs.get(0), // carMake
inputs.get(1), // carModel
Integer.parseInt(inputs.get(2)), // carYear
inputs.get(3)); // carCondition
Extracts each parameter by index from the message parts. The order matches the client’s method signature exactly.
Return the Result
TextPart responsePart = new TextPart(agentResponse, null);
List<Part<?>> parts = List.of(responsePart);
updater.addArtifact(parts, null, null, null);
updater.complete();
Creates a text part with the agent’s response and sends it back to the client via the TaskUpdater. This completes the A2A task.
Try It Out
You’ll need to run two applications simultaneously.
Terminal 1: Start the Remote A2A Server
Wait for:
The remote service is now running and ready to accept A2A requests for pricing!
Terminal 2: Start the Main Application
Open a new terminal and run:
Wait for:
Test the Complete Flow
Open your browser to http://localhost:8080.
You’ll see the Fleet Status grid with inline feedback forms in the Action column and the approval notification button.

iFind the Honda Civic (status: Rented) in the Fleet Status grid and enter feedback indicating severe damage:
Click Return.
What happens?
-
Parallel Analysis (
FeedbackAnalysisWorkflow):FeedbackTask.disposition()executed byFeedbackAnalysisAgent: “Disposition required — severe damage”FeedbackTask.maintenance()executed byFeedbackAnalysisAgent: “Major repairs needed”FeedbackTask.cleaning()executed byFeedbackAnalysisAgent: “Not applicable”
-
Supervisor Orchestration (FleetSupervisorAgent):
- Analyzes feedback and determines disposition is required
- Invokes PricingAgent (remote via A2A) to estimate vehicle value
- Invokes DispositionAgent (local) to determine disposition
-
A2A Communication (for pricing):
- Client sends task to
http://localhost:8888 AgentExecutorreceives and processes taskPricingAgent(AI service) estimates the vehicle value- Result flows back to client
- Client sends task to
-
Local Disposition:
DispositionAgentdetermines action based on value and condition
-
UI Update:
- Car status →
DISPOSED - Car status updates to
PENDING_DISPOSITIONin the Fleet Status grid
- Car status →
Check the Logs
Terminal 1 (Remote A2A Server):
Terminal 2 (Main Application):
[FeedbackAnalysisAgent/disposition] DISPOSITION_REQUIRED - Severe structural damage, uneconomical to repair
[FleetSupervisorAgent] Invoking PricingAgent for value estimation
[PricingAgent @A2AClientAgent] Sending task to http://localhost:8888
[PricingAgent @A2AClientAgent] Received result: Estimated Value: $12,500
[FleetSupervisorAgent] Invoking DispositionAgent
[DispositionAgent] Result: Car should be scrapped...
Notice the cross-application communication via A2A!
How It All Works Together
Let’s trace the complete flow:
sequenceDiagram
participant User
participant Service as CarManagementService
participant Workflow as CarProcessingWorkflow
participant FeedbackWF as FeedbackAnalysisWorkflow
participant Supervisor as FleetSupervisorAgent
participant PricingClient as PricingAgent<br/>@A2AClientAgent
participant A2A as A2A Protocol<br/>(JSON-RPC)
participant Executor as AgentExecutor
participant PricingRemote as PricingAgent<br/>AI Service
participant Disposition as DispositionAgent<br/>(Local)
User->>Service: Return car with severe damage
Service->>Workflow: processCarReturn(...)
rect rgb(255, 243, 205)
Note over Workflow,FeedbackWF: Parallel Analysis
Workflow->>FeedbackWF: Execute
par Concurrent Execution
FeedbackWF->>FeedbackWF: FeedbackAnalysisAgent<br/>with FeedbackTask.cleaning()
and
FeedbackWF->>FeedbackWF: FeedbackAnalysisAgent<br/>with FeedbackTask.maintenance()
and
FeedbackWF->>FeedbackWF: FeedbackAnalysisAgent<br/>with FeedbackTask.disposition()<br/>Result: "DISPOSITION_REQUIRED"
end
end
rect rgb(212, 237, 218)
Note over PricingClient,PricingRemote: A2A Communication (Pricing)
Supervisor->>PricingClient: Estimate vehicle value
PricingClient->>A2A: Create Task with inputs
A2A->>Executor: POST /tasks
Executor->>PricingRemote: estimateValue(...)
PricingRemote->>Executor: Return valuation
Executor->>A2A: Update task status
A2A->>PricingClient: Return result
end
rect rgb(248, 215, 218)
Note over Supervisor,Disposition: Local Disposition
Supervisor->>Disposition: processDisposition(...)
Disposition->>Supervisor: Return recommendation
end
Supervisor->>Workflow: Return disposition result
Workflow->>Service: Return CarConditions
Service->>Service: Set status to DISPOSED
Service->>User: Update UI
Understanding the A2A Implementation
Client Side (@A2AClientAgent)
The A2A client agent is remarkably simple:
@A2AClientAgent(a2aServerUrl = "http://localhost:8888", ...)
String estimateValue(...) // PricingAgent
Quarkus LangChain4j handles:
- Creating the A2A task
- Serializing method parameters as task inputs
- Sending the HTTP request via JSON-RPC
- Waiting for the response
- Deserializing the result
- Error handling and retries
Server Side (AgentCard + AgentExecutor)
The server requires more components:
| Component | Purpose |
|---|---|
| AgentCard | Describes agent capabilities, published at /card endpoint |
| AgentExecutor | Receives and processes A2A task requests |
| TaskUpdater | Updates task status and sends results back to client |
| AI Agent | The actual AI service (PricingAgent) |
This separation allows: - Agents to focus on business logic - A2A infrastructure to handle protocol details - Remote agents to be reused — any application can connect to the pricing service via A2A
Key Takeaways
- A2A enables distributed agents: Different teams can maintain specialized agents in separate systems
@A2AClientAgentis powerful: Simple annotation transforms a method into an A2A client- AgentCard describes capabilities: Clients can discover what remote agents can do
- AgentExecutor handles protocol: Separates A2A infrastructure from agent logic
- Tasks vs. Messages: A2A supports both task-based and conversational interactions
- Type-safe integration: Method parameters automatically become task inputs
- Remote agents integrate seamlessly: Works with existing workflows and local agents
- Two runtimes communicate: Real-world simulation of distributed agent systems
- Selective distribution: Not every agent needs to be remote — only distribute what benefits from it (e.g., the pricing service can be reused by other applications)
- Local + remote mix: Combining local agents (DispositionAgent) with remote A2A agents (PricingAgent) in the same workflow
Experiment Further
1. Add Agent Discovery
The AgentCard is published at http://localhost:8888/card. Try:
You’ll see the full agent description including the pricing skill, capabilities, and transport protocols.
2. Test Different Disposition Scenarios
Try these feedback examples:
Scenario 1: Sell the car
Scenario 2: Donate the car
Scenario 3: Scrap the car
Observe how the remote agent makes different decisions!
3. Create Your Own A2A Agent
What other specialized agents could be useful?
- Route Planner Agent: Plans maintenance schedules for the fleet
- Insurance Agent: Assesses insurance claims for damaged cars
- Inventory Agent: Tracks fleet availability across locations
Try creating a simple A2A server for one of these!
4. Monitor A2A Communication
Add logging to see the JSON-RPC messages:
This shows the raw A2A protocol messages.
Troubleshooting
Connection refused to localhost:8888
Make sure the remote A2A server is running in Terminal 1. Check for:
If you see “Port already in use”, another application is using port 8888. You can change it in remote-a2a-agent/src/main/resources/application.properties:
Then update the client’s a2aServerUrl accordingly.
Task execution timeout
If the remote agent takes too long to respond, you might see a timeout error. The default timeout is sufficient for most cases, but you can increase it if needed by configuring the A2A client.
Parameter mismatch errors
If you see errors about missing parameters, verify that:
- Client agent method parameter names match what AgentExecutor extracts
- The text parts are extracted in the correct order in the
AgentExecutor - All required parameters are being sent by the client
Both applications on same port
If you see “Port already in use” on 8080:
- Make sure you stopped the application from Step 06
- Only run the main application from
multi-agent-system, not from a previous step directory - Check for zombie Java processes:
ps aux | grep java
What’s Next?
You’ve successfully distributed the pricing service as a remote A2A agent while keeping the rest of the system local!
You learned how to:
- Convert local agents to remote A2A services
- Connect to remote agents using
@A2AClientAgent - Build A2A servers with AgentCard and AgentExecutor
- Integrate remote agents into complex workflows
- Run multiple Quarkus applications that communicate via A2A
- Understand the architectural trade-offs between local and distributed agents
Key Progression: - Step 4: Sophisticated local orchestration with Supervisor Pattern - Step 5: Human-in-the-Loop for safe, controlled autonomous decisions - Step 6: Multimodal image analysis for enriched feedback - Step 7: Distributed architecture with A2A protocol
Congratulations on completing the final step of Section 2! Ready to wrap up? Head to the conclusion to review everything you’ve learned and see how these patterns apply to real-world scenarios!
Continue to Conclusion - Mastering Agentic Systems