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

Limit data exposure in public grant views #13

Merged
merged 3 commits into from
Jan 8, 2025
Merged

Conversation

Pabl0cks
Copy link
Member

@Pabl0cks Pabl0cks commented Jan 7, 2025

Creating a new getPublicGrants function that doesn't get all grants data, so it doesn't get exposed in public views when users check their console.

GrantDetails page will keep all data from grants since it's displayed in the screen (only admins and the own grantee can access to that page)

To test:

In this repo:

  1. Switch to this branch.

  2. Add the following variables in .env.local:

POSTGRES_URL="postgresql://postgres:mysecretpassword@localhost:5432/postgres"
NEXTAUTH_URL=http://localhost:3000
NEXTAUTH_SECRET=somereallysecretsecret
  1. Setup the dev environment:

    Update `Stream.sol` to lower frequency
    // SPDX-License-Identifier: MIT
    pragma solidity >=0.8.0 <0.9.0;
    
    import "@openzeppelin/contracts/access/AccessControl.sol";
    
    contract Stream is AccessControl {
        bytes32 public constant OWNER_ROLE = keccak256("OWNER_ROLE");
    
        struct GrantStream {
    	    uint256 cap;
    	    uint256 last;
    	    uint256 amountLeft;
    	    uint8 grantNumber;
    	    uint8 stageNumber;
    	    address builder;
        }
    
        struct BuilderGrantData {
    	    uint256 grantId;
    	    uint8 grantNumber;
        }
    
        mapping(uint256 => GrantStream) public grantStreams;
        uint256 public nextGrantId = 1;
    
        mapping(address => BuilderGrantData[]) public builderGrants;
    
        uint256 public constant FULL_STREAM_UNLOCK_PERIOD = 60; // 1 min
        uint256 public constant DUST_THRESHOLD = 1000000000000000; // 0.001 ETH
    
        event Withdraw(
    	    address indexed to,
    	    uint256 amount,
    	    string reason,
    	    uint256 grantId,
    	    uint8 grantNumber,
    	    uint8 stageNumber
        );
        event AddGrant(uint256 indexed grantId, address indexed to, uint256 amount);
        event ReinitializeGrant(
    	    uint256 indexed grantId,
    	    address indexed to,
    	    uint256 amount
        );
        event MoveGrantToNextStage(
    	    uint256 indexed grantId,
    	    address indexed to,
    	    uint256 amount,
    	    uint8 grantNumber,
    	    uint8 stageNumber
        );
        event ReinitializeNextStage(
    	    uint256 indexed grantId,
    	    address indexed builder,
    	    uint256 amount,
    	    uint8 grantNumber,
    	    uint8 stageNumber
        );
        event UpdateGrant(
    	    uint256 indexed grantId,
    	    address indexed to,
    	    uint256 cap,
    	    uint256 last,
    	    uint256 amountLeft,
    	    uint8 grantNumber,
    	    uint8 stageNumber
        );
        event AddOwner(address indexed newOwner, address indexed addedBy);
        event RemoveOwner(address indexed removedOwner, address indexed removedBy);
    
        // Custom errors
        error NoActiveStream();
        error InsufficientContractFunds();
        error UnauthorizedWithdrawal();
        error InsufficientStreamFunds();
        error FailedToSendEther();
        error PreviousAmountNotFullyWithdrawn();
        error AlreadyWithdrawnFromGrant();
    
        constructor(address[] memory _initialOwners) {
    	    _setRoleAdmin(OWNER_ROLE, OWNER_ROLE);
    	    for (uint i = 0; i < _initialOwners.length; i++) {
    		    _grantRole(OWNER_ROLE, _initialOwners[i]);
    	    }
        }
    
        function unlockedGrantAmount(
    	    uint256 _grantId
        ) public view returns (uint256) {
    	    GrantStream memory grantStream = grantStreams[_grantId];
    	    if (grantStream.cap == 0) revert NoActiveStream();
    
    	    if (grantStream.amountLeft == 0) {
    		    return 0;
    	    }
    
    	    uint256 elapsedTime = block.timestamp - grantStream.last;
    	    uint256 unlockedAmount = (grantStream.cap * elapsedTime) /
    		    FULL_STREAM_UNLOCK_PERIOD;
    
    	    return
    		    unlockedAmount > grantStream.amountLeft
    			    ? grantStream.amountLeft
    			    : unlockedAmount;
        }
    
        function addGrantStream(
    	    address _builder,
    	    uint256 _cap,
    	    uint8 _grantNumber
        ) public onlyRole(OWNER_ROLE) returns (uint256) {
    	    // check if grantStream with same grantNumber already exists
    	    uint256 existingGrantId;
    	    BuilderGrantData[] memory existingBuilderGrants = builderGrants[
    		    _builder
    	    ];
    	    for (uint i = 0; i < existingBuilderGrants.length; i++) {
    		    GrantStream memory existingGrant = grantStreams[
    			    existingBuilderGrants[i].grantId
    		    ];
    		    if (existingGrant.grantNumber == _grantNumber) {
    			    if (existingGrant.cap != existingGrant.amountLeft) {
    				    revert AlreadyWithdrawnFromGrant();
    			    }
    			    existingGrantId = existingBuilderGrants[i].grantId;
    			    break;
    		    }
    	    }
    
    	    // update existing grant or create new one
    	    uint256 grantId = existingGrantId != 0
    		    ? existingGrantId
    		    : nextGrantId++;
    
    	    grantStreams[grantId] = GrantStream({
    		    cap: _cap,
    		    last: block.timestamp,
    		    amountLeft: _cap,
    		    grantNumber: _grantNumber,
    		    stageNumber: 1,
    		    builder: _builder
    	    });
    
    	    if (existingGrantId == 0) {
    		    builderGrants[_builder].push(
    			    BuilderGrantData({
    				    grantId: grantId,
    				    grantNumber: _grantNumber
    			    })
    		    );
    		    emit AddGrant(grantId, _builder, _cap);
    	    } else {
    		    emit ReinitializeGrant(grantId, _builder, _cap);
    	    }
    	    return grantId;
        }
    
        function moveGrantToNextStage(
    	    uint256 _grantId,
    	    uint256 _cap
        ) public onlyRole(OWNER_ROLE) {
    	    GrantStream storage grantStream = grantStreams[_grantId];
    	    if (grantStream.cap == 0) revert NoActiveStream();
    
    	    // If amountLeft equals cap, reinitialize with same stage number
    	    if (grantStream.amountLeft == grantStream.cap) {
    		    grantStream.cap = _cap;
    		    grantStream.last = block.timestamp;
    		    grantStream.amountLeft = _cap;
    		    // Stage number remains the same
    		    emit ReinitializeNextStage(
    			    _grantId,
    			    grantStream.builder,
    			    _cap,
    			    grantStream.grantNumber,
    			    grantStream.stageNumber
    		    );
    	    } else {
    		    if (grantStream.amountLeft > DUST_THRESHOLD)
    			    revert PreviousAmountNotFullyWithdrawn();
    
    		    if (grantStream.amountLeft > 0) {
    			    (bool sent, ) = payable(grantStream.builder).call{
    				    value: grantStream.amountLeft
    			    }("");
    			    if (!sent) revert FailedToSendEther();
    		    }
    
    		    grantStream.cap = _cap;
    		    grantStream.last = block.timestamp;
    		    grantStream.amountLeft = _cap;
    		    grantStream.stageNumber += 1;
    
    		    emit MoveGrantToNextStage(
    			    _grantId,
    			    grantStream.builder,
    			    _cap,
    			    grantStream.grantNumber,
    			    grantStream.stageNumber
    		    );
    	    }
        }
    
        function updateGrant(
    	    uint256 _grantId,
    	    uint256 _cap,
    	    uint256 _last,
    	    uint256 _amountLeft,
    	    uint8 _stageNumber
        ) public onlyRole(OWNER_ROLE) {
    	    GrantStream storage grantStream = grantStreams[_grantId];
    	    if (grantStream.cap == 0) revert NoActiveStream();
    	    grantStream.cap = _cap;
    	    grantStream.last = _last;
    	    grantStream.amountLeft = _amountLeft;
    	    grantStream.stageNumber = _stageNumber;
    
    	    emit UpdateGrant(
    		    _grantId,
    		    grantStream.builder,
    		    _cap,
    		    grantStream.last,
    		    grantStream.amountLeft,
    		    grantStream.grantNumber,
    		    grantStream.stageNumber
    	    );
        }
    
        function streamWithdraw(
    	    uint256 _grantId,
    	    uint256 _amount,
    	    string memory _reason
        ) public {
    	    if (address(this).balance < _amount) revert InsufficientContractFunds();
    	    GrantStream storage grantStream = grantStreams[_grantId];
    	    if (grantStream.cap == 0) revert NoActiveStream();
    	    if (msg.sender != grantStream.builder) revert UnauthorizedWithdrawal();
    
    	    uint256 totalAmountCanWithdraw = unlockedGrantAmount(_grantId);
    	    if (totalAmountCanWithdraw < _amount) revert InsufficientStreamFunds();
    
    	    uint256 elapsedTime = block.timestamp - grantStream.last;
    	    uint256 timeToDeduct = (elapsedTime * _amount) / totalAmountCanWithdraw;
    
    	    grantStream.last = grantStream.last + timeToDeduct;
    	    grantStream.amountLeft -= _amount;
    
    	    (bool sent, ) = msg.sender.call{ value: _amount }("");
    	    if (!sent) revert FailedToSendEther();
    
    	    emit Withdraw(
    		    msg.sender,
    		    _amount,
    		    _reason,
    		    _grantId,
    		    grantStream.grantNumber,
    		    grantStream.stageNumber
    	    );
        }
    
        function getBuilderGrantCount(
    	    address _builder
        ) public view returns (uint256) {
    	    return builderGrants[_builder].length;
        }
    
        function addOwner(address newOwner) public onlyRole(OWNER_ROLE) {
    	    grantRole(OWNER_ROLE, newOwner);
    	    emit AddOwner(newOwner, msg.sender);
        }
    
        function removeOwner(address owner) public onlyRole(OWNER_ROLE) {
    	    revokeRole(OWNER_ROLE, owner);
    	    emit RemoveOwner(owner, msg.sender);
        }
    
        function getGrantIdByBuilderAndGrantNumber(
    	    address _builder,
    	    uint8 _grantNumber
        ) public view returns (uint256) {
    	    for (uint256 i = 0; i < builderGrants[_builder].length; i++) {
    		    if (builderGrants[_builder][i].grantNumber == _grantNumber) {
    			    return builderGrants[_builder][i].grantId;
    		    }
    	    }
    	    return 0;
        }
    
        receive() external payable {}
    
        fallback() external payable {}
    }
    
  • Update export const MINIMAL_VOTES_FOR_FINAL_APPROVAL = 1;
  1. scaffold.config.ts => chains.hardhat

Copy link

vercel bot commented Jan 7, 2025

The latest updates on your projects. Learn more about Vercel for Git ↗︎

Name Status Preview Comments Updated (UTC)
ens-pg ✅ Ready (Inspect) Visit Preview 💬 Add feedback Jan 8, 2025 5:38am

@Pabl0cks Pabl0cks marked this pull request as ready for review January 8, 2025 00:36
Copy link
Member

@technophile-04 technophile-04 left a comment

Choose a reason for hiding this comment

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

Tysm @Pabl0cks!! Just pushed couple of commits:

  1. 4981004: removing requestedFunds field since we don't show that on frontend too.
  2. cfcf8c0: removed unnecessary map, I think we didn't need it coz it looks like we did that only for type assertion but without it also types are working nicely working

@technophile-04 technophile-04 merged commit 3666201 into main Jan 8, 2025
3 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants