1792 lines
55 KiB
PHP
1792 lines
55 KiB
PHP
<?php
|
|
namespace AIOSEO\Plugin\Pro\Main;
|
|
|
|
// Exit if accessed directly.
|
|
if ( ! defined( 'ABSPATH' ) ) {
|
|
exit;
|
|
}
|
|
|
|
use AIOSEO\Plugin\Common\Main as CommonMain;
|
|
use AIOSEO\Plugin\Common\Models as CommonModels;
|
|
use AIOSEO\Plugin\Pro\Models as ProModels;
|
|
|
|
/**
|
|
* Updater class.
|
|
*
|
|
* @since 4.0.0
|
|
*/
|
|
class Updates extends CommonMain\Updates {
|
|
/**
|
|
* Class constructor.
|
|
*
|
|
* @since 4.0.5
|
|
*/
|
|
public function __construct() {
|
|
parent::__construct();
|
|
|
|
add_action( 'aioseo_loaded', [ $this, 'runPreAddonUpdates' ], 1 );
|
|
add_action( 'aioseo_v4_migrate_post_og_image', [ $this, 'migratePostOgImage' ] );
|
|
add_action( 'aioseo_v4_migrate_term_og_image', [ $this, 'migrateTermOgImage' ] );
|
|
add_action( 'aioseo_v432_unschedule_objects_scan', [ $this, 'cancelDuplicateObjectsActions' ] );
|
|
}
|
|
|
|
/**
|
|
* Run pre-addon updates/migrations.
|
|
* This runs before our regular migrations, which run on "init".
|
|
*
|
|
* @since 4.1.7
|
|
* @version 4.4.0
|
|
*
|
|
* @return void
|
|
*/
|
|
public function runPreAddonUpdates() {
|
|
$lastActiveProVersion = aioseo()->internalOptions->internal->lastActiveProVersion;
|
|
|
|
// If the last active version hasn't been set and the user has license data, set it to the current version.
|
|
// This ensures that we don't rerun the migrations for existing users.
|
|
if (
|
|
! $lastActiveProVersion &&
|
|
( aioseo()->internalOptions->internal->license->level || aioseo()->internalOptions->internal->license->expires )
|
|
) {
|
|
aioseo()->internalOptions->internal->lastActiveProVersion = AIOSEO_VERSION;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Runs our migrations.
|
|
*
|
|
* @since 4.0.0
|
|
* @version 4.4.0
|
|
*
|
|
* @return void
|
|
*/
|
|
public function runUpdates() {
|
|
$lastActiveProVersion = aioseo()->internalOptions->internal->lastActiveProVersion;
|
|
// Don't run updates if the last active version is the same as the current version.
|
|
if ( aioseo()->version === $lastActiveProVersion ) {
|
|
// Allow addons to run their updates.
|
|
do_action( 'aioseo_run_updates', $lastActiveProVersion );
|
|
|
|
return;
|
|
}
|
|
|
|
// Try to acquire the lock.
|
|
if ( ! aioseo()->core->db->acquireLock( 'aioseo_pro_run_updates_lock', 0 ) ) {
|
|
// If we couldn't acquire the lock, exit early without doing anything.
|
|
// This means another process is already running updates.
|
|
return;
|
|
}
|
|
|
|
// Run the parent updates first.
|
|
parent::runUpdates();
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.0.0', '>=' ) && version_compare( $lastActiveProVersion, '4.0.5', '<' ) ) {
|
|
try {
|
|
aioseo()->core->cache->update( 'v4_migrate_post_og_image', time(), WEEK_IN_SECONDS );
|
|
aioseo()->core->cache->update( 'v4_migrate_term_og_image', time(), WEEK_IN_SECONDS );
|
|
|
|
as_schedule_single_action( time() + 10, 'aioseo_v4_migrate_post_og_image', [], 'aioseo' );
|
|
as_schedule_single_action( time() + 10, 'aioseo_v4_migrate_term_og_image', [], 'aioseo' );
|
|
} catch ( \Exception $e ) {
|
|
// Do nothing.
|
|
}
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.0.0', '>=' ) && version_compare( $lastActiveProVersion, '4.0.6', '<' ) ) {
|
|
$this->migratedGoogleAnalyticsToDeprecated();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.0.6', '<' ) ) {
|
|
$this->disableTwitterUseOgDefault();
|
|
$this->updateMaxImagePreviewDefault();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.1.5', '<' ) ) {
|
|
$this->removeUnusedColumns();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.1.8', '<' ) ) {
|
|
$this->migrateRemovedQaSchema();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.1.9', '<' ) ) {
|
|
// Clear addons cache to fix the PHP warning/error with the version_compare where the minimum version key for the REST API addon didn't exist yet locally.
|
|
aioseo()->core->cache->delete( 'addons' );
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.2.4', '<' ) ) {
|
|
$this->migrateImageSeoOptions();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.3.0', '<' ) ) {
|
|
$this->addSearchStatisticsTables();
|
|
aioseo()->access->addCapabilities();
|
|
|
|
// Clear addons cache to get the new features array to check for minimum plan levels.
|
|
aioseo()->core->cache->delete( 'addons' );
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.3.2', '<' ) ) {
|
|
$this->addOpenAiColumns();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.3.3', '<' ) ) {
|
|
aioseo()->actionScheduler->scheduleSingle( 'aioseo_v432_unschedule_objects_scan', 30 );
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.3.9.1', '<' ) ) {
|
|
$this->migratePriorityColumn();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.4.0', '<' ) ) {
|
|
$this->addRevisionsTable();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.5.0', '<' ) ) {
|
|
$this->updateSearchStatisticsObjectsForIndexStatus();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.5.5', '<' ) ) {
|
|
$this->cancelUrlInspectionScanActions();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.7.0', '<' ) ) {
|
|
$this->addSearchStatisticsKeywordRankTrackerTables();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.7.1', '<' ) ) {
|
|
$this->migrateRemoveCategoryBase();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.8.2', '<' ) ) {
|
|
$this->addColumnsToSearchStatisticsObjects();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.8.3', '<' ) ) {
|
|
$this->addBreadcrumbSettingsTermColumn();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.8.4', '<' ) ) {
|
|
$this->addNotificationForOpenAiUsers();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.8.6', '<' ) ) {
|
|
$this->addSeoAnalyzerObjectsTable();
|
|
$this->addSeoAnalyzerScanDateColumnForPosts();
|
|
$this->addSeoAnalyzerScanDateColumnForTerms();
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.9.1', '<' ) ) {
|
|
aioseo()->core->cache->delete( 'addons' );
|
|
$this->removeOldRedirectsAddon();
|
|
aioseo()->core->cache->delete( 'db_schema' );
|
|
$this->addRedirectsTables();
|
|
$this->migrateRedirectDefaults();
|
|
$this->addCustomRuleColumn();
|
|
$this->allowNullTargetUrl();
|
|
if ( ! empty( aioseo()->redirects->cache ) ) {
|
|
aioseo()->redirects->cache->clearRedirects();
|
|
}
|
|
$this->addRedirects404Tables();
|
|
$this->redirects404MigrateOptions();
|
|
$this->addPostIdColumn();
|
|
$this->fixDoubleEncodedSourceUrlMatch();
|
|
$this->addUrlHashColumns();
|
|
aioseo()->core->cache->delete( 'db_schema' );
|
|
}
|
|
|
|
if ( version_compare( $lastActiveProVersion, '4.9.6', '<' ) ) {
|
|
$this->migrateProSensitiveOptions();
|
|
$this->addRedirectsCommentColumn();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Updates the latest version after all migrations and updates have run.
|
|
*
|
|
* @since 4.4.0
|
|
*
|
|
* @return void
|
|
*/
|
|
public function updateLatestVersion() {
|
|
parent::updateLatestVersion();
|
|
|
|
if ( aioseo()->internalOptions->internal->lastActiveProVersion === aioseo()->version ) {
|
|
return;
|
|
}
|
|
|
|
aioseo()->internalOptions->internal->lastActiveProVersion = aioseo()->version;
|
|
}
|
|
|
|
/**
|
|
* Adds custom tables for V4.
|
|
*
|
|
* @since 4.0.0
|
|
*
|
|
* @return void
|
|
*/
|
|
public function addInitialCustomTablesForV4() {
|
|
parent::addInitialCustomTablesForV4();
|
|
|
|
$db = aioseo()->core->db->db;
|
|
$charsetCollate = '';
|
|
|
|
if ( ! empty( $db->charset ) ) {
|
|
$charsetCollate .= "DEFAULT CHARACTER SET {$db->charset}";
|
|
}
|
|
if ( ! empty( $db->collate ) ) {
|
|
$charsetCollate .= " COLLATE {$db->collate}";
|
|
}
|
|
|
|
if ( ! aioseo()->core->db->tableExists( 'aioseo_terms' ) ) {
|
|
$tableName = $db->prefix . 'aioseo_terms';
|
|
|
|
// Incorrect defaults are adjusted below through migrations.
|
|
aioseo()->core->db->execute(
|
|
"CREATE TABLE {$tableName} (
|
|
id bigint(20) unsigned NOT NULL AUTO_INCREMENT,
|
|
term_id bigint(20) unsigned NOT NULL,
|
|
title text DEFAULT NULL,
|
|
description text DEFAULT NULL,
|
|
keywords mediumtext DEFAULT NULL,
|
|
keyphrases longtext DEFAULT NULL,
|
|
page_analysis longtext DEFAULT NULL,
|
|
canonical_url text DEFAULT NULL,
|
|
og_title text DEFAULT NULL,
|
|
og_description text DEFAULT NULL,
|
|
og_object_type varchar(64) DEFAULT 'default',
|
|
og_image_type varchar(64) DEFAULT 'default',
|
|
og_image_custom_url text DEFAULT NULL,
|
|
og_image_custom_fields text DEFAULT NULL,
|
|
og_custom_image_width int(11) DEFAULT NULL,
|
|
og_custom_image_height int(11) DEFAULT NULL,
|
|
og_video varchar(255) DEFAULT NULL,
|
|
og_custom_url text DEFAULT NULL,
|
|
og_article_section text DEFAULT NULL,
|
|
og_article_tags text DEFAULT NULL,
|
|
twitter_use_og tinyint(1) DEFAULT 1,
|
|
twitter_card varchar(64) DEFAULT 'default',
|
|
twitter_image_type varchar(64) DEFAULT 'default',
|
|
twitter_image_custom_url text DEFAULT NULL,
|
|
twitter_image_custom_fields text DEFAULT NULL,
|
|
twitter_title text DEFAULT NULL,
|
|
twitter_description text DEFAULT NULL,
|
|
seo_score int(11) DEFAULT 0 NOT NULL,
|
|
pillar_content tinyint(1) DEFAULT NULL,
|
|
robots_default tinyint(1) DEFAULT 1 NOT NULL,
|
|
robots_noindex tinyint(1) DEFAULT 0 NOT NULL,
|
|
robots_noarchive tinyint(1) DEFAULT 0 NOT NULL,
|
|
robots_nosnippet tinyint(1) DEFAULT 0 NOT NULL,
|
|
robots_nofollow tinyint(1) DEFAULT 0 NOT NULL,
|
|
robots_noimageindex tinyint(1) DEFAULT 0 NOT NULL,
|
|
robots_noodp tinyint(1) DEFAULT 0 NOT NULL,
|
|
robots_notranslate tinyint(1) DEFAULT 0 NOT NULL,
|
|
robots_max_snippet int(11) DEFAULT NULL,
|
|
robots_max_videopreview int(11) DEFAULT NULL,
|
|
robots_max_imagepreview varchar(20) DEFAULT 'none',
|
|
priority tinytext DEFAULT NULL,
|
|
frequency tinytext DEFAULT NULL,
|
|
images longtext DEFAULT NULL,
|
|
videos longtext DEFAULT NULL,
|
|
video_scan_date datetime DEFAULT NULL,
|
|
tabs mediumtext DEFAULT NULL,
|
|
local_seo longtext DEFAULT NULL,
|
|
created datetime NOT NULL,
|
|
updated datetime NOT NULL,
|
|
PRIMARY KEY (id),
|
|
KEY ndx_aioseo_terms_term_id (term_id)
|
|
) {$charsetCollate};"
|
|
);
|
|
|
|
// Reset the cache for the installed tables.
|
|
aioseo()->core->cache->delete( 'db_schema' );
|
|
}
|
|
|
|
$this->addRedirectsTables();
|
|
}
|
|
|
|
/**
|
|
* Sets the default social images.
|
|
*
|
|
* @since 4.0.0
|
|
*
|
|
* @return void
|
|
*/
|
|
public function setDefaultSocialImages() {
|
|
parent::setDefaultSocialImages();
|
|
|
|
$siteLogo = aioseo()->helpers->getSiteLogoUrl();
|
|
if ( $siteLogo ) {
|
|
if ( ! aioseo()->options->social->facebook->general->defaultImageTerms ) {
|
|
aioseo()->options->social->facebook->general->defaultImageTerms = $siteLogo;
|
|
}
|
|
if ( ! aioseo()->options->social->twitter->general->defaultImageTerms ) {
|
|
aioseo()->options->social->twitter->general->defaultImageTerms = $siteLogo;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Migrates the post OG images for users between 4.0.0 and 4.0.4 again to fix a bug.
|
|
*
|
|
* @since 4.0.5
|
|
*
|
|
* @return void
|
|
*/
|
|
public function migratePostOgImage() {
|
|
// If the v3 migration is still running, postpone this.
|
|
if ( aioseo()->migration->isMigrationRunning() ) {
|
|
try {
|
|
as_schedule_single_action( time() + 300, 'aioseo_v4_migrate_post_og_image', [], 'aioseo' );
|
|
} catch ( \Exception $e ) {
|
|
// Do nothing.
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
$postsPerAction = 200;
|
|
$publicPostTypes = implode( "', '", aioseo()->helpers->getPublicPostTypes( true ) );
|
|
$timeStarted = gmdate( 'Y-m-d H:i:s', aioseo()->core->cache->get( 'v4_migrate_post_og_image' ) );
|
|
|
|
$postsToMigrate = aioseo()->core->db
|
|
->start( 'posts' . ' as p' )
|
|
->select( 'p.ID' )
|
|
->leftJoin( 'aioseo_posts as ap', '`p`.`ID` = `ap`.`post_id`' )
|
|
->whereRaw( "( ap.post_id IS NULL OR ap.updated < '$timeStarted' )" )
|
|
->whereRaw( "( p.post_type IN ( '$publicPostTypes' ) )" )
|
|
->orderBy( 'p.ID DESC' )
|
|
->limit( $postsPerAction )
|
|
->run()
|
|
->result();
|
|
|
|
if ( ! $postsToMigrate || ! count( $postsToMigrate ) ) {
|
|
aioseo()->core->cache->delete( 'v4_migrate_post_og_image' );
|
|
|
|
return;
|
|
}
|
|
|
|
foreach ( $postsToMigrate as $post ) {
|
|
$postMeta = aioseo()->core->db
|
|
->start( 'postmeta' . ' as pm' )
|
|
->select( 'pm.meta_key, pm.meta_value' )
|
|
->where( 'pm.post_id', $post->ID )
|
|
->whereLike( 'pm.meta_key', '_aioseop_opengraph_settings', false )
|
|
->run()
|
|
->result();
|
|
|
|
$aioseoPost = CommonModels\Post::getPost( $post->ID );
|
|
$meta = [
|
|
'post_id' => $post->ID,
|
|
];
|
|
|
|
if ( ! $postMeta || ! count( $postMeta ) ) {
|
|
$aioseoPost->set( $meta );
|
|
$aioseoPost->save();
|
|
continue;
|
|
}
|
|
|
|
foreach ( $postMeta as $record ) {
|
|
$name = $record->meta_key;
|
|
$value = $record->meta_value;
|
|
|
|
if ( '_aioseop_opengraph_settings' !== $name ) {
|
|
continue;
|
|
}
|
|
|
|
$ogMeta = aioseo()->helpers->maybeUnserialize( $value );
|
|
if ( ! is_array( $ogMeta ) ) {
|
|
continue;
|
|
}
|
|
|
|
foreach ( $ogMeta as $name => $value ) {
|
|
if (
|
|
'aioseop_opengraph_settings_image' !== $name ||
|
|
(
|
|
in_array( 'aioseop_opengraph_settings_customimg', array_keys( $ogMeta ), true ) &&
|
|
! empty( $post->og_image_custom_url ) &&
|
|
$post->og_image_custom_url !== $ogMeta['aioseop_opengraph_settings_customimg']
|
|
)
|
|
) {
|
|
continue;
|
|
}
|
|
|
|
$meta['og_image_type'] = 'custom_image';
|
|
$meta['og_image_custom_url'] = esc_url( $value );
|
|
break;
|
|
}
|
|
}
|
|
|
|
$aioseoPost->set( $meta );
|
|
$aioseoPost->save();
|
|
}
|
|
|
|
if ( count( $postsToMigrate ) === $postsPerAction ) {
|
|
try {
|
|
as_schedule_single_action( time() + 10, 'aioseo_v4_migrate_post_og_image', [], 'aioseo' );
|
|
} catch ( \Exception $e ) {
|
|
// Do nothing.
|
|
}
|
|
} else {
|
|
aioseo()->core->cache->delete( 'v4_migrate_post_og_image' );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Migrates the term OG images for users between 4.0.0 and 4.0.4 again to fix a bug.
|
|
*
|
|
* @since 4.0.5
|
|
*
|
|
* @return void
|
|
*/
|
|
public function migrateTermOgImage() {
|
|
// If the v3 migration is still running, postpone this.
|
|
if ( aioseo()->migration->isMigrationRunning() ) {
|
|
try {
|
|
as_schedule_single_action( time() + 300, 'aioseo_v4_migrate_term_og_image', [], 'aioseo' );
|
|
} catch ( \Exception $e ) {
|
|
// Do nothing.
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
$termsPerAction = 200;
|
|
$publicTaxonomies = implode( "', '", aioseo()->helpers->getPublicTaxonomies( true ) );
|
|
$timeStarted = gmdate( 'Y-m-d H:i:s', aioseo()->core->cache->get( 'v4_migrate_term_og_image' ) );
|
|
|
|
$termsToMigrate = aioseo()->core->db
|
|
->start( 'terms' . ' as t' )
|
|
->select( 't.term_id' )
|
|
->leftJoin( 'aioseo_terms as at', '`t`.`term_id` = `at`.`term_id`' )
|
|
->leftJoin( 'term_taxonomy as tt', '`t`.`term_id` = `tt`.`term_id`' )
|
|
->whereRaw( "( at.term_id IS NULL OR at.updated < '$timeStarted' )" )
|
|
->whereRaw( "( tt.taxonomy IN ( '$publicTaxonomies' ) )" )
|
|
->orderBy( 't.term_id DESC' )
|
|
->limit( $termsPerAction )
|
|
->run()
|
|
->result();
|
|
|
|
if ( ! $termsToMigrate || ! count( $termsToMigrate ) ) {
|
|
aioseo()->core->cache->delete( 'v4_migrate_term_og_image' );
|
|
|
|
return;
|
|
}
|
|
|
|
foreach ( $termsToMigrate as $term ) {
|
|
$termMeta = aioseo()->core->db
|
|
->start( 'termmeta' . ' as tm' )
|
|
->select( '`tm`.`meta_key`, `tm`.`meta_value`' )
|
|
->where( 'tm.term_id', $term->term_id )
|
|
->whereLike( 'tm.meta_key', '_aioseop_opengraph_settings%', true )
|
|
->run()
|
|
->result();
|
|
|
|
$aioseoTerm = ProModels\Term::getTerm( $term->term_id );
|
|
$meta = [
|
|
'term_id' => $term->term_id
|
|
];
|
|
|
|
if ( ! $termMeta || ! count( $termMeta ) ) {
|
|
$aioseoTerm->set( $meta );
|
|
$aioseoTerm->save();
|
|
continue;
|
|
}
|
|
|
|
foreach ( $termMeta as $record ) {
|
|
$name = $record->meta_key;
|
|
$value = $record->meta_value;
|
|
|
|
if ( '_aioseop_opengraph_settings' !== $name ) {
|
|
continue;
|
|
}
|
|
|
|
$ogMeta = aioseo()->helpers->maybeUnserialize( $value );
|
|
if ( ! is_array( $ogMeta ) ) {
|
|
continue;
|
|
}
|
|
|
|
foreach ( $ogMeta as $name => $value ) {
|
|
if (
|
|
'aioseop_opengraph_settings_image' !== $name ||
|
|
(
|
|
in_array( 'aioseop_opengraph_settings_customimg', array_keys( $ogMeta ), true ) &&
|
|
! empty( $term->og_image_custom_url ) &&
|
|
$term->og_image_custom_url !== $ogMeta['aioseop_opengraph_settings_customimg']
|
|
)
|
|
) {
|
|
continue;
|
|
}
|
|
|
|
$meta['og_image_type'] = 'custom_image';
|
|
$meta['og_image_custom_url'] = esc_url( $value );
|
|
break;
|
|
}
|
|
}
|
|
|
|
$aioseoTerm->set( $meta );
|
|
$aioseoTerm->save();
|
|
}
|
|
|
|
if ( count( $termsToMigrate ) === $termsPerAction ) {
|
|
try {
|
|
as_schedule_single_action( time() + 10, 'aioseo_v4_migrate_term_og_image', [], 'aioseo' );
|
|
} catch ( \Exception $e ) {
|
|
// Do nothing.
|
|
}
|
|
} else {
|
|
aioseo()->core->cache->delete( 'v4_migrate_term_og_image' );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Deprecate already migrated googleAnalytics.
|
|
*
|
|
* @since 4.0.6
|
|
*
|
|
* @return void
|
|
*/
|
|
private function migratedGoogleAnalyticsToDeprecated() {
|
|
$options = $this->getRawOptions();
|
|
if (
|
|
! empty( $options['webmasterTools'] ) &&
|
|
! empty( $options['webmasterTools']['googleAnalytics'] )
|
|
) {
|
|
foreach ( $options['webmasterTools']['googleAnalytics'] as $name => $value ) {
|
|
aioseo()->options->deprecated->webmasterTools->googleAnalytics->$name = $value;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Retrieve the raw options from the database for migration.
|
|
*
|
|
* @since 4.0.6
|
|
*
|
|
* @return array An array of options.
|
|
*/
|
|
private function getRawOptions() {
|
|
// Options from the DB.
|
|
$commonOptions = json_decode( get_option( aioseo()->options->optionsName ), true );
|
|
if ( empty( $commonOptions ) ) {
|
|
$commonOptions = [];
|
|
}
|
|
|
|
$proOptions = json_decode( get_option( aioseo()->options->optionsName . '_pro' ), true );
|
|
if ( empty( $proOptions ) ) {
|
|
$proOptions = [];
|
|
}
|
|
|
|
return array_merge_recursive( $commonOptions, $proOptions );
|
|
}
|
|
|
|
/**
|
|
* Modifes the default value of the twitter_use_og column.
|
|
*
|
|
* @since 4.0.6
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function disableTwitterUseOgDefault() {
|
|
parent::disableTwitterUseOgDefault();
|
|
|
|
if ( aioseo()->core->db->tableExists( 'aioseo_terms' ) ) {
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_terms';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
MODIFY twitter_use_og tinyint(1) DEFAULT 0"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Modifes the default value of the robots_max_imagepreview column.
|
|
*
|
|
* @since 4.0.6
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function updateMaxImagePreviewDefault() {
|
|
parent::updateMaxImagePreviewDefault();
|
|
|
|
if ( aioseo()->core->db->tableExists( 'aioseo_terms' ) ) {
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_terms';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
MODIFY robots_max_imagepreview varchar(20) DEFAULT 'large'"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Deletes duplicate records in our custom tables.
|
|
*
|
|
* @since 4.0.13
|
|
*
|
|
* @return void
|
|
*/
|
|
public function removeDuplicateRecords() {
|
|
parent::removeDuplicateRecords();
|
|
|
|
$duplicates = aioseo()->core->db->start( 'aioseo_terms' )
|
|
->select( 'term_id, min(id) as id' )
|
|
->groupBy( 'term_id having count(term_id) > 1' )
|
|
->orderByRaw( 'count(term_id) DESC' )
|
|
->run()
|
|
->result();
|
|
|
|
if ( empty( $duplicates ) ) {
|
|
return;
|
|
}
|
|
|
|
foreach ( $duplicates as $duplicate ) {
|
|
$termId = $duplicate->term_id;
|
|
$firstRecordId = $duplicate->id;
|
|
|
|
aioseo()->core->db->delete( 'aioseo_terms' )
|
|
->where( 'id >', $firstRecordId )
|
|
->where( 'term_id', $termId )
|
|
->run();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds the new capabilities for all the roles.
|
|
*
|
|
* @since 4.1.3
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function accessControlNewCapabilities() {
|
|
$newCapabilities = [
|
|
'dashboard',
|
|
'setupWizard'
|
|
];
|
|
|
|
$options = aioseo()->options->noConflict();
|
|
$dynamicOptions = aioseo()->dynamicOptions->noConflict();
|
|
|
|
foreach ( aioseo()->access->getRoles() as $role ) {
|
|
foreach ( $newCapabilities as $capability ) {
|
|
if ( $options->accessControl->has( $role ) ) {
|
|
$default = $options->accessControl->$role->getDefault( $capability );
|
|
$options->accessControl->$role->$capability = $default;
|
|
}
|
|
|
|
if ( $dynamicOptions->accessControl->has( $role ) ) {
|
|
$default = $dynamicOptions->accessControl->$role->getDefault( $capability );
|
|
$dynamicOptions->accessControl->$role->$capability = $default;
|
|
}
|
|
}
|
|
}
|
|
|
|
aioseo()->access->addCapabilities();
|
|
}
|
|
|
|
/**
|
|
* Migrate dynamic settings to a separate options structure.
|
|
*
|
|
* @since 4.1.4
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function migrateDynamicSettings() {
|
|
parent::migrateDynamicSettings();
|
|
|
|
$rawOptions = $this->getRawOptions();
|
|
$options = aioseo()->dynamicOptions->noConflict();
|
|
|
|
// Facebook post type object types.
|
|
if (
|
|
! empty( $rawOptions['social']['faceboook']['general']['dynamic']['taxonomies'] )
|
|
) {
|
|
foreach ( $rawOptions['social']['faceboook']['general']['dynamic']['taxonomies'] as $taxonomyName => $data ) {
|
|
if ( $options->social->facebook->general->taxonomies->has( $taxonomyName ) ) {
|
|
$options->social->facebook->general->taxonomies->$taxonomyName->objectType = $data['objectType'];
|
|
}
|
|
}
|
|
}
|
|
|
|
// Breadcrumbs post type templates.
|
|
if (
|
|
! empty( $rawOptions['breadcrumbs']['dynamic']['postTypes'] )
|
|
) {
|
|
foreach ( $rawOptions['breadcrumbs']['dynamic']['postTypes'] as $postTypeName => $data ) {
|
|
if ( $options->breadcrumbs->postTypes->has( $postTypeName ) ) {
|
|
$options->breadcrumbs->postTypes->$postTypeName->useDefaultTemplate = $data['useDefaultTemplate'];
|
|
$options->breadcrumbs->postTypes->$postTypeName->taxonomy = $data['taxonomy'];
|
|
$options->breadcrumbs->postTypes->$postTypeName->showArchiveCrumb = $data['showArchiveCrumb'];
|
|
$options->breadcrumbs->postTypes->$postTypeName->showTaxonomyCrumbs = $data['showTaxonomyCrumbs'];
|
|
$options->breadcrumbs->postTypes->$postTypeName->showHomeCrumb = $data['showHomeCrumb'];
|
|
$options->breadcrumbs->postTypes->$postTypeName->showPrefixCrumb = $data['showPrefixCrumb'];
|
|
$options->breadcrumbs->postTypes->$postTypeName->showParentCrumbs = $data['showParentCrumbs'];
|
|
$options->breadcrumbs->postTypes->$postTypeName->template = $data['template'];
|
|
|
|
if ( ! empty( $data['parentTemplate'] ) ) {
|
|
$options->breadcrumbs->postTypes->$postTypeName->parentTemplate = $data['parentTemplate'];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Breadcrumbs taxonomy templates.
|
|
if (
|
|
! empty( $rawOptions['breadcrumbs']['dynamic']['taxonomies'] )
|
|
) {
|
|
foreach ( $rawOptions['breadcrumbs']['dynamic']['taxonomies'] as $taxonomyName => $data ) {
|
|
if ( $options->breadcrumbs->taxonomies->has( $taxonomyName ) ) {
|
|
$options->breadcrumbs->taxonomies->$taxonomyName->useDefaultTemplate = $data['useDefaultTemplate'];
|
|
$options->breadcrumbs->taxonomies->$taxonomyName->showHomeCrumb = $data['showHomeCrumb'];
|
|
$options->breadcrumbs->taxonomies->$taxonomyName->showPrefixCrumb = $data['showPrefixCrumb'];
|
|
$options->breadcrumbs->taxonomies->$taxonomyName->showParentCrumbs = $data['showParentCrumbs'];
|
|
$options->breadcrumbs->taxonomies->$taxonomyName->template = $data['template'];
|
|
|
|
if ( ! empty( $data['parentTemplate'] ) ) {
|
|
$options->breadcrumbs->taxonomies->$taxonomyName->parentTemplate = $data['parentTemplate'];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Access control settings.
|
|
if (
|
|
! empty( $rawOptions['accessControl']['dynamic'] )
|
|
) {
|
|
foreach ( $rawOptions['accessControl']['dynamic'] as $role => $data ) {
|
|
if ( $options->accessControl->has( $role ) ) {
|
|
$options->accessControl->$role->useDefault = $data['useDefault'];
|
|
$options->accessControl->$role->dashboard = $data['dashboard'];
|
|
$options->accessControl->$role->generalSettings = $data['generalSettings'];
|
|
$options->accessControl->$role->searchAppearanceSettings = $data['searchAppearanceSettings'];
|
|
$options->accessControl->$role->socialNetworksSettings = $data['socialNetworksSettings'];
|
|
$options->accessControl->$role->sitemapSettings = $data['sitemapSettings'];
|
|
$options->accessControl->$role->redirectsSettings = $data['redirectsSettings'];
|
|
$options->accessControl->$role->seoAnalysisSettings = $data['seoAnalysisSettings'];
|
|
$options->accessControl->$role->toolsSettings = $data['toolsSettings'];
|
|
$options->accessControl->$role->featureManagerSettings = $data['featureManagerSettings'];
|
|
$options->accessControl->$role->pageAnalysis = $data['pageAnalysis'];
|
|
$options->accessControl->$role->pageGeneralSettings = $data['pageGeneralSettings'];
|
|
$options->accessControl->$role->pageAdvancedSettings = $data['pageAdvancedSettings'];
|
|
$options->accessControl->$role->pageSchemaSettings = $data['pageSchemaSettings'];
|
|
$options->accessControl->$role->pageSocialSettings = $data['pageSocialSettings'];
|
|
$options->accessControl->$role->localSeoSettings = $data['localSeoSettings'];
|
|
$options->accessControl->$role->pageLocalSeoSettings = $data['pageLocalSeoSettings'];
|
|
$options->accessControl->$role->setupWizard = $data['setupWizard'];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Removes a number of columns that Terms do not use.
|
|
*
|
|
* @since 4.1.5
|
|
*
|
|
* @return void
|
|
*/
|
|
private function removeUnusedColumns() {
|
|
$columnsToRemove = [
|
|
'cornerstone_content',
|
|
'seo_score',
|
|
'keyphrases',
|
|
'page_analysis'
|
|
];
|
|
|
|
foreach ( $columnsToRemove as $columnName ) {
|
|
if ( aioseo()->core->db->columnExists( 'aioseo_terms', $columnName ) ) {
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_terms';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
DROP COLUMN $columnName"
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Add in image width/height columns and image URL for caching.
|
|
*
|
|
* @since 4.1.6
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function migrateOgTwitterImageColumns() {
|
|
parent::migrateOgTwitterImageColumns();
|
|
|
|
if ( aioseo()->core->db->tableExists( 'aioseo_terms' ) ) {
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_terms';
|
|
|
|
// OG Columns.
|
|
if ( ! aioseo()->core->db->columnExists( 'aioseo_terms', 'og_image_url' ) ) {
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName} ADD og_image_url text DEFAULT NULL AFTER og_image_type"
|
|
);
|
|
}
|
|
|
|
if ( aioseo()->core->db->columnExists( 'aioseo_terms', 'og_custom_image_height' ) ) {
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName} CHANGE COLUMN og_custom_image_height og_image_height int(11) DEFAULT NULL AFTER og_image_url"
|
|
);
|
|
} elseif ( ! aioseo()->core->db->columnExists( 'aioseo_terms', 'og_image_height' ) ) {
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName} ADD og_image_height int(11) DEFAULT NULL AFTER og_image_url"
|
|
);
|
|
}
|
|
|
|
if ( aioseo()->core->db->columnExists( 'aioseo_terms', 'og_custom_image_width' ) ) {
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName} CHANGE COLUMN og_custom_image_width og_image_width int(11) DEFAULT NULL AFTER og_image_url"
|
|
);
|
|
} elseif ( ! aioseo()->core->db->columnExists( 'aioseo_terms', 'og_image_width' ) ) {
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName} ADD og_image_width int(11) DEFAULT NULL AFTER og_image_url"
|
|
);
|
|
}
|
|
|
|
// Twitter image url columnn.
|
|
if ( ! aioseo()->core->db->columnExists( 'aioseo_terms', 'twitter_image_url' ) ) {
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName} ADD twitter_image_url text DEFAULT NULL AFTER twitter_image_type"
|
|
);
|
|
}
|
|
|
|
// Reset the cache for the installed tables.
|
|
aioseo()->core->cache->delete( 'db_schema' );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Migrates the removed QAPage schema if it's used anywhere as a default.
|
|
*
|
|
* @since 4.1.8
|
|
*
|
|
* @return void
|
|
*/
|
|
private function migrateRemovedQaSchema() {
|
|
$searchAppearance = aioseo()->dynamicOptions->searchAppearance->all();
|
|
$postTypes = array_keys( $searchAppearance['postTypes'] );
|
|
|
|
foreach ( $postTypes as $postType ) {
|
|
if ( 'qapage' === strtolower( aioseo()->dynamicOptions->searchAppearance->postTypes->$postType->webPageType ) ) {
|
|
aioseo()->dynamicOptions->searchAppearance->postTypes->$postType->webPageType = 'WebPage';
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Remove the tabs column as it is unnecessary.
|
|
*
|
|
* @since 4.2.2
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function removeTabsColumn() {
|
|
parent::removeTabsColumn();
|
|
|
|
if ( aioseo()->core->db->columnExists( 'aioseo_terms', 'tabs' ) ) {
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_terms';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
DROP tabs"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Migrates the old Image SEO options to the new structure.
|
|
*
|
|
* @since 4.2.4
|
|
*
|
|
* @return void
|
|
*/
|
|
public function migrateImageSeoOptions() {
|
|
$image = aioseo()->options->image->all();
|
|
$format = ! empty( $image['format'] ) ? $image['format'] : '';
|
|
$stripPunctuation = ! empty( $image['stripPunctuation'] ) ? $image['stripPunctuation'] : '';
|
|
|
|
if ( $format ) {
|
|
foreach ( $format as $attribute => $option ) {
|
|
aioseo()->options->image->{$attribute}->format = $option;
|
|
}
|
|
}
|
|
|
|
if ( $stripPunctuation ) {
|
|
foreach ( $stripPunctuation as $attribute => $option ) {
|
|
aioseo()->options->image->{$attribute}->stripPunctuation = $option;
|
|
}
|
|
}
|
|
|
|
// If the user was already actively using the Image SEO addon, disable the upload settings.
|
|
if ( ! aioseo()->addons->getLoadedAddon( 'imageSeo' ) ) {
|
|
return;
|
|
}
|
|
|
|
aioseo()->options->image->caption->autogenerate = false;
|
|
aioseo()->options->image->description->autogenerate = false;
|
|
}
|
|
|
|
/**
|
|
* Adds the tables for Search Statistics.
|
|
*
|
|
* @since 4.3.0
|
|
*
|
|
* @return void
|
|
*/
|
|
public function addSearchStatisticsTables() {
|
|
$db = aioseo()->core->db->db;
|
|
$charsetCollate = '';
|
|
if ( ! empty( $db->charset ) ) {
|
|
$charsetCollate .= "DEFAULT CHARACTER SET {$db->charset}";
|
|
}
|
|
if ( ! empty( $db->collate ) ) {
|
|
$charsetCollate .= " COLLATE {$db->collate}";
|
|
}
|
|
|
|
// Check for search_statistics_objects table.
|
|
if ( ! aioseo()->core->db->tableExists( 'aioseo_search_statistics_objects' ) ) {
|
|
$tableName = $db->prefix . 'aioseo_search_statistics_objects';
|
|
|
|
aioseo()->core->db->execute(
|
|
"CREATE TABLE {$tableName} (
|
|
id bigint(20) unsigned NOT NULL auto_increment,
|
|
object_id bigint(20) unsigned DEFAULT NULL,
|
|
object_type varchar(100) DEFAULT NULL,
|
|
object_subtype varchar(100) DEFAULT NULL,
|
|
object_path varchar(500) NOT NULL,
|
|
object_path_hash varchar(40) NOT NULL,
|
|
inspection_result longtext DEFAULT NULL,
|
|
inspection_result_date datetime DEFAULT NULL,
|
|
verdict varchar(64) DEFAULT NULL,
|
|
robots_txt_state varchar(64) DEFAULT NULL,
|
|
indexing_state varchar(64) DEFAULT NULL,
|
|
page_fetch_state varchar(64) DEFAULT NULL,
|
|
coverage_state varchar(64) DEFAULT NULL,
|
|
crawled_as varchar(64) DEFAULT NULL,
|
|
last_crawl_time datetime DEFAULT NULL,
|
|
created datetime NOT NULL,
|
|
updated datetime NOT NULL,
|
|
PRIMARY KEY (id),
|
|
UNIQUE KEY ndx_aioseo_object_path_hash1 (object_path_hash),
|
|
KEY ndx_aioseo_object_id1 (object_id)
|
|
) {$charsetCollate};"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds the post column for the OpenAI data.
|
|
*
|
|
* @since 4.3.2
|
|
*
|
|
* @return void
|
|
*/
|
|
private function addOpenAiColumns() {
|
|
if ( ! aioseo()->core->db->columnExists( 'aioseo_posts', 'open_ai' ) ) {
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_posts';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
ADD open_ai mediumtext DEFAULT NULL AFTER options"
|
|
);
|
|
|
|
aioseo()->core->cache->delete( 'db_schema' );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Modify the columns to allow nullable values and add the column for the index status data.
|
|
*
|
|
* @since 4.5.0
|
|
*
|
|
* @return void
|
|
*/
|
|
private function updateSearchStatisticsObjectsForIndexStatus() {
|
|
if ( ! aioseo()->core->db->tableExists( 'aioseo_search_statistics_objects' ) ) {
|
|
return;
|
|
}
|
|
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_search_statistics_objects';
|
|
|
|
// Allow nullable values.
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
MODIFY `object_id` bigint(20) unsigned DEFAULT NULL,
|
|
MODIFY `object_type` varchar(100) DEFAULT NULL,
|
|
MODIFY `object_subtype` varchar(100) DEFAULT NULL"
|
|
);
|
|
|
|
// Create new inspection_result column.
|
|
if ( ! aioseo()->core->db->columnExists( 'aioseo_search_statistics_objects', 'inspection_result' ) ) {
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
ADD `inspection_result` longtext DEFAULT NULL AFTER object_path_hash,
|
|
ADD `inspection_result_date` datetime DEFAULT NULL AFTER inspection_result,
|
|
ADD `indexed` tinyint(1) DEFAULT 0 NOT NULL AFTER inspection_result_date"
|
|
);
|
|
}
|
|
|
|
// Reset the cache for the installed tables.
|
|
aioseo()->core->cache->delete( 'db_schema' );
|
|
|
|
// Reset the objects table to start scanning again.
|
|
aioseo()->searchStatistics->objects->reset();
|
|
}
|
|
|
|
/**
|
|
* Cancels all duplicate aioseo_search_statistics_objects_scan actions.
|
|
*
|
|
* @since 4.3.3
|
|
*
|
|
* @return void
|
|
*/
|
|
public function cancelDuplicateObjectsActions() {
|
|
as_unschedule_all_actions( 'aioseo_search_statistics_objects_scan' );
|
|
}
|
|
|
|
/**
|
|
* Cancels all aioseo_search_statistics_objects_scan actions.
|
|
*
|
|
* @since 4.5.5
|
|
*
|
|
* @return void
|
|
*/
|
|
public function cancelUrlInspectionScanActions() {
|
|
as_unschedule_all_actions( 'aioseo_search_statistics_url_inspection_scan' );
|
|
}
|
|
|
|
/**
|
|
* Casts the priority column to a float.
|
|
*
|
|
* @since 4.3.9.1
|
|
*
|
|
* @return void
|
|
*/
|
|
private function migratePriorityColumn() {
|
|
if ( ! aioseo()->core->db->columnExists( 'aioseo_terms', 'priority' ) ) {
|
|
return;
|
|
}
|
|
|
|
$prefix = aioseo()->core->db->prefix;
|
|
$aioseoTermsTable = $prefix . 'aioseo_terms';
|
|
|
|
// First, cast the default value to NULL since it's a string.
|
|
aioseo()->core->db->execute( "UPDATE {$aioseoTermsTable} SET priority = NULL WHERE priority = 'default'" );
|
|
|
|
// Then, alter the column to a float.
|
|
aioseo()->core->db->execute( "ALTER TABLE {$aioseoTermsTable} MODIFY priority float" );
|
|
}
|
|
|
|
/**
|
|
* Create the SEO Revisions table.
|
|
*
|
|
* @since 4.4.0
|
|
*
|
|
* @return void
|
|
*/
|
|
public function addRevisionsTable() {
|
|
$rawTableName = 'aioseo_revisions';
|
|
|
|
if ( aioseo()->core->db->tableExists( $rawTableName ) ) {
|
|
return;
|
|
}
|
|
|
|
$prefixedTableName = aioseo()->core->db->db->prefix . $rawTableName;
|
|
$charsetCollate = aioseo()->core->db->db->get_charset_collate();
|
|
$sql = "
|
|
CREATE TABLE $prefixedTableName (
|
|
id BIGINT(20) NOT NULL AUTO_INCREMENT,
|
|
object_id BIGINT(20) NOT NULL,
|
|
object_type varchar(20) NOT NULL,
|
|
author_id BIGINT(20) NOT NULL,
|
|
note varchar(255) DEFAULT NULL,
|
|
revision_data LONGTEXT NOT NULL,
|
|
created DATETIME NOT NULL,
|
|
updated DATETIME NOT NULL,
|
|
PRIMARY KEY (id),
|
|
KEY object (object_id, object_type)
|
|
) $charsetCollate;
|
|
";
|
|
|
|
aioseo()->core->db->execute( $sql );
|
|
}
|
|
|
|
/**
|
|
* Create the Keyword Rank Tracker tables.
|
|
*
|
|
* @since 4.7.0
|
|
*
|
|
* @return void
|
|
*/
|
|
public function addSearchStatisticsKeywordRankTrackerTables() {
|
|
$db = aioseo()->core->db->db;
|
|
$charsetCollate = $db->get_charset_collate();
|
|
$rawTableName = 'aioseo_search_statistics_keywords';
|
|
if ( ! aioseo()->core->db->tableExists( $rawTableName ) ) {
|
|
aioseo()->core->db->execute(
|
|
"CREATE TABLE {$db->prefix}{$rawTableName} (
|
|
id bigint(20) unsigned NOT NULL AUTO_INCREMENT,
|
|
name varchar(150) NOT NULL,
|
|
favorited tinyint(1) NOT NULL DEFAULT 0,
|
|
created DATETIME NOT NULL,
|
|
updated DATETIME NOT NULL,
|
|
PRIMARY KEY (`id`),
|
|
UNIQUE KEY ndx_aioseo_search_statistics_keyword_name (`name`)
|
|
) $charsetCollate;"
|
|
);
|
|
}
|
|
|
|
$rawTableName = 'aioseo_search_statistics_keyword_groups';
|
|
if ( ! aioseo()->core->db->tableExists( $rawTableName ) ) {
|
|
aioseo()->core->db->execute(
|
|
"CREATE TABLE {$db->prefix}{$rawTableName} (
|
|
id bigint(20) unsigned NOT NULL AUTO_INCREMENT,
|
|
name varchar(150) DEFAULT NULL,
|
|
created DATETIME NOT NULL,
|
|
updated DATETIME NOT NULL,
|
|
PRIMARY KEY (`id`),
|
|
UNIQUE KEY ndx_aioseo_search_statistics_keyword_groups_name (`name`)
|
|
) $charsetCollate;"
|
|
);
|
|
}
|
|
|
|
$rawTableName = 'aioseo_search_statistics_keyword_relationships';
|
|
if ( ! aioseo()->core->db->tableExists( $rawTableName ) ) {
|
|
aioseo()->core->db->execute(
|
|
"CREATE TABLE {$db->prefix}{$rawTableName} (
|
|
keyword_id bigint(20) unsigned NOT NULL,
|
|
keyword_group_id bigint(20) unsigned NOT NULL,
|
|
PRIMARY KEY (`keyword_id`, `keyword_group_id`)
|
|
) $charsetCollate;"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Migrates the "Remove Category Base" setting from Pro to Lite.
|
|
*
|
|
* @since 4.7.1
|
|
*
|
|
* @return void
|
|
*/
|
|
public function migrateRemoveCategoryBase() {
|
|
$options = $this->getRawOptions();
|
|
if ( empty( $options['searchAppearance']['advanced']['removeCatBase'] ) ) {
|
|
return;
|
|
}
|
|
|
|
aioseo()->options->searchAppearance->advanced->removeCategoryBase = true;
|
|
aioseo()->options->flushRewriteRules();
|
|
}
|
|
|
|
/**
|
|
* Add more columns to the search statistics objects table.
|
|
*
|
|
* @since 4.8.2
|
|
*
|
|
* @return void
|
|
*/
|
|
private function addColumnsToSearchStatisticsObjects() {
|
|
$rawTableName = 'aioseo_search_statistics_objects';
|
|
$prefixedTableName = aioseo()->core->db->db->prefix . $rawTableName;
|
|
if ( ! aioseo()->core->db->tableExists( $rawTableName ) ) {
|
|
return;
|
|
}
|
|
|
|
if ( ! aioseo()->core->db->columnExists( $rawTableName, 'verdict' ) ) {
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$prefixedTableName}
|
|
ADD `last_crawl_time` datetime DEFAULT NULL AFTER inspection_result_date,
|
|
ADD `crawled_as` varchar(64) DEFAULT NULL AFTER inspection_result_date,
|
|
ADD `coverage_state` varchar(64) DEFAULT NULL AFTER inspection_result_date,
|
|
ADD `page_fetch_state` varchar(64) DEFAULT NULL AFTER inspection_result_date,
|
|
ADD `indexing_state` varchar(64) DEFAULT NULL AFTER inspection_result_date,
|
|
ADD `robots_txt_state` varchar(64) DEFAULT NULL AFTER inspection_result_date,
|
|
ADD `verdict` varchar(64) DEFAULT NULL AFTER inspection_result_date"
|
|
);
|
|
}
|
|
|
|
// Make sure the object_path_hash column has a unique index.
|
|
$indexes = aioseo()->core->db->execute( "SHOW INDEXES FROM {$prefixedTableName} WHERE Key_name = 'ndx_aioseo_object_path_hash1'", true )->result();
|
|
if ( ! empty( $indexes[0] ) && $indexes[0]->Non_unique ?? 0 ) {
|
|
// 1. Clean up existing duplicate data (and prevent the error "Duplicate entry for key ndx_aioseo_object_path_hash1").
|
|
aioseo()->core->db->execute(
|
|
"DELETE t1 FROM {$prefixedTableName} t1
|
|
INNER JOIN {$prefixedTableName} t2
|
|
WHERE t1.id < t2.id AND t1.object_path_hash = t2.object_path_hash"
|
|
);
|
|
|
|
// 2. Enforce data integrity by adding a unique constraint.
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$prefixedTableName}
|
|
DROP INDEX ndx_aioseo_object_path_hash1,
|
|
ADD UNIQUE KEY ndx_aioseo_object_path_hash1 (object_path_hash)"
|
|
);
|
|
}
|
|
|
|
aioseo()->core->cache->delete( 'db_schema' );
|
|
aioseo()->searchStatistics->urlInspection->reset();
|
|
}
|
|
|
|
/**
|
|
* Adds the breadcrumb settings column to our terms table.
|
|
*
|
|
* @since 4.8.3
|
|
*
|
|
* @return void
|
|
*/
|
|
public function addBreadcrumbSettingsTermColumn() {
|
|
if ( ! aioseo()->core->db->columnExists( 'aioseo_terms', 'breadcrumb_settings' ) ) {
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_terms';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
ADD `breadcrumb_settings` longtext DEFAULT NULL AFTER local_seo"
|
|
);
|
|
|
|
aioseo()->core->cache->delete( 'db_schema' );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds a notification for OpenAI users.
|
|
*
|
|
* @since 4.8.4
|
|
*
|
|
* @return void
|
|
*/
|
|
private function addNotificationForOpenAiUsers() {
|
|
$options = $this->getRawOptions();
|
|
if ( empty( $options['advanced']['openAiKey'] ) ) {
|
|
return;
|
|
}
|
|
|
|
if ( CommonModels\Notification::getNotificationByName( 'open-ai-removed' )->exists() ) {
|
|
return;
|
|
}
|
|
|
|
CommonModels\Notification::addNotification( [
|
|
'slug' => uniqid(),
|
|
'notification_name' => 'open-ai-removed',
|
|
'title' => __( 'Introducing AI Content', 'aioseo-pro' ),
|
|
'content' => __( 'We are excited to announce the launch of our new AI Content Generator, allowing you to generate FAQs, Key Points, social posts, SEO titles, meta descriptions, and more!</br></br>The old OpenAI integration has been removed in favour of this new feature, so we can focus on providing you with the best AI experience possible. Your AIOSEO Pro license comes with credits that are included in your subscription, so you can start generating content right away.', 'aioseo-pro' ), // phpcs:ignore Generic.Files.LineLength.MaxExceeded
|
|
'type' => 'info',
|
|
'level' => [ 'all' ],
|
|
'button1_label' => __( 'Learn More', 'aioseo-pro' ),
|
|
'button1_action' => 'https://aioseo.com/docs/ai-content-generator/',
|
|
'start' => gmdate( 'Y-m-d H:i:s' )
|
|
] );
|
|
}
|
|
|
|
/**
|
|
* Adds our custom table for the SeoAnalysis/SeoAnalyzer issues/checkers.
|
|
*
|
|
* @since 4.8.6
|
|
*
|
|
* @return void
|
|
*/
|
|
public function addSeoAnalyzerObjectsTable() {
|
|
$db = aioseo()->core->db->db;
|
|
$charsetCollate = '';
|
|
|
|
if ( ! empty( $db->charset ) ) {
|
|
$charsetCollate .= "DEFAULT CHARACTER SET {$db->charset}";
|
|
}
|
|
if ( ! empty( $db->collate ) ) {
|
|
$charsetCollate .= " COLLATE {$db->collate}";
|
|
}
|
|
|
|
// Check for seo analyzer objects table.
|
|
if ( ! aioseo()->core->db->tableExists( 'aioseo_seo_analyzer_objects' ) ) {
|
|
$tableName = $db->prefix . 'aioseo_seo_analyzer_objects';
|
|
|
|
aioseo()->core->db->execute(
|
|
"CREATE TABLE {$tableName} (
|
|
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
|
|
`object_id` bigint(20) unsigned NOT NULL,
|
|
`object_type` varchar(40) NOT NULL,
|
|
`object_subtype` varchar(40) NOT NULL,
|
|
`code` varchar(50) NOT NULL,
|
|
`metadata` text,
|
|
`is_ignored` boolean NOT NULL DEFAULT 0,
|
|
`created` DATETIME NOT NULL,
|
|
`updated` DATETIME NOT NULL,
|
|
PRIMARY KEY (id),
|
|
KEY ndx_aioseo_seo_analyzer_objects_object (object_id, object_type),
|
|
KEY ndx_aioseo_seo_analyzer_objects_code (code)
|
|
) {$charsetCollate};"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds the seo analyzer scan date column to our posts table.
|
|
*
|
|
* @since 4.8.6
|
|
*
|
|
* @return void
|
|
*/
|
|
public function addSeoAnalyzerScanDateColumnForPosts() {
|
|
if ( ! aioseo()->core->db->columnExists( 'aioseo_posts', 'seo_analyzer_scan_date' ) ) {
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_posts';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
ADD seo_analyzer_scan_date datetime DEFAULT NULL AFTER local_seo"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds the seo analyzer scan date column to our terms table.
|
|
*
|
|
* @since 4.8.6
|
|
*
|
|
* @return void
|
|
*/
|
|
public function addSeoAnalyzerScanDateColumnForTerms() {
|
|
if ( ! aioseo()->core->db->columnExists( 'aioseo_terms', 'seo_analyzer_scan_date' ) ) {
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_terms';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
ADD seo_analyzer_scan_date datetime DEFAULT NULL AFTER local_seo"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds a column for custom redirect matching rules.
|
|
*
|
|
* @since 4.9.1
|
|
*
|
|
* @return void
|
|
*/
|
|
private function addCustomRuleColumn() {
|
|
if ( ! aioseo()->core->db->columnExists( 'aioseo_redirects', 'custom_rules' ) ) {
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_redirects';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
ADD custom_rules text DEFAULT NULL AFTER query_param"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds a column for custom redirect matching rules.
|
|
*
|
|
* @since 4.9.1
|
|
*
|
|
* @return void
|
|
*/
|
|
private function allowNullTargetUrl() {
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_redirects';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
MODIFY `target_url_hash` varchar(40) DEFAULT NULL"
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Add MySQL tables for redirect 404s.
|
|
*
|
|
* @since 4.9.1
|
|
*
|
|
* @return void
|
|
*/
|
|
private function addRedirects404Tables() {
|
|
$db = aioseo()->core->db->db;
|
|
$charsetCollate = '';
|
|
|
|
if ( ! empty( $db->charset ) ) {
|
|
$charsetCollate .= "DEFAULT CHARACTER SET {$db->charset}";
|
|
}
|
|
if ( ! empty( $db->collate ) ) {
|
|
$charsetCollate .= " COLLATE {$db->collate}";
|
|
}
|
|
|
|
if ( ! aioseo()->core->db->tableExists( 'aioseo_redirects_404' ) ) {
|
|
$tableName = $db->prefix . 'aioseo_redirects_404';
|
|
|
|
aioseo()->core->db->execute(
|
|
"CREATE TABLE {$tableName} (
|
|
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
|
|
`post_id` bigint(20) unsigned DEFAULT NULL,
|
|
`post_type` varchar(256) DEFAULT NULL,
|
|
`taxonomy` varchar(256) DEFAULT NULL,
|
|
`source_url` text,
|
|
`source_url_hash` varchar(40) NOT NULL,
|
|
`parent_posts` text,
|
|
`parent_terms` text,
|
|
`created` datetime NOT NULL,
|
|
`updated` datetime NOT NULL,
|
|
PRIMARY KEY (id),
|
|
UNIQUE KEY ndx_aioseo_redirects_404_source_url_hash (source_url_hash)
|
|
) {$charsetCollate};"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Add MySQL tables for redirects.
|
|
*
|
|
* @since 4.9.1
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function addRedirectsTables() {
|
|
// This update requires V4 to be active and since this could run on an activation hook, we need an extra sanity check.
|
|
$db = aioseo()->core->db->db;
|
|
$charsetCollate = '';
|
|
|
|
if ( ! empty( $db->charset ) ) {
|
|
$charsetCollate .= "DEFAULT CHARACTER SET {$db->charset}";
|
|
}
|
|
if ( ! empty( $db->collate ) ) {
|
|
$charsetCollate .= " COLLATE {$db->collate}";
|
|
}
|
|
|
|
// Check for redirects table.
|
|
if ( ! aioseo()->core->db->tableExists( 'aioseo_redirects' ) ) {
|
|
$tableName = $db->prefix . 'aioseo_redirects';
|
|
|
|
aioseo()->core->db->execute(
|
|
"CREATE TABLE {$tableName} (
|
|
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
|
|
`source_url` text NOT NULL,
|
|
`source_url_hash` varchar(40) NOT NULL,
|
|
`source_url_match` text NOT NULL,
|
|
`source_url_match_hash` varchar(40) NOT NULL,
|
|
`target_url` text NOT NULL,
|
|
`target_url_hash` varchar(40) NOT NULL,
|
|
`type` int(11) unsigned NOT NULL DEFAULT 301,
|
|
`query_param` varchar(40) NOT NULL DEFAULT 'ignore',
|
|
`group` varchar(256) NOT NULL DEFAULT 'manual',
|
|
`regex` tinyint(1) unsigned NOT NULL DEFAULT 0,
|
|
`ignore_slash` tinyint(1) unsigned NOT NULL DEFAULT 1,
|
|
`ignore_case` tinyint(1) unsigned NOT NULL DEFAULT 1,
|
|
`enabled` tinyint(1) unsigned NOT NULL DEFAULT 1,
|
|
`created` datetime NOT NULL,
|
|
`updated` datetime NOT NULL,
|
|
PRIMARY KEY (id),
|
|
UNIQUE KEY ndx_aioseo_redirects_source_url_hash (source_url_hash),
|
|
KEY ndx_aioseo_redirects_source_url_match_hash (source_url_match_hash),
|
|
KEY ndx_aioseo_redirects_target_url_hash (target_url_hash),
|
|
KEY ndx_aioseo_redirects_type (type),
|
|
KEY ndx_aioseo_redirects_enabled (enabled)
|
|
) {$charsetCollate};"
|
|
);
|
|
}
|
|
|
|
if ( ! aioseo()->core->db->tableExists( 'aioseo_redirects_hits' ) ) {
|
|
$tableName = $db->prefix . 'aioseo_redirects_hits';
|
|
|
|
aioseo()->core->db->execute(
|
|
"CREATE TABLE {$tableName} (
|
|
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
|
|
`redirect_id` bigint(20) unsigned NOT NULL,
|
|
`count` bigint(20) unsigned NOT NULL DEFAULT 0,
|
|
`created` datetime NOT NULL,
|
|
`updated` datetime NOT NULL,
|
|
PRIMARY KEY (id),
|
|
UNIQUE KEY ndx_aioseo_redirects_hits_redirect_id (redirect_id),
|
|
KEY ndx_aioseo_redirects_hits_count (count)
|
|
) {$charsetCollate};"
|
|
);
|
|
}
|
|
|
|
if ( ! aioseo()->core->db->tableExists( 'aioseo_redirects_logs' ) ) {
|
|
$tableName = $db->prefix . 'aioseo_redirects_logs';
|
|
|
|
aioseo()->core->db->execute(
|
|
"CREATE TABLE {$tableName} (
|
|
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
|
|
`url` mediumtext NOT NULL,
|
|
`domain` varchar(255) DEFAULT NULL,
|
|
`sent_to` mediumtext DEFAULT NULL,
|
|
`agent` mediumtext,
|
|
`referrer` mediumtext DEFAULT NULL,
|
|
`http_code` int(11) unsigned NOT NULL DEFAULT 0,
|
|
`request_method` varchar(10) DEFAULT NULL,
|
|
`request_data` mediumtext DEFAULT NULL,
|
|
`redirect_by` varchar(50) DEFAULT NULL,
|
|
`redirect_id` bigint(20) unsigned DEFAULT NULL,
|
|
`ip` varchar(45) DEFAULT NULL,
|
|
`created` datetime NOT NULL,
|
|
`updated` datetime NOT NULL,
|
|
PRIMARY KEY (`id`),
|
|
KEY ndx_aioseo_redirects_logs_created (`created`),
|
|
KEY ndx_aioseo_redirects_logs_redirect_id (`redirect_id`),
|
|
KEY ndx_aioseo_redirects_logs_ip (`ip`)
|
|
) {$charsetCollate};"
|
|
);
|
|
}
|
|
|
|
if ( ! aioseo()->core->db->tableExists( 'aioseo_redirects_404_logs' ) ) {
|
|
$tableName = $db->prefix . 'aioseo_redirects_404_logs';
|
|
|
|
aioseo()->core->db->execute(
|
|
"CREATE TABLE {$tableName} (
|
|
`id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
|
|
`url` mediumtext NOT NULL,
|
|
`domain` varchar(255) DEFAULT NULL,
|
|
`agent` mediumtext,
|
|
`referrer` mediumtext DEFAULT NULL,
|
|
`http_code` int(11) unsigned NOT NULL DEFAULT 0,
|
|
`request_method` varchar(10) DEFAULT NULL,
|
|
`request_data` mediumtext DEFAULT NULL,
|
|
`ip` varchar(45) DEFAULT NULL,
|
|
`created` datetime NOT NULL,
|
|
`updated` datetime NOT NULL,
|
|
PRIMARY KEY (`id`),
|
|
KEY ndx_aioseo_redirects_404_logs_created (`created`),
|
|
KEY ndx_aioseo_redirects_404_logs_ip (`ip`)
|
|
) {$charsetCollate};"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Migrate redirect default options.
|
|
*
|
|
* @since 4.9.1
|
|
*
|
|
* @return void
|
|
*/
|
|
private function migrateRedirectDefaults() {
|
|
if ( empty( aioseo()->redirects ) ) {
|
|
return;
|
|
}
|
|
|
|
$dbOptions = json_decode( get_option( aioseo()->redirects->options->optionsName ), true );
|
|
if ( empty( $dbOptions['defaults'] ) ) {
|
|
return;
|
|
}
|
|
|
|
if ( isset( $dbOptions['defaults']['ignoreCase'] ) ) {
|
|
aioseo()->redirects->options->redirectDefaults->ignoreCase = $dbOptions['defaults']['ignoreCase'];
|
|
}
|
|
|
|
if ( isset( $dbOptions['defaults']['ignoreSlash'] ) ) {
|
|
aioseo()->redirects->options->redirectDefaults->ignoreSlash = $dbOptions['defaults']['ignoreSlash'];
|
|
}
|
|
|
|
if ( isset( $dbOptions['defaults']['redirectType'] ) ) {
|
|
aioseo()->redirects->options->redirectDefaults->redirectType = $dbOptions['defaults']['redirectType'];
|
|
}
|
|
|
|
if ( isset( $dbOptions['defaults']['queryParam'] ) ) {
|
|
aioseo()->redirects->options->redirectDefaults->queryParam = $dbOptions['defaults']['queryParam'];
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* Migrate new options for advanced 404s.
|
|
*
|
|
* @since 4.9.1
|
|
*
|
|
* @return void
|
|
*/
|
|
private function redirects404MigrateOptions() {
|
|
if ( empty( aioseo()->redirects ) ) {
|
|
return;
|
|
}
|
|
|
|
// Enable 404 redirects if it was already on.
|
|
$oldOptions = aioseo()->redirects->options->advanced404s->all();
|
|
if (
|
|
! empty( $oldOptions['redirectToHome'] ) ||
|
|
! empty( $oldOptions['redirectToParent'] )
|
|
) {
|
|
aioseo()->redirects->options->advanced404s->enabled = true;
|
|
|
|
// Enable 404 default redirects if it was active for Home.
|
|
if ( ! empty( $oldOptions['redirectToHome'] ) ) {
|
|
aioseo()->redirects->options->advanced404s->redirectDefaultEnabled = true;
|
|
}
|
|
|
|
aioseo()->redirects->options->save( true );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds a column for custom redirect matching rules.
|
|
*
|
|
* @since 4.9.1
|
|
*
|
|
* @return void
|
|
*/
|
|
private function addPostIdColumn() {
|
|
if ( ! aioseo()->core->db->columnExists( 'aioseo_redirects', 'post_id' ) ) {
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_redirects';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
ADD post_id bigint unsigned DEFAULT NULL AFTER id,
|
|
ADD INDEX ndx_aioseo_redirects_post_id (post_id)"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Fixes source url matches that got double encoded.
|
|
*
|
|
* @since 4.9.1
|
|
*
|
|
* @return void
|
|
*/
|
|
private function fixDoubleEncodedSourceUrlMatch() {
|
|
$redirects = aioseo()->core->db->start( 'aioseo_redirects' )
|
|
->whereRaw( 'source_url_match REGEXP "%[0-9a-f]{2}"' )
|
|
->run()
|
|
->models( 'AIOSEO\\Plugin\\Pro\\Redirects\\Models\\Redirect' );
|
|
|
|
foreach ( $redirects as $redirect ) {
|
|
// This will trigger the model's transform which will regenerate source_url_match and hash.
|
|
$redirect->save();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds url_hash columns to both log tables for performance optimization.
|
|
*
|
|
* @since 4.9.1
|
|
*
|
|
* @return void
|
|
*/
|
|
private function addUrlHashColumns() {
|
|
$db = aioseo()->core->db->db;
|
|
|
|
// Add url_hash column to aioseo_redirects_logs table
|
|
if ( ! aioseo()->core->db->columnExists( 'aioseo_redirects_logs', 'url_hash' ) ) {
|
|
$tableName = $db->prefix . 'aioseo_redirects_logs';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
ADD url_hash VARCHAR(40) DEFAULT NULL AFTER url,
|
|
ADD INDEX ndx_aioseo_redirects_404_logs_url_hash (url_hash)"
|
|
);
|
|
|
|
// Populate existing records with url_hash
|
|
aioseo()->core->db->execute(
|
|
"UPDATE {$tableName} SET url_hash = SHA1(url) WHERE url_hash IS NULL"
|
|
);
|
|
}
|
|
|
|
// Add url_hash column to aioseo_redirects_404_logs table
|
|
if ( ! aioseo()->core->db->columnExists( 'aioseo_redirects_404_logs', 'url_hash' ) ) {
|
|
$tableName = $db->prefix . 'aioseo_redirects_404_logs';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
ADD url_hash VARCHAR(40) DEFAULT NULL AFTER url,
|
|
ADD INDEX ndx_aioseo_redirects_404_logs_url_hash (url_hash)"
|
|
);
|
|
|
|
// Populate existing records with url_hash
|
|
aioseo()->core->db->execute(
|
|
"UPDATE {$tableName} SET url_hash = SHA1(url) WHERE url_hash IS NULL"
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Migrates Pro-specific sensitive values (license keys) to the new SensitiveOptions storage.
|
|
*
|
|
* @since 4.9.6
|
|
*
|
|
* @return void
|
|
*/
|
|
private function migrateProSensitiveOptions() {
|
|
$rawOptions = json_decode( (string) get_option( 'aioseo_options', '' ), true );
|
|
if ( is_array( $rawOptions ) ) {
|
|
$licenseKey = aioseo()->helpers->getNestedValue( $rawOptions, [ 'general', 'licenseKey' ] );
|
|
if ( ! empty( $licenseKey ) && is_string( $licenseKey ) ) {
|
|
aioseo()->sensitiveOptions->set( 'licenseKey', $licenseKey );
|
|
}
|
|
|
|
// Remove the old license key from the options.
|
|
unset( $rawOptions['general']['licenseKey'] );
|
|
update_option( 'aioseo_options', wp_json_encode( $rawOptions ), false );
|
|
}
|
|
|
|
$rawProOptions = json_decode( (string) get_option( 'aioseo_options_pro', '' ), true );
|
|
if ( is_array( $rawProOptions ) ) {
|
|
$licenseKey = aioseo()->helpers->getNestedValue( $rawProOptions, [ 'general', 'licenseKey' ] );
|
|
if ( ! empty( $licenseKey ) && is_string( $licenseKey ) ) {
|
|
aioseo()->sensitiveOptions->set( 'licenseKey', $licenseKey );
|
|
}
|
|
|
|
// Remove the old license key from the options.
|
|
unset( $rawProOptions['general']['licenseKey'] );
|
|
update_option( 'aioseo_options_pro', wp_json_encode( $rawProOptions ), false );
|
|
}
|
|
|
|
// Force save the sensitive options.
|
|
aioseo()->sensitiveOptions->save( true );
|
|
|
|
// Migrate network license key.
|
|
if ( ! empty( aioseo()->networkSensitiveOptions ) ) {
|
|
aioseo()->helpers->switchToBlog( aioseo()->helpers->getNetworkId() );
|
|
|
|
$rawNetworkOptions = json_decode( (string) get_option( 'aioseo_options_network_pro', '' ), true );
|
|
if ( is_array( $rawNetworkOptions ) ) {
|
|
$networkLicenseKey = aioseo()->helpers->getNestedValue( $rawNetworkOptions, [ 'general', 'licenseKey' ] );
|
|
if ( ! empty( $networkLicenseKey ) && is_string( $networkLicenseKey ) ) {
|
|
aioseo()->networkSensitiveOptions->set( 'networkLicenseKey', $networkLicenseKey );
|
|
}
|
|
|
|
// Remove the old license key from the network options.
|
|
unset( $rawNetworkOptions['general']['licenseKey'] );
|
|
update_option( 'aioseo_options_network_pro', wp_json_encode( $rawNetworkOptions ) );
|
|
}
|
|
|
|
aioseo()->helpers->restoreCurrentBlog();
|
|
aioseo()->networkSensitiveOptions->save( true );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Removes the old Redirects addon.
|
|
*
|
|
* @since 4.9.1
|
|
*
|
|
* @return void
|
|
*/
|
|
private function removeOldRedirectsAddon() {
|
|
$pluginSlug = 'aioseo-redirects/aioseo-redirects.php';
|
|
|
|
if ( ! function_exists( 'get_plugins' ) ) {
|
|
require_once ABSPATH . 'wp-admin/includes/plugin.php';
|
|
}
|
|
|
|
$allPlugins = get_plugins();
|
|
if ( ! isset( $allPlugins[ $pluginSlug ] ) ) {
|
|
// Plugin doesn't exist, nothing to do.
|
|
return;
|
|
}
|
|
|
|
if ( is_plugin_active( $pluginSlug ) || is_plugin_active_for_network( $pluginSlug ) ) {
|
|
deactivate_plugins( $pluginSlug, true ); // Silent deactivation.
|
|
}
|
|
|
|
if ( ! function_exists( 'delete_plugins' ) ) {
|
|
require_once ABSPATH . 'wp-admin/includes/plugin.php';
|
|
}
|
|
|
|
delete_plugins( [ $pluginSlug ] );
|
|
}
|
|
|
|
/**
|
|
* Adds the comment column to the redirects table.
|
|
*
|
|
* @since 4.9.6
|
|
*
|
|
* @return void
|
|
*/
|
|
private function addRedirectsCommentColumn() {
|
|
if ( ! aioseo()->core->db->columnExists( 'aioseo_redirects', 'comment' ) ) {
|
|
$tableName = aioseo()->core->db->db->prefix . 'aioseo_redirects';
|
|
aioseo()->core->db->execute(
|
|
"ALTER TABLE {$tableName}
|
|
ADD comment text DEFAULT NULL AFTER enabled"
|
|
);
|
|
|
|
aioseo()->core->cache->delete( 'db_schema' );
|
|
}
|
|
}
|
|
} |