Доказуемо честный
Введите детали игры, чтобы проверить ваши результаты
Код
Выберите игру выше, чтобы изучить исходный код, который создает ее результаты.
import { createHmac } from 'crypto';
const UINT32_MAX = 0x100000000; // 2^32
/**
* Generates an infinite deterministic stream of cryptographically secure
* 32-bit unsigned integers derived from HMAC-SHA256.
*
* @param key - Secret cryptographic key (server seed). Must remain private until revealed.
* @param message - Public input string (e.g., client seed, nonce, or domain label).
*
* @remarks
* - Uses HMAC-SHA256 as a pseudorandom function (PRF) to produce a reproducible
* and verifiable sequence of numbers.
* - `step` ensures each 32-bit block is derived from a unique input.
* - Output is split into 32-bit chunks, suitable for unbiased integer
* full numeric precision and maximum entropy.
* - Fully deterministic: anyone with the same key and message can verify the results.
*/
function* getUint32Stream(key: string, message: string): Iterator<number> {
let step = 0;
while (true) {
const hmac = createHmac('sha256', key);
hmac.update(`${message}:${step}`);
const hash = hmac.digest(); // 32 bytes (256 bits)
for (let i = 0; i < hash.length; i += 4) {
yield (
(hash[i] << 24) |
(hash[i + 1] << 16) |
(hash[i + 2] << 8) |
hash[i + 3]
) >>> 0; // force unsigned 32-bit
}
step++;
}
}
/**
* Converts a 32-bit random number into an unbiased integer within [0, range).
*
* @param uint32StreamFactory - Function that returns a stream of 32-bit integers.
* @param range - The desired upper bound (exclusive) for the output.
*
* @remarks
* - Uses rejection sampling to avoid modulo bias.
* - Every integer within the range is equally likely.
* - Critical for fair gameplay and provably fair verification.
*/
function getUnbiasedInt(
uint32StreamFactory: () => Iterator<number>,
range: number
): number {
if (!Number.isInteger(range) || range <= 0) {
throw new Error('Invalid range');
}
const maxAcceptable = UINT32_MAX - (UINT32_MAX % range);
const stream = uint32StreamFactory();
let next = stream.next();
while (!next.done) {
const value = next.value;
if (value < maxAcceptable) {
return value % range;
}
next = stream.next();
}
throw new Error('Failed to generate unbiased value: entropy stream exhausted');
}
export const HiloCardRank = ['A', '2', '3', '4', '5', '6', '7', '8', '9', 'T', 'J', 'Q', 'K'] as const;
export const HiloCardSuit = ['c', 'd', 'h', 's'] as const;
function getHiloCard(options: {
serverSeed: string;
clientSeed: string;
nonce: number;
round: number;
}) {
const deck = HiloCardSuit.flatMap(suit =>
HiloCardRank.map(rank => [rank, suit])
);
const uint32Generator = getUint32Stream(options.serverSeed, `${options.clientSeed}:${options.nonce}:${options.round}:hilo`);
const cardIndex = getUnbiasedInt(() => uint32Generator, deck.length);
const card = deck[cardIndex];
return card;
}
// console.log(
// getHiloCard({
// clientSeed: 'player_input_client_seed',
// serverSeed: 'player_input_server_seed',
// nonce: 0,
// round: 1,
// })
// )