oont-contents/plugins/jetpack-boost/app/modules/class-module.php
2025-04-06 08:34:48 +02:00

205 lines
5.6 KiB
PHP

<?php
namespace Automattic\Jetpack_Boost\Modules;
use Automattic\Jetpack_Boost\Contracts\Changes_Output_After_Activation;
use Automattic\Jetpack_Boost\Contracts\Changes_Output_On_Activation;
use Automattic\Jetpack_Boost\Contracts\Feature;
use Automattic\Jetpack_Boost\Contracts\Has_Activate;
use Automattic\Jetpack_Boost\Contracts\Has_Deactivate;
use Automattic\Jetpack_Boost\Contracts\Needs_To_Be_Ready;
use Automattic\Jetpack_Boost\Contracts\Optimization;
use Automattic\Jetpack_Boost\Contracts\Sub_Feature;
use Automattic\Jetpack_Boost\Lib\Status;
class Module {
const DISABLE_MODULE_QUERY_VAR = 'jb-disable-modules';
/**
* @var Status
*/
private $status;
/**
* @var Feature
*/
public $feature;
public function __construct( Feature $feature ) {
$this->feature = $feature;
$this->status = new Status( $feature::get_slug() );
}
public function on_activate() {
if ( $this->feature instanceof Changes_Output_On_Activation ) {
$this->indicate_page_output_changed();
}
return $this->feature instanceof Has_Activate ? $this->feature::activate() : true;
}
public function on_deactivate() {
// If the module changes the page output, with or without preparation, deactivating the module should indicate a page output change.
if ( $this->feature instanceof Changes_Output_On_Activation || $this->feature instanceof Changes_Output_After_Activation ) {
$this->indicate_page_output_changed();
}
return $this->feature instanceof Has_Deactivate ? $this->feature::deactivate() : true;
}
public function indicate_page_output_changed() {
/**
* Indicate that the HTML output of front-end has changed.
*
* If there is any page cache, it should be invalidated when this action is triggered.
*/
do_action( 'jetpack_boost_page_output_changed' );
}
public function get_slug() {
return $this->feature::get_slug();
}
/**
* If the module has any submodules, this method will return an array of Module instances for each submodule.
*/
public function get_submodules() {
$subfeatures = Features_Index::get_sub_features_of( $this->feature );
$modules = array();
foreach ( $subfeatures as $subfeature ) {
$modules[ $subfeature::get_slug() ] = new Module( new $subfeature() );
}
return $modules;
}
public function get_available_submodules() {
$submodules = $this->get_submodules();
if ( empty( $submodules ) ) {
return array();
}
$available_submodules = array();
foreach ( $submodules as $slug => $submodule ) {
if ( $submodule->is_available() ) {
$available_submodules[ $slug ] = $submodule;
}
}
return $available_submodules;
}
/**
* Get the active parent modules.
*
* @return Module[] The active parent modules.
*/
public function get_active_parent_modules() {
if ( ! $this->feature instanceof Sub_Feature ) {
return array();
}
$parent_features = $this->feature->get_parent_features();
$modules = array();
foreach ( $parent_features as $parent_feature ) {
$parent_module = new Module( new $parent_feature() );
if ( $parent_module->is_enabled() ) {
$modules[ $parent_module->get_slug() ] = $parent_module;
}
}
return $modules;
}
public function update( $new_status ) {
return $this->status->set( $new_status );
}
/**
* Check if the module is enabled.
*
* If the module is always on, it is enabled. Otherwise, check database for the module status.
* If it's a submodule, the status is only about the submodule itself, not its parent modules.
*
* @return bool True if the module is enabled, false otherwise.
*/
public function is_enabled() {
$always_on = is_subclass_of( $this->feature, 'Automattic\Jetpack_Boost\Contracts\Is_Always_On' );
if ( $always_on ) {
return true;
}
return $this->status->get();
}
/**
* Check if the module is available.
*
* If the module is not available, it cannot be enabled.
*/
public function is_available() {
if ( ! $this->feature::is_available() ) {
return false;
}
if ( $this->is_force_disabled() ) {
return false;
}
// If the module is not a sub-module, and it already passed the availability check, it is available.
if ( ! $this->feature instanceof Sub_Feature ) {
return true;
}
// If the module is a sub-module, it is available if at least one of its parent modules is available.
foreach ( $this->feature::get_parent_features() as $parent_feature ) {
if ( ( new Module( new $parent_feature() ) )->is_available() ) {
return true;
}
}
return false;
}
private function is_force_disabled() {
$slug = $this->feature::get_slug();
// phpcs:disable WordPress.Security.NonceVerification.Recommended
if ( ! empty( $_GET[ self::DISABLE_MODULE_QUERY_VAR ] ) ) {
// phpcs:disable WordPress.Security.NonceVerification.Recommended
// phpcs:disable WordPress.Security.ValidatedSanitizedInput.MissingUnslash
// phpcs:disable WordPress.Security.ValidatedSanitizedInput.InputNotSanitized
$disabled_modules = array_map( 'sanitize_key', explode( ',', $_GET[ self::DISABLE_MODULE_QUERY_VAR ] ) );
return in_array( $slug, $disabled_modules, true ) || in_array( 'all', $disabled_modules, true );
}
return false;
}
/**
* Check if the module is active and ready to serve optimized output.
*/
public function is_optimizing() {
if ( ! $this->is_available() ) {
return false;
}
if ( ! ( $this->feature instanceof Optimization ) || ! $this->is_enabled() ) {
return false;
}
if ( $this->feature instanceof Needs_To_Be_Ready && ! $this->feature->is_ready() ) {
return false;
}
if ( $this->feature instanceof Sub_Feature ) {
$parent_modules = $this->get_active_parent_modules();
if ( empty( $parent_modules ) ) {
return false;
}
}
return true;
}
}