1666 lines
		
	
	
		
			51 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			1666 lines
		
	
	
		
			51 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | ||
| /**
 | ||
|  * WPSEO plugin file.
 | ||
|  *
 | ||
|  * @package WPSEO\Internal
 | ||
|  */
 | ||
| 
 | ||
| use Yoast\WP\Lib\Model;
 | ||
| use Yoast\WP\SEO\Helpers\Taxonomy_Helper;
 | ||
| use Yoast\WP\SEO\Integrations\Cleanup_Integration;
 | ||
| 
 | ||
| /**
 | ||
|  * This code handles the option upgrades.
 | ||
|  */
 | ||
| class WPSEO_Upgrade {
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * The taxonomy helper.
 | ||
| 	 *
 | ||
| 	 * @var Taxonomy_Helper
 | ||
| 	 */
 | ||
| 	private $taxonomy_helper;
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Class constructor.
 | ||
| 	 */
 | ||
| 	public function __construct() {
 | ||
| 		$this->taxonomy_helper = YoastSEO()->helpers->taxonomy;
 | ||
| 
 | ||
| 		$version = WPSEO_Options::get( 'version' );
 | ||
| 
 | ||
| 		WPSEO_Options::maybe_set_multisite_defaults( false );
 | ||
| 
 | ||
| 		$routines = [
 | ||
| 			'1.5.0'      => 'upgrade_15',
 | ||
| 			'2.0'        => 'upgrade_20',
 | ||
| 			'2.1'        => 'upgrade_21',
 | ||
| 			'2.2'        => 'upgrade_22',
 | ||
| 			'2.3'        => 'upgrade_23',
 | ||
| 			'3.0'        => 'upgrade_30',
 | ||
| 			'3.3'        => 'upgrade_33',
 | ||
| 			'3.6'        => 'upgrade_36',
 | ||
| 			'4.0'        => 'upgrade_40',
 | ||
| 			'4.4'        => 'upgrade_44',
 | ||
| 			'4.7'        => 'upgrade_47',
 | ||
| 			'4.9'        => 'upgrade_49',
 | ||
| 			'5.0'        => 'upgrade_50',
 | ||
| 			'5.5'        => 'upgrade_55',
 | ||
| 			'6.3'        => 'upgrade_63',
 | ||
| 			'7.0-RC0'    => 'upgrade_70',
 | ||
| 			'7.1-RC0'    => 'upgrade_71',
 | ||
| 			'7.3-RC0'    => 'upgrade_73',
 | ||
| 			'7.4-RC0'    => 'upgrade_74',
 | ||
| 			'7.5.3'      => 'upgrade_753',
 | ||
| 			'7.7-RC0'    => 'upgrade_77',
 | ||
| 			'7.7.2-RC0'  => 'upgrade_772',
 | ||
| 			'9.0-RC0'    => 'upgrade_90',
 | ||
| 			'10.0-RC0'   => 'upgrade_100',
 | ||
| 			'11.1-RC0'   => 'upgrade_111',
 | ||
| 			// Reset notifications because we removed the AMP Glue plugin notification.
 | ||
| 			'12.1-RC0'   => 'clean_all_notifications',
 | ||
| 			'12.3-RC0'   => 'upgrade_123',
 | ||
| 			'12.4-RC0'   => 'upgrade_124',
 | ||
| 			'12.8-RC0'   => 'upgrade_128',
 | ||
| 			'13.2-RC0'   => 'upgrade_132',
 | ||
| 			'14.0.3-RC0' => 'upgrade_1403',
 | ||
| 			'14.1-RC0'   => 'upgrade_141',
 | ||
| 			'14.2-RC0'   => 'upgrade_142',
 | ||
| 			'14.5-RC0'   => 'upgrade_145',
 | ||
| 			'14.9-RC0'   => 'upgrade_149',
 | ||
| 			'15.1-RC0'   => 'upgrade_151',
 | ||
| 			'15.3-RC0'   => 'upgrade_153',
 | ||
| 			'15.5-RC0'   => 'upgrade_155',
 | ||
| 			'15.7-RC0'   => 'upgrade_157',
 | ||
| 			'15.9.1-RC0' => 'upgrade_1591',
 | ||
| 			'16.2-RC0'   => 'upgrade_162',
 | ||
| 			'16.5-RC0'   => 'upgrade_165',
 | ||
| 			'17.2-RC0'   => 'upgrade_172',
 | ||
| 			'17.7.1-RC0' => 'upgrade_1771',
 | ||
| 			'17.9-RC0'   => 'upgrade_179',
 | ||
| 			'18.3-RC3'   => 'upgrade_183',
 | ||
| 			'18.6-RC0'   => 'upgrade_186',
 | ||
| 			'18.9-RC0'   => 'upgrade_189',
 | ||
| 			'19.1-RC0'   => 'upgrade_191',
 | ||
| 			'19.3-RC0'   => 'upgrade_193',
 | ||
| 			'19.6-RC0'   => 'upgrade_196',
 | ||
| 			'19.11-RC0'  => 'upgrade_1911',
 | ||
| 			'20.2-RC0'   => 'upgrade_202',
 | ||
| 			'20.5-RC0'   => 'upgrade_205',
 | ||
| 			'20.7-RC0'   => 'upgrade_207',
 | ||
| 			'20.8-RC0'   => 'upgrade_208',
 | ||
| 		];
 | ||
| 
 | ||
| 		array_walk( $routines, [ $this, 'run_upgrade_routine' ], $version );
 | ||
| 		if ( version_compare( $version, '12.5-RC0', '<' ) ) {
 | ||
| 			/*
 | ||
| 			 * We have to run this by hook, because otherwise:
 | ||
| 			 * - the theme support check isn't available.
 | ||
| 			 * - the notification center notifications are not filled yet.
 | ||
| 			 */
 | ||
| 			add_action( 'init', [ $this, 'upgrade_125' ] );
 | ||
| 		}
 | ||
| 
 | ||
| 		// Since 3.7.
 | ||
| 		$upsell_notice = new WPSEO_Product_Upsell_Notice();
 | ||
| 		$upsell_notice->set_upgrade_notice();
 | ||
| 
 | ||
| 		/**
 | ||
| 		 * Filter: 'wpseo_run_upgrade' - Runs the upgrade hook which are dependent on Yoast SEO.
 | ||
| 		 *
 | ||
| 		 * @api string - The current version of Yoast SEO
 | ||
| 		 */
 | ||
| 		do_action( 'wpseo_run_upgrade', $version );
 | ||
| 
 | ||
| 		$this->finish_up( $version );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Runs the upgrade routine.
 | ||
| 	 *
 | ||
| 	 * @param string $routine         The method to call.
 | ||
| 	 * @param string $version         The new version.
 | ||
| 	 * @param string $current_version The current set version.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	protected function run_upgrade_routine( $routine, $version, $current_version ) {
 | ||
| 		if ( version_compare( $current_version, $version, '<' ) ) {
 | ||
| 			$this->$routine( $current_version );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Adds a new upgrade history entry.
 | ||
| 	 *
 | ||
| 	 * @param string $current_version The old version from which we are upgrading.
 | ||
| 	 * @param string $new_version     The version we are upgrading to.
 | ||
| 	 */
 | ||
| 	protected function add_upgrade_history( $current_version, $new_version ) {
 | ||
| 		$upgrade_history = new WPSEO_Upgrade_History();
 | ||
| 		$upgrade_history->add( $current_version, $new_version, array_keys( WPSEO_Options::$options ) );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Runs the needed cleanup after an update, setting the DB version to latest version, flushing caches etc.
 | ||
| 	 *
 | ||
| 	 * @param string|null $previous_version The previous version.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	protected function finish_up( $previous_version = null ) {
 | ||
| 		if ( $previous_version ) {
 | ||
| 			WPSEO_Options::set( 'previous_version', $previous_version );
 | ||
| 		}
 | ||
| 		WPSEO_Options::set( 'version', WPSEO_VERSION );
 | ||
| 
 | ||
| 		// Just flush rewrites, always, to at least make them work after an upgrade.
 | ||
| 		add_action( 'shutdown', 'flush_rewrite_rules' );
 | ||
| 
 | ||
| 		// Flush the sitemap cache.
 | ||
| 		WPSEO_Sitemaps_Cache::clear();
 | ||
| 
 | ||
| 		// Make sure all our options always exist - issue #1245.
 | ||
| 		WPSEO_Options::ensure_options_exist();
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Run the Yoast SEO 1.5 upgrade routine.
 | ||
| 	 *
 | ||
| 	 * @param string $version Current plugin version.
 | ||
| 	 */
 | ||
| 	private function upgrade_15( $version ) {
 | ||
| 		// Clean up options and meta.
 | ||
| 		WPSEO_Options::clean_up( null, $version );
 | ||
| 		WPSEO_Meta::clean_up();
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Moves options that moved position in WPSEO 2.0.
 | ||
| 	 */
 | ||
| 	private function upgrade_20() {
 | ||
| 		/**
 | ||
| 		 * Clean up stray wpseo_ms options from the options table, option should only exist in the sitemeta table.
 | ||
| 		 * This could have been caused in many version of Yoast SEO, so deleting it for everything below 2.0.
 | ||
| 		 */
 | ||
| 		delete_option( 'wpseo_ms' );
 | ||
| 
 | ||
| 		$wpseo = $this->get_option_from_database( 'wpseo' );
 | ||
| 		$this->save_option_setting( $wpseo, 'pinterestverify' );
 | ||
| 
 | ||
| 		// Re-save option to trigger sanitization.
 | ||
| 		$this->cleanup_option_data( 'wpseo' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Detects if taxonomy terms were split and updates the corresponding taxonomy meta's accordingly.
 | ||
| 	 */
 | ||
| 	private function upgrade_21() {
 | ||
| 		$taxonomies = get_option( 'wpseo_taxonomy_meta', [] );
 | ||
| 
 | ||
| 		if ( ! empty( $taxonomies ) ) {
 | ||
| 			foreach ( $taxonomies as $taxonomy => $tax_metas ) {
 | ||
| 				foreach ( $tax_metas as $term_id => $tax_meta ) {
 | ||
| 					if ( function_exists( 'wp_get_split_term' ) ) {
 | ||
| 						$new_term_id = wp_get_split_term( $term_id, $taxonomy );
 | ||
| 						if ( $new_term_id !== false ) {
 | ||
| 							$taxonomies[ $taxonomy ][ $new_term_id ] = $taxonomies[ $taxonomy ][ $term_id ];
 | ||
| 							unset( $taxonomies[ $taxonomy ][ $term_id ] );
 | ||
| 						}
 | ||
| 					}
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 			update_option( 'wpseo_taxonomy_meta', $taxonomies );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs upgrade functions to Yoast SEO 2.2.
 | ||
| 	 */
 | ||
| 	private function upgrade_22() {
 | ||
| 		// Unschedule our tracking.
 | ||
| 		wp_clear_scheduled_hook( 'yoast_tracking' );
 | ||
| 
 | ||
| 		$this->cleanup_option_data( 'wpseo' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Schedules upgrade function to Yoast SEO 2.3.
 | ||
| 	 */
 | ||
| 	private function upgrade_23() {
 | ||
| 		add_action( 'wp', [ $this, 'upgrade_23_query' ], 90 );
 | ||
| 		add_action( 'admin_head', [ $this, 'upgrade_23_query' ], 90 );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs upgrade query to Yoast SEO 2.3.
 | ||
| 	 */
 | ||
| 	public function upgrade_23_query() {
 | ||
| 		$wp_query = new WP_Query( 'post_type=any&meta_key=_yoast_wpseo_sitemap-include&meta_value=never&order=ASC' );
 | ||
| 
 | ||
| 		if ( ! empty( $wp_query->posts ) ) {
 | ||
| 			$options = get_option( 'wpseo_xml' );
 | ||
| 
 | ||
| 			$excluded_posts = [];
 | ||
| 			if ( $options['excluded-posts'] !== '' ) {
 | ||
| 				$excluded_posts = explode( ',', $options['excluded-posts'] );
 | ||
| 			}
 | ||
| 
 | ||
| 			foreach ( $wp_query->posts as $post ) {
 | ||
| 				if ( ! in_array( (string) $post->ID, $excluded_posts, true ) ) {
 | ||
| 					$excluded_posts[] = $post->ID;
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 			// Updates the meta value.
 | ||
| 			$options['excluded-posts'] = implode( ',', $excluded_posts );
 | ||
| 
 | ||
| 			// Update the option.
 | ||
| 			update_option( 'wpseo_xml', $options );
 | ||
| 		}
 | ||
| 
 | ||
| 		// Remove the meta fields.
 | ||
| 		delete_post_meta_by_key( '_yoast_wpseo_sitemap-include' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs upgrade functions to Yoast SEO 3.0.
 | ||
| 	 */
 | ||
| 	private function upgrade_30() {
 | ||
| 		// Remove the meta fields for sitemap prio.
 | ||
| 		delete_post_meta_by_key( '_yoast_wpseo_sitemap-prio' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs upgrade functions to Yoast SEO 3.3.
 | ||
| 	 */
 | ||
| 	private function upgrade_33() {
 | ||
| 		// Notification dismissals have been moved to User Meta instead of global option.
 | ||
| 		delete_option( Yoast_Notification_Center::STORAGE_KEY );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs upgrade functions to Yoast SEO 3.6.
 | ||
| 	 */
 | ||
| 	private function upgrade_36() {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		// Between 3.2 and 3.4 the sitemap options were saved with autoloading enabled.
 | ||
| 		$wpdb->query( 'DELETE FROM ' . $wpdb->options . ' WHERE option_name LIKE "wpseo_sitemap_%" AND autoload = "yes"' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes the about notice when its still in the database.
 | ||
| 	 */
 | ||
| 	private function upgrade_40() {
 | ||
| 		$center = Yoast_Notification_Center::get();
 | ||
| 		$center->remove_notification_by_id( 'wpseo-dismiss-about' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Moves the content-analysis-active and keyword-analysis-acive options from wpseo-titles to wpseo.
 | ||
| 	 */
 | ||
| 	private function upgrade_44() {
 | ||
| 		$wpseo_titles = $this->get_option_from_database( 'wpseo_titles' );
 | ||
| 
 | ||
| 		$this->save_option_setting( $wpseo_titles, 'content-analysis-active', 'content_analysis_active' );
 | ||
| 		$this->save_option_setting( $wpseo_titles, 'keyword-analysis-active', 'keyword_analysis_active' );
 | ||
| 
 | ||
| 		// Remove irrelevant content from the option.
 | ||
| 		$this->cleanup_option_data( 'wpseo_titles' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Renames the meta name for the cornerstone content. It was a public meta field and it has to be private.
 | ||
| 	 */
 | ||
| 	private function upgrade_47() {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		// The meta key has to be private, so prefix it.
 | ||
| 		$wpdb->query(
 | ||
| 			$wpdb->prepare(
 | ||
| 				'UPDATE ' . $wpdb->postmeta . ' SET meta_key = %s WHERE meta_key = "yst_is_cornerstone"',
 | ||
| 				WPSEO_Cornerstone_Filter::META_NAME
 | ||
| 			)
 | ||
| 		);
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes the 'wpseo-dismiss-about' notice for every user that still has it.
 | ||
| 	 */
 | ||
| 	private function upgrade_49() {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		/*
 | ||
| 		 * Using a filter to remove the notification for the current logged in user. The notification center is
 | ||
| 		 * initializing the notifications before the upgrade routine has been executedd and is saving the stored
 | ||
| 		 * notifications on shutdown. This causes the returning notification. By adding this filter the shutdown
 | ||
| 		 * routine on the notification center will remove the notification.
 | ||
| 		 */
 | ||
| 		add_filter( 'yoast_notifications_before_storage', [ $this, 'remove_about_notice' ] );
 | ||
| 
 | ||
| 		$meta_key = $wpdb->get_blog_prefix() . Yoast_Notification_Center::STORAGE_KEY;
 | ||
| 
 | ||
| 		$usermetas = $wpdb->get_results(
 | ||
| 			$wpdb->prepare(
 | ||
| 				'
 | ||
| 				SELECT user_id, meta_value
 | ||
| 				FROM ' . $wpdb->usermeta . '
 | ||
| 				WHERE meta_key = %s AND meta_value LIKE %s
 | ||
| 				',
 | ||
| 				$meta_key,
 | ||
| 				'%wpseo-dismiss-about%'
 | ||
| 			),
 | ||
| 			ARRAY_A
 | ||
| 		);
 | ||
| 
 | ||
| 		if ( empty( $usermetas ) ) {
 | ||
| 			return;
 | ||
| 		}
 | ||
| 
 | ||
| 		foreach ( $usermetas as $usermeta ) {
 | ||
| 			$notifications = maybe_unserialize( $usermeta['meta_value'] );
 | ||
| 
 | ||
| 			foreach ( $notifications as $notification_key => $notification ) {
 | ||
| 				if ( ! empty( $notification['options']['id'] ) && $notification['options']['id'] === 'wpseo-dismiss-about' ) {
 | ||
| 					unset( $notifications[ $notification_key ] );
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 			update_user_option( $usermeta['user_id'], Yoast_Notification_Center::STORAGE_KEY, array_values( $notifications ) );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes the wpseo-dismiss-about notice from a list of notifications.
 | ||
| 	 *
 | ||
| 	 * @param Yoast_Notification[] $notifications The notifications to filter.
 | ||
| 	 *
 | ||
| 	 * @return Yoast_Notification[] The filtered list of notifications. Excluding the wpseo-dismiss-about notification.
 | ||
| 	 */
 | ||
| 	public function remove_about_notice( $notifications ) {
 | ||
| 		foreach ( $notifications as $notification_key => $notification ) {
 | ||
| 			if ( $notification->get_id() === 'wpseo-dismiss-about' ) {
 | ||
| 				unset( $notifications[ $notification_key ] );
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		return $notifications;
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Adds the yoast_seo_links table to the database.
 | ||
| 	 */
 | ||
| 	private function upgrade_50() {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		// Deletes the post meta value, which might created in the RC.
 | ||
| 		$wpdb->query( 'DELETE FROM ' . $wpdb->postmeta . ' WHERE meta_key = "_yst_content_links_processed"' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Register new capabilities and roles.
 | ||
| 	 */
 | ||
| 	private function upgrade_55() {
 | ||
| 		// Register roles.
 | ||
| 		do_action( 'wpseo_register_roles' );
 | ||
| 		WPSEO_Role_Manager_Factory::get()->add();
 | ||
| 
 | ||
| 		// Register capabilities.
 | ||
| 		do_action( 'wpseo_register_capabilities' );
 | ||
| 		WPSEO_Capability_Manager_Factory::get()->add();
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes some no longer used options for noindexing subpages and for meta keywords and its associated templates.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_63() {
 | ||
| 		$this->cleanup_option_data( 'wpseo_titles' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Perform the 7.0 upgrade, moves settings around, deletes several options.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_70() {
 | ||
| 
 | ||
| 		$wpseo_permalinks    = $this->get_option_from_database( 'wpseo_permalinks' );
 | ||
| 		$wpseo_xml           = $this->get_option_from_database( 'wpseo_xml' );
 | ||
| 		$wpseo_rss           = $this->get_option_from_database( 'wpseo_rss' );
 | ||
| 		$wpseo               = $this->get_option_from_database( 'wpseo' );
 | ||
| 		$wpseo_internallinks = $this->get_option_from_database( 'wpseo_internallinks' );
 | ||
| 
 | ||
| 		// Move some permalink settings, then delete the option.
 | ||
| 		$this->save_option_setting( $wpseo_permalinks, 'redirectattachment', 'disable-attachment' );
 | ||
| 		$this->save_option_setting( $wpseo_permalinks, 'stripcategorybase' );
 | ||
| 
 | ||
| 		// Move one XML sitemap setting, then delete the option.
 | ||
| 		$this->save_option_setting( $wpseo_xml, 'enablexmlsitemap', 'enable_xml_sitemap' );
 | ||
| 
 | ||
| 		// Move the RSS settings to the search appearance settings, then delete the RSS option.
 | ||
| 		$this->save_option_setting( $wpseo_rss, 'rssbefore' );
 | ||
| 		$this->save_option_setting( $wpseo_rss, 'rssafter' );
 | ||
| 
 | ||
| 		$this->save_option_setting( $wpseo, 'company_logo' );
 | ||
| 		$this->save_option_setting( $wpseo, 'company_name' );
 | ||
| 		$this->save_option_setting( $wpseo, 'company_or_person' );
 | ||
| 		$this->save_option_setting( $wpseo, 'person_name' );
 | ||
| 
 | ||
| 		// Remove the website name and altername name as we no longer need them.
 | ||
| 		$this->cleanup_option_data( 'wpseo' );
 | ||
| 
 | ||
| 		// All the breadcrumbs settings have moved to the search appearance settings.
 | ||
| 		foreach ( array_keys( $wpseo_internallinks ) as $key ) {
 | ||
| 			$this->save_option_setting( $wpseo_internallinks, $key );
 | ||
| 		}
 | ||
| 
 | ||
| 		// Convert hidden metabox options to display metabox options.
 | ||
| 		$title_options = get_option( 'wpseo_titles' );
 | ||
| 
 | ||
| 		foreach ( $title_options as $key => $value ) {
 | ||
| 			if ( strpos( $key, 'hideeditbox-tax-' ) === 0 ) {
 | ||
| 				$taxonomy = substr( $key, strlen( 'hideeditbox-tax-' ) );
 | ||
| 				WPSEO_Options::set( 'display-metabox-tax-' . $taxonomy, ! $value );
 | ||
| 				continue;
 | ||
| 			}
 | ||
| 
 | ||
| 			if ( strpos( $key, 'hideeditbox-' ) === 0 ) {
 | ||
| 				$post_type = substr( $key, strlen( 'hideeditbox-' ) );
 | ||
| 				WPSEO_Options::set( 'display-metabox-pt-' . $post_type, ! $value );
 | ||
| 				continue;
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		// Cleanup removed options.
 | ||
| 		delete_option( 'wpseo_xml' );
 | ||
| 		delete_option( 'wpseo_permalinks' );
 | ||
| 		delete_option( 'wpseo_rss' );
 | ||
| 		delete_option( 'wpseo_internallinks' );
 | ||
| 
 | ||
| 		// Remove possibly present plugin conflict notice for plugin that was removed from the list of conflicting plugins.
 | ||
| 		$yoast_plugin_conflict = WPSEO_Plugin_Conflict::get_instance();
 | ||
| 		$yoast_plugin_conflict->clear_error( 'header-footer/plugin.php' );
 | ||
| 
 | ||
| 		// Moves the user meta for excluding from the XML sitemap to a noindex.
 | ||
| 		global $wpdb;
 | ||
| 		$wpdb->query( "UPDATE $wpdb->usermeta SET meta_key = 'wpseo_noindex_author' WHERE meta_key = 'wpseo_excludeauthorsitemap'" );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Perform the 7.1 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_71() {
 | ||
| 		$this->cleanup_option_data( 'wpseo_social' );
 | ||
| 
 | ||
| 		// Move the breadcrumbs setting and invert it.
 | ||
| 		$title_options = $this->get_option_from_database( 'wpseo_titles' );
 | ||
| 
 | ||
| 		if ( array_key_exists( 'breadcrumbs-blog-remove', $title_options ) ) {
 | ||
| 			WPSEO_Options::set( 'breadcrumbs-display-blog-page', ! $title_options['breadcrumbs-blog-remove'] );
 | ||
| 
 | ||
| 			$this->cleanup_option_data( 'wpseo_titles' );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Perform the 7.3 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_73() {
 | ||
| 		global $wpdb;
 | ||
| 		// We've moved the cornerstone checkbox to our proper namespace.
 | ||
| 		$wpdb->query( "UPDATE $wpdb->postmeta SET meta_key = '_yoast_wpseo_is_cornerstone' WHERE meta_key = '_yst_is_cornerstone'" );
 | ||
| 
 | ||
| 		// Remove the previous Whip dismissed message, as this is a new one regarding PHP 5.2.
 | ||
| 		delete_option( 'whip_dismiss_timestamp' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 7.4 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_74() {
 | ||
| 		$this->remove_sitemap_validators();
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 7.5.3 upgrade.
 | ||
| 	 *
 | ||
| 	 * When upgrading purging media is potentially relevant.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_753() {
 | ||
| 		// Only when attachments are not disabled.
 | ||
| 		if ( WPSEO_Options::get( 'disable-attachment' ) === true ) {
 | ||
| 			return;
 | ||
| 		}
 | ||
| 
 | ||
| 		// Only when attachments are not no-indexed.
 | ||
| 		if ( WPSEO_Options::get( 'noindex-attachment' ) === true ) {
 | ||
| 			return;
 | ||
| 		}
 | ||
| 
 | ||
| 		// Set purging relevancy.
 | ||
| 		WPSEO_Options::set( 'is-media-purge-relevant', true );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 7.7 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_77() {
 | ||
| 		// Remove all OpenGraph content image cache.
 | ||
| 		$this->delete_post_meta( '_yoast_wpseo_post_image_cache' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 7.7.2 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_772() {
 | ||
| 		if ( YoastSEO()->helpers->woocommerce->is_active() ) {
 | ||
| 			$this->migrate_woocommerce_archive_setting_to_shop_page();
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 9.0 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_90() {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		// Invalidate all sitemap cache transients.
 | ||
| 		WPSEO_Sitemaps_Cache_Validator::cleanup_database();
 | ||
| 
 | ||
| 		// Removes all scheduled tasks for hitting the sitemap index.
 | ||
| 		wp_clear_scheduled_hook( 'wpseo_hit_sitemap_index' );
 | ||
| 
 | ||
| 		$wpdb->query( 'DELETE FROM ' . $wpdb->options . ' WHERE option_name LIKE "wpseo_sitemap_%"' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 10.0 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_100() {
 | ||
| 		// Removes recalibration notifications.
 | ||
| 		$this->clean_all_notifications();
 | ||
| 
 | ||
| 		// Removes recalibration options.
 | ||
| 		WPSEO_Options::clean_up( 'wpseo' );
 | ||
| 		delete_option( 'wpseo_recalibration_beta_mailinglist_subscription' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 11.1 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_111() {
 | ||
| 		// Set company_or_person to company when it's an invalid value.
 | ||
| 		$company_or_person = WPSEO_Options::get( 'company_or_person', '' );
 | ||
| 
 | ||
| 		if ( ! in_array( $company_or_person, [ 'company', 'person' ], true ) ) {
 | ||
| 			WPSEO_Options::set( 'company_or_person', 'company' );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 12.3 upgrade.
 | ||
| 	 *
 | ||
| 	 * Removes the about notice when its still in the database.
 | ||
| 	 */
 | ||
| 	private function upgrade_123() {
 | ||
| 		$plugins = [
 | ||
| 			'yoast-seo-premium',
 | ||
| 			'video-seo-for-wordpress-seo-by-yoast',
 | ||
| 			'yoast-news-seo',
 | ||
| 			'local-seo-for-yoast-seo',
 | ||
| 			'yoast-woocommerce-seo',
 | ||
| 			'yoast-acf-analysis',
 | ||
| 		];
 | ||
| 
 | ||
| 		$center = Yoast_Notification_Center::get();
 | ||
| 		foreach ( $plugins as $plugin ) {
 | ||
| 			$center->remove_notification_by_id( 'wpseo-outdated-yoast-seo-plugin-' . $plugin );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 12.4 upgrade.
 | ||
| 	 *
 | ||
| 	 * Removes the Google plus defaults from the database.
 | ||
| 	 */
 | ||
| 	private function upgrade_124() {
 | ||
| 		$this->cleanup_option_data( 'wpseo_social' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 12.5 upgrade.
 | ||
| 	 */
 | ||
| 	public function upgrade_125() {
 | ||
| 		// Disables the force rewrite title when the theme supports it through WordPress.
 | ||
| 		if ( WPSEO_Options::get( 'forcerewritetitle', false ) && current_theme_supports( 'title-tag' ) ) {
 | ||
| 			WPSEO_Options::set( 'forcerewritetitle', false );
 | ||
| 		}
 | ||
| 
 | ||
| 		global $wpdb;
 | ||
| 		$wpdb->query( "DELETE FROM $wpdb->usermeta WHERE meta_key = 'wp_yoast_promo_hide_premium_upsell_admin_block'" );
 | ||
| 
 | ||
| 		// Removes the WordPress update notification, because it is no longer necessary when WordPress 5.3 is released.
 | ||
| 		$center = Yoast_Notification_Center::get();
 | ||
| 		$center->remove_notification_by_id( 'wpseo-dismiss-wordpress-upgrade' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 12.8 upgrade.
 | ||
| 	 */
 | ||
| 	private function upgrade_128() {
 | ||
| 		// Re-save wpseo to make sure bf_banner_2019_dismissed key is gone.
 | ||
| 		$this->cleanup_option_data( 'wpseo' );
 | ||
| 
 | ||
| 		Yoast_Notification_Center::get()->remove_notification_by_id( 'wpseo-dismiss-page_comments-notice' );
 | ||
| 		Yoast_Notification_Center::get()->remove_notification_by_id( 'wpseo-dismiss-wordpress-upgrade' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 13.2 upgrade.
 | ||
| 	 */
 | ||
| 	private function upgrade_132() {
 | ||
| 		Yoast_Notification_Center::get()->remove_notification_by_id( 'wpseo-dismiss-tagline-notice' );
 | ||
| 		Yoast_Notification_Center::get()->remove_notification_by_id( 'wpseo-dismiss-permalink-notice' );
 | ||
| 		Yoast_Notification_Center::get()->remove_notification_by_id( 'wpseo-dismiss-onpageorg' );
 | ||
| 
 | ||
| 		// Transfers the onpage option value to the ryte option.
 | ||
| 		$ryte_option   = get_option( 'wpseo_ryte' );
 | ||
| 		$onpage_option = get_option( 'wpseo_onpage' );
 | ||
| 		if ( ! $ryte_option && $onpage_option ) {
 | ||
| 			update_option( 'wpseo_ryte', $onpage_option );
 | ||
| 			delete_option( 'wpseo_onpage' );
 | ||
| 		}
 | ||
| 
 | ||
| 		// Changes onpage_indexability to ryte_indexability.
 | ||
| 		$wpseo_option = get_option( 'wpseo' );
 | ||
| 		if ( isset( $wpseo_option['onpage_indexability'] ) && ! isset( $wpseo_option['ryte_indexability'] ) ) {
 | ||
| 			$wpseo_option['ryte_indexability'] = $wpseo_option['onpage_indexability'];
 | ||
| 			unset( $wpseo_option['onpage_indexability'] );
 | ||
| 			update_option( 'wpseo', $wpseo_option );
 | ||
| 		}
 | ||
| 
 | ||
| 		if ( wp_next_scheduled( 'wpseo_ryte_fetch' ) ) {
 | ||
| 			wp_clear_scheduled_hook( 'wpseo_ryte_fetch' );
 | ||
| 		}
 | ||
| 
 | ||
| 		/*
 | ||
| 		 * Re-register capabilities to add the new `view_site_health_checks`
 | ||
| 		 * capability to the SEO Manager role.
 | ||
| 		 */
 | ||
| 		do_action( 'wpseo_register_capabilities' );
 | ||
| 		WPSEO_Capability_Manager_Factory::get()->add();
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Perform the 14.0.3 upgrade.
 | ||
| 	 */
 | ||
| 	private function upgrade_1403() {
 | ||
| 		WPSEO_Options::set( 'ignore_indexation_warning', false );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 14.1 upgrade.
 | ||
| 	 */
 | ||
| 	private function upgrade_141() {
 | ||
| 		/*
 | ||
| 		 * These notifications are retrieved from storage on the `init` hook with
 | ||
| 		 * priority 1. We need to remove them after they're retrieved.
 | ||
| 		 */
 | ||
| 		add_action( 'init', [ $this, 'remove_notifications_for_141' ] );
 | ||
| 		add_action( 'init', [ $this, 'clean_up_private_taxonomies_for_141' ] );
 | ||
| 
 | ||
| 		$this->reset_permalinks_of_attachments_for_141();
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 14.2 upgrade.
 | ||
| 	 *
 | ||
| 	 * Removes the yoast-acf-analysis notice when it's still in the database.
 | ||
| 	 */
 | ||
| 	private function upgrade_142() {
 | ||
| 		add_action( 'init', [ $this, 'remove_acf_notification_for_142' ] );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 14.5 upgrade.
 | ||
| 	 */
 | ||
| 	private function upgrade_145() {
 | ||
| 		add_action( 'init', [ $this, 'set_indexation_completed_option_for_145' ] );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 14.9 upgrade.
 | ||
| 	 */
 | ||
| 	private function upgrade_149() {
 | ||
| 		$version = get_option( 'wpseo_license_server_version', 2 );
 | ||
| 		WPSEO_Options::set( 'license_server_version', $version );
 | ||
| 		delete_option( 'wpseo_license_server_version' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 15.1 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_151() {
 | ||
| 		$this->set_home_url_for_151();
 | ||
| 		$this->move_indexables_indexation_reason_for_151();
 | ||
| 
 | ||
| 		add_action( 'init', [ $this, 'set_permalink_structure_option_for_151' ] );
 | ||
| 		add_action( 'init', [ $this, 'store_custom_taxonomy_slugs_for_151' ] );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 15.3 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_153() {
 | ||
| 		WPSEO_Options::set( 'category_base_url', get_option( 'category_base' ) );
 | ||
| 		WPSEO_Options::set( 'tag_base_url', get_option( 'tag_base' ) );
 | ||
| 
 | ||
| 		// Rename a couple of options.
 | ||
| 		$indexation_started_value = WPSEO_Options::get( 'indexation_started' );
 | ||
| 		WPSEO_Options::set( 'indexing_started', $indexation_started_value );
 | ||
| 
 | ||
| 		$indexables_indexing_completed_value = WPSEO_Options::get( 'indexables_indexation_completed' );
 | ||
| 		WPSEO_Options::set( 'indexables_indexing_completed', $indexables_indexing_completed_value );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 15.5 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_155() {
 | ||
| 		// Unset the fbadminapp value in the wpseo_social option.
 | ||
| 		$wpseo_social_option = get_option( 'wpseo_social' );
 | ||
| 
 | ||
| 		if ( isset( $wpseo_social_option['fbadminapp'] ) ) {
 | ||
| 			unset( $wpseo_social_option['fbadminapp'] );
 | ||
| 			update_option( 'wpseo_social', $wpseo_social_option );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 15.7 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_157() {
 | ||
| 		add_action( 'init', [ $this, 'remove_plugin_updated_notification_for_157' ] );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 15.9.1 upgrade routine.
 | ||
| 	 */
 | ||
| 	private function upgrade_1591() {
 | ||
| 		$enabled_auto_updates = \get_option( 'auto_update_plugins' );
 | ||
| 		$addon_update_watcher = YoastSEO()->classes->get( \Yoast\WP\SEO\Integrations\Watchers\Addon_Update_Watcher::class );
 | ||
| 		$addon_update_watcher->toggle_auto_updates_for_add_ons( 'auto_update_plugins', [], $enabled_auto_updates );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 16.2 upgrade routine.
 | ||
| 	 */
 | ||
| 	private function upgrade_162() {
 | ||
| 		$enabled_auto_updates = \get_site_option( 'auto_update_plugins' );
 | ||
| 		$addon_update_watcher = YoastSEO()->classes->get( \Yoast\WP\SEO\Integrations\Watchers\Addon_Update_Watcher::class );
 | ||
| 		$addon_update_watcher->toggle_auto_updates_for_add_ons( 'auto_update_plugins', $enabled_auto_updates, [] );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 16.5 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_165() {
 | ||
| 		add_action( 'init', [ $this, 'copy_og_settings_from_social_to_titles' ], 99 );
 | ||
| 
 | ||
| 		// Run after the WPSEO_Options::enrich_defaults method which has priority 99.
 | ||
| 		add_action( 'init', [ $this, 'reset_og_settings_to_default_values' ], 100 );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 17.2 upgrade. Cleans out any unnecessary indexables. See $cleanup_integration->get_cleanup_tasks() to see what will be cleaned out.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function upgrade_172() {
 | ||
| 		\wp_unschedule_hook( 'wpseo_cleanup_orphaned_indexables' );
 | ||
| 		\wp_unschedule_hook( 'wpseo_cleanup_indexables' );
 | ||
| 
 | ||
| 		if ( ! \wp_next_scheduled( Cleanup_Integration::START_HOOK ) ) {
 | ||
| 			\wp_schedule_single_event( ( time() + ( MINUTE_IN_SECONDS * 5 ) ), Cleanup_Integration::START_HOOK );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 17.7.1 upgrade routine.
 | ||
| 	 */
 | ||
| 	private function upgrade_1771() {
 | ||
| 		$enabled_auto_updates = \get_site_option( 'auto_update_plugins' );
 | ||
| 		$addon_update_watcher = YoastSEO()->classes->get( \Yoast\WP\SEO\Integrations\Watchers\Addon_Update_Watcher::class );
 | ||
| 		$addon_update_watcher->toggle_auto_updates_for_add_ons( 'auto_update_plugins', $enabled_auto_updates, [] );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 17.9 upgrade routine.
 | ||
| 	 */
 | ||
| 	private function upgrade_179() {
 | ||
| 		WPSEO_Options::set( 'wincher_integration_active', true );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 18.3 upgrade routine.
 | ||
| 	 */
 | ||
| 	private function upgrade_183() {
 | ||
| 		$this->delete_post_meta( 'yoast-structured-data-blocks-images-cache' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 18.6 upgrade routine.
 | ||
| 	 */
 | ||
| 	private function upgrade_186() {
 | ||
| 		if ( is_multisite() ) {
 | ||
| 			WPSEO_Options::set( 'allow_wincher_integration_active', false );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 18.9 upgrade routine.
 | ||
| 	 */
 | ||
| 	private function upgrade_189() {
 | ||
| 		// Make old users not get the Installation Success page after upgrading.
 | ||
| 		WPSEO_Options::set( 'should_redirect_after_install_free', false );
 | ||
| 		// We're adding a hardcoded time here, so that in the future we can be able to identify whether the user did see the Installation Success page or not.
 | ||
| 		// If they did, they wouldn't have this hardcoded value in that option, but rather (roughly) the timestamp of the moment they saw it.
 | ||
| 		WPSEO_Options::set( 'activation_redirect_timestamp_free', 1652258756 );
 | ||
| 
 | ||
| 		// Transfer the Social URLs.
 | ||
| 		$other   = [];
 | ||
| 		$other[] = WPSEO_Options::get( 'instagram_url' );
 | ||
| 		$other[] = WPSEO_Options::get( 'linkedin_url' );
 | ||
| 		$other[] = WPSEO_Options::get( 'myspace_url' );
 | ||
| 		$other[] = WPSEO_Options::get( 'pinterest_url' );
 | ||
| 		$other[] = WPSEO_Options::get( 'youtube_url' );
 | ||
| 		$other[] = WPSEO_Options::get( 'wikipedia_url' );
 | ||
| 
 | ||
| 		WPSEO_Options::set( 'other_social_urls', array_values( array_unique( array_filter( $other ) ) ) );
 | ||
| 
 | ||
| 		// Transfer the progress of the old Configuration Workout.
 | ||
| 		$workout_data      = WPSEO_Options::get( 'workouts_data' );
 | ||
| 		$old_conf_progress = isset( $workout_data['configuration']['finishedSteps'] ) ? $workout_data['configuration']['finishedSteps'] : [];
 | ||
| 
 | ||
| 		if ( in_array( 'optimizeSeoData', $old_conf_progress, true ) && in_array( 'siteRepresentation', $old_conf_progress, true ) ) {
 | ||
| 			// If completed ‘SEO optimization’ and ‘Site representation’ step, we assume the workout was completed.
 | ||
| 			$configuration_finished_steps = [
 | ||
| 				'siteRepresentation',
 | ||
| 				'socialProfiles',
 | ||
| 				'personalPreferences',
 | ||
| 			];
 | ||
| 			WPSEO_Options::set( 'configuration_finished_steps', $configuration_finished_steps );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 19.1 upgrade routine.
 | ||
| 	 */
 | ||
| 	private function upgrade_191() {
 | ||
| 		if ( is_multisite() ) {
 | ||
| 			WPSEO_Options::set( 'allow_remove_feed_post_comments', true );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 19.3 upgrade routine.
 | ||
| 	 */
 | ||
| 	private function upgrade_193() {
 | ||
| 		if ( empty( get_option( 'wpseo_premium', [] ) ) ) {
 | ||
| 			WPSEO_Options::set( 'enable_index_now', true );
 | ||
| 			WPSEO_Options::set( 'enable_link_suggestions', true );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 19.6 upgrade routine.
 | ||
| 	 */
 | ||
| 	private function upgrade_196() {
 | ||
| 		WPSEO_Options::set( 'ryte_indexability', false );
 | ||
| 		WPSEO_Options::set( 'allow_ryte_indexability', false );
 | ||
| 		wp_clear_scheduled_hook( 'wpseo_ryte_fetch' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 19.11 upgrade routine.
 | ||
| 	 */
 | ||
| 	private function upgrade_1911() {
 | ||
| 		\add_action( 'shutdown', [ $this, 'remove_indexable_rows_for_non_public_post_types' ] );
 | ||
| 		\add_action( 'shutdown', [ $this, 'remove_indexable_rows_for_non_public_taxonomies' ] );
 | ||
| 		$this->deduplicate_unindexed_indexable_rows();
 | ||
| 		$this->remove_indexable_rows_for_disabled_authors_archive();
 | ||
| 		if ( ! \wp_next_scheduled( Cleanup_Integration::START_HOOK ) ) {
 | ||
| 			\wp_schedule_single_event( ( time() + ( MINUTE_IN_SECONDS * 5 ) ), Cleanup_Integration::START_HOOK );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 20.2 upgrade routine.
 | ||
| 	 */
 | ||
| 	private function upgrade_202() {
 | ||
| 		if ( WPSEO_Options::get( 'disable-attachment', true ) ) {
 | ||
| 			$attachment_cleanup_helper = YoastSEO()->helpers->attachment_cleanup;
 | ||
| 
 | ||
| 			$attachment_cleanup_helper->remove_attachment_indexables( true );
 | ||
| 			$attachment_cleanup_helper->clean_attachment_links_from_target_indexable_ids( true );
 | ||
| 		}
 | ||
| 
 | ||
| 		$this->clean_unindexed_indexable_rows_with_no_object_id();
 | ||
| 
 | ||
| 		if ( ! \wp_next_scheduled( Cleanup_Integration::START_HOOK ) ) {
 | ||
| 			// This schedules the cleanup routine cron again, since in combination of premium cleans up the prominent words table. We also want to cleanup possible orphaned hierarchies from the above cleanups.
 | ||
| 			\wp_schedule_single_event( ( time() + ( MINUTE_IN_SECONDS * 5 ) ), Cleanup_Integration::START_HOOK );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 20.5 upgrade routine.
 | ||
| 	 */
 | ||
| 	private function upgrade_205() {
 | ||
| 		if ( ! \wp_next_scheduled( Cleanup_Integration::START_HOOK ) ) {
 | ||
| 			\wp_schedule_single_event( ( time() + ( MINUTE_IN_SECONDS * 5 ) ), Cleanup_Integration::START_HOOK );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 20.7 upgrade routine.
 | ||
| 	 * Removes the metadata related to the settings page introduction modal for all the users.
 | ||
| 	 * Also, schedules another cleanup scheduled action.
 | ||
| 	 */
 | ||
| 	private function upgrade_207() {
 | ||
| 		add_action( 'shutdown', [ $this, 'delete_user_introduction_meta' ] );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Performs the 20.8 upgrade routine.
 | ||
| 	 * Schedules another cleanup scheduled action.
 | ||
| 	 */
 | ||
| 	private function upgrade_208() {
 | ||
| 		if ( ! \wp_next_scheduled( Cleanup_Integration::START_HOOK ) ) {
 | ||
| 			\wp_schedule_single_event( ( time() + ( MINUTE_IN_SECONDS * 5 ) ), Cleanup_Integration::START_HOOK );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Sets the home_url option for the 15.1 upgrade routine.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	protected function set_home_url_for_151() {
 | ||
| 		$home_url = WPSEO_Options::get( 'home_url' );
 | ||
| 
 | ||
| 		if ( empty( $home_url ) ) {
 | ||
| 			WPSEO_Options::set( 'home_url', get_home_url() );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Moves the `indexables_indexation_reason` option to the
 | ||
| 	 * renamed `indexing_reason` option.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	protected function move_indexables_indexation_reason_for_151() {
 | ||
| 		$reason = WPSEO_Options::get( 'indexables_indexation_reason', '' );
 | ||
| 		WPSEO_Options::set( 'indexing_reason', $reason );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Checks if the indexable indexation is completed.
 | ||
| 	 * If so, sets the `indexables_indexation_completed` option to `true`,
 | ||
| 	 * else to `false`.
 | ||
| 	 */
 | ||
| 	public function set_indexation_completed_option_for_145() {
 | ||
| 		WPSEO_Options::set( 'indexables_indexation_completed', YoastSEO()->helpers->indexing->get_limited_filtered_unindexed_count( 1 ) === 0 );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Cleans up the private taxonomies from the indexables table for the upgrade routine to 14.1.
 | ||
| 	 */
 | ||
| 	public function clean_up_private_taxonomies_for_141() {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		// If migrations haven't been completed successfully the following may give false errors. So suppress them.
 | ||
| 		$show_errors       = $wpdb->show_errors;
 | ||
| 		$wpdb->show_errors = false;
 | ||
| 
 | ||
| 		// Clean up indexables of private taxonomies.
 | ||
| 		$private_taxonomies = \get_taxonomies( [ 'public' => false ], 'names' );
 | ||
| 
 | ||
| 		if ( empty( $private_taxonomies ) ) {
 | ||
| 			return;
 | ||
| 		}
 | ||
| 
 | ||
| 		$indexable_table = Model::get_table_name( 'Indexable' );
 | ||
| 
 | ||
| 		// phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared -- Reason: Is it prepared already.
 | ||
| 		$query = $wpdb->prepare(
 | ||
| 			// phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared -- Reason: Too hard to fix.
 | ||
| 			"DELETE FROM $indexable_table
 | ||
| 			WHERE object_type = 'term'
 | ||
| 			AND object_sub_type IN ("
 | ||
| 				. \implode( ', ', \array_fill( 0, \count( $private_taxonomies ), '%s' ) )
 | ||
| 				. ')',
 | ||
| 			$private_taxonomies
 | ||
| 		);
 | ||
| 		$wpdb->query( $query ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared -- Reason: Is it prepared already.
 | ||
| 
 | ||
| 		$wpdb->show_errors = $show_errors;
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Resets the permalinks of attachments to `null` in the indexable table for the upgrade routine to 14.1.
 | ||
| 	 */
 | ||
| 	private function reset_permalinks_of_attachments_for_141() {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		// If migrations haven't been completed succesfully the following may give false errors. So suppress them.
 | ||
| 		$show_errors       = $wpdb->show_errors;
 | ||
| 		$wpdb->show_errors = false;
 | ||
| 
 | ||
| 		// Reset the permalinks of the attachments in the indexable table.
 | ||
| 		$indexable_table = Model::get_table_name( 'Indexable' );
 | ||
| 		$query           = "UPDATE $indexable_table SET permalink = NULL WHERE object_type = 'post' AND object_sub_type = 'attachment'";
 | ||
| 		$wpdb->query( $query ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared -- Reason: There is no user input.
 | ||
| 
 | ||
| 		$wpdb->show_errors = $show_errors;
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes notifications from the Notification center for the 14.1 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	public function remove_notifications_for_141() {
 | ||
| 		Yoast_Notification_Center::get()->remove_notification_by_id( 'wpseo-dismiss-recalculate' );
 | ||
| 		Yoast_Notification_Center::get()->remove_notification_by_id( 'wpseo-dismiss-blog-public-notice' );
 | ||
| 		Yoast_Notification_Center::get()->remove_notification_by_id( 'wpseo-links-table-not-accessible' );
 | ||
| 		Yoast_Notification_Center::get()->remove_notification_by_id( 'wpseo-post-type-archive-notification' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes the wpseo-suggested-plugin-yoast-acf-analysis notification from the Notification center for the 14.2 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	public function remove_acf_notification_for_142() {
 | ||
| 		Yoast_Notification_Center::get()->remove_notification_by_id( 'wpseo-suggested-plugin-yoast-acf-analysis' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes the wpseo-plugin-updated notification from the Notification center for the 15.7 upgrade.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	public function remove_plugin_updated_notification_for_157() {
 | ||
| 		Yoast_Notification_Center::get()->remove_notification_by_id( 'wpseo-plugin-updated' );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes all notifications saved in the database under 'wp_yoast_notifications'.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function clean_all_notifications() {
 | ||
| 		global $wpdb;
 | ||
| 		delete_metadata( 'user', 0, $wpdb->get_blog_prefix() . Yoast_Notification_Center::STORAGE_KEY, '', true );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes the post meta fields for a given meta key.
 | ||
| 	 *
 | ||
| 	 * @param string $meta_key The meta key.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function delete_post_meta( $meta_key ) {
 | ||
| 		global $wpdb;
 | ||
| 		$deleted = $wpdb->delete( $wpdb->postmeta, [ 'meta_key' => $meta_key ], [ '%s' ] );
 | ||
| 
 | ||
| 		if ( $deleted ) {
 | ||
| 			wp_cache_set( 'last_changed', microtime(), 'posts' );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes all sitemap validators.
 | ||
| 	 *
 | ||
| 	 * This should be executed on every upgrade routine until we have removed the sitemap caching in the database.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function remove_sitemap_validators() {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		// Remove all sitemap validators.
 | ||
| 		$wpdb->query( "DELETE FROM $wpdb->options WHERE option_name LIKE 'wpseo_sitemap%validator%'" );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Retrieves the option value directly from the database.
 | ||
| 	 *
 | ||
| 	 * @param string $option_name Option to retrieve.
 | ||
| 	 *
 | ||
| 	 * @return array|mixed The content of the option if exists, otherwise an empty array.
 | ||
| 	 */
 | ||
| 	protected function get_option_from_database( $option_name ) {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		// Load option directly from the database, to avoid filtering and sanitization.
 | ||
| 		$sql     = $wpdb->prepare( 'SELECT option_value FROM ' . $wpdb->options . ' WHERE option_name = %s', $option_name );
 | ||
| 		$results = $wpdb->get_results( $sql, ARRAY_A ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared -- Reason: Is is already prepared.
 | ||
| 		if ( ! empty( $results ) ) {
 | ||
| 			return maybe_unserialize( $results[0]['option_value'] );
 | ||
| 		}
 | ||
| 
 | ||
| 		return [];
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Cleans the option to make sure only relevant settings are there.
 | ||
| 	 *
 | ||
| 	 * @param string $option_name Option name save.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	protected function cleanup_option_data( $option_name ) {
 | ||
| 		$data = get_option( $option_name, [] );
 | ||
| 		if ( ! is_array( $data ) || $data === [] ) {
 | ||
| 			return;
 | ||
| 		}
 | ||
| 
 | ||
| 		/*
 | ||
| 		 * Clean up the option by re-saving it.
 | ||
| 		 *
 | ||
| 		 * The option framework will remove any settings that are not configured
 | ||
| 		 * for this option, removing any migrated settings.
 | ||
| 		 */
 | ||
| 		update_option( $option_name, $data );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Saves an option setting to where it should be stored.
 | ||
| 	 *
 | ||
| 	 * @param array       $source_data    The option containing the value to be migrated.
 | ||
| 	 * @param string      $source_setting Name of the key in the "from" option.
 | ||
| 	 * @param string|null $target_setting Name of the key in the "to" option.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	protected function save_option_setting( $source_data, $source_setting, $target_setting = null ) {
 | ||
| 		if ( $target_setting === null ) {
 | ||
| 			$target_setting = $source_setting;
 | ||
| 		}
 | ||
| 
 | ||
| 		if ( isset( $source_data[ $source_setting ] ) ) {
 | ||
| 			WPSEO_Options::set( $target_setting, $source_data[ $source_setting ] );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Migrates WooCommerce archive settings to the WooCommerce Shop page meta-data settings.
 | ||
| 	 *
 | ||
| 	 * If no Shop page is defined, nothing will be migrated.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function migrate_woocommerce_archive_setting_to_shop_page() {
 | ||
| 		$shop_page_id = wc_get_page_id( 'shop' );
 | ||
| 
 | ||
| 		if ( $shop_page_id === -1 ) {
 | ||
| 			return;
 | ||
| 		}
 | ||
| 
 | ||
| 		$title = WPSEO_Meta::get_value( 'title', $shop_page_id );
 | ||
| 
 | ||
| 		if ( empty( $title ) ) {
 | ||
| 			$option_title = WPSEO_Options::get( 'title-ptarchive-product' );
 | ||
| 
 | ||
| 			WPSEO_Meta::set_value(
 | ||
| 				'title',
 | ||
| 				$option_title,
 | ||
| 				$shop_page_id
 | ||
| 			);
 | ||
| 
 | ||
| 			WPSEO_Options::set( 'title-ptarchive-product', '' );
 | ||
| 		}
 | ||
| 
 | ||
| 		$meta_description = WPSEO_Meta::get_value( 'metadesc', $shop_page_id );
 | ||
| 
 | ||
| 		if ( empty( $meta_description ) ) {
 | ||
| 			$option_metadesc = WPSEO_Options::get( 'metadesc-ptarchive-product' );
 | ||
| 
 | ||
| 			WPSEO_Meta::set_value(
 | ||
| 				'metadesc',
 | ||
| 				$option_metadesc,
 | ||
| 				$shop_page_id
 | ||
| 			);
 | ||
| 
 | ||
| 			WPSEO_Options::set( 'metadesc-ptarchive-product', '' );
 | ||
| 		}
 | ||
| 
 | ||
| 		$bc_title = WPSEO_Meta::get_value( 'bctitle', $shop_page_id );
 | ||
| 
 | ||
| 		if ( empty( $bc_title ) ) {
 | ||
| 			$option_bctitle = WPSEO_Options::get( 'bctitle-ptarchive-product' );
 | ||
| 
 | ||
| 			WPSEO_Meta::set_value(
 | ||
| 				'bctitle',
 | ||
| 				$option_bctitle,
 | ||
| 				$shop_page_id
 | ||
| 			);
 | ||
| 
 | ||
| 			WPSEO_Options::set( 'bctitle-ptarchive-product', '' );
 | ||
| 		}
 | ||
| 
 | ||
| 		$noindex = WPSEO_Meta::get_value( 'meta-robots-noindex', $shop_page_id );
 | ||
| 
 | ||
| 		if ( $noindex === '0' ) {
 | ||
| 			$option_noindex = WPSEO_Options::get( 'noindex-ptarchive-product' );
 | ||
| 
 | ||
| 			WPSEO_Meta::set_value(
 | ||
| 				'meta-robots-noindex',
 | ||
| 				$option_noindex,
 | ||
| 				$shop_page_id
 | ||
| 			);
 | ||
| 
 | ||
| 			WPSEO_Options::set( 'noindex-ptarchive-product', false );
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Stores the initial `permalink_structure` option.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	public function set_permalink_structure_option_for_151() {
 | ||
| 		WPSEO_Options::set( 'permalink_structure', get_option( 'permalink_structure' ) );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Stores the initial slugs of custom taxonomies.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	public function store_custom_taxonomy_slugs_for_151() {
 | ||
| 		$taxonomies = $this->taxonomy_helper->get_custom_taxonomies();
 | ||
| 
 | ||
| 		$custom_taxonomies = [];
 | ||
| 
 | ||
| 		foreach ( $taxonomies as $taxonomy ) {
 | ||
| 			$slug = $this->taxonomy_helper->get_taxonomy_slug( $taxonomy );
 | ||
| 
 | ||
| 			$custom_taxonomies[ $taxonomy ] = $slug;
 | ||
| 		}
 | ||
| 
 | ||
| 		WPSEO_Options::set( 'custom_taxonomy_slugs', $custom_taxonomies );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Copies the frontpage social settings to the titles options.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	public function copy_og_settings_from_social_to_titles() {
 | ||
| 		$wpseo_social = get_option( 'wpseo_social' );
 | ||
| 		$wpseo_titles = get_option( 'wpseo_titles' );
 | ||
| 
 | ||
| 		$copied_options = [];
 | ||
| 		// Reset to the correct default value.
 | ||
| 		$copied_options['open_graph_frontpage_title'] = '%%sitename%%';
 | ||
| 
 | ||
| 		$options = [
 | ||
| 			'og_frontpage_title'    => 'open_graph_frontpage_title',
 | ||
| 			'og_frontpage_desc'     => 'open_graph_frontpage_desc',
 | ||
| 			'og_frontpage_image'    => 'open_graph_frontpage_image',
 | ||
| 			'og_frontpage_image_id' => 'open_graph_frontpage_image_id',
 | ||
| 		];
 | ||
| 
 | ||
| 		foreach ( $options as $social_option => $titles_option ) {
 | ||
| 			if ( ! empty( $wpseo_social[ $social_option ] ) ) {
 | ||
| 				$copied_options[ $titles_option ] = $wpseo_social[ $social_option ];
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		$wpseo_titles = array_merge( $wpseo_titles, $copied_options );
 | ||
| 
 | ||
| 		update_option( 'wpseo_titles', $wpseo_titles );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Reset the social options with the correct default values.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	public function reset_og_settings_to_default_values() {
 | ||
| 		$wpseo_titles    = get_option( 'wpseo_titles' );
 | ||
| 		$updated_options = [];
 | ||
| 
 | ||
| 		$updated_options['social-title-author-wpseo']  = '%%name%%';
 | ||
| 		$updated_options['social-title-archive-wpseo'] = '%%date%%';
 | ||
| 
 | ||
| 		/* translators: %s expands to the name of a post type (plural). */
 | ||
| 		$post_type_archive_default = sprintf( __( '%s Archive', 'wordpress-seo' ), '%%pt_plural%%' );
 | ||
| 
 | ||
| 		/* translators: %s expands to the variable used for term title. */
 | ||
| 		$term_archive_default = sprintf( __( '%s Archives', 'wordpress-seo' ), '%%term_title%%' );
 | ||
| 
 | ||
| 		$post_type_objects = get_post_types( [ 'public' => true ], 'objects' );
 | ||
| 
 | ||
| 		if ( $post_type_objects ) {
 | ||
| 			foreach ( $post_type_objects as $pt ) {
 | ||
| 				// Post types.
 | ||
| 				if ( isset( $wpseo_titles[ 'social-title-' . $pt->name ] ) ) {
 | ||
| 					$updated_options[ 'social-title-' . $pt->name ] = '%%title%%';
 | ||
| 				}
 | ||
| 				// Post type archives.
 | ||
| 				if ( isset( $wpseo_titles[ 'social-title-ptarchive-' . $pt->name ] ) ) {
 | ||
| 					$updated_options[ 'social-title-ptarchive-' . $pt->name ] = $post_type_archive_default;
 | ||
| 				}
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		$taxonomy_objects = get_taxonomies( [ 'public' => true ], 'object' );
 | ||
| 
 | ||
| 		if ( $taxonomy_objects ) {
 | ||
| 			foreach ( $taxonomy_objects as $tax ) {
 | ||
| 				if ( isset( $wpseo_titles[ 'social-title-tax-' . $tax->name ] ) ) {
 | ||
| 					$updated_options[ 'social-title-tax-' . $tax->name ] = $term_archive_default;
 | ||
| 				}
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		$wpseo_titles = array_merge( $wpseo_titles, $updated_options );
 | ||
| 
 | ||
| 		update_option( 'wpseo_titles', $wpseo_titles );
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes all indexables for posts that are not publicly viewable.
 | ||
| 	 * This method should be called after init, because post_types can still be registered.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	public function remove_indexable_rows_for_non_public_post_types() {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		// If migrations haven't been completed successfully the following may give false errors. So suppress them.
 | ||
| 		$show_errors       = $wpdb->show_errors;
 | ||
| 		$wpdb->show_errors = false;
 | ||
| 
 | ||
| 		$indexable_table = Model::get_table_name( 'Indexable' );
 | ||
| 
 | ||
| 		$included_post_types = \YoastSEO()->helpers->post_type->get_indexable_post_types();
 | ||
| 
 | ||
| 		// phpcs:disable WordPress.DB.PreparedSQL.InterpolatedNotPrepared -- Reason: Too hard to fix.
 | ||
| 		if ( empty( $included_post_types ) ) {
 | ||
| 			$delete_query = "
 | ||
| 				DELETE FROM $indexable_table
 | ||
| 				WHERE object_type = 'post'
 | ||
| 				AND object_sub_type IS NOT NULL";
 | ||
| 		}
 | ||
| 		else {
 | ||
| 			$delete_query = $wpdb->prepare(
 | ||
| 				"DELETE FROM $indexable_table
 | ||
| 				WHERE object_type = 'post'
 | ||
| 				AND object_sub_type IS NOT NULL
 | ||
| 				AND object_sub_type NOT IN ( " . \implode( ', ', \array_fill( 0, \count( $included_post_types ), '%s' ) ) . ' )',
 | ||
| 				$included_post_types
 | ||
| 			);
 | ||
| 		}
 | ||
| 		// phpcs:enable
 | ||
| 
 | ||
| 		// phpcs:disable WordPress.DB.DirectDatabaseQuery.DirectQuery -- Reason: Most performant way.
 | ||
| 		// phpcs:disable WordPress.DB.DirectDatabaseQuery.NoCaching -- Reason: No relevant caches.
 | ||
| 		// phpcs:disable WordPress.DB.PreparedSQL.NotPrepared -- Reason: Is it prepared already.
 | ||
| 		$wpdb->query( $delete_query );
 | ||
| 		// phpcs:enable
 | ||
| 
 | ||
| 		$wpdb->show_errors = $show_errors;
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes all indexables for terms that are not publicly viewable.
 | ||
| 	 * This method should be called after init, because taxonomies can still be registered.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	public function remove_indexable_rows_for_non_public_taxonomies() {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		// If migrations haven't been completed successfully the following may give false errors. So suppress them.
 | ||
| 		$show_errors       = $wpdb->show_errors;
 | ||
| 		$wpdb->show_errors = false;
 | ||
| 
 | ||
| 		$indexable_table = Model::get_table_name( 'Indexable' );
 | ||
| 
 | ||
| 		$included_taxonomies = \YoastSEO()->helpers->taxonomy->get_indexable_taxonomies();
 | ||
| 
 | ||
| 		// phpcs:disable WordPress.DB.PreparedSQL.InterpolatedNotPrepared -- Reason: Too hard to fix.
 | ||
| 		if ( empty( $included_taxonomies ) ) {
 | ||
| 			$delete_query = "DELETE FROM $indexable_table
 | ||
| 				WHERE object_type = 'term'
 | ||
| 				AND object_sub_type IS NOT NULL";
 | ||
| 		}
 | ||
| 		else {
 | ||
| 			$delete_query = $wpdb->prepare(
 | ||
| 				"DELETE FROM $indexable_table
 | ||
| 				WHERE object_type = 'term'
 | ||
| 				AND object_sub_type IS NOT NULL
 | ||
| 				AND object_sub_type NOT IN ( " . \implode( ', ', \array_fill( 0, \count( $included_taxonomies ), '%s' ) ) . ' )',
 | ||
| 				$included_taxonomies
 | ||
| 			);
 | ||
| 		}
 | ||
| 		// phpcs:enable
 | ||
| 
 | ||
| 		// phpcs:disable WordPress.DB.DirectDatabaseQuery.DirectQuery -- Reason: Most performant way.
 | ||
| 		// phpcs:disable WordPress.DB.DirectDatabaseQuery.NoCaching -- Reason: No relevant caches.
 | ||
| 		// phpcs:disable WordPress.DB.PreparedSQL.NotPrepared -- Reason: Is it prepared already.
 | ||
| 		$wpdb->query( $delete_query );
 | ||
| 		// phpcs:enable
 | ||
| 
 | ||
| 		$wpdb->show_errors = $show_errors;
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * De-duplicates indexables that have more than one "unindexed" rows for the same object. Keeps the newest indexable.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function deduplicate_unindexed_indexable_rows() {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		// If migrations haven't been completed successfully the following may give false errors. So suppress them.
 | ||
| 		$show_errors       = $wpdb->show_errors;
 | ||
| 		$wpdb->show_errors = false;
 | ||
| 
 | ||
| 		$indexable_table = Model::get_table_name( 'Indexable' );
 | ||
| 
 | ||
| 		$query = "
 | ||
| 			SELECT
 | ||
| 				MAX(id) as newest_id,
 | ||
| 				object_id,
 | ||
| 				object_type
 | ||
| 			FROM
 | ||
| 				$indexable_table
 | ||
| 			WHERE
 | ||
| 				post_status = 'unindexed'
 | ||
| 				AND object_type IN ( 'term', 'post', 'user' )
 | ||
| 			GROUP BY
 | ||
| 				object_id,
 | ||
| 				object_type
 | ||
| 			HAVING
 | ||
| 				count(*) > 1";
 | ||
| 
 | ||
| 		// phpcs:disable WordPress.DB.DirectDatabaseQuery.DirectQuery -- Reason: Most performant way.
 | ||
| 		// phpcs:disable WordPress.DB.DirectDatabaseQuery.NoCaching -- Reason: No relevant caches.
 | ||
| 		// phpcs:disable WordPress.DB.PreparedSQL.NotPrepared -- Reason: Is it prepared already.
 | ||
| 		$duplicates = $wpdb->get_results( $query, ARRAY_A );
 | ||
| 		// phpcs:enable
 | ||
| 
 | ||
| 		if ( empty( $duplicates ) ) {
 | ||
| 			$wpdb->show_errors = $show_errors;
 | ||
| 
 | ||
| 			return;
 | ||
| 		}
 | ||
| 
 | ||
| 		// Users, terms and posts may share the same object_id. So delete them in separate, more performant, queries.
 | ||
| 		$delete_queries = [
 | ||
| 			$this->get_indexable_deduplication_query_for_type( 'post', $duplicates, $wpdb ),
 | ||
| 			$this->get_indexable_deduplication_query_for_type( 'term', $duplicates, $wpdb ),
 | ||
| 			$this->get_indexable_deduplication_query_for_type( 'user', $duplicates, $wpdb ),
 | ||
| 		];
 | ||
| 
 | ||
| 		foreach ( $delete_queries as $delete_query ) {
 | ||
| 			if ( ! empty( $delete_query ) ) {
 | ||
| 				// phpcs:disable WordPress.DB.DirectDatabaseQuery.DirectQuery -- Reason: Most performant way.
 | ||
| 				// phpcs:disable WordPress.DB.DirectDatabaseQuery.NoCaching -- Reason: No relevant caches.
 | ||
| 				// phpcs:disable WordPress.DB.PreparedSQL.NotPrepared -- Reason: Is it prepared already.
 | ||
| 				$wpdb->query( $delete_query );
 | ||
| 				// phpcs:enable
 | ||
| 			}
 | ||
| 		}
 | ||
| 
 | ||
| 		$wpdb->show_errors = $show_errors;
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Cleans up "unindexed" indexable rows when appropriate, aka when there's no object ID even though it should.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function clean_unindexed_indexable_rows_with_no_object_id() {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		// If migrations haven't been completed successfully the following may give false errors. So suppress them.
 | ||
| 		$show_errors       = $wpdb->show_errors;
 | ||
| 		$wpdb->show_errors = false;
 | ||
| 
 | ||
| 		$indexable_table = Model::get_table_name( 'Indexable' );
 | ||
| 
 | ||
| 		// phpcs:disable WordPress.DB.PreparedSQL.InterpolatedNotPrepared -- Reason: No user input, just a table name.
 | ||
| 		// phpcs:disable WordPress.DB.DirectDatabaseQuery.DirectQuery -- Reason: Most performant way.
 | ||
| 		// phpcs:disable WordPress.DB.DirectDatabaseQuery.NoCaching -- Reason: No relevant caches.
 | ||
| 		$wpdb->query(
 | ||
| 			"DELETE FROM $indexable_table
 | ||
| 			WHERE post_status = 'unindexed'
 | ||
| 			AND object_type NOT IN ( 'home-page', 'date-archive', 'post-type-archive', 'system-page' )
 | ||
| 			AND object_id IS NULL"
 | ||
| 		);
 | ||
| 		// phpcs:enable
 | ||
| 
 | ||
| 		$wpdb->show_errors = $show_errors;
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes all user indexable rows when the author archive is disabled.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	private function remove_indexable_rows_for_disabled_authors_archive() {
 | ||
| 		global $wpdb;
 | ||
| 
 | ||
| 		if ( ! \YoastSEO()->helpers->author_archive->are_disabled() ) {
 | ||
| 			return;
 | ||
| 		}
 | ||
| 
 | ||
| 		// If migrations haven't been completed successfully the following may give false errors. So suppress them.
 | ||
| 		$show_errors       = $wpdb->show_errors;
 | ||
| 		$wpdb->show_errors = false;
 | ||
| 
 | ||
| 		$indexable_table = Model::get_table_name( 'Indexable' );
 | ||
| 
 | ||
| 		$delete_query = "DELETE FROM $indexable_table WHERE object_type = 'user'";
 | ||
| 		// phpcs:enable
 | ||
| 
 | ||
| 		// phpcs:disable WordPress.DB.DirectDatabaseQuery.NoCaching -- Reason: No relevant caches.
 | ||
| 		// phpcs:disable WordPress.DB.DirectDatabaseQuery.DirectQuery -- Reason: Most performant way.
 | ||
| 		// phpcs:disable WordPress.DB.PreparedSQL.NotPrepared -- Reason: Is it prepared already.
 | ||
| 		$wpdb->query( $delete_query );
 | ||
| 		// phpcs:enable
 | ||
| 
 | ||
| 		$wpdb->show_errors = $show_errors;
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Creates a query for de-duplicating indexables for a particular type.
 | ||
| 	 *
 | ||
| 	 * @param string $object_type The object type to deduplicate.
 | ||
| 	 * @param array  $duplicates  The result of the duplicate query.
 | ||
| 	 * @param wpdb   $wpdb        The wpdb object.
 | ||
| 	 *
 | ||
| 	 * @return string The query that removes all but one duplicate for each object of the object type.
 | ||
| 	 */
 | ||
| 	private function get_indexable_deduplication_query_for_type( $object_type, $duplicates, $wpdb ) {
 | ||
| 		$indexable_table = Model::get_table_name( 'Indexable' );
 | ||
| 
 | ||
| 		$filtered_duplicates = \array_filter(
 | ||
| 			$duplicates,
 | ||
| 			static function ( $duplicate ) use ( $object_type ) {
 | ||
| 				return $duplicate['object_type'] === $object_type;
 | ||
| 			}
 | ||
| 		);
 | ||
| 
 | ||
| 		if ( empty( $filtered_duplicates ) ) {
 | ||
| 			return '';
 | ||
| 		}
 | ||
| 
 | ||
| 		$object_ids           = wp_list_pluck( $filtered_duplicates, 'object_id' );
 | ||
| 		$newest_indexable_ids = wp_list_pluck( $filtered_duplicates, 'newest_id' );
 | ||
| 
 | ||
| 		// phpcs:disable WordPress.DB.PreparedSQL.InterpolatedNotPrepared -- Reason: Too hard to fix.
 | ||
| 		// phpcs:disable WordPress.DB.PreparedSQLPlaceholders.ReplacementsWrongNumber -- Reason: we're passing an array instead.
 | ||
| 		return $wpdb->prepare(
 | ||
| 			"DELETE FROM
 | ||
| 				$indexable_table
 | ||
| 			WHERE
 | ||
| 				object_id IN ( " . \implode( ', ', \array_fill( 0, \count( $filtered_duplicates ), '%d' ) ) . ' )
 | ||
| 				AND id NOT IN ( ' . \implode( ', ', \array_fill( 0, \count( $filtered_duplicates ), '%d' ) ) . ' )
 | ||
| 				AND object_type = %s',
 | ||
| 			array_merge( array_values( $object_ids ), array_values( $newest_indexable_ids ), [ $object_type ] )
 | ||
| 		);
 | ||
| 		// phpcs:enable
 | ||
| 	}
 | ||
| 
 | ||
| 	/**
 | ||
| 	 * Removes the settings' introduction modal data for users.
 | ||
| 	 *
 | ||
| 	 * @return void
 | ||
| 	 */
 | ||
| 	public function delete_user_introduction_meta() {
 | ||
| 		delete_metadata( 'user', 0, '_yoast_settings_introduction', '', true );
 | ||
| 	}
 | ||
| }
 |