# How to create an oracle
Waves Oracles is a catalog for the searching and creating of oracle cards. Using this service, you know which oracles have already been created by other developers and what data they write on the blockchain. Anyone can create their oracle card so that other users know it exists and can use it.
Creating the oracle as microservice which takes data from an outside source and writes on the blockchain, remains outside the scope of Waves Oracles.
This article based on a simple sample will shortly introduce you to full-cycle how and why to use Waves Oracles, how to implementation of the program part of the oracle and launch your solution, and then use the oracle data in the dApp.
# Example of an oracle
As a small example, the following use case should be considered now: imagine you have dApp and you want in it to access data of exchange rates: WAVES per USD and WAVES per BTC, e.g.
If off-chain data is required for dApp to be executed, this data must be obtained and write on the blockchain, because dApp have access only to data stored on the blockchain. To get data from off-chain to blockchain implements small programs called oracles.
Our dApp requires quotation data on the blockchain. Therefore, we will create a new oracle, which will receive the relevant quotation data from the public Waves Data Service API (you can use any other source) once an hour and write it on the blockchain. In addition, we will create an oracle card so that other users can alse use the data in their decentralized applications.
# Implementation of the program part of the oracle
The main part of the oracle is a program that has access to the API and records exchange rates in the oracles account data storage. Here use TypeScript with Node.js, but you can use Python or any other programming language. List of client libraries.
# Cron
Create a cron that will run our service to retrieve data from the API every hour:
import * as cron from 'node-cron';
import { WavesPrice } from './WavesPrice';
// run WavesPrice class every 1 hour
cron.schedule('0 0 */1 * * *', () => {
new WavesPrice(logger);
});
# Get data and transaction sending
Create a service that will request data from the public Waves Data Services API.
Get the lastPrice parameter from the API, shift the point by the required digits and assign a value to the corresponding keys:
lastPrice = await this.getLastprice('https://api.wavesplatform.com/v0/pairs/WAVES/Ft8X1v1LTa1ABafufpaCWyVj8KkaxUWE6xBhW6sNFJck');
dataParams.push({ key: 'waves_usd_2', value: lastPrice * Math.pow(10, 2) });
lastPrice = await this.getLastprice('https://api.wavesplatform.com/v0/pairs/WAVES/8LQW8f7P5d5PZM7GtZEBgaqRPGSzS3DfPuiXrURJ4AJS');
dataParams.push({ key: 'waves_btc_8', value: lastPrice * Math.pow(10, 8) });
The easiest and correct way is to create a new account for new oracle. You can create an account using Waves.Exchange, see the Create Account section.
Sign the data transaction by your oracle account SEED:
const signerDataTX = DataTX(params,'YOU ORACLE SEED HERE');
Send the signed data transaction to the blockchain:
- Testnet — nodes-testnet.wavesnodes.com
- Mainnet — nodes.wavesnodes.com.
const result = await broadcast(signerDataTX, 'https://nodes.wavesnodes.com');
Creating an oracle card and other transactions requires a fee, so don't forget to top up your account balance. The first time you can try to create an oracle in the test network.
The code ends up looking like this:
import axios from 'axios';
import { data as DataTX, broadcast } from '@waves/waves-transactions';
export class WavesPrice {
constructor() {
this.start();
}
private async getLastprice(url: string): Promise<number> {
const data = (await axios.get(url)).data.data;
return data.lastPrice;
}
private async broadcastTX(dataParams) {
const params = {
data: dataParams
};
const signerDataTX = DataTX(params,'YOU ORACLE SEED HERE');
const result = await broadcast(signerDataTX, 'https://nodes.wavesnodes.com');
}
public async start() {
try {
let dataParams = [];
let lastPrice = 0;
lastPrice = await this.getLastprice('https://api.wavesplatform.com/v0/pairs/WAVES/Ft8X1v1LTa1ABafufpaCWyVj8KkaxUWE6xBhW6sNFJck');
dataParams.push({ key: 'waves_usd_2', value: lastPrice * Math.pow(10, 2) });
lastPrice = await this.getLastprice('https://api.wavesplatform.com/v0/pairs/WAVES/8LQW8f7P5d5PZM7GtZEBgaqRPGSzS3DfPuiXrURJ4AJS');
dataParams.push({ key: 'waves_btc_8', value: lastPrice * Math.pow(10, 8) });
await this.broadcastTX(dataParams);
} catch(err) {
console.log(err);
}
}
}
# Project dependencies
This part sets the project dependencies:
{
"name": "balance-oracle",
"version": "0.1.0",
"private": true,
"dependencies": {
"axios": "0.19.0",
"node-cron": "2.0.3",
"@waves/waves-transactions": "3.16.3"
},
"scripts": {
"start": "tsc ./project/src/app.ts && node ./project/src/app.js"
},
"devDependencies": {
"@types/node": "^11.11.3"
}
}
# Launching oracle
First, install Node.js, if it is not yet installed: https://nodejs.org/en.
Setting project dependencies:
$ npm install
If you have not installed TypeScript, you need to install it globally:
$ npm install -g ts-node typescript
Now let's launch our oracle:
$ npm run start
In Waves Explorer we can see the data transaction with exchange rates in the format that we wanted (fig. 1):
Figure 1.
# Creation of the oracle card
For other users to know about our oracle, we use Waves Oracles to create an oracle card.
Waves Oracles is a service that utilizes Waves Keeper.
To create an oracle card, use the same account you used to sign transactions in the code.
Open Waves Oracles, log in with Waves Keeper and hit the Create an oracle on the sidebar menu (fig.2).
Figure 2.
On popup will be shown needs to fill certain information about the oracle.
Let's call our oracle "WAVES/USD and WAVES/BTC", select the category "Market data & exchange rates" — so users can find that oracle quickly. Сhoose the "Production" status, because the oracle is already working and available. Write a short oracle description in the About field. And the oracle updates the quotes data every hour, we specify it in the Update frequency field.
The up part of the form is shown in fig. 3.
Figure 3.
Coming down below and fill in the Specification and Example.
In our case, the oracle must write two quotation values: WAVES/USD and WAVES/BTC. Therefore, we define these two parameters, as shown in fig. 4.
Here we need some explanations: dApp on Ride cannot use float value, that's why we use an integer type with point shift by the necessary number of digits. Also in the key, we specify the number of digits by which the point shifts: for USD by two digits, for BTC by eight digits. It'll be easy to parse keys like this by Ride and realize how many characters the point is shifted.
The key like this is not unique and the account data storage will always retain the last value. You can add a timestamp to make the key unique and save historical values.
Figure 4.
After the form is filled, Approve the data transaction for the creation of the oracle card with Waves Keeper. Once we did so, the new oracle card is successfully registered in Waves Oracles. We can see the oracle card protocol in Waves Explorer, as shown in fig. 5. After a while, the card will appear in the Waves Oracles interface.
Figure 5.
# Usage of the oracle data
Congratulations, now our oracle is completely ready. Once data is written in the blockchain, every Ride based dApp's can access this data (via the getInteger(), getString(), getBinary() and getBoolean() methods) and use it for their calculations, e.g., decide on the amount of payouts, sending of transactions, winners of a contest, etc.
In our case, for example, to get WAVES/BTC data from the oracle, you need to specify the oracle address and the corresponding key in getInteger() method:
getInteger("3PPTrTo3AzR56N7ArzbU3Bpq9zYMgcf39Mk", "waves_btc_8")