From fcec99cd48859db8db38fe6c394e46b80116c583 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 08:03:57 +0200 Subject: [PATCH 01/22] Remove psalm-immutable from readonly classes --- src/Date.php | 1 - src/Month.php | 1 - src/Time.php | 1 - src/Year.php | 1 - 4 files changed, 4 deletions(-) diff --git a/src/Date.php b/src/Date.php index dcfca76..3bea034 100644 --- a/src/Date.php +++ b/src/Date.php @@ -4,7 +4,6 @@ namespace DigitalCraftsman\DateTimePrecision; -/** @psalm-immutable */ final readonly class Date implements \Stringable { private const DATE_FORMAT = 'Y-m-d'; diff --git a/src/Month.php b/src/Month.php index 32aa55a..9c3e36a 100644 --- a/src/Month.php +++ b/src/Month.php @@ -4,7 +4,6 @@ namespace DigitalCraftsman\DateTimePrecision; -/** @psalm-immutable */ final readonly class Month implements \Stringable { private const MONTH_FORMAT = 'Y-m'; diff --git a/src/Time.php b/src/Time.php index 7d097c8..65f2d69 100644 --- a/src/Time.php +++ b/src/Time.php @@ -4,7 +4,6 @@ namespace DigitalCraftsman\DateTimePrecision; -/** @psalm-immutable */ final readonly class Time implements \Stringable { private const TIME_FORMAT = 'H:i:s.u'; diff --git a/src/Year.php b/src/Year.php index 1798e5a..66b2f30 100644 --- a/src/Year.php +++ b/src/Year.php @@ -4,7 +4,6 @@ namespace DigitalCraftsman\DateTimePrecision; -/** @psalm-immutable */ final readonly class Year { // -- Construction From ef61ab75dce0cefe7daf5c41690f10ea9575d91d Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 08:04:55 +0200 Subject: [PATCH 02/22] Add weekday including type and normalizers --- .../DoctrineTypeRegisterCompilerPass.php | 2 + src/Doctrine/WeekdayType.php | 50 ++++++++ src/Resources/config/services.yaml | 3 + src/Serializer/WeekdayNormalizer.php | 51 ++++++++ src/Weekday.php | 111 +++++++++++++++++ tests/Doctrine/WeekdayTypeTest.php | 54 +++++++++ tests/Serializer/WeekdayNormalizerTest.php | 114 ++++++++++++++++++ 7 files changed, 385 insertions(+) create mode 100644 src/Doctrine/WeekdayType.php create mode 100644 src/Serializer/WeekdayNormalizer.php create mode 100644 src/Weekday.php create mode 100644 tests/Doctrine/WeekdayTypeTest.php create mode 100644 tests/Serializer/WeekdayNormalizerTest.php diff --git a/src/DependencyInjection/DoctrineTypeRegisterCompilerPass.php b/src/DependencyInjection/DoctrineTypeRegisterCompilerPass.php index 4a6a59c..997a68b 100644 --- a/src/DependencyInjection/DoctrineTypeRegisterCompilerPass.php +++ b/src/DependencyInjection/DoctrineTypeRegisterCompilerPass.php @@ -8,6 +8,7 @@ use DigitalCraftsman\DateTimePrecision\Doctrine\MomentType; use DigitalCraftsman\DateTimePrecision\Doctrine\MonthType; use DigitalCraftsman\DateTimePrecision\Doctrine\TimeType; +use DigitalCraftsman\DateTimePrecision\Doctrine\WeekdayType; use DigitalCraftsman\DateTimePrecision\Doctrine\YearType; use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface; use Symfony\Component\DependencyInjection\ContainerBuilder; @@ -23,6 +24,7 @@ public function process(ContainerBuilder $container): void $typeDefinitions['dtp_moment'] = ['class' => MomentType::class]; $typeDefinitions['dtp_time'] = ['class' => TimeType::class]; + $typeDefinitions['dtp_weekday'] = ['class' => WeekdayType::class]; $typeDefinitions['dtp_date'] = ['class' => DateType::class]; $typeDefinitions['dtp_month'] = ['class' => MonthType::class]; $typeDefinitions['dtp_year'] = ['class' => YearType::class]; diff --git a/src/Doctrine/WeekdayType.php b/src/Doctrine/WeekdayType.php new file mode 100644 index 0000000..e15c715 --- /dev/null +++ b/src/Doctrine/WeekdayType.php @@ -0,0 +1,50 @@ +value; + } + + /** @param string|null $value */ + public function convertToPHPValue($value, AbstractPlatform $platform): ?Weekday + { + return $value === null + ? null + : Weekday::from($value); + } + + /** @codeCoverageIgnore */ + public function getSQLDeclaration(array $column, AbstractPlatform $platform): string + { + $column['length'] = 9; + + return $platform->getStringTypeDeclarationSQL($column); + } + + /** @codeCoverageIgnore */ + public function requiresSQLCommentHint(AbstractPlatform $platform): bool + { + return true; + } +} diff --git a/src/Resources/config/services.yaml b/src/Resources/config/services.yaml index aeea9a5..7d5a2f2 100644 --- a/src/Resources/config/services.yaml +++ b/src/Resources/config/services.yaml @@ -13,6 +13,9 @@ services: DigitalCraftsman\DateTimePrecision\Serializer\TimeNormalizer: tags: [ { name: 'serializer.normalizer' } ] + DigitalCraftsman\DateTimePrecision\Serializer\WeekdayNormalizer: + tags: [ { name: 'serializer.normalizer' } ] + DigitalCraftsman\DateTimePrecision\Serializer\DateNormalizer: tags: [ { name: 'serializer.normalizer' } ] diff --git a/src/Serializer/WeekdayNormalizer.php b/src/Serializer/WeekdayNormalizer.php new file mode 100644 index 0000000..c337dbd --- /dev/null +++ b/src/Serializer/WeekdayNormalizer.php @@ -0,0 +1,51 @@ +value; + } + + /** @param ?string $data */ + public function denormalize($data, $type, $format = null, array $context = []): ?Weekday + { + return $data === null + ? null + : Weekday::from($data); + } + + /** + * @return array + * + * @codeCoverageIgnore + */ + public function getSupportedTypes(?string $format): array + { + return [ + Weekday::class => true, + ]; + } +} diff --git a/src/Weekday.php b/src/Weekday.php new file mode 100644 index 0000000..17e1a9a --- /dev/null +++ b/src/Weekday.php @@ -0,0 +1,111 @@ +format('N')); + } + + public static function fromDate(Date $date): self + { + return self::fromDayOfWeek((int) $date->format('N')); + } + + public static function fromDayOfWeek(int $dayOfIsoWeek): self + { + return match ($dayOfIsoWeek) { + 1 => self::MONDAY, + 2 => self::TUESDAY, + 3 => self::WEDNESDAY, + 4 => self::THURSDAY, + 5 => self::FRIDAY, + 6 => self::SATURDAY, + 7 => self::SUNDAY, + }; + } + + // -- Accessors + + public function dayOfWeek(): int + { + return match ($this) { + self::MONDAY => 1, + self::TUESDAY => 2, + self::WEDNESDAY => 3, + self::THURSDAY => 4, + self::FRIDAY => 5, + self::SATURDAY => 6, + self::SUNDAY => 7, + }; + } + + public function isEqualTo(self $weekday): bool + { + return $this === $weekday; + } + + public function isNotEqualTo(self $weekday): bool + { + return $this !== $weekday; + } + + public function isBefore(self $date): bool + { + return $this->dayOfWeek() < $date->dayOfWeek(); + } + + public function isNotBefore(self $date): bool + { + return !($this->dayOfWeek() < $date->dayOfWeek()); + } + + public function isBeforeOrEqualTo(self $date): bool + { + return $this->dayOfWeek() <= $date->dayOfWeek(); + } + + public function isNotBeforeOrEqualTo(self $date): bool + { + return !($this->dayOfWeek() <= $date->dayOfWeek()); + } + + public function isAfter(self $date): bool + { + return $this->dayOfWeek() > $date->dayOfWeek(); + } + + public function isNotAfter(self $date): bool + { + return !($this->dayOfWeek() > $date->dayOfWeek()); + } + + public function isAfterOrEqualTo(self $date): bool + { + return $this->dayOfWeek() >= $date->dayOfWeek(); + } + + public function isNotAfterOrEqualTo(self $date): bool + { + return !($this->dayOfWeek() >= $date->dayOfWeek()); + } + + public function compareTo(self $date): int + { + return $this->dayOfWeek() <=> $date->dayOfWeek(); + } +} diff --git a/tests/Doctrine/WeekdayTypeTest.php b/tests/Doctrine/WeekdayTypeTest.php new file mode 100644 index 0000000..fe1e49b --- /dev/null +++ b/tests/Doctrine/WeekdayTypeTest.php @@ -0,0 +1,54 @@ +convertToDatabaseValue($weekday, $platform); + $phpValue = $weekdayType->convertToPHPValue($databaseValue, $platform); + + // -- Assert + self::assertEquals($weekday, $phpValue); + } + + /** + * @test + * + * @covers ::convertToDatabaseValue + * @covers ::convertToPHPValue + */ + public function convert_from_and_to_null_value_works(): void + { + // -- Arrange + $weekdayType = new WeekdayType(); + $platform = new PostgreSQLPlatform(); + + // -- Act + $databaseValue = $weekdayType->convertToDatabaseValue(null, $platform); + $phpValue = $weekdayType->convertToPHPValue($databaseValue, $platform); + + // -- Assert + self::assertNull($phpValue); + } +} diff --git a/tests/Serializer/WeekdayNormalizerTest.php b/tests/Serializer/WeekdayNormalizerTest.php new file mode 100644 index 0000000..bbcebad --- /dev/null +++ b/tests/Serializer/WeekdayNormalizerTest.php @@ -0,0 +1,114 @@ +normalize($weekday); + $denormalizedData = $normalizer->denormalize($normalizedData, Weekday::class); + + // -- Assert + self::assertEquals($weekday, $denormalizedData); + } + + /** + * @test + * + * @covers ::normalize + */ + public function weekday_normalization_with_null_works(): void + { + // -- Arrange + $normalizer = new WeekdayNormalizer(); + + // -- Act + $normalizedData = $normalizer->normalize(null); + + // -- Assert + self::assertNull($normalizedData); + } + + /** + * @test + * + * @covers ::denormalize + */ + public function weekday_denormalization_with_null_works(): void + { + // -- Arrange + $normalizer = new WeekdayNormalizer(); + + // -- Act + $denormalizedData = $normalizer->denormalize(null, Weekday::class); + + // -- Assert + self::assertNull($denormalizedData); + } + + /** + * @test + * + * @covers ::supportsNormalization + */ + public function supports_normalization(): void + { + // -- Arrange + $weekday = Weekday::TUESDAY; + + $normalizer = new WeekdayNormalizer(); + + // -- Act & Assert + self::assertTrue($normalizer->supportsNormalization($weekday)); + } + + /** + * @test + * + * @covers ::supportsNormalization + */ + public function supports_normalization_fails(): void + { + // -- Arrange + $year = Year::fromString('2022'); + + $normalizer = new WeekdayNormalizer(); + + // -- Act & Assert + self::assertFalse($normalizer->supportsNormalization($year)); + } + + /** + * @test + * + * @covers ::supportsDenormalization + */ + public function supports_denormalization(): void + { + // -- Arrange + $normalizer = new WeekdayNormalizer(); + + // -- Act & Assert + self::assertTrue($normalizer->supportsDenormalization(null, Weekday::class)); + } +} From e2fb847a8e0ae1e8dd0453e977ed24b4d7b2dac4 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 08:08:10 +0200 Subject: [PATCH 03/22] Add default coverage make command --- Makefile | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/Makefile b/Makefile index e27acd5..4b636e0 100644 --- a/Makefile +++ b/Makefile @@ -103,6 +103,10 @@ php-8.2-tests: php-8.3-tests: docker-compose run --rm php-8.3 ./vendor/bin/phpunit +## php-tests-coverage Run the tests for default PHP version and create coverage report. +.PHONY: php-tests-coverage +php-tests-coverage: php-8.3-tests-html-coverage + ## php-8.2-tests-html-coverage Run the tests with PHP 8.2 including coverage report as HTML. .PHONY: php-8.2-tests-html-coverage php-8.2-tests-html-coverage: From 0f22610e35748805a3bf555bd296a03e898a5398 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 08:08:53 +0200 Subject: [PATCH 04/22] Extend moment with weekday operations --- src/Moment.php | 42 ++++++++++++++++++++++++++++++++---------- 1 file changed, 32 insertions(+), 10 deletions(-) diff --git a/src/Moment.php b/src/Moment.php index 36913e9..cf7f051 100644 --- a/src/Moment.php +++ b/src/Moment.php @@ -68,6 +68,18 @@ public function timeInTimeZone(\DateTimeZone $timeZone): Time ->time(); } + public function weekday(): Weekday + { + return Weekday::fromDateTime($this->dateTime); + } + + public function weekdayInTimeZone(\DateTimeZone $timeZone): Weekday + { + return $this + ->toTimeZone($timeZone) + ->weekday(); + } + public function month(): Month { return Month::fromDateTime($this->dateTime); @@ -98,11 +110,12 @@ public function isEqualTo(self $moment): bool } public function isEqualToInTimeZone( - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): bool { return match (true) { $comparator instanceof Time => $this->timeInTimeZone($timeZone)->isEqualTo($comparator), + $comparator instanceof Weekday => $this->weekdayInTimeZone($timeZone)->isEqualTo($comparator), $comparator instanceof Date => $this->dateInTimeZone($timeZone)->isEqualTo($comparator), $comparator instanceof Month => $this->monthInTimeZone($timeZone)->isEqualTo($comparator), $comparator instanceof Year => $this->yearInTimeZone($timeZone)->isEqualTo($comparator), @@ -115,11 +128,12 @@ public function isNotEqualTo(self $moment): bool } public function isNotEqualToInTimeZone( - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): bool { return match (true) { $comparator instanceof Time => $this->timeInTimeZone($timeZone)->isNotEqualTo($comparator), + $comparator instanceof Weekday => $this->weekdayInTimeZone($timeZone)->isNotEqualTo($comparator), $comparator instanceof Date => $this->dateInTimeZone($timeZone)->isNotEqualTo($comparator), $comparator instanceof Month => $this->monthInTimeZone($timeZone)->isNotEqualTo($comparator), $comparator instanceof Year => $this->yearInTimeZone($timeZone)->isNotEqualTo($comparator), @@ -132,11 +146,12 @@ public function isAfter(self $moment): bool } public function isAfterInTimeZone( - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): bool { return match (true) { $comparator instanceof Time => $this->timeInTimeZone($timeZone)->isAfter($comparator), + $comparator instanceof Weekday => $this->weekdayInTimeZone($timeZone)->isAfter($comparator), $comparator instanceof Date => $this->dateInTimeZone($timeZone)->isAfter($comparator), $comparator instanceof Month => $this->monthInTimeZone($timeZone)->isAfter($comparator), $comparator instanceof Year => $this->yearInTimeZone($timeZone)->isAfter($comparator), @@ -149,11 +164,12 @@ public function isNotAfter(self $moment): bool } public function isNotAfterInTimeZone( - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): bool { return match (true) { $comparator instanceof Time => $this->timeInTimeZone($timeZone)->isNotAfter($comparator), + $comparator instanceof Weekday => $this->weekdayInTimeZone($timeZone)->isNotAfter($comparator), $comparator instanceof Date => $this->dateInTimeZone($timeZone)->isNotAfter($comparator), $comparator instanceof Month => $this->monthInTimeZone($timeZone)->isNotAfter($comparator), $comparator instanceof Year => $this->yearInTimeZone($timeZone)->isNotAfter($comparator), @@ -166,11 +182,12 @@ public function isAfterOrEqualTo(self $moment): bool } public function isAfterOrEqualToInTimeZone( - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): bool { return match (true) { $comparator instanceof Time => $this->timeInTimeZone($timeZone)->isAfterOrEqualTo($comparator), + $comparator instanceof Weekday => $this->weekdayInTimeZone($timeZone)->isAfterOrEqualTo($comparator), $comparator instanceof Date => $this->dateInTimeZone($timeZone)->isAfterOrEqualTo($comparator), $comparator instanceof Month => $this->monthInTimeZone($timeZone)->isAfterOrEqualTo($comparator), $comparator instanceof Year => $this->yearInTimeZone($timeZone)->isAfterOrEqualTo($comparator), @@ -183,11 +200,12 @@ public function isNotAfterOrEqualTo(self $moment): bool } public function isNotAfterOrEqualToInTimeZone( - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): bool { return match (true) { $comparator instanceof Time => $this->timeInTimeZone($timeZone)->isNotAfterOrEqualTo($comparator), + $comparator instanceof Weekday => $this->weekdayInTimeZone($timeZone)->isNotAfterOrEqualTo($comparator), $comparator instanceof Date => $this->dateInTimeZone($timeZone)->isNotAfterOrEqualTo($comparator), $comparator instanceof Month => $this->monthInTimeZone($timeZone)->isNotAfterOrEqualTo($comparator), $comparator instanceof Year => $this->yearInTimeZone($timeZone)->isNotAfterOrEqualTo($comparator), @@ -200,11 +218,12 @@ public function isBeforeOrEqualTo(self $moment): bool } public function isBeforeOrEqualToInTimeZone( - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): bool { return match (true) { $comparator instanceof Time => $this->timeInTimeZone($timeZone)->isBeforeOrEqualTo($comparator), + $comparator instanceof Weekday => $this->weekdayInTimeZone($timeZone)->isBeforeOrEqualTo($comparator), $comparator instanceof Date => $this->dateInTimeZone($timeZone)->isBeforeOrEqualTo($comparator), $comparator instanceof Month => $this->monthInTimeZone($timeZone)->isBeforeOrEqualTo($comparator), $comparator instanceof Year => $this->yearInTimeZone($timeZone)->isBeforeOrEqualTo($comparator), @@ -217,11 +236,12 @@ public function isNotBeforeOrEqualTo(self $moment): bool } public function isNotBeforeOrEqualToInTimeZone( - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): bool { return match (true) { $comparator instanceof Time => $this->timeInTimeZone($timeZone)->isNotBeforeOrEqualTo($comparator), + $comparator instanceof Weekday => $this->weekdayInTimeZone($timeZone)->isNotBeforeOrEqualTo($comparator), $comparator instanceof Date => $this->dateInTimeZone($timeZone)->isNotBeforeOrEqualTo($comparator), $comparator instanceof Month => $this->monthInTimeZone($timeZone)->isNotBeforeOrEqualTo($comparator), $comparator instanceof Year => $this->yearInTimeZone($timeZone)->isNotBeforeOrEqualTo($comparator), @@ -235,11 +255,12 @@ public function isBefore( } public function isBeforeInTimeZone( - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): bool { return match (true) { $comparator instanceof Time => $this->timeInTimeZone($timeZone)->isBefore($comparator), + $comparator instanceof Weekday => $this->weekdayInTimeZone($timeZone)->isBefore($comparator), $comparator instanceof Date => $this->dateInTimeZone($timeZone)->isBefore($comparator), $comparator instanceof Month => $this->monthInTimeZone($timeZone)->isBefore($comparator), $comparator instanceof Year => $this->yearInTimeZone($timeZone)->isBefore($comparator), @@ -252,11 +273,12 @@ public function isNotBefore(self $moment): bool } public function isNotBeforeInTimeZone( - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): bool { return match (true) { $comparator instanceof Time => $this->timeInTimeZone($timeZone)->isNotBefore($comparator), + $comparator instanceof Weekday => $this->weekdayInTimeZone($timeZone)->isNotBefore($comparator), $comparator instanceof Date => $this->dateInTimeZone($timeZone)->isNotBefore($comparator), $comparator instanceof Month => $this->monthInTimeZone($timeZone)->isNotBefore($comparator), $comparator instanceof Year => $this->yearInTimeZone($timeZone)->isNotBefore($comparator), From b3e22d0a7cb1901b601cac174bd54f091e66c209 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 08:09:06 +0200 Subject: [PATCH 05/22] Add weekday getter to date --- src/Date.php | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/src/Date.php b/src/Date.php index 3bea034..f5461a0 100644 --- a/src/Date.php +++ b/src/Date.php @@ -151,6 +151,11 @@ public function datesUntil( return $dates; } + public function weekday(): Weekday + { + return Weekday::fromDate($this); + } + // Mutations public function format(string $format): string From 69862c1995f94e06a465c9672f395cc3810d7c00 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 08:17:48 +0200 Subject: [PATCH 06/22] Move and extend compiler pass test --- .../DoctrineTypeRegisterCompilerPassTest.php | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) rename tests/{DigitalCraftsman => }/DateTimePrecision/DependencyInjection/DoctrineTypeRegisterCompilerPassTest.php (83%) diff --git a/tests/DigitalCraftsman/DateTimePrecision/DependencyInjection/DoctrineTypeRegisterCompilerPassTest.php b/tests/DateTimePrecision/DependencyInjection/DoctrineTypeRegisterCompilerPassTest.php similarity index 83% rename from tests/DigitalCraftsman/DateTimePrecision/DependencyInjection/DoctrineTypeRegisterCompilerPassTest.php rename to tests/DateTimePrecision/DependencyInjection/DoctrineTypeRegisterCompilerPassTest.php index 7e8505f..2cd5933 100644 --- a/tests/DigitalCraftsman/DateTimePrecision/DependencyInjection/DoctrineTypeRegisterCompilerPassTest.php +++ b/tests/DateTimePrecision/DependencyInjection/DoctrineTypeRegisterCompilerPassTest.php @@ -2,12 +2,14 @@ declare(strict_types=1); -namespace DigitalCraftsman\DateTimePrecision\DependencyInjection; +namespace DigitalCraftsman\DateTimePrecision\DateTimePrecision\DependencyInjection; +use DigitalCraftsman\DateTimePrecision\DependencyInjection\DoctrineTypeRegisterCompilerPass; use DigitalCraftsman\DateTimePrecision\Doctrine\DateType; use DigitalCraftsman\DateTimePrecision\Doctrine\MomentType; use DigitalCraftsman\DateTimePrecision\Doctrine\MonthType; use DigitalCraftsman\DateTimePrecision\Doctrine\TimeType; +use DigitalCraftsman\DateTimePrecision\Doctrine\WeekdayType; use DigitalCraftsman\DateTimePrecision\Doctrine\YearType; use PHPUnit\Framework\TestCase; use Symfony\Component\DependencyInjection\ContainerBuilder; @@ -41,6 +43,9 @@ public function process_works(): void self::assertArrayHasKey('dtp_time', $updatedParameters); self::assertSame(['class' => TimeType::class], $updatedParameters['dtp_time']); + self::assertArrayHasKey('dtp_weekday', $updatedParameters); + self::assertSame(['class' => WeekdayType::class], $updatedParameters['dtp_weekday']); + self::assertArrayHasKey('dtp_date', $updatedParameters); self::assertSame(['class' => DateType::class], $updatedParameters['dtp_date']); From 0db48c7d2dc987aeabaa513e933372ef6076571a Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 08:18:07 +0200 Subject: [PATCH 07/22] Kill mutations for is not equal in time zone --- tests/Moment/IsNotEqualToInTimeZoneTest.php | 23 +++++++++++++++++++-- 1 file changed, 21 insertions(+), 2 deletions(-) diff --git a/tests/Moment/IsNotEqualToInTimeZoneTest.php b/tests/Moment/IsNotEqualToInTimeZoneTest.php index 9dac15b..d3e2789 100644 --- a/tests/Moment/IsNotEqualToInTimeZoneTest.php +++ b/tests/Moment/IsNotEqualToInTimeZoneTest.php @@ -8,6 +8,7 @@ use DigitalCraftsman\DateTimePrecision\Moment; use DigitalCraftsman\DateTimePrecision\Month; use DigitalCraftsman\DateTimePrecision\Time; +use DigitalCraftsman\DateTimePrecision\Weekday; use DigitalCraftsman\DateTimePrecision\Year; use PHPUnit\Framework\TestCase; @@ -24,7 +25,7 @@ final class IsNotEqualToInTimeZoneTest extends TestCase public function is_not_equal_to_in_time_zone_works( bool $expectedResult, Moment $moment, - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): void { // -- Act & Assert @@ -35,7 +36,7 @@ public function is_not_equal_to_in_time_zone_works( * @return array */ @@ -60,6 +61,24 @@ public static function dataProvider(): array Time::fromString('15:00:00'), new \DateTimeZone('Europe/Berlin'), ], + 'moment before weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-08 14:55:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SUNDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment same weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-08 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment after weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-08 15:05:00', new \DateTimeZone('Europe/Berlin')), + Weekday::THURSDAY, + new \DateTimeZone('Europe/Berlin'), + ], 'moment before date' => [ true, Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), From 4897afecb96171c67006b9f9c86ddb01e14398b7 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 08:18:33 +0200 Subject: [PATCH 08/22] Add isEqualTo test coverage for weekday --- tests/Weekday/IsEqualToTest.php | 56 +++++++++++++++++++++++++++++++++ 1 file changed, 56 insertions(+) create mode 100644 tests/Weekday/IsEqualToTest.php diff --git a/tests/Weekday/IsEqualToTest.php b/tests/Weekday/IsEqualToTest.php new file mode 100644 index 0000000..e4ac2c4 --- /dev/null +++ b/tests/Weekday/IsEqualToTest.php @@ -0,0 +1,56 @@ +isEqualTo($comparator)); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + '1 day before' => [ + false, + Weekday::MONDAY, + Weekday::TUESDAY, + ], + 'same day' => [ + true, + Weekday::TUESDAY, + Weekday::TUESDAY, + ], + '1 day later' => [ + false, + Weekday::TUESDAY, + Weekday::MONDAY, + ], + ]; + } +} From 2d9f7761c41564f71d38e88331a5028131e94a28 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 08:28:55 +0200 Subject: [PATCH 09/22] Add coverage for date->weekday() --- tests/Date/WeekdayTest.php | 48 ++++++++++++++++++++++++++++++++++++++ 1 file changed, 48 insertions(+) create mode 100644 tests/Date/WeekdayTest.php diff --git a/tests/Date/WeekdayTest.php b/tests/Date/WeekdayTest.php new file mode 100644 index 0000000..55b38e3 --- /dev/null +++ b/tests/Date/WeekdayTest.php @@ -0,0 +1,48 @@ +weekday()); + } + + /** + * @return array + */ + public static function dataProviderForWeekday(): array + { + return [ + 'saturday' => [ + Weekday::SATURDAY, + Date::fromString('2022-10-08'), + ], + 'monday' => [ + Weekday::MONDAY, + Date::fromString('2022-10-10'), + ], + ]; + } +} From 9c8306a4b290484e83b67f2405a6280a691d8833 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 08:40:48 +0200 Subject: [PATCH 10/22] Add missing isEqualToInTimezone --- tests/Moment/IsEqualToInTimeZoneTest.php | 23 +++++++++++++++++++++-- 1 file changed, 21 insertions(+), 2 deletions(-) diff --git a/tests/Moment/IsEqualToInTimeZoneTest.php b/tests/Moment/IsEqualToInTimeZoneTest.php index af1ea6f..9f6cf16 100644 --- a/tests/Moment/IsEqualToInTimeZoneTest.php +++ b/tests/Moment/IsEqualToInTimeZoneTest.php @@ -8,6 +8,7 @@ use DigitalCraftsman\DateTimePrecision\Moment; use DigitalCraftsman\DateTimePrecision\Month; use DigitalCraftsman\DateTimePrecision\Time; +use DigitalCraftsman\DateTimePrecision\Weekday; use DigitalCraftsman\DateTimePrecision\Year; use PHPUnit\Framework\TestCase; @@ -24,7 +25,7 @@ final class IsEqualToInTimeZoneTest extends TestCase public function is_equal_to_in_time_zone_works( bool $expectedResult, Moment $moment, - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): void { // -- Act & Assert @@ -35,7 +36,7 @@ public function is_equal_to_in_time_zone_works( * @return array */ @@ -60,6 +61,24 @@ public static function dataProvider(): array Time::fromString('15:00:00'), new \DateTimeZone('Europe/Berlin'), ], + 'moment before weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment same weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-08 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment after weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-09 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], 'moment before date' => [ false, Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), From b32f3b68e1cb3859aa165775ffcf684ce78563d1 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 10:15:14 +0200 Subject: [PATCH 11/22] Squashed all mutants --- tests/Moment/IsAfterInTimeZoneTest.php | 23 +++++++++++++++++-- .../Moment/IsAfterOrEqualToInTimeZoneTest.php | 23 +++++++++++++++++-- tests/Moment/IsBeforeInTimeZoneTest.php | 23 +++++++++++++++++-- .../IsBeforeOrEqualToInTimeZoneTest.php | 23 +++++++++++++++++-- tests/Moment/IsNotAfterInTimeZoneTest.php | 23 +++++++++++++++++-- .../IsNotAfterOrEqualToInTimeZoneTest.php | 23 +++++++++++++++++-- tests/Moment/IsNotBeforeInTimeZoneTest.php | 23 +++++++++++++++++-- .../IsNotBeforeOrEqualToInTimeZoneTest.php | 23 +++++++++++++++++-- 8 files changed, 168 insertions(+), 16 deletions(-) diff --git a/tests/Moment/IsAfterInTimeZoneTest.php b/tests/Moment/IsAfterInTimeZoneTest.php index bafc998..028339d 100644 --- a/tests/Moment/IsAfterInTimeZoneTest.php +++ b/tests/Moment/IsAfterInTimeZoneTest.php @@ -8,6 +8,7 @@ use DigitalCraftsman\DateTimePrecision\Moment; use DigitalCraftsman\DateTimePrecision\Month; use DigitalCraftsman\DateTimePrecision\Time; +use DigitalCraftsman\DateTimePrecision\Weekday; use DigitalCraftsman\DateTimePrecision\Year; use PHPUnit\Framework\TestCase; @@ -24,7 +25,7 @@ final class IsAfterInTimeZoneTest extends TestCase public function is_after_in_time_zone_works( bool $expectedResult, Moment $moment, - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): void { // -- Act & Assert @@ -35,7 +36,7 @@ public function is_after_in_time_zone_works( * @return array */ @@ -60,6 +61,24 @@ public static function dataProvider(): array Time::fromString('15:00:00'), new \DateTimeZone('Europe/Berlin'), ], + 'moment before weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment same weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-08 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment after weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-09 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], 'moment before date' => [ false, Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), diff --git a/tests/Moment/IsAfterOrEqualToInTimeZoneTest.php b/tests/Moment/IsAfterOrEqualToInTimeZoneTest.php index 45d76d6..17bb5a7 100644 --- a/tests/Moment/IsAfterOrEqualToInTimeZoneTest.php +++ b/tests/Moment/IsAfterOrEqualToInTimeZoneTest.php @@ -8,6 +8,7 @@ use DigitalCraftsman\DateTimePrecision\Moment; use DigitalCraftsman\DateTimePrecision\Month; use DigitalCraftsman\DateTimePrecision\Time; +use DigitalCraftsman\DateTimePrecision\Weekday; use DigitalCraftsman\DateTimePrecision\Year; use PHPUnit\Framework\TestCase; @@ -24,7 +25,7 @@ final class IsAfterOrEqualToInTimeZoneTest extends TestCase public function is_after_or_equal_to_in_time_zone_works( bool $expectedResult, Moment $moment, - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): void { // -- Act & Assert @@ -35,7 +36,7 @@ public function is_after_or_equal_to_in_time_zone_works( * @return array */ @@ -60,6 +61,24 @@ public static function dataProvider(): array Time::fromString('15:00:00'), new \DateTimeZone('Europe/Berlin'), ], + 'moment before weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment same weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-08 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment after weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-09 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], 'moment before date' => [ false, Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), diff --git a/tests/Moment/IsBeforeInTimeZoneTest.php b/tests/Moment/IsBeforeInTimeZoneTest.php index 197c34d..61bda0b 100644 --- a/tests/Moment/IsBeforeInTimeZoneTest.php +++ b/tests/Moment/IsBeforeInTimeZoneTest.php @@ -8,6 +8,7 @@ use DigitalCraftsman\DateTimePrecision\Moment; use DigitalCraftsman\DateTimePrecision\Month; use DigitalCraftsman\DateTimePrecision\Time; +use DigitalCraftsman\DateTimePrecision\Weekday; use DigitalCraftsman\DateTimePrecision\Year; use PHPUnit\Framework\TestCase; @@ -24,7 +25,7 @@ final class IsBeforeInTimeZoneTest extends TestCase public function is_before_in_time_zone_works( bool $expectedResult, Moment $moment, - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): void { // -- Act & Assert @@ -35,7 +36,7 @@ public function is_before_in_time_zone_works( * @return array */ @@ -60,6 +61,24 @@ public static function dataProvider(): array Time::fromString('15:00:00'), new \DateTimeZone('Europe/Berlin'), ], + 'moment before weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment same weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-08 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment after weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-09 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], 'moment before date' => [ true, Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), diff --git a/tests/Moment/IsBeforeOrEqualToInTimeZoneTest.php b/tests/Moment/IsBeforeOrEqualToInTimeZoneTest.php index 17066f7..4b2c6b4 100644 --- a/tests/Moment/IsBeforeOrEqualToInTimeZoneTest.php +++ b/tests/Moment/IsBeforeOrEqualToInTimeZoneTest.php @@ -8,6 +8,7 @@ use DigitalCraftsman\DateTimePrecision\Moment; use DigitalCraftsman\DateTimePrecision\Month; use DigitalCraftsman\DateTimePrecision\Time; +use DigitalCraftsman\DateTimePrecision\Weekday; use DigitalCraftsman\DateTimePrecision\Year; use PHPUnit\Framework\TestCase; @@ -24,7 +25,7 @@ final class IsBeforeOrEqualToInTimeZoneTest extends TestCase public function is_before_or_equal_to_in_time_zone_works( bool $expectedResult, Moment $moment, - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): void { // -- Act & Assert @@ -35,7 +36,7 @@ public function is_before_or_equal_to_in_time_zone_works( * @return array */ @@ -60,6 +61,24 @@ public static function dataProvider(): array Time::fromString('15:00:00'), new \DateTimeZone('Europe/Berlin'), ], + 'moment before weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment same weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-08 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment after weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-09 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], 'moment before date' => [ true, Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), diff --git a/tests/Moment/IsNotAfterInTimeZoneTest.php b/tests/Moment/IsNotAfterInTimeZoneTest.php index 829ab0c..e34c7b5 100644 --- a/tests/Moment/IsNotAfterInTimeZoneTest.php +++ b/tests/Moment/IsNotAfterInTimeZoneTest.php @@ -8,6 +8,7 @@ use DigitalCraftsman\DateTimePrecision\Moment; use DigitalCraftsman\DateTimePrecision\Month; use DigitalCraftsman\DateTimePrecision\Time; +use DigitalCraftsman\DateTimePrecision\Weekday; use DigitalCraftsman\DateTimePrecision\Year; use PHPUnit\Framework\TestCase; @@ -24,7 +25,7 @@ final class IsNotAfterInTimeZoneTest extends TestCase public function is_not_after_in_time_zone_works( bool $expectedResult, Moment $moment, - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): void { // -- Act & Assert @@ -35,7 +36,7 @@ public function is_not_after_in_time_zone_works( * @return array */ @@ -60,6 +61,24 @@ public static function dataProvider(): array Time::fromString('15:00:00'), new \DateTimeZone('Europe/Berlin'), ], + 'moment before weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment same weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-08 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment after weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-09 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], 'moment before date' => [ true, Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), diff --git a/tests/Moment/IsNotAfterOrEqualToInTimeZoneTest.php b/tests/Moment/IsNotAfterOrEqualToInTimeZoneTest.php index 21f0a7b..7e0f046 100644 --- a/tests/Moment/IsNotAfterOrEqualToInTimeZoneTest.php +++ b/tests/Moment/IsNotAfterOrEqualToInTimeZoneTest.php @@ -8,6 +8,7 @@ use DigitalCraftsman\DateTimePrecision\Moment; use DigitalCraftsman\DateTimePrecision\Month; use DigitalCraftsman\DateTimePrecision\Time; +use DigitalCraftsman\DateTimePrecision\Weekday; use DigitalCraftsman\DateTimePrecision\Year; use PHPUnit\Framework\TestCase; @@ -24,7 +25,7 @@ final class IsNotAfterOrEqualToInTimeZoneTest extends TestCase public function is_not_after_or_equal_to_in_time_zone_works( bool $expectedResult, Moment $moment, - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): void { // -- Act & Assert @@ -35,7 +36,7 @@ public function is_not_after_or_equal_to_in_time_zone_works( * @return array */ @@ -60,6 +61,24 @@ public static function dataProvider(): array Time::fromString('15:00:00'), new \DateTimeZone('Europe/Berlin'), ], + 'moment before weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment same weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-08 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment after weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-09 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], 'moment before date' => [ true, Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), diff --git a/tests/Moment/IsNotBeforeInTimeZoneTest.php b/tests/Moment/IsNotBeforeInTimeZoneTest.php index 0a1bcd1..4d1f5b0 100644 --- a/tests/Moment/IsNotBeforeInTimeZoneTest.php +++ b/tests/Moment/IsNotBeforeInTimeZoneTest.php @@ -8,6 +8,7 @@ use DigitalCraftsman\DateTimePrecision\Moment; use DigitalCraftsman\DateTimePrecision\Month; use DigitalCraftsman\DateTimePrecision\Time; +use DigitalCraftsman\DateTimePrecision\Weekday; use DigitalCraftsman\DateTimePrecision\Year; use PHPUnit\Framework\TestCase; @@ -24,7 +25,7 @@ final class IsNotBeforeInTimeZoneTest extends TestCase public function is_not_before_in_time_zone_works( bool $expectedResult, Moment $moment, - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): void { // -- Act & Assert @@ -35,7 +36,7 @@ public function is_not_before_in_time_zone_works( * @return array */ @@ -60,6 +61,24 @@ public static function dataProvider(): array Time::fromString('15:00:00'), new \DateTimeZone('Europe/Berlin'), ], + 'moment before weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment same weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-08 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment after weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-09 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], 'moment before date' => [ false, Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), diff --git a/tests/Moment/IsNotBeforeOrEqualToInTimeZoneTest.php b/tests/Moment/IsNotBeforeOrEqualToInTimeZoneTest.php index 7d28831..18df559 100644 --- a/tests/Moment/IsNotBeforeOrEqualToInTimeZoneTest.php +++ b/tests/Moment/IsNotBeforeOrEqualToInTimeZoneTest.php @@ -8,6 +8,7 @@ use DigitalCraftsman\DateTimePrecision\Moment; use DigitalCraftsman\DateTimePrecision\Month; use DigitalCraftsman\DateTimePrecision\Time; +use DigitalCraftsman\DateTimePrecision\Weekday; use DigitalCraftsman\DateTimePrecision\Year; use PHPUnit\Framework\TestCase; @@ -24,7 +25,7 @@ final class IsNotBeforeOrEqualToInTimeZoneTest extends TestCase public function is_not_before_or_equal_to_in_time_zone_works( bool $expectedResult, Moment $moment, - Time | Date | Month | Year $comparator, + Time | Weekday | Date | Month | Year $comparator, \DateTimeZone $timeZone, ): void { // -- Act & Assert @@ -35,7 +36,7 @@ public function is_not_before_or_equal_to_in_time_zone_works( * @return array */ @@ -60,6 +61,24 @@ public static function dataProvider(): array Time::fromString('15:00:00'), new \DateTimeZone('Europe/Berlin'), ], + 'moment before weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment same weekday' => [ + false, + Moment::fromStringInTimeZone('2022-10-08 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], + 'moment after weekday' => [ + true, + Moment::fromStringInTimeZone('2022-10-09 15:00:00', new \DateTimeZone('Europe/Berlin')), + Weekday::SATURDAY, + new \DateTimeZone('Europe/Berlin'), + ], 'moment before date' => [ false, Moment::fromStringInTimeZone('2022-10-07 15:00:00', new \DateTimeZone('Europe/Berlin')), From 03d6ed22a3af0acb5d81002b1ce12b6f4b4d16d4 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 10:21:27 +0200 Subject: [PATCH 12/22] Add weekday test for Moment --- tests/Moment/WeekdayTest.php | 93 ++++++++++++++++++++++++++++++++++++ 1 file changed, 93 insertions(+) create mode 100644 tests/Moment/WeekdayTest.php diff --git a/tests/Moment/WeekdayTest.php b/tests/Moment/WeekdayTest.php new file mode 100644 index 0000000..8d854f1 --- /dev/null +++ b/tests/Moment/WeekdayTest.php @@ -0,0 +1,93 @@ +weekday()); + } + + /** + * @return array + */ + public static function dataProviderForTime(): array + { + return [ + 'same weekday in UTC' => [ + Weekday::SATURDAY, + Moment::fromString('2022-10-08 15:00:00'), + ], + 'weekday adapted due to weekday zone difference' => [ + Weekday::SATURDAY, + Moment::fromStringInTimeZone('2022-10-08 00:00:00', new \DateTimeZone('Europe/Berlin')) + ->toTimeZone(new \DateTimeZone('Europe/Berlin')), + ], + ]; + } + + /** + * @test + * + * @dataProvider dataProviderForTimeInTimeZone + * + * @covers ::weekdayInTimeZone + */ + public function weekday_in_weekday_zone_works( + Weekday $expectedResult, + Moment $dateTime, + \DateTimeZone $weekdayZone, + ): void { + // -- Act & Assert + self::assertEquals($expectedResult, $dateTime->weekdayInTimeZone($weekdayZone)); + } + + /** + * @return array + */ + public static function dataProviderForTimeInTimeZone(): array + { + return [ + 'same weekday in UTC' => [ + Weekday::SATURDAY, + Moment::fromString('2022-10-08 15:00:00'), + new \DateTimeZone('UTC'), + ], + 'adapting weekday for two hours through timezone difference' => [ + Weekday::SUNDAY, + Moment::fromString('2022-10-08 23:00:00'), + new \DateTimeZone('Europe/Berlin'), + ], + 'same weekday when timezone was used for creation of datetime' => [ + Weekday::SATURDAY, + Moment::fromStringInTimeZone('2022-10-08 15:00:00', new \DateTimeZone('Europe/Berlin')), + new \DateTimeZone('Europe/Berlin'), + ], + ]; + } +} From 57628d7e7cab2080b7549954516d5ed84a045288 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 10:27:59 +0200 Subject: [PATCH 13/22] Add more test coverage --- tests/Weekday/IsAfterOrEqualToTest.php | 56 +++++++++++++++++++++++ tests/Weekday/IsNotAfterOrEqualToTest.php | 56 +++++++++++++++++++++++ tests/Weekday/IsNotEqualToTest.php | 56 +++++++++++++++++++++++ 3 files changed, 168 insertions(+) create mode 100644 tests/Weekday/IsAfterOrEqualToTest.php create mode 100644 tests/Weekday/IsNotAfterOrEqualToTest.php create mode 100644 tests/Weekday/IsNotEqualToTest.php diff --git a/tests/Weekday/IsAfterOrEqualToTest.php b/tests/Weekday/IsAfterOrEqualToTest.php new file mode 100644 index 0000000..33540a0 --- /dev/null +++ b/tests/Weekday/IsAfterOrEqualToTest.php @@ -0,0 +1,56 @@ +isAfterOrEqualTo($comparator)); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + '1 day before' => [ + false, + Weekday::MONDAY, + Weekday::TUESDAY, + ], + 'same day' => [ + true, + Weekday::TUESDAY, + Weekday::TUESDAY, + ], + '1 day later' => [ + true, + Weekday::TUESDAY, + Weekday::MONDAY, + ], + ]; + } +} diff --git a/tests/Weekday/IsNotAfterOrEqualToTest.php b/tests/Weekday/IsNotAfterOrEqualToTest.php new file mode 100644 index 0000000..15d1a7c --- /dev/null +++ b/tests/Weekday/IsNotAfterOrEqualToTest.php @@ -0,0 +1,56 @@ +isNotAfterOrEqualTo($comparator)); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + '1 day before' => [ + true, + Weekday::MONDAY, + Weekday::TUESDAY, + ], + 'same day' => [ + false, + Weekday::TUESDAY, + Weekday::TUESDAY, + ], + '1 day later' => [ + false, + Weekday::TUESDAY, + Weekday::MONDAY, + ], + ]; + } +} diff --git a/tests/Weekday/IsNotEqualToTest.php b/tests/Weekday/IsNotEqualToTest.php new file mode 100644 index 0000000..8c299ba --- /dev/null +++ b/tests/Weekday/IsNotEqualToTest.php @@ -0,0 +1,56 @@ +isNotEqualTo($comparator)); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + '1 day before' => [ + true, + Weekday::MONDAY, + Weekday::TUESDAY, + ], + 'same day' => [ + false, + Weekday::TUESDAY, + Weekday::TUESDAY, + ], + '1 day later' => [ + true, + Weekday::TUESDAY, + Weekday::MONDAY, + ], + ]; + } +} From 3b9101a6648581ec8fdce932cddf80db96f81e51 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 10:29:54 +0200 Subject: [PATCH 14/22] More test coverage --- tests/Weekday/IsBeforeTest.php | 56 +++++++++++++++++++++++++++++++ tests/Weekday/IsNotBeforeTest.php | 56 +++++++++++++++++++++++++++++++ 2 files changed, 112 insertions(+) create mode 100644 tests/Weekday/IsBeforeTest.php create mode 100644 tests/Weekday/IsNotBeforeTest.php diff --git a/tests/Weekday/IsBeforeTest.php b/tests/Weekday/IsBeforeTest.php new file mode 100644 index 0000000..685a572 --- /dev/null +++ b/tests/Weekday/IsBeforeTest.php @@ -0,0 +1,56 @@ +isBefore($comparator)); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + '1 day before' => [ + true, + Weekday::MONDAY, + Weekday::TUESDAY, + ], + 'same day' => [ + false, + Weekday::TUESDAY, + Weekday::TUESDAY, + ], + '1 day later' => [ + false, + Weekday::TUESDAY, + Weekday::MONDAY, + ], + ]; + } +} diff --git a/tests/Weekday/IsNotBeforeTest.php b/tests/Weekday/IsNotBeforeTest.php new file mode 100644 index 0000000..0d03f11 --- /dev/null +++ b/tests/Weekday/IsNotBeforeTest.php @@ -0,0 +1,56 @@ +isNotBefore($comparator)); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + '1 day before' => [ + false, + Weekday::MONDAY, + Weekday::TUESDAY, + ], + 'same day' => [ + true, + Weekday::TUESDAY, + Weekday::TUESDAY, + ], + '1 day later' => [ + true, + Weekday::TUESDAY, + Weekday::MONDAY, + ], + ]; + } +} From dfde9bc886c9ee219def343faba4e83023e6c847 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 10:45:09 +0200 Subject: [PATCH 15/22] Finalize coverage of is methods --- tests/Weekday/IsAfterTest.php | 56 ++++++++++++++++++++++ tests/Weekday/IsBeforeOrEqualToTest.php | 56 ++++++++++++++++++++++ tests/Weekday/IsNotAfterTest.php | 56 ++++++++++++++++++++++ tests/Weekday/IsNotBeforeOrEqualToTest.php | 56 ++++++++++++++++++++++ 4 files changed, 224 insertions(+) create mode 100644 tests/Weekday/IsAfterTest.php create mode 100644 tests/Weekday/IsBeforeOrEqualToTest.php create mode 100644 tests/Weekday/IsNotAfterTest.php create mode 100644 tests/Weekday/IsNotBeforeOrEqualToTest.php diff --git a/tests/Weekday/IsAfterTest.php b/tests/Weekday/IsAfterTest.php new file mode 100644 index 0000000..951d4d4 --- /dev/null +++ b/tests/Weekday/IsAfterTest.php @@ -0,0 +1,56 @@ +isAfter($comparator)); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + '1 day before' => [ + false, + Weekday::MONDAY, + Weekday::TUESDAY, + ], + 'same day' => [ + false, + Weekday::TUESDAY, + Weekday::TUESDAY, + ], + '1 day later' => [ + true, + Weekday::TUESDAY, + Weekday::MONDAY, + ], + ]; + } +} diff --git a/tests/Weekday/IsBeforeOrEqualToTest.php b/tests/Weekday/IsBeforeOrEqualToTest.php new file mode 100644 index 0000000..f6bbfde --- /dev/null +++ b/tests/Weekday/IsBeforeOrEqualToTest.php @@ -0,0 +1,56 @@ +isBeforeOrEqualTo($comparator)); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + '1 day before' => [ + true, + Weekday::MONDAY, + Weekday::TUESDAY, + ], + 'same day' => [ + true, + Weekday::TUESDAY, + Weekday::TUESDAY, + ], + '1 day later' => [ + false, + Weekday::TUESDAY, + Weekday::MONDAY, + ], + ]; + } +} diff --git a/tests/Weekday/IsNotAfterTest.php b/tests/Weekday/IsNotAfterTest.php new file mode 100644 index 0000000..f4c000b --- /dev/null +++ b/tests/Weekday/IsNotAfterTest.php @@ -0,0 +1,56 @@ +isNotAfter($comparator)); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + '1 day before' => [ + true, + Weekday::MONDAY, + Weekday::TUESDAY, + ], + 'same day' => [ + true, + Weekday::TUESDAY, + Weekday::TUESDAY, + ], + '1 day later' => [ + false, + Weekday::TUESDAY, + Weekday::MONDAY, + ], + ]; + } +} diff --git a/tests/Weekday/IsNotBeforeOrEqualToTest.php b/tests/Weekday/IsNotBeforeOrEqualToTest.php new file mode 100644 index 0000000..8698241 --- /dev/null +++ b/tests/Weekday/IsNotBeforeOrEqualToTest.php @@ -0,0 +1,56 @@ +isNotBeforeOrEqualTo($comparator)); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + '1 day before' => [ + false, + Weekday::MONDAY, + Weekday::TUESDAY, + ], + 'same day' => [ + false, + Weekday::TUESDAY, + Weekday::TUESDAY, + ], + '1 day later' => [ + true, + Weekday::TUESDAY, + Weekday::MONDAY, + ], + ]; + } +} From f5330516945ed045caeecac1e57826d80610b23b Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 11:57:45 +0200 Subject: [PATCH 16/22] Remove unnecessary method and extend test coverage --- src/Date.php | 2 +- src/Weekday.php | 12 +----- tests/Weekday/CompareToTest.php | 61 +++++++++++++++++++++++++++ tests/Weekday/ConstructionTest.php | 68 ++++++++++++++++++++++++++++++ 4 files changed, 132 insertions(+), 11 deletions(-) create mode 100644 tests/Weekday/CompareToTest.php create mode 100644 tests/Weekday/ConstructionTest.php diff --git a/src/Date.php b/src/Date.php index f5461a0..4219ccb 100644 --- a/src/Date.php +++ b/src/Date.php @@ -153,7 +153,7 @@ public function datesUntil( public function weekday(): Weekday { - return Weekday::fromDate($this); + return Weekday::fromDateTime($this->toDateTimeImmutable()); } // Mutations diff --git a/src/Weekday.php b/src/Weekday.php index 17e1a9a..4617711 100644 --- a/src/Weekday.php +++ b/src/Weekday.php @@ -18,17 +18,9 @@ enum Weekday: string public static function fromDateTime(\DateTimeImmutable $dateTime): self { - return self::fromDayOfWeek((int) $dateTime->format('N')); - } + $dayOfWeek = (int) $dateTime->format('N'); - public static function fromDate(Date $date): self - { - return self::fromDayOfWeek((int) $date->format('N')); - } - - public static function fromDayOfWeek(int $dayOfIsoWeek): self - { - return match ($dayOfIsoWeek) { + return match ($dayOfWeek) { 1 => self::MONDAY, 2 => self::TUESDAY, 3 => self::WEDNESDAY, diff --git a/tests/Weekday/CompareToTest.php b/tests/Weekday/CompareToTest.php new file mode 100644 index 0000000..1815b05 --- /dev/null +++ b/tests/Weekday/CompareToTest.php @@ -0,0 +1,61 @@ +compareTo($comparator)); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + '1 weekday later' => [ + 1, + Weekday::THURSDAY, + Weekday::WEDNESDAY, + ], + 'same weekday' => [ + 0, + Weekday::SATURDAY, + Weekday::SATURDAY, + ], + '1 weekday before' => [ + -1, + Weekday::TUESDAY, + Weekday::WEDNESDAY, + ], + '2 weekdays before' => [ + -1, + Weekday::MONDAY, + Weekday::WEDNESDAY, + ], + ]; + } +} diff --git a/tests/Weekday/ConstructionTest.php b/tests/Weekday/ConstructionTest.php new file mode 100644 index 0000000..b6386b8 --- /dev/null +++ b/tests/Weekday/ConstructionTest.php @@ -0,0 +1,68 @@ +isEqualTo(Weekday::fromDateTime($moment->dateTime))); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + 'monday' => [ + Weekday::MONDAY, + Moment::fromString('2022-10-10 22:15:00'), + ], + 'tuesday' => [ + Weekday::TUESDAY, + Moment::fromString('2022-10-11 22:15:00'), + ], + 'wednesday' => [ + Weekday::WEDNESDAY, + Moment::fromString('2022-10-12 22:15:00'), + ], + 'thursday' => [ + Weekday::THURSDAY, + Moment::fromString('2022-10-13 22:15:00'), + ], + 'friday' => [ + Weekday::FRIDAY, + Moment::fromString('2022-10-14 22:15:00'), + ], + 'saturday' => [ + Weekday::SATURDAY, + Moment::fromString('2022-10-15 22:15:00'), + ], + 'sunday' => [ + Weekday::SUNDAY, + Moment::fromString('2022-10-16 22:15:00'), + ], + ]; + } +} From e10fb10da17c776f0425a400b9c0714d3ac2fa01 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 11:59:56 +0200 Subject: [PATCH 17/22] Last test coverage for day of week --- tests/Weekday/DayOfWeekTest.php | 67 +++++++++++++++++++++++++++++++++ 1 file changed, 67 insertions(+) create mode 100644 tests/Weekday/DayOfWeekTest.php diff --git a/tests/Weekday/DayOfWeekTest.php b/tests/Weekday/DayOfWeekTest.php new file mode 100644 index 0000000..b42945e --- /dev/null +++ b/tests/Weekday/DayOfWeekTest.php @@ -0,0 +1,67 @@ +dayOfWeek()); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + 'monday' => [ + 1, + Weekday::MONDAY, + ], + 'tuesday' => [ + 2, + Weekday::TUESDAY, + ], + 'wednesday' => [ + 3, + Weekday::WEDNESDAY, + ], + 'thursday' => [ + 4, + Weekday::THURSDAY, + ], + 'friday' => [ + 5, + Weekday::FRIDAY, + ], + 'saturday' => [ + 6, + Weekday::SATURDAY, + ], + 'sunday' => [ + 7, + Weekday::SUNDAY, + ], + ]; + } +} From 78e5028183594157274b4c44b29aef619795c557 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 14:10:58 +0200 Subject: [PATCH 18/22] Add weekdays value object --- .../DoctrineTypeRegisterCompilerPass.php | 2 + src/Doctrine/WeekdaysType.php | 56 +++++++++++++++++++ src/Resources/config/services.yaml | 3 + src/Serializer/WeekdaysNormalizer.php | 49 ++++++++++++++++ src/Weekdays.php | 54 ++++++++++++++++++ .../DoctrineTypeRegisterCompilerPassTest.php | 4 ++ 6 files changed, 168 insertions(+) create mode 100644 src/Doctrine/WeekdaysType.php create mode 100644 src/Serializer/WeekdaysNormalizer.php create mode 100644 src/Weekdays.php diff --git a/src/DependencyInjection/DoctrineTypeRegisterCompilerPass.php b/src/DependencyInjection/DoctrineTypeRegisterCompilerPass.php index 997a68b..16d0395 100644 --- a/src/DependencyInjection/DoctrineTypeRegisterCompilerPass.php +++ b/src/DependencyInjection/DoctrineTypeRegisterCompilerPass.php @@ -8,6 +8,7 @@ use DigitalCraftsman\DateTimePrecision\Doctrine\MomentType; use DigitalCraftsman\DateTimePrecision\Doctrine\MonthType; use DigitalCraftsman\DateTimePrecision\Doctrine\TimeType; +use DigitalCraftsman\DateTimePrecision\Doctrine\WeekdaysType; use DigitalCraftsman\DateTimePrecision\Doctrine\WeekdayType; use DigitalCraftsman\DateTimePrecision\Doctrine\YearType; use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface; @@ -25,6 +26,7 @@ public function process(ContainerBuilder $container): void $typeDefinitions['dtp_moment'] = ['class' => MomentType::class]; $typeDefinitions['dtp_time'] = ['class' => TimeType::class]; $typeDefinitions['dtp_weekday'] = ['class' => WeekdayType::class]; + $typeDefinitions['dtp_weekdays'] = ['class' => WeekdaysType::class]; $typeDefinitions['dtp_date'] = ['class' => DateType::class]; $typeDefinitions['dtp_month'] = ['class' => MonthType::class]; $typeDefinitions['dtp_year'] = ['class' => YearType::class]; diff --git a/src/Doctrine/WeekdaysType.php b/src/Doctrine/WeekdaysType.php new file mode 100644 index 0000000..fabb878 --- /dev/null +++ b/src/Doctrine/WeekdaysType.php @@ -0,0 +1,56 @@ +normalize(); + + return json_encode($array, JSON_THROW_ON_ERROR); + } + + /** @param string|null $value */ + public function convertToPHPValue($value, AbstractPlatform $platform): ?Weekdays + { + if ($value === null) { + return null; + } + + $array = json_decode($value, true, 512, JSON_THROW_ON_ERROR); + + return Weekdays::denormalize($array); + } + + /** @codeCoverageIgnore */ + public function getSQLDeclaration(array $column, AbstractPlatform $platform): string + { + $column['jsonb'] = true; + + return $platform->getJsonTypeDeclarationSQL($column); + } + + /** @codeCoverageIgnore */ + public function requiresSQLCommentHint(AbstractPlatform $platform): bool + { + return true; + } +} diff --git a/src/Resources/config/services.yaml b/src/Resources/config/services.yaml index 7d5a2f2..10d0004 100644 --- a/src/Resources/config/services.yaml +++ b/src/Resources/config/services.yaml @@ -16,6 +16,9 @@ services: DigitalCraftsman\DateTimePrecision\Serializer\WeekdayNormalizer: tags: [ { name: 'serializer.normalizer' } ] + DigitalCraftsman\DateTimePrecision\Serializer\WeekdaysNormalizer: + tags: [ { name: 'serializer.normalizer' } ] + DigitalCraftsman\DateTimePrecision\Serializer\DateNormalizer: tags: [ { name: 'serializer.normalizer' } ] diff --git a/src/Serializer/WeekdaysNormalizer.php b/src/Serializer/WeekdaysNormalizer.php new file mode 100644 index 0000000..1cfeaf8 --- /dev/null +++ b/src/Serializer/WeekdaysNormalizer.php @@ -0,0 +1,49 @@ +normalize(); + } + + /** @param ?array $data */ + public function denormalize($data, $type, $format = null, array $context = []): ?Weekdays + { + return $data === null + ? null + : Weekdays::denormalize($data); + } + + /** + * @return array + * + * @codeCoverageIgnore + */ + public function getSupportedTypes(?string $format): array + { + return [ + Weekdays::class => true, + ]; + } +} diff --git a/src/Weekdays.php b/src/Weekdays.php new file mode 100644 index 0000000..87232d3 --- /dev/null +++ b/src/Weekdays.php @@ -0,0 +1,54 @@ + $weekdays */ + public function __construct( + /** @var array $weekdays */ + public array $weekdays, + ) { + // TODO: Validate for unique + } + + // -- Array normalizable + + /** @param array $array */ + public static function denormalize(array $array): self + { + $weekdays = []; + foreach ($array as $value) { + $weekdays[] = Weekday::from($value); + } + + return new self($weekdays); + } + + /** @return array */ + public function normalize(): array + { + $weekdayStrings = []; + foreach ($this->weekdays as $weekday) { + $weekdayStrings[] = $weekday->value; + } + + return $weekdayStrings; + } + + // Accessors + + public function contains(Weekday $weekday): bool + { + return in_array($weekday, $this->weekdays, true); + } + + public function notContains(Weekday $weekday): bool + { + return !in_array($weekday, $this->weekdays, true); + } +} diff --git a/tests/DateTimePrecision/DependencyInjection/DoctrineTypeRegisterCompilerPassTest.php b/tests/DateTimePrecision/DependencyInjection/DoctrineTypeRegisterCompilerPassTest.php index 2cd5933..4a321ea 100644 --- a/tests/DateTimePrecision/DependencyInjection/DoctrineTypeRegisterCompilerPassTest.php +++ b/tests/DateTimePrecision/DependencyInjection/DoctrineTypeRegisterCompilerPassTest.php @@ -9,6 +9,7 @@ use DigitalCraftsman\DateTimePrecision\Doctrine\MomentType; use DigitalCraftsman\DateTimePrecision\Doctrine\MonthType; use DigitalCraftsman\DateTimePrecision\Doctrine\TimeType; +use DigitalCraftsman\DateTimePrecision\Doctrine\WeekdaysType; use DigitalCraftsman\DateTimePrecision\Doctrine\WeekdayType; use DigitalCraftsman\DateTimePrecision\Doctrine\YearType; use PHPUnit\Framework\TestCase; @@ -46,6 +47,9 @@ public function process_works(): void self::assertArrayHasKey('dtp_weekday', $updatedParameters); self::assertSame(['class' => WeekdayType::class], $updatedParameters['dtp_weekday']); + self::assertArrayHasKey('dtp_weekdays', $updatedParameters); + self::assertSame(['class' => WeekdaysType::class], $updatedParameters['dtp_weekdays']); + self::assertArrayHasKey('dtp_date', $updatedParameters); self::assertSame(['class' => DateType::class], $updatedParameters['dtp_date']); From 30f19b1db4aa103470fb2dc7fb7068601da5acfc Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 14:26:54 +0200 Subject: [PATCH 19/22] Add test for weekdays type --- infection.json5 | 10 +++++ tests/Doctrine/WeekdaysTypeTest.php | 63 +++++++++++++++++++++++++++++ 2 files changed, 73 insertions(+) create mode 100644 tests/Doctrine/WeekdaysTypeTest.php diff --git a/infection.json5 b/infection.json5 index 64fc377..6c3e24c 100644 --- a/infection.json5 +++ b/infection.json5 @@ -12,6 +12,16 @@ "DigitalCraftsman\\DateTimePrecision\\Moment", ] }, + "IncrementInteger": { + "ignore": [ + "DigitalCraftsman\\DateTimePrecision\\Doctrine\\WeekdaysType::convertToPHPValue" + ] + }, + "DecrementInteger": { + "ignore": [ + "DigitalCraftsman\\DateTimePrecision\\Doctrine\\WeekdaysType::convertToPHPValue" + ] + } }, "minMsi": 100, "minCoveredMsi": 100 diff --git a/tests/Doctrine/WeekdaysTypeTest.php b/tests/Doctrine/WeekdaysTypeTest.php new file mode 100644 index 0000000..14cb373 --- /dev/null +++ b/tests/Doctrine/WeekdaysTypeTest.php @@ -0,0 +1,63 @@ +convertToDatabaseValue($weekdays, $platform); + $convertedValue = $doctrineType->convertToPHPValue($databaseValue, $platform); + + // -- Assert + self::assertSame($expectedDatabaseValue, $databaseValue); + self::assertEquals($weekdays, $convertedValue); + } + + /** + * @test + * + * @covers ::convertToDatabaseValue + * @covers ::convertToPHPValue + */ + public function convert_from_and_to_weekdays_php_value_works_with_null(): void + { + // -- Arrange + $doctrineType = new WeekdaysType(); + $platform = new PostgreSQLPlatform(); + + // -- Act + $databaseValue = $doctrineType->convertToDatabaseValue(null, $platform); + $phpValue = $doctrineType->convertToPHPValue($databaseValue, $platform); + + // -- Assert + self::assertNull($phpValue); + } +} From 2e8af172c3e7b66879cb089e7d763b6d49393b63 Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 20:41:13 +0200 Subject: [PATCH 20/22] Add test for normalizer --- tests/Serializer/WeekdaysNormalizerTest.php | 121 ++++++++++++++++++++ 1 file changed, 121 insertions(+) create mode 100644 tests/Serializer/WeekdaysNormalizerTest.php diff --git a/tests/Serializer/WeekdaysNormalizerTest.php b/tests/Serializer/WeekdaysNormalizerTest.php new file mode 100644 index 0000000..dc7d643 --- /dev/null +++ b/tests/Serializer/WeekdaysNormalizerTest.php @@ -0,0 +1,121 @@ +normalize($weekdays); + $denormalizedData = $normalizer->denormalize($normalizedData, Weekdays::class); + + // -- Assert + self::assertEquals($weekdays, $denormalizedData); + } + + /** + * @test + * + * @covers ::normalize + */ + public function weekday_normalization_with_null_works(): void + { + // -- Arrange + $normalizer = new WeekdaysNormalizer(); + + // -- Act + $normalizedData = $normalizer->normalize(null); + + // -- Assert + self::assertNull($normalizedData); + } + + /** + * @test + * + * @covers ::denormalize + */ + public function weekdays_denormalization_with_null_works(): void + { + // -- Arrange + $normalizer = new WeekdaysNormalizer(); + + // -- Act + $denormalizedData = $normalizer->denormalize(null, Weekdays::class); + + // -- Assert + self::assertNull($denormalizedData); + } + + /** + * @test + * + * @covers ::supportsNormalization + */ + public function supports_normalization(): void + { + // -- Arrange + $weekdays = new Weekdays([ + Weekday::MONDAY, + Weekday::SATURDAY, + ]); + + $normalizer = new WeekdaysNormalizer(); + + // -- Act & Assert + self::assertTrue($normalizer->supportsNormalization($weekdays)); + } + + /** + * @test + * + * @covers ::supportsNormalization + */ + public function supports_normalization_fails(): void + { + // -- Arrange + $year = Year::fromString('2022'); + + $normalizer = new WeekdaysNormalizer(); + + // -- Act & Assert + self::assertFalse($normalizer->supportsNormalization($year)); + } + + /** + * @test + * + * @covers ::supportsDenormalization + */ + public function supports_denormalization(): void + { + // -- Arrange + $normalizer = new WeekdaysNormalizer(); + + // -- Act & Assert + self::assertTrue($normalizer->supportsDenormalization(null, Weekdays::class)); + } +} From 7969ee85d570c5af5d4c06f59e46ee97f6bd205d Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 20:52:04 +0200 Subject: [PATCH 21/22] Validate for unique weekdays --- src/Weekdays.php | 8 ++++- tests/Weekdays/ConstructionTest.php | 47 +++++++++++++++++++++++++++++ 2 files changed, 54 insertions(+), 1 deletion(-) create mode 100644 tests/Weekdays/ConstructionTest.php diff --git a/src/Weekdays.php b/src/Weekdays.php index 87232d3..3282327 100644 --- a/src/Weekdays.php +++ b/src/Weekdays.php @@ -13,7 +13,13 @@ public function __construct( /** @var array $weekdays */ public array $weekdays, ) { - // TODO: Validate for unique + $enumValues = []; + foreach ($this->weekdays as $weekday) { + $enumValues[] = $weekday->value; + } + if (count($enumValues) !== count(array_unique($enumValues))) { + throw new \InvalidArgumentException('Weekdays must be unique.'); + } } // -- Array normalizable diff --git a/tests/Weekdays/ConstructionTest.php b/tests/Weekdays/ConstructionTest.php new file mode 100644 index 0000000..2a7625f --- /dev/null +++ b/tests/Weekdays/ConstructionTest.php @@ -0,0 +1,47 @@ +expectException(\InvalidArgumentException::class); + + // -- Act + new Weekdays([ + Weekday::MONDAY, + Weekday::MONDAY, + Weekday::TUESDAY, + ]); + } +} From bb7c66357cd48b979fcdc3d719b06a21d0a243bd Mon Sep 17 00:00:00 2001 From: Christian Kolb Date: Sun, 9 Jun 2024 21:01:05 +0200 Subject: [PATCH 22/22] Add final tests --- tests/Weekdays/ContainsTest.php | 58 +++++++++++++++++++++++++++ tests/Weekdays/NormalizationTest.php | 35 +++++++++++++++++ tests/Weekdays/NotContainsTest.php | 59 ++++++++++++++++++++++++++++ 3 files changed, 152 insertions(+) create mode 100644 tests/Weekdays/ContainsTest.php create mode 100644 tests/Weekdays/NormalizationTest.php create mode 100644 tests/Weekdays/NotContainsTest.php diff --git a/tests/Weekdays/ContainsTest.php b/tests/Weekdays/ContainsTest.php new file mode 100644 index 0000000..6dda7cc --- /dev/null +++ b/tests/Weekdays/ContainsTest.php @@ -0,0 +1,58 @@ +contains($comparator)); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + 'tuesday' => [ + true, + new Weekdays([ + Weekday::TUESDAY, + Weekday::WEDNESDAY, + ]), + Weekday::TUESDAY, + ], + 'thursday' => [ + false, + new Weekdays([ + Weekday::MONDAY, + Weekday::WEDNESDAY, + ]), + Weekday::THURSDAY, + ], + ]; + } +} diff --git a/tests/Weekdays/NormalizationTest.php b/tests/Weekdays/NormalizationTest.php new file mode 100644 index 0000000..17e8b85 --- /dev/null +++ b/tests/Weekdays/NormalizationTest.php @@ -0,0 +1,35 @@ +normalize(); + $denormalizedWeekdays = Weekdays::denormalize($normalizedWeekdays); + + // -- Assert + self::assertEquals($weekdays, $denormalizedWeekdays); + } +} diff --git a/tests/Weekdays/NotContainsTest.php b/tests/Weekdays/NotContainsTest.php new file mode 100644 index 0000000..58d302b --- /dev/null +++ b/tests/Weekdays/NotContainsTest.php @@ -0,0 +1,59 @@ +notContains($comparator)); + } + + /** + * @return array + */ + public static function dataProvider(): array + { + return [ + 'tuesday' => [ + true, + new Weekdays([ + Weekday::MONDAY, + Weekday::WEDNESDAY, + ]), + Weekday::TUESDAY, + ], + 'thursday' => [ + false, + new Weekdays([ + Weekday::MONDAY, + Weekday::WEDNESDAY, + Weekday::THURSDAY, + ]), + Weekday::THURSDAY, + ], + ]; + } +}