Description : You stand victorious, panting, over the fallen form of Eldorion. The beast’s eternal resilience proved no match for your cunning and skill, adventurer. The path to the city gates of Eldoria now lies open, but the journey is far from over. As you approach, a shimmering structure catches your eye: the HeliosDEX, a decentralized exchange powered by the radiant energy of Helios himself. Whispers tell of travelers using this exchange to amass fortunes, stocking up on rare items and crucial supplies before braving the perils of Eldoria. Perhaps you can use this opportunity to your advantage…
When converting ethers to MalakarEssence (MAL) or HeliosLuminaShards (HLS) token, it is rounded up to the next integer, even if we convert 0.000000000000000001 ether it gives, it is rounded up to 1 ETHER
Repeat the operation until we have enough token to withdraw.
Give enough allowance to the DEX, refund the tokens and get the ethers.
Introduction
For this challenge, we are dealing with three ERC20 contracts and a decentrized exchange (DEX) with which we can swap ethers for ERC20 tokens.
Before diving in the code, we must know what a DEX and ERC20 are.
Definitions
DEX in a nutshell
To sum up, a DEX is a platform with smart contracts working on the backend that provides an automated way to swap a token for one another. (example : ETH for ELD, the token created for this challenge).
.
So we can transfer ethers and get a certain amount of tokens, paying micro fees for each transaction.
ERC20 in a nutshell
ERC20 is a standard defining “Fongible Token” (FT) in the ethereum network.
Example functionalities ERC-20 provides:
transfer tokens from one account to another
get the current token balance of an account
get the total supply of the token available on the network
approve whether an amount of token from an account can be spent by a third-party account
The difference with the well-knowns “NFT” (ERC-721), is that NFTs or Non-Fongible Tokens cannot be replaced with another identical token, while a Fongible-Tokens can be replaced. For instance you can replace a “ELD” token with another identical “ELD” token.
The public functions defined for ERC20 tokens are the following :
functionswapForELD()externalpayable underHeliosEye {uint256 grossELD = Math.mulDiv(msg.value, exchangeRatioELD,1e18, Math.Rounding(0));uint256 fee =(grossELD * feeBps)/10_000;uint256 netELD = grossELD - fee;require(netELD <= reserveELD,"HeliosDEX: Helios grieves that the ELD reserves are not plentiful enough for this exchange. A smaller offering would be most welcome"); reserveELD -= netELD; eldorionFang.transfer(msg.sender, netELD); emit HeliosBarter(address(eldorionFang),msg.value, netELD);}functionswapForMAL()externalpayable underHeliosEye {uint256 grossMal = Math.mulDiv(msg.value, exchangeRatioMAL,1e18, Math.Rounding(1));uint256 fee =(grossMal * feeBps)/10_000;uint256 netMal = grossMal - fee;require(netMal <= reserveMAL,"HeliosDEX: Helios grieves that the MAL reserves are not plentiful enough for this exchange. A smaller offering would be most welcome"); reserveMAL -= netMal; malakarEssence.transfer(msg.sender, netMal); emit HeliosBarter(address(malakarEssence),msg.value, netMal);}functionswapForHLS()externalpayable underHeliosEye {uint256 grossHLS = Math.mulDiv(msg.value, exchangeRatioHLS,1e18, Math.Rounding(3));uint256 fee =(grossHLS * feeBps)/10_000;uint256 netHLS = grossHLS - fee;require(netHLS <= reserveHLS,"HeliosDEX: Helios grieves that the HSL reserves are not plentiful enough for this exchange. A smaller offering would be most welcome"); reserveHLS -= netHLS; heliosLuminaShards.transfer(msg.sender, netHLS); emit HeliosBarter(address(heliosLuminaShards),msg.value, netHLS);}
Each function has the modifier underHeliosEye which we’ll see later, and they have the same process, only some variables differ.
y is the exchangeRatio, which is the answer to “how many token I get for ether”, it is different for each tokens and for ELD it is 2 (defined in HeliosDEX contract)
denominator is $10^{18}$ (1 ether in wei)
rounding is Math.Rounding(0)
Then, the fees are computed with the following :
$$ fee = \frac {(grossELD * feeBps)}{10000} $$
with feeBps = 25 (defined in HeliosDEX contract) :
Meaning we lost 0.25% of the value we send in fees which is not so much.
After these formalities, the contract checks if there are enough token to give us in the total supply, if yes, it subtract the quantity of tokens wanted from the total supply.
And finally, the netELD tokens amount is transferred to our balance on balanceOf[address].
1
2
3
4
require(netELD <= reserveELD,"HeliosDEX: Helios grieves that the ELD reserves are not plentiful enough for this exchange. A smaller offering would be most welcome");reserveELD -= netELD;eldorionFang.transfer(msg.sender, netELD);
And the tokens are transferred to our balance on balanceOf[address]
This process is the same for the tree tokens except from two differences, the exchange ratio and the rounding direction :
for ELD : ratio = 2, and rounding direction is 0
for MAL : ratio = 4, and rounding direction is 1
for HLS : ratio = 10, and rounding direction is 3
So for 1 Ether I get around 2 ELD, 4 MAL and 10 HLS.
But for 0.99 Ether, I get 0 ELD, 4 MAL and 10 HLS, due to the rounding direction. With a positive value, the result is rounded up, if zero, it is rounded down.
Before I mentioned the underHeliosEye modifier applied to every swap function:
1
2
3
4
modifierunderHeliosEye{require(msg.value>0,"HeliosDEX: Helios sees your empty hand! Only true offerings are worthy of a HeliosBarter");_;}
The latter makes sure we send at least 1 wei of value to swap.
This done, let’s look at the oneTimeRefund function used to refund our tokens and get back ethers.
functiononeTimeRefund(address item,uint256 amount)external heliosGuardedTrade {require(!hasRefunded[msg.sender],"HeliosDEX: refund already bestowed upon thee");require(amount >0,"HeliosDEX: naught for naught is no trade. Offer substance, or be gone!");uint256 exchangeRatio;if(item ==address(eldorionFang)){ exchangeRatio = exchangeRatioELD;require(eldorionFang.transferFrom(msg.sender,address(this), amount),"ELD transfer failed"); reserveELD += amount;}elseif(item ==address(malakarEssence)){ exchangeRatio = exchangeRatioMAL;require(malakarEssence.transferFrom(msg.sender,address(this), amount),"MAL transfer failed"); reserveMAL += amount;}elseif(item ==address(heliosLuminaShards)){ exchangeRatio = exchangeRatioHLS;require(heliosLuminaShards.transferFrom(msg.sender,address(this), amount),"HLS transfer failed"); reserveHLS += amount;}else{revert("HeliosDEX: Helios descries forbidden offering");}uint256 grossEth = Math.mulDiv(amount,1e18, exchangeRatio);uint256 fee =(grossEth * feeBps)/10_000;uint256 netEth = grossEth - fee; hasRefunded[msg.sender]=true;payable(msg.sender).transfer(netEth); emit HeliosRefund(item, amount, netEth);}
The contracts check if we already refunded our token and if the amount to refund is greater than zero.
1
2
require(!hasRefunded[msg.sender],"HeliosDEX: refund already bestowed upon thee");require(amount >0,"HeliosDEX: naught for naught is no trade. Offer substance, or be gone!");
After that, the exchange ratio is set according to the token we want to refund and the supply of token is increased, example for ELD :
The tokens are transferred back from us to the contract, but if we have not enough balance or the contract has not enough allowance on our balance, the execution is reverted.
Then the amount of ethers subtracted by the fees is computed and transferred to us :