Разработка децентрализованной биржи (DEX)

Проектируем и разрабатываем блокчейн-решения полного цикла: от архитектуры смарт-контрактов до запуска DeFi-протоколов, NFT-маркетплейсов и криптобирж. Аудит безопасности, токеномика, интеграция с существующей инфраструктурой.
Показано 1 из 1 услугВсе 1306 услуг
Разработка децентрализованной биржи (DEX)
Сложная
от 2 недель до 3 месяцев
Часто задаваемые вопросы
Направления блокчейн-разработки
Этапы блокчейн-разработки
Последние работы
  • image_website-b2b-advance_0.png
    Разработка сайта компании B2B ADVANCE
    1221
  • image_web-applications_feedme_466_0.webp
    Разработка веб-приложения для компании FEEDME
    1163
  • image_websites_belfingroup_462_0.webp
    Разработка веб-сайта для компании БЕЛФИНГРУПП
    855
  • image_ecommerce_furnoro_435_0.webp
    Разработка интернет магазина для компании FURNORO
    1056
  • image_logo-advance_0.png
    Разработка логотипа компании B2B Advance
    561
  • image_crm_enviok_479_0.webp
    Разработка веб-приложения для компании Enviok
    828

Разработка DEX (децентрализованной биржи)

Децентрализованная биржа — это smart contracts, которые позволяют торговать без посредника. Пользователь никогда не передаёт контроль над средствами третьей стороне: all trades happen on-chain, settlement атомарный, custody — в кошельке пользователя. Uniswap V3 торгует > $1B в день. Задача: разработать DEX с нуля — от AMM механизма до фронтенда.

Выбор модели DEX

AMM (Automated Market Maker)

Ликвидность предоставляется в пулы, а не в order book. Цена определяется математической формулой.

Constant Product (x * y = k) — Uniswap V2 модель. Простая, работает для любой пары. Недостаток: capital inefficiency — большинство ликвидности никогда не используется.

Concentrated Liquidity — Uniswap V3 модель. LP указывает диапазон цен для своей ликвидности. Capital efficiency в 10–100× выше. Сложнее для LP — нужно активное управление.

Stableswap (Curve) — для активов с похожей ценой (USDC/USDT, stETH/ETH). Минимальный slippage при больших объёмах.

Order Book DEX

On-chain order book — дорогой по газу (каждое размещение/отмена = транзакция). Решения: off-chain orderbook с on-chain settlement (dYdX v3, Serum), или L2-based CLOB (dYdX v4 на Cosmos).

Для большинства DEX на EVM — AMM с concentrated liquidity.

Constant Product AMM: смарт-контракт

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;

import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/token/ERC20/ERC20.sol";
import "@openzeppelin/contracts/utils/math/Math.sol";

contract ConstantProductPool is ERC20 {
    address public immutable token0;
    address public immutable token1;
    
    uint256 private reserve0;
    uint256 private reserve1;
    
    uint256 private constant FEE_NUMERATOR = 997;   // 0.3% fee
    uint256 private constant FEE_DENOMINATOR = 1000;
    
    event Swap(address indexed sender, uint256 amount0In, uint256 amount1In,
               uint256 amount0Out, uint256 amount1Out, address indexed to);
    event Mint(address indexed sender, uint256 amount0, uint256 amount1);
    event Burn(address indexed sender, uint256 amount0, uint256 amount1, address indexed to);
    
    constructor(address _token0, address _token1) ERC20("LP Token", "LP") {
        token0 = _token0;
        token1 = _token1;
    }
    
    // Добавление ликвидности
    function mint(address to) external returns (uint256 liquidity) {
        uint256 balance0 = IERC20(token0).balanceOf(address(this));
        uint256 balance1 = IERC20(token1).balanceOf(address(this));
        uint256 amount0 = balance0 - reserve0;
        uint256 amount1 = balance1 - reserve1;
        
        uint256 totalSupply_ = totalSupply();
        
        if (totalSupply_ == 0) {
            // Первое добавление ликвидности: геометрическое среднее минус MINIMUM_LIQUIDITY
            liquidity = Math.sqrt(amount0 * amount1) - 1000;
            _mint(address(0xdead), 1000); // заблокировать минимум для предотвращения инфляционной атаки
        } else {
            liquidity = Math.min(
                amount0 * totalSupply_ / reserve0,
                amount1 * totalSupply_ / reserve1
            );
        }
        
        require(liquidity > 0, "INSUFFICIENT_LIQUIDITY_MINTED");
        _mint(to, liquidity);
        
        reserve0 = balance0;
        reserve1 = balance1;
        
        emit Mint(msg.sender, amount0, amount1);
    }
    
    // Вывод ликвидности
    function burn(address to) external returns (uint256 amount0, uint256 amount1) {
        uint256 liquidity = balanceOf(address(this));
        uint256 totalSupply_ = totalSupply();
        
        amount0 = liquidity * reserve0 / totalSupply_;
        amount1 = liquidity * reserve1 / totalSupply_;
        
        require(amount0 > 0 && amount1 > 0, "INSUFFICIENT_LIQUIDITY_BURNED");
        
        _burn(address(this), liquidity);
        IERC20(token0).transfer(to, amount0);
        IERC20(token1).transfer(to, amount1);
        
        reserve0 = IERC20(token0).balanceOf(address(this));
        reserve1 = IERC20(token1).balanceOf(address(this));
        
        emit Burn(msg.sender, amount0, amount1, to);
    }
    
    // Swap
    function swap(uint256 amount0Out, uint256 amount1Out, address to, bytes calldata data) external {
        require(amount0Out > 0 || amount1Out > 0, "INSUFFICIENT_OUTPUT_AMOUNT");
        require(amount0Out < reserve0 && amount1Out < reserve1, "INSUFFICIENT_LIQUIDITY");
        
        // Оптимистичный transfer (flash loan паттерн)
        if (amount0Out > 0) IERC20(token0).transfer(to, amount0Out);
        if (amount1Out > 0) IERC20(token1).transfer(to, amount1Out);
        
        // Flash loan callback если нужен
        if (data.length > 0) IUniswapV2Callee(to).uniswapV2Call(msg.sender, amount0Out, amount1Out, data);
        
        uint256 balance0 = IERC20(token0).balanceOf(address(this));
        uint256 balance1 = IERC20(token1).balanceOf(address(this));
        
        uint256 amount0In = balance0 > reserve0 - amount0Out ? balance0 - (reserve0 - amount0Out) : 0;
        uint256 amount1In = balance1 > reserve1 - amount1Out ? balance1 - (reserve1 - amount1Out) : 0;
        
        require(amount0In > 0 || amount1In > 0, "INSUFFICIENT_INPUT_AMOUNT");
        
        // Проверяем инвариант с учётом fee: (x + 0.997*dx)(y - dy) >= x*y
        uint256 balance0Adjusted = balance0 * FEE_DENOMINATOR - amount0In * (FEE_DENOMINATOR - FEE_NUMERATOR);
        uint256 balance1Adjusted = balance1 * FEE_DENOMINATOR - amount1In * (FEE_DENOMINATOR - FEE_NUMERATOR);
        
        require(
            balance0Adjusted * balance1Adjusted >= reserve0 * reserve1 * FEE_DENOMINATOR ** 2,
            "K_INVARIANT_VIOLATED"
        );
        
        reserve0 = uint256(balance0);
        reserve1 = uint256(balance1);
        
        emit Swap(msg.sender, amount0In, amount1In, amount0Out, amount1Out, to);
    }
    
    // Вычисление выходного количества по формуле AMM
    function getAmountOut(uint256 amountIn, uint256 reserveIn, uint256 reserveOut) 
        public pure returns (uint256) 
    {
        require(amountIn > 0, "INSUFFICIENT_INPUT_AMOUNT");
        require(reserveIn > 0 && reserveOut > 0, "INSUFFICIENT_LIQUIDITY");
        
        uint256 amountInWithFee = amountIn * FEE_NUMERATOR;
        uint256 numerator = amountInWithFee * reserveOut;
        uint256 denominator = reserveIn * FEE_DENOMINATOR + amountInWithFee;
        
        return numerator / denominator;
    }
}

Factory и Router

Uniswap-style архитектура: Factory создаёт пулы, Router — точка входа для пользователей.

contract DEXFactory {
    mapping(address => mapping(address => address)) public getPool;
    address[] public allPools;
    
    event PoolCreated(address indexed token0, address indexed token1, address pool);
    
    function createPool(address tokenA, address tokenB) external returns (address pool) {
        require(tokenA != tokenB, "IDENTICAL_ADDRESSES");
        
        (address token0, address token1) = tokenA < tokenB 
            ? (tokenA, tokenB) : (tokenB, tokenA);
        
        require(token0 != address(0), "ZERO_ADDRESS");
        require(getPool[token0][token1] == address(0), "POOL_EXISTS");
        
        bytes memory bytecode = type(ConstantProductPool).creationCode;
        bytes32 salt = keccak256(abi.encodePacked(token0, token1));
        assembly { pool := create2(0, add(bytecode, 32), mload(bytecode), salt) }
        
        ConstantProductPool(pool).initialize(token0, token1);
        
        getPool[token0][token1] = pool;
        getPool[token1][token0] = pool;
        allPools.push(pool);
        
        emit PoolCreated(token0, token1, pool);
    }
}

Router: находит путь обмена (прямой или через промежуточный токен — например, TOKEN_A → ETH → TOKEN_B):

contract DEXRouter {
    address public immutable factory;
    address public immutable WETH;
    
    // Swap с slippage protection
    function swapExactTokensForTokens(
        uint256 amountIn,
        uint256 amountOutMin,  // минимальный out — защита от slippage
        address[] calldata path,
        address to,
        uint256 deadline
    ) external returns (uint256[] memory amounts) {
        require(deadline >= block.timestamp, "EXPIRED");
        
        amounts = getAmountsOut(amountIn, path);
        require(amounts[amounts.length - 1] >= amountOutMin, "INSUFFICIENT_OUTPUT_AMOUNT");
        
        IERC20(path[0]).transferFrom(msg.sender, getPool(path[0], path[1]), amounts[0]);
        _swap(amounts, path, to);
    }
    
    // ETH → Token через WETH
    function swapExactETHForTokens(
        uint256 amountOutMin,
        address[] calldata path,
        address to,
        uint256 deadline
    ) external payable returns (uint256[] memory amounts) {
        require(path[0] == WETH, "INVALID_PATH");
        amounts = getAmountsOut(msg.value, path);
        require(amounts[amounts.length - 1] >= amountOutMin, "INSUFFICIENT_OUTPUT_AMOUNT");
        
        IWETH(WETH).deposit{value: amounts[0]}();
        IERC20(WETH).transfer(getPool(path[0], path[1]), amounts[0]);
        _swap(amounts, path, to);
    }
}

Безопасность AMM

Reentrancy Protection

// Все внешние вызовы после обновления state (Checks-Effects-Interactions)
// + ReentrancyGuard от OpenZeppelin
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";

contract ConstantProductPool is ERC20, ReentrancyGuard {
    function swap(...) external nonReentrant {
        // ...
    }
}

Price Manipulation (TWAP)

Spot price легко manipulate через flash loan на один блок. Для oracle — Time-Weighted Average Price:

// TWAP oracle — среднее за последние N секунд
uint256 price0CumulativeLast;
uint256 price1CumulativeLast;
uint32  blockTimestampLast;

function _updatePriceAccumulators() private {
    uint32 blockTimestamp = uint32(block.timestamp);
    uint32 timeElapsed = blockTimestamp - blockTimestampLast;
    
    if (timeElapsed > 0 && reserve0 != 0 && reserve1 != 0) {
        // Накапливаем price * time
        price0CumulativeLast += uint256(UQ112x112.encode(reserve1).uqdiv(reserve0)) * timeElapsed;
        price1CumulativeLast += uint256(UQ112x112.encode(reserve0).uqdiv(reserve1)) * timeElapsed;
    }
    
    blockTimestampLast = blockTimestamp;
}

MEV Protection

Frontrunning атаки на DEX — стандартная проблема. Mitigation:

  • Slippage tolerance: пользователь устанавливает max slippage (0.5–1%), транзакция реверсируется если превышен
  • Deadline: транзакция реверсируется если не исполнена до deadline
  • Flashbots / private mempool: для больших свапов — отправка через Flashbots для предотвращения frontrunning

Frontend DEX интерфейс

import { useAccount, useReadContract, useWriteContract } from 'wagmi';
import { parseEther, formatEther } from 'viem';

function SwapInterface() {
  const [tokenIn, setTokenIn] = useState<Token>(ETH);
  const [tokenOut, setTokenOut] = useState<Token>(USDC);
  const [amountIn, setAmountIn] = useState('');
  
  // Получаем quote из контракта
  const { data: amountOut } = useReadContract({
    address: ROUTER_ADDRESS,
    abi: routerAbi,
    functionName: 'getAmountsOut',
    args: amountIn ? [
      parseEther(amountIn),
      [tokenIn.address, tokenOut.address],
    ] : undefined,
    query: { enabled: !!amountIn && parseFloat(amountIn) > 0 },
  });
  
  const { writeContract, isPending } = useWriteContract();
  
  const handleSwap = () => {
    const deadline = Math.floor(Date.now() / 1000) + 1200; // +20 min
    const minAmountOut = amountOut[1] * 995n / 1000n;      // 0.5% slippage
    
    writeContract({
      address: ROUTER_ADDRESS,
      abi: routerAbi,
      functionName: 'swapExactETHForTokens',
      args: [minAmountOut, [WETH, tokenOut.address], account.address, deadline],
      value: parseEther(amountIn),
    });
  };
  
  return (
    <div>
      <TokenInput token={tokenIn} amount={amountIn} onChange={setAmountIn} />
      <SwapArrow onClick={switchTokens} />
      <TokenInput token={tokenOut} amount={amountOut ? formatEther(amountOut[1]) : ''} readonly />
      <PriceImpact amountIn={amountIn} amountOut={amountOut} pool={currentPool} />
      <SlippageSettings />
      <Button onClick={handleSwap} disabled={isPending}>
        {isPending ? 'Swapping...' : 'Swap'}
      </Button>
    </div>
  );
}

Subgraph для аналитики

The Graph — индексация on-chain событий для аналитики:

// schema.graphql
type Pool @entity {
  id: ID!
  token0: Token!
  token1: Token!
  reserve0: BigDecimal!
  reserve1: BigDecimal!
  totalSupply: BigDecimal!
  swapCount: BigInt!
  volumeUSD: BigDecimal!
  createdAt: BigInt!
}

type Swap @entity(immutable: true) {
  id: ID!
  pool: Pool!
  amount0In: BigDecimal!
  amount1In: BigDecimal!
  amount0Out: BigDecimal!
  amount1Out: BigDecimal!
  amountUSD: BigDecimal!
  timestamp: BigInt!
  sender: Bytes!
}
// mapping.ts — обработчик событий
export function handleSwap(event: SwapEvent): void {
  let pool = Pool.load(event.address.toHex())!;
  
  let swap = new Swap(event.transaction.hash.toHex() + "-" + event.logIndex.toString());
  swap.pool = pool.id;
  swap.amount0In = convertTokenToDecimal(event.params.amount0In, pool.token0.decimals);
  swap.amount1In = convertTokenToDecimal(event.params.amount1In, pool.token1.decimals);
  swap.amountUSD = calculateUSDValue(swap, pool);
  swap.timestamp = event.block.timestamp;
  swap.save();
  
  pool.swapCount = pool.swapCount.plus(BigInt.fromI32(1));
  pool.volumeUSD = pool.volumeUSD.plus(swap.amountUSD);
  pool.save();
}

Аудит

DEX smart contracts управляют реальными средствами пользователей. Аудит — не опциональный шаг:

  • Reentrancy: все пути исполнения через nonReentrant
  • Flash loan attacks: price oracle не использует spot price
  • Integer overflow/underflow: Solidity 0.8+ с встроенными проверками
  • Access control: только factory может создавать пулы
  • Precision loss: integer math, правильный порядок операций

Рекомендуемые аудиторы: Trail of Bits, OpenZeppelin Security, Halborn.

Сроки разработки

Компонент Срок
AMM core contracts 4–6 недель
Factory + Router 3–4 недели
TWAP oracle 1–2 недели
Subgraph 2–3 недели
Frontend (swap + liquidity) 4–6 недель
Smart contract аудит 4–8 недель

MVP DEX на mainnet: 4–6 месяцев с учётом аудита.