1
0
Fork 0
mirror of synced 2024-06-28 19:20:25 +12:00
appwrite/app/controllers/api/graphql.php

140 lines
4.6 KiB
PHP
Raw Normal View History

2020-01-04 10:16:26 +13:00
<?php
2022-04-07 18:40:28 +12:00
use Appwrite\GraphQL\Builder;
use Appwrite\Utopia\Response;
2021-03-17 02:34:11 +13:00
use GraphQL\Error\DebugFlag;
2022-04-06 01:48:51 +12:00
use GraphQL\GraphQL;
use GraphQL\Type;
2022-04-20 22:30:48 +12:00
use GraphQL\Validator\Rules\DisableIntrospection;
2022-04-07 18:40:28 +12:00
use GraphQL\Validator\Rules\QueryComplexity;
use GraphQL\Validator\Rules\QueryDepth;
use Swoole\Coroutine\WaitGroup;
2020-06-29 05:31:21 +12:00
use Utopia\App;
2022-04-07 18:40:28 +12:00
use Utopia\Validator\JSON;
use Utopia\Validator\Text;
2020-01-04 10:16:26 +13:00
2022-04-11 23:08:57 +12:00
App::get('/v1/graphql')
->desc('GraphQL Endpoint')
2022-04-07 18:39:33 +12:00
->groups(['api', 'grapgql'])
2021-03-05 07:40:52 +13:00
->label('scope', 'graphql')
2022-04-11 23:08:57 +12:00
->label('sdk.auth', [APP_AUTH_TYPE_KEY, APP_AUTH_TYPE_SESSION, APP_AUTH_TYPE_JWT])
2022-04-07 18:39:33 +12:00
->label('sdk.namespace', 'graphql')
2022-04-11 23:08:57 +12:00
->label('sdk.method', 'query')
->label('sdk.description', '/docs/references/graphql/query.md')
2022-04-07 18:39:33 +12:00
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_ANY)
->label('abuse-limit', 60)
2022-04-07 18:39:42 +12:00
->label('abuse-time', 60)
->param('query', '', new Text(1024), 'The query to execute. Max 1024 chars.')
2022-04-08 17:55:17 +12:00
->param('operationName', null, new Text(256), 'Name of the operation to execute', true)
2022-04-07 18:39:42 +12:00
->param('variables', [], new JSON(), 'Variables to use in the operation', true)
->inject('request')
->inject('response')
2021-03-11 04:22:19 +13:00
->inject('utopia')
->inject('register')
2022-04-06 01:48:51 +12:00
->inject('dbForProject')
->inject('promiseAdapter')
->inject('gqlSchema')
2022-04-11 23:08:57 +12:00
->action(Closure::fromCallable('graphqlRequest'));
App::post('/v1/graphql')
->desc('GraphQL Endpoint')
->groups(['api', 'grapgql'])
->label('scope', 'graphql')
->label('sdk.auth', [APP_AUTH_TYPE_KEY, APP_AUTH_TYPE_SESSION, APP_AUTH_TYPE_JWT])
->label('sdk.namespace', 'graphql')
->label('sdk.method', 'mutate')
->label('sdk.description', '/docs/references/graphql/mutate.md')
->label('sdk.response.code', Response::STATUS_CODE_OK)
->label('sdk.response.type', Response::CONTENT_TYPE_JSON)
->label('sdk.response.model', Response::MODEL_ANY)
->label('abuse-limit', 60)
->label('abuse-time', 60)
->param('query', '', new Text(1024), 'The query to execute. Max 1024 chars.', true)
->param('operationName', null, new Text(256), 'Name of the operation to execute', true)
->param('variables', [], new JSON(), 'Variables to use in the operation', true)
->inject('request')
->inject('response')
->inject('utopia')
->inject('dbForProject')
->inject('promiseAdapter')
->inject('gqlSchema')
2022-04-11 23:08:57 +12:00
->action(Closure::fromCallable('graphqlRequest'));
2021-11-25 21:04:39 +13:00
2022-04-11 23:08:57 +12:00
/**
* @throws Exception
*/
function graphqlRequest(
$query,
$operationName,
$variables,
$request,
$response,
$utopia,
$dbForProject,
$promiseAdapter,
$gqlSchema
2022-04-11 23:08:57 +12:00
)
{
/** @var Appwrite\Utopia\Request $request */
/** @var Appwrite\Utopia\Response $response */
/** @var Type\Schema $schema */
/** @var Utopia\App $utopia */
/** @var Utopia\Registry\Registry $register */
/** @var \Utopia\Database\Database $dbForProject */
2022-04-11 23:08:57 +12:00
if ($request->getHeader('content-type') === 'application/graphql') {
2022-04-20 22:30:48 +12:00
// TODO: Add getRawContent() method to Request
2022-04-11 23:08:57 +12:00
$query = $request->getSwoole()->rawContent();
}
if (empty($query)) {
throw new Exception('Query is empty', Response::STATUS_CODE_BAD_REQUEST);
}
2021-03-11 02:51:03 +13:00
2022-04-11 23:08:57 +12:00
$debugFlags = App::isDevelopment()
? DebugFlag::INCLUDE_DEBUG_MESSAGE | DebugFlag::INCLUDE_TRACE | DebugFlag::RETHROW_INTERNAL_EXCEPTIONS
: DebugFlag::NONE;
2022-04-06 01:48:51 +12:00
2022-04-11 23:08:57 +12:00
$validations = array_merge(
GraphQL::getStandardValidationRules(),
[
new QueryComplexity(App::getEnv('_APP_GRAPHQL_MAX_QUERY_COMPLEXITY', 200)),
new QueryDepth(App::getEnv('_APP_GRAPHQL_MAX_QUERY_DEPTH', 3)),
]
);
2022-04-07 18:40:28 +12:00
2022-04-20 22:30:48 +12:00
if (App::isProduction()) {
$validations[] = new DisableIntrospection();
}
2022-04-11 23:08:57 +12:00
$promise = GraphQL::promiseToExecute(
$promiseAdapter,
$gqlSchema,
2022-04-11 23:08:57 +12:00
$query,
variableValues: $variables,
operationName: $operationName,
validationRules: $validations
);
2022-04-07 18:40:28 +12:00
2022-04-11 23:08:57 +12:00
// Blocking wait while queries resolve asynchronously
$wg = new WaitGroup();
$wg->add();
2022-04-20 22:30:48 +12:00
$promise->then(function ($result) use ($response, $debugFlags, $wg) {
$result = $result->toArray($debugFlags);
\var_dump("Result:" . $result);
if (isset($result['errors'])) {
$response->json(['data' => [], ...$result]);
2022-04-11 23:08:57 +12:00
$wg->done();
2022-04-20 22:30:48 +12:00
return;
2022-04-11 23:08:57 +12:00
}
2022-04-20 22:30:48 +12:00
$response->json(['data' => $result]);
$wg->done();
},
function ($error) use ($response, $wg) {
$response->text(\json_encode(['errors' => [\json_encode($error)]]));
$wg->done();
});
2022-04-11 23:08:57 +12:00
$wg->wait();
}