237 lines
6.8 KiB
PHP
Raw Normal View History

2024-05-20 15:37:46 +03:00
<?php
/**
* WPSEO plugin file.
*
* @package WPSEO\Admin\Tracking
*/
use Yoast\WP\SEO\Analytics\Application\Missing_Indexables_Collector;
use Yoast\WP\SEO\Analytics\Application\To_Be_Cleaned_Indexables_Collector;
/**
* This class handles the tracking routine.
*/
class WPSEO_Tracking implements WPSEO_WordPress_Integration {
/**
* The tracking option name.
*
* @var string
*/
protected $option_name = 'wpseo_tracking_last_request';
/**
* The limit for the option.
*
* @var int
*/
protected $threshold = 0;
/**
* The endpoint to send the data to.
*
* @var string
*/
protected $endpoint = '';
/**
* The current time.
*
* @var int
*/
private $current_time;
/**
* WPSEO_Tracking constructor.
*
* @param string $endpoint The endpoint to send the data to.
* @param int $threshold The limit for the option.
*/
public function __construct( $endpoint, $threshold ) {
if ( ! $this->tracking_enabled() ) {
return;
}
$this->endpoint = $endpoint;
$this->threshold = $threshold;
$this->current_time = time();
}
/**
* Registers all hooks to WordPress.
*/
public function register_hooks() {
if ( ! $this->tracking_enabled() ) {
return;
}
// Send tracking data on `admin_init`.
add_action( 'admin_init', [ $this, 'send' ], 1 );
// Add an action hook that will be triggered at the specified time by `wp_schedule_single_event()`.
add_action( 'wpseo_send_tracking_data_after_core_update', [ $this, 'send' ] );
// Call `wp_schedule_single_event()` after a WordPress core update.
add_action( 'upgrader_process_complete', [ $this, 'schedule_tracking_data_sending' ], 10, 2 );
}
/**
* Schedules a new sending of the tracking data after a WordPress core update.
*
* @param bool|WP_Upgrader $upgrader Optional. WP_Upgrader instance or false.
* Depending on context, it might be a Theme_Upgrader,
* Plugin_Upgrader, Core_Upgrade, or Language_Pack_Upgrader.
* instance. Default false.
* @param array $data Array of update data.
*
* @return void
*/
public function schedule_tracking_data_sending( $upgrader = false, $data = [] ) {
// Return if it's not a WordPress core update.
if ( ! $upgrader || ! isset( $data['type'] ) || $data['type'] !== 'core' ) {
return;
}
/*
* To uniquely identify the scheduled cron event, `wp_next_scheduled()`
* needs to receive the same arguments as those used when originally
* scheduling the event otherwise it will always return false.
*/
if ( ! wp_next_scheduled( 'wpseo_send_tracking_data_after_core_update', [ true ] ) ) {
/*
* Schedule sending of data tracking 6 hours after a WordPress core
* update. Pass a `true` parameter for the callback `$force` argument.
*/
wp_schedule_single_event( ( time() + ( HOUR_IN_SECONDS * 6 ) ), 'wpseo_send_tracking_data_after_core_update', [ true ] );
}
}
/**
* Sends the tracking data.
*
* @param bool $force Whether to send the tracking data ignoring the two
* weeks time threshold. Default false.
*/
public function send( $force = false ) {
if ( ! $this->should_send_tracking( $force ) ) {
return;
}
// Set a 'content-type' header of 'application/json'.
$tracking_request_args = [
'headers' => [
'content-type:' => 'application/json',
],
];
$collector = $this->get_collector();
$request = new WPSEO_Remote_Request( $this->endpoint, $tracking_request_args );
$request->set_body( $collector->get_as_json() );
$request->send();
update_option( $this->option_name, $this->current_time, 'yes' );
}
/**
* Determines whether to send the tracking data.
*
* Returns false if tracking is disabled or the current page is one of the
* admin plugins pages. Returns true when there's no tracking data stored or
* the data was sent more than two weeks ago. The two weeks interval is set
* when instantiating the class.
*
* @param bool $ignore_time_treshhold Whether to send the tracking data ignoring
* the two weeks time treshhold. Default false.
*
* @return bool True when tracking data should be sent.
*/
protected function should_send_tracking( $ignore_time_treshhold = false ) {
global $pagenow;
// Only send tracking on the main site of a multi-site instance. This returns true on non-multisite installs.
if ( is_network_admin() || ! is_main_site() ) {
return false;
}
// Because we don't want to possibly block plugin actions with our routines.
if ( in_array( $pagenow, [ 'plugins.php', 'plugin-install.php', 'plugin-editor.php' ], true ) ) {
return false;
}
$last_time = get_option( $this->option_name );
// When tracking data haven't been sent yet or when sending data is forced.
if ( ! $last_time || $ignore_time_treshhold ) {
return true;
}
return $this->exceeds_treshhold( $this->current_time - $last_time );
}
/**
* Checks if the given amount of seconds exceeds the set threshold.
*
* @param int $seconds The amount of seconds to check.
*
* @return bool True when seconds is bigger than threshold.
*/
protected function exceeds_treshhold( $seconds ) {
return ( $seconds > $this->threshold );
}
/**
* Returns the collector for collecting the data.
*
* @return WPSEO_Collector The instance of the collector.
*/
public function get_collector() {
$collector = new WPSEO_Collector();
$collector->add_collection( new WPSEO_Tracking_Default_Data() );
$collector->add_collection( new WPSEO_Tracking_Server_Data() );
$collector->add_collection( new WPSEO_Tracking_Theme_Data() );
$collector->add_collection( new WPSEO_Tracking_Plugin_Data() );
$collector->add_collection( new WPSEO_Tracking_Settings_Data() );
$collector->add_collection( new WPSEO_Tracking_Addon_Data() );
$collector->add_collection( YoastSEO()->classes->get( Missing_Indexables_Collector::class ) );
$collector->add_collection( YoastSEO()->classes->get( To_Be_Cleaned_Indexables_Collector::class ) );
return $collector;
}
/**
* See if we should run tracking at all.
*
* @return bool True when we can track, false when we can't.
*/
private function tracking_enabled() {
// Check if we're allowing tracking.
$tracking = WPSEO_Options::get( 'tracking' );
if ( $tracking === false ) {
return false;
}
// Save this state.
if ( $tracking === null ) {
/**
* Filter: 'wpseo_enable_tracking' - Enables the data tracking of Yoast SEO Premium and add-ons.
*
* @api string $is_enabled The enabled state. Default is false.
*/
$tracking = apply_filters( 'wpseo_enable_tracking', false );
WPSEO_Options::set( 'tracking', $tracking );
}
if ( $tracking === false ) {
return false;
}
if ( ! YoastSEO()->helpers->environment->is_production_mode() ) {
return false;
}
return true;
}
}