Я просто розглядав, як вирішити цю саму проблему, але також хочу, щоб моя функція створила маркер, який можна використовувати і для пошуку паролів. Це означає, що мені потрібно обмежити здатність лексеми здогадуватися. Оскільки uniqid
ґрунтується на часі, а за даними php.net "повернене значення мало відрізняється від microtime ()", uniqid
не відповідає критеріям. PHP рекомендує openssl_random_pseudo_bytes()
замість цього генерувати криптографічно захищені маркери.
Швидка, коротка та точна відповідь:
bin2hex(openssl_random_pseudo_bytes($bytes))
який генерує випадкову рядок буквено-цифрових символів довжиною = $ байт * 2. На жаль, у цього лише алфавіт [a-f][0-9]
, але він працює.
Нижче наведена найсильніша функція, яку я міг би зробити, що відповідає критеріям (Це реалізована версія відповіді Еріка).
function crypto_rand_secure($min, $max)
{
$range = $max - $min;
if ($range < 1) return $min; // not so random...
$log = ceil(log($range, 2));
$bytes = (int) ($log / 8) + 1; // length in bytes
$bits = (int) $log + 1; // length in bits
$filter = (int) (1 << $bits) - 1; // set all lower bits to 1
do {
$rnd = hexdec(bin2hex(openssl_random_pseudo_bytes($bytes)));
$rnd = $rnd & $filter; // discard irrelevant bits
} while ($rnd > $range);
return $min + $rnd;
}
function getToken($length)
{
$token = "";
$codeAlphabet = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
$codeAlphabet.= "abcdefghijklmnopqrstuvwxyz";
$codeAlphabet.= "0123456789";
$max = strlen($codeAlphabet); // edited
for ($i=0; $i < $length; $i++) {
$token .= $codeAlphabet[crypto_rand_secure(0, $max-1)];
}
return $token;
}
crypto_rand_secure($min, $max)
працює як падіння заміни на rand()
або mt_rand
. Він використовує openssl_random_pseudo_bytes, щоб допомогти створити випадкове число між $ min та $ max.
getToken($length)
створює алфавіт для використання в токені, а потім створює рядок довжиною $length
.
EDIT: Я знехтував цитувати джерело - http://us1.php.net/manual/en/function.openssl-random-pseudo-bytes.php#104322
EDIT (PHP7): Після випуску PHP7 стандартна бібліотека тепер має дві нові функції, які можуть замінити / покращити / спростити функцію crypto_rand_secure вище. random_bytes($length)
іrandom_int($min, $max)
http://php.net/manual/en/function.random-bytes.php
http://php.net/manual/en/function.random-int.php
Приклад:
function getToken($length){
$token = "";
$codeAlphabet = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
$codeAlphabet.= "abcdefghijklmnopqrstuvwxyz";
$codeAlphabet.= "0123456789";
$max = strlen($codeAlphabet);
for ($i=0; $i < $length; $i++) {
$token .= $codeAlphabet[random_int(0, $max-1)];
}
return $token;
}