🧑🏫 Tutorial
Introduction
In this tutorial, we will walk through the process of creating a simple dApp (short for "Decentralized Application") using Taquito. We will:
- Create a simple command-line application that reads the balance of an address from the blockchain
- Establish a high-level understanding of the blockchain, Tezos, dApps, and Taquito
- Send a
Transfer
operation to the blockchain using Taquito - Interact with a smart contract using Taquito
- Implement a simple GUI dApp
Prerequisites
Prior knowledge
In order to follow this tutorial, you need to have some understanding about the following concepts:
- Basic knowledge of JavaScript and programming in general
- A high-level understanding of blockchain technology and ecosystem (we will also cover this briefly in the tutorial)
Development machine
We need a development machine with the following software installed:
- Node.js: one of the current versions (LTS recommended)
- A code editor like VS Code, or any other editor of your choice
- Optionally: docker (for creating your own key pair)
This tutorial should work on Windows, Linux, and macOS. On other systems like a Chromebook or a tablet, you might need additional setup not covered in the tutorial.
If you are using windows, there are two ways to run the commands in this tutorial: inside a WSL2 terminal, or inside the PowerShell. Generally, WSL2 is recommended for programming. But if you prefer, you can still use PowerShell. If you use WSL2, you can use the same commands as Linux. For PowerShell, we will provide commands in a separate tab only when they are different from Linux.
Let's start with a simple command-line application
In this section, we will create a simple command-line application that shows the balance of an address. This will help us understand the basics of Taquito and the flow of events in a dApp.
Open a terminal and run the following commands:
- Linux & mac
- Windows Powershell
mkdir my-cli-dappcd my-cli-dappnpm init -ynpm i -D typescript ts-node
md my-cli-dappcd my-cli-dappnpm init -ynpm i -D typescript ts-node
console.log("Hello Blockchain!");
Now, run the following command in the terminal:
npx ts-node index.ts
If everything is done right, you should be able to see the output Hello Blockchain!
in the terminal.
Now, we can start using Taquito to interact with the Tezos blockchain.
Run the following command to install Taquito:
npm i @mavrykdynamics/taquito
Now, open the file index.ts
and replace the code with the following:
import { TezosToolkit } from "@mavrykdynamics/taquito";var tezosToolkit = new TezosToolkit("https://rpc.mavryk,network/basenet");tezosToolkit.tz.getBalance("mv1RK8FjLrVza385ZaeszhTeMiJcDjJk9ZLr").then(balance => {console.log(balance.toNumber());});
Running npx ts-node index.ts
should now show the balance of the specified address. This balance is in units of Mumav (micro Mav). Mav is the currency of Mavryk, its code is MVRK
, and the symbol is ṁ
. 1 Mav is 1,000,000 Mumav.
Congratulations! You have just interacted with the Tezos blockchain using Taquito. In the next section, we will establish a high-level understanding of the blockchain, Tezos, dApps, and Taquito. If you are already familiar with these concepts, you can skip to Sending a Transfer operation to the blockchain using Taquito.
What is a blockchain?
The blockchain is a way to trust a network of computers run by strangers (so you don't have to trust the individual people). It might seem impossible, but it works. How?
All computers that form a blockchain run the same software. They also store all the information needed to verify the integrity of the data. So anyone can verify that the data is correct. Techniques from cryptography are used to make this possible.
In order to work with a blockchain, a high level understanding is enough:
- The blockchain is a network of computers that run the same software.
- The blockchain stores data in a way that anyone can verify the integrity of the data.
- The data is split into "blocks". Each block contains a list of operations (like sending some tokens from one account to another).
- Once a block is created and the blockchain reaches a consensus on the information in it, it is impossible to change the data in the block.
- In order to send an operation to the blockchain, you can send it to any of the nodes participating in the consensus. The node will forward the data to the other nodes.
- In order to read data from the blockchain, you can send a request to any of the nodes.
- Anyone can read data from the blockchain. But to send an operation to the blockchain, it needs to be cryptographically signed.
Like any complex system, the simple overview we just gave is vastly simplified by leaving out a lot of details, and avoiding unnecessary precision. I believe having a good mental model of the system is more important than being precise. As you keep working in the blockchain ecosystem, your mental model will become more accurate over time.
How does the blockchain work?
What makes Tezos different?
Some interesting features in Tezos are designed to address the shortcomings of the earlier generations of blockchain. When Tezos was created, it had these benefits over existing blockchains:
- Proof of stake (It is now being adopted by some other blockchains as well). This eliminates a big problem with earlier blockchains: the need for a lot of energy to run the network.
- Evolution of the blockchain. Remember that the blockchain is a network of computers that run the same software. This means that if you want to upgrade the software, all the nodes need to be upgraded at the same time. This is not easy to do. Tezos solves this problem by having evolution baked into the protocol. This means that the blockchain can evolve over time without the need for a "hard fork".
- Delegation: Users can "delegate" their funds to a "baker". The baker will participate in the network consensus and will receive rewards. The baker will then share the rewards with the delegators. This makes it possible for users to participate in the network consensus without the need to run a node themselves or give control of their funds to a third party.
Because Tezos is designed to evolve, it is now much more than what it was when it was created.
Different versions of Tezos protocol are named after historic cities. At the time of this writing, we are in the "Nairobi" era. But the next protocol "Atlas" is being implemented and will be voted on soon, then the next one would be a city that starts with "P". 🤔 I wonder what will happen after we reach Z. 🤷
The "mainnet" is the actual Tezos Blockchain. However, there are several "testnets" that are used for testing. One of them is named "basenet", and it evolves to the new protocol much earlier than the mainnet, so that the ecosystem has enough time to implement and test the new protocol's features.
What is a dApp?
A dApp is a web application that interacts with a blockchain. The blockchain is the source of truth for the dApp. The dApp usually interacts with the user, reads data from, and writes to the blockchain. The dApp might also communicate with other services, notably a blockchain indexer. But dApps are not limited to these ideas. For instance, one might create a dApp that works as part of an industrial process, and interacts with the blockchain to benefit from its features, like transparency, immutability, and provability.
What is Taquito?
Taquito is a JavaScript library that makes it easy to interact with the Tezos blockchain. It is designed to be used in dApps. It is also used in some wallets and other tools. It is open-source, developed and maintained by ECAD Labs.
Without Taquito, sending operations to the Tezos blockchain requires you to write a lot of code. Some of that is to:
- Estimate the costs for the operation
- Properly encode the operation
- Sign the operation
- Inject the operation
- Get the operation receipt
- Monitor the chain for inclusion of the operation
And to implement all these, you need detailed information about different data types, protocols, constants, and algorithms used in the Tezos blockchain. Taquito abstracts away all this complexity and provides a simple API for interacting with the blockchain.
Sending a Transfer
operation to the blockchain using Taquito
Now, we want to send an operation to the blockchain. When reading, we just sent a read request. When sending an operation, we need to prove that we own the address. To do this, we need to sign the operation with the private key of the address.
In the next step, we will simply store the private key in the source code. This is not secure, and you should never do this in a production application.
Taquito provides an "In Memory Signer" functionality. Use the following command to add that to your project:
npm i @mavrykdynamics/taquito-signer
Open the file index.ts
and replace the code with the following:
import { InMemorySigner } from "@mavrykdynamics/taquito-signer";import { TezosToolkit } from "@mavrykdynamics/taquito";async function main() {var tezosToolkit = new TezosToolkit("https://rpc.mavryk,network/basenet");// WARNING: DO NOT DO THIS IN PRODUCTION, KEEP YOUR SECRETS SAFEconst signer = await InMemorySigner.fromSecretKey('spsk29SxqYRjnreqGzsYiAUEqxyhDwD8j2J57pJjaGgGtReZVD2UiD');const pkh = await signer.publicKeyHash();console.log(pkh);tezosToolkit.setProvider({ signer });const op = await tezosToolkit.contract.transfer({ to: 'mv1RK8FjLrVza385ZaeszhTeMiJcDjJk9ZLr', amount: 1 });await op.confirmation();console.log(op.hash);}main().catch(console.error);
Now, if you run your code, you should be able to see the address of the signer (mv2DZLWLuDRKUuR4BrWetimZ1C6Pg6pPAo3n
) and the hash of the operation in the terminal.
What is happening here? Let's break it down:
- We create a new instance of
TezosToolkit
and pass the URL of the node we want to connect to. - We create a new instance of
InMemorySigner
and pass the private key of the address we want to use to sign the operation. - We get the public key hash of the address.
- We set the provider of the
TezosToolkit
instance to the signer we created in step 2. - We send a
Transfer
operation to the blockchain. Taquito will automatically do the following behind the scenes:- Estimate the cost of the operation.
- Encode the operation.
- Sign the operation with the private key.
- Send the signed operation to the blockchain.
- We wait for the operation to be included in a block.
- We print the hash of the operation.
For the purpose of this section, I have created a new address and funded it on testnet. You are sharing the same secret key with everyone else going through this tutorial. So don't use this address for anything important. Also, there are two possible reasons why you might not be able to send the operation:
- Other people testing the code have consumed all the ṁ in the address, so your operation will fail because of insufficient funds. You can head over to basenet faucet and send some ṁ to the address for free. Or you can get free Testnet Mav right from your terminal:
npx @mavrykdynamics/get-mav <your-address> --amount 100 --network basenet
- Another person is sending an operation from this address at the same time as you. One address can only send one operation to each block. This is very improbable, but at least you know someone else is going through this tutorial at the same time as you. 😄
How to create my own public/private key pair?
Congratulations! You have just sent an operation to the Tezos blockchain using Taquito.
Interacting with a smart contract using Taquito
In this section, we will interact with a smart contract using Taquito. We will mint an NFT on objkt.com NFT marketplace.
Well, objk.com is the production service. We will use the testnet one: basenet.objkt.com, because to mint NFT on it, you only need basenet ṁ, which is free. Also, because I want to put my test secret key here, and I don't like to share my mainnet secret key with actual ṁ in it with everyone.
The concepts you learn here are not limited to objkt.com or NFTs. You can use the same concepts to interact with any smart contract on the Tezos blockchain.
Most dApps interact with smart contracts. You can think of a smart contract as a program that runs on the blockchain. The smart contract can store data and execute code. The code is executed when a user sends an operation to one of the smart contract's entrypoint
s. The smart contract can also send operations to the blockchain, or to other smart contracts.
In Tezos, smart contracts are written using one of the high-level languages (like Ligo), and compiled to Michelson. Then the contract is originated (deployed) to the blockchain. During origination, an address prefixed with KT1
is created for the contract. You can then interact with the smart contract by sending operations to this address.
In objkt.com, any user can create a number of collections and then mint NFTs in any of these collections. I have already created a collection and made our test address (mv2DZLWLuDRKUuR4BrWetimZ1C6Pg6pPAo3n
) an operator of the collection. So, this user can now mint NFTs in this collection. Check out the collection here, and note the number of tokens in it.
Open the file index.ts
and change the main
function to the following:
async function main() {var tezosToolkit = new TezosToolkit("https://rpc.mavryk,network/basenet");const signer = await InMemorySigner.fromSecretKey('spsk29SxqYRjnreqGzsYiAUEqxyhDwD8j2J57pJjaGgGtReZVD2UiD');tezosToolkit.setProvider({ signer });const contract = await tezosToolkit.contract.at('KT1JarALvhDLjtFhraeTMGGoeNLUkuL6jGtM');const op = await contract.methodsObject.mint_artist({collection_id: 71947,editions: 1,metadata_cid: '697066733a2f2f516d52325672336775713467594d45366268676b47474a34714656647652786867766e47516d7a6672346d364635',target: 'mv2DZLWLuDRKUuR4BrWetimZ1C6Pg6pPAo3n'}).send();await op.confirmation();console.log(op.hash);}
What happens here? Let's break it down:
1- We set up the Tezos Toolkit, and the signer as before.
1- We get the contract metadata from the blockchain. This metadata contains information about the contract, including its entrypoints.
1- We call the mint_artist
entrypoint of the contract. We need to provide the data it expects, which is provided as a javascript object. Taquito will automatically encode this data to be sent to the contract.
1- We wait for the confirmation and print the hash as before.
Now, if you run your code, you should be able to see the hash of the operation in the terminal. After about a minute, you should be able to see the new NFT in the collection from this link. Because everyone following this tutorial is minting NFTs with the same metadata, all the NFTs in this collection will look the same. However, the number of tokens in the collection should increase by one.
Congratulations! You have just interacted with a smart contract using Taquito. Additionally, you programmatically minted an NFT.
Up until now, we have created programs that run in a terminal. While some of the code written to interact with the blockchain might actually be such an application, most of the time, we will usually create dApps that can be accessed in the browser.
For these "browser dApps", there is a problem: how do we sign the operations? We can't store the secret key in the source code because anyone can see it. We also should not ask users to enter their secret key in our dApp, because that requires them to fully trust our dApp.
As it turns out, there is an elegant solution to this problem.
dApps and Crypto Wallets
The wallet is a program that stores the user's secrets (like the private key) and signs operations on behalf of the user. The wallet is not part of the dApp. It is a separate program. The wallet is usually a browser extension, a website, a mobile app, or a hardware wallet. The wallet is also responsible for showing the operations to the user and asking them to approve the operation. The user only needs to fully trust the wallet. The dApp cannot make any write operations to the blockchain unless it is signed by the wallet.
The beacon SDK is a library that provides a standard way for dApps to connect to wallets. The beacon SDK supports several wallets, including Thanos, Temple, and Kukai. You don't need to use the beacon SDK directly. Taquito uses the beacon SDK internally.
Another interesting component is the indexer/explorer. The way data is stored on the blockchain is optimized for storage, and to facilitate data retrieval that's essential for new operations (like checking the balance of an account). But some other operations might be slow. A blockchain indexer reads all the data from the blockchain and stores it in an optimized way for fast retrieval. Users can interact with that data through the explorer, which is a web application that shows the data in a user-friendly way. Also, dApps can read the data from the indexer to reduce the load on the blockchain and/or to have a faster response time.
The smart contract is conceptually part of the dApp, but it lives on the blockchain.
The flow of events in the dApp
Here is a high-level summary of the flow of events in the dApp:
- The user visits the dApp in their browser (by entering the URL or clicking on a link)
- The browser loads the dApp's code from a web server
- The dApp is loaded, and the user can interact with it
- At this stage, the dApp can read data from the blockchain, as long as the data does not need to be limited to a specific user (in our example dApp, the list of ideas can be read by anyone, but to show a list of user's collection, votes, etc., the dApp needs to know who the user is)
- The user makes an interaction that requires connecting the wallet
- The dApp shows a popup to the user, asking them to choose a wallet to connect to
- The user selects a wallet
- The user visits their wallet (on their phone, computer, a browser extension, in another tab, or even a hardware wallet) and approves the connection
- The user revisits the dApp. This time, the dApp might be showing additional information (such as the user's collection) or allowing the user to send operations to the blockchain (such as registering an idea or voting on an idea)
- The user makes an interaction that requires sending an operation to the blockchain
- The dApp sends the operation to the wallet
- The wallet shows the operation to the user and asks them to approve it
- The user approves the operation
- The wallet sends the signed operation to the blockchain
- The blockchain processes the operation
- The dApp can wait for the operation to be included in a block
- The dApp can read the result of the operation from the blockchain
Alternatively, in a slightly different flow, the wallet sends the signed operation to the dApp, and dApp sends it to the blockchain. From the user's point of view, both flows look the same.
Creating a simple dApp that transfers ṁ from the user's wallet to another address
We will start by creating a simple dApp that transfers ṁ from the user's wallet to another address. This will help us understand the flow of events in a dApp and the role of Taquito and Beacon SDK in the process.
creating the React app
Open your terminal and run the following commands:
npm create vite@latest my-dapp -- --template react-tscd my-dappnpm inpm run dev
The terminal should show a message that says: ➜ Local: http://localhost:4173/
but the port number might be different.
Now open a browser and visit the URL printed in the terminal. You should see a page that says: "Hello, Vite + React".
Optional: Commit the initial code to git
adding Taquito and Beacon SDK to the React app
In the next step, we add Taquito and Beacon SDK to the React app, and create a minimal UI to connect to the wallet and transfer ṁ.
npm i @mavrykdynamics/taquito @mavrykdynamics/taquito-beacon-wallet @mavrykdynamics/beacon-dapp
Open the file index.html
and make the following changes:
- <title>Vite + React</title>+ <title>My dApp</title>
Open the file src/App.tsx
and replace the content with the following code:
import { useState } from "react";import { TezosToolkit } from "@mavrykdynamics/taquito";import "./App.css";import ConnectButton from "./components/ConnectWallet";import Transfer from "./components/Transfer";import { BeaconWallet } from "@mavrykdynamics/taquito-beacon-wallet";const App = () => {const [Tezos] = useState<TezosToolkit>(new TezosToolkit("https://rpc.mavryk,network/basenet"));const [wallet, setWallet] = useState<BeaconWallet | undefined>(undefined);const [userAddress, setUserAddress] = useState<string | undefined>(undefined);switch (userAddress) {case undefined: return <ConnectButtonTezos={Tezos}setUserAddress={setUserAddress}setWallet={setWallet}wallet={wallet}/>;default: return <TransferTezos={Tezos}/>;}};export default App;
Connecting to the wallet
The first step in interacting with the blockchain is connecting to the user's wallet. Taquito provides a BeaconWallet class that abstracts away the complexity of connecting to the wallet. The BeaconWallet class is a wrapper around the Beacon SDK.
Create a new file src/components/ConnectWallet.tsx
and add the following code:
import { Dispatch, SetStateAction, useEffect } from "react";import { TezosToolkit } from "@mavrykdynamics/taquito";import { BeaconWallet } from "@mavrykdynamics/taquito-beacon-wallet";import {NetworkType,} from "@airgap/beacon-dapp";type ButtonProps = {Tezos: TezosToolkit;setUserAddress: Dispatch<SetStateAction<string | undefined>>;setWallet: Dispatch<SetStateAction<BeaconWallet | undefined>>;wallet: BeaconWallet | undefined;};const ConnectButton = ({Tezos,setUserAddress,setWallet,wallet,}: ButtonProps): JSX.Element => {const connectWallet = async (): Promise<void> => {try {await wallet!.requestPermissions({network: {type: NetworkType.BASENET,rpcUrl: "https://rpc.mavryk,network/basenet",},});const userAddress = await wallet!.getPKH();setUserAddress(userAddress);} catch (error) {console.log(error);}};useEffect(() => {(async () => {const wallet = new BeaconWallet({name: "My dApp",preferredNetwork: NetworkType.BASENET,disableDefaultEvents: false,enableMetrics: true,});Tezos.setWalletProvider(wallet);setWallet(wallet);})();}, []);return (<div className="buttons"><button className="button" onClick={connectWallet}><span><i className="fas fa-wallet"></i> Connect wallet</span></button></div>);};export default ConnectButton;
Transferring ṁ from the user's wallet to another address
After you connect to the wallet, you can send operations to the blockchain. In this step, we will create a simple UI to transfer ṁ from the user's wallet to another address.
Create a new file src/components/Transfer.tsx
and add the following code:
import { useState } from "react";import { TezosToolkit } from "@mavrykdynamics/taquito";const Transfer = ({Tezos,}: {Tezos: TezosToolkit;}): JSX.Element => {const [recipient, setRecipient] = useState<string>("");const [amount, setAmount] = useState<string>("");const [loading, setLoading] = useState<boolean>(false);const sendTez = async (): Promise<void> => {if (recipient && amount) {setLoading(true);try {const op = await Tezos.wallet.transfer({ to: recipient, amount: parseInt(amount), mumav: true }).send();await op.confirmation();} catch (error) {console.log(error);} finally {setLoading(false);}}};return (<div id="transfer-inputs">Recipient: <inputtype="text"placeholder="Recipient"value={recipient}onChange={e => setRecipient(e.target.value)}/><br />Amount in uTez:<inputtype="number"placeholder="Amount"value={amount}onChange={e => setAmount(e.target.value)}/><br /><buttonclassName="button"disabled={!recipient && !amount}onClick={sendTez}>{loading ? (<span><i className="fas fa-spinner fa-spin"></i> Sending...</span>) : (<span><i className="far fa-paper-plane"></i> Send</span>)}</button></div>);};export default Transfer;
Fixing node-specific dependencies in the browser
The libraries Taquito and Beacon SDK are designed to run in a Node.js environment. However, we are running them in a browser. This causes some issues. For example, the Beacon SDK uses the Node.js buffer
, stream
, and util
modules. These modules are not available in the browser. Fortunately, there are browser-compatible versions of these modules. We can use these versions instead of the Node.js versions. To do this, we need to install the following packages:
npm i -D vite-plugin-node-polyfills
Now we need to tell Vite to use this plugin. To do this, open the file vite.config.ts
and add the following code:
import { defineConfig } from 'vite'import react from '@vitejs/plugin-react'import { nodePolyfills } from 'vite-plugin-node-polyfills'export default defineConfig({plugins: [react(), nodePolyfills()],});
Running the dApp
Make sure that the command npm run dev
is still running in the terminal, and that there are no build errors.
Now, you should be able to see the "Connect Wallet" button in the browser. Clicking on it opens the wallet selection modal. You can choose your favorite wallet and connect to it. After this, you need to visit your wallet to approve the connection. After that, you should be able to see the "Send" button. You can enter an address and an amount (in Mumav, notice the mumav: true
in Transfer.tsx
) and click on the "Send" button to send ṁ to the address.
If you have not set up a wallet before, clicking on the Kukai wallet opens a page that asks you to create a new wallet. Remember to visit the basenet faucet to fund your wallet with some ṁ. If you want to use that wallet for real ṁ, you need to back up the mnemonic phrase. But remember that the mnemonic phrase is a secret. Anyone who has access to it can steal your ṁ.
Closing thoughts
We've come a long way:
- We started with a simple command-line application that reads the balance of an address from the blockchain.
- Then, we established a high-level understanding of the blockchain, Tezos, dApps, and Taquito.
- After that, we sent a
Transfer
operation to the blockchain using Taquito. - Then, we interacted with a smart contract using Taquito.
- Finally, we implemented a simple GUI dApp.
Your journey does not end here. There are many more things to learn. Here are some ideas:
- Check out the Taquito documentation to learn more about Taquito.
- Learn a smart contract language like Ligo.
- Learn more about Tezos and its ecosystem, check out the Tezos Developer Portal.
- See what others are doing: Join Tezos community on Discord, Reddit, Twitter, and Telegram.
- Start building. There is no better way to learn than to build something. You can start with a simple idea and build on it.
What is needed to make the dApp production-ready
Please ensure you read and consider the checklist in dApp pre-launch checklist before you launch your dApp.