tags; $_POST['tag_edit__source'] = $metadata->source; $image = Image::by_id($post_id); if (!$image->is_locked() || $user->can(Permissions::EDIT_IMAGE_LOCK)) { send_event(new ImageInfoSetEvent($image)); } return Image::by_id($post_id); } } 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(), ]); $t2 = ftime(); $resp = $server->executeRequest(); 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(\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(\json_encode(self::handle_uploads())); } } private static function handle_uploads(): array { global $user; if (!$user->can(Permissions::CREATE_IMAGE)) { return ["error" => "User cannot create posts"]; } $common_tags = $_POST['common_tags']; $common_source = $_POST['common_source']; $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[] = self::handle_upload($n, $common_tags, $common_source); } catch(\Exception $e) { $results[] = ["error" => $e->getMessage()]; } } return ["results" => $results]; } private static function handle_upload(int $n, string $common_tags, string $common_source): array { if (!empty($_POST["url$n"])) { throw new \Exception("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: return ["error" => "File larger than PHP can handle"]; default: return ["error" => "Mystery error: $ec"]; } } $tags = trim($common_tags . " " . $_POST["tags$n"]); $source = $common_source; if (!empty($_POST["source$n"])) { $source = $_POST["source$n"]; } $event = send_event(new DataUploadEvent($tmpname, [ 'filename' => $filename, 'tags' => Tag::explode($tags), 'source' => $source, ])); return ["image_ids" => 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)->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 \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; }); } }