- WordPress core y plugins - Tema Twenty Twenty-Four configurado - Plugin allow-unfiltered-html.php simplificado - .gitignore configurado para excluir wp-config.php y uploads 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
426 lines
12 KiB
PHP
Executable File
426 lines
12 KiB
PHP
Executable File
<?php
|
|
/**
|
|
* Background Updater
|
|
*
|
|
* @version 1.7.4
|
|
*/
|
|
|
|
defined( 'ABSPATH' ) || exit;
|
|
if ( ! class_exists( 'WP_Async_Request', false ) ) {
|
|
include_once dirname( dirname( __FILE__ ) ) . '/libraries/wp-async-request.php';
|
|
}
|
|
|
|
if ( ! class_exists( 'WP_Background_Process', false ) ) {
|
|
include_once dirname( dirname( __FILE__ ) ) . '/libraries/wp-background-process.php';
|
|
}
|
|
if ( ! class_exists( 'WooFunnels_Background_Updater' ) ) {
|
|
/**
|
|
* WooFunnels_Background_Updater Class.
|
|
* Based on WC_Background_Updater concept
|
|
*/
|
|
#[AllowDynamicProperties]
|
|
class WooFunnels_Background_Updater extends WP_Background_Process {
|
|
|
|
const MAX_SAME_OFFSET_THRESHOLD = 5;
|
|
|
|
protected $prefix = 'bwf_1';
|
|
protected $action = 'updater';
|
|
|
|
/**
|
|
* Initiate new background process.
|
|
*
|
|
* WooFunnels_Background_Updater constructor.
|
|
*/
|
|
public function __construct() {
|
|
parent::__construct();
|
|
}
|
|
|
|
/**
|
|
* Schedule cron healthcheck.
|
|
*
|
|
* @param array $schedules Schedules.
|
|
*
|
|
* @return array
|
|
*/
|
|
public function schedule_cron_healthcheck( $schedules ) {
|
|
$interval = apply_filters( $this->identifier . '_cron_interval', 5 );
|
|
|
|
if ( property_exists( $this, 'cron_interval' ) ) {
|
|
$interval = apply_filters( $this->identifier . '_cron_interval', $this->cron_interval );
|
|
}
|
|
|
|
// Adds every 5 minutes to the existing schedules.
|
|
$schedules[ $this->identifier . '_cron_interval' ] = array(
|
|
'interval' => MINUTE_IN_SECONDS * $interval,
|
|
/* translators: %d: interval */
|
|
'display' => sprintf( __( 'Every %d minutes', 'woocommerce' ), $interval ), // phpcs:ignore WordPress.WP.I18n.TextDomainMismatch
|
|
);
|
|
|
|
return $schedules;
|
|
}
|
|
|
|
/**
|
|
* Handle cron healthcheck
|
|
*
|
|
* Restart the background process if not already running
|
|
* and data exists in the queue.
|
|
*/
|
|
public function handle_cron_healthcheck() {
|
|
if ( $this->is_process_running() ) {
|
|
|
|
// Background process already running.
|
|
return;
|
|
}
|
|
|
|
if ( $this->is_queue_empty() ) {
|
|
// No data to process.
|
|
$this->clear_scheduled_event();
|
|
BWF_Logger::get_instance()->log( 'Scheduled event cleared as queue is empty.', 'woofunnels_indexing' );
|
|
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* We are saving the last 5 offset value, due to any specific reason if last 5 offsets are same then it might be the time to kill the process.
|
|
*/
|
|
$offsets = $this->get_last_offsets();
|
|
if ( self::MAX_SAME_OFFSET_THRESHOLD === count( $offsets ) ) {
|
|
$unique = array_unique( $offsets );
|
|
if ( 1 === count( $unique ) ) {
|
|
$this->kill_process();
|
|
BWF_Logger::get_instance()->log( sprintf( 'Offset is stuck from last %d cron jobs, terminating the process.', self::MAX_SAME_OFFSET_THRESHOLD ), 'woofunnels_indexing' );
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
$this->manage_last_offsets();
|
|
BWF_Logger::get_instance()->log( 'Cron started again!!', 'woofunnels_indexing' );
|
|
|
|
/**
|
|
* Everything looks good, lets roll the indexing
|
|
*/
|
|
|
|
$this->handle();
|
|
}
|
|
|
|
/**
|
|
* Overriding parent protected function publically to use outside this class
|
|
* @return bool
|
|
*/
|
|
public function is_process_running() {
|
|
return parent::is_process_running();
|
|
}
|
|
|
|
/**
|
|
* Is queue empty.
|
|
*
|
|
* @return bool
|
|
*/
|
|
protected function is_queue_empty() {
|
|
global $wpdb;
|
|
|
|
$table = $wpdb->options;
|
|
$column = 'option_name';
|
|
|
|
if ( is_multisite() ) {
|
|
$table = $wpdb->sitemeta;
|
|
$column = 'meta_key';
|
|
}
|
|
|
|
$key = $wpdb->esc_like( $this->identifier . '_batch_' ) . '%';
|
|
$count = $wpdb->get_var( $wpdb->prepare( "SELECT COUNT(*) FROM {$table} WHERE {$column} LIKE %s", $key ) ); // @codingStandardsIgnoreLine.
|
|
|
|
return ! ( $count > 0 );
|
|
}
|
|
|
|
public function get_last_offsets() {
|
|
return get_option( '_bwf_last_offsets', array() );
|
|
}
|
|
|
|
/**
|
|
* Kill process.
|
|
*
|
|
* Stop processing queue items, clear cronjob and delete all batches.
|
|
*/
|
|
public function kill_process() {
|
|
$this->kill_process_safe();
|
|
WooFunnels_Dashboard::$classes['WooFunnels_DB_Updater']->set_upgrade_state( '1' );
|
|
|
|
}
|
|
|
|
public function kill_process_safe() {
|
|
if ( ! $this->is_queue_empty() ) {
|
|
$this->delete_all_batches();
|
|
wp_clear_scheduled_hook( $this->cron_hook_identifier );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Delete all batches.
|
|
*
|
|
* @return WooFunnels_Background_Updater
|
|
*/
|
|
public function delete_all_batches() {
|
|
global $wpdb;
|
|
|
|
$table = $wpdb->options;
|
|
$column = 'option_name';
|
|
|
|
if ( is_multisite() ) {
|
|
$table = $wpdb->sitemeta;
|
|
$column = 'meta_key';
|
|
}
|
|
|
|
$key = $wpdb->esc_like( $this->identifier . '_batch_' ) . '%';
|
|
$wpdb->query( $wpdb->prepare( "DELETE FROM {$table} WHERE {$column} LIKE %s", $key ) ); // @codingStandardsIgnoreLine.
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Manage last 5 offsets
|
|
*/
|
|
public function manage_last_offsets() {
|
|
$offsets = $this->get_last_offsets();
|
|
$current_offset = get_option( '_bwf_offset', 0 );
|
|
if ( self::MAX_SAME_OFFSET_THRESHOLD === count( $offsets ) ) {
|
|
$offsets = array_map( function ( $key ) use ( $offsets ) {
|
|
return isset( $offsets[ $key + 1 ] ) ? $offsets[ $key + 1 ] : 0;
|
|
}, array_keys( $offsets ) );
|
|
|
|
$offsets[ self::MAX_SAME_OFFSET_THRESHOLD - 1 ] = $current_offset;
|
|
} else {
|
|
$offsets[ count( $offsets ) ] = $current_offset;
|
|
}
|
|
|
|
$this->update_last_offsets( $offsets );
|
|
|
|
}
|
|
|
|
public function update_last_offsets( $offsets ) {
|
|
update_option( '_bwf_last_offsets', $offsets );
|
|
}
|
|
|
|
/**
|
|
* Handle.
|
|
*
|
|
* Pass each queue item to the task handler, while remaining
|
|
* within server memory and time limit constraints.
|
|
*/
|
|
protected function handle() {
|
|
$this->lock_process();
|
|
|
|
do {
|
|
$batch = $this->get_batch();
|
|
|
|
foreach ( $batch->data as $key => $value ) {
|
|
$task = $this->task( $value );
|
|
|
|
if ( false !== $task ) {
|
|
$batch->data[ $key ] = $task;
|
|
} else {
|
|
unset( $batch->data[ $key ] );
|
|
}
|
|
|
|
if ( $this->batch_limit_exceeded() ) {
|
|
// Batch limits reached.
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Update or delete current batch.
|
|
if ( ! empty( $batch->data ) ) {
|
|
$this->update( $batch->key, $batch->data );
|
|
} else {
|
|
$this->delete( $batch->key );
|
|
}
|
|
} while ( ! $this->batch_limit_exceeded() && ! $this->is_queue_empty() );
|
|
|
|
$this->unlock_process();
|
|
|
|
// Start next batch or complete process.
|
|
if ( ! $this->is_queue_empty() ) {
|
|
$this->dispatch();
|
|
} else {
|
|
$this->complete();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get batch.
|
|
*
|
|
* @return stdClass Return the first batch from the queue.
|
|
*/
|
|
protected function get_batch() {
|
|
global $wpdb;
|
|
|
|
$table = $wpdb->options;
|
|
$column = 'option_name';
|
|
$key_column = 'option_id';
|
|
$value_column = 'option_value';
|
|
|
|
if ( is_multisite() ) {
|
|
$table = $wpdb->sitemeta;
|
|
$column = 'meta_key';
|
|
$key_column = 'meta_id';
|
|
$value_column = 'meta_value';
|
|
}
|
|
|
|
$key = $wpdb->esc_like( $this->identifier . '_batch_' ) . '%';
|
|
$query = $wpdb->get_row( $wpdb->prepare( "SELECT * FROM {$table} WHERE {$column} LIKE %s ORDER BY {$key_column} ASC LIMIT 1", $key ) ); // @codingStandardsIgnoreLine.
|
|
|
|
$batch = new stdClass();
|
|
$batch->key = $query->$column;
|
|
$batch->data = array_filter( (array) maybe_unserialize( $query->$value_column ) );
|
|
|
|
return $batch;
|
|
}
|
|
|
|
/**
|
|
* Task
|
|
*
|
|
* Override this method to perform any actions required on each
|
|
* queue item. Return the modified item for further processing
|
|
* in the next pass through. Or, return false to remove the
|
|
* item from the queue.
|
|
*
|
|
* @param string $callback Update callback function.
|
|
*
|
|
* @return string|bool
|
|
* @SuppressWarnings(PHPMD.DevelopmentCodeFragment)
|
|
* @SuppressWarnings(PHPMD.ElseExpression)
|
|
*/
|
|
protected function task( $callback ) {
|
|
|
|
$result = false;
|
|
if ( is_callable( $callback ) ) {
|
|
BWF_Logger::get_instance()->log( 'Running the callback: ' . print_r( $callback, true ), 'woofunnels_indexing' ); // phpcs:ignore WordPress.PHP.DevelopmentFunctions.error_log_print_r
|
|
$result = (bool) call_user_func( $callback );
|
|
|
|
if ( $result ) {
|
|
/**sleep( 5 );*/
|
|
BWF_Logger::get_instance()->log( "Result: $result Need to run again the callback: " . print_r( $callback, true ), 'woofunnels_indexing' ); // phpcs:ignore WordPress.PHP.DevelopmentFunctions.error_log_print_r
|
|
} else {
|
|
BWF_Logger::get_instance()->log( "Result: $result Finished running the callback: " . print_r( $callback, true ), 'woofunnels_indexing' ); // phpcs:ignore WordPress.PHP.DevelopmentFunctions.error_log_print_r
|
|
}
|
|
} else {
|
|
BWF_Logger::get_instance()->log( "Result: $result Could not find the callback: " . print_r( $callback, true ), 'woofunnels_indexing' ); // phpcs:ignore WordPress.PHP.DevelopmentFunctions.error_log_print_r
|
|
}
|
|
|
|
return $result ? $callback : false;
|
|
}
|
|
|
|
/**
|
|
* See if the batch limit has been exceeded.
|
|
*
|
|
* @return bool
|
|
*/
|
|
protected function batch_limit_exceeded() {
|
|
return $this->time_exceeded() || $this->memory_exceeded();
|
|
}
|
|
|
|
/**
|
|
* Memory exceeded
|
|
*
|
|
* Ensures the batch process never exceeds 90%
|
|
* of the maximum WordPress memory.
|
|
*
|
|
* @return bool
|
|
*/
|
|
protected function memory_exceeded() {
|
|
$memory_limit = $this->get_memory_limit() * 0.9; // 90% of max memory
|
|
$current_memory = memory_get_usage( true );
|
|
$return = false;
|
|
|
|
if ( $current_memory >= $memory_limit ) {
|
|
$return = true;
|
|
}
|
|
|
|
return apply_filters( $this->identifier . '_memory_exceeded', $return );
|
|
}
|
|
|
|
/**
|
|
* Get memory limit.
|
|
*
|
|
* @return int
|
|
*/
|
|
protected function get_memory_limit() {
|
|
if ( function_exists( 'ini_get' ) ) {
|
|
$memory_limit = ini_get( 'memory_limit' );
|
|
} else {
|
|
// Sensible default.
|
|
$memory_limit = '128M';
|
|
}
|
|
|
|
if ( ! $memory_limit || - 1 === intval( $memory_limit ) ) {
|
|
// Unlimited, set to 32GB.
|
|
$memory_limit = '32G';
|
|
}
|
|
|
|
return wp_convert_hr_to_bytes( $memory_limit );
|
|
}
|
|
|
|
/**
|
|
* Complete
|
|
*
|
|
* Override if applicable, but ensure that the below actions are
|
|
* performed, or, call parent::complete().
|
|
*/
|
|
protected function complete() {
|
|
|
|
update_option( '_bwf_offset', 0 );
|
|
|
|
BWF_Logger::get_instance()->log( 'Background scanning completed for indexing order and creating updating contacts and customers.', 'woofunnels_indexing' );
|
|
do_action( 'bwf_order_index_completed' );
|
|
parent::complete();
|
|
}
|
|
|
|
/**
|
|
* Is the updater running?
|
|
*
|
|
* @return boolean
|
|
*/
|
|
public function is_updating() {
|
|
return false === $this->is_queue_empty();
|
|
}
|
|
|
|
public function maybe_re_dispatch_background_process() {
|
|
if ( 3 !== absint( WooFunnels_Dashboard::$classes['WooFunnels_DB_Updater']->get_upgrade_state() ) ) {
|
|
return;
|
|
}
|
|
if ( $this->is_queue_empty() ) {
|
|
return;
|
|
}
|
|
if ( $this->is_process_running() ) {
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* We are saving the last 5 offset value, due to any specific reason if last 5 offsets are same then it might be the time to kill the process.
|
|
*/
|
|
$offsets = $this->get_last_offsets();
|
|
if ( self::MAX_SAME_OFFSET_THRESHOLD === count( $offsets ) ) {
|
|
$unique = array_unique( $offsets );
|
|
if ( 1 === count( $unique ) ) {
|
|
$this->kill_process();
|
|
BWF_Logger::get_instance()->log( sprintf( 'Offset is stuck from last %d attempts, terminating the process.', self::MAX_SAME_OFFSET_THRESHOLD ), 'woofunnels_indexing' );
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
$this->manage_last_offsets();
|
|
$this->dispatch();
|
|
}
|
|
|
|
/**
|
|
* Schedule fallback event.
|
|
*/
|
|
protected function schedule_event() {
|
|
if ( ! wp_next_scheduled( $this->cron_hook_identifier ) ) {
|
|
wp_schedule_event( time() + 10, $this->cron_interval_identifier, $this->cron_hook_identifier );
|
|
}
|
|
}
|
|
}
|
|
} |