![]() Server : Apache System : Linux server2.corals.io 4.18.0-348.2.1.el8_5.x86_64 #1 SMP Mon Nov 15 09:17:08 EST 2021 x86_64 User : corals ( 1002) PHP Version : 7.4.33 Disable Function : exec,passthru,shell_exec,system Directory : /home/corals/old/vendor/webonyx/graphql-php/docs/ |
## Protection Against Malicious Queries GraphQL allows a large degree of dynamism and flexibility for clients to control what happens on the server during query execution. Malicious clients may abuse this by sending very deep and complex queries whose execution exhausts server resources. At a basic level, it is recommended to limit the resources a single HTTP request can use through PHP settings such as: - [`max_execution_time`](https://www.php.net/manual/en/info.configuration.php#ini.max-execution-time) - [`memory_limit`](https://www.php.net/manual/en/ini.core.php#ini.memory-limit) - [`post_max_size`](https://www.php.net/manual/en/ini.core.php#ini.post-max-size) In addition, graphql-php offers security mechanisms that are specific to GraphQL. ## Query Complexity Analysis This is a port of [Query Complexity Analysis in Sangria](https://sangria-graphql.github.io/learn#query-complexity-analysis). Complexity analysis is a separate validation rule which calculates query complexity score before execution. Every field in the query gets a default score 1 (including ObjectType nodes). Total complexity of the query is the sum of all field scores. For example, the complexity of introspection query is **109**. If this score exceeds a threshold, a query is not executed and an error is returned instead. Complexity analysis is disabled by default. You may enable it by setting a maximum query complexity: ```php use GraphQL\GraphQL; use GraphQL\Validator\Rules\QueryComplexity; use GraphQL\Validator\DocumentValidator; $rule = new QueryComplexity(100); DocumentValidator::addRule($rule); GraphQL::executeQuery(/*...*/); ``` This will set the rule globally. Alternatively, you can provide validation rules [per execution](executing-queries.md#custom-validation-rules). To customize field score add **complexity** function to field definition: ```php use GraphQL\Type\Definition\Type; use GraphQL\Type\Definition\ObjectType; $type = new ObjectType([ 'name' => 'MyType', 'fields' => [ 'someList' => [ 'type' => Type::listOf(Type::string()), 'args' => [ 'limit' => [ 'type' => Type::int(), 'defaultValue' => 10 ] ], 'complexity' => fn (int $childrenComplexity, array $args): int => $childrenComplexity * $args['limit'], ] ] ]); ``` ## Limiting Query Depth This is a port of [Limiting Query Depth in Sangria](https://sangria-graphql.github.io/learn#limiting-query-depth). This is a simpler approach that limits the nesting depth a query can have. For example, the depth of the default introspection query is **7**. This rule is disabled by default. You may enable it by setting a maximum query depth: ```php use GraphQL\GraphQL; use GraphQL\Validator\Rules\QueryDepth; use GraphQL\Validator\DocumentValidator; $rule = new QueryDepth(10); DocumentValidator::addRule($rule); GraphQL::executeQuery(/*...*/); ``` This will set the rule globally. Alternatively, you can provide validation rules [per execution](executing-queries.md#custom-validation-rules). ## Disabling Introspection [Introspection](https://graphql.org/learn/introspection/) is a mechanism for fetching schema structure. It is used by tools like GraphiQL for auto-completion, query validation, etc. Introspection is enabled by default. It means that anybody can get a full description of your schema by sending a special query containing meta fields **\_\_type** and **\_\_schema** . If you are not planning to expose your API to the general public, it makes sense to disable this feature. GraphQL PHP provides you separate validation rule which prohibits queries that contain **\_\_type** or **\_\_schema** fields. To disable introspection, add following rule: ```php use GraphQL\GraphQL; use GraphQL\Validator\Rules\DisableIntrospection; use GraphQL\Validator\DocumentValidator; $rule = new DisableIntrospection(DisableIntrospection::ENABLED); DocumentValidator::addRule($rule); GraphQL::executeQuery(/*...*/); ``` This will set the rule globally. Alternatively, you can provide validation rules [per execution](executing-queries.md#custom-validation-rules).