Find this useful? Enter your email to receive occasional updates for securing PHP code.
Signing you up...
Thank you for signing up!
PHP Decode
<?php declare(strict_types=1); namespace GraphQL\Validator\Rules; use GraphQL\Error\Erro..
Decoded Output download
<?php declare(strict_types=1);
namespace GraphQL\Validator\Rules;
use GraphQL\Error\Error;
use GraphQL\Language\AST\FieldNode;
use GraphQL\Language\AST\FragmentDefinitionNode;
use GraphQL\Language\AST\FragmentSpreadNode;
use GraphQL\Language\AST\InlineFragmentNode;
use GraphQL\Language\AST\Node;
use GraphQL\Language\AST\NodeKind;
use GraphQL\Language\AST\OperationDefinitionNode;
use GraphQL\Language\AST\SelectionSetNode;
use GraphQL\Validator\QueryValidationContext;
class QueryDepth extends QuerySecurityRule
{
/** @var array<string, bool> Fragment names which are already calculated in recursion */
protected array $calculatedFragments = [];
protected int $maxQueryDepth;
/** @throws \InvalidArgumentException */
public function __construct(int $maxQueryDepth)
{
$this->setMaxQueryDepth($maxQueryDepth);
}
public function getVisitor(QueryValidationContext $context): array
{
return $this->invokeIfNeeded(
$context,
[
NodeKind::OPERATION_DEFINITION => [
'leave' => function (OperationDefinitionNode $operationDefinition) use ($context): void {
$maxDepth = $this->fieldDepth($operationDefinition);
if ($maxDepth <= $this->maxQueryDepth) {
return;
}
$context->reportError(
new Error(static::maxQueryDepthErrorMessage($this->maxQueryDepth, $maxDepth))
);
},
],
]
);
}
/** @param OperationDefinitionNode|FieldNode|InlineFragmentNode|FragmentDefinitionNode $node */
protected function fieldDepth(Node $node, int $depth = 0, int $maxDepth = 0): int
{
if ($node->selectionSet instanceof SelectionSetNode) {
foreach ($node->selectionSet->selections as $childNode) {
$maxDepth = $this->nodeDepth($childNode, $depth, $maxDepth);
}
}
return $maxDepth;
}
protected function nodeDepth(Node $node, int $depth = 0, int $maxDepth = 0): int
{
switch (true) {
case $node instanceof FieldNode:
// node has children?
if ($node->selectionSet !== null) {
// update maxDepth if needed
if ($depth > $maxDepth) {
$maxDepth = $depth;
}
$maxDepth = $this->fieldDepth($node, $depth + 1, $maxDepth);
}
break;
case $node instanceof InlineFragmentNode:
$maxDepth = $this->fieldDepth($node, $depth, $maxDepth);
break;
case $node instanceof FragmentSpreadNode:
$fragment = $this->getFragment($node);
if ($fragment !== null) {
$name = $fragment->name->value;
if (isset($this->calculatedFragments[$name])) {
return $this->maxQueryDepth + 1;
}
$this->calculatedFragments[$name] = true;
$maxDepth = $this->fieldDepth($fragment, $depth, $maxDepth);
unset($this->calculatedFragments[$name]);
}
break;
}
return $maxDepth;
}
public function getMaxQueryDepth(): int
{
return $this->maxQueryDepth;
}
/**
* Set max query depth. If equal to 0 no check is done. Must be greater or equal to 0.
*
* @throws \InvalidArgumentException
*/
public function setMaxQueryDepth(int $maxQueryDepth): void
{
$this->checkIfGreaterOrEqualToZero('maxQueryDepth', $maxQueryDepth);
$this->maxQueryDepth = $maxQueryDepth;
}
public static function maxQueryDepthErrorMessage(int $max, int $count): string
{
return "Max query depth should be {$max} but got {$count}.";
}
protected function isEnabled(): bool
{
return $this->maxQueryDepth !== self::DISABLED;
}
}
?>
Did this file decode correctly?
Original Code
<?php declare(strict_types=1);
namespace GraphQL\Validator\Rules;
use GraphQL\Error\Error;
use GraphQL\Language\AST\FieldNode;
use GraphQL\Language\AST\FragmentDefinitionNode;
use GraphQL\Language\AST\FragmentSpreadNode;
use GraphQL\Language\AST\InlineFragmentNode;
use GraphQL\Language\AST\Node;
use GraphQL\Language\AST\NodeKind;
use GraphQL\Language\AST\OperationDefinitionNode;
use GraphQL\Language\AST\SelectionSetNode;
use GraphQL\Validator\QueryValidationContext;
class QueryDepth extends QuerySecurityRule
{
/** @var array<string, bool> Fragment names which are already calculated in recursion */
protected array $calculatedFragments = [];
protected int $maxQueryDepth;
/** @throws \InvalidArgumentException */
public function __construct(int $maxQueryDepth)
{
$this->setMaxQueryDepth($maxQueryDepth);
}
public function getVisitor(QueryValidationContext $context): array
{
return $this->invokeIfNeeded(
$context,
[
NodeKind::OPERATION_DEFINITION => [
'leave' => function (OperationDefinitionNode $operationDefinition) use ($context): void {
$maxDepth = $this->fieldDepth($operationDefinition);
if ($maxDepth <= $this->maxQueryDepth) {
return;
}
$context->reportError(
new Error(static::maxQueryDepthErrorMessage($this->maxQueryDepth, $maxDepth))
);
},
],
]
);
}
/** @param OperationDefinitionNode|FieldNode|InlineFragmentNode|FragmentDefinitionNode $node */
protected function fieldDepth(Node $node, int $depth = 0, int $maxDepth = 0): int
{
if ($node->selectionSet instanceof SelectionSetNode) {
foreach ($node->selectionSet->selections as $childNode) {
$maxDepth = $this->nodeDepth($childNode, $depth, $maxDepth);
}
}
return $maxDepth;
}
protected function nodeDepth(Node $node, int $depth = 0, int $maxDepth = 0): int
{
switch (true) {
case $node instanceof FieldNode:
// node has children?
if ($node->selectionSet !== null) {
// update maxDepth if needed
if ($depth > $maxDepth) {
$maxDepth = $depth;
}
$maxDepth = $this->fieldDepth($node, $depth + 1, $maxDepth);
}
break;
case $node instanceof InlineFragmentNode:
$maxDepth = $this->fieldDepth($node, $depth, $maxDepth);
break;
case $node instanceof FragmentSpreadNode:
$fragment = $this->getFragment($node);
if ($fragment !== null) {
$name = $fragment->name->value;
if (isset($this->calculatedFragments[$name])) {
return $this->maxQueryDepth + 1;
}
$this->calculatedFragments[$name] = true;
$maxDepth = $this->fieldDepth($fragment, $depth, $maxDepth);
unset($this->calculatedFragments[$name]);
}
break;
}
return $maxDepth;
}
public function getMaxQueryDepth(): int
{
return $this->maxQueryDepth;
}
/**
* Set max query depth. If equal to 0 no check is done. Must be greater or equal to 0.
*
* @throws \InvalidArgumentException
*/
public function setMaxQueryDepth(int $maxQueryDepth): void
{
$this->checkIfGreaterOrEqualToZero('maxQueryDepth', $maxQueryDepth);
$this->maxQueryDepth = $maxQueryDepth;
}
public static function maxQueryDepthErrorMessage(int $max, int $count): string
{
return "Max query depth should be {$max} but got {$count}.";
}
protected function isEnabled(): bool
{
return $this->maxQueryDepth !== self::DISABLED;
}
}
Function Calls
None |
Stats
MD5 | 2206aef14a6abf0e925b86d77b0d57cb |
Eval Count | 0 |
Decode Time | 107 ms |