Today Chainlink price feeds are used by many DeFi protocols to secure billions of dollars. Whilst feeds report fresh prices the majority of the time, L2 feeds will report stale price data whenever the L2 chain stops producing new blocks. This can happen whenever the L2 Sequencer fails to process any new transactions. Whenever this happens, an arbitrage opportunity is created for malicious actors to take advantage of the price difference between the price inside and outside the L2 chain.
The Starknet Emergency Protocol provides a way for Chainlink price feed consumers to guard against the scenario described above. The protocol tracks the last known health of the Sequencer and reports its health on chain along with the timestamp of when it either comes back online or goes offline. This allows consuming contracts to implement a grace period in their contracts to revert transactions whenever the Sequencer is down.
For more background information check the official docs.
WARNING: The current implementation of the protocol supports health status detection for Starknet centralized Sequencer architecture. As Starknet plans to decentralize the Sequencer in the future, this protocol will either need to be redesigned and reevaluated. The reason for this is that the current protocol relies on polling the pending
block from the Sequencer to determine if new transactions are being added. A decentralized Starknet sequencer will no longer allow this hence breaking the protocol. The documentation states:
Today, Starknet supports querying the new block before its construction is complete. This feature improves the responsiveness of the system prior to the decentralization phase, but will probably become obsolete once the system is decentralized, as full nodes will only propagate finalized blocks through the network.
For more information on how the pending block is used, take a look at the Layer2 Sequencer Health External Adapter section.
The diagram above illustrates the general path of how the Sequencer’s status is relayed from L1 to L2.
- L1 Ethereum (Solditiy):
- L2 Starknet (Cairo):
L1
- The EA is run by a network of Node operators to post the latest sequencer status to the
Aggregator
contract and relayed to theValidatorProxy
contract. TheAggregator
contract then calls thevalidate
function in theValidatorProxy
contract, which proxies the call to theStarknetValidator
contract. - The
StarknetValidator
then calls thesendMessageToL2
function on theStarknet
contract. This message will contain instructions to call theupdateStatus(bool status, uint64 timestamp)
function in theStarknetSequencerUptimeFeed
contract deployed on L2 - The core
Starknet
contract then emits a newLogMessageToL2
event to to signal that a new message needs to be sent from L1 to L2.
event LogMessageToL2(
address indexed fromAddress,
uint256 indexed toAddress,
uint256 indexed selector,
uint256[] payload,
uint256 nonce
);
- The
Sequencer
will then pickup theLogMessageToL2
event emitted above and forward the message to the target contract on L2.
L2
- The Sequencer posts the message to the
starknet_sequencer_uptime_feed
contract and calls theupdate_status
function to update the Sequencer status. - Consumers can then read from the
aggregator_proxy
contract, which fetches the latest round data from thestarknet_sequencer_uptime_feed
contract.
In the event that the Sequencer is down, messages will not be transmitted from L1 to L2 and no L2 transactions are executed. Instead messages will be enqueued in the Sequencer and only processed in the order they arrived later once the Sequencer comes back up. This means that as long as the message from the StarknetValidator
on L1 is already enqueued in the Starknet Sequencer, the flag on the starknet_sequencer_uptime_feed
on L2 will be guaranteed to be flipped prior to any subsequent transactions. This happens as the transaction flipping the flag on the uptime feed will get executed before transactions that were enqueued after it. This is further explained in the diagrams below.
During Sequencer downtime
- New
LogMessageToL2
events emitted are not picked up whilst the Sequencer is down. - When the Sequencer is down, all L2 transactions sent from L1 are stuck in the pending queue, which lives in Starknet’s centralized Sequencer.
- Tx1 contains Chainlink’s transaction to set the status of the Sequencer as being down on L2.
- Tx2 is a transaction made by a consumer that is dependent on
After Sequencer comes back online
LogMessageToL2
events are picked up and added to the pending queue.- Transactions in the pending queue are processed chronologically so Tx1 is processed before Tx2.
- As Tx1 happens before Tx2, Tx2 will read the status of the Sequencer as being down
As of writing, on version v0.11.0, Starknet has begun charging mandatory fees to send messages from L1 to L2. These fees are used to pay for the transaction
on L2. As the Emergency Protocol needs to send messages cross chain,
the protocol needs a way to estimate gas fees. Currently, the StarkwareValidator
contract on L1 does the following to estimate the amount of required
gas.
- Estimate gas fees by running the command below. The command is from Starkware's standard CLI (using version 0.11.0.x)
starknet estimate_message_fee \
--feeder_gateway_url=https://alpha4.starknet.io/feeder_gateway/
--from_address ${L1_SENDER_ADDR} \
--address ${UPTIME_FEED_ADDR} \
--function update_status \
--inputs ${STATUS} ${TIMESTAMP}
Make sure that the L1_SENDER_ADDR
is equal to the l1 sender storage variable on the uptime feed, or else the gateway will respond with a revert instead of the values. If you don't set the l1 sender storage variable, it'll be 0 by default (as in the example below)
Example Query and response:
starknet estimate_message_fee \
--feeder_gateway_url=https://alpha4.starknet.io/feeder_gateway/ \
--from_address 0x0 \
--address=0x06f4279f832de1afd94ab79aa1766628d2c1e70bc7f74bfba3335db8e728a7e6 \
--function update_status \
--inputs 0x1 123123
The estimated fee is: 3739595758116898 WEI (0.003740 ETH).
Gas usage: 17266
Gas price: 216587267353 WEI
In order to reliably ensure that cross chain messages are sent with sufficient gas, the estimate is multiplied by a buffer. At the time of writing (Starknet v.0.11.0), Starkware has told us that L2 gas prices are equal to L1 gas prices and are denominated in Ethereum Wei, so we use L1 gas price feed to get the gas price:
- Read the current L1 gas price from Chainlink's L1 gas price feed
- Multiply gas price by a buffer
- Multiply product of above by the number of gas units
gasFee = buffer * l1GasPrice * numGasUnits
The gas units that it costs is also derived from the starknet estimate_message_fee command (as shown above).
As of the time of writing (Starknet v. 0.11.0), we recommend a gasAdjustment of 130 (or 1.3x buffer) and a gas units to be 17300.
The emergency protocol requires an off chain component to tracks the health of the centralized Starkware sequencer. Today, this is made up by a DON (Decentralized Oracle Network) that triggers using OCR (Offchain Reporting). A new OCR round is initiated every 30s whereby each node in the DON checks the health of the Sequencer using the Layer2 Sequencer Health External Adapter. If the nodes in the DON determine that the Sequencer’s health has changed, they elect a new leader to write the updated result onto chain as shown in the diagram above.
How the External Adapter Works
Checking the Starkware Sequencer’s health is currently a two step process
-
Call the Sequencer directly to fetch the pending block’s details.
- Verify that a new block has been produced within 2 minutes by checking the pending block’s
parentHash
- If the pending block’s
parentHash
has not changed, then check the length of thetransactions
field to see if it has increased since the last round
- Verify that a new block has been produced within 2 minutes by checking the pending block’s
-
Send an empty transaction to a dummy contract at address
0x00000000000000000000000000000000000000000000000000000000000001
The EA sends the empty transaction using the StarknetJS library. This transaction tries to call the dummy contract’s
initialize
function with amaxFee
of 0const DUMMY_ADDRESS = '0x00000000000000000000000000000000000000000000000000000000000001' const DEFAULT_PRIVATE_KEY = '0x0000000000000000000000000000000000000000000000000000000000000001' const starkKeyPair = ec.genKeyPair(DEFAULT_PRIVATE_KEY) const starkKeyPub = ec.getStarkKey(starkKeyPair) const provider = config.starkwareConfig.provider const account = new Account(provider, DUMMY_ADDRESS, starkKeyPair) account.execute( { contractAddress: DUMMY_ADDRESS, entrypoint: 'initialize', calldata: [starkKeyPub, '0'], }, undefined, { maxFee: '0' }, )
-
As the above transaction is expected to fail, the EA will consider the Sequencer as healthy if it receives any of the expected error statuses
StarknetErrorCode.UNINITIALIZED_CONTRACT
if the dummy contract has not been initializedStarknetErrorCode.OUT_OF_RANGE_FEE
if the dummy contract has been initialized by accident. As Starknet is a permissionless network, we cannot guarantee that a user deploys and initializes a contract at the dummy address. As a result, the EA will set themaxFee
to 0 so that the transaction will fail with theStarknetErrorCode.OUT_OF_RANGE_FEE
status code.