oont-contents/plugins/wp-optimize/includes/class-updraft-smush-manager.php
2025-04-06 08:34:48 +02:00

1759 lines
58 KiB
PHP

<?php
/**
* Extends the generic task manager to manage smush related queues
*/
if (!defined('ABSPATH')) die('Access denied.');
if (!class_exists('Updraft_Task_Manager_1_4')) require_once(WPO_PLUGIN_MAIN_PATH . 'vendor/team-updraft/common-libs/src/updraft-tasks/class-updraft-task-manager.php');
if (!class_exists('Updraft_Smush_Manager')) :
class Updraft_Smush_Manager extends Updraft_Task_Manager_1_4 {
static protected $_instance = null;
/**
* Options used for smush jobs
*
* @var array
*/
public $options;
/**
* The service provider to use
*
* @var string
*/
public $webservice;
/**
* The logger for this instance
*
* @var mixed
*/
public $logger;
/**
* Require task-level locking
*
* @var integer
*/
protected $use_per_task_lock = 60;
/**
* The Task Manager constructor
*/
public function __construct() {
parent::__construct();
$this->commands = new Updraft_Smush_Manager_Commands($this);
$this->options = WP_Optimize()->get_options();
// we set default options when compression server is false - it means that options was not saved before
if (!$this->options->get_option('compression_server')) {
$this->set_default_options();
}
$this->webservice = $this->options->get_option('compression_server', 'resmushit');
// Ensure the saved service is valid
if (!in_array($this->webservice, $this->get_allowed_services())) {
$this->webservice = $this->get_default_webservice();
}
$this->logger = new Updraft_File_Logger($this->get_logfile_path());
$this->add_logger($this->logger);
add_action('wp_ajax_updraft_smush_ajax', array($this, 'updraft_smush_ajax'));
add_action('admin_enqueue_scripts', array($this, 'admin_enqueue_scripts'), 9);
add_action('elementor/editor/before_enqueue_scripts', array($this, 'admin_enqueue_scripts'));
add_action('add_attachment', array($this, 'autosmush_create_task'));
add_action('ud_task_initialised', array($this, 'set_task_logger'));
add_action('ud_task_started', array($this, 'set_task_logger'));
add_action('ud_task_completed', array($this, 'record_stats'));
add_action('ud_task_failed', array($this, 'record_stats'));
add_action('prune_smush_logs', array($this, 'prune_smush_logs'));
add_action('process_smush_tasks', array($this, 'process_smush_tasks'));
if ('show' == $this->options->get_option('show_smush_metabox', 'show')) {
add_action('add_meta_boxes_attachment', array($this, 'add_smush_metabox'), 10, 2);
add_filter('attachment_fields_to_edit', array($this, 'add_compress_button_to_media_modal' ), 10, 2);
}
add_action('delete_attachment', array($this, 'unscheduled_original_file_deletion'));
add_filter('manage_media_columns', array($this, 'manage_media_columns'));
add_action('manage_media_custom_column', array($this, 'manage_media_custom_column'), 10, 2);
// clean backup images cron action.
add_action('wpo_smush_clear_backup_images', array($this, 'clear_backup_images'));
// add filter for already compressed images by EWWW Image Optimizer.
add_filter('wpo_get_uncompressed_images_args', array($this, 'ewww_image_optimizer_compressed_images_args'));
// schedule or unschedule clear backup images cron if need
$scheduled = wp_next_scheduled('wpo_smush_clear_backup_images');
if ($this->options->get_option('back_up_delete_after', true)) {
if (!$scheduled) {
wp_schedule_event(time(), 'daily', 'wpo_smush_clear_backup_images');
}
} else {
if ($scheduled) {
wp_unschedule_event($scheduled, 'wpo_smush_clear_backup_images');
}
}
// Schedule CRON job for deleting failed smush tasks
add_action('wpo_smush_clear_failed_tasks', array($this, 'clear_failed_tasks'));
if (!wp_next_scheduled('wpo_smush_clear_failed_tasks')) {
wp_schedule_event(time(), 'wpo_monthly', 'wpo_smush_clear_failed_tasks');
}
}
/**
* Add custom column to Media Library.
*
* @param array $columns
* @return mixed
*/
public function manage_media_columns($columns) {
$columns['wpo_smush'] = 'WP-Optimize';
return $columns;
}
/**
* Display content in the custom column.
*
* @param string $column
* @param int $attachment_id
*/
public function manage_media_custom_column($column, $attachment_id) {
if ('wpo_smush' !== $column) return;
echo $this->get_smush_details($attachment_id); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped -- Output already escaped when generating smush details
}
/**
* Get smush details of given image ID
*
* @param int $attachment_id
*
* @return string smush details
*/
public function get_smush_details($attachment_id) {
$info = '<div class="wpo-smush-media-library-column" data-blog="'.esc_attr(get_current_blog_id()).'" data-id="'.esc_attr($attachment_id).'">';
$info .= $this->get_media_smush_column_content($attachment_id);
$info .= '</div>';
return $info;
}
/**
* Get content for Media Library column.
*
* @param int $attachment_id
*
* @return string
*/
public function get_media_smush_column_content($attachment_id) {
$file = get_attached_file($attachment_id);
$ext = WPO_Image_Utils::get_extension($file);
$allowed_extensions = WPO_Image_Utils::get_allowed_extensions();
$compressed = get_post_meta($attachment_id, 'smush-complete', true) ? true : false;
$has_backup = get_post_meta($attachment_id, 'original-file', true) ? true : false;
$smush_info = get_post_meta($attachment_id, 'smush-info', true);
$smush_stats = get_post_meta($attachment_id, 'smush-stats', true);
$marked = get_post_meta($attachment_id, 'smush-marked', false);
$extract = array(
'blog_id' => get_current_blog_id(),
'post_id' => $attachment_id,
'smush_info_display'=> $compressed ? "display:inline-block;" : "display:none;",
'smush_display' => $compressed ? "display:none;" : "display:inline-block;",
'before_smush_sep' => !$compressed ? '<span class="wpo-action-separator"> | </span>' : '',
'restore_action' => $has_backup && $compressed ? "display:block;" : "display:none;",
'smush_mark' => !$compressed && !$marked ? "display:inline-block;" : "display:none;",
'smush_unmark' => $marked ? "display:inline-block;" : "display:none;",
'smush_info' => $smush_info ? $smush_info : ' ',
'smush_details' => '',
'restore_tooltip' => $this->get_restore_image_tooltip_text(),
'smush_marked' => $marked,
);
if (!empty($smush_stats['sizes-info'])) {
$extract['smush_details'] = WP_Optimize()->include_template('images/smush-details.php', true, array('sizes_info' => $smush_stats['sizes-info']));
}
$extract['compressed_by_another_plugin'] = $this->is_image_compressed_by_another_plugin($attachment_id);
$output = '';
if (WPO_Image_Utils::is_supported_extension($ext, $allowed_extensions)) {
$output .= WP_Optimize()->include_template('admin-medialibrary-smush-column.php', true, $extract);
} else {
$message = __('Compressing this file type extension is not supported', 'wp-optimize');
$output .= sprintf("<p>%s</p>", esc_html($message));
}
if (WPO_Image_Utils::is_supported_extension($ext, array_diff($allowed_extensions, array('gif'))) && file_exists($file) && !file_exists($file . '.webp')) {
if (WPO_WebP_Utils::can_do_webp_conversion()) {
$link_text = __('Convert to WebP', 'wp-optimize');
$output .= sprintf('<a href="#" class="convert-to-webp" data-attachment-id="%d">%s</a><br>', esc_attr($attachment_id), esc_html($link_text));
}
}
return $output;
}
/**
* The Task Manager AJAX handler
*/
public function updraft_smush_ajax() {
$nonce = empty($_REQUEST['nonce']) ? '' : $_REQUEST['nonce'];
if (!wp_verify_nonce($nonce, 'updraft-task-manager-ajax-nonce') || empty($_REQUEST['subaction']))
die('Security check failed');
if (!current_user_can(WP_Optimize()->capability_required())) {
die('You are not allowed to run this command.');
}
$subaction = $_REQUEST['subaction'];
$allowed_commands = Updraft_Smush_Manager_Commands::get_allowed_ajax_commands();
if (in_array($subaction, $allowed_commands)) {
if (isset($_REQUEST['data'])) {
$data = $_REQUEST['data'];
$results = call_user_func(array($this->commands, $subaction), $data);
} else {
$results = call_user_func(array($this->commands, $subaction));
}
if (is_wp_error($results)) {
$results = array(
'status' => true,
'result' => false,
'error_code' => $results->get_error_code(),
'error_message' => $results->get_error_message(),
'error_data' => $results->get_error_data(),
);
}
echo wp_json_encode($results);
} else {
echo wp_json_encode(array('error' => 'No such command found'));
}
die();
}
/**
* Creates a task to auto compress an image on upload
*
* @param int $post_id - id of the post
*/
public function autosmush_create_task($post_id) {
$post = get_post($post_id);
$file = get_attached_file($post_id);
$ext = WPO_Image_Utils::get_extension($file);
$allowed_extensions = WPO_Image_Utils::get_allowed_extensions();
if(!in_array($ext, $allowed_extensions)) return;
if (!$this->options->get_option('autosmush', false))
return;
if (!'image' == substr($post->post_mime_type, 0, 5))
return;
if ($this->task_exists($post_id))
return;
$options = array(
'attachment_id' => $post_id,
'blog_id' => get_current_blog_id(),
'image_quality' => $this->options->get_option('image_quality', 92),
'keep_original' => $this->options->get_option('back_up_original', true),
'preserve_exif' => $this->options->get_option('preserve_exif', true),
'lossy_compression' => $this->options->get_option('lossy_compression', false)
);
if (filesize($file) > 5242880) {
$options['request_timeout'] = 180;
}
$server = $this->options->get_option('compression_server', $this->webservice);
$task_name = $this->get_associated_task($server);
$blog_info = is_multisite() ? ', blog ID : '.get_current_blog_id() : '';
$description = "$task_name with attachment ID : ".$post_id . $blog_info .", autocreated on : ".gmdate("F d, Y h:i:s", time());
$task = call_user_func(array($task_name, 'create_task'), 'smush', $description, $options, $task_name);
if ($task) $this->set_task_logger($task);
$this->log($description);
if (!wp_next_scheduled('process_smush_tasks')) {
wp_schedule_single_event(time() + 300, 'process_smush_tasks');
}
}
/**
* Processes the smush tasks in the queue, then cleans up the completed tasks.
*
* Before processing the queue, it first schedules a cron job to re-initiate the process after a certain
* interval, ensuring that the process will be completed later in case the current processing fails
* or is interrupted. This method can be invoked directly or scheduled as a cron job.
*/
public function process_smush_tasks() {
/*
* Only add log header when called as a cron job, assuming the log header is already added by the caller
* when called directly. This is to avoid duplicate log headers in the log file.
*/
if (defined('DOING_CRON') && DOING_CRON) {
$this->write_log_header();
}
// If there are no pending tasks, nothing to process. In that case, attempt to clean up old tasks and return
if ($this->is_queue_processed()) {
$this->clean_up_old_tasks('smush');
return;
}
if (!wp_next_scheduled('process_smush_tasks')) {
wp_schedule_single_event(time() + 600, 'process_smush_tasks');
}
// Process the queue
$this->clear_cached_data();
$this->process_queue('smush');
$this->clean_up_old_tasks('smush');
}
/**
* Process the compression of a single image
*
* @param int $image - ID of image
* @param array $options - options to use
* @param string $server - the server to process with
*
* @return boolean - Status of the task
*/
public function compress_single_image($image, $options, $server) {
$task_name = $this->get_associated_task($server);
$blog_info = is_multisite() ? ', blog ID : '.get_current_blog_id() : '';
$description = "$task_name - attachment ID : ". $image . $blog_info. ", started on : ". gmdate("F d, Y h:i:s", time());
$task = call_user_func(array($task_name, 'create_task'), 'smush', $description, $options, $task_name);
if ($task) $this->set_task_logger($task);
$this->clear_cached_data();
if (!wp_next_scheduled('prune_smush_logs')) {
wp_schedule_single_event(time() + 7200, 'prune_smush_logs');
}
return $this->process_task($task);
}
/**
* Restores a single image if a backup is available
*
* @param int $image_id - The id of the image
* @param int $blog_id - The id of the blog
*
* @return bool|WP_Error - success or failure
*/
public function restore_single_image($image_id, $blog_id) {
$switched_blog = false;
if (is_multisite() && current_user_can('manage_network_options')) {
switch_to_blog($blog_id);
$switched_blog = true;
} elseif (is_multisite() && get_current_blog_id() != $blog_id) {
return new WP_Error('restore_backup_wrong_blog_id', __('The blog ID provided does not match the current blog.', 'wp-optimize'));
}
$error = false;
$image_path = get_attached_file($image_id);
$backup_path = get_post_meta($image_id, 'original-file', true);
// If the file doesn't exist, check if it's relative
if (!is_file($backup_path)) {
$uploads_dir = wp_upload_dir();
$uploads_basedir = trailingslashit($uploads_dir['basedir']);
if (is_file($uploads_basedir . $backup_path)) {
$backup_path = $uploads_basedir . $backup_path;
}
}
// If the file still doesn't exist, the record could be an absolute path from a migrated site
// Use the current Uploads path
if (!is_file($backup_path)) {
$current_uploads_dir_folder = trailingslashit(substr($uploads_basedir, strlen(ABSPATH)));
// A strict operator (!==) needs to be used, as 0 is a positive result.
if (false !== strpos($backup_path, $current_uploads_dir_folder)) {
$temp_relative_backup_path = substr($backup_path, strpos($backup_path, $current_uploads_dir_folder) + strlen($current_uploads_dir_folder));
if (is_file($uploads_basedir . $temp_relative_backup_path)) {
$backup_path = $uploads_basedir . $temp_relative_backup_path;
}
}
}
// If the file still doesn't exist, the record could be an absolute path from a migrated site
// The current Uploads path failed, so try with the default uploads directory value
if (!is_file($backup_path)) {
// A strict operator (!==) needs to be used, as 0 is a positive result.
if (false !== strpos($backup_path, 'wp-content/uploads/')) {
$backup_path = substr($backup_path, strpos($backup_path, 'wp-content/uploads/') + strlen('wp-content/uploads/'));
$backup_path = $uploads_basedir . $backup_path;
}
}
if (!is_file($backup_path)) {
// Delete information about backup.
delete_post_meta($image_id, 'original-file');
$error = new WP_Error('restore_backup_not_found', __('The backup was not found; it may have been deleted or was already restored', 'wp-optimize'));
} elseif (!wp_is_writable($image_path)) {
$error = new WP_Error('restore_failed', __('The destination could not be written to.', 'wp-optimize').' '.__("Please check your folder's permissions", 'wp-optimize'));
} elseif (!copy($backup_path, $image_path)) {
$error = new WP_Error('restore_failed', __('The file could not be copied; check your PHP error logs for details', 'wp-optimize'));
} elseif (!wp_delete_file($backup_path)) {
// translators: %s is the backup file path
$error = new WP_Error('restore_failed', sprintf(__('The backup file %s could not be deleted.', 'wp-optimize'), $backup_path));
}
if (!$error) {
// if backup image deleted successfully
// then delete from attachment meta associated smush data
delete_post_meta($image_id, 'smush-complete');
delete_post_meta($image_id, 'smush-stats');
delete_post_meta($image_id, 'original-file');
delete_post_meta($image_id, 'smush-info');
}
if ($switched_blog) {
restore_current_blog();
}
if (is_wp_error($error)) return $error;
$this->delete_from_cache('uncompressed_images');
if (!wp_next_scheduled('prune_smush_logs')) {
wp_schedule_single_event(time() + 7200, 'prune_smush_logs');
}
return true;
}
/**
* Restore compressed images for selected blog.
*
* @param bool $restore_backup if true then restore images from backup otherwise just delete meta.
* @param int $blog_id blog id.
* @param int $images_limit how many images process per time.
* @param bool $delete_only_backups_meta meta fields will deleted only for images those will restored from backup.
*
* @return array ['completed' => (bool), 'message' => (string), 'error' => (string)]
*/
public function bulk_restore_compressed_images($restore_backup, $blog_id = 1, $images_limit = 100, $delete_only_backups_meta = false) {
global $wpdb;
if (is_multisite()) {
switch_to_blog($blog_id);
}
$result = array(
'completed' => false,
'message' => '',
'smushed_images_count' => 0,
);
$processed = 0;
if ($restore_backup) {
// get post ids those have backup meta field.
$image_ids = $wpdb->get_results($wpdb->prepare("SELECT post_id FROM {$wpdb->postmeta} WHERE meta_key = 'original-file' LIMIT %d;", $images_limit), ARRAY_A);
if (!empty($image_ids)) {
// run restore function for each found image.
foreach ($image_ids as $image) {
$restore_result = $this->restore_single_image($image['post_id'], $blog_id);
// if we get an error then we stop the work, except situation when "backup already restored'.
if (is_wp_error($restore_result) && 'restore_backup_not_found' != $restore_result->get_error_code()) {
// we need to stop the work as we haven't restored the backup.
$result['error'] = $restore_result->get_error_message();
$this->options->delete_option('smush_images_restored');
break;
}
$processed++;
}
}
$images_count = count($image_ids);
// if all images processed then set flag completed to true.
if ($processed == $images_count && $images_count < $images_limit) {
$this->options->delete_option('smush_images_restored');
$result['completed'] = true;
} else {
// save into options total processed count.
$processed += $this->options->get_option('smush_images_restored', 0);
$this->options->update_option('smush_images_restored', $processed);
if (is_multisite()) {
// translators: %1$s is the number of restored images, %2$s is the site url.
$result['message'] = sprintf(__('%1$s compressed images were restored from their backup for the site %2$s', 'wp-optimize'), $processed, get_site_url($blog_id));
} else {
// translators: %s is the number of restored images.
$result['message'] = sprintf(__('%s compressed images were restored from their backup', 'wp-optimize'), $processed);
}
}
} else {
// if we just delete compressed images meta then set complete flag to true.
$result['completed'] = true;
}
if ($result['completed']) {
$smushed_images_count = $wpdb->get_var("SELECT COUNT(*) FROM {$wpdb->postmeta} WHERE meta_key='smush-complete' AND meta_value=1");
$result['smushed_images_count'] = $smushed_images_count;
if ($delete_only_backups_meta) {
if (is_multisite()) {
if ($smushed_images_count > 0) {
// translators: %s is the site url
$result['message'] = sprintf(__('All the compressed images for site %s with backup copies of their original files were successfully restored.', 'wp-optimize'), get_site_url($blog_id));
// translators: %s is the number of images
$result['message'] .= ' '.sprintf(_n('Unable to restore %s image without backup files.', 'Unable to restore %s images without backup files.', $smushed_images_count, 'wp-optimize'), $smushed_images_count);
} else {
// translators: %s is the site url
$result['message'] = sprintf(__('All the compressed images for the site %s were successfully restored.', 'wp-optimize'), get_site_url($blog_id));
}
} else {
if ($smushed_images_count > 0) {
$result['message'] = __('All the compressed images with backup copies of their original files were successfully restored.', 'wp-optimize');
// translators: %s is the number of images
$result['message'] .= ' '.sprintf(_n('Unable to restore %s image without backup files.', 'Unable to restore %s images without backup files.', $smushed_images_count, 'wp-optimize'), $smushed_images_count);
} else {
$result['message'] = __('All the compressed images were successfully restored.', 'wp-optimize');
}
}
} else {
if (is_multisite()) {
// translators: %s is the site url
$result['message'] = sprintf(__('All the compressed images for the site %s were successfully marked as uncompressed.', 'wp-optimize'), get_site_url($blog_id));
} else {
$result['message'] = __('All the compressed images were successfully marked as uncompressed.', 'wp-optimize');
}
}
// clear all metas for smushed images after work completed.
// if $delete_only_backup_meta set to true then all meta fields was deleted in restore_single_image()
// and we don't need delete metas for other images.
if (!$delete_only_backups_meta) {
$wpdb->query("DELETE FROM {$wpdb->postmeta} WHERE meta_key IN ('smush-complete', 'smush-stats', 'original-file', 'smush-info');");
}
}
if (is_multisite()) {
restore_current_blog();
}
return $result;
}
/**
* Process bulk smushing operation
*
* @param array $images - the array of images to process
* @return bool - true if processing complete
*/
public function process_bulk_smush($images = array()) {
// Get a list of pending tasks so we can exclude those
$pending_tasks = $this->get_pending_tasks();
$queued_images = array();
$this->write_log_header();
if (!empty($pending_tasks)) {
foreach ($pending_tasks as $task) {
$queued_images[] = array(
'attachment_id' => $task->get_option('attachment_id'),
'blog_id' => $task->get_option('blog_id')
);
}
}
foreach ($images as $image) {
// Skip if already in the queue
if (in_array($image, $queued_images)) continue;
$options = array(
'attachment_id' => intval($image['attachment_id']),
'blog_id' => intval($image['blog_id']),
'image_quality' => $this->options->get_option('image_quality', 92),
'keep_original' => $this->options->get_option('back_up_original', true),
'preserve_exif' => $this->options->get_option('preserve_exif', true),
'lossy_compression' => $this->options->get_option('lossy_compression', false)
);
$server = $this->options->get_option('compression_server', $this->webservice);
$task_name = $this->get_associated_task($server);
$blog_info = is_multisite() ? ', Blog ID : '.intval($image['blog_id']) : '';
$description = "$task_name - Attachment ID : ". intval($image['attachment_id']) . $blog_info . ", Started on : ". gmdate("F d, Y h:i:s", time());
$task = call_user_func(array($task_name, 'create_task'), 'smush', $description, $options, $task_name);
if ($task) $this->set_task_logger($task);
}
$this->process_smush_tasks();
if (!wp_next_scheduled('prune_smush_logs')) {
wp_schedule_single_event(time() + 7200, 'prune_smush_logs');
}
return true;
}
/**
* Check if a specified server online
*
* @param string $server - the server to test
* @return bool - true if yes, false otherwise
*/
public function check_server_online($server = 'resmushit') {
$task = $this->get_associated_task($server);
$online = call_user_func(array($task, 'is_server_online'));
if ($online) {
update_option($task, strtotime('now'));
} else {
$this->log(get_option($task));
}
$this->log(sprintf('%s server status: %s', $task, $online ? 'Online' : 'Offline'));
return $online;
}
/**
* Checks if the queue for smushing is complete
*
* @return bool - true if processed, false otherwise
*/
public function is_queue_processed() {
$active = $this->get_pending_tasks();
if ($active && 0 != count($active)) return false;
return true;
}
/**
* Logs useful data once a smush task completes or if it fails
*
* @param mixed $task - A task object
*/
public function record_stats($task) {
$attachment_id = $task->get_option('attachment_id');
$completed_task_count = $this->options->get_option('completed_task_count', false);
$failed_task_count = $this->options->get_option('failed_task_count', 0);
$total_bytes_saved = $this->options->get_option('total_bytes_saved', false);
$total_percent_saved = $this->options->get_option('total_percent_saved', 0);
if ('ud_task_failed' == current_action()) {
$this->options->update_option('failed_task_count', ++$failed_task_count);
return;
}
if (empty($completed_task_count)) {
$completed_task_count = $total_bytes_saved = 0;
}
if (!$total_bytes_saved) {
$total_bytes_saved = 0;
}
if (is_multisite()) {
switch_to_blog($task->get_option('blog_id', 1));
$stats = get_post_meta($attachment_id, 'smush-stats', true);
restore_current_blog();
} else {
$stats = get_post_meta($attachment_id, 'smush-stats', true);
}
if (isset($stats['sizes-info'])) {
$original_size = 0;
$compressed_size = 0;
foreach ($stats['sizes-info'] as $info) {
$original_size += $info['original'];
$compressed_size += $info['compressed'];
}
$percent = round((($original_size - $compressed_size) / $original_size * 100), 2);
} else {
$original_size = isset($stats['original-size']) ? $stats['original-size'] : 0;
$compressed_size = isset($stats['smushed-size']) ? $stats['smushed-size'] : 0;
$percent = isset($stats['savings-percent']) ? $stats['savings-percent'] : 0;
}
$saved = $original_size - $compressed_size;
$completed_task_count++;
$total_bytes_saved += $saved;
$total_percent_saved = (($total_percent_saved * ($completed_task_count - 1)) + $percent) / $completed_task_count;
$this->options->update_option('completed_task_count', $completed_task_count);
$this->options->update_option('total_bytes_saved', $total_bytes_saved);
$this->options->update_option('total_percent_saved', $total_percent_saved);
}
/**
* Get current smush options.
*
* @return array
*/
public function get_smush_options() {
static $smush_options = array();
if (empty($smush_options)) {
$smush_options = array(
'compression_server' => $this->options->get_option('compression_server', $this->get_default_webservice()),
'image_quality' => $this->options->get_option('image_quality', 92),
'lossy_compression' => $this->options->get_option('lossy_compression', false),
'back_up_original' => $this->options->get_option('back_up_original', true),
'back_up_delete_after' => $this->options->get_option('back_up_delete_after', true),
'back_up_delete_after_days' => $this->options->get_option('back_up_delete_after_days', 50),
'preserve_exif' => $this->options->get_option('preserve_exif', false),
'autosmush' => $this->options->get_option('autosmush', false),
'show_smush_metabox' => $this->options->get_option('show_smush_metabox', 'show') == 'show' ? true : false,
'webp_conversion' => $this->options->get_option('webp_conversion', false)
);
}
return $smush_options;
}
/**
* Updates global smush options
*
* @param array $options - sent in via AJAX
* @return bool - status of the update
*/
public function update_smush_options($options) {
foreach ($options as $option => $value) {
$this->options->update_option($option, $value);
}
return true;
}
/**
* Clears smush related stats
*
* @return bool - status of the update
*/
public function clear_smush_stats() {
$this->options->update_option('failed_task_count', 0);
$this->options->update_option('completed_task_count', false);
$this->options->update_option('total_bytes_saved', false);
$this->options->update_option('total_percent_saved', 0);
return true;
}
/**
* Returns array of translations used in javascript code.
*
* @return array - translations used in JS
*/
public function smush_js_translations() {
$resmushit_article_link = WP_Optimize()->wp_optimize_url('https://resmush.it/api/', __('resmushIt', 'wp-optimize'), '', '', true);
return apply_filters('updraft_smush_js_translations', array(
'all_images_compressed' => __('No uncompressed images were found.', 'wp-optimize'),
'error_unexpected_response' => __('An unexpected response was received from the server.', 'wp-optimize') . ' ' . __('More information has been logged in the browser console.', 'wp-optimize'),
'compress_single_image_dialog' => __('Please wait: compressing the selected image.', 'wp-optimize'),
'error_try_again_later' => __('Please try again later.', 'wp-optimize'),
'server_check' => __('Connecting to the Smush API server, please wait', 'wp-optimize'),
'please_wait' => __('Please wait while the request is being processed', 'wp-optimize'),
'server_error' => __('There was an error connecting to the image compression server.', 'wp-optimize') .
'<br>' . __('This could mean either the server is temporarily unavailable or there are connectivity issues with your internet connection.', 'wp-optimize') . ' ' .
// translators: %s is a link
'<i>' . sprintf(__('(Also ensure IPs listed at the bottom of this %s page are whitelisted by your webserver).', 'wp-optimize'), $resmushit_article_link) . '</i>' .
'<br>' . __('Please try later.', 'wp-optimize'),
'please_select_images' => __('Please select the images you want compressed from the "Uncompressed images" panel first', 'wp-optimize'),
'please_updating_images_info' => __('Please wait: updating information about the selected image.', 'wp-optimize'),
'please_select_compressed_images' => __('Please select the images you want to mark as already compressed from the "Uncompressed images" panel first', 'wp-optimize'),
'view_image' => __('View Image', 'wp-optimize'),
'delete_image_backup_confirm' => __('Do you really want to delete all backup images now?', 'wp-optimize') . ' ' . __('This action is irreversible.', 'wp-optimize'),
'mark_all_images_uncompressed' => __('Do you really want to mark all the images as uncompressed?', 'wp-optimize') . ' ' . __('This action is irreversible.', 'wp-optimize'),
'restore_images_from_backup' => __('Do you want to restore the original images from the backup (where they exist?)', 'wp-optimize'),
'restore_all_compressed_images' => __('Do you really want to restore all the compressed images?', 'wp-optimize'),
'webp_conversion_tool_error' => __('No WebP conversion tools are available on your web-server.', 'wp-optimize'),
'webp_conversion_tool_how_to' => __('How to get the WebP conversion tools to work?', 'wp-optimize'),
'more' => __('More', 'wp-optimize'),
'less' => __('Less', 'wp-optimize'),
'converting_to_webp' => __('Converting image to WebP format, please wait', 'wp-optimize'),
'compress_image' => __('Compress Image', 'wp-optimize'),
'compress' => __('Compress', 'wp-optimize'),
'cancel' => __('Cancel', 'wp-optimize'),
'close' => __('Close', 'wp-optimize'),
));
}
/**
* Adds a smush metabox on the post edit screen for images
*
* @param WP_Post $post - a post object
*/
public function add_smush_metabox($post) {
if (!wp_attachment_is_image($post->ID)) return;
if (!file_exists(get_attached_file($post->ID))) {
return;
}
add_meta_box('smush-metabox', __('Compress Image', 'wp-optimize'), array($this, 'render_smush_metabox'), 'attachment', 'side');
}
/**
* Renders a metabox on the post edit screen for images
*
* @param WP_Post $post - a post object
*/
public function render_smush_metabox($post) {
$compressed = get_post_meta($post->ID, 'smush-complete', true) ? true : false;
$has_backup = get_post_meta($post->ID, 'original-file', true) ? true : false;
$smush_info = get_post_meta($post->ID, 'smush-info', true);
$smush_stats = get_post_meta($post->ID, 'smush-stats', true);
$marked = get_post_meta($post->ID, 'smush-marked', false);
$options = Updraft_Smush_Manager()->get_smush_options();
$file = get_attached_file($post->ID);
$ext = WPO_Image_Utils::get_extension($file);
$allowed_extensions = WPO_Image_Utils::get_allowed_extensions();
$file_size = ($file && is_file($file)) ? filesize($file) : 0;
$extract = array(
'post_id' => $post->ID,
'smush_display' => $compressed ? "display:none;" : "display:block;",
'restore_display' => $compressed ? "display:block;" : "display:none;",
'restore_action' => $has_backup ? "display:block;" : "display:none;",
'smush_mark' => !$compressed && !$marked ? "display:block;" : "display:none;",
'smush_unmark' => $marked ? "display:block;" : "display:none;",
'smush_info' => $smush_info ? $smush_info : ' ',
'file_size' => $file_size,
'smush_options' => $options,
'custom' => 90 >= $options['image_quality'] && 65 <= $options['image_quality'],
'smush_details' => '',
'restore_tooltip' => $this->get_restore_image_tooltip_text(),
'smush_marked' => $marked,
);
if (!empty($smush_stats['sizes-info'])) {
$extract['smush_details'] = WP_Optimize()->include_template('images/smush-details.php', true, array('sizes_info' => $smush_stats['sizes-info']));
}
$extract['smush_settings_form'] = WP_Optimize()->include_template('admin-metabox-smush-settings.php', true, $extract);
$extract['compressed_by_another_plugin'] = $this->is_image_compressed_by_another_plugin($post->ID);
if (WPO_Image_Utils::is_supported_extension($ext, $allowed_extensions)) {
WP_Optimize()->include_template('admin-metabox-smush.php', false, $extract);
} else {
printf("<p>%s</p>", esc_html__('Compressing this file type extension is not supported', 'wp-optimize'));
}
}
/**
* Get text for restrore image tooltip.
*
* @return string
*/
private function get_restore_image_tooltip_text() {
$text = __('Only the original image will be restored.', 'wp-optimize');
$text .= ' ';
$text .= __('In order to restore the other sizes, you should use a plugin such as "Regenerate Thumbnails".', 'wp-optimize');
return $text;
}
/**
* Check if a single image compressed by another plugin.
*
* @param int $image_id
* @return bool
*/
private function is_image_compressed_by_another_plugin($image_id) {
global $wpdb;
$meta = $wpdb->get_results("SELECT meta_key, meta_value FROM {$wpdb->postmeta} WHERE `post_id`={$image_id}", ARRAY_A);
if (is_array($meta)) {
foreach ($meta as $row) {
// Smush, Imagify, Compress JPEG & PNG images by TinyPNG.
if (in_array($row['meta_key'], array('wp-smpro-smush-data', '_imagify_optimization_level', 'tiny_compress_images'))) return true;
// ShortPixel Image Optimizer
if ('_shortpixel_status' == $row['meta_key'] && 2 <= $row['meta_key'] && 3 > $row['meta_key']) return true;
}
}
if (WP_Optimize()->get_db_info()->table_exists('ewwwio_images')) {
$old_show_errors = $wpdb->show_errors(false);
// EWWW Image Optimizer.
$ewww_image = $wpdb->get_col("SELECT attachment_id FROM {$wpdb->prefix}ewwwio_images WHERE attachment_id={$image_id} AND gallery='media' LIMIT 1");
if (!empty($ewww_image)) return true;
$wpdb->show_errors($old_show_errors);
}
return apply_filters('wpo_image_compressed_by_another_plugin', false);
}
/**
* Get attachment ids for images those already compressed with EWWW Image Optimizer.
* Used with filter `wpo_get_uncompressed_images_args`.
*
* @param array $args WP_Query arguments.
* @return array
*/
public function ewww_image_optimizer_compressed_images_args($args) {
global $wpdb;
if (!WP_Optimize()->get_db_info()->table_exists('ewwwio_images')) return $args;
$old_show_errors = $wpdb->show_errors(false);
$compressed_images = $wpdb->get_col("SELECT DISTINCT(attachment_id) FROM {$wpdb->prefix}ewwwio_images WHERE gallery='media'");
$wpdb->show_errors($old_show_errors);
if (isset($args['post__not_in'])) {
$args['post__not_in'] = array_merge($args['post__not_in'], $compressed_images);
} else {
$args['post__not_in'] = $compressed_images;
}
return $args;
}
/**
* Returns a list of images for smush (from cache if available)
*
* @param string $use_cache
* @return array - uncompressed images
*/
public function get_uncompressed_images($use_cache = "true") {
if ("true" == $use_cache) {
$uncompressed_images = $this->get_from_cache('uncompressed_images');
if ($uncompressed_images) return $uncompressed_images;
}
$uncompressed_images = array();
$accepted_mimes = array('image/jpeg', 'image/gif', 'image/png');
$args = array(
'post_type' => 'attachment',
'post_mime_type' => $accepted_mimes,
'post_status' => 'inherit',
'posts_per_page' => apply_filters('updraft_smush_posts_per_page', 1000),
'meta_query' => $this->get_uncompressed_images_meta_query(),
'no_found_rows' => true,
'fields' => 'ids'
);
$allowed_extensions = WPO_Image_Utils::get_allowed_extensions();
if (is_multisite()) {
$sites = WP_Optimize()->get_sites();
foreach ($sites as $site) {
switch_to_blog($site->blog_id);
$args = apply_filters('wpo_get_uncompressed_images_args', $args);
$images = new WP_Query($args);
foreach ($images->posts as $image) {
// If `field` is removed from $args it returns a WP_Post obj
$image_id = is_int($image) ? $image : $image->ID;
$file = get_attached_file($image_id);
$ext = WPO_Image_Utils::get_extension($file);
if (file_exists($file)) {
if (WPO_Image_Utils::is_supported_extension($ext, $allowed_extensions)) {
$uncompressed_images[$site->blog_id][] = array(
'id' => $image_id,
'thumb_url' => wp_get_attachment_thumb_url($image_id),
'filesize' => filesize(get_attached_file($image_id))
);
} else {
$this->log("Blog_id={$site->blog_id}, ID={$image_id}, File={$file} This image type is not supported.");
}
} else {
$this->log("Could not find file for image: blog_id={$site->blog_id}, ID={$image_id}, file={$file}");
}
}
restore_current_blog();
}
} else {
$args = apply_filters('wpo_get_uncompressed_images_args', $args);
$images = new WP_Query($args);
foreach ($images->posts as $image) {
// If `field` is removed from $args it returns a WP_Post obj
$image_id = is_int($image) ? $image : $image->ID;
$file = get_attached_file($image_id);
$ext = WPO_Image_Utils::get_extension($file);
if (file_exists($file)) {
if (WPO_Image_Utils::is_supported_extension($ext, $allowed_extensions)) {
$uncompressed_images[1][] = array(
'id' => $image_id,
'thumb_url' => wp_get_attachment_thumb_url($image_id),
'filesize' => filesize(get_attached_file($image_id))
);
} else {
$this->log("Image ID={$image_id}, File={$file} This image type is not supported.");
}
} else {
$this->log("Could not find file for image: ID={$image_id}, file={$file}");
}
}
}
$this->save_to_cache('uncompressed_images', $uncompressed_images);
return $uncompressed_images;
}
/**
* Returns a list of admin URLs. This is to prevent unnecessary bloat in the output of get_uncompressed_images() (and thus better performance over the network on sites with huge numbers of images)
*
* @return array - list of admin URLs
*/
public function get_admin_urls() {
$admin_urls = $this->get_from_cache('admin_urls');
if ($admin_urls) return $admin_urls;
$admin_urls = array();
if (is_multisite()) {
$sites = WP_Optimize()->get_sites();
foreach ($sites as $site) {
switch_to_blog($site->blog_id);
$admin_urls[$site->blog_id] = admin_url();
restore_current_blog();
}
} else {
// The pseudo-blog_id here (1) matches (and must match) what is used in get_uncompressed_images
$admin_urls[1] = admin_url();
}
$this->save_to_cache('admin_urls', $admin_urls);
return $admin_urls;
}
/**
* Check if a task exists for a given image
*
* @param string $image - The attachment ID of the image
* @return bool - true if yes, false otherwise
*/
public function task_exists($image) {
$blog_id = get_current_blog_id();
$pending_tasks = $this->get_active_tasks('smush');
if (!empty($pending_tasks)) {
foreach ($pending_tasks as $task) {
$task_attachment_id = $task->get_option('attachment_id');
$task_blog_id = $task->get_option('blog_id');
if ($image === $task_attachment_id && $blog_id === $task_blog_id) {
return true;
}
}
}
return false;
}
/**
* Returns the status of images compressed in this iteration of the bulk compress
*
* @param array $images - List of images in the current session
*
* @return array - status of the operation
*/
public function get_session_stats($images) {
$stats = array();
foreach ($images as $image) {
if (is_multisite()) {
switch_to_blog($image['blog_id']);
$stats[] = get_post_meta($image['attachment_id'], 'smush-complete', true) ? 'success' : 'fail';
restore_current_blog();
} else {
$stats[] = get_post_meta($image['attachment_id'], 'smush-complete', true) ? 'success' : 'fail';
}
}
return array_count_values($stats);
}
/**
* Returns a list of images for smush (from cache if available)
*
* @return array - List of task objects with uncompressed images
*/
public function get_pending_tasks() {
return $this->get_active_tasks('smush');
}
/**
* Deletes and removes any pending tasks from queue
*/
public function clear_pending_images() {
$pending_tasks = $this->get_active_tasks('smush');
if (!empty($pending_tasks)) {
foreach ($pending_tasks as $task) {
$task->delete_meta();
$task->delete();
}
}
return true;
}
/**
* Returns a count of failed tasks
*
* @return int - failed tasks
*/
public function get_failed_task_count() {
return $this->options->get_option('failed_task_count', 0);
}
/**
* Adds the required scripts and styles
*/
public function admin_enqueue_scripts() {
$current_screen = get_current_screen();
if (null === $current_screen) return;
// load scripts and styles only on WP-Optimize pages
if (!preg_match('/wp\-optimize|attachment|upload/i', $current_screen->id)) return;
$enqueue_version = WP_Optimize()->get_enqueue_version();
$min_or_not = WP_Optimize()->get_min_or_not_string();
$min_or_not_internal = WP_Optimize()->get_min_or_not_internal_string();
$js_variables = $this->smush_js_translations();
$js_variables['ajaxurl'] = admin_url('admin-ajax.php');
$js_variables['features'] = $this->get_features();
$js_variables['smush_ajax_nonce'] = wp_create_nonce('updraft-task-manager-ajax-nonce');
$js_variables['smush_settings'] = $this->get_smush_options();
$js_variables['blog_id'] = get_current_blog_id();
$js_variables['compress'] = esc_html__('Compress', 'wp-optimize');
$js_variables['cancel'] = esc_html__('Cancel', 'wp-optimize');
$js_variables['cancelling'] = esc_html__('Cancelling...', 'wp-optimize');
$js_variables['images_restored_successfully'] = esc_html__('The images were restored successfully', 'wp-optimize');
$js_variables['logo_src'] = esc_url(WPO_PLUGIN_URL.'images/notices/wp_optimize_logo.png');
wp_enqueue_script('block-ui-js', WPO_PLUGIN_URL.'includes/blockui/jquery.blockUI'.$min_or_not.'.js', array('jquery'), $enqueue_version);
wp_enqueue_script('wp-optimize-heartbeat-js', WPO_PLUGIN_URL.'js/heartbeat'.$min_or_not_internal.'.js', array('jquery'), $enqueue_version);
wp_localize_script('wp-optimize-heartbeat-js', 'wpo_heartbeat_ajax', array(
'ajaxurl' => admin_url('admin-ajax.php'),
'nonce' => wp_create_nonce('heartbeat-nonce'),
'interval' => WPO_Ajax::HEARTBEAT_INTERVAL
));
wp_enqueue_script('smush-js', WPO_PLUGIN_URL.'js/wposmush'.$min_or_not_internal.'.js', array('jquery', 'block-ui-js', 'wp-optimize-block-ui', 'wp-optimize-send-command', 'wp-optimize-heartbeat-js'), $enqueue_version);
wp_enqueue_style('smush-css', WPO_PLUGIN_URL.'css/smush'.$min_or_not_internal.'.css', array(), $enqueue_version);
wp_localize_script('smush-js', 'wposmush', $js_variables);
}
/**
* Gets default service provider for smush
*
* @return string - service name
*/
public function get_default_webservice() {
return 'resmushit';
}
/**
* Sets default options for smush
*/
public function set_default_options() {
$options = array(
'compression_server' => $this->get_default_webservice(),
'image_quality' => 92,
'lossy_compression' => false,
'back_up_original' => true,
'preserve_exif' => false,
'autosmush' => false,
'back_up_delete_after' => $this->options->get_option('back_up_delete_after', true),
'back_up_delete_after_days' => $this->options->get_option('back_up_delete_after_days', 50),
'webp_conversion' => false,
);
$this->update_smush_options($options);
}
/**
* Gets default service provider for smush
*
* @param string $server - The name of the server
* @return string - associated task type, default if none found
*/
public function get_associated_task($server) {
$allowed = $this->get_allowed_services();
if (key_exists($server, $allowed))
return $allowed[$server];
$default = $this->get_default_webservice();
return $allowed[$default];
}
/**
* Gets allowed service providers for smush
*
* @return array - key value pair of service name => task name
*/
public function get_allowed_services() {
return array(
'resmushit' => 'Re_Smush_It_Task',
);
}
/**
* Gets allowed service provider features smush
*
* @return array - key value pair of service name => features exposed
*/
public function get_features() {
$features = array();
foreach ($this->get_allowed_services() as $service => $class_name) {
$features[$service] = call_user_func(array($class_name, 'get_features'));
}
return $features;
}
/**
* Returns the path to the logfile
*
* @return string - file path
*/
public function get_logfile_path() {
return WP_Optimize_Utils::get_log_file_path('smush');
}
/**
* Delete all smush log files
*
* @deprecated 3.5.0
*/
public function delete_log_files() {
_deprecated_function(__METHOD__, '3.5.0');
}
/**
* Adds a logger to the task
*
* @param Mixed $task - a task object
*/
public function set_task_logger($task) {
if (!$this->logger) {
$this->logger = new Updraft_File_Logger($this->get_logfile_path());
}
if (!$task->get_loggers()) {
$task->add_logger($this->logger);
}
}
/**
* Writes a standardised header to the log file
*/
public function write_log_header() {
global $wpdb;
// phpcs:disable
$wp_version = $this->get_wordpress_version();
$mysql_version = $wpdb->db_version();
$disabled_functions = ini_get('disable_functions');
$max_execution_time = (int) @ini_get("max_execution_time");
$memory_limit = ini_get('memory_limit');
$memory_usage = round(@memory_get_usage(false)/1048576, 1);
$total_memory_usage = round(@memory_get_usage(true)/1048576, 1);
// Attempt to raise limit
@set_time_limit(330);
$log_header = array();
// phpcs:enable
$log_header[] = "\n";
$log_header[] = "Header for logs at time: ".gmdate('r')." on ".network_site_url();
$log_header[] = "WP: ".$wp_version;
$php_uname = '';
if (function_exists('php_uname')) {
$php_uname = ", " . php_uname();
}
$log_header[] = "PHP: ".phpversion()." (".PHP_SAPI.$php_uname.")";
$log_header[] = "MySQL: $mysql_version";
$log_header[] = "WPLANG: ".get_locale();
$log_header[] = "Server: ".$_SERVER["SERVER_SOFTWARE"];
$log_header[] = "Outbound connections: ".(defined('WP_HTTP_BLOCK_EXTERNAL') ? 'Y' : 'N');
$log_header[] = "Disabled Functions: $disabled_functions";
$log_header[] = "max_execution_time: $max_execution_time";
$log_header[] = "memory_limit: $memory_limit (used: {$memory_usage}M | {$total_memory_usage}M)";
$log_header[] = "multisite: ".(is_multisite() ? 'Y' : 'N');
$log_header[] = "openssl: ".(defined('OPENSSL_VERSION_TEXT') ? OPENSSL_VERSION_TEXT : 'N');
if (apply_filters("wpo_write_server_info_in_smush_log", false)) {
foreach ($log_header as $log_entry) {
$this->log($log_entry);
}
}
$memlim = $this->memory_check_current();
if ($memlim<65 && $memlim>0) {
$this->log(sprintf('The amount of memory (RAM) allowed for PHP is very low (%s Mb) - you should increase it to avoid failures due to insufficient memory (consult your web hosting company for more help)', round($memlim, 1)), 'warning');
}
if ($max_execution_time>0 && $max_execution_time<20) {
$this->log(sprintf('The amount of time allowed for WordPress plugins to run is very low (%s seconds) - you should increase it to avoid 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)', $max_execution_time, 90), 'warning');
}
}
/**
* Prunes the log file
*/
public function prune_smush_logs() {
$this->log("Pruning the smush log file");
$this->logger->prune_logs();
}
/**
* Get the WordPress version
*
* @return String - the version
*/
public function get_wordpress_version() {
static $got_wp_version = false;
if (!$got_wp_version) {
global $wp_version;
@include(ABSPATH.WPINC.'/version.php');// phpcs:ignore Generic.PHP.NoSilencedErrors.Discouraged -- suppress warning if `version.php` does not exists
$got_wp_version = $wp_version;
}
return $got_wp_version;
}
/**
* Get the current memory limit
*
* @return String - memory limit in megabytes
*/
public function memory_check_current($memory_limit = false) {
// Returns in megabytes
if (false == $memory_limit) $memory_limit = ini_get('memory_limit');
$memory_limit = rtrim($memory_limit);
$memory_unit = $memory_limit[strlen($memory_limit)-1];
if (0 == (int) $memory_unit && '0' !== $memory_unit) {
$memory_limit = substr($memory_limit, 0, strlen($memory_limit)-1);
} else {
$memory_unit = '';
}
switch ($memory_unit) {
case '':
$memory_limit = floor($memory_limit/1048576);
break;
case 'K':
case 'k':
$memory_limit = floor($memory_limit/1024);
break;
case 'G':
$memory_limit = $memory_limit*1024;
break;
case 'M':
// assumed size, no change needed
break;
}
return $memory_limit;
}
/**
* Saves a value to the cache.
*
* @param string $key
* @param mixed $value
* @param int $blog_id
*/
public function save_to_cache($key, $value, $blog_id = 1) {
$transient_limit = 3600 * 48;
$key = 'wpo_smush_cache_' . $blog_id . '_'. $key;
WP_Optimize_Transients_Cache::get_instance()->set_transient($key, $value, $transient_limit);
}
/**
* Gets value from the cache.
*
* @param string $key
* @param int $blog_id
* @return mixed
*/
public function get_from_cache($key, $blog_id = 1) {
$key = 'wpo_smush_cache_' . $blog_id . '_'. $key;
return WP_Optimize_Transients_Cache::get_instance()->get($key);
}
/**
* Deletes a value from the cache.
*
* @param string $key
* @param int $blog_id
*/
public function delete_from_cache($key, $blog_id = 1) {
$key = 'wpo_smush_cache_' . $blog_id . '_'. $key;
WP_Optimize_Transients_Cache::get_instance()->delete($key);
$this->delete_transient($key);
}
/**
* Wrapper for deleting a transient
*
* @param string $key
*/
public function delete_transient($key) {
if ($this->is_multisite_mode()) {
delete_site_transient($key);
} else {
delete_transient($key);
}
}
/**
* Removes all cached data
*/
public function clear_cached_data() {
global $wpdb;
// get list of cached data by optimization.
if ($this->is_multisite_mode()) {
$keys = $wpdb->get_col("SELECT meta_key FROM {$wpdb->sitemeta} WHERE meta_key LIKE '%wpo_smush_cache_%'");
} else {
$keys = $wpdb->get_col("SELECT option_name FROM {$wpdb->options} WHERE option_name LIKE '%wpo_smush_cache_%'");
}
if (!empty($keys)) {
$transient_keys = array();
foreach ($keys as $key) {
preg_match('/wpo_smush_cache_.+/', $key, $option_name);
$option_name = $option_name[0];
$transient_keys[] = $option_name;
}
// get unique keys.
$transient_keys = array_unique($transient_keys);
// delete transients.
foreach ($transient_keys as $key) {
$this->delete_transient($key);
}
}
}
/**
* Delete recursively all smush backup files created more that $days_ago days.
*
* @param string $directory upload directory
* @param int $days_ago
*/
public function clear_backup_images_directory($directory, $days_ago = 30) {
$directory = trailingslashit($directory);
$current_time = time();
if (preg_match('/(\d{4})\/(\d{2})\/$/', $directory, $match)) {
$check_date = false;
if ($days_ago > 0) {
// check if it is end directory then scan for backup images.
$year = (int) $match[1];
$month = (int) $match[2];
$limit = strtotime('-'.$days_ago.' '.(($days_ago > 1) ? 'days' : 'day'));
$year_limit = (int) gmdate('Y', $limit);
$month_limit = (int) gmdate('m', $limit);
$day_limit = (int) gmdate('j', $limit);
// if current directory is newer than needed then we skip it.
if ($year_limit < $year || ($year_limit == $year && $month_limit < $month)) {
return;
}
// we will check dates only in directory that contain limit date.
$check_date = ($year_limit == $year && $month_limit == $month);
}
// GLOB_BRACE isn't defined on some systems (Solaris, SunOS and more) > https://www.php.net/manual/en/function.glob.php
$files = glob($directory . '*-updraft-pre-smush-original.*', (defined('GLOB_BRACE') ? GLOB_BRACE : 0));
foreach ($files as $file) {
if ($check_date) {
$filedate_day = (int) gmdate('j', filectime($file));
if ($filedate_day >= $day_limit) continue;
}
wp_delete_file($file);
}
} else {
// scan directories recursively.
$handle = opendir($directory);
if (false === $handle) return;
$file = readdir($handle);
while (false !== $file) {
if ('.' == $file || '..' == $file) {
$file = readdir($handle);
continue;
}
if (is_dir($directory . $file)) {
$this->clear_backup_images_directory($directory . $file, $days_ago);
} elseif (is_file($directory . $file) && preg_match('/^.+-updraft-pre-smush-original\.\S{3,4}/i', $file)) {
// check the file time and compare with $days_ago.
$filedate_day = (int) filectime($directory . $file);
if ($filedate_day > 0 && ($current_time - $filedate_day) / 86400 >= $days_ago) wp_delete_file($directory . $file);
}
$file = readdir($handle);
}
}
}
/**
* Clean backup smush images according to saved options.
*/
public function clear_backup_images() {
$back_up_delete_after = $this->options->get_option('back_up_delete_after', false);
if (!$back_up_delete_after) return;
$back_up_delete_after_days = $this->options->get_option('back_up_delete_after_days', 50);
$upload_dir = wp_upload_dir(null, false);
$base_dir = $upload_dir['basedir'];
$this->clear_backup_images_directory($base_dir, $back_up_delete_after_days);
}
/**
* Check if attachment already compressed.
*
* @param int $attachment_id
*
* @return bool
*/
public function is_compressed($attachment_id) {
return (true == get_post_meta($attachment_id, 'smush-complete', true));
}
/**
* @param array $form_fields
* @param WP_Post $post
*
* @return array
*/
public function add_compress_button_to_media_modal($form_fields, $post) {
if (!is_admin() || !function_exists('get_current_screen')) return $form_fields;
/**
* In media modal get_current_screen() return null or id = 'async-upload' We don't need add smush fields elsewhere.
*/
$current_screen = get_current_screen();
if (null !== $current_screen && 'async-upload' != $current_screen->id) return $form_fields;
/**
* Don't show additional fields for non-image attachments.
*/
if (!wp_attachment_is_image($post->ID)) return $form_fields;
ob_start();
$this->render_smush_metabox($post);
$smush_metabox = ob_get_contents();
ob_end_clean();
$form_fields['wpo_compress_image'] = array(
'value' => '',
'label' => __('Compress image', 'wp-optimize'),
'input' => 'html',
'html' => $smush_metabox,
);
return $form_fields;
}
/**
* Returns true if multisite
*
* @return bool
*/
public function is_multisite_mode() {
return WP_Optimize()->is_multisite_mode();
}
/**
* This callback function is triggered due to delete_attachment action (wp-includes/post.php) and is executed prior to deletion of post-type attachment
*
* @param int $post_id - WordPress Post ID
*/
public function unscheduled_original_file_deletion($post_id) {
$the_original_file = get_post_meta($post_id, 'original-file', true);
$uploads_dir = wp_get_upload_dir();
$the_original_file = trailingslashit($uploads_dir['basedir']) . $the_original_file;
if ('' != $the_original_file && file_exists($the_original_file)) {
wp_delete_file($the_original_file);
}
}
/**
* Remove failed smush tasks from the wp_tm_tasks table
*/
public function clear_failed_tasks() {
$failed_tasks = $this->get_tasks('failed', 'smush');
if (empty($failed_tasks)) return;
foreach ($failed_tasks as $task) {
$task->delete_meta();
$task->delete();
}
}
/**
* Instance of WP_Optimize_Page_Cache_Preloader.
*
* @return self
*/
public static function instance() {
if (empty(self::$_instance)) {
self::$_instance = new self();
}
return self::$_instance;
}
/**
* Meta query array for getting uncompressed images
*
* @return array
*/
public function get_uncompressed_images_meta_query() {
return array(
'relation' => 'AND',
array(
'relation' => 'OR',
array(
'key' => 'smush-complete',
'compare' => '!=',
'value' => '1',
),
array(
'key' => 'smush-complete',
'compare' => 'NOT EXISTS',
'value' => '',
),
),
// ShortPixel Image Optimizer plugin
array(
'relation' => 'OR',
array(
'key' => '_shortpixel_status',
'compare' => '<',
'value' => '2',
),
array(
'key' => '_shortpixel_status',
'compare' => '>=',
'value' => '3',
),
array(
'key' => '_shortpixel_status',
'compare' => 'NOT EXISTS',
'value' => '',
),
),
// Smush plugin
array(
'key' => 'wp-smpro-smush-data',
'compare' => 'NOT EXISTS',
'value' => '',
),
// Imagify
array(
'key' => '_imagify_optimization_level',
'compare' => 'NOT EXISTS',
'value' => '',
),
// Compress JPEG & PNG images by TinyPNG
array(
'key' => 'tiny_compress_images',
'compare' => 'NOT EXISTS',
'value' => '',
),
);
}
}
/**
* Returns a Updraft_Smush_Manager instance
*/
function Updraft_Smush_Manager() {
return Updraft_Smush_Manager::instance();
}
endif;