Authentication

All authenticated endpoints require a Bearer token in the Authorization header.

curl https://domani.run/api/domains/search?q=example.com \
  -H "Authorization: Bearer domani_sk_xxx"

Get an API key by creating an account or from the dashboard. Some endpoints (search, TLDs, WHOIS) are public but have lower rate limits without auth.

Token Scopes

Tokens can be restricted to specific permission scopes. Use * for full access (default). Create scoped tokens via POST /api/tokens with a scopes array.

ScopeGrants access to
domains:readGET /api/domains, GET /api/domains/{domain}, GET /api/domains/{domain}/dns, /status, /email/check, /auth-code, /transfer-away, /transfer-status, /analytics
domains:writePUT /api/domains/{domain}/dns, POST /connect, POST /verify, PUT /settings, PUT /parking, PUT/DELETE /api/domains/{domain}/for-sale, POST /api/domains/import, POST /import/verify
domains:transferPOST /api/domains/buy, POST /transfer, POST /renew (involves payment, includes marketplace purchases)
tokens:readGET /api/tokens
tokens:writePOST /api/tokens, DELETE /api/tokens/{id}
webhooks:readGET /api/webhooks, GET /api/webhooks/{id}/deliveries
webhooks:writePOST /api/webhooks, PATCH /api/webhooks/{id}, DELETE /api/webhooks/{id}
email:readGET /api/emails, /api/emails/{address}, /api/emails/{address}/messages, /api/domains/{domain}/email/status
email:writePOST /api/emails, POST /api/emails/{address}/send, POST /api/domains/{domain}/email/setup
account:readGET /api/me
account:writeDELETE /api/me, POST /api/billing/setup, POST /api/me/resend-verification
billing:readGET /api/billing/invoices
searchGET /api/domains/search, /suggest, /whois, /dns-check, GET /api/tlds
deals:readGET /api/deals, GET /api/deals/{id}
deals:writePOST /api/domains/sell, PATCH /api/deals/{id}
notifications:readGET /api/notifications, GET /api/notifications/count

Scope attenuation: a token can only create child tokens with equal or fewer scopes — never more.

Missing scope returns 403 with code INSUFFICIENT_SCOPE and a hint naming the required scope.

Errors

All errors return a consistent JSON envelope. Use code for programmatic handling and hint for recovery guidance.

{
  "error": "Missing API key",
  "code": "MISSING_API_KEY",
  "hint": "Include 'Authorization: Bearer domani_sk_xxx' header",
  "documentation_url": "https://domani.run/llms.txt"
}
CodeHTTP
MISSING_API_KEY401
INVALID_API_KEY401
EXPIRED_API_KEY401
MISSING_PARAMETER400
INVALID_DOMAIN400
PAYMENT_REQUIRED402
DOMAIN_UNAVAILABLE409
RATE_LIMIT_EXCEEDED429
USER_EXISTS409
NOT_FOUND404
UNKNOWN_PROVIDER400
UNKNOWN_METHOD400
TARGET_REQUIRED400
INVALID_PARAMETER400
CONTACT_REQUIRED400
INVALID_CONTACT400
INSUFFICIENT_SCOPE403
SCOPE_ESCALATION400
DOMAIN_EXISTS400
DOMAIN_NOT_ACTIVE400
UNSUPPORTED_TLD400
NOT_AVAILABLE400
ALREADY_REGISTERED400
TRANSFER_NOT_ELIGIBLE400
ALREADY_LISTED400
LISTING_SOLD400
SELF_PURCHASE400
DESCRIPTION_TOO_LONG400
DEAL_NOT_FOUND400
INVALID_DEAL_STATE400
EPP_ATTEMPTS_EXCEEDED400
ACTIVE_DEAL_EXISTS400

Rate Limits

Every response includes rate limit headers:

  • X-RateLimit-Limit — max requests per window
  • X-RateLimit-Remaining — requests left
  • X-RateLimit-Reset — unix timestamp when window resets

When rate limited (HTTP 429), check the Retry-After header for seconds to wait.

Authentication

POST/api/auth/registerPublic

Create a new account

Creates a new user account and returns an API key. Use this key as a Bearer token for all authenticated endpoints. Each account also receives a unique referral code. Pass a ref code to permanently link the new user to a referrer - the referrer earns a commission on all future domain purchases by this user.

Request body

email*string (email)
refstringReferral code - permanently links this user to the referrer. The referrer earns a commission on every domain purchase.

Response 201

api_keystringYour API key - save this securely
referral_codestringYour unique referral code for earning commissions
has_payment_methodbooleanWhether the user has a card on file (always false for new accounts)
userobject

Example response

{
  "api_key": "domani_sk_a1B2c3D4e5F6g7H8i9J0kLmN",
  "referral_code": "x7k9m2",
  "has_payment_method": false,
  "user": {
    "id": "cm5abc123",
    "email": "dev@example.com"
  }
}

Example

curl -X POST https://domani.run/api/auth/register \
  -H "Content-Type: application/json" \
  -d '{"email": "dev@example.com"}'
POST/api/auth/loginPublic

Send a magic link sign-in email

Sends a sign-in link to the provided email. If the user doesn't exist, creates an account first. The magic link redirects to the dashboard with auth credentials. For programmatic API access, use POST /api/auth/register to get an API key directly.

Request body

email*string (email)

Response 200

okboolean
messagestring

Example response

{
  "ok": true,
  "message": "Check your email for a sign-in link."
}

Example

curl -X POST https://domani.run/api/auth/login \
  -H "Content-Type: application/json" \
  -d '{"email": "dev@example.com"}'

Account

GET/api/me

Get current account details

Returns the authenticated user's account information including email, payment status, contact info status, referral code, and usage counts. Agents should call this after authentication to check has_payment_method and has_contact before attempting purchases.

Response 200

idstring
emailstring (email)
has_payment_methodbooleanWhether the user has a card on file (required before buying domains)
has_contactbooleanWhether the user has set WHOIS contact info (required before purchasing domains)
contactRegistrantContactRegistrant contact info (null if not set)
referral_codestringYour unique referral code
referral_rateintegerCommission percentage on referred purchases
domain_countintegerNumber of domains owned
token_countintegerNumber of active API tokens
created_atstring (date-time)

Example response

{
  "id": "cm5abc123",
  "email": "dev@example.com",
  "has_payment_method": true,
  "has_contact": true,
  "referral_code": "x7k9m2",
  "referral_rate": 20,
  "domain_count": 3,
  "token_count": 2,
  "created_at": "2025-01-15T10:30:00.000Z"
}

Example

curl https://domani.run/api/me \
  -H "Authorization: Bearer domani_sk_xxx"
PUT/api/me

Update account preferences

Update the authenticated user's account preferences such as preferred payment method.

Request body

preferred_payment_method*string | null(card | usdc)Default payment method for purchases (card, usdc, or null for auto-select)

Response 200

preferred_payment_methodstring | null
hintstring
DELETE/api/me

Delete account and all associated data

Permanently delete the authenticated user's account, all domains, tokens, webhooks, and other associated data. This action cannot be undone.

Response 200

deletedboolean
hintstring
GET/api/me/contact

Get registrant contact info

Returns the authenticated user's WHOIS registrant contact information. Must be set before purchasing or transferring domains.

Response 200

has_contactboolean
contactRegistrantContact

Example response

{
  "has_contact": true,
  "contact": {
    "first_name": "Jane",
    "last_name": "Doe",
    "address1": "123 Main St",
    "city": "New York",
    "state": "NY",
    "postal_code": "10001",
    "country": "US",
    "phone": "+1.2125551234",
    "email": "jane@example.com"
  }
}

Example

curl https://domani.run/api/me/contact \
  -H "Authorization: Bearer domani_sk_xxx"
PUT/api/me/contact

Set registrant contact info

Set or update your WHOIS registrant contact information. Required by ICANN before purchasing domains. Used as the domain registrant and billing contact. When updated, the new contact is automatically propagated to all active domains at registrars that support contact updates.

Response 200

contactRegistrantContact

Example

curl -X PUT https://domani.run/api/me/contact \
  -H "Authorization: Bearer domani_sk_xxx" \
  -H "Content-Type: application/json" \
  -d '{"first_name":"Jane","last_name":"Doe","address1":"123 Main St","city":"New York","state":"NY","postal_code":"10001","country":"US","phone":"+1.2125551234","email":"jane@example.com"}'
POST/api/me/resend-verification

Resend contact email verification

Resend the verification email for the account's contact email address. Rate limited to once per 15 minutes per email.

Request body

emailstring (email)Email address to verify (defaults to account contact email)

Response 200

sentboolean
already_verifiedboolean
hintstring

TLDs

GET/api/tldsPublic

List all available TLDs with pricing

Returns all 1014+ available TLDs with registration, renewal, and transfer pricing. Supports filtering by price range, searching by TLD name, sorting, and pagination via limit/offset. When paginated, includes a Link header with the next page URL. Public endpoint - no authentication required. Authenticated requests get higher rate limits (60/min vs 30/min).

Parameters

max_pricenumberMaximum registration price in USD
min_pricenumberMinimum registration price in USD
sortstring(tld | price | renewal)Sort field - 'tld' (alphabetical), 'price' (registration), or 'renewal'
orderstring(asc | desc)Sort order
searchstringFilter TLD names containing this string
limitintegerMax results per page
offsetintegerNumber of results to skip

Response 200

tldsTldInfo[]
totalintegerTotal matching TLDs (before pagination)
offsetinteger
limitinteger | null

Example response

{
  "tlds": [
    {
      "tld": "xyz",
      "registration": 4.04,
      "renewal": 4.04,
      "transfer": 4.04
    },
    {
      "tld": "com",
      "registration": 10.88,
      "renewal": 13.08,
      "transfer": 13.08
    },
    {
      "tld": "dev",
      "registration": 15.85,
      "renewal": 15.85,
      "transfer": 15.85
    }
  ],
  "total": 1014,
  "offset": 0,
  "limit": null
}

Example

curl https://domani.run/api/tlds?max_price=10&sort=price

Domains

GET/api/domains

List all your registered domains

Returns all domains owned by the authenticated user, ordered by purchase date (newest first). Includes domain name, status, purchase date, and expiration date.

Response 200

domainsDomainRecord[]

Example response

{
  "domains": [
    {
      "domain": "mysite.com",
      "status": "active",
      "purchasedAt": "2025-01-15T10:30:00.000Z",
      "expiresAt": "2026-01-15T10:30:00.000Z",
      "emailEnabled": true,
      "emailVerified": true,
      "emailVerifiedAt": "2025-01-15T10:35:00.000Z",
      "mailboxCount": 2
    },
    {
      "domain": "myapp.dev",
      "status": "active",
      "purchasedAt": "2025-02-01T14:00:00.000Z",
      "expiresAt": "2026-02-01T14:00:00.000Z",
      "emailEnabled": false,
      "emailVerified": false,
      "emailVerifiedAt": null,
      "mailboxCount": 0
    }
  ]
}

Example

curl https://domani.run/api/domains \
  -H "Authorization: Bearer domani_sk_xxx"
POST/api/domains/buy

Purchase one or more domains

Purchases one or more domains and registers them immediately. Send {"domain": "..."} for single or {"domains": ["...", "..."]} for bulk (max 10). If the domain is listed for sale on the Domani marketplace, this endpoint automatically routes to a marketplace purchase instead of a fresh registration. The response includes years: 0 and marketplace_purchase: true to distinguish it from a registration. Marketplace purchases are single-domain only. The buyer pays the full listed price; the seller receives the price minus a platform commission (default 10%). Bulk purchases process each domain independently - failures don't block other purchases. The response includes results (succeeded), errors (failed), and summary with counts. Accepts three payment methods: 1. Card (default): Requires a card on file via POST /api/billing/setup. Charged per domain. 2. USDC (two-step): First call without payment to get a 402 with payment_options.crypto (wallet address, amount, chains). Send USDC on-chain, then retry with payment_tx and payment_chain in the request body. 3. x402 USDC (atomic): Include a PAYMENT-SIGNATURE header with a signed USDC payment on Base. Bulk purchases require a card on file (USDC not supported for multi-domain). If no payment is provided for single domain, returns HTTP 402 with payment instructions.

Request body

domainstringSingle domain to purchase (e.g. mysite.com)
domainsstring[]Array of domains to purchase (max 10, requires card on file)
payment_methodstring(card | usdc)Payment method: 'card' (default) or 'usdc'. When 'usdc', first call returns 402 with payment instructions.
payment_txstringTransaction hash of a USDC payment already sent on-chain. Required on the second call of the two-step USDC flow.
payment_chainstring(base | ethereum)Chain the USDC payment was sent on (e.g. 'base' or 'ethereum').
yearsintegerNumber of years to register (1-10, default 1). Price is multiplied by years.
refstringReferral code (optional) - the referrer earns a commission

Response 201

domainstring
statusstring
expiresstring (date-time)
pricenumber
currencystring
yearsintegerNumber of years registered
payment_methodstringHow the domain was paid for
payment_txstringOn-chain transaction hash (present for USDC payments)
marketplace_purchasebooleanTrue when the domain was bought from the Domani marketplace (not a fresh registration)

Example

curl -X POST https://domani.run/api/domains/buy \
  -H "Authorization: Bearer domani_sk_xxx" \
  -H "Content-Type: application/json" \
  -d '{"domain": "mysite.com"}'
GET/api/domains/{domain}

Get detailed information about a domain you own

Returns comprehensive information about a domain: status, purchase date, expiry, payment method, security lock, WHOIS privacy, auto-renew status, and registration dates. The registrar object may be null if extended data is unavailable.

Parameters

domain*stringDomain name you own

Response 200

domainstring
statusstring
auto_renewbooleanWhether auto-renew is enabled
purchased_atstring (date-time)
expires_atstring (date-time)
days_until_expiryintegerDays until domain expiry
payment_methodstringHow the domain was paid for
payment_txstring | nullOn-chain tx hash (USDC payments)
registrarobject | nullDomain registration details (null if unavailable)

Example response

{
  "domain": "mysite.com",
  "status": "active",
  "auto_renew": true,
  "purchased_at": "2025-06-01T12:00:00.000Z",
  "expires_at": "2026-06-01T12:00:00.000Z",
  "days_until_expiry": 365,
  "payment_method": "card",
  "registrar": {
    "security_lock": true,
    "whois_privacy": true,
    "auto_renew": true,
    "create_date": "2025-06-01T12:00:00Z",
    "expire_date": "2026-06-01T12:00:00Z"
  }
}

Example

curl https://domani.run/api/domains/mysite.com \
  -H "Authorization: Bearer domani_sk_xxx"
PUT/api/domains/{domain}/settings

Update domain settings

Update settings for a domain you own. Supports auto_renew, whois_privacy, and security_lock. At least one field must be provided.

Parameters

domain*stringDomain name you own

Request body

auto_renewbooleanWhether to enable auto-renew for this domain
whois_privacybooleanWhether to enable WHOIS privacy protection
security_lockbooleanWhether to lock the domain against transfers

Response 200

domainstring
auto_renewboolean
whois_privacyboolean
security_lockboolean
hintstring

Example

curl -X PUT https://domani.run/api/domains/mysite.com/settings \
  -H "Authorization: Bearer domani_sk_xxx" \
  -H "Content-Type: application/json" \
  -d '{"auto_renew": false}'
PUT/api/domains/{domain}/parking

Update parking settings

Update parking and listing settings for a domain you own. Enable/disable the parking page and set a 'For Sale' listing price. When parking is enabled, visitors to the domain see a branded page. When a listing price is set, visitors see a 'For Sale' page with a contact form.

Parameters

domain*stringDomain name

Request body

enabledbooleanWhether to enable the parking page
listing_pricenumber | nullSale price in USD, or null to remove listing

Response 200

domainstring
parking_enabledboolean
listing_pricenumber | null
hintstring
GET/api/domains/{domain}/analytics

Get parking analytics

Get visitor analytics for a parked domain - page views, inquiries, conversion rate, 30-day daily breakdown, and recent inquiries.

Parameters

domain*stringDomain name

Response 200

domainstring
views_7dintegerTotal page views in the last 7 days
views_30dintegerTotal page views in the last 30 days
inquiries_30dintegerTotal inquiries in the last 30 days
conversion_ratenumberInquiry-to-view ratio as percentage
daily_viewsobject[]Daily view counts for the last 30 days
recent_inquiriesobject[]5 most recent inquiries
hintstring

Example response

{
  "domain": "example.com",
  "views_7d": 42,
  "views_30d": 156,
  "inquiries_30d": 3,
  "conversion_rate": 1.92,
  "daily_views": [
    {
      "date": "2026-02-25",
      "views": 7
    }
  ],
  "recent_inquiries": [
    {
      "email": "buyer@example.com",
      "offer": 500,
      "date": "2026-02-25T10:00:00Z"
    }
  ],
  "hint": "example.com received 156 views and 3 inquiries in the last 30 days (1.92% conversion)."
}
GET/api/domains/{domain}/auth-code

Get EPP auth code

Get the EPP/auth code needed to transfer a domain to another registrar. Automatically unlocks the domain if it's locked.

Parameters

domain*stringDomain name you own

Response 200

domainstring
auth_codestringEPP auth code to give to the new registrar
was_unlockedbooleanWhether the domain was auto-unlocked
hintstring
next_stepsstring[]
GET/api/domains/{domain}/transfer-away

Get outbound transfer status

Check the status of an outbound domain transfer. Use after getting an auth code and initiating the transfer at the new registrar.

Parameters

domain*stringDomain name you own

Response 200

domainstring
statusstring
gaining_registrarstring
request_datestring (date-time)
hintstring
GET/api/domains/{domain}/transfer-status

Check inbound transfer status

Check the status of an inbound domain transfer. Returns detailed status with actionable hints. Use after initiating a transfer with POST /api/domains/transfer.

Parameters

domain*stringDomain name you own

Response 200

domainstring
statusstring
timestampstringTimestamp of last status update
hintstringActionable guidance based on the current status
POST/api/domains/import

Import an external domain

Import a domain you own at another registrar (GoDaddy, Namecheap, Cloudflare, etc.) to manage through domani.run. Free, no transfer needed. Returns a TXT record for DNS-based ownership verification.

Request body

domain*stringDomain to import

Response 200

domainstring
statusstring
tokenstring
txt_recordobject
hintstring
POST/api/domains/import/verify

Verify and complete domain import

Verify DNS TXT record and complete domain import. Call after adding the TXT record returned by POST /api/domains/import. Uses Cloudflare DNS-over-HTTPS to check propagation.

Request body

domain*stringDomain to verify

Response 201

domainstring
statusstring
expiresstring (date-time)
registrarstring
payment_methodstring
hintstring
GET/api/domains/transfer-check

Pre-check transfer eligibility

Check whether a domain is eligible for transfer before initiating. Returns transfer price, eligibility status, and any blockers (unsupported TLD, ICANN waiting period, domain locked). Call this before POST /api/domains/transfer.

Parameters

domain*stringDomain to check, e.g. mysite.com

Response 200

domainstring
tldstring
eligibleboolean
pricenumber | nullTransfer price in USD (null if TLD unsupported)
currencystring
reasonstringHuman-readable reason if not eligible
codestring
eligible_atstring (date)ISO date when domain becomes eligible (ICANN waiting period)
hintstring
POST/api/domains/transfer-watch

Watch a domain for transfer eligibility

Watch a domain and get notified (email + transfer.eligible webhook) when it becomes eligible for transfer. Uses RDAP to check EPP status codes and ICANN 60-day lock windows. If already eligible, returns immediately without creating a watch.

Request body

domain*stringDomain to watch, e.g. mysite.com

Response 200

domainstring
eligiblebooleanWhether the domain is currently eligible for transfer
eligible_atstring (date)ISO date when domain becomes eligible (if not yet eligible)
hintstringActionable guidance
watchingbooleanWhether a notification has been set up
POST/api/domains/transfer

Transfer domain from another provider

Initiate a domain transfer-in. Requires the authorization/EPP code from the current provider. Transfer typically takes 1-5 days to complete. Payment is charged upfront via card or USDC (two-step: get 402, send USDC, retry with payment_tx + payment_chain).

Request body

domain*stringDomain to transfer, e.g. mysite.com
auth_code*stringAuthorization/EPP code from the current provider
payment_methodstring(card | usdc)Payment method: 'card' (default) or 'usdc'.
payment_txstringTransaction hash of a USDC payment already sent on-chain.
payment_chainstring(base | ethereum)Chain the USDC payment was sent on.
extra_subdomainsstring[]Additional subdomains to include in the pre-transfer DNS snapshot. We auto-discover subdomains via CT logs, SPF, MX/DKIM inference, and a common wordlist - use this for any custom subdomains we might miss.

Response 202

domainstring
statusstring
expiresstring (date-time)
pricenumber
currencystring
payment_methodstring
payment_txstring
auto_renewboolean
hintstring
dns_snapshotobjectSummary of DNS records captured before the transfer. Records are auto-restored when the transfer completes.

Example

curl -X POST https://domani.run/api/domains/transfer \
  -H "Authorization: Bearer domani_sk_xxx" \
  -H "Content-Type: application/json" \
  -d '{"domain": "mysite.com", "auth_code": "EPP-AUTH-CODE"}'
POST/api/domains/{domain}/renew

Renew a domain

Renew a domain you own for additional years (1-10). Payment is charged upfront. The new expiry date is returned in the response.

Parameters

domain*stringDomain name you own

Request body

yearsintegerNumber of years to renew (default 1)
payment_methodstring(card | usdc)Payment method: 'card' (default) or 'usdc'.
payment_txstringTransaction hash of a USDC payment already sent on-chain.
payment_chainstring(base | ethereum)Chain the USDC payment was sent on.

Response 200

domainstring
renewed_yearsinteger
new_expirystring (date-time)
pricenumber
currencystring
payment_methodstring
payment_txstring

Example

curl -X POST https://domani.run/api/domains/mysite.com/renew \
  -H "Authorization: Bearer domani_sk_xxx" \
  -H "Content-Type: application/json" \
  -d '{"years": 1}'

DNS

GET/api/domains/{domain}/dns

Get DNS records for a domain you own

Returns all DNS records configured for a domain you own. Includes A, AAAA, CNAME, MX, TXT, NS, and other record types.

Parameters

domain*stringDomain name you own

Response 200

domainstring
recordsDnsRecord[]

Example response

{
  "domain": "mysite.com",
  "records": [
    {
      "type": "A",
      "name": "@",
      "value": "76.76.21.21",
      "ttl": 3600
    },
    {
      "type": "CNAME",
      "name": "www",
      "value": "mysite.com",
      "ttl": 3600
    },
    {
      "type": "MX",
      "name": "@",
      "value": "mail.example.com",
      "ttl": 3600,
      "priority": 10
    }
  ]
}

Example

curl https://domani.run/api/domains/mysite.com/dns \
  -H "Authorization: Bearer domani_sk_xxx"
PUT/api/domains/{domain}/dns

Set DNS records for a domain you own

Replaces all DNS records for a domain you own. Send the complete set of records - any existing records not included will be removed.

Parameters

domain*stringDomain name you own

Request body

records*DnsRecord[]

Response 200

successboolean
domainstring
recordsDnsRecord[]

Example response

{
  "success": true,
  "domain": "mysite.com",
  "records": [
    {
      "type": "A",
      "name": "@",
      "value": "76.76.21.21",
      "ttl": 3600
    },
    {
      "type": "CNAME",
      "name": "www",
      "value": "cname.vercel-dns.com",
      "ttl": 3600
    }
  ]
}

Example

curl -X PUT https://domani.run/api/domains/mysite.com/dns \
  -H "Authorization: Bearer domani_sk_xxx" \
  -H "Content-Type: application/json" \
  -d '{"records": [
    {"type": "A", "name": "@", "value": "76.76.21.21", "ttl": 3600},
    {"type": "CNAME", "name": "www", "value": "cname.vercel-dns.com", "ttl": 3600}
  ]}'
POST/api/domains/{domain}/dns/snapshot

Capture a DNS snapshot

Captures all DNS records for a domain via public DNS lookups (Cloudflare DoH). Discovers subdomains from Certificate Transparency logs, SPF records, and common names. The snapshot is also stored server-side as a backup.

Parameters

domain*stringDomain name you own

Request body

extra_subdomainsstring[]Additional subdomains to include in the snapshot

Response 200

domainstring
recordsDnsRecord[]
subdomainsstring[]
sourcesstring[]
capturedAtstring (date-time)

Example response

{
  "domain": "mysite.com",
  "records": [
    {
      "type": "A",
      "name": "@",
      "value": "76.76.21.21",
      "ttl": 3600
    },
    {
      "type": "CNAME",
      "name": "www",
      "value": "mysite.com",
      "ttl": 3600
    },
    {
      "type": "MX",
      "name": "@",
      "value": "aspmx.l.google.com",
      "ttl": 3600,
      "priority": 1
    }
  ],
  "subdomains": [
    "www",
    "mail",
    "api"
  ],
  "sources": [
    "ct",
    "wordlist"
  ],
  "capturedAt": "2026-03-06T12:00:00.000Z"
}

Example

curl -X POST https://domani.run/api/domains/mysite.com/dns/snapshot \
  -H "Authorization: Bearer domani_sk_xxx"
POST/api/domains/{domain}/dns/restore

Restore DNS from a snapshot

Applies DNS records from a snapshot to the domain's current registrar. Uses a diff algorithm to avoid duplicating records that already exist. If no snapshot is provided in the body, restores from the server-side backup (created by snapshot or transfer).

Parameters

domain*stringDomain name you own

Request body

snapshotobjectA DNS snapshot object (as returned by the snapshot endpoint). If omitted, the stored server-side backup is used.

Response 200

domainstring
appliedintegerNumber of records created
skippedintegerNumber of records already present
recordCountintegerTotal records in the snapshot
errorsstring[]

Example response

{
  "domain": "mysite.com",
  "applied": 5,
  "skipped": 2,
  "recordCount": 7,
  "errors": []
}

Example

curl -X POST https://domani.run/api/domains/mysite.com/dns/restore \
  -H "Authorization: Bearer domani_sk_xxx"
GET/api/domains/{domain}/nameservers

Get nameservers for a domain

Returns the authoritative nameservers configured for a domain you own. If empty, DNS operations (parking, email, connect) will fail until nameservers are assigned.

Parameters

domain*stringDomain name you own

Response 200

domainstring
nameserversstring[]
default_nameserversstring[]Registrar default nameservers (null if unknown)
hintstring

Example response

{
  "domain": "mysite.com",
  "nameservers": [
    "ns1.systemdns.com",
    "ns2.systemdns.com",
    "ns3.systemdns.com"
  ],
  "default_nameservers": [
    "ns1.systemdns.com",
    "ns2.systemdns.com",
    "ns3.systemdns.com"
  ],
  "hint": "3 nameservers configured for mysite.com."
}

Example

curl https://domani.run/api/domains/mysite.com/nameservers \
  -H "Authorization: Bearer domani_sk_xxx"
PUT/api/domains/{domain}/nameservers

Set nameservers for a domain

Replace the nameservers for a domain you own. Requires 2–13 valid hostnames. Common values: OpenSRS DNS (ns1.systemdns.com, ns2.systemdns.com, ns3.systemdns.com).

Parameters

domain*stringDomain name you own

Request body

nameservers*string[]Nameserver hostnames (2–13 required)

Response 200

successboolean
domainstring
nameserversstring[]
hintstring

Example response

{
  "success": true,
  "domain": "mysite.com",
  "nameservers": [
    "ns1.systemdns.com",
    "ns2.systemdns.com",
    "ns3.systemdns.com"
  ],
  "hint": "Nameservers for mysite.com updated to ns1.systemdns.com, ns2.systemdns.com, ns3.systemdns.com. Changes may take up to 48 hours to propagate."
}

Example

curl -X PUT https://domani.run/api/domains/mysite.com/nameservers \
  -H "Authorization: Bearer domani_sk_xxx" \
  -H "Content-Type: application/json" \
  -d '{"nameservers": ["ns1.systemdns.com", "ns2.systemdns.com", "ns3.systemdns.com"]}'

Connect

POST/api/domains/{domain}/connect

Connect a domain to a hosting or email provider

Connects a domain to a hosting or email provider by auto-detecting the provider from a target URL or accepting an explicit provider name. For domains managed through the platform, DNS records are set automatically (status: dns_set). For imported domains (external registrar), returns the records as instructions to add manually at your registrar (status: manual_setup_required, records have status pending). Supported hosting providers: vercel, netlify, cloudflare-pages, github-pages, railway, fly. Supported email providers: google-workspace, fastmail, proton. Use GET /api/domains/{domain}/connect to list all available providers and methods.

Parameters

domain*stringDomain name you own

Request body

targetstringTarget URL for auto-detection (e.g. my-app.vercel.app)
providerstringExplicit provider name (e.g. vercel, google-workspace)
methodstringConnection method if provider has multiple (e.g. cname-only)

Response 200

domainstring
providerstring
categorystring
methodstring
recordsDnsRecord[]
docsstring (uri)
statusstringdns_set = records written automatically. manual_setup_required = imported domain, add records at your registrar.
hintstring
next_stepsstring[]Provider-specific actions to complete after DNS records are set

Example

curl -X POST https://domani.run/api/domains/mysite.com/connect \
  -H "Authorization: Bearer domani_sk_xxx" \
  -H "Content-Type: application/json" \
  -d '{"target": "my-app.vercel.app"}'
GET/api/domains/{domain}/connect

List available providers for connecting a domain

Returns all supported hosting and email providers with their connection methods. Use this to discover what providers are available before calling POST.

Parameters

domain*stringDomain name you own

Response 200

domainstring
providersobject
POST/api/domains/{domain}/verify

Verify that a provider connection is working

Runs the provider's verification check to confirm DNS records have propagated and the connection is active. Use after POST /api/domains/{domain}/connect to verify.

Parameters

domain*stringDomain name you own

Request body

targetstringTarget for auto-detection
providerstringProvider name
methodstringMethod name

Response 200

domainstring
providerstring
methodstring
verifiedboolean
messagestring
hintstring
next_stepsstring[]Provider-specific troubleshooting steps (only when verified is false)

Example

curl -X POST https://domani.run/api/domains/mysite.com/verify \
  -H "Authorization: Bearer domani_sk_xxx" \
  -H "Content-Type: application/json" \
  -d '{"target": "my-app.vercel.app"}'

Status

GET/api/domains/{domain}/status

Check domain health: DNS, SSL, email, expiry

Checks the health of a domain via DNS-over-HTTPS: A/AAAA/CNAME/MX/TXT/NS records, SSL certificate status, email configuration, and expiry date. Supports SSE streaming via Accept: text/event-stream - closing the connection cancels in-flight DNS/SSL checks server-side. Use this after connecting to verify DNS propagation.

Parameters

domain*stringDomain name you own

Response 200

domainstring
statusstring
expiresstring (date-time)
days_until_expiryinteger
dnsobject
sslobject
emailobject

Example response

{
  "domain": "mysite.com",
  "status": "active",
  "expires": "2027-02-18T00:00:00.000Z",
  "days_until_expiry": 365,
  "dns": {
    "propagated": true,
    "a": [
      "76.76.21.21"
    ],
    "aaaa": [],
    "cname": [],
    "mx": [],
    "txt": [],
    "ns": [
      "ns1.porkbun.com",
      "ns2.porkbun.com"
    ],
    "registrar_records": [
      {
        "type": "A",
        "name": "@",
        "value": "76.76.21.21",
        "ttl": 3600
      }
    ]
  },
  "ssl": {
    "active": true
  },
  "email": {
    "configured": false,
    "mx": []
  }
}

Example

curl https://domani.run/api/domains/mysite.com/status \
  -H "Authorization: Bearer domani_sk_xxx"

Email

GET/api/domains/{domain}/email/check

Check email DNS health (MX, SPF, DMARC, DKIM)

Checks email DNS configuration for a domain: MX records, SPF, DMARC, and DKIM. Auto-detects the email provider from MX records (Google Workspace, Proton, Fastmail).

Parameters

domain*stringDomain name you own

Response 200

domainstring
providerstring | nullAuto-detected email provider (google-workspace, proton, fastmail) or null
mxobject
spfobject
dmarcobject
dkimobject

Example response

{
  "domain": "mysite.com",
  "provider": "google-workspace",
  "mx": {
    "configured": true,
    "records": [
      "1 aspmx.l.google.com",
      "5 alt1.aspmx.l.google.com"
    ]
  },
  "spf": {
    "configured": true,
    "value": "v=spf1 include:_spf.google.com ~all"
  },
  "dmarc": {
    "configured": true,
    "value": "v=DMARC1; p=quarantine; rua=mailto:dmarc@mysite.com"
  },
  "dkim": {
    "configured": true,
    "selectors": [
      "google"
    ]
  }
}

Example

curl https://domani.run/api/domains/mysite.com/email/check \
  -H "Authorization: Bearer domani_sk_xxx"
POST/api/domains/{domain}/email/setup

Set up email on a domain

Pre-configure email DNS on a domain (optional). This is called automatically by create_mailbox when needed. Pass force=true to override existing email provider (Google Workspace, Fastmail, Proton).

Parameters

domain*stringDomain name

Request body

forcebooleanOverride existing email provider (Google Workspace, Fastmail, Proton). Default: false

Response 200

domainstring
statusstring
verified_atstring | null
recordsDnsRecord[]
hintstring
DELETE/api/domains/{domain}/email/setup

Remove email from a domain

Remove email from a domain, delete all mailboxes and messages, and set emailEnabled to false.

Parameters

domain*stringDomain name
GET/api/domains/{domain}/email/status

Check email DNS status

Check email DNS verification status for a domain. Returns whether the domain is verified and ready, DNS record details, and mailbox count.

Parameters

domain*stringDomain name

Response 200

domainstring
verifiedboolean
recordsDnsRecord[]
mailbox_countinteger
POST/api/email

Create a mailbox on domani.run

Create a free email address on domani.run (e.g. hello@domani.run). No domain ownership required. Limited to 1 free @domani.run mailbox per user. Many common slugs are reserved.

Request body

slug*stringLocal part of the email address (e.g. 'hello' for hello@domani.run)
namestringDisplay name (defaults to capitalized slug)

Response 200

idstring
addressstring
slugstring
domainstring
GET/api/email

List all mailboxes

List all email addresses (mailboxes) owned by the user across all domains.

Parameters

domainstringFilter by domain

Response 200

mailboxesobject[]
POST/api/emails

Create a mailbox

Create an email address (mailbox). Provide a full address like "hello@mysite.com". Email DNS is auto-configured on first use. Pass force=true to override existing email provider. Max 5 mailboxes per user. For free @domani.run mailboxes use POST /api/email.

Request body

address*stringFull email address, e.g. hello@mysite.com
namestringDisplay name for outbound emails (e.g. "John Doe")
forcebooleanOverride existing MX records. Default: false

Response 200

idstring
addressstring
namestring | null
domainstring
slugstring
webhook_urlstring | null
forward_tostring | null
signing_secretstringHMAC secret for verifying inbound webhook payloads (shown once at creation)
GET/api/emails

List all mailboxes

List all email addresses (mailboxes) owned by the user across all domains. Same as GET /api/email.

Parameters

domainstringFilter by domain

Response 200

mailboxesobject[]
GET/api/emails/{address}

Get mailbox details

Get details for a specific mailbox by its full email address (URL-encoded, e.g. hello%40mysite.com).

Parameters

address*stringFull email address (URL-encoded, e.g. hello%40mysite.com)

Response 200

idstring
addressstring
slugstring
domainstring
webhook_urlstring | null
forward_tostring | null
signing_secretstringHMAC secret for verifying inbound webhook payloads
message_countinteger
created_atstring (date-time)
PATCH/api/emails/{address}

Update mailbox

Update a mailbox's display name, webhook URL, or forward-to email address by its full email address. Set to null to disable/clear.

Parameters

address*stringFull email address (URL-encoded)

Request body

namestring | nullDisplay name for the mailbox. Set to null to clear.
webhook_urlstring | nullHTTPS URL for inbound email webhook, or null to clear
forward_tostring | nullEmail address to forward inbound emails to, or null to clear

Response 200

idstring
addressstring
slugstring
domainstring
namestring | null
webhook_urlstring | null
forward_tostring | null
signing_secretstringHMAC secret for verifying inbound webhook payloads
DELETE/api/emails/{address}

Delete a mailbox

Delete a mailbox by its full email address. Requires confirm=true in body. Without confirm, returns a preview with message_count.

Parameters

address*stringFull email address (URL-encoded)

Request body

confirmbooleanMust be true to actually delete. Omit to preview.
POST/api/emails/{address}/send

Send an email

Send an email from a mailbox identified by its full email address. Rate limit: 100 emails per hour per mailbox.

Parameters

address*stringSender email address (URL-encoded)

Request body

to*stringRecipient email address(es)
ccanyCC recipient(s)
bccanyBCC recipient(s)
subjectstring
textstringPlain text body
htmlstringHTML body
reply_tostringReply-To address
in_reply_tostringMessage-ID of the email being replied to (for threading)
referencesstringSpace-separated Message-ID chain (for threading)
attachmentsobject[]File attachments (max 10, max 40MB total). Content must be base64-encoded.

Response 200

idstring
message_idstringRFC 5322 Message-ID
fromstring
tostring
statusstring
attachments_countinteger
GET/api/emails/{address}/messages

List emails

List emails sent and received by a mailbox identified by its full email address. Supports cursor-based pagination and direction filtering.

Parameters

address*stringMailbox email address (URL-encoded)
cursorstringPagination cursor from previous response
limitintegerMax messages to return
directionstring(in | out | all)Filter by direction (default: all)
sincestringISO timestamp - return only messages created after this date
fromstringFilter by sender address
tostringFilter by recipient address
subjectstringFilter by subject

Response 200

messagesobject[]
next_cursorstring | null
has_moreboolean
PATCH/api/emails/{address}/messages/read

Mark messages as read or unread

Mark one or more messages as read or unread in a mailbox identified by its full email address.

Parameters

address*stringMailbox email address (URL-encoded)

Request body

message_ids*string[]IDs of messages to mark
read*booleantrue to mark as read, false to mark as unread

Response 200

countinteger
POST/api/emails/{address}/messages/delete

Bulk delete messages

Soft-delete multiple messages at once in a mailbox identified by its full email address.

Parameters

address*stringMailbox email address (URL-encoded)

Request body

message_ids*string[]IDs of messages to delete

Response 200

countinteger
GET/api/emails/{address}/messages/{id}

Get a message

Fetch a single message by ID from a mailbox identified by its full email address.

Parameters

address*stringMailbox email address (URL-encoded)
id*stringMessage ID

Response 200

idstring
message_idstring | null
directionstring
fromstring
tostring
ccstring | null
reply_tostring | null
subjectstring
textstring | null
htmlstring | null
in_reply_tostring | null
referencesstring | null
attachmentsobject[]
statusstring
is_readboolean
read_atstring | null
eventsobject[]
created_atstring (date-time)
DELETE/api/emails/{address}/messages/{id}

Delete a message

Soft-delete a message from a mailbox identified by its full email address.

Parameters

address*stringMailbox email address (URL-encoded)
id*stringMessage ID

Response 200

idstring
deletedboolean
POST/api/emails/{address}/messages/{id}/reply

Reply to a message

Reply to a message in a mailbox identified by its full email address. Threading headers are auto-populated.

Parameters

address*stringMailbox email address (URL-encoded)
id*stringMessage ID to reply to

Request body

textstringPlain text reply body
htmlstringHTML reply body
ccanyAdditional CC recipient(s)
allbooleanReply-all: auto-populate CC from original To/CC

Response 200

idstring
message_idstring
fromstring
tostring
statusstring
POST/api/emails/{address}/messages/{id}/forward

Forward a message

Forward a message from a mailbox identified by its full email address to another recipient.

Parameters

address*stringMailbox email address (URL-encoded)
id*stringMessage ID to forward

Request body

to*stringRecipient email address
textstringOptional note to prepend

Response 200

idstring
message_idstring
fromstring
tostring
statusstring
GET/api/emails/{address}/avatar

Get mailbox avatar info

Returns the Gravatar URL (if set) and signup link for a mailbox identified by its full email address.

Parameters

address*stringMailbox email address (URL-encoded)

Response 200

addressstring
gravatar_urlstring | null
gravatar_signupstring
has_avatarboolean
hintstring
GET/api/emails/{address}/webhook

Get mailbox webhook config

Returns the webhook URL, signing secret, and status for a mailbox.

Parameters

address*stringMailbox email address (URL-encoded)

Response 200

addressstring
webhook_urlstring | null
signing_secretstring | null
has_webhookboolean
hintstring
PUT/api/emails/{address}/webhook

Set mailbox webhook

Set or update the inbound webhook URL for a mailbox. Incoming emails will be POSTed to this URL with HMAC-SHA256 signing.

Parameters

address*stringMailbox email address (URL-encoded)

Request body

url*stringHTTPS webhook URL

Response 200

addressstring
webhook_urlstring
signing_secretstring
has_webhookboolean
hintstring
DELETE/api/emails/{address}/webhook

Remove mailbox webhook

Remove the webhook URL and signing secret from a mailbox. Inbound emails will only be stored.

Parameters

address*stringMailbox email address (URL-encoded)

Response 200

addressstring
webhook_urlstring | null
signing_secretstring | null
has_webhookboolean
hintstring
POST/api/emails/{address}/webhook/rotate

Rotate webhook signing secret

Regenerate the HMAC-SHA256 signing secret for a mailbox webhook. The old secret is immediately invalidated.

Parameters

address*stringMailbox email address (URL-encoded)

Response 200

addressstring
webhook_urlstring
signing_secretstring
has_webhookboolean
hintstring
POST/api/emails/{address}/webhook/test

Test mailbox webhook

Send a signed test payload to the mailbox webhook URL. Returns HTTP status and whether delivery succeeded.

Parameters

address*stringMailbox email address (URL-encoded)

Response 200

addressstring
webhook_urlstring
statusinteger
successboolean
hintstring

Verify Service

POST/api/domains/{domain}/verify-service

Add DNS records to verify domain ownership for a third-party service

Adds the correct DNS records (TXT, CNAME) to prove domain ownership to a third-party service. Supports Stripe, Google Search Console, AWS SES, Postmark, Resend, Facebook, HubSpot, and Microsoft 365. Unknown service names fall back to a generic TXT record. Records are merged with existing DNS - existing records are preserved.

Parameters

domain*stringDomain name you own

Request body

service*stringService name (e.g. stripe, google-search-console, aws-ses, postmark, resend, facebook, hubspot, microsoft-365)
token*stringVerification token provided by the service

Response 200

domainstring
servicestring
service_display_namestring
known_serviceboolean
records_addedDnsRecord[]
total_recordsinteger
docsstring (uri) | null
hintstring

Example response

{
  "domain": "mysite.com",
  "service": "stripe",
  "service_display_name": "Stripe",
  "known_service": true,
  "records_added": [
    {
      "type": "TXT",
      "name": "@",
      "value": "stripe-verification=abc123def456",
      "ttl": 3600
    }
  ],
  "total_records": 3,
  "docs": "https://docs.stripe.com/identity/verification-checks/selfie#domain-verification",
  "hint": "Stripe verification records added. Return to Stripe to complete verification."
}

Example

curl -X POST https://domani.run/api/domains/mysite.com/verify-service \
  -H "Authorization: Bearer domani_sk_xxx" \
  -H "Content-Type: application/json" \
  -d '{"service": "stripe", "token": "abc123def456"}'
GET/api/domains/{domain}/verify-service

List supported services for domain verification

Returns all known services with their names, descriptions, and documentation URLs. Unknown services fall back to a generic TXT record.

Parameters

domain*stringDomain name you own

Response 200

domainstring
servicesobject[]
hintstring

Discovery

GET/api/domains/whoisPublic

Look up domain registration data via RDAP

Queries the RDAP (Registration Data Access Protocol) network for domain registration data. Returns registrar, creation/expiry/update dates, days until expiry, status codes, nameservers, DNSSEC status, and contact information (registrant, admin, tech, billing, abuse). Works for any domain - no ownership required. RDAP is the modern, JSON-native replacement for WHOIS. Falls back to WHOIS (port 43) for TLDs without RDAP, and supplements RDAP with WHOIS referral data when contacts are missing (common post-GDPR). Contact information may be partially or fully redacted due to GDPR or privacy protection services. Public endpoint - no authentication required. Authenticated requests get higher rate limits (30/min vs 10/min).

Parameters

q*stringDomain to look up (e.g. example.com)

Response 200

domainstring
registeredbooleanWhether the domain is currently registered
registrarstring | nullCurrent registrar name
registrar_urlstring | nullRegistrar website URL
registrar_iana_idstring | nullRegistrar IANA ID (from RDAP publicIds)
createdstring | nullRegistration date (YYYY-MM-DD)
expiresstring | nullExpiration date (YYYY-MM-DD)
updatedstring | nullLast updated date (YYYY-MM-DD)
days_until_expiryinteger | nullDays until expiry (null if not registered)
statusstring[]EPP status codes
nameserversstring[]Authoritative nameservers
dnssecbooleanWhether DNSSEC is enabled
redactedbooleanWhether WHOIS/RDAP data is redacted for privacy (GDPR)
contactsobjectContact information by role (null when no data available, fields may be null when redacted)

Example

curl "https://domani.run/api/domains/whois?q=google.com"
GET/api/domains/{domain}/ogPublic

Get website preview metadata for a domain

Returns Open Graph metadata (title, description, image, favicon) for a domain. Data is lazily fetched and cached for 7 days. First request for a domain may take 1-5s (live fetch); subsequent requests are instant (cache hit). Returns null fields if the site is unreachable or has no OG tags. Public endpoint - no authentication required.

Parameters

domain*stringDomain to fetch metadata for (e.g. google.com)

Response 200

titlestring | nullPage title (og:title > twitter:title > <title>)
descriptionstring | nullPage description (og:description > meta description)
imagestring | nullOG image URL (absolute)
faviconstring | nullFavicon URL (absolute)

Example

curl "https://domani.run/api/domains/google.com/og"

API Tokens

GET/api/tokens

List your API tokens

Returns all API tokens for the authenticated user. Keys are masked for security (first 12 + last 4 characters visible). Use this to audit active tokens.

Response 200

tokensTokenInfo[]

Example response

{
  "tokens": [
    {
      "id": "cm5abc123",
      "name": "Default",
      "key": "domani_sk_a...wxyz",
      "lastUsed": "2025-02-18T10:30:00.000Z",
      "expiresAt": null,
      "expired": false,
      "scopes": [
        "*"
      ],
      "createdAt": "2025-01-15T10:30:00.000Z"
    },
    {
      "id": "cm5def456",
      "name": "CI/CD",
      "key": "domani_sk_b...vwxy",
      "lastUsed": null,
      "expiresAt": "2025-05-01T14:00:00.000Z",
      "expired": false,
      "scopes": [
        "domains:read",
        "domains:write"
      ],
      "createdAt": "2025-02-01T14:00:00.000Z"
    }
  ]
}

Example

curl https://domani.run/api/tokens \
  -H "Authorization: Bearer domani_sk_xxx"
POST/api/tokens

Create a new API token

Creates a new API token. The full key is returned only in this response - store it securely. You can create multiple tokens to separate access for different applications.

Request body

namestringHuman-readable name to identify this token
expires_inintegerToken lifetime in seconds (min 3600 = 1h, max 31536000 = 1y). Omit for no expiration.
expires_atstring (date-time)Absolute expiration date (ISO 8601). Alternative to expires_in.
scopesstring[]Permission scopes for this token. Use ['*'] for full access (default). A token can only grant scopes it already has (scope attenuation). Scopes: domains:read (GET /api/domains, GET /api/domains/{domain}, GET /api/domains/{domain}/dns, /status, /email/check, /auth-code, /transfer-away, /transfer-status, /analytics), domains:write (PUT /api/domains/{domain}/dns, POST /connect, POST /verify, PUT /settings, PUT /parking, PUT/DELETE /api/domains/{domain}/for-sale, POST /api/domains/import, POST /import/verify), domains:transfer (POST /api/domains/buy, POST /transfer, POST /renew (involves payment, includes marketplace purchases)), tokens:read (GET /api/tokens), tokens:write (POST /api/tokens, DELETE /api/tokens/{id}), webhooks:read (GET /api/webhooks, GET /api/webhooks/{id}/deliveries), webhooks:write (POST /api/webhooks, PATCH /api/webhooks/{id}, DELETE /api/webhooks/{id}), email:read (GET /api/emails, /api/emails/{address}, /api/emails/{address}/messages, /api/domains/{domain}/email/status), email:write (POST /api/emails, POST /api/emails/{address}/send, POST /api/domains/{domain}/email/setup), account:read (GET /api/me), account:write (DELETE /api/me, POST /api/billing/setup, POST /api/me/resend-verification), billing:read (GET /api/billing/invoices), search (GET /api/domains/search, /suggest, /whois, /dns-check, GET /api/tlds), deals:read (GET /api/deals, GET /api/deals/{id}), deals:write (POST /api/domains/sell, PATCH /api/deals/{id}), notifications:read (GET /api/notifications, GET /api/notifications/count).

Response 201

idstring
namestring
keystringFull API key - save this, it won't be shown again
expiresAtstring (date-time) | nullWhen this token expires (null = never)
scopesstring[]Permission scopes granted to this token
createdAtstring (date-time)

Example response

{
  "id": "cm5ghi789",
  "name": "CI/CD Pipeline",
  "key": "domani_sk_x1Y2z3A4b5C6d7E8f9G0hIjK",
  "expiresAt": "2025-03-20T10:30:00.000Z",
  "scopes": [
    "domains:read",
    "domains:write"
  ],
  "createdAt": "2025-02-18T10:30:00.000Z"
}

Example

curl -X POST https://domani.run/api/tokens \
  -H "Authorization: Bearer domani_sk_xxx" \
  -H "Content-Type: application/json" \
  -d '{"name": "CI/CD Pipeline"}'
DELETE/api/tokens/{id}

Revoke an API token

Permanently revokes an API token. Any applications using this token will immediately lose access. This action cannot be undone.

Parameters

id*stringToken ID (from GET /api/tokens response)

Response 200

successboolean

Example response

{
  "success": true
}

Example

curl -X DELETE https://domani.run/api/tokens/cm5abc123 \
  -H "Authorization: Bearer domani_sk_xxx"

Billing

POST/api/billing/setup

Get checkout URL for adding a payment method

Creates a checkout session for adding a credit card. Redirect the user to the returned URL. This is required before purchasing domains. After the user completes checkout, they are redirected back to the dashboard.

Request body

modestring(setup_intent | checkout)setup_intent returns a client_secret for frontend Stripe.js. checkout returns a URL to redirect the user to Stripe Checkout.

Response 200

urlstring (uri)Checkout URL (checkout mode)
client_secretstringStripe SetupIntent client secret (setup_intent mode)
hintstringNext steps guidance

Example response

{
  "url": "https://checkout.domani.run/session/cs_abc123...",
  "hint": "Open this URL in a browser to add your payment card."
}

Example

curl -X POST https://domani.run/api/billing/setup \
  -H "Authorization: Bearer domani_sk_xxx" \
  -H "Content-Type: application/json" \
  -d '{"mode": "checkout"}'
DELETE/api/billing/card

Remove saved payment method

Detaches the user's saved credit card from their account. After removal, a new card must be added before purchasing domains.

Response 200

removedboolean

Example response

{
  "removed": true
}

Example

curl -X DELETE https://domani.run/api/billing/card \
  -H "Authorization: Bearer domani_sk_xxx"
GET/api/billing/invoices

List payment invoices

Returns a list of paid invoices with invoice number, amount, date, and links to view or download the PDF. Invoices are generated automatically for every domain purchase, renewal, and transfer.

Parameters

limitintegerMax invoices to return

Response 200

invoicesobject[]
has_moreboolean
hintstring

Example

curl https://domani.run/api/billing/invoices \
  -H "Authorization: Bearer domani_sk_xxx"

Referrals

GET/api/referrals

Get referral earnings and history

Returns your unique referral code, commission rate (20%), earnings breakdown (total, paid, pending), and a history of all referred purchases. Share your referral code with others - when they register with your code (via the ref parameter), you earn a commission on all their future domain purchases automatically.

Response 200

referral_codestringYour unique referral code to share
referral_rateintegerCommission percentage (currently 20%)
total_earned_centsintegerTotal earnings in cents
total_paid_centsintegerTotal paid out in cents
total_pending_centsintegerPending payout in cents
referralsobject[]

Example response

{
  "referral_code": "x7k9m2",
  "referral_rate": 20,
  "total_earned_cents": 400,
  "total_paid_cents": 200,
  "total_pending_cents": 200,
  "referrals": [
    {
      "id": "cm5ref001",
      "domains": [
        "friend-site.com"
      ],
      "earned_cents": 200,
      "paid_out": true,
      "created_at": "2025-01-20T10:00:00.000Z"
    },
    {
      "id": "cm5ref002",
      "domains": [
        "another.dev"
      ],
      "earned_cents": 200,
      "paid_out": false,
      "created_at": "2025-02-10T15:30:00.000Z"
    }
  ]
}

Example

curl https://domani.run/api/referrals \
  -H "Authorization: Bearer domani_sk_xxx"

Notifications

GET/api/notifications

List notifications

List your in-app notifications with cursor-based pagination. Includes deal events, listing updates, and system alerts. Requires notifications:read scope.

Parameters

limitintegerMax notifications to return
cursorstringCursor for pagination (from previous response's next_cursor)

Response 200

notificationsobject[]
has_moreboolean
next_cursorstring | null
GET/api/notifications/count

Get unread notification count

Lightweight endpoint for polling unread notification count. Requires notifications:read scope.

Response 200

unread_countinteger

Example response

{
  "unread_count": 3
}
PATCH/api/notifications/{id}

Mark notification as read

Mark a single notification as read. Requires notifications:read scope.

Parameters

id*stringNotification ID

Request body

read*booleanSet to true to mark as read

Response 200

okboolean
POST/api/notifications/read-all

Mark all notifications as read

Mark all unread notifications as read. Requires notifications:read scope.

Response 200

okboolean
countintegerNumber of notifications marked as read
GET/api/notifications/preferences

Get notification preferences

Returns the user's notification preferences for each category and channel (in_app, email). Missing preferences default to enabled. Requires notifications:read scope.

Response 200

preferencesobject[]
PUT/api/notifications/preferences

Update notification preferences

Update notification preferences for specific categories and channels. Requires notifications:read scope.

Request body

preferences*object[]

Response 200

okboolean

Webhooks

GET/api/webhooks

List webhooks

List all webhook endpoints configured for your account. Returns URL, subscribed events, and active status for each webhook. Secrets are never returned after creation.

Response 200

webhooksobject[]
hintstring

Example response

{
  "webhooks": [
    {
      "id": "cm5wh001",
      "url": "https://example.com/webhooks/domani",
      "events": [
        "domain.purchased",
        "dns.updated"
      ],
      "active": true,
      "created_at": "2026-02-25T10:00:00.000Z"
    }
  ],
  "hint": "1 webhook(s)."
}

Example

curl https://domani.run/api/webhooks \
  -H "Authorization: Bearer domani_sk_xxx"
POST/api/webhooks

Create webhook

Register a new webhook endpoint. The URL must use HTTPS. Choose which event types to subscribe to. The webhook secret is returned only once at creation - save it to verify incoming payloads with HMAC-SHA256. Maximum 5 webhooks per account.

Request body

url*string (uri)HTTPS URL that will receive webhook POST requests
events*string[]Event types to subscribe to. Use GET /api/webhooks/events to list available types.

Response 200

idstring
urlstring (uri)
eventsstring[]
secretstringHMAC-SHA256 signing secret (shown once)
activeboolean
created_atstring (date-time)
hintstring

Example response

{
  "id": "cm5wh001",
  "url": "https://example.com/webhooks/domani",
  "events": [
    "domain.purchased",
    "dns.updated"
  ],
  "secret": "whsec_aBcDeFgHiJkLmNoPqRsTuVwXyZ012345",
  "active": true,
  "created_at": "2026-02-25T10:00:00.000Z",
  "hint": "Webhook created. Save the secret - it won't be shown again. Use it to verify incoming payloads with HMAC-SHA256."
}

Example

curl -X POST https://domani.run/api/webhooks \
  -H "Authorization: Bearer domani_sk_xxx" \
  -H "Content-Type: application/json" \
  -d '{"url":"https://example.com/hook","events":["domain.purchased","dns.updated"]}'
PATCH/api/webhooks/{id}

Update webhook

Update a webhook's URL, subscribed events, or active status. Send only the fields you want to change.

Parameters

id*stringWebhook ID

Request body

urlstring (uri)New HTTPS URL
eventsstring[]New list of event types
activebooleanSet false to pause, true to resume

Response 200

idstring
urlstring (uri)
eventsstring[]
activeboolean
hintstring
DELETE/api/webhooks/{id}

Delete webhook

Delete a webhook endpoint. All pending deliveries will be cancelled. This cannot be undone.

Parameters

id*stringWebhook ID

Response 200

successboolean
hintstring
GET/api/webhooks/{id}/deliveries

List webhook deliveries

Get recent delivery attempts for a webhook. Shows event type, HTTP status, number of attempts, and any errors. Useful for debugging failed deliveries.

Parameters

id*stringWebhook ID
limitintegerMaximum number of deliveries to return

Response 200

deliveriesobject[]
hintstring

Example response

{
  "deliveries": [
    {
      "id": "cm5del001",
      "event_type": "domain.purchased",
      "status": "delivered",
      "http_status": 200,
      "attempts": 1,
      "last_error": null,
      "created_at": "2026-02-25T10:05:00.000Z"
    }
  ],
  "hint": "1 delivery(ies) for this webhook."
}
GET/api/webhooks/events

List webhook event types

List all available webhook event types with descriptions. No authentication required.

Response 200

eventsobject[]

Example response

{
  "events": [
    {
      "type": "domain.purchased",
      "description": "A new domain was successfully registered"
    },
    {
      "type": "dns.updated",
      "description": "DNS records were changed for a domain"
    },
    {
      "type": "transfer.completed",
      "description": "A domain transfer completed successfully"
    }
  ]
}

Example

curl https://domani.run/api/webhooks/events