- Resources
- Overview
- Detailed Description of the Building Blocks
- Smart Contract Documentation
- Node Deployment Guide
- Conclusion
Field | Value |
---|---|
Chain name | DIA Lasernet Testnet |
Chain ID | 10640 |
Block explorer | https://testnet-explorer.diadata.org |
RPC URL | https://testnet-rpc.diadata.org |
Websocket | wss://testnet-rpc.diadata.org |
Gas token | DIA on ETH Sepolia 0xa35a89390FcA5dB148859114DADe875280250Bd1 |
Faucet | https://faucet.diadata.org |
Documentation | https://docs.diadata.org |
This repository hosts a self-contained containerized application comprising three main components: scraper, collector, and processor. The scraper collects trade data from various centralized and decentralized exchanges. The collector and processor aggregate the data through a two-step process to produce a scalar value associated with an asset, which is subsequently published on-chain. In most cases, this value represents the asset's price in USD.
In the following sections, we describe the function and usage of the building blocks (the three components mentioned earlier) that make up the system (see figure). The explanation proceeds from left to right
Each scraper is implemented in a dedicated file in the folder /pkg/scrapers with the main function signature func NewExchangeScraper(pairs []models.ExchangePair, tradesChannel chan models.Trade, wg *sync.WaitGroup)
,
resp. pools
instead of pairs
for decentralized exchanges.
Its function is to continuously fetch trades data from a given exchange and send them to the channel tradesChannel
.
The expected input for a scraper is a set of pair tickers such as BTC-USDT
. Tickers are always capitalized and symbols separated by a hyphen. It's the role of the scraper to format the pair ticker such that it can subscribe to
the corresponding (websocket) stream.
For centralized exchanges, a json file in /config/symbolIdentification is needed that assigns blockchain and address to each ticker symbol the scraper is handling.
The collector gathers trades from all running scrapers. As soon as it receives a signal through a trigger channel it bundles trades in atomic tradesblocks. An atomic tradesblock is a set of trades restricted to one market on one exchange, for instance BTC-USDT
trades on Binance exchange. These tradesblocks are sent to the Processor
.
The processor is a 2-step aggregation procedure similar to mapReduce:
- Step 1: Aggregate trades from an atomic tradesblock. The type of aggregation can be selected through an environment variable (see Feeder/main). The only assumption on the aggregation implementation is that it returns a
float64
. - Step 2: Aggregate filter values obtained in step 1. The selection of aggregation method and assumptions are identical to Step 1. The obtained scalar value is sent to the Oracle feeder.
The feeder is feeding a simple key value oracle. It publishes the value obtained from the Processor. It is worth mentioning that the feeder can contain the trigger mechanism that initiates an iteration of the data flow diagram.
For monitoring, we use Prometheus client libraries for Go to create, manage, and expose metrics. A metrics struct is defined to track uptime (using a Prometheus gauge) and store Pushgateway details like URL, job name, and authentication credentials. The uptime metric is initialized, registered, and updated periodically based on the application's runtime. Metrics are pushed to the Pushgateway every 30 seconds, with authentication and error handling in place.
For more details about the contracts, refer to the following documentation:
This document outlines the procedures for deploying the diadata/decentralized-feeder:<VERSION>
containerized application. Replace <VERSION>
with the desired version (e.g.v0.0.5
) when deploying.
For the most recent Docker image tags, please refer to public docker hub: https://hub.docker.com/repository/docker/diadata/decentralized-feeder/general
-
Ensure that Docker or Docker Compose is installed on your machine.
-
Clone this repository to your local machine.
-
The container has minimal resource requirements, making it suitable for most machines, including Windows, macOS, Linux, and Raspberry Pi, as long as Docker is installed.
-
An ETH private key from MetaMask or any other Eth wallet. Alternatively to generate private key effortlesly eth-keys tool can be used for this ethereum/eth-keys
-
DIA tokens in your wallet (you can use faucet for this https://faucet.diadata.org)
- Locate the
docker-compose
folder in this repository. - Inside, you will find a file named
docker-compose.yaml
.
-
Create a
.env
file in the same directory asdocker-compose.yaml
. This file should contain the following variables:PRIVATE_KEY
: Your private key for the deployment.DEPLOYED_CONTRACT
: The contract address. Initially, leave this empty during the first deployment to retrieve the deployed contract.PUSHGATEWAY_USER
: to allow decentralized-feeder authenticate towards the monitoring server. Reach out to the team to get hold of these credentials, info [at] diadata.orgPUSHGATEWAY_PASSWORD
: to allow decentralized-feeder authenticate towards the monitoring server. Reach out to the team to get hold of these credentials, info [at] diadata.org
For additional environment variable configurations, refer to Adding Exchange Pairs and Watchdog environment variables
-
Example
.env
file:PRIVATE_KEY=myprivatekey DEPLOYED_CONTRACT= PUSHGATEWAY_USER= PUSHGATEWAY_PASSWORD=
-
Open a terminal in the
docker-compose
folder and start the deployment by running:docker-compose up
-
Once the container is deployed with
DEPLOYED_CONTRACT
env variable empty the logs will display the deployed contract address in the following format:│ time="2024-11-25T11:30:08Z" level=info msg="Contract pending deploy: 0xxxxxxxxxxxxxxxxxxxxxxxxxx."
-
Copy the displayed contract address (e.g.,
0xxxxxxxxxxxxxxxxxxxxxxxxxx
) and stop the container withdocker rm -f <container_name>
. -
Update your
.env
file withDEPLOYED_CONTRACT
variable mentioned above. Redeployed the container withdocker-compose up -d
PRIVATE_KEY=myprivatekey DEPLOYED_CONTRACT=0xxxxxxxxxxxxxxxxxxxxxxxxxx
-
Check if the container is running correctly by viewing the logs. Run the following command:
docker-compose logs -f
-
Expected Logs: Look for logs similar to the example below, which indicate a successful startup:
│ time="2024-10-29T13:39:35Z" level=info msg="Processor - Atomic filter value for market Binance:SUSHI-USDT with 20 trades: 0.7095307176575745." │ │ time="2024-10-29T13:39:35Z" level=info msg="Processor - Atomic filter value for market Simulation:UNI-USDC with 1 trades: 8.008539500390082." │ │ time="2024-10-29T13:39:35Z" level=info msg="Processor - Atomic filter value for market Crypto.com:USDT-USD with 5 trades: 0.99948." │ │ time="2024-10-29T13:39:35Z" level=info msg="Processor - filter median for MOVR: 9.864475653518195." │ │ time="2024-10-29T13:39:35Z" level=info msg="Processor - filter median for STORJ: 0.4672954012114179." │ │ time="2024-10-29T13:39:35Z" level=info msg="Processor - filter median for DIA: 0.9839597410694259." │ │ time="2024-10-29T13:39:35Z" level=info msg="Processor - filter median for WETH: 2626.9564003841315."
-
Cleanup the deployment:
docker rm -f <container_name>
-
Verify the container has been removed:
docker ps -a
This method is suitable for simple setups without orchestration.
- Deploy the feeder with
DEPLOYED_CONTRACT
initially empty:docker run -d \ -e PRIVATE_KEY=myprivatekey \ -e DEPLOYED_CONTRACT= \ -e PUSHGATEWAY_USER= \ -e PUSHGATEWAY_PASSWORD= \ --name decentralized-feeder \ diadata/decentralized-feeder:<VERSION>
- Retrieve the logs to get the deployed contract address:
docker logs <container_name>
- Stop the container, update the
DEPLOYED_CONTRACT
value, and restart:docker stop <container_name> docker run -d \ -e PRIVATE_KEY=myprivatekey \ -e DEPLOYED_CONTRACT=0xxxxxxxxxxxxxxxxxxxxxxxxxx \ -e PUSHGATEWAY_USER= \ -e PUSHGATEWAY_PASSWORD= \ -e EXCHANGEPAIRS="Binance:TON-USDT, Binance:TRX-USDT, ....." --name decentralized-feeder \ diadata/decentralized-feeder:<VERSION>
- Retrieve the logs to verify the container is running as expected
docker logs <container_name>
- For additional environment variable configurations, refer to Adding Exchange Pairs and Watchdog environment variables
Kubernetes is ideal for production environments requiring scalability and high availability.
- Create a Kubernetes
Deployment
manifest. Replace<VERSION>
with the desired version:apiVersion: apps/v1 kind: Deployment metadata: name: decentralized-feeder namespace: default spec: replicas: 1 selector: matchLabels: app: decentralized-feeder template: metadata: labels: app: decentralized-feeder spec: containers: - name: feeder-container image: diadata/decentralized-feeder:<VERSION> env: - name: PRIVATE_KEY value: "myprivatekey" - name: DEPLOYED_CONTRACT value: "" - name: EXCHANGEPAIRS value: "" - name: PUSHGATEWAY_USER= value: "" - name: PUSHGATEWAY_PASSWORD= value: "" - containerPort: 8080
For additional environment variable configurations, refer to Adding Exchange Pairs and Watchdog environment variables
- Deploy the feeder with
DEPLOYED_CONTRACT
set to an empty string (""
) in the Kubernetes manifest.kubectl apply -f deployment.yaml
- Monitor the logs for the deployed contract address:
kubectl logs <pod-name>
- Update the
DEPLOYED_CONTRACT
value in the manifest with the retrieved contract address. - Apply the updated manifest:
kubectl apply -f deployment.yaml
To configure exchange pairs for the decentralized feeder, use the EXCHANGEPAIRS
environment variable. This can be done regardless of the deployment method. The variable specifies pairs to scrape from various exchanges, formatted as a comma-separated list of <Exchange>:<Asset-Pair>
(e.g., Binance:BTC-USDT
).
Locate the environment configuration file or section for your deployment method:
- For Docker Compose: Use the
.env
file or add directly to thedocker-compose.yaml
file. - For Kubernetes: Update the kubernetes manifest file
manifest.yaml
- For Docker Run: Pass the variable directly using the
-e
flag.
- Example in docker-compose:
EXCHANGEPAIRS=" Binance:TON-USDT, Binance:TRX-USDT, Binance:UNI-USDT, Binance:USDC-USDT, Binance:WIF-USDT, CoinBase:AAVE-USD, CoinBase:ADA-USD, CoinBase:AERO-USD, CoinBase:APT-USD, CoinBase:ARB-USD, GateIO:ARB-USDT, GateIO:ATOM-USDT, GateIO:AVAX-USDT, GateIO:BNB-USDT, GateIO:BONK-USDT, Kraken:AAVE-USD, Kraken:ADA-USD, Kraken:ADA-USDT, Kraken:APT-USD, Kraken:ARB-USD, KuCoin:AAVE-USDT, KuCoin:ADA-USDT, KuCoin:AERO-USDT, KuCoin:APT-USDT, KuCoin:AR-USDT, Crypto.com:BONK-USD, Crypto.com:BTC-USDT, Crypto.com:BTC-USD, Crypto.com:CRV-USD "
- Example in Kubernetes manifest:
spec: containers: - name: feeder-container image: diadata/decentralized-feeder:<VERSION> env: - name: PRIVATE_KEY value: "myprivatekey" - name: DEPLOYED_CONTRACT value: "" - name: EXCHANGEPAIRS value: " Binance:TON-USDT, Binance:TRX-USDT, Binance:UNI-USDT, Binance:USDC-USDT, Binance:WIF-USDT, CoinBase:AAVE-USD, CoinBase:ADA-USD, CoinBase:AERO-USD, CoinBase:APT-USD, CoinBase:ARB-USD, GateIO:ARB-USDT, GateIO:ATOM-USDT, GateIO:AVAX-USDT, GateIO:BNB-USDT, GateIO:BONK-USDT, Kraken:AAVE-USD, Kraken:ADA-USD, Kraken:ADA-USDT, Kraken:APT-USD, Kraken:ARB-USD, KuCoin:AAVE-USDT, KuCoin:ADA-USDT, KuCoin:AERO-USDT, KuCoin:APT-USDT, KuCoin:AR-USDT, Crypto.com:BONK-USD, Crypto.com:BTC-USDT, Crypto.com:BTC-USD, Crypto.com:CRV-USD " ports: - containerPort: 8080
- Example in Docker Run:
docker run -d \ -e PRIVATE_KEY=your-private-key \ -e DEPLOYED_CONTRACT=your-contrract \ -e EXCHANGEPAIRS="Binance:TON-USDT, Binance:TRX-USDT, ....." \ --name decentralized-feeder \ diadata/decentralized-feeder:<VERSION>
- Docker Compose: Check logs with:
docker-compose logs -f
- Kubernetes: Check pod logs:
kubectl logs <pod-name>
- Docker Run: View logs with:
docker logs <container-name>
- The output should look like:
lasernet-feeder-1 | time="2024-11-26T12:22:47Z" level=info msg="Processor - Start......" lasernet-feeder-1 | time="2024-11-26T12:22:47Z" level=info msg="CoinBase - Started scraper." lasernet-feeder-1 | time="2024-11-26T12:22:47Z" level=info msg="Kraken - Started scraper." lasernet-feeder-1 | time="2024-11-26T12:22:47Z" level=info msg="GateIO - Started scraper." lasernet-feeder-1 | time="2024-11-26T12:22:47Z" level=info msg="KuCoin - Started scraper." lasernet-feeder-1 | time="2024-11-26T12:22:47Z" level=info msg="Crypto.com - Started scraper." lasernet-feeder-1 | time="2024-11-26T12:22:47Z" level=info msg="Binance - Started scraper at 2024-11-26 12:22:47.97428349 +0000 UTC m=+0.037715635."lasernet-feeder-1 | time="2024-11-26T12:23:08Z" level=info msg="Processor - Atomic filter value for market Binance:USDC-USDT with 89 trades: 0.9998099980000003." lasernet-feeder-1 | time="2024-11-26T12:23:08Z" level=info msg="Processor - Atomic filter value for market Binance:UNI-USDT with 47 trades: 10.817108170000003." lasernet-feeder-1 | time="2024-11-26T12:23:09Z" level=info msg="Processor - Atomic filter value for market Binance:TRX-USDT with 297 trades: 0.18920189200000007." lasernet-feeder-1 | time="2024-11-26T12:23:09Z" level=info msg="Processor - Atomic filter value for market CoinBase:APT-USD with 18 trades: 11.38." lasernet-feeder-1 | time="2024-11-26T12:23:09Z" level=info msg="Processor - Atomic filter value for market GateIO:BNB-USDT with 3 trades: 620.9062090000001." lasernet-feeder-1 | time="2024-11-26T12:23:09Z" level=info msg="Processor - Atomic filter value for market CoinBase:AERO-USD with 5 trades: 1.27824." ....
The decentralized feeders contain two different types of watchdog variables that monitor the liveliness of WebSocket connections used for subscribing to trades in exchange pairs.
- Exchange-wide watchdogs, such as
BINANCE_WATCHDOG
If no trades are recorded by a scraper for any pair on the given exchange withinBINANCE_WATCHDOG
seconds, the scraper is restarted and will resubscribe to all pairs specified in the feeder's configuration. - Pairwise watchdogs such as
BINANCE_WATCHDOG_BTC_USDT
: If no trades are recorded by a scraper for a specific pair withinEXCHANGE_WATCHDOG_ASSET1_ASSET2
seconds, the scraper will unsubscribe and subsequently resubscribe to the corresponding pair. All other subscriptions of this scraper will remain untouched. The first type of watchdog applies to cases where the scraper fails, for instance due to server-side issues, and require a restart. The second type of watchdog applies to dropping websocket subscriptions. These ocurr in websocket connections and are often "silent", i.e. there is no error message that allows for a proper handling. An example of how watchdog variable could look like in the context of kubernetes manifest.
- name: COINBASE_WATCHDOG
value: "240"
- name: CRYPTODOTOCOM_WATCHDOG
value: "240"
- name: GATEIO_WATCHDOG
value: "240"
- name: BINANCE_WATCHDOG_BTC_USDTs
value: "300"
- name: CRYPTODOTCOM_WATCHDOG_BTC_USDT
value: "300"
- name: KUCOIN_WATCHDOG_BTC_USDC
value: "300"
If any issues arise during deployment, follow these steps based on your deployment method:
- Docker Compose:
docker-compose logs -f
- Docker Run:
docker logs <container_name>
- Kubernetes:
kubectl logs <pod-name>
- Ensure all required variables (
PRIVATE_KEY
,DEPLOYED_CONTRACT
) are correctly set:- Docker Compose: Check
.env
file. - Docker Run: Verify
-e
flags. - Kubernetes: Check the Deployment manifest or ConfigMap.
- Docker Compose: Check
- Docker Compose:
docker-compose down && docker-compose up -d
- Docker Run:
docker stop <container_name> && docker rm <container_name> && docker run -d ...
- Kubernetes:
kubectl delete pod <pod-name>
- Ensure the correct image version is used and manifests/files are properly configured.
- Ensure you're using the correct image version:
docker pull diadata/decentralized-feeder:<VERSION>
- Apply fixes and redeploy.
The diadata/decentralized-feeder:<VERSION>
image can be deployed using various methods to accommodate different use cases. For production environments, Kubernetes or Helm is recommended for scalability and flexibility. For simpler setups or local testing, Docker Compose or Docker Run is sufficient.
If you encounter any issues or need further assistance, feel free to reach out to the team @ info [at] diadata.org