From ab526193ddb57c5a94e649076bc9621e72cc1f7f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bla=C5=BE=20Hrastnik?= Date: Wed, 3 Apr 2024 14:30:26 +0900 Subject: [PATCH 1/5] fix: off-by-one in SequencerUptimeFeed::round_data It didn't allow querying for the latest round id --- contracts/src/emergency/sequencer_uptime_feed.cairo | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/contracts/src/emergency/sequencer_uptime_feed.cairo b/contracts/src/emergency/sequencer_uptime_feed.cairo index d93cd5aa1..f67c96d35 100644 --- a/contracts/src/emergency/sequencer_uptime_feed.cairo +++ b/contracts/src/emergency/sequencer_uptime_feed.cairo @@ -134,7 +134,7 @@ mod SequencerUptimeFeed { fn round_data(self: @ContractState, round_id: u128) -> Round { self._require_read_access(); - assert(round_id < self._latest_round_id.read(), 'invalid round id'); + assert(round_id <= self._latest_round_id.read(), 'invalid round id'); let round_transmission = self._round_transmissions.read(round_id); Round { round_id: round_id.into(), From 8f0d923a5ac435233c1635a988ac4da0c35b843f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bla=C5=BE=20Hrastnik?= Date: Wed, 3 Apr 2024 15:48:36 +0900 Subject: [PATCH 2/5] fix: Correctly record sender on L1 calls --- .../src/emergency/sequencer_uptime_feed.cairo | 34 ++++++++++++------- 1 file changed, 21 insertions(+), 13 deletions(-) diff --git a/contracts/src/emergency/sequencer_uptime_feed.cairo b/contracts/src/emergency/sequencer_uptime_feed.cairo index f67c96d35..c1784298c 100644 --- a/contracts/src/emergency/sequencer_uptime_feed.cairo +++ b/contracts/src/emergency/sequencer_uptime_feed.cairo @@ -56,7 +56,7 @@ mod SequencerUptimeFeed { #[substorage(v0)] access_control: AccessControlComponent::Storage, // l1 sender is an starknet validator ethereum address - _l1_sender: felt252, + _l1_sender: EthAddress, // maps round id to round transmission _round_transmissions: LegacyMap, _latest_round_id: u128, @@ -85,7 +85,8 @@ mod SequencerUptimeFeed { #[derive(Drop, starknet::Event)] struct NewRound { round_id: u128, - started_by: ContractAddress, + #[key] + started_by: EthAddress, started_at: u64 } @@ -161,6 +162,8 @@ mod SequencerUptimeFeed { #[l1_handler] fn update_status(ref self: ContractState, from_address: felt252, status: u128, timestamp: u64) { + // Cairo enforces from_address to be a felt252 on the method signature, but we can cast it right after + let from_address: EthAddress = from_address.try_into().unwrap(); assert(self._l1_sender.read() == from_address, 'EXPECTED_FROM_BRIDGE_ONLY'); let latest_round_id = self._latest_round_id.read(); @@ -186,7 +189,7 @@ mod SequencerUptimeFeed { } else { // only increment round when status flips let round_id = latest_round_id + 1_u128; - self._record_round(round_id, status, timestamp); + self._record_round(from_address, round_id, status, timestamp); } } @@ -199,21 +202,19 @@ mod SequencerUptimeFeed { let old_address = self._l1_sender.read(); - if old_address != address.into() { - self._l1_sender.write(address.into()); + if old_address != address { + self._l1_sender.write(address); self .emit( Event::L1SenderTransferred( - L1SenderTransferred { - from_address: old_address.try_into().unwrap(), to_address: address - } + L1SenderTransferred { from_address: old_address, to_address: address } ) ); } } fn l1_sender(self: @ContractState) -> EthAddress { - self._l1_sender.read().try_into().unwrap() + self._l1_sender.read() } } @@ -245,10 +246,19 @@ mod SequencerUptimeFeed { self.access_control.initializer(); let round_id = 1_u128; let timestamp = starknet::info::get_block_timestamp(); - self._record_round(round_id, initial_status, timestamp); + let from_address = EthAddress { + address: 0 + }; // initial round is set by the constructor, not by an L1 sender + self._record_round(from_address, round_id, initial_status, timestamp); } - fn _record_round(ref self: ContractState, round_id: u128, status: u128, timestamp: u64) { + fn _record_round( + ref self: ContractState, + sender: EthAddress, + round_id: u128, + status: u128, + timestamp: u64 + ) { self._latest_round_id.write(round_id); let block_info = starknet::info::get_block_info().unbox(); let block_number = block_info.block_number; @@ -262,8 +272,6 @@ mod SequencerUptimeFeed { }; self._round_transmissions.write(round_id, round); - let sender = starknet::info::get_caller_address(); - self .emit( Event::NewRound( From 20b41fba44131a3e06fa3fc4f6e679937c16c0dd Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bla=C5=BE=20Hrastnik?= Date: Wed, 3 Apr 2024 14:44:12 +0900 Subject: [PATCH 3/5] Annotate events with index keys --- .../src/emergency/sequencer_uptime_feed.cairo | 8 ++++++++ contracts/src/libraries/access_control.cairo | 2 ++ contracts/src/libraries/token/erc677.cairo | 2 ++ contracts/src/libraries/upgradeable.cairo | 1 + contracts/src/multisig.cairo | 11 +++++++++++ contracts/src/ocr2/aggregator.cairo | 15 +++++++++++++++ 6 files changed, 39 insertions(+) diff --git a/contracts/src/emergency/sequencer_uptime_feed.cairo b/contracts/src/emergency/sequencer_uptime_feed.cairo index c1784298c..1a2c75c25 100644 --- a/contracts/src/emergency/sequencer_uptime_feed.cairo +++ b/contracts/src/emergency/sequencer_uptime_feed.cairo @@ -79,11 +79,13 @@ mod SequencerUptimeFeed { #[derive(Drop, starknet::Event)] struct RoundUpdated { status: u128, + #[key] updated_at: u64 } #[derive(Drop, starknet::Event)] struct NewRound { + #[key] round_id: u128, #[key] started_by: EthAddress, @@ -93,21 +95,27 @@ mod SequencerUptimeFeed { #[derive(Drop, starknet::Event)] struct AnswerUpdated { current: u128, + #[key] round_id: u128, + #[key] timestamp: u64 } #[derive(Drop, starknet::Event)] struct UpdateIgnored { latest_status: u128, + #[key] latest_timestamp: u64, incoming_status: u128, + #[key] incoming_timestamp: u64 } #[derive(Drop, starknet::Event)] struct L1SenderTransferred { + #[key] from_address: EthAddress, + #[key] to_address: EthAddress } diff --git a/contracts/src/libraries/access_control.cairo b/contracts/src/libraries/access_control.cairo index d9d872bb9..b0a910d55 100644 --- a/contracts/src/libraries/access_control.cairo +++ b/contracts/src/libraries/access_control.cairo @@ -37,11 +37,13 @@ mod AccessControlComponent { #[derive(Drop, starknet::Event)] struct AddedAccess { + #[key] user: ContractAddress } #[derive(Drop, starknet::Event)] struct RemovedAccess { + #[key] user: ContractAddress } diff --git a/contracts/src/libraries/token/erc677.cairo b/contracts/src/libraries/token/erc677.cairo index 6b23abc77..20fd6c54a 100644 --- a/contracts/src/libraries/token/erc677.cairo +++ b/contracts/src/libraries/token/erc677.cairo @@ -42,7 +42,9 @@ mod ERC677Component { #[derive(Drop, starknet::Event)] struct TransferAndCall { + #[key] from: ContractAddress, + #[key] to: ContractAddress, value: u256, data: Array diff --git a/contracts/src/libraries/upgradeable.cairo b/contracts/src/libraries/upgradeable.cairo index 388581be6..6b9e6af73 100644 --- a/contracts/src/libraries/upgradeable.cairo +++ b/contracts/src/libraries/upgradeable.cairo @@ -10,6 +10,7 @@ trait IUpgradeable { #[derive(Drop, starknet::Event)] struct Upgraded { + #[key] new_impl: ClassHash } diff --git a/contracts/src/multisig.cairo b/contracts/src/multisig.cairo index 5f78e4796..b1422df45 100644 --- a/contracts/src/multisig.cairo +++ b/contracts/src/multisig.cairo @@ -107,36 +107,47 @@ mod Multisig { #[derive(Drop, starknet::Event)] struct TransactionSubmitted { + #[key] signer: ContractAddress, + #[key] nonce: u128, + #[key] to: ContractAddress } #[derive(Drop, starknet::Event)] struct TransactionConfirmed { + #[key] signer: ContractAddress, + #[key] nonce: u128 } #[derive(Drop, starknet::Event)] struct ConfirmationRevoked { + #[key] signer: ContractAddress, + #[key] nonce: u128 } #[derive(Drop, starknet::Event)] struct TransactionExecuted { + #[key] executor: ContractAddress, + #[key] nonce: u128 } #[derive(Drop, starknet::Event)] struct SignersSet { + #[key] signers: Array } #[derive(Drop, starknet::Event)] struct ThresholdSet { + #[key] threshold: usize } diff --git a/contracts/src/ocr2/aggregator.cairo b/contracts/src/ocr2/aggregator.cairo index 00456383f..f5ca342cb 100644 --- a/contracts/src/ocr2/aggregator.cairo +++ b/contracts/src/ocr2/aggregator.cairo @@ -244,8 +244,10 @@ mod Aggregator { #[derive(Drop, starknet::Event)] struct NewTransmission { + #[key] round_id: u128, answer: u128, + #[key] transmitter: ContractAddress, observation_timestamp: u64, observers: felt252, @@ -418,7 +420,9 @@ mod Aggregator { #[derive(Drop, starknet::Event)] struct ConfigSet { + #[key] previous_config_block_number: u64, + #[key] latest_config_digest: felt252, config_count: u64, oracles: Array, @@ -848,7 +852,9 @@ mod Aggregator { #[derive(Drop, starknet::Event)] struct BillingAccessControllerSet { + #[key] old_controller: ContractAddress, + #[key] new_controller: ContractAddress, } @@ -859,6 +865,7 @@ mod Aggregator { #[derive(Drop, starknet::Event)] struct OraclePaid { + #[key] transmitter: ContractAddress, payee: ContractAddress, amount: u256, @@ -867,7 +874,9 @@ mod Aggregator { #[derive(Drop, starknet::Event)] struct LinkTokenSet { + #[key] old_link_token: ContractAddress, + #[key] new_link_token: ContractAddress } @@ -1150,15 +1159,21 @@ mod Aggregator { #[derive(Drop, starknet::Event)] struct PayeeshipTransferRequested { + #[key] transmitter: ContractAddress, + #[key] current: ContractAddress, + #[key] proposed: ContractAddress, } #[derive(Drop, starknet::Event)] struct PayeeshipTransferred { + #[key] transmitter: ContractAddress, + #[key] previous: ContractAddress, + #[key] current: ContractAddress, } From 96ee08dd7a405d04d5c3852e249dc2b37790f7e7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bla=C5=BE=20Hrastnik?= Date: Fri, 5 Apr 2024 15:32:24 +0900 Subject: [PATCH 4/5] Adjust event parsing to account for event keys --- .../src/wrapper/index.ts | 3 +- .../src/lib/encoding.ts | 6 +- .../test/commands/ocr2.test.ts | 2 +- relayer/pkg/chainlink/ocr2/client.go | 32 ++++----- relayer/pkg/chainlink/ocr2/client_test.go | 2 +- relayer/pkg/chainlink/ocr2/events.go | 39 ++++++----- relayer/pkg/chainlink/ocr2/events_test.go | 68 +++++++++++++------ 7 files changed, 92 insertions(+), 60 deletions(-) diff --git a/packages-ts/starknet-gauntlet-multisig/src/wrapper/index.ts b/packages-ts/starknet-gauntlet-multisig/src/wrapper/index.ts index 2720087cb..479da662e 100644 --- a/packages-ts/starknet-gauntlet-multisig/src/wrapper/index.ts +++ b/packages-ts/starknet-gauntlet-multisig/src/wrapper/index.ts @@ -285,7 +285,8 @@ export const wrapCommand = ( const txInfo = (await this.provider.provider.getTransactionReceipt( tx.hash, )) as InvokeTransactionReceiptResponse - proposalId = Number(num.hexToDecimalString((txInfo.events[0] as any).data[1])) + // TODO: use contract.parseEvents? + proposalId = Number(num.hexToDecimalString((txInfo.events[0] as any).keys[2])) // 0 == event_id, 1 == executor, 2 == nonce/proposal_id } const data = await this.afterExecute(result, proposalId) diff --git a/packages-ts/starknet-gauntlet-ocr2/src/lib/encoding.ts b/packages-ts/starknet-gauntlet-ocr2/src/lib/encoding.ts index f1734d4d4..e061cffd3 100644 --- a/packages-ts/starknet-gauntlet-ocr2/src/lib/encoding.ts +++ b/packages-ts/starknet-gauntlet-ocr2/src/lib/encoding.ts @@ -4,8 +4,8 @@ import { feltsToBytes } from '@chainlink/starknet-gauntlet' export const decodeOffchainConfigFromEventData = (data: string[]): encoding.OffchainConfig => { // The ConfigSet event is defined as: // fn ConfigSet( - // previous_config_block_number: u64, - // latest_config_digest: felt252, + // previous_config_block_number: u64, (key) + // latest_config_digest: felt252, (key) // config_count: u64, // oracles: Array, // f: u8, @@ -13,7 +13,7 @@ export const decodeOffchainConfigFromEventData = (data: string[]): encoding.Offc // offchain_config_version: u64, // offchain_config: Array, //) - const oraclesLenIndex = 3 + const oraclesLenIndex = 1 const oraclesLen = Number(BigInt(data[oraclesLenIndex])) const oracleStructSize = 2 const fIndex = oraclesLenIndex + oraclesLen * oracleStructSize + 1 diff --git a/packages-ts/starknet-gauntlet-ocr2/test/commands/ocr2.test.ts b/packages-ts/starknet-gauntlet-ocr2/test/commands/ocr2.test.ts index 1ecc87001..60690a659 100644 --- a/packages-ts/starknet-gauntlet-ocr2/test/commands/ocr2.test.ts +++ b/packages-ts/starknet-gauntlet-ocr2/test/commands/ocr2.test.ts @@ -180,7 +180,7 @@ describe('OCR2 Contract', () => { // reconstruct signers array from event const eventSigners: bigint[] = [] for (let i = 0; i < signers.length; i++) { - const signer = BigInt(eventData[4 + 2 * i]) // split according to event structure + const signer = BigInt(eventData[2 + 2 * i]) // split according to event structure eventSigners.push(signer) } diff --git a/relayer/pkg/chainlink/ocr2/client.go b/relayer/pkg/chainlink/ocr2/client.go index ecfff3459..7764f401e 100644 --- a/relayer/pkg/chainlink/ocr2/client.go +++ b/relayer/pkg/chainlink/ocr2/client.go @@ -186,7 +186,7 @@ func (c *Client) LinkAvailableForPayment(ctx context.Context, address *felt.Felt return ans, nil } -func (c *Client) fetchEventsFromBlock(ctx context.Context, address *felt.Felt, eventType string, blockNum uint64) (eventsAsFeltArrs [][]*felt.Felt, err error) { +func (c *Client) fetchEventsFromBlock(ctx context.Context, address *felt.Felt, eventType string, blockNum uint64) (events []starknetrpc.EmittedEvent, err error) { block := starknetrpc.WithBlockNumber(blockNum) eventKey := starknetutils.GetSelectorFromNameFelt(eventType) @@ -205,32 +205,30 @@ func (c *Client) fetchEventsFromBlock(ctx context.Context, address *felt.Felt, e ChunkSize: 10, }, } - events, err := c.r.Events(ctx, input) + chunk, err := c.r.Events(ctx, input) + events = chunk.Events // TODO: check events.isLastPage, query more if needed if err != nil { - return eventsAsFeltArrs, errors.Wrap(err, "couldn't fetch events for block") + return events, errors.Wrap(err, "couldn't fetch events for block") } - for _, event := range events.Events { - eventsAsFeltArrs = append(eventsAsFeltArrs, event.Data) - } - if len(eventsAsFeltArrs) == 0 { + if len(events) == 0 { return nil, errors.New("events not found in the block") } - return eventsAsFeltArrs, nil + return events, nil } func (c *Client) ConfigFromEventAt(ctx context.Context, address *felt.Felt, blockNum uint64) (cc ContractConfig, err error) { - eventsAsFeltArrs, err := c.fetchEventsFromBlock(ctx, address, "ConfigSet", blockNum) + events, err := c.fetchEventsFromBlock(ctx, address, "ConfigSet", blockNum) if err != nil { return cc, errors.Wrap(err, "failed to fetch config_set events") } - if len(eventsAsFeltArrs) != 1 { - return cc, fmt.Errorf("expected to find one config_set event in block %d for address %s but found %d", blockNum, address, len(eventsAsFeltArrs)) + if len(events) != 1 { + return cc, fmt.Errorf("expected to find one config_set event in block %d for address %s but found %d", blockNum, address, len(events)) } - configAtEvent := eventsAsFeltArrs[0] + configAtEvent := events[0] config, err := ParseConfigSetEvent(configAtEvent) if err != nil { return cc, errors.Wrap(err, "couldn't parse config event") @@ -243,16 +241,16 @@ func (c *Client) ConfigFromEventAt(ctx context.Context, address *felt.Felt, bloc // NewTransmissionsFromEventsAt finds events of type new_transmission emitted by the contract address in a given block number. func (c *Client) NewTransmissionsFromEventsAt(ctx context.Context, address *felt.Felt, blockNum uint64) (events []NewTransmissionEvent, err error) { - eventsAsFeltArrs, err := c.fetchEventsFromBlock(ctx, address, "NewTransmission", blockNum) + rawEvents, err := c.fetchEventsFromBlock(ctx, address, "NewTransmission", blockNum) if err != nil { return nil, errors.Wrap(err, "failed to fetch new_transmission events") } - if len(eventsAsFeltArrs) == 0 { - return nil, fmt.Errorf("expected to find at least one new_transmission event in block %d for address %s but found %d", blockNum, address, len(eventsAsFeltArrs)) + if len(rawEvents) == 0 { + return nil, fmt.Errorf("expected to find at least one new_transmission event in block %d for address %s but found %d", blockNum, address, len(rawEvents)) } events = []NewTransmissionEvent{} - for _, felts := range eventsAsFeltArrs { - event, err := ParseNewTransmissionEvent(felts) + for _, rawEvent := range rawEvents { + event, err := ParseNewTransmissionEvent(rawEvent) if err != nil { return nil, errors.Wrap(err, "couldn't parse new_transmission event") } diff --git a/relayer/pkg/chainlink/ocr2/client_test.go b/relayer/pkg/chainlink/ocr2/client_test.go index 2f0c550e2..734ae603a 100644 --- a/relayer/pkg/chainlink/ocr2/client_test.go +++ b/relayer/pkg/chainlink/ocr2/client_test.go @@ -20,7 +20,7 @@ import ( "github.com/smartcontractkit/chainlink-starknet/relayer/pkg/starknet" ) -const BLOCK_OUTPUT = `{"result": {"events": [ {"from_address": "0xd43963a4e875a361f5d164b2e70953598eb4f45fde86924082d51b4d78e489", "keys": ["0x9a144bf4a6a8fd083c93211e163e59221578efcc86b93f8c97c620e7b9608a"], "data": ["0x0", "0x4b791b801cf0d7b6a2f9e59daf15ec2dd7d9cdc3bc5e037bada9c86e4821c", "0x1", "0x4", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603730", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603734", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603731", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603735", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603732", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603736", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603733", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603737", "0x1", "0x3", "0x1", "0x0", "0xf4240", "0x2", "0x15", "0x263", "0x880a0d9e61d1080d88ee16f1880bcc1960b2080cab5ee01288090dfc04a30", "0x53a0201024220af400004fa5d02cd5170b5261032e71f2847ead36159cf8d", "0xee68affc3c8520904220af400004fa5d02cd5170b5261032e71f2847ead361", "0x59cf8dee68affc3c8520914220af400004fa5d02cd5170b5261032e71f2847", "0xead36159cf8dee68affc3c8520924220af400004fa5d02cd5170b5261032e7", "0x1f2847ead36159cf8dee68affc3c8520934a42307830346363316266613939", "0x65323832653433346165663238313563613137333337613932336364326336", "0x31636630633764653562333236643761383630333733304a42307830346363", "0x31626661393965323832653433346165663238313563613137333337613932", "0x33636432633631636630633764653562333236643761383630333733314a42", "0x30783034636331626661393965323832653433346165663238313563613137", "0x33333761393233636432633631636630633764653562333236643761383630", "0x333733324a4230783034636331626661393965323832653433346165663238", "0x31356361313733333761393233636432633631636630633764653562333236", "0x643761383630333733335200608094ebdc03688084af5f708084af5f788084", "0xaf5f82018c010a202ac49e648a1f84da5a143eeab68c8402c65a1567e63971", "0x7f5732d5e6310c2c761220a6c1ae85186dc981dc61cd14d7511ee5ab70258a", "0x10ac4e03e4d4991761b2c0a61a1090696dc7afed7f61a26887e78e683a1c1a", "0x10a29e5fa535f2edea7afa9acb4fd349b31a10d1b88713982955d79fa0e422", "0x685a748b1a10a07e0118cc38a71d2a9d60bf52938b4a"]}]}}` +const BLOCK_OUTPUT = `{"result": {"events": [ {"from_address": "0xd43963a4e875a361f5d164b2e70953598eb4f45fde86924082d51b4d78e489", "keys": ["0x9a144bf4a6a8fd083c93211e163e59221578efcc86b93f8c97c620e7b9608a", "0x0", "0x4b791b801cf0d7b6a2f9e59daf15ec2dd7d9cdc3bc5e037bada9c86e4821c"], "data": ["0x1", "0x4", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603730", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603734", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603731", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603735", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603732", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603736", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603733", "0x4cc1bfa99e282e434aef2815ca17337a923cd2c61cf0c7de5b326d7a8603737", "0x1", "0x3", "0x1", "0x0", "0xf4240", "0x2", "0x15", "0x263", "0x880a0d9e61d1080d88ee16f1880bcc1960b2080cab5ee01288090dfc04a30", "0x53a0201024220af400004fa5d02cd5170b5261032e71f2847ead36159cf8d", "0xee68affc3c8520904220af400004fa5d02cd5170b5261032e71f2847ead361", "0x59cf8dee68affc3c8520914220af400004fa5d02cd5170b5261032e71f2847", "0xead36159cf8dee68affc3c8520924220af400004fa5d02cd5170b5261032e7", "0x1f2847ead36159cf8dee68affc3c8520934a42307830346363316266613939", "0x65323832653433346165663238313563613137333337613932336364326336", "0x31636630633764653562333236643761383630333733304a42307830346363", "0x31626661393965323832653433346165663238313563613137333337613932", "0x33636432633631636630633764653562333236643761383630333733314a42", "0x30783034636331626661393965323832653433346165663238313563613137", "0x33333761393233636432633631636630633764653562333236643761383630", "0x333733324a4230783034636331626661393965323832653433346165663238", "0x31356361313733333761393233636432633631636630633764653562333236", "0x643761383630333733335200608094ebdc03688084af5f708084af5f788084", "0xaf5f82018c010a202ac49e648a1f84da5a143eeab68c8402c65a1567e63971", "0x7f5732d5e6310c2c761220a6c1ae85186dc981dc61cd14d7511ee5ab70258a", "0x10ac4e03e4d4991761b2c0a61a1090696dc7afed7f61a26887e78e683a1c1a", "0x10a29e5fa535f2edea7afa9acb4fd349b31a10d1b88713982955d79fa0e422", "0x685a748b1a10a07e0118cc38a71d2a9d60bf52938b4a"]}]}}` const ocr2ContractAddress = "0xd43963a4e875a361f5d164b2e70953598eb4f45fde86924082d51b4d78e489" // matches BLOCK_OUTPUT event func TestOCR2Client(t *testing.T) { diff --git a/relayer/pkg/chainlink/ocr2/events.go b/relayer/pkg/chainlink/ocr2/events.go index 01bf67c84..453faa5d9 100644 --- a/relayer/pkg/chainlink/ocr2/events.go +++ b/relayer/pkg/chainlink/ocr2/events.go @@ -10,6 +10,7 @@ import ( "github.com/smartcontractkit/libocr/offchainreporting2/types" + starknetrpc "github.com/NethermindEth/starknet.go/rpc" "github.com/smartcontractkit/chainlink-starknet/relayer/pkg/chainlink/ocr2/medianreport" "github.com/smartcontractkit/chainlink-starknet/relayer/pkg/starknet" ) @@ -40,33 +41,38 @@ type NewTransmissionEvent struct { } // ParseNewTransmissionEvent is decoding binary felt data as the NewTransmissionEvent type -func ParseNewTransmissionEvent(eventData []*felt.Felt) (NewTransmissionEvent, error) { +func ParseNewTransmissionEvent(event starknetrpc.EmittedEvent) (NewTransmissionEvent, error) { + eventData := event.Data { - const observationsLenIndex = 5 - const constNumOfElements = 11 + const observationsLenIndex = 3 + const constNumOfElements = 9 + const constNumOfKeys = 2 if len(eventData) < constNumOfElements { return NewTransmissionEvent{}, errors.New("invalid: event data") } + if len(event.Keys) < constNumOfKeys { + return NewTransmissionEvent{}, errors.New("invalid: event data") + } + observationsLen := eventData[observationsLenIndex].BigInt(big.NewInt(0)).Uint64() if len(eventData) != constNumOfElements+int(observationsLen) { return NewTransmissionEvent{}, errors.New("invalid: event data") } } + // keys[0] == event_id // round_id + roundId := uint32(event.Keys[1].BigInt(big.NewInt(0)).Uint64()) + // transmitter + transmitter := event.Keys[2] + index := 0 - roundId := uint32(eventData[index].BigInt(big.NewInt(0)).Uint64()) // answer - index++ latestAnswer := eventData[index].BigInt(big.NewInt(0)) - // transmitter - index++ - transmitter := eventData[index] - // observation_timestamp index++ unixTime := eventData[index].BigInt(big.NewInt(0)).Int64() @@ -130,9 +136,10 @@ func ParseNewTransmissionEvent(eventData []*felt.Felt) (NewTransmissionEvent, er } // ParseConfigSetEvent is decoding binary felt data as the libocr ContractConfig type -func ParseConfigSetEvent(eventData []*felt.Felt) (types.ContractConfig, error) { +func ParseConfigSetEvent(event starknetrpc.EmittedEvent) (types.ContractConfig, error) { + eventData := event.Data { - const oraclesLenIdx = 3 + const oraclesLenIdx = 1 if len(eventData) < oraclesLenIdx { return types.ContractConfig{}, errors.New("invalid: event data") } @@ -157,15 +164,15 @@ func ParseConfigSetEvent(eventData []*felt.Felt) (types.ContractConfig, error) { } } - index := 0 - // previous_config_block_number - skip + // keys[0] == event_id + // keys[1] == previous_config_block_number - skip // latest_config_digest - index++ - digest := eventData[index].Bytes() + digest := event.Keys[2].Bytes() + + index := 0 // config_count - index++ configCount := eventData[index].BigInt(big.NewInt(0)).Uint64() // oracles_len diff --git a/relayer/pkg/chainlink/ocr2/events_test.go b/relayer/pkg/chainlink/ocr2/events_test.go index e0f74fe90..5ad495a05 100644 --- a/relayer/pkg/chainlink/ocr2/events_test.go +++ b/relayer/pkg/chainlink/ocr2/events_test.go @@ -9,6 +9,7 @@ import ( "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" + starknetrpc "github.com/NethermindEth/starknet.go/rpc" starknetutils "github.com/NethermindEth/starknet.go/utils" "github.com/smartcontractkit/libocr/offchainreporting2/types" @@ -18,28 +19,37 @@ import ( ) var ( + // NewTransmission + newTransmissionEventKeysRaw = []string{ + "0x9", // event_id + "0x1", // round_id + "0x2c0dd77ce74b1667dc6fa782bbafaef5becbe2d04b052726ab236daeb52ac5d", // transmitter + } newTransmissionEventRaw = []string{ - "0x1", - "0x63", - "0x2c0dd77ce74b1667dc6fa782bbafaef5becbe2d04b052726ab236daeb52ac5d", - "0x1", - "0x10203000000000000000000000000000000000000000000000000000000", - "0x4", + "0x63", // answer + "0x1", // observation_timestamp + "0x10203000000000000000000000000000000000000000000000000000000", // observers + "0x4", // len(observations) "0x63", "0x63", "0x63", "0x63", - "0x1", - "0x1", - "0x485341c18461d70eac6ded4b8b17147f173308ddd56216a86f9ec4d994453", - "0x1", - "0x1", + "0x1", // juels_per_fee_coin + "0x1", // gas_price + "0x485341c18461d70eac6ded4b8b17147f173308ddd56216a86f9ec4d994453", // config_digest + "0x1", // epoch_and_round + "0x1", // reimbursement + } + + // ConfigSet + configSetEventKeysRaw = []string{ + "0x10", // event_id + "0x0", // previous_block_number + "0x485341c18461d70eac6ded4b8b17147f173308ddd56216a86f9ec4d994453", // lastest_config_digest } configSetEventRaw = []string{ - "0x0", - "0x485341c18461d70eac6ded4b8b17147f173308ddd56216a86f9ec4d994453", - "0x1", - "0x4", + "0x1", // config_count + "0x4", // len(oracles) "0x21e867aa6e6c545949a9c6f9f5401b70007bd93675857a0a7d5345b8bffcbf0", "0x2c0dd77ce74b1667dc6fa782bbafaef5becbe2d04b052726ab236daeb52ac5d", "0x64642f34e68436f45757b920f4cdfbdff82728844d740bac672a19ad72011ca", @@ -48,24 +58,32 @@ var ( "0x2f14e18cc198dd5133c8a9aa92992fc1a462f703401716f402d0ee383b54faa", "0x4fcf11b05ebd00a207030c04836defbec3d37a3f77e581f2d0962a20a55adcd", "0x5c35686f78db31d9d896bb425b3fd99be19019f8aeaf0f7a8767867903341d4", - "0x1", - "0x3", + "0x1", // f + "0x3", // len(onchain_config) "0x1", "0xa", "0x3b9aca00", - "0x2", - "0x2", + "0x2", // offchain_config_version + "0x2", // len(offchain_config) "0x1", "0x1", } ) func TestNewTransmissionEvent_Parse(t *testing.T) { + eventKeys, err := starknetutils.HexArrToFelt(newTransmissionEventKeysRaw) + assert.NoError(t, err) eventData, err := starknetutils.HexArrToFelt(newTransmissionEventRaw) assert.NoError(t, err) require.Equal(t, len(newTransmissionEventRaw), len(eventData)) - e, err := ParseNewTransmissionEvent(eventData) + event := starknetrpc.EmittedEvent{ + Event: starknetrpc.Event{ + Keys: eventKeys, + Data: eventData, + }, + } + e, err := ParseNewTransmissionEvent(event) assert.NoError(t, err) require.Equal(t, e.RoundId, uint32(1)) @@ -95,11 +113,19 @@ func TestNewTransmissionEvent_Parse(t *testing.T) { } func TestConfigSetEvent_Parse(t *testing.T) { + eventKeys, err := starknetutils.HexArrToFelt(configSetEventKeysRaw) + assert.NoError(t, err) eventData, err := starknetutils.HexArrToFelt(configSetEventRaw) assert.NoError(t, err) require.Equal(t, len(configSetEventRaw), len(eventData)) - e, err := ParseConfigSetEvent(eventData) + event := starknetrpc.EmittedEvent{ + Event: starknetrpc.Event{ + Keys: eventKeys, + Data: eventData, + }, + } + e, err := ParseConfigSetEvent(event) assert.NoError(t, err) configDigest := XXXMustBytesToConfigDigest(starknet.XXXMustHexDecodeString("000485341c18461d70eac6ded4b8b17147f173308ddd56216a86f9ec4d994453")) From a12998341bf5c33b9e90a586e2ffc15ec9030370 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bla=C5=BE=20Hrastnik?= Date: Mon, 8 Apr 2024 17:19:15 +0900 Subject: [PATCH 5/5] Adjust constNumOfKeys to also include event.ID in the count --- relayer/pkg/chainlink/ocr2/events.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/relayer/pkg/chainlink/ocr2/events.go b/relayer/pkg/chainlink/ocr2/events.go index 453faa5d9..22f0cd0d4 100644 --- a/relayer/pkg/chainlink/ocr2/events.go +++ b/relayer/pkg/chainlink/ocr2/events.go @@ -46,7 +46,7 @@ func ParseNewTransmissionEvent(event starknetrpc.EmittedEvent) (NewTransmissionE { const observationsLenIndex = 3 const constNumOfElements = 9 - const constNumOfKeys = 2 + const constNumOfKeys = 2 + 1 // additional 1 for the automatic event ID key if len(eventData) < constNumOfElements { return NewTransmissionEvent{}, errors.New("invalid: event data")