2009-04-19 21:36:28 +02:00
< ? php
/**
2009-04-26 20:17:19 +02:00
* A File upgrader class for WordPress .
2009-04-19 21:36:28 +02:00
*
2009-04-26 20:17:19 +02:00
* This set of classes are designed to be used to upgrade / install a local set of files on the filesystem via the Filesystem Abstraction classes .
2009-04-19 21:36:28 +02:00
*
2009-04-26 20:17:19 +02:00
* @ link http :// trac . wordpress . org / ticket / 7875 consolidate plugin / theme / core upgrade / install functions
*
* @ package WordPress
* @ subpackage Upgrader
* @ since 2.8 . 0
2009-04-19 21:36:28 +02:00
*/
2013-09-04 08:34:38 +02:00
require ABSPATH . 'wp-admin/includes/class-wp-upgrader-skins.php' ;
2009-04-26 20:17:19 +02:00
/**
* WordPress Upgrader class for Upgrading / Installing a local set of files via the Filesystem Abstraction classes from a Zip file .
*
* @ package WordPress
* @ subpackage Upgrader
* @ since 2.8 . 0
*/
2009-04-19 21:36:28 +02:00
class WP_Upgrader {
var $strings = array ();
var $skin = null ;
var $result = array ();
2009-04-20 20:18:39 +02:00
2009-04-21 10:10:41 +02:00
function __construct ( $skin = null ) {
2009-04-19 21:36:28 +02:00
if ( null == $skin )
$this -> skin = new WP_Upgrader_Skin ();
else
$this -> skin = $skin ;
2009-04-26 20:17:19 +02:00
}
2009-05-25 01:47:49 +02:00
2009-04-26 20:17:19 +02:00
function init () {
2009-04-19 21:36:28 +02:00
$this -> skin -> set_upgrader ( $this );
2009-04-26 20:17:19 +02:00
$this -> generic_strings ();
2009-04-19 21:36:28 +02:00
}
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
function generic_strings () {
$this -> strings [ 'bad_request' ] = __ ( 'Invalid Data provided.' );
$this -> strings [ 'fs_unavailable' ] = __ ( 'Could not access filesystem.' );
2010-01-21 22:37:43 +01:00
$this -> strings [ 'fs_error' ] = __ ( 'Filesystem error.' );
2009-04-19 21:36:28 +02:00
$this -> strings [ 'fs_no_root_dir' ] = __ ( 'Unable to locate WordPress Root directory.' );
$this -> strings [ 'fs_no_content_dir' ] = __ ( 'Unable to locate WordPress Content directory (wp-content).' );
$this -> strings [ 'fs_no_plugins_dir' ] = __ ( 'Unable to locate WordPress Plugin directory.' );
$this -> strings [ 'fs_no_themes_dir' ] = __ ( 'Unable to locate WordPress Theme directory.' );
2010-04-21 12:58:06 +02:00
/* translators: %s: directory name */
2009-04-19 21:36:28 +02:00
$this -> strings [ 'fs_no_folder' ] = __ ( 'Unable to locate needed folder (%s).' );
$this -> strings [ 'download_failed' ] = __ ( 'Download failed.' );
2009-12-24 10:46:57 +01:00
$this -> strings [ 'installing_package' ] = __ ( 'Installing the latest version…' );
2013-05-22 18:57:20 +02:00
$this -> strings [ 'no_files' ] = __ ( 'The package contains no files.' );
2009-04-19 21:36:28 +02:00
$this -> strings [ 'folder_exists' ] = __ ( 'Destination folder already exists.' );
$this -> strings [ 'mkdir_failed' ] = __ ( 'Could not create directory.' );
2011-11-10 20:41:44 +01:00
$this -> strings [ 'incompatible_archive' ] = __ ( 'The package could not be installed.' );
2009-04-20 20:18:39 +02:00
2009-12-24 10:46:57 +01:00
$this -> strings [ 'maintenance_start' ] = __ ( 'Enabling Maintenance mode…' );
$this -> strings [ 'maintenance_end' ] = __ ( 'Disabling Maintenance mode…' );
2009-04-19 21:36:28 +02:00
}
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
function fs_connect ( $directories = array () ) {
global $wp_filesystem ;
2009-04-20 20:18:39 +02:00
2009-04-20 20:15:08 +02:00
if ( false === ( $credentials = $this -> skin -> request_filesystem_credentials ()) )
2009-04-19 21:36:28 +02:00
return false ;
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
if ( ! WP_Filesystem ( $credentials ) ) {
$error = true ;
if ( is_object ( $wp_filesystem ) && $wp_filesystem -> errors -> get_error_code () )
$error = $wp_filesystem -> errors ;
$this -> skin -> request_filesystem_credentials ( $error ); //Failed to connect, Error and request again
return false ;
}
if ( ! is_object ( $wp_filesystem ) )
return new WP_Error ( 'fs_unavailable' , $this -> strings [ 'fs_unavailable' ] );
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
if ( is_wp_error ( $wp_filesystem -> errors ) && $wp_filesystem -> errors -> get_error_code () )
return new WP_Error ( 'fs_error' , $this -> strings [ 'fs_error' ], $wp_filesystem -> errors );
foreach ( ( array ) $directories as $dir ) {
2009-08-21 03:36:01 +02:00
switch ( $dir ) {
case ABSPATH :
if ( ! $wp_filesystem -> abspath () )
return new WP_Error ( 'fs_no_root_dir' , $this -> strings [ 'fs_no_root_dir' ]);
break ;
case WP_CONTENT_DIR :
if ( ! $wp_filesystem -> wp_content_dir () )
return new WP_Error ( 'fs_no_content_dir' , $this -> strings [ 'fs_no_content_dir' ]);
break ;
case WP_PLUGIN_DIR :
if ( ! $wp_filesystem -> wp_plugins_dir () )
return new WP_Error ( 'fs_no_plugins_dir' , $this -> strings [ 'fs_no_plugins_dir' ]);
break ;
2013-08-22 06:49:06 +02:00
case get_theme_root () :
if ( ! $wp_filesystem -> wp_themes_dir () )
2009-08-21 03:36:01 +02:00
return new WP_Error ( 'fs_no_themes_dir' , $this -> strings [ 'fs_no_themes_dir' ]);
break ;
default :
if ( ! $wp_filesystem -> find_folder ( $dir ) )
2013-06-21 07:54:40 +02:00
return new WP_Error ( 'fs_no_folder' , sprintf ( $this -> strings [ 'fs_no_folder' ], esc_html ( basename ( $dir ) ) ) );
2009-08-21 03:36:01 +02:00
break ;
}
2009-04-19 21:36:28 +02:00
}
return true ;
} //end fs_connect();
function download_package ( $package ) {
2013-09-13 14:52:28 +02:00
/**
* Filter whether to return the package .
*
* @ since 3.7 . 0
*
* @ param bool $reply Whether to bail without returning the package . Default is false .
* @ param string $package The package file name .
* @ param object $this The WP_Upgrader instance .
*/
$reply = apply_filters ( 'upgrader_pre_download' , false , $package , $this );
if ( false !== $reply )
return $reply ;
2009-04-19 21:36:28 +02:00
if ( ! preg_match ( '!^(http|https|ftp)://!i' , $package ) && file_exists ( $package ) ) //Local file or remote?
return $package ; //must be a local file..
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
if ( empty ( $package ) )
return new WP_Error ( 'no_package' , $this -> strings [ 'no_package' ]);
$this -> skin -> feedback ( 'downloading_package' , $package );
$download_file = download_url ( $package );
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
if ( is_wp_error ( $download_file ) )
return new WP_Error ( 'download_failed' , $this -> strings [ 'download_failed' ], $download_file -> get_error_message ());
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
return $download_file ;
}
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
function unpack_package ( $package , $delete_package = true ) {
global $wp_filesystem ;
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
$this -> skin -> feedback ( 'unpack_package' );
$upgrade_folder = $wp_filesystem -> wp_content_dir () . 'upgrade/' ;
//Clean up contents of upgrade directory beforehand.
$upgrade_files = $wp_filesystem -> dirlist ( $upgrade_folder );
if ( ! empty ( $upgrade_files ) ) {
foreach ( $upgrade_files as $file )
$wp_filesystem -> delete ( $upgrade_folder . $file [ 'name' ], true );
}
//We need a working directory
$working_dir = $upgrade_folder . basename ( $package , '.zip' );
// Clean up working directory
if ( $wp_filesystem -> is_dir ( $working_dir ) )
$wp_filesystem -> delete ( $working_dir , true );
// Unzip package to working directory
$result = unzip_file ( $package , $working_dir ); //TODO optimizations, Copy when Move/Rename would suffice?
// Once extracted, delete the package if required.
if ( $delete_package )
unlink ( $package );
if ( is_wp_error ( $result ) ) {
$wp_filesystem -> delete ( $working_dir , true );
2011-08-28 17:02:48 +02:00
if ( 'incompatible_archive' == $result -> get_error_code () ) {
return new WP_Error ( 'incompatible_archive' , $this -> strings [ 'incompatible_archive' ], $result -> get_error_data () );
}
2009-04-19 21:36:28 +02:00
return $result ;
}
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
return $working_dir ;
}
2009-04-20 20:18:39 +02:00
2013-09-06 03:32:09 +02:00
function install_package ( $args = array () ) {
2013-08-22 06:49:06 +02:00
global $wp_filesystem , $wp_theme_directories ;
2013-09-06 03:32:09 +02:00
$defaults = array (
'source' => '' , // Please always pass this
'destination' => '' , // and this
'clear_destination' => false ,
'clear_working' => false ,
'abort_if_destination_exists' => true ,
'hook_extra' => array ()
);
2009-04-19 21:36:28 +02:00
$args = wp_parse_args ( $args , $defaults );
extract ( $args );
@ set_time_limit ( 300 );
if ( empty ( $source ) || empty ( $destination ) )
return new WP_Error ( 'bad_request' , $this -> strings [ 'bad_request' ]);
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
$this -> skin -> feedback ( 'installing_package' );
$res = apply_filters ( 'upgrader_pre_install' , true , $hook_extra );
if ( is_wp_error ( $res ) )
return $res ;
//Retain the Original source and destinations
$remote_source = $source ;
$local_destination = $destination ;
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
$source_files = array_keys ( $wp_filesystem -> dirlist ( $remote_source ) );
$remote_destination = $wp_filesystem -> find_folder ( $local_destination );
//Locate which directory to copy to the new folder, This is based on the actual folder holding the files.
if ( 1 == count ( $source_files ) && $wp_filesystem -> is_dir ( trailingslashit ( $source ) . $source_files [ 0 ] . '/' ) ) //Only one folder? Then we want its contents.
$source = trailingslashit ( $source ) . trailingslashit ( $source_files [ 0 ]);
elseif ( count ( $source_files ) == 0 )
2013-10-11 18:05:13 +02:00
return new WP_Error ( 'incompatible_archive_empty' , $this -> strings [ 'incompatible_archive' ], $this -> strings [ 'no_files' ] ); // There are no files?
2012-12-20 16:55:32 +01:00
else //It's only a single file, the upgrader will use the foldername of this file as the destination folder. foldername is based on zip filename.
2011-11-02 13:16:47 +01:00
$source = trailingslashit ( $source );
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
//Hook ability to change the source file location..
$source = apply_filters ( 'upgrader_source_selection' , $source , $remote_source , $this );
if ( is_wp_error ( $source ) )
return $source ;
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
//Has the source location changed? If so, we need a new source_files list.
if ( $source !== $remote_source )
$source_files = array_keys ( $wp_filesystem -> dirlist ( $source ) );
2009-04-20 20:18:39 +02:00
2013-08-22 06:49:06 +02:00
// Protection against deleting files in any important base directories.
// Theme_Upgrader & Plugin_Upgrader also trigger this, as they pass the destination directory (WP_PLUGIN_DIR / wp-content/themes)
// intending to copy the directory into the directory, whilst they pass the source as the actual files to copy.
2013-08-30 09:34:24 +02:00
$protected_directories = array ( ABSPATH , WP_CONTENT_DIR , WP_PLUGIN_DIR , WP_CONTENT_DIR . '/themes' );
if ( is_array ( $wp_theme_directories ) )
$protected_directories = array_merge ( $protected_directories , $wp_theme_directories );
if ( in_array ( $destination , $protected_directories ) ) {
2009-04-19 21:36:28 +02:00
$remote_destination = trailingslashit ( $remote_destination ) . trailingslashit ( basename ( $source ));
$destination = trailingslashit ( $destination ) . trailingslashit ( basename ( $source ));
}
2011-04-03 13:54:07 +02:00
if ( $clear_destination ) {
2011-09-03 16:18:10 +02:00
//We're going to clear the destination if there's something there
2011-04-03 13:54:07 +02:00
$this -> skin -> feedback ( 'remove_old' );
$removed = true ;
if ( $wp_filesystem -> exists ( $remote_destination ) )
2009-04-19 21:36:28 +02:00
$removed = $wp_filesystem -> delete ( $remote_destination , true );
2011-04-03 13:54:07 +02:00
$removed = apply_filters ( 'upgrader_clear_destination' , $removed , $local_destination , $remote_destination , $hook_extra );
if ( is_wp_error ( $removed ) )
return $removed ;
else if ( ! $removed )
return new WP_Error ( 'remove_old_failed' , $this -> strings [ 'remove_old_failed' ]);
2013-04-04 23:10:54 +02:00
} elseif ( $abort_if_destination_exists && $wp_filesystem -> exists ( $remote_destination ) ) {
2011-09-03 16:18:10 +02:00
//If we're not clearing the destination folder and something exists there already, Bail.
2011-04-03 13:54:07 +02:00
//But first check to see if there are actually any files in the folder.
$_files = $wp_filesystem -> dirlist ( $remote_destination );
if ( ! empty ( $_files ) ) {
$wp_filesystem -> delete ( $remote_source , true ); //Clear out the source files.
return new WP_Error ( 'folder_exists' , $this -> strings [ 'folder_exists' ], $remote_destination );
2009-09-14 01:00:45 +02:00
}
2009-04-19 21:36:28 +02:00
}
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
//Create destination if needed
if ( ! $wp_filesystem -> exists ( $remote_destination ) )
if ( ! $wp_filesystem -> mkdir ( $remote_destination , FS_CHMOD_DIR ) )
2013-10-11 18:05:13 +02:00
return new WP_Error ( 'mkdir_failed_destination' , $this -> strings [ 'mkdir_failed' ], $remote_destination );
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
// Copy new version of item into place.
$result = copy_dir ( $source , $remote_destination );
if ( is_wp_error ( $result ) ) {
if ( $clear_working )
$wp_filesystem -> delete ( $remote_source , true );
return $result ;
}
2009-04-20 20:18:39 +02:00
//Clear the Working folder?
2009-04-19 21:36:28 +02:00
if ( $clear_working )
$wp_filesystem -> delete ( $remote_source , true );
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
$destination_name = basename ( str_replace ( $local_destination , '' , $destination ) );
if ( '.' == $destination_name )
$destination_name = '' ;
$this -> result = compact ( 'local_source' , 'source' , 'source_name' , 'source_files' , 'destination' , 'destination_name' , 'local_destination' , 'remote_destination' , 'clear_destination' , 'delete_source_dir' );
$res = apply_filters ( 'upgrader_post_install' , true , $hook_extra , $this -> result );
if ( is_wp_error ( $res ) ) {
$this -> result = $res ;
return $res ;
}
//Bombard the calling function will all the info which we've just used.
return $this -> result ;
}
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
function run ( $options ) {
2013-09-06 03:32:09 +02:00
$defaults = array (
'package' => '' , // Please always pass this.
'destination' => '' , // And this
'clear_destination' => false ,
'abort_if_destination_exists' => true , // Abort if the Destination directory exists, Pass clear_destination as false please
'clear_working' => true ,
'is_multi' => false ,
'hook_extra' => array () // Pass any extra $hook_extra args here, this will be passed to any hooked filters.
);
2009-04-19 21:36:28 +02:00
$options = wp_parse_args ( $options , $defaults );
extract ( $options );
//Connect to the Filesystem first.
$res = $this -> fs_connect ( array ( WP_CONTENT_DIR , $destination ) );
if ( ! $res ) //Mainly for non-connected filesystem.
return false ;
2013-09-12 04:42:10 +02:00
if ( ! $is_multi ) // call $this->header separately if running multiple times
$this -> skin -> header ();
$this -> skin -> before ();
2009-04-19 21:36:28 +02:00
if ( is_wp_error ( $res ) ) {
$this -> skin -> error ( $res );
2013-09-12 04:42:10 +02:00
$this -> skin -> after ();
if ( ! $is_multi )
$this -> skin -> footer ();
2009-04-19 21:36:28 +02:00
return $res ;
}
2009-04-20 20:15:08 +02:00
2009-04-19 21:36:28 +02:00
//Download the package (Note, This just returns the filename of the file if the package is a local file)
$download = $this -> download_package ( $package );
if ( is_wp_error ( $download ) ) {
$this -> skin -> error ( $download );
2010-03-06 09:39:50 +01:00
$this -> skin -> after ();
2013-09-12 04:42:10 +02:00
if ( ! $is_multi )
$this -> skin -> footer ();
2009-04-19 21:36:28 +02:00
return $download ;
}
2009-04-20 20:18:39 +02:00
2011-04-19 07:54:01 +02:00
$delete_package = ( $download != $package ); // Do not delete a "local" file
2011-09-03 16:18:10 +02:00
//Unzips the file into a temporary directory
2011-04-19 07:54:01 +02:00
$working_dir = $this -> unpack_package ( $download , $delete_package );
2009-04-19 21:36:28 +02:00
if ( is_wp_error ( $working_dir ) ) {
$this -> skin -> error ( $working_dir );
2010-03-06 09:39:50 +01:00
$this -> skin -> after ();
2013-09-12 04:42:10 +02:00
if ( ! $is_multi )
$this -> skin -> footer ();
2009-04-19 21:36:28 +02:00
return $working_dir ;
}
//With the given options, this installs it to the destination directory.
2009-04-20 20:18:39 +02:00
$result = $this -> install_package ( array (
2013-09-06 03:32:09 +02:00
'source' => $working_dir ,
'destination' => $destination ,
'clear_destination' => $clear_destination ,
'abort_if_destination_exists' => $abort_if_destination_exists ,
'clear_working' => $clear_working ,
'hook_extra' => $hook_extra
) );
2009-04-19 21:36:28 +02:00
$this -> skin -> set_result ( $result );
if ( is_wp_error ( $result ) ) {
$this -> skin -> error ( $result );
$this -> skin -> feedback ( 'process_failed' );
} else {
2011-09-03 16:18:10 +02:00
//Install Succeeded
2009-04-19 21:36:28 +02:00
$this -> skin -> feedback ( 'process_success' );
}
2013-09-12 04:42:10 +02:00
2009-04-19 21:36:28 +02:00
$this -> skin -> after ();
2009-10-26 08:04:30 +01:00
2013-09-12 04:42:10 +02:00
if ( ! $is_multi )
2009-10-26 08:04:30 +01:00
$this -> skin -> footer ();
2009-04-19 21:36:28 +02:00
return $result ;
}
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
function maintenance_mode ( $enable = false ) {
global $wp_filesystem ;
$file = $wp_filesystem -> abspath () . '.maintenance' ;
if ( $enable ) {
$this -> skin -> feedback ( 'maintenance_start' );
// Create maintenance file to signal that we are upgrading
$maintenance_string = '<?php $upgrading = ' . time () . '; ?>' ;
$wp_filesystem -> delete ( $file );
$wp_filesystem -> put_contents ( $file , $maintenance_string , FS_CHMOD_FILE );
} else if ( ! $enable && $wp_filesystem -> exists ( $file ) ) {
$this -> skin -> feedback ( 'maintenance_end' );
$wp_filesystem -> delete ( $file );
}
}
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
}
2009-04-26 20:17:19 +02:00
/**
* Plugin Upgrader class for WordPress Plugins , It is designed to upgrade / install plugins from a local zip , remote zip URL , or uploaded zip file .
*
* @ package WordPress
* @ subpackage Upgrader
* @ since 2.8 . 0
*/
2009-04-19 21:36:28 +02:00
class Plugin_Upgrader extends WP_Upgrader {
var $result ;
2009-11-13 22:28:40 +01:00
var $bulk = false ;
2009-10-26 08:04:30 +01:00
var $show_before = '' ;
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
function upgrade_strings () {
$this -> strings [ 'up_to_date' ] = __ ( 'The plugin is at the latest version.' );
2010-12-02 22:45:47 +01:00
$this -> strings [ 'no_package' ] = __ ( 'Update package not available.' );
2009-12-24 10:46:57 +01:00
$this -> strings [ 'downloading_package' ] = __ ( 'Downloading update from <span class="code">%s</span>…' );
$this -> strings [ 'unpack_package' ] = __ ( 'Unpacking the update…' );
$this -> strings [ 'remove_old' ] = __ ( 'Removing the old version of the plugin…' );
2009-04-19 21:36:28 +02:00
$this -> strings [ 'remove_old_failed' ] = __ ( 'Could not remove the old plugin.' );
2010-12-02 22:45:47 +01:00
$this -> strings [ 'process_failed' ] = __ ( 'Plugin update failed.' );
$this -> strings [ 'process_success' ] = __ ( 'Plugin updated successfully.' );
2009-04-19 21:36:28 +02:00
}
function install_strings () {
$this -> strings [ 'no_package' ] = __ ( 'Install package not available.' );
2009-12-24 10:46:57 +01:00
$this -> strings [ 'downloading_package' ] = __ ( 'Downloading install package from <span class="code">%s</span>…' );
$this -> strings [ 'unpack_package' ] = __ ( 'Unpacking the package…' );
$this -> strings [ 'installing_package' ] = __ ( 'Installing the plugin…' );
2013-05-22 18:57:20 +02:00
$this -> strings [ 'no_files' ] = __ ( 'The plugin contains no files.' );
2009-12-24 10:46:57 +01:00
$this -> strings [ 'process_failed' ] = __ ( 'Plugin install failed.' );
$this -> strings [ 'process_success' ] = __ ( 'Plugin installed successfully.' );
2009-04-19 21:36:28 +02:00
}
2013-09-06 03:32:09 +02:00
function install ( $package , $args = array () ) {
$defaults = array (
'clear_update_cache' => true ,
);
2013-09-19 10:39:51 +02:00
$parsed_args = wp_parse_args ( $args , $defaults );
2009-04-20 20:18:39 +02:00
2009-04-26 20:17:19 +02:00
$this -> init ();
2009-04-19 21:36:28 +02:00
$this -> install_strings ();
2013-09-05 18:33:50 +02:00
add_filter ( 'upgrader_source_selection' , array ( $this , 'check_package' ) );
2011-08-28 17:02:48 +02:00
2013-09-06 03:32:09 +02:00
$this -> run ( array (
'package' => $package ,
'destination' => WP_PLUGIN_DIR ,
'clear_destination' => false , // Do not overwrite files.
'clear_working' => true ,
'hook_extra' => array ()
) );
2009-04-20 20:18:39 +02:00
2013-09-05 18:33:50 +02:00
remove_filter ( 'upgrader_source_selection' , array ( $this , 'check_package' ) );
2011-08-28 17:02:48 +02:00
2011-08-28 07:51:38 +02:00
if ( ! $this -> result || is_wp_error ( $this -> result ) )
return $this -> result ;
2009-04-19 21:36:28 +02:00
// Force refresh of plugin update information
2013-09-06 03:32:09 +02:00
wp_clean_plugins_cache ( $parsed_args [ 'clear_update_cache' ] );
2013-04-04 23:10:54 +02:00
do_action ( 'upgrader_process_complete' , $this , array ( 'action' => 'install' , 'type' => 'plugin' ), $package );
2009-04-19 21:36:28 +02:00
2011-08-28 07:51:38 +02:00
return true ;
2009-04-19 21:36:28 +02:00
}
2013-09-06 03:32:09 +02:00
function upgrade ( $plugin , $args = array () ) {
$defaults = array (
'clear_update_cache' => true ,
);
2013-09-19 10:39:51 +02:00
$parsed_args = wp_parse_args ( $args , $defaults );
2009-04-20 20:18:39 +02:00
2009-04-26 20:17:19 +02:00
$this -> init ();
2009-04-19 21:36:28 +02:00
$this -> upgrade_strings ();
2009-04-20 20:18:39 +02:00
2010-01-08 21:49:55 +01:00
$current = get_site_transient ( 'update_plugins' );
2009-04-19 21:36:28 +02:00
if ( ! isset ( $current -> response [ $plugin ] ) ) {
2010-03-06 09:39:50 +01:00
$this -> skin -> before ();
2009-06-06 10:14:41 +02:00
$this -> skin -> set_result ( false );
2009-04-19 21:36:28 +02:00
$this -> skin -> error ( 'up_to_date' );
2009-06-06 10:14:41 +02:00
$this -> skin -> after ();
2009-04-19 21:36:28 +02:00
return false ;
}
// Get the URL to the zip file
$r = $current -> response [ $plugin ];
2009-04-20 20:18:39 +02:00
2013-09-05 18:33:50 +02:00
add_filter ( 'upgrader_pre_install' , array ( $this , 'deactivate_plugin_before_upgrade' ), 10 , 2 );
add_filter ( 'upgrader_clear_destination' , array ( $this , 'delete_old_plugin' ), 10 , 4 );
//'source_selection' => array($this, 'source_selection'), //there's a trac ticket to move up the directory for zip's which are made a bit differently, useful for non-.org plugins.
2009-04-19 21:36:28 +02:00
2013-09-06 03:32:09 +02:00
$this -> run ( array (
'package' => $r -> package ,
'destination' => WP_PLUGIN_DIR ,
'clear_destination' => true ,
'clear_working' => true ,
'hook_extra' => array (
'plugin' => $plugin
),
) );
2009-04-19 21:36:28 +02:00
2011-09-03 16:18:10 +02:00
// Cleanup our hooks, in case something else does a upgrade on this connection.
2013-09-05 18:33:50 +02:00
remove_filter ( 'upgrader_pre_install' , array ( $this , 'deactivate_plugin_before_upgrade' ));
remove_filter ( 'upgrader_clear_destination' , array ( $this , 'delete_old_plugin' ));
2009-04-19 21:36:28 +02:00
if ( ! $this -> result || is_wp_error ( $this -> result ) )
return $this -> result ;
// Force refresh of plugin update information
2013-09-06 03:32:09 +02:00
wp_clean_plugins_cache ( $parsed_args [ 'clear_update_cache' ] );
2013-04-04 23:10:54 +02:00
do_action ( 'upgrader_process_complete' , $this , array ( 'action' => 'update' , 'type' => 'plugin' ), $plugin );
2013-08-30 09:37:32 +02:00
return true ;
2009-04-19 21:36:28 +02:00
}
2009-04-20 20:18:39 +02:00
2013-09-06 03:32:09 +02:00
function bulk_upgrade ( $plugins , $args = array () ) {
$defaults = array (
'clear_update_cache' => true ,
);
2013-09-19 10:39:51 +02:00
$parsed_args = wp_parse_args ( $args , $defaults );
2009-10-23 23:49:12 +02:00
$this -> init ();
$this -> bulk = true ;
$this -> upgrade_strings ();
2010-01-08 21:49:55 +01:00
$current = get_site_transient ( 'update_plugins' );
2009-10-23 23:49:12 +02:00
2013-09-05 18:33:50 +02:00
add_filter ( 'upgrader_clear_destination' , array ( $this , 'delete_old_plugin' ), 10 , 4 );
2009-10-23 23:49:12 +02:00
2009-10-26 08:04:30 +01:00
$this -> skin -> header ();
2009-11-09 19:53:21 +01:00
// Connect to the Filesystem first.
$res = $this -> fs_connect ( array ( WP_CONTENT_DIR , WP_PLUGIN_DIR ) );
if ( ! $res ) {
$this -> skin -> footer ();
return false ;
}
2010-05-25 15:57:06 +02:00
$this -> skin -> bulk_header ();
2013-08-17 07:35:17 +02:00
// Only start maintenance mode if:
// - running Multisite and there are one or more plugins specified, OR
// - a plugin with an update available is currently active.
// @TODO: For multisite, maintenance mode should only kick in for individual sites if at all possible.
$maintenance = ( is_multisite () && ! empty ( $plugins ) );
2010-07-24 05:59:53 +02:00
foreach ( $plugins as $plugin )
2013-08-17 07:35:17 +02:00
$maintenance = $maintenance || ( is_plugin_active ( $plugin ) && isset ( $current -> response [ $plugin ] ) );
2010-07-24 05:59:53 +02:00
if ( $maintenance )
$this -> maintenance_mode ( true );
2009-11-09 19:53:21 +01:00
2010-02-13 07:08:15 +01:00
$results = array ();
2010-03-13 04:59:40 +01:00
$this -> update_count = count ( $plugins );
$this -> update_current = 0 ;
2009-10-23 23:49:12 +02:00
foreach ( $plugins as $plugin ) {
2010-03-13 04:59:40 +01:00
$this -> update_current ++ ;
2010-03-06 09:39:50 +01:00
$this -> skin -> plugin_info = get_plugin_data ( WP_PLUGIN_DIR . '/' . $plugin , false , true );
2009-10-26 08:04:30 +01:00
2009-10-23 23:49:12 +02:00
if ( ! isset ( $current -> response [ $plugin ] ) ) {
2012-07-09 05:40:32 +02:00
$this -> skin -> set_result ( true );
2010-03-06 09:39:50 +01:00
$this -> skin -> before ();
2012-07-09 05:40:32 +02:00
$this -> skin -> feedback ( 'up_to_date' );
2009-10-23 23:49:12 +02:00
$this -> skin -> after ();
2012-07-09 05:40:32 +02:00
$results [ $plugin ] = true ;
2009-10-23 23:49:12 +02:00
continue ;
}
// Get the URL to the zip file
$r = $current -> response [ $plugin ];
$this -> skin -> plugin_active = is_plugin_active ( $plugin );
2013-09-06 03:32:09 +02:00
$result = $this -> run ( array (
'package' => $r -> package ,
'destination' => WP_PLUGIN_DIR ,
'clear_destination' => true ,
'clear_working' => true ,
'is_multi' => true ,
'hook_extra' => array (
'plugin' => $plugin
)
) );
2009-10-26 08:04:30 +01:00
2009-10-23 23:49:12 +02:00
$results [ $plugin ] = $this -> result ;
// Prevent credentials auth screen from displaying multiple times
if ( false === $result )
break ;
2010-03-06 09:39:50 +01:00
} //end foreach $plugins
2010-05-23 14:29:00 +02:00
2009-11-09 19:53:21 +01:00
$this -> maintenance_mode ( false );
2010-05-25 15:57:06 +02:00
$this -> skin -> bulk_footer ();
2010-05-23 14:29:00 +02:00
2009-10-26 08:04:30 +01:00
$this -> skin -> footer ();
2009-10-23 23:49:12 +02:00
2011-09-03 16:18:10 +02:00
// Cleanup our hooks, in case something else does a upgrade on this connection.
2013-09-05 18:33:50 +02:00
remove_filter ( 'upgrader_clear_destination' , array ( $this , 'delete_old_plugin' ));
2009-10-23 23:49:12 +02:00
// Force refresh of plugin update information
2013-09-06 03:32:09 +02:00
wp_clean_plugins_cache ( $parsed_args [ 'clear_update_cache' ] );
2013-04-04 23:10:54 +02:00
do_action ( 'upgrader_process_complete' , $this , array ( 'action' => 'update' , 'type' => 'plugin' , 'bulk' => true ), $plugins );
2009-10-23 23:49:12 +02:00
return $results ;
}
2011-08-28 17:02:48 +02:00
function check_package ( $source ) {
global $wp_filesystem ;
if ( is_wp_error ( $source ) )
return $source ;
$working_directory = str_replace ( $wp_filesystem -> wp_content_dir (), trailingslashit ( WP_CONTENT_DIR ), $source );
if ( ! is_dir ( $working_directory ) ) // Sanity check, if the above fails, lets not prevent installation.
return $source ;
// Check the folder contains at least 1 valid plugin.
$plugins_found = false ;
foreach ( glob ( $working_directory . '*.php' ) as $file ) {
$info = get_plugin_data ( $file , false , false );
if ( ! empty ( $info [ 'Name' ] ) ) {
$plugins_found = true ;
break ;
}
}
if ( ! $plugins_found )
2013-10-11 18:05:13 +02:00
return new WP_Error ( 'incompatible_archive_no_plugins' , $this -> strings [ 'incompatible_archive' ], __ ( 'No valid plugins were found.' ) );
2011-08-28 17:02:48 +02:00
return $source ;
}
2009-04-19 21:36:28 +02:00
//return plugin info.
function plugin_info () {
if ( ! is_array ( $this -> result ) )
return false ;
if ( empty ( $this -> result [ 'destination_name' ]) )
2009-04-20 20:18:39 +02:00
return false ;
2009-04-19 21:36:28 +02:00
$plugin = get_plugins ( '/' . $this -> result [ 'destination_name' ]); //Ensure to pass with leading slash
if ( empty ( $plugin ) )
return false ;
$pluginfiles = array_keys ( $plugin ); //Assume the requested plugin is the first in the list
2009-04-20 20:18:39 +02:00
return $this -> result [ 'destination_name' ] . '/' . $pluginfiles [ 0 ];
2009-04-19 21:36:28 +02:00
}
//Hooked to pre_install
function deactivate_plugin_before_upgrade ( $return , $plugin ) {
if ( is_wp_error ( $return ) ) //Bypass.
return $return ;
$plugin = isset ( $plugin [ 'plugin' ]) ? $plugin [ 'plugin' ] : '' ;
if ( empty ( $plugin ) )
return new WP_Error ( 'bad_request' , $this -> strings [ 'bad_request' ]);
if ( is_plugin_active ( $plugin ) ) {
//Deactivate the plugin silently, Prevent deactivation hooks from running.
deactivate_plugins ( $plugin , true );
2009-04-20 20:18:39 +02:00
}
2009-04-19 21:36:28 +02:00
}
//Hooked to upgrade_clear_destination
function delete_old_plugin ( $removed , $local_destination , $remote_destination , $plugin ) {
global $wp_filesystem ;
2011-04-03 13:55:27 +02:00
2009-04-19 21:36:28 +02:00
if ( is_wp_error ( $removed ) )
return $removed ; //Pass errors through.
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
$plugin = isset ( $plugin [ 'plugin' ]) ? $plugin [ 'plugin' ] : '' ;
if ( empty ( $plugin ) )
return new WP_Error ( 'bad_request' , $this -> strings [ 'bad_request' ]);
$plugins_dir = $wp_filesystem -> wp_plugins_dir ();
$this_plugin_dir = trailingslashit ( dirname ( $plugins_dir . $plugin ) );
2011-04-03 13:55:27 +02:00
2012-12-20 16:55:32 +01:00
if ( ! $wp_filesystem -> exists ( $this_plugin_dir ) ) //If it's already vanished.
2009-04-19 21:36:28 +02:00
return $removed ;
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
// If plugin is in its own directory, recursively delete the directory.
2012-12-20 16:55:32 +01:00
if ( strpos ( $plugin , '/' ) && $this_plugin_dir != $plugins_dir ) //base check on if plugin includes directory separator AND that it's not the root plugin folder
2009-04-19 21:36:28 +02:00
$deleted = $wp_filesystem -> delete ( $this_plugin_dir , true );
else
$deleted = $wp_filesystem -> delete ( $plugins_dir . $plugin );
2011-04-03 13:55:27 +02:00
2009-04-19 21:36:28 +02:00
if ( ! $deleted )
return new WP_Error ( 'remove_old_failed' , $this -> strings [ 'remove_old_failed' ]);
2009-04-20 20:18:39 +02:00
2011-04-03 13:54:07 +02:00
return true ;
2009-04-19 21:36:28 +02:00
}
}
2009-04-26 20:17:19 +02:00
/**
* Theme Upgrader class for WordPress Themes , It is designed to upgrade / install themes from a local zip , remote zip URL , or uploaded zip file .
*
* @ package WordPress
* @ subpackage Upgrader
* @ since 2.8 . 0
*/
2009-04-19 21:36:28 +02:00
class Theme_Upgrader extends WP_Upgrader {
var $result ;
2011-08-10 21:26:36 +02:00
var $bulk = false ;
2009-04-19 21:36:28 +02:00
function upgrade_strings () {
$this -> strings [ 'up_to_date' ] = __ ( 'The theme is at the latest version.' );
2010-12-02 22:45:47 +01:00
$this -> strings [ 'no_package' ] = __ ( 'Update package not available.' );
2009-12-24 10:46:57 +01:00
$this -> strings [ 'downloading_package' ] = __ ( 'Downloading update from <span class="code">%s</span>…' );
$this -> strings [ 'unpack_package' ] = __ ( 'Unpacking the update…' );
$this -> strings [ 'remove_old' ] = __ ( 'Removing the old version of the theme…' );
2009-04-19 21:36:28 +02:00
$this -> strings [ 'remove_old_failed' ] = __ ( 'Could not remove the old theme.' );
2010-12-02 22:45:47 +01:00
$this -> strings [ 'process_failed' ] = __ ( 'Theme update failed.' );
$this -> strings [ 'process_success' ] = __ ( 'Theme updated successfully.' );
2009-04-19 21:36:28 +02:00
}
function install_strings () {
$this -> strings [ 'no_package' ] = __ ( 'Install package not available.' );
2009-12-24 10:46:57 +01:00
$this -> strings [ 'downloading_package' ] = __ ( 'Downloading install package from <span class="code">%s</span>…' );
$this -> strings [ 'unpack_package' ] = __ ( 'Unpacking the package…' );
$this -> strings [ 'installing_package' ] = __ ( 'Installing the theme…' );
2013-05-22 18:57:20 +02:00
$this -> strings [ 'no_files' ] = __ ( 'The theme contains no files.' );
2009-12-24 10:46:57 +01:00
$this -> strings [ 'process_failed' ] = __ ( 'Theme install failed.' );
$this -> strings [ 'process_success' ] = __ ( 'Theme installed successfully.' );
2012-03-23 11:10:06 +01:00
/* translators: 1: theme name, 2: version */
$this -> strings [ 'process_success_specific' ] = __ ( 'Successfully installed the theme <strong>%1$s %2$s</strong>.' );
$this -> strings [ 'parent_theme_search' ] = __ ( 'This theme requires a parent theme. Checking if it is installed…' );
/* translators: 1: theme name, 2: version */
$this -> strings [ 'parent_theme_prepare_install' ] = __ ( 'Preparing to install <strong>%1$s %2$s</strong>…' );
/* translators: 1: theme name, 2: version */
$this -> strings [ 'parent_theme_currently_installed' ] = __ ( 'The parent theme, <strong>%1$s %2$s</strong>, is currently installed.' );
/* translators: 1: theme name, 2: version */
$this -> strings [ 'parent_theme_install_success' ] = __ ( 'Successfully installed the parent theme, <strong>%1$s %2$s</strong>.' );
$this -> strings [ 'parent_theme_not_found' ] = __ ( '<strong>The parent theme could not be found.</strong> You will need to install the parent theme, <strong>%s</strong>, before you can use this child theme.' );
}
function check_parent_theme_filter ( $install_result , $hook_extra , $child_result ) {
// Check to see if we need to install a parent theme
$theme_info = $this -> theme_info ();
2012-03-23 11:20:26 +01:00
if ( ! $theme_info -> parent () )
2012-03-23 11:10:06 +01:00
return $install_result ;
2012-03-23 11:20:26 +01:00
$this -> skin -> feedback ( 'parent_theme_search' );
2012-03-23 11:10:06 +01:00
2012-03-23 11:20:26 +01:00
if ( ! $theme_info -> parent () -> errors () ) {
$this -> skin -> feedback ( 'parent_theme_currently_installed' , $theme_info -> parent () -> display ( 'Name' ), $theme_info -> parent () -> display ( 'Version' ) );
2012-03-23 11:10:06 +01:00
// We already have the theme, fall through.
return $install_result ;
}
// We don't have the parent theme, lets install it
2012-03-23 11:20:26 +01:00
$api = themes_api ( 'theme_information' , array ( 'slug' => $theme_info -> get ( 'Template' ), 'fields' => array ( 'sections' => false , 'tags' => false ) ) ); //Save on a bit of bandwidth.
2012-03-23 11:10:06 +01:00
if ( ! $api || is_wp_error ( $api ) ) {
2012-03-23 11:20:26 +01:00
$this -> skin -> feedback ( 'parent_theme_not_found' , $theme_info -> get ( 'Template' ) );
2012-03-23 11:10:06 +01:00
// Don't show activate or preview actions after install
2013-09-05 18:33:50 +02:00
add_filter ( 'install_theme_complete_actions' , array ( $this , 'hide_activate_preview_actions' ) );
2012-03-23 11:10:06 +01:00
return $install_result ;
}
// Backup required data we're going to override:
$child_api = $this -> skin -> api ;
$child_success_message = $this -> strings [ 'process_success' ];
// Override them
$this -> skin -> api = $api ;
$this -> strings [ 'process_success_specific' ] = $this -> strings [ 'parent_theme_install_success' ]; //, $api->name, $api->version);
$this -> skin -> feedback ( 'parent_theme_prepare_install' , $api -> name , $api -> version );
2012-04-10 04:25:03 +02:00
2012-03-23 11:10:06 +01:00
add_filter ( 'install_theme_complete_actions' , '__return_false' , 999 ); // Don't show any actions after installing the theme.
// Install the parent theme
$parent_result = $this -> run ( array (
'package' => $api -> download_link ,
2013-08-22 06:49:06 +02:00
'destination' => get_theme_root (),
2012-03-23 11:10:06 +01:00
'clear_destination' => false , //Do not overwrite files.
'clear_working' => true
) );
if ( is_wp_error ( $parent_result ) )
2013-09-05 18:33:50 +02:00
add_filter ( 'install_theme_complete_actions' , array ( $this , 'hide_activate_preview_actions' ) );
2012-03-23 11:10:06 +01:00
// Start cleaning up after the parents installation
remove_filter ( 'install_theme_complete_actions' , '__return_false' , 999 );
// Reset child's result and data
$this -> result = $child_result ;
$this -> skin -> api = $child_api ;
$this -> strings [ 'process_success' ] = $child_success_message ;
return $install_result ;
}
2012-04-10 04:25:03 +02:00
2012-03-23 11:10:06 +01:00
function hide_activate_preview_actions ( $actions ) {
unset ( $actions [ 'activate' ], $actions [ 'preview' ]);
return $actions ;
2009-04-19 21:36:28 +02:00
}
2013-09-06 03:32:09 +02:00
function install ( $package , $args = array () ) {
$defaults = array (
'clear_update_cache' => true ,
);
2013-09-19 10:39:51 +02:00
$parsed_args = wp_parse_args ( $args , $defaults );
2009-05-25 01:47:49 +02:00
2009-04-26 20:17:19 +02:00
$this -> init ();
2009-04-19 21:36:28 +02:00
$this -> install_strings ();
2013-09-05 18:33:50 +02:00
add_filter ( 'upgrader_source_selection' , array ( $this , 'check_package' ) );
add_filter ( 'upgrader_post_install' , array ( $this , 'check_parent_theme_filter' ), 10 , 3 );
2011-08-28 17:02:48 +02:00
2013-09-06 03:32:09 +02:00
$this -> run ( array (
2013-08-22 06:49:06 +02:00
'package' => $package ,
'destination' => get_theme_root (),
'clear_destination' => false , //Do not overwrite files.
'clear_working' => true
2013-09-06 03:32:09 +02:00
) );
2009-04-20 20:18:39 +02:00
2013-09-05 18:33:50 +02:00
remove_filter ( 'upgrader_source_selection' , array ( $this , 'check_package' ) );
remove_filter ( 'upgrader_post_install' , array ( $this , 'check_parent_theme_filter' ));
2011-08-28 17:02:48 +02:00
2009-04-19 21:36:28 +02:00
if ( ! $this -> result || is_wp_error ( $this -> result ) )
return $this -> result ;
2009-04-20 20:18:39 +02:00
2013-09-06 03:32:09 +02:00
// Refresh the Theme Update information
wp_clean_themes_cache ( $parsed_args [ 'clear_update_cache' ] );
2013-04-04 23:10:54 +02:00
do_action ( 'upgrader_process_complete' , $this , array ( 'action' => 'install' , 'type' => 'theme' ), $package );
2009-04-20 20:18:39 +02:00
2011-08-28 07:51:38 +02:00
return true ;
2009-04-19 21:36:28 +02:00
}
2013-09-06 03:32:09 +02:00
function upgrade ( $theme , $args = array () ) {
$defaults = array (
'clear_update_cache' => true ,
);
2013-09-19 10:39:51 +02:00
$parsed_args = wp_parse_args ( $args , $defaults );
2009-04-20 20:18:39 +02:00
2009-04-26 20:17:19 +02:00
$this -> init ();
2009-04-19 21:36:28 +02:00
$this -> upgrade_strings ();
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
// Is an update available?
2010-01-08 21:49:55 +01:00
$current = get_site_transient ( 'update_themes' );
2009-06-06 10:14:41 +02:00
if ( ! isset ( $current -> response [ $theme ] ) ) {
2010-03-06 09:39:50 +01:00
$this -> skin -> before ();
2009-06-06 10:14:41 +02:00
$this -> skin -> set_result ( false );
$this -> skin -> error ( 'up_to_date' );
$this -> skin -> after ();
return false ;
}
2009-09-14 16:03:32 +02:00
2009-04-19 21:36:28 +02:00
$r = $current -> response [ $theme ];
2013-09-05 18:33:50 +02:00
add_filter ( 'upgrader_pre_install' , array ( $this , 'current_before' ), 10 , 2 );
add_filter ( 'upgrader_post_install' , array ( $this , 'current_after' ), 10 , 2 );
add_filter ( 'upgrader_clear_destination' , array ( $this , 'delete_old_theme' ), 10 , 4 );
2009-04-19 21:36:28 +02:00
2013-09-06 03:32:09 +02:00
$this -> run ( array (
2013-08-22 06:49:06 +02:00
'package' => $r [ 'package' ],
'destination' => get_theme_root ( $theme ),
'clear_destination' => true ,
'clear_working' => true ,
'hook_extra' => array (
'theme' => $theme
),
2013-09-06 03:32:09 +02:00
) );
2009-04-19 21:36:28 +02:00
2013-09-05 18:33:50 +02:00
remove_filter ( 'upgrader_pre_install' , array ( $this , 'current_before' ));
remove_filter ( 'upgrader_post_install' , array ( $this , 'current_after' ));
remove_filter ( 'upgrader_clear_destination' , array ( $this , 'delete_old_theme' ));
2011-08-28 17:02:48 +02:00
2009-04-19 21:36:28 +02:00
if ( ! $this -> result || is_wp_error ( $this -> result ) )
return $this -> result ;
2013-09-06 03:32:09 +02:00
wp_clean_themes_cache ( $parsed_args [ 'clear_update_cache' ] );
2013-04-04 23:10:54 +02:00
do_action ( 'upgrader_process_complete' , $this , array ( 'action' => 'update' , 'type' => 'theme' ), $theme );
2009-04-19 21:36:28 +02:00
return true ;
}
2009-04-20 20:18:39 +02:00
2013-09-06 03:32:09 +02:00
function bulk_upgrade ( $themes , $args = array () ) {
$defaults = array (
'clear_update_cache' => true ,
);
2013-09-19 10:39:51 +02:00
$parsed_args = wp_parse_args ( $args , $defaults );
2010-03-13 04:59:40 +01:00
$this -> init ();
$this -> bulk = true ;
$this -> upgrade_strings ();
$current = get_site_transient ( 'update_themes' );
2013-09-05 18:33:50 +02:00
add_filter ( 'upgrader_pre_install' , array ( $this , 'current_before' ), 10 , 2 );
add_filter ( 'upgrader_post_install' , array ( $this , 'current_after' ), 10 , 2 );
add_filter ( 'upgrader_clear_destination' , array ( $this , 'delete_old_theme' ), 10 , 4 );
2010-03-13 04:59:40 +01:00
$this -> skin -> header ();
// Connect to the Filesystem first.
$res = $this -> fs_connect ( array ( WP_CONTENT_DIR ) );
if ( ! $res ) {
$this -> skin -> footer ();
return false ;
}
2010-05-25 15:57:06 +02:00
$this -> skin -> bulk_header ();
2013-08-17 07:35:17 +02:00
// Only start maintenance mode if:
// - running Multisite and there are one or more themes specified, OR
// - a theme with an update available is currently in use.
// @TODO: For multisite, maintenance mode should only kick in for individual sites if at all possible.
$maintenance = ( is_multisite () && ! empty ( $themes ) );
2010-07-24 05:59:53 +02:00
foreach ( $themes as $theme )
$maintenance = $maintenance || $theme == get_stylesheet () || $theme == get_template ();
if ( $maintenance )
$this -> maintenance_mode ( true );
2010-03-13 04:59:40 +01:00
$results = array ();
$this -> update_count = count ( $themes );
$this -> update_current = 0 ;
foreach ( $themes as $theme ) {
$this -> update_current ++ ;
2012-06-30 14:08:18 +02:00
$this -> skin -> theme_info = $this -> theme_info ( $theme );
2010-03-13 04:59:40 +01:00
if ( ! isset ( $current -> response [ $theme ] ) ) {
2012-07-09 05:40:32 +02:00
$this -> skin -> set_result ( true );
2010-03-13 04:59:40 +01:00
$this -> skin -> before ();
2012-07-09 05:40:32 +02:00
$this -> skin -> feedback ( 'up_to_date' );
2010-03-13 04:59:40 +01:00
$this -> skin -> after ();
2012-07-09 05:40:32 +02:00
$results [ $theme ] = true ;
2010-03-13 04:59:40 +01:00
continue ;
}
// Get the URL to the zip file
$r = $current -> response [ $theme ];
2013-09-06 03:32:09 +02:00
$result = $this -> run ( array (
2013-08-22 06:49:06 +02:00
'package' => $r [ 'package' ],
'destination' => get_theme_root ( $theme ),
'clear_destination' => true ,
'clear_working' => true ,
'hook_extra' => array (
'theme' => $theme
),
2013-09-06 03:32:09 +02:00
) );
2010-03-13 04:59:40 +01:00
$results [ $theme ] = $this -> result ;
// Prevent credentials auth screen from displaying multiple times
if ( false === $result )
break ;
} //end foreach $plugins
2010-05-23 14:29:00 +02:00
2010-03-13 04:59:40 +01:00
$this -> maintenance_mode ( false );
2010-05-25 15:57:06 +02:00
$this -> skin -> bulk_footer ();
2010-03-13 04:59:40 +01:00
$this -> skin -> footer ();
2011-09-03 16:18:10 +02:00
// Cleanup our hooks, in case something else does a upgrade on this connection.
2013-09-05 18:33:50 +02:00
remove_filter ( 'upgrader_pre_install' , array ( $this , 'current_before' ));
remove_filter ( 'upgrader_post_install' , array ( $this , 'current_after' ));
remove_filter ( 'upgrader_clear_destination' , array ( $this , 'delete_old_theme' ));
2010-03-13 04:59:40 +01:00
2013-09-06 03:32:09 +02:00
// Refresh the Theme Update information
wp_clean_themes_cache ( $parsed_args [ 'clear_update_cache' ] );
2013-04-04 23:10:54 +02:00
do_action ( 'upgrader_process_complete' , $this , array ( 'action' => 'update' , 'type' => 'theme' , 'bulk' => true ), $themes );
2010-03-13 04:59:40 +01:00
return $results ;
}
2011-08-28 17:02:48 +02:00
function check_package ( $source ) {
global $wp_filesystem ;
if ( is_wp_error ( $source ) )
return $source ;
// Check the folder contains a valid theme
$working_directory = str_replace ( $wp_filesystem -> wp_content_dir (), trailingslashit ( WP_CONTENT_DIR ), $source );
if ( ! is_dir ( $working_directory ) ) // Sanity check, if the above fails, lets not prevent installation.
return $source ;
2012-03-23 11:20:26 +01:00
// A proper archive should have a style.css file in the single subdirectory
if ( ! file_exists ( $working_directory . 'style.css' ) )
2013-10-11 18:05:13 +02:00
return new WP_Error ( 'incompatible_archive_theme_no_style' , $this -> strings [ 'incompatible_archive' ], __ ( 'The theme is missing the <code>style.css</code> stylesheet.' ) );
2011-08-28 17:02:48 +02:00
2012-03-23 11:20:26 +01:00
$info = get_file_data ( $working_directory . 'style.css' , array ( 'Name' => 'Theme Name' , 'Template' => 'Template' ) );
if ( empty ( $info [ 'Name' ] ) )
2013-10-11 18:05:13 +02:00
return new WP_Error ( 'incompatible_archive_theme_no_name' , $this -> strings [ 'incompatible_archive' ], __ ( " The <code>style.css</code> stylesheet doesn't contain a valid theme header. " ) );
2011-08-28 17:02:48 +02:00
2012-03-23 11:20:26 +01:00
// If it's not a child theme, it must have at least an index.php to be legit.
if ( empty ( $info [ 'Template' ] ) && ! file_exists ( $working_directory . 'index.php' ) )
2013-10-11 18:05:13 +02:00
return new WP_Error ( 'incompatible_archive_theme_no_index' , $this -> strings [ 'incompatible_archive' ], __ ( 'The theme is missing the <code>index.php</code> file.' ) );
2011-08-28 17:02:48 +02:00
return $source ;
}
2009-04-19 21:36:28 +02:00
function current_before ( $return , $theme ) {
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
if ( is_wp_error ( $return ) )
return $return ;
$theme = isset ( $theme [ 'theme' ]) ? $theme [ 'theme' ] : '' ;
if ( $theme != get_stylesheet () ) //If not current
return $return ;
2010-07-24 05:59:53 +02:00
//Change to maintenance mode now.
2010-03-13 04:59:40 +01:00
if ( ! $this -> bulk )
$this -> maintenance_mode ( true );
2009-04-19 21:36:28 +02:00
return $return ;
}
2011-08-28 17:02:48 +02:00
2009-04-19 21:36:28 +02:00
function current_after ( $return , $theme ) {
if ( is_wp_error ( $return ) )
return $return ;
$theme = isset ( $theme [ 'theme' ]) ? $theme [ 'theme' ] : '' ;
2012-03-23 11:20:26 +01:00
if ( $theme != get_stylesheet () ) // If not current
2009-04-19 21:36:28 +02:00
return $return ;
2012-06-26 07:21:04 +02:00
// Ensure stylesheet name hasn't changed after the upgrade:
2009-04-19 21:36:28 +02:00
if ( $theme == get_stylesheet () && $theme != $this -> result [ 'destination_name' ] ) {
2012-06-26 07:21:04 +02:00
wp_clean_themes_cache ();
2009-04-19 21:36:28 +02:00
$stylesheet = $this -> result [ 'destination_name' ];
2012-06-26 07:21:04 +02:00
switch_theme ( $stylesheet );
2009-04-19 21:36:28 +02:00
}
2010-07-24 05:59:53 +02:00
//Time to remove maintenance mode
2010-03-13 04:59:40 +01:00
if ( ! $this -> bulk )
$this -> maintenance_mode ( false );
2009-04-19 21:36:28 +02:00
return $return ;
}
2009-04-20 20:18:39 +02:00
2013-08-22 06:49:06 +02:00
function delete_old_theme ( $removed , $local_destination , $remote_destination , $theme ) {
2009-04-19 21:36:28 +02:00
global $wp_filesystem ;
2009-04-20 20:18:39 +02:00
2013-08-22 06:49:06 +02:00
if ( is_wp_error ( $removed ) )
return $removed ; // Pass errors through.
2009-04-20 20:18:39 +02:00
2013-08-22 06:49:06 +02:00
if ( ! isset ( $theme [ 'theme' ] ) )
return $removed ;
2009-04-20 20:18:39 +02:00
2013-08-22 06:49:06 +02:00
$theme = $theme [ 'theme' ];
$themes_dir = trailingslashit ( $wp_filesystem -> wp_themes_dir ( $theme ) );
if ( $wp_filesystem -> exists ( $themes_dir . $theme ) ) {
if ( ! $wp_filesystem -> delete ( $themes_dir . $theme , true ) )
2009-04-19 21:36:28 +02:00
return false ;
2013-08-22 06:49:06 +02:00
}
2009-04-19 21:36:28 +02:00
return true ;
}
2009-04-20 20:18:39 +02:00
2010-03-13 04:59:40 +01:00
function theme_info ( $theme = null ) {
if ( empty ( $theme ) ) {
if ( ! empty ( $this -> result [ 'destination_name' ]) )
$theme = $this -> result [ 'destination_name' ];
else
return false ;
}
2013-08-22 06:49:06 +02:00
return wp_get_theme ( $theme );
2009-04-19 21:36:28 +02:00
}
}
2013-09-23 04:07:51 +02:00
add_action ( 'upgrader_process_complete' , array ( 'Language_Pack_Upgrader' , 'async_upgrade' ), 20 , 3 );
class Language_Pack_Upgrader extends WP_Upgrader {
var $result ;
var $bulk = true ;
static function async_upgrade ( $upgrader , $context , $package ) {
// Avoid recursion.
if ( $upgrader instanceof Language_Pack_Upgrader )
return ;
$lp_upgrader = new Language_Pack_Upgrader ( new Headerless_Upgrader_Skin () );
$lp_upgrader -> upgrade ();
}
function upgrade_strings () {
2013-10-02 20:50:45 +02:00
$this -> strings [ 'starting_upgrade' ] = __ ( 'Some of your translations need updating. Sit tight for a few more seconds while we update them as well.' );
$this -> strings [ 'up_to_date' ] = __ ( 'The translation is up to date.' ); // We need to silently skip this case
2013-09-23 04:07:51 +02:00
$this -> strings [ 'no_package' ] = __ ( 'Update package not available.' );
2013-10-02 20:50:45 +02:00
$this -> strings [ 'downloading_package' ] = __ ( 'Downloading translation from <span class="code">%s</span>…' );
2013-09-23 04:07:51 +02:00
$this -> strings [ 'unpack_package' ] = __ ( 'Unpacking the update…' );
2013-10-02 20:50:45 +02:00
$this -> strings [ 'process_failed' ] = __ ( 'Translation update failed.' );
$this -> strings [ 'process_success' ] = __ ( 'Translation updated successfully.' );
2013-09-23 04:07:51 +02:00
}
2013-09-23 11:37:25 +02:00
function upgrade ( $update = false ) {
if ( $update )
$update = array ( $update );
2013-09-28 01:20:08 +02:00
$results = $this -> bulk_upgrade ( $update );
return $results [ 0 ];
2013-09-23 04:07:51 +02:00
}
2013-09-23 11:37:25 +02:00
function bulk_upgrade ( $language_updates = array () ) {
2013-09-28 01:20:08 +02:00
global $wp_filesystem ;
2013-09-23 04:07:51 +02:00
$this -> init ();
$this -> upgrade_strings ();
2013-09-23 11:37:25 +02:00
if ( ! $language_updates )
$language_updates = wp_get_translation_updates ();
2013-09-23 04:07:51 +02:00
if ( empty ( $language_updates ) )
return true ;
2013-09-23 11:37:25 +02:00
if ( 'upgrader_process_complete' == current_filter () )
$this -> skin -> feedback ( 'starting_upgrade' );
2013-09-23 04:07:51 +02:00
add_filter ( 'upgrader_source_selection' , array ( & $this , 'check_package' ), 10 , 3 );
$this -> skin -> header ();
// Connect to the Filesystem first.
$res = $this -> fs_connect ( array ( WP_CONTENT_DIR , WP_LANG_DIR ) );
if ( ! $res ) {
$this -> skin -> footer ();
return false ;
}
$results = array ();
$this -> update_count = count ( $language_updates );
$this -> update_current = 0 ;
2013-09-28 01:20:08 +02:00
// The filesystem's mkdir() is not recursive. Make sure WP_LANG_DIR exists,
// as we then may need to create a /plugins or /themes directory inside of it.
$remote_destination = $wp_filesystem -> find_folder ( WP_LANG_DIR );
if ( ! $wp_filesystem -> exists ( $remote_destination ) )
if ( ! $wp_filesystem -> mkdir ( $remote_destination , FS_CHMOD_DIR ) )
2013-10-11 18:05:13 +02:00
return new WP_Error ( 'mkdir_failed_lang_dir' , $this -> strings [ 'mkdir_failed' ], $remote_destination );
2013-09-28 01:20:08 +02:00
2013-09-23 04:07:51 +02:00
foreach ( $language_updates as $language_update ) {
$destination = WP_LANG_DIR ;
if ( 'plugin' == $language_update -> type )
$destination .= '/plugins' ;
elseif ( 'theme' == $language_update -> type )
$destination .= '/themes' ;
$this -> update_current ++ ;
$options = array (
'package' => $language_update -> package ,
'destination' => $destination ,
'clear_destination' => false ,
'abort_if_destination_exists' => false , // We expect the destination to exist.
'clear_working' => true ,
'is_multi' => true ,
'hook_extra' => array (
'language_update_type' => $language_update -> type ,
'language_update' => $language_update ,
)
);
$result = $this -> run ( $options );
$results [] = $this -> result ;
// Prevent credentials auth screen from displaying multiple times.
if ( false === $result )
break ;
}
// Clean up our hooks, in case something else does an upgrade on this connection.
remove_filter ( 'upgrader_source_selection' , array ( & $this , 'check_package' ), 10 , 2 );
return $results ;
}
function check_package ( $source , $remote_source ) {
global $wp_filesystem ;
if ( is_wp_error ( $source ) )
return $source ;
// Check that the folder contains a valid language.
$files = $wp_filesystem -> dirlist ( $remote_source );
// Check to see if a .po and .mo exist in the folder.
$po = $mo = false ;
foreach ( ( array ) $files as $file => $filedata ) {
if ( '.po' == substr ( $file , - 3 ) )
$po = true ;
elseif ( '.mo' == substr ( $file , - 3 ) )
$mo = true ;
}
if ( ! $mo || ! $po )
2013-10-11 18:05:13 +02:00
return new WP_Error ( 'incompatible_archive_pomo' , $this -> strings [ 'incompatible_archive' ],
2013-09-23 04:07:51 +02:00
__ ( 'The language pack is missing either the <code>.po</code> or <code>.mo</code> files.' ) );
return $source ;
}
}
2009-04-26 20:17:19 +02:00
/**
2011-09-03 16:18:10 +02:00
* Core Upgrader class for WordPress . It allows for WordPress to upgrade itself in combination with the wp - admin / includes / update - core . php file
2009-04-26 20:17:19 +02:00
*
* @ package WordPress
* @ subpackage Upgrader
* @ since 2.8 . 0
*/
2009-04-19 21:36:28 +02:00
class Core_Upgrader extends WP_Upgrader {
function upgrade_strings () {
$this -> strings [ 'up_to_date' ] = __ ( 'WordPress is at the latest version.' );
2010-12-02 22:45:47 +01:00
$this -> strings [ 'no_package' ] = __ ( 'Update package not available.' );
2009-12-24 10:46:57 +01:00
$this -> strings [ 'downloading_package' ] = __ ( 'Downloading update from <span class="code">%s</span>…' );
$this -> strings [ 'unpack_package' ] = __ ( 'Unpacking the update…' );
2009-04-19 21:36:28 +02:00
$this -> strings [ 'copy_failed' ] = __ ( 'Could not copy files.' );
2012-04-27 20:57:59 +02:00
$this -> strings [ 'copy_failed_space' ] = __ ( 'Could not copy files. You may have run out of disk space.' );
2013-10-02 19:43:15 +02:00
$this -> strings [ 'start_rollback' ] = __ ( 'Attempting to roll back to previous version.' );
2013-10-02 19:44:09 +02:00
$this -> strings [ 'rollback_was_required' ] = __ ( 'Due to an error during updating, WordPress has rolled back to your previous version.' );
2009-04-19 21:36:28 +02:00
}
2013-09-06 03:32:09 +02:00
function upgrade ( $current , $args = array () ) {
2011-06-10 08:22:33 +02:00
global $wp_filesystem , $wp_version ;
2009-04-20 20:18:39 +02:00
2013-09-06 03:32:09 +02:00
$defaults = array (
2013-09-30 02:20:35 +02:00
'pre_check_md5' => true ,
'attempt_rollback' => false ,
'do_rollback' => false ,
2013-09-06 03:32:09 +02:00
);
2013-09-19 10:39:51 +02:00
$parsed_args = wp_parse_args ( $args , $defaults );
2013-09-06 03:32:09 +02:00
2009-04-26 20:17:19 +02:00
$this -> init ();
$this -> upgrade_strings ();
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
// Is an update available?
if ( ! isset ( $current -> response ) || $current -> response == 'latest' )
return new WP_Error ( 'up_to_date' , $this -> strings [ 'up_to_date' ]);
$res = $this -> fs_connect ( array ( ABSPATH , WP_CONTENT_DIR ) );
if ( is_wp_error ( $res ) )
return $res ;
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
$wp_dir = trailingslashit ( $wp_filesystem -> abspath ());
2009-04-20 20:18:39 +02:00
2013-09-21 08:48:20 +02:00
// Pre-cache the checksums for the versions we care about
get_core_checksums ( array ( $wp_version , $current -> version ) );
2013-10-02 23:49:34 +02:00
$partial = true ;
2013-09-30 02:20:35 +02:00
if ( $parsed_args [ 'do_rollback' ] )
2013-10-02 23:49:34 +02:00
$partial = false ;
2013-09-30 02:20:35 +02:00
elseif ( $parsed_args [ 'pre_check_md5' ] && ! $this -> check_files () )
2013-10-02 23:49:34 +02:00
$partial = false ;
2013-09-21 08:48:20 +02:00
2011-06-10 07:47:44 +02:00
// If partial update is returned from the API, use that, unless we're doing a reinstall.
// If we cross the new_bundled version number, then use the new_bundled zip.
// Don't though if the constant is set to skip bundled items.
// If the API returns a no_content zip, go with it. Finally, default to the full zip.
2013-09-30 02:20:35 +02:00
if ( $parsed_args [ 'do_rollback' ] && $current -> packages -> rollback )
$to_download = 'rollback' ;
2013-10-02 23:49:34 +02:00
elseif ( $current -> packages -> partial && 'reinstall' != $current -> response && $wp_version == $current -> partial_version && $partial )
2011-06-10 07:47:44 +02:00
$to_download = 'partial' ;
elseif ( $current -> packages -> new_bundled && version_compare ( $wp_version , $current -> new_bundled , '<' )
&& ( ! defined ( 'CORE_UPGRADE_SKIP_NEW_BUNDLED' ) || ! CORE_UPGRADE_SKIP_NEW_BUNDLED ) )
$to_download = 'new_bundled' ;
elseif ( $current -> packages -> no_content )
$to_download = 'no_content' ;
else
$to_download = 'full' ;
$download = $this -> download_package ( $current -> packages -> $to_download );
2009-04-19 21:36:28 +02:00
if ( is_wp_error ( $download ) )
return $download ;
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
$working_dir = $this -> unpack_package ( $download );
if ( is_wp_error ( $working_dir ) )
return $working_dir ;
// 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 ) ) {
$wp_filesystem -> delete ( $working_dir , true );
2013-10-11 18:05:13 +02:00
return new WP_Error ( 'copy_failed_for_update_core_file' , $this -> strings [ 'copy_failed' ] );
2009-04-19 21:36:28 +02:00
}
$wp_filesystem -> chmod ( $wp_dir . 'wp-admin/includes/update-core.php' , FS_CHMOD_FILE );
2009-04-20 20:18:39 +02:00
2013-09-30 02:20:35 +02:00
require_once ( ABSPATH . 'wp-admin/includes/update-core.php' );
2009-04-20 20:18:39 +02:00
2012-04-27 20:57:59 +02:00
if ( ! function_exists ( 'update_core' ) )
return new WP_Error ( 'copy_failed_space' , $this -> strings [ 'copy_failed_space' ] );
2013-09-30 02:29:19 +02:00
$result = update_core ( $working_dir , $wp_dir );
2013-09-30 02:20:35 +02:00
// In the event of an error, rollback to the previous version
2013-10-10 03:31:22 +02:00
if ( is_wp_error ( $result ) && $parsed_args [ 'attempt_rollback' ] && $current -> packages -> rollback && ! $parsed_args [ 'do_rollback' ] ) {
2013-09-30 02:20:35 +02:00
apply_filters ( 'update_feedback' , $result );
apply_filters ( 'update_feedback' , $this -> strings [ 'start_rollback' ] );
2013-10-10 03:31:22 +02:00
$rollback_result = $this -> upgrade ( $current , array_merge ( $parsed_args , array ( 'do_rollback' => true ) ) );
2013-09-30 02:20:35 +02:00
2013-10-10 03:31:22 +02:00
$result = new WP_Error ( 'rollback_was_required' , $this -> strings [ 'rollback_was_required' ], array ( 'rollback' => $rollback_result , 'update' => $result ) );
2013-09-30 02:20:35 +02:00
}
2013-04-04 23:10:54 +02:00
do_action ( 'upgrader_process_complete' , $this , array ( 'action' => 'update' , 'type' => 'core' ), $result );
return $result ;
2009-04-19 21:36:28 +02:00
}
2013-09-13 08:18:16 +02:00
// Determines if this WordPress Core version should update to $offered_ver or not
static function should_upgrade_to_version ( $offered_ver /* x.y.z */ ) {
include ABSPATH . WPINC . '/version.php' ; // $wp_version; // x.y.z
$current_branch = implode ( '.' , array_slice ( preg_split ( '/[.-]/' , $wp_version ), 0 , 2 ) ); // x.y
$new_branch = implode ( '.' , array_slice ( preg_split ( '/[.-]/' , $offered_ver ), 0 , 2 ) ); // x.y
$current_is_development_version = ( bool ) strpos ( $wp_version , '-' );
// Defaults:
2013-09-24 05:35:53 +02:00
$upgrade_dev = true ;
$upgrade_minor = true ;
2013-09-13 08:18:16 +02:00
$upgrade_major = false ;
// WP_AUTO_UPDATE_CORE = true (all), 'minor', false.
if ( defined ( 'WP_AUTO_UPDATE_CORE' ) ) {
if ( false === WP_AUTO_UPDATE_CORE ) {
// Defaults to turned off, unless a filter allows it
$upgrade_dev = $upgrade_minor = $upgrade_major = false ;
} elseif ( true === WP_AUTO_UPDATE_CORE ) {
// ALL updates for core
$upgrade_dev = $upgrade_minor = $upgrade_major = true ;
} elseif ( 'minor' === WP_AUTO_UPDATE_CORE ) {
// Only minor updates for core
$upgrade_dev = $upgrade_major = false ;
$upgrade_minor = true ;
}
}
// 1: If we're already on that version, not much point in updating?
if ( $offered_ver == $wp_version )
return false ;
// 2: If we're running a newer version, that's a nope
2013-09-19 10:45:06 +02:00
if ( version_compare ( $wp_version , $offered_ver , '>' ) )
2013-09-13 08:18:16 +02:00
return false ;
// 3: 3.7-alpha-25000 -> 3.7-alpha-25678 -> 3.7-beta1 -> 3.7-beta2
if ( $current_is_development_version ) {
if ( ! apply_filters ( 'allow_dev_auto_core_updates' , $upgrade_dev ) )
return false ;
// else fall through to minor + major branches below
}
// 4: Minor In-branch updates (3.7.0 -> 3.7.1 -> 3.7.2 -> 3.7.4)
if ( $current_branch == $new_branch )
return apply_filters ( 'allow_minor_auto_core_updates' , $upgrade_minor );
// 5: Major version updates (3.7.0 -> 3.8.0 -> 3.9.1)
if ( version_compare ( $new_branch , $current_branch , '>' ) )
return apply_filters ( 'allow_major_auto_core_updates' , $upgrade_major );
// If we're not sure, we don't want it
return false ;
}
2013-09-21 08:48:20 +02:00
function check_files () {
global $wp_version ;
$checksums = get_core_checksums ( $wp_version );
if ( empty ( $checksums [ $wp_version ] ) || ! is_array ( $checksums [ $wp_version ] ) )
return false ;
foreach ( $checksums [ $wp_version ] as $file => $checksum ) {
2013-09-30 01:07:59 +02:00
// Skip files which get updated
if ( 'wp-content' == substr ( $file , 0 , 10 ) )
continue ;
if ( ! file_exists ( ABSPATH . $file ) || md5_file ( ABSPATH . $file ) !== $checksum )
2013-09-21 08:48:20 +02:00
return false ;
}
return true ;
}
2009-04-19 21:36:28 +02:00
}
2009-04-26 20:17:19 +02:00
/**
2012-12-20 16:55:32 +01:00
* Upgrade Skin helper for File uploads . This class handles the upload process and passes it as if it ' s a local file to the Upgrade / Installer functions .
2009-04-26 20:17:19 +02:00
*
* @ package WordPress
* @ subpackage Upgrader
* @ since 2.8 . 0
*/
2009-04-19 21:36:28 +02:00
class File_Upload_Upgrader {
var $package ;
var $filename ;
2011-08-28 10:42:07 +02:00
var $id = 0 ;
2009-04-19 21:36:28 +02:00
function __construct ( $form , $urlholder ) {
2009-04-20 20:18:39 +02:00
2009-04-19 21:36:28 +02:00
if ( empty ( $_FILES [ $form ][ 'name' ]) && empty ( $_GET [ $urlholder ]) )
wp_die ( __ ( 'Please select a file' ));
2009-04-20 20:18:39 +02:00
2012-12-20 16:55:32 +01:00
//Handle a newly uploaded file, Else assume it's already been uploaded
2011-08-28 10:42:07 +02:00
if ( ! empty ( $_FILES ) ) {
$overrides = array ( 'test_form' => false , 'test_type' => false );
$file = wp_handle_upload ( $_FILES [ $form ], $overrides );
if ( isset ( $file [ 'error' ] ) )
wp_die ( $file [ 'error' ] );
2009-04-20 20:18:39 +02:00
2011-08-28 10:42:07 +02:00
$this -> filename = $_FILES [ $form ][ 'name' ];
$this -> package = $file [ 'file' ];
// Construct the object array
$object = array (
'post_title' => $this -> filename ,
'post_content' => $file [ 'url' ],
'post_mime_type' => $file [ 'type' ],
'guid' => $file [ 'url' ],
'context' => 'upgrader' ,
'post_status' => 'private'
);
// Save the data
$this -> id = wp_insert_attachment ( $object , $file [ 'file' ] );
// schedule a cleanup for 2 hours from now in case of failed install
wp_schedule_single_event ( time () + 7200 , 'upgrader_scheduled_cleanup' , array ( $this -> id ) );
} elseif ( is_numeric ( $_GET [ $urlholder ] ) ) {
// Numeric Package = previously uploaded file, see above.
$this -> id = ( int ) $_GET [ $urlholder ];
$attachment = get_post ( $this -> id );
if ( empty ( $attachment ) )
2011-10-24 21:13:23 +02:00
wp_die ( __ ( 'Please select a file' ));
2011-08-28 10:42:07 +02:00
$this -> filename = $attachment -> post_title ;
$this -> package = get_attached_file ( $attachment -> ID );
2009-04-19 21:36:28 +02:00
} else {
2011-08-28 10:42:07 +02:00
// Else, It's set to something, Back compat for plugins using the old (pre-3.3) File_Uploader handler.
if ( ! ( ( $uploads = wp_upload_dir () ) && false === $uploads [ 'error' ] ) )
wp_die ( $uploads [ 'error' ] );
$this -> filename = $_GET [ $urlholder ];
2009-04-19 21:36:28 +02:00
$this -> package = $uploads [ 'basedir' ] . '/' . $this -> filename ;
}
}
2011-08-28 05:51:35 +02:00
function cleanup () {
2011-08-28 10:42:07 +02:00
if ( $this -> id )
wp_delete_attachment ( $this -> id );
elseif ( file_exists ( $this -> package ) )
return @ unlink ( $this -> package );
2011-08-28 05:51:35 +02:00
return true ;
}
2013-06-21 07:54:40 +02:00
}
2013-09-13 08:18:16 +02:00
/**
* WordPress Automatic Upgrader helper class
*
* @ since 3.7 . 0
*/
class WP_Automatic_Upgrader {
2013-09-19 10:45:06 +02:00
static $upgrade_results = array ();
2013-09-13 08:18:16 +02:00
static function upgrader_disabled () {
// That's a no if you don't want files changes
2013-09-24 09:15:58 +02:00
if ( defined ( 'DISALLOW_FILE_MODS' ) && DISALLOW_FILE_MODS )
2013-09-13 08:18:16 +02:00
return true ;
// More fine grained control can be done through the WP_AUTO_UPDATE_CORE constant and filters
if ( defined ( 'AUTOMATIC_UPDATER_DISABLED' ) && AUTOMATIC_UPDATER_DISABLED )
return true ;
if ( defined ( 'WP_INSTALLING' ) )
return true ;
return apply_filters ( 'auto_upgrader_disabled' , false );
}
2013-09-28 01:20:08 +02:00
/**
* Check for GIT / SVN checkouts .
*/
2013-09-28 08:32:14 +02:00
static function is_vcs_checkout ( $context ) {
2013-10-06 17:45:09 +02:00
$context_dirs = array ( untrailingslashit ( $context ) );
if ( $context !== ABSPATH )
$context_dirs [] = untrailingslashit ( ABSPATH );
$vcs_dirs = array ( '.svn' , '.git' , '.hg' , '.bzr' );
$check_dirs = array ();
foreach ( $context_dirs as $context_dir ) {
// Walk up from $context_dir to the root.
do {
$check_dirs [] = $context_dir ;
} while ( $context_dir != dirname ( $context_dir ) && $context_dir = dirname ( $context_dir ) );
}
2013-09-28 01:20:08 +02:00
2013-10-06 17:45:09 +02:00
$check_dirs = array_unique ( $check_dirs );
// Search all directories we've found for evidence of version control.
foreach ( $vcs_dirs as $vcs_dir ) {
foreach ( $check_dirs as $check_dir ) {
2013-10-11 19:10:18 +02:00
if ( $checkout = @ is_dir ( rtrim ( $check_dir , '\\/' ) . " / $vcs_dir " ) )
2013-10-06 17:45:09 +02:00
break 2 ;
2013-09-28 01:20:08 +02:00
}
}
return apply_filters ( 'auto_upgrade_is_vcs_checkout' , $checkout , $context );
}
2013-09-13 08:18:16 +02:00
/**
* Tests to see if we should upgrade a specific item , does not test to see if we CAN update the item .
*/
static function should_auto_update ( $type , $item , $context ) {
if ( self :: upgrader_disabled () )
return false ;
2013-09-28 01:20:08 +02:00
if ( self :: is_vcs_checkout ( $context ) )
return false ;
2013-09-13 08:18:16 +02:00
// Next up, do we actually have it enabled for this type of update?
switch ( $type ) {
2013-09-23 11:37:25 +02:00
case 'language' :
2013-09-24 05:35:53 +02:00
$upgrade = true ;
2013-09-23 11:37:25 +02:00
break ;
2013-09-13 08:18:16 +02:00
case 'core' :
$upgrade = Core_Upgrader :: should_upgrade_to_version ( $item -> current );
break ;
default :
case 'plugin' :
case 'theme' :
$upgrade = false ;
break ;
}
// And does the user / plugins want it?
// Plugins may filter on 'auto_upgrade_plugin', and check the 2nd param, $item, to only enable it for certain Plugins/Themes
if ( ! apply_filters ( 'auto_upgrade_' . $type , $upgrade , $item ) )
return false ;
// If it's a core update, are we actually compatible with it's requirements?
if ( 'core' == $type ) {
global $wpdb ;
$php_compat = version_compare ( phpversion (), $item -> php_version , '>=' );
if ( file_exists ( WP_CONTENT_DIR . '/db.php' ) && empty ( $wpdb -> is_mysql ) )
$mysql_compat = true ;
else
$mysql_compat = version_compare ( $wpdb -> db_version (), $item -> mysql_version , '>=' );
if ( ! $php_compat || ! $mysql_compat )
return false ;
}
return true ;
}
// Checks to see if WP_Filesystem is setup to allow unattended upgrades
2013-09-19 10:45:06 +02:00
static function can_auto_update ( $context , $skin = false ) {
if ( ! $skin )
$skin = new Automatic_Upgrader_Skin ();
return ( bool ) $skin -> request_filesystem_credentials ( false , $context );
2013-09-13 08:18:16 +02:00
}
static function upgrade ( $type , $item ) {
2013-09-19 10:45:06 +02:00
$skin = new Automatic_Upgrader_Skin ();
2013-09-13 08:18:16 +02:00
switch ( $type ) {
case 'core' :
// The Core upgrader doesn't use the Upgrader's skin during the actual main part of the upgrade, instead, firing a filter
2013-09-19 10:45:06 +02:00
add_filter ( 'update_feedback' , array ( $skin , 'feedback' ) );
$upgrader = new Core_Upgrader ( $skin );
2013-09-13 08:18:16 +02:00
$context = ABSPATH ;
break ;
case 'plugin' :
2013-09-19 10:45:06 +02:00
$upgrader = new Plugin_Upgrader ( $skin );
2013-09-13 08:18:16 +02:00
$context = WP_PLUGIN_DIR ; // We don't support custom Plugin directories, or updates for WPMU_PLUGIN_DIR
break ;
case 'theme' :
2013-09-19 10:45:06 +02:00
$upgrader = new Theme_Upgrader ( $skin );
2013-09-13 08:18:16 +02:00
$context = get_theme_root ( $item );
break ;
2013-09-23 11:37:25 +02:00
case 'language' :
$upgrader = new Language_Pack_Upgrader ( $skin );
2013-09-28 01:20:08 +02:00
$context = WP_CONTENT_DIR ; // WP_LANG_DIR;
2013-09-23 11:37:25 +02:00
break ;
2013-09-13 08:18:16 +02:00
}
// Determine if we can perform this upgrade or not
2013-09-28 01:20:08 +02:00
if ( ! self :: should_auto_update ( $type , $item , $context ) || ! self :: can_auto_update ( $context , $skin ) )
2013-09-13 08:18:16 +02:00
return false ;
2013-09-19 10:45:06 +02:00
switch ( $type ) {
case 'core' :
$skin -> feedback ( __ ( 'Updating to WordPress %s' ), $item -> version );
$item_name = sprintf ( __ ( 'WordPress %s' ), $item -> version );
break ;
case 'theme' :
$theme = wp_get_theme ( $item );
$item_name = $theme -> Get ( 'Name' );
2013-10-02 20:50:45 +02:00
$skin -> feedback ( __ ( 'Updating theme: %s' ), $item_name );
2013-09-19 10:45:06 +02:00
break ;
case 'plugin' :
$plugin_data = get_plugin_data ( $context . '/' . $item );
$item_name = $plugin_data [ 'Name' ];
2013-10-02 20:50:45 +02:00
$skin -> feedback ( __ ( 'Updating plugin: %s' ), $item_name );
2013-09-23 11:37:25 +02:00
break ;
case 'language' :
if ( 'theme' == $item -> type ) {
$theme = wp_get_theme ( $item -> slug );
$skin -> feedback ( sprintf (
2013-10-02 20:50:45 +02:00
__ ( 'Updating the %1$s translation for the %2$s theme' ),
2013-09-23 11:37:25 +02:00
$item -> language ,
$theme -> Get ( 'Name' )
) );
$item_name = sprintf (
2013-10-02 20:50:45 +02:00
__ ( '%1$s translation for the %2$s theme' ),
2013-09-23 11:37:25 +02:00
$item -> language ,
$theme -> Get ( 'Name' )
);
} elseif ( 'plugin' == $item -> type ) {
$plugin_data = get_plugins ( '/' . $item -> slug );
$plugin_data = array_shift ( $plugin_data );
$skin -> feedback ( sprintf (
2013-10-02 20:50:45 +02:00
__ ( 'Updating the %1$s translation for the %2$s plugin' ),
2013-09-23 11:37:25 +02:00
$item -> language ,
$plugin_data [ 'Name' ]
) );
$item_name = sprintf (
2013-10-02 20:50:45 +02:00
__ ( '%1$s translation for the %2$s plugin' ),
2013-09-23 11:37:25 +02:00
$item -> language ,
$plugin_data [ 'Name' ]
);
} else {
$skin -> feedback ( sprintf (
__ ( 'Updating %s translation' ),
$item -> language
) );
$item_name = sprintf (
__ ( '%s translation' ),
$item -> language
);
}
2013-09-19 10:45:06 +02:00
break ;
}
2013-09-13 08:18:16 +02:00
// Boom, This sites about to get a whole new splash of paint!
$upgrade_result = $upgrader -> upgrade ( $item , array (
'clear_update_cache' => false ,
2013-09-30 00:59:46 +02:00
'pre_check_md5' => false , /* always use partial builds if possible for core updates */
2013-09-30 02:20:35 +02:00
'attempt_rollback' => true , /* only available for core updates */
2013-09-13 08:18:16 +02:00
) );
// Core doesn't output this, so lets append it so we don't get confused
if ( 'core' == $type ) {
if ( is_wp_error ( $upgrade_result ) ) {
2013-09-19 10:45:06 +02:00
$skin -> error ( __ ( 'Installation Failed' ), $upgrade_result );
2013-09-13 08:18:16 +02:00
} else {
2013-09-19 10:45:06 +02:00
$skin -> feedback ( __ ( 'WordPress updated successfully' ) );
2013-09-13 08:18:16 +02:00
}
}
2013-09-19 10:45:06 +02:00
self :: $upgrade_results [ $type ][] = ( object ) array (
'item' => $item ,
2013-09-28 01:20:08 +02:00
'result' => $upgrade_result ,
2013-09-19 10:45:06 +02:00
'name' => $item_name ,
'messages' => $skin -> get_upgrade_messages ()
2013-09-13 08:18:16 +02:00
);
return $upgrade_result ;
}
/**
* Kicks off a upgrade request for each item in the upgrade " queue "
*/
static function perform_auto_updates () {
$lock_name = 'auto_upgrader.lock' ;
2013-09-17 04:27:21 +02:00
if ( get_site_option ( $lock_name ) ) {
2013-09-13 08:18:16 +02:00
// Test to see if it was set more than an hour ago, if so, cleanup.
2013-09-17 04:27:21 +02:00
if ( get_site_option ( $lock_name ) < ( time () - HOUR_IN_SECONDS ) )
delete_site_option ( $lock_name );
2013-09-17 06:07:35 +02:00
else // The process is already locked
2013-09-13 08:18:16 +02:00
return ;
}
// Lock upgrades for us for half an hour
2013-09-17 04:27:21 +02:00
if ( ! add_site_option ( $lock_name , microtime ( true ), HOUR_IN_SECONDS / 2 ) )
2013-09-13 08:18:16 +02:00
return ;
2013-09-23 11:37:25 +02:00
// Don't automatically run these thins, as we'll handle it ourselves
remove_action ( 'upgrader_process_complete' , array ( 'Language_Pack_Upgrader' , 'async_upgrade' ), 20 , 3 );
remove_action ( 'upgrader_process_complete' , 'wp_version_check' );
remove_action ( 'upgrader_process_complete' , 'wp_update_plugins' );
remove_action ( 'upgrader_process_complete' , 'wp_update_themes' );
2013-09-13 08:18:16 +02:00
// Next, Plugins
wp_update_plugins (); // Check for Plugin updates
$plugin_updates = get_site_transient ( 'update_plugins' );
if ( $plugin_updates && ! empty ( $plugin_updates -> response ) ) {
foreach ( array_keys ( $plugin_updates -> response ) as $plugin ) {
self :: upgrade ( 'plugin' , $plugin );
}
// Force refresh of plugin update information
wp_clean_plugins_cache ();
}
// Next, those themes we all love
wp_update_themes (); // Check for Theme updates
$theme_updates = get_site_transient ( 'update_themes' );
if ( $theme_updates && ! empty ( $theme_updates -> response ) ) {
foreach ( array_keys ( $theme_updates -> response ) as $theme ) {
self :: upgrade ( 'theme' , $theme );
}
// Force refresh of theme update information
wp_clean_themes_cache ();
}
2013-09-23 11:37:25 +02:00
// Next, Process any core upgrade
2013-09-13 08:18:16 +02:00
wp_version_check (); // Check for Core updates
2013-10-10 03:31:22 +02:00
$extra_update_stats = array ();
2013-09-13 08:18:16 +02:00
$core_update = find_core_auto_update ();
2013-09-19 10:45:06 +02:00
if ( $core_update ) {
2013-10-10 03:31:22 +02:00
$start_time = time ();
$core_update_result = self :: upgrade ( 'core' , $core_update );
2013-09-19 10:45:06 +02:00
delete_site_transient ( 'update_core' );
2013-10-10 03:31:22 +02:00
$extra_update_stats [ 'success' ] = is_wp_error ( $core_update_result ) ? $core_update_result -> get_error_code () : true ;
if ( is_wp_error ( $core_update_result ) && 'rollback_was_required' == $core_update_result -> get_error_code () ) {
$rollback_data = $core_update_result -> get_error_data ();
$extra_update_stats [ 'success' ] = is_wp_error ( $rollback_data [ 'update' ] ) ? $rollback_data [ 'update' ] -> get_error_code () : $rollback_data [ 'update' ];
2013-10-10 03:56:44 +02:00
$extra_update_stats [ 'rollback' ] = is_wp_error ( $rollback_data [ 'rollback' ] ) ? $rollback_data [ 'rollback' ] -> get_error_code () : true ; // If it's not a WP_Error, the rollback was successful.
2013-10-10 03:31:22 +02:00
}
$extra_update_stats [ 'fs_method' ] = $GLOBALS [ 'wp_filesystem' ] -> method ;
$extra_update_stats [ 'time_taken' ] = ( time () - $start_time );
2013-10-10 16:45:55 +02:00
$extra_update_stats [ 'attempted' ] = $core_update -> version ;
2013-09-19 10:45:06 +02:00
}
2013-09-13 08:18:16 +02:00
2013-09-23 11:37:25 +02:00
// Cleanup, and check for any pending translations
2013-10-10 03:31:22 +02:00
wp_version_check ( $extra_update_stats ); // check for Core updates
2013-09-13 08:18:16 +02:00
wp_update_themes (); // Check for Theme updates
wp_update_plugins (); // Check for Plugin updates
2013-09-23 11:37:25 +02:00
// Finally, Process any new translations
$language_updates = wp_get_translation_updates ();
if ( $language_updates ) {
foreach ( $language_updates as $update ) {
self :: upgrade ( 'language' , $update );
}
// Clear existing caches
wp_clean_plugins_cache ();
wp_clean_themes_cache ();
delete_site_transient ( 'update_core' );
wp_version_check (); // check for Core updates
wp_update_themes (); // Check for Theme updates
wp_update_plugins (); // Check for Plugin updates
}
2013-09-21 08:59:25 +02:00
/**
* Filter whether to email an update summary to the site administrator .
*
* @ since 3.7 . 0
*
2013-09-23 11:37:25 +02:00
* @ param bool Whether or not email should be sent to administrator . Default true .
* @ param bool | array $core_update An array of core update data , false otherwise .
* @ param object $theme_updates Object containing theme update properties .
* @ param object $plugin_updates Object containing plugin update properties .
* @ param array $language_updates Array containing the Language updates available .
* @ param array $upgrade_results Array of the upgrade results keyed by upgrade type , and plugin / theme slug .
2013-09-21 08:59:25 +02:00
*/
2013-09-23 11:37:25 +02:00
if ( apply_filters ( 'enable_auto_upgrade_email' , true , $core_update , $theme_updates , $plugin_updates , $language_updates , self :: $upgrade_results ) )
2013-09-21 08:59:25 +02:00
self :: send_email ();
2013-09-19 10:45:06 +02:00
2013-09-17 04:27:21 +02:00
// Clear the lock
delete_site_option ( $lock_name );
2013-09-13 08:18:16 +02:00
}
2013-09-19 10:45:06 +02:00
static function send_email () {
if ( empty ( self :: $upgrade_results ) )
return ;
$upgrade_count = 0 ;
foreach ( self :: $upgrade_results as $type => $upgrades )
$upgrade_count += count ( $upgrades );
2013-09-28 01:20:08 +02:00
$body = array ();
$failures = 0 ;
2013-09-19 10:45:06 +02:00
2013-10-12 22:17:20 +02:00
$body [] = 'WordPress site: ' . network_home_url ( '/' );
2013-09-28 01:20:08 +02:00
// Core
2013-09-19 10:45:06 +02:00
if ( isset ( self :: $upgrade_results [ 'core' ] ) ) {
$result = self :: $upgrade_results [ 'core' ][ 0 ];
2013-09-28 01:20:08 +02:00
if ( $result -> result && ! is_wp_error ( $result -> result ) ) {
$body [] = sprintf ( 'SUCCESS: WordPress was successfully updated to %s' , $result -> name );
} else {
$body [] = sprintf ( 'FAILED: WordPress failed to update to %s' , $result -> name );
$failures ++ ;
2013-09-19 10:45:06 +02:00
}
2013-09-28 01:20:08 +02:00
$body [] = '' ;
2013-09-19 10:45:06 +02:00
}
2013-09-28 01:20:08 +02:00
// Plugins, Themes, Languages
foreach ( array ( 'plugin' , 'theme' , 'language' ) as $type ) {
if ( ! isset ( self :: $upgrade_results [ $type ] ) )
continue ;
$success_items = wp_list_filter ( self :: $upgrade_results [ $type ], array ( 'result' => true ) );
if ( $success_items ) {
$body [] = " The following { $type } s were successfully updated: " ;
foreach ( wp_list_pluck ( $success_items , 'name' ) as $name )
$body [] = ' * SUCCESS: ' . $name ;
}
if ( $success_items != self :: $upgrade_results [ $type ] ) {
2013-09-19 10:45:06 +02:00
// Failed updates
2013-09-28 01:20:08 +02:00
$body [] = " The following { $type } s failed to update: " ;
foreach ( self :: $upgrade_results [ $type ] as $item ) {
if ( ! $item -> result || is_wp_error ( $item -> result ) ) {
$body [] = ' * FAILED: ' . $item -> name ;
$failures ++ ;
}
2013-09-19 10:45:06 +02:00
}
}
2013-09-28 01:20:08 +02:00
$body [] = '' ;
2013-09-19 10:45:06 +02:00
}
2013-09-28 01:20:08 +02:00
if ( $failures ) {
$body [] = '' ;
$body [] = 'BETA TESTING?' ;
$body [] = '=============' ;
$body [] = '' ;
$body [] = 'If you think these failures might be due to a bug in WordPress 3.7, could you report it?' ;
$body [] = ' * Open a thread in the support forums: http://wordpress.org/support/forum/alphabeta' ;
$body [] = " * Or, if you're comfortable writing a bug report: http://core.trac.wordpress.org/ " ;
$body [] = '' ;
$body [] = 'Thanks! -- The WordPress Team' ;
$body [] = '' ;
$subject = sprintf ( '[%s] There were failures during background updates' , get_bloginfo ( 'name' ) );
} else {
$subject = sprintf ( '[%s] Background updates have finished' , get_bloginfo ( 'name' ) );
2013-09-23 11:37:25 +02:00
}
2013-09-28 01:20:08 +02:00
$body [] = 'UPGRADE LOG' ;
$body [] = '===========' ;
2013-09-19 10:45:06 +02:00
$body [] = '' ;
2013-09-23 11:37:25 +02:00
foreach ( array ( 'core' , 'plugin' , 'theme' , 'language' ) as $type ) {
2013-09-19 10:45:06 +02:00
if ( ! isset ( self :: $upgrade_results [ $type ] ) )
continue ;
foreach ( self :: $upgrade_results [ $type ] as $upgrade ) {
$body [] = $upgrade -> name ;
2013-09-28 01:20:08 +02:00
$body [] = str_repeat ( '-' , strlen ( $upgrade -> name ) );
2013-09-19 10:45:06 +02:00
foreach ( $upgrade -> messages as $message )
2013-09-28 01:20:08 +02:00
$body [] = " " . html_entity_decode ( str_replace ( '…' , '...' , $message ) );
if ( is_wp_error ( $upgrade -> result ) )
$body [] = ' Error: [' . $upgrade -> result -> get_error_code () . '] ' . $upgrade -> result -> get_error_message ();
2013-09-19 10:45:06 +02:00
$body [] = '' ;
}
}
2013-09-28 01:20:08 +02:00
//echo "<h1>\n$subject\n</h1>\n";
//echo "<pre>\n" . implode( "\n", $body ) . "\n</pre>";
2013-09-19 10:45:06 +02:00
2013-09-28 01:20:08 +02:00
wp_mail ( get_site_option ( 'admin_email' ), $subject , implode ( " \n " , $body ) );
2013-09-19 10:45:06 +02:00
}
2013-09-13 08:18:16 +02:00
}