HTB Cyber Apocalypse CTF 2025 - Eldorion [blockchain]
Difficulty : Easy
Team : Phreaks 2600
Description : “Welcome to the realms of Eldoria, adventurer. You’ve found yourself trapped in this mysterious digital domain, and the only way to escape is by overcoming the trials laid before you. But your journey has barely begun, and already an overwhelming obstacle stands in your path. Before you can even reach the nearest city, seeking allies and information, you must face Eldorion, a colossal beast with terrifying regenerative powers. This creature, known for its ““eternal resilience”” guards the only passage forward. It’s clear: you must defeat Eldorion to continue your quest.”
Author : PerryThePwner
Topics : timestamp, transactions
TL;DR
-
write a contract that will trigger
attack()
multiple time in one function. -
get the flag
Introduction
For this challenge we have one main contract to study. Through this contract, we can carry out attacks
to reduce its health to 0.
Recon
Smart Contracts
Here are the smart contracts :
Setup.sol
|
|
Eldorion.sol
|
|
Code Reading
Setup.sol
This contract do nothing extravagant except deploying the target contract (Eldorion
) :
|
|
And check if we solved the challenge :
|
|
And it calls isDefeated
from the target contract.
Eldorion.sol
We will start directly by reading isDefeated
:
|
|
It checks if the health is equal to 0.
And coincidence, there’s an attack function above.
|
|
We can submit a quantity of damage to inflict. But there are two conditions :
- One attack can’t inflict damages above 100.
- The damage inflicted can’t be greater than the remaining health.
If satisfied, the health is subtracted by the damage inflicted.
But what I didn’t mentionned, is that a modifier is applied to the function, eternalResilience
.
A modifier allow us to, modify the execution of a function :
|
|
It checks if the current block timestamp is greater than lastAttackTimestamp
, if yes, the health is set back to MAX_HEALTH
(300), and lastAttackTimestamp
set to the current block timestamp.
The underscore _;
, states to continue the flow of the function that called this modifier.
So what we can deduce is that each time we attack, the health is put back to the max if the current block timestamp is greater than the last attack timestamp.
And that’s pretty all.
Solving
So to solve, we know we have to put the health to zero.
BUT,
We can’t inflict more than 100 damage, and every attacks, if the block timestamp change, the health return to the max health.
To solve we need to have in mind two concepts which are transactions and blocks.
From the ethereum documentation about transactions :
Transactions are cryptographically signed instructions from accounts. An account will initiate a transaction to update the state of the Ethereum network. The simplest transaction is transferring ETH from one account to another.
A transaction can also be a call to a function, like attack()
.
And for the blocks :
Blocks are batches of transactions with a hash of the previous block in the chain.
Also each block have other properties :
- block number
- block timestamp
- block hash
- …
In shorts a block can contains multiple transaction in it that will share the same block properties, thus we can manage to have multiple call of attack()
in one block so the timestamp stay the same.
To do that, we can write a contract with a function that will call three times attack()
with a value of 100 :
Attack.sol
|
|
We can deploy the contract with forge from foundry-rs :
|
|
And trigger the attack :
|
|
HTB{w0w_tr1pl3_hit_c0mbo_ggs_y0u_d...}
.