v4 UUIDを生成するPHP関数 質問する

v4 UUIDを生成するPHP関数 質問する

そこで、いろいろ調べて、PHP で有効な v4 UUID を生成する関数をまとめようとしました。これが私ができた中で最も近いものです。16 進数、10 進数、2 進数、PHP のビット演算子などについての私の知識はほとんどありません。この関数は、1 つの領域まで有効な v4 UUID を生成します。v4 UUID は次の形式になります。

xxxxxxxxx-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;
}

?>

ベストアンサー1

UUIDv4 を構築するには、128 ビット相当のランダム データを生成し、いくつかのフィールドをパッチしてデータが標準に準拠するようにし、16 進グループとしてフォーマットします。

によるとRFC 4122 - セクション 4.4以下のビットを変更する必要があります:

  1. time_hi_and_version(第7オクテットのビット4-7)
  2. clock_seq_hi_and_reserved(第9オクテットのビット6と7)

以下のコードは与えられたデータの順列を作成し、bin2hex()そしてvsprintf()最終的なフォーマットを行います。

function uuidv4()
{
  $data = 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));
}

PHP 5以前

このrandom_bytes関数は PHP 7.0 で導入されました。古いバージョンの PHP でこれを実装する方法の例を次に示します。

$data = openssl_random_pseudo_bytes(16, $strong);
// ensure the result is cryptographically strong
assert($data !== false && $strong);

ランダムデータの生成

暗号的に強力なランダムデータを生成するには、 を使用する代わりにmt_rand()、次のいずれかを使用することをお勧めします。openssl_random_pseudo_bytes()またはrandom_bytes()(php7 以降) ランダム データを生成する別の方法を選択することもできます。

おすすめ記事