/protomerkle

build merkle trees on-chain

Primary LanguageSolidity

Protomerkle

Build merkle trees on-chain. Efficient and low mana costs!

This library attempts to be as generic as solidity allows without compromising on security or efficiency.

Mana efficiency

It's cheap!

Running 1 test for test/MerkleTests.sol:MerkleTests
[PASS] testStressTestTree() (gas: 898692)
Logs:
  insert index 0 (cold)
  mana: 36802
  insert index 1 (warm)
  mana: 27494
  update index 1 (warmer)
  mana: 5601

UPDATE: forge test is lying about consumption. based on internal tests (not committed to the git at the moment) the actual numbers are more like:

insert index 0 (cold): 61773
insert index 1 (warm): 35083
insert index 1 (warmer): 12921
3 items batch (cold): 134033
3 items batch (warm): 40887

Testing

forge test -vv

Integration

Install the library:

forge install hananbeer/protomerkle

The base library is MerkleTree, but there is also an extended MerkleSumTree and CachedMerkleTree (great for L2, has cheaper appendItem at insignificantly costlier inserts):

import "protomerkle/src/MerkleTree.sol";

The library has a method updateItems which is public payable (I probably need to change this to internal soon) which accepts an index of an item to update, inclusion proof and arbitrary update parameters. (bytes memory params)

If the item exists in the merkle tree then the _onBeforeUpdate function is called. Modify the item in this function and return the new item abi.encode()ed. Once the item has been updated in storage the _onAfterUpdate function is called with the new item and the update params.

The default _onBeforeUpdate simply copies the update params into the item, but you may choose to implement partial updates, apply deltas or simply perform validation logic first. (note this function is currently view to restrict state modifications - this might need to change in the future)

Note how you don't need to define the storage for your data - the library does it for you! Simply define the data structures for your items (MyStruct here) and update params (MyUpdateParam here) and decode them in the callback _onBeforeUpdate.

import "protomerkle/src/MerkleTree.sol";

uint8 constant TREE_HEIGHT = 5;
uint16 constant ITEM_SIZE = 64; // this is sizeof(MyStruct)

contract ExampleMerkleTree is MerkleTree {
    struct MyStruct {
        uint256 value;
        uint256 lastChanged;
    }

    struct MyUpdateParam {
        uint256 amount;
    }

    constructor() MerkleTree(TREE_HEIGHT, ITEM_SIZE) {
    }

    function _onBeforeUpdate(
        uint256 index,
        bytes memory item,
        bytes calldata params
    ) internal view override returns (bytes memory) {
        // permission check
        require(hasRole(msg.sender, MERKLE_MODIFIER), "not approved");

        // parse the change
        MyStruct memory data = abi.decode(item, (MyStruct));
        MyUpdateParam memory update = abi.decode(params, (MyUpdateParam));

        // soft limit the amount
        if (update.amount > 1000)
            update.amount = 1000;

        // apply a delta change
        data.value += update.amount * 1e18;

        // apply a change not coming from params
        data.lastChanged = block.timestamp;

        // make sure to return the item as bytes using abi.encode()
        // until solidity gets generics we are stuck with this raw bytes horror
        return abi.encode(data);
    }
    
    function _onAfterUpdate(
        uint256 index,
        bytes memory item,
        bytes calldata params
    ) internal override {
        // do your state changes here
        //MyUpdateParam memory update = abi.decode(params, (MyUpdateParam));
        //address WETH = address(0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2);
        //WETH.transferFrom(msg.sender, address(this), data.amount * 1e18);
    }
}

One small caveat of Solidity lacking generics is the need to specify the size of your items data structure in the constructor.

Here it is done manually, but you can also do something like this:

    function _getItemSize() internal {
        MyStruct memory empty;
        return abi.encode(empty).length;
    }

Note that the library does not keep track of number of items (this actually saved a ton of mana - up to 38%!!) so if you need to do that - do it yourself!