Fetch multiple prices
Overview
Follow this tutorial to implement a more advanced example of handling HTTP requests with x/async
.
You'll create a contract that fetches prices of multiple cryptocurrencies from the CoinGecko API and extracts price values from the CBOR-encoded response.
Step 1. Create a contract
Create a new file MultiCoinPrices.sol
:
warden-http-examples/src/MultiCoinPrices.sol
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.25;
import "./interfaces/IAsync.sol";
import "./interfaces/Http.sol";
contract MultiCoinPrices {
uint64 public lastFutureId;
uint256 public bitcoinPrice;
uint256 public tetherPrice;
uint256 public uniswapPrice;
bytes public responseBody;
// Fetch different prices from the CoinGecko API:
function run() public returns (Http.Request memory request) {
request.url = "https://api.coingecko.com/api/v3/simple/price?ids=bitcoin,tether,uniswap&vs_currencies=usd";
request.method = "GET";
request.body = "";
lastFutureId = IASYNC_CONTRACT.addFuture("http", abi.encode(request), address(this));
bitcoinPrice = 0;
tetherPrice = 0;
uniswapPrice = 0;
}
// A callback function: the Warden node calls it automatically
function cb() external {
FutureByIdResponse memory future = IASYNC_CONTRACT.futureById(lastFutureId);
if (future.futureResponse.result.id == 0) {
revert("Not ready yet");
}
Http.Response memory response = abi.decode(future.futureResponse.result.output, (Http.Response));
require(response.status == 200, "HTTP request failed");
responseBody = response.body;
// The CBOR map structure: {"bitcoin":{"usd":value},"tether":{"usd":value},"uniswap":{"usd":value}}
// Skip the initial byte (map marker) and get to the first key
uint256 offset = 1;
offset = parseCoinPrice(response.body, offset, "bitcoin");
offset = parseCoinPrice(response.body, offset, "tether");
parseCoinPrice(response.body, offset, "uniswap");
}
// A helper function for parsing a coin price from CBOR data
function parseCoinPrice(bytes memory data, uint256 startOffset, string memory coinName) private returns (uint256) {
// Skip the coin name string: we don't need to validate it as we're following a known structure
uint256 offset = skipCborString(data, startOffset);
// Skip the map marker for the inner map
offset += 1;
// Skip the "usd" string
offset = skipCborString(data, offset);
// Read the price value, which is encoded as a float in CBOR
if (data[offset] == 0xfb) {
// A double precision float marker
offset += 1; // Skip the marker
// Extract the 8 bytes of the float
uint64 bits;
for (uint256 i = 0; i < 8; i++) {
bits = (bits << 8) | uint64(uint8(data[offset + i]));
}
// Convert to a fixed point representation (multiply by 1e6 to preserve decimals)
uint256 price = uint256(convertFloatBitsToUint(bits));
// Store the price in an appropriate variable
if (keccak256(bytes(coinName)) == keccak256(bytes("bitcoin"))) {
bitcoinPrice = price;
} else if (keccak256(bytes(coinName)) == keccak256(bytes("tether"))) {
tetherPrice = price;
} else if (keccak256(bytes(coinName)) == keccak256(bytes("uniswap"))) {
uniswapPrice = price;
}
return offset + 8; // Return the new offset after the float
}
return offset; // Fallback if not a float
}
// A helper function for skipping the CBOR string and returning the new offset
function skipCborString(bytes memory data, uint256 offset) private pure returns (uint256) {
uint8 initialByte = uint8(data[offset]);
uint8 majorType = initialByte >> 5;
uint8 additionalInfo = initialByte & 0x1f;
require(majorType == 3, "Expected string type"); // The string major type is 3
offset += 1; // Skip the initial byte
uint256 stringLength;
if (additionalInfo < 24) {
stringLength = additionalInfo;
} else if (additionalInfo == 24) {
// 1-byte length
stringLength = uint8(data[offset]);
offset += 1;
} else if (additionalInfo == 25) {
// 2-byte length
stringLength = uint16(uint8(data[offset])) << 8 | uint16(uint8(data[offset + 1]));
offset += 2;
}
return offset + stringLength; // Skip the string content
}
// Convert IEEE 754 double precision float bits to uint
// This is a simplified conversion that works for our specific use case
function convertFloatBitsToUint(uint64 bits) private pure returns (uint256) {
// Extract components of the IEEE 754 double
uint64 sign = bits >> 63;
uint64 exponent = (bits >> 52) & 0x7FF;
uint64 mantissa = bits & 0xFFFFFFFFFFFFF;
// Handle special cases
if (exponent == 0) {
return 0; // Zero or subnormal
}
// For normal numbers
uint256 value = (1 << 52) | mantissa; // Add implicit leading 1
// Adjust by exponent (bias is 1023)
int256 adjustedExp = int256(uint256(exponent)) - 1023;
// Scale to a fixed point with 6 decimal places (multiply by 1e6)
uint256 scaledValue;
if (adjustedExp > 0) {
scaledValue = value << uint256(adjustedExp);
} else if (adjustedExp < 0) {
scaledValue = value >> uint256(-adjustedExp);
} else {
scaledValue = value;
}
// Apply scaling factor to get 6 decimal places
scaledValue = scaledValue * 1000000 / (1 << 52);
// Apply sign
if (sign == 1) {
return 0; // We don't expect negative prices
}
return scaledValue;
}
}
Step 2. Deploy and test
-
Deploy the contract:
forge create --rpc-url $RPC_URL --private-key $PRIVATE_KEY \
src/MultiCoinPrices.sol:MultiCoinPrices --broadcast -
Note down the value returned as
Deployed to
and set it as an environment variable:export CONTRACT_ADDRESS=my-contract-address
-
Make an HTTP request by calling the
run()
function:cast send $CONTRACT_ADDRESS "run()" \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL -
Wait a few seconds, then use the
cb()
function to process the result:cast send $CONTRACT_ADDRESS "cb()" \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL -
Check the prices:
cast call $CONTRACT_ADDRESS "bitcoinPrice()(uint256)" --rpc-url $RPC_URL
cast call $CONTRACT_ADDRESS "tetherPrice()(uint256)" --rpc-url $RPC_URL
cast call $CONTRACT_ADDRESS "uniswapPrice()(uint256)" --rpc-url $RPC_URLIn the output, you'll see the price of each cryptocurrency displayed with 6 decimal places of precision.
Next steps
Learn how to use other Plugins: