This package provides utilities for consuming
- dynamic price/custom feeds
- automation services
- on-demand delivery of data
from the Orally Network Oracle using Solidity.
###Truffle/Hardhat
If you are using Truffle or Hardhat, simply install the NPM package:
npm install @orally-network/solidity-sdk
###Foundry
If you are using Foundry, you will need to create an NPM project if you don't already have one. From the root directory of your project, run:
npm init -y
npm install @orally-network/solidity-sdk
Then add the following line to your remappings.txt
file:
@orally-network/solidity-sdk/=node_modules/@orally-network/solidity-sdk
To consume prices you should use the IOrallyVerifierOracle
interface.
To verify price feed you should call the verifyPriceFeedWithCache
function with priceData
with a proof from HTTP Gateway:
https://tysiw-qaaaa-aaaak-qcikq-cai.icp0.io/get_xrc_data_with_proof?id=DOGE/SHIB&bytes=true
.
You will get PriceFeed
struct as a return with the price, decimals, and timestamp.
Later to read the latest price, you can call getPriceFeed
with the feed ID of the price feed you're interested in.
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;
import "@orally-network/solidity-sdk/IOrallyVerifierOracle.sol";
import "@orally-network/solidity-sdk/OrallyStructs.sol";
contract ExampleContract {
IOrallyVerifierOracle oracle;
constructor(address orallyVerifierOracleAddress) {
oracle = IOrallyVerifierOracle(orallyVerifierOracleAddress);
}
// price data from
// https://tysiw-qaaaa-aaaak-qcikq-cai.icp0.io/get_xrc_data_with_proof?id=DOGE/SHIB&bytes=true&api_key={YOUR_API_KEY}
function getDogeShibPrice(
bytes memory priceFeedData
) public view returns (OrallyStructs.PriceFeed memory) {
// Verify the price feed data and get the price, decimals, and timestamp.
OrallyStructs.PriceFeed priceFeed = oracle.verifyPriceFeed(priceFeedData);
// priceFeed.price is the price of DOGE/SHIB
// priceFeed.decimals is the number of decimals in the price
// priceFeed.timestamp is the timestamp when price feed was aggregated
return priceFeed;
}
// with cache
// https://tysiw-qaaaa-aaaak-qcikq-cai.icp0.io/get_xrc_data_with_proof?id=DOGE/SHIB&bytes=true&API_KEY={YOUR_API_KEY}
function updatePriceFeed(
bytes memory priceFeedData
) public returns (OrallyStructs.PriceFeed memory) {
// Verify the price feed data and get the price, decimals, and timestamp.
oracle.updatePriceFeed(priceFeedData);
// Get the price feed data from the cache.
OrallyStructs.PriceFeed priceFeed = oracle.getPriceFeed("DOGE/SHIB");
// priceFeed.price is the price of DOGE/SHIB
// priceFeed.decimals is the number of decimals in the price
// priceFeed.timestamp is the timestamp when price feed was aggregated
return priceFeed;
}
}
To verify if address holding some token on the side chain.
To consume side chain data you should use the IOrallyVerifierOracle
interface.
To verify side chain data you should call the verifyChainData
function with chainData
with a proof from HTTP Gateway:
https://tysiw-qaaaa-aaaak-qcikq-cai.icp0.io/read_contract_with_proof?chain_id=42161&function_signature="function balanceOf(address account) external view returns (uint256)"&contract_addr=0xA533f744B179F2431f5395978e391107DC76e103&method=balanceOf¶ms=(0x654DFF41D51c230FA400205A633101C5C1f1969C)&bytes=true
.
You will get (bytes data, bytes metaData)
as a return. metaData could be decoded as OrallyStructs.ReadContractMetadata
and data
depend on what you requested from side chain.
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;
import "@orally-network/solidity-sdk/IOrallyVerifierOracle.sol";
import "@orally-network/solidity-sdk/OrallyStructs.sol";
contract ExampleReadContract {
IOrallyVerifierOracle oracle;
constructor(address orallyVerifierOracleAddress) {
oracle = IOrallyVerifierOracle(orallyVerifierOracleAddress);
}
// chain data from
// https://tysiw-qaaaa-aaaak-qcikq-cai.icp0.io/read_contract_with_proof?chain_id=42161&function_signature="function balanceOf(address account) external view returns (uint256)"&contract_addr=0xA533f744B179F2431f5395978e391107DC76e103&method=balanceOf¶ms=(0x654DFF41D51c230FA400205A633101C5C1f1969C)&bytes=true
function getSideChainUserTokenBalance(
bytes calldata chainData
) public view returns (uint256) {
// Verify the chain data and get the balance of the user.
(bytes memory dataBytes, bytes memory metaBytes) = oracle.verifyReadContractData(chainData);
(uint256 balance) = abi.decode(dataBytes, (uint256));
(OrallyStructs.ReadContractMetadata memory meta) = abi.decode(metaBytes, (OrallyStructs.ReadContractMetadata));
// balance is the balance of the user of the requested token
// meta.chainId is the chain id of the side chain
// meta.contractAddr is the address of the contract on the side chain
// meta.method is the method that was called on the contract
// meta.params is the parameters that were passed to the method
return balance;
}
}
contract ExampleGetLogsContract {
IOrallyVerifierOracle oracle;
constructor(address orallyVerifierOracleAddress) {
oracle = IOrallyVerifierOracle(orallyVerifierOracleAddress);
}
// chain data from
// https://tysiw-qaaaa-aaaak-qcikq-cai.icp0.io/read_logs_with_proof?chain_id=42161&block_from=204767826&block_to=204767826&topics0=0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef&addresses=0xa533f744b179f2431f5395978e391107dc76e103&bytes=true
function getSideChainTransferLog(
bytes calldata chainData
) public view returns (uint256) {
// Verify the chain data and get the balance of the user.
(bytes memory dataBytes, bytes memory metaBytes) = oracle.verifyReadLogsData(chainData);
(OrallyStructs.ReadLogsData[] memory data) = abi.decode(dataBytes, (OrallyStructs.ReadLogsData[]));
(OrallyStructs.ReadLogsMetadata memory meta) = abi.decode(metaBytes, (OrallyStructs.ReadLogsMetadata));
(uint256 transferTokenAmount) = abi.decode(data[0].data, (uint256));
// transferTokenAmount is the amount of tokens transferred in the log
// meta.chainId is the chain id of the side chain
// meta.blockFrom is the block number from which the logs were read
// meta.blockTo is the block number to which the logs were read
// meta.topics0 is the topic of the log
// meta.addresses is the address of the contract from which the logs were read
// data[0].data is the data of the log
return balance;
}
}
To request feed from your contract you should use the IApolloCoordinator
interface.
You will receive requested data in fulfillData
callback. You should utilize ApolloReceiver
for that.
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;
import {ApolloReceiver} from "@orally-network/solidity-sdk/ApolloReceiver.sol";
contract RequestingPriceFeedExample is ApolloReceiver {
constructor(address _executorsRegistry, address _apolloCoordinator)
ApolloReceiver(_executorsRegistry, _apolloCoordinator) {}
// Example function to request a price feed
// `apolloCoordinator` is passing as public var from ApolloReceiver contract
function requestPriceFeed() public {
// Requesting the ARB/UNI price feed with a specified callback gas limit
uint256 requestId = apolloCoordinator.requestDataFeed("ARB/UNI", 100000);
}
// Overriding the fulfillData function to handle incoming data
function fulfillData(bytes memory data) internal override {
(
uint256 _requestId,
string memory _dataFeedId,
uint256 _rate,
uint256 _decimals,
uint256 _timestamp
) = abi.decode(data, (
uint256,
string,
uint256,
uint256,
uint256
));
// ...
}
}
contract RequestingRandomnessExample is ApolloReceiver {
constructor(address _executorsRegistry, address _apolloCoordinator)
ApolloReceiver(_executorsRegistry, _apolloCoordinator) {}
// Example function to request a price feed
// `apolloCoordinator` is passing as public var from ApolloReceiver contract
function requestRandomness() public {
// Requesting the randomness with a specified callback gas limit and number of random words
apolloCoordinator.requestRandomFeed(300000, 1);
}
// Overriding the fulfillData function to handle incoming data
function fulfillData(bytes memory data) internal override {
(, uint256[] memory randomWords) = abi.decode(data, (uint256, uint256[]));
// transform the result to a number between 1 and 100 inclusively
uint256 randomNumber = (randomWords[0] % 100) + 1;
// ...
}
}
To allow MPC wallet to make regular updates to your smart contract you should inherit from OrallyPythiaConsumer
.
You will utilize onlyExecutor(workflowId)
modifier to allow only correct workflow from MPC wallet to update data in your contract.
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;
import {OrallyPythiaConsumer} from "@orally-network/solidity-sdk/OrallyPythiaConsumer.sol";
contract ExampleContract is OrallyPythiaConsumer {
constructor(address _executorsRegistry)
OrallyPythiaConsumer(_executorsRegistry, msg.sender) {}
// Automation target method for receiving custom weather condition feed
function updateTemperature(
uint256 workflowId,
string memory _feedId,
uint256 _temperature,
uint256 _decimals,
uint256 _timestamp
) public onlyExecutor(workflowId) {
// do something with that data
}
// ...
}