Skip to main content

Module error_handling

Module error_handling 

Source
Expand description

Error handling patterns in agent-client-protocol.

This chapter explains how errors work in agent-client-protocol callbacks and the difference between protocol errors (sent to the peer) and connection errors (which shut down the connection).

§Callback Return Types

Almost all agent-client-protocol callbacks return Result<_, crate::Error>. What happens when you return an Err depends on the context:

Returning Err from a callback shuts down the connection.

This is appropriate for truly unrecoverable situations—internal bugs, resource exhaustion, or when you want to terminate the connection. But most of the time, you want to send an error to the peer while keeping the connection alive.

§Sending Protocol Errors

To send an error response to a request (without closing the connection), use the request context’s respond method:

Client.builder()
    .on_receive_request(async |request: ValidateRequest, responder, _cx| {
        if request.data.is_empty() {
            // Send error to peer, keep connection alive
            responder.respond_with_error(agent_client_protocol::Error::invalid_params())?;
            return Ok(());
        }

        // Process valid request...
        responder.respond(ValidateResponse { is_valid: true, error: None })?;
        Ok(())
    }, agent_client_protocol::on_receive_request!())

For sending error notifications (one-way error messages), use send_error_notification:

cx.send_error_notification(agent_client_protocol::Error::internal_error()
    .data("Something went wrong"))?;

§The into_internal_error Helper

When working with external libraries that return their own error types, you need to convert them to agent_client_protocol::Error. The Error::into_internal_error method provides a convenient way to do this:

use agent_client_protocol::Error;

// Convert any error type to agent_client_protocol::Error
let value = serde_json::to_value(&data)
    .map_err(Error::into_internal_error)?;

// Or with a file operation
let contents = std::fs::read_to_string(path)
    .map_err(Error::into_internal_error);

This wraps the original error’s message in an internal error, which is appropriate for unexpected failures. For expected error conditions that should be communicated to the peer, create specific error types instead.

§Error Types

The Error type provides factory methods for common JSON-RPC error codes:

You can add context with .data():

let error = agent_client_protocol::Error::invalid_params()
    .data(serde_json::json!({
        "field": "timeout",
        "reason": "must be positive"
    }));

§Summary

SituationWhat to do
Send error response to requestresponder.respond(Err(error)) then Ok(())
Send error notificationcx.send_error_notification(error) then Ok(())
Shut down connectionReturn Err(error) from callback
Convert external error.map_err(Error::into_internal_error)?