Search
API reference for ChainlinkClient
contract.
ChainlinkClient
contracts can communicate with legacy Oracle
contracts or Operator
contracts.
Name | Description |
---|---|
setChainlinkOracle | Sets the stored address for the oracle contract |
setChainlinkToken | Sets the stored address for the LINK token |
buildChainlinkRequest | Instantiates a Request object with the required parameters |
buildOperatorRequest | Instantiates a Request object with the required parameters. Note the oracle must be an Operator contract |
sendChainlinkRequest | Sends the request payload to the stored address stored as chainlinkOracleAddress |
sendChainlinkRequestTo | Sends a request to the oracle address specified |
sendOperatorRequest | Sends the request payload to the stored address stored as chainlinkOracleAddress. Note the oracle must be an Operator contract |
sendOperatorRequestTo | Sends a request to the oracle address specified. Note the oracle must be an Operator contract |
validateChainlinkCallback | Secures the fulfillment callback to make sure it is only called by permissioned senders |
addChainlinkExternalRequest | Allows a Chainlinked contract to track unfulfilled requests that it hasn't created itself |
cancelChainlinkRequest | Cancels Chainlink requests attempting to contact an unresponsive node |
useChainlinkWithENS | Looks up the addresses of the LINK token and Oracle contract through ENS |
updateChainlinkOracleWithENS | Updates the stored oracle address with the latest address resolved through ENS |
chainlinkTokenAddress | Returns the stored address of the LINK token |
chainlinkOracleAddress | Returns the stored address of the oracle contract |
Name | Description |
---|---|
ChainlinkRequested | Emitted from a Chainlinked contract when a request is sent to an oracle |
ChainlinkFulfilled | Emitted from a Chainlinked contract when a request is fulfilled by an oracle |
ChainlinkCancelled | Emitted from a Chainlinked contract when a request is cancelled |
Name | Description |
---|---|
recordChainlinkFulfillment | Used on fulfillment callbacks to ensure that the caller and requestId are valid. This is the modifier equivalent of the method validateChainlinkCallback |
Name | Description |
---|---|
LINK_DIVISIBILITY | Helper uint256 to represent the divisibility of a LINK token. Equivalent to 10^18 |
Name | Description |
---|---|
Chainlink.Request | All of the parameters that can be passed over in a Chainlink request |
Below you'll find each helper explained in greater detail alongside respective implementation examples to help you leverage these methods once you start building your own Chainlinked contract.
After the function signature and a short description, two code examples are provided, one focusing on the exact usage of the method and one where the helper is presented in the context of a full contract.
function setChainlinkOracle(
address _oracle
)
Sets a private storage variable provided for convenience if your contract only needs to talk to one oracle and you do not want to specify it on every request. Once an oracle is set with setChainlinkOracle
that is the address used with sendChainlinkRequest.
Retrieve the oracle address using chainlinkOracleAddress. These getters and setters are provided to enforce that changes to the oracle are explicitly made in the code.
constructor(address _oracle)
{
setChainlinkOracle(_oracle);
}
setChainlinkToken(
address _link
)
Sets the stored address for the LINK token which is used to send requests to Oracles. There are different token addresses on different network. See LINK Token Contracts for the address of the LINK token on the network you're deploying to.
constructor(address _link)
public
{
setChainlinkToken(_link);
}
Use buildOperatorRequest
function if the oracle is an Operator
contract.
function buildChainlinkRequest(
bytes32 _jobId,
address _callbackAddress,
bytes4 _callbackFunctionSignature
) returns (Chainlink.Request memory request)
Instantiates a Request from the Chainlink contract. A Request is a struct which contains the necessary parameters to be sent to the oracle contract. The buildChainlinkRequest
function takes an ID, which can be a Job ID, a callback address to receive the resulting data, and a callback function signature to call on the callback address.
function requestPrice()
public
{
bytes32 jobId = "493610cff14346f786f88ed791ab7704";
bytes4 selector = this.myCallback.selector;
// build a request that calls the myCallback function defined
// below by specifying the address of this contract and the function
// selector of the myCallback
Chainlink.Request memory request = buildChainlinkRequest(
jobId,
address(this),
selector);
}
This function is similar to buildChainlinkRequest
function. One major difference is that buildOperatorRequest
does not allow setting up the address of the callback. The callback address is set to the address of the calling contract.
It is recommended to use buildOperatorRequest
but make sure the oracle you are contacting is an Operator
contract.
function buildOperatorRequest(
bytes32 _jobId,
bytes4 _callbackFunctionSignature
) returns (Chainlink.Request memory request)
Instantiates a Request from the Chainlink contract. A Request is a struct that contains the necessary parameters to be sent to the oracle contract. The buildOperatorRequest
function takes an ID, which can be a Job ID, and a callback function signature to call on the calling contract address.
function requestPrice()
public
{
bytes32 jobId = "493610cff14346f786f88ed791ab7704";
bytes4 selector = this.myCallback.selector;
// build a request that calls the myCallback function defined
// below by specifying the function selector of myCallback
Chainlink.Request memory request = buildOperatorRequest(
jobId,
selector);
}
Use sendOperatorRequest
function if the oracle is an Operator
contract.
function sendChainlinkRequest(
Chainlink.Request memory _req,
uint256 _payment
) returns (bytes32 requestId)
Sends the request payload to the stored oracle address. It takes a Chainlink.Request and the amount of LINK to send amount as parameters. The request is serialized and calls oracleRequest
on the address stored in chainlinkOracleAddress
via the LINK token's transferAndCall
method.
sendChainlinkRequest
returns the ID of the request. If your application needs to, your contract can store that ID, but you don't need to. The ChainlinkClient helpers will store the ID under the hood, along with the oracle address, and use them when you call recordChainlinkFulfillment
in your callback function to make sure only that the address you want can call your Chainlink callback function.
sendChainlinkRequest
emits a ChainlinkRequested event containing the request ID, if you would like to use it in your Web3 application.
function requestPrice()
public
{
Chainlink.Request memory request = buildChainlinkRequest(jobId, address(this), this.callback.selector);
uint256 paymentAmount = 1 * LINK_DIVISIBILITY / 10; // Equivalent to 0.1 LINK
// send the request that you just built
sendChainlinkRequest(request, paymentAmount);
}
Use sendOperatorRequestTo
function if the oracle is an Operator
contract.
function sendChainlinkRequestTo(
address _oracle,
Chainlink.Request memory _req,
uint256 _payment
) returns (bytes32 requestId)
Similar to sendChainlinkRequest, sendChainlinkRequestTo
sends a Request but allows the target oracle to be specified. It requires an address, a Request, and an amount, and returns the requestId
. This allows a requesting contract to create and track requests sent to multiple oracle contract addresses.
sendChainlinkRequestTo
emits a ChainlinkRequested event containing the request ID, if you would like to use it in your Web3 application.
function requestPriceFrom(address _oracle)
public
{
Chainlink.Request memory request = buildChainlinkRequest(jobId, address(this), this.callback.callbackSelector);
uint256 paymentAmount = 1 * LINK_DIVISIBILITY; // = 1 LINK
// send the request that you just built to a specified oracle
sendChainlinkRequestTo(_oracle, request, paymentAmount);
}
This function is similar to sendChainlinkRequest
function.
It is recommended to use sendOperatorRequest
but make sure the oracle you are contacting is an Operator
contract.
function sendOperatorRequest(
Chainlink.Request memory _req,
uint256 _payment
) returns (bytes32 requestId)
The sendOperatorRequest
function sends the request payload to the stored oracle address. It takes a Chainlink.Request and the amount of LINK to send amount as parameters. The request is serialized and calls operatorRequest
on the address stored in chainlinkOracleAddress
using the LINK token's transferAndCall
method.
sendOperatorRequest
returns the ID of the request. Optionally, your contract can store the ID if your application needs it. The ChainlinkClient
helpers store the ID and the oracle address and use them when you call recordChainlinkFulfillment
in your callback function. This ensures that only the specified address can call your Chainlink callback function.
sendOperatorRequest
emits a ChainlinkRequested event containing the request ID that you can use in your Web3 application.
function requestPrice()
public
{
Chainlink.Request memory request = buildOperatorRequest(jobId, this.callback.selector);
uint256 paymentAmount = 1 * LINK_DIVISIBILITY / 10; // Equivalent to 0.1 LINK
// send the request that you just built
sendOperatorRequest(request, paymentAmount);
}
This function is similar to sendChainlinkRequestTo
function.
It is recommended to use sendOperatorRequestTo
, but make sure the oracle you are contacting is an Operator
contract.
function sendChainlinkRequestTo(
address _oracle,
Chainlink.Request memory _req,
uint256 _payment
) returns (bytes32 requestId)
Similar to sendOperatorRequest, sendOperatorRequestTo
sends a Request but allows the target oracle to be specified. It requires an address, a Request, and an amount, and returns the requestId
. This allows a requesting contract to create and track requests sent to multiple oracle contract addresses.
sendOperatorRequestTo
emits a ChainlinkRequested event containing a request ID that you can use in your Web3 application.
function requestPriceFrom(address _oracle)
public
{
Chainlink.Request memory request = buildOperatorRequest(jobId, this.callback.callbackSelector);
uint256 paymentAmount = 1 * LINK_DIVISIBILITY; // = 1 LINK
// send the request that you just built to a specified oracle
sendOperatorRequestTo(_oracle, request, paymentAmount);
}
function validateChainlinkCallback(
bytes32 _requestId
)
Used on fulfillment callbacks to ensure that the caller and requestId
are valid. They protect ChainlinkClient callbacks from being called by malicious callers. validateChainlinkCallback
allows for a request to be called
This is the method equivalent of the modifier recordChainlinkFulfillment
. Either validateChainlinkCallback
or recordChainlinkFulfillment
should be used on all fulfillment functions to ensure that the caller and requestId
are valid. Use the modifier or the method, not both.
validateChainlinkCallback
emits a ChainlinkFulfilled event.
function myCallback(bytes32 _requestId, uint256 _price)
public
{
validateChainlinkCallback(_requestId);
currentPrice = _price;
}
Do not call multiple times
Do not call validateChainlinkCallback
multiple times. The nature of validating the callback is to ensure the response is only received once and not replayed. Calling a second time with the same method ID will trigger a revert. Similarly, your callback should validate using either validateChainlinkCallback
or recordChainlinkFulfillment
, not both.
function addChainlinkExternalRequest(
address _oracle,
bytes32 _requestId
)
addChainlinkExternalRequest
allows a Chainlink contract to track unfulfilled requests that it hasn't created itself. For example, contract A creates a request and sets the callback for contract B. Contract B needs to know about the request created by contract A so that it can validate the callback when it is executed.
function expectResponseFor(bytes32 _requestId)
public
{
addChainlinkExternalRequest(chainlinkOracleAddress(), _requestId);
}
Be careful adding external requests
Being able to change a request means that you can change the data fed into a contract. Permissioning someone to make external requests can allow them to change the outcome of your contract. You should be sure to make sure that they are a trusted to do so. If they are not trusted to do so, you should put the request making logic on-chain where it is auditable and tamperproof.
function cancelChainlinkRequest(bytes32 _requestId,
uint256 _payment,
bytes4 _callbackFunc,
uint256 _expiration
)
In case an oracle node does not respond, it may be necessary to retrieve the LINK used to pay for the unfulfilled request. The cancelChainlinkRequest
will send the cancel request to the address used for the request, which transfers the amount of LINK back to the requesting contract, and delete it from the tracked requests.
The default expiration for a request is five minutes, after which it can be cancelled. The cancellation must be sent by the address which was specified as the callback location of the contract.
For the sake of efficient gas usage, only a hash of the request's parameters are stored on-chain. In order to validate the terms of the request and that it can be calculated, the request parameters must be provided. Additionally, cancellation must be called by the address which the callback would otherwise have been called on.
cancelChainlinkRequest
emits a ChainlinkCancelled event.
function cancelRequest(
bytes32 _requestId,
uint256 _payment,
bytes4 _callbackFunc,
uint256 _expiration
) public {
cancelChainlinkRequest(_requestId, _payment, _callbackFunc, _expiration);
}
function useChainlinkWithENS(
address _ens,
bytes32 _node
)
Allows a Chainlink contract to store the addresses of the LINK token and oracle contract addresses without supplying the addresses themselves. We use ENS where available to resolve these addresses. It requires the address of the ENS contract and the node (which is a hash) for the domain.
If your Oracle provider supports using ENS for rolling upgrades to their oracle contract, once you've pointed your Chainlinked contract to the ENS records then you can update the records using updateChainlinkOracleWithENS.
address constant ROPSTEN_ENS = 0x112234455C3a32FD11230C42E7Bccd4A84e02010;
bytes32 constant ROPSTEN_CHAINLINK_ENS = 0xead9c0180f6d685e43522fcfe277c2f0465fe930fb32b5b415826eacf9803727;
constructor(){
useChainlinkWithENS(ROPSTEN_ENS, ROPSTEN_CHAINLINK_ENS);
}
Updating oracle addresses
If an oracle provider supports listing their oracle on ENS, that provides the added security of being able to update any issues that may arise. The tradeoff here is that by using their ENS record, you are allowing whoever controls that record and the corresponding code it points to. If your contract does this, you must either audit the updated code and make sure it matches Oracle.sol or trust whoever can update the records.
function updateChainlinkOracleWithENS()
Updates the stored oracle contract address with the latest address resolved through the ENS contract. This requires the oracle provider to support listing their address on ENS.
This method only works after useChainlinkWithENS has been called on the contract.
function updateOracleAddressToLatest() public {
updateChainlinkOracleWithENS();
}
Updating oracle addresses
If an oracle provider supports listing their oracle on ENS, that provides the added security of being able to update any issues that may arise. The tradeoff here is that by using their ENS record, you are allowing whoever controls that record and the corresponding code it points to. If your contract does this, you must either audit the updated code and make sure it matches Oracle.sol or trust whoever can update the records.
function chainlinkTokenAddress() returns (address)
The chainlinkTokenAddress
function is a helper used to return the stored address of the Chainlink token. This variable is protected and so only made available through getters and setters.
function withdrawLink() public {
LinkTokenInterface link = LinkTokenInterface(chainlinkTokenAddress());
require(link.transfer(msg.sender, link.balanceOf(address(this))), "Unable to transfer");
}
The chainlinkOracleAddress
function is a helper used to return the stored address of the oracle contract.
function getOracle() public view returns (address) {
return chainlinkOracleAddress();
}
event ChainlinkRequested(
bytes32 indexed id
)
Emitted when sendChainlinkRequest and sendChainlinkRequestTo are called. Includes the request ID as an event topic.
event ChainlinkFulfilled(
bytes32 indexed id
)
Emitted when validateChainlinkCallback or recordChainlinkFulfillment are called. Includes the request ID as an event topic.
event ChainlinkCancelled(
bytes32 indexed id
)
Emitted when cancelChainlinkRequest is called. Includes the request ID as an event topic.
LINK_DIVISIBILITY
is a uint256 constant to represent one whole unit of the LINK token (1000000000000000000). It can be used with another value to specify payment in an easy-to-read format, instead of hardcoding magic numbers.
uint256 constant private ORACLE_PAYMENT = 100 * LINK_DIVISIBILITY; // = 100 LINK
recordChainlinkFulfillment
is used on fulfillment callbacks to ensure that the caller and requestId
are valid. This is the method equivalent of the method validateChainlinkCallback
.
Either validateChainlinkCallback
or recordChainlinkFulfillment
should be used on all Chainlink callback functions to ensure that the sender and requestId
are valid. They protect ChainlinkClient callbacks from being called by malicious callers. Do not call both of them, or your callback may revert before you can record the reported response.
function myCallback(bytes32 _requestId, uint256 _price)
public
recordChainlinkFulfillment(_requestId) // always validate callbacks
{
currentPrice = _price;
}
library Chainlink {
struct Request {
bytes32 id;
address callbackAddress;
bytes4 callbackFunctionId;
uint256 nonce;
Buffer.buffer buf;
}
}
The Chainlink Request struct encapsulates all of the fields needed for a Chainlink request and its corresponding response callback.
The Chainlink protocol aims to be flexible and not restrict application developers. The Solidity Chainlink Request model is a great example of that. It is exceptionally flexible, given the limitations of Solidity. The request can contain an arbitrary amount of keys and values to be passed off-chain to the oracles for each request. It does so by converting the parameters into CBOR, and then storing them in a buffer. This allows for any number of parameters all of different types to be encoded on-chain.
The request's ID is generated by hashing the sender's address and the request's nonce. This scheme ensures that only the requester can generate their request ID, and no other contract can trigger a response from an oracle with that ID. New requests whose IDs match an unfulfilled request ID will not be accepted by the oracle.
Intended for memory
The Request object was intended to be stored in memory. If you have a reason to persist the struct in storage, it is recommended that you do so by copying each attribute over and explicitly copying the bytes in the buffer.
Name | Description |
---|---|
id | Identifier for the request |
callbackAddress | Address that the response will be sent to upon fulfillment |
callbackFunctionId | Selector of the function on the callbackAddress that will be invoked with the response upon fulfillment |
nonce | Used to generate the request ID |
buf | Buffer that stores additional user defined parameters as CBOR |
Name | Description |
---|---|
add | Add a string value to the run request parameters |
addBytes | Add a bytes value to the run request parameters |
addInt | Add an integer value to the run request parameters |
addUint | Add an unsigned integer to the run request parameters |
addStringArray | Add an array of strings as a value in the run request parameters |
setBuffer | Directly set the CBOR of the run request parameters |
function add(
Request memory self,
string _key,
string _value
)
Add a string value to the run request parameters. Commonly used for get
with jobs using httpGet
tasks.
function requestEthereumPrice()
public
{
Chainlink.Request memory req = buildChainlinkRequest(jobId, this, this.fulfill.selector);
req.add("get", "https://min-api.cryptocompare.com/data/price?fsym=ETH&tsyms=USD,EUR,JPY");
sendChainlinkRequest(req, 1 * LINK_DIVISIBILITY); // =1 LINK
}
function addBytes(
Request memory self,
string _key,
bytes _value
)
Add a CBOR bytes type value to the run request parameters.
function requestEmojiPopularity(bytes _unicode)
public
{
Chainlink.Request memory req = buildChainlinkRequest(jobId, this, this.fulfill.selector);
req.addBytes("emojiUnicode", _unicode);
sendChainlinkRequest(req, LINK_DIVISIBILITY * 1);
}
Note that this can also be used as a workaround to pass other data types like arrays or addresses. For instance, to add an address, one would first encode it using abi.encode
then pass the result to addBytes
:
Chainlink.Request memory req = buildChainlinkRequest(jobId, this, this.fulfill.selector);
req.addBytes("address", abi.encode(msg.sender)); // msg.sender used in this example. Replace it with your address
function addInt(
Request memory self,
string _key,
int256 _value
)
Add a CBOR signed integer type value to the run request parameters. Commonly used with the times
parameter of any job using a multiply
task.
function requestPrice()
public
{
Chainlink.Request memory req = buildChainlinkRequest(jobId, this, this.fulfill.selector);
req.addInt("times", 100);
sendChainlinkRequest(req, LINK_DIVISIBILITY * 1);
}
function addUint(
Request memory self,
string _key,
uint256 _value
)
Add a CBOR unsigned integer type value to the run request parameters. Commonly used with the times
parameter of any job using a multiply
task.
function requestPrice()
public
{
Chainlink.Request memory req = buildChainlinkRequest(jobId, this, this.fulfill.selector);
req.addUint("times", 100);
sendChainlinkRequest(req, LINK_DIVISIBILITY * 1);
}
function addStringArray(
Request memory self,
string _key,
string[] memory _values
)
Add a CBOR array of strings to the run request parameters. Commonly used with the path
parameter for any job including a jsonParse
task.
function requestPrice(string _currency)
public
{
Chainlink.Request memory req = buildChainlinkRequest(JOB_ID, this, this.myCallback.selector);
string[] memory path = new string[](2);
path[0] = _currency;
path[1] = "recent";
// specify templated fields in a job specification
req.addStringArray("path", path);
sendChainlinkRequest(req, PAYMENT);
}
function setBuffer(
Request memory self,
bytes _data
)
Set the CBOR payload directly on the request object, avoiding the cost of encoding the parameters in CBOR. This can be helpful when reading the bytes from storage or having them passed in from off-chain where they were pre-encoded.
function requestPrice(bytes _cbor)
public
{
Chainlink.Request memory req = buildChainlinkRequest(JOB_ID, this, this.myCallback.selector);
req.setBuffer(_cbor);
sendChainlinkRequest(req, PAYMENT);
}
Be careful setting the request buffer directly
Moving the CBOR encoding logic off-chain can save some gas, but it also opens up the opportunity for people to encode parameters that not all parties agreed to. Be sure that whoever is permissioned to call setBuffer
is trusted or auditable.