Hello World Smart Contract for Beginners
If you are new to blockchain development and donβt know where to start, or if you just want to understand how to deploy and interact with smart contracts, this guide is for you. We will walk through creating and deploying a simple smart contract on the Goerli test network using a virtual wallet MetaMask(opens in a new tab), Solidity(opens in a new tab), Hardhat(opens in a new tab), and Alchemy(opens in a new tab) (donβt worry if you donβt understand what any of this means yet, we will explain it).
Warning
π§ Deprecation Notice
For the entirety of this guide, the Goerli test network is being used for creating and deploying a smart contract. However, please note that the Ethereum Foundation has announced that Goerli will soon be deprecated(opens in a new tab).
We recommend you to use the Sepolia(opens in a new tab) and Sepolia faucet(opens in a new tab) for this tutorial.
In part 2(opens in a new tab) of this tutorial weβll go through how we can interact with our smart contract once itβs deployed here, and in part 3(opens in a new tab) weβll cover how to publish it on Etherscan.
If you have questions at any point feel free to reach out in the Alchemy Discord(opens in a new tab)!
Step 1: Connect to the Ethereum network
There are many ways to make requests to the Ethereum chain. For simplicity, weβll use a free account on Alchemy, a blockchain developer platform and API that allows us to communicate with the Ethereum chain without having to run our own nodes. The platform also has developer tools for monitoring and analytics that weβll take advantage of in this tutorial to understand whatβs going on under the hood in our smart contract deployment. If you donβt already have an Alchemy account, you can sign up for free here(opens in a new tab).
Step 2: Create your app (and API key)
Once youβve created an Alchemy account, you can generate an API key by creating an app. This will allow us to make requests to the Goerli test network. If youβre not familiar with testnets, check out this page.
- Navigate to the βCreate Appβ page in your Alchemy Dashboard by hovering over βAppsβ in the nav bar and clicking βCreate Appβ
- Name your app βHello Worldβ, offer a short description, select βStagingβ for the Environment (used for your app bookkeeping), and choose βGoerliβ for your network.
- Click βCreate appβ and thatβs it! Your app should appear in the table below.
Step 3: Create an Ethereum account (address)
We need an Ethereum account to send and receive transactions. For this tutorial, weβll use MetaMask, a virtual wallet in the browser used to manage your Ethereum account address. More on transactions.
You can download and create a MetaMask account for free here(opens in a new tab). When you are creating an account, or if you already have an account, make sure to switch over to the βGoerli Test Networkβ in the upper right (so that weβre not dealing with real money).
Step 4: Add ether from a Faucet
In order to deploy our smart contract to the test network, weβll need some fake Eth. To get Eth you can go to the Goerli faucet(opens in a new tab) and log into your Alchemy account and enter your wallet address, then click βSend Me Eth.β It may take some time to receive your fake Eth due to network traffic. (At the time of writing this, it took around 30 minutes.) You should see Eth in your Metamask account soon after!
Step 5: Check your Balance
To double check our balance is there, letβs make an eth_getBalance(opens in a new tab) request using Alchemyβs composer tool(opens in a new tab). This will return the amount of ETH in our wallet. After you input your MetaMask account address and click βSend Requestβ, you should see a response like this:
1{ "jsonrpc": "2.0", "id": 0, "result": "0x2B5E3AF16B1880000" }Copy
NOTE: This result is in wei not ETH. Wei is used as the smallest denomination of ether. The conversion from wei to ETH is: 1 eth = 1018 wei. So if we convert 0x2B5E3AF16B1880000 to decimal we get 5*10ΒΉβΈ which equals 5 ETH.
Phew! Our fake money is all there .
Step 6: Initialize our project
First, weβll need to create a folder for our project. Navigate to your command line and type:
1mkdir hello-world2cd hello-world
Now that weβre inside our project folder, weβll use npm init
to initialize the project. If you donβt already have npm installed, follow these instructions(opens in a new tab) (weβll also need Node.js so download that too!).
1npm init
It doesnβt really matter how you answer the installation questions, here is how we did it for reference:
1package name: (hello-world)2version: (1.0.0)3description: hello world smart contract4entry point: (index.js)5test command:6git repository:7keywords:8author:9license: (ISC)10About to write to /Users/.../.../.../hello-world/package.json:1112{13 "name": "hello-world",14 "version": "1.0.0",15 "description": "hello world smart contract",16 "main": "index.js",17 "scripts": {18 "test": "echo \\"Error: no test specified\\" && exit 1"19 },20 "author": "",21 "license": "ISC"22}Show all
Approve the package.json and weβre good to go!
Step 7: Download Hardhat(opens in a new tab)
Hardhat is a development environment to compile, deploy, test, and debug your Ethereum software. It helps developers when building smart contracts and dapps locally before deploying to the live chain.
Inside our hello-world
project run:
1npm install --save-dev hardhat
Check out this page for more details on installation instructions(opens in a new tab).
Step 8: Create Hardhat project
Inside our project folder run:
1npx hardhat
You should then see a welcome message and option to select what you want to do. Select βcreate an empty hardhat.config.jsβ:
1888 888 888 888 8882888 888 888 888 8883888 888 888 888 88848888888888 8888b. 888d888 .d88888 88888b. 8888b. 8888885888 888 "88b 888P" d88" 888 888 "88b "88b 8886888 888 .d888888 888 888 888 888 888 .d888888 8887888 888 888 888 888 Y88b 888 888 888 888 888 Y88b.8888 888 "Y888888 888 "Y88888 888 888 "Y888888 "Y888910π· Welcome to Hardhat v2.0.11 π·β?1112What do you want to do? β¦13Create a sample project14β― Create an empty hardhat.config.js15QuitShow all
This will generate a hardhat.config.js
file for us which is where weβll specify all of the set up for our project (on step 13).
Step 9: Add project folders
To keep our project organized weβll create two new folders. Navigate to the root directory of your project in your command line and type:
1mkdir contracts2mkdir scripts
contracts/
is where weβll keep our hello world smart contract code filescripts/
is where weβll keep scripts to deploy and interact with our contract
Step 10: Write our contract
You might be asking yourself, when the heck are we going to write code?? Well, here we are, on step 10.
Open up the hello-world project in your favorite editor (we like VSCode(opens in a new tab)). Smart contracts are written in a language called Solidity which is what we will use to write our HelloWorld.sol smart contract.β
- Navigate to the βcontractsβ folder and create a new file called HelloWorld.sol
- Below is a sample Hello World smart contract from the Ethereum Foundation that we will be using for this tutorial. Copy and paste in the contents below into your HelloWorld.sol file, and be sure to read the comments to understand what this contract does:
1// Specifies the version of Solidity, using semantic versioning.2// Learn more: https://solidity.readthedocs.io/en/v0.5.10/layout-of-source-files.html#pragma3pragma solidity ^0.7.0;45// Defines a contract named `HelloWorld`.6// A contract is a collection of functions and data (its state). Once deployed, a contract resides at a specific address on the Ethereum blockchain. Learn more: https://solidity.readthedocs.io/en/v0.5.10/structure-of-a-contract.html7contract HelloWorld {89 // Declares a state variable `message` of type `string`.10 // State variables are variables whose values are permanently stored in contract storage. The keyword `public` makes variables accessible from outside a contract and creates a function that other contracts or clients can call to access the value.11 string public message;1213 // Similar to many class-based object-oriented languages, a constructor is a special function that is only executed upon contract creation.14 // Constructors are used to initialize the contract's data. Learn more:https://solidity.readthedocs.io/en/v0.5.10/contracts.html#constructors15 constructor(string memory initMessage) {1617 // Accepts a string argument `initMessage` and sets the value into the contract's `message` storage variable).18 message = initMessage;19 }2021 // A public function that accepts a string argument and updates the `message` storage variable.22 function update(string memory newMessage) public {23 message = newMessage;24 }25}Show allCopy
This is a super simple smart contract that stores a message upon creation and can be updated by calling the update
function.
Step 11: Connect MetaMask & Alchemy to your project
Weβve created a MetaMask wallet, Alchemy account, and written our smart contract, now itβs time to connect the three.
Every transaction sent from your virtual wallet requires a signature using your unique private key. To provide our program with this permission, we can safely store our private key (and Alchemy API key) in an environment file.
To learn more about sending transactions, check out this tutorial on sending transactions using web3.
First, install the dotenv package in your project directory:
1npm install dotenv --save
Then, create a .env
file in the root directory of our project, and add your MetaMask private key and HTTP Alchemy API URL to it.
- Follow these instructions(opens in a new tab) to export your private key
- See below to get HTTP Alchemy API URL
Copy Alchemy API URL
Your .env
should look like this:
1API_URL = "https://eth-goerli.alchemyapi.io/v2/your-api-key"2PRIVATE_KEY = "your-metamask-private-key"
To actually connect these to our code, weβll reference these variables in our hardhat.config.js
file on step 13.
.env
! Please make sure never to share or expose your .env
file with anyone, as you are compromising your secrets in doing so. If you are using version control, add your .env
to a gitignore(opens in a new tab) file.Step 12: Install Ethers.js
Ethers.js is a library that makes it easier to interact and make requests to Ethereum by wrapping standard JSON-RPC methods with more user friendly methods.
Hardhat makes it super easy to integrate Plugins(opens in a new tab) for additional tooling and extended functionality. Weβll be taking advantage of the Ethers plugin(opens in a new tab) for contract deployment (Ethers.js(opens in a new tab) has some super clean contract deployment methods).
In your project directory type:
1npm install --save-dev @nomiclabs/hardhat-ethers "ethers@^5.0.0"
Weβll also require ethers in our hardhat.config.js
in the next step.
Step 13: Update hardhat.config.js
Weβve added several dependencies and plugins so far, now we need to update hardhat.config.js
so that our project knows about all of them.
Update your hardhat.config.js
to look like this:
1require('dotenv').config();23require("@nomiclabs/hardhat-ethers");4const { API_URL, PRIVATE_KEY } = process.env;56/**7* @type import('hardhat/config').HardhatUserConfig8*/9module.exports = {10 solidity: "0.7.3",11 defaultNetwork: "goerli",12 networks: {13 hardhat: {},14 goerli: {15 url: API_URL,16 accounts: [`0x${PRIVATE_KEY}`]17 }18 },19}Show all
Step 14: Compile our contract
To make sure everything is working so far, letβs compile our contract. The compile
task is one of the built-in hardhat tasks.
From the command line run:
1npx hardhat compile
You might get a warning about SPDX license identifier not provided in source file
, but no need to worry about that β hopefully everything else looks good! If not, you can always message in the Alchemy discord(opens in a new tab).
Step 15: Write our deploy script
Now that our contract is written and our configuration file is good to go, itβs time to write our contract deploy script.
Navigate to the scripts/
folder and create a new file called deploy.js
, adding the following contents to it:
1async function main() {2 const HelloWorld = await ethers.getContractFactory("HelloWorld");34 // Start deployment, returning a promise that resolves to a contract object5 const hello_world = await HelloWorld.deploy("Hello World!");6 console.log("Contract deployed to address:", hello_world.address);}78main()9 .then(() => process.exit(0))10 .catch(error => {11 console.error(error);12 process.exit(1);13 });Show all
Hardhat does an amazing job of explaining what each of these lines of code does in their Contracts tutorial(opens in a new tab), weβve adopted their explanations here.
1const HelloWorld = await ethers.getContractFactory("HelloWorld");
A ContractFactory
in ethers.js is an abstraction used to deploy new smart contracts, so HelloWorld
here is a factory for instances of our hello world contract. When using the hardhat-ethers
plugin ContractFactory
and Contract
instances are connected to the first signer by default.
1const hello_world = await HelloWorld.deploy();
Calling deploy()
on a ContractFactory
will start the deployment, and return a Promise
that resolves to a Contract
. This is the object that has a method for each of our smart contract functions.
Step 16: Deploy our contract
Weβre finally ready to deploy our smart contract! Navigate to the command line and run:
1npx hardhat run scripts/deploy.js --network goerli
You should then see something like:
1Contract deployed to address: 0x6cd7d44516a20882cEa2DE9f205bF401c0d23570
If we go to the Goerli etherscan(opens in a new tab) and search for our contract address we should able to see that it has been deployed successfully. The transaction will look something like this:
The From
address should match your MetaMask account address and the To address will say βContract Creationβ but if we click into the transaction weβll see our contract address in the To
field:
Congrats! You just deployed a smart contract to the Ethereum chain π
To understand whatβs going on under the hood, letβs navigate to the Explorer tab in our Alchemy dashboard(opens in a new tab). If you have multiple Alchemy apps make sure to filter by app and select βHello Worldβ.
Here youβll see a handful of JSON-RPC calls that Hardhat/Ethers made under the hood for us when we called the .deploy()
function. Two important ones to call out here are eth_sendRawTransaction
(opens in a new tab), which is the request to actually write our contract onto the Goerli chain, and eth_getTransactionByHash
(opens in a new tab) which is a request to read information about our transaction given the hash (a typical pattern when
transactions). To learn more about sending transactions, check out this tutorial on sending transactions using Web3
Thatβs all for part 1 of this tutorial, in part 2 weβll actually interact with our smart contract(opens in a new tab) by updated our initial message, and in part 3 weβll publish our smart contract to Etherscan(opens in a new tab) so everyone will know how to interact with it.
Want to learn more about Alchemy? Check out our website(opens in a new tab). Never want to miss an update? Subscribe to our newsletter here(opens in a new tab)! Be sure to also follow our Twitter(opens in a new tab) and join our Discord(opens in a new tab).
Last edit: @pettinarip(opens in a new tab), November 23, 2023