Integrating Hardhat with IPFS (InterPlanetary File System) allows developers to store and retrieve files in a decentralized manner. This is particularly useful for decentralized applications (dApps) that require off-chain storage, such as images, documents, or any other type of data. This guide will show you how to upload files to IPFS and link them to your smart contracts using Hardhat.

Prerequisites

  • Basic knowledge of JavaScript and Solidity.
  • A Hardhat project set up with smart contracts.
  • Node.js installed on your machine.
  • An IPFS node or a service like Infura or Pinata for IPFS.

Setting Up Your Hardhat Project

If you haven't set up a Hardhat project yet, you can do so by following these steps:

mkdir my-hardhat-ipfs-project
cd my-hardhat-ipfs-project
npm init --yes
npm install --save-dev hardhat
npx hardhat

Create a simple smart contract in the contracts directory, such as IPFSStorage.sol:

pragma solidity ^0.8.0;

contract IPFSStorage {
string public ipfsHash;

function setIPFSHash(string memory _ipfsHash) public {
ipfsHash = _ipfsHash;
}

function getIPFSHash() public view returns (string memory) {
return ipfsHash;
}
}

Installing IPFS Libraries

To interact with IPFS, you can use the ipfs-http-client library. Install it in your Hardhat project:

npm install ipfs-http-client

Uploading Files to IPFS

Create a new JavaScript file in the scripts directory named uploadToIPFS.js. This script will upload a file to IPFS and interact with your smart contract:

const { create } = require("ipfs-http-client");
const { ethers } = require("hardhat");

// Connect to IPFS
const ipfs = create({ url: "https://ipfs.infura.io:5001/api/v0" }); // Using Infura as an IPFS provider

async function main() {
// Upload a file to IPFS
const file = await ipfs.add({ path: "example.txt", content: "Hello, IPFS!" });
console.log("Uploaded file to IPFS with hash:", file.path);

// Get the deployed contract
const IPFSStorage = await ethers.getContractFactory("IPFSStorage");
const ipfsStorage = await IPFSStorage.deploy();
await ipfsStorage.deployed();

// Store the IPFS hash in the smart contract
const tx = await ipfsStorage.setIPFSHash(file.path);
await tx.wait();

console.log("Stored IPFS hash in contract:", await ipfsStorage.getIPFSHash());
}

main()
.then(() => process.exit(0))
.catch((error) => {
console.error(error);
process.exit(1);
});

Running the Script

Before running the script, make sure your Hardhat local network is running:

npx hardhat node

Then, run the script to upload a file to IPFS and store the IPFS hash in the smart contract:

npx hardhat run scripts/uploadToIPFS.js --network localhost

Retrieving Data from IPFS

To retrieve the file from IPFS using the hash stored in your smart contract, you can create another script. Create a new JavaScript file in the scripts directory named retrieveFromIPFS.js:

const { create } = require("ipfs-http-client");
const { ethers } = require("hardhat");

// Connect to IPFS
const ipfs = create({ url: "https://ipfs.infura.io:5001/api/v0" });

async function main() {
// Get the deployed contract
const IPFSStorage = await ethers.getContractFactory("IPFSStorage");
const ipfsStorage = await IPFSStorage.attach("your_contract_address"); // Replace with your deployed contract address

// Retrieve the IPFS hash from the contract
const ipfsHash = await ipfsStorage.getIPFSHash();
console.log("Retrieved IPFS hash from contract:", ipfsHash);

// Fetch the file from IPFS
const stream = ipfs.cat(ipfsHash);
let data = '';

for await (const chunk of stream) {
data += chunk.toString();
}

console.log("Retrieved file content from IPFS:", data);
}

main()
.then(() => process.exit(0))
.catch((error) => {
console.error(error);
process.exit(1);
});

Conclusion

By integrating Hardhat with IPFS, you can effectively store and retrieve files in a decentralized manner. This setup is essential for dApps that require off-chain storage solutions. With the provided scripts, you can upload files to IPFS, store their hashes in smart contracts, and retrieve the files when needed, ensuring a seamless experience in decentralized application development.