WooCommerce Code Reference

SchemaHandle.php

Source code

<?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, '__' );
	}
}