Игры Профиль Чат ? Бонус Еще
Режим
Server seed
Client seed
Salt

Перевод сидов в байты


Перевод байтов в числа


Перевод чисел в результат


Доказуемая честность


Основополагающая концепция доказуемой честности заключается в том, что игроки имеют возможность доказать и подтвердить, что результаты их игр являются честными и не подвергались манипуляциям со стороны сайта. Это достигается за счет использования так называемой схемы обязательства, где случайные результаты игр генерируются на основе набора параметров, часть которых известна игроку перед началом раунда. Схема обязательства используется, чтобы гарантировать, что игрок оказывает влияние на все полученные результаты, вместе с тем параметры подобраны таким образом, чтобы игрок не мог предугадать результат той или иной игры наперед, но всегда мог детерминированным образом подтвердить факт честности данной игры.

Параметры генератора случайных чисел


В предложенной системе в качестве входных параметров генератора случайных чисел используется следующий набор из четырех параметров

Server seed

Серверный сид генерируется нашей системой и представляет собой 32 байта, представленные в виде 64-символьной строки. Для режимов, в которых в одном раунде участвует только один игрок, у игрока есть возможность запросить у системы генерацию нового серверного сида, новое значение которого будет сразу же показано. К этим режимам относятся:

  • Mines
  • Dice
  • Overgo

Для режимов, в которых в одном раунде участвуют несколько игроков, у отдельного игрока нет возможности запросить у системы генерацию нового серверного сида (это вызвано тем, что системе неясно, запрос какого из игроков удовлетворить), однако гарантией честности в данных режимах является тот факт, что серверный сид является постоянным, то есть не меняется от раунда к раунду. В качестве значения серверного сида взят заранее посчитанный хеш определенного bitcoin-блока. Режимы и соответствующие им постоянные серверные сиды приведены ниже:

Client seed

Клиентский сид принадлежит игроку и используется для того, чтобы игрок имел возможность оказывать влияние на генерируемые числа. Изначально клиентский сид генерируется системой при регистрации на сайте, однако в любой момент у пользователя есть возможность сменить свой клиентский сид (обратите внимание, что, в отличие от серверного сида, есть возможность ввести значение желаемого клиентского сида). Для режимов, в которых в одном раунде участвует только один игрок, в качестве клиентского сида используется пользовательский клиентский сид. Эти режимы приведены ниже:

  • Mines
  • Dice

Для режимов, в которых в одном раунде участвуют несколько игроков, в качестве клиентского сида не может использоваться пользовательский клиентский сид (это вызвано тем, что системе неясно, сид какого из игроков использовать), однако гарантией честности в данных режимах является тот факт, что клиентский сид является постоянным, то есть не меняется от раунда к раунду. В качестве значения клиентского сида взят заранее посчитанный хеш определенного bitcoin-блока. Режимы и соответствующие им постоянные клиентские сиды приведены ниже:

Salt

Поскольку серверный сид и клиентский сид постоянны и известны пользователю, параметр salt (соль) используется в схеме для того, чтобы получать разные результаты с одной и той же парой client seed + server seed. Соль представляет собой случайные 32 байта, представленные в виде 64-символьной строки. Данный параметр уникален для каждого раунда и недоступен игроку перед началом игры, однако отображается в конце игры (так называемая фаза раскрытия в схеме обязательства).

Cursor

Генератор случайных чисел, используя в качестве входных параметров server seed, client seed и salt, генерирует случайные 64 байта, представленные в виде 128-символьной строки. Эта строка используется для генерации 8 случайных чисел. Однако некоторые режимы требуют генерации большего количества чисел. Для этого вводится дополнительный параметр cursor, который изначательно равен 0 и увеличивается на единицу каждый раз, когда необходимо сгенерировать очередные 8 чисел в рамках одного раунда. Режимы, которые используют больше одного курсора:

  • Mines (3 курсора)

Режимы, которые используют ровно один курсор:

  • Dice
  • Double
  • Jackpot
  • Crash
  • Overgo

Генератор случайных чисел


Генерация случайных байт на основе входных параметров осуществляется с использованием функций HMAC-SHA512 и SHA256

bytes = HMAC-SHA512(SHA256(server_seed:salt), client_seed:cursor)

Результатом генерации являются случайные 64 байта, которые затем делятся на 8 блоков по 8 байт, и каждый блок используется для генерации случайного числа в диапазоне [0, 1). Алгоритм конвертации блока в число приведен в разделе "Реализация", а также графически демонстрируется при проверке игры.

Батарея тестов Dieharder


Батарея тестов Dieharder — это набор статистических тестов для измерения качества генератора случайных чисел. Она состоит из множества тестов (включая набор тестов Diehard и набор тестов NIST Test Suite), которые вместе рассматриваются как один из наиболее строгих существующих наборов тестов. Мы сгенерировали 1 миллиард случайных чисел с помощью вышеприведенного генератора и протестировали этот набор с помощью Dieharder со следующими опциями

dieharder -g 202 -f ./rng_output_test_random_CS -a |& tee -a output_dh_random_CS.txt

Для большей статистической значимости мы запустили тест дважды на разных наборах. Результаты тестирования показывают, что приведенный генератор случайных чисел действительно генерирует несмещенные случайные последовательности, в которых отсутствуют статистически значимые закономерности. Отчеты в виде текстовых файлов доступны по ссылкам:
output_dh_random_CS_1.txt
output_dh_random_CS_2.txt

Генерация случайных чисел


Для любого раунда, client seed, server seed, salt и cursor служат единственными входными параметрами генератора. Генератор преобразует эти параметры в последовательность случайных байтов с помощью хеш-функций HMAC-SHA512 и SHA256, используя свойство лавинного эффекта

function seedsToBytes(serverSeed, clientSeed, salt, cursor) {
    // SHA-256 calculation
    const hmacKey = createHash(serverSeed + ":" + salt);
    const hmacMessage = clientSeed + ":" + cursor;
    // HMAC calculation
    return createHmac(hmacKey, hmacMessage);
}

Перевод байтов в числа


Результатом работы генератора являются 64 случайных байта (128-символьная строка), которые разбиваются на 8 блоков по 8 байт каждый, и каждый блок используется для получения одного числа в интервале [0, 1) с помощью нижеприведенного алгоритма.

function numbersFromBytes(bytes) {
    for (let t = 0; t < 8; t++) {
        let value = 0;
        let pw = 256;
        for (let i = t * 8; i < t * 8 + 8; i++) {
            value += bytes[i] / pw;
            pw *= 256;
        }
        result.push(value);
    }
    return result;
}

Генерация результата Mines


Для генерации результата в режиме Mines требуется 24 случайных числа в интервале [0..1), поэтому генерируется 3 массива из 8 чисел, которые затем объединяются в один массив, называемый индексным. Данный массив используется для выбора позиций мин на поле с помощью алгоритма Фишера-Йетса

// numbers - index array
function minesShuffling(numbers, minesCount) {
    // Initial array
    let range = [];
    // Resulting array
    let permutation = [];
    for (var i = 0; i <= 24; i++) range.push(i);
    for (var i = 0; i < numbers.length; i++) { 
        let index = Math.floor(numbers[i] * range.length);
        permutation.push(range.splice(index, 1)[0]);
    }
    return permutation;
}

Генерация результата Dice


Для генерации результата в режиме Dice требуется 1 случайное число в интервале [0..1), которое затем переводится в результат Dice в интервале [0..100] по следующему алгоритму.

// number has interval [0..1)
function diceOutcome(number) {
    return Math.floor(number * 10001) / 100;
}

Генерация результата Double


Для генерации результата в режиме Double требуется 1 случайное число в интервале [0..1), которое затем переводится в сектор колеса в интервале [0..14] по следующему алгоритму.

// number has interval [0..1)
function doubleOutcome(number) {
    return Math.floor(number * 15);
}

Генерация результата Jackpot


Для генерации результата в режиме Jackpot требуется 1 случайное число в интервале [0..1), которое затем переводится в выигрышный билет путем арифметического умножения на общее число билетов, округленного вниз, с поправкой на единицу (поскольку нумерация билетов начинается с 1).

// number has interval [0..1)
function jackpotOutcome(number, ticketsCount) {
    return Math.floor(number * ticketsCount) + 1;
}

Генерация результата Crash


Для генерации результата в режиме Crash требуется 1 случайное число в интервале [0..1), которое затем переводится в коэффициент Crash, имеющий экспоненциальное распределение, по следующему алгоритму.

// number has interval [0..1)
function crashOutcome(number) {
    return max(1, 1000000 / (Math.floor(number * 1000000) + 1) * (1 - 0.05)).toFixed(2);
}

Генерация результата Overgo


Для генерации результата в режиме Overgo требуется 1 случайное число в интервале [0..1), которое затем переводится в коэффициент Overgo, имеющий экспоненциальное распределение, по следующему алгоритму.

// number has interval [0..1)
function overgoOutcome(number) {
    return max(1, 1000000 / (Math.floor(number * 1000000) + 1) * (1 - 0.03)).toFixed(2);
}
Онлайн чат
Quiz
Приз:
дал правильный ответ:
Женя Лоскутов
Егор Попов, за 3 минуты бывало. Бывало и за 15 часов
Василий Лавренов
Алексея перетопили и на 7 к выебали
Максим Новоселов
заебись
Максим Новоселов
закинул 185 вывожу 430
Максим Новоселов
Ух
Максим Новоселов
Спс мани икс
Хабиб Миронов
Максим Новоселов, на каких режимах играешь?
Максим Новоселов
Краш мины
Максим Новоселов
И доубле
Хабиб Миронов
Максим Новоселов, на минах просто играешь или больште х
Максим Новоселов
Просто
Хабиб Миронов
Хорош чё)
Василий Лавренов
А кто в джекпоте выиграл?
Василий Лавренов
Алексея Скворцова выебали?
Женя Лоскутов
Василий Лавренов, наврятли
Бенедикт Попов
эх
Бенедикт Попов
надо было до 100
Ваня Осипчук
Женя Лоскутов, сливает или держит?
Женя Лоскутов
Ваня Осипчук, чуть дало и слил и держит
Ваня Осипчук
Женя Лоскутов, понял принял
Сергей Федосеев
Хахахахахахха
Бенедикт Попов
ебу
Женя Лоскутов
Ваня Осипчук, аахах накаркал
Абдулла Алимов
Женя Лоскутов, оо уже депнул?
Ваня Осипчук
Женя Лоскутов, вижу)
Абдулла Алимов
Женя Лоскутов, скок баланс
Женя Лоскутов
Абдулла Алимов, ужк депнул и уже осталось всего половина депа ахах
Абдулла Алимов
Женя Лоскутов, бля
Ваня Осипчук
Женя Лоскутов, а скок деп?
Абдулла Алимов
А если рили скок баланс в монетах?
Илья Ярошенко
Зел
Абдулла Алимов
Аууу
Абдулла Алимов
Ля краш заебал
Василий Лавренов
Кто в долг даст 120 руб?
Василий Лавренов
Прям в долг по дпроценты
Василий Лавренов
150 должен буду ...
Женя Лоскутов
Фух деп откамбекал
Абдулла Алимов
Женя Лоскутов, кросс
Василий Лавренов
Женя Лоскутов, дашь в долг?
Абдулла Алимов
Василий Лавренов, смешно
Ваня Осипчук
Женя Лоскутов, хорош
Василий Лавренов
Абдулла Алимов, почему смешно?
Женя Лоскутов
Василий Лавренов, я сам в должниках ахах
Василий Лавренов
Ахахах да да да
Абдулла Алимов
Василий Лавренов, Ему деньги нужны как минимум чтоб долги вернуть +себе
Абдулла Алимов
БЛЯЯЯЯЯЯЯЯ
Абдулла Алимов
СУКАААА
Василий Лавренов
Абдулла Алимов, а ты дашь 200 в долг?
Женя Лоскутов
Василий Лавренов, что ты мне да да дакал?
Василий Лавренов
Я вирну 310
Абдулла Алимов
Неееее я сам жду свой долг гыг
Абдулла Алимов
Женя Лоскутов, баланс скажи
Василий Лавренов
Женя Лоскутов, ну да согланэшался с тобой
Женя Лоскутов
Василий Лавренов, да и я тебе разве раньше сколькоты просил давал тебе? Ты думаешь щас что-то поменялось?
Василий Лавренов
Женя Лоскутов, ну так а что тебе жалко дать?
Василий Лавренов
Я не богатый как бы ..
Женя Лоскутов
Василий Лавренов, вот именно. Потом хуй бабок дождусь)
Абдулла Алимов
Женя Лоскутов, сижу в краше и переживаю за тебя ахахах
Владислав Морозов
Женя Лоскутов, А можно тупо промики собирать?
Владислав Морозов
Женя Лоскутов, А потом на вывод?
Абдулла Алимов
Ля краш сука
0