Skip to content
This repository has been archived by the owner on Jun 11, 2023. It is now read-only.

chaduke - noCircularLinkage() might fail to detect circles in the tree #28

Closed
sherlock-admin opened this issue Mar 9, 2023 · 7 comments
Labels
Disagree With (non-)Duplication The sponsor disputed the duplication state of this issue Escalation Resolved This issue's escalations have been approved/rejected Hats.sol Non-Reward This issue will not receive a payout

Comments

@sherlock-admin
Copy link
Contributor

sherlock-admin commented Mar 9, 2023

chaduke

medium

noCircularLinkage() might fail to detect circles in the tree

Summary

noCircularLinkage(A, B) can detect circles that A is involved, but fail to detect the presence of circles of the whole tree in which A is not involved. As a result, when there is a circle in other branches of the tree, the function will get into infinite recursion and revert due to out of gas. The function will fail to detect when there is a circle in another branch that A is not part of.

Vulnerability Detail

The main issue of noCircularLinkage(A, B) is that it assumes: if there is a circle in the tree, then A must be involved. However, circles might exist in various branches, not just in the branch of A.

https://github.com/Hats-Protocol/hats-protocol/blob/fafcfdf046c0369c1f9e077eacd94a328f9d7af0/src/HatsIdUtilities.sol#L194-L200

In the following code POC, we show that five tophats 1, 2, 3, 4, 5, when there is a cycle 2->3->4->5->3, and 1 is not involved, then noCircularLinkage(1, 2) will fail to detect the circle and revert due to infinite recursion.

 function testCircularLinks() public {
        uint32 TopHat1 = 1;
        uint32 TopHat2 = 2;
        uint32 TopHat3 = 3;
        uint32 TopHat4 = 4;
        uint32 TopHat5 = 5;
  
        uint id1 = utils.buildHatId(1 << 224, 1);
        uint id2 = utils.buildHatId(2 << 224, 2);
        uint id3 = utils.buildHatId(3 << 224, 3);
        uint id4 = utils.buildHatId(4 << 224, 4);
        uint id5 = utils.buildHatId(5 << 224, 5);
        assertFalse(utils.isTopHat(id1));
  
        utils.linkTree(TopHat1, id2);
        utils.linkTree(TopHat2, id3);
        utils.linkTree(TopHat3, id4);
        utils.linkTree(TopHat4, id5);
        utils.linkTree(TopHat5, id3);
  
        vm.expectRevert();  
        utils.noCircularLinkage(TopHat1, id2);
    }

Impact

noCircularLinkage() might fail to detect circles in the tree.

This might create circular management and confuse the power hierarchy in an organization. In addition, many functions (those that use recursions) will revert due to out of gas since they depend on the the property that there is no circle in the tree. For example, getHatLevel() will revert when there is a circle in the tree.

Code Snippet

See above

Tool used

VSCode

Manual Review

Recommendation

  • record each tophat domains that have been visited, when a tophat domain is visited again, then we detect the presence of a circle.
@github-actions github-actions bot added Medium A valid Medium severity issue Duplicate A valid issue that is a duplicate of an issue with `Has Duplicates` label labels Mar 12, 2023
@spengrah spengrah added the Disagree With (non-)Duplication The sponsor disputed the duplication state of this issue label Mar 13, 2023
@spengrah
Copy link

spengrah commented Mar 13, 2023

This is not a duplicate of #96 since it deals with circular admins rather than a DOS/stack-too-deep issue stemming from recursion.

@spengrah
Copy link

spengrah commented Mar 13, 2023

I'm still thinking this through, but I think that we can rely on the fact that if there is an undetectable circle then there will be infinite recursion, which (as noted) will result in a revert. While not as clean as bubbling up the CircularLinkage error, that should ensure that no circular links can be created.

TODO:

  • check the hypothesis that undetected circles will result in reverts, ie because of infinite recursion. cc @zobront
  • add note in natspec/docs that noCircularLinkage will revert if there is an undetectable circle (and that this is a good thing)

@spengrah
Copy link

Upon further inspection, I don't think this is an issue because it would not be possible to enter into this scenario in the first place. In the poc test above, utils.linkTree(TopHat5, id3) would fail because it itself is creating a circle, so we would never get to the purported issue.

@sherlock-admin sherlock-admin added the Reward A payout will be made for this issue label Mar 29, 2023
@cducrest
Copy link

Escalate for 10 USDC

This scenario cannot happen, because noCircularLinkage() is called by _linkTopHatToTree every time a link is created. Even though it only checks for circular linkage involving the newly created link, it is sufficient as it will be run for every link created. It cannot be that the tree has a circular linkage in another branch because noCircularLinkage() would have been called when that other branch was been created.

@sherlock-admin
Copy link
Contributor Author

Escalate for 10 USDC

This scenario cannot happen, because noCircularLinkage() is called by _linkTopHatToTree every time a link is created. Even though it only checks for circular linkage involving the newly created link, it is sufficient as it will be run for every link created. It cannot be that the tree has a circular linkage in another branch because noCircularLinkage() would have been called when that other branch was been created.

You've created a valid escalation for 10 USDC!

To remove the escalation from consideration: Delete your comment.

You may delete or edit your escalation comment anytime before the 48-hour escalation window closes. After that, the escalation becomes final.

@sherlock-admin sherlock-admin added the Escalated This issue contains a pending escalation label Mar 31, 2023
@hrishibhat
Copy link
Contributor

Escalation accepted

Not a valid issue
Not a valid duplicate of #96
And the above-mentioned scenario with noCircularLinkage is not possible as shown in the comments from both Sponsor and Escalation

@sherlock-admin
Copy link
Contributor Author

Escalation accepted

Not a valid issue
Not a valid duplicate of #96
And the above-mentioned scenario with noCircularLinkage is not possible as shown in the comments from both Sponsor and Escalation

This issue's escalations have been accepted!

Contestants' payouts and scores will be updated according to the changes made on this issue.

@sherlock-admin sherlock-admin added Escalation Resolved This issue's escalations have been approved/rejected Non-Reward This issue will not receive a payout and removed Escalated This issue contains a pending escalation Medium A valid Medium severity issue Duplicate A valid issue that is a duplicate of an issue with `Has Duplicates` label Reward A payout will be made for this issue labels Apr 2, 2023
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
Disagree With (non-)Duplication The sponsor disputed the duplication state of this issue Escalation Resolved This issue's escalations have been approved/rejected Hats.sol Non-Reward This issue will not receive a payout
Projects
None yet
Development

No branches or pull requests

4 participants