67 lines
2.0 KiB
PHP
67 lines
2.0 KiB
PHP
|
<?php
|
||
|
|
||
|
/*
|
||
|
* This file is part of the Symfony package.
|
||
|
*
|
||
|
* (c) Fabien Potencier <fabien@symfony.com>
|
||
|
*
|
||
|
* For the full copyright and license information, please view the LICENSE
|
||
|
* file that was distributed with this source code.
|
||
|
*/
|
||
|
|
||
|
namespace Symfony\Component\Uid;
|
||
|
|
||
|
/**
|
||
|
* A v6 UUID is lexicographically sortable and contains a 60-bit timestamp and 62 extra unique bits.
|
||
|
*
|
||
|
* Unlike UUIDv1, this implementation of UUIDv6 doesn't leak the MAC address of the host.
|
||
|
*
|
||
|
* @author Nicolas Grekas <p@tchwork.com>
|
||
|
*/
|
||
|
class UuidV6 extends Uuid implements TimeBasedUidInterface
|
||
|
{
|
||
|
protected const TYPE = 6;
|
||
|
|
||
|
private static string $node;
|
||
|
|
||
|
public function __construct(string $uuid = null)
|
||
|
{
|
||
|
if (null === $uuid) {
|
||
|
$this->uid = static::generate();
|
||
|
} else {
|
||
|
parent::__construct($uuid, true);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public function getDateTime(): \DateTimeImmutable
|
||
|
{
|
||
|
return BinaryUtil::hexToDateTime('0'.substr($this->uid, 0, 8).substr($this->uid, 9, 4).substr($this->uid, 15, 3));
|
||
|
}
|
||
|
|
||
|
public function getNode(): string
|
||
|
{
|
||
|
return substr($this->uid, 24);
|
||
|
}
|
||
|
|
||
|
public static function generate(\DateTimeInterface $time = null, Uuid $node = null): string
|
||
|
{
|
||
|
$uuidV1 = UuidV1::generate($time, $node);
|
||
|
$uuid = substr($uuidV1, 15, 3).substr($uuidV1, 9, 4).$uuidV1[0].'-'.substr($uuidV1, 1, 4).'-6'.substr($uuidV1, 5, 3).substr($uuidV1, 18, 6);
|
||
|
|
||
|
if ($node) {
|
||
|
return $uuid.substr($uuidV1, 24);
|
||
|
}
|
||
|
|
||
|
// uuid_create() returns a stable "node" that can leak the MAC of the host, but
|
||
|
// UUIDv6 prefers a truly random number here, let's XOR both to preserve the entropy
|
||
|
|
||
|
if (!isset(self::$node)) {
|
||
|
$seed = [random_int(0, 0xFFFFFF), random_int(0, 0xFFFFFF)];
|
||
|
$node = unpack('N2', hex2bin('00'.substr($uuidV1, 24, 6)).hex2bin('00'.substr($uuidV1, 30)));
|
||
|
self::$node = sprintf('%06x%06x', ($seed[0] ^ $node[1]) | 0x010000, $seed[1] ^ $node[2]);
|
||
|
}
|
||
|
|
||
|
return $uuid.self::$node;
|
||
|
}
|
||
|
}
|