Разработка системы multi-hop свапов

Проектируем и разрабатываем блокчейн-решения полного цикла: от архитектуры смарт-контрактов до запуска DeFi-протоколов, NFT-маркетплейсов и криптобирж. Аудит безопасности, токеномика, интеграция с существующей инфраструктурой.
Показано 1 из 1 услугВсе 1306 услуг
Разработка системы multi-hop свапов
Сложная
~1-2 недели
Часто задаваемые вопросы
Направления блокчейн-разработки
Этапы блокчейн-разработки
Последние работы
  • 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
    1062
  • image_logo-advance_0.png
    Разработка логотипа компании B2B Advance
    561
  • image_crm_enviok_479_0.webp
    Разработка веб-приложения для компании Enviok
    828

Разработка системы multi-hop свапов

Протокол агрегирует ликвидность с трёх DEX, но маршрут USDC→WBTC идёт через единственный пул с глубиной $200k. Результат — проскальзывание 1.8% на сделке в $50k. Пользователь торгует в минус относительно рыночной цены, а алгоритм об этом молчит. Multi-hop решает это через разбивку маршрута: USDC→WETH через Uniswap v3, WETH→WBTC через Curve tricrypto. Суммарное проскальзывание — 0.3%. Разница существенная, но реализовать правильно — не тривиально.

Где ломается наивная реализация multi-hop

Path encoding и stack overflow в Solidity

Uniswap v3 кодирует маршрут как bytes path — последовательность address fee address fee address. При трёх хопах это 20+3+20+3+20 = 66 байт. Кажется просто. Проблема начинается, когда разработчик пытается строить path динамически в Solidity — abi.encodePacked в цикле с uint24[] fees и address[] tokens. Если ввод не валидируется, можно собрать путь с несовпадением длин: 4 токена, 2 fee. Контракт скомпилируется. На swap уйдёт в revert на уровне декодирования в UniswapV3Pool, и без внятного error message.

Второй вектор — callback manipulation. В uniswapV3SwapCallback контракт должен проверить, что вызывающий — легитимный пул, вычисленный через PoolAddress.computeAddress. Если этой проверки нет, любой может вызвать callback напрямую, передав произвольный amount0Delta / amount1Delta, и вытащить токены из контракта. Именно так был дренирован один из форков агрегатора в 2023 году — отсутствие caller validation в callback.

Price impact calculation через несколько пулов

Считать price impact по multi-hop маршруту сложнее, чем по одному пулу. Наивный подход — вызвать quoteExactInput у Quoter, получить amountOut, сравнить с spot price. Работает. Но Quoter v2 требует simulate через eth_call, а при частых запросах это создаёт нагрузку на RPC. Более правильный путь — off-chain расчёт через математику CPMM и CLMM: для каждого пула считаем sqrtPriceX96 после свапа, затем агрегируем. Это позволяет считать impact без on-chain запросов.

Ещё тонкость: при hop через Curve стабильный пул (3pool, Frax) математика другая — StableSwap invariant вместо x*y=k. Смешивать расчёты нельзя, иначе оценка выйдет неверной.

MEV и sandwich-атаки на multi-hop маршруты

Длинный маршрут с несколькими пулами — лакомый кусок для MEV-ботов. Каждый пул — отдельная точка атаки. Классический sandwich: бот фронтранит первый хоп, повышает цену, затем бэкранит после выполнения транзакции. Защита — жёсткий amountOutMinimum на весь маршрут (не на каждый хоп отдельно) и использование private mempool: Flashbots Protect, MEV Blocker, или прямая отправка через eth_sendPrivateTransaction.

Как мы строим multi-hop систему

Архитектура: off-chain routing + on-chain execution

Разделение обязанностей принципиально. Off-chain router считает оптимальный маршрут — это Python/TypeScript сервис, который строит граф из пулов Uniswap v2/v3, Curve, Balancer, и запускает Dijkstra или Bellman-Ford для поиска пути с минимальным impact. On-chain контракт только исполняет: принимает закодированный путь, валидирует его, исполняет свапы через ISwapRouter / ICurvePool, отдаёт amountOut.

Компонент Инструменты Задача
Graph builder viem, The Graph, subgraph Актуальный snapshot пулов
Path optimizer TypeScript, custom Dijkstra Поиск маршрута с min slippage
Quote engine UniswapV3 Quoter v2, Curve calc Точная оценка amountOut
Executor contract Solidity 0.8.x, Foundry On-chain исполнение
Slippage guard amountOutMinimum + deadline Защита от MEV

Реализация executor-контракта

Контракт реализует IUniversalRouter-подобный интерфейс. Ключевая функция — executeMultiHop(bytes calldata path, uint256 amountIn, uint256 amountOutMin, address recipient). Внутри: декодируем path, определяем тип первого пула (Uniswap v3 по наличию fee uint24, или Curve по address registry), маршрутизируем в соответствующий адаптер.

Каждый адаптер — отдельный контракт, зарегистрированный в IAdapterRegistry. Это позволяет добавлять поддержку новых DEX без переписывания executor. Паттерн — strategy через интерфейс ISwapAdapter с методом swap(address tokenIn, address tokenOut, uint256 amountIn, bytes calldata data) returns (uint256 amountOut).

Кэш адресов пулов в mapping(bytes32 => address) — ключ это keccak256(abi.encodePacked(token0, token1, fee)). Позволяет не обращаться к factory на каждый хоп.

Тестирование на fork mainnet

Multi-hop нельзя протестировать без реального состояния пулов. Используем Foundry fork-тесты:

vm.createSelectFork(vm.envString("ETH_RPC_URL"), blockNumber);

Фиксируем конкретный блок — воспроизводимость тестов. Прогоняем сценарии: USDC→WETH→WBTC через Uniswap v3, DAI→USDC→ETH→stETH через микс Curve+Uniswap. Проверяем, что amountOut совпадает с предсказанием Quoter с допуском ±0.01%.

Fuzzing на размер входных сумм — amountIn от 1 до 10^9 единиц токена. Ищем edge cases где path расчёт даёт amountOut = 0 из-за integer overflow/underflow при промежуточных вычислениях.

Процесс работы

Аналитика (2-3 дня). Инвентаризация пулов: какие DEX, какие чейны, нужна ли кросс-чейн поддержка. Определяем, нужен ли собственный subgraph или достаточно публичных эндпоинтов.

Проектирование (3-5 дней). Схема граф-роутера, интерфейсы адаптеров, storage layout executor-контракта. На этом этапе решаем вопрос апгрейдаемости: если добавление новых DEX планируется, адаптер-реестр должен поддерживать registerAdapter с access control.

Разработка (1-2 недели). Off-chain router + on-chain executor + набор адаптеров под конкретные DEX. Fork-тесты на Ethereum и целевых L2 (Arbitrum, Optimism, Base).

Интеграция. wagmi/viem хуки для frontend: useMultiHopQuote, useMultiHopSwap. WebSocket подписка на обновление цен через The Graph.

Аудит и деплой. Slither + ручной review callback-функций. Деплой через Foundry script с Gnosis Safe мультисиг на функции owner.

Ориентиры по срокам

MVP с поддержкой Uniswap v2/v3 и одним чейном — 1-2 недели. Полноценный агрегатор с Curve, Balancer, кастомным subgraph и поддержкой 3-4 чейнов — 6-8 недель. Сроки зависят от количества поддерживаемых DEX и требований к точности quote-движка.

Стоимость рассчитывается после анализа технического задания.