Using the Push model means retrieving data from on-chain contract storage. This requires the data to exist in the on-chain contract storage, uploaded by a publisher or node operator beforehand. Entangle maintains a node that continuously publishes updates for supported assets, but anyone is free to run their own node instance to ensure redundancy or customize their setup.
To utilize an update in your contract, you can read the value of the asset of interest from the UDFOracle contract deployed on each supported chain. For demonstration purposes, let's create a consumer contract whose logic depends on the NGL/USD price.
Step #1: Initialize hardhat project
First of all, let's create new project and initialize hardhat in it.
Step #2: Create PushReader contract and hardhat ignition script
Next, let's create our consumer contract. The consumer contract, referred to as the PushReader, will retrieve data stored in PushMarketplace, which assumes the availability of verified updates pushed on-chain beforehand.
For this example, the contract will read the latest stored price of the NGL/USD asset and store it. To achieve this, we'll use the getFeedPrice function, a free data retrieval method for previously subscribed readers. For convenience, this function returns both the price and the timestamp in a single call. Providing both values ensures that users can verify the freshness of the data, which is critical for applications where price accuracy and timing are essential for decision-making.
./contracts/PushReader.sol
// SPDX-License-Identifier: BSL1.1
pragma solidity ^0.8.20;
interface IPushMarketplace {
function getFeedPrice(
bytes32 feed
) external returns (uint256 price, uint256 timestamp);
}
contract PushReader {
IPushMarketplace public pushMarketplace;
uint public latestPrice;
uint public latestTimestamp;
constructor(address _pushMarketplace){
pushMarketplace = IPushMarketplace(_pushMarketplace);
}
function getData(bytes32 feedKey) external{
(latestPrice, latestTimestamp) = pushMarketplace.getFeedPrice(feedKey);
}
}
Next, create an ignition script to help with deployment of the new PushReader contract.
To deploy your contract on testnet, you'll need ETH eth_sepolia testnet tokens. In case you don't have ETH on eth_sepolia, you can spin up a local development fork of eth_sepolia by using anvil node like this:
To validate that we have our price, we will write an example backend script that send transaction to execute PushConsumer.consumePrice. Let's create a separate directory scripts and our new script consumePrice.ts to do just that.
import {
PushReader,
} from "../typechain-types";
import { ethers } from "hardhat";
// PushConsumer address we got from the deployment
const PushReaderAddress = "0xe23650424625B602c0f996F25b287203632A3f16";
async function main() {
const pushConsumer = await ethers.getContractAt(
"PushReader",
PushReaderAddress
) as PushReader
;
const feedKey = ethers.encodeBytes32String("NGL/USD");
let tx = await pushReader.getFeedPrice(feedKey);
await tx.wait();
console.log("sent PushReader.getData tx", tx.hash);
let price = await pushReader.latestPrice();
let timestamp = await pushReader.latestTimestamp();
console.log("price:", price );
console.log("timestamp:", timestamp );
}
main()
.then(() => process.exit(0))
.catch(error => {
console.error(error);
process.exit(1);
});
Change the PushReaderAddress to the address you got from deployment and execute the script and execute this script to send the transaction to the blockchain.
Copy
$ yarn hardhat run ./scripts/consumePrice.ts --network eth_sepolia
sent PushReader.getData tx 0xcb2cc0657759caaa2a0b70daaae61673f5f7a4202b9f7033d1b48c38ef7a508d
price: 1060140419373985n
timestamp: 1743088059n
You can use cast to examine the transaction calltrace and logs. The emitted event shows the latest update that we verified.
Note that gas usage may vary depending on the result and the version of the library in use.
cast run 0xcb2cc0657759caaa2a0b70daaae61673f5f7a4202b9f7033d1b48c38ef7a508d -r https://ethereum-sepolia-rpc.publicnode.com