From 7093d9b1887e8f869f060762ec7c5ac219776d66 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?J=C3=A9r=C3=B4me=20Tamarelle?= Date: Mon, 16 Sep 2024 09:20:30 +0200 Subject: [PATCH 1/2] Replace arrayHasKey with assertArrayHasKey in tests (#1403) --- tests/Operation/WatchFunctionalTest.php | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/tests/Operation/WatchFunctionalTest.php b/tests/Operation/WatchFunctionalTest.php index 8715bb6b1..562192f1b 100644 --- a/tests/Operation/WatchFunctionalTest.php +++ b/tests/Operation/WatchFunctionalTest.php @@ -307,15 +307,15 @@ function (array $event) use (&$events): void { $this->assertCount(3, $events); $this->assertSame('getMore', $events[0]['started']->getCommandName()); - $this->arrayHasKey('failed', $events[0]); + $this->assertArrayHasKey('failed', $events[0]); $this->assertSame('aggregate', $events[1]['started']->getCommandName()); $this->assertResumeAfter($postBatchResumeToken, $events[1]['started']->getCommand()); - $this->arrayHasKey('succeeded', $events[1]); + $this->assertArrayHasKey('succeeded', $events[1]); // Original cursor is freed immediately after the change stream resumes $this->assertSame('killCursors', $events[2]['started']->getCommandName()); - $this->arrayHasKey('succeeded', $events[2]); + $this->assertArrayHasKey('succeeded', $events[2]); $this->assertFalse($changeStream->valid()); } @@ -380,15 +380,15 @@ function (array $event) use (&$events): void { $this->assertCount(3, $events); $this->assertSame('getMore', $events[0]['started']->getCommandName()); - $this->arrayHasKey('failed', $events[0]); + $this->assertArrayHasKey('failed', $events[0]); $this->assertSame('aggregate', $events[1]['started']->getCommandName()); $this->assertStartAtOperationTime($operationTime, $events[1]['started']->getCommand()); - $this->arrayHasKey('succeeded', $events[1]); + $this->assertArrayHasKey('succeeded', $events[1]); // Original cursor is freed immediately after the change stream resumes $this->assertSame('killCursors', $events[2]['started']->getCommandName()); - $this->arrayHasKey('succeeded', $events[2]); + $this->assertArrayHasKey('succeeded', $events[2]); $this->assertFalse($changeStream->valid()); } From f2842bcd8be9cad85a2bdf9ab85d1312e8b7d1ce Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?J=C3=A9r=C3=B4me=20Tamarelle?= Date: Mon, 16 Sep 2024 10:30:29 +0200 Subject: [PATCH 2/2] Fix types accepted by $round (#1401) --- generator/config/expression/round.yaml | 20 ++++++++++---- src/Builder/Expression/FactoryTrait.php | 6 ++--- src/Builder/Expression/RoundOperator.php | 10 +++---- tests/Builder/Expression/Pipelines.php | 26 +++++++++++++++++++ .../Builder/Expression/RoundOperatorTest.php | 18 +++++++++++++ 5 files changed, 67 insertions(+), 13 deletions(-) diff --git a/generator/config/expression/round.yaml b/generator/config/expression/round.yaml index 52e6004a0..9bc6961c7 100644 --- a/generator/config/expression/round.yaml +++ b/generator/config/expression/round.yaml @@ -8,15 +8,12 @@ type: - resolvesToLong encode: array description: | - Rounds a number to to a whole integer or to a specified decimal place. + Rounds a number to a whole integer or to a specified decimal place. arguments: - name: number type: - - resolvesToInt - - resolvesToDouble - - resolvesToDecimal - - resolvesToLong + - resolvesToNumber description: | Can be any valid expression that resolves to a number. Specifically, the expression must resolve to an integer, double, decimal, or long. $round returns an error if the expression resolves to a non-numeric data type. @@ -38,3 +35,16 @@ tests: $round: - '$value' - 1 + - + name: 'Round Average Rating' + pipeline: + - + $project: + roundedAverageRating: + $avg: + - + $round: + - + $avg: + - '$averageRating' + - 2 diff --git a/src/Builder/Expression/FactoryTrait.php b/src/Builder/Expression/FactoryTrait.php index 7828951ea..78f460fa0 100644 --- a/src/Builder/Expression/FactoryTrait.php +++ b/src/Builder/Expression/FactoryTrait.php @@ -1546,15 +1546,15 @@ public static function reverseArray(PackedArray|ResolvesToArray|BSONArray|array } /** - * Rounds a number to to a whole integer or to a specified decimal place. + * Rounds a number to a whole integer or to a specified decimal place. * * @see https://www.mongodb.com/docs/manual/reference/operator/aggregation/round/ - * @param Decimal128|Int64|ResolvesToDecimal|ResolvesToDouble|ResolvesToInt|ResolvesToLong|float|int $number Can be any valid expression that resolves to a number. Specifically, the expression must resolve to an integer, double, decimal, or long. + * @param Decimal128|Int64|ResolvesToNumber|float|int $number Can be any valid expression that resolves to a number. Specifically, the expression must resolve to an integer, double, decimal, or long. * $round returns an error if the expression resolves to a non-numeric data type. * @param Optional|ResolvesToInt|int $place Can be any valid expression that resolves to an integer between -20 and 100, exclusive. */ public static function round( - Decimal128|Int64|ResolvesToDecimal|ResolvesToDouble|ResolvesToInt|ResolvesToLong|float|int $number, + Decimal128|Int64|ResolvesToNumber|float|int $number, Optional|ResolvesToInt|int $place = Optional::Undefined, ): RoundOperator { return new RoundOperator($number, $place); diff --git a/src/Builder/Expression/RoundOperator.php b/src/Builder/Expression/RoundOperator.php index 7f3cad53e..323ae99eb 100644 --- a/src/Builder/Expression/RoundOperator.php +++ b/src/Builder/Expression/RoundOperator.php @@ -15,7 +15,7 @@ use MongoDB\Builder\Type\Optional; /** - * Rounds a number to to a whole integer or to a specified decimal place. + * Rounds a number to a whole integer or to a specified decimal place. * * @see https://www.mongodb.com/docs/manual/reference/operator/aggregation/round/ */ @@ -24,21 +24,21 @@ class RoundOperator implements ResolvesToInt, ResolvesToDouble, ResolvesToDecima public const ENCODE = Encode::Array; /** - * @var Decimal128|Int64|ResolvesToDecimal|ResolvesToDouble|ResolvesToInt|ResolvesToLong|float|int $number Can be any valid expression that resolves to a number. Specifically, the expression must resolve to an integer, double, decimal, or long. + * @var Decimal128|Int64|ResolvesToNumber|float|int $number Can be any valid expression that resolves to a number. Specifically, the expression must resolve to an integer, double, decimal, or long. * $round returns an error if the expression resolves to a non-numeric data type. */ - public readonly Decimal128|Int64|ResolvesToDecimal|ResolvesToDouble|ResolvesToInt|ResolvesToLong|float|int $number; + public readonly Decimal128|Int64|ResolvesToNumber|float|int $number; /** @var Optional|ResolvesToInt|int $place Can be any valid expression that resolves to an integer between -20 and 100, exclusive. */ public readonly Optional|ResolvesToInt|int $place; /** - * @param Decimal128|Int64|ResolvesToDecimal|ResolvesToDouble|ResolvesToInt|ResolvesToLong|float|int $number Can be any valid expression that resolves to a number. Specifically, the expression must resolve to an integer, double, decimal, or long. + * @param Decimal128|Int64|ResolvesToNumber|float|int $number Can be any valid expression that resolves to a number. Specifically, the expression must resolve to an integer, double, decimal, or long. * $round returns an error if the expression resolves to a non-numeric data type. * @param Optional|ResolvesToInt|int $place Can be any valid expression that resolves to an integer between -20 and 100, exclusive. */ public function __construct( - Decimal128|Int64|ResolvesToDecimal|ResolvesToDouble|ResolvesToInt|ResolvesToLong|float|int $number, + Decimal128|Int64|ResolvesToNumber|float|int $number, Optional|ResolvesToInt|int $place = Optional::Undefined, ) { $this->number = $number; diff --git a/tests/Builder/Expression/Pipelines.php b/tests/Builder/Expression/Pipelines.php index f5a4bcec1..f84d00984 100644 --- a/tests/Builder/Expression/Pipelines.php +++ b/tests/Builder/Expression/Pipelines.php @@ -4266,6 +4266,32 @@ enum Pipelines: string ] JSON; + /** Round Average Rating */ + case RoundRoundAverageRating = <<<'JSON' + [ + { + "$project": { + "roundedAverageRating": { + "$avg": [ + { + "$round": [ + { + "$avg": [ + "$averageRating" + ] + }, + { + "$numberInt": "2" + } + ] + } + ] + } + } + } + ] + JSON; + /** * Example * diff --git a/tests/Builder/Expression/RoundOperatorTest.php b/tests/Builder/Expression/RoundOperatorTest.php index accae5074..758668818 100644 --- a/tests/Builder/Expression/RoundOperatorTest.php +++ b/tests/Builder/Expression/RoundOperatorTest.php @@ -27,4 +27,22 @@ public function testExample(): void $this->assertSamePipeline(Pipelines::RoundExample, $pipeline); } + + public function testRoundAverageRating(): void + { + $pipeline = new Pipeline( + Stage::project( + roundedAverageRating: Expression::avg( + Expression::round( + Expression::avg( + Expression::doubleFieldPath('averageRating'), + ), + 2, + ), + ), + ), + ); + + $this->assertSamePipeline(Pipelines::RoundRoundAverageRating, $pipeline); + } }