is_locked() || $user->can(Permissions::EDIT_IMAGE_LOCK)) { send_event(new ImageInfoSetEvent($image, 0, [ 'tags' => $metadata->tags, 'source' => $metadata->source, ])); } return Image::by_id_ex($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(); 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} */ 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[] = ["image_ids" => self::handle_upload($n, $common_tags, $common_source)]; } catch(\Exception $e) { $results[] = ["error" => $e->getMessage()]; } } return ["results" => $results]; } /** * @return int[] */ private static function handle_upload(int $n, string $common_tags, string $common_source): 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: $ec"); } } $tags = trim($common_tags . " " . $_POST["tags$n"]); $source = $common_source; if (!empty($_POST["source$n"])) { $source = $_POST["source$n"]; } $event = $database->with_savepoint(function () use ($tmpname, $filename, $n, $tags, $source) { return send_event(new DataUploadEvent($tmpname, $filename, $n, [ 'tags' => Tag::explode($tags), 'source' => $source, ])); }); 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)->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; }); } }