From d1faf683afd6380d1195cf8219d25dc9e70e14be Mon Sep 17 00:00:00 2001 From: Julien Deniau Date: Tue, 11 Sep 2018 09:47:50 +0200 Subject: [PATCH 1/4] increase JSON context DX --- src/Context/JsonContext.php | 55 +++++++++++++++++++------------------ 1 file changed, 29 insertions(+), 26 deletions(-) diff --git a/src/Context/JsonContext.php b/src/Context/JsonContext.php index 99630567..060e028f 100644 --- a/src/Context/JsonContext.php +++ b/src/Context/JsonContext.php @@ -3,13 +3,11 @@ namespace Behatch\Context; use Behat\Gherkin\Node\PyStringNode; - use Behat\Gherkin\Node\TableNode; -use Behat\Mink\Exception\ExpectationException; +use Behatch\HttpCall\HttpCallResultPool; use Behatch\Json\Json; -use Behatch\Json\JsonSchema; use Behatch\Json\JsonInspector; -use Behatch\HttpCall\HttpCallResultPool; +use Behatch\Json\JsonSchema; class JsonContext extends BaseContext { @@ -49,17 +47,17 @@ public function theResponseShouldNotBeInJson() /** * Checks, that given JSON node is equal to given value * - * @Then the JSON node :node should be equal to :text + * @Then the JSON node :node should be equal to :expected */ - public function theJsonNodeShouldBeEqualTo($node, $text) + public function theJsonNodeShouldBeEqualTo($node, $expected) { $json = $this->getJson(); $actual = $this->inspector->evaluate($json, $node); - if ($actual != $text) { + if ($actual != $expected) { throw new \Exception( - sprintf("The node value is '%s'", json_encode($actual)) + sprintf("The node '%s' value is '%s', '%s' expected", $node, json_encode($actual), json_encode($expected)) ); } } @@ -89,7 +87,7 @@ public function theJsonNodeShouldMatch($node, $pattern) if (preg_match($pattern, $actual) === 0) { throw new \Exception( - sprintf("The node value is '%s'", json_encode($actual)) + sprintf("The node '%s' value is '%s', '%s' pattern expected", $node, json_encode($actual), $pattern) ); } } @@ -107,7 +105,7 @@ public function theJsonNodeShouldBeNull($node) if (null !== $actual) { throw new \Exception( - sprintf('The node value is `%s`', json_encode($actual)) + sprintf("The node '%s' value is '%s', null expected", $node, json_encode($actual)) ); } } @@ -119,9 +117,15 @@ public function theJsonNodeShouldBeNull($node) */ public function theJsonNodeShouldNotBeNull($node) { - $this->not(function () use ($node) { - return $this->theJsonNodeShouldBeNull($node); - }, sprintf('The node %s should not be null', $node)); + $json = $this->getJson(); + + $actual = $this->inspector->evaluate($json, $node); + + if (null === $actual) { + throw new \Exception( + sprintf("The node '%s' value is null, '%s' expected", $node, json_encode($actual)) + ); + } } /** @@ -137,7 +141,7 @@ public function theJsonNodeShouldBeTrue($node) if (true !== $actual) { throw new \Exception( - sprintf('The node value is `%s`', json_encode($actual)) + sprintf("The node '%s' value is '%s', 'true' expected", $node, json_encode($actual)) ); } } @@ -155,7 +159,7 @@ public function theJsonNodeShouldBeFalse($node) if (false !== $actual) { throw new \Exception( - sprintf('The node value is `%s`', json_encode($actual)) + sprintf("The node '%s' value is '%s', 'false' expected", $node, json_encode($actual)) ); } } @@ -163,17 +167,17 @@ public function theJsonNodeShouldBeFalse($node) /** * Checks, that given JSON node is equal to the given string * - * @Then the JSON node :node should be equal to the string :text + * @Then the JSON node :node should be equal to the string :expected */ - public function theJsonNodeShouldBeEqualToTheString($node, $text) + public function theJsonNodeShouldBeEqualToTheString($node, $expected) { $json = $this->getJson(); $actual = $this->inspector->evaluate($json, $node); - if ($actual !== $text) { + if ($actual !== $expected) { throw new \Exception( - sprintf('The node value is `%s`', json_encode($actual)) + sprintf("The node '%s' value is '%s', string '%s' expected", $node, json_encode($actual), $expected) ); } } @@ -191,7 +195,7 @@ public function theJsonNodeShouldBeEqualToTheNumber($node, $number) if ($actual !== (float) $number && $actual !== (int) $number) { throw new \Exception( - sprintf('The node value is `%s`', json_encode($actual)) + sprintf("The node '%s' value is '%s', numder '%6' expected", $node, json_encode($actual), (float) $number) ); } } @@ -207,7 +211,7 @@ public function theJsonNodeShouldHaveElements($node, $count) $actual = $this->inspector->evaluate($json, $node); - $this->assertSame($count, sizeof((array) $actual)); + $this->assertSame($count, count((array) $actual)); } /** @@ -276,6 +280,7 @@ public function theJsonNodeShouldExist($name) } catch (\Exception $e) { throw new \Exception("The node '$name' does not exist."); } + return $node; } @@ -337,7 +342,7 @@ public function theJsonShouldBeInvalidAccordingToTheSchema($filename) $this->not(function () use ($filename) { return $this->theJsonShouldBeValidAccordingToTheSchema($filename); - }, "The schema was valid"); + }, 'The schema was valid'); } /** @@ -356,7 +361,7 @@ public function theJsonShouldBeEqualTo(PyStringNode $content) $this->assertSame( (string) $expected, (string) $actual, - "The json is equal to:\n". $actual->encode() + "The json is equal to:\n" . $actual->encode() ); } @@ -390,8 +395,8 @@ public function theJsonShouldBeValidAccordingToTheSwaggerSchema($dumpPath, $sche ) ); } + /** - * * Checks, that response JSON not matches with a swagger dump * * @Then the JSON should not be valid according to swagger :dumpPath dump schema :schemaName @@ -403,8 +408,6 @@ public function theJsonShouldNotBeValidAccordingToTheSwaggerSchema($dumpPath, $s }, 'JSON Schema matches but it should not'); } - - protected function getJson() { return new Json($this->httpCallResultPool->getResult()->getValue()); From 200aa8adab02be92006d6d41ef3e4d9be63c3183 Mon Sep 17 00:00:00 2001 From: Julien Deniau Date: Tue, 11 Sep 2018 15:01:27 +0200 Subject: [PATCH 2/4] add tests --- src/Context/JsonContext.php | 21 ++- tests/units/Context/JsonContext.php | 195 ++++++++++++++++++++++++++++ 2 files changed, 211 insertions(+), 5 deletions(-) create mode 100644 tests/units/Context/JsonContext.php diff --git a/src/Context/JsonContext.php b/src/Context/JsonContext.php index 060e028f..973c0262 100644 --- a/src/Context/JsonContext.php +++ b/src/Context/JsonContext.php @@ -57,7 +57,7 @@ public function theJsonNodeShouldBeEqualTo($node, $expected) if ($actual != $expected) { throw new \Exception( - sprintf("The node '%s' value is '%s', '%s' expected", $node, json_encode($actual), json_encode($expected)) + sprintf("The node '%s' value is '%s', '%s' expected", $node, json_encode($actual), $expected) ); } } @@ -69,8 +69,19 @@ public function theJsonNodeShouldBeEqualTo($node, $expected) */ public function theJsonNodesShouldBeEqualTo(TableNode $nodes) { - foreach ($nodes->getRowsHash() as $node => $text) { - $this->theJsonNodeShouldBeEqualTo($node, $text); + $json = $this->getJson(); + + $errors = []; + foreach ($nodes->getRowsHash() as $node => $expected) { + $actual = $this->inspector->evaluate($json, $node); + + if ($actual != $expected) { + $errors[] = sprintf("The node '%s' value is '%s', '%s' expected", $node, json_encode($actual), $expected); + } + } + + if (!empty($errors)) { + throw new \Exception(implode("\n", $errors)); } } @@ -123,7 +134,7 @@ public function theJsonNodeShouldNotBeNull($node) if (null === $actual) { throw new \Exception( - sprintf("The node '%s' value is null, '%s' expected", $node, json_encode($actual)) + sprintf("The node '%s' value is null, non-null value expected", $node) ); } } @@ -195,7 +206,7 @@ public function theJsonNodeShouldBeEqualToTheNumber($node, $number) if ($actual !== (float) $number && $actual !== (int) $number) { throw new \Exception( - sprintf("The node '%s' value is '%s', numder '%6' expected", $node, json_encode($actual), (float) $number) + sprintf("The node '%s' value is '%s', number '%s' expected", $node, json_encode($actual), (string) $number) ); } } diff --git a/tests/units/Context/JsonContext.php b/tests/units/Context/JsonContext.php new file mode 100644 index 00000000..1e138a71 --- /dev/null +++ b/tests/units/Context/JsonContext.php @@ -0,0 +1,195 @@ +mockGenerator->orphanize('__construct'); + $httpCallResult = $this->newMockInstance(HttpCallResult::class); + $httpCallResult->getMockController()->getValue = json_encode([ + 'a string node' => 'some string', + 'another string node' => 'some other string', + 'a null node' => null, + 'a true node' => true, + 'a false node' => false, + 'a number node' => 3, + 'an array node' => [ + 'one', + 'two', + 'three', + ], + ]); + + $this->httpCallResultPool = $this->newMockInstance(HttpCallResultPool::class); + $this->httpCallResultPool->getMockController()->getResult = $httpCallResult; + } + + public function testTheJsonNodeShouldBeEqualTo() + { + $this + ->given($this->newTestedInstance($this->httpCallResultPool)) + ->then + + ->if($this->testedInstance->theJsonNodeShouldBeEqualTo('a string node', 'some string')) + + ->exception(function () { + $this->testedInstance->theJsonNodeShouldBeEqualTo('a string node', 'expectedstring'); + }) + ->hasMessage("The node 'a string node' value is '\"some string\"', 'expectedstring' expected") + ; + } + + public function testTheJsonNodesShouldBeEqualTo() + { + $this + ->given($this->newTestedInstance($this->httpCallResultPool)) + ->and($validTableNode = new TableNode([ + 1 => ['a string node', 'some string'], + 2 => ['another string node', 'some other string'], + ])) + ->then + + ->if($this->testedInstance->theJsonNodesShouldBeEqualTo($validTableNode)) + + ->exception(function () { + $invalidTableNode = new TableNode([ + 1 => ['a string node', 'may the force'], + 2 => ['another string node', 'be with you'], + ]); + $this->testedInstance->theJsonNodesShouldBeEqualTo($invalidTableNode); + }) + ->hasMessage("The node 'a string node' value is '\"some string\"', 'may the force' expected\nThe node 'another string node' value is '\"some other string\"', 'be with you' expected") + ; + } + public function testTheJsonNodeShouldMatch() + { + $this + ->given($this->newTestedInstance($this->httpCallResultPool)) + ->then + + ->if($this->testedInstance->theJsonNodeShouldMatch('a string node', '/some/')) + + ->exception(function () { + $this->testedInstance->theJsonNodeShouldMatch('a string node', '/nomatch/'); + }) + ->hasMessage("The node 'a string node' value is '\"some string\"', '/nomatch/' pattern expected") + ; + } + + public function testTheJsonNodeShouldBeNull() + { + $this + ->given($this->newTestedInstance($this->httpCallResultPool)) + ->then + + ->if($this->testedInstance->theJsonNodeShouldBeNull('a null node')) + + ->exception(function () { + $this->testedInstance->theJsonNodeShouldBeNull('a string node'); + }) + ->hasMessage("The node 'a string node' value is '\"some string\"', null expected") + ; + } + + public function testTheJsonNodeShouldNotBeNull() + { + $this + ->given($this->newTestedInstance($this->httpCallResultPool)) + ->then + + ->if($this->testedInstance->theJsonNodeShouldNotBeNull('a string node')) + + ->exception(function () { + $this->testedInstance->theJsonNodeShouldNotBeNull('a null node'); + }) + ->hasMessage("The node 'a null node' value is null, non-null value expected") + ; + } + + public function testTheJsonNodeShouldBeTrue() + { + $this + ->given($this->newTestedInstance($this->httpCallResultPool)) + ->then + + ->if($this->testedInstance->theJsonNodeShouldBeTrue('a true node')) + + ->exception(function () { + $this->testedInstance->theJsonNodeShouldBeTrue('a false node'); + }) + ->hasMessage("The node 'a false node' value is 'false', 'true' expected") + ; + } + + public function testTheJsonNodeShouldBeFalse() + { + $this + ->given($this->newTestedInstance($this->httpCallResultPool)) + ->then + + ->if($this->testedInstance->theJsonNodeShouldBeFalse('a false node')) + + ->exception(function () { + $this->testedInstance->theJsonNodeShouldBeFalse('a true node'); + }) + ->hasMessage("The node 'a true node' value is 'true', 'false' expected") + ; + } + + public function testTheJsonNodeShouldBeEqualToTheString() + { + $this + ->given($this->newTestedInstance($this->httpCallResultPool)) + ->then + + ->if($this->testedInstance->theJsonNodeShouldBeEqualToTheString('a string node', 'some string')) + + ->exception(function () { + $this->testedInstance->theJsonNodeShouldBeEqualToTheString('a string node', 'expected'); + }) + ->hasMessage("The node 'a string node' value is '\"some string\"', string 'expected' expected") + ; + } + + public function testTheJsonNodeShouldBeEqualToTheNumber() + { + $this + ->given($this->newTestedInstance($this->httpCallResultPool)) + ->then + + ->if($this->testedInstance->theJsonNodeShouldBeEqualToTheNumber('a number node', 3)) + + ->exception(function () { + $this->testedInstance->theJsonNodeShouldBeEqualToTheNumber('a number node', 2); + }) + ->hasMessage("The node 'a number node' value is '3', number '2' expected") + ; + } + + public function testTheJsonNodeShouldExist() + { + $this + ->given($this->newTestedInstance($this->httpCallResultPool)) + ->then + + ->if($this->testedInstance->theJsonNodeShouldExist('a string node')) + + ->exception(function () { + $this->testedInstance->theJsonNodeShouldExist('invalid key'); + }) + ->hasMessage("The node 'invalid key' does not exist.") + ; + } +} From 389cf969ccdd9bef928bf0860ebfc732adedd498 Mon Sep 17 00:00:00 2001 From: didier2l Date: Wed, 21 Sep 2022 09:41:27 +0200 Subject: [PATCH 3/4] fix ReflectionClass::getClass deprecated (php8) --- src/HttpCall/HttpCallResultPoolResolver.php | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/HttpCall/HttpCallResultPoolResolver.php b/src/HttpCall/HttpCallResultPoolResolver.php index 988d8533..a1d10376 100644 --- a/src/HttpCall/HttpCallResultPoolResolver.php +++ b/src/HttpCall/HttpCallResultPoolResolver.php @@ -24,10 +24,10 @@ public function resolveArguments(\ReflectionClass $classReflection, array $argum $parameters = $constructor->getParameters(); foreach ($parameters as $parameter) { if ( - null !== $parameter->getClass() - && isset($this->dependencies[$parameter->getClass()->name]) + null !== $parameter->getType() + && isset($this->dependencies[$parameter->getType()->getName()]) ) { - $arguments[$parameter->name] = $this->dependencies[$parameter->getClass()->name]; + $arguments[$parameter->name] = $this->dependencies[$parameter->getType()->getName()]; } } } From 1f986cd4216dbe3c5e17700ae88b51878d49a55b Mon Sep 17 00:00:00 2001 From: Julien Deniau Date: Mon, 14 Nov 2022 12:17:15 +0100 Subject: [PATCH 4/4] Reespace \n special gherkin char --- src/Context/JsonContext.php | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/src/Context/JsonContext.php b/src/Context/JsonContext.php index 973c0262..7a4628a2 100644 --- a/src/Context/JsonContext.php +++ b/src/Context/JsonContext.php @@ -53,6 +53,8 @@ public function theJsonNodeShouldBeEqualTo($node, $expected) { $json = $this->getJson(); + $expected = self::reespaceSpecialGherkinValue($expected); + $actual = $this->inspector->evaluate($json, $node); if ($actual != $expected) { @@ -75,6 +77,8 @@ public function theJsonNodesShouldBeEqualTo(TableNode $nodes) foreach ($nodes->getRowsHash() as $node => $expected) { $actual = $this->inspector->evaluate($json, $node); + $expected = self::reespaceSpecialGherkinValue($expected); + if ($actual != $expected) { $errors[] = sprintf("The node '%s' value is '%s', '%s' expected", $node, json_encode($actual), $expected); } @@ -432,4 +436,9 @@ private function checkSchemaFile($filename) ); } } + + public static function reespaceSpecialGherkinValue(string $value): string + { + return str_replace("\\n", "\n", $value); + } }