mirror of
https://github.com/ramsey/uuid.git
synced 2026-06-15 16:07:55 +03:00
a252f29847
As documented in https://wiki.php.net/rfc/use_global_elements, the engine (by default) does a local namespace lookup, then falls back to global namespace when first calling a global function referenced in namespaced code, unless that function is referenced via fully qualified name (FQN). By using the FQN, the actual symbol can be looked up at compile-time, both by the PHP engine and by static analysis tooling, allowing for compiler (in particular) optimizations to replace known hot-path functions with specialized opcodes. Sadly, no actual benchmark at hand: the improvements can be minimal or massive, depending on where this library is used (tight loops being most relevant).
190 lines
6.5 KiB
PHP
190 lines
6.5 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Ramsey\Uuid\Test\Converter\Time;
|
|
|
|
use Brick\Math\BigInteger;
|
|
use Mockery;
|
|
use Ramsey\Uuid\Converter\Time\GenericTimeConverter;
|
|
use Ramsey\Uuid\Converter\Time\PhpTimeConverter;
|
|
use Ramsey\Uuid\Exception\InvalidArgumentException;
|
|
use Ramsey\Uuid\Math\BrickMathCalculator;
|
|
use Ramsey\Uuid\Test\TestCase;
|
|
use Ramsey\Uuid\Type\Hexadecimal;
|
|
|
|
use function sprintf;
|
|
|
|
class PhpTimeConverterTest extends TestCase
|
|
{
|
|
public function testCalculateTimeReturnsArrayOfTimeSegments(): void
|
|
{
|
|
$seconds = BigInteger::of(5);
|
|
$microSeconds = BigInteger::of(3);
|
|
|
|
$calculatedTime = BigInteger::zero()
|
|
->plus($seconds->multipliedBy(10000000))
|
|
->plus($microSeconds->multipliedBy(10))
|
|
->plus(BigInteger::fromBase('01b21dd213814000', 16));
|
|
|
|
$maskLow = BigInteger::fromBase('ffffffff', 16);
|
|
$maskMid = BigInteger::fromBase('ffff', 16);
|
|
$maskHi = BigInteger::fromBase('0fff', 16);
|
|
|
|
$expected = sprintf('%04s', $calculatedTime->shiftedRight(48)->and($maskHi)->toBase(16));
|
|
$expected .= sprintf('%04s', $calculatedTime->shiftedRight(32)->and($maskMid)->toBase(16));
|
|
$expected .= sprintf('%08s', $calculatedTime->and($maskLow)->toBase(16));
|
|
|
|
$converter = new PhpTimeConverter();
|
|
$returned = $converter->calculateTime((string) $seconds, (string) $microSeconds);
|
|
|
|
$this->assertSame($expected, $returned->toString());
|
|
}
|
|
|
|
public function testCalculateTimeThrowsExceptionWhenSecondsIsNotOnlyDigits(): void
|
|
{
|
|
/** @var Mockery\MockInterface & PhpTimeConverter $converter */
|
|
$converter = Mockery::mock(PhpTimeConverter::class)->makePartial();
|
|
|
|
$this->expectException(InvalidArgumentException::class);
|
|
$this->expectExceptionMessage(
|
|
'Value must be a signed integer or a string containing only digits '
|
|
. '0-9 and, optionally, a sign (+ or -)'
|
|
);
|
|
|
|
$converter->calculateTime('12.34', '5678');
|
|
}
|
|
|
|
public function testCalculateTimeThrowsExceptionWhenMicroSecondsIsNotOnlyDigits(): void
|
|
{
|
|
/** @var Mockery\MockInterface & PhpTimeConverter $converter */
|
|
$converter = Mockery::mock(PhpTimeConverter::class)->makePartial();
|
|
|
|
$this->expectException(InvalidArgumentException::class);
|
|
$this->expectExceptionMessage(
|
|
'Value must be a signed integer or a string containing only digits '
|
|
. '0-9 and, optionally, a sign (+ or -)'
|
|
);
|
|
|
|
$converter->calculateTime('1234', '56.78');
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideConvertTime
|
|
*/
|
|
public function testConvertTime(Hexadecimal $uuidTimestamp, string $unixTimestamp, string $microSeconds): void
|
|
{
|
|
$calculator = new BrickMathCalculator();
|
|
$fallbackConverter = new GenericTimeConverter($calculator);
|
|
$converter = new PhpTimeConverter($calculator, $fallbackConverter);
|
|
|
|
$result = $converter->convertTime($uuidTimestamp);
|
|
|
|
$this->assertSame($unixTimestamp, $result->getSeconds()->toString());
|
|
$this->assertSame($microSeconds, $result->getMicroSeconds()->toString());
|
|
}
|
|
|
|
/**
|
|
* @phpcsSuppress SlevomatCodingStandard.TypeHints.ReturnTypeHint.MissingTraversableTypeHintSpecification
|
|
*/
|
|
public function provideConvertTime(): array
|
|
{
|
|
return [
|
|
[
|
|
'uuidTimestamp' => new Hexadecimal('1e1c57dff6f8cb0'),
|
|
'unixTimestamp' => '1341368074',
|
|
'microSeconds' => '491000',
|
|
],
|
|
[
|
|
'uuidTimestamp' => new Hexadecimal('1ea333764c71df6'),
|
|
'unixTimestamp' => '1578612359',
|
|
'microSeconds' => '521023',
|
|
],
|
|
[
|
|
'uuidTimestamp' => new Hexadecimal('fffffffff9785f6'),
|
|
'unixTimestamp' => '103072857659',
|
|
'microSeconds' => '999999',
|
|
],
|
|
|
|
// This is the last possible time supported by v1 UUIDs. When
|
|
// converted to a Unix timestamp, the microseconds are lost.
|
|
// 60038-03-11 05:36:10.955161
|
|
[
|
|
'uuidTimestamp' => new Hexadecimal('fffffffffffffffa'),
|
|
'unixTimestamp' => '1832455114570',
|
|
'microSeconds' => '955161',
|
|
],
|
|
|
|
// This is the earliest possible date supported by v1 UUIDs:
|
|
// 1582-10-15 00:00:00.000000
|
|
[
|
|
'uuidTimestamp' => new Hexadecimal('000000000000'),
|
|
'unixTimestamp' => '-12219292800',
|
|
'microSeconds' => '0',
|
|
],
|
|
|
|
// This is the Unix epoch:
|
|
// 1970-01-01 00:00:00.000000
|
|
[
|
|
'uuidTimestamp' => new Hexadecimal('1b21dd213814000'),
|
|
'unixTimestamp' => '0',
|
|
'microSeconds' => '0',
|
|
],
|
|
];
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideCalculateTime
|
|
*/
|
|
public function testCalculateTime(string $seconds, string $microSeconds, string $expected): void
|
|
{
|
|
$calculator = new BrickMathCalculator();
|
|
$fallbackConverter = new GenericTimeConverter($calculator);
|
|
$converter = new PhpTimeConverter($calculator, $fallbackConverter);
|
|
|
|
$result = $converter->calculateTime($seconds, $microSeconds);
|
|
|
|
$this->assertSame($expected, $result->toString());
|
|
}
|
|
|
|
/**
|
|
* @phpcsSuppress SlevomatCodingStandard.TypeHints.ReturnTypeHint.MissingTraversableTypeHintSpecification
|
|
*/
|
|
public function provideCalculateTime(): array
|
|
{
|
|
return [
|
|
[
|
|
'seconds' => '-12219146756',
|
|
'microSeconds' => '0',
|
|
'expected' => '000001540901e600',
|
|
],
|
|
[
|
|
'seconds' => '103072857659',
|
|
'microseconds' => '999999',
|
|
'expected' => '0fffffffff9785f6',
|
|
],
|
|
[
|
|
'seconds' => '1578612359',
|
|
'microseconds' => '521023',
|
|
'expected' => '01ea333764c71df6',
|
|
],
|
|
|
|
// This is the earliest possible date supported by v1 UUIDs:
|
|
// 1582-10-15 00:00:00.000000
|
|
[
|
|
'seconds' => '-12219292800',
|
|
'microSeconds' => '0',
|
|
'expected' => '0000000000000000',
|
|
],
|
|
|
|
// This is the last possible time supported by v1 UUIDs:
|
|
// 60038-03-11 05:36:10.955161
|
|
[
|
|
'seconds' => '1832455114570',
|
|
'microseconds' => '955161',
|
|
'expected' => 'fffffffffffffffa',
|
|
],
|
|
];
|
|
}
|
|
}
|