From cf56dad59645dce4712bd8ec650645ba789f83ee Mon Sep 17 00:00:00 2001 From: "Thiago Rodrigues (xthiago)" Date: Mon, 19 Jun 2023 01:02:58 -0300 Subject: [PATCH] add FreezesUuidTrait to help with unit tests --- README.md | 55 +++++++++++++++++++++++++ src/Testing/FreezesUuidTrait.php | 71 ++++++++++++++++++++++++++++++++ 2 files changed, 126 insertions(+) create mode 100644 src/Testing/FreezesUuidTrait.php diff --git a/README.md b/README.md index e7393b1..4262458 100644 --- a/README.md +++ b/README.md @@ -184,6 +184,61 @@ $serializer = new Serializer( ); ``` +### `FreezesUuidTrait` utility to help with tests + +When creating tests, we may want to know the value of the generated IDs beforehand. This lets us write more simpler +assertions. This library provides a trait [`FreezesUuidTrait`](src/Testing/FreezesUuidTrait.php) that could be used +to freeze the uuid v4 generation or to set a known sequence. + +```php +class MyAwesomeTest extends Testcase +{ + use FreezesUuidTrait; + + protected function setUp(): void + { + // we could freeze the uuids here :) + } + + protected function tearDown(): void + { + $this->unfreezeUuid(); // <-- This is important to unfreeze the uuid generation. + } + + public function test_fixed_uuid(): void + { + // Fixing the uuid value (this can also be set on `setUp()` method): + $this->freezeUuidV4WithFixedValue('866cc948-b6de-4cdc-8f5e-3b53a58a9f63'); + + // All generated Id will have the same uuid value. + $this->assertSame('866cc948-b6de-4cdc-8f5e-3b53a58a9f63', (string) Id::generate()); + $this->assertSame('866cc948-b6de-4cdc-8f5e-3b53a58a9f63', (string) Id::generate()); + $this->assertSame('866cc948-b6de-4cdc-8f5e-3b53a58a9f63', (string) Id::generate()); + } + + public function test_fixed_uuid_sequence(): void + { + // Fixing the uuid value with a known sequence: + $this->freezeUuidV4WithKnownSequence( + 'e2d5d0fd-a719-4da7-976d-b5cd184fa615' + '4c98d212-19ed-4c41-9ea2-4b2d48d7410d' + '2acfaf05-25c3-4db9-9fea-5d71a0c3f909' + ); + + // Each new Id instance will assume one uuid from the known sequence: + $this->assertSame('e2d5d0fd-a719-4da7-976d-b5cd184fa615', (string) Id::generate()); + $this->assertSame('4c98d212-19ed-4c41-9ea2-4b2d48d7410d', (string) Id::generate()); + $this->assertSame('2acfaf05-25c3-4db9-9fea-5d71a0c3f909', (string) Id::generate()); + + // If we call again Id::generate(), it will throw RuntimeException because there is no remaining uuid in the + // available list. + $this->assertException(RunTimeException::class); + Id::generate(); + $this->fail('The expected exception was not thrown.'); + } +} +``` + ## Contributing Pull requests are welcome. For major changes, please open an issue first to discuss what you would like to change. diff --git a/src/Testing/FreezesUuidTrait.php b/src/Testing/FreezesUuidTrait.php new file mode 100644 index 0000000..c897d9d --- /dev/null +++ b/src/Testing/FreezesUuidTrait.php @@ -0,0 +1,71 @@ +uuid = $uuid; + parent::__construct(); + } + + public function uuid4(): UuidInterface + { + return Uuid::fromString($this->uuid); + } + }; + + Uuid::setFactory($factory); + } + + private function freezeUuidV4WithKnownSequence(string ...$uuids): void + { + $sequence = new ArrayIterator($uuids); + $factory = new class($sequence) extends UuidFactory { + /** @var Iterator */ + private $uuidSequence; + + public function __construct(Iterator $uuidSequence) + { + $this->uuidSequence = $uuidSequence; + parent::__construct(); + } + + public function uuid4(): UuidInterface + { + if (false === $this->uuidSequence->valid()) { + throw new RuntimeException( + 'The UUID sequence are over. Maybe more UUIDs were used than initially configured.' + ); + } + $uuid = $this->uuidSequence->current(); + $this->uuidSequence->next(); + + return Uuid::fromString($uuid); + } + }; + + Uuid::setFactory($factory); + } + + private function unfreezeUuid(): void + { + Uuid::setFactory(new UuidFactory()); + } +}