123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267 |
- // SPDX-License-Identifier: MIT
- // OpenZeppelin Contracts (last updated v5.3.0) (utils/structs/MerkleTree.sol)
- pragma solidity ^0.8.20;
- import {Hashes} from "../cryptography/Hashes.sol";
- import {Arrays} from "../Arrays.sol";
- import {Panic} from "../Panic.sol";
- import {StorageSlot} from "../StorageSlot.sol";
- /**
- * @dev Library for managing https://wikipedia.org/wiki/Merkle_Tree[Merkle Tree] data structures.
- *
- * Each tree is a complete binary tree with the ability to sequentially insert leaves, changing them from a zero to a
- * non-zero value and updating its root. This structure allows inserting commitments (or other entries) that are not
- * stored, but can be proven to be part of the tree at a later time if the root is kept. See {MerkleProof}.
- *
- * A tree is defined by the following parameters:
- *
- * * Depth: The number of levels in the tree, it also defines the maximum number of leaves as 2**depth.
- * * Zero value: The value that represents an empty leaf. Used to avoid regular zero values to be part of the tree.
- * * Hashing function: A cryptographic hash function used to produce internal nodes. Defaults to {Hashes-commutativeKeccak256}.
- *
- * NOTE: Building trees using non-commutative hashing functions (i.e. `H(a, b) != H(b, a)`) is supported. However,
- * proving the inclusion of a leaf in such trees is not possible with the {MerkleProof} library since it only supports
- * _commutative_ hashing functions.
- *
- * _Available since v5.1._
- */
- library MerkleTree {
- /// @dev Error emitted when trying to update a leaf that was not previously pushed.
- error MerkleTreeUpdateInvalidIndex(uint256 index, uint256 length);
- /// @dev Error emitted when the proof used during an update is invalid (could not reproduce the side).
- error MerkleTreeUpdateInvalidProof();
- /**
- * @dev A complete `bytes32` Merkle tree.
- *
- * The `sides` and `zero` arrays are set to have a length equal to the depth of the tree during setup.
- *
- * Struct members have an underscore prefix indicating that they are "private" and should not be read or written to
- * directly. Use the functions provided below instead. Modifying the struct manually may violate assumptions and
- * lead to unexpected behavior.
- *
- * NOTE: The `root` and the updates history is not stored within the tree. Consider using a secondary structure to
- * store a list of historical roots from the values returned from {setup} and {push} (e.g. a mapping, {BitMaps} or
- * {Checkpoints}).
- *
- * WARNING: Updating any of the tree's parameters after the first insertion will result in a corrupted tree.
- */
- struct Bytes32PushTree {
- uint256 _nextLeafIndex;
- bytes32[] _sides;
- bytes32[] _zeros;
- }
- /**
- * @dev Initialize a {Bytes32PushTree} using {Hashes-commutativeKeccak256} to hash internal nodes.
- * The capacity of the tree (i.e. number of leaves) is set to `2**treeDepth`.
- *
- * Calling this function on MerkleTree that was already setup and used will reset it to a blank state.
- *
- * Once a tree is setup, any push to it must use the same hashing function. This means that values
- * should be pushed to it using the default {xref-MerkleTree-push-struct-MerkleTree-Bytes32PushTree-bytes32-}[push] function.
- *
- * IMPORTANT: The zero value should be carefully chosen since it will be stored in the tree representing
- * empty leaves. It should be a value that is not expected to be part of the tree.
- */
- function setup(Bytes32PushTree storage self, uint8 treeDepth, bytes32 zero) internal returns (bytes32 initialRoot) {
- return setup(self, treeDepth, zero, Hashes.commutativeKeccak256);
- }
- /**
- * @dev Same as {xref-MerkleTree-setup-struct-MerkleTree-Bytes32PushTree-uint8-bytes32-}[setup], but allows to specify a custom hashing function.
- *
- * Once a tree is setup, any push to it must use the same hashing function. This means that values
- * should be pushed to it using the custom push function, which should be the same one as used during the setup.
- *
- * IMPORTANT: Providing a custom hashing function is a security-sensitive operation since it may
- * compromise the soundness of the tree.
- *
- * NOTE: Consider verifying that the hashing function does not manipulate the memory state directly and that it
- * follows the Solidity memory safety rules. Otherwise, it may lead to unexpected behavior.
- */
- function setup(
- Bytes32PushTree storage self,
- uint8 treeDepth,
- bytes32 zero,
- function(bytes32, bytes32) view returns (bytes32) fnHash
- ) internal returns (bytes32 initialRoot) {
- // Store depth in the dynamic array
- Arrays.unsafeSetLength(self._sides, treeDepth);
- Arrays.unsafeSetLength(self._zeros, treeDepth);
- // Build each root of zero-filled subtrees
- bytes32 currentZero = zero;
- for (uint256 i = 0; i < treeDepth; ++i) {
- Arrays.unsafeAccess(self._zeros, i).value = currentZero;
- currentZero = fnHash(currentZero, currentZero);
- }
- // Set the first root
- self._nextLeafIndex = 0;
- return currentZero;
- }
- /**
- * @dev Insert a new leaf in the tree, and compute the new root. Returns the position of the inserted leaf in the
- * tree, and the resulting root.
- *
- * Hashing the leaf before calling this function is recommended as a protection against
- * second pre-image attacks.
- *
- * This variant uses {Hashes-commutativeKeccak256} to hash internal nodes. It should only be used on merkle trees
- * that were setup using the same (default) hashing function (i.e. by calling
- * {xref-MerkleTree-setup-struct-MerkleTree-Bytes32PushTree-uint8-bytes32-}[the default setup] function).
- */
- function push(Bytes32PushTree storage self, bytes32 leaf) internal returns (uint256 index, bytes32 newRoot) {
- return push(self, leaf, Hashes.commutativeKeccak256);
- }
- /**
- * @dev Insert a new leaf in the tree, and compute the new root. Returns the position of the inserted leaf in the
- * tree, and the resulting root.
- *
- * Hashing the leaf before calling this function is recommended as a protection against
- * second pre-image attacks.
- *
- * This variant uses a custom hashing function to hash internal nodes. It should only be called with the same
- * function as the one used during the initial setup of the merkle tree.
- */
- function push(
- Bytes32PushTree storage self,
- bytes32 leaf,
- function(bytes32, bytes32) view returns (bytes32) fnHash
- ) internal returns (uint256 index, bytes32 newRoot) {
- // Cache read
- uint256 treeDepth = depth(self);
- // Get leaf index
- index = self._nextLeafIndex++;
- // Check if tree is full.
- if (index >= 1 << treeDepth) {
- Panic.panic(Panic.RESOURCE_ERROR);
- }
- // Rebuild branch from leaf to root
- uint256 currentIndex = index;
- bytes32 currentLevelHash = leaf;
- for (uint256 i = 0; i < treeDepth; i++) {
- // Reaching the parent node, is currentLevelHash the left child?
- bool isLeft = currentIndex % 2 == 0;
- // If so, next time we will come from the right, so we need to save it
- if (isLeft) {
- Arrays.unsafeAccess(self._sides, i).value = currentLevelHash;
- }
- // Compute the current node hash by using the hash function
- // with either its sibling (side) or the zero value for that level.
- currentLevelHash = fnHash(
- isLeft ? currentLevelHash : Arrays.unsafeAccess(self._sides, i).value,
- isLeft ? Arrays.unsafeAccess(self._zeros, i).value : currentLevelHash
- );
- // Update node index
- currentIndex >>= 1;
- }
- return (index, currentLevelHash);
- }
- /**
- * @dev Change the value of the leaf at position `index` from `oldValue` to `newValue`. Returns the recomputed "old"
- * root (before the update) and "new" root (after the update). The caller must verify that the reconstructed old
- * root is the last known one.
- *
- * The `proof` must be an up-to-date inclusion proof for the leaf being updated. This means that this function is
- * vulnerable to front-running. Any {push} or {update} operation (that changes the root of the tree) would render
- * all "in flight" updates invalid.
- *
- * This variant uses {Hashes-commutativeKeccak256} to hash internal nodes. It should only be used on merkle trees
- * that were setup using the same (default) hashing function (i.e. by calling
- * {xref-MerkleTree-setup-struct-MerkleTree-Bytes32PushTree-uint8-bytes32-}[the default setup] function).
- */
- function update(
- Bytes32PushTree storage self,
- uint256 index,
- bytes32 oldValue,
- bytes32 newValue,
- bytes32[] memory proof
- ) internal returns (bytes32 oldRoot, bytes32 newRoot) {
- return update(self, index, oldValue, newValue, proof, Hashes.commutativeKeccak256);
- }
- /**
- * @dev Change the value of the leaf at position `index` from `oldValue` to `newValue`. Returns the recomputed "old"
- * root (before the update) and "new" root (after the update). The caller must verify that the reconstructed old
- * root is the last known one.
- *
- * The `proof` must be an up-to-date inclusion proof for the leaf being update. This means that this function is
- * vulnerable to front-running. Any {push} or {update} operation (that changes the root of the tree) would render
- * all "in flight" updates invalid.
- *
- * This variant uses a custom hashing function to hash internal nodes. It should only be called with the same
- * function as the one used during the initial setup of the merkle tree.
- */
- function update(
- Bytes32PushTree storage self,
- uint256 index,
- bytes32 oldValue,
- bytes32 newValue,
- bytes32[] memory proof,
- function(bytes32, bytes32) view returns (bytes32) fnHash
- ) internal returns (bytes32 oldRoot, bytes32 newRoot) {
- unchecked {
- // Check index range
- uint256 length = self._nextLeafIndex;
- if (index >= length) revert MerkleTreeUpdateInvalidIndex(index, length);
- // Cache read
- uint256 treeDepth = depth(self);
- // Workaround stack too deep
- bytes32[] storage sides = self._sides;
- // This cannot overflow because: 0 <= index < length
- uint256 lastIndex = length - 1;
- uint256 currentIndex = index;
- bytes32 currentLevelHashOld = oldValue;
- bytes32 currentLevelHashNew = newValue;
- for (uint32 i = 0; i < treeDepth; i++) {
- bool isLeft = currentIndex % 2 == 0;
- lastIndex >>= 1;
- currentIndex >>= 1;
- if (isLeft && currentIndex == lastIndex) {
- StorageSlot.Bytes32Slot storage side = Arrays.unsafeAccess(sides, i);
- if (side.value != currentLevelHashOld) revert MerkleTreeUpdateInvalidProof();
- side.value = currentLevelHashNew;
- }
- bytes32 sibling = proof[i];
- currentLevelHashOld = fnHash(
- isLeft ? currentLevelHashOld : sibling,
- isLeft ? sibling : currentLevelHashOld
- );
- currentLevelHashNew = fnHash(
- isLeft ? currentLevelHashNew : sibling,
- isLeft ? sibling : currentLevelHashNew
- );
- }
- return (currentLevelHashOld, currentLevelHashNew);
- }
- }
- /**
- * @dev Tree's depth (set at initialization)
- */
- function depth(Bytes32PushTree storage self) internal view returns (uint256) {
- return self._zeros.length;
- }
- }
|