selectWinner() pushes ETH directly to the winner's address.
If the winner is a contract without a receive() or fallback() function, the call reverts, and because players[] and raffleStartTime are only reset after the send, the raffle is permanently stuck.
Likelihood:
Any smart contract entered as a player is a candidate it need not be malicious, just non-payable
A malicious actor deliberately enters with a reverting contract to grief the protocol
Impact:
selectWinner() cannot complete; the raffle is permanently frozen
All entrance fees are locked in the contract with no recovery path
A contract with no receive() enters the raffle; if selected as a winner, every subsequent selectWinner() call reverts on the ETH transfer, permanently freezing the raffle.
Use a pull-payment pattern so the winner claims their prize rather than having it pushed.
## Description If a player submits a smart contract as a player, and if it doesn't implement the `receive()` or `fallback()` function, the call use to send the funds to the winner will fail to execute, compromising the functionality of the protocol. ## Vulnerability Details The vulnerability comes from the way that are programmed smart contracts, if the smart contract doesn't implement a `receive() payable` or `fallback() payable` functions, it is not possible to send ether to the program. ## Impact High - Medium: The protocol won't be able to select a winner but players will be able to withdraw funds with the `refund()` function ## Recommendations Restrict access to the raffle to only EOAs (Externally Owned Accounts), by checking if the passed address in enterRaffle is a smart contract, if it is we revert the transaction. We can easily implement this check into the function because of the Adress library from OppenZeppelin. I'll add this replace `enterRaffle()` with these lines of code: ```solidity function enterRaffle(address[] memory newPlayers) public payable { require(msg.value == entranceFee * newPlayers.length, "PuppyRaffle: Must send enough to enter raffle"); for (uint256 i = 0; i < newPlayers.length; i++) { require(Address.isContract(newPlayers[i]) == false, "The players need to be EOAs"); players.push(newPlayers[i]); } // Check for duplicates for (uint256 i = 0; i < players.length - 1; i++) { for (uint256 j = i + 1; j < players.length; j++) { require(players[i] != players[j], "PuppyRaffle: Duplicate player"); } } emit RaffleEnter(newPlayers); } ```
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.