Implement HTTP requests
Overview
This tutorial explains how implement simple HTTP requests to multiple APIs:
- The CoinGecko API
- The GitHub API
- The JSONPlaceholder API
Step 1. Create a contract
Create a new file called MultiApiTest.sol
:
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.25;
import "./interfaces/IAsync.sol";
import "./interfaces/Http.sol";
contract MultiApiTest {
uint64 public lastFutureId;
bytes public responseBody;
uint256 public statusCode;
string public currentApi;
// Try the CoinGecko API: fetch the current Bitcoin price
function tryCoinGecko() public returns (Http.Request memory request) {
currentApi = "CoinGecko";
request.url = "https://api.coingecko.com/api/v3/simple/price?ids=bitcoin&vs_currencies=usd";
request.method = "GET";
request.body = "";
lastFutureId = IASYNC_CONTRACT.addFuture("http", abi.encode(request), address(this));
}
// Try the GitHub API
function tryGitHub() public returns (Http.Request memory request) {
currentApi = "GitHub";
request.url = "https://api.github.com/users/ethereum";
request.method = "GET";
request.body = "";
lastFutureId = IASYNC_CONTRACT.addFuture("http", abi.encode(request), address(this));
}
// Try the JSONPlaceholder API
function tryJSONPlaceholder() public returns (Http.Request memory request) {
currentApi = "JSONPlaceholder";
request.url = "https://jsonplaceholder.typicode.com/posts/1";
request.method = "GET";
request.body = "";
lastFutureId = IASYNC_CONTRACT.addFuture("http", abi.encode(request), address(this));
}
// Try OpenWeatherMap API
function tryOpenWeather() public returns (Http.Request memory request) {
currentApi = "OpenWeather";
request.url = "https://api.openweathermap.org/data/2.5/weather?q=London&appid=b6907d289e10d714a6e88b30761fae22";
request.method = "GET";
request.body = "";
lastFutureId = IASYNC_CONTRACT.addFuture("http", abi.encode(request), address(this));
}
// Check if the response is ready
function isReady() public view returns (bool) {
FutureByIdResponse memory future = IASYNC_CONTRACT.futureById(lastFutureId);
return future.futureResponse.result.id != 0;
}
// Try to process the response without reverting
function tryProcess() public returns (bool) {
FutureByIdResponse memory future = IASYNC_CONTRACT.futureById(lastFutureId);
if (future.futureResponse.result.id == 0) {
return false; // Not ready yet
}
// Decode the response
Http.Response memory response = abi.decode(future.futureResponse.result.output, (Http.Response));
// Store the response data
statusCode = response.status;
responseBody = response.body;
return true;
}
// A callback function: the precompile will call it when the response is ready
function cb() external {
FutureByIdResponse memory future = IASYNC_CONTRACT.futureById(lastFutureId);
if (future.futureResponse.result.id == 0) {
return; // Not ready yet
}
// Decode the response
Http.Response memory response = abi.decode(future.futureResponse.result.output, (Http.Response));
// Store the response data
statusCode = response.status;
responseBody = response.body;
}
// Get the response as a string for easier reading
function getResponseAsString() public view returns (string memory) {
return string(responseBody);
}
}
Step 2. Deploy and test
2.1. Deploy
-
Deploy the contract:
forge create --rpc-url $RPC_URL --private-key $PRIVATE_KEY \
src/MultiApiTest.sol:MultiApiTest --broadcast -
Note down the value returned as
Deployed to
and set it as an environment variable:export CONTRACT_ADDRESS=my-contract-address
2.2. Test the CoinGecko API
Test HTTP requests to the CoinGecko API:
-
Call the
tryCoinGecko()
function. It'll fetch the current Bitcoin price.cast send $CONTRACT_ADDRESS "tryCoinGecko()" \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL -
Wait a few seconds. Now you can check whether the response is ready by calling
isReady()
(this step is optional):cast call $CONTRACT_ADDRESS "isReady()(bool)" --rpc-url $RPC_URL
The expected output is the following:
true
-
Call
tryProcess()
. This function will process the response without reverting in case it's not ready yet.cast send $CONTRACT_ADDRESS "tryProcess()" \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL -
Finally, get the status code by calling
statusCode()
:cast call $CONTRACT_ADDRESS "statusCode()(uint256)" --rpc-url $RPC_URL
The expected output is the following:
200
-
You can also try to get the response as a string:
cast call $CONTRACT_ADDRESS "getResponseAsString()(string)" --rpc-url $RPC_URL
This will print a CBOR-encoded output:
"�gbitcoin�cusd�@��P\0\0\0\0"
2.3. Test the GitHub API
Test requests to the GitHub API in a similar way:
cast send $CONTRACT_ADDRESS "tryGitHub()" \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
cast send $CONTRACT_ADDRESS "tryProcess()"
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
cast call $CONTRACT_ADDRESS "statusCode()(uint256)" --rpc-url $RPC_URL
The expected output is the following:
200
2.4. Test the JSONPlaceholder API
Test requests to the JSONPlaceholder API:
cast send $CONTRACT_ADDRESS "tryJSONPlaceholder()" \
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
cast send $CONTRACT_ADDRESS "tryProcess()"
--private-key $PRIVATE_KEY \
--rpc-url $RPC_URL
cast call $CONTRACT_ADDRESS "statusCode()(uint256)" --rpc-url $RPC_URL
This might fail with the following error:
Error: server returned a null response when a non-null response was expected
If you're experiencing troubles querying a particular API with x/async
, please contact us in Discord.
Next steps
When you made a request to the CoinGecko API, you received a CBOR-encoded output. To learn how to extract data from such responses, follow the next guide: Extract data.