Introduction¶
Polkadot Hub supports two execution paths for running smart contracts: PVM (which compiles Solidity to the Polkadot Virtual Machine via the revive compiler) and EVM (powered by REVM, a Rust implementation of the Ethereum Virtual Machine, which runs standard EVM bytecode with zero modifications). This tutorial follows the EVM path.
With EVM, you deploy the same unmodified Solidity contracts using the same standard Hardhat toolchain you already know. No special compiler plugins, no contract rewrites, and no porting effort. If your project compiles with vanilla Hardhat, it runs on Polkadot Hub through EVM.
This tutorial walks you through cloning, compiling, testing, and deploying Uniswap V2 on Polkadot Hub using Hardhat and TypeScript. By the end, you will have a fully functioning UniswapV2Factory contract deployed to the Polkadot Hub TestNet.
Prerequisites¶
Before starting, make sure you have:
- Node.js v22.0.0 or later and npm installed
- Basic understanding of Solidity and TypeScript
- Familiarity with the Hardhat development environment
- Some test tokens to cover transaction fees, obtained from the Polkadot faucet (see Get Test Tokens for a guide to using the faucet)
- A wallet with a private key for signing transactions
- Basic understanding of how AMMs and liquidity pools work
Set Up the Project¶
Start by cloning the EVM Hardhat examples repository, which contains the Uniswap V2 Core project with a standard Hardhat and TypeScript configuration:
-
Clone the repository and navigate to the Uniswap V2 project:
-
Install the required dependencies:
-
Compile the contracts:
If the compilation is successful, you should see output similar to the following:
npx hardhat compile Downloading solc 0.5.16 Compiling 12 Solidity files Successfully compiled 12 Solidity filesAfter running this command, the compiled artifacts (ABI and bytecode) appear in the
artifactsdirectory.
Configure Secure Key Management¶
This project uses Hardhat Configuration Variables to manage private keys securely. Unlike .env files, configuration variables are stored outside your project directory and are never at risk of being committed to version control.
To set your private key for TestNet deployment, run:
When prompted, paste your private key. Hardhat stores it securely and makes it available through vars.get("TESTNET_PRIVATE_KEY") in the configuration file.
Warning
Keep your private key safe and never share it with anyone. If it is compromised, your funds can be stolen.
The hardhat.config.ts file references the variable conditionally, so the project works without it for local development:
networks: {
localNode: {
url: "http://127.0.0.1:8545",
},
polkadotTestnet: {
url: "https://services.polkadothub-rpc.com/testnet",
accounts: vars.has("TESTNET_PRIVATE_KEY")
? [vars.get("TESTNET_PRIVATE_KEY")]
: [],
},
},
Note
You only need the TESTNET_PRIVATE_KEY variable when deploying to the Polkadot Hub TestNet. Local development against the development node does not require any key configuration.
Uniswap V2 Core Architecture¶
Before interacting with the contracts, it is essential to understand the core architecture that powers Uniswap V2. This model forms the basis of nearly every modern DEX implementation and operates under automated market making, token pair liquidity pools, and deterministic pricing principles.
At the heart of Uniswap V2 lies a simple but powerful system composed of two major smart contracts:
- Factory contract: Acts as a registry and creator of new trading pairs. When two ERC-20 tokens are to be traded, the Factory contract generates a new Pair contract that manages that specific token pair's liquidity pool. It tracks all deployed pairs and ensures uniqueness, so no duplicate pools can exist for the same token combination.
- Pair contract: Each Pair contract is a decentralized liquidity pool that holds reserves of two ERC-20 tokens. These contracts implement the core AMM logic, maintaining a constant product invariant (x * y = k) to facilitate swaps and price determination. Users contribute tokens to these pools in return for LP (liquidity provider) tokens, which represent their proportional share of the reserves.
This architecture enables Uniswap to be highly modular, trustless, and extensible. By distributing responsibilities across these components, developers and users can engage with the protocol in a composable and predictable manner.
The project scaffolding is as follows:
uniswap-v2-core-hardhat/
├── contracts/
│ ├── interfaces/
│ │ ├── IERC20.sol
│ │ ├── IUniswapV2Callee.sol
│ │ ├── IUniswapV2ERC20.sol
│ │ ├── IUniswapV2Factory.sol
│ │ └── IUniswapV2Pair.sol
│ ├── libraries/
│ │ ├── Math.sol
│ │ ├── SafeMath.sol
│ │ └── UQ112x112.sol
│ ├── test/
│ │ └── ERC20.sol
│ ├── UniswapV2ERC20.sol
│ ├── UniswapV2Factory.sol
│ └── UniswapV2Pair.sol
├── ignition/
│ └── modules/
│ └── UniswapV2Factory.ts
├── scripts/
│ └── deploy.ts
├── test/
│ ├── shared/
│ │ └── utilities.ts
│ ├── UniswapV2ERC20.test.ts
│ ├── UniswapV2Factory.test.ts
│ └── UniswapV2Pair.test.ts
├── hardhat.config.ts
├── package.json
└── tsconfig.json
Key differences from a typical Ethereum Hardhat project are minimal. The Solidity contracts are the original Uniswap V2 source (Solidity 0.5.16) with no modifications. The test files use TypeScript (.test.ts) and avoid loadFixture for compatibility with the Polkadot execution environment.
Test the Contracts¶
The project includes a comprehensive test suite with 28 tests across three test files covering ERC-20 functionality, factory operations, and pair contract interactions including liquidity management and swaps.
To run the tests locally:
-
Start the local development node. Follow the steps in the Local Development Node guide to set it up.
-
In a new terminal, run the test suite against the local node:
The tests are configured with a 120-second Mocha timeout to accommodate Polkadot network block times. The result should look similar to the following:
npx hardhat test --network localNode Compiling 12 Solidity files Successfully compiled 12 Solidity files UniswapV2ERC20 ✔ name, symbol, decimals, totalSupply, balanceOf, DOMAIN_SEPARATOR, PERMIT_TYPEHASH (41ms) ✔ approve (4983ms) ✔ transfer (5047ms) ✔ transfer:fail ✔ transferFrom (6104ms) ✔ transferFrom:max (6138ms) UniswapV2Factory ✔ feeTo, feeToSetter, allPairsLength ✔ createPair (168ms) ✔ createPair:reverse (1198ms) ✔ setFeeTo (1112ms) ✔ setFeeToSetter (1099ms) UniswapV2Pair ✔ mint (11208ms) ✔ getInputPrice:0 (12374ms) ✔ getInputPrice:1 (17382ms) ✔ getInputPrice:2 (17401ms) ✔ getInputPrice:3 (17488ms) ✔ getInputPrice:4 (17433ms) ✔ getInputPrice:5 (17378ms) ✔ getInputPrice:6 (13427ms) ✔ optimistic:0 (17431ms) ✔ optimistic:1 (17730ms) ✔ optimistic:2 (17441ms) ✔ optimistic:3 (21409ms) ✔ swap:token0 (12449ms) ✔ swap:token1 (17415ms) ✔ burn (17474ms) ✔ feeTo:off (23684ms) ✔ feeTo:on (24775ms) 28 passing (12m)
Tip
If tests time out, ensure your local development node is running and accessible at http://127.0.0.1:8545.
Deploy the Contracts¶
After successfully testing the contracts, you can deploy them to the Polkadot Hub TestNet using Hardhat Ignition. The Ignition module at ignition/modules/UniswapV2Factory.ts deploys the UniswapV2Factory contract.
Make sure you have configured your private key and that your account has test tokens. Then run:
When prompted, confirm the target network name and chain ID. Ignition deploys the Factory contract and prints the deployed address. The output should look similar to the following:
npx hardhat ignition deploy ./ignition/modules/UniswapV2Factory.ts --network polkadotTestnet ✔ Confirm deploy to network polkadotTestnet (420420417)? … yes Hardhat Ignition 🚀 Deploying [ UniswapV2FactoryModule ] Batch #1 Executed UniswapV2FactoryModule#UniswapV2Factory [ UniswapV2FactoryModule ] successfully deployed 🚀 Deployed Addresses UniswapV2FactoryModule#UniswapV2Factory - 0x85b108660f47caDfAB9e0503104C08C1c96e0DA9
Where to Go Next¶
-
Tutorial Deploy Uniswap V2 Periphery
Deploy Router contracts for user-facing swaps, liquidity management, and WETH wrapping on top of V2 Core.
-
Guide Hardhat on Polkadot
Learn how to create, compile, test, and deploy smart contracts on Polkadot Hub using Hardhat.
-
Guide Local Development Node
Set up and run a local development node for testing your smart contracts against Polkadot.
Introduction¶
Decentralized exchanges (DEXs) are a cornerstone of the DeFi ecosystem, allowing for permissionless token swaps without intermediaries. Uniswap V2, with its Automated Market Maker (AMM) model, revolutionized DEXs by enabling liquidity provision for any ERC-20 token pair.
This tutorial will guide you through how Uniswap V2 works so you can take advantage of it in your projects deployed to Polkadot Hub. By understanding these contracts, you'll gain hands-on experience with one of the most influential DeFi protocols and understand how it functions across blockchain ecosystems.
Prerequisites¶
Before starting, make sure you have:
- Node.js (v16.0.0 or later) and npm installed.
- Basic understanding of Solidity and JavaScript.
- Familiarity with Hardhat development environment.
- Some test tokens to cover transaction fees (obtained from the Polkadot faucet).
- Basic understanding of how AMMs and liquidity pools work.
Set Up the Project¶
Start by cloning the Uniswap V2 project:
-
Clone the Uniswap V2 repository:
-
Install the required dependencies:
-
Create a
.envfile in your project root to store your private keys (you can use as an example theenv.examplefile):Ensure to replace
"INSERT_LOCAL_PRIVATE_KEY"with a private key available in the local environment (you can get them from this file). And"INSERT_AH_PRIVATE_KEY"with the account's private key you want to use to deploy the contracts. You can get this by exporting the private key from your wallet (e.g., MetaMask).Warning
Keep your private key safe, and never share it with anyone. If it is compromised, your funds can be stolen.
-
Compile the contracts:
If the compilation is successful, you should see the following output:
After running the above command, you should see the compiled contracts in the artifacts directory. This directory contains the ABI and bytecode of your contracts.
Understanding Uniswap V2 Architecture¶
Before interacting with the contracts, it's essential to understand the core architecture that powers Uniswap V2. This model forms the basis of nearly every modern DEX implementation and operates under automated market making, token pair liquidity pools, and deterministic pricing principles.
At the heart of Uniswap V2 lies a simple but powerful system composed of two major smart contracts:
- Factory contract: The factory acts as a registry and creator of new trading pairs. When two ERC-20 tokens are to be traded, the Factory contract is responsible for generating a new Pair contract that will manage that specific token pair’s liquidity pool. It keeps track of all deployed pairs and ensures uniqueness—no duplicate pools can exist for the same token combination.
- Pair contract: Each pair contract is a decentralized liquidity pool that holds reserves of two ERC-20 tokens. These contracts implement the core logic of the AMM, maintaining a constant product invariant (x * y = k) to facilitate swaps and price determination. Users can contribute tokens to these pools in return for LP (liquidity provider) tokens, which represent their proportional share of the reserves.
This minimal architecture enables Uniswap to be highly modular, trustless, and extensible. By distributing responsibilities across these components, developers, and users can engage with the protocol in a composable and predictable manner, making it an ideal foundation for DEX functionality across ecosystems, including Polkadot Hub.
The project scaffolding is as follows:
uniswap-V2-polkadot
├── bin/
├── contracts/
│ ├── interfaces/
│ │ ├── IERC20.sol
│ │ ├── IUniswapV2Callee.sol
│ │ ├── IUniswapV2ERC20.sol
│ │ ├── IUniswapV2Factory.sol
│ │ └── IUniswapV2Pair.sol
│ ├── libraries/
│ │ ├── Math.sol
│ │ ├── SafeMath.sol
│ │ └── UQ112x112.sol
│ ├── test/
│ │ └── ERC20.sol
│ ├── UniswapV2ERC20.sol
│ ├── UniswapV2Factory.sol
│ └── UniswapV2Pair.sol
├── ignition/
├── scripts/
│ └── deploy.js
├── node_modules/
├── test/
│ ├── shared/
│ │ ├── fixtures.js
│ │ └── utilities.js
│ ├── UniswapV2ERC20.js
│ ├── UniswapV2Factory.js
│ └── UniswapV2Pair.js
├── .env.example
├── .gitignore
├── hardhat.config.js
├── package.json
└── README.md
Test the Contracts¶
You can run the provided test suite to ensure the contracts are working as expected. The tests cover various scenarios, including creating pairs, adding liquidity, and executing swaps.
To test it locally, you can run the following commands:
-
Run the local
revive-dev-node, for this, you can check the Local Development Node guide. -
In a new terminal, run the tests:
The result should look like this:
Deploy the Contracts¶
After successfully testing the contracts, you can deploy them to the local node or Polkadot Hub. The deployment script is located in the scripts directory and is named deploy.js. This script deploys the Factory and Pair contracts to the network.
To deploy the contracts, run the following command:
This command deploys the contracts to your local blockchain for development and testing. If you want to deploy to Polkadot Hub, you can use the following command:
The command above deploys to the actual Polkadot Hub TestNet. It requires test tokens, persists on the network, and operates under real network conditions.
The deployment script will output the addresses of the deployed contracts. Save these addresses, as you will need them to interact with the contracts. For example, the output should look like this:
Conclusion¶
This tutorial guided you through deploying Uniswap V2 contracts to Polkadot Hub. This implementation brings the powerful AMM architecture to the Polkadot ecosystem, laying the foundation for the decentralized trading of ERC-20 token pairs.
By following this guide, you've gained practical experience with:
- Setting up a Hardhat project for deploying to Polkadot Hub.
- Understanding the Uniswap V2 architecture.
- Testing Uniswap V2 contracts in a local environment.
- Deploying contracts to both local and testnet environments.
To build on this foundation, you could extend this project by implementing functionality to create liquidity pools, execute token swaps, and build a user interface for interacting with your deployment.
This knowledge can be leveraged to build more complex DeFi applications or to integrate Uniswap V2 functionality into your existing projects on Polkadot.
| Created: April 6, 2026