The AgenticAdvertising.org registry provides a public REST API for resolving brands and properties, discovering agents, and validating authorization in the AdCP ecosystem.Documentation Index
Fetch the complete documentation index at: https://docs.adcontextprotocol.org/llms.txt
Use this file to discover all available pages before exploring further.
Base URL
/.well-known/openapi.yaml.
Quick Start
Resolve a brand domain to its canonical identity:Response
Rate Limits
| Endpoint | Limit |
|---|---|
Bulk resolve (/api/brands/resolve/bulk, /api/properties/resolve/bulk) | 20 requests/minute per IP |
Save endpoints (/api/brands/save, /api/properties/save) | 60 requests/hour per user |
Crawl request (/api/registry/crawl-request) | 5 minutes per domain, 30 requests/hour per user |
| All other endpoints | No limit |
429 Too Many Requests when the limit is exceeded.
Endpoint Groups
Brand Resolution
Resolve domains to canonical brand identities, fetch brand.json files, and browse the brand registry.
Property Resolution
Resolve publisher domains to property information, validate adagents.json, and browse properties.
Agent Discovery
List, search, and filter agents by inventory profile. Browse publishers and view registry statistics.
Change Feed
Poll a cursor-based feed of registry changes for local sync.
Lookups & Authorization
Look up agents by domain, validate product authorization, and check property authorization in real time.
Lookups by entity
Three endpoints answer different questions about who is in the registry. They span two tag groups in the API reference, so the table below is the entry point.| Endpoint | Question it answers |
|---|---|
GET /api/registry/agents | βGive me the catalog.β |
GET /api/registry/operator?domain=X | βWhat does this entity operate?β |
GET /api/registry/publisher?domain=X | βWhat does this entity publish?β |
GET /api/registry/agents β the catalog of AAO-attested member-enrolled agents in the registry. Use this when you want to browse or filter the public agent population. See Registering an agent for how agents end up in this catalog. Reference: List agents endpoint under Agent Discovery.
GET /api/registry/operator?domain=X β given a domain, returns the agents that entity operates and the publishers that trust them. Use this when you have one entity in hand and want its agent footprint. Response shape is auth-aware: anonymous callers see only public agents, AAO API-tier callers also see members_only, and profile owners additionally see private (server/src/routes/registry-api.ts:5486). Reference: Operator lookup under Authorization Lookups.
GET /api/registry/publisher?domain=X β given a domain, returns the inventory that entity publishes (properties[]) and which agents it authorizes (authorized_agents[] from its adagents.json). Use this when you have one publisher in hand and want its inventory and delegations. The endpoint is unauthenticated and returns the same shape for every caller (server/src/routes/registry-api.ts:5550). Reference: Publisher lookup under Authorization Lookups.
Brand Resolution
These endpoints resolve domains to brand identities. Thesource field in the response indicates where the data came from:
| Source | Meaning |
|---|---|
brand_json | Resolved from the domainβs /.well-known/brand.json file |
enriched | Enriched via Brandfetch API |
community | Submitted by a community member |
/api/brands/enrich or look up the brand in the registry.
| Method | Path | Description |
|---|---|---|
| GET | /api/brands/resolve | Resolve a domain to its canonical brand |
| POST | /api/brands/resolve/bulk | Resolve up to 100 domains at once |
| GET | /api/brands/brand-json | Fetch raw brand.json for a domain |
| GET | /api/brands/registry | List all brands (search, pagination) |
| GET | /api/brands/enrich | Enrich brand data via Brandfetch |
| GET | /api/brands/history | Edit history for a brand |
| POST | /api/brands/save | Save or update a community brand (auth required) |
Property Resolution
| Method | Path | Description |
|---|---|---|
| GET | /api/properties/resolve | Resolve a domain to its property info |
| POST | /api/properties/resolve/bulk | Resolve up to 100 domains at once |
| GET | /api/properties/registry | List all properties (search, pagination) |
| GET | /api/properties/validate | Validate a domainβs adagents.json |
| GET | /api/properties/history | Edit history for a property |
| POST | /api/properties/save | Save or update a hosted property (auth required) |
Agent Discovery
| Method | Path | Description | |||||||
|---|---|---|---|---|---|---|---|---|---|
| GET | /api/registry/agents | List all member-enrolled agents β filter by type (`?type=brand | rights | measurement | governance | creative | sales | buying | signals`) |
| GET | /api/registry/agents/search | Search agents by inventory profile (auth required) | |||||||
| GET | /api/registry/publishers | List all publishers | |||||||
| GET | /api/registry/stats | Registry statistics | |||||||
| POST | /api/registry/crawl-request | Request re-crawl of a publisher domain (auth required) |
Measurement-vendor discovery
To discover measurement vendors specifically (Adelaide-style attention, Scope3-style emissions, Nielsen DAR, IAS/DV custom quality, etc.), filter the agent list bytype=measurement:
get_adcp_capabilities.measurement.metrics[] β the canonical, vendor-controlled source of truth. AAO crawls each measurement agentβs get_adcp_capabilities on a TTL and stores the result; passing ?capabilities=true folds the catalog into the response next to creative_capabilities and signals_capabilities.
Filter parameters (all imply type=measurement when present; an explicit non-measurement type returns 400):
| Param | Match | Notes |
|---|---|---|
metric_id=attention_units | Exact match on metrics[].metric_id | Repeatable; multiple values are ORβd within the param. |
accreditation=MRC | Exact match on metrics[].accreditations[].accrediting_body | Repeatable. Vendor-asserted β verified_by_aao is always false in the response; renderers should mark these as vendor claims, not AAO endorsements. |
q=attention | Case-insensitive substring on metric_id | v1 scope: metric_id only. Max 64 chars; SQL wildcards (%, _) rejected. Description/standard fuzzy search is a follow-up. |
| Use case | Path | Why |
|---|---|---|
| Discovery / planning (βwhich vendors offer attention?β) | AAO index (?metric_id=...) | Pre-aggregated, cached, fast. Cross-vendor in one call. Stale up to the AAO TTL window (typically 24h). |
| Settlement / audit (βdoes Adelaide currently support metric X?β) | Direct call to get_adcp_capabilities | Live, canonical, no staleness. The buyer is already calling the measurement agent at delivery / reconciliation time; one extra call is cheap and removes index staleness from the audit trail. |
Filtering at get_products time | AAO index | Buyer is in a fast-path query and the sellerβs product catalog already needs to know which vendors are valid. |
Change Feed
| Method | Path | Description |
|---|---|---|
| GET | /api/registry/feed | Poll cursor-based registry change feed (auth required) |
Lookups & Authorization
| Method | Path | Description |
|---|---|---|
| GET | /api/registry/lookup/domain/{domain} | Find agents authorized for a domain |
| GET | /api/registry/lookup/property | Find agents by property identifier |
| GET | /api/registry/lookup/agent/{agentUrl}/domains | Get all domains for an agent |
| POST | /api/registry/validate/product-authorization | Validate agent product authorization |
| POST | /api/registry/expand/product-identifiers | Expand property selectors to identifiers |
| GET | /api/registry/validate/property-authorization | Real-time authorization check |
adagents.json (does it authorize this agent with the claimed delegation_type?) and the operatorβs brand.json (does it declare this property with a matching relationship?).
Validation Tools
| Method | Path | Description |
|---|---|---|
| POST | /api/adagents/validate | Validate adagents.json for a domain |
| POST | /api/adagents/create | Generate adagents.json content |
Search
| Method | Path | Description |
|---|---|---|
| GET | /api/search | Search across brands, publishers, and properties |
| GET | /api/manifest-refs/lookup | Find manifest references for a domain |
Agent Probing
| Method | Path | Description |
|---|---|---|
| GET | /api/public/discover-agent | Probe an agent URL for capabilities |
| GET | /api/public/agent-formats | Get creative formats from an agent |
| GET | /api/public/agent-products | Get products from a sales agent |
| GET | /api/public/validate-publisher | Validate a publisher domain |
Activity history
GET /api/brands/history?domain={domain} and GET /api/properties/history?domain={domain} return the edit history for a registry entry, newest first. These are public endpoints β no authentication required.
Response
editor_name: "system" were written by automated enrichment. When is_rollback is true, rolled_back_to contains the revision number that was restored. Pagination uses limit (max 100) and offset query parameters.
Anti-abuse and anti-homograph controls
Because/api/brands/save, /api/properties/save, and the adagents validation endpoints accept domain strings from authenticated member organizations, the hosted registry applies a layered floor of anti-abuse controls at save time. These are operational behaviors of the AgenticAdvertising.org registry in the 3.x era β not a new wire surface β and exist so that typosquats, confusable lookalikes, and drive-by brand hijacks cannot get written into the index by a single authenticated caller.
- Domain normalization (IDNA 2008 + confusable detection). Save endpoints SHOULD apply IDNA 2008 to normalize internationalized domain names to ASCII before persistence, and SHOULD then run Unicode confusable-detection (for example, ICU
uspoofor equivalent) against two corpora: (1) already-registered entries in the index, and (2) a curated high-value-brand deny list maintained by the registry operator. The deny list catches typosquats of well-known brands before those brands themselves are registered (e.g., ag00gle.comsubmission collides with the deny-list entry even if Google has not yet claimed an index row). Ambiguous submissions β mixed-script labels, homograph collisions, disallowed Unicode classes β SHOULD be rejected or flagged for human review rather than silently committed. - Ownership proof before commit. Save endpoints MUST require evidence of domain control before a new brand or property entry is committed to the index β this is the threat that motivates the control, since an attacker with a compromised member API key would otherwise be free to bulk-register fresh confusable variants that have no prior entry to conflict with. For revisions to an existing community-source entry by the same authenticated organization, re-proof SHOULD be required on a rolling basis (for example, once the prior proof is older than 90 days) but MAY be skipped within that window. Accepted proofs are either a DNS TXT record at
_adcp-owner.{domain}matching a server-issued nonce, or an HTTP challenge hosted at/.well-known/adcp-ownership.txton the domain. Nonces MUST be single-use, scoped to the(organization, domain)pair, and MUST expire within 15 minutes of issuance; verification MUST consume the nonce on success and invalidate it on failure. A leaked or unused nonce after expiry is dead. Revisions to an existing authoritative entry (i.e., one backed bybrand.json/adagents.json) continue to follow the 409 Conflict semantics in Save brand and Save property; ownership proof covers the community-source save path. - Per-organization rate limits on saves. In addition to the per-IP rate limits documented in Rate Limits, save endpoints SHOULD apply per-organization limits so that a single compromised API key cannot bulk-register confusable variants. The hosted implementation uses a burst-tolerant cap (indicative: tens of saves per hour per org, low-hundreds per day per org); callers exceeding the per-org bucket receive
429 Too Many Requests.
specs/registry-change-feed.md Β§Advisory identity material): the feed is change-detection, not a trust anchor, and the publisherβs own adagents.json pin remains the authoritative identity source (see adagents.json Β§signing_keys). Operators running an alternative registry implementation SHOULD apply equivalent save-time controls before accepting community-source writes.
Authentication
Public endpoints (resolution, discovery, search) require no authentication. Write endpoints accept either an organization API key (server-to-server) or a user JWT obtained via OAuth 2.1 (interactive / agent clients). Both are sent in theAuthorization: Bearer ... header.
Option A: Organization API key
Long-lived, org-scoped. Best for server-to-server integrations where no user is present.- Sign in at agenticadvertising.org/dashboard/api-keys
- Click Create key and copy the generated key
Authorization header:
Option B: User SSO via OAuth 2.1
Short-lived, user-scoped. Best for agent clients (MCP, AI assistants, custom apps) where a human is signing in to AAO. A single token works against both/mcp and the REST API.
Discovery follows RFC 8414 and RFC 9728:
- Authorization server metadata:
GET /.well-known/oauth-authorization-server - Protected-resource metadata (REST API):
GET /.well-known/oauth-protected-resource/api - Protected-resource metadata (MCP):
GET /.well-known/oauth-protected-resource/mcp
/register. Users authenticate via AuthKit; the token is a WorkOS-signed JWT.
403 if the authenticated user lacks the required standing.
Authenticated endpoints
These endpoints require a valid API key.Save brand
POST /api/brands/save
Save or update a community brand in the registry. For existing brands, creates a revision-tracked edit. Cannot edit authoritative brands managed via brand.json β those return 409 Conflict.
Request body:
domain and brand_name are required. brand_manifest (brand identity data) is optional. The brandβs source is set to "community" by the server. Domains are normalized (protocol stripped, lowercased).
Response (create)
Response (update)
Save property
POST /api/properties/save
Save or update a hosted property in the registry. For existing properties, creates a revision-tracked edit. Cannot edit authoritative properties managed via adagents.json β those return 409 Conflict.
Request body:
publisher_domain and authorized_agents (each with a required url and optional authorized_for) are required. properties (each requiring type and name) and contact are optional. Domains are normalized (protocol stripped, lowercased).
Response (create)
Response (update)
Change feed
GET /api/registry/feed
Poll a cursor-based feed of registry changes. Use this to keep a local copy of the registry in sync without re-fetching the full dataset. Events are ordered by UUID v7 event_id, providing monotonic cursor progression. The feed retains events for 90 days β expired cursors return 410 Gone.
Query parameters:
| Parameter | Type | Default | Description |
|---|---|---|---|
cursor | UUID | β | Resume after this event ID. Omit for the earliest available events. |
types | string | β | Comma-separated event type filters. Supports glob patterns (e.g. property.*). |
limit | number | 100 | Max events per page (1β10,000). |
| Type | Description |
|---|---|
property.created | A new property was added to the registry |
property.updated | Property metadata changed |
property.merged | Two property records were merged |
property.stale | Property failed re-crawl validation |
property.reactivated | A stale property passed re-crawl |
agent.discovered | A new agent_url appeared in a publisherβs adagents.json (authorization graph; does not imply the agent is in /api/registry/agents) |
agent.removed | An agent was removed from the registry |
agent.profile_updated | Agent inventory profile changed |
publisher.adagents_changed | A publisherβs adagents.json was updated |
authorization.granted | An agent was authorized for a property |
authorization.revoked | An authorization was removed |
Response
has_more is true, pass the returned cursor value in the next request to continue polling. When false, youβve reached the end of the current feed β poll again later with the same cursor to pick up new events.
If the cursor has expired (older than 90 days or not found), the response is 410 Gone:
410 Gone
Agent search
GET /api/registry/agents/search
Search agents by inventory profile β channels, markets, content categories, property types, and more. Filters use AND across dimensions and OR within a dimension. Results are ranked by a relevance score based on filter match breadth, inventory depth, and TMP support.
Query parameters:
| Parameter | Type | Default | Description |
|---|---|---|---|
channels | CSV | β | Filter by channel (e.g. ctv,olv,display) |
property_types | CSV | β | Filter by property type (e.g. ctv_app,website) |
markets | CSV | β | Filter by market/country code (e.g. US,GB) |
categories | CSV | β | Filter by IAB content category (e.g. IAB-7,IAB-7-1) |
tags | CSV | β | Filter by tag (e.g. premium,brand_safe) |
delivery_types | CSV | β | Filter by delivery type (e.g. guaranteed,programmatic) |
has_tmp | boolean | β | Require TMP support (true or false) |
min_properties | number | β | Minimum number of properties in inventory |
cursor | string | β | Pagination cursor from a previous response |
limit | number | 50 | Max results per page (1β200) |
Response
matched_filters array shows which filter dimensions matched, useful for understanding why a result was returned. The relevance_score combines filter match breadth, ln(property_count + 1) weighted at 0.1, and a 0.05 boost for TMP support.
Crawl request
POST /api/registry/crawl-request
Request an immediate re-crawl of a publisher domain. Use this after updating an adagents.json file so the registry picks up changes without waiting for the next scheduled crawl. The crawl runs asynchronously β the endpoint returns 202 Accepted immediately.
Rate-limited to one request per domain every 5 minutes and 30 requests per user per hour.
Request body:
domain is required. Domains are normalized (lowercased, trimmed). The endpoint validates the domain format and performs a DNS lookup to reject private/reserved IP addresses.
202 Accepted
429 Too Many Requests
retry_after is the number of seconds to wait before retrying.
Submit brand (legacy)
POST /api/brands/discovered/community
Submit a brand for review. This endpoint predates /api/brands/save β prefer the save endpoint for new integrations.
Error responses
| Status | Description |
|---|---|
| 400 | Missing required fields or invalid domain |
| 401 | Missing or invalid API key |
| 409 | Cannot edit an authoritative brand/property (managed via brand.json or adagents.json) |
| 410 | Cursor expired (change feed β older than 90-day retention window) |
| 429 | Rate limit exceeded |
Protocol vs REST API
The AdCP protocol defines MCP and A2A tasks for agent-to-agent communication (e.g.get_products, create_media_buy). The registry REST API is separate β it provides HTTP endpoints for looking up entities in the AgenticAdvertising.org registry.
Use the REST API for discovery and authorization:
- Resolve brand or property domains before making protocol calls
- Discover which agents exist and what theyβre authorized for
- Validate authorization in real time during ad serving
- Build integrations that browse or search the registry
- Fetching products from a sales agent (
get_products) - Creating media buys (
create_media_buy) - Building creatives (
build_creative) - Getting signals (
get_signals)