Разработка системы турниров крипто-казино
Турниры — механика удержания и монетизации. Вместо (или в дополнение к) обычному PvH (player vs house), игроки соревнуются между собой за призовой фонд. Retention эффект значительный: игрок, участвующий в 7-дневном турнире, с высокой вероятностью играет каждый день. Для крипто-казино турниры с on-chain prize distribution дают дополнительное преимущество: верифицируемые правила и выплаты.
Технически система турниров — это комбинация: накопление очков (tracking), leaderboard (рейтинг), prize pool management (смарт-контракт), и distribution (автоматическая выплата победителям).
Архитектура турнирной системы
Типы турниров
Freeroll: бесплатное участие, prize pool от казино. Engagement tool, не revenue generator. Привлекает новых игроков.
Buy-in: игрок платит entry fee (криптой), fees формируют prize pool (minus rake). Rake = 5–15% от total fees → доход казино.
Leaderboard tournament: накапливаются очки за обычную игру за период (7 дней, 24 часа). Prize pool фиксированный от казино. Не требует отдельной записи — автоматически участвуют все игроки.
Sit & Go: начинается когда набирается N игроков. Быстрые форматы: 5–10 минут, $10 buy-in, топ 3 в деньгах.
Prize pool смарт-контракт
contract TournamentManager {
struct Tournament {
bytes32 id;
TournamentType tournType;
uint256 startTime;
uint256 endTime;
uint256 prizePool; // total prize pool
uint256 entryFee; // 0 для freeroll
uint256 rake; // в basis points: 1000 = 10%
uint8 maxParticipants;
address[] participants;
bool distributed;
PrizeStructure[] prizes; // [{rank: 1, percent: 5000}, ...]
}
struct PrizeStructure {
uint8 rank;
uint16 percent; // basis points: 5000 = 50%
}
mapping(bytes32 => Tournament) public tournaments;
// Регистрация с buy-in
function enterTournament(bytes32 tournamentId) external payable {
Tournament storage t = tournaments[tournamentId];
require(block.timestamp < t.startTime, "Tournament started");
require(t.participants.length < t.maxParticipants, "Full");
require(!_isParticipant(tournamentId, msg.sender), "Already entered");
if (t.entryFee > 0) {
require(msg.value == t.entryFee, "Wrong entry fee");
// Rake удерживаем сразу
uint256 rakeAmount = msg.value * t.rake / 10_000;
uint256 prizeContribution = msg.value - rakeAmount;
t.prizePool += prizeContribution;
// Rake → treasury
treasury.transfer(rakeAmount);
}
t.participants.push(msg.sender);
emit PlayerEntered(tournamentId, msg.sender);
}
// Дистрибьюция призов после окончания (вызывается оператором с результатами)
function distributePrizes(
bytes32 tournamentId,
address[] calldata rankedPlayers, // топ N игроков по очкам
bytes calldata operatorSignature
) external {
Tournament storage t = tournaments[tournamentId];
require(block.timestamp > t.endTime, "Not ended");
require(!t.distributed, "Already distributed");
// Верифицируем подпись оператора на ranked list
_verifyRankingSignature(tournamentId, rankedPlayers, operatorSignature);
t.distributed = true;
// Выплачиваем по prize structure
for (uint8 i = 0; i < t.prizes.length && i < rankedPlayers.length; i++) {
uint256 prize = t.prizePool * t.prizes[i].percent / 10_000;
payable(rankedPlayers[i]).transfer(prize);
emit PrizeAwarded(tournamentId, rankedPlayers[i], i + 1, prize);
}
}
}
Leaderboard и очки: off-chain с on-chain settlement
Обновлять leaderboard on-chain при каждом игровом действии — нерационально. Правильная архитектура: очки накапливаются off-chain (game server), settlement происходит on-chain только при финализации.
// Backend: Tournament Score Tracker
class TournamentScoreTracker {
private scores = new Map<string, Map<string, bigint>>(); // tournId → playerId → score
// Вызывается при каждой игровой транзакции
async onGameResult(event: GameResultEvent): Promise<void> {
const activeTournaments = await this.getActiveTournaments(event.timestamp);
for (const tournament of activeTournaments) {
if (!this.isEligible(event, tournament)) continue;
const points = this.calculatePoints(event, tournament.scoringRules);
const key = `${tournament.id}:${event.playerId}`;
const current = this.scores.get(tournament.id)?.get(event.playerId) ?? 0n;
this.scores.get(tournament.id)?.set(event.playerId, current + points);
// Публикуем в Redis для real-time leaderboard API
await redis.zadd(
`tournament:${tournament.id}:leaderboard`,
{ score: Number(current + points), member: event.playerId },
);
}
}
// Финализируем и подготавливаем для on-chain settlement
async finalizeAndSign(tournamentId: string): Promise<SignedRanking> {
const scores = this.scores.get(tournamentId);
if (!scores) throw new Error('Tournament not found');
// Сортируем по очкам
const ranked = Array.from(scores.entries())
.sort(([, a], [, b]) => (b > a ? 1 : -1))
.map(([playerId]) => playerId);
// Строим Merkle дерево из ranked list
const leaves = ranked.map((id, index) =>
keccak256(abi.encode(tournamentId, id, index + 1))
);
const merkleRoot = buildMerkleTree(leaves).getRoot();
// Подписываем оператором
const signature = await operatorSigner.signMessage(
keccak256(abi.encode(tournamentId, merkleRoot, ranked.slice(0, 20)))
);
return { ranked: ranked.slice(0, 20), merkleRoot, signature };
}
}
Scoring rules — очки за разные игры
Разные игры дают разные очки в зависимости от правил турнира:
interface ScoringRules {
gameType: 'slots' | 'crash' | 'keno' | 'blackjack';
multiplier: number; // базовый множитель очков
minBet?: bigint; // минимальная ставка для зачёта
bonusOnWin: number; // бонусные очки за выигрыш
bonusOnBigWin: number; // бонус за выигрыш > X
bigWinThreshold: number; // порог "большого выигрыша" в % от ставки
}
// Примерная формула:
function calculatePoints(event: GameResultEvent, rules: ScoringRules): bigint {
if (rules.minBet && event.betAmount < rules.minBet) return 0n;
// Базовые очки = ставка × multiplier
let points = event.betAmount * BigInt(rules.multiplier) / 100n;
// Бонус за выигрыш
if (event.payout > event.betAmount) {
points += event.betAmount * BigInt(rules.bonusOnWin) / 100n;
// Бонус за большой выигрыш
const winRatio = Number(event.payout * 100n / event.betAmount);
if (winRatio >= rules.bigWinThreshold) {
points += event.betAmount * BigInt(rules.bonusOnBigWin) / 100n;
}
}
return points;
}
Real-time leaderboard API
Для хорошего UX: обновление leaderboard каждые 10–30 секунд во время турнира.
// WebSocket endpoint для real-time leaderboard
wss.on('connection', (ws, req) => {
const tournamentId = extractTournamentId(req.url);
const sendLeaderboard = async () => {
// Redis Sorted Set: O(log N) для top-N запроса
const top20 = await redis.zrange(
`tournament:${tournamentId}:leaderboard`,
0, 19,
{ REV: true, WITHSCORES: true }
);
const leaderboard = top20.map(({ value: playerId, score }, index) => ({
rank: index + 1,
playerId,
score: BigInt(score),
displayName: playerNames.get(playerId),
prize: calculatePrize(tournamentId, index + 1),
}));
ws.send(JSON.stringify({ type: 'leaderboard_update', data: leaderboard }));
};
// Отправляем сразу и потом каждые 15 секунд
sendLeaderboard();
const interval = setInterval(sendLeaderboard, 15_000);
ws.on('close', () => clearInterval(interval));
});
Для игрока важно видеть свою позицию: Redis поддерживает ZRANK — O(log N) получение ранга конкретного игрока.
Специальные механики турниров
Rebuy система
// Игрок может купить дополнительные попытки (rebuys) в ранней фазе турнира
function rebuy(bytes32 tournamentId) external payable {
Tournament storage t = tournaments[tournamentId];
require(block.timestamp < t.rebuyDeadline, "Rebuy period ended");
require(msg.value == t.rebuyFee, "Wrong rebuy fee");
PlayerTournamentData storage pd = playerData[tournamentId][msg.sender];
require(pd.rebuys < t.maxRebuys, "Max rebuys reached");
pd.rebuys++;
uint256 rakeAmount = msg.value * t.rake / 10_000;
t.prizePool += msg.value - rakeAmount;
treasury.transfer(rakeAmount);
// Game server получает сигнал: сбросить очки игрока до стартовых
emit RebuyPurchased(tournamentId, msg.sender, pd.rebuys);
}
Satellite tournaments: вход в крупные турниры через мелкие
Классическая poker механика: выиграй в $5 сателлите → получи билет на $100 турнир. В крипто: prize — NFT ticket или SBT (Soulbound Token), дающий право входа.
// Prize: NFT tournament ticket
function distributeSatellitePrizes(
bytes32 satelliteId,
address[] calldata winners,
bytes calldata signature
) external {
_verifyRankingSignature(satelliteId, winners, signature);
Tournament storage sat = tournaments[satelliteId];
require(block.timestamp > sat.endTime, "Not ended");
uint256 numTickets = sat.prizes.length; // только топ N получают билеты
for (uint256 i = 0; i < numTickets && i < winners.length; i++) {
// Минтим tournament ticket NFT
uint256 ticketId = ticketNFT.mint(winners[i], sat.targetTournamentId);
emit TicketAwarded(satelliteId, winners[i], ticketId, sat.targetTournamentId);
}
}
Fraud prevention и fairness
Collusion detection
В PvP турнирах игроки могут договариваться. On-chain методы ограничены, но:
- Rate limiting: не более 1 ставки в N секунд в турнирные игры
- IP/device fingerprinting на backend (off-chain)
- Ограничение: игроки из одного wallet cluster не могут участвовать в одном турнире
Anti-self-dealing
Оператор не должен иметь возможность манипулировать итогами. Решение: результаты записываются от game server через GAME_SERVER_ROLE, ranking подписывается оператором через multisig (не single signer).
// Многоподписная финализация (2-of-3 среди операторов)
function finalizeRanking(
bytes32 tournamentId,
address[] calldata ranked,
bytes[] calldata signatures // 2 из 3 операторов
) external {
require(_verifyMultisig(tournamentId, ranked, signatures, 2), "Need 2 signatures");
_distributePrizes(tournamentId, ranked);
}
Стек разработки
Контракты: Solidity + Foundry. OpenZeppelin AccessControl. ERC-721 для tournament tickets. Backend: Node.js + TypeScript. Redis Sorted Sets для leaderboard. PostgreSQL для истории. Real-time: WebSocket сервер. Bull/BullMQ для фоновых задач (score aggregation, finalization). Frontend: React + wagmi + Socket.io client.
| Компонент | Технология |
|---|---|
| Prize pool | Solidity смарт-контракт |
| Score tracking | Redis Sorted Set |
| Leaderboard API | WebSocket + Redis ZRANGE |
| Ranking finalization | Operator multisig + Merkle |
| Tournament tickets | ERC-721 SBT |
Ориентиры по срокам
MVP (leaderboard tournament, один тип игры, автоматическая дистрибьюция): 4–5 недель. Полная система (все типы турниров, rebuy, сателлиты, real-time leaderboard, fraud detection): 10–14 недель.







