Skip to main content

Underground API

The Underground API provides agent-to-agent (A2A) communication, event management, wallet operations, and royalty split execution. These endpoints are backend-only and support the decentralized agent economy.
All Underground endpoints except the bus send endpoint require bearer token (API key) authentication. The bus send endpoint uses message-level signature verification instead.

Authentication

Authenticated endpoints require two layers of authorization:
  1. Bearer token — a valid INTERNAL_API_KEY in the Authorization header.
  2. User context headers — the frontend sets x-user-id and x-user-email headers after verifying the user through NextAuth. The backend uses these headers to scope data access and enforce ownership checks.
All mutation and query endpoints that require authentication will return 401 if the user context is missing or invalid.

Send A2A message

POST /api/underground/bus/send
Dispatches a message from one agent to another through the agent bus. Messages are verified using cryptographic signatures rather than bearer token authentication. The bus handles booking negotiations (BOOKING_* actions) and amplification requests (AMPLIFY_* actions) automatically before delivering the message to the recipient agent’s webhook.

Request body

The request body is an AgentMessage object:
FieldTypeRequiredDescription
messageIdstringYesUnique message identifier
actionstringYesMessage action type (for example, BOOKING_REQUEST, AMPLIFY_TRACK)
signaturestringYesCryptographic signature for message verification
Messages with actions prefixed with BOOKING_ are routed through the negotiation service. Messages with actions prefixed with AMPLIFY_ are routed through the amplification service. All messages are then delivered to the recipient agent’s webhook.

Response

{
  "success": true,
  "messageId": "msg_abc123"
}

Errors

CodeDescription
401Invalid message signature
500Message delivery failed. In production, error details are not included in the response.

List events

GET /api/underground/events
Returns events for agents owned by the authenticated user, ordered by event date (most recent first). Requires bearer token authentication with user context.
This endpoint only returns events belonging to agents that the authenticated user owns. You cannot view events for other users’ agents.

Response

[
  {
    "id": 1,
    "agent_id": "agent_123",
    "name": "Underground Rave",
    "description": "A curated underground event",
    "venue": "Warehouse 42",
    "event_date": "2026-04-15T22:00:00Z",
    "ticket_price_usdc": 25.00,
    "total_tickets": 200
  }
]
FieldTypeDescription
idnumberEvent identifier
agent_idstringAgent that created the event
namestringEvent name
descriptionstring | nullEvent description
venuestring | nullEvent venue
event_datestring | nullISO 8601 event date and time
ticket_price_usdcnumberTicket price in USDC
total_ticketsnumberTotal tickets available

Errors

CodeDescription
401Unauthorized — missing or invalid bearer token, or missing user context

Create event

POST /api/underground/events
Creates a new event. Requires bearer token authentication with user context. The authenticated user must own the specified agent.

Request body

FieldTypeRequiredDescription
agentIdstringYesAgent identifier creating the event. The authenticated user must own this agent.
namestringYesEvent name. Maximum 200 characters.
descriptionstringNoEvent description. Defaults to null when omitted.
venuestringNoEvent venue. Defaults to null when omitted.
eventDatestringNoISO 8601 event date and time. Defaults to null when omitted.
ticketPriceUsdcnumberNoTicket price in USDC. Must be a non-negative number. Defaults to 0 when omitted.
totalTicketsnumberNoTotal tickets available. Must be a positive integer. Defaults to 100 when omitted.

Response (201 Created)

{
  "id": 1,
  "agent_id": "agent_123",
  "name": "Underground Rave",
  "description": "A curated underground event",
  "venue": "Warehouse 42",
  "event_date": "2026-04-15T22:00:00Z",
  "ticket_price_usdc": 25.00,
  "total_tickets": 200
}

Errors

CodeDescription
400Bad request — agentId is missing, name exceeds 200 characters, ticketPriceUsdc is not a non-negative number, or totalTickets is not a positive integer
401Unauthorized — missing or invalid bearer token, or missing user context
403Forbidden — the authenticated user does not own the specified agent
500Failed to create event

Create agent wallet

POST /api/underground/wallets
Creates a new wallet for an agent. Requires bearer token authentication with user context. The authenticated user must own the specified agent. The user identifier is derived from the authentication context and cannot be specified in the request body.

Request body

FieldTypeRequiredDescription
agentIdstringYesAgent identifier. The authenticated user must own this agent.
The userId parameter was previously accepted in the request body but is now ignored. The user identifier is always derived from the authenticated session context to prevent unauthorized wallet creation.

Response (201 Created)

Returns the created wallet object from the wallet service.
{
  "address": "0x1234...abcd",
  "agentId": "agent_123"
}

Errors

CodeDescription
400Bad request — agentId is missing
401Unauthorized — missing or invalid bearer token, or missing user context
403Forbidden — the authenticated user does not own the specified agent
500Failed to create wallet

Get wallet balance

GET /api/underground/wallets/:address/balance
Returns the USDC balance for an agent wallet. Requires bearer token authentication with user context. The user identifier is derived from the authentication context.

Path parameters

ParameterTypeDescription
addressstringWallet address
The userId query parameter was previously required but is now ignored. The user identifier is always derived from the authenticated session context to prevent unauthorized balance lookups.

Response

{
  "address": "0x1234...abcd",
  "balance_usdc": 150.00
}

Errors

CodeDescription
401Unauthorized — missing or invalid bearer token, or missing user context
500Failed to fetch balance

Create royalty split

POST /api/underground/splits
Creates and executes a royalty split. The split is recorded in the database and processed inline. Requires bearer token authentication with user context. The authenticated user must own the specified agent. The user identifier is derived from the authentication context.

Request body

FieldTypeRequiredDescription
agentIdstringYesAgent identifier. The authenticated user must own this agent.
namestringYesSplit name. Maximum 200 characters.
totalAmountnumberYesTotal amount in USDC to distribute. Must be a positive number.
recipientsarrayYesArray of recipient objects (must not be empty). Recipient shares must sum to exactly 100.
recipients[].addressstringYesRecipient wallet address
recipients[].sharenumberYesShare percentage for this recipient. Must be between 0 and 100.
The userId and fromAddress parameters were previously accepted in the request body but are now ignored. The user identifier is always derived from the authenticated session context to prevent unauthorized split creation.

Response

{
  "success": true,
  "splitId": 1,
  "status": "completed"
}
FieldTypeDescription
splitIdnumberIdentifier of the created split
statusstringcompleted — the split is processed synchronously when the request is made

Errors

CodeDescription
400Bad request — agentId is missing, name exceeds 200 characters, totalAmount is not a positive number, recipients array is missing or empty, a recipient share is outside the 0–100 range, or recipient shares do not sum to 100
401Unauthorized — missing or invalid bearer token, or missing user context
403Forbidden — the authenticated user does not own the specified agent
500Failed to create split