This repository has been archived on 2024-09-05. You can view files and clone it, but cannot push or open issues or pull requests.
shimmie2/ext/graphql/main.php
Shish 91e1cf3abe [graphql] take post metadata as arbitrary key-value pairs
I don't want "tags" and "source" to be special things where we explicitly handle those and fail to handle anything else. Making all metadata fields {string:string} pairs seems nicer, and matches with HTTP POST syntax
2024-02-25 20:06:18 +00:00

241 lines
8.2 KiB
PHP

<?php
declare(strict_types=1);
namespace Shimmie2;
use Symfony\Component\Console\Command\Command;
use Symfony\Component\Console\Input\{InputInterface,InputArgument};
use Symfony\Component\Console\Output\OutputInterface;
use GraphQL\GraphQL as GQL;
use GraphQL\Server\StandardServer;
use GraphQL\Error\DebugFlag;
use GraphQL\Type\Schema;
use GraphQL\Utils\SchemaPrinter;
#[\GQLA\InputObjectType]
class MetadataInput
{
public function __construct(
#[\GQLA\Field]
public string $key,
#[\GQLA\Field]
public string $value,
) {
}
/**
* @param array<MetadataInput> $metadata
*/
#[\GQLA\Mutation(args: ["post_id" => "Int!", "metadata" => "[MetadataInput!]!"])]
public static function update_post_metadata(int $post_id, array $metadata): Image
{
$image = Image::by_id_ex($post_id);
$pairs = [];
foreach ($metadata as $m) {
$pairs[$m->key] = $m->value;
}
send_event(new ImageInfoSetEvent($image, 0, $pairs));
return Image::by_id_ex($post_id);
}
}
function shmFieldResolver(
mixed $objectValue,
mixed $args,
mixed $context,
\GraphQL\Type\Definition\ResolveInfo $info
): mixed {
$fieldName = $info->fieldName;
$property = null;
if (is_array($objectValue)) {
if (isset($objectValue[$fieldName])) {
$property = $objectValue[$fieldName];
}
} elseif ($objectValue instanceof \ArrayAccess) {
if (isset($objectValue->{$fieldName})) {
$property = $objectValue->{$fieldName};
} elseif (isset($objectValue[$fieldName])) {
$property = $objectValue[$fieldName];
}
} elseif (is_object($objectValue)) {
if (isset($objectValue->{$fieldName})) {
$property = $objectValue->{$fieldName};
}
}
return $property instanceof \Closure
? $property($objectValue, $args, $context, $info)
: $property;
}
class GraphQL extends Extension
{
public static function get_schema(): Schema
{
global $_tracer;
$_tracer->begin("Create Schema");
$schema = new \GQLA\Schema();
$_tracer->end(null);
return $schema;
}
private function cors(): void
{
global $config;
$pat = $config->get_string("graphql_cors_pattern");
if ($pat && isset($_SERVER['HTTP_ORIGIN'])) {
if (preg_match("#$pat#", $_SERVER['HTTP_ORIGIN'])) {
header("Access-Control-Allow-Origin: {$_SERVER['HTTP_ORIGIN']}");
header('Access-Control-Allow-Credentials: true');
header('Access-Control-Max-Age: 86400');
}
}
if ($_SERVER['REQUEST_METHOD'] == 'OPTIONS') {
if (isset($_SERVER['HTTP_ACCESS_CONTROL_REQUEST_METHOD'])) {
header("Access-Control-Allow-Methods: GET, POST, OPTIONS");
}
if (isset($_SERVER['HTTP_ACCESS_CONTROL_REQUEST_HEADERS'])) {
header("Access-Control-Allow-Headers: {$_SERVER['HTTP_ACCESS_CONTROL_REQUEST_HEADERS']}");
}
exit(0);
}
}
public function onInitExt(InitExtEvent $event): void
{
global $config;
$config->set_default_string('graphql_cors_pattern', "");
$config->set_default_bool('graphql_debug', false);
}
public function onPageRequest(PageRequestEvent $event): void
{
global $config, $page;
if ($event->page_matches("graphql")) {
$this->cors();
$t1 = ftime();
$server = new StandardServer([
'schema' => $this->get_schema(),
'fieldResolver' => "\Shimmie2\shmFieldResolver",
]);
$t2 = ftime();
$resp = $server->executeRequest();
assert(!is_array($resp));
assert(is_a($resp, \GraphQL\Executor\ExecutionResult::class));
if ($config->get_bool("graphql_debug")) {
$debug = DebugFlag::INCLUDE_DEBUG_MESSAGE | DebugFlag::RETHROW_INTERNAL_EXCEPTIONS;
$body = $resp->toArray($debug);
} else {
$body = $resp->toArray();
}
$t3 = ftime();
$body['stats'] = get_debug_info_arr();
$body['stats']['graphql_schema_time'] = round($t2 - $t1, 2);
$body['stats']['graphql_execute_time'] = round($t3 - $t2, 2);
// sleep(1);
$page->set_mode(PageMode::DATA);
$page->set_mime("application/json");
$page->set_data(\Safe\json_encode($body, JSON_UNESCAPED_UNICODE));
}
if ($event->page_matches("graphql_upload")) {
$this->cors();
$page->set_mode(PageMode::DATA);
$page->set_mime("application/json");
$page->set_data(\Safe\json_encode(self::handle_uploads()));
}
}
/**
* @return array{error?:string,results?:array<array{error?:string,image_ids?:int[]}>}
*/
private static function handle_uploads(): array
{
global $user;
if (!$user->can(Permissions::CREATE_IMAGE)) {
return ["error" => "User cannot create posts"];
}
$metadata = only_strings($_POST);
$results = [];
for ($n = 0; $n < 100; $n++) {
if (empty($_POST["url$n"]) && empty($_FILES["data$n"])) {
break;
}
if (isset($_FILES["data$n"]) && ($_FILES["data$n"]["size"] == 0 || $_FILES["data$n"]["error"] == UPLOAD_ERR_NO_FILE)) {
break;
}
try {
$results[] = ["image_ids" => self::handle_upload($n, $metadata)];
} catch(\Exception $e) {
$results[] = ["error" => $e->getMessage()];
}
}
return ["results" => $results];
}
/**
* @param array<string, string> $metadata
* @return int[]
*/
private static function handle_upload(int $n, array $metadata): array
{
global $database;
if (!empty($_POST["url$n"])) {
throw new UploadException("URLs not handled yet");
} else {
$ec = $_FILES["data$n"]["error"];
switch($ec) {
case UPLOAD_ERR_OK:
$tmpname = $_FILES["data$n"]["tmp_name"];
$filename = $_FILES["data$n"]["name"];
break;
case UPLOAD_ERR_INI_SIZE:
throw new UploadException("File larger than PHP can handle");
default:
throw new UploadException("Mystery error: ".var_export($ec, true));
}
}
$event = $database->with_savepoint(function () use ($tmpname, $filename, $n, $metadata) {
return send_event(new DataUploadEvent($tmpname, $filename, $n, $metadata));
});
return array_map(fn ($im) => $im->id, $event->images);
}
public function onCliGen(CliGenEvent $event): void
{
$event->app->register('graphql:query')
->addArgument('query', InputArgument::REQUIRED)
->setDescription('Run a GraphQL query')
->setCode(function (InputInterface $input, OutputInterface $output): int {
$query = $input->getArgument('query');
$t1 = ftime();
$schema = $this->get_schema();
$t2 = ftime();
$debug = DebugFlag::INCLUDE_DEBUG_MESSAGE | DebugFlag::RETHROW_INTERNAL_EXCEPTIONS;
$body = GQL::executeQuery($schema, $query, fieldResolver: "\Shimmie2\shmFieldResolver")->toArray($debug);
$t3 = ftime();
$body['stats'] = get_debug_info_arr();
$body['stats']['graphql_schema_time'] = round($t2 - $t1, 2);
$body['stats']['graphql_execute_time'] = round($t3 - $t2, 2);
echo \Safe\json_encode($body, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT);
return Command::SUCCESS;
});
$event->app->register('graphql:schema')
->setDescription('Print out the GraphQL schema')
->setCode(function (InputInterface $input, OutputInterface $output): int {
$schema = $this->get_schema();
echo(SchemaPrinter::doPrint($schema));
return Command::SUCCESS;
});
}
}