diff --git a/docs/building-on-lisk/using-oracle-data/redstone-pull.md b/docs/building-on-lisk/using-oracle-data/redstone-pull.md index 2c24b7b9c..1d39080e7 100644 --- a/docs/building-on-lisk/using-oracle-data/redstone-pull.md +++ b/docs/building-on-lisk/using-oracle-data/redstone-pull.md @@ -1,7 +1,7 @@ --- title: ...with Redstone (Pull) slug: /building-on-lisk/using-oracle-data/redstone-pull -description: A guide on using Redstone Data Feeds to access real-world data such as asset prices, directly from your smart contracts on the Lisk testnet. +description: A guide on using Redstone Pull to access real-world data such as asset prices, directly from your smart contracts on the Lisk testnet. keywords: [ Oracle Oracles, @@ -17,10 +17,11 @@ keywords: [ dapp development, build a dapp on Lisk, build on Lisk, + Redstone Pull, ] --- -# Accessing real-world data using the Redstone Oracle +# Accessing real-world data using the Redstone Oracle (Pull) This page will explain how you can access oracle data using Redstone. diff --git a/docs/building-on-lisk/using-oracle-data/redstone-push.md b/docs/building-on-lisk/using-oracle-data/redstone-push.md new file mode 100644 index 000000000..13630433e --- /dev/null +++ b/docs/building-on-lisk/using-oracle-data/redstone-push.md @@ -0,0 +1,24 @@ +--- +title: ...with Redstone (Push) +slug: /building-on-lisk/using-oracle-data/redstone-push +description: A guide on using Redstone Push to access real-world data such as asset prices, directly from your smart contracts on the Lisk testnet. +keywords: [ + Oracle + Oracles, + Redstone, + price feeds, + data feeds, + smart contract, + Lisk blockchain, + Lisk network, + Lisk testnet, + Lisk test network, + app development, + dapp development, + build a dapp on Lisk, + build on Lisk, + Redstone Push, + ] +--- + +# Accessing real-world data using the Redstone Oracle (Push) \ No newline at end of file diff --git a/i18n/ind/docusaurus-plugin-content-docs/current/building-on-lisk/use-oracle-data.md b/i18n/ind/docusaurus-plugin-content-docs/current/building-on-lisk/using-oracle-data/redstone-pull.md similarity index 85% rename from i18n/ind/docusaurus-plugin-content-docs/current/building-on-lisk/use-oracle-data.md rename to i18n/ind/docusaurus-plugin-content-docs/current/building-on-lisk/using-oracle-data/redstone-pull.md index f682e1e88..9c59f22d7 100644 --- a/i18n/ind/docusaurus-plugin-content-docs/current/building-on-lisk/use-oracle-data.md +++ b/i18n/ind/docusaurus-plugin-content-docs/current/building-on-lisk/using-oracle-data/redstone-pull.md @@ -1,7 +1,7 @@ --- -title: Accessing real-world data with Oracles -slug: /building-on-lisk/use-oracle-data -description: A guide on using Redstone Data Feeds to access real-world data such as asset prices, directly from your smart contracts on the Lisk testnet. +title: ...with Redstone (Pull) +slug: /building-on-lisk/using-oracle-data/redstone-pull +description: A guide on using Redstone Pull to access real-world data such as asset prices, directly from your smart contracts on the Lisk blockchain. keywords: [ Oracle Oracles, @@ -17,12 +17,13 @@ keywords: [ dapp development, build a dapp on Lisk, build on Lisk, + Redstone Pull, ] --- -# Accessing real-world data using the Redstone Oracle +# Accessing real-world data using the Redstone Oracle (Pull) -This page will explain how you can access real world / off-chain data using Oracles such as Redstone. +This page will explain how you can access oracle data using [Redstone Pull](https://docs.redstone.finance/docs/get-started/models/redstone-pull/). RedStone is a data ecosystem that delivers frequently updated, reliable, and diverse data for your dApp and smart contracts deployed on Lisk. @@ -32,7 +33,7 @@ To create a smart contract that directly fetches the latest data from the Redsto This guide uses the [Redstone Pull model](https://docs.redstone.finance/docs/get-started/models/redstone-pull) to fetch the data. -For an overview of the different modules that Redstone offers to receive oracle data, go to [Oracles > Redstone](../lisk-tools/oracles#redstone). +For an overview of the different modules that Redstone offers to receive oracle data, go to [Oracles > Redstone](../../lisk-tools/oracles#redstone). [Hardhat](https://hardhat.org/) is used in this guide to create the smart contract. In case you want to use Foundry, check out the [Redstone docs](https://docs.redstone.finance/docs/get-started/models/redstone-pull#foundry) for instructions. @@ -158,5 +159,5 @@ Latest ETH price: To deploy the smart contract on Lisk Sepolia or Lisk Mainnet, follow the guides -- [Deploying a smart contract with Hardhat](deploying-smart-contract/with-Hardhat.md), or -- [Deploying a smart contract with Foundry](deploying-smart-contract/with-Foundry.md) \ No newline at end of file +- [Deploying a smart contract with Hardhat](../deploying-smart-contract/with-Hardhat), or +- [Deploying a smart contract with Foundry](../deploying-smart-contract/with-Foundry) \ No newline at end of file diff --git a/i18n/ind/docusaurus-plugin-content-docs/current/building-on-lisk/using-oracle-data/redstone-push.md b/i18n/ind/docusaurus-plugin-content-docs/current/building-on-lisk/using-oracle-data/redstone-push.md new file mode 100644 index 000000000..6857475e1 --- /dev/null +++ b/i18n/ind/docusaurus-plugin-content-docs/current/building-on-lisk/using-oracle-data/redstone-push.md @@ -0,0 +1,189 @@ +--- +title: ...with Redstone (Push) +slug: /building-on-lisk/using-oracle-data/redstone-push +description: A guide on using Redstone Push to access real-world data such as asset prices, directly from your smart contracts on the Lisk blockchain. +keywords: [ + Oracle + Oracles, + Redstone, + price feeds, + data feeds, + smart contract, + Lisk blockchain, + Lisk network, + Lisk testnet, + Lisk test network, + app development, + dapp development, + build a dapp on Lisk, + build on Lisk, + Redstone Push, + ] +--- + +# Accessing real-world data using the Redstone Oracle (Push) + +This page will explain how you can access oracle data using [Redstone Push](https://docs.redstone.finance/docs/get-started/models/redstone-push). + +RedStone is a data ecosystem that delivers frequently updated, reliable, and diverse data for your dApp and smart contracts deployed on Lisk. + + +## Import + +To use the RedStone data in your contract, import the [AggregatorV3Interface](https://docs.chain.link/data-feeds/using-data-feeds#solidity) from Chainlink. + +For every data feed you like to store, create a new constant with type `AggregatorV3Interface`. + +In the constructor, set the above defined constants to point to the respective data feeds: +Use the `AggregatorV3Interface()` function and pass the adress of the respective data feed contract as parameter. + +```solidity +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.28; + +import {AggregatorV3Interface} from "@chainlink/contracts/src/v0.8/shared/interfaces/AggregatorV3Interface.sol"; + +contract RedStoneDataConsumer { + AggregatorV3Interface internal dataFeedETH; + AggregatorV3Interface internal dataFeedLSK; + AggregatorV3Interface internal dataFeedUSDT; + + /** + * Network: Lisk + * Aggregator: ETH/USD + * Address: 0x6b7AB4213c77A671Fc7AEe8eB23C9961fDdaB3b2 + */ + /** + * Network: Lisk + * Aggregator: LSK/USD + * Address: 0xa1EbA9E63ed7BA328fE0778cFD67699F05378a96 + */ + /** + * Network: Lisk + * Aggregator: USDT/USD + * Address: 0xd2176Dd57D1e200c0A8ec9e575A129b511DBD3AD + */ + constructor() { + dataFeedETH = AggregatorV3Interface( + 0x6b7AB4213c77A671Fc7AEe8eB23C9961fDdaB3b2 + ); + dataFeedLSK = AggregatorV3Interface( + 0xa1EbA9E63ed7BA328fE0778cFD67699F05378a96 + ); + dataFeedUSDT = AggregatorV3Interface( + 0xd2176Dd57D1e200c0A8ec9e575A129b511DBD3AD + ); + } +} +``` + +## Reading data +To read the data of the price feeds, we define the following functions in the contract: + +- `getRedStoneETHDataFeedLatestAnswer()` +- `getRedStoneLSKDataFeedLatestAnswer()` +- `getRedStoneUSDTDataFeedLatestAnswer()` + +Inside of the functions, call the [latestRoundData](https://docs.chain.link/data-feeds/api-reference#latestrounddata) on the respective price feeds to receive the latest price feeds for the respective token. + +The `latestRoundData()` function returns the following values: + +- `roundId`(uint80): The round ID. +The "round" the answer was created in. +Every time a price feed is updated, it adds +1 to the round ID. +- `answer`(int256): The data that this specific feed provides. +Depending on the feed you selected, this answer provides asset prices, reserves, and other types of data. +- `startedAt`(uint256): Timestamp of when the round started. +- `updatedAt`(uint256): Timestamp of when the round was updated. +- `answeredInRound`(uint80): Deprecated - Previously used when answers could take multiple rounds to be computed. + +In this example, we will only use `answer` and `updatedAt`. +The `updatedAt` value should be used to make sure that the `answer` is recent enough for your application to use it. +You can compare `updatedAt` to the latest block time (`uint256 currentTime = block.timestamp;`) to ensure you are only using the latest oracle data in your application. + +```solidity +// SPDX-License-Identifier: MIT +pragma solidity ^0.8.28; + +/** + * THIS IS AN EXAMPLE CONTRACT THAT USES HARDCODED + * VALUES FOR CLARITY. + * THIS IS AN EXAMPLE CONTRACT THAT USES UN-AUDITED CODE. + * DO NOT USE THIS CODE IN PRODUCTION. + */ + +import {AggregatorV3Interface} from "@chainlink/contracts/src/v0.8/shared/interfaces/AggregatorV3Interface.sol"; + +contract DataConsumerV3 { + AggregatorV3Interface internal priceFeedETH; + AggregatorV3Interface internal priceFeedLSK; + AggregatorV3Interface internal priceFeedUSDT; + + /** + * Network: Lisk + * Aggregator: ETH/USD + * Address: 0x6b7AB4213c77A671Fc7AEe8eB23C9961fDdaB3b2 + */ + /** + * Network: Lisk + * Aggregator: LSK/USD + * Address: 0xa1EbA9E63ed7BA328fE0778cFD67699F05378a96 + */ + /** + * Network: Lisk + * Aggregator: USDT/USD + * Address: 0xd2176Dd57D1e200c0A8ec9e575A129b511DBD3AD + */ + constructor() { + priceFeedETH = AggregatorV3Interface( + 0x6b7AB4213c77A671Fc7AEe8eB23C9961fDdaB3b2 + ); + priceFeedLSK = AggregatorV3Interface( + 0xa1EbA9E63ed7BA328fE0778cFD67699F05378a96 + ); + priceFeedUSDT = AggregatorV3Interface( + 0xd2176Dd57D1e200c0A8ec9e575A129b511DBD3AD + ); + } + + /** + * Returns the latest ETH price, and when it was updated. + */ + function getRedStoneETHDataFeedLatestAnswer() public view returns (int, uint) { + ( + /* uint80 roundID */, + int answer, + uint updatedAt, + /*uint startedAt*/, + /*uint80 answeredInRound*/ + ) = priceFeedETH.latestRoundData(); + return (answer, updatedAt); + } + /** + * Returns the latest LSK price, and when it was updated. + */ + function getRedStoneLSKDataFeedLatestAnswer() public view returns (int, uint) { + ( + /* uint80 roundID */, + int answer, + uint updatedAt, + /*uint startedAt*/, + /*uint80 answeredInRound*/ + ) = priceFeedLSK.latestRoundData(); + return (answer, updatedAt); + } + /** + * Returns the latest USDT price, and when it was updated. + */ + function getRedStoneUSDTDataFeedLatestAnswer() public view returns (int, uint) { + ( + /* uint80 roundID */, + int answer, + uint updatedAt, + /*uint startedAt*/, + /*uint80 answeredInRound*/ + ) = priceFeedUSDT.latestRoundData(); + return (answer, updatedAt); + } +} +``` diff --git a/i18n/ind/docusaurus-plugin-content-docs/current/building-on-lisk/using-oracle-data/tellor.md b/i18n/ind/docusaurus-plugin-content-docs/current/building-on-lisk/using-oracle-data/tellor.md new file mode 100644 index 000000000..2988d0431 --- /dev/null +++ b/i18n/ind/docusaurus-plugin-content-docs/current/building-on-lisk/using-oracle-data/tellor.md @@ -0,0 +1,207 @@ +--- +title: ...with Tellor +slug: /building-on-lisk/using-oracle-data/tellor +description: A guide on using Tellor Data Feeds to access real-world data such as asset prices, directly from your smart contracts on the Lisk blockchain. +keywords: [ + Oracle + Oracles, + Redstone, + price feeds, + data feeds, + smart contract, + Lisk blockchain, + Lisk network, + Lisk testnet, + Lisk test network, + app development, + dapp development, + build a dapp on Lisk, + build on Lisk, + ] +--- + +# Using oracle data with Tellor +This page will explain how you can access oracle data using Tellor. + +[Tellor](https://tellor.io/) is an immutable decentralized oracle protocol where parties can request the value of an offchain data point (e.g. ETH/USD, LSK/USD) and reporters compete to add this value to an onchain databank. +The inputs to this databank are secured by a network of staked reporters. + +Tellor utilizes crypto-economic incentive mechanisms, rewarding honest data submissions by reporters and punishing bad actors through the issuance of Tellor’s token, Tributes (TRB) and a dispute mechanism. + +This incentivizes an open, permissionless network of data reporting and data validation, ensuring that data can be provided by anyone and checked by everyone. + +## Installation +The first thing you'll want to do is install the basic tools necessary for using Tellor as your oracle. + +To install [usingtellor](https://github.com/tellor-io/usingtellor), run one the following commands: + +- Hardhat: `npm install usingtellor` +- Foundry: `forge install tellor-io/usingtellor` + +Once installed, this will allow your contracts to inherit the functions from the 'UsingTellor' contract. + +Great! Now that you've got the tools ready, let's go through a simple exercise where we retrieve the `eth/usd` and `lsk/usd` prices from Tellor. + +## Import + +```solidity +pragma solidity >=0.8.0; + +import "usingtellor/contracts/UsingTellor.sol"; +/** + * Network: Lisk Mainnet + * Address: 0x896419Ed2E0dC848a1f7d2814F4e5Df4b9B9bFcc +*/ +contract MyContract is UsingTellor { + + constructor(address payable _tellorOracle) UsingTellor(_tellorOracle) { + + } + + // ... + +} +``` + +To import the UsingTellor contract into your Solidity file, pass the desired Tellor Oracle address as a parameter. +For the Lisk Mainnet, the Tellor Oracle address is: [0x896419Ed2E0dC848a1f7d2814F4e5Df4b9B9bFcc](https://blockscout.lisk.com/address/0x896419Ed2E0dC848a1f7d2814F4e5Df4b9B9bFcc) + +## Reading data + +In the example below, we add two functions: + +- `getETHSpotPrice()` that reads the ETH/USD price feed and another function +- `getLSKSpotPrice()` that reads the LSK/USD price feed from the Oracle + +```solidity +// SPDX-License-Identifier: MIT +pragma solidity >=0.8.0; + +import { UsingTellor } from "usingtellor/contracts/UsingTellor.sol"; + +contract MyContract is UsingTellor { + + uint256 public ethLastStoredTimestamp; + uint256 public ethLastStoredPrice; + uint256 public lskLastStoredTimestamp; + uint256 public lskLastStoredPrice; + bytes32 public immutable ethQueryId; + bytes32 public immutable lskQueryId; + uint256 public constant DISPUTE_BUFFER = 20 minutes; + uint256 public constant STALENESS_AGE = 12 hours; + + error StalePrice(uint256 price, uint256 timestamp); + error NoValueRetrieved(uint256 timestamp); + + /** + * @dev the constructor sets the Tellor address and the ETH and LSK queryIds + * @param _tellorOracle is the address of the Tellor oracle + */ + constructor (address payable _tellorOracle) UsingTellor(_tellorOracle) { + // set the ETH queryId + bytes memory _ethQueryData = abi.encode("SpotPrice", abi.encode("eth", "usd")); + ethQueryId = keccak256(_ethQueryData); + // set the LSK queryId + bytes memory _lskQueryData = abi.encode("SpotPrice", abi.encode("lsk", "usd")); + lskQueryId = keccak256(_lskQueryData); + } + + /** + * @dev Allows a user contract to read the ETH price from Tellor and perform some + * best practice checks on the retrieved data + * @return _value the ETH spot price from Tellor, with 18 decimal places + * @return timestamp the value's timestamp + */ + function getETHSpotPrice() + public + returns ( + uint256 _value, + uint256 timestamp + ) + { + // retrieve the most recent 20+ minute old ETH price. + // the buffer allows time for a bad value to be disputed + (bytes memory _data, uint256 _timestamp) = _getDataBefore(ethQueryId, block.timestamp - DISPUTE_BUFFER); + + // check whether any value was retrieved + if (_timestamp == 0 || _data.length == 0) revert NoValueRetrieved(_timestamp); + + // decode the value from bytes to uint256 + _value = abi.decode(_data, (uint256)); + + // prevent a back-in-time dispute attack by caching the most recent value and timestamp. + // this stops an attacker from disputing tellor values to manupulate which price is used + // by your protocol + if (_timestamp > ethLastStoredTimestamp) { + // if the new value is newer than the last stored value, update the cache + ethLastStoredTimestamp = _timestamp; + ethLastStoredPrice = _value; + } else { + // if the new value is older than the last stored value, use the cached value + _value = ethLastStoredPrice; + _timestamp = ethLastStoredTimestamp; + } + + // check whether value is too old + if (block.timestamp - _timestamp > STALENESS_AGE) revert StalePrice(_value, _timestamp); + + // return the value and timestamp + return (_value, _timestamp); + } + + /** + * @dev Allows a user contract to read the LSK price from Tellor and perform some + * best practice checks on the retrieved data + * @return _value the LSK spot price from Tellor, with 18 decimal places + * @return timestamp the value's timestamp + */ + function getLSKSpotPrice() + public + returns ( + uint256 _value, + uint256 timestamp + ) + { + (bytes memory _data, uint256 _timestamp) = _getDataBefore(lskQueryId, block.timestamp - DISPUTE_BUFFER); + + if (_timestamp == 0 || _data.length == 0) revert NoValueRetrieved(_timestamp); + + _value = abi.decode(_data, (uint256)); + + if (_timestamp > lskLastStoredTimestamp) { + lskLastStoredTimestamp = _timestamp; + lskLastStoredPrice = _value; + } else { + _value = lskLastStoredPrice; + _timestamp = lskLastStoredTimestamp; + } + + if (block.timestamp - _timestamp > STALENESS_AGE) revert StalePrice(_value, _timestamp); + + return (_value, _timestamp); + } +} +``` + +You can adapt this contract to your needs. +The example utilizes some best practices[^1] for using Tellor by implementing a dispute time buffer and a data staleness check. +In addition, it also seeks to mitigate back-in-time dispute attacks by caching the most recent value and timestamp. + +[^1]: Based on examples in [Tellor's best practices repository](https://github.com/tellor-io/best-practices-user/tree/main) + +:::tip +For a general overview of best practices using Tellor, go to the [User checklists](https://docs.tellor.io/tellor/getting-data/user-checklists) in the Tellor docs. +::: + +## Deploying on Lisk + +To deploy the smart contract on Lisk Sepolia or Lisk Mainnet, follow the guides + +- [Deploying a smart contract with Hardhat](../deploying-smart-contract/with-Hardhat), or +- [Deploying a smart contract with Foundry](../deploying-smart-contract/with-Foundry) + +## Further resources + +- For a more robust implementation of the Tellor oracle, check out the full list of available functions [here](https://github.com/tellor-io/usingtellor/blob/master/README.md). +- Have a specific Data Feed Request? [Fill out this form](https://github.com/tellor-io/dataSpecs/issues/new?assignees=&labels=&template=new_query_type.yaml&title=%5BNew+Data+Request+Form%5D%3A+). +- Still have question? Reach out to the Tellor team on Discord [here](https://discord.gg/tellor). \ No newline at end of file diff --git a/sidebars.js b/sidebars.js index 0a199518b..1ba87194b 100644 --- a/sidebars.js +++ b/sidebars.js @@ -116,6 +116,7 @@ const sidebars = { items: [ 'building-on-lisk/using-oracle-data/tellor', 'building-on-lisk/using-oracle-data/redstone-pull', + 'building-on-lisk/using-oracle-data/redstone-push', ], } ],