PHP функция для генерации v4 UUID
Так что я немного покопался и пытался собрать воедино функцию, которая генерирует действительный UUID v4 в PHP. Это самое близкое, что мне удалось приехать. Мои знания в шестнадцатеричном, десятичном, двоичном, побитовых операторах PHP и тому подобном практически не существуют. Эта функция генерирует действительный UUID v4 вплоть до одной области. UUID версии 4 должен иметь вид:
xxxxxxxx-xxxx-4 xxx-y xxx-xxxxxxxxxxxx
где y равно 8, 9, A или B. Это то место, где происходит сбой функций, поскольку он не придерживается этого.
Я надеялся, что кто-то, обладающий большими знаниями в этой области, чем я, сможет протянуть мне руку и помочь мне исправить эту функцию, чтобы она соответствовала этому правилу.
Функция выглядит следующим образом:
<?php
function gen_uuid() {
$uuid = array(
'time_low' => 0,
'time_mid' => 0,
'time_hi' => 0,
'clock_seq_hi' => 0,
'clock_seq_low' => 0,
'node' => array()
);
$uuid['time_low'] = mt_rand(0, 0xffff) + (mt_rand(0, 0xffff) << 16);
$uuid['time_mid'] = mt_rand(0, 0xffff);
$uuid['time_hi'] = (4 << 12) | (mt_rand(0, 0x1000));
$uuid['clock_seq_hi'] = (1 << 7) | (mt_rand(0, 128));
$uuid['clock_seq_low'] = mt_rand(0, 255);
for ($i = 0; $i < 6; $i++) {
$uuid['node'][$i] = mt_rand(0, 255);
}
$uuid = sprintf('%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x',
$uuid['time_low'],
$uuid['time_mid'],
$uuid['time_hi'],
$uuid['clock_seq_hi'],
$uuid['clock_seq_low'],
$uuid['node'][0],
$uuid['node'][1],
$uuid['node'][2],
$uuid['node'][3],
$uuid['node'][4],
$uuid['node'][5]
);
return $uuid;
}
?>
Спасибо всем, кто может мне помочь.
19 ответов
Из этого комментария к руководству по PHP вы можете использовать это:
function gen_uuid() {
return sprintf( '%04x%04x-%04x-%04x-%04x-%04x%04x%04x',
// 32 bits for "time_low"
mt_rand( 0, 0xffff ), mt_rand( 0, 0xffff ),
// 16 bits for "time_mid"
mt_rand( 0, 0xffff ),
// 16 bits for "time_hi_and_version",
// four most significant bits holds version number 4
mt_rand( 0, 0x0fff ) | 0x4000,
// 16 bits, 8 bits for "clk_seq_hi_res",
// 8 bits for "clk_seq_low",
// two most significant bits holds zero and one for variant DCE1.1
mt_rand( 0, 0x3fff ) | 0x8000,
// 48 bits for "node"
mt_rand( 0, 0xffff ), mt_rand( 0, 0xffff ), mt_rand( 0, 0xffff )
);
}
Вместо того, чтобы разбивать его на отдельные поля, проще генерировать случайный блок данных и изменять отдельные позиции байтов. Вам также следует использовать лучший генератор случайных чисел, чем mt_rand().
Согласно RFC 4122 - раздел 4.4, вам необходимо изменить эти поля:
time_hi_and_version
(биты 4-7 7-го октета),clock_seq_hi_and_reserved
(биты 6 и 7 из 9-го октета)
Все остальные 122 бита должны быть достаточно случайными.
Следующий подход генерирует 128 бит случайных данных, используя openssl_random_pseudo_bytes()
, делает перестановки на октеты, а затем использует bin2hex()
а также vsprintf()
сделать окончательное форматирование.
function guidv4($data)
{
assert(strlen($data) == 16);
$data[6] = chr(ord($data[6]) & 0x0f | 0x40); // set version to 0100
$data[8] = chr(ord($data[8]) & 0x3f | 0x80); // set bits 6-7 to 10
return vsprintf('%s%s-%s-%s-%s-%s%s%s', str_split(bin2hex($data), 4));
}
echo guidv4(openssl_random_pseudo_bytes(16));
В PHP 7 генерировать случайные последовательности байтов еще проще, используя random_bytes()
:
echo guidv4(random_bytes(16));
Любой, кто использует зависимости композитора, вы можете рассмотреть эту библиотеку: https://github.com/ramsey/uuid
Это не становится легче, чем это:
Uuid::uuid4();
В системах Unix используйте ядро системы, чтобы сгенерировать для вас uuid.
file_get_contents('/proc/sys/kernel/random/uuid')
Кредит Samveen на /questions/8998271/vs2008-molcha-umiraet-mnogo-pri-rabote-s-wpf/8998296#8998296
Примечание!: Использование этого метода для получения uuid действительно очень быстро истощает пул энтропии! Я бы не использовал это там, где это будет часто вызываться.
$uuid = vsprintf('%s%s-%s-%s-%s-%s%s%s', str_split(bin2hex(random_bytes(16)), 4));
Небольшое изменение в ответе Джека о добавлении поддержки PHP < 7:
// Get an RFC-4122 compliant globaly unique identifier
function get_guid() {
$data = PHP_MAJOR_VERSION < 7 ? openssl_random_pseudo_bytes(16) : random_bytes(16);
$data[6] = chr(ord($data[6]) & 0x0f | 0x40); // Set version to 0100
$data[8] = chr(ord($data[8]) & 0x3f | 0x80); // Set bits 6-7 to 10
return vsprintf('%s%s-%s-%s-%s-%s%s%s', str_split(bin2hex($data), 4));
}
В своем поиске создания vuuuid v4 я сначала пришел на эту страницу, а затем нашел ее на http://php.net/manual/en/function.com-create-guid.php
function guidv4()
{
if (function_exists('com_create_guid') === true)
return trim(com_create_guid(), '{}');
$data = openssl_random_pseudo_bytes(16);
$data[6] = chr(ord($data[6]) & 0x0f | 0x40); // set version to 0100
$data[8] = chr(ord($data[8]) & 0x3f | 0x80); // set bits 6-7 to 10
return vsprintf('%s%s-%s-%s-%s-%s%s%s', str_split(bin2hex($data), 4));
}
кредит: павел.волинцев
Изменить: чтобы уточнить, эта функция всегда даст вам vuuid v4 (PHP >= 5.3.0).
Когда функция com_create_guid доступна (обычно только в Windows), она использует ее и удаляет фигурные скобки.
Если его нет (Linux), он обратится к этой сильной случайной функции openssl_random_pseudo_bytes, а затем использует vsprintf для форматирования в v4 uuid.
Если вы используете CakePHP
Вы можете использовать их метод CakeText::uuid();
из класса CakeText для создания uuid RFC4122.
Криптографически безопасный UUID v4 для PHP >= 7.
<?php
function uuid4() {
/* 32 random HEX + space for 4 hyphens */
$out = bin2hex(random_bytes(18));
$out[8] = "-";
$out[13] = "-";
$out[18] = "-";
$out[23] = "-";
/* UUID v4 */
$out[14] = "4";
/* variant 1 - 10xx */
$out[19] = ["8", "9", "a", "b"][random_int(0, 3)];
return $out;
}
echo uuid4();
выход:c68469d2-065b-4b17-b36f-5c40efb5f6cd
Используйте Symfony Polyfill / Uuid
Затем вы можете просто сгенерировать uuid как собственную функцию php:
$uuid = uuid_create(UUID_TYPE_RANDOM);
Подробнее об этом читайте в официальном сообщении Symfony blop - https://symfony.com/blog/introducing-the-new-symfony-uuid-polyfill
Мой ответ основан на комментарии пользователя uniqid, но он использует функцию openssl_random_pseudo_bytes для генерации случайной строки вместо чтения из /dev/urandom
function guid()
{
$randomString = openssl_random_pseudo_bytes(16);
$time_low = bin2hex(substr($randomString, 0, 4));
$time_mid = bin2hex(substr($randomString, 4, 2));
$time_hi_and_version = bin2hex(substr($randomString, 6, 2));
$clock_seq_hi_and_reserved = bin2hex(substr($randomString, 8, 2));
$node = bin2hex(substr($randomString, 10, 6));
/**
* Set the four most significant bits (bits 12 through 15) of the
* time_hi_and_version field to the 4-bit version number from
* Section 4.1.3.
* @see http://tools.ietf.org/html/rfc4122#section-4.1.3
*/
$time_hi_and_version = hexdec($time_hi_and_version);
$time_hi_and_version = $time_hi_and_version >> 4;
$time_hi_and_version = $time_hi_and_version | 0x4000;
/**
* Set the two most significant bits (bits 6 and 7) of the
* clock_seq_hi_and_reserved to zero and one, respectively.
*/
$clock_seq_hi_and_reserved = hexdec($clock_seq_hi_and_reserved);
$clock_seq_hi_and_reserved = $clock_seq_hi_and_reserved >> 2;
$clock_seq_hi_and_reserved = $clock_seq_hi_and_reserved | 0x8000;
return sprintf('%08s-%04s-%04x-%04x-%012s', $time_low, $time_mid, $time_hi_and_version, $clock_seq_hi_and_reserved, $node);
} // guid
preg_replace_callback('/[xy]/', function ($matches)
{
return dechex('x' == $matches[0] ? mt_rand(0, 15) : (mt_rand(0, 15) & 0x3 | 0x8));
}
, 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx');
Или, если вы не можете использовать анонимные функции.
preg_replace_callback('/[xy]/', create_function(
'$matches',
'return dechex("x" == $matches[0] ? mt_rand(0, 15) : (mt_rand(0, 15) & 0x3 | 0x8));'
)
, 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx');
Я искал точно такую же вещь и почти реализовал ее версию, я подумал, что стоит упомянуть, что, если вы делаете это в рамках WordPress, у WP есть своя супер-удобная функция именно для этого:
$myUUID = wp_generate_uuid4();
Вы можете прочитать описание и источник здесь.
Как насчет использования mysql для генерации uuid для вас?
$conn = new mysqli($servername, $username, $password, $dbname, $port);
$query = 'SELECT UUID()';
echo $conn->query($query)->fetch_row()[0];
Это может быть проще?
$uuid = bin2hex(openssl_random_pseudo_bytes(16));
for($cnt = 8; $cnt <=23; $cnt+=5)
$uuid = substr($uuid, 0, $cnt) . "-" . substr($uuid, $cnt);
echo $uuid . "\n";
Я уверен, что есть более элегантный способ сделать преобразование из двоичного в десятичное для 4xxx
а также yxxx
части. Но если вы хотите использовать openssl_random_pseudo_bytes
в качестве вашего криптографически безопасного генератора чисел вот что я использую:
return sprintf('%s-%s-%04x-%04x-%s',
bin2hex(openssl_random_pseudo_bytes(4)),
bin2hex(openssl_random_pseudo_bytes(2)),
hexdec(bin2hex(openssl_random_pseudo_bytes(2))) & 0x0fff | 0x4000,
hexdec(bin2hex(openssl_random_pseudo_bytes(2))) & 0x3fff | 0x8000,
bin2hex(openssl_random_pseudo_bytes(6))
);
От Тома, на http://www.php.net/manual/en/function.uniqid.php
$r = unpack('v*', fread(fopen('/dev/random', 'r'),16));
$uuid = sprintf('%04x%04x-%04x-%04x-%04x-%04x%04x%04x',
$r[1], $r[2], $r[3], $r[4] & 0x0fff | 0x4000,
$r[5] & 0x3fff | 0x8000, $r[6], $r[7], $r[8])
Можно использовать функцию uuid_create, доступную в PHP 7.2 и более поздних версиях.
$uuid = uuid_create();
echo $uuid;
Просто идея, но в конечном итоге для получения GUID V4 я использовал сервер базы данных. Я использую SQL Server, и в сценарии, где мне нужен GUID, я уже выполнял запрос, поэтому я просто добавил newid() в качестве одного из полей результатов запроса. Это дало мне GUID V4, в котором я нуждался.
Это, очевидно, зависит от сервера базы данных, который вы используете, и от того, что еще происходит в коде, где вам нужен GUID (и сколько GUID вам нужно), но если ваш сервер БД генерирует GUIDv4, и особенно если вы используете в любом случае, это быстрый и простой способ получить ваш GUID, не зависящий от версии PHP.