diff --git a/docs/error-handling.md b/docs/error-handling.md index 8582949..17b0a2a 100644 --- a/docs/error-handling.md +++ b/docs/error-handling.md @@ -84,7 +84,7 @@ To change default **"Internal server error"** message to something else, use: GraphQL\Error\FormattedError::setInternalErrorMessage("Unexpected error"); ``` -#Debugging tools +# Debugging tools During development or debugging use `$result->toArray(true)` to add **debugMessage** key to each formatted error entry. If you also want to add exception trace - pass flags instead: @@ -116,7 +116,7 @@ This will make each error entry to look like this: ]; ``` -If you prefer first resolver exception to be re-thrown, use following flags: +If you prefer the first resolver exception to be re-thrown, use following flags: ```php toArray($debug); ``` +If you only want to re-throw Exceptions that are not marked as safe through the `ClientAware` interface, use +the flag `Debug::RETHROW_UNSAFE_EXCEPTIONS`. + # Custom Error Handling and Formatting It is possible to define custom **formatter** and **handler** for result errors. diff --git a/src/Error/Debug.php b/src/Error/Debug.php index 6e98087..a48b81f 100644 --- a/src/Error/Debug.php +++ b/src/Error/Debug.php @@ -12,4 +12,5 @@ class Debug const INCLUDE_DEBUG_MESSAGE = 1; const INCLUDE_TRACE = 2; const RETHROW_INTERNAL_EXCEPTIONS = 4; + const RETHROW_UNSAFE_EXCEPTIONS = 8; } diff --git a/src/Error/FormattedError.php b/src/Error/FormattedError.php index 25d364d..8ca886a 100644 --- a/src/Error/FormattedError.php +++ b/src/Error/FormattedError.php @@ -245,9 +245,15 @@ class FormattedError } } - $isInternal = ! $e instanceof ClientAware || ! $e->isClientSafe(); + $isUnsafe = ! $e instanceof ClientAware || ! $e->isClientSafe(); - if (($debug & Debug::INCLUDE_DEBUG_MESSAGE) && $isInternal) { + if (($debug & Debug::RETHROW_UNSAFE_EXCEPTIONS) && $isUnsafe) { + if ($e->getPrevious()) { + throw $e->getPrevious(); + } + } + + if (($debug & Debug::INCLUDE_DEBUG_MESSAGE) && $isUnsafe) { // Displaying debugMessage as a first entry: $formattedError = ['debugMessage' => $e->getMessage()] + $formattedError; } diff --git a/tests/Server/QueryExecutionTest.php b/tests/Server/QueryExecutionTest.php index 2cde99f..d816bce 100644 --- a/tests/Server/QueryExecutionTest.php +++ b/tests/Server/QueryExecutionTest.php @@ -81,21 +81,21 @@ class QueryExecutionTest extends ServerTestCase $query = ' { - fieldWithException + fieldWithSafeException f1 } '; $expected = [ - 'data' => [ - 'fieldWithException' => null, + 'data' => [ + 'fieldWithSafeException' => null, 'f1' => 'f1', ], 'errors' => [ [ 'message' => 'This is the exception we want', - 'path' => ['fieldWithException'], - 'trace' => [], + 'path' => ['fieldWithSafeException'], + 'trace' => [], ], ], ]; @@ -104,6 +104,18 @@ class QueryExecutionTest extends ServerTestCase $this->assertArraySubset($expected, $result); } + public function testRethrowUnsafeExceptions() : void + { + $this->config->setDebug(Debug::RETHROW_UNSAFE_EXCEPTIONS); + $this->expectException(Unsafe::class); + + $this->executeQuery(' + { + fieldWithUnsafeException + } + ')->toArray(); + } + public function testPassesRootValueAndContext() : void { $rootValue = 'myRootValue'; @@ -268,7 +280,7 @@ class QueryExecutionTest extends ServerTestCase { $batch = [ ['query' => '{invalid}'], - ['query' => '{f1,fieldWithException}'], + ['query' => '{f1,fieldWithSafeException}'], ]; $result = $this->executeBatchedQuery($batch); @@ -453,7 +465,7 @@ class QueryExecutionTest extends ServerTestCase $batch = [ ['query' => '{invalid}'], - ['query' => '{f1,fieldWithException}'], + ['query' => '{f1,fieldWithSafeException}'], [ 'query' => ' query ($a: String!, $b: String!) { @@ -472,9 +484,9 @@ class QueryExecutionTest extends ServerTestCase 'errors' => [['message' => 'Cannot query field "invalid" on type "Query".']], ], [ - 'data' => [ - 'f1' => 'f1', - 'fieldWithException' => null, + 'data' => [ + 'f1' => 'f1', + 'fieldWithSafeException' => null, ], 'errors' => [ ['message' => 'This is the exception we want'], @@ -618,7 +630,7 @@ class QueryExecutionTest extends ServerTestCase return ['test' => 'formatted']; }); - $result = $this->executeQuery('{fieldWithException}'); + $result = $this->executeQuery('{fieldWithSafeException}'); $this->assertFalse($called); $formatted = $result->toArray(); $expected = [ @@ -658,7 +670,7 @@ class QueryExecutionTest extends ServerTestCase ]; }); - $result = $this->executeQuery('{fieldWithException,test: fieldWithException}'); + $result = $this->executeQuery('{fieldWithSafeException,test: fieldWithSafeException}'); $this->assertFalse($called); $formatted = $result->toArray(); diff --git a/tests/Server/ServerTestCase.php b/tests/Server/ServerTestCase.php index 1e67d87..4ad9698 100644 --- a/tests/Server/ServerTestCase.php +++ b/tests/Server/ServerTestCase.php @@ -41,12 +41,18 @@ abstract class ServerTestCase extends TestCase return $info->fieldName; }, ], - 'fieldWithException' => [ - 'type' => Type::string(), - 'resolve' => function ($root, $args, $context, $info) { + 'fieldWithSafeException' => [ + 'type' => Type::string(), + 'resolve' => function () { throw new UserError('This is the exception we want'); }, ], + 'fieldWithUnsafeException' => [ + 'type' => Type::string(), + 'resolve' => function () { + throw new Unsafe('This exception should not be shown to the user'); + }, + ], 'testContextAndRootValue' => [ 'type' => Type::string(), 'resolve' => function ($root, $args, $context, $info) { diff --git a/tests/Server/Unsafe.php b/tests/Server/Unsafe.php new file mode 100644 index 0000000..2eb3742 --- /dev/null +++ b/tests/Server/Unsafe.php @@ -0,0 +1,34 @@ +