Core Upgrader: Add a locking mechanism to avoid two concurrent updates of WordPress occuring.

Fixes #34878


git-svn-id: https://develop.svn.wordpress.org/trunk@36349 602fd350-edb4-49c9-b593-d223f7449a82
This commit is contained in:
Dion Hulse 2016-01-19 05:06:46 +00:00
parent 5eee5cfc3c
commit b9e62671f6
2 changed files with 88 additions and 27 deletions

View File

@ -751,6 +751,64 @@ class WP_Upgrader {
} }
} }
/**
* Create a Lock using WordPress options.
*
* @since 4.5.0
* @access public
*
* @param string $lock_name The name of this unique lock.
* @param int $release_timeout The duration in seconds to respect an existing lock. Default: 1 hour.
* @return bool
*/
public function create_lock( $lock_name, $release_timeout = null ) {
global $wpdb;
if ( ! $release_timeout ) {
$release_timeout = HOUR_IN_SECONDS;
}
$lock_option = $lock_name . '.lock';
// Try to lock
$lock_result = $wpdb->query( $wpdb->prepare( "INSERT IGNORE INTO `$wpdb->options` ( `option_name`, `option_value`, `autoload` ) VALUES (%s, %s, 'no') /* LOCK */", $lock_option, time() ) );
if ( ! $lock_result ) {
$lock_result = get_option( $lock_option );
// If we couldn't create a lock, and there isn't a lock, bail
if ( ! $lock_result ) {
return false;
}
// Check to see if the lock is still valid
if ( $lock_result > ( time() - $release_timeout ) ) {
return false;
}
// There must exist an expired lock, clear it and re-gain it.
$this->release_lock( $lock_name );
return $this->create_lock( $lock_name, $release_timeout );
}
// Update the lock, as by this point we've definitely got a lock, just need to fire the actions
update_option( $lock_option, time() );
return true;
}
/**
* Release a lock created by `WP_Upgrader::create_lock()`.
*
* @since 4.5.0
* @access public
*
* @param string $lock_name The name of this unique lock.
* @return bool
*/
public function release_lock( $lock_name ) {
return delete_option( $lock_name . '.lock' );
}
} }
/** /**
@ -2168,6 +2226,7 @@ class Core_Upgrader extends WP_Upgrader {
*/ */
public function upgrade_strings() { public function upgrade_strings() {
$this->strings['up_to_date'] = __('WordPress is at the latest version.'); $this->strings['up_to_date'] = __('WordPress is at the latest version.');
$this->strings['locked'] = __('Another update is currently in progress.');
$this->strings['no_package'] = __('Update package not available.'); $this->strings['no_package'] = __('Update package not available.');
$this->strings['downloading_package'] = __('Downloading update from <span class="code">%s</span>&#8230;'); $this->strings['downloading_package'] = __('Downloading update from <span class="code">%s</span>&#8230;');
$this->strings['unpack_package'] = __('Unpacking the update&#8230;'); $this->strings['unpack_package'] = __('Unpacking the update&#8230;');
@ -2252,25 +2311,38 @@ class Core_Upgrader extends WP_Upgrader {
else else
$to_download = 'full'; $to_download = 'full';
// Lock to prevent multiple Core Updates occuring
$lock = $this->create_lock( 'core_updater', 15 * MINUTE_IN_SECONDS );
if ( ! $lock ) {
return new WP_Error( 'locked', $this->strings['locked'] );
}
$download = $this->download_package( $current->packages->$to_download ); $download = $this->download_package( $current->packages->$to_download );
if ( is_wp_error($download) ) if ( is_wp_error( $download ) ) {
$this->release_lock( 'core_updater' );
return $download; return $download;
}
$working_dir = $this->unpack_package( $download ); $working_dir = $this->unpack_package( $download );
if ( is_wp_error($working_dir) ) if ( is_wp_error( $working_dir ) ) {
$this->release_lock( 'core_updater' );
return $working_dir; return $working_dir;
}
// Copy update-core.php from the new version into place. // Copy update-core.php from the new version into place.
if ( !$wp_filesystem->copy($working_dir . '/wordpress/wp-admin/includes/update-core.php', $wp_dir . 'wp-admin/includes/update-core.php', true) ) { if ( !$wp_filesystem->copy($working_dir . '/wordpress/wp-admin/includes/update-core.php', $wp_dir . 'wp-admin/includes/update-core.php', true) ) {
$wp_filesystem->delete($working_dir, true); $wp_filesystem->delete($working_dir, true);
$this->release_lock( 'core_updater' );
return new WP_Error( 'copy_failed_for_update_core_file', __( 'The update cannot be installed because we will be unable to copy some files. This is usually due to inconsistent file permissions.' ), 'wp-admin/includes/update-core.php' ); return new WP_Error( 'copy_failed_for_update_core_file', __( 'The update cannot be installed because we will be unable to copy some files. This is usually due to inconsistent file permissions.' ), 'wp-admin/includes/update-core.php' );
} }
$wp_filesystem->chmod($wp_dir . 'wp-admin/includes/update-core.php', FS_CHMOD_FILE); $wp_filesystem->chmod($wp_dir . 'wp-admin/includes/update-core.php', FS_CHMOD_FILE);
require_once( ABSPATH . 'wp-admin/includes/update-core.php' ); require_once( ABSPATH . 'wp-admin/includes/update-core.php' );
if ( ! function_exists( 'update_core' ) ) if ( ! function_exists( 'update_core' ) ) {
$this->release_lock( 'core_updater' );
return new WP_Error( 'copy_failed_space', $this->strings['copy_failed_space'] ); return new WP_Error( 'copy_failed_space', $this->strings['copy_failed_space'] );
}
$result = update_core( $working_dir, $wp_dir ); $result = update_core( $working_dir, $wp_dir );
@ -2345,6 +2417,8 @@ class Core_Upgrader extends WP_Upgrader {
wp_version_check( $stats ); wp_version_check( $stats );
} }
$this->release_lock( 'core_updater' );
return $result; return $result;
} }
@ -2938,8 +3012,13 @@ class WP_Automatic_Updater {
$upgrade_result = new WP_Error( 'fs_unavailable', __( 'Could not access filesystem.' ) ); $upgrade_result = new WP_Error( 'fs_unavailable', __( 'Could not access filesystem.' ) );
} }
// Core doesn't output this, so let's append it so we don't get confused.
if ( 'core' == $type ) { if ( 'core' == $type ) {
if ( is_wp_error( $upgrade_result ) && ( 'up_to_date' == $upgrade_result->get_error_code() || 'locked' == $upgrade_result->get_error_code() ) ) {
// These aren't actual errors, treat it as a skipped-update instead to avoid triggering the post-core update failure routines.
return false;
}
// Core doesn't output this, so let's append it so we don't get confused.
if ( is_wp_error( $upgrade_result ) ) { if ( is_wp_error( $upgrade_result ) ) {
$skin->error( __( 'Installation Failed' ), $upgrade_result ); $skin->error( __( 'Installation Failed' ), $upgrade_result );
} else { } else {
@ -2975,25 +3054,8 @@ class WP_Automatic_Updater {
if ( ! is_main_network() || ! is_main_site() ) if ( ! is_main_network() || ! is_main_site() )
return; return;
$lock_name = 'auto_updater.lock'; if ( ! $this->create_lock( 'auto_updater' ) )
return;
// Try to lock
$lock_result = $wpdb->query( $wpdb->prepare( "INSERT IGNORE INTO `$wpdb->options` ( `option_name`, `option_value`, `autoload` ) VALUES (%s, %s, 'no') /* LOCK */", $lock_name, time() ) );
if ( ! $lock_result ) {
$lock_result = get_option( $lock_name );
// If we couldn't create a lock, and there isn't a lock, bail
if ( ! $lock_result )
return;
// Check to see if the lock is still valid
if ( $lock_result > ( time() - HOUR_IN_SECONDS ) )
return;
}
// Update the lock, as by this point we've definitely got a lock, just need to fire the actions
update_option( $lock_name, time() );
// Don't automatically run these thins, as we'll handle it ourselves // Don't automatically run these thins, as we'll handle it ourselves
remove_action( 'upgrader_process_complete', array( 'Language_Pack_Upgrader', 'async_upgrade' ), 20 ); remove_action( 'upgrader_process_complete', array( 'Language_Pack_Upgrader', 'async_upgrade' ), 20 );
@ -3092,8 +3154,7 @@ class WP_Automatic_Updater {
do_action( 'automatic_updates_complete', $this->update_results ); do_action( 'automatic_updates_complete', $this->update_results );
} }
// Clear the lock $this->release_lock( 'auto_updater' );
delete_option( $lock_name );
} }
/** /**
@ -3163,7 +3224,7 @@ class WP_Automatic_Updater {
* the issue could actually be on WordPress.org's side.) If that one fails, then email. * the issue could actually be on WordPress.org's side.) If that one fails, then email.
*/ */
$send = true; $send = true;
$transient_failures = array( 'incompatible_archive', 'download_failed', 'insane_distro' ); $transient_failures = array( 'incompatible_archive', 'download_failed', 'insane_distro', 'locked' );
if ( in_array( $error_code, $transient_failures ) && ! get_site_option( 'auto_core_update_failed' ) ) { if ( in_array( $error_code, $transient_failures ) && ! get_site_option( 'auto_core_update_failed' ) ) {
wp_schedule_single_event( time() + HOUR_IN_SECONDS, 'wp_maybe_auto_update' ); wp_schedule_single_event( time() + HOUR_IN_SECONDS, 'wp_maybe_auto_update' );
$send = false; $send = false;

View File

@ -484,7 +484,7 @@ function do_core_upgrade( $reinstall = false ) {
if ( is_wp_error($result) ) { if ( is_wp_error($result) ) {
show_message($result); show_message($result);
if ('up_to_date' != $result->get_error_code() ) if ( 'up_to_date' != $result->get_error_code() && 'locked' != $result->get_error_code() )
show_message( __('Installation Failed') ); show_message( __('Installation Failed') );
echo '</div>'; echo '</div>';
return; return;