HEX
Server: Apache
System: Linux WWW 6.1.0-40-amd64 #1 SMP PREEMPT_DYNAMIC Debian 6.1.153-1 (2025-09-20) x86_64
User: web11 (1011)
PHP: 8.2.29
Disabled: NONE
Upload Files
File: /var/www/studis.kauko.lt/wp-content/plugins/the-events-calendar/common/src/Common/Hooks.php
<?php
/**
 * TEC Common Hooks
 *
 * @since 6.5.3
 *
 * @package TEC\Common;
 */

namespace TEC\Common;

use TEC\Common\Contracts\Provider\Controller as Controller_Contract;
use TEC\Common\Libraries\Provider as Library_Provider;
use TEC\Common\Contracts\Container;

/**
 * Class Hooks
 *
 * @since 6.5.3
 */
class Hooks extends Controller_Contract {
	/**
	 * The hook prefix.
	 *
	 * @since 6.7.1
	 *
	 * @var string
	 */
	private string $hook_prefix;

	/**
	 * The cached paths.
	 *
	 * @since 6.9.2
	 *
	 * @var array
	 */
	protected static array $cached_paths = [];

	/**
	 * Constructor.
	 *
	 * @since 6.7.1
	 *
	 * @param Container        $container The container.
	 * @param Library_Provider $library_provider The library provider.
	 */
	public function __construct( Container $container, Library_Provider $library_provider ) {
		parent::__construct( $container );
		$this->hook_prefix = $library_provider->get_hook_prefix();
	}

	/**
	 * Registers the hooks added by the controller.
	 *
	 * @since 6.5.3
	 * @since 6.7.1 Added hook for group paths to follow symlinks.
	 */
	public function do_register(): void {
		add_action( 'current_screen', [ $this, 'admin_headers_about_to_be_sent' ], PHP_INT_MAX );
		add_action( 'shutdown', [ $this, 'tec_shutdown' ], 0 );
		add_filter( 'tec_common_group_path', [ $this, 'group_paths_should_follow_symlinks' ] );
		add_filter( "stellarwp/assets/{$this->hook_prefix}/group_path", [ $this, 'group_paths_should_follow_symlinks' ] );
	}

	/**
	 * Removes hooks added by the controller.
	 *
	 * @since 6.5.3
	 * @since 6.7.1 Removed hook for group paths to follow symlinks.
	 */
	public function unregister(): void {
		remove_action( 'current_screen', [ $this, 'admin_headers_about_to_be_sent' ], PHP_INT_MAX );
		remove_action( 'shutdown', [ $this, 'tec_shutdown' ], 0 );
		remove_filter( 'tec_common_group_path', [ $this, 'group_paths_should_follow_symlinks' ] );
		remove_filter( "stellarwp/assets/{$this->hook_prefix}/group_path", [ $this, 'group_paths_should_follow_symlinks' ] );
	}

	/**
	 * Fires an action just before headers are sent.
	 *
	 * @since 6.5.3
	 */
	public function admin_headers_about_to_be_sent() {
		/**
		 * Fires just before headers are sent.
		 *
		 * We can use this action instead of headers_sent().
		 *
		 * Especially where a functionality would trigger a fatal error if headers are
		 * sent using an action is more forgiving.
		 *
		 * @since 6.5.3
		 */
		do_action( 'tec_admin_headers_about_to_be_sent' );
	}

	/**
	 * Fires an action during the shutdown action.
	 *
	 * @since 6.5.3
	 */
	public function tec_shutdown() {
		/**
		 * Fires during the shutdown action.
		 *
		 * This is mostly useful for testing code. We can trigger this action
		 * instead of triggering the whole shutdown.
		 *
		 * In production code, it can help us only in the sense of adding our own
		 * actions in a specific order.
		 *
		 * @since 6.5.3
		 */
		do_action( 'tec_shutdown' );
	}

	/**
	 * Ensure we follow symlinks for the group paths.
	 *
	 * @since 6.7.1
	 *
	 * @param array $group_path_data The group path data.
	 *
	 * @return array The group path data.
	 */
	public function group_paths_should_follow_symlinks( array $group_path_data ): array {
		if ( empty( $group_path_data['root'] ) ) {
			return $group_path_data;
		}

		if ( isset( self::$cached_paths[ $group_path_data['root'] ] ) ) {
			$group_path_data['root'] = self::$cached_paths[ $group_path_data['root'] ];

			return $group_path_data;
		}

		$test_path         = $group_path_data['root'] ?? '';
		$is_inside_plugins = $test_path !== str_replace( trailingslashit( WP_PLUGIN_DIR ), '', $test_path );

		$is_inside_plugins = $is_inside_plugins || $test_path !== str_replace( trailingslashit( dirname( __DIR__, 4 ) ), '', $test_path );

		$following_symlinks_root = str_replace( trailingslashit( dirname( __DIR__, 4 ) ), trailingslashit( WP_PLUGIN_DIR ), $group_path_data['root'] ?? '' );

		if ( $is_inside_plugins ) {
			$group_path_data['root'] = $following_symlinks_root;

			self::$cached_paths[ $group_path_data['root'] ] = $group_path_data['root'];

			return $group_path_data;
		}

		$identified_plugin     = false;
		$possible_plugin_slugs = explode( DIRECTORY_SEPARATOR, $group_path_data['root'] );

		foreach ( $possible_plugin_slugs as $key => $possible_plugin_slug ) {
			unset( $possible_plugin_slugs[ $key ] );
			if ( ! is_dir( WP_PLUGIN_DIR . DIRECTORY_SEPARATOR . $possible_plugin_slug ) ) {
				continue;
			}

			$group_path_data['root'] = trailingslashit( WP_PLUGIN_DIR . DIRECTORY_SEPARATOR . $possible_plugin_slug );
			$identified_plugin       = true;
			break;
		}

		if ( ! $identified_plugin ) {
			$group_path_data['root'] = $following_symlinks_root;

			self::$cached_paths[ $group_path_data['root'] ] = $group_path_data['root'];

			return $group_path_data;
		}

		if ( ! empty( $possible_plugin_slugs ) ) {
			foreach ( $possible_plugin_slugs as $path_part ) {
				if ( ! is_dir( $group_path_data['root'] . $path_part ) ) {
					continue;
				}

				$group_path_data['root'] = trailingslashit( $group_path_data['root'] . $path_part );
			}
		}

		self::$cached_paths[ $group_path_data['root'] ] = $group_path_data['root'];

		return $group_path_data;
	}
}