Unified error types for exchange operations.
All exchange errors are normalized to this struct, providing consistent error handling across every configured exchange. Each error carries its type, the original exchange error code and message, and whether it's recoverable.
Error Types
Recoverable (can retry automatically)
:rate_limit_exceeded- Too many requests, retry afterretry_afterms:network_error- Connection or timeout issue:exchange_not_available- Exchange down, on maintenance, or market closed
Non-recoverable (require intervention)
:authentication_error- API key/secret rejected or invalid nonce:insufficient_funds- Not enough funds for the operation:invalid_order- Order parameters rejected by exchange:order_not_found- Order ID does not exist:bad_request- Invalid request parameters:bad_symbol- Symbol not recognized by exchange:permission_denied- API key lacks permissions or account suspended:access_restricted- Geographic/IP block or wrong-URL HTML response (non-Cloudflare):cloudflare_challenge- Cloudflare anti-bot challenge page (exchange reachable but requires browser/approved client):not_supported- Method not supported by this exchange:operation_failed- Operation rejected or failed:invalid_parameters- Invalid request parameters (code bug):market_closed- Market is not currently trading:circuit_open- Circuit breaker tripped due to consecutive failures
Generic
:exchange_error- Unmapped error (seecodeandmessage)
Example
case CCXT.HTTP.request(exchange, :post, "/v5/order/create", params: params) do
{:ok, response} -> handle_response(response)
{:error, %CCXT.Error{type: :insufficient_funds}} -> notify_low_balance()
{:error, %CCXT.Error{type: :rate_limit_exceeded, retry_after: ms}} -> Process.sleep(ms)
{:error, %CCXT.Error{} = err} -> Logger.error("Exchange error: #{err.message}")
end
Summary
Functions
Creates an access restricted error.
Creates an authentication error.
Creates a bad request error.
Creates a bad symbol error.
Creates a circuit breaker open error.
Creates a Cloudflare challenge error.
Creates a generic exchange error.
Creates an exchange not available error.
Maps a CCXT spec exception class to an error type atom.
Creates an insufficient funds error.
Creates an invalid order error.
Creates an invalid parameters error.
Creates a market closed error.
Creates a network error.
Returns all non-recoverable error types.
Creates a not supported error.
Creates an operation failed error.
Creates an order not found error.
Creates a permission denied error.
Creates a rate limit exceeded error.
Returns the recoverability classification for an error type.
Returns all recoverable error types.
Returns the full spec class to error type mapping.
Types
@type error_type() ::
:rate_limit_exceeded
| :network_error
| :exchange_not_available
| :authentication_error
| :insufficient_funds
| :invalid_order
| :order_not_found
| :bad_request
| :bad_symbol
| :permission_denied
| :access_restricted
| :cloudflare_challenge
| :not_supported
| :operation_failed
| :invalid_parameters
| :market_closed
| :circuit_open
| :exchange_error
@type t() :: %CCXT.Error{ __exception__: true, code: String.t() | integer() | nil, exchange: String.t() | nil, hints: [String.t()], http_status: non_neg_integer() | nil, message: String.t(), raw: map() | nil, recoverable: boolean() | nil, retry_after: non_neg_integer() | nil, type: error_type() }
Functions
Creates an access restricted error.
Used when exchange returns HTML instead of JSON without Cloudflare
markers — typically a wrong URL/prefix, geo/IP block, or landing page.
Cloudflare challenges use cloudflare_challenge/1 instead.
Creates an authentication error.
Creates a bad request error.
Creates a bad symbol error.
Creates a circuit breaker open error.
Creates a Cloudflare challenge error.
Used when the exchange is reachable but served a Cloudflare anti-bot challenge page (e.g. "Just a moment..."). Inconclusive for integration tests — the client is reaching the right host but needs a browser or approved path to pass the challenge.
Creates a generic exchange error.
Use this for errors that don't fit other categories.
Creates an exchange not available error.
@spec from_spec_class(String.t()) :: error_type()
Maps a CCXT spec exception class to an error type atom.
Accepts both raw class names and __function: prefixed strings from specs.
Examples
from_spec_class("AuthenticationError")
#=> :authentication_error
from_spec_class("__function:InsufficientFunds")
#=> :insufficient_funds
from_spec_class("UnknownClass")
#=> :exchange_error
Creates an insufficient funds error.
Creates an invalid order error.
Creates an invalid parameters error.
Creates a market closed error.
Creates a network error.
@spec non_recoverable_types() :: [error_type()]
Returns all non-recoverable error types.
Creates a not supported error.
Creates an operation failed error.
Creates an order not found error.
Creates a permission denied error.
Creates a rate limit exceeded error.
Options
:retry_after- Milliseconds until retry is allowed:exchange- Exchange ID string:raw- Original error response from exchange:hints- List of debugging hint strings
@spec recoverable?(error_type()) :: boolean() | nil
Returns the recoverability classification for an error type.
true— recoverable (can retry automatically)false— not recoverable (requires intervention)nil— unknown (generic exchange_error)
@spec recoverable_types() :: [error_type()]
Returns all recoverable error types.
@spec spec_class_mapping() :: %{required(String.t()) => error_type()}
Returns the full spec class to error type mapping.