Un contrat intelligent « Hello World » pour les débutants - Fullstack
Ce guide s'adresse à vous si vous êtes novice en matière de développement blockchain et que vous ne savez pas par où commencer ou comment déployer et interagir avec les contrats intelligents. Nous allons vous guider dans la création et le déploiement d'un contrat intelligent simple sur le réseau de test Goerli en utilisant MetaMask (opens in a new tab), Solidity (opens in a new tab), Hardhat (opens in a new tab), et Alchemy (opens in a new tab).
Vous aurez besoin d'un compte Alchemy pour achever ce tutoriel. Inscrivez-vous pour un compte gratuit (opens in a new tab).
Si vous avez des questions à un moment ou à un autre, n'hésitez pas à nous contacter sur le Discord d'Alchemy (opens in a new tab) !
Partie 1 - Créer et déployer votre contrat intelligent avec Hardhat
Se connecter au réseau Ethereum
Il existe de nombreuses façons de faire des requêtes sur la chaîne Ethereum. Pour plus de simplicité, nous allons utiliser un compte gratuit sur Alchemy, une plateforme de développement blockchain et une API qui nous permet de communiquer avec la chaîne Ethereum sans avoir à exécuter notre propre nœud. Alchemy dispose également d'outils de développement pour la surveillance et l'analyse, dont nous allons tirer parti dans ce tutoriel, pour comprendre ce qui se passe sous le capot dans le déploiement de notre contrat intelligent.
Créez votre application et votre clé API
Une fois votre compte Alchemy créé, vous pouvez générer une clé API en créant une application. Cela va vous permettre d'émettre des requêtes sur le réseau de test Goerli. Si vous n'êtes pas familier avec les réseaux de test, vous pouvez lire le guide d'Alchemy pour choisir un réseau (opens in a new tab).
Sur le tableau de bord Alchemy, trouvez le menu déroulant Apps dans la barre de navigation et cliquez sur Create App.
Donnez à votre application le nom « Hello World » et écrivez une courte description. Sélectionnez Staging comme environnement et Goerli comme réseau.
Note : assurez-vous de sélectionner Goerli, sinon ce tutoriel ne fonctionnera pas.
Cliquez sur Créer une application. Votre application apparaîtra dans le tableau ci-dessous.
Créer un compte Ethereum
Vous avez besoin d'un compte Ethereum pour envoyer et recevoir des transactions. Nous utiliserons MetaMask, un portefeuille virtuel intégré au navigateur permettant aux utilisateurs de gérer l'adresse de leur compte Ethereum.
Vous pouvez télécharger et créer un compte MetaMask gratuitement ici (opens in a new tab). Lorsque vous créez un compte, ou si vous en avez déjà un, assurez-vous de basculer sur le « Réseau de test Goerli » en haut à droite (afin de ne pas utiliser d'argent réel).
Étape 4 : Ajouter de l'ether à partir d'un robinet
Afin de déployer votre contrat intelligent sur le réseau de test, vous aurez besoin de faux ETH. Pour obtenir de l'ETH sur le réseau Goerli, rendez-vous sur un robinet Goerli et entrez l'adresse de votre compte Goerli. Notez que les robinets Goerli peuvent avoir quelques difficultés de fonctionnement ces derniers temps - consultez la page des réseaux de test pour une liste d'options à essayer :
Note : en raison de la congestion du réseau, cela peut prendre un certain temps. ``
Étape 5 : Vérifiez votre solde
Pour vérifier que les ETH sont bien dans votre portefeuille, nous allons effectuer une requête eth_getBalance (opens in a new tab) en utilisant l'outil Composer d'Alchemy (opens in a new tab). Cela va renvoyer la quantité d'ETH dans notre portefeuille. Pour en savoir plus, consultez le court tutoriel d'Alchemy sur la manière d'utiliser l'outil Composer (opens in a new tab).
Saisissez l'adresse de votre compte MetaMask et cliquez sur Envoyer la requête. Vous verrez une réponse qui ressemble à l'extrait de code ci-dessous.
1{ "jsonrpc": "2.0", "id": 0, "result": "0x2B5E3AF16B1880000" }Note : Ce résultat est en wei, pas en ETH. Le wei est utilisé comme la plus petite dénomination d'ether.
Ouf ! Notre faux argent est bien là.
Étape 6 : Initialiser notre projet
Tout d'abord, nous devons créer un dossier pour notre projet. Accédez à votre ligne de commande et entrez ce qui suit.
1mkdir hello-world2cd hello-worldMaintenant que nous sommes dans notre dossier de projet, nous allons utiliser npm init pour initialiser le projet.
Si vous n'avez pas encore installé npm, suivez ces instructions pour installer Node.js et npm (opens in a new tab).
Pour les besoins de ce tutoriel, peu importe comment vous répondez aux questions d'initialisation. Voici comment nous l'avons fait à titre de référence :
1package name: (hello-world)2version: (1.0.0)3description: hello world smart contract4entry point: (index.js)5test command:6git repository:7keywords:8author:9license: (ISC)1011About to write to /Users/.../.../.../hello-world/package.json:1213{14 "name": "hello-world",15 "version": "1.0.0",16 "description": "hello world smart contract",17 "main": "index.js",18 "scripts": {19 "test": "echo \"Error: no test specified\" && exit 1"20 },21 "author": "",22 "license": "ISC"23}Afficher toutApprouvez le fichier package.json et nous sommes prêts !
Étape 7 : Télécharger Hardhat
Hardhat est un environnement de développement qui permet de compiler, déployer, tester et déboguer votre logiciel Ethereum. Il aide les développeurs à construire des contrats intelligents et des dApps localement avant de les déployer sur la chaîne en production.
Dans notre projet hello-world, exécutez :
1npm install --save-dev hardhatConsultez cette page pour plus de détails sur les instructions d'installation (opens in a new tab).
Étape 8 : Créer un projet Hardhat
Dans le dossier de notre projet hello-world, exécutez :
1npx hardhatVous devriez maintenant voir un message de bienvenue ainsi qu'une option pour sélectionner ce que vous voulez faire. Sélectionnez : « 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👷 Bienvenue dans Hardhat v2.0.11 👷1112Que voulez-vous faire ? …13Créer un projet d'exemple14❯ Créer un fichier hardhat.config.js vide15QuitterAfficher toutCela générera un fichier hardhat.config.js dans le projet. Nous l'utiliserons plus tard dans le tutoriel pour spécifier la configuration de notre projet.
Étape 9 : Ajouter les dossiers du projet
Pour garder le projet organisé, créons deux nouveaux dossiers. Dans la ligne de commande, naviguez vers le répertoire racine de votre projet hello-world et tapez :
1mkdir contracts2mkdir scriptscontracts/est l'endroit où nous conserverons le fichier de code de notre contrat intelligent hello world.scripts/est l'endroit où nous conserverons les scripts pour déployer notre contrat et interagir avec lui.
Étape 10 : Écrire notre contrat
Vous vous demandez peut-être : quand allons-nous écrire du code ? C'est maintenant !
Ouvrez le projet hello-world dans votre éditeur préféré. Les contrats intelligents sont le plus souvent écrits en Solidity, que nous utiliserons pour écrire notre contrat intelligent.
- Naviguez vers le dossier
contractset créez un nouveau fichier appeléHelloWorld.sol - Ci-dessous se trouve un exemple de contrat intelligent Hello World que nous utiliserons pour ce tutoriel. Copiez le contenu ci-dessous dans le fichier
HelloWorld.sol.
Note : Assurez-vous de lire les commentaires pour comprendre ce que fait ce contrat.
1// Spécifie la version de Solidity, en utilisant le versionnage sémantique.2// En savoir plus : https://solidity.readthedocs.io/en/v0.5.10/layout-of-source-files.html#pragma3pragma solidity >=0.7.3;45// Définit un contrat nommé `HelloWorld`.6// Un contrat est une collection de fonctions et de données (son état). Une fois déployé, un contrat réside à une adresse spécifique sur la blockchain Ethereum. En savoir plus : https://solidity.readthedocs.io/en/v0.5.10/structure-of-a-contract.html7contract HelloWorld {89 //Émis lorsque la fonction de mise à jour est appelée10 //Les événements de contrat intelligent sont un moyen pour votre contrat de communiquer que quelque chose s'est passé sur la blockchain à votre interface d'application, qui peut « écouter » certains événements et prendre des mesures lorsqu'ils se produisent.11 event UpdatedMessages(string oldStr, string newStr);1213 // Déclare une variable d'état `message` de type `string`.14 // Les variables d'état sont des variables dont les valeurs sont stockées en permanence dans le stockage du contrat. Le mot-clé `public` rend les variables accessibles depuis l'extérieur d'un contrat et crée une fonction que d'autres contrats ou clients peuvent appeler pour accéder à la valeur.15 string public message;1617 // Semblable à de nombreux langages orientés objet basés sur les classes, un constructeur est une fonction spéciale qui n'est exécutée qu'à la création du contrat.18 // Les constructeurs sont utilisés pour initialiser les données du contrat. En savoir plus : https://solidity.readthedocs.io/en/v0.5.10/contracts.html#constructors19 constructor(string memory initMessage) {2021 // Accepte un argument de chaîne `initMessage` et définit la valeur dans la variable de stockage `message` du contrat).22 message = initMessage;23 }2425 // Une fonction publique qui accepte un argument de chaîne et met à jour la variable de stockage `message`.26 function update(string memory newMessage) public {27 string memory oldMsg = message;28 message = newMessage;29 emit UpdatedMessages(oldMsg, newMessage);30 }31}Afficher toutIl s'agit d'un contrat intelligent basique qui stocke un message lors de sa création. Il peut être mis à jour en appelant la fonction update.
Étape 11 : Connecter MetaMask et Alchemy à votre projet
Nous avons créé un portefeuille MetaMask, un compte Alchemy et rédigé notre contrat intelligent. Il est maintenant temps de connecter les trois.
Chaque transaction envoyée depuis votre portefeuille nécessite une signature à l'aide de votre clé privée unique. Pour fournir cette autorisation à notre programme, nous pouvons stocker en toute sécurité notre clé privée dans un fichier d'environnement. Nous y stockerons également une clé API pour Alchemy.
Pour en savoir plus sur l'envoi de transactions, consultez ce tutoriel (opens in a new tab) sur l'envoi de transactions à l'aide de web3.
Premièrement, installez le paquet dotenv dans votre dossier de projet :
1npm install dotenv --saveEnsuite, créez un fichier .env dans le répertoire racine du projet. Ajoutez-y votre clé privée MetaMask et l'URL API HTTP d'Alchemy.
Votre fichier d'environnement doit être nommé .env ou il ne sera pas reconnu comme un fichier d'environnement.
Ne le nommez pas process.env ou .env-custom ou quoi que ce soit d'autre.
- Suivez ces instructions (opens in a new tab) pour exporter votre clé privée
- Voir ci-dessous pour obtenir l'URL de l'API HTTP Alchemy
Votre .env devrait ressembler à ceci :
1API_URL = "https://eth-goerli.alchemyapi.io/v2/your-api-key"2PRIVATE_KEY = "your-metamask-private-key"Pour les relier à notre code, nous ferons référence à ces variables dans notre fichier hardhat.config.js à l'étape 13.
Étape 12 : Installer Ethers.js
Ethers.js est une bibliothèque qui facilite l'interaction et les requêtes vers Ethereum en enveloppant les méthodes JSON-RPC standard (opens in a new tab) avec des méthodes plus conviviales.
Hardhat nous permet d'intégrer des plugins (opens in a new tab) pour des outils supplémentaires et des fonctionnalités étendues. Nous allons profiter du plugin Ethers (opens in a new tab) pour le déploiement de contrats.
Dans votre dossier de projet, tapez :
npm install --save-dev @nomiclabs/hardhat-ethers "ethers@^5.0.0"Étape 13 : Mettre à jour hardhat.config.js
À ce stade, nous avons ajouté plusieurs dépendances et plugins. Nous devons maintenant mettre à jour hardhat.config.js pour que notre projet les reconnaisse.
Mettez à jour votre hardhat.config.js pour qu'il ressemble à ceci :
1/**2 * @type import('hardhat/config').HardhatUserConfig3 */45require("dotenv").config()6require("@nomiclabs/hardhat-ethers")78const { API_URL, PRIVATE_KEY } = process.env910module.exports = {11 solidity: "0.7.3",12 defaultNetwork: "goerli",13 networks: {14 hardhat: {},15 goerli: {16 url: API_URL,17 accounts: [`0x${PRIVATE_KEY}`],18 },19 },20}Afficher toutÉtape 14 : Compiler notre contrat
Pour s’assurer à ce stade que tout fonctionne, compilons notre contrat. La tâche compile est l'une des tâches intégrées de hardhat.
À partir de la ligne de commande, exécutez :
npx hardhat compileVous pourriez recevoir un avertissement concernant l'« identifiant de licence SPDX non fourni dans le fichier source », mais ne vous inquiétez pas — espérons que tout le reste se passe bien ! Sinon, vous pouvez toujours envoyer un message sur le Discord d'Alchemy (opens in a new tab).
Étape 15 : Écrire notre script de déploiement
Maintenant que notre contrat est codé et que notre fichier de configuration est bon, il est temps d’écrire notre script de déploiement pour notre contrat.
Naviguez vers le dossier scripts/ et créez un nouveau fichier appelé deploy.js, en y ajoutant le contenu suivant :
1async function main() {2 const HelloWorld = await ethers.getContractFactory("HelloWorld")34 // Démarrer le déploiement, en retournant une promesse qui se résout en un objet de contrat5 const hello_world = await HelloWorld.deploy("Hello World!")6 console.log("Contrat déployé à l'adresse :", hello_world.address)7}89main()10 .then(() => process.exit(0))11 .catch((error) => {12 console.error(error)13 process.exit(1)14 })Afficher toutHardhat explique très bien ce que fait chacune de ces lignes de code dans son tutoriel sur les contrats (opens in a new tab), nous avons repris leurs explications ici.
1const HelloWorld = await ethers.getContractFactory("HelloWorld")Un ContractFactory dans ethers.js est une abstraction utilisée pour déployer de nouveaux contrats intelligents, donc HelloWorld ici est une factory (fabrique) (opens in a new tab) pour les instances de notre contrat hello world. Lors de l'utilisation du plugin hardhat-ethers, les instances ContractFactory et Contract sont connectées par défaut au premier signataire (propriétaire).
1const hello_world = await HelloWorld.deploy()Appeler deploy() sur une ContractFactory lancera le déploiement et retournera une Promise qui se résout en un objet Contract. C'est l'objet qui possède une méthode pour chacune des fonctions de notre contrat intelligent.
Étape 16 : Déployer notre contrat
Nous sommes enfin prêts à déployer notre contrat intelligent ! Naviguez vers la ligne de commande et exécutez :
npx hardhat run scripts/deploy.js --network goerliVous devriez maintenant voir quelque chose comme :
Contrat déployé à l'adresse : 0x6cd7d44516a20882cEa2DE9f205bF401c0d23570Veuillez sauvegarder cette adresse. Nous l'utiliserons plus tard dans le tutoriel.
Si nous allons sur l'Etherscan de Goerli (opens in a new tab) et que nous recherchons l'adresse de notre contrat, nous devrions être en mesure de voir qu'il a été déployé avec succès. La transaction ressemblera à quelque chose comme :
L'adresse From devrait correspondre à l'adresse de votre compte MetaMask et l'adresse To indiquera Création de Contrat. Si nous cliquons sur la transaction, nous verrons l'adresse de notre contrat dans le champ To.
Félicitations ! Vous venez de déployer un contrat intelligent sur un réseau de test Ethereum.
Pour comprendre ce qui se passe en coulisses, naviguons vers l'onglet Explorer dans notre tableau de bord Alchemy (opens in a new tab). Si vous avez plusieurs applications Alchemy, assurez-vous de filtrer par application et de sélectionner Hello World.
Ici, vous verrez une poignée de méthodes JSON-RPC que Hardhat/Ethers ont créées en coulisses pour nous lorsque nous avons appelé la fonction .deploy(). Deux méthodes importantes ici sont eth_sendRawTransaction (opens in a new tab), qui est la requête pour écrire notre contrat sur la chaîne Goerli, et eth_getTransactionByHash (opens in a new tab), qui est une requête pour lire les informations sur notre transaction à partir du hachage. Pour en savoir plus sur l'envoi de transactions, consultez notre tutoriel sur l'envoi de transactions à l'aide de Web3.
Partie 2 : Interagir avec votre contrat intelligent
Maintenant que nous avons déployé avec succès un contrat intelligent sur le réseau Goerli, apprenons à interagir avec lui.
Créer un fichier interact.js
C'est dans ce fichier que nous écrirons notre script d'interaction. Nous utiliserons la bibliothèque Ethers.js que vous avez précédemment installée dans la Partie 1.
À l'intérieur du dossier scripts/, créez un nouveau fichier nommé interact.js et ajoutez le code suivant :
1// interact.js23const API_KEY = process.env.API_KEY4const PRIVATE_KEY = process.env.PRIVATE_KEY5const CONTRACT_ADDRESS = process.env.CONTRACT_ADDRESSMettez à jour votre fichier .env
Nous allons utiliser de nouvelles variables d'environnement, nous devons donc les définir dans le fichier .env que nous avons créé précédemment.
Nous devrons ajouter une définition pour notre API_KEY Alchemy et le CONTRACT_ADDRESS où votre contrat intelligent a été déployé.
Votre fichier .env devrait ressembler à ceci :
# .envAPI_URL = "https://eth-goerli.alchemyapi.io/v2/<your-api-key>"API_KEY = "<your-api-key>"PRIVATE_KEY = "<your-metamask-private-key>"CONTRACT_ADDRESS = "0x<your contract address>"Récupérez l'ABI de votre contrat
L' de notre contrat est l'interface pour interagir avec notre contrat intelligent. Hardhat génère automatiquement un ABI et le sauvegarde dans HelloWorld.json. Pour utiliser l'ABI, nous devons en extraire le contenu en ajoutant les lignes de code suivantes à notre fichier interact.js :
1// interact.js2const contract = require("../artifacts/contracts/HelloWorld.sol/HelloWorld.json")Si vous voulez voir l'ABI, vous pouvez l'afficher dans votre console :
1console.log(JSON.stringify(contract.abi))Pour voir votre ABI s'afficher dans la console, naviguez vers votre terminal et exécutez :
npx hardhat run scripts/interact.jsCréer une instance de votre contrat
Pour interagir avec notre contrat, nous devons créer une instance de contrat dans notre code. Pour ce faire avec Ethers.js, nous devrons travailler avec trois concepts :
- Fournisseur - un fournisseur de nœud qui vous donne un accès en lecture et écriture à la blockchain
- Signataire - représente un compte Ethereum qui peut signer des transactions
- Contrat - un objet Ethers.js représentant un contrat spécifique déployé sur la chaîne
Nous utiliserons l'ABI de contrat de l'étape précédente pour créer notre instance du contrat :
1// interact.js23// Fournisseur4const alchemyProvider = new ethers.providers.AlchemyProvider(5 (network = "goerli"),6 API_KEY7)89// Signataire10const signer = new ethers.Wallet(PRIVATE_KEY, alchemyProvider)1112// Contrat13const helloWorldContract = new ethers.Contract(14 CONTRACT_ADDRESS,15 contract.abi,16 signer17)Afficher toutEn savoir plus sur les Fournisseurs, les Signataires et les Contrats dans la documentation d'ethers.js (opens in a new tab).
Lire le message d'initialisation
Vous souvenez-vous lorsque nous avons déployé notre contrat avec initMessage = "Hello world!" ? Nous allons maintenant lire ce message stocké dans notre contrat intelligent et l'afficher sur la console.
En JavaScript, des fonctions asynchrones sont utilisées lors de l'interaction avec des réseaux. Pour en savoir plus sur les fonctions asynchrones, lisez cet article sur Medium (opens in a new tab).
Utilisez le code ci-dessous pour appeler la fonction message dans notre contrat intelligent et lire le message d'initialisation :
1// interact.js23// ...45async function main() {6 const message = await helloWorldContract.message()7 console.log("Le message est : " + message)8}9main()Afficher toutAprès avoir exécuté le fichier en utilisant npx hardhat run scripts/interact.js dans le terminal, nous devrions voir cette réponse :
1Le message est : Hello world!Félicitations ! Vous venez de lire avec succès des données de contrat intelligent depuis la blockchain Ethereum, bravo !
Mettre à jour le message
Au lieu de simplement lire le message, nous pouvons également mettre à jour le message sauvegardé dans notre contrat intelligent en utilisant la fonction update ! Plutôt cool, n'est-ce pas ?
Pour mettre à jour le message, nous pouvons appeler directement la fonction update sur notre objet de Contrat instancié :
1// interact.js23// ...45async function main() {6 const message = await helloWorldContract.message()7 console.log("Le message est : " + message)89 console.log("Mise à jour du message...")10 const tx = await helloWorldContract.update("Ceci est le nouveau message.")11 await tx.wait()12}13main()Afficher toutNotez qu'à la ligne 11, nous faisons un appel à .wait() sur l'objet de transaction renvoyé. Cela garantit que notre script attend que la transaction soit minée sur la blockchain avant de quitter la fonction. Si l'appel .wait() n'est pas inclus, le script risque de ne pas voir la valeur mise à jour du message dans le contrat.
Lire le nouveau message
Vous devriez pouvoir répéter l'étape précédente pour lire la valeur du message mis à jour. Prenez un moment et voyez si vous pouvez apporter les modifications nécessaires pour afficher cette nouvelle valeur !
Si vous avez besoin d'un indice, voici à quoi devrait ressembler votre fichier interact.js à ce stade :
1// interact.js23const API_KEY = process.env.API_KEY4const PRIVATE_KEY = process.env.PRIVATE_KEY5const CONTRACT_ADDRESS = process.env.CONTRACT_ADDRESS67const contract = require("../artifacts/contracts/HelloWorld.sol/HelloWorld.json")89// fournisseur - Alchemy10const alchemyProvider = new ethers.providers.AlchemyProvider(11 (network = "goerli"),12 API_KEY13)1415// signataire - vous16const signer = new ethers.Wallet(PRIVATE_KEY, alchemyProvider)1718// instance de contrat19const helloWorldContract = new ethers.Contract(20 CONTRACT_ADDRESS,21 contract.abi,22 signer23)2425async function main() {26 const message = await helloWorldContract.message()27 console.log("Le message est : " + message)2829 console.log("Mise à jour du message...")30 const tx = await helloWorldContract.update("ceci est le nouveau message")31 await tx.wait()3233 const newMessage = await helloWorldContract.message()34 console.log("Le nouveau message est : " + newMessage)35}3637main()Afficher toutExécutez simplement le script et vous devriez pouvoir voir l'ancien message, le statut de la mise à jour, et le nouveau message affiché sur votre terminal !
npx hardhat run scripts/interact.js --network goerli
1Le message est : Hello World!2Mise à jour du message...3Le nouveau message est : Ceci est le nouveau message.Lors de l'exécution de ce script, vous remarquerez peut-être que l'étape « Mise à jour du message... » prend du temps à charger avant que le nouveau message ne s'affiche. Cela est dû au processus de minage ; si vous êtes curieux de suivre les transactions pendant qu'elles sont en cours de minage, visitez la mempool d'Alchemy (opens in a new tab) pour voir le statut d'une transaction. Si la transaction est abandonnée, il est également utile de consulter l'Etherscan de Goerli (opens in a new tab) et de rechercher le hachage de votre transaction.
Partie 3 : Publier votre contrat intelligent sur Etherscan
Vous avez fait tout le travail difficile pour donner vie à votre contrat intelligent ; il est maintenant temps de le partager avec le monde !
En vérifiant votre contrat intelligent sur Etherscan, n'importe qui peut voir votre code source et interagir avec votre contrat intelligent. Commençons !
Étape 1 : Générer une clé API sur votre compte Etherscan
Une clé API Etherscan est nécessaire pour vérifier que vous possédez le contrat intelligent que vous essayez de publier.
Si vous n'avez pas encore de compte Etherscan, inscrivez-vous pour un compte (opens in a new tab).
Une fois connecté, trouvez votre nom d'utilisateur dans la barre de navigation, survolez-le et sélectionnez le bouton Mon profil.
Sur votre page de profil, vous devriez voir une barre de navigation latérale. Depuis la barre de navigation latérale, sélectionnez Clés API. Ensuite, appuyez sur le bouton « Ajouter » pour créer une nouvelle clé API, nommez votre application hello-world et appuyez sur le bouton Créer une nouvelle clé API.
Votre nouvelle clé API devrait apparaître dans le tableau des clés API. Copiez la clé API dans votre presse-papiers.
Ensuite, nous devons ajouter la clé API d'Etherscan à notre fichier .env.
Après l'avoir ajoutée, votre fichier .env devrait ressembler à ceci :
1API_URL = "https://eth-goerli.alchemyapi.io/v2/your-api-key"2PUBLIC_KEY = "your-public-account-address"3PRIVATE_KEY = "your-private-account-address"4CONTRACT_ADDRESS = "your-contract-address"5ETHERSCAN_API_KEY = "your-etherscan-key"Contrats intelligents déployés par Hardhat
Installer hardhat-etherscan
Publier votre contrat sur Etherscan à l'aide de Hardhat est simple. Vous devrez d'abord installer le plugin hardhat-etherscan pour commencer. hardhat-etherscan vérifiera automatiquement le code source et l'ABI du contrat intelligent sur Etherscan. Pour l'ajouter, dans le répertoire hello-world, exécutez :
1npm install --save-dev @nomiclabs/hardhat-etherscanUne fois installé, incluez la déclaration suivante en haut de votre hardhat.config.js, et ajoutez les options de configuration d'Etherscan :
1// hardhat.config.js23require("dotenv").config()4require("@nomiclabs/hardhat-ethers")5require("@nomiclabs/hardhat-etherscan")67const { API_URL, PRIVATE_KEY, ETHERSCAN_API_KEY } = process.env89module.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 etherscan: {20 // Votre clé API pour Etherscan21 // Obtenez-en une sur https://etherscan.io/22 apiKey: ETHERSCAN_API_KEY,23 },24}Afficher toutVérifiez votre contrat intelligent sur Etherscan
Assurez-vous que tous les fichiers sont sauvegardés et que toutes les variables .env sont correctement configurées.
Exécutez la tâche verify, en passant l'adresse du contrat et le réseau sur lequel il est déployé :
1npx hardhat verify --network goerli DEPLOYED_CONTRACT_ADDRESS 'Hello World!'Assurez-vous que DEPLOYED_CONTRACT_ADDRESS est l'adresse de votre contrat intelligent déployé sur le réseau de test Goerli. De plus, l'argument final ('Hello World!') doit être la même valeur de chaîne que celle utilisée lors de l'étape de déploiement de la partie 1.
Si tout se passe bien, vous verrez le message suivant dans votre terminal :
1Code source du contrat soumis avec succès2contracts/HelloWorld.sol:HelloWorld at 0xdeployed-contract-address3pour vérification sur Etherscan. En attente du résultat de la vérification...456Contrat HelloWorld vérifié avec succès sur Etherscan.7https://goerli.etherscan.io/address/<contract-address>#contractsFélicitations ! Votre code de contrat intelligent est sur Etherscan !
Consultez votre contrat intelligent sur Etherscan !
Lorsque vous naviguez vers le lien fourni dans votre terminal, vous devriez pouvoir voir votre code de contrat intelligent et l'ABI publiés sur Etherscan !
Wahooo - vous l'avez fait, champion ! Désormais, n'importe qui peut appeler ou écrire dans votre contrat intelligent ! Nous sommes impatients de voir ce que vous construirez ensuite !
Partie 4 - Intégration de votre contrat intelligent avec le frontend
À la fin de ce tutoriel, vous saurez comment :
- Connecter un portefeuille MetaMask à votre dapp
- Lire des données de votre contrat intelligent en utilisant l'API Alchemy Web3 (opens in a new tab)
- Signer des transactions Ethereum en utilisant MetaMask
Pour cette dapp, nous utiliserons React (opens in a new tab) comme notre framework frontend ; cependant, il est important de noter que nous ne passerons pas beaucoup de temps à décomposer ses fondamentaux, car nous nous concentrerons principalement sur l'intégration de la fonctionnalité Web3 à notre projet.
En prérequis, vous devriez avoir une compréhension de niveau débutant de React. Sinon, nous vous recommandons de suivre le tutoriel officiel d'introduction à React (opens in a new tab).
Cloner les fichiers de démarrage
Tout d'abord, rendez-vous sur le dépôt GitHub hello-world-part-four (opens in a new tab) pour obtenir les fichiers de démarrage de ce projet et clonez ce dépôt sur votre machine locale.
Ouvrez le dépôt cloné localement. Remarquez qu'il contient deux dossiers : starter-files et completed.
starter-files- nous travaillerons dans ce répertoire, nous connecterons l'interface utilisateur à votre portefeuille Ethereum et au contrat intelligent que nous avons publié sur Etherscan dans la Partie 3.completedcontient le tutoriel complet et ne doit être utilisé que comme référence si vous êtes bloqué.
Ensuite, ouvrez votre copie de starter-files avec votre éditeur de code préféré, puis naviguez dans le dossier src.
Tout le code que nous allons écrire se trouvera dans le dossier src. Nous modifierons le composant HelloWorld.js et les fichiers JavaScript util/interact.js pour donner à notre projet des fonctionnalités Web3.
Consultez les fichiers de démarrage
Avant de commencer à coder, explorons ce qui nous est fourni dans les fichiers de départ.
Faire fonctionner votre projet React
Commençons par exécuter le projet React dans notre navigateur. La beauté de React est qu'une fois que notre projet est en cours d'exécution dans notre navigateur, toutes les modifications que nous sauvegardons seront mises à jour en direct dans notre navigateur.
Pour lancer le projet, naviguez vers le répertoire racine du dossier starter-files, et exécutez npm install dans votre terminal pour installer les dépendances du projet :
cd starter-filesnpm installUne fois l'installation terminée, exécutez npm start dans votre terminal :
npm startCela devrait ouvrir http://localhost:3000/ (opens in a new tab) dans votre navigateur, où vous verrez le frontend de notre projet. Il devrait se composer d'un champ (un endroit pour mettre à jour le message stocké dans votre contrat intelligent), d'un bouton « Connecter le portefeuille » et d'un bouton « Mettre à jour ».
Si vous essayez de cliquer sur l'un ou l'autre des boutons, vous remarquerez qu'ils ne fonctionnent pas - c'est parce que nous devons encore programmer leur fonctionnalité.
Le composant HelloWorld.js
Retournons dans le dossier src de notre éditeur et ouvrons le fichier HelloWorld.js. Il est très important de comprendre tout ce qui se trouve dans ce fichier, car c'est le composant principal de React sur lequel nous allons travailler.
En haut de ce fichier, vous remarquerez que nous avons plusieurs instructions d'importation nécessaires pour faire fonctionner notre projet, y compris la bibliothèque React, les hooks useEffect et useState, certains éléments de ./util/interact.js (nous les décrirons plus en détail bientôt !), et le logo d'Alchemy.
1// HelloWorld.js23import React from "react"4import { useEffect, useState } from "react"5import {6 helloWorldContract,7 connectWallet,8 updateMessage,9 loadCurrentMessage,10 getCurrentWalletConnected,11} from "./util/interact.js"1213import alchemylogo from "./alchemylogo.svg"Afficher toutEnsuite, nous avons nos variables d'état que nous mettrons à jour après des événements spécifiques.
1// HelloWorld.js23//Variables d'état4const [walletAddress, setWallet] = useState("")5const [status, setStatus] = useState("")6const [message, setMessage] = useState("Pas de connexion au réseau.")7const [newMessage, setNewMessage] = useState("")Voici ce que chacune des variables représente :
walletAddress: une chaîne qui stocke l'adresse du portefeuille de l'utilisateurstatus- une chaîne de caractères qui stocke un message utile guidant l'utilisateur sur la façon d'interagir avec la dappmessage- une chaîne qui stocke le message actuel dans le contrat intelligentnewMessage- une chaîne qui stocke le nouveau message qui sera écrit dans le contrat intelligent
Après les variables d'état, vous verrez cinq fonctions non implémentées : useEffect, addSmartContractListener, addWalletListener, connectWalletPressed, et onUpdatePressed. Nous expliquerons ce qu'elles font ci-dessous :
1// HelloWorld.js23//appelé une seule fois4useEffect(async () => {5 //TODO: implémenter6}, [])78function addSmartContractListener() {9 //TODO: implémenter10}1112function addWalletListener() {13 //TODO: implémenter14}1516const connectWalletPressed = async () => {17 //TODO: implémenter18}1920const onUpdatePressed = async () => {21 //TODO: implémenter22}Afficher toutuseEffect(opens in a new tab) - c'est un hook React qui est appelé après que votre composant est rendu. Comme il a un tableau vide[]passé en paramètre (voir ligne 4), il ne sera appelé qu'au premier rendu du composant. Ici, nous chargerons le message actuel stocké dans notre contrat intelligent, appellerons nos écouteurs de contrat intelligent et de portefeuille, et mettrons à jour notre interface pour refléter si un portefeuille est déjà connecté.addSmartContractListener- cette fonction configure un écouteur qui surveillera l'événementUpdatedMessagesde notre contrat HelloWorld et mettra à jour notre interface utilisateur lorsque le message sera modifié dans notre contrat intelligent.addWalletListener- cette fonction configure un écouteur qui détecte les changements dans l'état du portefeuille MetaMask de l'utilisateur, par exemple lorsque l'utilisateur déconnecte son portefeuille ou change d'adresse.connectWalletPressed- cette fonction sera appelée pour connecter le portefeuille MetaMask de l'utilisateur à notre dapp.onUpdatePressed- cette fonction sera appelée lorsque l'utilisateur souhaite mettre à jour le message stocké dans le contrat intelligent.
Vers la fin de ce fichier, nous avons l'interface utilisateur de notre composant.
1// HelloWorld.js23//l'interface utilisateur de notre composant4return (5 <div id="container">6 <img id="logo" src={alchemylogo}></img>7 <button id="walletButton" onClick={connectWalletPressed}>8 {walletAddress.length > 0 ? (9 "Connecté : " +10 String(walletAddress).substring(0, 6) +11 "..." +12 String(walletAddress).substring(38)13 ) : (14 <span>Connecter le portefeuille</span>15 )}16 </button>1718 <h2 style={{ paddingTop: "50px" }}>Message actuel :</h2>19 <p>{message}</p>2021 <h2 style={{ paddingTop: "18px" }}>Nouveau message :</h2>2223 <div>24 <input25 type="text"26 placeholder="Mettez à jour le message dans votre contrat intelligent."27 onChange={(e) => setNewMessage(e.target.value)}28 value={newMessage}29 />30 <p id="status">{status}</p>3132 <button id="publishButton" onClick={onUpdatePressed}>33 Mettre à jour34 </button>35</div>36 37</div>38)Afficher toutSi vous analysez attentivement ce code, vous remarquerez où nous utilisons nos différentes variables d'état dans notre interface utilisateur :
- Aux lignes 6-12, si le portefeuille de l'utilisateur est connecté (c'est-à-dire
walletAddress.length > 0), nous affichons une version tronquée de l'walletAddressde l'utilisateur dans le bouton avec l'ID "walletButton"; sinon, il indique simplement "Connecter le portefeuille". - À la ligne 17, nous affichons le message actuel stocké dans le contrat intelligent, qui est capturé dans la chaîne
message. - Aux lignes 23-26, nous utilisons un composant contrôlé (opens in a new tab) pour mettre à jour notre variable d'état
newMessagelorsque l'entrée dans le champ de texte change.
En plus de nos variables d'état, vous verrez également que les fonctions connectWalletPressed et onUpdatePressed sont appelées lorsque les boutons avec les ID publishButton et walletButton sont cliqués respectivement.
Enfin, voyons où ce composant HelloWorld.js est ajouté.
Si vous allez dans le fichier App.js, qui est le composant principal de React agissant comme un conteneur pour tous les autres composants, vous verrez que notre composant HelloWorld.js est injecté à la ligne 7.
Enfin et surtout, examinons un autre fichier qui vous est fourni, le fichier interact.js.
Le fichier interact.js
Parce que nous voulons adhérer au paradigme M-V-C (opens in a new tab), nous voudrons un fichier séparé qui contient toutes nos fonctions pour gérer la logique, les données et les règles de notre dapp, puis pouvoir exporter ces fonctions vers notre frontend (notre composant HelloWorld.js).
👆🏽C'est exactement le but de notre fichier interact.js !
Naviguez vers le dossier util dans votre répertoire src, et vous remarquerez que nous avons inclus un fichier appelé interact.js qui contiendra toutes nos fonctions et variables d'interaction avec le contrat intelligent et le portefeuille.
1// interact.js23//export const helloWorldContract;45export const loadCurrentMessage = async () => {}67export const connectWallet = async () => {}89const getCurrentWalletConnected = async () => {}1011export const updateMessage = async (message) => {}Afficher toutVous remarquerez en haut du fichier que nous avons commenté l'objet helloWorldContract. Plus tard dans ce tutoriel, nous décommenterons cet objet et instancierons notre contrat intelligent dans cette variable, que nous exporterons ensuite dans notre composant HelloWorld.js.
Les quatre fonctions non implémentées après notre objet helloWorldContract font ce qui suit :
loadCurrentMessage- cette fonction gère la logique de chargement du message actuel stocké dans le contrat intelligent. Elle effectuera un appel en lecture au contrat intelligent Hello World en utilisant l'API Alchemy Web3 (opens in a new tab).connectWallet- cette fonction connectera le MetaMask de l'utilisateur à notre dapp.getCurrentWalletConnected- cette fonction vérifiera si un compte Ethereum est déjà connecté à notre dapp lors du chargement de la page et mettra à jour notre interface utilisateur en conséquence.updateMessage- cette fonction mettra à jour le message stocké dans le contrat intelligent. Elle effectuera un appel en écriture au contrat intelligent Hello World, donc le portefeuille MetaMask de l'utilisateur devra signer une transaction Ethereum pour mettre à jour le message.
Maintenant que nous comprenons avec quoi nous travaillons, voyons comment lire notre contrat intelligent !
Étape 3 : Lire à partir de votre contrat intelligent
Pour lire à partir de votre contrat intelligent, vous devrez configurer avec succès :
- Une connexion API à la chaîne Ethereum
- Une instance chargée de votre contrat intelligent
- Une fonction pour appeler la fonction de votre contrat intelligent
- Un écouteur pour surveiller les mises à jour lorsque les données que vous lisez à partir du contrat intelligent changent
Cela peut sembler beaucoup d'étapes, mais ne vous inquiétez pas ! Nous allons vous guider étape par étape ! :)
Établir une connexion API à la chaîne Ethereum
Alors, souvenez-vous comment, dans la partie 2 de ce tutoriel, nous avons utilisé notre clé Alchemy Web3 pour lire notre contrat intelligent (opens in a new tab) ? Vous aurez également besoin d'une clé Alchemy Web3 dans votre dapp pour lire à partir de la chaîne.
Si vous ne l'avez pas déjà, installez d'abord Alchemy Web3 (opens in a new tab) en naviguant vers le répertoire racine de vos starter-files et en exécutant la commande suivante dans votre terminal :
1npm install @alch/alchemy-web3Alchemy Web3 (opens in a new tab) est un wrapper autour de Web3.js (opens in a new tab), qui fournit des méthodes API améliorées et d'autres avantages cruciaux pour vous faciliter la vie en tant que développeur web3. Il est conçu pour nécessiter une configuration minimale afin que vous puissiez commencer à l'utiliser immédiatement dans votre application !
Ensuite, installez le paquet dotenv (opens in a new tab) dans le répertoire de votre projet, afin que nous ayons un endroit sécurisé pour stocker notre clé API après l'avoir récupérée.
1npm install dotenv --savePour notre dapp, nous utiliserons notre clé API Websockets au lieu de notre clé API HTTP, car cela nous permettra de mettre en place un écouteur qui détecte lorsque le message stocké dans le contrat intelligent change.
Une fois que vous avez votre clé API, créez un fichier .env dans votre répertoire racine et ajoutez-y votre URL Websockets Alchemy. Ensuite, votre fichier .env devrait ressembler à ceci :
1REACT_APP_ALCHEMY_KEY = wss://eth-goerli.ws.alchemyapi.io/v2/<key>Maintenant, nous sommes prêts à configurer notre point de terminaison Alchemy Web3 dans notre dapp ! Retournons à notre interact.js, qui est imbriqué dans notre dossier util, et ajoutons le code suivant en haut du fichier :
1// interact.js23require("dotenv").config()4const alchemyKey = process.env.REACT_APP_ALCHEMY_KEY5const { createAlchemyWeb3 } = require("@alch/alchemy-web3")6const web3 = createAlchemyWeb3(alchemyKey)78//export const helloWorldContract;Ci-dessus, nous avons d'abord importé la clé Alchemy depuis notre fichier .env, puis nous avons passé notre alchemyKey à createAlchemyWeb3 pour établir notre point de terminaison Alchemy Web3.
Avec ce point de terminaison prêt, il est temps de charger notre contrat intelligent !
Chargement de votre contrat intelligent Hello World
Pour charger votre contrat intelligent Hello World, vous aurez besoin de son adresse de contrat et de son ABI, qui peuvent tous deux être trouvés sur Etherscan si vous avez terminé la Partie 3 de ce tutoriel.
Comment obtenir l'ABI de votre contrat depuis Etherscan
Si vous avez sauté la partie 3 de ce tutoriel, vous pouvez utiliser le contrat HelloWorld avec l'adresse 0x6f3f635A9762B47954229Ea479b4541eAF402A6A (opens in a new tab). Son ABI peut être trouvé ici (opens in a new tab).
L'ABI d'un contrat est nécessaire pour spécifier quelle fonction un contrat invoquera, ainsi que pour s'assurer que la fonction renverra des données dans le format que vous attendez. Une fois que nous avons copié l'ABI de notre contrat, sauvegardons-le en tant que fichier JSON appelé contract-abi.json dans votre répertoire src.
Votre contract-abi.json doit être stocké dans votre dossier src.
Armés de l'adresse de notre contrat, de l'ABI et du point de terminaison Alchemy Web3, nous pouvons utiliser la méthode de contrat (opens in a new tab) pour charger une instance de notre contrat intelligent. Importez l'ABI de votre contrat dans le fichier interact.js et ajoutez l'adresse de votre contrat.
1// interact.js23const contractABI = require("../contract-abi.json")4const contractAddress = "0x6f3f635A9762B47954229Ea479b4541eAF402A6A"Nous pouvons maintenant enfin décommenter notre variable helloWorldContract et charger le contrat intelligent en utilisant notre point de terminaison AlchemyWeb3 :
1// interact.js2export const helloWorldContract = new web3.eth.Contract(3 contractABI,4 contractAddress5)Pour récapituler, les 12 premières lignes de votre interact.js devraient maintenant ressembler à ceci :
1// interact.js23require("dotenv").config()4const alchemyKey = process.env.REACT_APP_ALCHEMY_KEY5const { createAlchemyWeb3 } = require("@alch/alchemy-web3")6const web3 = createAlchemyWeb3(alchemyKey)78const contractABI = require("../contract-abi.json")9const contractAddress = "0x6f3f635A9762B47954229Ea479b4541eAF402A6A"1011export const helloWorldContract = new web3.eth.Contract(12 contractABI,13 contractAddress14)Afficher toutMaintenant que notre contrat est chargé, nous pouvons implémenter notre fonction loadCurrentMessage !
Implémentation de loadCurrentMessage dans votre fichier interact.js
Cette fonction est très simple. Nous allons faire un simple appel web3 asynchrone pour lire notre contrat. Notre fonction renverra le message stocké dans le contrat intelligent :
Mettez à jour la fonction loadCurrentMessage dans votre fichier interact.js comme suit :
1// interact.js23export const loadCurrentMessage = async () => {4 const message = await helloWorldContract.methods.message().call()5 return message6}Puisque nous voulons afficher ce contrat intelligent dans notre interface utilisateur, mettons à jour la fonction useEffect dans notre composant HelloWorld.js comme suit :
1// HelloWorld.js23//appelé une seule fois4useEffect(async () => {5 const message = await loadCurrentMessage()6 setMessage(message)7}, [])Notez que nous voulons que notre loadCurrentMessage ne soit appelé qu'une seule fois lors du premier rendu du composant. Nous allons bientôt implémenter addSmartContractListener pour mettre à jour automatiquement l'interface utilisateur après que le message dans le contrat intelligent a changé.
Avant de nous plonger dans notre écouteur, voyons ce que nous avons jusqu'à présent ! Enregistrez vos fichiers HelloWorld.js et interact.js, puis allez sur http://localhost:3000/ (opens in a new tab)
Vous remarquerez que le message actuel ne dit plus « Pas de connexion au réseau ». Au lieu de cela, il reflète le message stocké dans le contrat intelligent. Génial !
Votre interface utilisateur devrait maintenant refléter le message stocké dans le contrat intelligent
Maintenant, en parlant de cet écouteur...
Implémenter addSmartContractListener
Si vous repensez au fichier HelloWorld.sol que nous avons écrit dans la Partie 1 de cette série de tutoriels (opens in a new tab), vous vous souviendrez qu'il y a un événement de contrat intelligent appelé UpdatedMessages qui est émis après l'invocation de la fonction update de notre contrat intelligent (voir lignes 9 et 27) :
1// HelloWorld.sol23// Spécifie la version de Solidity, en utilisant le versionnage sémantique.4// En savoir plus : https://solidity.readthedocs.io/en/v0.5.10/layout-of-source-files.html#pragma5pragma solidity ^0.7.3;67// Définit un contrat nommé `HelloWorld`.8// Un contrat est une collection de fonctions et de données (son état). Une fois déployé, un contrat réside à une adresse spécifique sur la blockchain Ethereum. En savoir plus : https://solidity.readthedocs.io/en/v0.5.10/structure-of-a-contract.html9contract HelloWorld {1011 //Émis lorsque la fonction de mise à jour est appelée12 //Les événements de contrat intelligent sont un moyen pour votre contrat de communiquer que quelque chose s'est passé sur la blockchain à votre interface d'application, qui peut « écouter » certains événements et prendre des mesures lorsqu'ils se produisent.13 event UpdatedMessages(string oldStr, string newStr);1415 // Déclare une variable d'état `message` de type `string`.16 // Les variables d'état sont des variables dont les valeurs sont stockées en permanence dans le stockage du contrat. Le mot-clé `public` rend les variables accessibles depuis l'extérieur d'un contrat et crée une fonction que d'autres contrats ou clients peuvent appeler pour accéder à la valeur.17 string public message;1819 // Semblable à de nombreux langages orientés objet basés sur les classes, un constructeur est une fonction spéciale qui n'est exécutée qu'à la création du contrat.20 // Les constructeurs sont utilisés pour initialiser les données du contrat. En savoir plus : https://solidity.readthedocs.io/en/v0.5.10/contracts.html#constructors21 constructor(string memory initMessage) {2223 // Accepte un argument de chaîne `initMessage` et définit la valeur dans la variable de stockage `message` du contrat).24 message = initMessage;25 }2627 // Une fonction publique qui accepte un argument de chaîne et met à jour la variable de stockage `message`.28 function update(string memory newMessage) public {29 string memory oldMsg = message;30 message = newMessage;31 emit UpdatedMessages(oldMsg, newMessage);32 }33}Afficher toutLes événements de contrat intelligent sont un moyen pour votre contrat de communiquer qu'un événement s'est produit (c'est-à-dire qu'il y a eu un événement) sur la blockchain à votre application frontale, qui peut être à l'« écoute » d'événements spécifiques et prendre des mesures lorsqu'ils se produisent.
La fonction addSmartContractListener va spécifiquement écouter l'événement UpdatedMessages de notre contrat intelligent Hello World et mettre à jour notre interface utilisateur pour afficher le nouveau message.
Modifiez addSmartContractListener comme suit :
1// HelloWorld.js23function addSmartContractListener() {4 helloWorldContract.events.UpdatedMessages({}, (error, data) => {5 if (error) {6 setStatus("😥 " + error.message)7 } else {8 setMessage(data.returnValues[1])9 setNewMessage("")10 setStatus("🎉 Votre message a été mis à jour !")11 }12 })13}Afficher toutAnalysons ce qui se passe lorsque l'écouteur détecte un événement :
- Si une erreur se produit lorsque l'événement est émis, elle sera reflétée dans l'interface utilisateur via notre variable d'état
status. - Sinon, nous utiliserons l'objet
datarenvoyé. L'objetdata.returnValuesest un tableau indexé à zéro où le premier élément du tableau contient le message précédent et le second le message mis à jour. En somme, lors d'un événement réussi, nous définirons notre chaînemessagesur le message mis à jour, effacerons la chaînenewMessageet mettrons à jour notre variable d'étatstatuspour indiquer qu'un nouveau message a été publié sur notre contrat intelligent.
Enfin, appelons notre écouteur dans notre fonction useEffect afin qu'il soit initialisé lors du premier rendu du composant HelloWorld.js. Globalement, votre fonction useEffect devrait ressembler à ceci :
1// HelloWorld.js23useEffect(async () => {4 const message = await loadCurrentMessage()5 setMessage(message)6 addSmartContractListener()7}, [])Maintenant que nous pouvons lire notre contrat intelligent, il serait formidable de savoir comment y écrire aussi ! Cependant, pour écrire dans notre dapp, nous devons d'abord y connecter un portefeuille Ethereum.
Donc, nous allons ensuite nous occuper de la configuration de notre portefeuille Ethereum (MetaMask) puis de sa connexion à notre dapp !
Étape 4 : Configurer votre portefeuille Ethereum
Pour écrire quoi que ce soit sur la chaîne Ethereum, les utilisateurs doivent signer des transactions à l'aide des clés privées de leur portefeuille virtuel. Pour ce tutoriel, nous utiliserons MetaMask (opens in a new tab), un portefeuille virtuel dans le navigateur utilisé pour gérer l'adresse de votre compte Ethereum, car il rend cette signature de transaction très facile pour l'utilisateur final.
Si vous voulez mieux comprendre le fonctionnement des transactions sur Ethereum, consultez cette page de la Fondation Ethereum.
Télécharger MetaMask
Vous pouvez télécharger et créer un compte MetaMask gratuitement ici (opens in a new tab). Lorsque vous créez un compte, ou si vous en avez déjà un, assurez-vous de basculer vers le « Réseau de test Goerli » en haut à droite (afin de ne pas utiliser d'argent réel).
Ajouter de l'ether depuis un robinet
Pour signer une transaction sur la blockchain Ethereum, nous aurons besoin de faux Eth. Pour obtenir de l'Eth, vous pouvez vous rendre sur FaucETH (opens in a new tab), saisir l'adresse de votre compte Goerli, cliquer sur « Request funds », puis sélectionner « Ethereum Testnet Goerli » dans le menu déroulant et enfin cliquer à nouveau sur le bouton « Request funds ». Vous devriez voir les ETH dans votre compte MetaMask peu de temps après !
Vérifiez votre solde
Pour vérifier que notre solde est bien là, faisons une requête eth_getBalance (opens in a new tab) en utilisant l'outil de composition d'Alchemy (opens in a new tab). Cela va retourner la quantité d'ETH que contient votre portefeuille. Après avoir entré l'adresse de votre compte MetaMask et cliqué sur « Send Request », vous devriez voir une réponse comme celle-ci :
1{"jsonrpc": "2.0", "id": 0, "result": "0xde0b6b3a7640000"}REMARQUE : ce résultat est en wei et non en eth. Le wei est utilisé comme la plus petite dénomination d'ether. La conversion de wei vers eth est : 1 eth = 10¹⁸ wei. Donc si on convertit 0xde0b6b3a7640000 en nombre décimal, nous obtenons 1*10¹⁸ ce qui correspond à 1 eth.
Ouf ! Notre faux argent est bien là ! 🤑
Étape 5 : Connecter MetaMask à votre interface utilisateur
Maintenant que notre portefeuille MetaMask est configuré, connectons-y notre dApp !
La fonction connectWallet
Dans notre fichier interact.js, implémentons la fonction connectWallet, que nous pourrons ensuite appeler dans notre composant HelloWorld.js.
Modifions connectWallet comme suit :
1// interact.js23export const connectWallet = async () => {4 if (window.ethereum) {5 try {6 const addressArray = await window.ethereum.request({7 method: "eth_requestAccounts",8 })9 const obj = {10 status: "👆🏽 Écrivez un message dans le champ de texte ci-dessus.",11 address: addressArray[0],12 }13 return obj14 } catch (err) {15 return {16 address: "",17 status: "😥 " + err.message,18 }19 }20 } else {21 return {22 address: "",23 status: (24 <span>25 <p>26 {" "}27 🦊 <a target="_blank" href={`https://metamask.io/download`}>28 Vous devez installer MetaMask, un portefeuille Ethereum virtuel, dans votre29 navigateur.30 </a>31 </p>32 </span>33 ),34 }35 }36}Afficher toutAlors, que fait exactement ce bloc de code géant ?
Eh bien, d'abord, il vérifie si window.ethereum est activé dans votre navigateur.
window.ethereum est une API globale injectée par MetaMask et d'autres fournisseurs de portefeuilles qui permet aux sites Web de demander les comptes Ethereum des utilisateurs. S'il est approuvé, il peut lire des données des blockchains auxquelles l'utilisateur est connecté, et suggérer à l'utilisateur de signer des messages et des transactions. Consultez la documentation de MetaMask (opens in a new tab) pour plus d'informations !
Si window.ethereum n'est pas présent, cela signifie que MetaMask n'est pas installé. Cela se traduit par le renvoi d'un objet JSON, où l'address renvoyée est une chaîne vide, et où l'objet status JSX relaie que l'utilisateur doit installer MetaMask.
Maintenant, si window.ethereum est présent, c'est là que les choses deviennent intéressantes.
À l'aide d'une boucle try/catch, nous allons essayer de nous connecter à MetaMask en appelant window.ethereum.request({ method: "eth_requestAccounts" }); (opens in a new tab). L'appel de cette fonction ouvrira MetaMask dans le navigateur, où l'utilisateur sera invité à connecter son portefeuille à votre dApp.
- Si l'utilisateur choisit de se connecter,
method: "eth_requestAccounts"renverra un tableau contenant toutes les adresses de compte de l'utilisateur connectées à la dapp. Au total, notre fonctionconnectWalletrenverra un objet JSON qui contient la premièreadressede ce tableau (voir ligne 9) et un messaged'étatqui invite l'utilisateur à écrire un message au contrat intelligent. - Si l'utilisateur rejette la connexion, l'objet JSON contiendra une chaîne vide pour l'
addressrenvoyée et un messaged'étatindiquant que l'utilisateur a rejeté la connexion.
Maintenant que nous avons écrit cette fonction connectWallet, la prochaine étape est de l'appeler dans notre composant HelloWorld.js.
Ajouter la fonction connectWallet à votre composant d'interface HelloWorld.js
Naviguez vers la fonction connectWalletPressed dans HelloWorld.js, et mettez-la à jour comme suit :
1// HelloWorld.js23const connectWalletPressed = async () => {4 const walletResponse = await connectWallet()5 setStatus(walletResponse.status)6 setWallet(walletResponse.address)7}Remarquez comment la plupart de nos fonctionnalités sont abstraites de notre composant HelloWorld.js à partir du fichier interact.js ? C'est ainsi que nous respectons le paradigme M-V-C !
Dans connectWalletPressed, nous effectuons simplement un appel en attente vers notre fonction importée connectWallet et, à l'aide de sa réponse, nous mettons à jour nos variables status et walletAddress via leurs hooks d'état.
Maintenant, enregistrons les deux fichiers (HelloWorld.js et interact.js) et testons notre interface utilisateur jusqu'à présent.
Ouvrez votre navigateur sur la page http://localhost:3000/ (opens in a new tab), et appuyez sur le bouton « Connecter le portefeuille » en haut à droite de la page.
Si MetaMask est installé, vous devriez être invité à connecter votre portefeuille à votre dApp. Acceptez l'invitation à se connecter.
Vous devriez voir que le bouton du portefeuille indique maintenant que votre adresse est connectée ! Yessss 🔥
Ensuite, essayez de rafraîchir la page... c'est étrange. Notre bouton de portefeuille nous invite à connecter MetaMask bien qu'il soit déjà connecté...
Cependant, n'ayez crainte ! Nous pouvons facilement résoudre ce problème en implémentant getCurrentWalletConnected, qui vérifiera si une adresse est déjà connectée à notre dapp et mettra à jour notre interface utilisateur en conséquence !
La fonction getCurrentWalletConnected
Mettez à jour votre fonction getCurrentWalletConnected dans le fichier interact.js comme suit :
1// interact.js23export const getCurrentWalletConnected = async () => {4 if (window.ethereum) {5 try {6 const addressArray = await window.ethereum.request({7 method: "eth_accounts",8 })9 if (addressArray.length > 0) {10 return {11 address: addressArray[0],12 status: "👆🏽 Écrivez un message dans le champ de texte ci-dessus.",13 }14 } else {15 return {16 address: "",17 status: "🦊 Connectez-vous à MetaMask en utilisant le bouton en haut à droite.",18 }19 }20 } catch (err) {21 return {22 address: "",23 status: "😥 " + err.message,24 }25 }26 } else {27 return {28 address: "",29 status: (30 <span>31 <p>32 {" "}33 🦊 <a target="_blank" href={`https://metamask.io/download`}>34 Vous devez installer MetaMask, un portefeuille Ethereum virtuel, dans votre35 navigateur.36 </a>37 </p>38 </span>39 ),40 }41 }42}Afficher toutCe code est très similaire à la fonction connectWallet que nous venons d'écrire à l'étape précédente.
La principale différence est qu'au lieu d'appeler la méthode eth_requestAccounts, qui ouvre MetaMask pour que l'utilisateur connecte son portefeuille, nous appelons ici la méthode eth_accounts, qui renvoie simplement un tableau contenant les adresses MetaMask actuellement connectées à notre dapp.
Pour voir cette fonction en action, appelons-la dans notre fonction useEffect de notre composant HelloWorld.js :
1// HelloWorld.js23useEffect(async () => {4 const message = await loadCurrentMessage()5 setMessage(message)6 addSmartContractListener()78 const { address, status } = await getCurrentWalletConnected()9 setWallet(address)10 setStatus(status)11}, [])Afficher toutNotez que nous utilisons la réponse de notre appel à getCurrentWalletConnected pour mettre à jour nos variables d'état walletAddress et status.
Maintenant que vous avez ajouté ce code, essayons de rafraîchir la fenêtre de notre navigateur.
Géniaaaal ! Le bouton devrait indiquer que vous êtes connecté et afficher un aperçu de l'adresse de votre portefeuille connecté, même après avoir été actualisé !
Implémenter addWalletListener
La dernière étape de la configuration de notre dApp de portefeuille consiste à mettre en place le listener de portefeuille afin que notre interface utilisateur soit mise à jour lorsque l'état de notre portefeuille change, par exemple lorsque l'utilisateur se déconnecte ou change de compte.
Dans votre fichier HelloWorld.js, modifiez votre fonction addWalletListener comme suit :
1// HelloWorld.js23function addWalletListener() {4 if (window.ethereum) {5 window.ethereum.on("accountsChanged", (accounts) => {6 if (accounts.length > 0) {7 setWallet(accounts[0])8 setStatus("👆🏽 Écrivez un message dans le champ de texte ci-dessus.")9 } else {10 setWallet("")11 setStatus("🦊 Connectez-vous à MetaMask en utilisant le bouton en haut à droite.")12 }13 })14 } else {15 setStatus(16 <p>17 {" "}18 🦊 <a target="_blank" href={`https://metamask.io/download`}>19 Vous devez installer MetaMask, un portefeuille Ethereum virtuel, dans votre navigateur.20 </a>21 </p>22 )23 }24}Afficher toutJe parie que vous n'avez même pas besoin de notre aide pour comprendre ce qui se passe ici à ce stade, mais pour des raisons d'exhaustivité, décomposons rapidement :
- Tout d'abord, notre fonction vérifie si
window.ethereumest activé (c'est-à-dire si MetaMask est installé).- Si ce n'est pas le cas, nous définissons simplement notre variable d'état
statussur une chaîne JSX qui invite l'utilisateur à installer MetaMask. - S'il est activé, nous configurons l'écouteur
window.ethereum.on("accountsChanged")à la ligne 3 qui écoute les changements d'état dans le portefeuille MetaMask, qui incluent le moment où l'utilisateur connecte un compte supplémentaire à la dapp, change de compte ou déconnecte un compte. S'il y a au moins un compte connecté, la variable d'étatwalletAddressest mise à jour en tant que premier compte dans le tableauaccountsrenvoyé par l'écouteur. Sinon,walletAddressest défini comme une chaîne vide.
- Si ce n'est pas le cas, nous définissons simplement notre variable d'état
Enfin et surtout, nous devons l'appeler dans notre fonction useEffect :
1// HelloWorld.js23useEffect(async () => {4 const message = await loadCurrentMessage()5 setMessage(message)6 addSmartContractListener()78 const { address, status } = await getCurrentWalletConnected()9 setWallet(address)10 setStatus(status)1112 addWalletListener()13}, [])Afficher toutEt c'est tout ! Nous avons terminé avec succès la programmation de toutes les fonctionnalités de notre portefeuille ! Passons maintenant à notre dernière tâche : mettre à jour le message stocké dans notre contrat intelligent !
Étape 6 : Implémenter la fonction updateMessage
C'est parti, nous sommes dans la dernière ligne droite ! Dans la fonction updateMessage de votre fichier interact.js, nous allons faire ce qui suit :
- S'assurer que le message que nous souhaitons publier dans notre contact intelligent est valide
- Signer notre transaction en utilisant MetaMask
- Appeler cette fonction depuis notre composant frontend
HelloWorld.js
Cela ne prendra pas très longtemps ; finissons cette dapp !
Gestion des erreurs de saisie
Naturellement, il est logique d'avoir une sorte de gestion des erreurs de saisie au début de la fonction.
Nous voudrons que notre fonction se termine prématurément s'il n'y a pas d'extension MetaMask installée, si aucun portefeuille n'est connecté (c'est-à-dire que l'address transmise est une chaîne vide), ou si le message est une chaîne vide. Ajoutons la gestion des erreurs suivante à updateMessage :
1// interact.js23export const updateMessage = async (address, message) => {4 if (!window.ethereum || address === null) {5 return {6 status:7 "💡 Connectez votre portefeuille MetaMask pour mettre à jour le message sur la blockchain.",8 }9 }1011 if (message.trim() === "") {12 return {13 status: "❌ Votre message ne peut pas être une chaîne vide.",14 }15 }16}Afficher toutMaintenant que nous avons une bonne gestion des erreurs de saisie, il est temps de signer la transaction via MetaMask !
Signer notre transaction
Si vous êtes déjà à l'aise avec les transactions Ethereum traditionnelles de web3, le code que nous écrirons ensuite vous sera très familier. Sous votre code de gestion des erreurs de saisie, ajoutez ce qui suit à updateMessage :
1// interact.js23//configurer les paramètres de la transaction4const transactionParameters = {5 to: contractAddress, // Requis sauf lors de la publication de contrats.6 from: address, // doit correspondre à l'adresse active de l'utilisateur.7 data: helloWorldContract.methods.update(message).encodeABI(),8}910//signer la transaction11try {12 const txHash = await window.ethereum.request({13 method: "eth_sendTransaction",14 params: [transactionParameters],15 })16 return {17 status: (18 <span>19 ✅{" "}20 <a target="_blank" href={`https://goerli.etherscan.io/tx/${txHash}`}>21 Consultez l'état de votre transaction sur Etherscan !22 </a>23 <br />24 ℹ️ Une fois la transaction vérifiée par le réseau, le message sera25 mis à jour automatiquement.26 </span>27 ),28 }29} catch (error) {30 return {31 status: "😥 " + error.message,32 }33}Afficher toutAnalysons ce qui se passe. Tout d'abord, nous configurons les paramètres de nos transactions, où :
tospécifie l'adresse du destinataire (notre contrat intelligent)fromspécifie le signataire de la transaction, la variableaddressque nous avons passée dans notre fonctiondatacontient l'appel à la méthodeupdatede notre contrat intelligent Hello World, recevant notre variable de chaînemessageen entrée
Ensuite, nous effectuons un appel await, window.ethereum.request, où nous demandons à MetaMask de signer la transaction. Remarquez, aux lignes 11 et 12, nous spécifions notre méthode eth, eth_sendTransaction, et nous passons nos transactionParameters.
À ce stade, MetaMask s'ouvrira dans le navigateur, et demandera à l'utilisateur de signer ou rejeter la transaction.
- Si la transaction réussit, la fonction renverra un objet JSON où la chaîne JSX
statusinvite l'utilisateur à consulter Etherscan pour plus d'informations sur sa transaction. - Si la transaction échoue, la fonction renverra un objet JSON où la chaîne
statusrelaie le message d'erreur.
Globalement, notre fonction updateMessage devrait ressembler à ceci :
1// interact.js23export const updateMessage = async (address, message) => {4 //gestion des erreurs de saisie5 if (!window.ethereum || address === null) {6 return {7 status:8 "💡 Connectez votre portefeuille MetaMask pour mettre à jour le message sur la blockchain.",9 }10 }1112 if (message.trim() === "") {13 return {14 status: "❌ Votre message ne peut pas être une chaîne vide.",15 }16 }1718 //configurer les paramètres de la transaction19 const transactionParameters = {20 to: contractAddress, // Requis sauf lors de la publication de contrats.21 from: address, // doit correspondre à l'adresse active de l'utilisateur.22 data: helloWorldContract.methods.update(message).encodeABI(),23 }2425 //signer la transaction26 try {27 const txHash = await window.ethereum.request({28 method: "eth_sendTransaction",29 params: [transactionParameters],30 })31 return {32 status: (33 <span>34 ✅{" "}35 <a target="_blank" href={`https://goerli.etherscan.io/tx/${txHash}`}>36 Consultez l'état de votre transaction sur Etherscan !37 </a>38 <br />39 ℹ️ Une fois la transaction vérifiée par le réseau, le message sera40 mis à jour automatiquement.41 </span>42 ),43 }44 } catch (error) {45 return {46 status: "😥 " + error.message,47 }48 }49}Afficher toutEnfin et surtout, nous devons connecter notre fonction updateMessage à notre composant HelloWorld.js.
Connecter updateMessage au frontend HelloWorld.js
Notre fonction onUpdatePressed devrait faire un appel await à la fonction updateMessage importée et modifier la variable d'état status pour refléter si notre transaction a réussi ou échoué :
1// HelloWorld.js23const onUpdatePressed = async () => {4 const { status } = await updateMessage(walletAddress, newMessage)5 setStatus(status)6}C'est super propre et simple. Et devinez quoi... VOTRE DAPP EST TERMINÉE !!!
Allez-y et testez le bouton Mettre à jour !
Créez votre propre dapp personnalisée
Wooooo, vous êtes arrivé à la fin du tutoriel ! Pour récapituler, vous avez appris à :
- Connecter un portefeuille MetaMask à votre projet de dapp
- Lire des données de votre contrat intelligent en utilisant l'API Alchemy Web3 (opens in a new tab)
- Signer des transactions Ethereum en utilisant MetaMask
Maintenant, vous êtes pleinement équipé pour appliquer les compétences de ce tutoriel afin de créer votre propre projet de dapp personnalisé ! Comme toujours, si vous avez des questions, n'hésitez pas à nous contacter pour obtenir de l'aide sur le Discord d'Alchemy (opens in a new tab). 🧙♂️
Une fois que vous avez terminé ce tutoriel, faites-nous savoir comment s'est passée votre expérience ou si vous avez des commentaires en nous identifiant sur Twitter @alchemyplatform (opens in a new tab) !
Dernière mise à jour de la page : 26 février 2026





