HEX
Server: Apache/2.4.65 (Debian)
System: Linux d8108ab83977 5.15.0-124-generic #134-Ubuntu SMP Fri Sep 27 20:20:17 UTC 2024 x86_64
User: www-data (33)
PHP: 8.1.33
Disabled: NONE
Upload Files
File: /var/www/html/wp-content/plugins/wp-graphql/src/AppContext.php
<?php

namespace WPGraphQL;

use GraphQL\Error\UserError;
use WPGraphQL\Data\Loader\CommentAuthorLoader;
use WPGraphQL\Data\Loader\CommentLoader;
use WPGraphQL\Data\Loader\EnqueuedScriptLoader;
use WPGraphQL\Data\Loader\EnqueuedStylesheetLoader;
use WPGraphQL\Data\Loader\PluginLoader;
use WPGraphQL\Data\Loader\PostObjectLoader;
use WPGraphQL\Data\Loader\PostTypeLoader;
use WPGraphQL\Data\Loader\TaxonomyLoader;
use WPGraphQL\Data\Loader\TermObjectLoader;
use WPGraphQL\Data\Loader\ThemeLoader;
use WPGraphQL\Data\Loader\UserLoader;
use WPGraphQL\Data\Loader\UserRoleLoader;
use WPGraphQL\Data\NodeResolver;

/**
 * Class AppContext
 * Creates an object that contains all of the context for the GraphQL query
 * This class gets instantiated and populated in the main WPGraphQL class.
 *
 * The context is passed to each resolver during execution.
 *
 * Resolvers have the ability to read and write to context to pass info to nested resolvers.
 *
 * @package WPGraphQL
 */
#[\AllowDynamicProperties]
class AppContext {
	/**
	 * The default loaders for the AppContext.
	 */
	private const DEFAULT_LOADERS = [
		'comment_author'      => CommentAuthorLoader::class,
		'comment'             => CommentLoader::class,
		'enqueued_script'     => EnqueuedScriptLoader::class,
		'enqueued_stylesheet' => EnqueuedStylesheetLoader::class,
		'plugin'              => PluginLoader::class,
		'nav_menu_item'       => PostObjectLoader::class,
		'post'                => PostObjectLoader::class,
		'post_type'           => PostTypeLoader::class,
		'taxonomy'            => TaxonomyLoader::class,
		'term'                => TermObjectLoader::class,
		'theme'               => ThemeLoader::class,
		'user'                => UserLoader::class,
		'user_role'           => UserRoleLoader::class,
	];

	/**
	 * Stores the class to use for the connection query.
	 *
	 * @var \WP_Query|null
	 */
	public $connection_query_class = null;

	/**
	 * Stores the url string for the current site
	 *
	 * @var string $root_url
	 */
	public $root_url;

	/**
	 * Stores the WP_User object of the current user
	 *
	 * @var \WP_User $viewer
	 */
	public $viewer;

	/**
	 * @var \WPGraphQL\Registry\TypeRegistry
	 */
	public $type_registry;

	/**
	 * Stores everything from the $_REQUEST global
	 *
	 * @var mixed $request
	 */
	public $request;

	/**
	 * Stores additional $config properties
	 *
	 * @var mixed $config
	 */
	public $config;

	/**
	 * Passes context about the current connection being resolved
	 *
	 * @todo These properties and methods are unused. We should consider deprecating/removing them.
	 *
	 * @var mixed|string|null
	 */
	public $currentConnection = null;

	/**
	 * Passes context about the current connection
	 *
	 * @todo These properties and methods are unused. We should consider deprecating/removing them.
	 *
	 * @var array<string,mixed>
	 */
	public $connectionArgs = [];

	/**
	 * Stores the loaders for the class
	 *
	 * @var array<string,\WPGraphQL\Data\Loader\AbstractDataLoader>
	 *
	 * phpcs:disable SlevomatCodingStandard.Namespaces.FullyQualifiedClassNameInAnnotation, -- For phpstan type hinting
	 *
	 * @template T of key-of<self::DEFAULT_LOADERS>
	 *
	 * @phpstan-var array<T, new<self::DEFAULT_LOADERS[T]>>|array<string,\WPGraphQL\Data\Loader\AbstractDataLoader>
	 *
	 * phpcs:enable
	 */
	public $loaders = [];

	/**
	 * Instance of the NodeResolver class to resolve nodes by URI
	 *
	 * @var \WPGraphQL\Data\NodeResolver
	 */
	public $node_resolver;

	/**
	 * The loader classes, before they are instantiated.
	 *
	 * @var array<string,class-string<\WPGraphQL\Data\Loader\AbstractDataLoader>>
	 */
	private $loader_classes = self::DEFAULT_LOADERS;

	/**
	 * AppContext constructor.
	 */
	public function __construct() {

		// Prime the loader classes (and their instances) for the AppContext.
		$this->prepare_data_loaders();

		/**
		 * This sets up the NodeResolver to allow nodes to be resolved by URI
		 */
		$this->node_resolver = new NodeResolver( $this );

		/**
		 * This filters the config for the AppContext.
		 *
		 * This can be used to store additional context config, which is available to resolvers
		 * throughout the resolution of a GraphQL request.
		 *
		 * @param mixed[] $config The config array of the AppContext object
		 */
		$this->config = apply_filters( 'graphql_app_context_config', $this->config );
	}

	/**
	 * Prepares the data loaders for the AppContext.
	 *
	 * This method instantiates the loader classes and prepares them for use in the AppContext.
	 * It also applies filters to allow customization of the loader classes.
	 *
	 * @uses graphql_data_loader_classes filter.
	 * @uses graphql_data_loaders filter (deprecated).
	 */
	private function prepare_data_loaders(): void {
		/**
		 * Filter to change the data loader classes.
		 *
		 * This allows for additional loaders to be added to the AppContext or replaced as needed.
		 *
		 * @param array<string,class-string<\WPGraphQL\Data\Loader\AbstractDataLoader>> $loader_classes The loader classes accessible in the AppContext
		 * @param \WPGraphQL\AppContext                                                $context        The AppContext
		 */
		$this->loader_classes = apply_filters( 'graphql_data_loader_classes', $this->loader_classes, $this );

		/**
		 * Prime the loaders if needed
		 *
		 * @todo Remove this when the loaders are instantiated on demand.
		 */
		if ( has_filter( 'graphql_data_loaders' ) ) {
			$loaders = array_map(
				function ( $loader_class ) {
					return new $loader_class( $this );
				},
				$this->loader_classes
			);

			/**
			 * @deprecated next-version in favor of graphql_data_loader_classes.
			 * @todo Remove in a future version.
			 *
			 * @param array<string,\WPGraphQL\Data\Loader\AbstractDataLoader> $loaders The loaders accessible in the AppContext
			 * @param \WPGraphQL\AppContext                                   $context The AppContext
			 */
			$this->loaders = apply_filters_deprecated(
				'graphql_data_loaders',
				[ $loaders, $this ],
				'2.3.2',
				'graphql_data_loader_classes',
				esc_html__( 'The graphql_data_loaders filter is deprecated and will be removed in a future version. Instead, use the graphql_data_loader_classes filter to add/change data loader classes before they are instantiated.', 'wp-graphql' ),
			);
		}
	}

	/**
	 * Retrieves loader assigned to $key
	 *
	 * @template T of key-of<self::DEFAULT_LOADERS>
	 *
	 * @param T|string $key The name of the loader to get.
	 *
	 * @return \WPGraphQL\Data\Loader\AbstractDataLoader
	 * @throws \GraphQL\Error\UserError If the loader is not found.
	 *
	 * @phpstan-return ( $key is T ? new<self::DEFAULT_LOADERS[T]> : \WPGraphQL\Data\Loader\AbstractDataLoader )
	 */
	public function get_loader( $key ) {
		// @todo: Remove the isset() when `graphql_data_loaders` is removed.
		if ( ! array_key_exists( $key, $this->loader_classes ) && ! isset( $this->loaders[ $key ] ) ) {
			// translators: %s is the key of the loader that was not found.
			throw new UserError( esc_html( sprintf( __( 'No loader assigned to the key %s', 'wp-graphql' ), $key ) ) );
		}

		// If the loader is not instantiated, instantiate it.
		if ( ! isset( $this->loaders[ $key ] ) ) {
			try {
				$this->loaders[ $key ] = new $this->loader_classes[ $key ]( $this );
			} catch ( \Throwable $e ) {
				// translators: %s is the key of the loader that failed to instantiate.
				throw new UserError( esc_html( sprintf( __( 'Failed to instantiate %1$s: %2$s', 'wp-graphql' ), $this->loader_classes[ $key ], $e->getMessage() ) ) );
			}
		}

		return $this->loaders[ $key ];
	}

	/**
	 * Magic getter used to warn about accessing the loaders property directly.
	 *
	 * @todo Remove this when we change the property visibility.
	 *
	 * @param string $key The name of the property to get.
	 * @return mixed
	 */
	public function __get( $key ) {
		// Use default handling if the key is not a loader.
		if ( 'loaders' !== $key ) {
			return $this->$key;
		}

		/** @todo Remove in v3.0.0 */
		_doing_it_wrong(
			__METHOD__,
			esc_html__( 'Accessing the AppContext::$loaders property from outside the AppContext class is deprecated and will throw an error in the next major version of WPGraphQL. Use AppContext::get_loader() instead.', 'wp-graphql' ),
			'2.3.2'
		);

		// Return the actual loaders array.
		return $this->loaders;
	}

	/**
	 * Returns the $args for the connection the field is a part of
	 *
	 * @todo These properties and methods are unused. We should consider deprecating/removing them.
	 *
	 * @return mixed[]|mixed
	 */
	public function get_connection_args() {
		return isset( $this->currentConnection ) && isset( $this->connectionArgs[ $this->currentConnection ] ) ? $this->connectionArgs[ $this->currentConnection ] : [];
	}

	/**
	 * Returns the current connection
	 *
	 * @todo These properties and methods are unused. We should consider deprecating/removing them.
	 *
	 * @return mixed|string|null
	 */
	public function get_current_connection() {
		return isset( $this->currentConnection ) ? $this->currentConnection : null;
	}

	/**
	 * @todo remove in v3.0.0
	 * @deprecated use get_connection_args() instead
	 * @codeCoverageIgnore
	 *
	 * @return mixed[]|mixed
	 */
	public function getConnectionArgs() {
		_doing_it_wrong(
			__METHOD__,
			sprintf(
				// translators: %s is the method name.
				esc_html__( 'This function will be removed in the next major version of WPGraphQL. Use %s instead.', 'wp-graphql' ),
				esc_html( self::class . '::get_connection_args()' )
			),
			'0.8.4',
		);
		return $this->get_connection_args();
	}

	/**
	 * @todo Remove in v3.0.0
	 * @deprecated Use get_loader instead.
	 * @codeCoverageIgnore
	 *
	 * @param string $key The name of the loader to get
	 *
	 * @return \WPGraphQL\Data\Loader\AbstractDataLoader
	 */
	public function getLoader( $key ) {
		_doing_it_wrong(
			__METHOD__,
			sprintf(
				// translators: %s is the method name.
				esc_html__( 'This function will be removed in the next major version of WPGraphQL. Use %s instead.', 'wp-graphql' ),
				esc_html( self::class . '::get_loader()' )
			),
			'0.8.4',
		);
		return $this->get_loader( $key );
	}

	/**
	 * @todo Remove in v3.0.0
	 * @deprecated use get_current_connection instead.
	 * @codeCoverageIgnore
	 *
	 * @return mixed|string|null
	 */
	public function getCurrentConnection() {
		_doing_it_wrong(
			__METHOD__,
			sprintf(
				// translators: %s is the method name.
				esc_html__( 'This function will be removed in the next major version of WPGraphQL. Use %s instead.', 'wp-graphql' ),
				esc_html( self::class . '::get_current_connection()' )
			),
			'0.8.4',
		);
		return $this->get_current_connection();
	}
}