How to customize your Orbit chain's precompiles
Orbit chains are currently a public preview capability. This offering and its supporting documentation may change significantly as we capture feedback from readers like you.
To provide feedback, click the Request an update button at the top of this document, join the Arbitrum Discord, or reach out to our team directly by completing this form.
The guidance in this document will work only if you use eth_call
to call the new precompiles. If you're calling from other contracts or adding non-view/pure methods, this approach will break block validation.
To support these additional use-cases, follow the instructions described in How to customize your Orbit chain's behavior.
This tutorial provides three ways to customize your chain's precompiles:
- Add new methods to an existing precompile.
- Create a new precompile.
- Define a new event.
Prerequisites
Clone the Nitro repository before you begin:
git clone --branch v2.1.1 <https://github.com/OffchainLabs/nitro.git>
cd nitro
git submodule update --init --recursive --force
Option 1: Add new methods to an existing precompile
Using your favorite code editor, open an existing precompile from the precompiles implementation directory, /precompiles
. We'll use ArbSys.go
as an example. Open the corresponding Go implementation file (ArbSys.go
) and add a simple SayHi
method:
func (con *ArbSys) SayHi(c ctx, evm mech) (string, error) {
return "hi", nil
}
Then, open the corresponding Solidity interface file (ArbSys.sol
) from the precompiles interface directory, /src/precompiles
, and add the required interface. Ensure that the method name on the interface matches the name of the function you introduced in the previous step, camelCased
:
function sayHi() external view returns(string memory);
Next, follow the steps in How to customize your Orbit chain's behavior to build a modified Arbitrum Nitro node docker image and run it.
Note that the instructions provided in How to run a full node will not work with your Orbit node. See Command-line options (Orbit) for Orbit-specific CLI flags.
Once your node is running, you can call ArbSys.sol
either directly using curl
, or through Foundry's cast call
.
Call your function directly using curl
curl Your_IP_Address:8547\
-X POST \
-H "Content-Type: application/json" \
--data '{"method":"eth_call","params":[{"from":null,"to":"0x0000000000000000000000000000000000000064","data":"0x0c49c36c"}, "latest"],"id":1,"jsonrpc":"2.0"}'
You should see something like this:
{"jsonrpc":"2.0","id":1,"result":"0x000000000000000000000000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000000026869000000000000000000000000000000000000000000000000000000000000"}
0x6869
is the hex-encoded utf8 representation of hi
, which you'll see embedded in the result
hex string.
Call your function using Foundry's cast call
cast call 0x0000000000000000000000000000000000000064 "sayHi()(string)”
You should see something like this:
hi
Option 2: Create a new precompile
First, navigate to the precompiles implementation directory, /precompiles
, and create a new precompile implementation file called ArbHi.go
. We'll define a new method, and we'll give it an address:
package precompiles
// ArbGasInfo provides insight into the cost of using the rollup.
type ArbHi struct {
Address addr // 0x11a, for example
}
func (con *ArbHi) SayHi(c ctx, evm mech) (string, error) {
return "hi", nil
}
Then, update precompile.go to register the new precompile under the Precompiles()
method:
insert(MakePrecompile(templates.ArbHiMetaData, &ArbHi{Address: hex("11a")})) // 0x011a here is an example address
Navigate to the precompiles interface directory, /src/precompiles
, create ArbHi.sol
, and add the required interface. Ensure that the method name on the interface matches the name of the function you introduced in the previous step, camelCased
:
pragma solidity >=0.4.21 <0.9.0;
/// @title Say hi.
/// @notice just for test
/// This custom contract will set on 0x000000000000000000000000000000000000011a since we set it in precompile.go.
interface ArbHi {
function sayHi() external view returns(string memory);
}
Next, follow the steps in How to customize your Orbit chain's behavior to build a modified Arbitrum Nitro node docker image and run it.
Note that the instructions provided in How to run a full node will not work with your Orbit node. See Command-line options (Orbit) for Orbit-specific CLI flags.
Once your node is running, you can call ArbHi.sol
either directly using curl
, or through Foundry's cast call
.
Call your function directly using curl
curl Your_IP_Address:8547 \
-X POST \
-H "Content-Type: application/json" \
--data '{"method":"eth_call","params":[{"from":null,"to":"0x000000000000000000000000000000000000011a","data":"0x0c49c36c"}, "latest"],"id":1,"jsonrpc":"2.0"}'
You should see something like this:
{"jsonrpc":"2.0","id":1,"result":"0x000000000000000000000000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000000026869000000000000000000000000000000000000000000000000000000000000"}
Call your function using Foundry's cast call
cast call 0x000000000000000000000000000000000000011a "sayHi()(string)”
You should see something like this:
hi
Option 3: Define a new event
Here we will still use the precompile Arbsys
as an example and add a simple Hi
event to be emitted in SayHi
method which we added in ArbSys.sol
contract at Option 1, first go to the precompiles implementation directory, and find ArbSys.go
, edit the ArbSys
struct:
// ArbSys provides system-level functionality for interacting with L1 and understanding the call stack.
type ArbSys struct {
Address addr // 0x64
L2ToL1Tx func(ctx, mech, addr, addr, huge, huge, huge, huge, huge, huge, []byte) error
L2ToL1TxGasCost func(addr, addr, huge, huge, huge, huge, huge, huge, []byte) (uint64, error)
SendMerkleUpdate func(ctx, mech, huge, bytes32, huge) error
SendMerkleUpdateGasCost func(huge, bytes32, huge) (uint64, error)
InvalidBlockNumberError func(huge, huge) error
// deprecated event
L2ToL1Transaction func(ctx, mech, addr, addr, huge, huge, huge, huge, huge, huge, huge, []byte) error
L2ToL1TransactionGasCost func(addr, addr, huge, huge, huge, huge, huge, huge, huge, []byte) (uint64, error)
// Add your customize event here:
Hi func(ctx, mech, addr) error
// This is needed which will tell you how much gas it will cost, the param is the same as your event but without the first two (ctx, mech), the return param is always (uint64, error)
HiGasCost func(addr) (uint64, error)
}
Then add the event to SayHi
method:
func (con *ArbSys) SayHi(c ctx, evm mech) (string, error) {
err := con.Hi(c, evm, c.caller)
return "hi", err
}
Now navigate to the precompiles interface directory, open Arbsys.sol
, and add the required interface. Ensure that the Event name on the interface matches the name of the function you introduced in ArbSys
struct in the previous step:
event Hi(address caller);
If you want to make the param in event as index
, just add index to solidity interface as this:
event Hi(address indexed caller);
Now as emiting events need to cost gas, so we should remove view
function behavior from the method:
function sayHi() external returns(string memory);
Next, build Nitro by following the instructions in How to build Nitro locally. Note that if you've already built the Docker image, you still need run the last step to rebuild.
Run Nitro with the following command:
docker run --rm -it -v /some/local/dir/arbitrum:/home/user/.arbitrum -p 0.0.0.0:8547:8547 -p 0.0.0.0:8548:8548 offchainlabs/nitro-node:v2.1.1-e9d8842 --parent-chain.connection.url=<YourParentChainUrl> --chain.id=<YourOrbitChainId> --http.api=net,web3,eth,debug --http.corsdomain=* --http.addr=0.0.0.0 --http.vhosts=*
Note that the instructions provided in How to run a full node will not work with your Orbit node. See Command-line options (Orbit) for Orbit-specific CLI flags.
Send the transaction and get the transaction receipt
Send transaction to ArbSys
, note as the function is not view/pure function, we need to send transaction with gas cost:
cast send 0x0000000000000000000000000000000000000064 "sayHi()(string)"
Then we will have a transaction hash result, call eth_getTransactionReceipt
with that transaction hash, you may get:
{"jsonrpc":"2.0","id":1,"result":{"blockHash":"Your_blockHash","blockNumber":"Your_blockNumber","contractAddress":null,"cumulativeGasUsed":"0x680b","effectiveGasPrice":"0x5f5e100","from":"Your_address","gasUsed":"0x680b","gasUsedForL1":"0xe35","l1BlockNumber":"l1_blockNumber","logs":[{"address":"0x0000000000000000000000000000000000000064","topics":["0xa9378d5bd800fae4d5b8d4c6712b2b64e8ecc86fdc831cb51944000fc7c8ecfa","0x000000000000000000000000{Your_address}"],"data":"0x","blockNumber":"Your_blockNumber","transactionHash":"Your_txHash","transactionIndex":"0x1","blockHash":"Your_blockHash","logIndex":"0x0","removed":false}],"logsBloom":"0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000100000000000000040000000000000080004000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000040000000000000000000000000000000004000000000000200000000000000000000000000000000000000000000000000000000000000000000000000000000000","status":"0x1","to":"0x0000000000000000000000000000000000000064","transactionHash":"Your_txHash","transactionIndex":"0x1","type":"0x2"}}
You should see logs
field within the transaction receipt like this:
"logs":[
{
"address":"0x0000000000000000000000000000000000000064",
"topics":[
"0xa9378d5bd800fae4d5b8d4c6712b2b64e8ecc86fdc831cb51944000fc7c8ecfa",
"0x000000000000000000000000{Your_address}"
],
"data":"0x",
"blockNumber":"0x40",
"transactionHash":"{Your_txHash}",
"transactionIndex":"0x1",
"blockHash":"0x0b367d705002b3575db99354a0964c033f929f26f4442ed347e47ae43a8f28e4",
"logIndex":"0x0",
"removed":false
}
]
From the logs
field, we can see the topics[0] is 0xa937..cfa
, which is the event signature of Hi(address)
, you can check it using 4 byte directory, and topics[1] is Your_address
which is exactly what we defined above!