registry = PUM_Upgrade_Registry::instance(); } /** * Register core upgrade processes. * * @param PUM_Upgrade_Registry $registry The upgrade registry instance. * @return void */ public function register_processes( PUM_Upgrade_Registry $registry ) { // v1.7 Upgrades $registry->add_upgrade( 'core-v1_7-popups', [ 'rules' => [ version_compare( self::$initial_version, '1.7', '<' ), ], 'class' => 'PUM_Upgrade_v1_7_Popups', 'file' => Popup_Maker::$DIR . 'includes/batch/upgrade/class-upgrade-v1_7-popups.php', ] ); $registry->add_upgrade( 'core-v1_7-settings', [ 'rules' => [ version_compare( self::$initial_version, '1.7', '<' ), ], 'class' => 'PUM_Upgrade_v1_7_Settings', 'file' => Popup_Maker::$DIR . 'includes/batch/upgrade/class-upgrade-v1_7-settings.php', ] ); $registry->add_upgrade( 'core-v1_8-themes', [ 'rules' => [ $this->needs_v1_8_theme_upgrade(), ], 'class' => 'PUM_Upgrade_v1_8_Themes', 'file' => Popup_Maker::$DIR . 'includes/batch/upgrade/class-upgrade-v1_8-themes.php', ] ); } /** * Check if v1.8 theme upgrade is needed. * * @return bool True if upgrade is needed, false otherwise. */ public function needs_v1_8_theme_upgrade() { if ( pum_has_completed_upgrade( 'core-v1_8-themes' ) ) { return false; } $needs_upgrade = get_transient( 'pum_needs_1_8_theme_upgrades' ); if ( false === $needs_upgrade ) { $query = new WP_Query( [ 'post_type' => 'popup_theme', 'post_status' => 'any', 'fields' => 'ids', // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query 'meta_query' => [ 'relation' => 'OR', [ 'key' => 'popup_theme_data_version', 'compare' => 'NOT EXISTS', 'value' => 'deprecated', // Here for WP 3.9 or less. ], [ 'key' => 'popup_theme_data_version', 'compare' => '<', 'value' => 3, ], ], ] ); $needs_upgrade = $query->post_count; } if ( $needs_upgrade <= 0 ) { pum_set_upgrade_complete( 'core-v1_8-themes' ); delete_transient( 'pum_needs_1_8_theme_upgrades' ); return false; } set_transient( 'pum_needs_1_8_theme_upgrades', $needs_upgrade ); return (bool) $needs_upgrade; } /** * Registers a new upgrade routine. * * @param string $upgrade_id Upgrade ID. * @param array $args Arguments for registering a new upgrade routine. { * @type bool[] $rules Upgrade rules. * @type string $class Upgrade class name. * @type string $file Upgrade file path. * } * * @return bool True if the upgrade routine was added, otherwise false. */ public function add_routine( $upgrade_id, $args ) { return $this->registry->add_upgrade( $upgrade_id, $args ); } /** * Displays upgrade notices. * * @return void */ public function upgrade_notices() { if ( ! $this->has_uncomplete_upgrades() || ! current_user_can( 'manage_options' ) ) { return; } // Enqueue admin JS for the batch processor. wp_enqueue_script( 'pum-admin-batch' ); wp_enqueue_style( 'pum-admin-batch' ); ?>
render_upgrade_notice(); ?> render_form(); ?>
Updated alerts list. */ public function upgrade_alert( $alerts = [] ) { if ( ! $this->has_uncomplete_upgrades() || ! current_user_can( 'manage_options' ) ) { return $alerts; } // Enqueue admin JS for the batch processor. wp_enqueue_script( 'pum-admin-batch' ); wp_enqueue_style( 'pum-admin-batch' ); ob_start(); $this->render_upgrade_notice(); $this->render_form(); $html = ob_get_clean(); $alerts[] = [ 'code' => 'upgrades_required', 'type' => 'warning', 'html' => $html, 'priority' => 1000, 'dismissible' => false, 'global' => true, ]; return $alerts; } /** * Renders the upgrade notification message. * * Message only, no form. * * @return void */ public function render_upgrade_notice() { $resume_upgrade = $this->maybe_resume_upgrade(); ?>

$this->get_current_upgrade_id(), 'step' => 1, ]; $resume_upgrade = $this->maybe_resume_upgrade(); if ( $resume_upgrade && is_array( $resume_upgrade ) ) { $args = wp_parse_args( $resume_upgrade, $args ); } ?>

registry->get( $upgrade_id ); } /** * Get all upgrade routines. * * Note: Unfiltered. * * @return array Associative array of upgrade ID => upgrade data. */ public function get_routines() { return $this->registry->get_upgrades(); } /** * Adds an upgrade action to the completed upgrades array. * * @param string $upgrade_id The action to add to the completed upgrades array. * * @return bool True if the upgrade was successfully marked complete, false otherwise. */ public function set_upgrade_complete( $upgrade_id = '' ) { if ( empty( $upgrade_id ) ) { return false; } $completed_upgrades = $this->get_completed_upgrades(); if ( ! in_array( $upgrade_id, $completed_upgrades, true ) ) { $completed_upgrades[] = $upgrade_id; do_action( 'pum_set_upgrade_complete', $upgrade_id ); } // Remove any blanks, and only show uniques $completed_upgrades = array_unique( array_values( $completed_upgrades ) ); return update_option( 'pum_completed_upgrades', $completed_upgrades ); } /** * Get's the array of completed upgrade actions. * * @return array The array of completed upgrade IDs. */ public function get_completed_upgrades() { $completed_upgrades = get_option( 'pum_completed_upgrades' ); if ( false === $completed_upgrades ) { $completed_upgrades = []; update_option( 'pum_completed_upgrades', $completed_upgrades ); } return get_option( 'pum_completed_upgrades', [] ); } /** * Check if the upgrade routine has been run for a specific action. * * @param string $upgrade_id The upgrade action to check completion for. * * @return bool True if the action has been completed, false otherwise. */ public function has_completed_upgrade( $upgrade_id = '' ) { if ( empty( $upgrade_id ) ) { return false; } $completed_upgrades = $this->get_completed_upgrades(); return in_array( $upgrade_id, $completed_upgrades, true ); } /** * Check if there are uncompleted upgrades available. * * @return bool True if upgrades are needed, false otherwise. */ public function has_uncomplete_upgrades() { return (bool) count( $this->get_uncompleted_upgrades() ); } /** * Returns array of uncompleted upgrades. * * This doesn't return an upgrade if: * - It was previously complete. * - If any false values in the upgrades $rules array are found. * * @return array Associative array of upgrade ID => upgrade data. */ public function get_uncompleted_upgrades() { $required_upgrades = $this->get_routines(); foreach ( $required_upgrades as $upgrade_id => $upgrade ) { // If the upgrade has already completed or one of the rules failed remove it from the list. if ( $this->has_completed_upgrade( $upgrade_id ) || in_array( false, $upgrade['rules'], true ) ) { unset( $required_upgrades[ $upgrade_id ] ); } } return $required_upgrades; } /** * Handles Ajax for processing an upgrade request. * * @return void */ public function process_upgrade_request() { $upgrade_id = isset( $_REQUEST['upgrade_id'] ) ? sanitize_key( $_REQUEST['upgrade_id'] ) : false; if ( ! $upgrade_id && ! $this->has_uncomplete_upgrades() ) { wp_send_json_error( [ 'error' => __( 'A batch process ID must be present to continue.', 'popup-maker' ), ] ); } // Nonce. if ( ! check_ajax_referer( 'pum_upgrade_ajax_nonce', 'nonce' ) ) { wp_send_json_error( [ 'error' => __( 'You do not have permission to initiate this request. Contact an administrator for more information.', 'popup-maker' ), ] ); } if ( ! $upgrade_id ) { $upgrade_id = $this->get_current_upgrade_id(); } $step = ! empty( $_REQUEST['step'] ) ? absint( $_REQUEST['step'] ) : 1; /** * Instantiate the upgrade class. * * @var PUM_Interface_Batch_Process|PUM_Interface_Batch_PrefetchProcess $upgrade */ $upgrade = $this->get_upgrade( $upgrade_id, $step ); if ( false === $upgrade ) { wp_send_json_error( [ 'error' => sprintf( /* translators: 1: Batch process ID. */ __( '%s is an invalid batch process ID.', 'popup-maker' ), esc_html( $upgrade_id ) ), ] ); } /** * Garbage collect any old temporary data in the case step is 1. * Here to prevent case ajax passes step 1 without resetting process counts. */ $first_step = $step < 2; if ( $first_step ) { $upgrade->finish(); } $using_prefetch = ( $upgrade instanceof PUM_Interface_Batch_PrefetchProcess ); // Handle pre-fetching data. if ( $using_prefetch ) { // Initialize any data needed to process a step. $data = isset( $_REQUEST['form'] ) ? sanitize_key( $_REQUEST['form'] ) : []; $upgrade->init( $data ); $upgrade->pre_fetch(); } /** @var int|string|WP_Error $step */ $step = $upgrade->process_step(); if ( ! is_wp_error( $step ) ) { $response_data = [ 'step' => $step, 'next' => null, ]; // Finish and set the status flag if done. if ( 'done' === $step ) { $response_data['done'] = true; $response_data['message'] = $upgrade->get_message( 'done' ); // Once all calculations have finished, run cleanup. $upgrade->finish(); // Set the upgrade complete. pum_set_upgrade_complete( $upgrade_id ); if ( $this->has_uncomplete_upgrades() ) { // Since the other was complete return the next (now current) upgrade_id. $response_data['next'] = $this->get_current_upgrade_id(); } } else { $response_data['done'] = false; $response_data['message'] = $first_step ? $upgrade->get_message( 'start' ) : ''; $response_data['percentage'] = $upgrade->get_percentage_complete(); } wp_send_json_success( $response_data ); } else { wp_send_json_error( $step ); } } /** * Returns the first key in the uncompleted upgrades. * * @return string|null The upgrade ID or null if no upgrades pending. */ public function get_current_upgrade_id() { $upgrades = $this->get_uncompleted_upgrades(); reset( $upgrades ); return key( $upgrades ); } /** * Returns the current upgrade processor instance. * * @return PUM_Interface_Batch_Process|PUM_Interface_Batch_PrefetchProcess|false False if no upgrade found. */ public function get_current_upgrade() { $upgrade_id = $this->get_current_upgrade_id(); return $this->get_upgrade( $upgrade_id ); } /** * Gets the upgrade process object. * * @param string $upgrade_id The upgrade identifier. * @param int $step The current step number. * * @return PUM_Interface_Batch_Process|PUM_Interface_Batch_PrefetchProcess|false The upgrade processor instance or false if not found. */ public function get_upgrade( $upgrade_id = '', $step = 1 ) { $upgrade = $this->registry->get( $upgrade_id ); if ( ! $upgrade ) { return false; } $class = isset( $upgrade['class'] ) ? sanitize_text_field( $upgrade['class'] ) : ''; $class_file = isset( $upgrade['file'] ) ? $upgrade['file'] : ''; if ( ! class_exists( $class ) && ! empty( $class_file ) && file_exists( $class_file ) ) { require_once $class_file; } else { wp_send_json_error( [ 'error' => sprintf( /* translators: 1: Batch process ID. */ __( 'An invalid file path is registered for the %1$s batch process handler.', 'popup-maker' ), "{$upgrade_id}" ), ] ); } if ( empty( $class ) || ! class_exists( $class ) ) { wp_send_json_error( [ 'error' => sprintf( /* translators: 1: Class name, 2: Batch process ID. */ __( '%1$s is an invalid handler for the %2$s batch process. Please try again.', 'popup-maker' ), "{$class}", "{$upgrade_id}" ), ] ); } /** * @var PUM_Interface_Batch_Process|PUM_Interface_Batch_PrefetchProcess */ return new $class( $step ); } /** * Add upgrades tab to tools page if there are upgrades available. * * @param array $tabs Existing tabs array where key is tab ID and value is tab label. * * @return array Updated tabs array. */ public function tools_page_tabs( $tabs = [] ) { if ( $this->has_uncomplete_upgrades() ) { $tabs['upgrades'] = __( 'Upgrades', 'popup-maker' ); } return $tabs; } /** * Renders upgrade form on the tools page upgrade tab. * * @return void */ public function tools_page_tab_content() { if ( ! $this->has_uncomplete_upgrades() ) { esc_html_e( 'No upgrades currently required.', 'popup-maker' ); return; } // Enqueue admin JS for the batch processor. wp_enqueue_script( 'pum-admin-batch' ); wp_enqueue_style( 'pum-admin-batch' ); $this->render_upgrade_notice(); $this->render_form(); } }