$this->parseRawRequest('application/graphql', $query), 'psr' => $this->parsePsrRequest('application/graphql', $query) ]; foreach ($parsed as $source => $parsedBody) { $this->assertValidOperationParams($parsedBody, $query, null, null, null, $source); $this->assertFalse($parsedBody->isReadOnly(), $source); } } public function testParsesUrlencodedRequest() { $query = '{my query}'; $variables = ['test' => 1, 'test2' => 2]; $operation = 'op'; $post = [ 'query' => $query, 'variables' => $variables, 'operation' => $operation ]; $parsed = [ 'raw' => $this->parseRawFormUrlencodedRequest($post), 'psr' => $this->parsePsrFormUrlEncodedRequest($post) ]; foreach ($parsed as $method => $parsedBody) { $this->assertValidOperationParams($parsedBody, $query, null, $variables, $operation, $method); $this->assertFalse($parsedBody->isReadOnly(), $method); } } public function testParsesGetRequest() { $query = '{my query}'; $variables = ['test' => 1, 'test2' => 2]; $operation = 'op'; $get = [ 'query' => $query, 'variables' => $variables, 'operation' => $operation ]; $parsed = [ 'raw' => $this->parseRawGetRequest($get), 'psr' => $this->parsePsrGetRequest($get) ]; foreach ($parsed as $method => $parsedBody) { $this->assertValidOperationParams($parsedBody, $query, null, $variables, $operation, $method); $this->assertTrue($parsedBody->isReadonly(), $method); } } public function testParsesJSONRequest() { $query = '{my query}'; $variables = ['test' => 1, 'test2' => 2]; $operation = 'op'; $body = [ 'query' => $query, 'variables' => $variables, 'operation' => $operation ]; $parsed = [ 'raw' => $this->parseRawRequest('application/json', json_encode($body)), 'psr' => $this->parsePsrRequest('application/json', json_encode($body)) ]; foreach ($parsed as $method => $parsedBody) { $this->assertValidOperationParams($parsedBody, $query, null, $variables, $operation, $method); $this->assertFalse($parsedBody->isReadOnly(), $method); } } public function testParsesVariablesAsJSON() { $query = '{my query}'; $variables = ['test' => 1, 'test2' => 2]; $operation = 'op'; $body = [ 'query' => $query, 'variables' => json_encode($variables), 'operation' => $operation ]; $parsed = [ 'raw' => $this->parseRawRequest('application/json', json_encode($body)), 'psr' => $this->parsePsrRequest('application/json', json_encode($body)) ]; foreach ($parsed as $method => $parsedBody) { $this->assertValidOperationParams($parsedBody, $query, null, $variables, $operation, $method); $this->assertFalse($parsedBody->isReadOnly(), $method); } } public function testIgnoresInvalidVariablesJson() { $query = '{my query}'; $variables = '"some invalid json'; $operation = 'op'; $body = [ 'query' => $query, 'variables' => $variables, 'operation' => $operation ]; $parsed = [ 'raw' => $this->parseRawRequest('application/json', json_encode($body)), 'psr' => $this->parsePsrRequest('application/json', json_encode($body)), ]; foreach ($parsed as $method => $parsedBody) { $this->assertValidOperationParams($parsedBody, $query, null, $variables, $operation, $method); $this->assertFalse($parsedBody->isReadOnly(), $method); } } public function testParsesBatchJSONRequest() { $body = [ [ 'query' => '{my query}', 'variables' => ['test' => 1, 'test2' => 2], 'operation' => 'op' ], [ 'queryId' => 'my-query-id', 'variables' => ['test' => 1, 'test2' => 2], 'operation' => 'op2' ], ]; $parsed = [ 'raw' => $this->parseRawRequest('application/json', json_encode($body)), 'psr' => $this->parsePsrRequest('application/json', json_encode($body)) ]; foreach ($parsed as $method => $parsedBody) { $this->assertInternalType('array', $parsedBody, $method); $this->assertCount(2, $parsedBody, $method); $this->assertValidOperationParams($parsedBody[0], $body[0]['query'], null, $body[0]['variables'], $body[0]['operation'], $method); $this->assertValidOperationParams($parsedBody[1], null, $body[1]['queryId'], $body[1]['variables'], $body[1]['operation'], $method); } } public function testFailsParsingInvalidRawJsonRequest() { $body = 'not really{} a json'; try { $this->parseRawRequest('application/json', $body); $this->fail('Expected exception not thrown'); } catch (RequestError $e) { $this->assertEquals('Could not parse JSON: Syntax error', $e->getMessage()); } try { $this->parsePsrRequest('application/json', $body); $this->fail('Expected exception not thrown'); } catch (InvariantViolation $e) { // Expecting parsing exception to be thrown somewhere else: $this->assertEquals( 'PSR request is expected to provide parsed body for "application/json" requests but got null', $e->getMessage() ); } } // There is no equivalent for psr request, because it should throw public function testFailsParsingNonArrayOrObjectJsonRequest() { $body = '"str"'; try { $this->parseRawRequest('application/json', $body); $this->fail('Expected exception not thrown'); } catch (RequestError $e) { $this->assertEquals('GraphQL Server expects JSON object or array, but got "str"', $e->getMessage()); } try { $this->parsePsrRequest('application/json', $body); $this->fail('Expected exception not thrown'); } catch (RequestError $e) { $this->assertEquals('GraphQL Server expects JSON object or array, but got "str"', $e->getMessage()); } } public function testFailsParsingInvalidContentType() { $contentType = 'not-supported-content-type'; $body = 'test'; try { $this->parseRawRequest($contentType, $body); $this->fail('Expected exception not thrown'); } catch (RequestError $e) { $this->assertEquals('Unexpected content type: "not-supported-content-type"', $e->getMessage()); } try { $this->parsePsrRequest($contentType, $body); $this->fail('Expected exception not thrown'); } catch (RequestError $e) { $this->assertEquals('Unexpected content type: "not-supported-content-type"', $e->getMessage()); } } public function testFailsWithMissingContentType() { try { $this->parseRawRequest(null, 'test'); $this->fail('Expected exception not thrown'); } catch (RequestError $e) { $this->assertEquals('Missing "Content-Type" header', $e->getMessage()); } try { $this->parsePsrRequest(null, 'test'); $this->fail('Expected exception not thrown'); } catch (RequestError $e) { $this->assertEquals('Missing "Content-Type" header', $e->getMessage()); } } public function testFailsOnMethodsOtherThanPostOrGet() { try { $this->parseRawRequest('application/json', json_encode([]), "PUT"); $this->fail('Expected exception not thrown'); } catch (RequestError $e) { $this->assertEquals('HTTP Method "PUT" is not supported', $e->getMessage()); } try { $this->parsePsrRequest('application/json', json_encode([]), "PUT"); $this->fail('Expected exception not thrown'); } catch (RequestError $e) { $this->assertEquals('HTTP Method "PUT" is not supported', $e->getMessage()); } } /** * @param string $contentType * @param string $content * @param $method * * @return OperationParams|OperationParams[] */ private function parseRawRequest($contentType, $content, $method = 'POST') { $_SERVER['CONTENT_TYPE'] = $contentType; $_SERVER['REQUEST_METHOD'] = $method; $helper = new Helper(); return $helper->parseHttpRequest(function() use ($content) { return $content; }); } /** * @param string $contentType * @param string $content * @param $method * * @return OperationParams|OperationParams[] */ private function parsePsrRequest($contentType, $content, $method = 'POST') { $psrRequestBody = new PsrStreamStub(); $psrRequestBody->content = $content; $psrRequest = new PsrRequestStub(); $psrRequest->headers['content-type'] = [$contentType]; $psrRequest->method = $method; $psrRequest->body = $psrRequestBody; if ($contentType === 'application/json') { $parsedBody = json_decode($content, true); $parsedBody = $parsedBody === false ? null : $parsedBody; } else { $parsedBody = null; } $psrRequest->parsedBody = $parsedBody; $helper = new Helper(); return $helper->parsePsrRequest($psrRequest); } /** * @param array $postValue * @return OperationParams|OperationParams[] */ private function parseRawFormUrlencodedRequest($postValue) { $_SERVER['CONTENT_TYPE'] = 'application/x-www-form-urlencoded'; $_SERVER['REQUEST_METHOD'] = 'POST'; $_POST = $postValue; $helper = new Helper(); return $helper->parseHttpRequest(function() { throw new InvariantViolation("Shouldn't read from php://input for urlencoded request"); }); } /** * @param $postValue * @return array|Helper */ private function parsePsrFormUrlEncodedRequest($postValue) { $psrRequest = new PsrRequestStub(); $psrRequest->headers['content-type'] = ['application/x-www-form-urlencoded']; $psrRequest->method = 'POST'; $psrRequest->parsedBody = $postValue; $helper = new Helper(); return $helper->parsePsrRequest($psrRequest); } /** * @param $getValue * @return OperationParams */ private function parseRawGetRequest($getValue) { $_SERVER['REQUEST_METHOD'] = 'GET'; $_GET = $getValue; $helper = new Helper(); return $helper->parseHttpRequest(function() { throw new InvariantViolation("Shouldn't read from php://input for urlencoded request"); }); } /** * @param $getValue * @return array|Helper */ private function parsePsrGetRequest($getValue) { $psrRequest = new PsrRequestStub(); $psrRequest->method = 'GET'; $psrRequest->queryParams = $getValue; $helper = new Helper(); return $helper->parsePsrRequest($psrRequest); } /** * @param OperationParams $params * @param string $query * @param string $queryId * @param array $variables * @param string $operation */ private function assertValidOperationParams($params, $query, $queryId = null, $variables = null, $operation = null, $message = '') { $this->assertInstanceOf(OperationParams::class, $params, $message); $this->assertSame($query, $params->query, $message); $this->assertSame($queryId, $params->queryId, $message); $this->assertSame($variables, $params->variables, $message); $this->assertSame($operation, $params->operation, $message); } }