v5 (LATEST)
Devnet
Testnet
Mainnet
Coreum DEX API server
The API server provides a RESTFUL API for the Coreum DEX. The goal of the API server is to flatten the learning curve for the Coreum DEX (and with that of the Cosmos SDK) and to provide a simple and easy to use API for developers to interact with the Coreum DEX. The only part where there is some Cosmos knowledge required is where the developer has to interact with a wallet application to sign the transaction.
The API design principles used are:
- Reduce/eliminate blockchain required knowledge as much as possible for a frontend/business developer
- Remove required calculations to work with blockchain denomination notations
- Reduce business rules related code in the frontend.
This means that:
- Numeric values are published in the internal representation and as human-readable values: HumanReadablePrice, SymbolAmount
- Submission of values into the order/create endpoint is in human-readable values (e.g., 0.25 for a price instead of 25000 (Depending on precision: The developer does not need to know how to work with precision))
- Submission into endpoints like order/create check all the relevant business rules so that the frontend dev can be concerned with UX/UI instead of what order works or not
- The frontend uses a single javascript package to communicate with the backend, thus abstracting the communication between this API and the frontend, making it easier to implement a personalized DEX
The goal is that by using this API, any developer can launch a Coreum DEX with a lower investment and commoditize the DEX market.
Requirement for running the API
The API depends on the store to be running. The data in the store is populated by the data-aggregator, which is also present in this repository (see /apps/data-aggregator).
Contents of the API
The following is provided:
- OHLC data
- Trade history & currencies
- Order book
- Update service using a websocket for real-time updates
- Endpoint for transaction code generation
- Endpoint for order submission
In the following paragraphs, the endpoints are described in more detail.
Routes
The API server provides the following routes:
- GET /api/ohlc : Returns OHLC data
- GET /api/trades : Returns trade data (filterable)
- GET /api/tickers?symbols=base64encoded : Returns the latest price for one or more trading pairs
- GET /api/currencies : Returns the currencies
- GET /api/market : Returns the market data (provides information for trade tick size)
- GET /api/ws : Websocket for real-time updates
- POST /api/order/create : Create an order
- POST /api/order/cancel : Cancel an order
- POST /api/order/submit : Submit an order
- GET /api/order/orderbook : Returns the order book
- GET /api/wallet/assets : Returns the assets of a wallet
GET /ohlc
Params:
- symbolrequired - symbol for which OHLC should be returned. NOTE:- symbolshould be urlsafe encoded.
- periodrequired - one of- ["1m","3m","5m","15m","30m","1h","3h","6h","12h","1d","3d","1w"]
- fromrequired - unix timestamp of OHLC start
- torequired - unix timestamp of OHLC end
NOTE: API has limitation of 2000 points per request
It means that you can request 2000 of 1m points (2000/60/24) ~1.3 days. Or 2000 of 1h points ~83.3 days etc.
Returns:
[
  [
    1611069600, // timestamp
    "5.54016620498615", // open
    "5.54016620498615", // high
    "5.54016620498615", // low
    "5.54016620498615", // close
    "1.9855", // volume
  ],
  [
    1611069660,
    "5.54016620498615",
    "5.54016620498615",
    "5.54016620498615",
    "5.54016620498615",
    "0",
  ],
]
Example
curl "https://coredex.test.coreum.dev/api/ohlc?symbol=dextestdenom9-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs_dextestdenom1-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs&period=1m&from=1611007200&to=1611070980" \
--header "Network: devnet"
Get /trades
Params:
- symbolrequired - symbol for which trades should be returned. NOTE:- symbolshould be urlsafe encoded.
- fromoptional - unix timestamp of trades start. If not set, the last 50 trades are returned.
- tooptional - unix timestamp of trades end. Only optional if- fromis not set.
- accountoptional - account address for which trades should be returned
- sideoptional - side of the trade (1 - buy, 2 - sell (see enum domain/order-properties))
Returns:
{
  "Trades": [
    {
      "Account": "devcore1fpdgztw4aepgy8vezs9hx27yqua4fpewygdspc",
      "OrderID": "8b341e25-482e-487f-b9e2-9467d98c16ac",
      "Sequence": 27388,
      "Amount": {
        "Value": 2080,
        "Exp": -6,
      },
      "Price": 35.015385,
      "Denom1": {
        "Currency": "dextestdenom8",
        "Issuer": "devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
        "Precision": 6,
        "Denom": "dextestdenom8-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
      },
      "Denom2": {
        "Currency": "dextestdenom3",
        "Issuer": "devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
        "Precision": 6,
        "Denom": "dextestdenom3-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
      },
      "Side": 1,
      "BlockTime": {
        "seconds": 1736358800,
        "nanos": 634506142,
      },
      "MetaData": {
        "Network": 3,
        "UpdatedAt": {
          "seconds": 1737747689,
          "nanos": 498590000,
        },
        "CreatedAt": {
          "seconds": 1737747689,
          "nanos": 498029000,
        },
      },
      "TXID": "29E2362BE19BE53B5A38CFAAB4B777484F5956972C656A4378D7620A6E8F4A36",
      "BlockHeight": 6714462,
      "HumanReadablePrice": "35.015385",
      "SymbolAmount": "0.002080",
      "Status": 3,
    },
    //...
  ]
}
For retrieving the exchange history, the dev should only retrieve one side: If both sides are retrieved the list contains duplicates (party and counter party) of the trades, which would be confusing the end users.
tickers
The tickers endpoint returns the latest price for all trading pairs.
Params:
- symbolsmandatory - a base64 encoded list of symbols for which the ticker should be returned.
Maximum 20 symbols. Watch out for overflow of the URL in certain browsers: The symbol strings are quite long, so most likely the queries should be limited to 10 symbols or even less
The content to be base64 encode is a JSON array: ["USD-..._BTC-...].
Example call:
curl -H "Network: devnet" \
-X "GET" "https://coredex.test.coreum.dev/api/tickers?symbols=WyJVU0QrckQ5VzdVTHZlYXZ6OHFCR00xUjVqTWdLMlFLc0VEUFFWaS9YUlAiLCAiWFJQL1VTRCtyRDlXN1VMdmVhdno4cUJHTTFSNWpNZ0syUUtzRURQUVZpIl0="
Returns:
{
  "Tickers": {
    "dextestdenom9-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs_dextestdenom1-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs": {
      "OpenTime": 1723084712,
      "CloseTime": 1723171112,
      "OpenPrice": 0.17726004,
      "HighPrice": 3135.5825912454534,
      "LowPrice": 0.1730303609235772,
      "LastPrice": 0.1783099502981715,
      "FirstPrice": 0.1730303609235772,
      "Volume": 170639.98819271981,
      "InvertedVolume": 30319.373563999994,
    }
  },
  "USDTickers": {
    "dextestdenom9-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs_dextestdenom1-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs": {
      "OpenTime": 1723084712,
      "CloseTime": 1723171112,
      "OpenPrice": 1.24082028,
      "HighPrice": 21949.078138718174,
      "LowPrice": 1.2112125264650404,
      "LastPrice": 1.2481696520872005,
      "FirstPrice": 1.2112125264650404,
      "Volume": 24377.141170388546,
      "InvertedVolume": 212235.61494799994,
    }
  }
}
Currencies
Retrieve the currencies. Returns a list of currencies.
{
  "Currencies": [
    {
      "Denom": {
        "Currency": "ATOM",
        "Issuer": "cosmoshub-4",
        "Precision": 6,
        "IsIBC": false,
        "Denom": "uatom",
        "Name": "Cosmos",
        "Description": "Cosmos Hub native token",
        "Icon": "https://cosmos.network/logo.png",
      },
      "SendCommission": {
        "Value": 1,
        "Exp": -2,
      },
      "BurnRate": {
        "Value": 1,
        "Exp": -3,
      },
      "InitialAmount": {
        "Value": 1000000,
        "Exp": 0,
      },
      "Chain": "cosmoshub-4",
      "OriginChain": "cosmoshub-4",
      "ChainSupply": "100000000",
      "Description": "Cosmos Hub native token",
      "SkipDisplay": false,
      "MetaData": {
        "Network": 1,
        "UpdatedAt": "2023-10-01T12:00:00Z",
        "CreatedAt": "2020-01-01T00:00:00Z",
      },
    },
    //...
  ],
  "Offset": 0,
}
Example call:
curl -H "Network: devnet" \
-X "GET" "https://coredex.test.coreum.dev/api/currencies"
/market
Returns the market data for a certain symbol.
Params:
- symbolrequired - symbol for which market data should be returned. NOTE:- symbolshould be urlsafe encoded.
Returns:
{
    "Denom1": {
        "Currency": "alb",
        "Issuer": "devcore19p7572k4pj00szx36ehpnhs8z2gqls8ky3ne43",
        "Precision": 6,
        "Denom": "alb-devcore19p7572k4pj00szx36ehpnhs8z2gqls8ky3ne43"
    },
    "Denom2": {
        "Currency": "nor",
        "Issuer": "devcore19p7572k4pj00szx36ehpnhs8z2gqls8ky3ne43",
        "Precision": 6,
        "Denom": "nor-devcore19p7572k4pj00szx36ehpnhs8z2gqls8ky3ne43"
    },
    "MetaData": {
        "Network": 3,
        "UpdatedAt": {
            "seconds": 1746050943,
            "nanos": 326356000
        },
        "CreatedAt": {
            "seconds": 1746050943,
            "nanos": 325398000
        }
    },
    "PriceTick": {
        "Value": 1,
        "Exp": -6
    },
    "QuantityStep": 10000
}
/order/create
The order create uses a POST request to create an order. The function returns a to-be-signed transaction. The function does not persist the order in the order book (persistence is done only after submitting the order to the blockchain).
The call takes a POST:
{
  "Sender": "devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
  "Type": ORDER_TYPE_LIMIT, // Enum value
  "OrderID": "string", // Optional, can be an empty string
  "BaseDenom": "dextestdenom5-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
  "QuoteDenom": " dextestdenom9-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
  // price is value of one unit of the base_denom expressed in terms of the quote_denom.
  "Price": "0.25",
  "Quantity": "1000",
  "Side": SIDE_BUY, // Enum value
  "GoodTil": {
    "GoodTilBlockHeight": 1000,
    "GoodTilBlockTime": "2023-10-01T12:00:00Z"
  },
  "TimeInForce": TIME_IN_FORCE_GTC // Enum value
}
The response is a transaction that needs to be signed and submitted to the blockchain.
Below is a matrix of possible values for the order object before it can be successfully submitted using /order/submit.
Example call:
curl -H "Network: devnet" \
-X "POST" "https://coredex.test.coreum.dev/api/order/create" \
-d '{
    "Sender": "devcore1878pk82zlndhldglx26r606qcd886562mad59y",
    "Type": 1,
    "BaseDenom": "dextestdenom5-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
    "QuoteDenom": "dextestdenom9-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
    "Price": "0.25",
    "Quantity": "1000",
    "Side": 1,
    "GoodTil": {
        "GoodTilBlockTime": "2025-12-30T12:00:00Z"
    },
    "TimeInForce": 1
    }'
The response object is compatible with the javascript and the golang proto marshallers (which are different in the COSMOS SDK), and due to that contain some replicated fields. The response object is corrected for the precision and goes from human-readable to a bit harder to read in the format required for the DEX.
Example response:
{
  "Sequence":126378,
  "OrderData":{
    "sender": "devcore1878pk82zlndhldglx26r606qcd886562mad59y",
    "type": 1,
    "baseDenom": "dextestdenom5-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
    "quoteDenom": "dextestdenom9-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
    "price": "25e-2",
    "quantity": "1000000000",
    "side": 1,
    "goodTil": {
        "goodTilBlockTime": "2025-12-30T12:00:00Z"
    },
    "timeInForce": 1,
    "base_denom": "dextestdenom5-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
    "quote_denom": "dextestdenom9-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
    "time_in_force": 1,
    "good_til": {
        "good_til_block_time": "2025-12-30T12:00:00Z"
    }
  }
}
In which the TXBytes is the base64 encoded transaction that needs to be signed and submitted to the blockchain.
/order/cancel
The order cancel uses a POST request to cancel an order. The function returns a transaction hash.
Example call:
curl -H "Network: devnet" \
-X "POST" "https://coredex.test.coreum.dev/api/order/cancel" \
-d '{
    "Sender": "devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
    "OrderID": "8b341e25-482e-487f-b9e2-9467d98c16ac"
}'
The response object is:
{
  "Sequence":126378,
  "OrderCancel":{
    "sender": "devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
    "id": "8b341e25-482e-487f-b9e2-9467d98c16ac"
  }
}
The difference here is minimal. The cancel in this format is mainly present to have a consistent API interface.
/order/submit
The order submit uses a POST request to submit an order. The function returns a transaction hash.
Example call:
curl -H "Network: devnet" \
-X "POST" "https://coredex.test.coreum.dev/api/order/submit" \
-d '{
  "TX": "CqcCCqQCChwvY29zbW9zLmJhbmsudjFiZXRhMS5Nc2dTZW5kEoMCCvwBCvkBChwvY29yZXVtLmRleC52MS5Nc2dQbGFjZU9yZGVyEtgBCi5kZXZjb3JlMXAwZWR6eXpwYXpwdDY4dmRyankyMGM0Mmx2d3NqcHZmemFoeWdzEAEaBnN0cmluZyI8ZGV4dGVzdGRlbm9tNS1kZXZjb3JlMXAwZWR6eXpwYXpwdDY4dmRyankyMGM0Mmx2d3NqcHZmemFoeWdzKjxkZXh0ZXN0ZGVub205LWRldmNvcmUxcDBlZHp5enBhenB0Njh2ZHJqeTIwYzQybHZ3c2pwdmZ6YWh5Z3MyBTI1ZS0yOgoxMDAwMDAwMDAwQAFKCwjoBxIGCMCv9MYGEgISABJSCk4KRgofL2Nvc21vcy5jcnlwdG8uc2VjcDI1NmsxLlB1YktleRIjCiEDWIusTf5MdJVCJlqRR1MVNO5So6We2b5v42yCQl+k3D0SBAoCCAESABpAa+STFz77MqaUIU2bbS5uKtXeOyGW/m2qLcf96WEAJPJCSnO/3em/QZtGKnF1eg4ylCawYOH/8gpDLrxI1ybsiw=="
}
Where the TX is the signed TX in base64 encoding.
Response:
{
  "TXHash": "0x1234567890"
}
/order/orderbook
Params:
- symbolrequired - symbol for which order book should be returned. NOTE:- symbolshould be urlsafe encoded.
- accountoptional - account address for which the orderbook should be returned
Symbol is defined as denom1_denom2 where denom1 and denom2 are the denom strings of the two assets in the trading pair.
The order book returns the first 50 buys, and the first 50 sells around the spread. In this buy is defined as I want to buy this from you at this price and sell is defined as You want to sell this to you at this price
Sample return:
{
  "Buy": [
    {
      "Price": "3140",
      "HumanReadablePrice": "3140",
      "Amount": "2272",
      "SymbolAmount": "0.002272",
      "Sequence": 41567,
      "OrderID": "8b341e25-482e-487f-b9e2-9467d98c16ac",
      "RemainingAmount": "2272",
      "RemainingSymbolAmount": "0.002272",
    },
  ],
  "Sell": [
    {
      "Price": "3360",
      "HumanReadablePrice": "3360",
      "Amount": "2071",
      "SymbolAmount": "0.002071",
      "Sequence": 41760,
      "OrderID": "8b341e25-482e-487f-b9e2-9467d98c16ac",
      "RemainingAmount": "2071",
      "RemainingSymbolAmount": "0.002071",
    },
  ],
}
where a frontend could display (SymbolAmount-RemainingSymbolAmount)/SymbolAmount as an indicator of progress of the order.
/wallet/assets
Returns the assets for a certain account.
Params:
- addressrequired - the address of the account for which the assets should be returned.
Sample return:
[
  {
    "Denom": "dextestdenom0-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
    "Amount": "1000000000000",
    "SymbolAmount": "1000.0000",
  },
  {
    "Denom": "dextestdenom1-devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs",
    "Amount": "1000000000000",
    "SymbolAmount": "1000.0000",
  },
  //...
]
Example call:
curl -H "Network: devnet" \
-X "GET" "https://coredex.test.coreum.dev/api/wallet/assets?address=devcore1p0edzyzpazpt68vdrjy20c42lvwsjpvfzahygs"
Update service for real-time updates
The update service uses a websocket for a subscription system in which the user subscribes to certain information.
See the Update service for extensive information.
Application start parameters
The API server can be started with the following parameters:
- NETWORKS- The networks handled by the blockchain scanner.
- STATE_STORE- Location where the store is running, typically at grpc port 50051, syntax for non-google cloudrun usage:- {host}:50051
- TRADE_STORE- Location where the store is running, typically at grpc port 50051, syntax for non-google cloudrun usage:- {host}:50051
- OHLC_STORE- Location where the store is running, typically at grpc port 50051, syntax for non-google cloudrun usage:- {host}:50051
- ORDER_STORE- Location where the store is running, typically at grpc port 50051, syntax for non-google cloudrun usage:- {host}:50051
- CURRENCY_STORE- Location where the store is running, typically at grpc port 50051, syntax for non-google cloudrun usage:- {host}:50051
- LOG_LEVEL- info. Other values: debug, error, warn
- HTTP_CONFIG- HTTP configuration with CORS settings
- BASE_COIN- Native/system coin configuration
- BASE_USDC- USDC reference for resolving equivalent USD values
Networks
The networks are configured in a JSON array which depending on usage might require escaping. The network configuration is as follows:
{
  "Node": [
    {
      "Network": "devnet",
      "GRPCHost": "full-node.devnet-1.coreum.dev:9090",
      "RPCHost": "https://full-node.devnet-1.coreum.dev:26657"
    },
    {
      "Network": "testnet",
      "GRPCHost": "full-node.testnet-1.coreum.dev:9090",
      "RPCHost": "https://full-node.testnet-1.coreum.dev:26657"
    }
  ]
}
For production purposes, it is advised to use a private node, and the API server functions are proxy for that scenario.
HTTP configuration
The HTTP configuration is configured in a JSON array which depending on usage might require escaping. The HTTP configuration is as follows:
{
  "HTTP": {
    "port": ":8080",
    "cors": {
      "allowedOrigins": ["http://localhost:3000", "http://localhost:3001"]
    },
    "timeouts": {
      "read": "10s",
      "write": "10s",
      "idle": "10s",
      "shutdown": "10s"
    }
  }
}
The most often changed values are the cors.AllowOrigins and the port.
Base coin
The base coin is configured in a JSON array which depending on usage might require escaping. The base coin configuration is as follows:
{
  "BaseCoin": [
    {
      "Network": "devnet",
      "Coin": "ucore"
    },
    {
      "Network": "testnet",
      "Coin": "utestcore"
    },
    {
      "Network": "devnet",
      "Coin": "udevcore"
    }
  ]
}
Base USDC
Base USDC is used by a path resolving algorithm (non-weighted Dijkstra algorithm) to find the path between two assets and be able to resolve that to the representative USD value using a USDC as base. Often devnet and testnet will not have an IBC USDC currency available, resulting in a devnet configuration only. On devnet/testnet the user most likely will not see any USD values in the data. The base USDC is configured in a JSON array which depending on usage might require escaping. The base USDC configuration is as follows:
{
  "BaseCoin": [
    {
      "Network": "devnet",
      "Coin": "uusdc-E1E3674A0E4E1EF9C69646F9AF8D9497173821826074622D831BAB73CCB99A2D"
    }
  ]
}