237 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			237 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?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;
 | |
| 	}
 | |
| }
 |