Ïåðåéòè èç ôîðóìà íà ñàéò.

ÍîâîñòèÔàéëîâûå àðõèâû
ÏîèñêÀêòèâíûå òåìûÒîï ëèñò
ÏðàâèëàÊòî â on-line?
Âõîä Çàáûëè ïàðîëü? Ïåðâûé ðàç íà ýòîì ñàéòå? Ðåãèñòðàöèÿ
Êîìïüþòåðíûé ôîðóì Ru.Board » Èíòåðíåò » Â ïîìîùü âåáìàñòåðó » WordPress

Ìîäåðèðóåò : Cheery

 Âåðñèÿ äëÿ ïå÷àòè • ÏîäïèñàòüñÿÄîáàâèòü â çàêëàäêè
Íà ïåðâóþ ñòðàíèöóê ýòîìó ñîîáùåíèþê ïîñëåäíåìó ñîîáùåíèþ

Îòêðûòü íîâóþ òåìó     Íàïèñàòü îòâåò â ýòó òåìó

vell



Full Member
Ðåäàêòèðîâàòü | Ïðîôèëü | Ñîîáùåíèå | Öèòèðîâàòü | Ñîîáùèòü ìîäåðàòîðó

<?php
 
if (!defined('UPDRAFTPLUS_DIR')) die('No direct access allowed');
 
// Admin-area code lives here. This gets called in admin_menu, earlier than admin_init
 
global $updraftplus_admin;
if (!is_a($updraftplus_admin, 'UpdraftPlus_Admin')) $updraftplus_admin = new UpdraftPlus_Admin();
 
class UpdraftPlus_Admin {
 
    public $logged = array();
 
    private $template_directories;
 
    private $backups_instance_ids;
 
    private $auth_instance_ids = array('dropbox' => array(), 'onedrive' => array(), 'googledrive' => array(), 'googlecloud' => array());
 
    public function __construct() {
        $this->admin_init();
    }
    
    private function wp_normalize_path($path) {
        // wp_normalize_path is not present before WP 3.9
        if (function_exists('wp_normalize_path')) return wp_normalize_path($path);
        // Taken from WP 4.6
        $path = str_replace('\\', '/', $path);
        $path = preg_replace('|(?<=.)/+|', '/', $path);
        if (':' === substr($path, 1, 1)) {
            $path = ucfirst($path);
        }
        return $path;
    }
    
    /**
     * Get the path to the UI templates directory
     *
     * @return String - a filesystem directory path
     */
    public function get_templates_dir() {
        return apply_filters('updraftplus_templates_dir', $this->wp_normalize_path(UPDRAFTPLUS_DIR.'/templates'));
    }
 
    private function register_template_directories() {
 
        $template_directories = array();
 
        $templates_dir = $this->get_templates_dir();
 
        if ($dh = opendir($templates_dir)) {
            while (($file = readdir($dh)) !== false) {
                if ('.' == $file || '..' == $file) continue;
                if (is_dir($templates_dir.'/'.$file)) {
                    $template_directories[$file] = $templates_dir.'/'.$file;
                }
            }
            closedir($dh);
        }
 
        // This is the optimal hook for most extensions to hook into
        $this->template_directories = apply_filters('updraftplus_template_directories', $template_directories);
 
    }
 
    public function include_template($path, $return_instead_of_echo = false, $extract_these = array()) {
        if ($return_instead_of_echo) ob_start();
 
        if (preg_match('#^([^/]+)/(.*)$#', $path, $matches)) {
            $prefix = $matches[1];
            $suffix = $matches[2];
            if (isset($this->template_directories[$prefix])) {
                $template_file = $this->template_directories[$prefix].'/'.$suffix;
            }
        }
 
        if (!isset($template_file)) {
            $template_file = UPDRAFTPLUS_DIR.'/templates/'.$path;
        }
 
        $template_file = apply_filters('updraftplus_template', $template_file, $path);
 
        do_action('updraftplus_before_template', $path, $template_file, $return_instead_of_echo, $extract_these);
 
        if (!file_exists($template_file)) {
            error_log("UpdraftPlus: template not found: $template_file");
            echo __('Error:', 'updraftplus').' '.__('template not found', 'updraftplus')." ($path)";
        } else {
            extract($extract_these);
            global $updraftplus, $wpdb;
            $updraftplus_admin = $this;
            include $template_file;
        }
 
        do_action('updraftplus_after_template', $path, $template_file, $return_instead_of_echo, $extract_these);
 
        if ($return_instead_of_echo) return ob_get_clean();
    }
    
    /**
     * Add actions for any needed dashboard notices for remote storage services
     *
     * @param String|Array $services - a list of services, or single service
     */
    private function setup_all_admin_notices_global($services) {
        
        global $updraftplus;
 
        if ('googledrive' === $services || (is_array($services) && in_array('googledrive', $services))) {
            $settings = $updraftplus->update_remote_storage_options_format('googledrive');
            
            if (is_wp_error($settings)) {
                if (!isset($this->storage_module_option_errors)) $this->storage_module_option_errors = '';
                $this->storage_module_option_errors .= "Google Drive (".$settings->get_error_code()."): ".$settings->get_error_message();
                add_action('all_admin_notices', array($this, 'show_admin_warning_multiple_storage_options'));
                $updraftplus->log_wp_error($settings, true, true);
            } elseif (!empty($settings['settings'])) {
                foreach ($settings['settings'] as $instance_id => $storage_options) {
                    if ((defined('UPDRAFTPLUS_CUSTOM_GOOGLEDRIVE_APP') && UPDRAFTPLUS_CUSTOM_GOOGLEDRIVE_APP) || !empty($storage_options['clientid'])) {
                        if (!empty($storage_options['clientid'])) {
                            $clientid = $storage_options['clientid'];
                            $token = empty($storage_options['token']) ? '' : $storage_options['token'];
                        }
                        if (!empty($clientid) && '' == $token) {
                            if (!in_array($instance_id, $this->auth_instance_ids['googledrive'])) $this->auth_instance_ids['googledrive'][] = $instance_id;
                            if (false === has_action('all_admin_notices', array($this, 'show_admin_warning_googledrive'))) add_action('all_admin_notices', array($this, 'show_admin_warning_googledrive'));
                        }
                        unset($clientid);
                        unset($token);
                    } else {
                        if (empty($storage_options['user_id'])) {
                            if (!in_array($instance_id, $this->auth_instance_ids['googledrive'])) $this->auth_instance_ids['googledrive'][] = $instance_id;
                            if (false === has_action('all_admin_notices', array($this, 'show_admin_warning_googledrive'))) add_action('all_admin_notices', array($this, 'show_admin_warning_googledrive'));
                        }
                    }
                }
            }
        }
        if ('googlecloud' === $services || (is_array($services) && in_array('googlecloud', $services))) {
            $settings = $updraftplus->update_remote_storage_options_format('googlecloud');
            
            if (is_wp_error($settings)) {
                if (!isset($this->storage_module_option_errors)) $this->storage_module_option_errors = '';
                $this->storage_module_option_errors .= "Google Cloud (".$settings->get_error_code()."): ".$settings->get_error_message();
                add_action('all_admin_notices', array($this, 'show_admin_warning_multiple_storage_options'));
                $updraftplus->log_wp_error($settings, true, true);
            } elseif (!empty($settings['settings'])) {
                foreach ($settings['settings'] as $instance_id => $storage_options) {
                    $clientid = $storage_options['clientid'];
                    $token = (empty($storage_options['token'])) ? '' : $storage_options['token'];
                    
                    if (!empty($clientid) && empty($token)) {
                        if (!in_array($instance_id, $this->auth_instance_ids['googlecloud'])) $this->auth_instance_ids['googlecloud'][] = $instance_id;
                        if (false === has_action('all_admin_notices', array($this, 'show_admin_warning_googlecloud'))) add_action('all_admin_notices', array($this, 'show_admin_warning_googlecloud'));
                    }
                }
            }
        }
        
        if ('dropbox' === $services || (is_array($services) && in_array('dropbox', $services))) {
            $settings = $updraftplus->update_remote_storage_options_format('dropbox');
            
            if (is_wp_error($settings)) {
                if (!isset($this->storage_module_option_errors)) $this->storage_module_option_errors = '';
                $this->storage_module_option_errors .= "Dropbox (".$settings->get_error_code()."): ".$settings->get_error_message();
                add_action('all_admin_notices', array($this, 'show_admin_warning_multiple_storage_options'));
                $updraftplus->log_wp_error($settings, true, true);
            } elseif (!empty($settings['settings'])) {
                foreach ($settings['settings'] as $instance_id => $storage_options) {
                    if (empty($storage_options['tk_access_token'])) {
                        if (!in_array($instance_id, $this->auth_instance_ids['dropbox'])) $this->auth_instance_ids['dropbox'][] = $instance_id;
                        if (false === has_action('all_admin_notices', array($this, 'show_admin_warning_dropbox'))) add_action('all_admin_notices', array($this, 'show_admin_warning_dropbox'));
                    }
                }
            }
        }
        
        if ('onedrive' === $services || (is_array($services) && in_array('onedrive', $services))) {
            $settings = $updraftplus->update_remote_storage_options_format('onedrive');
            
            if (is_wp_error($settings)) {
                if (!isset($this->storage_module_option_errors)) $this->storage_module_option_errors = '';
                $this->storage_module_option_errors .= "OneDrive (".$settings->get_error_code()."): ".$settings->get_error_message();
                add_action('all_admin_notices', array($this, 'show_admin_warning_multiple_storage_options'));
                $updraftplus->log_wp_error($settings, true, true);
            } elseif (!empty($settings['settings'])) {
                foreach ($settings['settings'] as $instance_id => $storage_options) {
                    if ((defined('UPDRAFTPLUS_CUSTOM_ONEDRIVE_APP') && UPDRAFTPLUS_CUSTOM_ONEDRIVE_APP)) {
                        if (!empty($storage_options['clientid']) && !empty($storage_options['secret']) && empty($storage_options['refresh_token'])) {
                                if (!in_array($instance_id, $this->auth_instance_ids['onedrive'])) $this->auth_instance_ids['onedrive'][] = $instance_id;
                                if (false === has_action('all_admin_notices', array($this, 'show_admin_warning_onedrive'))) add_action('all_admin_notices', array($this, 'show_admin_warning_onedrive'));
                        } elseif (empty($storage_options['refresh_token'])) {
                            if (!in_array($instance_id, $this->auth_instance_ids['onedrive'])) $this->auth_instance_ids['onedrive'][] = $instance_id;
                            if (false === has_action('all_admin_notices', array($this, 'show_admin_warning_onedrive'))) add_action('all_admin_notices', array($this, 'show_admin_warning_onedrive'));
                        }
                    } else {
                        if (empty($storage_options['refresh_token'])) {
                            if (!in_array($instance_id, $this->auth_instance_ids['onedrive'])) $this->auth_instance_ids['onedrive'][] = $instance_id;
                            if (false === has_action('all_admin_notices', array($this, 'show_admin_warning_onedrive'))) add_action('all_admin_notices', array($this, 'show_admin_warning_onedrive'));
                        }
                    }
                }
            }
        }
 
        if ('updraftvault' === $services || (is_array($services) && in_array('updraftvault', $services))) {
            $settings = $updraftplus->update_remote_storage_options_format('updraftvault');
            
            if (is_wp_error($settings)) {
                if (!isset($this->storage_module_option_errors)) $this->storage_module_option_errors = '';
                $this->storage_module_option_errors .= "UpdraftVault (".$settings->get_error_code()."): ".$settings->get_error_message();
                add_action('all_admin_notices', array($this, 'show_admin_warning_multiple_storage_options'));
                $updraftplus->log_wp_error($settings, true, true);
            } elseif (!empty($settings['settings'])) {
                foreach ($settings['settings'] as $instance_id => $storage_options) {
                    if (empty($storage_options['token']) && empty($storage_options['email'])) {
                        add_action('all_admin_notices', array($this, 'show_admin_warning_updraftvault'));
                    }
                }
            }
        }
 
        if ($this->disk_space_check(1048576*35) === false) add_action('all_admin_notices', array($this, 'show_admin_warning_diskspace'));
    }
    
    private function setup_all_admin_notices_udonly($service, $override = false) {
        global $wp_version;
 
        if (UpdraftPlus_Options::user_can_manage() && defined('DISABLE_WP_CRON') && DISABLE_WP_CRON && (!defined('UPDRAFTPLUS_DISABLE_WP_CRON_NOTICE') || !UPDRAFTPLUS_DISABLE_WP_CRON_NOTICE)) {
            add_action('all_admin_notices', array($this, 'show_admin_warning_disabledcron'));
        }
 
        if (UpdraftPlus_Options::get_updraft_option('updraft_debug_mode')) {
            @ini_set('display_errors', 1);
            @error_reporting(E_ALL & ~E_NOTICE & ~E_DEPRECATED);
            add_action('all_admin_notices', array($this, 'show_admin_debug_warning'));
        }
 
        if (null === UpdraftPlus_Options::get_updraft_option('updraft_interval')) {
            add_action('all_admin_notices', array($this, 'show_admin_nosettings_warning'));
            $this->no_settings_warning = true;
        }
 
        // Avoid false positives, by attempting to raise the limit (as happens when we actually do a backup)
        @set_time_limit(UPDRAFTPLUS_SET_TIME_LIMIT);
        $max_execution_time = (int) @ini_get('max_execution_time');
        if ($max_execution_time>0 && $max_execution_time<20) {
            add_action('all_admin_notices', array($this, 'show_admin_warning_execution_time'));
        }
 
        // LiteSpeed has a generic problem with terminating cron jobs
        if (isset($_SERVER['SERVER_SOFTWARE']) && strpos($_SERVER['SERVER_SOFTWARE'], 'LiteSpeed') !== false) {
            if (!is_file(ABSPATH.'.htaccess') || !preg_match('/noabort/i', file_get_contents(ABSPATH.'.htaccess'))) {
                add_action('all_admin_notices', array($this, 'show_admin_warning_litespeed'));
            }
        }
 
        if (version_compare($wp_version, '3.2', '<')) add_action('all_admin_notices', array($this, 'show_admin_warning_wordpressversion'));
    }
    
    /**
     * Used to output the information for the next scheduled backup.
     * moved to function for the ajax saves
     */
    public function next_scheduled_backups_output() {
        // UNIX timestamp
        $next_scheduled_backup = wp_next_scheduled('updraft_backup');
        if ($next_scheduled_backup) {
            // Convert to GMT
            $next_scheduled_backup_gmt = gmdate('Y-m-d H:i:s', $next_scheduled_backup);
            // Convert to blog time zone
            $next_scheduled_backup = get_date_from_gmt($next_scheduled_backup_gmt, 'D, F j, Y H:i');
            // $next_scheduled_backup = date_i18n('D, F j, Y H:i', $next_scheduled_backup);
        } else {
            $next_scheduled_backup = __('Nothing currently scheduled', 'updraftplus');
            $files_not_scheduled = true;
        }
        
        $next_scheduled_backup_database = wp_next_scheduled('updraft_backup_database');
        if (UpdraftPlus_Options::get_updraft_option('updraft_interval_database', UpdraftPlus_Options::get_updraft_option('updraft_interval')) == UpdraftPlus_Options::get_updraft_option('updraft_interval')) {
            if (isset($files_not_scheduled)) {
                $next_scheduled_backup_database = $next_scheduled_backup;
                $database_not_scheduled = true;
            } else {
                $next_scheduled_backup_database = __("At the same time as the files backup", 'updraftplus');
                $next_scheduled_backup_database_same_time = true;
            }
        } else {
            if ($next_scheduled_backup_database) {
                // Convert to GMT
                $next_scheduled_backup_database_gmt = gmdate('Y-m-d H:i:s', $next_scheduled_backup_database);
                // Convert to blog time zone
                $next_scheduled_backup_database = get_date_from_gmt($next_scheduled_backup_database_gmt, 'D, F j, Y H:i');
                // $next_scheduled_backup_database = date_i18n('D, F j, Y H:i', $next_scheduled_backup_database);
            } else {
                $next_scheduled_backup_database = __('Nothing currently scheduled', 'updraftplus');
                $database_not_scheduled = true;
            }
        }
        ?>
        <tr>
        <?php if (isset($files_not_scheduled) && isset($database_not_scheduled)) { ?>
            <td colspan="2" class="not-scheduled"><?php _e('Nothing currently scheduled', 'updraftplus'); ?></td>
        <?php } else { ?>
            <td class="updraft_scheduled"><?php echo empty($next_scheduled_backup_database_same_time) ? __('Files', 'updraftplus') : __('Files and database', 'updraftplus'); ?>:</td><td class="updraft_all-files"><?php echo $next_scheduled_backup; ?></td>
            </tr>
            <?php if (empty($next_scheduled_backup_database_same_time)) { ?>
                <tr>
                    <td class="updraft_scheduled"><?php _e('Database', 'updraftplus');?>: </td><td class="updraft_all-files"><?php echo $next_scheduled_backup_database; ?></td>
                </tr>
            <?php } ?>
        <?php
        }
    }
    
    /**
     * Run upon the WP admin_init action
     */
    private function admin_init() {
 
        add_action('core_upgrade_preamble', array($this, 'core_upgrade_preamble'));
        add_action('admin_action_upgrade-plugin', array($this, 'admin_action_upgrade_pluginortheme'));
        add_action('admin_action_upgrade-theme', array($this, 'admin_action_upgrade_pluginortheme'));
 
        add_action('admin_head', array($this, 'admin_head'));
        add_filter((is_multisite() ? 'network_admin_' : '').'plugin_action_links', array($this, 'plugin_action_links'), 10, 2);
        add_action('wp_ajax_updraft_download_backup', array($this, 'updraft_download_backup'));
        add_action('wp_ajax_updraft_ajax', array($this, 'updraft_ajax_handler'));
        add_action('wp_ajax_updraft_ajaxrestore', array($this, 'updraft_ajaxrestore'));
        add_action('wp_ajax_nopriv_updraft_ajaxrestore', array($this, 'updraft_ajaxrestore'));
        
        add_action('wp_ajax_plupload_action', array($this, 'plupload_action'));
        add_action('wp_ajax_plupload_action2', array($this, 'plupload_action2'));
 
        add_action('wp_before_admin_bar_render', array($this, 'wp_before_admin_bar_render'));
 
        // Add a new Ajax action for saving settings
        add_action('wp_ajax_updraft_savesettings', array($this, 'updraft_ajax_savesettings'));
        
        // Ajax for settings import and export
        add_action('wp_ajax_updraft_importsettings', array($this, 'updraft_ajax_importsettings'));
 
        // UpdraftPlus templates
        $this->register_template_directories();
        
        global $updraftplus, $wp_version, $pagenow;
        add_filter('updraftplus_dirlist_others', array($updraftplus, 'backup_others_dirlist'));
        add_filter('updraftplus_dirlist_uploads', array($updraftplus, 'backup_uploads_dirlist'));
 
        // First, the checks that are on all (admin) pages:
 
        $service = UpdraftPlus_Options::get_updraft_option('updraft_service');
 
        if (UpdraftPlus_Options::user_can_manage()) {
 
            $this->print_restore_in_progress_box_if_needed();
 
            // Main dashboard page advert
            // Since our nonce is printed, make sure they have sufficient credentials
            if ('index.php' == $pagenow && current_user_can('update_plugins') && (!file_exists(UPDRAFTPLUS_DIR.'/udaddons') || (defined('UPDRAFTPLUS_FORCE_DASHNOTICE') && UPDRAFTPLUS_FORCE_DASHNOTICE))) {
 
                $dismissed_until = UpdraftPlus_Options::get_updraft_option('updraftplus_dismisseddashnotice', 0);
                
                $backup_dir = $updraftplus->backups_dir_location();
                // N.B. Not an exact proxy for the installed time; they may have tweaked the expert option to move the directory
                $installed = @filemtime($backup_dir.'/index.html');
                $installed_for = time() - $installed;
 
                if (($installed && time() > $dismissed_until && $installed_for > 28*86400 && !defined('UPDRAFTPLUS_NOADS_B')) || (defined('UPDRAFTPLUS_FORCE_DASHNOTICE') && UPDRAFTPLUS_FORCE_DASHNOTICE)) {
                    add_action('all_admin_notices', array($this, 'show_admin_notice_upgradead'));
                }
            }
            
            // Moved out for use with Ajax saving
            $this->setup_all_admin_notices_global($service);
        }
 
        // Next, the actions that only come on the UpdraftPlus page
        if (UpdraftPlus_Options::admin_page() != $pagenow || empty($_REQUEST['page']) || 'updraftplus' != $_REQUEST['page']) return;
        $this->setup_all_admin_notices_udonly($service);
        
        add_action('admin_enqueue_scripts', array($this, 'admin_enqueue_scripts'), 99999);
    }
 
    /**
     * Sets up what is needed to allow an in-page backup to be run. Will enqueue scripts and output appropriate HTML (so, should be run when at a suitable place). Not intended for use on the UpdraftPlus settings page.
     *
     * @param string   $title    Text to use for the title of the modal
     * @param callable $callback Callable function to output the contents of the updraft_inpage_prebackup element - i.e. what shows in the modal before a backup beings.
     */
    public function add_backup_scaffolding($title, $callback) {
        $this->admin_enqueue_scripts();
        ?>
        <script>
        // TODO: This is not the best way.
        var updraft_credentialtest_nonce='<?php echo wp_create_nonce('updraftplus-credentialtest-nonce');?>';
        </script>
        <div id="updraft-poplog" >
            <pre id="updraft-poplog-content" style="white-space: pre-wrap;"></pre>
        </div>
        
        <div id="updraft-backupnow-inpage-modal" title="UpdraftPlus - <?php echo $title; ?>">
 
            <div id="updraft_inpage_prebackup" style="float:left; clear:both;">
                <?php call_user_func($callback); ?>
            </div>
 
            <div id="updraft_inpage_backup" style="float:left; clear:both;">
 
                <h2><?php echo $title;?></h2>
 
                <div id="updraft_backup_started" class="updated" style="display:none; max-width: 560px; font-size:100%; line-height: 100%; padding:6px; clear:left;"></div>
 
                <?php $this->render_active_jobs_and_log_table(true, false); ?>
                
            </div>
 
        </div>
        <?php
    }
    
    public function updraft_ajaxrestore() {
// TODO: All needs testing with restricted filesystem permissions. Those credentials need to be POST-ed too - currently not.
// TODO
// error_log(serialize($_POST));
 
        if (empty($_POST['subaction']) || 'restore' != $_POST['subaction']) {
            echo json_encode(array('e' => 'Illegitimate data sent (0)'));
            die();
        }
 
        if (empty($_POST['restorenonce'])) {
            echo json_encode(array('e' => 'Illegitimate data sent (1)'));
            die();
        }
 
        $restore_nonce = (string) $_POST['restorenonce'];
 
        if (empty($_POST['ajaxauth'])) {
            echo json_encode(array('e' => 'Illegitimate data sent (2)'));
            die();
        }
 
        global $updraftplus;
 
        $ajax_auth = get_site_option('updraft_ajax_restore_'.$restore_nonce);
 
        if (!$ajax_auth) {
            echo json_encode(array('e' => 'Illegitimate data sent (3)'));
            die();
        }
 
        if (!preg_match('/^([0-9a-f]+)\d+)/i', $ajax_auth, $matches)) {
            echo json_encode(array('e' => 'Illegitimate data sent (4)'));
            die();
        }
 
        $nonce_time = $matches[2];
        $auth_code_sent = $matches[1];
        if (time() > $nonce_time + 600) {
            echo json_encode(array('e' => 'Illegitimate data sent (5)'));
            die();
        }
 
// TODO: Deactivate the auth code whilst the operation is underway
 
        $last_one = empty($_POST['lastone']) ? false : true;
 
        @set_time_limit(UPDRAFTPLUS_SET_TIME_LIMIT);
 
        $updraftplus->backup_time_nonce($restore_nonce);
        $updraftplus->logfile_open($restore_nonce);
 
        $timestamp = empty($_POST['timestamp']) ? false : (int) $_POST['timestamp'];
        $multisite = empty($_POST['multisite']) ? false : (bool) $_POST['multisite'];
        $created_by_version = empty($_POST['created_by_version']) ? false : (int) $_POST['created_by_version'];
 
        // TODO: We need to know about first_one (not yet sent), as well as last_one
 
        // TODO: Verify the values of these
        $type = empty($_POST['type']) ? false : (int) $_POST['type'];
        $backupfile = empty($_POST['backupfile']) ? false : (string) $_POST['backupfile'];
 
        $updraftplus->log("Deferred restore resumption: $type: $backupfile (timestamp=$timestamp, last_one=$last_one)");
 
 
 
        $backupable_entities = $updraftplus->get_backupable_file_entities(true);
 
        if (!isset($backupable_entities[$type])) {
            echo json_encode(array('e' => 'Illegitimate data sent (6 - no such entity)', 'data' => $type));
            die();
        }
 
 
        if ($last_one) {
            // Remove the auth nonce from the DB to prevent abuse
            delete_site_option('updraft_ajax_restore_'.$restore_nonce);
        } else {
            // Reset the counter after a successful operation
            update_site_option('updraft_ajax_restore_'.$restore_nonce, $auth_code_sent.':'.time());
        }
 
        echo json_encode(array('e' => 'TODO', 'd' => $_POST));
        die;
    }
 
    public function wp_before_admin_bar_render() {
        global $wp_admin_bar;
        
        if (!UpdraftPlus_Options::user_can_manage()) return;
        if (defined('UPDRAFTPLUS_ADMINBAR_DISABLE') && UPDRAFTPLUS_ADMINBAR_DISABLE) return;
 
        if (false == apply_filters('updraftplus_settings_page_render', true)) return;
 
        $option_location = UpdraftPlus_Options::admin_page_url();
        
        $args = array(
            'id' => 'updraft_admin_node',
            'title' => apply_filters('updraftplus_admin_node_title', 'UpdraftPlus')
        );
        $wp_admin_bar->add_node($args);
        
        $args = array(
            'id' => 'updraft_admin_node_status',
            'title' => __('Current Status', 'updraftplus').' / '.__('Backup Now', 'updraftplus'),
            'parent' => 'updraft_admin_node',
            'href' => $option_location.'?page=updraftplus&tab=status'
        );
        $wp_admin_bar->add_node($args);
        
        $args = array(
            'id' => 'updraft_admin_node_backups',
            'title' => __('Existing Backups', 'updraftplus'),
            'parent' => 'updraft_admin_node',
            'href' => $option_location.'?page=updraftplus&tab=backups'
        );
        $wp_admin_bar->add_node($args);
        
        $args = array(
            'id' => 'updraft_admin_node_settings',
            'title' => __('Settings', 'updraftplus'),
            'parent' => 'updraft_admin_node',
            'href' => $option_location.'?page=updraftplus&tab=settings'
        );
        $wp_admin_bar->add_node($args);
        
        $args = array(
            'id' => 'updraft_admin_node_expert_content',
            'title' => __('Advanced Tools', 'updraftplus'),
            'parent' => 'updraft_admin_node',
            'href' => $option_location.'?page=updraftplus&tab=expert'
        );
        $wp_admin_bar->add_node($args);
        
        $args = array(
            'id' => 'updraft_admin_node_addons',
            'title' => __('Extensions', 'updraftplus'),
            'parent' => 'updraft_admin_node',
            'href' => $option_location.'?page=updraftplus&tab=addons'
        );
        $wp_admin_bar->add_node($args);
        
        global $updraftplus;
        if (!$updraftplus->have_addons) {
            $args = array(
                'id' => 'updraft_admin_node_premium',
                'title' => 'UpdraftPlus Premium',
                'parent' => 'updraft_admin_node',
                'href' => apply_filters('updraftplus_com_link', 'https://updraftplus.com/shop/updraftplus-premium/')
            );
            $wp_admin_bar->add_node($args);
        }
    }
 
    /**
     * Output HTML for a dashboard notice highlighting the benefits of upgrading to Premium
     */
    public function show_admin_notice_upgradead() {
        $this->include_template('wp-admin/notices/thanks-for-using-main-dash.php');
    }
 
    private function ensure_sufficient_jquery_and_enqueue() {
        global $updraftplus, $wp_version;
        
        $enqueue_version = (defined('SCRIPT_DEBUG') && SCRIPT_DEBUG) ? $updraftplus->version.'.'.time() : $updraftplus->version;
        $min_or_not = (defined('SCRIPT_DEBUG') && SCRIPT_DEBUG) ? '' : '.min';
        
        if (version_compare($wp_version, '3.3', '<')) {
            // Require a newer jQuery (3.2.1 has 1.6.1, so we go for something not too much newer). We use .on() in a way that is incompatible with < 1.7
            wp_deregister_script('jquery');
            $jquery_enqueue_version = (defined('SCRIPT_DEBUG') && SCRIPT_DEBUG) ? '1.7.2'.'.'.time() : '1.7.2';
            wp_register_script('jquery', 'https://ajax.googleapis.com/ajax/libs/jquery/1.7.2/jquery'.$min_or_not.'.js', false, $jquery_enqueue_version, false);
            wp_enqueue_script('jquery');
            // No plupload until 3.3
            wp_enqueue_script('updraftplus-admin', UPDRAFTPLUS_URL.'/includes/updraft-admin'.$min_or_not.'.js', array('jquery', 'jquery-ui-dialog'), $enqueue_version, true);
        } else {
            wp_enqueue_script('updraftplus-admin', UPDRAFTPLUS_URL.'/includes/updraft-admin'.$min_or_not.'.js', array('jquery', 'jquery-ui-dialog', 'plupload-all'), $enqueue_version);
        }
        
    }
 
    /**
     * This is also called directly from the auto-backup add-on
     */
    public function admin_enqueue_scripts() {
 
        global $updraftplus, $wp_locale;
        
        $enqueue_version = (defined('SCRIPT_DEBUG') && SCRIPT_DEBUG) ? $updraftplus->version.'.'.time() : $updraftplus->version;
        $min_or_not = (defined('SCRIPT_DEBUG') && SCRIPT_DEBUG) ? '' : '.min';
 
        // Defeat other plugins/themes which dump their jQuery UI CSS onto our settings page
        wp_deregister_style('jquery-ui');
        $jquery_ui_css_enqueue_version = (defined('SCRIPT_DEBUG') && SCRIPT_DEBUG) ? '1.11.4'.'.'.time() : '1.11.4';
        wp_enqueue_style('jquery-ui', UPDRAFTPLUS_URL.'/includes/jquery-ui.custom'.$min_or_not.'.css', array(), $jquery_ui_css_enqueue_version);
    
        wp_enqueue_style('updraft-admin-css', UPDRAFTPLUS_URL.'/css/admin'.$min_or_not.'.css', array(), $enqueue_version);
        // add_filter('style_loader_tag', array($this, 'style_loader_tag'), 10, 2);
 
        $this->ensure_sufficient_jquery_and_enqueue();
        $jquery_blockui_enqueue_version = (defined('SCRIPT_DEBUG') && SCRIPT_DEBUG) ? '2.70.0'.'.'.time() : '2.70.0';
        wp_enqueue_script('jquery-blockui', UPDRAFTPLUS_URL.'/includes/jquery.blockUI'.$min_or_not.'.js', array('jquery'), $jquery_blockui_enqueue_version);
    
        wp_enqueue_script('jquery-labelauty', UPDRAFTPLUS_URL.'/includes/labelauty/jquery-labelauty'.$min_or_not.'.js', array('jquery'), $enqueue_version);
        wp_enqueue_style('jquery-labelauty', UPDRAFTPLUS_URL.'/includes/labelauty/jquery-labelauty'.$min_or_not.'.css', array(), $enqueue_version);
        $serialize_js_enqueue_version = (defined('SCRIPT_DEBUG') && SCRIPT_DEBUG) ? '2.8.1'.'.'.time() : '2.8.1';
        wp_enqueue_script('jquery.serializeJSON', UPDRAFTPLUS_URL.'/includes/jquery.serializeJSON/jquery.serializejson'.$min_or_not.'.js', array('jquery'), $serialize_js_enqueue_version);
 
        $this->enqueue_jstree();
        
        do_action('updraftplus_admin_enqueue_scripts');
        
        $day_selector = '';
        for ($day_index = 0; $day_index <= 6; $day_index++) {
            // $selected = ($opt == $day_index) ? 'selected="selected"' : '';
            $selected = '';
            $day_selector .= "\n\t<option value='" . $day_index . "' $selected>" . $wp_locale->get_weekday($day_index) . '</option>';
        }
 
        $mday_selector = '';
        for ($mday_index = 1; $mday_index <= 28; $mday_index++) {
            // $selected = ($opt == $mday_index) ? 'selected="selected"' : '';
            $selected = '';
            $mday_selector .= "\n\t<option value='" . $mday_index . "' $selected>" . $mday_index . '</option>';
        }
 
        wp_localize_script('updraftplus-admin', 'updraftlion', array(
            'sendonlyonwarnings' => __('Send a report only when there are warnings/errors', 'updraftplus'),
            'wholebackup' => __('When the Email storage method is enabled, also send the entire backup', 'updraftplus'),
            'emailsizelimits' => esc_attr(sprintf(__('Be aware that mail servers tend to have size limits; typically around %s Mb; backups larger than any limits will likely not arrive.', 'updraftplus'), '10-20')),
            'rescanning' => __('Rescanning (looking for backups that you have uploaded manually into the internal backup store)...', 'updraftplus'),
            'rescanningremote' => __('Rescanning remote and local storage for backup sets...', 'updraftplus'),
            'enteremailhere' => esc_attr(__('To send to more than one address, separate each address with a comma.', 'updraftplus')),
            'excludedeverything' => __('If you exclude both the database and the files, then you have excluded everything!', 'updraftplus'),
            'nofileschosen' => __('You have chosen to backup files, but no file entities have been selected', 'updraftplus'),
            'notableschosen' => __('You have chosen to backup a database, but no tables have been selected', 'updraftplus'),
            'restore_proceeding' => __('The restore operation has begun. Do not press stop or close your browser until it reports itself as having finished.', 'updraftplus'),
            'unexpectedresponse' => __('Unexpected response:', 'updraftplus'),
            'servererrorcode' => __('The web server returned an error code (try again, or check your web server logs)', 'updraftplus'),
            'newuserpass' => __("The new user's RackSpace console password is (this will not be shown again):", 'updraftplus'),
            'trying' => __('Trying...', 'updraftplus'),
            'fetching' => __('Fetching...', 'updraftplus'),
            'calculating' => __('calculating...', 'updraftplus'),
            'begunlooking' => __('Begun looking for this entity', 'updraftplus'),
            'stilldownloading' => __('Some files are still downloading or being processed - please wait.', 'updraftplus'),
            'processing' => __('Processing files - please wait...', 'updraftplus'),
            'emptyresponse' => __('Error: the server sent an empty response.', 'updraftplus'),
            'warnings' => __('Warnings:', 'updraftplus'),
            'errors' => __('Errors:', 'updraftplus'),
            'jsonnotunderstood' => __('Error: the server sent us a response which we did not understand.', 'updraftplus'),
            'errordata' => __('Error data:', 'updraftplus'),
            'error' => __('Error:', 'updraftplus'),
            'errornocolon' => __('Error', 'updraftplus'),
            'existing_backups' => __('Existing Backups', 'updraftplus'),
            'fileready' => __('File ready.', 'updraftplus'),
            'actions' => __('Actions', 'updraftplus'),
            'deletefromserver' => __('Delete from your web server', 'updraftplus'),
            'downloadtocomputer' => __('Download to your computer', 'updraftplus'),
            'browse_contents' => __('Browse contents', 'updraftplus'),
            'notunderstood' => __('Download error: the server sent us a response which we did not understand.', 'updraftplus'),
            'requeststart' => __('Requesting start of backup...', 'updraftplus'),
            'phpinfo' => __('PHP information', 'updraftplus'),
            'delete_old_dirs' => __('Delete Old Directories', 'updraftplus'),
            'raw' => __('Raw backup history', 'updraftplus'),
            'notarchive' => __('This file does not appear to be an UpdraftPlus backup archive (such files are .zip or .gz files which have a name like: backup_(time)_(site name)_(code)_(type).(zip|gz)).', 'updraftplus').' '.__('However, UpdraftPlus archives are standard zip/SQL files - so if you are sure that your file has the right format, then you can rename it to match that pattern.', 'updraftplus'),
            'notarchive2' => '<p>'.__('This file does not appear to be an UpdraftPlus backup archive (such files are .zip or .gz files which have a name like: backup_(time)_(site name)_(code)_(type).(zip|gz)).', 'updraftplus').'</p> '.apply_filters('updraftplus_if_foreign_then_premium_message', '<p><a href="'.apply_filters('updraftplus_com_link', "https://updraftplus.com/shop/updraftplus-premium/").'">'.__('If this is a backup created by a different backup plugin, then UpdraftPlus Premium may be able to help you.', 'updraftplus').'</a></p>'),
            'makesure' => __('(make sure that you were trying to upload a zip file previously created by UpdraftPlus)', 'updraftplus'),
            'uploaderror' => __('Upload error:', 'updraftplus'),
            'notdba' => __('This file does not appear to be an UpdraftPlus encrypted database archive (such files are .gz.crypt files which have a name like: backup_(time)_(site name)_(code)_db.crypt.gz).', 'updraftplus'),
            'uploaderr' => __('Upload error', 'updraftplus'),
            'followlink' => __('Follow this link to attempt decryption and download the database file to your computer.', 'updraftplus'),
            'thiskey' => __('This decryption key will be attempted:', 'updraftplus'),
            'unknownresp' => __('Unknown server response:', 'updraftplus'),
            'ukrespstatus' => __('Unknown server response status:', 'updraftplus'),
            'uploaded' => __('The file was uploaded.', 'updraftplus'),
            'backupnow' => __('Backup Now', 'updraftplus'),
            'cancel' => __('Cancel', 'updraftplus'),
            'deletebutton' => __('Delete', 'updraftplus'),
            'createbutton' => __('Create', 'updraftplus'),
            'youdidnotselectany' => __('You did not select any components to restore. Please select at least one, and then try again.', 'updraftplus'),
            'proceedwithupdate' => __('Proceed with update', 'updraftplus'),
            'close' => __('Close', 'updraftplus'),
            'restore' => __('Restore', 'updraftplus'),
            'downloadlogfile' => __('Download log file', 'updraftplus'),
            'automaticbackupbeforeupdate' => __('Automatic backup before update', 'updraftplus'),
            'unsavedsettings' => __('You have made changes to your settings, and not saved.', 'updraftplus'),
            'saving' => __('Saving...', 'updraftplus'),
            'connect' => __('Connect', 'updraftplus'),
            'connecting' => __('Connecting...', 'updraftplus'),
            'disconnect' => __('Disconnect', 'updraftplus'),
            'disconnecting' => __('Disconnecting...', 'updraftplus'),
            'counting' => __('Counting...', 'updraftplus'),
            'updatequotacount' => __('Update quota count', 'updraftplus'),
            'addingsite' => __('Adding...', 'updraftplus'),
            'addsite' => __('Add site', 'updraftplus'),
            // 'resetting' => __('Resetting...', 'updraftplus'),
            'creating_please_allow' => __('Creating...', 'updraftplus').(function_exists('openssl_encrypt') ? '' : ' ('.__('your PHP install lacks the openssl module; as a result, this can take minutes; if nothing has happened by then, then you should either try a smaller key size, or ask your web hosting company how to enable this PHP module on your setup.', 'updraftplus').')'),
            'sendtosite' => __('Send to site:', 'updraftplus'),
            'checkrpcsetup' => sprintf(__('You should check that the remote site is online, not firewalled, does not have security modules that may be blocking access, has UpdraftPlus version %s or later active and that the keys have been entered correctly.', 'updraftplus'), '2.10.3'),
            'pleasenamekey' => __('Please give this key a name (e.g. indicate the site it is for):', 'updraftplus'),
            'key' => __('Key', 'updraftplus'),
            'nokeynamegiven' => sprintf(__("Failure: No %s was given.", 'updraftplus'), __('key name', 'updraftplus')),
            'deleting' => __('Deleting...', 'updraftplus'),
            'enter_mothership_url' => __('Please enter a valid URL', 'updraftplus'),
            'delete_response_not_understood' => __("We requested to delete the file, but could not understand the server's response", 'updraftplus'),
            'testingconnection' => __('Testing connection...', 'updraftplus'),
            'send' => __('Send', 'updraftplus'),
            'migratemodalheight' => class_exists('UpdraftPlus_Addons_Migrator') ? 555 : 300,
            'migratemodalwidth' => class_exists('UpdraftPlus_Addons_Migrator') ? 770 : 500,
            'download' => _x('Download', '(verb)', 'updraftplus'),
            'browse_download_link' => apply_filters('updraftplus_browse_download_link', '<a id="updraft_zip_download_notice" href="'.apply_filters('updraftplus_com_link', "https://updraftplus.com/landing/updraftplus-premium").'">'.__("With UpdraftPlus Premium, you can directly download individual files from here.", "updraftplus").'</a>'),
            'unsavedsettingsbackup' => __('You have made changes to your settings, and not saved.', 'updraftplus')."\n".__('You should save your changes to ensure that they are used for making your backup.', 'updraftplus'),
            'unsaved_settings_export' => __('You have made changes to your settings, and not saved.', 'updraftplus')."\n".__('Your export file will be of your displayed settings, not your saved ones.', 'updraftplus'),
            'dayselector' => $day_selector,
            'mdayselector' => $mday_selector,
            'day' => __('day', 'updraftplus'),
            'inthemonth' => __('in the month', 'updraftplus'),
            'days' => __('day(s)', 'updraftplus'),
            'hours' => __('hour(s)', 'updraftplus'),
            'weeks' => __('week(s)', 'updraftplus'),
            'forbackupsolderthan' => __('For backups older than', 'updraftplus'),
            'ud_url' => UPDRAFTPLUS_URL,
            'processing' => __('Processing...', 'updraftplus'),
            'pleasefillinrequired' => __('Please fill in the required information.', 'updraftplus'),
            'test_settings' => __('Test %s Settings', 'updraftplus'),
            'testing_settings' => __('Testing %s Settings...', 'updraftplus'),
            'settings_test_result' => __('%s settings test result:', 'updraftplus'),
            'nothing_yet_logged' => __('Nothing yet logged', 'updraftplus'),
            'import_select_file' => __('You have not yet selected a file to import.', 'updraftplus'),
            'import_invalid_json_file' => __('Error: The chosen file is corrupt. Please choose a valid UpdraftPlus export file.', 'updraftplus'),
            'updraft_settings_url' => UpdraftPlus_Options::admin_page_url().'?page=updraftplus',
            'network_site_url' => network_site_url(),
            'importing' => __('Importing...', 'updraftplus'),
            'importing_data_from' => __('This will import data from:', 'updraftplus'),
            'exported_on' => __('Which was exported on:', 'updraftplus'),
            'continue_import' => __('Do you want to carry out the import?', 'updraftplus'),
            'complete' => __('Complete', 'updraftplus'),
            'remote_delete_limit' => defined('UPDRAFTPLUS_REMOTE_DELETE_LIMIT') ? UPDRAFTPLUS_REMOTE_DELETE_LIMIT : 15,
            'remote_files_deleted' => __('remote files deleted', 'updraftplus'),
            'http_code' => __('HTTP code:', 'updraftplus'),
            'makesure2' => __('The file failed to upload. Please check the following:', 'updraftplus')."\n\n - ".__('Any settings in your .htaccess or web.config file that affects the maximum upload or post size.', 'updraftplus')."\n - ".__('The available memory on the server.', 'updraftplus')."\n - ".__('That you are attempting to upload a zip file previously created by UpdraftPlus.', 'updraftplus')."\n\n".__('Further information may be found in the browser JavaScript console, and the server PHP error logs.', 'updraftplus'),
            'zip_file_contents' => __('Browsing zip file', 'updraftplus'),
            'zip_file_contents_info' => __('Select a file to view information about it', 'updraftplus'),
            'search' => __('Search', 'updraftplus'),
            'download_timeout' => __('Unable to download file. This could be caused by a timeout. It would be best to download the zip to your computer.', 'updraftplus'),
            'loading_log_file' => __('Loading log file', 'updraftplus'),
            'updraftplus_version' => $updraftplus->version,
            'updraftcentral_wizard_empty_url' => __('Please enter the URL where your UpdraftCentral dashboard is hosted.'),
            'updraftcentral_wizard_invalid_url' => __('Please enter a valid URL e.g http://example.com', 'updraftplus')
        ));
    }
    
    /**
     * Despite the name, this fires irrespective of what capabilities the user has (even none - so be careful)
     */
    public function core_upgrade_preamble() {
        // They need to be able to perform backups, and to perform updates
        if (!UpdraftPlus_Options::user_can_manage() || (!current_user_can('update_core') && !current_user_can('update_plugins') && !current_user_can('update_themes'))) return;
 
        if (!class_exists('UpdraftPlus_Addon_Autobackup')) {
            if (defined('UPDRAFTPLUS_NOADS_B')) return;
        }
        
        ?>
        <?php
            if (!class_exists('UpdraftPlus_Addon_Autobackup')) {
                if (!class_exists('UpdraftPlus_Notices')) include_once(UPDRAFTPLUS_DIR.'/includes/updraftplus-notices.php');
                global $updraftplus_notices;
                echo apply_filters('updraftplus_autobackup_blurb', $updraftplus_notices->do_notice('autobackup', 'autobackup', true));
            } else {
                echo '<div class="updraft-ad-container updated">';
                echo '<h3 style="margin-top: 2px;">'. __('Be safe with an automatic backup', 'updraftplus').'</h3>';
                echo apply_filters('updraftplus_autobackup_blurb', '');
                echo '</div>';
            }
        ?>
        <script>
        jQuery(document).ready(function() {
            jQuery('.updraft-ad-container').appendTo('.wrap p:first');
        });
        </script>
        <?php
    }
 
    /**
     * Run upon the WP admin_head action
     */
    public function admin_head() {
 
        global $pagenow;
 
        if (UpdraftPlus_Options::admin_page() != $pagenow || !isset($_REQUEST['page']) || 'updraftplus' != $_REQUEST['page'] || !UpdraftPlus_Options::user_can_manage()) return;
 
        $chunk_size = min(wp_max_upload_size()-1024, 1048576*2);
 
        // The multiple_queues argument is ignored in plupload 2.x (WP3.9+) - http://make.wordpress.org/core/2014/04/11/plupload-2-x-in-wordpress-3-9/
        // max_file_size is also in filters as of plupload 2.x, but in its default position is still supported for backwards-compatibility. Likewise, our use of filters.extensions below is supported by a backwards-compatibility option (the current way is filters.mime-types.extensions
 
        $plupload_init = array(
            'runtimes' => 'html5,flash,silverlight,html4',
            'browse_button' => 'plupload-browse-button',
            'container' => 'plupload-upload-ui',
            'drop_element' => 'drag-drop-area',
            'file_data_name' => 'async-upload',
            'multiple_queues' => true,
            'max_file_size' => '100Gb',
            'chunk_size' => $chunk_size.'b',
            'url' => admin_url('admin-ajax.php', 'relative'),
            'multipart' => true,
            'multi_selection' => true,
            'urlstream_upload' => true,
            // additional post data to send to our ajax hook
            'multipart_params' => array(
                '_ajax_nonce' => wp_create_nonce('updraft-uploader'),
                'action' => 'plupload_action'
            )
        );
 
        // WP 3.9 updated to plupload 2.0 - https://core.trac.wordpress.org/ticket/25663
        if (is_file(ABSPATH.WPINC.'/js/plupload/Moxie.swf')) {
            $plupload_init['flash_swf_url'] = includes_url('js/plupload/Moxie.swf');
        } else {
            $plupload_init['flash_swf_url'] = includes_url('js/plupload/plupload.flash.swf');
        }
 
        if (is_file(ABSPATH.WPINC.'/js/plupload/Moxie.xap')) {
            $plupload_init['silverlight_xap_url'] = includes_url('js/plupload/Moxie.xap');
        } else {
            $plupload_init['silverlight_xap_url'] = includes_url('js/plupload/plupload.silverlight.swf');
        }
 
        ?><script>
            var updraft_credentialtest_nonce = '<?php echo wp_create_nonce('updraftplus-credentialtest-nonce');?>';
            var updraftplus_settings_nonce = '<?php echo wp_create_nonce('updraftplus-settings-nonce');?>';
            var updraft_siteurl = '<?php echo esc_js(site_url('', 'relative'));?>';
            var updraft_plupload_config = <?php echo json_encode($plupload_init); ?>;
            var updraft_download_nonce = '<?php echo wp_create_nonce('updraftplus_download');?>';
            var updraft_accept_archivename = <?php echo apply_filters('updraftplus_accept_archivename_js', "[]");?>;
            <?php
            $plupload_init['browse_button'] = 'plupload-browse-button2';
            $plupload_init['container'] = 'plupload-upload-ui2';
            $plupload_init['drop_element'] = 'drag-drop-area2';
            $plupload_init['multipart_params']['action'] = 'plupload_action2';
            $plupload_init['filters'] = array(array('title' => __('Allowed Files'), 'extensions' => 'crypt'));
            ?>
            var updraft_plupload_config2 = <?php echo json_encode($plupload_init); ?>;
            var updraft_downloader_nonce = '<?php wp_create_nonce("updraftplus_download"); ?>'
            <?php
                $overdue = $this->howmany_overdue_crons();
                if ($overdue >= 4) {
                    ?>
                    jQuery(document).ready(function() {
                        setTimeout(function(){ updraft_check_overduecrons(); }, 11000);
                    });
                <?php } ?>
        </script>
        <?php
    }
 
    private function disk_space_check($space) {
        global $updraftplus;
        $updraft_dir = $updraftplus->backups_dir_location();
        $disk_free_space = @disk_free_space($updraft_dir);
        if (false == $disk_free_space) return -1;
        return ($disk_free_space > $space) ? true : false;
    }
 
    /**
     * Adds the settings link under the plugin on the plugin screen.
     *
     * @param  Array  $links Set of links for the plugin, before being filtered
     * @param  String $file  File name (relative to the plugin directory)
     * @return Array filtered results
     */
    public function plugin_action_links($links, $file) {
        if (is_array($links) && 'updraftplus/updraftplus.php' == $file) {
            $settings_link = '<a href="'.UpdraftPlus_Options::admin_page_url().'?page=updraftplus">'.__("Settings", "updraftplus").'</a>';
            array_unshift($links, $settings_link);
            $settings_link = '<a href="'.apply_filters('updraftplus_com_link', "https://updraftplus.com/").'">'.__("Add-Ons / Pro Support", "updraftplus").'</a>';
            array_unshift($links, $settings_link);
        }
        return $links;
    }
 
    public function admin_action_upgrade_pluginortheme() {
        if (isset($_GET['action']) && ('upgrade-plugin' == $_GET['action'] || 'upgrade-theme' == $_GET['action']) && !class_exists('UpdraftPlus_Addon_Autobackup') && !defined('UPDRAFTPLUS_NOADS_B')) {
 
            if ('upgrade-plugin' == $_GET['action']) {
                if (!current_user_can('update_plugins')) return;
            } else {
                if (!current_user_can('update_themes')) return;
            }
 
            $dismissed_until = UpdraftPlus_Options::get_updraft_option('updraftplus_dismissedautobackup', 0);
            if ($dismissed_until > time()) return;
 
            if ('upgrade-plugin' == $_GET['action']) {
                $title = __('Update Plugin');
                $parent_file = 'plugins.php';
                $submenu_file = 'plugins.php';
            } else {
                $title = __('Update Theme');
                $parent_file = 'themes.php';
                $submenu_file = 'themes.php';
            }
 
            include_once(ABSPATH.'wp-admin/admin-header.php');
            
            if (!class_exists('UpdraftPlus_Notices')) include_once(UPDRAFTPLUS_DIR.'/includes/updraftplus-notices.php');
            global $updraftplus_notices;
            $updraftplus_notices->do_notice('autobackup', 'autobackup');
        }
    }
 
    public function show_admin_warning($message, $class = 'updated') {
        echo '<div class="updraftmessage '.$class.'">'."<p>$message</p></div>";
    }
 
    public function show_admin_warning_multiple_storage_options() {
        $this->show_admin_warning('<strong>UpdraftPlus:</strong> '.__('An error occurred when fetching storage module options: ', 'updraftplus').htmlspecialchars($this->storage_module_option_errors), 'error');
    }
 
    public function show_admin_warning_unwritable() {
        $unwritable_mess = htmlspecialchars(__("The 'Backup Now' button is disabled as your backup directory is not writable (go to the 'Settings' tab and find the relevant option).", 'updraftplus'));
        $this->show_admin_warning($unwritable_mess, "error");
    }
    
    public function show_admin_nosettings_warning() {
        $this->show_admin_warning('<strong>'.__('Welcome to UpdraftPlus!', 'updraftplus').'</strong> '.__('To make a backup, just press the Backup Now button.', 'updraftplus').' <a href="#" id="updraft-navtab-settings2">'.__('To change any of the default settings of what is backed up, to configure scheduled backups, to send your backups to remote storage (recommended), and more, go to the settings tab.', 'updraftplus').'</a>', 'updated notice is-dismissible');
    }
 
    public function show_admin_warning_execution_time() {
        $this->show_admin_warning('<strong>'.__('Warning', 'updraftplus').':</strong> '.sprintf(__('The amount of time allowed for WordPress plugins to run is very low (%s seconds) - you should increase it to avoid backup failures due to time-outs (consult your web hosting company for more help - it is the max_execution_time PHP setting; the recommended value is %s seconds or more)', 'updraftplus'), (int) @ini_get('max_execution_time'), 90));
    }
 
    public function show_admin_warning_disabledcron() {
        $this->show_admin_warning('<strong>'.__('Warning', 'updraftplus').':</strong> '.__('The scheduler is disabled in your WordPress install, via the DISABLE_WP_CRON setting. No backups can run (even &quot;Backup Now&quot;) unless either you have set up a facility to call the scheduler manually, or until it is enabled.', 'updraftplus').' <a href="'.apply_filters('updraftplus_com_link', "https://updraftplus.com/faqs/my-scheduled-backups-and-pressing-backup-now-does-nothing-however-pressing-debug-backup-does-produce-a-backup/#disablewpcron/").'">'.__('Go here for more information.', 'updraftplus').'</a>', 'updated updraftplus-disable-wp-cron-warning');
    }
 
    public function show_admin_warning_diskspace() {
        $this->show_admin_warning('<strong>'.__('Warning', 'updraftplus').':</strong> '.sprintf(__('You have less than %s of free disk space on the disk which UpdraftPlus is configured to use to create backups. UpdraftPlus could well run out of space. Contact your the operator of your server (e.g. your web hosting company) to resolve this issue.', 'updraftplus'), '35 MB'));
    }
 
    public function show_admin_warning_wordpressversion() {
        $this->show_admin_warning('<strong>'.__('Warning', 'updraftplus').':</strong> '.sprintf(__('UpdraftPlus does not officially support versions of WordPress before %s. It may work for you, but if it does not, then please be aware that no support is available until you upgrade WordPress.', 'updraftplus'), '3.2'));
    }
 
    public function show_admin_warning_litespeed() {
        $this->show_admin_warning('<strong>'.__('Warning', 'updraftplus').':</strong> '.sprintf(__('Your website is hosted using the %s web server.', 'updraftplus'), 'LiteSpeed').' <a href="'.apply_filters('updraftplus_com_link', "https://updraftplus.com/faqs/i-am-having-trouble-backing-up-and-my-web-hosting-company-uses-the-litespeed-webserver/").'">'.__('Please consult this FAQ if you have problems backing up.', 'updraftplus').'</a>');
    }
 
    public function show_admin_debug_warning() {
        $this->show_admin_warning('<strong>'.__('Notice', 'updraftplus').':</strong> '.__('UpdraftPlus\'s debug mode is on. You may see debugging notices on this page not just from UpdraftPlus, but from any other plugin installed. Please try to make sure that the notice you are seeing is from UpdraftPlus before you raise a support request.', 'updraftplus').'</a>');
    }
 
    public function show_admin_warning_overdue_crons($howmany) {
        $ret = '<div class="updraftmessage updated"><p>';
        $ret .= '<strong>'.__('Warning', 'updraftplus').':</strong> '.sprintf(__('WordPress has a number (%d) of scheduled tasks which are overdue. Unless this is a development site, this probably means that the scheduler in your WordPress install is not working.', 'updraftplus'), $howmany).' <a href="'.apply_filters('updraftplus_com_link', "https://updraftplus.com/faqs/scheduler-wordpress-installation-working/").'">'.__('Read this page for a guide to possible causes and how to fix it.', 'updraftplus').'</a>';
        $ret .= '</p></div>';
        return $ret;
    }
 
    /**
     * Output authorisation links for any un-authorised Dropbox settings instances
     */
    public function show_admin_warning_dropbox() {
        foreach ($this->auth_instance_ids['dropbox'] as $instance_id) {
            $this->show_admin_warning('<strong>'.__('UpdraftPlus notice:', 'updraftplus').'</strong> <a class="updraft_authlink" href="'.UpdraftPlus_Options::admin_page_url().'?page=updraftplus&action=updraftmethod-dropbox-auth&updraftplus_dropboxauth=doit&updraftplus_instance='.$instance_id.'">'.sprintf(__('Follow this link to authorize access to your %s account (you will not be able to back up to %s without it).', 'updraftplus'), 'Dropbox', 'Dropbox').'</a>', 'updated updraft_authenticate_dropbox');
        }
    }
 
    /**
     * Output authorisation links for any un-authorised OneDrive settings instances
     */
    public function show_admin_warning_onedrive() {
        foreach ($this->auth_instance_ids['onedrive'] as $instance_id) {
            $this->show_admin_warning('<strong>'.__('UpdraftPlus notice:', 'updraftplus').'</strong> <a class="updraft_authlink" href="'.UpdraftPlus_Options::admin_page_url().'?page=updraftplus&action=updraftmethod-onedrive-auth&updraftplus_onedriveauth=doit&updraftplus_instance='.$instance_id.'">'.sprintf(__('Follow this link to authorize access to your %s account (you will not be able to back up to %s without it).', 'updraftplus'), 'OneDrive', 'OneDrive').'</a>', 'updated updraft_authenticate_onedrive');
        }
    }
 
    public function show_admin_warning_updraftvault() {
        $this->show_admin_warning('<strong>'.__('UpdraftPlus notice:', 'updraftplus').'</strong> '.sprintf(__('%s has been chosen for remote storage, but you are not currently connected.', 'updraftplus'), 'UpdraftPlus Vault').' '.__('Go to the remote storage settings in order to connect.', 'updraftplus'), 'updated');
    }
 
    /**
     * Output authorisation links for any un-authorised Google Drive settings instances
     */
    public function show_admin_warning_googledrive() {
        foreach ($this->auth_instance_ids['googledrive'] as $instance_id) {
            $this->show_admin_warning('<strong>'.__('UpdraftPlus notice:', 'updraftplus').'</strong> <a class="updraft_authlink" href="'.UpdraftPlus_Options::admin_page_url().'?page=updraftplus&action=updraftmethod-googledrive-auth&updraftplus_googleauth=doit&updraftplus_instance='.$instance_id.'">'.sprintf(__('Follow this link to authorize access to your %s account (you will not be able to back up to %s without it).', 'updraftplus'), 'Google Drive', 'Google Drive').'</a>', 'updated updraft_authenticate_googledrive');
        }
    }
 
    /**
     * Output authorisation links for any un-authorised Google Cloud settings instances
     */
    public function show_admin_warning_googlecloud() {
        foreach ($this->auth_instance_ids['googlecloud'] as $instance_id) {
            $this->show_admin_warning('<strong>'.__('UpdraftPlus notice:', 'updraftplus').'</strong> <a class="updraft_authlink" href="'.UpdraftPlus_Options::admin_page_url().'?page=updraftplus&action=updraftmethod-googlecloud-auth&updraftplus_googleauth=doit&updraftplus_instance='.$instance_id.'">'.sprintf(__('Follow this link to authorize access to your %s account (you will not be able to back up to %s without it).', 'updraftplus'), 'Google Cloud', 'Google Cloud').'</a>', 'updated updraft_authenticate_googlecloud');
        }
    }
 
    /**
     * This options filter removes ABSPATH off the front of updraft_dir, if it is given absolutely and contained within it
     *
     * @param  string $updraft_dir Directory
     * @return string
     */
    public function prune_updraft_dir_prefix($updraft_dir) {
        if ('/' == substr($updraft_dir, 0, 1) || "\\" == substr($updraft_dir, 0, 1) || preg_match('/^[a-zA-Z]:/', $updraft_dir)) {
            $wcd = trailingslashit(WP_CONTENT_DIR);
            if (strpos($updraft_dir, $wcd) === 0) {
                $updraft_dir = substr($updraft_dir, strlen($wcd));
            }
        }
        return $updraft_dir;
    }
 
    /**
     * Start a download of a backup. This method is called via the AJAX action updraft_download_backup. May die instead of returning depending upon the mode in which it is called.
     */
    public function updraft_download_backup() {
 
        if (empty($_REQUEST['_wpnonce']) || !wp_verify_nonce($_REQUEST['_wpnonce'], 'updraftplus_download')) die;
 
        if (empty($_REQUEST['timestamp']) || !is_numeric($_REQUEST['timestamp']) || empty($_REQUEST['type'])) exit;
 
        $findex = empty($_REQUEST['findex']) ? 0 : (int) $_REQUEST['findex'];
        $stage = empty($_REQUEST['stage']) ? '' : $_REQUEST['stage'];
        $file_path = empty($_REQUEST['filepath']) ? '' : $_REQUEST['filepath'];
 
        // This call may not actually return, depending upon what mode it is called in
        $result = $this->do_updraft_download_backup($findex, $_REQUEST['type'], $_REQUEST['timestamp'], $stage, false, $file_path);
        
        // In theory, if a response was already sent, then Connection: close has been issued, and a Content-Length. However, in https://updraftplus.com/forums/topic/pclzip_err_bad_format-10-invalid-archive-structure/ a browser ignores both of these, and then picks up the second output and complains.
        if (empty($result['already_closed'])) echo json_encode($result);
        
        die();
    }
    
    /**
     * Ensure that a specified backup is present, downloading if necessary (or delete it, if the parameters so indicate). N.B. This function may die(), depending on the request being made in $stage
     *
     * @param Integer           $findex                      - the index number of the backup archive requested
     * @param String           $type                      - the entity type (e.g. 'plugins') being requested
     * @param Integer           $timestamp                  - identifier for the backup being requested (UNIX epoch time)
     * @param Mixed               $stage                      - the stage; valid values include (have not audited for other possibilities) at least 'delete' and 2.
     * @param Callable|Boolean $close_connection_callable - function used to close the connection to the caller; an array of data to return is passed. If false, then UpdraftPlus::close_browser_connection is called with a JSON version of the data.
     * @param String           $file_path                  - an over-ride for where to download the file to (basename only)
     *
     * @return Array - sumary of the results. May also just die.
     */
    public function do_updraft_download_backup($findex, $type, $timestamp, $stage, $close_connection_callable = false, $file_path = '') {
    
        @set_time_limit(UPDRAFTPLUS_SET_TIME_LIMIT);
 
        global $updraftplus;
        
        // This is a bit ugly; these variables get placed back into $_POST (where they may possibly have come from), so that UpdraftPlus::log() can detect exactly where to log the download status.
        $_POST['findex'] = $findex;
        $_POST['type'] = $type;
        $_POST['timestamp'] = $timestamp;
 
        // Check that it is a known entity type; if not, die
        if ('db' != substr($type, 0, 2)) {
            $backupable_entities = $updraftplus->get_backupable_file_entities(true);
            foreach ($backupable_entities as $t => $info) {
                if ($type == $t) $type_match = true;
            }
            if (empty($type_match)) return array('result' => 'error', 'code' => 'no_such_type');
        }
 
        // We already know that no possible entities have an MD5 clash (even after 2 characters)
        // Also, there's nothing enforcing a requirement that nonces are hexadecimal
        $job_nonce = dechex($timestamp).$findex.substr(md5($type), 0, 3);
 
        // You need a nonce before you can set job data. And we certainly don't yet have one.
        $updraftplus->backup_time_nonce($job_nonce);
 
        $debug_mode = UpdraftPlus_Options::get_updraft_option('updraft_debug_mode');
 
        // Set the job type before logging, as there can be different logging destinations
        $updraftplus->jobdata_set('job_type', 'download');
        $updraftplus->jobdata_set('job_time_ms', $updraftplus->job_time_ms);
 
        // Retrieve the information from our backup history
        $backup_history = UpdraftPlus_Backup_History::get_history();
        // Base name
        $file = $backup_history[$timestamp][$type];
 
        // Deal with multi-archive sets
        if (is_array($file)) $file = $file[$findex];
 
        if (false !== strpos($file_path, '..')) {
            error_log("UpdraftPlus_Admin::do_updraft_download_backup : invalid file_path: $file_path");
            return array('result' => __('Error: invalid path', 'updraftplus'));
        }
 
        if (!empty($file_path)) $file = $file_path;
 
        // Where it should end up being downloaded to
        $fullpath = $updraftplus->backups_dir_location().'/'.$file;
 
        if (!empty($file_path) && strpos(realpath($fullpath), realpath($updraftplus->backups_dir_location())) === false) {
            error_log("UpdraftPlus_Admin::do_updraft_download_backup : invalid fullpath: $fullpath");
            return array('result' => __('Error: invalid path', 'updraftplus'));
        }
 
        if (2 == $stage) {
            $updraftplus->spool_file($fullpath);
            // We only want to remove if it was a temp file from the zip browser
            if (!empty($file_path)) @unlink($fullpath);
            // Do not return - we do not want the caller to add any output
            die;
        }
 
        if ('delete' == $stage) {
            @unlink($fullpath);
            $updraftplus->log("The file has been deleted ($file)");
            return array('result' => 'deleted');
        }
 
        // TODO: FIXME: Failed downloads may leave log files forever (though they are small)
        if ($debug_mode) $updraftplus->logfile_open($updraftplus->nonce);
 
        set_error_handler(array($updraftplus, 'php_error'), E_ALL & ~E_STRICT);
 
        $updraftplus->log("Requested to obtain file: timestamp=$timestamp, type=$type, index=$findex");
 
        $itext = empty($findex) ? '' : $findex;
        $known_size = isset($backup_history[$timestamp][$type.$itext.'-size']) ? $backup_history[$timestamp][$type.$itext.'-size'] : 0;
 
        $services = isset($backup_history[$timestamp]['service']) ? $backup_history[$timestamp]['service'] : false;
        if (is_string($services)) $services = array($services);
 
        $updraftplus->jobdata_set('service', $services);
 
        // Fetch it from the cloud, if we have not already got it
 
        $needs_downloading = false;
 
        if (!file_exists($fullpath)) {
            // If the file doesn't exist and they're using one of the cloud options, fetch it down from the cloud.
            $needs_downloading = true;
            $updraftplus->log('File does not yet exist locally - needs downloading');
        } elseif ($known_size > 0 && filesize($fullpath) < $known_size) {
            $updraftplus->log("The file was found locally (".filesize($fullpath).") but did not match the size in the backup history ($known_size) - will resume downloading");
            $needs_downloading = true;
        } elseif ($known_size > 0) {
            $updraftplus->log('The file was found locally and matched the recorded size from the backup history ('.round($known_size/1024, 1).' KB)');
        } else {
            $updraftplus->log('No file size was found recorded in the backup history. We will assume the local one is complete.');
            $known_size = filesize($fullpath);
        }
 
        // The AJAX responder that updates on progress wants to see this
        $updraftplus->jobdata_set('dlfile_'.$timestamp.'_'.$type.'_'.$findex, "downloading:$known_size:$fullpath");
 
        if ($needs_downloading) {
 
            // Update the "last modified" time to dissuade any other instances from thinking that no downloaders are active
            @touch($fullpath);
 
            $msg = array(
                'result' => 'needs_download',
                'request' => array(
                        'type' => $type,
                        'timestamp' => $timestamp,
                        'findex' => $findex
                )
            );
        
            if ($close_connection_callable && is_callable($close_connection_callable)) {
                call_user_func($close_connection_callable, $msg);
            } else {
                $updraftplus->close_browser_connection(json_encode($msg));
            }
            $this->get_remote_file($services, $file, $timestamp);
        }
 
        // Now, be ready to spool the thing to the browser
        if (is_file($fullpath) && is_readable($fullpath)) {
 
            // That message is then picked up by the AJAX listener
            $updraftplus->jobdata_set('dlfile_'.$timestamp.'_'.$type.'_'.$findex, 'downloaded:'.filesize($fullpath).":$fullpath");
 
            $result = 'downloaded';
            
        } else {
 
            $updraftplus->jobdata_set('dlfile_'.$timestamp.'_'.$type.'_'.$findex, 'failed');
            $updraftplus->jobdata_set('dlerrors_'.$timestamp.'_'.$type.'_'.$findex, $updraftplus->errors);
            $updraftplus->log('Remote fetch failed. File '.$fullpath.' did not exist or was unreadable. If you delete local backups then remote retrieval may have failed.');
            
            $result = 'download_failed';
        }
 
        restore_error_handler();
 
        @fclose($updraftplus->logfile_handle);
        if (!$debug_mode) @unlink($updraftplus->logfile_name);
 
        // The browser connection was possibly already closed, but not necessarily
        return array('result' => $result, 'already_closed' => $needs_downloading);
 
    }
 
    /**
     * This method gets the remote storage information and objects and loops over each of them until we get a successful download of the passed in file.
     *
     * @param  Array   $services  - a list of connected service identifiers (e.g. 'dropbox', 's3', etc.)
     * @param  String  $file      - the name of the file
     * @param  Integer $timestamp - the backup timestamp
     * @param  Boolean $restore   - a boolean to indicate if the caller of this method is a restore or not; if so, different messages are logged
     */
    private function get_remote_file($services, $file, $timestamp, $restore = false) {
        global $updraftplus;
        
        $fullpath = $updraftplus->backups_dir_location().'/'.$file;
 
        $storage_objects_and_ids = $updraftplus->get_storage_objects_and_ids($services);
 
        $is_downloaded = false;
 
        $updraftplus->register_wp_http_option_hooks();
 
        foreach ($services as $service) {
 
            if (empty($service) || 'none' == $service) continue;
        
            if ($restore) {
                $service_description = empty($updraftplus->backup_methods[$service]) ? $service : $updraftplus->backup_methods[$service];
                $updraftplus->log(__("File is not locally present - needs retrieving from remote storage", 'updraftplus')." ($service_description)", 'notice-restore');
            }
 
            $object = $storage_objects_and_ids[$service]['object'];
 
            if (!$object->supports_feature('multi_options')) {
                error_log("UpdraftPlus_Admin::get_remote_file(): Multi-options not supported by: ".$service);
                continue;
            }
            
            $instance_ids = $storage_objects_and_ids[$service]['instance_settings'];
            $backups_instance_ids = isset($backup_history[$timestamp]['service_instance_ids'][$service]) ? $backup_history[$timestamp]['service_instance_ids'][$service] : array(false);
 
            foreach ($backups_instance_ids as $instance_id) {
 
                if (isset($instance_ids[$instance_id])) {
                    $options = $instance_ids[$instance_id];
                } else {
                    // If we didn't find a instance id match, it could be a new UpdraftPlus upgrade or a wipe settings with the same details entered so try the default options saved.
                    $options = $object->get_options();
                }
 
                $object->set_options($options, false, $instance_id);
 
                $download = $this->download_file($file, $object);
 
                if (is_readable($fullpath) && false !== $download) {
                    if ($restore) {
                        $updraftplus->log(__("OK", 'updraftplus'), 'notice-restore');
                    } else {
                        clearstatcache();
                        $updraftplus->log('Remote fetch was successful (file size: '.round(filesize($fullpath)/1024, 1).' KB)');
                    }
                    break 2;
                } else {
                    if ($restore) {
                        $updraftplus->log(__("Error", 'updraftplus'), 'notice-restore');
                    } else {
                        clearstatcache();
                        if (0 === @filesize($fullpath)) @unlink($fullpath);
                        $updraftplus->log('Remote fetch failed');
                    }
                }
            }
        }
        $updraftplus->register_wp_http_option_hooks(false);
    }
 
    /**
     * Downloads a specified file into UD's directory
     *
     * @param String $file   The name of the file
     * @param array  $object The object of the service to use to download with. UpdraftPlus_BackupModule.
     * @return Boolean - Whether the operation succeeded. Inherited from the storage module's download() method. N.B. At the time of writing it looks like not all modules necessarily return true upon success; but false can be relied upon for detecting failure.
     */
    private function download_file($file, $object) {
 
        global $updraftplus;
 
        @set_time_limit(UPDRAFTPLUS_SET_TIME_LIMIT);
 
        $service = $object->get_id();
        
        $updraftplus->log("Requested file from remote service: $service: $file");
 
        if (method_exists($object, 'download')) {
        
            try {
                return $object->download($file);
            } catch (Exception $e) {
                $log_message = 'Exception ('.get_class($e).') occurred during download: '.$e->getMessage().' (Code: '.$e->getCode().', line '.$e->getLine().' in '.$e->getFile().')';
                error_log($log_message);
                // @codingStandardsIgnoreLine
                if (function_exists('wp_debug_backtrace_summary')) $log_message .= ' Backtrace: '.wp_debug_backtrace_summary();
                $updraftplus->log($log_message);
                $updraftplus->log(sprintf(__('A PHP exception (%s) has occurred: %s', 'updraftplus'), get_class($e), $e->getMessage()), 'error');
                return false;
            // @codingStandardsIgnoreLine
            } catch (Error $e) {
                $log_message = 'PHP Fatal error ('.get_class($e).') has occurred during download. Error Message: '.$e->getMessage().' (Code: '.$e->getCode().', line '.$e->getLine().' in '.$e->getFile().')';
                error_log($log_message);
                // @codingStandardsIgnoreLine
                if (function_exists('wp_debug_backtrace_summary')) $log_message .= ' Backtrace: '.wp_debug_backtrace_summary();
                $updraftplus->log($log_message);
                $updraftplus->log(sprintf(__('A PHP fatal error (%s) has occurred: %s', 'updraftplus'), get_class($e), $e->getMessage()), 'error');
                return false;
            }
        } else {
            $updraftplus->log("Automatic backup restoration is not available with the method: $service.");
            $updraftplus->log("$file: ".sprintf(__("The backup archive for this file could not be found. The remote storage method in use (%s) does not allow us to retrieve files. To perform any restoration using UpdraftPlus, you will need to obtain a copy of this file and place it inside UpdraftPlus's working folder", 'updraftplus'), $service)." (".$this->prune_updraft_dir_prefix($updraftplus->backups_dir_location()).")", 'error');
            return false;
        }
 
    }
 
    /**
     * This is used as a callback
     *
     * @param  Mixed $msg The data to be JSON encoded and sent back
     */
    public function _updraftplus_background_operation_started($msg) {
        global $updraftplus;
        // The extra spaces are because of a bug seen on one server in handling of non-ASCII characters; see HS#11739
        $updraftplus->close_browser_connection(json_encode($msg).'        ');
    }
    
    public function updraft_ajax_handler() {
 
        global $updraftplus;
 
        $nonce = empty($_REQUEST['nonce']) ? '' : $_REQUEST['nonce'];
 
        if (!wp_verify_nonce($nonce, 'updraftplus-credentialtest-nonce') || empty($_REQUEST['subaction'])) die('Security check');
 
        // Mitigation in case the nonce leaked to an unauthorised user
        if ('dismissautobackup' == $_REQUEST['subaction']) {
            if (!current_user_can('update_plugins') && !current_user_can('update_themes')) return;
        } elseif ('dismissexpiry' == $_REQUEST['subaction'] || 'dismissdashnotice' == $_REQUEST['subaction']) {
            if (!current_user_can('update_plugins')) return;
        } else {
            if (!UpdraftPlus_Options::user_can_manage()) return;
        }
        
        $subaction = $_REQUEST['subaction'];
        
        // All others use _POST
        $data_in_get = array('get_log', 'get_fragment');
        
        // UpdraftPlus_WPAdmin_Commands extends UpdraftPlus_Commands - i.e. all commands are in there
        if (!class_exists('UpdraftPlus_WPAdmin_Commands')) include_once(UPDRAFTPLUS_DIR.'/includes/class-wpadmin-commands.php');
        $commands = new UpdraftPlus_WPAdmin_Commands($this);
        
        if (method_exists($commands, $subaction)) {
 
            $data = in_array($subaction, $data_in_get) ? $_GET : $_POST;
            // TODO: Once all commands come through here and through updraft_send_command(), the data should always come from this attribute (once updraft_send_command() is modified appropriately).
            if (isset($data['action_data'])) $data = $data['action_data'];
            $results = call_user_func(array($commands, $subaction), $data);
            
            if (is_wp_error($results)) {
                $results = array(
                    'result' => false,
                    'error_code' => $results->get_error_code(),
                    'error_message' => $results->get_error_message(),
                    'error_data' => $results->get_error_data(),
                );
            }
            
            if (is_string($results)) {
                // A handful of legacy methods, and some which are directly the source for iframes, for which JSON is not appropriate.
                echo $results;
            } else {
                echo json_encode($results);
            }
            die;
        
        }
        
        // Below are all the commands not ported over into class-commands.php or class-wpadmin-commands.php
 
        if ('activejobs_list' == $subaction) {
        
            // N.B. Also called from autobackup.php
            // TODO: This should go into UpdraftPlus_Commands, once the add-ons have been ported to use updraft_send_command()
            echo json_encode($this->get_activejobs_list($_GET));
 
        } elseif ('httpget' == $_REQUEST['subaction']) {
        
            // httpget
            $curl = empty($_REQUEST['curl']) ? false : true;
             echo $this->http_get($_REQUEST['uri'], $curl);
              
        } elseif ('doaction' == $_REQUEST['subaction'] && !empty($_REQUEST['subsubaction']) && 'updraft_' == substr($_REQUEST['subsubaction'], 0, 8)) {
        
            // These generally echo and die - they will need further work to port to one of the command classes. Some may already have equivalents in UpdraftPlus_Commands, if they are used from UpdraftCentral.
            do_action($_REQUEST['subsubaction']);
        } else {
            // These can be removed after a couple of releases
            include(UPDRAFTPLUS_DIR.'/includes/deprecated-actions.php');
        }
        
        die;
 
    }
    
    /**
     * Run a credentials test for the indicated remote storage module
     *
     * @param Array   $test_settings          The test parameters, including the method itself indicated in the key 'method'
     * @param Boolean $return_instead_of_echo Whether to return or echo the results. N.B. More than just the results to echo will be returned
     * @return Array|Void - the results, if they are being returned (rather than echoed). Keys: 'output' (the output), 'data' (other data)
     */
    public function do_credentials_test($test_settings, $return_instead_of_echo = false) {
    
        $method = (!empty($test_settings['method']) && preg_match("/^[a-z0-9]+$/", $test_settings['method'])) ? $test_settings['method'] : "";
        
        $objname = "UpdraftPlus_BackupModule_$method";
        
        $this->logged = array();
        // TODO: Add action for WP HTTP SSL stuff
        set_error_handler(array($this, 'get_php_errors'), E_ALL & ~E_STRICT);
        
        if (!class_exists($objname)) include_once(UPDRAFTPLUS_DIR."/methods/$method.php");
 
        $ret = '';
        $data = null;
        
        // TODO: Add action for WP HTTP SSL stuff
        if (method_exists($objname, "credentials_test")) {
            $obj = new $objname;
            if ($return_instead_of_echo) ob_start();
            $data = $obj->credentials_test($test_settings);
            if ($return_instead_of_echo) $ret .= ob_get_clean();
        }
        
        if (count($this->logged) >0) {
            $ret .= "\n\n".__('Messages:', 'updraftplus')."\n";
            foreach ($this->logged as $err) {
                $ret .= "* $err\n";
            }
            if (!$return_instead_of_echo) echo $ret;
        }
        restore_error_handler();
        
        if ($return_instead_of_echo) return array('output' => $ret, 'data' => $data);
        
    }
    
    /**
     * Delete a backup set, whilst respecting limits on how much to delete in one go
     *
     * @uses remove_backup_set_cleanup()
     * @param Array $opts - deletion options; with keys backup_timestamp, delete_remote, [remote_delete_limit]
     * @return Array - as from remove_backup_set_cleanup()
     */
    public function delete_set($opts) {
        
        global $updraftplus;
        
        $backups = UpdraftPlus_Backup_History::get_history();
        $timestamps = (string) $opts['backup_timestamp'];
 
        $remote_delete_limit = (isset($opts['remote_delete_limit']) && $opts['remote_delete_limit'] > 0) ? (int) $opts['remote_delete_limit'] : PHP_INT_MAX;
        
        $timestamps = explode(',', $timestamps);
        $delete_remote = empty($opts['delete_remote']) ? false : true;
 
        // You need a nonce before you can set job data. And we certainly don't yet have one.
        $updraftplus->backup_time_nonce();
        // Set the job type before logging, as there can be different logging destinations
        $updraftplus->jobdata_set('job_type', 'delete');
        $updraftplus->jobdata_set('job_time_ms', $updraftplus->job_time_ms);
 
        if (UpdraftPlus_Options::get_updraft_option('updraft_debug_mode')) {
            $updraftplus->logfile_open($updraftplus->nonce);
            set_error_handler(array($updraftplus, 'php_error'), E_ALL & ~E_STRICT);
        }
 
        $updraft_dir = $updraftplus->backups_dir_location();
        $backupable_entities = $updraftplus->get_backupable_file_entities(true, true);
 
        $local_deleted = 0;
        $remote_deleted = 0;
        $sets_removed = 0;
 
        foreach ($timestamps as $i => $timestamp) {
 
            if (!isset($backups[$timestamp])) {
                return array('result' => 'error', 'message' => __('Backup set not found', 'updraftplus'));
            }
 
            $nonce = isset($backups[$timestamp]['nonce']) ? $backups[$timestamp]['nonce'] : '';
 
            $delete_from_service = array();
 
            if ($delete_remote) {
                // Locate backup set
                if (isset($backups[$timestamp]['service'])) {
                    // Convert to an array so that there is no uncertainty about how to process it
                    $services = is_string($backups[$timestamp]['service']) ? array($backups[$timestamp]['service']) : $backups[$timestamp]['service'];
                    if (is_array($services)) {
                        foreach ($services as $service) {
                            if ($service && 'none' != $service && 'email' != $service) $delete_from_service[] = $service;
                        }
                    }
                }
            }
 
            $files_to_delete = array();
            foreach ($backupable_entities as $key => $ent) {
                if (isset($backups[$timestamp][$key])) {
                    $files_to_delete[$key] = $backups[$timestamp][$key];
                }
            }
            // Delete DB
            foreach ($backups[$timestamp] as $key => $value) {
                if ('db' == strtolower(substr($key, 0, 2)) && '-size' != substr($key, -5, 5)) {
                    $files_to_delete[$key] = $backups[$timestamp][$key];
                }
            }
 
            // Also delete the log
            if ($nonce && !UpdraftPlus_Options::get_updraft_option('updraft_debug_mode')) {
                $files_to_delete['log'] = "log.$nonce.txt";
            }
            
            $updraftplus->register_wp_http_option_hooks();
 
            foreach ($files_to_delete as $key => $files) {
 
                if (is_string($files)) {
                    $was_string = true;
                    $files = array($files);
                } else {
                    $was_string = false;
                }
 
                foreach ($files as $file) {
                    if (is_file($updraft_dir.'/'.$file) && @unlink($updraft_dir.'/'.$file)) $local_deleted++;
                }
 
                if ('log' != $key && count($delete_from_service) > 0) {
 
                    $storage_objects_and_ids = $updraftplus->get_storage_objects_and_ids($delete_from_service);
 
                    foreach ($delete_from_service as $service) {
                    
                        if ('email' == $service || 'none' == $service || !$service) continue;
 
                        $deleted = -1;
 
                        $remote_obj = $storage_objects_and_ids[$service]['object'];
 
                        $instance_settings = $storage_objects_and_ids[$service]['instance_settings'];
                        $this->backups_instance_ids = empty($backups[$timestamp]['service_instance_ids'][$service]) ? array() : $backups[$timestamp]['service_instance_ids'][$service];
 
                        uksort($instance_settings, array($this, 'instance_ids_sort'));
 
                        foreach ($instance_settings as $instance_id => $options) {
 
                            $remote_obj->set_options($options, false, $instance_id);
 
                            foreach ($files as $index => $file) {
                                if ($remote_deleted == $remote_delete_limit) {
                                    return $this->remove_backup_set_cleanup(false, $backups, $local_deleted, $remote_deleted, $sets_removed);
                                }
 
                                $deleted = $remote_obj->delete($file);
                                
                                if (-1 === $deleted) {
                                    // echo __('Did not know how to delete from this cloud service.', 'updraftplus');
                                } elseif (false !== $deleted) {
                                    $remote_deleted++;
                                }
                                
                                $itext = $index ? (string) $index : '';
                                if ($was_string) {
                                    unset($backups[$timestamp][$key]);
                                    if ('db' == strtolower(substr($key, 0, 2))) unset($backups[$timestamp][$key][$index.'-size']);
                                } else {
                                    unset($backups[$timestamp][$key][$index]);
                                    unset($backups[$timestamp][$key.$itext.'-size']);
                                    if (empty($backups[$timestamp][$key])) unset($backups[$timestamp][$key]);
                                }
                                if (isset($backups[$timestamp]['checksums']) && is_array($backups[$timestamp]['checksums'])) {
                                    foreach (array_keys($backups[$timestamp]['checksums']) as $algo) {
                                        unset($backups[$timestamp]['checksums'][$algo][$key.$index]);
                                    }
                                }
                                
                                // If we don't save the array back, then the above section will fire again for the same files - and the remote storage will be requested to delete already-deleted files, which then means no time is actually saved by the browser-backend loop method.
                                UpdraftPlus_Backup_History::save_history($backups);
                            }
                        }
                    }
                }
            }
 
            unset($backups[$timestamp]);
            UpdraftPlus_Backup_History::save_history($backups);
            $sets_removed++;
        }
 
        return $this->remove_backup_set_cleanup(true, $backups, $local_deleted, $remote_deleted, $sets_removed);
 
    }
 
    /**
     * This function sorts the array of instance ids currently saved so that any instance id that is in both the saved settings and the backup history move to the top of the array, as these are likely to work. Then values that don't appear in the backup history move to the bottom.
     *
     * @param  String $a - the first instance id
     * @param  String $b - the second instance id
     * @return Integer   - returns an integer to indicate what position the $b value should be moved in
     */
    public function instance_ids_sort($a, $b) {
        if (in_array($a, $this->backups_instance_ids)) {
            if (in_array($b, $this->backups_instance_ids)) return 0;
            return -1;
        }
        return in_array($b, $this->backups_instance_ids) ? 1 : 0;
    }
 
    /**
     * Called by self::delete_set() to finish up before returning (whether the complete deletion is finished or not)
     *
     * @param Boolean $delete_complete - whether the whole set is now gone (i.e. last round)
     * @param Array      $backups           - the backup history
     * @param Integer $local_deleted   - how many backup archives were deleted from local storage
     * @param Integer $remote_deleted  - how many backup archives were deleted from remote storage
     * @param Integer $sets_removed       - how many complete sets were removed
     *
     * @return Array - information on the status, suitable for returning to the UI
     */
    public function remove_backup_set_cleanup($delete_complete, $backups, $local_deleted, $remote_deleted, $sets_removed) {
 
        global $updraftplus;
 
        $updraftplus->register_wp_http_option_hooks(false);
 
        UpdraftPlus_Backup_History::save_history($backups);
 
        $updraftplus->log("Local files deleted: $local_deleted. Remote files deleted: $remote_deleted");
 
        if ($delete_complete) {
            $set_message = __('Backup sets removed:', 'updraftplus');
            $local_message = __('Local files deleted:', 'updraftplus');
            $remote_message = __('Remote files deleted:', 'updraftplus');
 
            if (UpdraftPlus_Options::get_updraft_option('updraft_debug_mode')) {
                restore_error_handler();
            }
            
            return array('result' => 'success', 'set_message' => $set_message, 'local_message' => $local_message, 'remote_message' => $remote_message, 'backup_sets' => $sets_removed, 'backup_local' => $local_deleted, 'backup_remote' => $remote_deleted);
        } else {
        
            return array('result' => 'continue', 'backup_local' => $local_deleted, 'backup_remote' => $remote_deleted, 'backup_sets' => $sets_removed);
        }
    }
 
    public function get_history_status($rescan, $remotescan) {
    
        global $updraftplus;
    
        if ($rescan) $messages = UpdraftPlus_Backup_History::rebuild($remotescan);
        $backup_history = UpdraftPlus_Backup_History::get_history();
        $output = $this->existing_backup_table($backup_history);
        $data = array();
 
        if (!empty($messages) && is_array($messages)) {
            $noutput = '<div style="margin-left: 100px; margin-top: 10px;"><ul style="list-style: disc inside;">';
            foreach ($messages as $msg) {
                $noutput .= '<li>'.(empty($msg['desc']) ? '' : $msg['desc'].': ').'<em>'.$msg['message'].'</em></li>';
                if (!empty($msg['data'])) {
                    if (!empty($msg['desc'])) {
                        $data['desc'] = $msg['data'];
                    } else {
                        // At the time of authorship, this code branch is not known to be used
                        $data[] = $msg['data'];
                    }
                }
            }
            $noutput .= '</ul></div>';
            $output = $noutput.$output;
        }
        
        $logs_exist = (false !== strpos($output, 'downloadlog'));
        if (!$logs_exist) {
            list($mod_time, $log_file, $nonce) = $updraftplus->last_modified_log();
            if ($mod_time) $logs_exist = true;
        }
        
        return apply_filters('updraftplus_get_history_status_result', array(
            'n' => sprintf(__('Existing Backups', 'updraftplus').' (%d)', count($backup_history)),
            't' => $output,
            'data' => $data,
            'cksum' => md5($output),
            'logs_exist' => $logs_exist,
        ));
    }
    
    /**
     * Get information on disk space used by an entity, or by UD's internal directory. Returns as a human-readable string.
     *
     * @param String $entity - the entity (e.g. 'plugins'; 'all' for all entities, or 'ud' for UD's internal directory)
     *
     * @return String
     */
    public function get_disk_space_used($entity) {
        global $updraftplus;
        if ('updraft' == $entity) {
            return $this->recursive_directory_size($updraftplus->backups_dir_location());
        } else {
            $backupable_entities = $updraftplus->get_backupable_file_entities(true, false);
            if ('all' == $entity) {
                $total_size = 0;
                foreach ($backupable_entities as $entity => $data) {
                    // Might be an array
                    $basedir = $backupable_entities[$entity];
                    $dirs = apply_filters('updraftplus_dirlist_'.$entity, $basedir);
                    $size = $this->recursive_directory_size($dirs, $updraftplus->get_exclude($entity), $basedir, 'numeric');
                    if (is_numeric($size) && $size>0) $total_size += $size;
                }
                return $updraftplus->convert_numeric_size_to_text($total_size);
            } elseif (!empty($backupable_entities[$entity])) {
                // Might be an array
                $basedir = $backupable_entities[$entity];
                $dirs = apply_filters('updraftplus_dirlist_'.$entity, $basedir);
                return $this->recursive_directory_size($dirs, $updraftplus->get_exclude($entity), $basedir);
            }
        }
        return __('Error', 'updraftplus');
    }
    
    /**
     * Stop an active backup job
     *
     * @param String $job_id - job ID of the job to stop
     *
     * @return Array - information on the outcome of the attempt
     */
    public function activejobs_delete($job_id) {
            
        if (preg_match("/^[0-9a-f]{12}$/", $job_id)) {
        
            global $updraftplus;
            $cron = get_option('cron');
            $found_it = false;
        
            $updraft_dir = $updraftplus->backups_dir_location();
            if (file_exists($updraft_dir.'/log.'.$job_id.'.txt')) touch($updraft_dir.'/deleteflag-'.$job_id.'.txt');
            
            foreach ($cron as $time => $job) {
                if (isset($job['updraft_backup_resume'])) {
                    foreach ($job['updraft_backup_resume'] as $hook => $info) {
                        if (isset($info['args'][1]) && $info['args'][1] == $job_id) {
                            $args = $cron[$time]['updraft_backup_resume'][$hook]['args'];
                            wp_unschedule_event($time, 'updraft_backup_resume', $args);
                            if (!$found_it) return array('ok' => 'Y', 'c' => 'deleted', 'm' => __('Job deleted', 'updraftplus'));
                            $found_it = true;
                        }
                    }
                }
            }
        }
 
        if (!$found_it) return array('ok' => 'N', 'c' => 'not_found', 'm' => __('Could not find that job - perhaps it has already finished?', 'updraftplus'));
 
    }
 
    /**
     * Input: an array of items
     * Each item is in the format: <base>,<timestamp>,<type>(,<findex>)
     * The 'base' is not for us: we just pass it straight back
     *
     * @param  array $downloaders Array of Items to download
     * @return array
     */
    public function get_download_statuses($downloaders) {
        global $updraftplus;
        $download_status = array();
        foreach ($downloaders as $downloader) {
            // prefix, timestamp, entity, index
            if (preg_match('/^([^,]+),(\d+),([-a-z]+|db[0-9]+),(\d+)$/', $downloader, $matches)) {
                $findex = (empty($matches[4])) ? '0' : $matches[4];
                $updraftplus->nonce = dechex($matches[2]).$findex.substr(md5($matches[3]), 0, 3);
                $updraftplus->jobdata_reset();
                $status = $this->download_status($matches[2], $matches[3], $matches[4]);
                if (is_array($status)) {
                    $status['base'] = $matches[1];
                    $status['timestamp'] = $matches[2];
                    $status['what'] = $matches[3];
                    $status['findex'] = $findex;
                    $download_status[] = $status;
                }
            }
        }
        return $download_status;
    }
    
    public function get_activejobs_list($request) {
    
        global $updraftplus;
        
        $download_status = empty($request['downloaders']) ? array() : $this->get_download_statuses(explode(':', $request['downloaders']));
 
        if (!empty($request['oneshot'])) {
            $job_id = get_site_option('updraft_oneshotnonce', false);
            // print_active_job() for one-shot jobs that aren't in cron
            $active_jobs = (false === $job_id) ? '' : $this->print_active_job($job_id, true);
        } elseif (!empty($request['thisjobonly'])) {
            // print_active_jobs() is for resumable jobs where we want the cron info to be included in the output
            $active_jobs = $this->print_active_jobs($request['thisjobonly']);
        } else {
            $active_jobs = $this->print_active_jobs();
        }
 
        $logupdate_array = array();
        if (!empty($request['log_fetch'])) {
            if (isset($request['log_nonce'])) {
                $log_nonce = $request['log_nonce'];
                $log_pointer = isset($request['log_pointer']) ? absint($request['log_pointer']) : 0;
                $logupdate_array = $this->fetch_log($log_nonce, $log_pointer);
            }
        }
 
        return array(
            // We allow the front-end to decide what to do if there's nothing logged - we used to (up to 1.11.29) send a pre-defined message
            'l' => htmlspecialchars(UpdraftPlus_Options::get_updraft_option('updraft_lastmessage', '')),
            'j' => $active_jobs,
            'ds' => $download_status,
            'u' => $logupdate_array
        );
    
    }
    
    public function request_backupnow($request, $close_connection_callable = false) {
        global $updraftplus;
        
        $backupnow_nocloud = (empty($request['backupnow_nocloud'])) ? false : true;
        $event = (!empty($request['backupnow_nofiles'])) ? 'updraft_backupnow_backup_database' : ((!empty($request['backupnow_nodb'])) ? 'updraft_backupnow_backup' : 'updraft_backupnow_backup_all');
 
        // The call to backup_time_nonce() allows us to know the nonce in advance, and return it
        $nonce = $updraftplus->backup_time_nonce();
 
        $msg = array(
            'nonce' => $nonce,
            'm' => '<strong>'.__('Start backup', 'updraftplus').':</strong> '.htmlspecialchars(__('OK. You should soon see activity in the "Last log message" field below.', 'updraftplus'))
        );
        
        if ($close_connection_callable && is_callable($close_connection_callable)) {
            call_user_func($close_connection_callable, $msg);
        } else {
            $updraftplus->close_browser_connection(json_encode($msg));
        }
 
        $options = array('nocloud' => $backupnow_nocloud, 'use_nonce' => $nonce);
        if (!empty($request['onlythisfileentity']) && is_string($request['onlythisfileentity'])) {
            // Something to see in the 'last log' field when it first appears, before the backup actually starts
            $updraftplus->log(__('Start backup', 'updraftplus'));
            $options['restrict_files_to_override'] = explode(',', $request['onlythisfileentity']);
        }
 
        if (!empty($request['extradata'])) {
            $options['extradata'] = $request['extradata'];
        }
 
        do_action($event, apply_filters('updraft_backupnow_options', $options, $request));
    }
    
    /**
     * Get the contents of a log file
     *
     * @param String  $backup_nonce     - the backup id; or empty, for the most recently modified
     * @param Integer $log_pointer     - the byte count to fetch from
     * @param String  $output_format - the format to return in; allowed as 'html' (which will escape HTML entities in what is returned) and 'raw'
     *
     * @return String
     */
    public function fetch_log($backup_nonce = '', $log_pointer = 0, $output_format = 'html') {
        global $updraftplus;
 
        if (empty($backup_nonce)) {
            list($mod_time, $log_file, $nonce) = $updraftplus->last_modified_log();
        } else {
            $nonce = $backup_nonce;
        }
 
        if (!preg_match('/^[0-9a-f]+$/', $nonce)) die('Security check');
        
        $log_content = '';
        $new_pointer = $log_pointer;
        
        if (!empty($nonce)) {
            $updraft_dir = $updraftplus->backups_dir_location();
 
            $potential_log_file = $updraft_dir."/log.".$nonce.".txt";
 
            if (is_readable($potential_log_file)) {
                
                $templog_array = array();
                $log_file = fopen($potential_log_file, "r");
                if ($log_pointer > 0) fseek($log_file, $log_pointer);
                
                while (($buffer = fgets($log_file, 4096)) !== false) {
                    $templog_array[] = $buffer;
                }
                if (!feof($log_file)) {
                    $templog_array[] = __('Error: unexpected file read fail', 'updraftplus');
                }
                
                $new_pointer = ftell($log_file);
                $log_content = implode("", $templog_array);
 
                
            } else {
                $log_content .= __('The log file could not be read.', 'updraftplus');
            }
 
        } else {
            $log_content .= __('The log file could not be read.', 'updraftplus');
        }
        
        if ('html' == $output_format) $log_content = htmlspecialchars($log_content);
        
        $ret_array = array(
            'log' => $log_content,
            'nonce' => $nonce,
            'pointer' => $new_pointer
        );
        
        return $ret_array;
    }
 
    /**
     * Get a count for the number of overdue cron jobs
     *
     * @return Integer - how many cron jobs are overdue
     */
    public function howmany_overdue_crons() {
        $how_many_overdue = 0;
        if (function_exists('_get_cron_array') || (is_file(ABSPATH.WPINC.'/cron.php') && include_once(ABSPATH.WPINC.'/cron.php') && function_exists('_get_cron_array'))) {
            $crons = _get_cron_array();
            if (is_array($crons)) {
                $timenow = time();
                foreach ($crons as $jt => $job) {
                    if ($jt < $timenow) $how_many_overdue++;
                }
            }
        }
        return $how_many_overdue;
    }
 
    public function get_php_errors($errno, $errstr, $errfile, $errline) {
        global $updraftplus;
        if (0 == error_reporting()) return true;
        $logline = $updraftplus->php_error_to_logline($errno, $errstr, $errfile, $errline);
        if (false !== $logline) $this->logged[] = $logline;
        // Don't pass it up the chain (since it's going to be output to the user always)
        return true;
    }
 
    private function download_status($timestamp, $type, $findex) {
        global $updraftplus;
        $response = array('m' => $updraftplus->jobdata_get('dlmessage_'.$timestamp.'_'.$type.'_'.$findex).'<br>');
        if ($file = $updraftplus->jobdata_get('dlfile_'.$timestamp.'_'.$type.'_'.$findex)) {
            if ('failed' == $file) {
                $response['e'] = __('Download failed', 'updraftplus').'<br>';
                $response['failed'] = true;
                $errs = $updraftplus->jobdata_get('dlerrors_'.$timestamp.'_'.$type.'_'.$findex);
                if (is_array($errs) && !empty($errs)) {
                    $response['e'] .= '<ul class="disc">';
                    foreach ($errs as $err) {
                        if (is_array($err)) {
                            $response['e'] .= '<li>'.htmlspecialchars($err['message']).'</li>';
                        } else {
                            $response['e'] .= '<li>'.htmlspecialchars($err).'</li>';
                        }
                    }
                    $response['e'] .= '</ul>';
                }
            } elseif (preg_match('/^downloaded\d+).*)$/', $file, $matches) && file_exists($matches[2])) {
                $response['p'] = 100;
                $response['f'] = $matches[2];
                $response['s'] = (int) $matches[1];
                $response['t'] = (int) $matches[1];
                $response['m'] = __('File ready.', 'updraftplus');
                if ('db' != substr($type, 0, 2)) $response['can_show_contents'] = true;
            } elseif (preg_match('/^downloading\d+).*)$/', $file, $matches) && file_exists($matches[2])) {
                // Convert to bytes
                $response['f'] = $matches[2];
                $total_size = (int) max($matches[1], 1);
                $cur_size = filesize($matches[2]);
                $response['s'] = $cur_size;
                $file_age = time() - filemtime($matches[2]);
                if ($file_age > 20) $response['a'] = time() - filemtime($matches[2]);
                $response['t'] = $total_size;
                $response['m'] .= __("Download in progress", 'updraftplus').' ('.round($cur_size/1024).' / '.round(($total_size/1024)).' KB)';
                $response['p'] = round(100*$cur_size/$total_size);
            } else {
                $response['m'] .= __('No local copy present.', 'updraftplus');
                $response['p'] = 0;
                $response['s'] = 0;
                $response['t'] = 1;
            }
        }
        return $response;
    }
 
    public function upload_dir($uploads) {
        global $updraftplus;
        $updraft_dir = $updraftplus->backups_dir_location();
        if (is_writable($updraft_dir)) $uploads['path'] = $updraft_dir;
        return $uploads;
    }
 
    /**
     * We do actually want to over-write
     *
     * @param  String $dir  Directory
     * @param  String $name Name
     * @param  String $ext  File extension
     * @return String
     */
    public function unique_filename_callback($dir, $name, $ext) {
        return $name.$ext;
    }
 
    public function sanitize_file_name($filename) {
        // WordPress 3.4.2 on multisite (at least) adds in an unwanted underscore
        return preg_replace('/-db(.*)\.gz_\.crypt$/', '-db$1.gz.crypt', $filename);
    }
 
    public function plupload_action() {
        // check ajax nonce
 
        global $updraftplus;
        @set_time_limit(UPDRAFTPLUS_SET_TIME_LIMIT);
 
        if (!UpdraftPlus_Options::user_can_manage()) exit;
        check_ajax_referer('updraft-uploader');
 
        $updraft_dir = $updraftplus->backups_dir_location();
        if (!@$updraftplus->really_is_writable($updraft_dir)) {
            echo json_encode(array('e' => sprintf(__("Backup directory (%s) is not writable, or does not exist.", 'updraftplus'), $updraft_dir).' '.__('You will find more information about this in the Settings section.', 'updraftplus')));
            exit;
        }
        
        add_filter('upload_dir', array($this, 'upload_dir'));
        add_filter('sanitize_file_name', array($this, 'sanitize_file_name'));
        // handle file upload
 
        $farray = array('test_form' => true, 'action' => 'plupload_action');
 
        $farray['test_type'] = false;
        $farray['ext'] = 'x-gzip';
        $farray['type'] = 'application/octet-stream';
 
        if (!isset($_POST['chunks'])) {
            $farray['unique_filename_callback'] = array($this, 'unique_filename_callback');
        }
 
        $status = wp_handle_upload(
            $_FILES['async-upload'],
            $farray
        );
        remove_filter('upload_dir', array($this, 'upload_dir'));
        remove_filter('sanitize_file_name', array($this, 'sanitize_file_name'));
 
        if (isset($status['error'])) {
            echo json_encode(array('e' => $status['error']));
            exit;
        }
 
        // If this was the chunk, then we should instead be concatenating onto the final file
        if (isset($_POST['chunks']) && isset($_POST['chunk']) && preg_match('/^[0-9]+$/', $_POST['chunk'])) {
            $final_file = basename($_POST['name']);
            if (!rename($status['file'], $updraft_dir.'/'.$final_file.'.'.$_POST['chunk'].'.zip.tmp')) {
                @unlink($status['file']);
                echo json_encode(array('e' => sprintf(__('Error: %s', 'updraftplus'), __('This file could not be uploaded', 'updraftplus'))));
                exit;
            }
            $status['file'] = $updraft_dir.'/'.$final_file.'.'.$_POST['chunk'].'.zip.tmp';
 
            // Final chunk? If so, then stich it all back together
            if ($_POST['chunk'] == $_POST['chunks']-1) {
                if ($wh = fopen($updraft_dir.'/'.$final_file, 'wb')) {
                    for ($i=0; $i<$_POST['chunks']; $i++) {
                        $rf = $updraft_dir.'/'.$final_file.'.'.$i.'.zip.tmp';
                        if ($rh = fopen($rf, 'rb')) {
                            while ($line = fread($rh, 32768)) {
                                fwrite($wh, $line);
                            }
                            fclose($rh);
                            @unlink($rf);
                        }
                    }
                    fclose($wh);
                    $status['file'] = $updraft_dir.'/'.$final_file;
                    if ('.tar' == substr($final_file, -4, 4)) {
                        if (file_exists($status['file'].'.gz')) unlink($status['file'].'.gz');
                        if (file_exists($status['file'].'.bz2')) unlink($status['file'].'.bz2');
                    } elseif ('.tar.gz' == substr($final_file, -7, 7)) {
                        if (file_exists(substr($status['file'], 0, strlen($status['file'])-3))) unlink(substr($status['file'], 0, strlen($status['file'])-3));
                        if (file_exists(substr($status['file'], 0, strlen($status['file'])-3).'.bz2')) unlink(substr($status['file'], 0, strlen($status['file'])-3).'.bz2');
                    } elseif ('.tar.bz2' == substr($final_file, -8, 8)) {
                        if (file_exists(substr($status['file'], 0, strlen($status['file'])-4))) unlink(substr($status['file'], 0, strlen($status['file'])-4));
                        if (file_exists(substr($status['file'], 0, strlen($status['file'])-4).'.gz')) unlink(substr($status['file'], 0, strlen($status['file'])-3).'.gz');
                    }
                }
            }
 
        }
 
        $response = array();
        if (!isset($_POST['chunks']) || (isset($_POST['chunk']) && $_POST['chunk'] == $_POST['chunks']-1)) {
            $file = basename($status['file']);
            if (!preg_match('/^log\.[a-f0-9]{12}\.txt/i', $file) && !preg_match('/^backup_([\-0-9]{15})_.*_([0-9a-f]{12})-([\-a-z]+)([0-9]+)?(\.(zip|gz|gz\.crypt))?$/i', $file, $matches)) {
                $accept = apply_filters('updraftplus_accept_archivename', array());
                if (is_array($accept)) {
                    foreach ($accept as $acc) {
                        if (preg_match('/'.$acc['pattern'].'/i', $file)) $accepted = $acc['desc'];
                    }
                }
                if (!empty($accepted)) {
                    $response['dm'] = sprintf(__('This backup was created by %s, and can be imported.', 'updraftplus'), $accepted);
                } else {
                    @unlink($status['file']);
                    echo json_encode(array('e' => sprintf(__('Error: %s', 'updraftplus'), __('Bad filename format - this does not look like a file created by UpdraftPlus', 'updraftplus'))));
                    exit;
                }
            } else {
                $backupable_entities = $updraftplus->get_backupable_file_entities(true);
                $type = isset($matches[3]) ? $matches[3] : '';
                if (!preg_match('/^log\.[a-f0-9]{12}\.txt/', $file) && 'db' != $type && !isset($backupable_entities[$type])) {
                    @unlink($status['file']);
                    echo json_encode(array('e' => sprintf(__('Error: %s', 'updraftplus'), sprintf(__('This looks like a file created by UpdraftPlus, but this install does not know about this type of object: %s. Perhaps you need to install an add-on?', 'updraftplus'), htmlspecialchars($type)))));
                    exit;
                }
            }
        }
 
        // send the uploaded file url in response
        $response['m'] = $status['url'];
        echo json_encode($response);
        exit;
    }
 
    /**
     * Database decrypter
     */
    public function plupload_action2() {
 
        @set_time_limit(UPDRAFTPLUS_SET_TIME_LIMIT);
        global $updraftplus;
 
        if (!UpdraftPlus_Options::user_can_manage()) exit;
        check_ajax_referer('updraft-uploader');
 
        $updraft_dir = $updraftplus->backups_dir_location();
        if (!is_writable($updraft_dir)) exit;
 
        add_filter('upload_dir', array($this, 'upload_dir'));
        add_filter('sanitize_file_name', array($this, 'sanitize_file_name'));
        // handle file upload
 
        $farray = array('test_form' => true, 'action' => 'plupload_action2');
 
        $farray['test_type'] = false;
        $farray['ext'] = 'crypt';
        $farray['type'] = 'application/octet-stream';
 
        if (isset($_POST['chunks'])) {
            // $farray['ext'] = 'zip';
            // $farray['type'] = 'application/zip';
        } else {
            $farray['unique_filename_callback'] = array($this, 'unique_filename_callback');
        }
 
        $status = wp_handle_upload(
            $_FILES['async-upload'],
            $farray
        );
        remove_filter('upload_dir', array($this, 'upload_dir'));
        remove_filter('sanitize_file_name', array($this, 'sanitize_file_name'));
 
        if (isset($status['error'])) {
            echo 'ERROR:'.$status['error'];
            exit;
        }
 
        // If this was the chunk, then we should instead be concatenating onto the final file
        if (isset($_POST['chunks']) && isset($_POST['chunk']) && preg_match('/^[0-9]+$/', $_POST['chunk'])) {
            $final_file = basename($_POST['name']);
            rename($status['file'], $updraft_dir.'/'.$final_file.'.'.$_POST['chunk'].'.zip.tmp');
            $status['file'] = $updraft_dir.'/'.$final_file.'.'.$_POST['chunk'].'.zip.tmp';
 
            // Final chunk? If so, then stich it all back together
            if ($_POST['chunk'] == $_POST['chunks']-1) {
                if ($wh = fopen($updraft_dir.'/'.$final_file, 'wb')) {
                    for ($i=0; $i<$_POST['chunks']; $i++) {
                        $rf = $updraft_dir.'/'.$final_file.'.'.$i.'.zip.tmp';
                        if ($rh = fopen($rf, 'rb')) {
                            while ($line = fread($rh, 32768)) {
                                fwrite($wh, $line);
                            }
                            fclose($rh);
                            @unlink($rf);
                        }
                    }
                    fclose($wh);
                    $status['file'] = $updraft_dir.'/'.$final_file;
                }
            }
 
        }
 
        if (!isset($_POST['chunks']) || (isset($_POST['chunk']) && $_POST['chunk'] == $_POST['chunks']-1)) {
            $file = basename($status['file']);
            if (!preg_match('/^backup_([\-0-9]{15})_.*_([0-9a-f]{12})-db([0-9]+)?\.(gz\.crypt)$/i', $file)) {
 
                @unlink($status['file']);
                echo 'ERROR:'.__('Bad filename format - this does not look like an encrypted database file created by UpdraftPlus', 'updraftplus');
 
                exit;
            }
        }
 
        // send the uploaded file url in response
        echo 'OK:'.$file;
        exit;
    }
 
    /**
     * Include the settings header template
     */
    public function settings_header() {
        $this->include_template('wp-admin/settings/header.php');
    }
 
    public function settings_output() {
 
        if (false == ($render = apply_filters('updraftplus_settings_page_render', true))) {
            do_action('updraftplus_settings_page_render_abort', $render);
            return;
        }
 
        do_action('updraftplus_settings_page_init');
 
        global $updraftplus;
 
        /**
         * We use request here because the initial restore is triggered by a POSTed form. we then may need to obtain credential for the WP_Filesystem. to do this WP outputs a form, but we don't pass our parameters via that. So the values are passed back in as GET parameters.
         */
        if (isset($_REQUEST['action']) && (('updraft_restore' == $_REQUEST['action'] && isset($_REQUEST['backup_timestamp'])) || ('updraft_restore_continue' == $_REQUEST['action'] && !empty($_REQUEST['restoreid'])))) {
 
            $is_continuation = ('updraft_restore_continue' == $_REQUEST['action']) ? true : false;
 
            if ($is_continuation) {
                $restore_in_progress = get_site_option('updraft_restore_in_progress');
                if ($restore_in_progress != $_REQUEST['restoreid']) {
                    $abort_restore_already = true;
                    $updraftplus->log(__('Sufficient information about the in-progress restoration operation could not be found.', 'updraftplus').' (restoreid_mismatch)', 'error', 'restoreid_mismatch');
                } else {
 
                    $restore_jobdata = $updraftplus->jobdata_getarray($restore_in_progress);
                    if (is_array($restore_jobdata) && isset($restore_jobdata['job_type']) && 'restore' == $restore_jobdata['job_type'] && isset($restore_jobdata['second_loop_entities']) && !empty($restore_jobdata['second_loop_entities']) && isset($restore_jobdata['job_time_ms']) && isset($restore_jobdata['backup_timestamp'])) {
                        $backup_timestamp = $restore_jobdata['backup_timestamp'];
                        $continuation_data = $restore_jobdata;
                    } else {
                        $abort_restore_already = true;
                        $updraftplus->log(__('Sufficient information about the in-progress restoration operation could not be found.', 'updraftplus').' (restoreid_nojobdata)', 'error', 'restoreid_nojobdata');
                    }
                }
 
            } else {
                $backup_timestamp = $_REQUEST['backup_timestamp'];
                $continuation_data = null;
            }
 
            if (empty($abort_restore_already)) {
                $backup_success = $this->restore_backup($backup_timestamp, $continuation_data);
            } else {
                $backup_success = false;
            }
 
            if (empty($updraftplus->errors) && true === $backup_success) {
                // TODO: Deal with the case of some of the work having been deferred
                // If we restored the database, then that will have out-of-date information which may confuse the user - so automatically re-scan for them.
                UpdraftPlus_Backup_History::rebuild();
                echo '<p><strong>';
                $updraftplus->log_e('Restore successful!');
                echo '</strong></p>';
                $updraftplus->log("Restore successful");
                $s_val = 1;
                if (!empty($this->entities_to_restore) && is_array($this->entities_to_restore)) {
                    foreach ($this->entities_to_restore as $k => $v) {
                        if ('db' != $v) $s_val = 2;
                    }
                }
                $pval = ($updraftplus->have_addons) ? 1 : 0;
 
                echo '<strong>'.__('Actions', 'updraftplus').':</strong> <a href="'.UpdraftPlus_Options::admin_page_url().'?page=updraftplus&updraft_restore_success='.$s_val.'&pval='.$pval.'">'.__('Return to UpdraftPlus Configuration', 'updraftplus').'</a>';
                return;
            } elseif (is_wp_error($backup_success)) {
                echo '<p>';
                $updraftplus->log_e('Restore failed...');
                echo '</p>';
                $updraftplus->log_wp_error($backup_success);
                $updraftplus->log("Restore failed");
                $updraftplus->list_errors();
                echo '<strong>'.__('Actions', 'updraftplus').':</strong> <a href="'.UpdraftPlus_Options::admin_page_url().'?page=updraftplus">'.__('Return to UpdraftPlus Configuration', 'updraftplus').'</a>';
                return;
            } elseif (false === $backup_success) {
                // This means, "not yet - but stay on the page because we may be able to do it later, e.g. if the user types in the requested information"
                echo '<p>';
                $updraftplus->log_e('Restore failed...');
                echo '</p>';
                $updraftplus->log("Restore failed");
                $updraftplus->list_errors();
                echo '<strong>'.__('Actions', 'updraftplus').':</strong> <a href="'.UpdraftPlus_Options::admin_page_url().'?page=updraftplus">'.__('Return to UpdraftPlus Configuration', 'updraftplus').'</a>';
                return;
            }
        }
 
        if (isset($_REQUEST['action']) && 'updraft_delete_old_dirs' == $_REQUEST['action']) {
            $nonce = (empty($_REQUEST['_wpnonce'])) ? "" : $_REQUEST['_wpnonce'];
            if (!wp_verify_nonce($nonce, 'updraftplus-credentialtest-nonce')) die('Security check');
            $this->delete_old_dirs_go();
            return;
        }
 
        if (!empty($_REQUEST['action']) && 'updraftplus_broadcastaction' == $_REQUEST['action'] && !empty($_REQUEST['subaction'])) {
            $nonce = (empty($_REQUEST['nonce'])) ? "" : $_REQUEST['nonce'];
            if (!wp_verify_nonce($nonce, 'updraftplus-credentialtest-nonce')) die('Security check');
            do_action($_REQUEST['subaction']);
            return;
        }
 
        if (isset($_GET['error'])) {
            // This is used by Microsoft OneDrive authorisation failures (May 15). I am not sure what may have been using the 'error' GET parameter otherwise - but it is harmless.
            if (!empty($_GET['error_description'])) {
                $this->show_admin_warning(htmlspecialchars($_GET['error_description']).' ('.htmlspecialchars($_GET['error']).')', 'error');
            } else {
                $this->show_admin_warning(htmlspecialchars($_GET['error']), 'error');
            }
        }
 
        if (isset($_GET['message'])) $this->show_admin_warning(htmlspecialchars($_GET['message']));
 
        if (isset($_GET['action']) && 'updraft_create_backup_dir' == $_GET['action'] && isset($_GET['nonce']) && wp_verify_nonce($_GET['nonce'], 'create_backup_dir')) {
            $created = $this->create_backup_dir();
            if (is_wp_error($created)) {
                echo '<p>'.__('Backup directory could not be created', 'updraftplus').'...<br>';
                echo '<ul class="disc">';
                foreach ($created->get_error_messages() as $key => $msg) {
                    echo '<li>'.htmlspecialchars($msg).'</li>';
                }
                echo '</ul></p>';
            } elseif (false !== $created) {
                echo '<p>'.__('Backup directory successfully created.', 'updraftplus').'</p><br>';
            }
            echo '<b>'.__('Actions', 'updraftplus').':</b> <a href="'.UpdraftPlus_Options::admin_page_url().'?page=updraftplus">'.__('Return to UpdraftPlus Configuration', 'updraftplus').'</a>';
            return;
        }
 
        echo '<div id="updraft_backup_started" class="updated updraft-hidden" style="display:none;"></div>';
 
        if (isset($_POST['action']) && 'updraft_wipesettings' == $_POST['action']) {
            $this->updraft_wipe_settings();
        }
 
        // This opens a div
        $this->settings_header();
        ?>
 
            <div id="updraft-hidethis">
            <p>
            <strong><?php _e('Warning:', 'updraftplus'); ?> <?php _e("If you can still read these words after the page finishes loading, then there is a JavaScript or jQuery problem in the site.", 'updraftplus'); ?></strong>
 
            <?php if (false !== strpos(basename(UPDRAFTPLUS_URL), ' ')) { ?>
                <strong><?php _e('The UpdraftPlus directory in wp-content/plugins has white-space in it; WordPress does not like this. You should rename the directory to wp-content/plugins/updraftplus to fix this problem.', 'updraftplus');?></strong>
            <?php } else { ?>
                <a href="<?php echo apply_filters('updraftplus_com_link', "https://updraftplus.com/do-you-have-a-javascript-or-jquery-error/");?>"><?php _e('Go here for more information.', 'updraftplus'); ?></a>
            <?php } ?>
            </p>
            </div>
 
            <?php
 
            $include_deleteform_div = true;
 
            // Opens a div, which needs closing later
            if (isset($_GET['updraft_restore_success'])) {
 
                if (get_template() === 'optimizePressTheme' || is_plugin_active('optimizePressPlugin') || is_plugin_active_for_network('optimizePressPlugin')) {
                    $this->show_admin_warning("<a href='https://optimizepress.zendesk.com/hc/en-us/articles/203699826-Update-URL-References-after-moving-domain' target='_blank'>" . __("OptimizePress 2.0 encodes its contents, so search/replace does not work.", "updraftplus") . ' ' . __("To fix this problem go here.", "updraftplus") . "</a>", "notice notice-warning");
                }
                $success_advert = (isset($_GET['pval']) && 0 == $_GET['pval'] && !$updraftplus->have_addons) ? '<p>'.__('For even more features and personal support, check out ', 'updraftplus').'<strong><a href="'.apply_filters("updraftplus_com_link", 'https://updraftplus.com/shop/updraftplus-premium/').'" target="_blank">UpdraftPlus Premium</a>.</strong></p>' : "";
 
                echo "<div class=\"updated backup-restored\"><span><strong>".__('Your backup has been restored.', 'updraftplus').'</strong></span><br>';
                // Unnecessary - will be advised of this below
                // if (2 == $_GET['updraft_restore_success']) echo ' '.__('Your old (themes, uploads, plugins, whatever) directories have been retained with "-old" appended to their name. Remove them when you are satisfied that the backup worked properly.');
                echo $success_advert;
                $include_deleteform_div = false;
 
            }
 
            if ($this->scan_old_dirs(true)) $this->print_delete_old_dirs_form(true, $include_deleteform_div);
 
            // Close the div opened by the earlier section
            if (isset($_GET['updraft_restore_success'])) echo '</div>';
 
            if (empty($success_advert) && empty($this->no_settings_warning)) {
 
                if (!class_exists('UpdraftPlus_Notices')) include_once(UPDRAFTPLUS_DIR.'/includes/updraftplus-notices.php');
                global $updraftplus_notices;
                $updraftplus_notices->do_notice();
            }
 
            if (!$updraftplus->memory_check(64)) {
                // HS8390 - A case where UpdraftPlus::memory_check_current() returns -1
                $memory_check_current = $updraftplus->memory_check_current();
                if ($memory_check_current > 0) {
                ?>
                    <div class="updated memory-limit"><?php _e('Your PHP memory limit (set by your web hosting company) is very low. UpdraftPlus attempted to raise it but was unsuccessful. This plugin may struggle with a memory limit of less than 64 Mb  - especially if you have very large files uploaded (though on the other hand, many sites will be successful with a 32Mb limit - your experience may vary).', 'updraftplus');?> <?php _e('Current limit is:', 'updraftplus');?> <?php echo $updraftplus->memory_check_current(); ?> MB</div>
                <?php }
            }
 
 
            if (!empty($updraftplus->errors)) {
                echo '<div class="error updraft_list_errors">';
                $updraftplus->list_errors();
                echo '</div>';
            }
 
            $backup_history = UpdraftPlus_Backup_History::get_history();
            if (empty($backup_history)) {
                UpdraftPlus_Backup_History::rebuild();
                $backup_history = UpdraftPlus_Backup_History::get_history();
            }
 
            
            ?>
 
        <?php
        
            $tabflag = 1;
            
            if (isset($_REQUEST['tab'])) {
                switch ($_REQUEST['tab']) {
                    case 'status':
                        $tabflag = 1;
                        break;
                    case 'backups':
                        $tabflag = 2;
                        break;
                    case 'settings':
                        $tabflag = 3;
                        break;
                    case 'expert':
                        $tabflag = 4;
                        break;
                    case 'addons':
                        $tabflag = 5;
                        break;
                    default:
                        $tabflag = 1;
                }
            }
        
            $this->include_template('wp-admin/settings/tab-bar.php', false, array('backup_history' => $backup_history, 'tabflag' => $tabflag));
 
            $updraft_dir = $updraftplus->backups_dir_location();
            $backup_disabled = $updraftplus->really_is_writable($updraft_dir) ? '' : 'disabled="disabled"';
        ?>
        
        <div id="updraft-poplog" >
            <pre id="updraft-poplog-content"></pre>
        </div>
        
        <?php $this->include_template('wp-admin/settings/tab-status.php', false, array('tabflag' => $tabflag, 'backup_disabled' => $backup_disabled)); ?>
        
        <div id="updraft-navtab-backups-content" <?php if (2 != $tabflag) echo 'class="updraft-hidden"'; ?> style="<?php if (2 != $tabflag) echo 'display:none;'; ?>">
            <?php
                $is_opera = (false !== strpos($_SERVER['HTTP_USER_AGENT'], 'Opera') || false !== strpos($_SERVER['HTTP_USER_AGENT'], 'OPR/'));
                $tmp_opts = array('include_opera_warning' => $is_opera);
                $this->settings_downloading_and_restoring($backup_history, false, $tmp_opts);
                $this->include_template('wp-admin/settings/delete-and-restore-modals.php');
            ?>
        </div>
 
        <div id="updraft-navtab-settings-content" <?php if (3 != $tabflag) echo 'class="updraft-hidden"'; ?> style="<?php if (3 != $tabflag) echo 'display:none;'; ?>">
            <h2 class="updraft_settings_sectionheading"><?php _e('Backup Contents And Schedule', 'updraftplus');?></h2>
            <?php UpdraftPlus_Options::options_form_begin(); ?>
                <?php $this->settings_formcontents(); ?>
            </form>
        </div>
 
        <div id="updraft-navtab-expert-content"<?php if (4 != $tabflag) echo ' class="updraft-hidden"'; ?> style="<?php if (4 != $tabflag) echo 'display:none;'; ?>">
            <?php $this->settings_advanced_tools(); ?>
        </div>
 
        <div id="updraft-navtab-addons-content"<?php if (5 != $tabflag) echo ' class="updraft-hidden"'; ?> style="<?php if (5 != $tabflag) echo 'display:none;'; ?>">
        
            <?php
                $tab_addons = $this->include_template('wp-admin/settings/tab-addons.php', true, array('tabflag' => $tabflag));
                
                echo apply_filters('updraftplus_addonstab_content', $tab_addons);
                
            ?>
        
        </div>
        
        <?php
        // settings_header() opens a div
        echo '</div>';
    }
 
    private function print_restore_in_progress_box_if_needed() {
        $restore_in_progress = get_site_option('updraft_restore_in_progress');
        if (!empty($restore_in_progress)) {
            global $updraftplus;
            $restore_jobdata = $updraftplus->jobdata_getarray($restore_in_progress);
            if (is_array($restore_jobdata) && !empty($restore_jobdata)) {
                // Only print if within the last 24 hours; and only after 2 minutes
                if (isset($restore_jobdata['job_type']) && 'restore' == $restore_jobdata['job_type'] && isset($restore_jobdata['second_loop_entities']) && !empty($restore_jobdata['second_loop_entities']) && isset($restore_jobdata['job_time_ms']) && (time() - $restore_jobdata['job_time_ms'] > 120 || (defined('UPDRAFTPLUS_RESTORE_PROGRESS_ALWAYS_SHOW') && UPDRAFTPLUS_RESTORE_PROGRESS_ALWAYS_SHOW)) && time() - $restore_jobdata['job_time_ms'] < 86400 && (empty($_REQUEST['action']) || ('updraft_restore' != $_REQUEST['action'] && 'updraft_restore_continue' != $_REQUEST['action']))) {
                    $restore_jobdata['jobid'] = $restore_in_progress;
                    $this->restore_in_progress_jobdata = $restore_jobdata;
                    add_action('all_admin_notices', array($this, 'show_admin_restore_in_progress_notice'));
                }
            }
        }
    }
 
    public function show_admin_restore_in_progress_notice() {
    
        if (isset($_REQUEST['action']) && 'updraft_restore_abort' == $_REQUEST['action'] && !empty($_REQUEST['restoreid'])) {
            delete_site_option('updraft_restore_in_progress');
            return;
        }
    
        $restore_jobdata = $this->restore_in_progress_jobdata;
        $seconds_ago = time() - (int) $restore_jobdata['job_time_ms'];
        $minutes_ago = floor($seconds_ago/60);
        $seconds_ago = $seconds_ago - $minutes_ago*60;
        $time_ago = sprintf(__("%s minutes, %s seconds", 'updraftplus'), $minutes_ago, $seconds_ago);
        ?><div class="updated show_admin_restore_in_progress_notice">
            <span class="unfinished-restoration"><strong><?php echo 'UpdraftPlus: '.__('Unfinished restoration', 'updraftplus'); ?> </strong></span><br>
            <p><?php printf(__('You have an unfinished restoration operation, begun %s ago.', 'updraftplus'), $time_ago);?></p>
        <form method="post" action="<?php echo UpdraftPlus_Options::admin_page_url().'?page=updraftplus'; ?>">
            <?php wp_nonce_field('updraftplus-credentialtest-nonce'); ?>
            <input id="updraft_restore_continue_action" type="hidden" name="action" value="updraft_restore_continue">
            <input type="hidden" name="restoreid" value="<?php echo $restore_jobdata['jobid'];?>" value="<?php echo esc_attr($restore_jobdata['jobid']);?>">
            <button onclick="jQuery('#updraft_restore_continue_action').val('updraft_restore_continue'); jQuery(this).parent('form').submit();" type="submit" class="button-primary"><?php _e('Continue restoration', 'updraftplus'); ?></button>
            <button onclick="jQuery('#updraft_restore_continue_action').val('updraft_restore_abort'); jQuery(this).parent('form').submit();" class="button-secondary"><?php _e('Dismiss', 'updraftplus');?></button>
        </form><?php
        echo "</div>";
 
    }
 
    /**
     * Return widgetry for the 'backup now' modal
     *
     * @return String
     */
    public function backupnow_modal_contents() {
        return $this->backup_now_widgetry();
    }
    
    private function backup_now_widgetry() {
 
        $ret = '';
 
        $ret .= '<p><input type="checkbox" id="backupnow_includedb" checked="checked"> <label for="backupnow_includedb">'.__("Include the database in the backup", 'updraftplus').'</label> ';
 
        $ret .= '(<a href="#" id="backupnow_database_showmoreoptions">...</a>)<br>';
 
        $ret .= '<div id="backupnow_database_moreoptions" class="updraft-hidden" style="display:none;">';
 
        $premium_link = apply_filters('updraftplus_com_link', 'https://updraftplus.com/landing/updraftplus-premium');
 
        $free_ret = '<em>'.__('All WordPress tables will be backed up.', 'updraftplus').' <a href="'.$premium_link.'">'. __('With UpdraftPlus Premium, you can choose to backup non-WordPress tables, backup only specified tables, and backup other databases too.', 'updraftplus').'</a></em>';
 
        $ret .= apply_filters('updraft_backupnow_database_showmoreoptions', $free_ret, '') . '</p>';
 
        $ret .= '</div>';
            
        $ret .= '<p><input type="checkbox" id="backupnow_includefiles" checked="checked"> <label for="backupnow_includefiles">'.__("Include any files in the backup", 'updraftplus').'</label> (<a href="#" id="backupnow_includefiles_showmoreoptions">...</a>)<br>';
 
        $ret .= '<div id="backupnow_includefiles_moreoptions" class="updraft-hidden" style="display:none;"><em>'.__('Your saved settings also affect what is backed up - e.g. files excluded.', 'updraftplus').'</em><br>'.$this->files_selector_widgetry('backupnow_files_', false, 'sometimes').'</div></p>';
 
        $ret .= '<span id="backupnow_remote_container">'.$this->backup_now_remote_message().'</span>';
 
        $ret .= apply_filters('updraft_backupnow_modal_afteroptions', '', '');
 
        return $ret;
    }
 
    /**
     * Also used by the auto-backups add-on
     *
     * @param  Boolean $wide_format       Whether to return data in a wide format
     * @param  Boolean $print_active_jobs Whether to include currently active jobs
     * @return String - the HTML output
     */
    public function render_active_jobs_and_log_table($wide_format = false, $print_active_jobs = true) {
        ?>
            <table class="form-table" id="updraft_activejobs_table">
 
                <?php $active_jobs = ($print_active_jobs) ? $this->print_active_jobs() : '';?>
                <tr id="updraft_activejobsrow" class="<?php
                    if (!$active_jobs && !$wide_format) {
                        echo 'hidden';
                    }
                    if ($wide_format) {
                        echo ".minimum-height";
                    }
                ?>">
                    <?php if ($wide_format) { ?>
                        <td id="updraft_activejobs" colspan="2">
                            <?php echo $active_jobs;?>
                        </td>
                    <?php } else { ?>
                        <th><?php _e('Backups in progress:', 'updraftplus');?></th>
                        <td id="updraft_activejobs"><?php echo $active_jobs;?></td>
                    <?php } ?>
                </tr>
 
                <tr id="updraft_lastlogmessagerow">
                    <?php if ($wide_format) {
                        // Hide for now - too ugly
                        ?>
                        <td colspan="2" class="last-message"><strong><?php _e('Last log message', 'updraftplus');?>:</strong><br>
                            <span id="updraft_lastlogcontainer"><?php echo htmlspecialchars(UpdraftPlus_Options::get_updraft_option('updraft_lastmessage', __('(Nothing yet logged)', 'updraftplus'))); ?></span><br>
                            <?php $this->most_recently_modified_log_link(); ?>
                        </td>
                    <?php } else { ?>
                        <th><?php _e('Last log message', 'updraftplus');?>:</th>
                        <td>
                            <span id="updraft_lastlogcontainer"><?php echo htmlspecialchars(UpdraftPlus_Options::get_updraft_option('updraft_lastmessage', __('(Nothing yet logged)', 'updraftplus'))); ?></span><br>
                            <?php $this->most_recently_modified_log_link(); ?>
                        </td>
                    <?php } ?>
                </tr>
 
                <?php
                // Currently disabled - not sure who we want to show this to
                if (1==0 && !defined('UPDRAFTPLUS_NOADS_B')) {
                    $feed = $updraftplus->get_updraftplus_rssfeed();
                    if (is_a($feed, 'SimplePie')) {
                        echo '<tr><th style="vertical-align:top;">'.__('Latest UpdraftPlus.com news:', 'updraftplus').'</th><td class="updraft_simplepie">';
                        echo '<ul class="disc;">';
                        foreach ($feed->get_items(0, 5) as $item) {
                            echo '<li>';
                            echo '<a href="'.esc_attr($item->get_permalink()).'">';
                            echo htmlspecialchars($item->get_title());
                            // D, F j, Y H:i
                            echo "</a> (".htmlspecialchars($item->get_date('j F Y')).")";
                            echo '</li>';
                        }
                        echo '</ul></td></tr>';
                    }
                }
            ?>
            </table>
        <?php
    }
 
    /**
     * Output directly a link allowing download of the most recently modified log file
     */
    private function most_recently_modified_log_link() {
 
        global $updraftplus;
        list($mod_time, $log_file, $nonce) = $updraftplus->last_modified_log();
        
        ?>
            <a href="?page=updraftplus&amp;action=downloadlatestmodlog&amp;wpnonce=<?php echo wp_create_nonce('updraftplus_download'); ?>" <?php if (!$mod_time) echo 'style="display:none;"'; ?> class="updraft-log-link" onclick="event.preventDefault(); updraft_popuplog('');"><?php _e('Download most recently modified log file', 'updraftplus');?></a>
        <?php
    }
    
    public function settings_downloading_and_restoring($backup_history = array(), $return_result = false, $options = array()) {
        return $this->include_template('wp-admin/settings/downloading-and-restoring.php', $return_result, array('backup_history' => $backup_history, 'options' => $options));
    }
 
    public function settings_debugrow($head, $content) {
        echo "<tr class=\"updraft_debugrow\"><th>$head</th><td>$content</td></tr>";
    }
 
    public function settings_advanced_tools($return_instead_of_echo = false, $pass_through = array()) {
        return $this->include_template('wp-admin/advanced/advanced-tools.php', $return_instead_of_echo, $pass_through);
    }
 
    private function print_delete_old_dirs_form($include_blurb = true, $include_div = true) {
        if ($include_blurb) {
            if ($include_div) {
                echo '<div id="updraft_delete_old_dirs_pagediv" class="updated delete-old-directories">';
            }
            echo '<p>'.__('Your WordPress install has old directories from its state before you restored/migrated (technical information: these are suffixed with -old). You should press this button to delete them as soon as you have verified that the restoration worked.', 'updraftplus').'</p>';
        }
        ?>
        <form method="post" action="<?php echo esc_url(add_query_arg(array('error' => false, 'updraft_restore_success' => false, 'action' => false, 'page' => 'updraftplus'))); ?>">
            <?php wp_nonce_field('updraftplus-credentialtest-nonce'); ?>
            <input type="hidden" name="action" value="updraft_delete_old_dirs">
            <input type="submit" class="button-primary" value="<?php echo esc_attr(__('Delete Old Directories', 'updraftplus'));?>">
        </form>
        <?php
        if ($include_blurb && $include_div) echo '</div>';
    }
 
    /**
     * Return cron status information about a specified in-progress job
     *
     * @param Boolean|String $job_id - the job to get information about; or, if not specified, all jobs
     *
     * @return Array|Boolean - the requested information, or false if it was not found. Format differs depending on whether info on all jobs, or a single job, was requested.
     */
    public function get_cron($job_id = false) {
    
        $cron = get_option('cron');
        if (!is_array($cron)) $cron = array();
        if (false === $job_id) return $cron;
 
        foreach ($cron as $time => $job) {
            if (!isset($job['updraft_backup_resume'])) continue;
            foreach ($job['updraft_backup_resume'] as $hook => $info) {
                if (isset($info['args'][1]) && $job_id == $info['args'][1]) {
                    global $updraftplus;
                    $jobdata = $updraftplus->jobdata_getarray($job_id);
                    return is_array($jobdata) ? array($time, $jobdata) : false;
                }
            }
        }
    }
 
    /**
     * Print active Jobs
     *
     * @param  boolean $this_job_only A value for $this_job_only also causes something to always be returned (to allow detection of the job having started on the front-end)
     * @return [type]                 [description]
     */
    private function print_active_jobs($this_job_only = false) {
        $cron = $this->get_cron();
        $ret = '';
 
        foreach ($cron as $time => $job) {
            if (isset($job['updraft_backup_resume'])) {
                foreach ($job['updraft_backup_resume'] as $hook => $info) {
                    if (isset($info['args'][1])) {
                        $job_id = $info['args'][1];
                        if (false === $this_job_only || $job_id == $this_job_only) {
                            $ret .= $this->print_active_job($job_id, false, $time, $info['args'][0]);
                        }
                    }
                }
            }
        }
 
        // A value for $this_job_only implies that output is required
        if (false !== $this_job_only && !$ret) {
            $ret = $this->print_active_job($this_job_only);
            if ('' == $ret) {
                // The presence of the exact ID matters to the front-end - indicates that the backup job has at least begun
                $ret = '<div class="active-jobs updraft_finished" id="updraft-jobid-'.$this_job_only.'"><em>'.__('The backup has finished running', 'updraftplus').'</em> - <a class="updraft-log-link" data-jobid="'.$this_job_only.'">'.__('View Log', 'updraftplus').'</a></div>';
            }
        }
 
        return $ret;
    }
 
    private function print_active_job($job_id, $is_oneshot = false, $time = false, $next_resumption = false) {
 
        $ret = '';
 
        global $updraftplus;
        $jobdata = $updraftplus->jobdata_getarray($job_id);
 
        if (false == apply_filters('updraftplus_print_active_job_continue', true, $is_oneshot, $next_resumption, $jobdata)) return '';
 
        if (!isset($jobdata['backup_time'])) return '';
 
        $backupable_entities = $updraftplus->get_backupable_file_entities(true, true);
 
        $began_at = (isset($jobdata['backup_time'])) ? get_date_from_gmt(gmdate('Y-m-d H:i:s', (int) $jobdata['backup_time']), 'D, F j, Y H:i') : '?';
 
        $jobstatus = empty($jobdata['jobstatus']) ? 'unknown' : $jobdata['jobstatus'];
        $stage = 0;
        switch ($jobstatus) {
            // Stage 0
            case 'begun':
            $curstage = __('Backup begun', 'updraftplus');
                break;
            // Stage 1
            case 'filescreating':
            $stage = 1;
            $curstage = __('Creating file backup zips', 'updraftplus');
            if (!empty($jobdata['filecreating_substatus']) && isset($backupable_entities[$jobdata['filecreating_substatus']['e']]['description'])) {
            
                $sdescrip = preg_replace('/ \(.*\)$/', '', $backupable_entities[$jobdata['filecreating_substatus']['e']]['description']);
                if (strlen($sdescrip) > 20 && isset($jobdata['filecreating_substatus']['e']) && is_array($jobdata['filecreating_substatus']['e']) && isset($backupable_entities[$jobdata['filecreating_substatus']['e']]['shortdescription'])) $sdescrip = $backupable_entities[$jobdata['filecreating_substatus']['e']]['shortdescription'];
                $curstage .= ' ('.$sdescrip.')';
                if (isset($jobdata['filecreating_substatus']['i']) && isset($jobdata['filecreating_substatus']['t'])) {
                    $stage = min(2, 1 + ($jobdata['filecreating_substatus']['i']/max($jobdata['filecreating_substatus']['t'], 1)));
                }
            }
                break;
            case 'filescreated':
            $stage = 2;
            $curstage = __('Created file backup zips', 'updraftplus');
                break;
            // Stage 4
            case 'clouduploading':
            $stage = 4;
            $curstage = __('Uploading files to remote storage', 'updraftplus');
            if (isset($jobdata['uploading_substatus']['t']) && isset($jobdata['uploading_substatus']['i'])) {
                $t = max((int) $jobdata['uploading_substatus']['t'], 1);
                $i = min($jobdata['uploading_substatus']['i']/$t, 1);
                $p = min($jobdata['uploading_substatus']['p'], 1);
                $pd = $i + $p/$t;
                $stage = 4 + $pd;
                $curstage .= ' '.sprintf(__('(%s%%, file %s of %s)', 'updraftplus'), floor(100*$pd), $jobdata['uploading_substatus']['i']+1, $t);
            }
                break;
            case 'pruning':
            $stage = 5;
            $curstage = __('Pruning old backup sets', 'updraftplus');
                break;
            case 'resumingforerrors':
            $stage = -1;
            $curstage = __('Waiting until scheduled time to retry because of errors', 'updraftplus');
                break;
            // Stage 6
            case 'finished':
            $stage = 6;
            $curstage = __('Backup finished', 'updraftplus');
                break;
            default:
            // Database creation and encryption occupies the space from 2 to 4. Databases are created then encrypted, then the next databae is created/encrypted, etc.
            if ('dbcreated' == substr($jobstatus, 0, 9)) {
                $jobstatus = 'dbcreated';
                $whichdb = substr($jobstatus, 9);
                if (!is_numeric($whichdb)) $whichdb = 0;
                $howmanydbs = max((empty($jobdata['backup_database']) || !is_array($jobdata['backup_database'])) ? 1 : count($jobdata['backup_database']), 1);
                $perdbspace = 2/$howmanydbs;
 
                $stage = min(4, 2 + ($whichdb+2)*$perdbspace);
 
                $curstage = __('Created database backup', 'updraftplus');
 
            } elseif ('dbcreating' == substr($jobstatus, 0, 10)) {
                $whichdb = substr($jobstatus, 10);
                if (!is_numeric($whichdb)) $whichdb = 0;
                $howmanydbs = (empty($jobdata['backup_database']) || !is_array($jobdata['backup_database'])) ? 1 : count($jobdata['backup_database']);
                $perdbspace = 2/$howmanydbs;
                $jobstatus = 'dbcreating';
 
                $stage = min(4, 2 + $whichdb*$perdbspace);
 
                $curstage = __('Creating database backup', 'updraftplus');
                if (!empty($jobdata['dbcreating_substatus']['t'])) {
                    $curstage .= ' ('.sprintf(__('table: %s', 'updraftplus'), $jobdata['dbcreating_substatus']['t']).')';
                    if (!empty($jobdata['dbcreating_substatus']['i']) && !empty($jobdata['dbcreating_substatus']['a'])) {
                        $substage = max(0.001, ($jobdata['dbcreating_substatus']['i'] / max($jobdata['dbcreating_substatus']['a'], 1)));
                        $stage += $substage * $perdbspace * 0.5;
                    }
                }
            } elseif ('dbencrypting' == substr($jobstatus, 0, 12)) {
                $whichdb = substr($jobstatus, 12);
                if (!is_numeric($whichdb)) $whichdb = 0;
                $howmanydbs = (empty($jobdata['backup_database']) || !is_array($jobdata['backup_database'])) ? 1 : count($jobdata['backup_database']);
                $perdbspace = 2/$howmanydbs;
                $stage = min(4, 2 + $whichdb*$perdbspace + $perdbspace*0.5);
                $jobstatus = 'dbencrypting';
                $curstage = __('Encrypting database', 'updraftplus');
            } elseif ('dbencrypted' == substr($jobstatus, 0, 11)) {
                $whichdb = substr($jobstatus, 11);
                if (!is_numeric($whichdb)) $whichdb = 0;
                $howmanydbs = (empty($jobdata['backup_database']) || !is_array($jobdata['backup_database'])) ? 1 : count($jobdata['backup_database']);
                $jobstatus = 'dbencrypted';
                $perdbspace = 2/$howmanydbs;
                $stage = min(4, 2 + $whichdb*$perdbspace + $perdbspace);
                $curstage = __('Encrypted database', 'updraftplus');
            } else {
                $curstage = __('Unknown', 'updraftplus');
            }
        }
 
        $runs_started = (empty($jobdata['runs_started'])) ? array() : $jobdata['runs_started'];
        $time_passed = (empty($jobdata['run_times'])) ? array() : $jobdata['run_times'];
        $last_checkin_ago = -1;
        if (is_array($time_passed)) {
            foreach ($time_passed as $run => $passed) {
                if (isset($runs_started[$run])) {
                    $time_ago = microtime(true) - ($runs_started[$run] + $time_passed[$run]);
                    if ($time_ago < $last_checkin_ago || -1 == $last_checkin_ago) $last_checkin_ago = $time_ago;
                }
            }
        }
 
        $next_res_after = (int) $time-time();
        $next_res_txt = ($is_oneshot) ? '' : ' - '.sprintf(__("next resumption: %d (after %ss)", 'updraftplus'), $next_resumption, $next_res_after). ' ';
        $last_activity_txt = ($last_checkin_ago >= 0) ? ' - '.sprintf(__('last activity: %ss ago', 'updraftplus'), floor($last_checkin_ago)).' ' : '';
 
        if (($last_checkin_ago < 50 && $next_res_after>30) || $is_oneshot) {
            $show_inline_info = $last_activity_txt;
            $title_info = $next_res_txt;
        } else {
            $show_inline_info = $next_res_txt;
            $title_info = $last_activity_txt;
        }
 
        // Existence of the 'updraft-jobid-(id)' id is checked for in other places, so do not modify this
        $ret .= '<div class="job-id" id="updraft-jobid-'.$job_id.'"><span class="updraft_jobtimings next-resumption';
 
        if (!empty($jobdata['is_autobackup'])) $ret .= ' isautobackup';
 
        $ret .= '" data-jobid="'.$job_id.'" data-lastactivity="'.(int) $last_checkin_ago.'" data-nextresumption="'.$next_resumption.'" data-nextresumptionafter="'.$next_res_after.'" title="'.esc_attr(sprintf(__('Job ID: %s', 'updraftplus'), $job_id)).$title_info.'">'.$began_at.'</span> ';
 
        $ret .= $show_inline_info;
        $ret .= '- <a data-jobid="'.$job_id.'" href="'.UpdraftPlus_Options::admin_page_url().'?page=updraftplus&action=downloadlog&updraftplus_backup_nonce='.$job_id.'" class="updraft-log-link">'.__('show log', 'updraftplus').'</a>';
 
        if (!$is_oneshot) $ret .=' - <a href="#" data-jobid="'.$job_id.'" title="'.esc_attr(__('Note: the progress bar below is based on stages, NOT time. Do not stop the backup simply because it seems to have remained in the same place for a while - that is normal.', 'updraftplus')).'" class="updraft_jobinfo_delete">'.__('stop', 'updraftplus').'</a>';
 
        $ret .= apply_filters('updraft_printjob_beforewarnings', '', $jobdata, $job_id);
 
        if (!empty($jobdata['warnings']) && is_array($jobdata['warnings'])) {
            $ret .= '<ul class="disc">';
            foreach ($jobdata['warnings'] as $warning) {
                $ret .= '<li>'.sprintf(__('Warning: %s', 'updraftplus'), make_clickable(htmlspecialchars($warning))).'</li>';
            }
            $ret .= '</ul>';
        }
 
        $ret .= '<div class="curstage">';
        $ret .= htmlspecialchars($curstage);
        // we need to add this data-progress attribute in order to be able to update the progress bar in UDC
        $ret .= '<div class="updraft_percentage" data-info="'.esc_attr($curstage).'" data-progress="'.(($stage>0) ? (ceil((100/6)*$stage)) : '0').'" style="height: 100%; width:'.(($stage>0) ? (ceil((100/6)*$stage)) : '0').'%"></div>';
        $ret .= '</div></div>';
 
        $ret .= '</div>';
 
        return $ret;
 
    }
 
    private function delete_old_dirs_go($show_return = true) {
        echo $show_return ? '<h1>UpdraftPlus - '.__('Remove old directories', 'updraftplus').'</h1>' : '<h2>'.__('Remove old directories', 'updraftplus').'</h2>';
 
        if ($this->delete_old_dirs()) {
            echo '<p>'.__('Old directories successfully removed.', 'updraftplus').'</p><br>';
        } else {
            echo '<p>',__('Old directory removal failed for some reason. You may want to do this manually.', 'updraftplus').'</p><br>';
        }
        if ($show_return) echo '<b>'.__('Actions', 'updraftplus').':</b> <a href="'.UpdraftPlus_Options::admin_page_url().'?page=updraftplus">'.__('Return to UpdraftPlus Configuration', 'updraftplus').'</a>';
    }
 
    /**
     * deletes the -old directories that are created when a backup is restored.
     *
     * @return String. Can also exit (something we ought to probably review)
     */
    private function delete_old_dirs() {
        global $wp_filesystem, $updraftplus;
        $credentials = request_filesystem_credentials(wp_nonce_url(UpdraftPlus_Options::admin_page_url()."?page=updraftplus&action=updraft_delete_old_dirs", 'updraftplus-credentialtest-nonce'));
        WP_Filesystem($credentials);
        if ($wp_filesystem->errors->get_error_code()) {
            foreach ($wp_filesystem->errors->get_error_messages() as $message) show_message($message);
            exit;
        }
        // From WP_CONTENT_DIR - which contains 'themes'
        $ret = $this->delete_old_dirs_dir($wp_filesystem->wp_content_dir());
 
        $updraft_dir = $updraftplus->backups_dir_location();
        if ($updraft_dir) {
            $ret4 = $updraft_dir ? $this->delete_old_dirs_dir($updraft_dir, false) : true;
        } else {
            $ret4 = true;
        }
 
        $plugs = untrailingslashit($wp_filesystem->wp_plugins_dir());
        if ($wp_filesystem->is_dir($plugs.'-old')) {
            echo "<strong>".__('Delete', 'updraftplus').": </strong>plugins-old: ";
            if (!$wp_filesystem->delete($plugs.'-old', true)) {
                $ret3 = false;
                echo "<strong>".__('Failed', 'updraftplus')."</strong><br>";
            } else {
                $ret3 = true;
                echo "<strong>".__('OK', 'updraftplus')."</strong><br>";
            }
        } else {
            $ret3 = true;
        }
 
        return $ret && $ret3 && $ret4;
    }
 
    private function delete_old_dirs_dir($dir, $wpfs = true) {
 
        $dir = trailingslashit($dir);
 
        global $wp_filesystem, $updraftplus;
 
        if ($wpfs) {
            $list = $wp_filesystem->dirlist($dir);
        } else {
            $list = scandir($dir);
        }
        if (!is_array($list)) return false;
 
        $ret = true;
        foreach ($list as $item) {
            $name = (is_array($item)) ? $item['name'] : $item;
            if ("-old" == substr($name, -4, 4)) {
                // recursively delete
                print "<strong>".__('Delete', 'updraftplus').": </strong>".htmlspecialchars($name).": ";
 
                if ($wpfs) {
                    if (!$wp_filesystem->delete($dir.$name, true)) {
                        $ret = false;
                        echo "<strong>".__('Failed', 'updraftplus')."</strong><br>";
                    } else {
                        echo "<strong>".__('OK', 'updraftplus')."</strong><br>";
                    }
                } else {
                    if ($updraftplus->remove_local_directory($dir.$name)) {
                        echo "<strong>".__('OK', 'updraftplus')."</strong><br>";
                    } else {
                        $ret = false;
                        echo "<strong>".__('Failed', 'updraftplus')."</strong><br>";
                    }
                }
            }
        }
        return $ret;
    }
 
    /**
     * The aim is to get a directory that is writable by the webserver, because that's the only way we can create zip files
     *
     * @return Boolean|WP_Error true if successful, otherwise false or a WP_Error
     */
    private function create_backup_dir() {
 
        global $wp_filesystem, $updraftplus;
 
        if (false === ($credentials = request_filesystem_credentials(UpdraftPlus_Options::admin_page().'?page=updraftplus&action=updraft_create_backup_dir&nonce='.wp_create_nonce('create_backup_dir')))) {
            return false;
        }
 
        if (!WP_Filesystem($credentials)) {
            // our credentials were no good, ask the user for them again
            request_filesystem_credentials(UpdraftPlus_Options::admin_page().'?page=updraftplus&action=updraft_create_backup_dir&nonce='.wp_create_nonce('create_backup_dir'), '', true);
            return false;
        }
 
        $updraft_dir = $updraftplus->backups_dir_location();
 
        $default_backup_dir = $wp_filesystem->find_folder(dirname($updraft_dir)).basename($updraft_dir);
 
        $updraft_dir = ($updraft_dir) ? $wp_filesystem->find_folder(dirname($updraft_dir)).basename($updraft_dir) : $default_backup_dir;
 
        if (!$wp_filesystem->is_dir($default_backup_dir) && !$wp_filesystem->mkdir($default_backup_dir, 0775)) {
            $wperr = new WP_Error;
            if ($wp_filesystem->errors->get_error_code()) {
                foreach ($wp_filesystem->errors->get_error_messages() as $message) {
                    $wperr->add('mkdir_error', $message);
                }
                return $wperr;
            } else {
                return new WP_Error('mkdir_error', __('The request to the filesystem to create the directory failed.', 'updraftplus'));
            }
        }
 
        if ($wp_filesystem->is_dir($default_backup_dir)) {
 
            if ($updraftplus->really_is_writable($updraft_dir)) return true;
 
            @$wp_filesystem->chmod($default_backup_dir, 0775);
            if ($updraftplus->really_is_writable($updraft_dir)) return true;
 
            @$wp_filesystem->chmod($default_backup_dir, 0777);
 
            if ($updraftplus->really_is_writable($updraft_dir)) {
                echo '<p>'.__('The folder was created, but we had to change its file permissions to 777 (world-writable) to be able to write to it. You should check with your hosting provider that this will not cause any problems', 'updraftplus').'</p>';
                return true;
            } else {
                @$wp_filesystem->chmod($default_backup_dir, 0775);
                $show_dir = (0 === strpos($default_backup_dir, ABSPATH)) ? substr($default_backup_dir, strlen(ABSPATH)) : $default_backup_dir;
                return new WP_Error('writable_error', __('The folder exists, but your webserver does not have permission to write to it.', 'updraftplus').' '.__('You will need to consult with your web hosting provider to find out how to set permissions for a WordPress plugin to write to the directory.', 'updraftplus').' ('.$show_dir.')');
            }
        }
 
        return true;
    }
 
    /**
     * scans the content dir to see if any -old dirs are present
     *
     * @param  Boolean $print_as_comment Echo information in an HTML comment
     * @return Boolean
     */
    private function scan_old_dirs($print_as_comment = false) {
        global $updraftplus;
        $dirs = scandir(untrailingslashit(WP_CONTENT_DIR));
        if (!is_array($dirs)) $dirs = array();
        $dirs_u = @scandir($updraftplus->backups_dir_location());
        if (!is_array($dirs_u)) $dirs_u = array();
        foreach (array_merge($dirs, $dirs_u) as $dir) {
            if (preg_match('/-old$/', $dir)) {
                if ($print_as_comment) echo '<!--'.htmlspecialchars($dir).'-->';
                return true;
            }
        }
        // No need to scan ABSPATH - we don't backup there
        if (is_dir(untrailingslashit(WP_PLUGIN_DIR).'-old')) {
            if ($print_as_comment) echo '<!--'.htmlspecialchars(untrailingslashit(WP_PLUGIN_DIR).'-old').'-->';
            return true;
        }
        return false;
    }
 
    /**
     * Outputs html for a storage method using the parameters passed in, this version should be removed when all remote storages use the multi version
     *
     * @param String $method   a list of methods to be used when
     * @param String $header   the table header content
     * @param String $contents the table contents
     */
    public function storagemethod_row($method, $header, $contents) {
        ?>
            <tr class="updraftplusmethod <?php echo $method;?>">
                <th><?php echo $header;?></th>
                <td><?php echo $contents;?></td>
            </tr>
        <?php
    }
 
    /**
     * Outputs html for a storage method using the parameters passed in, this version of the method is compatible with multi storage options
     *
     * @param  string $classes  a list of classes to be used when
     * @param  string $header   the table header content
     * @param  string $contents the table contents
     */
    public function storagemethod_row_multi($classes, $header, $contents) {
        ?>
            <tr class="<?php echo $classes;?>">
                <th><?php echo $header;?></th>
                <td><?php echo $contents;?></td>
            </tr>
        <?php
    }
    
    /**
     * Returns html for a storage method using the parameters passed in, this version of the method is compatible with multi storage options
     *
     * @param  string $classes  a list of classes to be used when
     * @param  string $header   the table header content
     * @param  string $contents the table contents
     * @return string handlebars html template
     */
    public function get_storagemethod_row_multi_configuration_template($classes, $header, $contents) {
        return '<tr class="'.esc_attr($classes).'">
                    <th>'.$header.'</th>
                    <td>'.$contents.'</td>
                </tr>';
    }
 
    /**
     * Get HTML suitable for the admin area for the status of the last backup
     *
     * @return String
     */
    public function last_backup_html() {
 
        global $updraftplus;
 
        $updraft_last_backup = UpdraftPlus_Options::get_updraft_option('updraft_last_backup');
 
        if ($updraft_last_backup) {
 
            // Convert to GMT, then to blog time
            $backup_time = (int) $updraft_last_backup['backup_time'];
 
            $print_time = get_date_from_gmt(gmdate('Y-m-d H:i:s', $backup_time), 'D, F j, Y H:i');
 
            if (empty($updraft_last_backup['backup_time_incremental'])) {
                $last_backup_text = "<span style=\"color:".(($updraft_last_backup['success']) ? 'green' : 'black').";\">".$print_time.'</span>';
            } else {
                $inc_time = get_date_from_gmt(gmdate('Y-m-d H:i:s', $updraft_last_backup['backup_time_incremental']), 'D, F j, Y H:i');
                $last_backup_text = "<span style=\"color:".(($updraft_last_backup['success']) ? 'green' : 'black').";\">$inc_time</span> (".sprintf(__('incremental backup; base backup: %s', 'updraftplus'), $print_time).')';
            }
 
            $last_backup_text .= '<br>';
 
            // Show errors + warnings
            if (is_array($updraft_last_backup['errors'])) {
                foreach ($updraft_last_backup['errors'] as $err) {
                    $level = (is_array($err)) ? $err['level'] : 'error';
                    $message = (is_array($err)) ? $err['message'] : $err;
                    $last_backup_text .= ('warning' == $level) ? "<span style=\"color:orange;\">" : "<span style=\"color:red;\">";
                    if ('warning' == $level) {
                        $message = sprintf(__("Warning: %s", 'updraftplus'), make_clickable(htmlspecialchars($message)));
                    } else {
                        $message = htmlspecialchars($message);
                    }
                    $last_backup_text .= $message;
                    $last_backup_text .= '</span><br>';
                }
            }
 
            // Link log
            if (!empty($updraft_last_backup['backup_nonce'])) {
                $updraft_dir = $updraftplus->backups_dir_location();
 
                $potential_log_file = $updraft_dir."/log.".$updraft_last_backup['backup_nonce'].".txt";
                if (is_readable($potential_log_file)) $last_backup_text .= "<a href=\"?page=updraftplus&action=downloadlog&updraftplus_backup_nonce=".$updraft_last_backup['backup_nonce']."\" class=\"updraft-log-link\" onclick=\"event.preventDefault(); updraft_popuplog('".$updraft_last_backup['backup_nonce']."');\">".__('Download log file', 'updraftplus')."</a>";
            }
 
        } else {
            $last_backup_text = "<span style=\"color:blue;\">".__('No backup has been completed', 'updraftplus')."</span>";
        }
 
        return $last_backup_text;
 
    }
 
    /**
     * Get a list of backup intervals
     *
     * @return Array - keys are used as identifiers in the UI drop-down; values are user-displayed text describing the interval
     */
    public function get_intervals() {
        return apply_filters('updraftplus_backup_intervals', array(
            'manual' => _x("Manual", 'i.e. Non-automatic', 'updraftplus'),
            'every4hours' => sprintf(__("Every %s hours", 'updraftplus'), '4'),
            'every8hours' => sprintf(__("Every %s hours", 'updraftplus'), '8'),
            'twicedaily' => sprintf(__("Every %s hours", 'updraftplus'), '12'),
            'daily' => __("Daily", 'updraftplus'),
            'weekly' => __("Weekly", 'updraftplus'),
            'fortnightly' => __("Fortnightly", 'updraftplus'),
            'monthly' => __("Monthly", 'updraftplus')
        ));
    }
    
    public function really_writable_message($really_is_writable, $updraft_dir) {
        if ($really_is_writable) {
            $dir_info = '<span style="color:green;">'.__('Backup directory specified is writable, which is good.', 'updraftplus').'</span>';
        } else {
            $dir_info = '<span style="color:red;">';
            if (!is_dir($updraft_dir)) {
                $dir_info .= __('Backup directory specified does <b>not</b> exist.', 'updraftplus');
            } else {
                $dir_info .= __('Backup directory specified exists, but is <b>not</b> writable.', 'updraftplus');
            }
            $dir_info .= '<span class="updraft-directory-not-writable-blurb"><span class="directory-permissions"><a class="updraft_create_backup_dir" href="'.UpdraftPlus_Options::admin_page_url().'?page=updraftplus&action=updraft_create_backup_dir&nonce='.wp_create_nonce('create_backup_dir').'">'.__('Follow this link to attempt to create the directory and set the permissions', 'updraftplus').'</a></span>, '.__('or, to reset this option', 'updraftplus').' <a href="#" class="updraft_backup_dir_reset">'.__('click here', 'updraftplus').'</a>. '.__('If that is unsuccessful check the permissions on your server or change it to another directory that is writable by your web server process.', 'updraftplus').'</span>';
        }
        return $dir_info;
    }
 
    /**
     * Directly output the settings form (suitable for the admin area)
     *
     * @param Array $options - current options (passed on to the template)
     */
    public function settings_formcontents($options = array()) {
        $this->include_template('wp-admin/settings/form-contents.php', false, array('options' => $options));
    }
 
    public function get_settings_js($method_objects, $really_is_writable, $updraft_dir, $active_service) {
 
        global $updraftplus;
        
        ob_start();
        ?>
        jQuery(document).ready(function() {
            <?php
                if (!$really_is_writable) echo "jQuery('.backupdirrow').show();\n";
            ?>
            <?php
                if (!empty($active_service)) {
                    if (is_array($active_service)) {
                        foreach ($active_service as $serv) {
                            echo "jQuery('.${serv}').show();\n";
                        }
                    } else {
                        echo "jQuery('.${active_service}').show();\n";
                    }
                } else {
                    echo "jQuery('.none').show();\n";
                }
                foreach ($updraftplus->backup_methods as $method => $description) {
                    // already done: require_once(UPDRAFTPLUS_DIR.'/methods/'.$method.'.php');
                    $call_method = "UpdraftPlus_BackupModule_$method";
                    if (method_exists($call_method, 'config_print_javascript_onready')) {
                        $method_objects[$method]->config_print_javascript_onready();
                    }
                }
            ?>
        });
        <?php
        $ret = ob_get_contents();
        ob_end_clean();
        return $ret;
    }
    
    /**
     * This is the files selector widget
     *
     * @param  string  $prefix                 Prefix for the ID
     * @param  boolean $show_exclusion_options True or False for exclusion options
     * @param  boolean $include_more           $include_more can be (bool) or (string)"sometimes"
     * @return string
     */
    public function files_selector_widgetry($prefix = '', $show_exclusion_options = true, $include_more = true) {
 
        $ret = '';
 
        global $updraftplus;
        $backupable_entities = $updraftplus->get_backupable_file_entities(true, true);
        // The true (default value if non-existent) here has the effect of forcing a default of on.
        $include_more_paths = UpdraftPlus_Options::get_updraft_option('updraft_include_more_path');
        foreach ($backupable_entities as $key => $info) {
            $included = (UpdraftPlus_Options::get_updraft_option("updraft_include_$key", apply_filters("updraftplus_defaultoption_include_".$key, true))) ? 'checked="checked"' : "";
            if ('others' == $key || 'uploads' == $key) {
 
                $data_toggle_exclude_field = $show_exclusion_options ? 'data-toggle_exclude_field="'.$key.'"' : '';
            
                $ret .= '<input class="updraft_include_entity" id="'.$prefix.'updraft_include_'.$key.'" '.$data_toggle_exclude_field.' type="checkbox" name="updraft_include_'.$key.'" value="1" '.$included.'> <label '.(('others' == $key) ? 'title="'.sprintf(__('Your wp-content directory server path: %s', 'updraftplus'), WP_CONTENT_DIR).'" ' : '').' for="'.$prefix.'updraft_include_'.$key.'">'.(('others' == $key) ? __('Any other directories found inside wp-content', 'updraftplus') : htmlspecialchars($info['description'])).'</label><br>';
                
                if ($show_exclusion_options) {
                    $include_exclude = UpdraftPlus_Options::get_updraft_option('updraft_include_'.$key.'_exclude', ('others' == $key) ? UPDRAFT_DEFAULT_OTHERS_EXCLUDE : UPDRAFT_DEFAULT_UPLOADS_EXCLUDE);
 
                    $display = ($included) ? '' : 'class="updraft-hidden" style="display:none;"';
 
                    $ret .= "<div id=\"".$prefix."updraft_include_".$key."_exclude\" $display>";
 
                    $ret .= '<label for="'.$prefix.'updraft_include_'.$key.'_exclude">'.__('Exclude these:', 'updraftplus').'</label>';
 
                    $ret .= '<input title="'.__('If entering multiple files/directories, then separate them with commas. For entities at the top level, you can use a * at the start or end of the entry as a wildcard.', 'updraftplus').'" type="text" id="'.$prefix.'updraft_include_'.$key.'_exclude" name="updraft_include_'.$key.'_exclude" size="54" value="'.htmlspecialchars($include_exclude).'" />';
 
                    $ret .= '<br></div>';
                }
 
            } else {
 
                if ('more' != $key || true === $include_more || ('sometimes' === $include_more && !empty($include_more_paths))) {
                
                    $data_toggle_exclude_field = $show_exclusion_options ? 'data-toggle_exclude_field="'.$key.'"' : '';
                
                    $ret .= "<input class=\"updraft_include_entity\" $data_toggle_exclude_field id=\"".$prefix."updraft_include_$key\" type=\"checkbox\" name=\"updraft_include_$key\" value=\"1\" $included /><label for=\"".$prefix."updraft_include_$key\"".((isset($info['htmltitle'])) ? ' title="'.htmlspecialchars($info['htmltitle']).'"' : '')."> ".htmlspecialchars($info['description']);
 
                    $ret .= "</label><br>";
                    $ret .= apply_filters("updraftplus_config_option_include_$key", '', $prefix);
                }
            }
        }
 
        return $ret;
    }
 
    /**
     * Output or echo HTML for an error condition relating to a remote storage method
     *
     * @param String  $text          - the text of the message; this should already be escaped (no more is done)
     * @param String  $extraclass - a CSS class for the resulting DOM node
     * @param Integer $echo          - if set, then the results will be echoed as well as returned
     *
     * @return String - the results
     */
    public function show_double_warning($text, $extraclass = '', $echo = true) {
 
        $ret = "<div class=\"error updraftplusmethod $extraclass\"><p>$text</p></div>";
        $ret .= "<p class=\"double-warning\">$text</p>";
 
        if ($echo) echo $ret;
        return $ret;
 
    }
 
    public function optionfilter_split_every($value) {
        return max(absint($value), UPDRAFTPLUS_SPLIT_MIN);
    }
 
    /**
     * Check if curl exists; if not, print or return appropriate error messages
     *
     * @param String  $service                the service description (used only for user-visible messages - so, use the description)
     * @param Boolean $has_fallback           set as true if the lack of Curl only affects the ability to connect over SSL
     * @param String  $extraclass             an extra CSS class for any resulting message, passed on to show_double_warning()
     * @param Boolean $echo_instead_of_return whether the result should be echoed or returned
     * @return String                         any resulting message, if $echo_instead_of_return was set
     */
    public function curl_check($service, $has_fallback = false, $extraclass = '', $echo_instead_of_return = true) {
 
        $ret = '';
 
        // Check requirements
        if (!function_exists("curl_init") || !function_exists('curl_exec')) {
        
            $ret .= $this->show_double_warning('<strong>'.__('Warning', 'updraftplus').':</strong> '.sprintf(__("Your web server's PHP installation does not included a <strong>required</strong> (for %s) module (%s). Please contact your web hosting provider's support and ask for them to enable it.", 'updraftplus'), $service, 'Curl').' ', $extraclass, false);
 
        } else {
            $curl_version = curl_version();
            $curl_ssl_supported= ($curl_version['features'] & CURL_VERSION_SSL);
            if (!$curl_ssl_supported) {
                if ($has_fallback) {
                    $ret .= '<p><strong>'.__('Warning', 'updraftplus').':</strong> '.sprintf(__("Your web server's PHP/Curl installation does not support https access. Communications with %s will be unencrypted. Ask your web host to install Curl/SSL in order to gain the ability for encryption (via an add-on).", 'updraftplus'), $service).'</p>';
                } else {
                    $ret .= $this->show_double_warning('<p><strong>'.__('Warning', 'updraftplus').':</strong> '.sprintf(__("Your web server's PHP/Curl installation does not support https access. We cannot access %s without this support. Please contact your web hosting provider's support. %s <strong>requires</strong> Curl+https. Please do not file any support requests; there is no alternative.", 'updraftplus'), $service, $service).'</p>', $extraclass, false);
                }
            } else {
                $ret .= '<p><em>'.sprintf(__("Good news: Your site's communications with %s can be encrypted. If you see any errors to do with encryption, then look in the 'Expert Settings' for more help.", 'updraftplus'), $service).'</em></p>';
            }
        }
        if ($echo_instead_of_return) {
            echo $ret;
        } else {
            return $ret;
        }
    }
 
    /**
     * If $basedirs is passed as an array, then $directorieses must be too
     * Note: Reason $directorieses is being used because $directories is used within the foreach-within-a-foreach further down
     *
     * @param Array|String $directorieses List of of directories, or a single one
     * @param Array           $exclude       An exclusion array of directories
     * @param Array|String $basedirs      A list of base directories, or a single one
     * @param String       $format        Return format - 'text' or 'numeric'
     * @return String|Integer
     */
    private function recursive_directory_size($directorieses, $exclude = array(), $basedirs = '', $format = 'text') {
   
        $size = 0;
 
        if (is_string($directorieses)) {
          $basedirs = $directorieses;
          $directorieses = array($directorieses);
        }
 
        if (is_string($basedirs)) $basedirs = array($basedirs);
 
        foreach ($directorieses as $ind => $directories) {
            if (!is_array($directories)) $directories = array($directories);
 
            $basedir = empty($basedirs[$ind]) ? $basedirs[0] : $basedirs[$ind];
 
            foreach ($directories as $dir) {
                if (is_file($dir)) {
                    $size += @filesize($dir);
                } else {
                    $suffix = ('' != $basedir) ? ((0 === strpos($dir, $basedir.'/')) ? substr($dir, 1+strlen($basedir)) : '') : '';
                    $size += $this->recursive_directory_size_raw($basedir, $exclude, $suffix);
                }
            }
 
        }
 
        if ('numeric' == $format) return $size;
 
        global $updraftplus;
        return $updraftplus->convert_numeric_size_to_text($size);
 
    }
 
    private function recursive_directory_size_raw($prefix_directory, &$exclude = array(), $suffix_directory = '') {
 
        $directory = $prefix_directory.('' == $suffix_directory ? '' : '/'.$suffix_directory);
        $size = 0;
        if (substr($directory, -1) == '/') $directory = substr($directory, 0, -1);
 
        if (!file_exists($directory) || !is_dir($directory) || !is_readable($directory)) return -1;
        if (file_exists($directory.'/.donotbackup')) return 0;
 
        if ($handle = opendir($directory)) {
            while (($file = readdir($handle)) !== false) {
                if ('.' != $file && '..' != $file) {
                    $spath = ('' == $suffix_directory) ? $file : $suffix_directory.'/'.$file;
                    if (false !== ($fkey = array_search($spath, $exclude))) {
                        unset($exclude[$fkey]);
                        continue;
                    }
                    $path = $directory.'/'.$file;
                    if (is_file($path)) {
                        $size += filesize($path);
                    } elseif (is_dir($path)) {
                        $handlesize = $this->recursive_directory_size_raw($prefix_directory, $exclude, $suffix_directory.('' == $suffix_directory ? '' : '/').$file);
                        if ($handlesize >= 0) {
                            $size += $handlesize;
                        }
                    }
                }
            }
            closedir($handle);
        }
 
        return $size;
 
    }
 
    private function raw_backup_info($backup_history, $key, $nonce) {
 
        global $updraftplus;
 
        $backup = $backup_history[$key];
 
        $pretty_date = get_date_from_gmt(gmdate('Y-m-d H:i:s', (int) $key), 'M d, Y G:i');
 
        $rawbackup = "<h2 title=\"$key\">$pretty_date</h2>";
 
        if (!empty($backup['label'])) $rawbackup .= '<span class="raw-backup-info">'.$backup['label'].'</span>';
 
        $rawbackup .= '<hr><p>';
 
        $backupable_entities = $updraftplus->get_backupable_file_entities(true, true);
 
        if (!empty($nonce)) {
            $jd = $updraftplus->jobdata_getarray($nonce);
        } else {
            $jd = array();
        }
        
        $checksums = $updraftplus->which_checksums();
 
        foreach ($backupable_entities as $type => $info) {
            if (!isset($backup[$type])) continue;
 
            $rawbackup .= $updraftplus->printfile($info['description'], $backup, $type, $checksums, $jd, true);
        }
 
        $total_size = 0;
        foreach ($backup as $ekey => $files) {
            if ('db' == strtolower(substr($ekey, 0, 2)) && '-size' != substr($ekey, -5, 5)) {
                $rawbackup .= $updraftplus->printfile(__('Database', 'updraftplus'), $backup, $ekey, $checksums, $jd, true);
            }
            if (!isset($backupable_entities[$ekey]) && ('db' != substr($ekey, 0, 2) || '-size' == substr($ekey, -5, 5))) continue;
            if (is_string($files)) $files = array($files);
            foreach ($files as $findex => $file) {
                $size_key = (0 == $findex) ? $ekey.'-size' : $ekey.$findex.'-size';
                $total_size = (false === $total_size || !isset($backup[$size_key]) || !is_numeric($backup[$size_key])) ? false : $total_size + $backup[$size_key];
            }
        }
 
        $services = empty($backup['service']) ? array('none') : $backup['service'];
        if (!is_array($services)) $services = array('none');
 
        $rawbackup .= '<strong>'.__('Uploaded to:', 'updraftplus').'</strong> ';
 
        $show_services = '';
        foreach ($services as $serv) {
            if ('none' == $serv || '' == $serv) {
                $add_none = true;
            } elseif (isset($updraftplus->backup_methods[$serv])) {
                $show_services .= ($show_services) ? ', '.$updraftplus->backup_methods[$serv] : $updraftplus->backup_methods[$serv];
            } else {
                $show_services .= ($show_services) ? ', '.$serv : $serv;
            }
        }
        if ('' == $show_services && $add_none) $show_services .= __('None', 'updraftplus');
 
        $rawbackup .= $show_services;
 
        if (false !== $total_size) {
            $rawbackup .= '</p><strong>'.__('Total backup size:', 'updraftplus').'</strong> '.$updraftplus->convert_numeric_size_to_text($total_size).'<p>';
        }
        
 
        
        $rawbackup .= '</p><hr><p><pre>'.print_r($backup, true).'</p></pre>';
 
        if (!empty($jd) && is_array($jd)) {
            $rawbackup .= '<p><pre>'.print_r($jd, true).'</pre></p>';
        }
 
        return esc_attr($rawbackup);
    }
 
    /**
     * Get the HTML for the table of existing backups
     *
     * @param Array|Boolean $backup_history - a list of backups to use, or false to get the current list from the database
     *
     * @return String - HTML for the table
     */
    public function existing_backup_table($backup_history = false) {
 
        global $updraftplus;
 
        if (false === $backup_history) $backup_history = UpdraftPlus_Backup_History::get_history();
        
        if (!is_array($backup_history) || empty($backup_history)) return "<p><em>".__('You have not yet made any backups.', 'updraftplus')."</em></p>";
 
        $pass_values = array(
            'backup_history' => $backup_history,
            'updraft_dir' => $updraftplus->backups_dir_location(),
            'backupable_entities' => $updraftplus->get_backupable_file_entities(true, true)
        );
        
        return $this->include_template('wp-admin/settings/existing-backups-table.php', true, $pass_values);
    
    }
 
    private function download_db_button($bkey, $key, $esc_pretty_date, $backup, $accept = array()) {
 
        if (!empty($backup['meta_foreign']) && isset($accept[$backup['meta_foreign']])) {
            $desc_source = $accept[$backup['meta_foreign']]['desc'];
        } else {
            $desc_source = __('unknown source', 'updraftplus');
        }
 
        $ret = '';
 
        if ('db' == $bkey) {
            $dbt = empty($backup['meta_foreign']) ? esc_attr(__('Database', 'updraftplus')) : esc_attr(sprintf(__('Database (created by %s)', 'updraftplus'), $desc_source));
        } else {
            $dbt = __('External database', 'updraftplus').' ('.substr($bkey, 2).')';
        }
 
        $ret .= $this->download_button($bkey, $key, 0, null, '', $dbt, $esc_pretty_date, '0');
        
        return $ret;
    }
 
    /**
     * Go through each of the file entities
     *
     * @param Array   $backup          An array of meta information
     * @param Integer $key             Backup timestamp (epoch time)
     * @param Array   $accept          An array of values to be accepted from vaules within $backup
     * @param String  $entities        Entities to be added
     * @param String  $esc_pretty_date Whether the button needs to escape the pretty date format
     * @return String - the resulting HTML
     */
    public function download_buttons($backup, $key, $accept, &$entities, $esc_pretty_date) {
        global $updraftplus;
        $ret = '';
        $backupable_entities = $updraftplus->get_backupable_file_entities(true, true);
 
        $first_entity = true;
 
        foreach ($backupable_entities as $type => $info) {
            if (!empty($backup['meta_foreign']) && 'wpcore' != $type) continue;
 
            $ide = '';
            if ('wpcore' == $type) $wpcore_restore_descrip = $info['description'];
            if (empty($backup['meta_foreign'])) {
                $sdescrip = preg_replace('/ \(.*\)$/', '', $info['description']);
                if (strlen($sdescrip) > 20 && isset($info['shortdescription'])) $sdescrip = $info['shortdescription'];
            } else {
                $info['description'] = 'WordPress';
 
                if (isset($accept[$backup['meta_foreign']])) {
                    $desc_source = $accept[$backup['meta_foreign']]['desc'];
                    $ide .= sprintf(__('Backup created by: %s.', 'updraftplus'), $accept[$backup['meta_foreign']]['desc']).' ';
                } else {
                    $desc_source = __('unknown source', 'updraftplus');
                    $ide .= __('Backup created by unknown source (%s) - cannot be restored.', 'updraftplus').' ';
                }
 
                $sdescrip = (empty($accept[$backup['meta_foreign']]['separatedb'])) ? sprintf(__('Files and database WordPress backup (created by %s)', 'updraftplus'), $desc_source) : sprintf(__('Files backup (created by %s)', 'updraftplus'), $desc_source);
                if ('wpcore' == $type) $wpcore_restore_descrip = $sdescrip;
            }
            if (isset($backup[$type])) {
                if (!is_array($backup[$type])) $backup[$type] = array($backup[$type]);
                $howmanyinset = count($backup[$type]);
                $expected_index = 0;
                $index_missing = false;
                $set_contents = '';
                $entities .= "/$type=";
                $whatfiles = $backup[$type];
                ksort($whatfiles);
                foreach ($whatfiles as $findex => $bfile) {
                    $set_contents .= ('' == $set_contents) ? $findex : ",$findex";
                    if ($findex != $expected_index) $index_missing = true;
                    $expected_index++;
                }
                $entities .= $set_contents.'/';
                if (!empty($backup['meta_foreign'])) {
                    $entities .= '/plugins=0//themes=0//uploads=0//others=0/';
                }
                $printing_first = true;
                foreach ($whatfiles as $findex => $bfile) {
                    
                    $pdescrip = ($findex > 0) ? $sdescrip.' ('.($findex+1).')' : $sdescrip;
                    if ($printing_first) {
                        $ide .= __('Press here to download or browse', 'updraftplus').' '.strtolower($info['description']);
                    } else {
                        $ret .= '<div class="updraft-hidden" style="display:none;">';
                    }
                    if (count($backup[$type]) >0) {
                        if ($printing_first) $ide .= ' '.sprintf(__('(%d archive(s) in set).', 'updraftplus'), $howmanyinset);
                    }
                    if ($index_missing) {
                        if ($printing_first) $ide .= ' '.__('You appear to be missing one or more archives from this multi-archive set.', 'updraftplus');
                    }
 
                    if (!$first_entity) {
                    } else {
                        $first_entity = false;
                    }
 
                    $ret .= $this->download_button($type, $key, $findex, $info, $ide, $pdescrip, $esc_pretty_date, $set_contents);
 
                    if (!$printing_first) {
                        $ret .= '</div>';
                    } else {
                        $printing_first = false;
                    }
                }
            }
        }
        return $ret;
    }
 
    public function date_label($pretty_date, $key, $backup, $jobdata, $nonce, $simple_format = false) {
 
        $pretty_date = $simple_format ? $pretty_date : '<div class="clear-right">'.$pretty_date.'</div>';
 
        $ret = apply_filters('updraftplus_showbackup_date', $pretty_date, $backup, $jobdata, (int) $key, $simple_format);
        if (is_array($jobdata) && !empty($jobdata['resume_interval']) && (empty($jobdata['jobstatus']) || 'finished' != $jobdata['jobstatus'])) {
            if ($simple_format) {
                $ret .= ' '.__('(Not finished)', 'updraftplus');
            } else {
                $ret .= apply_filters('updraftplus_msg_unfinishedbackup', "<br><span title=\"".esc_attr(__('If you are seeing more backups than you expect, then it is probably because the deletion of old backup sets does not happen until a fresh backup completes.', 'updraftplus'))."\">".__('(Not finished)', 'updraftplus').'</span>', $jobdata, $nonce);
            }
        }
        return $ret;
    }
 
    public function download_button($type, $backup_timestamp, $findex, $info, $title, $pdescrip, $esc_pretty_date, $set_contents) {
    
        $ret = '';
 
        $wp_nonce = wp_create_nonce('updraftplus_download');
        
        // updraft_downloader(base, backup_timestamp, what, whicharea, set_contents, prettydate, async)
        $ret .= '<button data-wp_nonce="'.esc_attr($wp_nonce).'" data-backup_timestamp="'.esc_attr($backup_timestamp).'" data-what="'.esc_attr($type).'" data-set_contents="'.esc_attr($set_contents).'" data-prettydate="'.esc_attr($esc_pretty_date).'" type="button" class="updraft_download_button '."uddownloadform_${type}_${backup_timestamp}_${findex}".'" title="'.$title.'">'.$pdescrip.'</button>';
        // onclick="'."return updraft_downloader('uddlstatus_', '$backup_timestamp', '$type', '.ud_downloadstatus', '$set_contents', '$esc_pretty_date', true)".'"
                
        return $ret;
    }
 
    public function restore_button($backup, $key, $pretty_date, $entities = '') {
        $ret = '<div class="restore-button">';
 
        if ($entities) {
            $show_data = $pretty_date;
            if (isset($backup['native']) && false == $backup['native']) {
                $show_data .= ' '.__('(backup set imported from remote location)', 'updraftplus');
            }
 
            $ret .= '<button data-showdata="'.esc_attr($show_data).'" data-backup_timestamp="'.$key.'" data-entities="'.esc_attr($entities).'" title="'.__('After pressing this button, you will be given the option to choose which components you wish to restore', 'updraftplus').'" type="button" style="float:left; clear:none;" class="button-primary choose-components-button">'.__('Restore', 'updraftplus').'</button>';
        }
        $ret .= "</div>\n";
        return $ret;
    }
 
    /**
     * Get HTML for the 'Delete' button for a particular backup in the 'Existing Backups' tab
     *
     * @param Integer $backup_time - backup timestamp (epoch time)
     * @param String  $nonce       - backup nonce
     * @param Array      $backup       - backup information array
     *
     * @return String - the resulting HTML
     */
    public function delete_button($backup_time, $nonce, $backup) {
        $sval = (!empty($backup['service']) && 'email' != $backup['service'] && 'none' != $backup['service'] && array('email') !== $backup['service'] && array('none') !== $backup['service']) ? '1' : '0';
        return '<div class="updraftplus-remove" style="float: left; clear: none;" data-hasremote="'.$sval.'">
            <a data-hasremote="'.$sval.'" data-nonce="'.$nonce.'" data-key="'.$backup_time.'" class="no-decoration updraft-delete-link" href="#" title="'.esc_attr(__('Delete this backup set', 'updraftplus')).'">'.__('Delete', 'updraftplus').'</a>
        </div>';
    }
 
    public function log_button($backup) {
        global $updraftplus;
        $updraft_dir = $updraftplus->backups_dir_location();
        $ret = '';
        if (isset($backup['nonce']) && preg_match("/^[0-9a-f]{12}$/", $backup['nonce']) && is_readable($updraft_dir.'/log.'.$backup['nonce'].'.txt')) {
            $nval = $backup['nonce'];
            $lt = __('View Log', 'updraftplus');
            $url = esc_attr(UpdraftPlus_Options::admin_page()."?page=updraftplus&action=downloadlog&amp;updraftplus_backup_nonce=$nval");
            $ret .= <<<ENDHERE
                <div style="clear:none;" class="updraft-viewlogdiv">
                    <a class="no-decoration updraft-log-link" href="$url" data-jobid="$nval">
                        $lt
                    </a>
                    <!--
                    <form action="$url" method="get">
                        <input type="hidden" name="action" value="downloadlog" />
                        <input type="hidden" name="page" value="updraftplus" />
                        <input type="hidden" name="updraftplus_backup_nonce" value="$nval" />
                        <input type="submit" value="$lt" class="updraft-log-link" onclick="event.preventDefault(); updraft_popuplog('$nval');" />
                    </form>
                    -->
                </div>
ENDHERE;
            return $ret;
        }
    }
 
    /**
     * Carry out the restore process
     *
     * @param  Integer      $timestamp         Identifying the backup to be restored
     * @param  Array|null $continuation_data For continuing a multi-stage restore (code believed to be incomplete)
     * @return Boolean|WP_Error                    WP_Error indicates a terminal failure; false indicates not-yet complete (not necessarily terminal); true indicates complete.
     */
    private function restore_backup($timestamp, $continuation_data = null) {
 
        global $wp_filesystem, $updraftplus;
        
        @set_time_limit(UPDRAFTPLUS_SET_TIME_LIMIT);
 
        $backup_set = UpdraftPlus_Backup_History::get_history($timestamp);
        
        if (empty($backup_set)) {
            echo '<p>'.__('This backup does not exist in the backup history - restoration aborted. Timestamp:', 'updraftplus')." $timestamp</p><br>";
            return new WP_Error('does_not_exist', __('Backup does not exist in the backup history', 'updraftplus'));
        }
 
        // request_filesystem_credentials passes on fields just via hidden name/value pairs.
        // Build array of parameters to be passed via this
        $extra_fields = array();
        if (isset($_POST['updraft_restore']) && is_array($_POST['updraft_restore'])) {
            foreach ($_POST['updraft_restore'] as $entity) {
                $_POST['updraft_restore_'.$entity] = 1;
                $extra_fields[] = 'updraft_restore_'.$entity;
            }
        }
 
        if (is_array($continuation_data)) {
            foreach ($continuation_data['second_loop_entities'] as $type => $files) {
                $_POST['updraft_restore_'.$type] = 1;
                if (!in_array('updraft_restore_'.$type, $extra_fields)) $extra_fields[] = 'updraft_restore_'.$type;
            }
            if (!empty($continuation_data['restore_options'])) $restore_options = $continuation_data['restore_options'];
        }
 
        // Now make sure that updraft_restorer_ option fields get passed along to request_filesystem_credentials
        foreach ($_POST as $key => $value) {
            if (0 === strpos($key, 'updraft_restorer_')) $extra_fields[] = $key;
        }
 
        $credentials = request_filesystem_credentials(UpdraftPlus_Options::admin_page()."?page=updraftplus&action=updraft_restore&backup_timestamp=$timestamp", '', false, false, $extra_fields);
        WP_Filesystem($credentials);
        if ($wp_filesystem->errors->get_error_code()) {
            echo '<p><em><a href="'.apply_filters('updraftplus_com_link', "https://updraftplus.com/faqs/asked-ftp-details-upon-restorationmigration-updates/").'">'.__('Why am I seeing this?', 'updraftplus').'</a></em></p>';
            foreach ($wp_filesystem->errors->get_error_messages() as $message) show_message($message);
            exit;
        }
 
        // If we make it this far then WP_Filesystem has been instantiated and is functional
 
        // Set up logging
        $updraftplus->backup_time_nonce();
        $updraftplus->jobdata_set('job_type', 'restore');
        $updraftplus->jobdata_set('job_time_ms', $updraftplus->job_time_ms);
        $updraftplus->logfile_open($updraftplus->nonce);
 
        // Provide download link for the log file
        // TODO: Automatic purging of old log files
        // TODO: Provide option to auto-email the log file
 
        echo '<h1>'.__('UpdraftPlus Restoration: Progress', 'updraftplus').'</h1><div id="updraft-restore-progress">';
 
        $this->show_admin_warning('<a target="_blank" href="?action=downloadlog&page=updraftplus&updraftplus_backup_nonce='.htmlspecialchars($updraftplus->nonce).'">'.__('Follow this link to download the log file for this restoration (needed for any support requests).', 'updraftplus').'</a>');
 
        $updraft_dir = trailingslashit($updraftplus->backups_dir_location());
        $foreign_known = apply_filters('updraftplus_accept_archivename', array());
 
        $service = isset($backup_set['service']) ? $backup_set['service'] : array('none');
        if (is_string($service)) $service = array($service);
 
        // Now, need to turn any updraft_restore_<entity> fields (that came from a potential WP_Filesystem form) back into parts of the _POST array (which we want to use)
        if (empty($_POST['updraft_restore']) || (!is_array($_POST['updraft_restore']))) $_POST['updraft_restore'] = array();
 
        $entities_to_restore = array();
        foreach ($_POST['updraft_restore'] as $entity) {
            if (empty($backup_set['meta_foreign'])) {
                $entities_to_restore[$entity] = $entity;
            } else {
                if ('db' == $entity && !empty($foreign_known[$backup_set['meta_foreign']]) && !empty($foreign_known[$backup_set['meta_foreign']]['separatedb'])) {
                    $entities_to_restore[$entity] = 'db';
                } else {
                    $entities_to_restore[$entity] = 'wpcore';
                }
            }
        }
 
        foreach ($_POST as $key => $value) {
            if (0 === strpos($key, 'updraft_restore_')) {
                $nkey = substr($key, 16);
                if (!isset($entities_to_restore[$nkey])) {
                    $_POST['updraft_restore'][] = $nkey;
                    if (empty($backup_set['meta_foreign'])) {
                        $entities_to_restore[$nkey] = $nkey;
                    } else {
                        if ('db' == $entity && !empty($foreign_known[$backup_set['meta_foreign']]['separatedb'])) {
                            $entities_to_restore[$nkey] = 'db';
                        } else {
                            $entities_to_restore[$nkey] = 'wpcore';
                        }
                    }
                }
            }
        }
 
        if (0 == count($_POST['updraft_restore'])) {
            echo '<p>'.__('ABORT: Could not find the information on which entities to restore.', 'updraftplus').'</p>';
            echo '<p>'.__('If making a request for support, please include this information:', 'updraftplus').' '.count($_POST).' : '.htmlspecialchars(serialize($_POST)).'</p>';
            return new WP_Error('missing_info', 'Backup information not found');
        }
 
        $this->entities_to_restore = $entities_to_restore;
 
        set_error_handler(array($updraftplus, 'php_error'), E_ALL & ~E_STRICT);
 
        /*
        $_POST['updraft_restore'] is typically something like: array(0=>'db', 1=>'plugins', 2=>'themes'), etc.
        i.e. array ('db', 'plugins', themes')
        */
 
        if (empty($restore_options)) {
            // Gather the restore optons into one place - code after here should read the options, and not the HTTP layer
            $restore_options = array();
            if (!empty($_POST['updraft_restorer_restore_options'])) {
                parse_str(stripslashes($_POST['updraft_restorer_restore_options']), $restore_options);
            }
            $restore_options['updraft_restorer_replacesiteurl'] = empty($_POST['updraft_restorer_replacesiteurl']) ? false : true;
            $restore_options['updraft_encryptionphrase'] = empty($_POST['updraft_encryptionphrase']) ? '' : (string) stripslashes($_POST['updraft_encryptionphrase']);
            $restore_options['updraft_restorer_wpcore_includewpconfig'] = empty($_POST['updraft_restorer_wpcore_includewpconfig']) ? false : true;
            $updraftplus->jobdata_set('restore_options', $restore_options);
        }
        
        // Restore in the most helpful order
        uksort($backup_set, array($this, 'sort_restoration_entities'));
        
        // Now log
        $copy_restore_options = $restore_options;
        if (!empty($copy_restore_options['updraft_encryptionphrase'])) $copy_restore_options['updraft_encryptionphrase'] = '***';
        $updraftplus->log("Restore job started. Entities to restore: ".implode(', ', array_flip($entities_to_restore)).'. Restore options: '.json_encode($copy_restore_options));
        
        $backup_set['timestamp'] = $timestamp;
 
        $backupable_entities = $updraftplus->get_backupable_file_entities(true, true);
 
        // Allow add-ons to adjust the restore directory (but only in the case of restore - otherwise, they could just use the filter built into UpdraftPlus::get_backupable_file_entities)
        $backupable_entities = apply_filters('updraft_backupable_file_entities_on_restore', $backupable_entities, $restore_options, $backup_set);
        
        // We use a single object for each entity, because we want to store information about the backup set
        include_once(UPDRAFTPLUS_DIR.'/restorer.php');
 
        global $updraftplus_restorer;
        
        $updraftplus_restorer = new Updraft_Restorer(new Updraft_Restorer_Skin, $backup_set, false, $restore_options);
 
        $second_loop = array();
 
        echo "<h2>".__('Final checks', 'updraftplus').'</h2>';
 
        if (empty($backup_set['meta_foreign'])) {
            $entities_to_download = $entities_to_restore;
        } else {
            if (!empty($foreign_known[$backup_set['meta_foreign']]['separatedb'])) {
                $entities_to_download = array();
                if (in_array('db', $entities_to_restore)) {
                    $entities_to_download['db'] = 1;
                }
                if (count($entities_to_restore) > 1 || !in_array('db', $entities_to_restore)) {
                    $entities_to_download['wpcore'] = 1;
                }
            } else {
                $entities_to_download = array('wpcore' => 1);
            }
        }
 
        // First loop: make sure that files are present + readable; and populate array for second loop
        foreach ($backup_set as $type => $files) {
            // All restorable entities must be given explicitly, as we can store other arbitrary data in the history array
            if (!isset($backupable_entities[$type]) && 'db' != $type) continue;
            if (isset($backupable_entities[$type]['restorable']) && false == $backupable_entities[$type]['restorable']) continue;
 
            if (!isset($entities_to_download[$type])) continue;
            if ('wpcore' == $type && is_multisite() && 0 === $updraftplus_restorer->ud_backup_is_multisite) {
                echo "<p>$type: <strong>";
                $updraftplus->log(__('Skipping restoration of WordPress core when importing a single site into a multisite installation. If you had anything necessary in your WordPress directory then you will need to re-add it manually from the zip file.', 'updraftplus'), 'notice-restore');
                // TODO
                // $updraftplus->log_e('Skipping restoration of WordPress core when importing a single site into a multisite installation. If you had anything necessary in your WordPress directory then you will need to re-add it manually from the zip file.');
                echo "</strong></p>";
                continue;
            }
 
            if (is_string($files)) $files = array($files);
 
            foreach ($files as $ind => $file) {
 
                $fullpath = $updraft_dir.$file;
                $updraftplus->log(sprintf(__("Looking for %s archive: file name: %s", 'updraftplus'), $type, $file), 'notice-restore');
 
                if (is_array($continuation_data) && isset($continuation_data['second_loop_entities'][$type]) && !in_array($file, $continuation_data['second_loop_entities'][$type])) {
                    echo __('Skipping: this archive was already restored.', 'updraftplus')."<br>";
                    // Set the marker so that the existing directory isn't moved out of the way
                    $updraftplus_restorer->been_restored[$type] = true;
                    continue;
                }
 
                $this->get_remote_file($service, $file, $timestamp, true);
 
                $index = (0 == $ind) ? '' : $ind;
                // If a file size is stored in the backup data, then verify correctness of the local file
                if (isset($backup_set[$type.$index.'-size'])) {
                    $fs = $backup_set[$type.$index.'-size'];
                    $print_message = __("Archive is expected to be size:", 'updraftplus')." ".round($fs/1024, 1)." KB: ";
                    $as = @filesize($fullpath);
                    if ($as == $fs) {
                        $updraftplus->log($print_message.__('OK', 'updraftplus'), 'notice-restore');
                    } else {
                        $updraftplus->log($print_message.__('Error:', 'updraftplus')." ".__('file is size:', 'updraftplus')." ".round($as/1024)." ($fs, $as)", 'warning-restore');
                    }
                } else {
                    $updraftplus->log(__("The backup records do not contain information about the proper size of this file.", 'updraftplus'), 'notice-restore');
                }
                if (!is_readable($fullpath)) {
                    $updraftplus->log(__('Could not find one of the files for restoration', 'updraftplus')." ($file)", 'warning-restore');
                    $updraftplus->log("$file: ".__('Could not find one of the files for restoration', 'updraftplus'), 'error');
                    echo '</div>';
                    restore_error_handler();
                    return false;
                }
            }
 
            if (empty($updraftplus_restorer->ud_foreign)) {
                $types = array($type);
            } else {
                if ('db' != $type || empty($foreign_known[$updraftplus_restorer->ud_foreign]['separatedb'])) {
                    $types = array('wpcore');
                } else {
                    $types = array('db');
                }
            }
 
            foreach ($types as $check_type) {
                $info = (isset($backupable_entities[$check_type])) ? $backupable_entities[$check_type] : array();
                $val = $updraftplus_restorer->pre_restore_backup($files, $check_type, $info, $continuation_data);
                if (is_wp_error($val)) {
                    $updraftplus->log_wp_error($val);
                    foreach ($val->get_error_messages() as $msg) {
                        $updraftplus->log(__('Error:',  'updraftplus').' '.$msg, 'warning-restore');
                    }
                    foreach ($val->get_error_codes() as $code) {
                        if ('already_exists' == $code) $this->print_delete_old_dirs_form(false);
                    }
                    echo '</div>'; // close the updraft_restore_progress div even if we error
                    restore_error_handler();
                    return $val;
                } elseif (false === $val) {
                    echo '</div>'; // close the updraft_restore_progress div even if we error
                    restore_error_handler();
                    return false;
                }
            }
 
            foreach ($entities_to_restore as $entity => $via) {
                if ($via == $type) {
                    if ('wpcore' == $via && 'db' == $entity && count($files) > 1) {
                        $second_loop[$entity] = apply_filters('updraftplus_select_wpcore_file_with_db', $files, $updraftplus_restorer->ud_foreign);
                    } else {
                        $second_loop[$entity] = $files;
                    }
                }
            }
        
        }
 
        $updraftplus_restorer->delete = UpdraftPlus_Options::get_updraft_option('updraft_delete_local') ? true : false;
        if ('none' === $service || 'email' === $service || empty($service) || (is_array($service) && 1 == count($service) && (in_array('none', $service) || in_array('', $service) || in_array('email', $service))) || !empty($updraftplus_restorer->ud_foreign)) {
            if ($updraftplus_restorer->delete) $updraftplus->log_e('Will not delete any archives after unpacking them, because there was no cloud storage for this backup');
            $updraftplus_restorer->delete = false;
        }
 
        if (!empty($updraftplus_restorer->ud_foreign)) $updraftplus->log("Foreign backup; created by: ".$updraftplus_restorer->ud_foreign);
 
        // Second loop: now actually do the restoration
        uksort($second_loop, array($this, 'sort_restoration_entities'));
 
        // If continuing, then prune those already done
        if (is_array($continuation_data)) {
            foreach ($second_loop as $type => $files) {
                if (isset($continuation_data['second_loop_entities'][$type])) $second_loop[$type] = $continuation_data['second_loop_entities'][$type];
            }
        }
 
        $updraftplus->jobdata_set('second_loop_entities', $second_loop);
        $updraftplus->jobdata_set('backup_timestamp', $timestamp);
        // use a site option, as otherwise on multisite when all the array of options is updated via UpdraftPlus_Options::update_site_option(), it will over-write any restored UD options from the backup
        update_site_option('updraft_restore_in_progress', $updraftplus->nonce);
 
        foreach ($second_loop as $type => $files) {
            // Types: uploads, themes, plugins, others, db
            $info = (isset($backupable_entities[$type])) ? $backupable_entities[$type] : array();
 
            echo ('db' == $type) ? "<h2>".__('Database', 'updraftplus')."</h2>" : "<h2>".$info['description']."</h2>";
            $updraftplus->log("Entity: ".$type);
 
            if (is_string($files)) $files = array($files);
            foreach ($files as $fkey => $file) {
                $last_one = (1 == count($second_loop) && 1 == count($files));
                try {
                    $val = $updraftplus_restorer->restore_backup($file, $type, $info, $last_one);
                } catch (Exception $e) {
                    $log_message = 'Exception ('.get_class($e).') occurred during restore: '.$e->getMessage().' (Code: '.$e->getCode().', line '.$e->getLine().' in '.$e->getFile().')';
                    $display_log_message = sprintf(__('A PHP exception (%s) has occurred: %s', 'updraftplus'), get_class($e), $e->getMessage());
                    error_log($log_message);
                    // @codingStandardsIgnoreLine
                    if (function_exists('wp_debug_backtrace_summary')) $log_message .= ' Backtrace: '.wp_debug_backtrace_summary();
                    $updraftplus->log($log_message);
                    $updraftplus->log($display_log_message, 'notice-restore');
                    die();
                // @codingStandardsIgnoreLine
                } catch (Error $e) {
                    $log_message = 'PHP Fatal error ('.get_class($e).') has occurred. Error Message: '.$e->getMessage().' (Code: '.$e->getCode().', line '.$e->getLine().' in '.$e->getFile().')';
                    error_log($log_message);
                    // @codingStandardsIgnoreLine
                    if (function_exists('wp_debug_backtrace_summary')) $log_message .= ' Backtrace: '.wp_debug_backtrace_summary();
                    $updraftplus->log($log_message);
                    $display_log_message = sprintf(__('A PHP fatal error (%s) has occurred: %s', 'updraftplus'), get_class($e), $e->getMessage());
                    $updraftplus->log($display_log_message, 'notice-restore');
                    die();
                }
                if (is_wp_error($val)) {
                    $codes = $val->get_error_codes();
                    if (is_array($codes) && in_array('not_found', $codes) && !empty($updraftplus_restorer->ud_foreign) && apply_filters('updraftplus_foreign_allow_missing_entity', false, $type, $updraftplus_restorer->ud_foreign)) {
                        $updraftplus->log("Entity to move not found in this zip - but this is possible with this foreign backup type");
                    } else {
                
                        $updraftplus->log_e($val);
                        foreach ($val->get_error_messages() as $msg) {
                            $updraftplus->log(__('Error message',  'updraftplus').': '.$msg, 'notice-restore');
                        }
                        $codes = $val->get_error_codes();
                        if (is_array($codes)) {
                            foreach ($codes as $code) {
                                $data = $val->get_error_data($code);
                                if (!empty($data)) {
                                    $pdata = (is_string($data)) ? $data : serialize($data);
                                    $updraftplus->log(__('Error data:', 'updraftplus').' '.$pdata, 'warning-restore');
                                    if (false !== strpos($pdata, 'PCLZIP_ERR_BAD_FORMAT (-10)')) {
                                        echo '<a href="'.apply_filters('updraftplus_com_link', "https://updraftplus.com/faqs/error-message-pclzip_err_bad_format-10-invalid-archive-structure-mean/").'"><strong>'.__('Please consult this FAQ for help on what to do about it.', 'updraftplus').'</strong></a><br>';
                                    }
                                }
                            }
                        }
                        echo '</div>'; // close the updraft_restore_progress div even if we error
                        restore_error_handler();
                        return $val;
                    }
                } elseif (false === $val) {
                    echo '</div>'; // close the updraft_restore_progress div even if we error
                    restore_error_handler();
                    return false;
                }
                unset($files[$fkey]);
                $second_loop[$type] = $files;
                $updraftplus->jobdata_set('second_loop_entities', $second_loop);
                $updraftplus->jobdata_set('backup_timestamp', $timestamp);
 
                do_action('updraft_restored_archive', $file, $type, $val, $fkey, $timestamp);
 
            }
            unset($second_loop[$type]);
            update_site_option('updraft_restore_in_progress', $updraftplus->nonce);
            $updraftplus->jobdata_set('second_loop_entities', $second_loop);
            $updraftplus->jobdata_set('backup_timestamp', $timestamp);
        }
 
        // All done - remove
        delete_site_option('updraft_restore_in_progress');
 
        foreach (array('template', 'stylesheet', 'template_root', 'stylesheet_root') as $opt) {
            add_filter('pre_option_'.$opt, array($this, 'option_filter_'.$opt));
        }
 
        // Clear any cached pages after the restore
        $updraftplus_restorer->clear_cache();
 
        if (!function_exists('validate_current_theme')) include_once(ABSPATH.WPINC.'/themes');
 
        // Have seen a case where the current theme in the DB began with a capital, but not on disk - and this breaks migrating from Windows to a case-sensitive system
        $template = get_option('template');
        if (!empty($template) && WP_DEFAULT_THEME != $template && strtolower($template) != $template) {
 
            $theme_root = get_theme_root($template);
            $theme_root2 = get_theme_root(strtolower($template));
 
            if (!file_exists("$theme_root/$template/style.css") && file_exists("$theme_root/".strtolower($template)."/style.css")) {
                $updraftplus->log_e("Theme directory (%s) not found, but lower-case version exists; updating database option accordingly", $template);
                update_option('template', strtolower($template));
            }
 
        }
 
        if (!validate_current_theme()) {
            echo '<strong>';
            $updraftplus->log_e("The current theme was not found; to prevent this stopping the site from loading, your theme has been reverted to the default theme");
            echo '</strong>';
        }
 
        echo '</div>'; // Close the updraft_restore_progress div
 
        restore_error_handler();
        
        return true;
    }
 
    public function option_filter_template($val) {
        global $updraftplus;
        return $updraftplus->option_filter_get('template');
    }
 
    public function option_filter_stylesheet($val) {
        global $updraftplus;
        return $updraftplus->option_filter_get('stylesheet');
    }
 
    public function option_filter_template_root($val) {
        global $updraftplus;
        return $updraftplus->option_filter_get('template_root');
    }
 
    public function option_filter_stylesheet_root($val) {
        global $updraftplus;
        return $updraftplus->option_filter_get('stylesheet_root');
    }
 
    public function sort_restoration_entities($a, $b) {
        if ($a == $b) return 0;
        // Put the database first
        // Put wpcore after plugins/uploads/themes (needed for restores of foreign all-in-one formats)
        if ('db' == $a || 'wpcore' == $b) return -1;
        if ('db' == $b || 'wpcore' == $a) return 1;
        // After wpcore, next last is others
        if ('others' == $b) return -1;
        if ('others' == $a) return 1;
        // And then uploads - this is only because we want to make sure uploads is after plugins, so that we know before we get to the uploads whether the version of UD which might have to unpack them can do this new-style or not.
        if ('uploads' == $b) return -1;
        if ('uploads' == $a) return 1;
        return strcmp($a, $b);
    }
 
    public function return_array($input) {
        if (!is_array($input)) $input = array();
        return $input;
    }
    
    public function updraft_ajax_savesettings() {
        global $updraftplus;
        
        if (empty($_POST) || empty($_POST['subaction']) || 'savesettings' != $_POST['subaction'] || !isset($_POST['nonce']) || !is_user_logged_in() || !UpdraftPlus_Options::user_can_manage() || !wp_verify_nonce($_POST['nonce'], 'updraftplus-settings-nonce')) die('Security check');
 
        if (empty($_POST['settings']) || !is_string($_POST['settings'])) die('Invalid data');
 
        parse_str(stripslashes($_POST['settings']), $posted_settings);
        // We now have $posted_settings as an array
        if (!empty($_POST['updraftplus_version'])) $posted_settings['updraftplus_version'] = $_POST['updraftplus_version'];
        
        echo json_encode($this->save_settings($posted_settings));
 
        die;
    }
    
    public function updraft_ajax_importsettings() {
        global $updraftplus;
          
        if (empty($_POST) || empty($_POST['subaction']) || 'importsettings' != $_POST['subaction'] || !isset($_POST['nonce']) || !is_user_logged_in() || !UpdraftPlus_Options::user_can_manage() || !wp_verify_nonce($_POST['nonce'], 'updraftplus-settings-nonce')) die('Security check');
          
        if (empty($_POST['settings']) || !is_string($_POST['settings'])) die('Invalid data');
 
        $this->import_settings($_POST);
    }
    
    /**
     * This method handles the imported json settings it will convert them into a readable format for the existing save settings function, it will also update some of the options to match the new remote storage options format (Apr 2017)
     *
     * @param  Array $settings - The settings from the imported json file
     */
    public function import_settings($settings) {
        global $updraftplus;
 
        // A bug in UD releases around 1.12.40 - 1.13.3 meant that it was saved in URL-string format, instead of JSON
        $perhaps_not_yet_parsed = json_decode(stripslashes($settings['settings']), true);
 
        if (!is_array($perhaps_not_yet_parsed)) {
            parse_str($perhaps_not_yet_parsed, $posted_settings);
        } else {
            $posted_settings = $perhaps_not_yet_parsed;
        }
 
        if (!empty($settings['updraftplus_version'])) $posted_settings['updraftplus_version'] = $settings['updraftplus_version'];
 
        // Handle the settings name change of WebDAV and SFTP (Apr 2017) if someone tries to import an old settings to this version
        if (isset($posted_settings['updraft_webdav_settings'])) {
            $posted_settings['updraft_webdav'] = $posted_settings['updraft_webdav_settings'];
            unset($posted_settings['updraft_webdav_settings']);
        }
 
        if (isset($posted_settings['updraft_sftp_settings'])) {
            $posted_settings['updraft_sftp'] = $posted_settings['updraft_sftp_settings'];
            unset($posted_settings['updraft_sftp_settings']);
        }
 
        // We also need to wrap some of the options in the new style settings array otherwise later on we will lose the settings if this information is missing
        if (empty($posted_settings['updraft_webdav']['settings'])) $posted_settings['updraft_webdav'] = $updraftplus->wrap_remote_storage_options($posted_settings['updraft_webdav']);
        if (empty($posted_settings['updraft_googledrive']['settings'])) $posted_settings['updraft_googledrive'] = $updraftplus->wrap_remote_storage_options($posted_settings['updraft_googledrive']);
        if (empty($posted_settings['updraft_googlecloud']['settings'])) $posted_settings['updraft_googlecloud'] = $updraftplus->wrap_remote_storage_options($posted_settings['updraft_googlecloud']);
        if (empty($posted_settings['updraft_onedrive']['settings'])) $posted_settings['updraft_onedrive'] = $updraftplus->wrap_remote_storage_options($posted_settings['updraft_onedrive']);
        if (empty($posted_settings['updraft_azure']['settings'])) $posted_settings['updraft_azure'] = $updraftplus->wrap_remote_storage_options($posted_settings['updraft_azure']);
        if (empty($posted_settings['updraft_dropbox']['settings'])) $posted_settings['updraft_dropbox'] = $updraftplus->wrap_remote_storage_options($posted_settings['updraft_dropbox']);
 
        echo json_encode($this->save_settings($posted_settings));
 
        die;
    }
    
    private function backup_now_remote_message() {
        global $updraftplus;
        
        $service = $updraftplus->just_one(UpdraftPlus_Options::get_updraft_option('updraft_service'));
        if (is_string($service)) $service = array($service);
        if (!is_array($service)) $service = array();
 
        $no_remote_configured = (empty($service) || array('none') === $service || array('') === $service) ? true : false;
 
        if ($no_remote_configured) {
            return '<input type="checkbox" disabled="disabled" id="backupnow_includecloud"> <em>'.sprintf(__("Backup won't be sent to any remote storage - none has been saved in the %s", 'updraftplus'), '<a href="'.UpdraftPlus_Options::admin_page_url().'?page=updraftplus&amp;tab=settings" id="updraft_backupnow_gotosettings">'.__('settings', 'updraftplus')).'</a>. '.__('Not got any remote storage?', 'updraftplus').' <a href="'.apply_filters('updraftplus_com_link', "https://updraftplus.com/landing/vault/").'">'.__("Check out UpdraftPlus Vault.", 'updraftplus').'</a></em>';
        } else {
            return '<input type="checkbox" id="backupnow_includecloud" checked="checked"> <label for="backupnow_includecloud">'.__("Send this backup to remote storage", 'updraftplus').'</label>';
        }
    }
    
    /**
     * This method works through the passed in settings array and saves the settings to the database clearing old data and setting up a return array with content to update the page via ajax
     *
     * @param  array $settings An array of settings taking from the admin page ready to be saved to the database
     * @return array           An array response containing the status of the update along with content to be used to update the admin page.
     */
    public function save_settings($settings) {
    
        global $updraftplus;
        
        // Make sure that settings filters are registered
        UpdraftPlus_Options::admin_init();
        
        $more_files_path_updated = false;
 
        if (isset($settings['updraftplus_version']) && $updraftplus->version == $settings['updraftplus_version']) {
 
            $return_array = array('saved' => true);
            
            $add_to_post_keys = array('updraft_interval', 'updraft_interval_database', 'updraft_starttime_files', 'updraft_starttime_db', 'updraft_startday_files', 'updraft_startday_db');
            
            // If database and files are on same schedule, override the db day/time settings
            if (isset($settings['updraft_interval_database']) && isset($settings['updraft_interval_database']) && $settings['updraft_interval_database'] == $settings['updraft_interval'] && isset($settings['updraft_starttime_files'])) {
                $settings['updraft_starttime_db'] = $settings['updraft_starttime_files'];
                $settings['updraft_startday_db'] = $settings['updraft_startday_files'];
            }
            foreach ($add_to_post_keys as $key) {
                // For add-ons that look at $_POST to find saved settings, add the relevant keys to $_POST so that they find them there
                if (isset($settings[$key])) {
                    $_POST[$key] = $settings[$key];
                }
            }
 
            // Check if updraft_include_more_path is set, if it is then we need to update the page, if it's not set but there's content already in the database that is cleared down below so again we should update the page.
            $more_files_path_updated = false;
 
            // i.e. If an option has been set, or if it was currently active in the settings
            if (isset($settings['updraft_include_more_path']) || UpdraftPlus_Options::get_updraft_option('updraft_include_more_path')) {
                $more_files_path_updated = true;
            }
            
            // Wipe the extra retention rules, as they are not saved correctly if the last one is deleted
            UpdraftPlus_Options::update_updraft_option('updraft_retain_extrarules', array());
            UpdraftPlus_Options::update_updraft_option('updraft_email', array());
            UpdraftPlus_Options::update_updraft_option('updraft_report_warningsonly', array());
            UpdraftPlus_Options::update_updraft_option('updraft_report_wholebackup', array());
            UpdraftPlus_Options::update_updraft_option('updraft_extradbs', array());
            UpdraftPlus_Options::update_updraft_option('updraft_include_more_path', array());
            
            $relevant_keys = $updraftplus->get_settings_keys();
            
            if (method_exists('UpdraftPlus_Options', 'mass_options_update')) {
                $original_settings = $settings;
                $settings = UpdraftPlus_Options::mass_options_update($settings);
                $mass_updated = true;
            }
 
            foreach ($settings as $key => $value) {
 
                if (in_array($key, $relevant_keys)) {
                    if ('updraft_service' == $key && is_array($value)) {
                        foreach ($value as $subkey => $subvalue) {
                            if ('0' == $subvalue) unset($value[$subkey]);
                        }
                    }
 
                    // This flag indicates that either the stored database option was changed, or that the supplied option was changed before being stored. It isn't comprehensive - it's only used to update some UI elements with invalid input.
                    $updated = empty($mass_updated) ? (is_string($value) && UpdraftPlus_Options::get_updraft_option($key) != $value) : (is_string($value) && (!isset($original_settings[$key]) || $original_settings[$key] != $value));
                    
                    $db_updated = empty($mass_updated) ? UpdraftPlus_Options::update_updraft_option($key, $value) : true;
                    
                    // Add information on what has changed to array to loop through to update links etc.
                    // Restricting to strings for now, to prevent any unintended leakage (since this is just used for UI updating)
                    if ($updated) {
                        $value = UpdraftPlus_Options::get_updraft_option($key);
                        if (is_string($value)) $return_array['changed'][$key] = $value;
                    }
                // @codingStandardsIgnoreLine
                } else {
                    // This section is ignored by CI otherwise it will complain the ELSE is empty.
                    
                    // When last active, it was catching: option_page, action, _wpnonce, _wp_http_referer, updraft_s3_endpoint, updraft_dreamobjects_endpoint. The latter two are empty; probably don't need to be in the page at all.
                    // error_log("Non-UD key when saving from POSTed data: ".$key);
                }
            }
        } else {
            $return_array = array('saved' => false, 'error_message' => sprintf(__('UpdraftPlus seems to have been updated to version (%s), which is different to the version running when this settings page was loaded. Please reload the settings page before trying to save settings.', 'updraftplus'), $updraftplus->version));
        }
        
        // Checking for various possible messages
        $updraft_dir = $updraftplus->backups_dir_location(false);
        $really_is_writable = $updraftplus->really_is_writable($updraft_dir);
        $dir_info = $this->really_writable_message($really_is_writable, $updraft_dir);
        $button_title = esc_attr(__('This button is disabled because your backup directory is not writable (see the settings).', 'updraftplus'));
        
        $return_array['backup_now_message'] = $this->backup_now_remote_message();
        
        $return_array['backup_dir'] = array('writable' => $really_is_writable, 'message' => $dir_info, 'button_title' => $button_title);
 
        // Check if $more_files_path_updated is true, is so then there's a change and we should update the backup modal
        if ($more_files_path_updated) {
            $return_array['updraft_include_more_path'] = $this->files_selector_widgetry('backupnow_files_', false, 'sometimes');
        }
        
        // Because of the single AJAX call, we need to remove the existing UD messages from the 'all_admin_notices' action
        remove_all_actions('all_admin_notices');
        
        // Moving from 2 to 1 ajax call
        ob_start();
 
        $service = UpdraftPlus_Options::get_updraft_option('updraft_service');
        
        $this->setup_all_admin_notices_global($service);
        $this->setup_all_admin_notices_udonly($service);
        
        do_action('all_admin_notices');
        
        if (!$really_is_writable) { // Check if writable
            $this->show_admin_warning_unwritable();
        }
        
        if ($return_array['saved']) { //
            $this->show_admin_warning(__('Your settings have been saved.', 'updraftplus'), 'updated fade');
        } else {
            if (isset($return_array['error_message'])) {
                $this->show_admin_warning($return_array['error_message'], 'error');
            } else {
                $this->show_admin_warning(__('Your settings failed to save. Please refresh the settings page and try again', 'updraftplus'), 'error');
            }
        }
        
        $messages_output = ob_get_contents();
        
        ob_clean();
        
        // Backup schedule output
        $this->next_scheduled_backups_output();
        
        $scheduled_output = ob_get_clean();
        
        $return_array['messages'] = $messages_output;
        $return_array['scheduled'] = $scheduled_output;
        
        // Add the updated options to the return message, so we can update on screen
        return $return_array;
        
    }
 
    /**
     * A method to remove UpdraftPlus settings from the options table.
     *
     * @param  boolean $wipe_all_settings Set to true as default as we want to remove all options, set to false if calling from UpdraftCentral, as we do not want to remove the UpdraftCentral key or we will lose connection to the site.
     * @return boolean
     */
    public function updraft_wipe_settings($wipe_all_settings = true) {
        
        global $updraftplus;
 
        $settings = $updraftplus->get_settings_keys();
 
        // if this is false the UDC has called it we don't want to remove the UDC key other wise we will lose connection to the remote site.
        if (false == $wipe_all_settings) {
            $key = array_search('updraft_central_localkeys', $settings);
            unset($settings[$key]);
        }
 
        foreach ($settings as $s) UpdraftPlus_Options::delete_updraft_option($s);
 
        // These aren't in get_settings_keys() because they are always in the options table, regardless of context
        global $wpdb;
        $wpdb->query("DELETE FROM $wpdb->options WHERE (option_name LIKE 'updraftplus_unlocked_%' OR option_name LIKE 'updraftplus_locked_%' OR option_name LIKE 'updraftplus_last_lock_time_%' OR option_name LIKE 'updraftplus_semaphore_%' OR option_name LIKE 'updraft_jobdata_%' OR option_name LIKE 'updraft_last_scheduled_%' )");
 
        $site_options = array('updraft_oneshotnonce');
        foreach ($site_options as $s) delete_site_option($s);
 
        $this->show_admin_warning(__("Your settings have been wiped.", 'updraftplus'));
 
        return true;
    }
 
    /**
     * This get the details for updraft vault and to be used globally
     *
     * @param  string $instance_id - the instance_id of the current instance being used
     * @return object              - the UpdraftVault option setup to use the passed in instance id or if one wasn't passed then use the default set of options
     */
    public function get_updraftvault($instance_id = '') {
        global $updraftplus;
 
        $storage_objects_and_ids = $updraftplus->get_storage_objects_and_ids(array('updraftvault'));
 
        if (isset($storage_objects_and_ids['updraftvault']['instance_settings'][$instance_id])) {
            $opts = $storage_objects_and_ids['updraftvault']['instance_settings'][$instance_id];
            $vault = $storage_objects_and_ids['updraftvault']['object'];
            $vault->set_options($opts, false, $instance_id);
        } else {
            include_once(UPDRAFTPLUS_DIR.'/methods/updraftvault.php');
            $vault = new UpdraftPlus_BackupModule_updraftvault();
        }
 
        return $vault;
    }
 
    /**
     * http_get will allow the HTTP Fetch execute available in advanced tools
     *
     * @param  String  $uri  Specific URL passed to curl
     * @param  Boolean $curl True or False if cURL is to be used
     * @return String - JSON encoded results
     */
    public function http_get($uri = null, $curl = false) {
 
        if (!preg_match('/^https?/', $uri)) return json_encode(array('e' => 'Non-http URL specified'));
    
        if ($curl) {
            if (!function_exists('curl_exec')) {
                return json_encode(array('e' => 'No Curl installed'));
                die;
            }
            $ch = curl_init();
            curl_setopt($ch, CURLOPT_URL, $uri);
            curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
            curl_setopt($ch, CURLOPT_FAILONERROR, true);
            curl_setopt($ch, CURLOPT_HEADER, false);
            curl_setopt($ch, CURLOPT_VERBOSE, true);
            curl_setopt($ch, CURLOPT_STDERR, $output = fopen('php://temp', "w+"));
            $response = curl_exec($ch);
            $error = curl_error($ch);
            $getinfo = curl_getinfo($ch);
            curl_close($ch);
 
            rewind($output);
            $verb = stream_get_contents($output);
 
            $resp = array();
            if (false === $response) {
                $resp['e'] = htmlspecialchars($error);
            }
            $resp['r'] = (empty($response)) ? '' : htmlspecialchars(substr($response, 0, 2048));
 
            if (!empty($verb)) $resp['r'] = htmlspecialchars($verb)."\n\n".$resp['r'];
 
            // Extra info returned for Central
            $resp['verb'] = $verb;
            $resp['response'] = $response;
            $resp['status'] = $getinfo;
 
            return json_encode($resp);
        } else {
            $response = wp_remote_get($uri, array('timeout' => 10));
            if (is_wp_error($response)) {
                return json_encode(array('e' => htmlspecialchars($response->get_error_message())));
            }
            return json_encode(
                array(
                    'r' => wp_remote_retrieve_response_code($response).': '.htmlspecialchars(substr(wp_remote_retrieve_body($response), 0, 2048)),
                    'code' => wp_remote_retrieve_response_code($response),
                    'html_response' => htmlspecialchars(substr(wp_remote_retrieve_body($response), 0, 2048)),
                    'response' => $response
                )
            );
        }
    }
 
    /**
     * This will return all the details for raw backup and file list, in HTML format
     *
     * @param Boolean $no_pre_tags - if set, then <pre></pre> tags will be removed from the output
     *
     * @return String
     */
    public function show_raw_backups($no_pre_tags = false) {
        global $updraftplus;
        
        $response = array();
        
        $response['html'] = '<h3 id="ud-debuginfo-rawbackups">'.__('Known backups (raw)', 'updraftplus').'</h3><pre>';
        ob_start();
        $history = UpdraftPlus_Backup_History::get_history();
        var_dump($history);
        $response["html"] .= ob_get_clean();
        $response['html'] .= '</pre>';
 
        $response['html'] .= '<h3 id="ud-debuginfo-files">'.__('Files', 'updraftplus').'</h3><pre>';
        $updraft_dir = $updraftplus->backups_dir_location();
        $raw_output = array();
        $d = dir($updraft_dir);
        while (false !== ($entry = $d->read())) {
            $fp = $updraft_dir.'/'.$entry;
            $mtime = filemtime($fp);
            if (is_dir($fp)) {
                $size = '       d';
            } elseif (is_link($fp)) {
                $size = '       l';
            } elseif (is_file($fp)) {
                $size = sprintf("%8.1f", round(filesize($fp)/1024, 1)).' '.gmdate('r', $mtime);
            } else {
                $size = '       ?';
            }
            if (preg_match('/^log\.(.*)\.txt$/', $entry, $lmatch)) $entry = '<a target="_top" href="?action=downloadlog&amp;page=updraftplus&amp;updraftplus_backup_nonce='.htmlspecialchars($lmatch[1]).'">'.$entry.'</a>';
            $raw_output[$mtime] = empty($raw_output[$mtime]) ? sprintf("%s %s\n", $size, $entry) : $raw_output[$mtime].sprintf("%s %s\n", $size, $entry);
        }
        @$d->close();
        krsort($raw_output, SORT_NUMERIC);
 
        foreach ($raw_output as $line) {
            $response['html'] .= $line;
        }
 
        $response['html'] .= '</pre>';
 
        $response['html'] .= '<h3 id="ud-debuginfo-options">'.__('Options (raw)', 'updraftplus').'</h3>';
        $opts = $updraftplus->get_settings_keys();
        asort($opts);
        // <tr><th>'.__('Key', 'updraftplus').'</th><th>'.__('Value', 'updraftplus').'</th></tr>
        $response['html'] .= '<table><thead></thead><tbody>';
        foreach ($opts as $opt) {
            $response['html'] .= '<tr><td>'.htmlspecialchars($opt).'</td><td>'.htmlspecialchars(print_r(UpdraftPlus_Options::get_updraft_option($opt), true)).'</td>';
        }
        $response['html'] .= '</tbody></table>';
 
        ob_start();
        do_action('updraftplus_showrawinfo');
        $response['html'] .= ob_get_clean();
 
        if (true == $no_pre_tags) {
            $response['html'] = str_replace('<pre>', '', $response['html']);
            $response['html'] = str_replace('</pre>', '', $response['html']);
        }
 
        return $response;
    }
 
    /**
     * This will call any wp_action
     *
     * @param  Array            $data                      The array of data with the vaules for wpaction
     * @param  Callable|Boolean    $close_connection_callable A callable to call to close the browser connection, or true for a default suitable for internal use, or false for none
     * @return Array - results
     */
    public function call_wp_action($data = null, $close_connection_callable = false) {
        global $updraftplus;
 
        ob_start();
 
        $res = '<em>Request received: </em>';
 
        if (preg_match('/^([^:]+)+.*)$/', stripslashes($data['wpaction']), $matches)) {
            $action = $matches[1];
            if (null === ($args = json_decode($matches[2], true))) {
                $res .= "The parameters (should be JSON) could not be decoded";
                $action = false;
            } else {
                if (is_string($args)) $args = array($args);
                $res .= "Will despatch action: ".htmlspecialchars($action).", parameters: ".htmlspecialchars(implode(',', $args));
            }
        } else {
            $action = $data['wpaction'];
            $res .= "Will despatch action: ".htmlspecialchars($action).", no parameters";
        }
 
        $ret = ob_get_clean();
 
        // Need to add this as the close browser should only work for UDP
        if ($close_connection_callable) {
            if (is_callable($close_connection_callable)) {
                call_user_func($close_connection_callable, array('r' => $res));
            } else {
                $updraftplus->close_browser_connection(json_encode(array('r' => $res)));
            }
        }
 
        if (!empty($action)) {
            if (!empty($args)) {
                ob_start();
                $returned = do_action_ref_array($action, $args);
                $output = ob_get_clean();
                $res .= " - do_action_ref_array Trigger ";
            } else {
                ob_start();
                do_action($action);
                $output = ob_get_contents();
                ob_end_clean();
                $res .= " - do_action Trigger ";
            }
        }
        $response['response']     = $res;
        $response['log']         = $output;
 
        // Check if response is empty
        if (!empty($returned)) $response['status'] = $returned;
 
        return $response;
    }
 
    /**
     * Enqueue JSTree JavaScript and CSS, taking into account whether it is already enqueued, and current debug settings
     */
    public function enqueue_jstree() {
 
        static $already_enqueued = false;
        if ($already_enqueued) return;
        
        $already_enqueued = true;
        $jstree_enqueue_version = (defined('SCRIPT_DEBUG') && SCRIPT_DEBUG) ? '3.3'.'.'.time() : '3.3';
        $min_or_not = (defined('SCRIPT_DEBUG') && SCRIPT_DEBUG) ? '' : '.min';
        
        wp_enqueue_script('jstree', UPDRAFTPLUS_URL.'/includes/jstree/jstree'.$min_or_not.'.js', array('jquery'), $jstree_enqueue_version);
        wp_enqueue_style('jstree', UPDRAFTPLUS_URL.'/includes/jstree/themes/default/style'.$min_or_not.'.css', array(), $jstree_enqueue_version);
    }
}

Âñåãî çàïèñåé: 600 | Çàðåãèñòð. 06-10-2004 | Îòïðàâëåíî: 15:13 06-01-2018 | Èñïðàâëåíî: vell, 15:52 06-01-2018
Îòêðûòü íîâóþ òåìó     Íàïèñàòü îòâåò â ýòó òåìó

Íà ïåðâóþ ñòðàíèöóê ýòîìó ñîîáùåíèþê ïîñëåäíåìó ñîîáùåíèþ

Êîìïüþòåðíûé ôîðóì Ru.Board » Èíòåðíåò » Â ïîìîùü âåáìàñòåðó » WordPress


Ðåêëàìà íà ôîðóìå Ru.Board.

Powered by Ikonboard "v2.1.7b" © 2000 Ikonboard.com
Modified by Ru.B0ard
© Ru.B0ard 2000-2024

BitCoin: 1NGG1chHtUvrtEqjeerQCKDMUi6S6CG4iC

Ðåéòèíã.ru