Разработка whitelist/allowlist для минтинга NFT

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

Разработка whitelist/allowlist для минтинга NFT

Типичная ситуация: коллекция на 10,000 NFT, whitelist-минт для 3,000 адресов за 12 часов до публичного. Если хранить whitelist on-chain как mapping(address => bool) — деплой контракта с записью 3,000 адресов обходится примерно в 15-20M gas только на SSTORE. При 20 gwei это $300-400 только на whitelist. Merkle proof решает это за 50k gas на весь список независимо от размера.

Merkle proof: как работает и где ошибаются

Merkle tree строится off-chain: каждый адрес хэшируется через keccak256(abi.encodePacked(address)), из листьев строится дерево попарным хэшированием. Результат — 32-байтный merkleRoot. Этот root деплоится в контракт. При минте пользователь передаёт proof[] — массив хэшей-братьев по пути от его листа до корня. Контракт верифицирует через MerkleProof.verify() из OpenZeppelin.

Распространённая ошибка — двойной минт. Если в контракте нет mapping(address => bool) public hasMinted (или mapping(address => uint256) public mintedCount), пользователь из whitelist может минтить неограниченное количество раз. Proof остаётся валидным. Контракт не знает, что адрес уже минтил.

Вторая ошибка — leaf encoding. OpenZeppelin MerkleProof ожидает, что лист — это keccak256(keccak256(data)) (double hash) для защиты от preimage attacks в случае если узлы дерева совпадают с листьями. Если генерируешь дерево через merkletreejs с одинарным hash, а контракт использует _leaf = keccak256(abi.encodePacked(account)) без double hash — может сработать collision attack при определённых конфигурациях. Используем keccak256(bytes.concat(keccak256(abi.encode(addr)))) или стандартную связку @openzeppelin/merkle-tree JS библиотека + MerkleProof.sol.

Варианты реализации

Merkle proof (основной)

Подходит для: списков от 100 до миллиона адресов. Стоимость деплоя не зависит от размера списка. Proof передаётся пользователем при минте (frontend генерирует автоматически).

Ограничение: нельзя добавить адрес после деплоя без пересборки дерева и обновления root. Если контракт upgradeable или owner может вызвать setMerkleRoot(bytes32) — это решается. Если immutable — нет.

Подпись backend (ECDSA)

Owner контракта держит приватный ключ. Для каждого whitelist-адреса backend подписывает keccak256(abi.encodePacked(address, nonce)). Контракт проверяет подпись через ECDSA.recover() и сверяет с signer address.

Преимущество: динамическое управление — можно добавлять адреса без изменения контракта, можно выдавать приоритеты, разные квоты. Недостаток: централизованный signer — single point of failure и trust. Если ключ утечёт — любой может заминтить.

Применение: gaming mint (backend знает achievement пользователя), динамические кампании.

On-chain mapping

Только для очень маленьких списков (<100 адресов) или когда список известен заранее и не изменится. addToWhitelist(address[]) с onlyOwner модификатором. 20,000 gas на адрес.

Дополнительная механика

Multi-tier whitelist — разные квоты для разных уровней. Merkle дерево содержит листья keccak256(abi.encode(address, maxMintAmount)). Пользователь передаёт proof + свой maxMintAmount, контракт верифицирует оба параметра вместе.

Temporal phases — WL → Allowlist → Public. Контракт держит enum SalePhase { PAUSED, WHITELIST, ALLOWLIST, PUBLIC }. Разные root'ы для каждой фазы, разные цены. owner меняет фазу через setPhase().

Batch mint с WL — пользователь может минтить N NFT за одну транзакцию, если его квота позволяет. mintedCount[msg.sender] += amount вместо bool флага.

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

Реализация (1-2 дня). Контракт с Merkle verify + hasMinted mapping + phase management. Тесты в Foundry с edge cases: повторный минт, невалидный proof, исчерпанная квота.

Frontend интеграция (1 день). Генерация proof через @openzeppelin/merkle-tree, wagmi hook useMint().

Деплой. Foundry script с автоматической верификацией Etherscan.

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

Whitelist контракт с Merkle proof — 2-3 дня включая тесты и frontend интеграцию.

Стоимость рассчитывается индивидуально.