Files
woocommerce-subscriptions/includes/core/class-wcs-core-autoloader.php
2025-05-21 10:18:23 +00:00

249 lines
6.8 KiB
PHP

<?php
// Exit if accessed directly.
if ( ! defined( 'ABSPATH' ) ) {
die();
}
/**
* WooCommerce Subscriptions Core Autoloader.
*
* @class WCS_Autoloader
*/
class WCS_Core_Autoloader {
/**
* The base path for autoloading.
*
* @var string
*/
protected $base_path = '';
/**
* WCS_Autoloader constructor.
*
* @param string $base_path
*/
public function __construct( $base_path ) {
$this->base_path = untrailingslashit( $base_path );
}
/**
* Destructor.
*/
public function __destruct() {
$this->unregister();
}
/**
* Register the autoloader.
*/
public function register() {
spl_autoload_register( array( $this, 'autoload' ) );
}
/**
* Unregister the autoloader.
*/
public function unregister() {
spl_autoload_unregister( array( $this, 'autoload' ) );
}
/**
* Autoload a class.
*
* @param string $class The class name to autoload.
*/
public function autoload( $class ) {
$class = strtolower( $class );
if ( ! $this->should_autoload( $class ) ) {
return;
}
$full_path = $this->get_class_base_path( $class ) . $this->get_relative_class_path( $class ) . $this->get_file_name( $class );
if ( is_readable( $full_path ) ) {
require_once $full_path; // nosemgrep: audit.php.lang.security.file.inclusion-arg -- This is a safe file inclusion, since the autoloader is only used for classes in the includes dir.
}
}
/**
* Gets the base path for a given class.
*
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v4.0.0
* @return string
*/
public function get_class_base_path( $class ) {
return $this->base_path;
}
/**
* Determine whether we should autoload a given class.
*
* @param string $class The class name.
*
* @return bool
*/
protected function should_autoload( $class ) {
// We're not using namespaces, so if the class has namespace separators, skip.
if ( false !== strpos( $class, '\\' ) ) {
return false;
}
return false !== strpos( $class, 'wcs_' ) || 0 === strpos( $class, 'wc_subscription' ) || ( false !== strpos( $class, 'wc_' ) && false !== strpos( $class, 'subscription' ) );
}
/**
* Convert the class name into an appropriate file name.
*
* @param string $class The class name.
*
* @return string The file name.
*/
protected function get_file_name( $class ) {
$file_prefix = 'class-';
if ( $this->is_class_abstract( $class ) ) {
$file_prefix = 'abstract-';
} elseif ( $this->is_class_interface( $class ) ) {
$file_prefix = 'interface-';
}
return $file_prefix . str_replace( '_', '-', $class ) . '.php';
}
/**
* Determine if the class is one of our abstract classes.
*
* @param string $class The class name.
*
* @return bool
*/
protected function is_class_abstract( $class ) {
static $abstracts = array(
'wcs_background_repairer' => true,
'wcs_background_updater' => true,
'wcs_background_upgrader' => true,
'wcs_cache_manager' => true,
'wcs_debug_tool' => true,
'wcs_debug_tool_cache_updater' => true,
'wcs_dynamic_hook_deprecator' => true,
'wcs_hook_deprecator' => true,
'wcs_scheduler' => true,
'wcs_sv_api_base' => true,
'wcs_customer_store' => true,
'wcs_related_order_store' => true,
'wcs_migrator' => true,
'wcs_table_maker' => true,
'wcs_deprecated_functions_handler' => true,
);
return isset( $abstracts[ $class ] );
}
/**
* Determine if the class is one of our class interfaces.
*
* @param string $class The class name.
* @return bool
*/
protected function is_class_interface( $class ) {
static $interfaces = array(
'wcs_cache_updater' => true,
'wcs_batch_processor' => true,
);
return isset( $interfaces[ $class ] );
}
/**
* Determine if the class is one of our data stores.
*
* @param string $class The class name.
* @return bool
*/
protected function is_class_data_store( $class ) {
static $data_stores = array(
'wcs_orders_table_subscription_data_store' => true,
'wcs_orders_table_data_store_controller' => true,
'wcs_related_order_store_cached_cpt' => true,
'wcs_related_order_store_cpt' => true,
'wcs_customer_store_cached_cpt' => true,
'wcs_customer_store_cpt' => true,
'wcs_product_variable_data_store_cpt' => true,
'wcs_subscription_data_store_cpt' => true,
);
return isset( $data_stores[ $class ] );
}
/**
* Get the relative path for the class location.
*
* This handles all of the special class locations and exceptions.
*
* @param string $class The class name.
*
* @return string The relative path (from the plugin root) to the class file.
*/
protected function get_relative_class_path( $class ) {
$path = '';
$is_admin = ( false !== strpos( $class, 'admin' ) );
if ( $this->is_class_abstract( $class ) ) {
if ( 'wcs_sv_api_base' === $class ) {
$path .= '/gateways/paypal/includes/abstracts';
} else {
$path .= '/abstracts';
}
} elseif ( $this->is_class_interface( $class ) ) {
$path .= '/interfaces';
} elseif ( false !== strpos( $class, 'paypal' ) ) {
$path .= '/gateways/paypal';
if ( 'wcs_paypal' === $class ) {
$path .= '';
} elseif ( 'wcs_repair_suspended_paypal_subscriptions' === $class ) {
// Deliberately avoid concatenation for this class, using the base path.
$path = '/includes/upgrades';
} elseif ( $is_admin ) {
$path .= '/includes/admin';
} elseif ( 'wc_paypal_standard_subscriptions' === $class ) {
$path .= '/includes/deprecated';
} else {
$path .= '/includes';
}
} elseif ( $is_admin && 'wcs_change_payment_method_admin' !== $class ) {
$path .= '/admin';
} elseif ( false !== strpos( $class, 'meta_box' ) ) {
$path .= '/admin/meta-boxes';
} elseif ( false !== strpos( $class, 'debug_tool' ) ) {
$path .= '/admin/debug-tools';
} elseif ( $this->is_class_data_store( $class ) ) {
$path .= '/data-stores';
} elseif ( false !== strpos( $class, 'deprecat' ) ) {
$path .= '/deprecated';
if ( false !== strpos( $class, 'handler' ) ) {
$path .= '/deprecation-handlers';
}
} elseif ( false !== strpos( $class, 'email' )
&& 'wc_subscriptions_email' !== $class
&& 'wc_subscriptions_email_notifications' !== $class
) {
$path .= '/emails';
} elseif ( false !== strpos( $class, 'gateway' ) && 'wc_subscriptions_change_payment_gateway' !== $class ) {
$path .= '/gateways';
} elseif ( false !== strpos( $class, 'legacy' ) || 'wcs_array_property_post_meta_black_magic' === $class ) {
$path .= '/legacy';
} elseif ( false !== strpos( $class, 'privacy' ) ) {
$path .= '/privacy';
} elseif ( false !== strpos( $class, 'upgrade' ) || false !== strpos( $class, 'repair' ) ) {
$path .= '/upgrades';
}
return trailingslashit( $path );
}
}