Snowman Merkle Airdrop

AI First Flight #10
Beginner FriendlyFoundrySolidityNFT
EXP
View results
Submission Details
Severity: low
Valid

Snow::earnSnow - Global s_earnTimer shared across all users allows any buySnow() call to reset the weekly cooldown for every user, permanently blocking free claims

Root + Impact

Description

  • Snow::earnSnow() is designed to allow each individual user to claim one free Snow token once per week.

  • s_earnTimer is a single contract-level global variable rather than a per-user mapping. Every call to buySnow() unconditionally overwrites s_earnTimer with block.timestamp, resetting the one-week cooldown window for every user simultaneously.

function buySnow(uint256 amount) external payable canFarmSnow {
// ...mint logic...
// @> Global timer overwritten on every purchase — affects all users
s_earnTimer = block.timestamp;
}
function earnSnow() external canFarmSnow {
// @> Reads global timer, not a per-caller cooldown
if (s_earnTimer != 0 && block.timestamp < (s_earnTimer + 1 weeks)) {
revert S__Timer();
}
_mint(msg.sender, 1);
s_earnTimer = block.timestamp;
}

Risk

Likelihood:

  • Any call to buySnow() — even purchasing a single token — triggers the global reset

  • A malicious actor can spend the cost of 1 Snow token every 6 days to permanently prevent all users from ever calling earnSnow() successfully

Impact:

  • The "free weekly Snow" guarantee advertised by the protocol is rendered inaccessible whenever any purchase activity occurs

  • All users are forced to pay to acquire Snow tokens, fundamentally breaking a core protocol promise

Proof of Concept

The following Foundry test demonstrates the denial-of-service. A buyer purchases one
Snow token, resetting the global timer. Alice then attempts to call earnSnow()
immediately after and is blocked. The test further shows that repeated purchases
every six days can keep the timer perpetually reset, locking all users out of free
claims for the entire farming duration.

function test_buySnowResetsEarnTimerForAllUsers() public {
address alice = makeAddr("alice");
address buyer = makeAddr("buyer");
// buyer purchases 1 Snow — resets the global earnSnow timer
deal(address(weth), buyer, snow.s_buyFee());
vm.startPrank(buyer);
weth.approve(address(snow), snow.s_buyFee());
snow.buySnow(1);
vm.stopPrank();
// alice attempts to earn Snow immediately after — reverts
vm.prank(alice);
vm.expectRevert(Snow.S__Timer.selector);
snow.earnSnow();
// Even after 6 days, another purchase blocks alice for another week
vm.warp(block.timestamp + 6 days);
deal(address(weth), buyer, snow.s_buyFee());
vm.startPrank(buyer);
weth.approve(address(snow), snow.s_buyFee());
snow.buySnow(1);
vm.stopPrank();
vm.prank(alice);
vm.expectRevert(Snow.S__Timer.selector);
snow.earnSnow(); // still blocked
}

Recommended Mitigation

The fix replaces the single global uint256 with a per-address mapping, giving each
caller an independent cooldown window. buySnow() no longer writes to the timer at
all, since purchasing tokens should not affect any user's ability to earn free Snow.

- uint256 private s_earnTimer;
+ mapping(address => uint256) private s_earnTimer;
function buySnow(uint256 amount) external payable canFarmSnow {
// ...mint logic...
- s_earnTimer = block.timestamp;
// buySnow should not affect the earnSnow cooldown
}
function earnSnow() external canFarmSnow {
- if (s_earnTimer != 0 && block.timestamp < (s_earnTimer + 1 weeks)) {
+ if (s_earnTimer[msg.sender] != 0 && block.timestamp < (s_earnTimer[msg.sender] + 1 weeks)) {
revert S__Timer();
}
_mint(msg.sender, 1);
- s_earnTimer = block.timestamp;
+ s_earnTimer[msg.sender] = block.timestamp;
}
Updates

Lead Judging Commences

ai-first-flight-judge Lead Judge about 20 hours ago
Submission Judgement Published
Validated
Assigned finding tags:

[L-02] Global Timer Reset in Snow::buySnow Denies Free Claims for All Users

## Description: The `Snow::buySnow` function contains a critical flaw where it resets a global timer `(s_earnTimer)` to the current block timestamp on every invocation. This timer controls eligibility for free token claims via `Snow::earnSnow()`, which requires 1 week to pass since the last timer reset. As a result: Any token purchase `(via buySnow)` blocks all free claims for all users for 7 days Malicious actors can permanently suppress free claims with micro-transactions Contradicts protocol documentation promising **"free weekly claims per user"** ## Impact: * **Complete Denial-of-Service:** Free claim mechanism becomes unusable * **Broken Protocol Incentives:** Undermines core user acquisition strategy * **Economic Damage:** Eliminates promised free distribution channel * **Reputation Harm:** Users perceive protocol as dishonest ```solidity function buySnow(uint256 amount) external payable canFarmSnow { if (msg.value == (s_buyFee * amount)) { _mint(msg.sender, amount); } else { i_weth.safeTransferFrom(msg.sender, address(this), (s_buyFee * amount)); _mint(msg.sender, amount); } @> s_earnTimer = block.timestamp; emit SnowBought(msg.sender, amount); } ``` ## Risk **Likelihood**: • Triggered by normal protocol usage (any purchase) • Requires only one transaction every 7 days to maintain blockage • Incentivized attack (low-cost disruption) **Impact**: • Permanent suppression of core protocol feature • Loss of user trust and adoption • Violates documented tokenomics ## Proof of Concept **Attack Scenario:** Permanent Free Claim Suppression * Attacker calls **buySnow(1)** with minimum payment * **s\_earnTimer** sets to current timestamp (T0) * All **earnSnow()** calls revert for **next 7 days** * On day 6, attacker repeats **buySnow(1)** * New timer reset (T1 = T0+6 days) * Free claims blocked until **T1+7 days (total 13 days)** * Repeat step **4 every 6 days → permanent blockage** **Test Case:** ```solidity // Day 0: Deploy contract snow = new Snow(...); // s_earnTimer = 0 // UserA claims successfully snow.earnSnow(); // Success (first claim always allowed) // Day 1: UserB buys 1 token snow.buySnow(1); // Resets global timer to day 1 // Day 2: UserA attempts claim snow.earnSnow(); // Reverts! Requires day 1+7 = day 8 // Day 7: UserC buys 1 token (day 7 < day 1+7) snow.buySnow(1); // Resets timer to day 7 // Day 8: UserA retries snow.earnSnow(); // Still reverts! Now requires day 7+7 = day 14 ``` ## Recommended Mitigation **Step 1:** Remove Global Timer Reset from `buySnow` ```diff function buySnow(uint256 amount) external payable canFarmSnow { // ... existing payment logic ... - s_earnTimer = block.timestamp; emit SnowBought(msg.sender, amount); } ``` **Step 2:** Implement Per-User Timer in `earnSnow` ```solidity // Add new state variable mapping(address => uint256) private s_lastClaimTime; function earnSnow() external canFarmSnow { // Check per-user timer instead of global if (s_lastClaimTime[msg.sender] != 0 && block.timestamp < s_lastClaimTime[msg.sender] + 1 weeks ) { revert S__Timer(); } _mint(msg.sender, 1); s_lastClaimTime[msg.sender] = block.timestamp; // Update user-specific timer emit SnowEarned(msg.sender, 1); // Add missing event } ``` **Step 3:** Initialize First Claim (Constructor) ```solidity constructor(...) { // Initialize with current timestamp to prevent immediate claims s_lastClaimTime[address(0)] = block.timestamp; } ```

Support

FAQs

Can't find an answer? Chat with us on Discord, Twitter or Linkedin.

Give us feedback!