NAV
json

X10 API Documentation

By using the X10 API, you agree to the X10 Terms & Privacy Policy. If you do not agree to the foregoing terms, do not use the X10 API.

Introduction

Welcome to the X10 API Documentation! This guide is designed to assist traders and developers in integrating with our hybrid perpetuals exchange.

X10 operates as a hybrid Central Limit Order Book (CLOB) exchange. While order processing, matching, position risk assessment, and transaction sequencing are handled off-chain, trade settlement occurs on-chain through the StarkEx Layer 2 engine.

StarkEx, built by Starkware, is a battle-tested scalability solution that brings lightning-fast transaction processing to the forefront, guaranteeing an unparalleled trading experience. This is achieved while preserving complete trustlessness and transparency through three key mechanisms embedded into StarkEx:

  1. On-chain validations of the trading logic ensure that fraudulent or incorrect transactions, including liquidations, contrary to the on-chain trading logic are never permitted.

  2. The mark prices, which determine the liquidability of positions, are obtained from multiple independent oracle providers to prevent potential price manipulation and ensure fair and accurate asset valuation within our ecosystem. X10 utilizes 5 Stork nodes to define the median mark price.

  3. Publication of zero-knowledge (ZK) proofs on the Ethereum Layer 1 blockchain validates Layer 2 transactions, ensuring both their integrity and security.

For deeper insights on X10 Exchange, please refer to the blogs X10 Vision and Architecture.

To optimize high-frequency trading performance, the X10 API operates asynchronously. This means that when you place an order, it immediately returns an order ID, even before the order is officially recorded in the book. To track your order status in real time, it's essential to subscribe to the Order WebSocket stream. This stream delivers instant updates about any changes to your orders, including confirmations, cancellations, and rejections, enabling you to manage your trading strategies effectively.

Python SDK

SDK configuration

from dataclasses import dataclass


@dataclass
class EndpointConfig:
    chain_rpc_url: str
    api_base_url: str
    stream_url: str
    onboarding_url: str
    signing_domain: str
    collateral_asset_contract: str
    asset_operations_contract: str
    collateral_asset_on_chain_id: str
    collateral_decimals: int


TESTNET_CONFIG = EndpointConfig(
    chain_rpc_url="https://rpc.sepolia.org",
    api_base_url="https://api.testnet.x10.exchange/api/v1",
    stream_url="wss://api.testnet.x10.exchange/stream.x10.exchange/v1",
    onboarding_url="https://api.testnet.x10.exchange",
    signing_domain="testnet.x10.exchange",
    collateral_asset_contract="0x0c9165046063b7bcd05c6924bbe05ed535c140a1",
    asset_operations_contract="0x7f0C670079147C5c5C45eef548E55D2cAc53B391",
    collateral_asset_on_chain_id="0x31857064564ed0ff978e687456963cba09c2c6985d8f9300a1de4962fafa054",
    collateral_decimals=6,
)

TESTNET_CONFIG_LEGACY_SIGNING_DOMAIN = EndpointConfig(
    chain_rpc_url="https://rpc.sepolia.org",
    api_base_url="https://api.testnet.x10.exchange/api/v1",
    stream_url="wss://api.testnet.x10.exchange/stream.x10.exchange/v1",
    onboarding_url="https://api.testnet.x10.exchange",
    signing_domain="x10.exchange",
    collateral_asset_contract="0x0c9165046063b7bcd05c6924bbe05ed535c140a1",
    asset_operations_contract="0x7f0C670079147C5c5C45eef548E55D2cAc53B391",
    collateral_asset_on_chain_id="0x31857064564ed0ff978e687456963cba09c2c6985d8f9300a1de4962fafa054",
    collateral_decimals=6,
)

MAINNET_CONFIG = EndpointConfig(
    chain_rpc_url="https://cloudflare-eth.com",
    api_base_url="https://api.prod.x10.exchange/api/v1",
    stream_url="wss://api.prod.x10.exchange/stream.x10.exchange/v1",
    onboarding_url="https://api.prod.x10.exchange",
    signing_domain="x10.exchange",
    collateral_asset_contract="0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48",
    asset_operations_contract="0x1cE5D7f52A8aBd23551e91248151CA5A13353C65",
    collateral_asset_on_chain_id="0x2893294412a4c8f915f75892b395ebbf6859ec246ec365c3b1f56f47c3a0a5d",
    collateral_decimals=6,
)

To enhance your algorithmic trading experience, X10 has released the 0.3.0-alpha version of our Python SDK.

Getting Started:

Supported Features:

We are committed to enhancing the SDK with more functionalities based on user feedback and evolving market needs.

Mainnet

Our Mainnet is running on Ethereum.

Base URL for the Mainnet API endpoints: https://api.prod.x10.exchange/.

Testnet

Our Testnet is running on Sepolia.

Base URL for the Testnet API endpoints: https://api.testnet.x10.exchange/.

On the testnet, users can claim $100,000 worth of test USDC per day for each L1 wallet. This can be done by clicking the 'Claim' button in the 'Account' section, located at the bottom right of the X10 Testnet Trade screen.

Allowed HTTP Verbs

GET: Retrieves a resource or list of resources.

POST: Creates a resource.

PATCH: Updates a resource.

DELETE: Deletes a resource.

Authentication

Due to the trustless and self-custody nature of the X10 exchange, transactions involving user funds require not only an API Key but also a valid Stark signature.

For order management, both an API Key and Stark signatures are necessary. For other endpoints, only the API Key signature is required. Stark signatures are generated using a private Stark key.

Account Creation, API and Stark Key Management

Currently, accounts can be created through the SDK or through the User Interface.

To create an account through the SDK, please refer to the onboarding example.

To create an account through the User Interface, connect your wallet on X10.exchange, and your first X10 account will be created.

You can create up to ten X10 sub-accounts per one L1 wallet address. You can add and manage all sub-accounts associated with your connected L1 wallet in the 'Account' section, located at the bottom right of the X10 Trade screen.

On the API management page, you can obtain API keys, Stark keys, and Vault numbers for each of your sub-accounts. Note that each sub-account is a separate StarkEx vault and therefore has unique API and Stark keys.

Authenticate Using API Key

X10 employs a simplified authentication scheme for API access. Authenticate by using your API key, which should be included in an HTTP header as follows: X-Api-Key: <API_KEY_FROM_API_MANAGEMENT_PAGE_OF_UI>.

Rate Limits

REST API endpoints are subject to rate limits. For real-time data, consider using the WebSockets API instead.

All REST API endpoints are throttled by IP address. Currently, the rate limit is set at 1,000 requests per minute, shared across all endpoints. We plan to increase these limits as our system expands. If you require an increase in the rate limit now, please reach out to our team on Discord.

Higher rate limit of 6,000 requests per minute apply for the market makers. Please refer to the Market Makers program for the details.

When a REST API rate limit is exceeded, a 429 status code will be returned.

Pagination

Paginated response schema:

type PaginatedResponse = {
  "status": "ok" | "error"
  "data": object | object[] | string | number,
  "error": {
    "code": number,
    "message": string
  },
  "pagination": {
    "cursor": number // Current cursor
    "count": number  // Count of the items in the response
  }
}

General not paginated response schema:

type GeneralResponse = {
  "status": "ok" | "error",
  "data": object | object[] | string | number,
  "error": {
    "code": number,
    "message": string
  }
}

The X10 API uses a cursor-based pagination model across all endpoints that may return large volumes of items.

Items are automatically sorted in descending order by ID unless otherwise specified in the endpoint description. As IDs increase over time, the most recent items are always returned first.

Pagination parameters are passed via the query string. These parameters include:

Parameter Required Type Description
cursor no number Determines the offset of the returned result. It represents the ID of the item after which you want to retrieve the next result. To get the next result page, use the cursor from the pagination section of the previous response.
limit no number The maximum number of items that should be returned.

Public REST-API

The following Public REST API endpoints enable users to access comprehensive information about available markets, their configurations, and trading statistics.

Get markets

HTTP Request

GET /api/v1/info/markets?market={market}

Get a list of available markets, their configurations, and trading statistics.

To request data for several markets, use the following format: GET /api/v1/info/markets?market=market1&market2.

Please note that the margin schedule by market is not covered by this endpoint. For more details on the margin schedule, please refer to the documentation.

Query Parameters

Parameter Required Type Description
market no string[] List of names of the requested markets.

Response example:

{
  "status": "ok",
  "data": [
    {
      "name": "BTC-USD",
      "assetName": "BTC",
      "assetPrecision": 6,
      "collateralAssetName": "USD",
      "collateralAssetPrecision": 6,
      "active": true,
      "marketStats": {
        "dailyVolume": "39659164065",
        "dailyVolumeBase": "39659164065",
        "dailyPriceChangePercentage": "5.57",
        "dailyLow": "39512",
        "dailyHigh": "42122",
        "lastPrice": "42000",
        "askPrice": "42005",
        "bidPrice": "39998",
        "markPrice": "39950",
        "indexPrice": "39940",
        "fundingRate": "0.001",
        "nextFundingRate": 1701563440,
        "openInterest": "1245.2",
        "openInterestBase": "1245.2"
      },
      "tradingConfig": {
        "minOrderSize": "0.001",
        "minOrderSizeChange": "0.001",
        "minPriceChange": "0.001",
        "maxMarketOrderValue": "1000000",
        "maxLimitOrderValue": "5000000",
        "maxPositionValue": "10000000",
        "maxLeverage": "50",
        "maxNumOrders": "200",
        "limitPriceCap": "0.05",
        "limitPriceFloor": "0.05"
      },
      "l2Config": {
        "type": "STARKX",
        "collateralId": "0x35596841893e0d17079c27b2d72db1694f26a1932a7429144b439ba0807d29c",
        "collateralResolution": 1000000,
        "syntheticId": "0x4254432d3130000000000000000000",
        "syntheticResolution": 10000000000
      }
    }
  ]
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data[].name yes string Name of the market.
data[].assetName yes string Name of the base asset.
data[].assetPrecision yes number Number of decimals for the base asset.
data[].collateralAssetName yes string Name of the collateral asset.
data[].collateralAssetPrecision yes number Number of decimals for the collateral asset.
data[].active yes boolean Indicates if the market is currently active. Can be true or false
data[].marketStats.dailyVolume yes string Trading volume of the market in the previous 24 hours in the collateral asset.
data[].marketStats.dailyVolumeBase yes string Trading volume of the market in the previous 24 hours in the base asset.
data[].marketStats.dailyPriceChange yes string Absolute price change of the last trade price over the past 24 hours.
data[].marketStats.dailyPriceChangePercentage yes string Percent price change of the last trade price over the past 24 hours.
data[].marketStats.dailyLow yes string Lowest trade price over the past 24 hours.
data[].marketStats.dailyHigh yes string Highest trade price over the past 24 hours.
data[].marketStats.lastPrice yes string Last price of the market.
data[].marketStats.askPrice yes string Current best ask price of the market.
data[].marketStats.bidPrice yes string Current best bid price of the market.
data[].marketStats.markPrice yes string Current mark price of the market.
data[].marketStats.indexPrice yes string Current index price of the market.
data[].marketStats.fundingRate yes string Current funding rate, calculated every minute.
data[].marketStats.nextFundingRate yes number Timestamp of the next funding update.
data[].marketStats.openInterest yes string Open interest in collateral asset.
data[].marketStats.openInterestBase yes string Open interest in base asset.
data[].tradingConfig.minOrderSize yes string Minimum order size for the market.
data[].tradingConfig.minOrderSizeChange yes string Minimum order size change for the market.
data[].tradingConfig.minPriceChange yes string Minimum price change for the market.
data[].tradingConfig.maxMarketOrderValue yes string Maximum market order value for the market.
data[].tradingConfig.maxLimitOrderValue yes string Maximum limit order value for the market.
data[].tradingConfig.maxPositionValue yes string Maximum position value for the market.
data[].tradingConfig.maxLeverage yes string Maximum leverage available for the market.
data[].tradingConfig.maxNumOrders yes string Maximum number of open orders for the market.
data[].tradingConfig.limitPriceCap yes string Limit order price cap.
data[].tradingConfig.limitPriceFloor yes string Limit order floor ratio.
data[].l2Config.type yes string Type of Layer 2 solution. Currently, only 'STARKX' is supported.
data[].l2Config.collateralId yes string StarkEx collateral asset ID.
data[].l2Config.collateralResolution yes number Collateral asset resolution, the number of quantums (StarkEx units) that fit within one "human-readable" unit of the collateral asset.
data[].l2Config.syntheticId yes string StarkEx synthetic asset ID.
data[].l2Config.syntheticResolution yes number Synthetic asset resolution, the number of quantums (StarkEx units) that fit within one "human-readable" unit of the synthetic asset.

Get market statistics

HTTP Request

GET /api/v1/info/markets/{market}/stats

Get the latest trading statistics for an individual market.

Please note that the returned funding rate represents the most recent funding rate, which is calculated every minute.

URL Parameters

Parameter Required Type Description
market yes string Name of the requested market.

Successful response example:

{
  "status": "OK",
  "data": {
    "dailyVolume": "10283410.122959",
    "dailyVolumeBase": "3343.1217",
    "dailyPriceChange": "-26.00",
    "dailyPriceChangePercentage": "-0.0084",
    "dailyLow": "3057.98",
    "dailyHigh": "3133.53",
    "lastPrice": "3085.70",
    "askPrice": "3089.05",
    "bidPrice": "3087.50",
    "markPrice": "3088.439710293828",
    "indexPrice": "3089.556987078441",
    "fundingRate": "-0.000059",
    "nextFundingRate": 1716192000000,
    "openInterest": "35827242.257619",
    "openInterestBase": "11600.4344",
    "deleverageLevels": {
      "shortPositions": [
        {
          "level": 1,
          "rankingLowerBound": "-1354535.1454"
        },
        {
          "level": 2,
          "rankingLowerBound": "-6.3450"
        },
        {
          "level": 3,
          "rankingLowerBound": "-0.3419"
        },
        {
          "level": 4,
          "rankingLowerBound": "0.0000"
        }
      ],
      "longPositions": [
        {
          "level": 1,
          "rankingLowerBound": "-2978.4427"
        },
        {
          "level": 2,
          "rankingLowerBound": "0.0000"
        },
        {
          "level": 3,
          "rankingLowerBound": "0.0000"
        },
        {
          "level": 4,
          "rankingLowerBound": "0.0001"
        }
      ]
    }
  }
}

Error response example:

{
  "status": "status: error",
  "error": {
    "code": "NOT_FOUND",
    "message": "Market not found"
  }
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data.dailyVolume yes string Trading volume of the market in the previous 24 hours in the collateral asset.
data.dailyVolumeBase yes string Trading volume of the market in the previous 24 hours in the base asset.
data.dailyPriceChange yes string Absolute price change of the last trade price over the past 24 hours.
data.dailyPriceChangePercentage yes string Percent price change of the last trade price over the past 24 hours.
data.dailyLow yes string Lowest trade price over the past 24 hours.
data.dailyHigh yes string Highest trade price over the past 24 hours.
data.lastPrice yes string Last price of the market.
data.askPrice yes string Current best ask price of the market.
data.bidPrice yes string Current best bid price of the market.
data.markPrice yes string Current mark price of the market.
data.indexPrice yes string Current index price of the market.
data.fundingRate yes string Current funding rate, calculated every minute.
data.nextFundingRate yes number Timestamp of the next funding update.
data.openInterest yes string Open interest in collateral asset.
data.openInterestBase yes string Open interest in base asset.
data.deleverageLevels yes enum Auto Deleveraging (ADL) levels for long and short positions, ranging from level 1 (lowest risk) to level 4 (highest risk) of ADL. For details, please refer to the documentation.

Get candles history

HTTP Request

GET /api/v1/info/candles/{market}/{candleType}

Get the candles history for an individual market for the timeframe specified in the request. Candles are sorted by timestamp in descending order.

Available price types include:

  1. Trades (last) price: GET /api/v1/info/candles/{market}/trades.

  2. Mark price: GET /api/v1/info/candles/{market}/mark-prices.

  3. Index price: GET /api/v1/info/candles/{market}/index-prices.

The endpoint returns a maximum of 10,000 records.

URL Parameters

Parameter Required Type Description
market yes string Name of the requested market.
candleType yes string Price type. Can be trades, mark-prices, or index-prices.

Query Parameters

Parameter Required Type Description
interval yes string The time interval between data points.
limit no number The maximum number of items that should be returned.
endTime no number End timestamp (in epoch milliseconds) for the requested period.

Response example:

{
  "status": "OK",
  "data": [
    {
      "o": "65206.2",
      "l": "65206.2",
      "h": "65206.2",
      "c": "65206.2",
      "v": "0.0",
      "T": 1715797320000
    }
  ]
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data[].o yes string Open price.
data[].c yes string Close price.
data[].h yes string Highest price.
data[].l yes string Lowest price.
data[].v yes string Trading volume (Only for trades candles).
data[].T yes number Starting timestamp (in epoch milliseconds) for the candle.

Get funding rates history

HTTP Request

GET /api/v1/info/{market}/funding?&startTime={startTime}&endTime={endTime}

Get the funding rates history for an individual market for the timeframe specified in the request. The funding rates are sorted by timestamp in descending order.

The endpoint returns a maximum of 10,000 records; pagination should be used to access records beyond this limit.

While the funding rate is calculated every minute, it is only applied once per hour. The records represent the 1-hour rates that were applied for the payment of funding fees.

For details on how the funding rate is calculated on X10, please refer to the documentation.

URL Parameters

Parameter Required Type Description
market yes string Names of the requested market.

Query Parameters

Parameter Required Type Description
startTime yes number Starting timestamp (in epoch milliseconds) for the requested period.
endTime yes number Ending timestamp (in epoch milliseconds) for the requested period.
cursor no number Determines the offset of the returned result. To get the next result page, you can use the cursor from the pagination section of the previous response.
limit no number Maximum number of items that should be returned.

Response example:

{
  "status": "OK",
  "data": [
    {
      "m": "BTC-USD",
      "T": 1701563440,
      "f": "0.001"
    }
  ],
  "pagination": {
    "cursor": 1784963886257016832,
    "count": 1
  }
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data[].m yes string Name of the requested market.
data[].T yes number Timestamp (in epoch milliseconds) when the funding rate was calculated and applied.
data[].f yes string Funding rates used for funding fee payments.

Get Market Makers' Q Scores

HTTP Request

GET /api/v1/info/rewards/maker/leaderboard?market={market}

Get the current rankings, reward shares, and Q scores across market makers in the current epoch for the specified markets. If no market is specified, data for all markets will be returned. The rankings are sorted by total score in descending order.

All scores are recalculated every minute, and the records represent the latest state of the epoch.

For details on the Market Makers program and how to participate, please refer to the documentation.

Query Parameters

Parameter Required Type Description
market no string[] List of names of the requested markets.

Response example:

{
  "status": "OK",
  "data": [
    {
      "marketName": "BTC-USD",
      "scores": [
        {
          "ranking": 1,
          "l1WalletAddress": "0x1b...f0c69",
          "volumeShare": "0",
          "liquidityScore": "0",
          "availabilityScore": "0",
          "volumeScore": "0",
          "totalScore": "0",
          "rewardsShare": "0"
        },
        {
          "ranking": 2,
          "l1WalletAddress": "0x14...af780",
          "volumeShare": "0",
          "liquidityScore": "0",
          "availabilityScore": "0",
          "volumeScore": "0",
          "totalScore": "0",
          "rewardsShare": "0"
        }
      ]
    }
  ]
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data[].marketName yes string Name of the market.
data[].scores[].ranking yes number Rank in the current epoch
data[].scores[].l1WalletAddress yes string Masked wallet.
data[].scores[].volumeShare yes number Maker volume share (Client's maker volume / Total maker volume).
data[].scores[].liquidityScore yes number (Sum of Qmin) ^ 0.3.
data[].scores[].availabilityScore yes number (Sum(Count(Qmin > 0))) ^ 5.
data[].scores[].volumeScore yes number (Maker volume) ^ 0.7.
data[].scores[].totalScore yes number Total Q score.
data[].scores[].rewardsShare yes number Reward share.

Private REST-API

Account

You can create up to ten X10 sub-accounts for each L1 wallet address. For more details, please refer to the Authentication section of the API Documentation.

The Private API endpoints listed below grant access to details specific to each sub-account, such as balances, transactions, positions, orders, trades, and the fee rates applied. Additionally, there are endpoints for retrieving the current leverage and adjusting it.

Please note that all endpoints in this section will only return records for the authenticated sub-account.

Get balance

HTTP Request

GET /api/v1/user/balance

Get key balance details for the authenticated sub-account:

  1. Account Balance = Deposits - Withdrawals + Realised PnL.

  2. Equity = Account Balance + Unrealised PnL.

  3. Available Balance for Trading = Equity - Initial Margin Requirement.

  4. Available Balance for Withdrawals = max(0, Wallet Balance + min(0,Unrealised PnL) - Initial Margin Requirement).

  5. Unrealised PnL = The sum of unrealised PnL across open positions, calculated as Position Size * (Mark Price - Entry Price).

  6. Initial Margin Requirement for a given market = Max(Abs(Position Value + Value of Buy Orders), Abs(Position Value + Value of Sell Orders))*1/Leverage.

  7. Account Margin Ratio = Maintenance Margin requirement of all open positions / Equity. Liquidation is triggered when Account Margin Ratio > 100%.

  8. Account Exposure = Sum(All positions value)

  9. Account Leverage = Exposure / Equity.

Response example:

{
  "status": "OK",
  "data": {
    "collateralName": "USDC",
    "balance": "13500",
    "equity": "12000",
    "availableForTrade": "1200",
    "availableForWithdrawal": "100",
    "unrealisedPnl": "-10.1",
    "initialMargin": "160",
    "marginRatio": "1.5",
    "exposure": "12751.859629",
    "leverage": "1275.1860",
    "updatedTime": 1701563440
  }
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data.collateralName yes string Name of the collateral asset used for the account.
data.balance yes string Account balance expressed in the collateral asset, also known as Wallet balance.
data.equity yes string Equity of the account.
data.availableForTrade yes string Available Balance for Trading.
data.availableForWithdrawal yes string Available Balance for Withdrawals.
data.unrealisedPnl yes string Current unrealised PnL of the account.
data.initialMargin yes string Collateral used to open the positions and orders.
data.marginRatio yes string Margin ratio of the account.
data.exposure yes string Exposure of the account.
data.leverage yes string Leverage of the account.
data.updatedTime yes number Timestamp (in epoch milliseconds) when the server generated the balance message.

Get deposits, withdrawals, transfers history

HTTP Request

GET /api/v1/user/assetOperations?&type={type}&status={status}

Get the history of deposits, withdrawals, and transfers between sub-accounts for the authenticated sub-account. Optionally, the request can be filtered by a specific transaction type or status.

The endpoint returns 50 records per page; pagination should be used to access records beyond this limit. Transactions are sorted by timestamp in descending order.

Transactions types

Status Description
DEPOSIT Deposit.
CLAIM Testing funds claim. Available only on X10 Testnet.
TRANSFER Transfer between sub-accounts within one L1 wallet.
SLOW_WITHDRAWAL Slow withdrawal.
FAST_WITHDRAWAL Fast withdrawal.

Transactions statuses

Status Description
CREATED Transaction created on X10.
IN_PROGRESS Transaction is being processed by X10 or StarkEx.
READY_FOR_CLAIM Slow withdrawal or rejected deposit ready for claim.
COMPLETED Transaction completed.
REJECTED Transaction rejected.

Response example:

{
    "status": "OK",
    "data": [
        {
            "id": "1236",
            "type": "TRANSFER",
            "status": "COMPLETED",
            "amount": "-5.0000000000000000",
            "fee": "0",
            "asset": 1,
            "time": 1722627477268,
            "accountId": 3001,
            "counterpartyAccountId": 3025
        },
        {
            "id": "0x3e8adae72bd576efa0410f3b18f3fe77b638d6fdd8fb49c0b38309ccc01839ac:0x029398cecc60f4e68c3ebeb28fa98df003ac4e0328b086aad5496a4d56815ccc:0xe7",
            "type": "DEPOSIT",
            "status": "COMPLETED",
            "amount": "10.0000000000000000",
            "fee": "0",
            "asset": 1,
            "time": 1722607553870,
            "accountId": 3001,
            "transactionHash": "0x3e8adae72bd576efa0410f3b18f3fe77b638d6fdd8fb49c0b38309ccc01839ac"
        },
        {
            "id": "1235",
            "type": "SLOW_WITHDRAWAL",
            "status": "COMPLETED",
            "amount": "-5.0000000000000000",
            "fee": "0",
            "asset": 1,
            "time": 1722607138130,
            "accountId": 3001,
            "transactionHash": "0xfd8d1bc06ee36b1a4b0bd98377818ac9a22b2346920aa2eb20015d4356a66319"
        },
        {
            "id": "1234",
            "type": "FAST_WITHDRAWAL",
            "status": "COMPLETED",
            "amount": "-5.0000000000000000",
            "fee": "4.2057230000000000",
            "asset": 1,
            "time": 1722607127646,
            "accountId": 3011,
            "transactionHash": "0x3babb6d6b0eecc62e47f71ff7873381312f25c91d576072f1d0b332a25d035b5"
        }
    ],
    "pagination": {
        "cursor": 23,
        "count": 23
    }
}

Query Parameters

Parameter Required Type Description
type no string Transaction type. Refer to the list of transaction types in the endpoint description above.
status no string Transaction status. Refer to the list of statuses in the endpoint description above.
cursor no Determines the offset of the returned result. It represents the ID of the item after which you want to retrieve the next result. To get the next result page, you can use the cursor from the pagination section of the previous response.
limit no number Maximum number of items that should be returned.

Response

Parameter Required Type Description
status yes string Response status. Can be ok or error.
data[].id yes number or string Transaction ID. A number assigned by X10 for transfers and withdrawals. An onchain id string for deposits.
data[].type yes string Transaction type. Refer to the list of transaction types in the endpoint description above.
data[].status yes string Transaction status. Refer to the list of statuses in the endpoint description above.
data[].amount yes string Transaction amount, absolute value in collateral asset.
data[].fee yes string Fee paid.
data[].asset yes string Collateral asset name.
data[].time yes number Timestamp (epoch milliseconds) when the transaction was updated.
data[].accountId yes number Account ID; source account for transfers and withdrawals; destination account for deposits.
data[].counterpartyAccountId no number Account ID; destination account for transfers.
data[].transactionHash no string Onchain transaction hash. Not available for transfers.

Get positions

HTTP Request

GET /api/v1/user/positions?market={market}&side={side}

Get all open positions for the authenticated sub-account. Optionally, the request can be filtered by a specific market or position side (long or short).

To request data for multiple markets, use the following format: GET /api/v1/user/positions?market=market1&market2.

Query Parameters

Parameter Required Type Description
market no string List of names of the requested markets.
side no string Position side. Can be long or short.

Response example:

{
  "status": "OK",
  "data": [
    {
      "id": 1,
      "accountId": 1,
      "market": "BTC-USD",
      "side": "LONG",
      "leverage": "10",
      "size": "0.1",
      "value": "4000",
      "openPrice": "39000",
      "markPrice": "40000",
      "liquidationPrice": "38200",
      "margin": "20",
      "unrealisedPnl": "1000",
      "realisedPnl": "1.2",
      "tpTriggerPrice": "41000",
      "tpLimitPrice": "41500",
      "slTriggerPrice": "39500",
      "slLimitPrice": "39000",
      "adl": "2.5",
      "maxPositionSize": "0.2",
      "createdTime": 1701563440000,
      "updatedTime": 1701563440
    }
  ]
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data[].id yes number Position ID assigned by X10.
data[].accountId yes number Account ID.
data[].market yes string Market name.
data[].side yes string Position side. Can be long or short.
data[].leverage yes string Position leverage.
data[].size yes string Position size, absolute value in base asset.
data[].value yes string Position value, absolute value in collateral asset.
data[].openPrice yes string Position's open (entry) price.
data[].markPrice yes string Current mark price of the market.
data[].liquidationPrice yes string Position's liquidation price.
data[].margin yes string Position's margin in collateral asset.
data[].unrealisedPnl yes string Position's Unrealised PnL.
data[].realisedPnl yes string Position's Realised PnL.
data[].tpTriggerPrice no string Take Profit Trigger price.
data[].tpLimitPrice no string Take Profit Limit price.
data[].slTriggerPrice no string Stop Loss Trigger price.
data[].slLimitPrice no string Stop Loss Limit price.
data[].maxPositionSize yes string Maximum allowed position size, absolute value in base asset.
data[].adl yes string Position's Auto-Deleveraging (ADL) ranking in the queue, expressed as a percentile. A value closer to 100 indicates a higher likelihood of being ADLed.
data[].createdTime yes number Timestamp (epoch milliseconds) when the position was created.
data[].updatedTime yes number Timestamp (epoch milliseconds) when the position was updated.

Get positions history

HTTP Request

GET /api/v1/user/positions/history?market={market}&side={side}

Get all open and closed positions for the authenticated sub-account. Optionally, the request can be filtered by a specific market or position side (long or short).

To request data for several markets, use the following format: GET /api/v1/user/positions/history?market=market1&market2.

The endpoint returns a maximum of 10,000 records; pagination should be used to access records beyond this limit.

Query Parameters

Parameter Required Type Description
market no string List of names of the requested markets.
side no string Position side. Can be long or short.
cursor no number Determines the offset of the returned result. It represents the ID of the item after which you want to retrieve the next result. To get the next result page, you can use the cursor from the pagination section of the previous response.
limit no number Maximum number of items that should be returned.

Response example:

{
  "status": "OK",
  "data": [
    {
      "id": 1784963886257016832,
      "accountId": 1,
      "market": "BTC-USD",
      "side": "LONG",
      "exitType": "TRADE",
      "leverage": "10",
      "size": "0.1",
      "maxPositionSize": "0.2",
      "openPrice": "39000",
      "exitPrice": "40000",
      "realisedPnl": "1.2",
      "createdTime": 1701563440000,
      "closedTime": 1701563440
    }
  ],
  "pagination": {
    "cursor": 1784963886257016832,
    "count": 1
  }
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data[].id yes number Position ID assigned by X10.
data[].accountId yes number Account ID.
data[].market yes string Market name.
data[].side yes string Position side. Can be long or short.
data[].exitType no string The exit type of the last trade that reduced the position. Can be trade, liquidation, or deleverage.
data[].leverage yes string Position leverage.
data[].size yes string Position size, absolute value in base asset.
data[].maxPositionSize yes string Maximum position size during the position's lifetime, absolute value in base asset.
data[].openPrice yes string The weighted average price of trades that contributed to increasing the position.
data[].exitPrice no string The weighted average price of trades that contributed to decreasing the position.
data[].realisedPnl yes string Position Realised PnL.
data[].createdTime yes number Timestamp (in epoch milliseconds) when the position was created.
data[].closedTime no number Timestamp (in epoch milliseconds) when the position was closed, applicable only for closed positions.

Get open orders

HTTP Request

GET /api/v1/user/orders?market={market}&type={type}&side={side}

Get all open orders for the authenticated sub-account. Optionally, the request can be filtered by a specific market or order type (limit, conditional, or tpsl).

Open orders correspond to the following order statuses from the list below: new, partially filled, untriggered.

To request data for several markets, use the following format: GET /api/v1/user/orders?market=market1&market2.

Order statuses

Status Description
NEW Order in the order book, unfilled.
PARTIALLY_FILLED Order in the order book, partially filled.
FILLED Order fully filled.
UNTRIGGERED Conditional order waiting for the trigger price.
CANCELLED Order cancelled.
REJECTED Order rejected.
EXPIRED Order expired.
TRIGGERED Technical status, transition from UNTRIGGERED to NEW.

Order status reasons (when cancelled or rejected)

Reason Description
NONE Order was accepted.
UNKNOWN Technical status reason.
UNKNOWN_MARKET Market does not exist.
DISABLED_MARKET Market is not active.
NOT_ENOUGH_FUNDS Insufficient balance to create order.
NO_LIQUIDITY Not enough liquidity in the market to execute the order.
INVALID_FEE Fee specified in the create order request is invalid.
INVALID_QTY Quantity specified is invalid.
INVALID_PRICE Price specified is invalid.
INVALID_VALUE Order exceeds the maximum value.
UNKNOWN_ACCOUNT Account does not exist.
SELF_TRADE_PROTECTION Order cancelled to prevent self-trading.
POST_ONLY_FAILED Order could not be posted as a post-only order.
REDUCE_ONLY_FAILED Reduce-only order failed due to position size conflict.
INVALID_EXPIRE_TIME Expiration time specified is invalid.
POSITION_TPSL_CONFLICT TPSL order for the entire position already exists.
INVALID_LEVERAGE Leverage specified is invalid.
PREV_ORDER_NOT_FOUND The order to be replaced does not exist.
PREV_ORDER_TRIGGERED The order to be replaced has been triggered and cannot be replaced.

Query Parameters

Parameter Required Type Description
market no string List of names of the requested markets.
type no string Order type. Can be limit, conditional or tpsl.
side no string Order side. Can be buy or sell.

Response example:

{
  "status": "OK",
  "data": [
    {
      "id": 1775511783722512384,
      "accountId": 3017,
      "externalId": "2554612759479898620327573136214120486511160383028978112799136270841501275076",
      "market": "ETH-USD",
      "type": "LIMIT",
      "side": "BUY",
      "status": "PARTIALLY_FILLED",
      "price": "3300",
      "averagePrice": "3297.00",
      "qty": "0.2",
      "filledQty": "0.1",
      "payedFee": "0.0120000000000000",
      "trigger": {
        "triggerPrice": "3300",
        "triggerPriceType": "LAST",
        "triggerPriceDirection": "UP",
        "executionPriceType": "MARKET"
      },
      "takeProfit": {
        "triggerPrice": "3500",
        "triggerPriceType": "LAST",
        "price": "3340",
        "priceType": "MARKET"
      },
      "stopLoss": {
        "triggerPrice": "2800",
        "triggerPriceType": "LAST",
        "price": "2660",
        "priceType": "MARKET"
      },
      "reduceOnly": false,
      "postOnly": false,
      "createdTime": 1701563440000,
      "updatedTime": 1701563440000,
      "timeInForce": "FOK",
      "expireTime": 1712754771819
    }
  ]
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data[].id yes number Order ID assigned by X10.
data[].externalId yes string Order ID assigned by user.
data[].accountId yes number Account ID.
data[].market yes string Market name.
data[].status yes string Order status.
data[].statusReason no string Reason for REJECTED or CANCELLED status.
data[].type yes string Order type. Can be limit, conditional or tpsl.
data[].side yes string Order side. Can be buy or sell.
data[].price no string Worst accepted price in the collateral asset.
data[].averagePrice no string Actual filled price, empty if not filled.
data[].qty yes string Order size in base asset.
data[].filledQty no string Actual filled quantity in base asset.
data[].payedFee no string Paid fee.
data[].reduceOnly no boolean Whether the order is Reduce-only.
data[].postOnly no boolean Whether the order is Post-only.
data[].trigger.triggerPrice no string Trigger price for conditional orders.
data[].trigger.triggerPriceType no string Trigger price type . Can be last, mark or index.
data[].trigger.triggerPriceDirection no string Indicates whether the order should be triggered when the price is above or below the set trigger price. It can be up (the order will be triggered when the price reaches or surpasses the set trigger price) or down (the order will be triggered when the price reaches or drops below the set trigger price).
data[].trigger.executionPriceType no string Execution price type. Can be limit or market.
data[].tpSlType no string TPSL type determining TPSL order size. Can be order or position.
data[].takeProfit.triggerPrice no string Take Profit Trigger price.
data[].takeProfit.triggerPriceType no string Take Profit Trigger price type. Can be last, mark or index.
data[].takeProfit.price no string Take Profit order price.
data[].takeProfit.priceType no string Indicates whether the Take profit order should be executed as market or limit order.
data[].stopLoss.triggerPrice no string Stop loss Trigger price.
data[].stopLoss.triggerPriceType no string Stop Loss Trigger price type. Can be last, mark or index.
data[].stopLoss.price no string Stop loss order price.
data[].stopLoss.priceType no string Indicates whether the Stop loss order should be executed as market or limit order.
data[].createdTime yes number Timestamp (in epoch milliseconds) of order creation.
data[].updatedTime yes number Timestamp (in epoch milliseconds) of order update.
data[].timeInForce yes string Time-in-force. Can be GTT (Good till time), FOK (Fill or kill) or IOC (Immediate or cancel).
data[].expireTime yes number Timestamp (in epoch milliseconds) when the order expires.

Get orders history

HTTP Request

GET /api/v1/user/orders/history?market={market}&type={type}&side={side}&id={id}&externalId={externalId}

Get orders history for the authenticated sub-account. Optionally, the request can be filtered by a specific market or order type (limit, market, conditional, or tpsl).

Orders history corresponds to the following order statuses from the list below: filled, cancelled, rejected, expired.

To request data for several markets, use the following format: GET /api/v1/user/orders/history?market=market1&market2.

The endpoint returns a maximum of 10,000 records; pagination should be used to access records beyond this limit. The records for non-filled orders are available only for the past 3 days.

Order statuses

Status Description
NEW Order in the order book, unfilled.
PARTIALLY_FILLED Order in the order book, partially filled.
FILLED Order fully filled.
UNTRIGGERED Conditional order waiting for the trigger price.
CANCELLED Order cancelled.
REJECTED Order rejected.
EXPIRED Order expired.
TRIGGERED Technical status, transition from UNTRIGGERED to NEW.

Query Parameters

Parameter Required Type Description
id no number List of internal Ids of the requested orders.
externalId no string[] List of external Ids of the requested orders.
market no string[] List of names of the requested markets.
type no string Order type. Can be limit, market, conditional or tpsl.
side no string Order side. Can be buy or sell.
cursor no number Determines the offset of the returned result. It represents the ID of the item after which you want to retrieve the next result. To get the next result page, you can use the cursor from the pagination section of the previous response.
limit no number Maximum number of items that should be returned.

Response example:

{
  "status": "OK",
  "data": [
    {
      "id": 1784963886257016832,
      "externalId": "ExtId-1",
      "accountId": 1,
      "market": "BTC-USD",
      "status": "FILLED",
      "type": "LIMIT",
      "side": "BUY",
      "price": "39000",
      "averagePrice": "39000",
      "qty": "0.2",
      "filledQty": "0.1",
      "payedFee": "0.0120000000000000",
      "reduceOnly": false,
      "postOnly": false,
      "trigger": {
        "triggerPrice": "34000",
        "triggerPriceType": "LAST",
        "triggerPriceDirection": "UP",
        "executionPriceType": "MARKET"
      },
      "tpslType": "ORDER",
      "takeProfit": {
        "triggerPrice": "34000",
        "triggerPriceType": "LAST",
        "price": "35000",
        "priceType": "MARKET",
        "starkExSignature": ""
      },
      "stopLoss": {
        "triggerPrice": "34000",
        "triggerPriceType": "LAST",
        "price": "35000",
        "priceType": "MARKET",
        "starkExSignature": ""
      },
      "createdTime": 1701563440000,
      "updatedTime": 1701563440000,
      "timeInForce": "FOK",
      "expireTime": 1706563440
    }
  ],
  "pagination": {
    "cursor": 1784963886257016832,
    "count": 1
  }
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data[].id yes number Order ID assigned by X10.
data[].externalId yes string Order ID assigned by user.
data[].accountId yes number Account ID.
data[].market yes string Market name.
data[].status yes string Order status.
data[].statusReason no string Reason for REJECTED or CANCELLED status.
data[].type yes string Order type. Can be limit, market, conditional or tpsl.
data[].side yes string Order side. Can be buy or sell.
data[].price no string Worst accepted price in the collateral asset.
data[].averagePrice no string Actual filled price, empty if not filled.
data[].qty yes string Order size in base asset.
data[].filledQty no string Actual filled quantity in base asset.
data[].payedFee no string Paid fee.
data[].reduceOnly no boolean Whether the order is Reduce-only.
data[].postOnly no boolean Whether the order is Post-only.
data[].trigger.triggerPrice no string Trigger price for conditional orders.
data[].trigger.triggerPriceType no string Trigger price type . Can be last, mark or index.
data[].trigger.triggerPriceDirection no string Indicates whether the order should be triggered when the price is above or below the set trigger price. It can be up (the order will be triggered when the price reaches or surpasses the set trigger price) or down (the order will be triggered when the price reaches or drops below the set trigger price).
data[].trigger.executionPriceType no string Execution price type. Can be limit or market.
data[].tpSlType no string TPSL type determining TPSL order size. Can be order or position.
data[].takeProfit.triggerPrice no string Take Profit Trigger price.
data[].takeProfit.triggerPriceType no string Take Profit Trigger price type. Can be last, mark or index.
data[].takeProfit.price no string Take Profit order price.
data[].takeProfit.priceType no string Indicates whether the Take profit order should be executed as market or limit order.
data[].stopLoss.triggerPrice no string Stop loss Trigger price.
data[].stopLoss.triggerPriceType no string Stop Loss Trigger price type. Can be last, mark or index.
data[].stopLoss.price no string Stop loss order price.
data[].stopLoss.priceType no string Indicates whether the Stop loss order should be executed as market or limit order.
data[].createdTime yes number Timestamp (in epoch milliseconds) of order creation.
data[].updatedTime yes number Timestamp (in epoch milliseconds) of order update.
data[].timeInForce yes string Time-in-force. Can be GTT (Good till time), FOK (Fill or kill) or IOC (Immediate or cancel).
data[].expireTime yes number Timestamp (in epoch milliseconds) when the order expires.

Get trades

HTTP Request

GET /api/v1/user/trades?market={market}&type={type}&side={side}

Get trades history for the authenticated sub-account. Optionally, the request can be filtered by a specific market, by trade type (trade, liquidation or deleverage) and side (buy or sell).

To request data for several markets, use the following format: GET /api/v1/user/trades?market=market1&market2.

The endpoint returns a maximum of 10,000 records; pagination should be used to access records beyond this limit.

Query Parameters

Parameter Required Type Description
market no string List of names of the requested markets.
type no string Trade type. Can be trade, liquidation or deleverage.
side no string Order side. Can be buy or sell.

Response example:

{
  "status": "OK",
  "data": [
    {
      "id": 1784963886257016832,
      "accountId": 3017,
      "market": "BTC-USD",
      "orderId": 9223372036854775808,
      "externalId": "ext-1",
      "side": "BUY",
      "price": "58853.4000000000000000",
      "qty": "0.0900000000000000",
      "value": "5296.8060000000000000",
      "fee": "0.0000000000000000",
      "tradeType": "DELEVERAGE",
      "createdTime": 1701563440000,
      "isTaker": true
    }
  ],
  "pagination": {
    "cursor": 1784963886257016832,
    "count": 1
  }
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data[].id yes number Trade ID assigned by X10.
data[].accountId yes number Account ID.
data[].market yes string Market name.
data[].orderId yes string Order ID assigned by X10.
data[].externalOrderId yes string Order ID assigned by user. Populated only on websocket stream.
data[].side yes string Order side. Can be buy or sell.
data[].averagePrice yes string Actual filled price.
data[].filledQty yes string Actual filled quantity in base asset.
data[].value yes string Actual filled absolute nominal value in collateral asset.
data[].fee yes string Paid fee.
data[].isTaker yes boolean Whether the trade was executed as a taker.
data[].tradeType yes string Trade type. Can be trade (for regular trades), liquidation (for liquidaton trades) or deleverage (for ADL trades).
data[].createdTime yes number Timestamp (in epoch milliseconds) when the trade happened.

Get funding payments

HTTP Request

GET /api/v1/user/funding/history?market={market}&side={side}&fromTime={fromTime}

Get funding payments history for the authenticated sub-account. Optionally, the request can be filtered by a specific market, by side (long or short) and from time as a start point.

To request data for several markets, use the following format: GET /api/v1/user/funding/history?market=market1&market2.

The endpoint returns a maximum of 10,000 records; pagination should be used to access records beyond this limit.

Query Parameters

Parameter Required Type Description
market no string List of names of the requested markets.
side no string Position side. Can be long or short.
fromTime yes number Starting timestamp (in epoch milliseconds).

Response example:

{
  "status": "OK",
  "data": [
    {
      "id": 8341,
      "accountId": 3137,
      "market": "BNB-USD",
      "positionId": 1821237954501148672,
      "side": "LONG",
      "size": "1.116",
      "value": "560.77401888",
      "markPrice": "502.48568",
      "fundingFee": "0",
      "fundingRate": "0",
      "paidTime": 1723147241346
    }
  ],
  "pagination": {
    "cursor": 8341,
    "count": 1
  }
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data[].id yes number Funding payment ID assigned by X10.
data[].accountId yes number Account ID.
data[].market yes string Market name.
data[].positionId yes number Position ID assigned by X10.
data[].side yes string Position side. Can be long or short.
data[].value yes string Position value at funding payment time.
data[].markPrice yes string Mark price at funding payment time
data[].fundingFee yes string Funding payment size.
data[].fundingRate yes string Funding rate.
data[].paidTime yes number Timestamp (in epoch milliseconds) when the funding payment happened.

Get current leverage

HTTP Request

GET /api/v1/user/leverage?market={market}

Get current leverage for the authenticated sub-account. You can get current leverage for all markets, a single market, or multiple specific markets.

To request data for several markets, use the format GET/api/v1/user/leverage?market=market1&market=market2.

Query Parameters

Parameter Required Type Description
market no string Name of the requested market.

Response example:

{
  "status": "OK",
  "data": [
    {
      "market": "SOL-USD",
      "leverage": "10"
    }
  ]
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data.market yes string Market name.
data.leverage yes string Current leverage.

Update leverage

HTTP Request

PATCH /api/v1/user/leverage

Update leverage for an individual market.

Modifying your leverage will impact your Available balance and Initial Margin requirements of your open position and orders in the market.

To adjust your leverage, you must meet two requirements: - The total value of your open position and triggered orders must remain below the maximum position value allowed for the selected leverage. - Your Available balance must be sufficient to cover the additional Margin requirements (if any) associated with the new leverage.

Failure to meet either of these criteria will result in an error.

For details on Margin requirements, please refer to the documentation.

Request example:

{
  "market": "BTC-USD",
  "leverage": "10"
}

Body Parameters

Parameter Required Type Description
market yes string Name of the requested market.
leverage yes string Target leverage.

Response example:

{
  "status": "OK",
  "data": {
    "market": "BTC-USD",
    "leverage": "10"
  }
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data.market yes string Market name.
data.leverage yes string Updated leverage.

Get fees

HTTP Request

GET /api/v1/user/fees?market={market}

Get current fees for the sub-account.

Fees on X10 are determined by:

  1. The type of order: Maker vs. Taker,

  2. The cumulative trading volume over the past 30 days.

For details on the Fee Schedule, please refer to the documentation.

Query Parameters

Parameter Required Type Description
market no string Name of the requested market.

Response example:

{
  "status": "OK",
  "data": [
    {
      "market": "BTC-USD",
      "makerFeeRate": "0.0002",
      "takerFeeRate": "0.0005"
    }
  ]
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data.market yes string Market name.
data.makerFeeRate yes string Maker fee rate.
data.takerFeeRate yes string Taker fee rate.

Order management

The Private API endpoints listed below allow you to create, cancel, and manage orders from the authenticated sub-account.

StarkEx-Specific Logic

X10's use of the StarkEx Layer 2 engine introduces unique elements to order creation that differ from centralized exchanges:

  1. StarkKey Signature: Required for all order management endpoints. For details, please refer to the reference implementation in the Python SDK.

  2. Price Parameter: All orders, including market orders, require a price as a mandatory parameter.

  3. Fee Parameter: All orders require a fee as a mandatory parameter. The Fee parameter represents the maximum fee a user is willing to pay for an order. Use the maker fee for Post-only orders and the taker fee for all other orders. Enter the fee in decimal format (e.g., 0.1 for 10%). To view current fees, utilize the Get fees endpoint, which displays applicable fee rates.

  4. Expiration Timestamp: All orders, including Fill or Kill and Immediate or Cancel orders, require an expiration timestamp as a mandatory parameter. When submitting orders via the API, enter the expiration time as an epoch timestamp in milliseconds. On the Mainnet, the maximum allowable expiration time is 90 days from the order creation date. On the Testnet, 28 days from the order creation date.

  5. Market Orders: StarkEx does not natively support market orders. On the UI, market orders are created as limit IOC (Immediate-or-Cancel) orders with a price parameter set to ensure immediate execution. For example, Market Buy Orders are set at the best ask price multiplied by 1.05, and Market Sell Orders at the best bid price multiplied by 0.95 (subtracting 5%).

  6. TPSL Orders: Orders with Take Profit and/or Stop Loss require multiple signatures. Please refer to the (documentation)[https://docs.x10.exchange/x10-resources/trading/order-types] to see supported types of TPSL orders.

Create or edit order

HTTP Request

POST /api/v1/user/order

Create a new order or edit (replace) an open order. When you create an order via our REST API, the initial response will confirm whether the order has been successfully accepted. Please be aware that, although rare, orders can be canceled or rejected by the Matching Engine even after acceptance at the REST API level. To receive real-time updates on your order status, it is crucial to subscribe to the Account updates WebSocket stream. This stream provides immediate notifications of any changes to your orders, including confirmations, cancellations, and rejections, allowing you to manage your trading strategies effectively.

Currently, we support limit, market, conditional and tpsl order types, along with reduce-only and post-only settings. For API trading, we offer the following Time-in-force settings: GTT (Good till time - default), FOK (Fill or kill) and IOC (Immediate or cancel). On the Mainnet, the maximum allowable expiration time for GTT orders is 90 days from the order creation date. On the Testnet, 28 days from the order creation date. For details on supported order types and settings, please refer to the documentation

To successfully place an order, it must meet the following requirements:

  1. Trading Rules. For detailed information, please refer to the trading rules documentation.

  2. Order Cost Requirements. For detailed information, please refer to the order cost documentation.

  3. Margin Schedule Requirements. For detailed information, please refer to the margin schedule documentation.

  4. Price requirements, which are described below.

Price requirements

  1. Limit Orders
    • Long Limit Orders: Order Price ≤ Mark Price * (1+Limit Order Price Cap)
    • Short Limit Orders: Order Price ≥ Mark Price * (1-Limit Order Floor Ratio)
  2. Market Orders
    • Long Market Order: Order Price ≤ Mark Price * (1 + 15%)
    • Short Market Order: Order Price ≥ Mark Price * (1 - 15%)
  3. Conditional Orders
    • Short Conditional Orders: Order Price ≥ Trigger price * (1-Limit Order Floor Ratio)
    • Long Conditional Orders: Order Price ≤ Trigger Price * (1+Limit Order Price Cap)
    • If a conditional order is triggered at placement, we apply the same validations as those for limit and market orders.
  4. TPSL Orders

Entry order: Buy; TPSL order: Sell.

Validation Stop loss Take profit
Trigger price validation Trigger price < Entry order price Trigger price > Entry order price.
Limit price validation Order Price ≥ Trigger price * (1-Limit Order Floor Ratio) Order Price ≥ Trigger price * (1-Limit Order Floor Ratio)

Entry order: Sell; TPSL order: Buy.

Validation Stop loss Take profit
Trigger price validation Trigger price > Entry order price. Trigger price < Entry order price.
Limit price validation Order Price ≤ Trigger Price * (1+Limit Order Price Cap) Order Price ≤ Trigger Price * (1+Limit Order Price Cap)

Orders Edit

To edit (replace) an open order, add its ID as the cancelId parameter. You can edit multiple parameters at once. Editing is available for all orders except for triggered TPSL orders.

Order editing and validations:

Editable Order Parameters:

Request

Request example:

{
  "id": "e581a9ca-c3a2-4318-9706-3f36a2b858d3",
  "market": "BTC-USDT",
  "type": "CONDITIONAL",
  "side": "BUY",
  "qty": "1",
  "price": "1000",
  "timeInForce": "GTT",
  "expiryEpochMillis": 1715884049245,
  "fee": "0.0002",
  "nonce": "876542",
  "settlement": {
    "signature": {
      "r": "0x17a89cb97c64f546d2dc9189e1ef73547487b228945dcda406cd0e4b8301bd3",
      "s": "0x385b65811a0fc92f109d5ebc30731efd158ee4e502945cd2fcb35a4947b045e"
    },
    "starkKey": "0x23830b00378d17755775b5a73a5967019222997eb2660c2dbfbc74877c2730f",
    "collateralPosition": "4272448241247734333"
  },
  "reduceOnly": true,
  "postOnly": false,
  "trigger": {
    "triggerPrice": "12000",
    "triggerPriceType": "LAST",
    "direction": "UP",
    "executionPriceType": "LIMIT"
  },
  "tpSlType": "ORDER",
  "takeProfit": {
    "triggerPrice": "1050",
    "triggerPriceType": "LAST",
    "price": "1300",
    "priceType": "LIMIT",
    "settlement": {
      "signature": {
        "r": "0x5b45f0fb2b8e075f6a5f9b4c039ccf1c01c56aa212c63f943337b920103c3a1",
        "s": "0x46133ab89d90a3ae2a3a7680d2a27e30fa015c0c4979931164c51b52b27758a"
      },
      "starkKey": "0x23830b00378d17755775b5a73a5967019222997eb2660c2dbfbc74877c2730f",
      "collateralPosition": "4272448241247734333"
    }
  },
  "stopLoss": {
    "triggerPrice": "950",
    "triggerPriceType": "LAST",
    "price": "900",
    "priceType": "LIMIT",
    "settlement": {
      "signature": {
        "r": "0x5033ad23fe851d16ceec5dd99f2f0c9585c5abec3f09ec89a32a961536ba55",
        "s": "0x1234ee151a8b5c68efb4adaa2eaf1dcc4a5107d4446274a69389ef8abd2dcf"
      },
      "starkKey": "0x23830b00378d17755775b5a73a5967019222997eb2660c2dbfbc74877c2730f",
      "collateralPosition": "4272448241247734333"
    }
  }
}

Body Parameters

Parameter Required Type Description
id yes string Order ID assigned by user.
market yes string Market name.
type yes string Order type. Can be limit, market, conditional or tpsl.
side yes string Order side. Can be buy or sell.
qty yes string Order size in base asset.
price yes string Worst accepted price in collateral asset. Note that price is optional for a tpsl type position.
reduceOnly no boolean Whether the order should be Reduce-only.
postOnly no boolean Whether the order should be Post-only.
timeInForce yes string Time-in-force setting. Can be GTT (Good till time), FOK (Fill or kill) or IOC (Immediate or cancel). This parameter will default to GTT.
expiryEpochMillis yes string Timestamp (in epoch milliseconds) when the order expires if not filled. Cannot exceed 28 days from the order creation time on Testnet.
fee yes string Highest accepted fee for the trade, expressed in decimal format (e.g., 0.1 for 10%). Use the maker fee for Post-only orders and the taker fee for all other orders.
cancelId no string External ID of the order that this order is replacing.
settlement yes object StarkKey signature, including nonce and signed order parameters. For details, please refer to the Python SDK reference implementation.
nonce yes string Nonce is part of the settlement and must be a number ≥1 and ≤2^31. Please make sure to check the Python SDK reference implementation.
trigger.triggerPrice no string Price threshold for triggering a conditional order.
trigger.triggerPriceType no string Type of price used for the order triggering. Can be last, mark or index.
trigger.triggerPriceDirection no string Indicates whether the order should be triggered when the price is above or below the set trigger price. It can be up (the order will be triggered when the price reaches or surpasses the set trigger price) or down (the order will be triggered when the price reaches or drops below the set trigger price).
trigger.executionPriceType no string Type of price used for the order execution. Can be limit or market.
tpSlType no string TPSL type determining TPSL order size. Can be order or position.
takeProfit.triggerPrice no string Take Profit Trigger price.
takeProfit.triggerPriceType no string Type of price used for the Take Profit order triggering. Can be last, mark or index.
takeProfit.price no string Take Profit order price.
takeProfit.priceType no string Indicates whether the Take profit order should be executed as market or limit order.
takeProfit.settlement no object StarkKey signature, including nonce and signed order parameters. For details, please refer to the Python SDK reference implementation.
triggerPrice no string Stop loss Trigger price.
stopLoss.triggerPriceType no string Type of price used for the Stop Loss order triggering. Can be last, mark or index.
stopLoss.price no string Stop loss order price.
stopLoss.priceType no string Indicates whether the Stop loss order should be executed as market or limit order.
stopLoss.settlement no object StarkKey signature, including nonce and signed order parameters. For details, please refer to the Python SDK reference implementation.

Response example:

{
  "status": "OK",
  "data": {
    "id": 1791389621914243072,
    "externalId": "31097979600959341921260192820644698907062844065707793749567497227004358262"
  }
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data.id yes number Order ID assigned by X10.
data.externalId yes string Order ID assigned by user.

Cancel order

HTTP Request

DELETE /api/v1/user/order/{id}

Cancel an individual order by X10 ID.

The cancellation process is asynchronous; the endpoint returns only the status of the cancellation.

URL Parameters

Parameter Required Type Description
id yes number Order to be canceled, ID assigned by X10.

DELETE /api/v1/user/order?externalId={externalId}

Cancel an individual order by user ID.

The cancellation process is asynchronous; the endpoint returns only the status of the cancellation.

URL Parameters

Parameter Required Type Description
externalId yes string Order to be canceled, Order ID assigned by user.

Response

Parameter Required Type Description
status yes string Can be ok or error.

Mass Cancel

HTTP Request

POST /api/v1/user/order/massCancel

Mass Cancel enables the cancellation of multiple orders by ID, by specific market, or for all orders within an account.

The cancellation process is asynchronous; the endpoint returns only the status of the cancellation request.

Although all parameters are optional, at least one must be specified.

Request example:

{
  "orderIds": [
    1,
    2
  ],
  "externalOrderIds": [
    "ExtId-1",
    "ExtId-2"
  ],
  "markets": [
    "BTC-USD",
    "ETH-USD"
  ],
  "cancelAll": true
}

Body Parameters

Parameter Required Type Description
markets no string[] Market names where all orders should be cancelled.
cancelAll no boolean Indicates whether all open orders for the account should be cancelled.
orderIds no number[] Cancel by X10 IDs.
externalOrderIds no string[] Cancel by external IDs.

Response

Parameter Required Type Description
status yes string Can be ok or error.

Mass Auto-Cancel (Dead Man's Switch)

HTTP Request

POST /api/v1/user/deadmanswitch?countdownTime={countdownTime}

The dead man's switch automatically cancels all open orders for the account at the end of the specified countdown if no Mass Auto-Cancel request is received within this timeframe. Setting the time to zero will remove any outstanding scheduled cancellations.

Positions and account status are not affected by the dead man's switch.

Request Parameters

Parameter Required Type Description
countdownTime yes number Time till Scheduled Mass Cancel (in seconds), should be non-negative. Setting the time to zero will remove any outstanding scheduled cancellations.

Response

Parameter Required Type Description
status yes string Can be ok or error.

Deposits

To deposit, please invoke the StarkEx contract. The contract address is 0x1cE5D7f52A8aBd23551e91248151CA5A13353C65.

Currently, we only support USDC deposits via the Ethereum network.

Create transfer

HTTP Request

POST /api/v1/user/transfer

Create a transfer between sub-accounts associated with the same L1 wallet.

Request

Request example:

{
  "fromAccount": 3004,
  "toAccount": 7349,
  "amount": "1000",
  "transferredAsset": "USD",
  "settlement": {
    "amount": 1000000000,
    "assetId": "0x31857064564ed0ff978e687456963cba09c2c6985d8f9300a1de4962fafa054",
    "expirationTimestamp": 478932,
    "nonce": 758978120,
    "receiverPositionId": 104350,
    "receiverPublicKey": "0x3895139a98a6168dc8b0db251bcd0e6dcf97fd1e96f7a87d9bd3f341753a844",
    "senderPositionId": 100005,
    "senderPublicKey": "0x3895139a98a6168dc8b0db251bcd0e6dcf97fd1e96f7a87d9bd3f341753a844",
    "signature": {
      "r": "6be1839e2ca76484a1a0fcaca9cbbe3792a23656d42ecee306c31e65aadb877",
      "s": "7b8f81258e16f0f90cd12f02e81427e54b4ebf7646e9b14b57f74c2cb44bff6"
    }
  }
}

Body Parameters

Parameter Required Type Description
fromAccount yes number Source account ID.
toAccount yes number Destination account ID.
amount yes string Transfer amount, absolute value in collateral asset.
transferredAsset yes string Collateral asset name.
settlement yes object Transfer object StarkKey signature (including nonce and transfer parameters). For details, please refer to the Python SDK.

Response example:

{
  "status": "OK",
  "data": {
    "validSignature": true,
    "id": 1820778187672010752
  }
}

Response

Parameter Required Type Description
status yes string Can be ok or error.
data.validSignature yes boolean Indicates whether the signature is valid.
data.id yes number Transfer ID assigned by X10.

Withdrawals

We support two types of withdrawals: fast and slow. Slow withdrawals can be made through both the API and UI, while fast withdrawals are available only via the UI. Withdrawals are permitted only to L1 wallets associated with the authorised account.

X10 doesn't charge fees on both types of withdrawals, but there are differences in processing.

Slow withdrawals:

Slow withdrawals involve a two-step process. The first step initiates your withdrawal request, which is processed on Layer 2. Once your funds are ready (this can take up to 12 hours), you can claim them to your wallet.

To initiate a slow withdrawal, send a "Create Slow Withdrawal" request as described below or use the corresponding SDK method slow_withdrawal, signed with a private L2 key.

To check if your funds are available for claim, use SDK method call_stark_perpetual_withdraw_balance.

To claim available funds, use SDK method call_stark_perpetual_withdraw, signed with a private L1 key.

Gas fee selection is currently not supported through the API. The gas value will be populated using the estimate_gas() function with an additional buffer of 100,000 gas up to the gasLimit of the latest block.

There is no maximum limit on the amount for slow withdrawals.

Fast withdrawals:

Processed almost instantly with a daily limit of $50,000 per calendar day.

A charge of 2x the current gas fee is applied to ensure transaction settlement in the event of gas cost volatility.

In rare instances, fast withdrawals may be unavailable due to liquidity gaps in the exchange's L1 wallet. If this occurs, please consider using slow withdrawals or try again later.

Fast withdrawals are not available for Market Makers and other institutional clients.

Create slow withdrawal

HTTP Request

POST /api/v1/user/withdrawal

Create a slow withdrawal to the L1 wallet associated with the authorised account. To avoid rejection, ensure that the withdrawal amount does not exceed your Available Balance for Withdrawals.

Available Balance for Withdrawals = max(0, Wallet Balance + min(0,Unrealised PnL) - Initial Margin Requirement).

Request

Request example: json { "type": "SLOW_SELF", "accountId": 3004, "amount": "1000", "asset": "USD", "settlement": { "amount": 1000000000, "collateralAssetId": "0x31857064564ed0ff978e687456963cba09c2c6985d8f9300a1de4962fafa054", "ethAddress": "0x1CE5161147db031d838F4BCDf86412C176d5a0D3", "expirationTimestamp": 479125, "nonce": 1696065254, "positionId": 100005, "publicKey": "0x3895139a98a6168dc8b0db251bcd0e6dcf97fd1e96f7a87d9bd3f341753a844", "signature": { "r": "34947b8a82c268d484dc5b879014a70acd6f071db2d80c356f47c5c56e8041d", "s": "37d5b153c786ba6b8b75a594e2e86f710404f0f045ac243a8db0002ff3aed27" } } }

Body Parameters

Parameter Required Type Description
type yes string For slow withdrawals, the type should be SLOW_SELF.
accountId yes number Source account ID.
amount yes string Withdrawal amount, absolute value in collateral asset.
asset yes string Collateral asset name.
settlement yes object Withdrawal object StarkKey signature (including nonce and withdrawal parameters). For details, please refer to the Python SDK.

Response example: json { "status": "OK", "data": 1820796462590083072 }

Response

Parameter Required Type Description
status yes string Can be ok or error.
data yes number Withdrawal ID, assigned by X10.

Public WebSocket streams

X10 offers a WebSocket API for streaming updates.

Connect to the WebSocket streams using wss://api.prod.x10.exchange as the host.

The server sends pings every 15 seconds and expects a pong response within 10 seconds. Although the server does not require pings from the client, it will respond with a pong if one is received.

Orderbooks stream

HTTP Request

GET /stream.x10.exchange/v1/orderbooks/{market}

Subscribe to the orderbooks stream for a specific market or for all available markets. If the market parameter is not submitted, the stream will include data for all available markets.

The initial response from the stream will be a snapshot of the order book. Subsequent updates occur every minute. All updates after the initial snapshot are delivered in delta format, reflecting only changes since the last update.

In the current version we support the following depth specifications:

URL Parameters

Parameter Required Type Description
market no string Select an individual market. If not specified, the subscription includes all markets.

Query Parameters

Parameter Required Type Description
depth no string Specify '1' to receive updates for best bid & ask only.

Response example:

{
  "ts": 1701563440,
  "type": "SNAPSHOT",
  "data": {
    "m": "BTC-USD",
    "b": [
      {
        "p": "25670",
        "q": "0.1"
      }
    ],
    "a": [
      {
        "p": "25770",
        "q": "0.1"
      }
    ]
  },
  "seq": 1
}

Response

Parameter Type Description
type string Type of message. Can be snapshot or delta.
ts number Timestamp (in epoch milliseconds) when the system generated the data.
data.m string Market name.
data.b object[] List of bid orders. For a snapshot, bids are sorted by price in descending order.
data.b[].p string Bid price.
data.b[].q string Bid size. For a snapshot, this represents the absolute size; for a delta, the change in size.
data.a object[] List of ask orders. For a snapshot, asks are sorted by price in ascending order.
data.a[].p string Ask price.
data.a[].q string Ask size. For a snapshot, this represents the absolute size; for a delta, the change in size.
seq number Monothonic sequence number. '1' corresponds to the first snapshot, and all subsequent numbers correspond to deltas. If a client receives a sequence out of order, it should reconnect.

Trades stream

HTTP Request

GET /stream.x10.exchange/v1/publicTrades/{market}

Subscribe to the trades stream for a specific market or for all available markets. If the market parameter is not submitted, the stream will include data for all available markets.

Historical trades data is currently available only for an authorised account through the Private REST-API.

URL Parameters

Parameter Required Type Description
market no string Select an individual market. If not specified, the subscription includes all markets.

Response example:

{
  "ts": 1701563440,
  "data": [
    {
      "m": "BTC-USD",
      "S": "BUY",
      "tT": "TRADE",
      "T": 1701563440,
      "p": "25670",
      "q": "0.1",
      "i": 25124
    }
  ],
  "seq": 2
}

Response

Parameter Type Description
ts number Timestamp (in epoch milliseconds) when the system generated the data.
data[].m string Market name.
data[].S string Side of taker trades. Can be buy or sell.
data[].tT string Trade type. Can be trade, liquidation or deleverage.
data[].T number Timestamp (in epoch milliseconds) when the trade happened.
data[].p string Trade price.
data[].q string Trade quantity in base asset.
data[].i number Trade ID.
seq number Monotonic sequence: Since there are no deltas, clients can skip trades that arrive out of sequence.

Funding rates stream

HTTP Request

GET /stream.x10.exchange/v1/funding/{market}

Subscribe to the funding rates stream for a specific market or for all available markets. If the market parameter is not submitted, the stream will include data for all available markets.

For historical funding rates data, use the Get funding rates history endpoint.

While the funding rate is calculated every minute, it is applied only once per hour. The records include only those funding rates that were used for funding fee payments.

URL Parameters

Parameter Required Type Description
market no string Select an individual market. If not specified, the subscription includes all markets.

Response example:

{
  "ts": 1701563440,
  "data": [
    {
      "m": "BTC-USD",
      "T": 1701563440,
      "f": "0.001"
    }
  ],
  "seq": 2
}

Response

Parameter Type Description
ts number Timestamp (in epoch milliseconds) when the system generated the data.
data[].m string Market name.
data[].T number Timestamp (in epoch milliseconds) when the funding rate was calculated and applied.
data[].f string Funding rates that were applied for funding fee payments.
seq number Monotonic sequence: Since there are no deltas, clients can skip funding rates that arrive out of sequence.

Candles stream

HTTP Request

GET /stream.x10.exchange/v1/candles/{market}/{candleType}?interval={interval}

Subscribe to the candles stream for a specific market or for all available markets. If the market parameter is not submitted, the stream will include data for all available markets.

The interval parameter should be specified in the ISO 8601 duration format. Available intervals include:

Trades price response example:

{
  "ts": 1695738675123,
  "data": [ 
    {
      "T": 1695738674000,
      "o": "1000.0000",
      "l": "800.0000",
      "h": "2400.0000",
      "c": "2100.0000",
      "v": "10.0000"
    }
  ],
  "seq": 1
}

Mark and Index price response example:

{
  "ts": 1695738675123,
  "data": [
    {
      "T": 1695738674000,
      "o": "1000.0000",
      "l": "800.0000",
      "h": "2400.0000",
      "c": "2100.0000"
    }
  ],
  "seq": 1
}

Available price types include:

  1. Trades price: GET /stream.x10.exchange/v1/candles/{market}/trades?interval=PT1M

  2. Mark price: GET /stream.x10.exchange/v1/candles/{market}/mark-prices?interval=PT1M

  3. Index price: GET /stream.x10.exchange/v1/candles/{market}/index-prices?interval=PT1M

Push frequency: 1-10s.

URL Parameters

Parameter Required Type Description
market no string Select an individual market. If not specified, the subscription includes all markets.
candleType yes string Price type. Can be trades, mark-prices or index-prices.

Query Parameters

Parameter Required Type Description
interval yes string Duration of candle (duration in ISO 8601).

Response

Parameter Type Description
ts number Timestamp (in epoch milliseconds) when the system generated the data.
data[].T number Starting timestamp (in epoch milliseconds) of the candle.
data[].o string Open price.
data[].c string Close price.
data[].h string Highest price.
data[].l string Lowest price.
data[].v string Trading volume (only for trade candles).
seq number Monothonic sequence number. '1' corresponds to the first snapshot, and all subsequent numbers correspond to deltas. If a client receives a sequence out of order, it should reconnect.

Private WebSocket streams

Connect to the WebSocket streams using ws://api.prod.x10.exchange as the host.

The server sends pings every 15 seconds and expects a pong response within 10 seconds. Although the server does not require pings from clients, it will respond with a pong if it receives one.

Account updates stream

HTTP Request

GET /stream.x10.exchange/v1/account

Orders updates response example:

{
  "type": "ORDER",
  "data": {
    "orders": [
      {
        "id": 1791181340771614723,
        "accountId": 1791181340771614721,
        "externalId": "-1771812132822291885",
        "market": "BTC-USD",
        "type": "LIMIT",
        "side": "BUY",
        "status": "NEW",
        "price": "12400.000000",
        "averagePrice": "13140.000000",
        "qty": "10.000000",
        "filledQty": "3.513000",
        "payedFee": "0.513000",
        "trigger": {
          "triggerPrice": "1220.00000",
          "triggerPriceType": "LAST",
          "direction": "UP",
          "executionPriceType": "LIMIT"
        },
        "tpSlType": "ORDER",
        "takeProfit": {
          "triggerPrice": "1.00000",
          "triggerPriceType": "LAST",
          "price": "1.00000",
          "priceType": "LIMIT"
        },
        "stopLoss": {
          "triggerPrice": "1.00000",
          "triggerPriceType": "LAST",
          "price": "1.00000",
          "priceType": "LIMIT"
        },
        "reduceOnly": true,
        "postOnly": false,
        "createdTime": 1715885888571,
        "updatedTime": 1715885888571,
        "expireTime": 1715885888571
      }
    ]
  },
  "ts": 1715885884837,
  "seq": 1
}

Trades updates response example:

{
  "type": "TRADE",
  "data": {
    "trades": [
      {
        "id": 1784963886257016832,
        "accountId": 3017,
        "market": "BTC-USD",
        "orderId": 9223372036854775808,
        "externalOrderId": "ext-1",
        "side": "BUY",
        "price": "58853.4000000000000000",
        "qty": "0.0900000000000000",
        "value": "5296.8060000000000000",
        "fee": "0.0000000000000000",
        "tradeType": "DELEVERAGE",
        "createdTime": 1701563440000,
        "isTaker": true
      }
    ]
  },
  "ts": 1715885884837,
  "seq": 1
}

Account balance updates response example:

{
  "type": "BALANCE",
  "data": {
    "balance": {
      "collateralName": "BTC",
      "balance": "100.000000",
      "equity": "20.000000",
      "availableForTrade": "3.000000",
      "availableForWithdrawal": "4.000000",
      "unrealisedPnl": "1.000000",
      "initialMargin": "0.140000",
      "marginRatio": "1.500000",
      "updatedTime": 1699976104901,
      "exposure": "12751.859629",
      "leverage": "1275.1860"
    }
  },
  "ts": 1715885952304,
  "seq": 1
}

Positions updates response example:

{
  "type": "POSITION",
  "data": {
    "positions": [
      {
        "id": 1791183357858545669,
        "accountId": 1791183357858545665,
        "market": "BTC-USD",
        "side": "SHORT",
        "leverage": "5.0",
        "size": "0.3",
        "value": "12751.8596295830",
        "openPrice": "42508.00",
        "markPrice": "42506.1987652769",
        "liquidationPrice": "75816.37",
        "margin": "637.59",
        "unrealisedPnl": "100.000000",
        "realisedPnl": "200.000000",
        "tpTriggerPrice": "1600.0000",
        "tpLimitPrice": "1500.0000",
        "slTriggerPrice": "1300.0000",
        "slLimitPrice": "1250.0000",
        "adl": 1,
        "createdAt": 1715886365748,
        "updatedAt": 1715886365748
      }
    ]
  },
  "ts": 1715886365748,
  "seq": 1
}

Subscribe to the account updates stream.

The initial responses will include comprehensive information about the account, including balance, open positions, and open orders, i.e. everything from GET /v1/user/balance, GET /v1/user/positions, GET /v1/user/orders.

Subsequent responses will contain all updates related to open orders, trades, account balance or open positions in a single message.

The response attributes will align with the responses from the corresponding REST API endpoints: Get trades, Get positions, Get open orders and Get balance. Refer to the Account section for details.

Market Makers Program

We provide a level playing field for all traders on X10, and we are introducing a maker's rewards program available to all traders without specific requirements. Details of the program and the application process are described here.

Market Makers are also eligible for higher Rate Limits. Currently, the Rate Limit for Market Makers is set at 6,000 requests per minute, shared across all endpoints. As our system grows, we plan to increase these limits.

Error responses

Unless specified otherwise for a particular endpoint and HTTP status code, the error response model follows the general response format and includes an error code along with a descriptive message for most errors.

Error code Error Description
GENERAL
400 BadRequest Invalid or missing parameters.
401 Unauthorized Authentication failure.
403 Forbidden Access denied.
404 NotFound Resource not found.
422 UnprocessableEntity Request format is correct, but data is invalid.
500 InternalServerError Internal server error.
MARKET, ASSET & ACCOUNT
1000 AssetNotFound Asset not found.
1001 MarketNotFound Market not found.
1002 MarketDisabled Market is disabled.
1003 MarketGroupNotFound Market group not found.
1004 AccountNotFound Account not found.
1005 NotSupportedInterval Not supported interval.
ORDER
1120 OrderQtyLessThanMinTradeSize Order quantity less than min trade size, based on market-specific trading rules.
1121 InvalidQtyWrongSizeIncrement Invalid quantity due to the wrong size increment, based on market-specific Minimum Change in Trade Size trading rule.
1122 OrderValueExceedsMaxOrderValue Order value exceeds max order value, based on market-specific trading rules.
1123 InvalidQtyPrecision Invalid quantity precision, currently equals to market-specific Minimum Change in Trade Size.
1124 InvalidPriceWrongPriceMovement Invalid price due to wrong price movement, based on market-specific Minimum Price Change trading rule.
1125 InvalidPricePrecision Invalid price precision, currently equals to market-specific Minimum Price Change.
1126 MaxOpenOrdersNumberExceeded Max open orders number exceeded, currently 200 orders per market.
1127 MaxPositionValueExceeded Max position value exceeded, based on the Margin schedule.
1128 InvalidTradingFees Trading fees are invalid. Refer to Order management section for details.
1129 InvalidPositionTpslQty Invalid quantity for position TPSL.
1130 MissingOrderPrice Order price is missing.
1131 MissingTpslTrigger TPSL order trigger is missing.
1132 NotAllowedOrderType Order type is not allowed.
1133 InvalidOrderParameters Invalid order parameters.
1134 DuplicateOrder Duplicate Order.
1135 InvalidOrderExpiration Order expiration date must be within 90 days for the Mainnet, 28 days for the Testnet.
1136 ReduceOnlyOrderSize ExceedsPositionSize Reduce-only order size exceeds open position size.
1137 ReduceOnlyOrder PositionIsMissing Position is missing for a reduce-only order.
1138 ReduceOnlyOrder PositionSameSide Position is the same side as a reduce-only order.
1139 MarketOrderMustBeIOC Market order must have time in force IOC.
1140 OrderCostExceedsBalance New order cost exceeds available balance.
1141 InvalidPriceAmount Invalid price value.
1142 EditOrderNotFound Edit order not found.
1143 MissingConditionalTrigger Conditional order trigger is missing.
1144 PostOnlyCantBeOn ConditionalMarketOrder Conditional market order can't be Post-only.