Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Snap sync: use zero-element proof for checking validity of final, empty range result #3047

Merged
merged 26 commits into from
Feb 26, 2024

Conversation

scorbajio
Copy link
Contributor

@scorbajio scorbajio commented Sep 20, 2023

During snap sync, if the client requests a range that is both empty and has no elements remaining to the right of it, the peer responds with just a proof. This proof can be checked as part of a zero-element proof to validate that there really are no remaining elements to be fetched past the range. This helps avoid premature halting, corrupt peers, and DoS attacks on snap sync.

@codecov
Copy link

codecov bot commented Sep 20, 2023

Codecov Report

Attention: Patch coverage is 70.96774% with 18 lines in your changes are missing coverage. Please review.

Project coverage is 86.92%. Comparing base (0e186fd) to head (34a7c12).

Additional details and impacted files

Impacted file tree graph

Flag Coverage Δ
block 88.34% <ø> (ø)
blockchain 91.61% <ø> (ø)
client 84.72% <97.43%> (+0.04%) ⬆️
common 98.25% <ø> (ø)
devp2p 82.12% <ø> (ø)
ethash ∅ <ø> (∅)
evm 74.33% <ø> (ø)
genesis 99.98% <ø> (ø)
rlp ∅ <ø> (∅)
statemanager 77.02% <ø> (ø)
trie 89.24% <26.08%> (-0.27%) ⬇️
tx 95.45% <ø> (ø)
util 89.13% <ø> (ø)
vm 80.20% <ø> (ø)
wallet 88.35% <ø> (ø)

Flags with carried forward coverage won't be shown. Click here to find out more.

@holgerd77
Copy link
Member

Updated this via UI

Can this get a review at some point? 🙏

@holgerd77
Copy link
Member

Will put this back in WIP since it has failing tests.

@holgerd77
Copy link
Member

@scorbajio What is the status of this PR?

Copy link
Member

@jochem-brouwer jochem-brouwer left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Two comments.

The implementation looks correct to me, but it seems that it lacks tests of the "unhappy" case that a peer gives us a zero element proof when there are clearly elements left in the trie (for both account + storage fetcher). Plus a case of the all-elements proof which is not covered 😄 👍

useKeyHashingFunction
)

if (value !== null || (await hasRightElement(trie, firstKey))) {
if (value !== null || (await hasRightElement(trie, firstKey as any))) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why does the as any has to be added here? This was not necessary before (also in other occurrences of this file)

Copy link
Contributor Author

@scorbajio scorbajio Feb 16, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

See my other comment bellow (#3047 (comment)). It seems like the implementation wasn't allowing the zero-element parameter combination of an existing firstKey, a null lastKey, and an existing proof to continue to the zero-element proof verification section of the verifyRangeProof implementation because it assumed if firstKey, lastKey, and proof aren't all non-null values after moving past the all-elements proof verification section of the code, that there is an input error, when this is not the case for a zero-element proof. I ended up moving the conditional that checks firstKey, lastKey, or proof aren't null to right after the zero-element proof check, so those three variables aren't being checked to be non-null in that section.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've added a check to make sure the parameters are as expected when running a zero-element proof, so the casting to any has been removed.

throw new Error('invalid zero element proof: value mismatch')
}

return false
}

if (proof === null || firstKey === null || lastKey === null) {
throw new Error(
'invalid all elements proof: proof, firstKey, lastKey must be null at the same time'
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is new, but has to do with an all elements proof 🤔 (looks correct to me, but maybe add a test 😄 )

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This check is actually not new, I've only moved it bellow where zero-element proof checking happens. It was previously like this:

  if (proof === null || firstKey === null || lastKey === null) {
    throw new Error(
      'invalid all elements proof: proof, firstKey, lastKey must be null at the same time'
    )
  }

  // Zero element proof
  if (keys.length === 0) {
    const { trie, value } = await verifyProof(
      rootHash,
      nibblestoBytes(firstKey),
      proof,
      useKeyHashingFunction
    )

    if (value !== null || (await hasRightElement(trie, firstKey))) {
      throw new Error('invalid zero element proof: value mismatch')
    }

    return false
  }

The reason I did that was because zero-element proof checking was failing when I was passing in just the root, firstKey, and proof, with null for lastKey and empty arrays for keys and values, as such:

          const isMissingRightRange = await Trie.verifyRangeProof(
            this.root,
            origin,
            null,
            [],
            [],
            <any>rangeResult.proof,
            { useKeyHashingFunction: keccak256 }
          )

The error resulting from this was confusing a zero-element request to be a failed all-elements request:

2024-02-16T00:22:13.531Z client:AccountFetcher Error: invalid all elements proof: proof, firstKey, lastKey must be null at the same time
    at Module.verifyRangeProof (/Users/indigoalpha/code/ethjs/0/ethereumjs-monorepo/packages/trie/src/proof/range.ts:450:11)
    at Function.verifyRangeProof (/Users/indigoalpha/code/ethjs/0/ethereumjs-monorepo/packages/trie/src/trie.ts:208:12)
    at AccountFetcher.request (/Users/indigoalpha/code/ethjs/0/ethereumjs-monorepo/packages/client/src/sync/fetcher/accountfetcher.ts:400:50)
    at /Users/indigoalpha/code/ethjs/0/ethereumjs-monorepo/packages/client/test/sync/fetcher/accountfetcher.spec.ts:313:17
    at runTest (file:///Users/indigoalpha/code/ethjs/0/ethereumjs-monorepo/node_modules/@vitest/runner/dist/index.js:486:11)
    at runSuite (file:///Users/indigoalpha/code/ethjs/0/ethereumjs-monorepo/node_modules/@vitest/runner/dist/index.js:594:15)
    at runSuite (file:///Users/indigoalpha/code/ethjs/0/ethereumjs-monorepo/node_modules/@vitest/runner/dist/index.js:594:15)
    at runFiles (file:///Users/indigoalpha/code/ethjs/0/ethereumjs-monorepo/node_modules/@vitest/runner/dist/index.js:645:5)
    at startTests (file:///Users/indigoalpha/code/ethjs/0/ethereumjs-monorepo/node_modules/@vitest/runner/dist/index.js:654:3)
    at file:///Users/indigoalpha/code/ethjs/0/ethereumjs-monorepo/node_modules/vitest/dist/entry.js:278:7

@holgerd77
Copy link
Member

@scorbajio This is a somewhat strange order of things with some likelihood to cause confusion with Jochem doing a review and leave some comments and then this PR being set to "Needs review" without the things mentioned actually being adressed. 🤔

Maybe as some general suggestion always good to add some short comment when one is changing the PR state, this always helps to avoid misunderstandings and additionaly avoid of PR state changes getting lost since GitHub does not notify.

Thanks 🙏

@scorbajio
Copy link
Contributor Author

scorbajio commented Feb 16, 2024

Two comments.

The implementation looks correct to me, but it seems that it lacks tests of the "unhappy" case that a peer gives us a zero element proof when there are clearly elements left in the trie (for both account + storage fetcher). Plus a case of the all-elements proof which is not covered 😄 👍

That's a good point 🙂. I've added a basic test for the storage fetcher and account fetcher to test if they will reject a zero-element proof if an element to the right exists, will be able to write more range proof tests once the flat db work is further along and we can create our own range proofs, but the all-elements proofs are already being tested in the trie package (e.g.

it('create all element range proof and verify it', async () => {
).

@holgerd77
Copy link
Member

Updated this via UI

@jochem-brouwer can this get a review update?

@holgerd77
Copy link
Member

Updated this via UI

Copy link
Member

@holgerd77 holgerd77 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Review comments seem to be addressed.

Will merge due to being already outstanding for a couple of days.

@holgerd77 holgerd77 merged commit d89a963 into master Feb 26, 2024
45 of 46 checks passed
@holgerd77 holgerd77 deleted the snap-sync-no-element-proof branch February 26, 2024 13:16
jochem-brouwer pushed a commit that referenced this pull request Mar 1, 2024
…ty range result (#3047)

* Use no-elements proof for final check in account fetcher

* Use no-elements proof for final check in storage fetcher

* Check inputs after zero element proof code section

* Cleanup

* Reject peer if zero-element proof fails

* Add tests for zero-element proof

* Remove proofTrie usage

* Use hashing function in static version of verifyRangeProof

* Include useKeyHashing in call to fromProof so that hashing function is used in proof verification

* Use appliedKey for hashing function instead of the function directly passed in TrieOpts

* Pass in hashing function for use in static proof verification calls

* Fix static range proof verification errors

* Use custom hashing function from opts if available

* Add test to check if zero element range proof verification fails with remaining elements to the right

* Check if parameters are as expected for zero-element proof

* Fix linting issues

---------

Co-authored-by: Scotty <66335769+ScottyPoi@users.noreply.github.com>
Co-authored-by: Holger Drewes <Holger.Drewes@gmail.com>
Co-authored-by: Indigo Alpha <indigoalpha@indigos-mbp.mynetworksettings.com>
jochem-brouwer added a commit that referenced this pull request Mar 1, 2024
* client: PreimageManager and Preimage storage in MetaDB

* common: add getAppliedKey method to StateManagerInterface

* evm: add preimage reporting

* statemanager: add getAppliedKey methods to the statemanager implementations

* vm: improve some types

* vm: implement preimage reporting

* client: add preimage saving

* evm: update to unprefixedHexToBytes

* vm: add reportPreimages test to runTx tests

* vm: pass down reportPreimages arg from block to tx

* client: pass down reportPreimages from client to vm through runBlock

* client: unify config options naming

* client: preimageManager test

* client: test preimage, wip

* common: add todo to gasPrices from 6800

* statemanager: fix type issues

* dev accessWitness class on the lines of geth impl

* integrate accesswitness in evm/vm/runtx flow

* plugin the gas schedule for tx origin and destination accesses

* complete, debug and fix the call and create access charges

* plug the access gas usage on the evm opcode runs

* debug and fix the code chunk accesses and the poststate setting

* implement access merging and accessed state tracking and traversing

* also provide chunkKey for easy reference

* decode raw accesses to structured ones and debug log them

* debug and add the missing accesses for coinbase, withdrawals

* stateManager: implement cache handling in the context of statelessness

* modify poststate to use accesses from the accesswitness to compare and fix mising chunkKey in returned accesses

* stateManager: getComputedValue

* fixes for the getcomputedval fn helper for the code

* correctly implement the getcontractcode with partial accessed segments available  and corresponding error handling in evm run

* statemanager: tentative fixes & improvements to code cache handling

* statemanager: checkpoint code cache

* fix the code chunk comparision in post state verification

* rename kaustinen2 local testnet data for

* fix pre and post state null chunks handling and corresponding get computed fixes

* setup the client to statelessly execute the verkle blocks randomly from anywhere the chain

* setup to statelessly run block13 of kaustinen2 in client test spec for easy debugging

* setup the client kaustinen2 stateless test to test and debug block 16

* improve the post state witness mismatch logging to be more comprehensible for debugging

* improve chunk verification tracking logging and fix post/prestate key coding from the witnesses

* debug and fix code accesses and overhaul/fix accesscharges on create/call

* add a more complete matching of generated and provided executed witnesses

* fix return

* debug, discuss and remove proof of absence charges for contract create for fixing the extra witnesses

* only charge code accesses if accessed from state and when code is written on contract creation

* handle bigint treeIndex for the bigint slot and debug/match the predersenhash with kauntinen2 usage

* client: fix kaustinen tests and migrate to new testing framework

* shift kaustinen 2 test to block 13

* client: remove stale kaustinen2 test data

* vm: adjust rewardAccount new common arg ordering and make it optional

* add block12 to the spec list

* move invalid opcode check outside jump analysis

* small cleanup in runtx

* add preimages spec

* build various block scenarios for preimages gen

* client: revert vmexecution preimages test

* client: remove unnecessary boolean explicit comparison

* client: remove unused import

* client: revert re-ordering

* stateManager: support non keccak256 applied keys

* debug and persist preimages on runWithoutSetHead as well

* add preimages for coinbase,withdrawals to the test and fix the spec to validate them

* client: fix preimage tests ts errors

* client: minor adjustments to preimage test

* client: save preimages helper

* vm: save preimages at the block level, new applyBlockResult type

* common: make getAppliedKey optional

* evm/vm: handle optional getAppliedKey

* client: add preimage doc

* trie: export "Path" interface (#3292)

* trie: export "Path" interface

* Move over Path interface export to types for consistency

---------

Co-authored-by: Holger Drewes <Holger.Drewes@gmail.com>

* Snap sync: use zero-element proof for checking validity of final, empty range result (#3047)

* Use no-elements proof for final check in account fetcher

* Use no-elements proof for final check in storage fetcher

* Check inputs after zero element proof code section

* Cleanup

* Reject peer if zero-element proof fails

* Add tests for zero-element proof

* Remove proofTrie usage

* Use hashing function in static version of verifyRangeProof

* Include useKeyHashing in call to fromProof so that hashing function is used in proof verification

* Use appliedKey for hashing function instead of the function directly passed in TrieOpts

* Pass in hashing function for use in static proof verification calls

* Fix static range proof verification errors

* Use custom hashing function from opts if available

* Add test to check if zero element range proof verification fails with remaining elements to the right

* Check if parameters are as expected for zero-element proof

* Fix linting issues

---------

Co-authored-by: Scotty <66335769+ScottyPoi@users.noreply.github.com>
Co-authored-by: Holger Drewes <Holger.Drewes@gmail.com>
Co-authored-by: Indigo Alpha <indigoalpha@indigos-mbp.mynetworksettings.com>

* Integrate `kzg-wasm` into monorepo (#3294)

* Update to official trusted setup

* Remove devnet6

* Add kzg-wasm

* Update block tests to use kzg-wasm

* Update tests

* Add more 4844 tests to browser run

* Initial integration of kzg-wasm on git

* Update kzg-wasm build

* Fix linter weirdness

* Move initKzg to `runTests`

* Fix tests

* More cleanup

* Goodbye c-kzg

* fix kzg references

* Replace c-kzg with kzg-wasm in package.json

* Update kzg wasm commit and vm tester config

* Update initKzg to createKZG

* fix copy pasta

* Fix more copy pasta

* update kzg-wasm to npm release

* One last bit of copy pasta

* Address feedback

* client: remove try/catch blocks createKZG() and remove the initKZG stale comment

---------

Co-authored-by: Jochem Brouwer <jochembrouwer96@gmail.com>

* Make trustedSetupPath in Util kzg module optional (#3296)

* client: add tx preimages to preimage test cases

---------

Co-authored-by: acolytec3 <17355484+acolytec3@users.noreply.github.com>
Co-authored-by: harkamal <gajinder@g11.in>
Co-authored-by: Jochem Brouwer <jochembrouwer96@gmail.com>
Co-authored-by: Scotty <66335769+ScottyPoi@users.noreply.github.com>
Co-authored-by: Holger Drewes <Holger.Drewes@gmail.com>
Co-authored-by: Scorbajio <indigophi@protonmail.com>
Co-authored-by: Indigo Alpha <indigoalpha@indigos-mbp.mynetworksettings.com>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants