Sono obbligato a creare un generatore casuale di numeri casuali (CS) determinabilmente equo (deterministico &) crittografato in PHP. Stiamo eseguendo PHP 5 e PHP 7 non è davvero un'opzione in questo momento. Tuttavia, ho trovato un polyfill per le nuove funzioni CS di PHP 7, quindi ho implementato quella soluzione (https://github.com/paragonie/random_compat).PHP PRNG seminato, deterministico, crittograficamente sicuro (generatore di numeri pseudo-casuali). È possibile?
Ho pensato che srand() potesse essere utilizzato per seed random_int(), ma ora non sono sicuro se questo è il caso. È possibile seminare un CSPRNG? Se può essere seminato, l'output sarà deterministico (stesso risultato casuale, dato lo stesso seme)?
Ecco il mio codice:
require_once($_SERVER['DOCUMENT_ROOT']."/lib/assets/random_compat/lib/random.php");
$seed_a = 8138707157292429635;
$seed_b = 'JuxJ1XLnBKk7gPASR80hJfq5Ey8QWEIc8Bt';
class CSPRNG{
private static $RNGseed = 0;
public function generate_seed_a(){
return random_int(0, PHP_INT_MAX);
}
public function generate_seed_b($length = 35){
$characters = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ';
$randomString = '';
for($i = 0; $i < $length; $i++){
$randomString .= $characters[random_int(0, strlen($characters) - 1)];
}
return $randomString;
}
public function seed($s = 0) {
if($s == 0){
$this->RNGseed = $this->generate_seed_a();
}else{
$this->RNGseed = $s;
}
srand($this->RNGseed);
}
public function generate_random_integer($min=0, $max=PHP_INT_MAX, $pad_zeros = true){
if($this->RNGseed == 0){
$this->seed();
}
$rnd_num = random_int($min, $max);
if($pad_zeros == true){
$num_digits = strlen((string)$max);
$format_str = "%0".$num_digits."d";
return sprintf($format_str, $rnd_num);
}else{
return $rnd_num;
}
}
public function drawing_numbers($seed_a, $num_of_balls = 6){
$this->seed($seed_a);
$draw_numbers = array();
for($i = 0; $i < $num_of_balls; $i++) {
$number = ($this->generate_random_integer(1, 49));
if(in_array($number, $draw_numbers)){
$i = $i-1;
}else{
array_push($draw_numbers, $number);
}
}
sort($draw_numbers);
return $draw_numbers;
}
}
$CSPRNG= new CSPRNG();
echo '<p>Seed A: '.$seed_a.'</p>';
echo '<p>Seed B: '.$seed_b.'</p>';
$hash = hash('sha1', $seed_a.$seed_b);
echo '<p>Hash: '.$hash.'</p>';
$drawNumbers = $CSPRNG->drawing_numbers($seed_a);
$draw_str = implode("-", $drawNumbers);
echo "<br>Drawing: $draw_str<br>";
Quando viene eseguito questo codice, il disegno ($ draw_str) dovrebbe essere la stessa su ogni esecuzione, ma non lo è.
Per dimostrare che il disegno è giusto, un seme (seme A) viene scelto prima che il numero di vincita venga selezionato e mostrato. Viene generato anche un altro numero casuale (seme B). Il seme B è usato come un sale e combinato con seme A e il risultato è un hash. Questo hash viene mostrato all'utente prima del disegno. Verrà inoltre fornito il codice sorgente in modo che, quando viene selezionato il numero vincente, vengano rivelati entrambi i semi. Possono verificare che le corrispondenze hash e tutto è stato fatto in modo equo.
Vuoi dire https://github.com/paragonie/seedspring? –
Sì, ma richiede PHP 7. Non un'opzione. – compcentral
Posso facilmente regolare il compositore.json per richiedere^5.6 |^7.0 e random_compat: \ –