PluginProbe ʕ •ᴥ•ʔ
Elementor Website Builder – more than just a page builder / 3.27.0-beta1
Elementor Website Builder – more than just a page builder v3.27.0-beta1
4.1.1 4.1.0 4.1.0-beta3 4.1.0-dev3 4.0.9 4.1.0-beta2 4.1.0-dev2 4.0.8 4.1.0-beta1 4.1.0-dev1 4.0.7 4.0.6 4.0.5 4.0.4 4.0.3 3.22.0-dev1 4.0.0-beta3 3.22.0-dev2 4.0.0-beta4 3.22.0-dev3 4.0.0-beta5 3.22.0-dev4 4.0.0-dev1 3.22.0-dev5 4.0.0-dev2 3.22.0-dev6 4.0.0-dev3 3.22.1 4.0.0-dev4 3.22.2 4.0.0-dev5 3.22.3 4.0.1 3.23.0 4.0.2 3.23.0-beta1 3.23.0-beta2 3.23.0-beta3 3.23.0-beta4 3.23.0-beta5 3.23.0-beta6 3.23.0-dev1 3.23.0-dev2 3.23.0-dev3 3.23.0-dev4 3.23.0-dev5 3.23.0-dev6 3.23.1 3.23.2 3.23.3 3.23.4 3.24.0 3.24.0-beta1 3.24.0-beta2 3.24.0-beta3 3.24.0-dev1 3.24.0-dev2 3.24.0-dev3 3.24.1 3.24.2 3.24.3 3.24.4 3.24.5 3.24.6 3.24.7 3.24.8 3.25.0 3.25.0-beta1 3.25.0-beta2 3.25.0-beta3 3.25.0-dev1 3.25.0-dev2 3.25.0-dev3 3.25.1 3.25.10 3.25.11 3.25.2 3.25.3 3.25.4 3.25.5 3.25.6 3.25.7 3.25.8 3.25.9 3.26.0 3.26.0-beta1 3.26.0-beta2 3.26.0-beta3 3.26.0-beta4 3.26.0-beta5 3.26.0-dev1 3.26.0-dev2 3.26.0-dev3 3.26.0-dev4 3.26.0-dev5 3.26.1 3.26.2 3.26.3 3.26.4 3.26.5 3.27.0 3.27.0-beta1 3.27.0-beta2 3.27.0-dev1 3.27.0-dev2 3.27.1 3.27.2 3.27.3 3.27.4 3.27.5 3.27.6 3.27.7 3.28.0 3.28.0-beta1 3.28.0-beta2 3.28.0-beta3 3.28.0-dev1 3.28.0-dev2 3.28.0-dev3 3.28.1 3.28.2 3.28.3 3.28.4 3.29.0 3.29.0-beta1 trunk 3.29.0-beta2 3.0.0 3.29.0-beta3 3.0.1 3.29.0-beta4 3.0.10 3.29.0-dev1 3.0.11 3.29.0-dev2 3.0.12 3.29.0-dev3 3.0.13 3.29.0-dev4 3.0.14 3.29.1 3.0.15 3.29.2 3.0.16 3.3.0 3.0.2 3.3.1 3.0.3 3.30.0 3.0.4 3.30.0-beta1 3.0.5 3.30.0-beta2 3.0.6 3.30.0-beta3 3.0.7 3.30.0-dev1 3.0.8 3.30.0-dev2 3.0.8.1 3.30.0-dev3 3.0.9 3.30.1 3.1.0 3.30.2 3.1.0-beta1 3.30.3 3.1.0-beta2 3.30.4 3.1.0-beta3 3.31.0 3.1.0-beta4 3.31.0-beta1 3.1.0-dev1 3.31.0-beta2 3.1.0-dev2 3.31.0-dev1 3.1.0-dev3 3.31.0-dev2 3.1.1 3.31.1 3.1.2 3.31.2 3.1.3 3.31.3 3.1.4 3.31.4 3.10.0 3.31.5 3.10.0-dev1 3.32.0 3.10.1 3.32.0-beta1 3.10.2 3.32.0-beta2 3.11.0 3.32.0-beta3 3.11.0-beta1 3.32.0-dev1 3.11.0-beta2 3.32.0-dev2 3.11.0-beta3 3.32.0-dev3 3.11.0-dev1 3.32.1 3.11.0-dev2 3.32.2 3.11.0-dev3 3.32.3 3.11.1 3.32.4 3.11.2 3.32.5 3.11.3 3.33.0 3.11.4 3.33.0-beta1 3.11.5 3.33.0-beta2 3.12.0 3.33.0-beta3 3.12.1 3.33.0-beta4 3.12.2 3.33.0-dev1 3.13.0 3.33.0-dev2 3.13.0-beta1 3.33.0-dev3 3.13.0-beta2 3.33.0-dev4 3.13.0-beta3 3.33.1 3.13.0-dev3 3.33.2 3.13.0-dev4 3.33.3 3.13.1 3.33.4 3.13.2 3.33.5 3.13.3 3.33.6 3.13.4 3.34.0 3.14.0 3.34.0-beta1 3.14.0-beta1 3.34.0-beta2 3.14.0-beta2 3.34.0-beta3 3.14.0-beta3 3.34.0-dev1 3.14.0-beta4 3.34.0-dev2 3.14.0-beta5 3.34.1 3.14.1 3.34.2 3.15.0 3.34.3 3.15.1 3.34.4 3.15.2 3.35.0 3.15.3 3.35.0-beta1 3.16.0 3.35.0-beta2 3.16.0-beta3 3.35.0-beta3 3.16.0-beta4 3.35.0-beta4 3.16.0-dev1 3.35.0-dev1 3.16.0-dev2 3.35.0-dev2 3.16.1 3.35.0-dev3 3.16.2 3.35.0-dev4 3.16.3 3.35.1 3.16.4 3.35.2 3.16.5 3.35.3 3.16.6 3.35.4 3.17.0 3.35.5 3.17.0-dev2 3.35.6 3.17.0-dev3 3.35.7 3.17.0-dev4 3.35.8 3.17.1 3.35.9 3.17.2 3.4.0 3.17.3 3.4.0-dev7 3.18.0 3.4.0-dev8 3.18.0-beta1 3.4.0-dev9 3.18.0-beta2 3.4.1 3.18.0-beta3 3.4.2 3.18.0-beta4 3.4.3 3.18.0-dev1 3.4.4 3.18.1 3.4.5 3.18.2 3.4.6 3.18.3 3.4.7 3.19.0 3.4.8 3.19.0-beta1 3.5.0 3.19.0-beta2 3.5.0-beta1 3.19.0-beta3 3.5.0-beta2 3.19.0-beta4 3.5.0-beta3 3.19.0-beta5 3.5.0-beta4 3.19.0-beta6 3.5.0-beta5 3.19.0-dev1 3.5.0-beta7 3.19.0-dev2 3.5.0-beta8 3.19.0-dev3 3.5.0-dev8 3.19.0-dev4 3.5.0-dev9 3.19.0-dev5 3.5.1 3.19.0-dev6 3.5.2 3.19.1 3.5.3 3.19.2 3.5.4 3.19.3 3.5.5 3.19.4 3.5.6 3.2.0 3.6.0 3.2.1 3.6.0-beta1 3.2.2 3.6.0-beta2 3.2.3 3.6.0-beta3 3.2.4 3.6.0-beta4 3.2.5 3.6.0-beta5 3.20.0 3.6.0-dev1 3.20.0-beta1 3.6.0-dev10 3.20.0-beta2 3.6.1 3.20.0-beta3 3.6.2 3.20.0-beta4 3.6.3 3.20.0-dev1 3.6.4 3.20.0-dev2 3.6.5 3.20.0-dev3 3.6.6 3.20.0-dev4 3.6.7 3.20.1 3.6.8 3.20.2 3.7.0 3.20.3 3.7.0-beta1 3.20.4 3.7.0-beta2 3.21.0 3.7.0-beta3 3.21.0-beta1 3.7.0-beta4 3.21.0-beta2 3.7.0-dev1 3.21.0-beta3 3.7.1 3.21.0-dev1 3.7.2 3.21.0-dev2 3.7.3 3.21.0-dev3 3.7.4 3.21.1 3.7.5 3.21.2 3.7.6 3.21.3 3.7.7 3.21.4 3.7.8 3.21.5 3.8.0 3.21.6 3.8.0-beta1 3.21.7 3.8.0-beta2 3.21.8 3.8.0-beta3 3.22.0 3.8.1 3.22.0-beta1 3.9.0 3.22.0-beta2 3.9.1 3.22.0-beta3 3.9.2 3.22.0-beta4 4.0.0 3.22.0-beta5 4.0.0-beta1 3.22.0-beta6 4.0.0-beta2
elementor / core / base / background-task.php
elementor / core / base Last commit date
background-process 1 year ago elements-iteration-actions 1 year ago providers 1 year ago traits 1 year ago app.php 3 years ago background-task-manager.php 3 years ago background-task.php 1 year ago base-object.php 3 years ago db-upgrades-manager.php 3 years ago document.php 1 year ago module.php 1 year ago
background-task.php
386 lines
1 <?php
2 namespace Elementor\Core\Base;
3
4 use Elementor\Plugin;
5 use Elementor\Core\Base\BackgroundProcess\WP_Background_Process;
6
7 /**
8 * Based on https://github.com/woocommerce/woocommerce/blob/master/includes/abstracts/class-wc-background-process.php
9 * & https://github.com/woocommerce/woocommerce/blob/master/includes/class-wc-background-updater.php
10 */
11
12 defined( 'ABSPATH' ) || exit;
13
14 /**
15 * WC_Background_Process class.
16 */
17 abstract class Background_Task extends WP_Background_Process {
18 protected $current_item;
19
20 /**
21 * Dispatch updater.
22 *
23 * Updater will still run via cron job if this fails for any reason.
24 */
25 public function dispatch() {
26 $dispatched = parent::dispatch();
27
28 if ( is_wp_error( $dispatched ) ) {
29 wp_die( esc_html( $dispatched ) );
30 }
31 }
32
33 public function query_col( $sql ) {
34 global $wpdb;
35
36 // Add Calc.
37 $item = $this->get_current_item();
38 if ( empty( $item['total'] ) ) {
39 $sql = preg_replace( '/^SELECT/', 'SELECT SQL_CALC_FOUND_ROWS', $sql );
40 }
41
42 // Add offset & limit.
43 $sql = preg_replace( '/;$/', '', $sql );
44 $sql .= ' LIMIT %d, %d;';
45
46 $results = $wpdb->get_col( $wpdb->prepare( $sql, $this->get_current_offset(), $this->get_limit() ) ); // phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared
47
48 if ( ! empty( $results ) ) {
49 $this->set_total();
50 }
51
52 return $results;
53 }
54
55 public function should_run_again( $updated_rows ) {
56 return count( $updated_rows ) === $this->get_limit();
57 }
58
59 public function get_current_offset() {
60 $limit = $this->get_limit();
61 return ( $this->current_item['iterate_num'] - 1 ) * $limit;
62 }
63
64 public function get_limit() {
65 return $this->manager->get_query_limit();
66 }
67
68 public function set_total() {
69 global $wpdb;
70
71 if ( empty( $this->current_item['total'] ) ) {
72 $total_rows = $wpdb->get_var( 'SELECT FOUND_ROWS();' );
73 $total_iterates = ceil( $total_rows / $this->get_limit() );
74 $this->current_item['total'] = $total_iterates;
75 }
76 }
77
78 /**
79 * Complete
80 *
81 * Override if applicable, but ensure that the below actions are
82 * performed, or, call parent::complete().
83 */
84 protected function complete() {
85 $this->manager->on_runner_complete( true );
86
87 parent::complete();
88 }
89
90 public function continue_run() {
91 // Used to fire an action added in WP_Background_Process::_construct() that calls WP_Background_Process::handle_cron_healthcheck().
92 // This method will make sure the database updates are executed even if cron is disabled. Nothing will happen if the updates are already running.
93 do_action( $this->cron_hook_identifier );
94 }
95
96 /**
97 * @return mixed
98 */
99 public function get_current_item() {
100 return $this->current_item;
101 }
102
103 /**
104 * Get batch.
105 *
106 * @return \stdClass Return the first batch from the queue.
107 */
108 protected function get_batch() {
109 $batch = parent::get_batch();
110 $batch->data = array_filter( (array) $batch->data );
111
112 return $batch;
113 }
114
115 /**
116 * Handle cron healthcheck
117 *
118 * Restart the background process if not already running
119 * and data exists in the queue.
120 */
121 public function handle_cron_healthcheck() {
122 if ( $this->is_process_running() ) {
123 // Background process already running.
124 return;
125 }
126
127 if ( $this->is_queue_empty() ) {
128 // No data to process.
129 $this->clear_scheduled_event();
130 return;
131 }
132
133 $this->handle();
134 }
135
136 /**
137 * Schedule fallback event.
138 */
139 protected function schedule_event() {
140 if ( ! wp_next_scheduled( $this->cron_hook_identifier ) ) {
141 wp_schedule_event( time() + 10, $this->cron_interval_identifier, $this->cron_hook_identifier );
142 }
143 }
144
145 /**
146 * Is the updater running?
147 *
148 * @return boolean
149 */
150 public function is_running() {
151 return false === $this->is_queue_empty();
152 }
153
154 /**
155 * See if the batch limit has been exceeded.
156 *
157 * @return bool
158 */
159 protected function batch_limit_exceeded() {
160 return $this->time_exceeded() || $this->memory_exceeded();
161 }
162
163 /**
164 * Handle.
165 *
166 * Pass each queue item to the task handler, while remaining
167 * within server memory and time limit constraints.
168 */
169 protected function handle() {
170 $this->manager->on_runner_start();
171
172 $this->lock_process();
173
174 do {
175 $batch = $this->get_batch();
176
177 foreach ( $batch->data as $key => $value ) {
178 $task = $this->task( $value );
179
180 if ( false !== $task ) {
181 $batch->data[ $key ] = $task;
182 } else {
183 unset( $batch->data[ $key ] );
184 }
185
186 if ( $this->batch_limit_exceeded() ) {
187 // Batch limits reached.
188 break;
189 }
190 }
191
192 // Update or delete current batch.
193 if ( ! empty( $batch->data ) ) {
194 $this->update( $batch->key, $batch->data );
195 } else {
196 $this->delete( $batch->key );
197 }
198 } while ( ! $this->batch_limit_exceeded() && ! $this->is_queue_empty() );
199
200 $this->unlock_process();
201
202 // Start next batch or complete process.
203 if ( ! $this->is_queue_empty() ) {
204 $this->dispatch();
205 } else {
206 $this->complete();
207 }
208 }
209
210 /**
211 * Use the protected `is_process_running` method as a public method.
212 * @return bool
213 */
214 public function is_process_locked() {
215 return $this->is_process_running();
216 }
217
218 public function handle_immediately( $callbacks ) {
219 $this->manager->on_runner_start();
220
221 $this->lock_process();
222
223 foreach ( $callbacks as $callback ) {
224 $item = [
225 'callback' => $callback,
226 ];
227
228 do {
229 $item = $this->task( $item );
230 } while ( $item );
231 }
232
233 $this->unlock_process();
234 }
235
236 /**
237 * Task
238 *
239 * Override this method to perform any actions required on each
240 * queue item. Return the modified item for further processing
241 * in the next pass through. Or, return false to remove the
242 * item from the queue.
243 *
244 * @param array $item
245 *
246 * @return array|bool
247 */
248 protected function task( $item ) {
249 $result = false;
250
251 if ( ! isset( $item['iterate_num'] ) ) {
252 $item['iterate_num'] = 1;
253 }
254
255 $logger = Plugin::$instance->logger->get_logger();
256 $callback = $this->format_callback_log( $item );
257
258 if ( is_callable( $item['callback'] ) ) {
259 $progress = '';
260
261 if ( 1 < $item['iterate_num'] ) {
262 if ( empty( $item['total'] ) ) {
263 $progress = sprintf( '(x%s)', $item['iterate_num'] );
264 } else {
265 $percent = ceil( $item['iterate_num'] / ( $item['total'] / 100 ) );
266 $progress = sprintf( '(%s of %s, %s%%)', $item['iterate_num'], $item['total'], $percent );
267 }
268 }
269
270 $logger->info( sprintf( '%s Start %s', $callback, $progress ) );
271
272 $this->current_item = $item;
273
274 $result = (bool) call_user_func( $item['callback'], $this );
275
276 // get back the updated item.
277 $item = $this->current_item;
278 $this->current_item = null;
279
280 if ( $result ) {
281 if ( empty( $item['total'] ) ) {
282 $logger->info( sprintf( '%s callback needs to run again', $callback ) );
283 } elseif ( 1 === $item['iterate_num'] ) {
284 $logger->info( sprintf( '%s callback needs to run more %d times', $callback, $item['total'] - $item['iterate_num'] ) );
285 }
286
287 $item['iterate_num']++;
288 } else {
289 $logger->info( sprintf( '%s Finished', $callback ) );
290 }
291 } else {
292 $logger->notice( sprintf( 'Could not find %s callback', $callback ) );
293 }
294
295 return $result ? $item : false;
296 }
297
298 /**
299 * Schedule cron healthcheck.
300 *
301 * @param array $schedules Schedules.
302 * @return array
303 */
304 public function schedule_cron_healthcheck( $schedules ) {
305 $interval = apply_filters( $this->identifier . '_cron_interval', 5 );
306
307 // Adds every 5 minutes to the existing schedules.
308 $schedules[ $this->identifier . '_cron_interval' ] = [
309 'interval' => MINUTE_IN_SECONDS * $interval,
310 'display' => sprintf(
311 /* translators: %d: Interval in minutes. */
312 esc_html__( 'Every %d minutes', 'elementor' ),
313 $interval
314 ),
315 ];
316
317 return $schedules;
318 }
319
320 /**
321 * See if the batch limit has been exceeded.
322 *
323 * @return bool
324 */
325 public function is_memory_exceeded() {
326 return $this->memory_exceeded();
327 }
328
329 /**
330 * Delete all batches.
331 *
332 * @return self
333 */
334 public function delete_all_batches() {
335 global $wpdb;
336
337 $table = $wpdb->options;
338 $column = 'option_name';
339
340 if ( is_multisite() ) {
341 $table = $wpdb->sitemeta;
342 $column = 'meta_key';
343 }
344
345 $key = $wpdb->esc_like( $this->identifier . '_batch_' ) . '%';
346
347 $wpdb->query( $wpdb->prepare( "DELETE FROM {$table} WHERE {$column} LIKE %s", $key ) ); // @codingStandardsIgnoreLine.
348
349 return $this;
350 }
351
352 /**
353 * Kill process.
354 *
355 * Stop processing queue items, clear cronjob and delete all batches.
356 */
357 public function kill_process() {
358 if ( ! $this->is_queue_empty() ) {
359 $this->delete_all_batches();
360 wp_clear_scheduled_hook( $this->cron_hook_identifier );
361 }
362 }
363
364 public function set_current_item( $item ) {
365 $this->current_item = $item;
366 }
367
368 protected function format_callback_log( $item ) {
369 return implode( '::', (array) $item['callback'] );
370 }
371
372 /**
373 * @var \Elementor\Core\Base\Background_Task_Manager
374 */
375 protected $manager;
376
377 public function __construct( $manager ) {
378 $this->manager = $manager;
379 // Uses unique prefix per blog so each blog has separate queue.
380 $this->prefix = 'elementor_' . get_current_blog_id();
381 $this->action = $this->manager->get_action();
382
383 parent::__construct();
384 }
385 }
386