📁 File Manager Pro
v10.0.3 | PHP: 8.2.31
Server: LiteSpeed
2026-07-03 12:40:55
📂
/ (Root)
/
home
/
orkouolp
/
web
/
orkofarms.com
/
wp-content
/
plugins
/
woocommerce
/
src
/
Api
/
Utils
📍 /home/orkouolp/web/orkofarms.com/wp-content/plugins/woocommerce/src/Api/Utils
🔄 Refresh
✏️
Editing: SchemaHandle.php
Writable
<?php declare(strict_types=1); namespace Automattic\WooCommerce\Api\Utils; use Automattic\WooCommerce\Vendor\GraphQL\Type\Definition\EnumType; use Automattic\WooCommerce\Vendor\GraphQL\Type\Definition\HasFieldsType; use Automattic\WooCommerce\Vendor\GraphQL\Type\Definition\InputObjectType; use Automattic\WooCommerce\Vendor\GraphQL\Type\Definition\Type; /** * Opaque handle to a dual-API GraphQL schema, exposing the runtime inspection * operations the dual-API surface supports. * * The handle wraps the live engine schema but does not expose it. Clients * therefore depend only on the methods this class declares — never on the * underlying engine type — which keeps a future engine swap as a non-public * API change. * * Construction is reserved for the dual-API infrastructure. Obtain a handle * via your dual-API `GraphQLController`'s `get_schema()` method: * * $schema = wc_get_container() * ->get( \Automattic\WooCommerce\Internal\Api\Autogenerated\GraphQLController::class ) * ->get_schema(); * * WooCommerce plugins implementing their own dual API reach a handle through * their own concrete autogenerated controller the same way. * * The current public surface is the discovery channel: {@see self::get_all_metadata()} * returns every row in the schema that carries either `#[Metadata]`-derived * entries or authorization attributes, and {@see self::find_metadata()} applies * filter-narrows semantics (`name`, `type`, `field`, `attribute`) over the same * set. Authorization descriptors are exposed as a parallel `authorization` * slice on each row, alongside the existing `entries`. * * @phpstan-type MetadataRow array{ * type: string, * field: ?string, * argument: ?string, * enumValue: ?string, * entries: array<string, bool|int|float|string|null>, * authorization: list<array{attribute: string, args: list<mixed>}> * } */ final class SchemaHandle { /** * The wrapped engine schema. Typed as `object` (rather than the engine's * `Schema` class) so the class signature carries no engine-specific * symbol; the inspection methods cast to engine APIs internally. * * @var object */ private object $engine_schema; /** * Wrap an engine schema in a handle. * * @internal Reserved for dual-API infrastructure (the controller's `get_schema()` accessor and similarly placed code). Plugins obtain a handle through their own controller, not by instantiating directly. * * @param object $engine_schema Engine-specific schema instance the handle wraps. */ public function __construct( object $engine_schema ) { $this->engine_schema = $engine_schema; } /** * Return every metadata row in the schema (introspection types excluded). * * Each row describes one *target* (a type, a field, an argument, or an * enum value) and carries the name=>value entries declared on it. The * same row shape is used for every target kind; the three nullable * position fields (`field`, `argument`, `enumValue`) discriminate. * * @return list<array{type: string, field: ?string, argument: ?string, enumValue: ?string, entries: array<string, bool|int|float|string|null>, authorization: list<array{attribute: string, args: list<mixed>}>}> */ public function get_all_metadata(): array { $rows = array(); $schema = $this->engine_schema; foreach ( $schema->getTypeMap() as $type_name => $type ) { if ( self::is_introspection_name( $type_name ) ) { continue; } $type_metadata = self::read_type_metadata( $type ); $type_authorization = self::read_type_authorization( $type ); if ( ! empty( $type_metadata ) || ! empty( $type_authorization ) ) { $rows[] = self::make_row( $type_name, null, null, null, $type_metadata, $type_authorization ); } if ( $type instanceof HasFieldsType ) { foreach ( $type->getFields() as $field_name => $field ) { $field_metadata = self::read_element_metadata( $field ); $field_authorization = self::read_element_authorization( $field ); if ( ! empty( $field_metadata ) || ! empty( $field_authorization ) ) { $rows[] = self::make_row( $type_name, $field_name, null, null, $field_metadata, $field_authorization ); } foreach ( $field->args as $arg ) { $arg_metadata = self::read_element_metadata( $arg ); if ( ! empty( $arg_metadata ) ) { $rows[] = self::make_row( $type_name, $field_name, $arg->name, null, $arg_metadata, array() ); } } } continue; } if ( $type instanceof InputObjectType ) { foreach ( $type->getFields() as $field_name => $field ) { $field_metadata = self::read_element_metadata( $field ); $field_authorization = self::read_element_authorization( $field ); if ( ! empty( $field_metadata ) || ! empty( $field_authorization ) ) { $rows[] = self::make_row( $type_name, $field_name, null, null, $field_metadata, $field_authorization ); } } continue; } if ( $type instanceof EnumType ) { foreach ( $type->getValues() as $value ) { $value_metadata = self::read_element_metadata( $value ); if ( ! empty( $value_metadata ) ) { $rows[] = self::make_row( $type_name, null, null, $value->name, $value_metadata, array() ); } } } } return $rows; } /** * Filter-narrows view over {@see self::get_all_metadata()}. * * Each filter argument independently restricts the result set; supplying * multiple composes as AND. When `$name` is supplied, the surviving rows * have their `entries` trimmed to the single matching entry; so a caller * asking "which elements are marked X" gets focused rows back, not the * full multi-entry shape. * * @param ?string $name Optional metadata name to match. When set, only rows containing this entry survive and their `entries` are trimmed to it. * @param ?string $type Optional GraphQL type name to match. * @param ?string $field Optional GraphQL field name to match. * @param ?string $attribute Optional authorization-attribute short name to match. When set, only rows carrying this attribute survive and their `authorization` is trimmed to the matching descriptors. * * @return list<array{type: string, field: ?string, argument: ?string, enumValue: ?string, entries: array<string, bool|int|float|string|null>, authorization: list<array{attribute: string, args: list<mixed>}>}> */ public function find_metadata( ?string $name = null, ?string $type = null, ?string $field = null, ?string $attribute = null ): array { $rows = $this->get_all_metadata(); $result = array(); foreach ( $rows as $row ) { if ( null !== $type && $row['type'] !== $type ) { continue; } if ( null !== $field && $row['field'] !== $field ) { continue; } if ( null !== $name ) { if ( ! array_key_exists( $name, $row['entries'] ) ) { continue; } $row['entries'] = array( $name => $row['entries'][ $name ] ); } if ( null !== $attribute ) { $matching = array_values( array_filter( $row['authorization'], static fn( array $descriptor ): bool => ( $descriptor['attribute'] ?? null ) === $attribute, ) ); if ( empty( $matching ) ) { continue; } $row['authorization'] = $matching; } $result[] = $row; } return $result; } /** * Read type-level metadata from a wrapped engine type. * * The wrapper subclasses in `Internal/Api/Schema/` expose `get_metadata()`; * non-wrapper types (e.g. the built-in scalars, the introspection types we * already filtered out) don't carry metadata and contribute an empty array. * * @param Type $type The GraphQL type to inspect. * @return array<string, bool|int|float|string|null> */ private static function read_type_metadata( Type $type ): array { if ( method_exists( $type, 'get_metadata' ) ) { $metadata = $type->get_metadata(); return is_array( $metadata ) ? $metadata : array(); } return array(); } /** * Read field-/arg-/enum-value-level metadata from the original config array. * * FieldDefinition, Argument, InputObjectField and EnumValueDefinition all * preserve their construction config in a public `$config` property, so * the `metadata` key emitted by ApiBuilder is reachable here without any * wrapper-side plumbing. * * @param object $element FieldDefinition | Argument | InputObjectField | EnumValueDefinition. * @return array<string, bool|int|float|string|null> */ private static function read_element_metadata( object $element ): array { if ( ! property_exists( $element, 'config' ) ) { return array(); } $metadata = $element->config['metadata'] ?? array(); return is_array( $metadata ) ? $metadata : array(); } /** * Read authorization descriptors attached to a wrapped engine type. * * The wrapper subclasses preserve the original config in `$type->config`; * authorization descriptors emitted by ApiBuilder live under the * `authorization` key as a list of `{attribute, args}` records. * * @param Type $type The GraphQL type to inspect. * @return list<array{attribute: string, args: list<mixed>}> */ private static function read_type_authorization( Type $type ): array { if ( ! property_exists( $type, 'config' ) ) { return array(); } $authorization = $type->config['authorization'] ?? array(); return is_array( $authorization ) ? $authorization : array(); } /** * Read authorization descriptors from a field-/arg-/enum-value-level config. * * Mirrors {@see self::read_element_metadata()} but pulls the * `authorization` key. Returns an empty list when the element carries * no authorization descriptors. * * @param object $element FieldDefinition | Argument | InputObjectField | EnumValueDefinition. * @return list<array{attribute: string, args: list<mixed>}> */ private static function read_element_authorization( object $element ): array { if ( ! property_exists( $element, 'config' ) ) { return array(); } $authorization = $element->config['authorization'] ?? array(); return is_array( $authorization ) ? $authorization : array(); } /** * Build a metadata row in the standard shape. * * @param string $type GraphQL type name. * @param ?string $field Field name when the row describes a field; null otherwise. * @param ?string $argument Argument name when the row describes a field argument; null otherwise. * @param ?string $enum_value Enum value name when the row describes an enum value; null otherwise. * @param array<string, bool|int|float|string|null> $entries Name=>value entries to attach to the row. * @param list<array{attribute: string, args: list<mixed>}> $authorization Authorization descriptors attached to the row, or an empty list. * @return array{type: string, field: ?string, argument: ?string, enumValue: ?string, entries: array<string, bool|int|float|string|null>, authorization: list<array{attribute: string, args: list<mixed>}>} */ private static function make_row( string $type, ?string $field, ?string $argument, ?string $enum_value, array $entries, array $authorization ): array { return array( 'type' => $type, 'field' => $field, 'argument' => $argument, 'enumValue' => $enum_value, 'entries' => $entries, 'authorization' => $authorization, ); } /** * Whether a type name belongs to GraphQL's introspection system (and so should be skipped). * * @param string $name Type name. */ private static function is_introspection_name( string $name ): bool { return str_starts_with( $name, '__' ); } }
💾 Save Changes
❌ Cancel