Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
The table outlines the differences between Nexus and Lumina, which are both oracle solutions supported by DIA:
Phase
🟢 Production-ready
🟠 Beta Mainnet
Oracle Types
Push
Push & Pull
Architecture
Oracles built on highly scalable infrastructure. Enabling gas cost minimization while maintaining data flexibility.
Powered by a permissionless network of nodes that submit verifiable data on-chain, creating a system built on decentralization and trustlessness.
Data Feeds
Token price feed
RWA price feed
Randomness
Fair-value feed
Token price feed
Data sourcing
First-party providers
First-party provider
Frequency
App-specific
~ 30s
Security
Battle-tested over multiple market cycles and is currently deployed across 50 chains and 200 dapps
Enforces crypto-economic and cryptographic security
DIA xMarket feeds provide smart contracts real-time price information of 3,000+ cryptocurrencies, transparently sourced from 80+ trusted, high-volume DEXs and CEXs.
The feeds facilitate the development of DeFi use cases such as money markets, lending/borrowing, synthetic asset issuance, options, derivatives and futures markets, and many more.
Browse all tokens that DIA's data library currently supports via the link below. New assets can also be supported on demand.
You can also check the list of data sources on the app.
DIA Nexus is composed of three main stages, each critical to ensuring the accuracy, reliability, and accessibility of your data.
Data Sourcing
A mechanism that fetches granular market data (trade ticks) from a broad range of on-chain and off-chain exchanges, including CEXs and DEXs, through scrapers. These scrapers are integral in the creation of resilient price feeds for assets across all networks – there is no reliance on third-party premium data providers.
Data Computation
Nexus employs transparent computational methodologies to process raw trade data into reliable feeds. From outlier cleansing filters to sophisticated pricing techniques, Nexus' computation system can handle diverse requirements.
Data Delivery
A network of nodes takes on the responsibility of pushing oracle data on-chain, currently servicing all major L1/L2 blockchains.
DIA xReal is designed for Real-World Asset (RWA) applications, bringing hundreds of data feeds for stocks, commodities, FX rates, and more on-chain.
It consists of two core components:
xReal Price Feeds – Secure, transparent price oracles for RWA-powered dApps (options, futures, tokenized index funds & more)
xReal Custom Solutions – On-demand oracles for tokenized assets, bond yields, FX rates, & real-world financial metrics
You can access all the available RWA data feeds & integration options below:
Nexus is a flexible oracle solution built with dApps' cross-chain data needs in mind. It provides:
Customizable token price feeds sourced directly from the exchanges rather than relying on third-party providers
RWA price feeds for traditional financial assets like stocks, commodities, and more.
Verifiable random numbers through drand's distributed randomness beacon.
Transparent and collateral-proofed price feeds for illiquid assets such as LSTs.
Complex computation
More compute-intensive methodologies can be used. More trades are leveraged for the computation of the final output.
Economical to use
Few on-chain transactions and gas costs are utilised for the creation of the final oracle data.
Fast integration for VM
Oracles can be quickly deployed in any env. (WASM, EVM or other) thanks to its chain-agnostic architecture.
Off-chain/on-access paths
Besides oracles, data can be access off-chain via Rest APIs and GraphQL for more flexibility.
Fast - from source to feed
Trade data is quickly (<30s) processed into feeds. Requiered for certain on-chian use cases.
Historica data avail.
Oracles deliver historical data up to 1 month off-the-shelf. Longer windows can be set up on demand.
This guide shows you how to use the token price feeds within your smart contract and get the latest prices available to your dApp!
DIA xFair is a new oracle product that provides DeFi protocols with transparent and collateral-proofed price feeds for illiquid assets such as LSTs, computed through on-chain collateral ratio checks.
This system allows for more accurate and fair pricing, supporting various DeFi use cases like lending, borrowing, money markets, options, and helps access idle total value locked (TVL) in LSDs.
Browse all LST price feeds that DIA's data library currently supports:
If you're building on any of the , you can find the deployed oracle contract addresses and a code example in the dedicated chain guide .
DIA xRandom provides smart contracts with unpredictable random numbers. DIA leverages ’s distributed randomness beacon, enabling verifiable, unpredictable and unbiased random numbers. Currently, the randomness is updated every 30 seconds and is expected to upgrade soon.
This page provides a comprehensive list of all the blockchain networks where DIA Nexus is accessible.
Aleph Zero
Alephium
Arbitrum
Arbitrum Nova
Astar
Aurora
Avalanche
Base
BEVM
BNB Chain
Boba network
BOB Build on Bitcoin
Celo
Clover
Cross Finance (CrossFi)
Edu Chain
Ethereum
Evmos
Fantom
Fuse
Gnosis Chain
Goat Network
HydraDX
Kadena
Linea
LUKSO
Metis
Moonbase
Moonbeam
Moonriver
Near
Nervos
OKT Chain
Optimism
Polygon
Polygon zkEVM
Shiden
Shibuya
Shido
Somnia
Soroban
Stacks
Telos
Velas
Wanchain
XRPL
zkSync
There are two ways that you can access the oracle in your dApp. You can either declare an IDIAOracleV2 interface or import the solidity library.
Access any DIA oracle smart contract.
Call getValue(pair_name)
with pair_name
being the full pair name such as BTC/USD
. You can use the "Read" section on Etherscan to execute this call.
The response of the call contains four values:
The current asset's price in USD with a fix-comma notation of 8 decimals.
Find the detailed description of the functions and how to run tests on our GitHub page:
DIA has a dedicated Solidity library to facilitate the integration of DIA oracles in your own contracts. The library consists of two functions, getPrice
and getPriceIfNotOlderThan
.
Returns the price of a specified asset along with the update timestamp.
Parameters:
Return Values:
Checks if the oracle price is older than maxTimePassed
Parameters:
Return Values:
Find a detailed integration example and how to run a test on our GitHub page:
The following is an example of how to retrieve price value from a standard DIA oracle. For the purpose of this example, we will be using the following demo oracle on Ethereum: .
The of the last oracle update.
oracle
address
Address of the oracle that we want to use
key
string
The asset that we want to use e.g. "ETH/USD"
latestPrice
uint128
Price of the specified asset returned by the DIAOracle
timestampOflatestPrice
uint128
The update timestamp of the latest price
oracle
address
Address of the oracle that we want to use
key
string
The asset that we want to use e.g. "ETH/USD"
maxTimePassed
uint128
The maximum acceptable time passed in seconds since the the price was updated
Price
uint128
Price of the specified asset returned by the DIAOracle
inTime
uint128
A boolian that is true
if the price was updated at most maxTimePassed seconds ago, otherwise false
Arbitrum Nova
1h heartbeat with 0.5% deviation
All DIA demo oracles include the following assets price information, delivered in USD:
BTC/USD
USDC/USD
DIA/USD
Ethereum Sepolia
1h heartbeat with 0.5% deviation
Arbitrum Sepolia
1h heartbeat with 0.5% deviation
Astar Shibuya
1h heartbeat with 0.5% deviation
Avalanche Fuji
1h heartbeat with 0.5% deviation
Base Sepolia
1h heartbeat with 0.5% deviation
Linea Sepolia
1h heartbeat with 0.5% deviation
Optimism Sepolia
1h heartbeat with 0.5% deviation
Polygon Amoy
1h heartbeat with 0.5% deviation
Polygon zkEVM Cardona
1h heartbeat with 0.5% deviation
zkSync Sepolia
1h heartbeat with 0.5% deviation
dApps built on Aleph Zero can leverage DIA oracles to access up-to-date asset price information.
Mainnet
Pricing Methodology
Deviation (%) & Refresh Frequency
0.5% and 120 seconds
Heartbeat
24h
USDT
USDT/USD
USDC
USDC/USD
BTC
BTC/USD
ETH
ETH/USD
stETH
stETH/USD
AZERO
AZERO/USD
ARB
ARB/USD
Below is an example of a contract consuming data from our oracle on Aleph Zero mainnet. If you pass BTC/USD
as the key, it will return the most recent price of BTC in USD with 8 decimal places (e.g. 9601458065403 is $96,014.58065403) along with the Unix timestamp of the last price update.
The following are demo oracles built for testing purposes. For production-ready oracles, visit this .
is a cross-chain oracle provider that sources granular market data from diverse exchanges, including CEXs and DEXs. Its data sourcing is thorough, enabling unparalleled transparency and customizability for resilient price feeds for 20,000+ assets. Its versatile data processing and delivery ensures adaptability and reliability for any decentralized application.
To consume price data, you’ll need to invoke the getValue
method on the oracle contract which you can access through the or the .
See the full example .
For developer assistance, connect with the DIA team directly on or .
Developers seeking other specialized, production-grade oracle with tailored price feeds and configurations can initiate the request by .
dApps built on EduChain can leverage DIA oracles to access up-to-date asset price information.
Mainnet
Testnet
Pricing Methodology
Deviation (%) & Refresh Frequency
0.5% and 120 seconds
Heartbeat
24h
EDU
EDU/USD
ETH
ETH/USD
WBTC
WBTC/USD
USDC
USDC/USD
USDT
USDT/USD
DAI
DAI/USD
ARB
ARB/USD
wstETH
wstETH/USD
GRAIL
GRAIL/USD
Below is an example of a contract consuming data from the oracle on EduChain testnet using the IDIAOracleV2 interface. If you pass WBTC/USD
as the key, it will return the most recent price of BTC in USD with 8 decimal places (e.g. 8258705325665 is $82,587.05325665) along with the Unix timestamp of the last price update.
dApps built on Alephium can leverage DIA oracles to access up-to-date asset price information.
Mainnet
Testnet
Pricing Methodology
Deviation (%) & Refresh Frequency
0.2% and 120 seconds
Heartbeat
10mins
BTC
BTC/USD
USDC
USDC/USD
ETH
ETH/USD
WBTC
WBTC/USD
USDT
USDT/USD
ALPH
ALPH/USD
AYIN
AYIN/USD
Locate one of the deployed contracts (either testnet or mainnet) and call the getValue
function. This function expects one parameter, which is the symbols of the asset you want to retrieve and a “/USD”, so for example for the Bitcoin price the parameter must be “BTC/USD”.
This will return two values:
The price of the asset, with 8 decimals.
The timestamp of the last update in Unix time format, in UTC timezone.
The following is an example of how you can integrate DIA's oracle into your smart contract with Vyper:
Find the detailed description of the functions and how to run tests on our GitHub page
dApps built on Hydration can leverage DIA oracles to access up-to-date asset price information.
Mainnet
Testnet
0xc756bd338a97c1d2faab4f13b5444a08a1566917
Mainnet
Testnet
0x5d8320f3ced9575d8e25b6f437e610fc6a03bf52
Pricing Methodology
Deviation (%) & Refresh Frequency
0.5% and 120 seconds
Heartbeat
24h
WBTC
WETH
DOT
USDT
USDC
tBTC
Aave
vDOT
WBTC
0xC9cCBe99bdD9538871f9756Ca5Ea64C2267cb0a7
WETH
0xBd763043861CAF4E7e4E7Ffe951A03dF2Ea7E5AC
DOT
0x422E745797EC0Ef399c17cE3E2348394F2944727
USDT
0xb4aC9f0E6E207D5d81B756F8aF6efe3fe7B0E72c
USDC
0xEE7aFb45c094DC9fA404D6A86A7d795d4aA33D28
tBTC
0x9f7abE73e656DB45F56B1d3755940CB3fc82Fcaa
Aave
0xC9E1415Eb64281aD449f50C454873d92F312d6fE
vDOT
0x234F96059d628Da80B76A40c0E50a9D16a8F3191
Below is an example of a contract consuming data from the oracle on Hydration testnet using the IDIAOracleV2 interface. If you pass BTC/USD
as the key, it will return the most recent price of BTC in USD with 8 decimal places (e.g. 9601458065403 is $96,014.58065403) along with the Unix timestamp of the last price update.
is a cross-chain oracle provider that sources granular market data from diverse exchanges, including CEXs and DEXs. Its data sourcing is thorough, enabling unparalleled transparency and customizability for resilient price feeds for 20,000+ assets. Its versatile data processing and delivery ensures adaptability and reliability for any decentralized application.
To consume price data, you’ll need to invoke the getValue
method on the oracle contract which you can access through the or the .
See the full example .
For developer assistance, connect with the DIA team directly on or .
Developers seeking other specialized, production-grade oracle with tailored price feeds and configurations can initiate the request by .
is a cross-chain oracle provider that sources granular market data from diverse exchanges, including CEXs and DEXs. Its data sourcing is thorough, enabling unparalleled transparency and customizability for resilient price feeds for 20,000+ assets. Its versatile data processing and delivery ensures adaptability and reliability for any decentralized application.
Below is the DIAOracle
contract implementation in :
For developer assistance, connect with the DIA team directly on or .
Developers seeking other specialized, production-grade oracle with tailored price feeds and configurations can initiate the request by .
To consume price data, you’ll need to invoke the getValue
method on the oracle contract which you can access through the or the .
See the full example .
To consume price data from our oracle, you can use the adapter smart contract located at the for each asset. This will allow you to access the same methods on the AggregatorV3Interface
such as getRoundData
& latestRoundData
. You can learn more .
For developer assistance, connect with the DIA team directly on or .
Developers seeking other specialized, production-grade oracle with tailored price feeds and configurations can initiate the request by .
This guide shows you how to use the randomness oracle within your smart contract and get verifiable random numbers available to your dApp!
dApps built on Superseed can leverage DIA oracles to access up-to-date asset price information.
Mainnet
Testnet
Pricing Methodology
Deviation (%) & Refresh Frequency
0.5% and 120 seconds
Heartbeat
24h
WETH
cbBTC
WETH
cbBTC
Below is an example of a contract consuming data from our oracle on Sepolia testnet using the IDIAOracleV2 interface. If you pass WETH/USD
as the key, it will return the most recent price of WETH in USD with 8 decimal places (e.g. 177101990135 is $1,771.01990135) along with the Unix timestamp of the last price update.
If you're building on any of the , you can find the deployed oracle contract address and a code example in the dedicated chain guide.
is a cross-chain oracle provider that sources granular market data from diverse exchanges, including CEXs and DEXs. Its data sourcing is thorough, enabling unparalleled transparency and customizability for resilient price feeds for 20,000+ assets. Its versatile data processing and delivery ensures adaptability and reliability for any decentralized application.
To consume price data, you’ll need to invoke the getValue
method on the oracle contract which you can access through the or the .
See the full example .
To consume price data from our oracle, you can use the adapter smart contract located at the for each asset. This will allow you to access the same methods on the AggregatorV3Interface
such as getRoundData
& latestRoundData
. You can learn more .
For developer assistance, connect with the DIA team directly on or .
Developers seeking other specialized, production-grade oracle with tailored price feeds and configurations can initiate the request by .
dApps built on Somnia can leverage DIA oracles to access up-to-date asset price information.
is a cross-chain oracle provider that sources granular market data from diverse exchanges, including CEXs and DEXs. Its data sourcing is thorough, enabling unparalleled transparency and customizability for resilient price feeds for 20,000+ assets. Its versatile data processing and delivery ensures adaptability and reliability for any decentralized application.
To consume price data, you’ll need to invoke the getValue
method on the oracle contract which you can access through the or the . Below is an example of a contract consuming data from our oracle on Sepolia testnet. If you pass BTC/USD
as the key, it will return the most recent price of BTC in USD with 8 decimal places (e.g. 9601458065403 is $96,014.58065403) along with the Unix timestamp of the last price update.
See the full example .
To consume price data from our oracle, you can use the adapter smart contract located at the for each asset. This will allow you to access the same methods on the AggregatorV3Interface
such as getRoundData
& latestRoundData
. You can learn more .
For developer assistance, connect with the DIA team directly on or .
Developers seeking other specialized, production-grade oracle with tailored price feeds and configurations can initiate the request by .
The RandomOracle
contract in ink! is designed to store random values for different rounds. Each round has an associated RandomData
struct containing a randomness value, a signature, and the previous signature. The contract provides methods for setting random values and retrieving random values for a given round.
Clone the repository:
Install required tools and dependencies:
Build the contract:
Deploy contract using this UI
To use the RandomOracle
contract in your own project, follow these steps:
In your own project, add the random-wasm-oracle
as a dependency. Open your project's Cargo.toml and add the following:
In your project's smart contract, import the necessary modules and types from the random-wasm-oracle
crate:
In your contract's storage, add a RandomOracleRef field:
In your contract's constructor, add a parameter for the RandomOracle
contract's address and initialize the RandomOracleRef
:
Implement methods to interact with the RandomOracle
contract, e.g., getting random values for a specific round:
Build and deploy your contract, providing the RandomOracle
contract's address https://shibuya.subscan.io/account/Y9YVxsyH8bza5zyK1AVW4iw1r7twVdoXMoDzWwdwraapvSM
when instantiating your contract.
Interact with your contract to call the methods that use the RandomOracle
contract.
DIA uses the drand.love public randomness beacon, and updates its oracle with round number, randomness and signature. Anyone can access published random values via round ID.
The DIA randomness smart contract is structured as follows
Users can call getLastRound()
to obtain the ID of the latest published round. To obtain the randomness of a certain round, users can call getRandomValueFromRound(uint256 _round)
using the obtaines round ID.
The signature can also be requested by calling getRandomValueFromRoundWithSignature(uint256 _round)
.
Please be aware that you should always let all inputs commit before any randomness is used in a later round. For example, if you build a lottery, only call randomness after the last participant has committed their stake. To show this in an example, we will build a simple dice game.
Imagine a simple game where two players play against each other. Both throw a dice and the user with the higher rolled number wins. In case of a draw, nobody wins. Both players need to "seed" the game with an initial number between 1 and 6, that will be added to the randomness (modulo 6). It is a fair game where everyone has a 50% chance of winning.
Both players need to commit their seeds. The latest published roundID is stored for the duration of the game. When both committed their values and the wait time of 10 rounds has passed, the dice can be rolled and the player ID of the winner is returned (or 0 in case of a draw).
You can follow the guide for how to access the price feeds deployed on each chain in your dApp:
You can follow the guide for how to access the price feeds deployed on each chain in your dApp:
dApps built on LUKSO can leverage DIA oracles to access up-to-date asset price information.
The DIA oracle on LUKSO is free of use. dApps built on LUKSO can leverage the oracle to access up-to-date asset price information. This oracle is designed for production environments and comes with a predefined list of feeds and settings.
Developers can access specific asset prices via dedicated smart contract adapters listed in the next section. Each adapter contract provides price data by calling functions without additional parameters.
latestAnswer
- provides the latest available price data for BTC/USD with 8 decimals accuracy
latestRoundData
- provides the latest available price data for BTC/USD (8 decimals) with the Unix timestamp (UTC timezone) of the last update
Below is the list of included price feeds. Use the corresponding adapter address to query the latest price information.
EUR/USD
0xbCfD839664B5Ad4D0C0C58db0c716D7a28dCd15E
LYX/USD
0x3153e4d03Cf97B230fc9c9d0ECCE5b2F0834d130
GBP/USD
0x7EA5b4bb2D89B1EfEF00FCC95E139B77Da27d460
BTC/USD
0xD19856Bd3Dbd9Cb00BaC76Fc90603FA3bB05aCEA
ETH/USD
0xc7f3aceBe05482eeCD668df1FCF1B59e6d14f77b
wstETH/USD
0x18f2182B37d46deC1dc236Cf3Aad3a02981796cB
DAI/USD
0x1bB6c1A90c9449A14C184Dd141D2c7d7b788a679
USDC/USD
0xc675A8d220E1ba6d14a77C9FbcFDd5f9474BD07A
USDT/USD
0xC00c10499DEF24352475Ae873C51FaFcFC894f03
stETH/USD
0x2b94002cfFA638B37E4DDe54EDfcF6Efdcb29E6A
Each asset is updated every two minutes, if the new price deviates from the old one by more than 0.5%. Additionally, a heartbeat of 24 hours is applied, which means that each price is updated at least once per day even if it moves by less than 0.5%.
dApps built on Kadena can leverage DIA oracles to access up-to-date asset price information.
As Kadena allows for several chains in parallel, it is important to retrieve the price from the correct chain. Currently, DIA oracles are deployed on these chains:
mainnet01
1
n_bfb76eab37bf8c84359d6552a1d96a309e030b71.dia-oracle
mainnet01
2
n_bfb76eab37bf8c84359d6552a1d96a309e030b71.dia-oracle
mainnet01
4
n_bfb76eab37bf8c84359d6552a1d96a309e030b71.dia-oracle
testnet04
4
n_bfb76eab37bf8c84359d6552a1d96a309e030b71.dia-oracle
Locate one of the deployed contracts (either testnet or mainnet) and call the get-value
function. This function expects one parameter, which is the symbols of the asset you want to retrieve and a “/USD”, so for example for the Bitcoin price the parameter must be “BTC/USD”.
This will return two values:
The price of the asset, with 8 decimals.
The timestamp of the last update in Unix time format, in UTC timezone.
The Kadena oracle includes the following price feeds. Use the Query Symbol as a parameter in the get-value
function to receive the latest price information for any of these assets.
Asset
Query Symbol
Sources Overview
Bitcoin (BTC)
BTC/USD
USD Coin (USDC)
USDC/USD
Staked ETH (stETH)
STETH/USD
Ether (ETH)
ETH/USD
Wrapped Bitcoin (WBTC)
WBTC/USD
Moonbeam (GLMR)
GLMR/USD
Optimism (OP)
OP/USD
DIA Token (DIA)
DIA/USD
Arbitrum (ARB)
ARB/USD
Binance Coin (BNB)
BNB/USD
Avalanche (AVAX)
AVAX/USD
Celo Dollar (CUSD)
CUSD/USD
Kadena (KDA)
KDA/USD
Gnosis (GNO)
GNO/USD
Wrapped Ether (WETH)
WETH/USD
Each asset is updated every two minutes, if the new price deviates from the old one by more than 0.5%. Additionally, a heartbeat of 24 hours is applied, which means that each price is updated at least once per day even if it moves by less than 0.5%.
Oracles require a constant stream of updates, thus gas costs reimbursement for updates is needed. To achieve this, all oracles deployed by DIA have a dedicated wallet which calls oracle contracts for updates. This is referred to as the "gas wallet".
For the gas wallet to perform new oracle contract updates, it must have a sufficient balance of the blockchain's native token to cover the transaction's gas costs.
The gas wallet can be funded with a one-time lump sum, allowing it to keep updating for a while. However, monitoring the gas wallet's balance is necessary to ensure continuity in the oracle's operations.
DIA has set up internal gas wallet balance trackers. We recommend protocols to actively monitor them to ensure that the wallet balance remains sufficient for pushing new oracle updates.
DIA has a dedicated endpoint for asset historical updates count to help you estimate the number of oracle updates. This allows to foresee potential gas costs that would be required for running the oracle.
GET
https://api.diadata.org/v1/assetUpdates/:blockchain/:address/:deviation/:updateFreq
The endpoint returns information on the asset deviation and updates count if it was used in the oracle. It takes price change (deviation) and oracle's ticker frequency as the main parameters to determine estimated updates count based on historical data.
is a cross-chain oracle provider that aggregates market data from a wide range of sources, including CEXs and DEXs. Its first-party data sourcing is thorough, enabling unparalleled transparency and customizability for resilient price feeds for 20,000+ assets. Its versatile data processing and delivery ensures adaptability and reliability for any decentralized application.
The main oracle contract is deployed at the following address .
If dApps require a custom oracle with different assets and configurations, please .
For example, to retrieve the latest , you can call:
More info on the .
Learn more about DIA’s and architecture.
The final price point for each asset is calculated by computing the assets' trade information across multiple DEXs and CEXs. This is done using a .
For developer assistance, connect with the DIA team directly on or .
Developers seeking other oracles with tailored price feeds and configurations can initiate the request by .
is a cross-chain oracle provider that sources granular market data from diverse exchanges, including CEXs and DEXs. Its data sourcing is thorough, enabling unparalleled transparency and customizability for resilient price feeds for 20,000+ assets. Its versatile data processing and delivery ensures adaptability and reliability for any decentralized application.
DIA Oracles on Kadena are open source, and their source code is available in . dApps built on Kadena can leverage DIA oracles to access up-to-date asset price information. These deployed oracles are designed for production environments and come with a predefined list of supported assets and settings.
If dApps require a custom oracle with different assets and configurations, please .
Learn more about DIA’s and architecture.
The final price point for each asset is calculated by computing the assets' trade information across multiple DEXs and CEXs. This is done using a .
For developer assistance, connect with the DIA team directly on or .
Developers seeking other specialized, production-grade oracle with tailored price feeds and configurations can initiate the request by .
Each oracle deployed by DIA has a gas wallet, which is the only entity capable of calling for an oracle update. The wallet initiates the transaction to import a new value into the oracle smart contract, based on the (that have been previously configured).
Example:
By default it will return data for the last 24h but you can configure the timeframe with starttime and endtime parameters. See example below:
The following are demo oracles built for testing purposes. For production-ready oracles, visit this .
blockchain*
String
A valid blockchain from GET /v1/blockchains, e.g., Bitcoin.
address*
String
A valid asset address from GET /v1/token/:symbol, e.g., 0x000000000000000000000000000000000000000 for BTC.
deviation*
Integer
Deviation measure in per mille, e.g. 20 will result in 2% deviation
updateFreq*
Integer
Oracle's price feed ticker frequency in seconds, e.g. 120 will calculate asset price in 2 minute trade blocks
starttime
Integer
Unix timestamp setting the start of the return array
endtime
Integer
Unix timestamp setting the end of the return array
dApps built on Stacks can leverage DIA oracles to access up-to-date asset price information.
Access the oracles in the smart contracts below:
Stacks Testnet
ST3Q982CNNQ00E3FH6853EMTA5FPF1M3ENJTHB8PY
Stacks Mainnet
SP1G48FZ4Y7JY8G2Z0N51QTCYGBQ6F4J43J77BQC0
You can query for assets like "STX/USD" or "stSTX/USD" in the get-value()
read function. It returns two values:
The price of STX/USD with 8 decimals
The timestamp of the last update (UTC timezone)
The Stacks oracle includes the following price feeds:
STX
Stacks
0x0000000000000000000000000000000000000000
stSTX
Stacks
SP4SZE494VC2YC5JYG7AYFQ44F5Q4PYV7DVMDPBG.ststx-token
DIKO
Stacks
SP2C2YFP12AJZB4MABJBAJ55XECVS7E4PMMZ89YZR.arkadiko-token
WELSH
Stacks
SP3NE50GEXFG9SZGTT51P40X2CKYSZ5CC4ZTZ7A2G.welshcorgicoin-token
VELAR
Stacks
SP1Y5YSTAHZ88XYK1VPDH24GY0HPX5J4JECTMY4A1.velar-token
ALEX
Stacks
SP3K8BC0PPEVCV7NZ6QSRWPQ2JE9E5B6N3PA0KBR9.age000-governance-token
sBTC
Stacks
SM3VDXK3WZZSA84XXFKAFAF15NNZX32CTSG82JFQ4.sbtc-token
aeUSDC
Stacks
SP3Y2ZSH8P7D50B0VBTSX11S7XSG24M1VB9YFQA4K.token-aeusdc
USDh
Stacks
SPN5AKG35QZSK2M8GAMR4AFX45659RJHDW353HSG.usdh-token-v1
BTC
Bitcoin
0x0000000000000000000000000000000000000000
A consistent heartbeat refreshes all asset prices every 15 minutes.
The DiaAssetSpecificCallingConvention
contract is compatible with AggregatorV3Interface
, allowing developers to read price data from a DIA oracle contract deployed for each asset by calling latestRoundData()
and returning the latest available price of that specific asset.
Influx
PostgreSQL
Redis
Kafka
Filterblock
Tradesblock
Foreign
Liquidity
Collector
Ping jobs
Prepare job (when install does not depend on external image)
Grafana
Loki
Influx
DIA Helm chart: DIA platform
Influx: Timeseries db
Grafana: Dashboard for monitoring
Loki: Logging the services
kube-prometheus: Monitoring Minikube cluster
Developing, Testing, and Building the DIA platform with Docker Compose
On prepare step if you want to connect services between other Compose files, create a deployments/local/exchange-scraper/docker-compose.override.yml
file:
And instead of run the services using this command:
Now in other Compose file just use like this:
Now in Docker context, service-x
can resolve the http://influxdb:8086
hostname
Developing, Testing, and Building the DIA platform with Minikube
Clone repo and change directory:
Next, start the development environment by running the following command to start a Minikube node:
Build the necessary service containers by executing the following commands:
Run the installation script to set up the necessary services (Redis cache, Kafka streams, PostgreSQL, and InfluxDB databases) by executing the following commands:
Now that you have the local cluster running, you are ready to start.
import the test data from the diadata snapshot service: This requires the default structure to be empty and created inside the postgres server. The script will automatically download the snapshot, install it and remove the downloaded snapshot again.
To stop the cluster:
To delete the cluster node:
Develop applications and sources on top of DIA
DIA has a dedicated infrastructure for testing various integrations into the system. In the following pages you can learn how it works.
Start here:
Or proceed directly to the scraper integration tutorials:
Astar Network
Shibuya (ink!)
Shiden Network
soon
Aurora
Aurora Testnet
Evmos
Evmos Testnet
Fuse
Moonbeam Alpha
Polygon Mumbai
is a cross-chain oracle provider that sources granular market data from diverse exchanges, including CEXs and DEXs. Its data sourcing is thorough, enabling unparalleled transparency and customizability for resilient price feeds for 20,000+ assets. Its versatile data processing and delivery ensures adaptability and reliability for any decentralized application.
dApps building on Stacks can utilize DIA oracles to obtain up-to-date asset price information. These deployed oracles are suitable for use in production environments. They come with a list of supported assets and settings. However, if dApps require a custom oracle with a different set of assets and configurations, they should .
Learn more about DIA’s and architecture.
The final price point for each asset is calculated by computing the assets' trade information across multiple DEXs and CEXs. This is done using a Moving Average with Interquartile Range (MAIR) methodology.
For assistance, connect with the DIA team directly on or . Developers seeking other specialized, production-grade oracle with tailored price feeds and configurations can initiate the request here:
: Tracing system for services
Note: Minikube runs on a virtualization layer on the host. The driver is usually the recommended and default option and should be seen as a requirement. For other supported drivers, please refer to . You can also check for advanced command usage.
First, install dependencies: and .
Implement LiquidityScraper
interface at pkg/dia/scraper/liquidity-scrapers/ScraperInterface.go
file
Add a scraper implementation, MyLiquidityScraper.go
file to pkg/dia/scraper/liquidity-scrapers/
folder implement the functions:
Add a constructor for the scraper:
PlatypusFinance: go mod tidy -go=1.16 && go mod tidy -go=1.17 && go install && liquidityScraper -exchange=PlatypusFinance
Orca: go mod tidy -go=1.16 && go mod tidy -go=1.17 && go install && SOLANA_URI_REST=https://try-rpc.mainnet.solana.blockdaemon.tech/ liquidityScraper -exchange=Orca
Implement ForeignScrapperer
interface at pkg/dia/scraper/foreign-scrapers/Scrapper.go
file
Add the scraper implementation a MyForeignScraper.go
Golang file to pkg/dia/scraper/foreign-scrapers/
folder and add the next implementations:
Add a constructor for the scraper:
Add MyForeignScraper
case at scraper file cmd/foreignscraper/foreign.go
:
DIA offers highly customizable oracles that are individually tailored to each dApp’s needs. Each oracle can be customized in several ways, including but not limited to price feeds, data sources, data cleaning filters, pricing methodologies, and oracle update mechanisms. This ensures that the data and oracle remain robust and resilient to the market conditions and provide a global market as well as a specific individual or cross-chain market prices.
DApps can always get in touch with DIA in two ways:
dApps built on XRPL can leverage DIA oracles to access up-to-date asset price information.
Mainnet
Provider: diadata (hex. 64696164617461)
Oracle Document ID: 42
Testnet
Provider: diadata (hex: 64696164617461)
Oracle Document ID: 1
Pricing Methodology
Deviation (%) & Refresh Frequency
0.5% and 120 seconds
Heartbeat
24h
BTC
4254430000000000000000000000000000000000
ETH
4554480000000000000000000000000000000000
XRP
5852500000000000000000000000000000000000
RLUSD
524C555344000000000000000000000000000000
USDC
5553444300000000000000000000000000000000
USDT
5553445400000000000000000000000000000000
This is the Request JSON for mainnet:
It will return the current price of all the assets in hexadecimal format, which can then be converted to decimal and scaled to obtain the actual price with 8 decimal places.
dApps built on Alephium can leverage DIA oracles to consume verifiable & decentralized random numbers on-chain, generated by the drand protocol.
Alephium Mainnet
Alephium Testnet
To consume randomness data, you’ll need to invoke the getLastRound
method on the oracle contract. It will return the round ID of the latest update.
Using this round ID, you can call getRandomValue
and will receive a return value of that randomness, the round ID and the BLS signature from the drand API.
Note that round IDs are used round-robin and will repeat after 1000 iterations.
Instead of distributing pre-calculated data feeds, the DIA Nexus system covers the whole data journey from individual trade information collection, and data computation to the last mile of the feed delivery. To create, for instance, an ETH/USD price feed oracle, Nexus sources data points from dozens of ETH pairs (ETH/USD, ETH/USDT, ETH/BTC, etc) trades from across a wide range of exchanges.
Learn the layers of data aggregation, filtering, and computation that ensure highly accurate and resilient feeds for dApps.
is a cross-chain oracle provider that sources granular market data from diverse exchanges, including CEXs and DEXs. Its data sourcing is thorough, enabling unparalleled transparency and customizability for resilient price feeds for 20,000+ assets. Its versatile data processing and delivery ensures adaptability and reliability for any decentralized application.
The DIA oracle object can be retrieved with the call by specifying the account and oracle_document_id.
For developer assistance, connect with the DIA team directly on or .
Developers seeking other specialized, production-grade oracle with tailored price feeds and configurations can initiate the request by .
The oracle uses randomness from the quicknet mainnet to provide randomness on Alephium.
Below is the DIARandomOracle
contract implementation in :
You can learn more .
For developer assistance, connect with the DIA team directly on or .
Developers seeking other specialized, production-grade oracle with tailored data feeds and configurations can initiate the request by .
Find the right data for your needs Show your users the most transparent data on the market with our API. Whether you're building a financial service, a portfolio management tool, a new media offering, or more, we have the most advanced and updated data on the market for your product.
Backtest your strategies Use the most efficient and transparent crypto data to run simulations and backtest your trading or investing strategies. With crowd-aggregated hundreds of exchanges, you can be sure that you're getting the right picture every single time.
Run Experiments Build your own models with our data, to further your interest or just for fun. With our flexible and powerful API, we provide you with a set of data that will help you draw insights and make conclusions.
The DIA base url is https://api.diadata.org/v1
. All API paths are sub-paths of this base URL.
We apply the highest performance standards for all our API feeds.
99.9% uptime
<1 sec response times
The price estimation process needs to be resilient against trades with prices diverting from the current market price. Reasons for diverting trades can be market manipulation, errors or flash crashes on certain exchanges, among others.
To avoid these irregularities, the first processing step involves data cleaning and outliers removal, to avoid building a feed using data that is completely away from the median. Detecting and excluding outliers is an important processing task, especially in low-volume and illiquid markets. Otherwise, a single low-volume trade can offset the price estimation and serve as a base token for other assets, leading to a chain reaction of misaligned price data.
How does this look in practice? By applying an Interquartile Range (IR) filter, DIA excludes data points and entire sets that lie outside of an acceptable range relative to the interquartile range. The filter analyses all trades of a predefined time range and sorts them by their recorded price. After that, this range of prices is divided into four price blocks, the quartiles. The boundaries of the full price range determine the boundaries of the first and the last quartile. To clear out outliers, any trades falling into the first or the last quartile are filtered out and subsequently, only trades falling into the "middle" quartiles move forward into further processing.
Now that we understand how the process of clearing diverting trade data works, how is the final price actually calculated from all remaining data points? To retrieve a single USD price value for every asset, DIA uses trade-based price determination methodologies. These filters are functions to get a single price point from a collection of trades in a block.
Filters are calculated for each asset on each exchange individually, as well as for each asset on all exchanges combined. This combined filter result represents the result closest to the true "whole market" that can be determined by this system. As each use case requires different data needs, DIA can provide multiple filters to best serve the requirements of each application.
Let’s look at a couple of examples:
Volume Weighted Average Price (VWAP): is a methodology for trade-based price determination that takes into account the different volumes of trades. All trades from the queried time range are collected and weighted by their volume. Weighting means that the (normalized) volume of the trade is multiplied by its executed price. This is done by accumulating the previously calculated volume-price-products of all trades and dividing them by the sum of all volumes combined.
Moving Average with Interquartile Range Filter (MAIR): in this case, all the trades collected in the queried time range are ordered by timestamp. For each second in this time range, a block is created where trades are put into. As soon as the collection of all trades for each block is finalized, then it's weighted against the volume for each data point and the weighted average price is taken to arrive at the final price
Additionally, DIA can always implement new on-demand methodologies as well as build new methodologies together with users.
Off-chain and on-chain delivery methods can be configured with custom update triggers, enhancing the feed's usability and reliability across various applications.
Update triggers are the rules that dictate how oracles refresh to mirror the most recent value of the data feed. These triggers are adaptable and can be tailored to optimally cater to particular scenarios, such as the price feeds of assets with low liquidity.
Each oracle can be customized to push data feed updates live based on the following parameters:
Time-based (push)
Updates are triggered in predefined time intervals (seconds, minutes or hours)
15 minutes
Deviation-based (push)
Updates are triggered by a deviation from the last reported value.
5% deviation
Time + deviation (push)
The oracle will update whenever any of the two conditions is met.
5% deviation + 24h time
Before you begin writing a scraper, please check if the exchange offers integration through WebSockets. If it does, implement the scraper using the WebSocket protocol instead of a RESTful API.
To scrape data from an exchange data source, called "MySource" for instance, follow these steps:
Create a new file in the directory pkg/dia/scraper/exchange-scrapers/
with the filename MySourceScraper.go
.
Start building your scraper by writing a function with the signature NewMySourceScraper(exchangeName string) *MySourceScraper
. We suggest that this function calls a mainLoop()
method in a go routine, constantly receiving trade information through the trade channel of MySourceScraper as long as the channel is open.
Also, to ensure proper error handling and cleanup, it is recommended to include a method Error()
, which returns an error as soon as the scraper's channel closes. Additionally, Close()
and cleanup()
methods should be included to handle the closing and shutting down of channels.
To make the scraper visible to the system, add a reference to it in Config.go
in the dia package:
Add a case for your scraper in the switch statement in the pkg/dia/scraper/exchange-scrapers/APIScraper.go
file:
Finally, add exchange's pairs to config/MySourceExchange.json
config:
Modify the build/Dockerfile-genericCollector
and build/Dockerfile-pairDiscoveryService
files and add the following two lines before the RUN go mod tidy
step:
Build the necessary service containers by running the following commands:
Create a manifest for the new exchange scraper by creating a new mysource.yaml
file. You can refer to existing files for guidance or use the following template:
Before running the manifest, create a new entry in the database for the new exchange:
Deploy the manifest using the following kubectl command:
Hooray 🎉 Your scraper should now be running.
CEX
DEX
Centralized:
Bitfinex: go mod tidy -go=1.16 && go mod tidy -go=1.17 && go install && collector -exchange=Bitfinex -mode=current -pairsfile=true
rm ./collector.go && cp /mnt/env-context/cmd/exchange-scrapers/collector/collector.go ./collector.go && go mod edit -replace github.com/diadata-org/diadata=/mnt/env-context && go mod tidy -go=1.16 && go mod tidy -go=1.17 && go install -a && collector -exchange=Bitfinex -mode=current -pairsfile=true
Bittrex: go mod tidy -go=1.16 && go mod tidy -go=1.17 && go install && collector -exchange=Bittrex -mode=current -pairsfile=true
CoinBase: go mod tidy -go=1.16 && go mod tidy -go=1.17 && go install && collector -exchange=CoinBase -mode=current -pairsfile=true
MEXC: go mod tidy -go=1.16 && go mod tidy -go=1.17 && go install && collector -exchange=MEXC -mode=current -pairsfile=true
Decentralized:
PlatypusFinance
Orca: go mod tidy -go=1.16 && go mod tidy -go=1.17 && go install && SOLANA_URI_REST=https://try-rpc.mainnet.solana.blockdaemon.tech/ collector -exchange=Orca -mode=current -pairsfile=true
GET
https://api.diadata.org/v1/assetQuotation/:blockchain/:asset
Returns the quotation for a fully qualified asset (i.e. distinguished by blockchain and address). Quotations are obtained by filtering trades data with a Moving Average filter and Interquartile Range outlier detection (MAIR).
blockchain*
String
Name of the blockchain for requested asset
asset*
String
Address of the requested asset
GET
https://api.diadata.org/v1/quotation/:symbol
Get most recent information on the currency corresponding to symbol. Quotations are obtained by filtering trades data with a Moving Average filter and Interquartile Range outlier detection (MAIR).
Note: this endpoint is intended for testing purposes only; if multiple assets with the same symbol exists, data for the highest volume generating asset will be returned
Symbol*
String
Which symbol to get a quotation for, e.g., BTC.
GET
https://api.diadata.org/v1/quotedAssets
Returns a list of assets available for quotations.
Use the query parameter blockchain in order to obtain exclusively assets on the selected blockchain.
blockchain
String
Name of the blockchain for the requested assets.
GET
https://api.diadata.org/v1/synthasset/:blockchain/:protocol
The endpoint provides information on the health factor of DeFi protocols.
It takes into consideration outstanding loans, contracts' balance and compares it to synthetic token issuance. This helps to make sure that all assets are fully collateralized.
Exemplary query:
Currently only Aave V2 and Aave V3 are supported on Ethereum and Avalanche blockchains.
It is possible to receive collateral information for a single token by using token filter as in example below:
To receive all asset updates data during the selected period use starttime
and endtime
filter parameters with timestamps as inputs. Here is an example:
blockchain*
String
Ethereum or Avalanche
protocol*
String
Aave
address
String
Token contract address. You can use either underlying token address or synthetic token to receive the data
starttime
Integer
Unix timestamp setting the start of the return array
endtime
Integer
Unix timestamp setting the end of the return array
GET
https://api.diadata.org/v1/poolLiquidity/:blockchain/:pool_address
blockchain*
String
Name of the blockchain of the requested pool.
address*
String
Address of the requested pool.
GET
https://api.diadata.org/v1/poolSlippage/:blockchain/:pool_address/:token1_address/:pool_type/:desired_slippage
To predict the amount of tokens required to reach a specified slippage, you can use Pool Slippage
endpoint. It works for any pool that exists on Uniswap V2 or any fork DEX on supported list of exchanges.
The logic is as follows:
Provide the address of the pool, select a token in which you wish to have slippage calculated (e.g. if selected ETH from ETH-USDC, it will show how much is ETH is required to move the price by selected amount) and determine the goal for price movement.
:blockchain*
String
Name of the blockchain of the requested pool (e.g. Ethereum, Moonbeam, etc.)
:pool_address*
String
Address of the requested pool
:token1_address*
String
Address of either of the tokens in the pool
:pool_type*
String
Type of the pool (e.g. UniswapV2)
:desired_slippage*
Integer
Per mille (e.g. 100 will result in 10% slippage)
GET
https://api.diadata.org/v1/blockchains
Get a list of all available blockchains.
GET
https://api.diadata.org/v1/exchanges
Get a list of all available crypto exchanges.
GET
https://api.diadata.org/v1/pairsCex/:exchange
Get all pairs on a given exchange for which DIA is collecting trades data at the moment.
Use the query parameter verified in order to get either verified or unverified pairs.
exchange*
String
Name of the crypto exchange.
verified
Bool
If set to true, only verified pairs are returned. If set to false, only unverified pairs are returned.
GET
https://api.diadata.org/v1/pairsAssetCex/:blockchain/:address
Get all pairs on a centralized exchange for which DIA is collecting trades data at the moment. Include pairs across all exchanges involving the requested asset as either base- or quote token.
Use the query parameter verified in order to get either verified or unverified pairs.
blockchain*
String
Name of the blockchain for the requested asset.
address*
String
Address of the requested asset.
verified
Bool
If set to true, only verified pairs are returned. If set to false, only unverified pairs are returned.
GET
https://api.diadata.org/v1/lastTradesAsset/:blockchain/:address
Get last trade timestamp for an asset.
blockchain*
String
A valid blockchain from GET /v1/blockchains, e.g., Bitcoin.
address*
String
A valid asset address from GET /v1/token/:symbol, e.g., 0x000000000000000000000000000000000000000 for BTC.
GET
https://api.diadata.org/v1/chartPoints/:filter/:exchange/:symbol
Get chart points for an exchange.
filter*
string
Which filter should be applied (Available options: MA120, VOL120, MEDIR120 and MAIR120).
exchange*
string
A valid exchange from GET /v1/exchanges, e.g., Binance
symbol*
string
A valid symbol from GET /v1/coins, e.g., BTC.
scale
string
Which scale the graph points distance should have. Available options: 5m 30m 1h 4h 1d 1w.
GET
https://api.diadata.org/v1/assetChartPoints/:filter/:blockchain/:address
Get asset details for all exchanges.
Remark: Careful! Successful responses can be rather large.
starttime
integer
Unix timestamp setting the start of the return array
endtime
integer
Unix timestamp setting the end of the return array
filter*
string
Which filter should be applied (Available options: MA120, MEDIR120, VOL120 and MAIR120).
blockchain*
string
A valid blockchain from GET /v1/blockchains, e.g., Bitcoin.
address*
String
A valid asset address from GET /v1/token/:symbol, e.g., 0x000000000000000000000000000000000000000 for BTC.
scale
string
Which scale the graph points distance should have. Available options: 5m 30m 1h 4h 1d 1w
GET
https://api.diadata.org/v1/foreignSymbols/:source
Get the list of available symbols along with their ITIN for guest quotations.
source*
string
source of the quotation
GET
https://api.diadata.org/v1/foreignQuotation/:source/:symbol
Get the latest quotation for a token from a guest source.
source*
string
source of the quotation
symbol*
string
Which symbol to get a quotation for, e.g. BTC
time
number
Unix timestamp.
By leveraging a network of WebSockets and decentralized node providers, DIA Nexus retrieves token trade ticks and metadata from exchanges. This, in turn, represents an enormous data availability (≈15bn trades/day) to, later on, enable DIA to capture the highest data precision and build flexible and customizable price feeds.
Trades for every asset pair get recorded and collected into blocks of 120 seconds into the DIA Nexus platform. For each block, a range of filters are applied. These are functions to get a single price point from the collection of trades in a block. For example, the Moving Average filter returns the volume-weighted moving average of the trades block.
DIA Nexus retrieves trading data from as close to the data source as possible in order to ensure the highest precision possible. The data retrieval is processed through a series of asset collectors:
For cross-chain bridges, we collect the swaps, enabling us to port prices and map assets across blockchains. Cross-Chain bridges allow for swapping a crypto asset on a given blockchain to the corresponding asset on a different blockchain. For instance, swap USDC on Ethereum to USDC on Arbitrum. Cross-Chain bridges can be seen as ranging somewhere between centralized and decentralized: Parts of the complete bridging transactions can be seen publicly on the respective blockchain. However, by its very nature, the bridging itself is done by a centralized entity.
One of the biggest risks for DeFi and Web3 applications is not having enough sources for its oracle and using a feed that relies on one single market. These single-source oracles, as explained in the beginning, are relatively easily exposed to price manipulations and man-in-the-middle attacks.
DIA is designed to avoid these scenarios by leveraging multiple markets as sources, including CEXs and DEXs and enabling the customisation of methodologies and sources to cater best for the needs of specific use cases. So far in DeFi’s (short) history, a failure in the oracle node network has never been the reason for a loss of funds, but attacks on the markets that provide the price data to the sources. Take for instance the example of the Cream Finance attack in October 2021.
DIA prioritises the transparency aspect of its data and oracle services and enables any dApp developer and user to know precisely how the data feeding the application is built, what data sources are been used, and what methodologies are been applied.
To start the Kubernetes Dashboard, use the following command:
Once the dashboard service is ready, you can visit the web interface in your browser.
Forward the ports of the data stores services to localhost:
PostgreSQL Database: kubectl port-forward deployment/postgres 5432:5432
Redis Cache: kubectl port-forward deployment/redis 6379:6379
InfluxDB Database: kubectl port-forward deployment/influx 8086:8086
REST Server: port 8081
To debug InfluxDB writes, just change points in batch to see more frequent writes to influx (pkg/model/db.go
):
A .testenv.local
file at root folder is required to pull private images from Docker:
DIA currently offers a set of price feeds for a selection of traditional financial assets, delivered via RestAPI endpoints.
GET
https://api.diadata.org/v1/rwa/Fiat/:FiatQuote
Returns the latest available exchange rate for fiat currencies.
GET
https://api.diadata.org/v1/rwa/Commodities/:CommoditySymbol
Returns the latest price of a commodity in USD.
GET
https://api.diadata.org/v1/rwa/ETF/:ETFSymbol
Returns the latest price of an ETF in USD.
To request custom data feeds, you can always refer to the following section:
This section describes how we filter for outliers in ourprice determination system.
The price estimation process needs to be robust against trades with prices that are diverting from the current market price of an asset pair. Reasons for diverting trades can be market manipulation, errors, flash crashes on certain exchanges, or other market irregularities.
Detecting and excluding outliers and market manipulation is an important data processing task, especially in small and (somewhat) intransparent markets. Otherwise, a single low-volume trade can offset the price estimation and serve as a base token for other assets, leading to a chain reaction of misaligned price data.
The Interquartile Range filter in DIA examined all trades in a trades block (i.e. all trades in the predefined time range) and sorts them by their recorded price.
After that, this range of prices is divided into four price blocks, the Quartiles. The boundaries of the full price range also determine the boundaries of the first and the last quartile.
To clear out outliers, any trades falling into the first or the last quartile are filtered out and subsequently, only trades falling into the "middle" quartiles are returned to the caller.
To simplify your DIA API integration journey, we created sample API calls for retrieving price data in the most widely used programming languages.
All examples make a call to DIA Asset Quotation API for Bitcoin price.
Discover the methodologies DIA uses to filter and aggregate prices of assets.
You can find examples of how to query the endpoint in different programming languages below:
DIA made API data easily accessible through GraphQL endpoint.
You can use the following link to send GraphQL requests:
Also you can use a dedicated front-end to try out queries:
Currently only tokens price feed data is available on GraphQL.
Example of schema that can be used for quoting data:
The schema will return the moving average price of ETH calculated every 5min (300 sec) by counting 2min trade blocks.
FeedSelection
filtering optionsThere are a number of ways to filter out exchanges with DIA's GraphQL endpoints.
Here's an example how to define FeedSelection query to return a price for a given range with all sources included:
There are number of ways to filter desired sources from the entire sources library for a given asset. We'll go over examples of these filters.
To select specific exchanges you can use the following format:
It is also possible to select specific pairs from a given exchange. The query would look as follows:
Another filtering option is to include sources based on available liquidity in the pool. For example if you select a liquidity threshold of $5 000 000, the response will calculate the price of an asset only by including liquidity pools from DEXes which have higher than $5 000 000 liquidity. The query looks as follows:
It is also possible to merge pairs selection with LiquidityThreshold
, which would result in such query:
One more option is to use two different assets to calculate one unified price. It might be useful for cross-chain assets, e.g. DAI on ETH mainnet & DAI on Arbitrum One. The query which can be used for such cases would look as follows:
In the table below you can find the description of response fields:
With all , DIA performs many layers of processing to build high-quality feeds resilient to bad actors. All steps can be customized with tailor-made methodologies to best serve each use case.
->
->
The price feed is created and published via an API endpoint, utilizing the previously and . This feed can be accessed off-chain using either RestAPI or DIA GraphQL, which allows for greater adaptability and direct modification of feed attributes.
To allow the platform to call your scraper, implement a scraper that conforms to the interface from the scrapers package:
In order to implement the interface, it is necessary to include the ScrapePair
method, which returns a MySourcePairScraper for a specific pair, so our main collection method can iterate over all possible trading pairs. Note that MySourcePairScraper should respect and implement the interface
For a better understanding of how to set up a scraper, you can refer to the file, as it provides an example of an existing exchange scraper and its overall structure and logic.
The page includes a list of all available Token data endpoints on DIA network. To review the data sources and methodologies used for pricing, visit the page.
Example:
Example:
Example:
Example:
Get the latest state of a DEX pool's liquidity. Example:
Here is an exemplary API call:
Example:
Example:
Example:
Example:
Example:
Example: Note: Successful responses can be rather large.
Example:
Example:
Example: Use the query parameter time in order to get the latest quotation before the specified timestamp
Example:
Filters are calculated for each asset pair on each exchange individually, as well as for each asset on all exchanges combined. This combined filter result represents the result closest to the true "whole market" that can be determined by this system. Learn more about .
For centralised exchanges such as Coinbase, Kraken or Binance, DIA scrapers collect trades directly from the exchange’s databases. The data collection happens in retrieval periods from 1 to 7 seconds and varies from exchange to exchange. All trading data is retrieved through Rest APIs or WebSocket APIs, developed and maintained by the respective trading platform. More on .
For decentralised exchanges, DIA collects data from decentralized exchanges on various blockchains by subscribing to swap events in the corresponding liquidity pools – LPs. In contrast to centralized exchanges, in decentralized exchanges, it is possible to retrieve trading data directly from the respective blockchain by subscribing to the corresponding pool events. Examples of sources are Uniswap, curve.finance or PancakeSwap. .
GraphQL: port 1111, and
Kafka: port 8080, and
Grafana: port 3000, and
Change image of pre-populated data from file
We calculate our cryptocurrency spot-prices using 120-second moving average (SMA) with a weighting of trade volumes (VWAP). Small volume trades have a proportionally smaller influence on the determined price than bigger volume trades. In order to create reliable pricing, we also exclude data points and sets entirely that lie outside of an relative to the of all data points.
After this clearing process is finished, the data is ready to be processed further in other filters like the or filter.
First of all, you should define the desired asset. DIA uses blockchain-address asset identification methodology. You can search for available assets and find out blockchain/address information in the .
Test DIA's APIs dedicated to token prices. You can learn more about the product .
:FiatQuote*
String
Exchange pair, e.g. EUR-USD; GBP-USD
:CommoditySymbol*
String
Commodity symbol e.g. XAU-USD (Gold)
:ETFSymbol*
String
ETF symbol e.g. IVV (iShares Core S&P 500 ETF)
filter
Select trades processing methodology (e.g. VWAP
, MAIR
, find out all available filters here)
BlockSizeSeconds
Determine the window for trades inclusion in seconds (e.g. 120
will include all trades into price calculation that happened during the last 2 minutes)
BlockShiftSeconds
Determine the frequency of price updates in seconds during the selected timeframe (e.g. 600
will return price for every 5min during the selected period)
StartTime
The start time of the preferred period for price calculation in timestamp format (e.g. 1655084380
will start calculating price from 13.06.2022 01:39:40 GMT time)
EndTime
The end time of the preferred period for price calculation in timestamp format (e.g. 1655090649
will calculate price until 13.06.2022 03:24:09 GMT time)
FeedSelection
This is the main module of the query which is responsible for parametrizing asset(s) for which the price data should be retrieved. Available parameters for FeedSelection
are described in detail below.
Time
Time at which the value was generated
Value
Price of an asset at a give time
Pools
Pools which had a trade during the block time period and were involved in the price calculation
Pairs
CEX pairs which had a trade during the block time period and were involved in the price calculation
Blockchain
Blockchain of an asset
Address
Address of an asset
The usual structure for calculating fair price of Liquid Staked Tokens (LST) is as follows:
The main components of the formula are:
BaseAssetPrice - price of the non-staked asset which is locked against LST (e.g. for stETH, ETH acts as a base asset)
LockedBaseAsset - number of base assets locked in the LST smart contract (e.g. for stETH, quantity of ETH locked in stETH smart contract)
IssuedLST - amount of LST tokens issued (e.g. for stETH, total supply of stETH represents it)
This is the standard methodology and the data is queried directly from the Smart Contract of the LST issuer. It allows sudden devaluation in case malicious actor manages to mint a large amount of LST tokens and ensures that the issuer is keeping the funds safe.
Depending on the contract type we apply the following logic for valuation:
If the assets can be redeemed 1:1 (e.g. stETH), price will always be the same as the base asset, even though the collateralization rate is higher (if collateralization ratio is lower, the price will be adjusted to negative side)
If the LST asset increases in value against the base asset because rewards are not paid out directly but rather represent a share of the total pool, the pricing will be dynamic depending on the reewards accrued in the pool (e.g. rETH)
These are the standard methodologies applied to the feeds but some assets might require custom methodologies.
This page contains information about the VWAPIR pricing methodology.
Description of data retrieval and processing of cryptocurrency data
The data retrieval is processed through a series of asset collectors, capable of fetching all kinds of financial data. These collectors are classified into the following groups:
Centralized Exchange Scrapers: Collect data from centralized exchanges for cryptocurrencies and other assets (like stocks, futures, and rare earth metals) such as Binance, Coinbase or Kraken.
Decentralized Exchange Scrapers: Collect data from decentralized exchanges on various blockchains by subscribing to swap events in the corresponding LP pools. Examples for sources are Uniswap, curve.finance or PancakeSwap.
Bridge Scrapers: Collect swaps data from bridges bridging crypto assets between different blockchains. Examples of sources are Multichain, Anyswap or Hop.
The raw trading data is the basis of all further derived quantities such as prices and circulating supply numbers for all cryptocurrency exchanges. We retrieve and store trading data from as close to the data source as possible and with the highest precision possible.
In this section you will find information about the methodology for the following data:
This page contains information about the VWAP pricing methodology.
VWAP (Volume Weighted Average Price) is a methodology for trade-based price determination that takes into account the different volumes of trades.
All trades from the queried time range are collected and weighted by their volume. Weighting means that the (normalized) volume of the trade is multiplied by its executed price.
As soon as the block has been finalized, the VWAP price of these prices is calculated. This is done by accumulating the previously calculated volume-price-products of all trades and dividing them by the sum of all volumes combined.
The result is then returned as the result of the filter operation.
The VWAP filter is used in DIA's price determination. Our API can display the latest VWAP filter values, i.e., the filter results from a 120 second interval of all recorded trades for an asset.
All trading data is retrieved through Rest APIs or websocket APIs, developed and maintained by the respective trading platform. See the below table for detailed information on all sources. We remark that by the very nature of a websocket API, there is no retrieval frequency, instead, trading data comes in continuously as it is produced.
All trades from the queried time range are ordered by timestamp.
For each second in the time range, there exists a "slot" where trades are put into.
As soon as all trades in the block have been finalized (usually 120 seconds per block), for each 1 second slot the closing price is taken.
These slots are then weighted against the volume for each data point and the weighted average price is taken to arrive at the final price.
The result is then returned as the result of the filter operation.
The MAIR filter is used in DIA's price determination. Our price quotations are the latest MAIR-120 filter values, i.e., the filter results from a 120 second interval of all recorded trades for an asset.
Distributed key generation (DKG)
Randomness is the property of lacking any sensible predictability. It is very difficult to create random events on-chain due to the deterministic nature of any EVM or EVM-like environment.
Centralized randomness is susceptible for attacks by the randomness source, as a single non-random outcome cannot be distinguished from a random one. Thus, having a single RNG provide randomness via an oracle is not enough.
Random numbers can be very relevant for on-chain applications such as games, lotteries, prediction markets, or NFT launches.
Relying on pseudo-random values like the last blockhash can be manipulated by miners and is not advisable.
From random number generation to on-chain distribution, DIA xRandom operates as follows:
DIA on-chain distribution: the final randomness signature is shipped on-chain as an oracle smart contract. In order to effectively and securely execute these on-chain transactions, DIA has built a robust, decentralised node infrastructure — dubbed DIA xNode. DIA xNode is a network of third-party, decentralized node providers that grant DIA the blockchain infrastructure to push data on-chain ensuring high rate limits.
It is important to understand the risks of the randomness oracle before using it and to be able to mitigate them.
Additionally, there are new risks introduced by using the oracle.
Oracle stops serving data
Check that the oracle has recent updates in its history.
Specific Round is missed by the oracle
Have your dApp use the next round if a certain round is unavailable (but later ones exist).
Oracle serves compromised data
Check the associated BLS signature provided by drand (Note: Currently not availabe on most EVMs).
In contrast to centralized exchanges, in decentralized exchanges, it is possible to retrieve trading data directly from the respective blockchain by subscribing to the corresponding pool events.
Base
Base
BNB Chain
Astar
Alephium
Ethereum
Polygon
Arbitrum
Fantom
Bifrost
BNB Chain
Stacks
Arbitrum
Arbitrum
Ethereum
Arbitrum
Fantom
Moonbeam
Polygon
Polygon
EVMOS
Metis
Moonriver
Hydration
Ethereum
Metis
Linea
Linea
Telos
Solana
Cosmos
Binance Smart Chain
Binance Smart Chain
Avalanche
Unreal
Polygon
Arbitrum
Arbitrum
Moonriver
Fantom
Moonbeam
Ethereum
Arbitrum
Fantom
Polygon
Metis
Binance Smart Chain
Binance Smart Chain
Avalanche
Arbitrum
Avalanche
Aurora
Celo
Ethereum
Ethereum
Arbitrum
Polygon
Celo
Base
Stacks
Optimism
Wanchain
Bifrost
Bifrost-Polkadot
The CVI is calculated on top of bid/ask information received from crypto (BTC) option markets. First of all, the current level of the orderbook is recorded and placed into a timestamped order. Instruments for each strike price are monitored, both for Call and Put options.
Instruments are named uniquely to identify them. For example, BTC-03JUN2020-6000-C is the name of a Bitcoin call option expiring at 3rd of June 2020 for a strike price of 6000 USD.
To calculate the CVI, the option levels of two dates are considered: The next Friday that is at least one month away from now and the first Friday with available options before that date. With that we ensure that the future market is observed appropriately.
The CVI is then calculated using this formula:
where N describe the minutes until the respective event (e.g. minutes in a year, 30 days or minutes until expiry of option T1). T1 and T2 are the fractions of minutes left until expiry of these options compared to the minutes in a full year.
The variances are calculated as such:
with the sum of deltas being defined as the intervals between strike prices for each strike price. R is the risk-free lending rate and Q(k) is used to describe the average of the bid-ask-spread for an option K_i.
The CVI starts at a value of 1000.
You can connect with us and find support through the following channels:
If you report security issues within our backend, smart contracts, and production operations, we offer a security bounty program. To be eligible for a bounty, be sure to follow these requirements:
Make sure you can describe the security issue you found in a concise and reproducible way.
Give us time to assess and address the issue. Sometimes behaviour can be perceived as security issue.
We will grade severity of reported issues and use the CVSS scale as a guideline. The ultimate decision about the severity we consider to be achieved remains in our discretion.
Payouts will be conducted in an established ERC20 stablecoin like USDC. Please make sure to be able to receive ERC20 tokens.
In order to reduce the impact of outliers in daily fluctuations of financial markets, many financial products use simple or compounded averages of a given reference rate. For instance, on every business day the New York Federal Reserve Bank publishes the average rate of the Secured Overnight Financing Rate, compounded over the last 30 calendar days under the name of SOFR30.
Here, we present the methodology of compounded rates that is used by large banks such as the New York Federal Reserve Bank (compounded SOFR) and the Bank of England (compounded SONIA).
which is the formulation of compounded rates used by the FED and the BOE amongst others.
Link to API documentation: Coming soon!
The VWAPIR (Volume Weighted Average Price with Interquartile Range Clearing) filter is the application of the filter to data that was previously cleared by the filtering system.
The filter is implemented as part of the FiltersBlockService in .
To see all integrated data sources visit:
If you didn't find the desired data/source, you can request a custom integration .
The filter is implemented as part of the FiltersBlockService in .
BitStamp
The MAIR (Moving Average with Interquartile Range Clearing) filter is the application of the Moving Average filter to data that was previously cleared by the filtering system.
The filter is implemented as part of the FiltersBlockService .
To offer distributed and verifiable randomness on-chain, DIA leverages ’s distributed randomness beacon. The beacon is run by a group of independent actors called the League of Entropy.
The is a collaborative project to provide a verifiable, decentralized randomness beacon. A decentralized randomness beacon combines randomness from multiple independent high entropy sources to generate a truly unbiased random number for anyone that may need a public source of randomness.
Please visit the project site at for the most up-to-date information on current operations. To learn how the how distributed randomness beacon is built please .
Drand run distributed nodes to produce their randomness beacon. They use to create collective private/public key. Participants in their League of Entropy then generate randomness in rounds and broadcast it together with its signature.
Drand randomness generation: in every drand epoch (currently set at 30s for the existing LoE beacons, but on future “unchained” beacons) each of the nodes that form the drand network generates a partial signature, which it broadcasts to the rest of the nodes. Once any node has enough, i.e., a threshold number of signatures, it computes the new randomness beacon, which is the hash of the signature aggregate.
Drand randomness propagation and consumption: the new beacon is propagated through the network to all other nodes. Any node can verify and accept or reject the hash of the signature (i.e., beacon) that it received. Any client or application can consume randomness out of band (i.e., without needing to be part of the drand network) either through or through , Gossipsub. Clients can also verify that the randomness they received is indeed the one produced by the drand network.
An extensive risk evaluation of the underlying protocol can be found . All risks listed there also affect the randomness guest oracle, as it serves as an underlying data provider.
, , sources
, , , , sources
PanCakeSwapV3
, , , sources
, , , , sources
, and sources
Contact us first. If you use or publish the vulnerability you will not be eligible for a bounty payout. Our contact address is .
Consider a unit investment over a time period of calendar days containing business days. Let be an interest rate that is published once every business day and assume that the business day convention is such that the year has days. Then, the compounded gain over the whole interest period results to
.
We remark that for constant interest rates and this simplifies to the well-known compounded gain formula . However, in general, the rate factor is an integer value that accounts for being a business day or a non-business day. More precisely, if is a business day followed by non-business days, we set . For instance, if is followed by a business day, i.e., , we have . For a friday, which is usually followed by two non-business days, we would have . Now, in order to get the average interest from the compounded gain , we subtract the original investment and normalize, thus obtaining
Sources:
The methodology from the previous section has a special feature in that it mixes compounded and non-compounded rates. More precisely, investments are not compounded for weekends and holidays. This behaviour is reflected in the rate factor . In the Index presented below, investments are compounded over all calendar days in the respective interest period.
Consider a unit investment over a time period of calendar days. Let be an interest rate that is published once every business day and assume that the business day convention is such that the year has days. We define an interest rate such that coincides with on business days and is set to the rate of the previous business day if is a holiday or a weekend. In this straightforward manner we obtain an interest rate for all calendar days and can now set
Severity
Payout
Low
500 - 2500 USDC
Medium
2500 - 5000 USDC
High
5000 - 7500 USDC
We welcome every contributor who wants to add something to our project. If you want to submit a feature request or a bug report please submit an issue on our issue page. In case you want to add code, read the documentation about the area you want to contribute to and generate a pull request after you finished programming. We will then take a look at your contribution and eventually integrate it.
In the interest of fostering an open and welcoming environment, we as contributors and maintainers pledge to making participation in our project and our community a harassment-free experience for everyone.
Examples of behavior that contributes to creating a positive environment include:
Using welcoming and inclusive language
Being respectful of differing viewpoints and experiences
Gracefully accepting constructive criticism
Focusing on what is best for the community
Showing empathy towards other community members
Examples of unacceptable behavior by participants include:
The use of sexualized language or imagery and unwelcome sexual attention or advances
Trolling, insulting/derogatory comments, and personal or political attacks
Public or private harassment
Publishing others' private information, such as a physical or electronic address, without explicit permission
Other conduct which could reasonably be considered inappropriate in a professional setting
Project maintainers are responsible for clarifying the standards of acceptable behavior and are expected to take appropriate and fair corrective action in response to any instances of unacceptable behavior.
Project maintainers have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, or to ban temporarily or permanently any contributor for other behaviors that they deem inappropriate, threatening, offensive, or harmful.
This Code of Conduct applies both within project spaces and in public spaces when an individual is representing the project or its community. Examples of representing a project or community include using an official project e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event. Representation of a project may be further defined and clarified by project maintainers.
Instances of abusive, harassing, or otherwise unacceptable behavior may be reported by contacting the project team at coc@diadata.org. The project team will review and investigate all complaints, and will respond in a way that it deems appropriate to the circumstances. The project team is obligated to maintain confidentiality with regard to the reporter of an incident. Further details of specific enforcement policies may be posted separately.
Project maintainers who do not follow or enforce the Code of Conduct in good faith may face temporary or permanent repercussions as determined by other members of the project's leadership.
For the usage of DIA's data and services
Important disclaimer information for visitors of DIA, outlining the boundaries and limitations of the content provided.
General Information: All content, including but not limited to texts, graphics, and services provided on this website, its linked platforms, applications, forums, blogs, social media, and other platforms associated with DIA (collectively referred to as the "Site"), are primarily sourced from third parties and are made available exclusively for general informational purposes.
No Representations or Warranties: DIA expressly disclaims all warranties, express or implied, related to the accuracy, completeness, security, or timeliness of the content and services. Nothing contained herein constitutes financial, legal, or any specialized advice. Users contemplating trading or other financial decisions should seek the counsel of a certified financial advisor. The Site does not constitute a solicitation, endorsement, or offer of any nature.
'As Is' and 'As Available' Provision: The Site, including all content and functionalities, is provided on an "AS IS" and "AS AVAILABLE" basis without any representation or endorsement. While DIA endeavors to ensure the seamless functioning of the Site, it does not warrant that the Site will be uninterrupted, timely, secure, or error-free, nor that defects, if any, will be corrected.
Risk of Data Loss: Users acknowledge that the data accessed or stored on our Site may be inadvertently lost, corrupted, or become temporarily inaccessible due to various factors. DIA shall not be held responsible for any loss, damage, or corruption of user data arising from technical failures, Internet disruptions, force majeure events, or other factors beyond or within our control.
Exclusion of Implied Warranties: To the fullest extent permissible under applicable law, all implied warranties, including but not limited to the implied warranties of merchantability, fitness for a particular purpose, and non-infringement, are hereby disclaimed. Some jurisdictions may prohibit the exclusion of certain warranties; thus, the above disclaimers may not apply to you in such jurisdictions.
Limitation of Liability: Under no circumstances, including negligence, shall DIA or its affiliates be liable for any indirect, special, incidental, punitive, exemplary, or consequential damages, including but not limited to loss of data, revenue, profits, or other economic advantage, arising out of or in any way connected with the use or performance of the Site or services, even if DIA had been previously advised of the possibility of such damages.
By accessing or using the Site, you hereby agree to be bound by the aforementioned terms and conditions. It is recommended to seek appropriate legal or professional advice when necessary.
You are free to:
Share — copy and redistribute the material in any medium or format
Adapt — remix, transform, and build upon the material
The licensor cannot revoke these freedoms as long as you follow the license terms.
Under the following terms:
Disclaimer - The licensor provides the material as-is, without any guarantees or warranties. This means there's no promise about the material's quality, accuracy, or any other aspects.
No Liability - The licensor won't be responsible for any kind of damages or costs that come from using the material, even if they know there's a risk.
Notices:
This example demonstrates the functionality of the Laminar chain's oracle pallet including the aggregation of price values.
make init
will setup the environment by downloading and installing all necessary dependencies and the required toolchain.
make build
builds the chain's code so we can execute it.
After that copy the chainSpec.json
file from this repository into the chain's directory. More on the spec file will be explained a little bit later.
When running a node being part of the laminar chain network it creates and stores lots of data because it's constantly building new blocks and extending the actual blockchain. We are going to store all that data in a subdirectory of /tmp
for each node (its base directory). If the base directory doesn't exist it will be created but it is necessary to purge all data of a specific node on subsequent launches:
This wipes the base directory of Alice's node. For wiping Bob's node's data adjust the base-path parameter accordingly. The action asks for confirmation which is given by answering with y
.
Bootstrap the network
We are going to launch Alice's node first:
This creates and runs a node that stores its data in the specified base directory.
We also pass the chain specification file into the node. We use this file to configure the scenario in which all participating parties will own a sufficient amount of gas in order to be able to "pay" for their transmitted transactions.
The parameter --alice
will make Alice the node's authority uses her predefined key pair.
We specify three ports: One for the other peers to connect to the node, a websocket port for the off-chain worker to connect to and a port for incoming RPCs.
After that we fire up Bob's node and tell it to connect to Alice. So in another terminal session we are executing
The main difference is the bootnodes
parameter which specifies the network and identity information of Alice's node.
Now the two nodes start working together and our laminar chain network is ready for playing with the oracle.
About oracles
A blockchain oracle is an interface for the chain to the real world and vice versa. By feeding off-chain data to the oracle they enter the chain's scope and can get processed by smart contracts. In this case our oracle is called an inbound oracle. For our example imagine a smart contract that operates on currency prices. It might trigger an action as soon as the price for a specific currency exceeds a specific value. By feeding the current prices from external sources periodically to the oracle we provide the smart contract with the required input data on-chain.
Oracle pallet
Feeding new key-value pairs to the oracle via signed transactions.
Retrieving key-value pairs from the oracle either by other modules on-chain or by off-chain workers via RPC.
The Laminar implementation configures the oracle pallet to use the type CurrencyId
as key type. That means it can be one of the predefined values FBTC, FETH, FEUR, etc. The value uses the type Price
.
In this example we are going to use an off-chain worker to feed currency prices and fetch them afterwards.
Our off-chain worker
Let's start by switching the directory
and setting up the dependencies in the package.json
file:
Next step is coding the off-chain worker in the file index.ts
. We import the required types
and build the frame for the program:
Now we can run
for fetching the dependencies and launching the program. However nothing is going on right now so let's write the actual code. From now on we put everything into the async block that we just prepared.
We start creating an API object:
Please note that our API object uses a websocket to communicate with the node. And this websocket connects to port 9945 which we have specified above when firing up Alice's node. So all the requests are submitted to her node.
Next step is configuring the key material. We want to transmit signed transactions that contain the prices to be fed in the oracle. Therefore the off-chain worker needs to know the key pairs of Alice, Bob and Charlie (yes, he's also going to take part in this example). Usually the key pairs would be imported, e. g. from a file. But as for the chain nodes we use the predefined key material which is addressed by //[Name]
. We derive those key pairs from the keyring we create specifying the key pair format SR25519.
Alice wants to trigger a transaction that feeds the oracle. If we start typing api.tx.
the code completion shows us the available extrinsics but the oracle isn't one of them. That's because the default methods of the Substrate runtime are known at compile time but not the custom chain specific extrinsics that might be part of the runtime. Let's find out what's really available by querying the node:
When executing the program we get a list of much more actually available extrinsics - and one of them is laminarOracle
. The list tells us even more about the oracle: the available query method feedValues
. This is exactly what we needed to know.x
Now Alice reports a current BTC price of 1, i. e. she feeds the oracle with the corresponding key-value pair:
This line consists of several interesting pieces:
We use the API object to trigger an extrinsic (tx
for transaction). The available extrinsics depend on the used chain. In this case we address the oracle pallet of the laminar chain which is addressed by laminarOracle
.
On that extrinsic we call the method feedValues
in order to submit key-value pairs that should be fed into the oracle. In this case we just submit a single key-value pair for the Bitcoin price.
As this operation is a transaction that is going to be included in a block and costs Alice some gas it needs to be signed by her. This is what happens in the last part of the line.
When this line returns the transaction has been finalized and we can now query the oracle with the current Bitcoin price. But there's again the question of how to do that. As mentioned earlier the oracle offers retrieving the current values via RPC. So let's find out the available RPCs:
The output again gives us what we need. Two calls should catch our attention: oracle_getAllValues
and oracle_getValue
. That means we have the two functions getAllValues
and getValue
available for the RPC oracle.
Now we query the current BTC price:
As you can see we use an RPC as the query doesn't change the state of the chain and therefore the operation is free. We specify the correct RPC (which is oracle
) and query the value to the key associated with the Bitcoin price (which is FBTC
). The output is a pair of two values: The current price and the timestamp of the transaction that contained the feeding operation.
That's it! Now we can run the program again with yarn start
. The oracle will be fed and queried and the output is the expected BTC price of 1.
Data aggregation
In the previous section we fed the oracle with one single data point and queried the oracle to retrieve a value. It isn't surprising at all that we read the value that we had written into the oracle just before. But it requires a closer look to what value is actually returned by getValue
. The oracle pallet states that this method returns a "combined value" which indicates that there is some data aggregation happening.
In fact the way the prices of each currency are aggregated is entirely up to the chain implementation. Currently the Laminar chain doesn't implement any aggregation algorithm but the oracle pallet comes with a default implementation: It takes all fed values of a specific currency submitted within the last 3 days and returns the median value. If there aren't any values to calculate the median from it just returns the most recent value.
Let's try it out. Disable the lines of the previous section that interacted with the oracle (via tx and RPC) and add the following:
When we query the oracle for the BTC price it returns the median value of 5 as expected.
Important! Remember to purge the chain status and restart the two nodes before executing the code of this section.
Download sources for the example and the laminar chain:
Setup laminar chain:
Run first chain node (Answer with "y" when asked for confirmation before purging):
Run second chain node (e.g. in another terminal session):
After both nodes have started to communicate with each other run medianizer example (in yet another terminal session):
Read through the smart contract implementations in the Nexus system.
A basic introduction to some fundamental concepts in crypto asset farming
Decentralized Finance (DeFi) has attracted a lot of attention in the recent years. A rather new and hot topic inside the world of DeFi is yield farming, which allows to earn passive income on token holdings using various investment strategies coded into smart contracts. There is an ever growing number of products labeled as yield farming. Many of those are advertised with an APY (Annual Percentage Yield) rate. We emphasize that in this context, the term APY is to be read with caution. In classical finance, it refers to the annual rate of return in an investment and is a predefined rate. Hence, an investor can be sure to obtain the corresponding interest in their investment. However, most farming products heavily rely on the investment strategy in volatile crypto markets. In contrast to APYs in lending and borrowing, rates published in farming are either APYs of past periods or projections based on past data.
and hence a return rate of 1%.
Again, whether the amount of tokens in a pool increases or decreases solely depends on the investment strategy. The return rates for past time ranges can be computed by considering pool rate differences, such as done in the example above. In order to estimate future return rates, one can apply mathematical methods which allow for the estimation of future values based on past data. The simplest way of doing so is to fit a linear function to the data, which is also known under the name of (linear) regression. However, for most cases, such simple models do not yield good results for bigger time ranges. Mathematicians have tackled such problems since a long time and there is a wide range of techniques available. Nowadays, many of these are used under the name of machine learning.
A slightly different approach for farming is presented by other platforms such as Synthetix. Apart from constant fee rates on transactions such as trades, the dynamical return rate depends on the average return of all investments on the platform. The corresponding gains (or losses) for the investors are paid out periodically, for instance weekly. We will first explain the general mechanism and then illustrate it with an example.
Example:
This section represents DIAs research efforts aimed at providing genuine data for the users. Here are some examples of our work:
DIA research is dedicated for expanding the oracles ecosystem with novel solutions. Treat these as indicative of our work and feel free to contact us if you'd like to develop something new & exciting!
This is a human-readable summary of (and not a substitute for) the . .
Attribution — You must give , provide a link to the license, and . You may do so in any reasonable manner, but not in any way that suggests the licensor endorses you or your use.
NonCommercial — You may not use the material for .
ShareAlike — If you remix, transform, or build upon the material, you must distribute your contributions under the as the original.
No additional restrictions — You may not apply legal terms or that legally restrict others from doing anything the license permits.
You do not have to comply with the license for elements of the material in the public domain or where your use is permitted by an applicable .
No warranties are given. The license may not give you all of the permissions necessary for your intended use. For example, other rights such as may limit how you use the material.
This is a human-readable summary of (and not a substitute for) the . .
Attribution — You must give , provide a link to the license, and . You may do so in any reasonable manner, but not in any way that suggests the licensor endorses you or your use.
NonCommercial — You may not use the material for .
ShareAlike — If you remix, transform, or build upon the material, you must distribute your contributions under the as the original.
No additional restrictions — You may not apply legal terms or that legally restrict others from doing anything the license permits.
You do not have to comply with the license for elements of the material in the public domain or where your use is permitted by an applicable .
No warranties are given. The license may not give you all of the permissions necessary for your intended use. For example, other rights such as may limit how you use the material.
The Laminar chain is built using the . In our example we are going to setup a network consisting if two nodes, one is operated by Alice and one is operated by Bob.
First we download the Laminar chain and bootstrap the network. After that we execute the following statements in the chain's directory.
A Substrate chain's runtime consists of modules called pallets. In the case of Laminar one of these pallets is our . It can be interacted with in two ways:
The component we create to simulate new price reports and oracle queries lives off-chain and will be written in Typescript. It is completely separate from anything stored on the Laminar chain itself. We rather interact with the chain through specific calls made using the . This API defines the technical aspects of how we can interact with the chain (and specifically with the oracle), i. e. how to build transactions and make RPCs. For the specific parameters regarding the Laminar chain we also make use of the .
For more convenience we define a function feedPrice
which takes a price value and a sender and triggers the transaction accordingly. We use the of Polkadot JS in order to make sure that we wait for every transaction to be finalized in a block. Now our three users submit their values:
You can also find the audit report .
Due to the complexity of this topic, our first step consists of publishing the pool rates as emitted in the smart contracts of various farming products such as yearn.finance or CVault. The basic idea of pool rates is pretty straightforward and can be understood as follows. Consider a farming pool of token and let be the amount of token at block number . Assume, farming starts at block number , then we have the following equation
where is the price corresponding to the pool of token . Initially, and hence . Motivated by this fact, we write
Here, is the pool rate corresponding to the pool of token .
As the initial supply is constant, the price depends on the evolution of and hence on the pool's investment strategy - if the number of tokens increases with block number , so will the pool rate. In other words, the return of an investment is determined by the change in the pool rate. The following example illustrates the relation between the pool rate and the return on an investment. We remark that in general, as for interest rates in traditional finance, differences of pool rates can be used for simple interest calculation as well as for compounded interest.
Example: Assume an investor puts an amount of tokens into an empty pool at block number . With the notation from above this means and . Further assume that blocks later, the investment strategy has increased the number of tokens in the pool to tokens, so . The return on the initial investment after these blocks can be obtained by the difference of the pool rates at blocks and respectively. Indeed, we have and thus . By the above equation for we have and thus . This yields
Consider a platform with different pools associated to an investment strategy. In most cases, the investment strategy is short or long on a cryptocurrency, represented by a minted synthetic asset. Assume an investor makes an investment by buying from a pool of such a token worth US-Dollar (it could be any other currency obviously). She thereby increases the so-called system's total debt from the previous value to (also in US-Dollar). Here, the total debt is just the value of all investments across the platform. Now assume that after one farming period has elapsed, the value of her investment is . Furthermore, the system's total debt, i.e. the US-Dollar value of all investments together changed to . The idea is that the system does not owe the investor her initial investment, but rather her initial proportion of the system's total debt, which is . Hence, her actual personal debt is
In other words, the total debt is distributed among all investors according to their initial share of the total debt. In particular, everyone wins if , that is, the total debt increases, and everyone looses if . Furthermore, the investor's total gain after one farming period is given by her personal debt minus her initial investment
Let us quickly illustrate this equation before making the connection to the pool rate . Assume the total debt has increased from by to . Plugging into the above equation for the gain leads to and hence, the investor makes a gain of . In order to see the relation to the pool rate from the previous section, we rewrite the personal debt in a somewhat artificial manner as
And hence, the pool rate is given by
It should come as no surprise that this leads to the pool rate for the above example. Let us now consider a slightly more comprehensive example.
Consider a protocol with just two pools, namely sBTC and sETH, and three investors as shown in the first row. The system's total debt is the sum of the investments. Assume that after one farming period the ETH price miraculously remains the same, and the price of BTC doubles, leading to the balances shown in row 2. In the third row, the personal debt for each investor is computed using the above formula for . In the last row, the corresponding total gains are computed. Although investor 3's strategy made a gain of , the investor himself only made a gain of , as he was the only one winning and his total gain is distributed among all investors. Furthermore, it is distributed according to the initial shares. As Investors 1 and 2 have far bigger initial shares, they get a large amount of Investment 3's gain, although their strategy did not win. Finally, we can easily compute the farming period's pool rate for this example, using the above formula
Some platforms such as Loopring emit total system rewards, i.e. accumulated rewards distributed among all pools. We remark that, given a particular staking pool's balance and the total reward , an average pool rate can be estimated as follows
In fixed reward systems as used by Barnbridge, a fixed (and predefined) reward is distributed among all investors in a pool after a given staking period. The reward is distributed according to an investor's share of the pool. More precisely, let again be the total staking pool's balance at the end of the staking period. Assume a particular investor has staked the amount . Then the investor's reward is given by
For instance, assume the reward is , the staking pool's balance at the end of the staking period of one week is given by and an investor invested of an arbitrary currency. Then the reward is given by
Here, the pool rate is simply given by the quotient , so in the above example We remark that there might be slight differences on how to implement these systems. Especially whether to allow deposits in the middle of a staking period and if so, how to distribute the reward among these "late" investors.
Step
Investment 1 in $
Investment 2 in $
Investment 3 in $
Total Debt in $
sBTC
sETH
0
100
0 48
2
0
sBTC
sETH
0
100
0
48
4
0