Skip to content

Latest commit

 

History

History
270 lines (211 loc) · 23.7 KB

README.md

File metadata and controls

270 lines (211 loc) · 23.7 KB

EthernautChallenges

The following are the solutions to the Ethernaut levels. For a detailed walkthrough please have a look at the videos.

As mentioned in the Solidity documentation : "When deploying contracts, you should use the latest released version of Solidity. This is because breaking changes as well as new features and bug fixes are introduced regularly."

Here are some useful links:

All the comments I made are subject to my own interpretation of how things work. Please feel free to contact me if something is not clear to you or needs to be corrected.

Solution:

Level 1 Fallback:

Solidity documentation release 0.6.4 :
"A contract can have at most one fallback function, declared using fallback () external [payable] (without the function keyword). This function cannot have arguments, cannot return anything and must have external visibility. It is executed on a call to the contract if none of the other functions match the given function signature, or if no data was supplied at all and there is no receive Ether function. The fallback function always receives data, but in order to also receive Ether it must be marked payable." p. 99

"msg.value (uint): number of wei sent with the message" p. 73

In the case of this contract, in order to execute the fallback function we need to pass the require condition:
require (msg.value > 0 && contributions[msg.sender] > 0)
Therefore, before calling the fallback function with an amount attached to it, we also need to increase our contributions (using the contribute function). Once that’s done we will become the new owner of the contract. We can then use the withdraw function to reduce its balance to 0.

We will solve this challenge using the following command in the console (can be access by clicking F12):
await contract.owner()
player
await contract.contributions(player)
await contract.contribute({value:1})
await contract.contributions(player)
await contract.sendTransaction({value:1})
await contract.owner()
player
await contract.withdraw()

Level 2 Fallout:

Solidity documentation release 0.6.4 :
"A constructor is an optional function declared with the constructor keyword which is executed upon contract creation, and where you can run contract initialisation code." p. 110
“Prior to version 0.4.22, constructors were defined as functions with the same name as the contract. This syntax was deprecated and is not allowed anymore in version 0.5.0.” p. 110

In this contract, the constructor syntax is deprecated and misspelled (fal1out written with the number 1 instead of the letter l). Therefore, to claim ownership of this contract you just need to call the fal1out function.

Level 3 Token:

Solidity documentation release 0.6.4 :
“A blockchain is a globally shared, transactional database. This means that everyone can read entries in the database just by participating in the network” p. 10
“Everything you use in a smart contract is publicly visible, even local variables and state variables marked private. Using random numbers in smart contracts is quite tricky if you do not want miners to be able to cheat.” p. 153

Ethereum Yellow paper:
“Providing random numbers within a deterministic system is, naturally, an impossible task. However, we can approximate with pseudo-random numbers by utilising data which is generally unknowable at the time of transacting. Such data might include the block’s hash, the block’s timestamp and the block’s beneficiary address”.

In this case, the block number is knowable at the time of transacting. Thus, we can create a malicious contract (Level3_CoinFlipSolution.sol) that computes the right guess and use this value to call the flip function of the CoinFlip contract (before a new block gets mined). Therefore, we are able to guess the right outcome everytime.

Level 4 Telephone:

Solidity documentation release 0.6.4 :
“msg.sender (address payable): sender of the message (current call)” p. 73
“tx.origin (address payable): sender of the transaction (full call chain)” p. 73

In other words, tx.origin is the original address that sends a transaction while msg.sender is the current (i.e. last, closest) sender of a message. For instance, assume user/contract A calls contract B which triggers it to call contract C which triggers it to call contract D, we have the following:

tel_graph2

To solve this level, we (the user) will call the function of a malicious contract (Level4_TelephoneSolution.sol) that will call the changeOwner function of the Telephone contract. Thus, for the Telephone contract: tx.origin (= user’s address) msg.sender (= malicious contract’s address). This will allow us to pass the if statement and become the new owner of the contract.

Level 5 Token:

Solidity documentation release 0.6.4 :
“As in many programming languages, Solidity’s integer types are not actually integers. They resemble integers when the values are small, but behave differently if the numbers are larger. For example, the following is true: uint8(255)+ uint8(1) == 0. This situation is called an overflow. It occurs when an operation is performed that requires a fixed size variable to store a number (or piece of data) that is outside the range of the variable’s data type. An underflow is the converse situation:uint8(0) - uint8(1) == 255” p. 156

As suggested by the level, it’s similar to how an odometer (instrument measuring the distance traveled by a vehicle) works:

Explanation2

To solve this level we will perform an underflow by using the transfer function with the following two inputs: another address (than the one we are currently using) and a number bigger than 20 (= amount of tokens given). We used the following command in the console:
await contract.balanceOf(player)
await contract.transfer("0x6E0B06770144b7b5923f3d759C19E1938Fe67807", 21)
await contract.balanceOf(player)

Level 6 Delegation:

Solidity documentation release 0.6.4 :
“There exists a special variant of a message call, named delegatecall which is identical to a message call apart from the fact that the code at the target address is executed in the context of the calling contract and msg.sender and msg.value do not change their values. This means that a contract can dynamically load code from a different address at runtime. Storage, current address and balance still refer to the calling contract, only the code is taken from the called address. This makes it possible to implement the “library” feature in Solidity: Reusable library code that can be applied to a contract’s storage, e.g. in order to implement a complex data structure” p. 13
“The first four bytes of the call data for a function call specifies the function to be called. It is the first (left, high-order in big-endian) four bytes of the Keccak-256 (SHA-3) hash of the signature of the function. The signature is defined as the canonical expression of the basic prototype without data location specifier” p. 179
“Any interaction with another contract imposes a potential danger, especially if the source code of the contract
is not known in advance.”
p. 78

In other words, by using a delegatecall you let another contract’s code run inside the calling contract. This code is executed using the calling contract state (i.e. data, variables) and can potentially modify it. It’s a double-edged sword. Here is an example:

Explanation2

To solve this level we will use the pwn function in the context of the Delegation contract by using a delegatecall (located in its fallback function). In order to precisely call the pwn function, we need to pass its function signature (i.e. first four bytes of the Keccak-256 hash).

Level 7 Force:

Solidity documentation release 0.6.4 :
“A contract without a receive Ether function can receive Ether as a recipient of a coinbase transaction (aka miner block reward) or as a destination of a selfdestruct. A contract cannot react to such Ether transfers and thus also cannot reject them. This is a design choice of the EVM and Solidity cannot work around it.” p. 99

To solve this level we will deploy a malicious contract (Level7_Force.sol) and send some fund to it. Then, we will designate the Force contract as owner of the malicious contract and destroy our malicious contract. Thus, sending fund to the Force contract that cannot be rejected.

Level 8 Vault:

Solidity documentation release 0.6.4 :
“Everything that is inside a contract is visible to all observers external to the blockchain. Making something private only prevents other contracts from reading or modifying the information, but it will still be visible to the whole world outside of the blockchain.” p. 90
“Even if a contract is removed by “selfdestruct”, it is still part of the history of the blockchain and probably retained by most Ethereum nodes. So, using “selfdestruct” is not the same as deleting data from a hard disk.” p. 14
“Statically-sized variables (everything except mapping and dynamically-sized array types) are laid out contiguously in storage starting from position 0. Multiple, contiguous items that need less than 32 bytes are packed into a single storage slot if possible [...] ”. p. 123

Everything you use in a smart contract is publicly visible. Moreover, keep in mind that a blockchain is an append-only ledger. If you change the state of your contract or even destroy it, it will still be part of the history of the blockchain. Thus, everyone can have access to it.

To solve this level we will unlock the vault by using the function unlock with the value of password as argument. To get the value of password we need to access the state and get the value stored at slot 1 (slot 0 contains the bool value).

Level 9 King:

Solidity documentation release 0.6.4 :
“The transfer function fails if the balance of the current contract is not large enough or if the Ether transfer is rejected by the receiving account. The transfer function reverts on failure. Note: If x is a contract address, its code (more specifically: its Receive Ether Function, if present, or otherwise its Fallback Function, if present) will be executed together with the transfer call (this is a feature of the EVM and cannot be prevented). If that execution runs out of gas or fails in any way, the Ether transfer will be reverted and the current contract will stop with an exception.” p. 49-50
“Any interaction with another contract imposes a potential danger, especially if the source code of the contract
is not known in advance.”
p. 78

In order to solve this level we first need to become the new King by sending an amount >= 1 Ether. Then, we must prevent others of dethroning us by forcing the transfer function to revert. This can be implemented in several ways. We will create a malicious contract (Level9_King.sol) with a fallback function that will revert anytime it’s called.

Level 10 Re-entrancy:

Solidity documentation release 0.6.4 :
“You should avoid using .call() whenever possible when executing another contract function as it bypasses type checking, function existence check, and argument packing.” p. 76
“Any interaction with another contract imposes a potential danger, especially if the source code of the contract
is not known in advance. The current contract hands over control to the called contract and that may potentially do just about anything.”
p. 78

In order to solve this level we will create a malicious contract with a fallback function that calls back the withdraw function. Thus, this will prevent the withdraw function completion until all the contract funds are drained (as shown below). Before calling the withdraw function we need to increase the balance of our malicious contract (by using the donate function of the Reentrance contract).

reentrance

Level 11 Elevator:

Solidity documentation release 0.6.4 :
“Interfaces are similar to abstract contracts, but they cannot have any functions implemented.” p. 113
“All functions declared in interfaces are implicitly virtual, which means that they can be overridden. This does not automatically mean that an overriding function can be overridden again - this is only possible if the overriding function is marked virtual.” p. 114

To solve this level we will create a malicious contract that will implement the isLastFloor function. Then we will invoke the goTo function from the malicious contract. This will ensure that it’s the isLastFloor function from the malicious contract that will be used. The isLastFloor function needs to return false the first time it’s called (to pass the if statement) and true the second time it’s called (to change the boolean top value to true).

Level 12 Privacy:

Solidity documentation release 0.6.4 :
“Everything that is inside a contract is visible to all observers external to the blockchain. Making something private only prevents other contracts from reading or modifying the information, but it will still be visible to the whole world outside of the blockchain.” p. 90
“Statically-sized variables (everything except mapping and dynamically-sized array types) are laid out contiguously in storage starting from position 0. Multiple, contiguous items that need less than 32 bytes are packed into a single storage slot if possible [...]”. p. 123

This level is similar to level 8 Vault. Remember that all on-chain data are publicly visible (marking them private only makes them inaccessible to other contracts). Please have a look at Nicole Zhu’s walkthrough in order to gain more insight on how variables are stored. This article by Steve Marx is also very helpful. To unlock this contract, we need to use the unlock function with an input equal to bytes16(data[2]) which is the first 16 bytes stored at slot 5 (as seen below). Note that:

  • bytes32 takes the same amount of storage as uint256; (2^8)^32 = 2^256.
  • in the video I was able to input a bytes32 instead of a bytes16 as expected by the unlock function. This might be due to the contract's ABI that truncates the input.

storage

Level 13 Gatekeeper One :

To solve this level we need to call the enter function and pass the conditions of each function modifier.
To pass:

  • gateOne: we will create a malicious contract with a function letMeIn that calls the enter function of the Gatekeeper contract. By calling letMeIn in Remix-IDE tx.origin ≠ msg.sender (see solution of level 4 Telephone for more details).
  • gateTwo: to see the remaining gas we can use the functionality of Remix-IDE. The value we are looking for will be shown after the opcode GAS, i.e. in the opcode PUSH2 (see picture below). Knowing this value we could add the proper gas amount to our call in order to pass gateTwo. However, as mentioned by Spalladino, the proper gas offset to use will vary depending on the compiler. We will use his solution in our malicious contract and brute-force a range of possible gas values. To account for it, we will increase the gas limit in Remix-IDE.

RemainingGas2

To learn more about opcode, please have a look at this article from Alejandro Santander in collaboration with Leo Arias.
Solidity documentation release 0.6.5:
“If an integer is explicitly converted to a smaller type, higher-order bits are cut off:
uint32 a = 0x12345678;
uint16 b = uint16(a); // b will be 0x5678 now ” p. 71

  • gateThree:
    • 1st condition: the last 8 hex need to be equal to the last 4 hex -> only possible if we mask part of _gateKey with 0, so that: 0x0000???? = 0x????.
    • 2nd condition: is achieved if the rest of the key is different from 0 so that 0x0000???? ≠ 0x????????0000????
    • 3rd condition: 0x0000???? needs to be equal to the last 4 hex of tx.origin
      We will create a variable to store the key. One possible solution is to use the value of tx.origin and only mask part of it with 0 (as described in the 1st condition).

In summary, we will create a malicious contract in Remix-IDE that calls the enter function of the Gatekeeper contract (thus passing gateOne). We will append a gas value to our call that will vary in order to brute force gateTwo (using Spalladino’s solution). Finally, we will pass to our call a parameter made by masking part or the value of tx.origin (in order to pass gateThree).

Level 14 Gatekeeper Two :

Solidity documentation release 0.6.5:
“extcodesize(a): size of the code at address a” p. 209
“^ (bitwise exclusive or)” p. 46
Ethereum Yellow Paper (section 7, as suggested by this level):
“Note that while the initialisation code is executing, the newly created address exists but with no intrinsic body code (during initialization code execution, EXTCODESIZE on the address should return zero).”

To pass:

  • gateOne: as with Gatekeeper One, we will create a malicious contract with a function letMeIn that calls the enter function of the Gatekeeper contract. By calling letMeIn in Remix-IDE tx.origin ≠ msg.sender (see solution of level 4 Telephone for more details).
  • gateTwo: the size of the code at the caller address needs to be equal to 0. This can be achieved by placing the letMeIn function inside the constructor of the malicious contract.
  • gateThree: by modifying this equation: a^b = c -> a^(a^b) = a^c -> 0^b = a^c -> b = a^c
    Knowing a and c we can thus easily compute b.

In summary, we will create a malicious contract in Remix-IDE with a function letMeIn that will call the enter function of the Gatekeeper contract (thus passing gateOne). By placing the letMeIn function inside the constructor of our malicious contract, we will satisfy the requirement of gateTwo. Finally, by passing as parameter the result of uint64(bytes8(keccak256(abi.encodePacked(address(this))))) ^ (uint64(0) - 1) we will pass gateThree.

Level 15 Naught Coin :

Solidity documentation release 0.6.5:
“Solidity supports multiple inheritance including polymorphism. [...] Polymorphism means that a function call (internal and external) always executes the function of the same name (and parameter types) in the most derived contract in the inheritance hierarchy. This has to be explicitly enabled on each function in the hierarchy using the virtual and override keywords. [...] Use ’is’ to derive from another contract. Derived contracts can access all non-private members including internal functions and state variables” p.105 -106

The contract we are given inherits from the ERC20 contract and only overrides the transfer function. It turns out that other functions from the ERC20 contract are available to us. Thus, we can use some of them to bypass the constraints imposed by the overridden transfer function.
Note that:

  • before using the transferFrom function we need to increase our allowance using the increaseAllowance function
  • we will have to do some workaround due to some issues with web3.js when dealing with big numbers

Instructions used:
await contract.balanceOf(player)
(await contract.balanceOf(player)).toString()
await contract.increaseAllowance(player, "1000000000000000000000000")
await contract.transferFrom (player, "another_address" , "1000000000000000000000000") // replace another_address by another address -_-
(await contract.balanceOf(player)).toString()

Level 16 Preservation :

To pass this level you will have to use the knowledge you’ve acquired from Ethernaut level 6 and 12.
In particular how:

  • delegatecall works and
  • how statically-sized variables (everything except mapping and dynamically-sized array types) are laid out in storage

The function setFirstTime executes a delegatecall to the function setTime of LibraryContract. Namely, setTime will be executed in the context of the Preservation contract, i.e. it will modify slot n°0 (timeZone1Library) of the Preservation contract with the argument you provide. Therefore, you could change this address with the address of a malicious contract. Thus, when you call setFristTime again, the function setTime of your malicious contract will be executed in the context of the Preservation contract. By writing this function such that it modifies its slot n°2, you could in the context of the Preservation contract modify the owner variable (and claim ownership of this contract).

Level 17 Recovery :

Solidity documentation release 0.6.6:
“The address of an external account is determined from the public key while the address of a contract is determined at the time the contract is created (it is derived from the creator address and the number of transactions sent from that address, the so-called “nonce”).” p. 11
Ethereum yellow paper:
“The address of the new account is defined as being the rightmost 160 bits of the Keccak hash of the RLP encoding of the structure containing only the sender and the account nonce.”
“The account's nonce is initially defined as one [...].”

To solve this level you will have to recover the address of the SimpleToken contract. You can either compute this address* or use Etherscan (we will use both approaches). Once you know this address you can use it to execute the destroy function (thus removing the 0.5 ether and pass this level).

* to compute the address of the contract you need to know the creator’s address (in our case the instance address) and the nonce (in this case 1, as this contract’s creation is its first transaction). In the console use the command: web3.utils.soliditySha3("0xd6", "0x94", instance_address, “0x01”), the address will be the first 20 bytes (=160 bits). For more detail, please have a look at the RLP documentation or this post.