🤖 Smart contracts are kind of like "always on" vending machines that anyone can access. Let's make a decentralized, digital currency. Then, let's build an unstoppable vending machine that will buy and sell the currency. We'll learn about the "approve" pattern for ERC20s and how contract to contract interactions work.
🏵 Create
YourToken.sol
smart contract that inherits the ERC20 token standard from OpenZeppelin. Set your token to_mint()
1000 (* 10 ** 18) tokens to themsg.sender
. Then create aVendor.sol
contract that sells your token using a payablebuyTokens()
function.
🎛 Edit the frontend that invites the user to
<input\>
an amount of tokens they want to buy. We'll display a preview of the amount of ETH (or USD) it will cost with a confirm button.
🔍 It will be important to verify your token's source code in the block explorer after you deploy. Supporters will want to be sure that it has a fixed supply and you can't just mint more.
🌟 The final deliverable is an app that lets users purchase and transfer your token. Deploy your contracts on your public chain of choice and then
yarn build
andyarn surge
your app to a public web server. Submit the url on SpeedRunEthereum.com!
💬 Meet other builders working on this challenge and get help in the Challenge 2 telegram!
🧫 Everything starts by ✏️ Editing YourToken.sol
in packages/hardhat/contracts
git clone https://github.com/scaffold-eth/scaffold-eth-typescript-challenges.git challenge-2-token-vendor
cd challenge-2-token-vendor
git checkout challenge-2-token-vendor
yarn install
🔏 Edit your smart contract YourToken.sol
in packages/hardhat-ts/contracts
You'll have three terminals up for:
yarn chain
(hardhat backend)
yarn deploy
(to compile, deploy, and publish your contracts to the frontend)
yarn start
(react app frontend)
Make sure you run the commands in the above order. The contract types get generated as part of the deploy, which will be required to build and start the app.
👀 Visit your frontend at http://localhost:3000
👩💻 Rerun
yarn deploy --reset
whenever you want to deploy new contracts to the frontend.
ignore any warnings, we'll get to that...
👩💻 Edit
YourToken.sol
to inherit the ERC20 token standard from OpenZeppelin
Mint 1000 (* 10 ** 18) in the constructor (to the msg.sender
) and then send them to your frontend address in the deploy/00_deploy_your_token.ts
:
const result = await yourToken.transfer(
"**YOUR FRONTEND ADDRESS**",
ethers.utils.parseEther("1000")
);
(Your frontend address is the address in the top right of your frontend. Go to localhost:3000 and copy the address from the top right.)
- Can you check the
balanceOf()
your frontend address in the YourToken of theDebug Contracts
tab? - Can you
transfer()
your token to another account and check that account'sbalanceOf
?
(Use an incognito window to create a new address and try sending to that new address. Use the transfer()
function in the Debug Contracts
tab.)
👩💻 Create a
Vendor.sol
contract with a payablebuyTokens()
function
Use a price variable named tokensPerEth
set to 100:
uint256 public constant tokensPerEth = 100;
📝 The
buyTokens()
function inVendor.sol
should usemsg.value
andtokensPerEth
to calculate an amount of tokens toyourToken.transfer()
tomsg.sender
.
📟 Emit event
BuyTokens(address buyer, uint256 amountOfEth, uint256 amountOfTokens)
when tokens are purchased.
Edit deploy/01_deploy_vendor.ts
to deploy the Vendor
(uncomment Vendor deploy lines).
You will also want to change 00_deploy_your_token.ts
and 01_deploy_vendor.ts
so you transfer the tokens to the vendor.address
instead of your frontend address.
const result = await yourToken.transfer(
vendor.address,
ethers.utils.parseEther("1000")
);
(You will use the YourToken
UI tab and the frontend for most of your testing. Most of the UI is already built for you for this challenge.)
📝 Edit
Vendor.sol
to inherit Ownable.
In deploy/01_deploy_vendor.ts
you will need to call transferOwnership()
on the Vendor
to make your frontend address the owner
:
await vendor.transferOwnership("**YOUR FRONTEND ADDRESS**");
📝 Finally, add a
withdraw()
function inVendor.sol
that lets the owner withdraw ETH from the vendor.
- Does the
Vendor
address start with abalanceOf
1000 inYourToken
on theDebug Contracts
tab? - Can you buy 10 tokens for 0.1 ETH?
- Can you transfer tokens to a different account?
- Can the
owner
withdraw the ETH from theVendor
?
- Can anyone withdraw? Test everything!
- What if you minted 2000 and only sent 1000 to the
Vendor
?
👩🏫 The hardest part of this challenge is to build your Vendor
to buy the tokens back.
🧐 The reason why this is hard is the approve()
pattern in ERC20s.
😕 First, the user has to call approve()
on the YourToken
contract, approving the Vendor
contract address to take some amount of tokens.
🤨 Then, the user makes a second transaction to the Vendor
contract to sellTokens()
.
🤓 The Vendor
should call yourToken.transferFrom(msg.sender, address(this), theAmount)
and if the user has approved the Vendor
correctly, tokens should transfer to the Vendor
and ETH should be sent to the user.
(Use the Debug Contracts
tab to call the approve and sellTokens() at first but then look in the YourToken.tsx
for the extra approve/sell UI to uncomment.)
- Can you sell tokens back to the vendor and receive ETH?
- Should we disable the
owner
withdraw to keep liquidity in theVendor
?
- Now is a good time to run
yarn test
to run the automated testing function. It will test that you hit the core checkpoints. You are looking for all green checkmarks and passing tests!
📡 Edit the defaultNetwork
in packages/hardhat-ts/hardhat.config.ts
, as well as targetNetworkInfo
in packages/vite-app-ts/src/config/providersConfig.ts
, to your choice of public EVM networks
👩🚀 You will want to run yarn account
to see if you have a deployer address.
🔐 If you don't have one, run yarn generate
to create a mnemonic and save it locally for deploying.
🛰 Use a faucet like faucet.paradigm.xyz to fund your deployer address (run yarn account
again to view balances)
🚀 Run
yarn deploy
to deploy to your public network of choice (😅 wherever you can get ⛽️ gas)
🔬 Inspect the block explorer for the network you deployed to... make sure your contract is there.
📦 Run yarn build
to package up your frontend.
💽 Upload your app to surge with yarn surge
(you could also yarn s3
or maybe even yarn ipfs
?)
🚔 Traffic to your url might break the Infura rate limit, edit your key: constants.ts
in packages/vite-app-ts/src/models/constants
.
Update the api-key in packages/hardhat-ts/package.json file. You can get your key here.
Now you are ready to run the
yarn verify --network your_network
command to verify your contracts on etherscan 🛰
This will be the URL you submit to SpeedRun.
🏃 Head to your next challenge here.
💬 Problems, questions, comments on the stack? Post them to the 🏗 scaffold-eth developers chat