Medium
High
likeUser() never credits deposited ETH, so matched funds remain locked in LikeRegistry.
The expected behavior is that each paid like is credited to the sender so that, upon a mutual match, both users' deposited ETH can be combined, fee-adjusted, andforwarded to the newly deployed multisig wallet.
Instead, likeUser() records the like but never increments userBalances[msg.sender], so matchRewards() reads 0 for both matched users, deploys the multisig with no rewards, and leaves the actual ETH trapped inside LikeRegistry.
Likelihood: High
The issue occurs in the normal protocol flow and requires no special conditions beyond two users mutually liking each other with the required payment.
Impact: Medium
Users' paid likes are never forwarded to the match wallet and remain locked in LikeRegistry, breaking the protocol's core ETH distribution flow.
The following test shows that likeUser() accepts ETH from both users and records the mutual match, but never credits those payments into userBalances. As a result, matchRewards() deploys the multisig wallet with 0 ETH, while the full 2 ether paid by both users remains trapped in LikeRegistry.
Credit each user's payment when the like is submitted so that matchRewards() can account for deposited ETH correctly andforward the expected amount to the match wallet.
## Description User A calls `likeUser` and sends `value > 1` ETH. According to the design of DatingDapp, the amount for user A should be accumulated by `userBalances`. Otherwise, in the subsequent calculations, the balance for each user will be 0. ## Vulnerability Details When User A calls `likeUser`, the accumulation of `userBalances` is not performed. ```solidity function likeUser( address liked ) external payable { require(msg.value >= 1 ether, "Must send at least 1 ETH"); require(!likes[msg.sender][liked], "Already liked"); require(msg.sender != liked, "Cannot like yourself"); require(profileNFT.profileToToken(msg.sender) != 0, "Must have a profile NFT"); require(profileNFT.profileToToken(liked) != 0, "Liked user must have a profile NFT"); likes[msg.sender][liked] = true; emit Liked(msg.sender, liked); // Check if mutual like if (likes[liked][msg.sender]) { matches[msg.sender].push(liked); matches[liked].push(msg.sender); emit Matched(msg.sender, liked); matchRewards(liked, msg.sender); } } ``` This will result in `totalRewards` always being 0, affecting all subsequent calculations: ```solidity uint256 totalRewards = matchUserOne + matchUserTwo; uint256 matchingFees = (totalRewards * FIXEDFEE ) / 100; uint256 rewards = totalRewards - matchingFees; totalFees += matchingFees; ``` ## POC ```solidity function testUserBalanceshouldIncreaseAfterLike() public { vm.prank(user1); likeRegistry.likeUser{value: 20 ether}(user2); assertEq(likeRegistry.userBalances(user1), 20 ether, "User1 balance should be 20 ether"); } ``` Then we will get an error: ```shell [FAIL: User1 balance should be 20 ether: 0 != 20000000000000000000] ``` ## Impact - Users will be unable to receive rewards. - The contract owner will also be unable to withdraw ETH from the contract. ## Recommendations Add processing for `userBalances` in the `likeUser` function: ```diff function likeUser( address liked ) external payable { require(msg.value >= 1 ether, "Must send at least 1 ETH"); require(!likes[msg.sender][liked], "Already liked"); require(msg.sender != liked, "Cannot like yourself"); require(profileNFT.profileToToken(msg.sender) != 0, "Must have a profile NFT"); require(profileNFT.profileToToken(liked) != 0, "Liked user must have a profile NFT"); likes[msg.sender][liked] = true; + userBalances[msg.sender] += msg.value; emit Liked(msg.sender, liked); [...] } ```
The contest is live. Earn rewards by submitting a finding.
Submissions are being reviewed by our AI judge. Results will be available in a few minutes.
View all submissionsThe contest is complete and the rewards are being distributed.