HEX
Server: Apache
System: Linux server1.panigaletech.com 5.4.0-1103-aws #111~18.04.1-Ubuntu SMP Tue May 23 20:04:10 UTC 2023 x86_64
User: ubuntu (1000)
PHP: 7.4.30
Disabled: pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals,pcntl_unshare,
Upload Files
File: //proc/thread-self/cwd/wp-content/plugins/wp-crontrol/wp-crontrol.php
<?php
/**
 * Plugin Name: WP Crontrol
 * Plugin URI:  https://wordpress.org/plugins/wp-crontrol/
 * Description: WP Crontrol lets you view and control what's happening in the WP-Cron system.
 * Author:      John Blackbourn & crontributors
 * Author URI:  https://github.com/johnbillion/wp-crontrol/graphs/contributors
 * Version:     1.7.1
 * Text Domain: wp-crontrol
 * Domain Path: /languages/
 * License:     GPL v2 or later
 *
 * LICENSE
 * This file is part of WP Crontrol.
 *
 * WP Crontrol is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * @package    WP Crontrol
 * @author     Edward Dale <scompt@scompt.com> & John Blackbourn <john@johnblackbourn.com>
 * @copyright  Copyright 2008 Edward Dale, 2012-2019 John Blackbourn
 * @license    http://www.gnu.org/licenses/gpl.txt GPL 2.0
 * @link       https://wordpress.org/plugins/wp-crontrol/
 * @since      0.2
 */

defined( 'ABSPATH' ) || die();

/**
 * The main encapsulating class for WP Control.
 */
class Crontrol {

	/**
	 * Hook onto all of the actions and filters needed by the plugin.
	 */
	protected function __construct() {

		$plugin_file = plugin_basename( __FILE__ );

		add_action( 'init',                               array( $this, 'action_init' ) );
		add_action( 'init',                               array( $this, 'action_handle_posts' ) );
		add_action( 'admin_menu',                         array( $this, 'action_admin_menu' ) );
		add_filter( "plugin_action_links_{$plugin_file}", array( $this, 'plugin_action_links' ), 10, 4 );
		add_filter( 'removable_query_args',               array( $this, 'filter_removable_query_args' ) );

		add_action( 'load-tools_page_crontrol_admin_manage_page', array( $this, 'enqueue_code_editor' ) );

		register_activation_hook( __FILE__, array( $this, 'action_activate' ) );

		add_filter( 'cron_schedules',    array( $this, 'filter_cron_schedules' ) );
		add_action( 'crontrol_cron_job', array( $this, 'action_php_cron_event' ) );
	}

	/**
	 * Evaluates the provided code using eval.
	 *
	 * @param string $code The PHP code to evaluate.
	 */
	public function action_php_cron_event( $code ) {
		eval( $code ); // @codingStandardsIgnoreLine
	}

	/**
	 * Run using the 'init' action.
	 */
	public function action_init() {
		load_plugin_textdomain( 'wp-crontrol', false, dirname( plugin_basename( __FILE__ ) ) . '/languages' );
	}

	/**
	 * Handles any POSTs made by the plugin. Run using the 'init' action.
	 */
	public function action_handle_posts() {
		if ( isset( $_POST['new_cron'] ) ) {
			if ( ! current_user_can( 'manage_options' ) ) {
				wp_die( esc_html__( 'You are not allowed to add new cron events.', 'wp-crontrol' ), 401 );
			}
			check_admin_referer( 'new-cron' );
			extract( wp_unslash( $_POST ), EXTR_PREFIX_ALL, 'in' );
			if ( 'crontrol_cron_job' === $in_hookname && ! current_user_can( 'edit_files' ) ) {
				wp_die( esc_html__( 'You are not allowed to add new PHP cron events.', 'wp-crontrol' ), 401 );
			}
			$in_args = json_decode( $in_args, true );
			$next_run = ( 'custom' === $in_next_run_date ) ? $in_next_run_date_custom : $in_next_run_date;
			$this->add_cron( $next_run, $in_schedule, $in_hookname, $in_args );
			$redirect = array(
				'page'             => 'crontrol_admin_manage_page',
				'crontrol_message' => '5',
				'crontrol_name'    => rawurlencode( $in_hookname ),
			);
			wp_safe_redirect( add_query_arg( $redirect, admin_url( 'tools.php' ) ) );
			exit;

		} elseif ( isset( $_POST['new_php_cron'] ) ) {
			if ( ! current_user_can( 'edit_files' ) ) {
				wp_die( esc_html__( 'You are not allowed to add new PHP cron events.', 'wp-crontrol' ), 401 );
			}
			check_admin_referer( 'new-cron' );
			extract( wp_unslash( $_POST ), EXTR_PREFIX_ALL, 'in' );
			$args = array(
				'code' => $in_hookcode,
				'name' => $in_eventname,
			);
			$next_run = ( 'custom' === $in_next_run_date ) ? $in_next_run_date_custom : $in_next_run_date;
			$this->add_cron( $next_run, $in_schedule, 'crontrol_cron_job', $args );
			$hookname = ( ! empty( $in_eventname ) ) ? $in_eventname : __( 'PHP Cron', 'wp-crontrol' );
			$redirect = array(
				'page'             => 'crontrol_admin_manage_page',
				'crontrol_message' => '5',
				'crontrol_name'    => rawurlencode( $hookname ),
			);
			wp_safe_redirect( add_query_arg( $redirect, admin_url( 'tools.php' ) ) );
			exit;

		} elseif ( isset( $_POST['edit_cron'] ) ) {
			if ( ! current_user_can( 'manage_options' ) ) {
				wp_die( esc_html__( 'You are not allowed to edit cron events.', 'wp-crontrol' ), 401 );
			}

			extract( wp_unslash( $_POST ), EXTR_PREFIX_ALL, 'in' );
			check_admin_referer( "edit-cron_{$in_original_hookname}_{$in_original_sig}_{$in_original_next_run}" );

			if ( 'crontrol_cron_job' === $in_hookname && ! current_user_can( 'edit_files' ) ) {
				wp_die( esc_html__( 'You are not allowed to edit PHP cron events.', 'wp-crontrol' ), 401 );
			}

			$in_args = json_decode( $in_args, true );
			$i = $this->delete_cron( $in_original_hookname, $in_original_sig, $in_original_next_run );
			$next_run = ( 'custom' === $in_next_run_date ) ? $in_next_run_date_custom : $in_next_run_date;
			$i = $this->add_cron( $next_run, $in_schedule, $in_hookname, $in_args );
			$redirect = array(
				'page'             => 'crontrol_admin_manage_page',
				'crontrol_message' => '4',
				'crontrol_name'    => rawurlencode( $in_hookname ),
			);
			wp_safe_redirect( add_query_arg( $redirect, admin_url( 'tools.php' ) ) );
			exit;

		} elseif ( isset( $_POST['edit_php_cron'] ) ) {
			if ( ! current_user_can( 'edit_files' ) ) {
				wp_die( esc_html__( 'You are not allowed to edit PHP cron events.', 'wp-crontrol' ), 401 );
			}

			extract( wp_unslash( $_POST ), EXTR_PREFIX_ALL, 'in' );
			check_admin_referer( "edit-cron_{$in_original_hookname}_{$in_original_sig}_{$in_original_next_run}" );
			$args['code'] = $in_hookcode;
			$args['name'] = $in_eventname;
			$args = array(
				'code' => $in_hookcode,
				'name' => $in_eventname,
			);
			$i = $this->delete_cron( $in_original_hookname, $in_original_sig, $in_original_next_run );
			$next_run = ( 'custom' === $in_next_run_date ) ? $in_next_run_date_custom : $in_next_run_date;
			$i = $this->add_cron( $next_run, $in_schedule, 'crontrol_cron_job', $args );
			$hookname = ( ! empty( $in_eventname ) ) ? $in_eventname : __( 'PHP Cron', 'wp-crontrol' );
			$redirect = array(
				'page'             => 'crontrol_admin_manage_page',
				'crontrol_message' => '4',
				'crontrol_name'    => rawurlencode( $hookname ),
			);
			wp_safe_redirect( add_query_arg( $redirect, admin_url( 'tools.php' ) ) );
			exit;

		} elseif ( isset( $_POST['new_schedule'] ) ) {
			if ( ! current_user_can( 'manage_options' ) ) {
				wp_die( esc_html__( 'You are not allowed to add new cron schedules.', 'wp-crontrol' ), 401 );
			}
			check_admin_referer( 'new-sched' );
			$name = wp_unslash( $_POST['internal_name'] );
			$interval = wp_unslash( $_POST['interval'] );
			$display = wp_unslash( $_POST['display_name'] );

			// The user entered something that wasn't a number.
			// Try to convert it with strtotime.
			if ( ! is_numeric( $interval ) ) {
				$now = time();
				$future = strtotime( $interval, $now );
				if ( false === $future || -1 == $future || $now > $future ) {
					$redirect = array(
						'page'             => 'crontrol_admin_options_page',
						'crontrol_message' => '7',
						'crontrol_name'    => rawurlencode( $interval ),
					);
					wp_safe_redirect( add_query_arg( $redirect, admin_url( 'options-general.php' ) ) );
					exit;
				}
				$interval = $future - $now;
			} elseif ( $interval <= 0 ) {
				$redirect = array(
					'page'             => 'crontrol_admin_options_page',
					'crontrol_message' => '7',
					'crontrol_name'    => rawurlencode( $interval ),
				);
				wp_safe_redirect( add_query_arg( $redirect, admin_url( 'options-general.php' ) ) );
				exit;
			}

			$this->add_schedule( $name, $interval, $display );
			$redirect = array(
				'page'             => 'crontrol_admin_options_page',
				'crontrol_message' => '3',
				'crontrol_name'    => rawurlencode( $name ),
			);
			wp_safe_redirect( add_query_arg( $redirect, admin_url( 'options-general.php' ) ) );
			exit;

		} elseif ( isset( $_GET['action'] ) && 'delete-sched' == $_GET['action'] ) {
			if ( ! current_user_can( 'manage_options' ) ) {
				wp_die( esc_html__( 'You are not allowed to delete cron schedules.', 'wp-crontrol' ), 401 );
			}
			$id = wp_unslash( $_GET['id'] );
			check_admin_referer( "delete-sched_{$id}" );
			$this->delete_schedule( $id );
			$redirect = array(
				'page'             => 'crontrol_admin_options_page',
				'crontrol_message' => '2',
				'crontrol_name'    => rawurlencode( $id ),
			);
			wp_safe_redirect( add_query_arg( $redirect, admin_url( 'options-general.php' ) ) );
			exit;

		} elseif ( isset( $_POST['delete_crons'] ) ) {
			if ( ! current_user_can( 'manage_options' ) ) {
				wp_die( esc_html__( 'You are not allowed to delete cron events.', 'wp-crontrol' ), 401 );
			}
			check_admin_referer( 'bulk-delete-crons' );

			if ( empty( $_POST['delete'] ) ) {
				return;
			}

			$delete  = wp_unslash( $_POST['delete'] );
			$deleted = 0;

			foreach ( $delete as $next_run => $events ) {
				foreach ( $events as $id => $sig ) {
					if ( 'crontrol_cron_job' === $id && ! current_user_can( 'edit_files' ) ) {
						continue;
					}
					if ( $this->delete_cron( urldecode( $id ), $sig, $next_run ) ) {
						$deleted++;
					}
				}
			}

			$redirect = array(
				'page'             => 'crontrol_admin_manage_page',
				'crontrol_name'    => $deleted,
				'crontrol_message' => '9',
			);
			wp_safe_redirect( add_query_arg( $redirect, admin_url( 'tools.php' ) ) );
			exit;

		} elseif ( isset( $_GET['action'] ) && 'delete-cron' == $_GET['action'] ) {
			if ( ! current_user_can( 'manage_options' ) ) {
				wp_die( esc_html__( 'You are not allowed to delete cron events.', 'wp-crontrol' ), 401 );
			}
			$id = wp_unslash( $_GET['id'] );
			$sig = wp_unslash( $_GET['sig'] );
			$next_run = intval( $_GET['next_run'] );
			check_admin_referer( "delete-cron_{$id}_{$sig}_{$next_run}" );

			if ( 'crontrol_cron_job' === $id && ! current_user_can( 'edit_files' ) ) {
				wp_die( esc_html__( 'You are not allowed to delete PHP cron events.', 'wp-crontrol' ), 401 );
			}

			if ( $this->delete_cron( $id, $sig, $next_run ) ) {
				$redirect = array(
					'page'             => 'crontrol_admin_manage_page',
					'crontrol_message' => '6',
					'crontrol_name'    => rawurlencode( $id ),
				);
				wp_safe_redirect( add_query_arg( $redirect, admin_url( 'tools.php' ) ) );
				exit;
			} else {
				$redirect = array(
					'page'             => 'crontrol_admin_manage_page',
					'crontrol_message' => '7',
					'crontrol_name'    => rawurlencode( $id ),
				);
				wp_safe_redirect( add_query_arg( $redirect, admin_url( 'tools.php' ) ) );
				exit;

			};

		} elseif ( isset( $_GET['action'] ) && 'run-cron' == $_GET['action'] ) {
			if ( ! current_user_can( 'manage_options' ) ) {
				wp_die( esc_html__( 'You are not allowed to run cron events.', 'wp-crontrol' ), 401 );
			}
			$id = wp_unslash( $_GET['id'] );
			$sig = wp_unslash( $_GET['sig'] );
			check_admin_referer( "run-cron_{$id}_{$sig}" );
			if ( $this->run_cron( $id, $sig ) ) {
				$redirect = array(
					'page'             => 'crontrol_admin_manage_page',
					'crontrol_message' => '1',
					'crontrol_name'    => rawurlencode( $id ),
				);
				wp_safe_redirect( add_query_arg( $redirect, admin_url( 'tools.php' ) ) );
				exit;
			} else {
				$redirect = array(
					'page'             => 'crontrol_admin_manage_page',
					'crontrol_message' => '8',
					'crontrol_name'    => rawurlencode( $id ),
				);
				wp_safe_redirect( add_query_arg( $redirect, admin_url( 'tools.php' ) ) );
				exit;
			}
		}
	}

	/**
	 * Executes a cron event immediately.
	 *
	 * Executes an event by scheduling a new single event with the same arguments.
	 *
	 * @param string $hookname The hook name of the cron event to run.
	 * @param string $sig      The cron event signature.
	 */
	public function run_cron( $hookname, $sig ) {
		$crons = _get_cron_array();
		foreach ( $crons as $time => $cron ) {
			if ( isset( $cron[ $hookname ][ $sig ] ) ) {
				$args = $cron[ $hookname ][ $sig ]['args'];
				delete_transient( 'doing_cron' );
				wp_schedule_single_event( time() - 1, $hookname, $args );
				spawn_cron();
				return true;
			}
		}
		return false;
	}

	/**
	 * Adds a new cron event.
	 *
	 * @param string $next_run A GMT time that the event should be run at.
	 * @param string $schedule The recurrence of the cron event.
	 * @param string $hookname The name of the hook to execute.
	 * @param array  $args     Arguments to add to the cron event.
	 */
	public function add_cron( $next_run, $schedule, $hookname, $args ) {
		$next_run = strtotime( $next_run );
		if ( false === $next_run ) {
			$next_run = time();
		} else {
			$next_run = get_gmt_from_date( date( 'Y-m-d H:i:s', $next_run ), 'U' );
		}
		if ( ! is_array( $args ) ) {
			$args = array();
		}
		if ( '_oneoff' === $schedule ) {
			return wp_schedule_single_event( $next_run, $hookname, $args ) === null;
		} else {
			return wp_schedule_event( $next_run, $schedule, $hookname, $args ) === null;
		}
	}

	/**
	 * Deletes a cron event.
	 *
	 * @param string $to_delete The hook name of the event to delete.
	 * @param string $sig       The cron event signature.
	 * @param string $next_run  The GMT time that the event would be run at.
	 */
	public function delete_cron( $to_delete, $sig, $next_run ) {
		$crons = _get_cron_array();
		if ( isset( $crons[ $next_run ][ $to_delete ][ $sig ] ) ) {
			$args = $crons[ $next_run ][ $to_delete ][ $sig ]['args'];
			wp_unschedule_event( $next_run, $to_delete, $args );
			return true;
		}
		return false;
	}

	/**
	 * Adds a new custom cron schedule.
	 *
	 * @param string $name     The internal name of the schedule.
	 * @param int    $interval The interval between executions of the new schedule.
	 * @param string $display  The display name of the schedule.
	 */
	public function add_schedule( $name, $interval, $display ) {
		$old_scheds = get_option( 'crontrol_schedules', array() );
		$old_scheds[ $name ] = array(
			'interval' => $interval,
			'display'  => $display,
		);
		update_option( 'crontrol_schedules', $old_scheds );
	}

	/**
	 * Deletes a custom cron schedule.
	 *
	 * @param string $name The internal_name of the schedule to delete.
	 */
	public function delete_schedule( $name ) {
		$scheds = get_option( 'crontrol_schedules', array() );
		unset( $scheds[ $name ] );
		update_option( 'crontrol_schedules', $scheds );
	}

	/**
	 * Sets up the plugin environment upon first activation.
	 *
	 * Run using the 'activate_' action.
	 */
	public function action_activate() {
		add_option( 'crontrol_schedules', array() );

		// If there's never been a cron event, _get_cron_array will return false.
		if ( _get_cron_array() === false ) {
			_set_cron_array( array() );
		}
	}

	/**
	 * Adds options & management pages to the admin menu.
	 *
	 * Run using the 'admin_menu' action.
	 */
	public function action_admin_menu() {
		add_options_page( esc_html__( 'Cron Schedules', 'wp-crontrol' ), esc_html__( 'Cron Schedules', 'wp-crontrol' ), 'manage_options', 'crontrol_admin_options_page', array( $this, 'admin_options_page' ) );
		add_management_page( esc_html__( 'Cron Events', 'wp-crontrol' ), esc_html__( 'Cron Events', 'wp-crontrol' ), 'manage_options', 'crontrol_admin_manage_page', array( $this, 'admin_manage_page' ) );
	}

	/**
	 * Adds items to the plugin's action links on the Plugins listing screen.
	 *
	 * @param string[] $actions     Array of action links.
	 * @param string   $plugin_file Path to the plugin file relative to the plugins directory.
	 * @param array    $plugin_data An array of plugin data.
	 * @param string   $context     The plugin context.
	 * @return string[] Array of action links.
	 */
	public function plugin_action_links( $actions, $plugin_file, $plugin_data, $context ) {
		$actions['crontrol-events'] = sprintf(
			'<a href="%s">%s</a>',
			esc_url( admin_url( 'tools.php?page=crontrol_admin_manage_page' ) ),
			esc_html__( 'Cron Events', 'wp-crontrol' )
		);
		$actions['crontrol-schedules'] = sprintf(
			'<a href="%s">%s</a>',
			esc_url( admin_url( 'options-general.php?page=crontrol_admin_options_page' ) ),
			esc_html__( 'Cron Schedules', 'wp-crontrol' )
		);
		return $actions;
	}

	/**
	 * Gives WordPress the plugin's set of cron schedules.
	 *
	 * Called by the 'cron_schedules' filter.
	 *
	 * @param array $scheds The current cron schedules. Usually an empty array.
	 * @return array The existing cron schedules along with the plugin's schedules.
	 */
	public function filter_cron_schedules( $scheds ) {
		$new_scheds = get_option( 'crontrol_schedules', array() );
		return array_merge( $new_scheds, $scheds );
	}

	/**
	 * Displays the options page for the plugin.
	 */
	public function admin_options_page() {
		$schedules = $this->get_schedules();
		$events = $this->get_cron_events();
		$custom_schedules = get_option( 'crontrol_schedules', array() );
		$custom_keys = array_keys( $custom_schedules );

		if ( is_wp_error( $events ) ) {
			$events = array();
		}

		$used_schedules = array_unique( wp_list_pluck( $events, 'schedule' ) );

		$messages = array(
			/* translators: 1: The name of the cron schedule. */
			'2' => __( 'Successfully deleted the cron schedule %s.', 'wp-crontrol' ),
			/* translators: 1: The name of the cron schedule. */
			'3' => __( 'Successfully added the cron schedule %s.', 'wp-crontrol' ),
			/* translators: 1: The name of the cron schedule. */
			'7' => __( 'Cron schedule not added because there was a problem parsing %s.', 'wp-crontrol' ),
		);
		if ( isset( $_GET['crontrol_message'] ) && isset( $_GET['crontrol_name'] ) && isset( $messages[ $_GET['crontrol_message'] ] ) ) {
			$hook = wp_unslash( $_GET['crontrol_name'] );
			$message = wp_unslash( $_GET['crontrol_message'] );
			$msg  = sprintf( esc_html( $messages[ $message ] ), '<strong>' . esc_html( $hook ) . '</strong>' );

			printf( '<div id="message" class="updated notice is-dismissible"><p>%s</p></div>', $msg ); // WPCS:: XSS ok.
		}

		?>
		<div class="wrap">
		<h1><?php esc_html_e( 'WP-Cron Schedules', 'wp-crontrol' ); ?></h1>
		<p><?php esc_html_e( 'WP-Cron schedules are the time intervals that are available for scheduling events. You can only delete custom schedules.', 'wp-crontrol' ); ?></p>
		<table class="widefat striped">
		<thead>
			<tr>
				<th scope="col"><?php esc_html_e( 'Name', 'wp-crontrol' ); ?></th>
				<th scope="col"><?php esc_html_e( 'Interval', 'wp-crontrol' ); ?></th>
				<th scope="col"><?php esc_html_e( 'Display Name', 'wp-crontrol' ); ?></th>
				<th scope="col"><?php esc_html_e( 'Delete', 'wp-crontrol' ); ?></th>
			</tr>
		</thead>
		<tbody>
		<?php
		if ( empty( $schedules ) ) {
			?>
			<tr colspan="4"><td><?php esc_html_e( 'You currently have no schedules. Add one below.', 'wp-crontrol' ); ?></td></tr>
			<?php
		} else {
			foreach ( $schedules as $name => $data ) {
				printf(
					'<tr id="sched-%s">',
					esc_attr( $name )
				);
				printf(
					'<td>%s</td>',
					esc_html( $name )
				);
				printf(
					'<td>%s (%s)</td>',
					esc_html( $data['interval'] ),
					esc_html( $this->interval( $data['interval'] ) )
				);
				printf(
					'<td>%s</td>',
					esc_html( $data['display'] )
				);

				echo '<td>';
				if ( in_array( $name, $custom_keys, true ) ) {
					if ( in_array( $name, $used_schedules, true ) ) {
						esc_html_e( 'This custom schedule is in use and cannot be deleted', 'wp-crontrol' );
					} else {
						$url = add_query_arg( array(
							'page'   => 'crontrol_admin_options_page',
							'action' => 'delete-sched',
							'id'     => rawurlencode( $name ),
						), admin_url( 'options-general.php' ) );
						$url = wp_nonce_url( $url, 'delete-sched_' . $name );
						printf( '<span class="row-actions visible"><span class="delete"><a href="%s">%s</a></span></span>',
							esc_url( $url ),
							esc_html__( 'Delete', 'wp-crontrol' )
						);
					}
				} else {
					echo '&nbsp;';
				}
				echo '</td>';
				echo '</tr>';
			}
		}
		?>
		</tbody>
		</table>
		</div>
		<div class="wrap">
			<p class="description">
				<?php
					printf(
						'<a href="%s">%s</a>',
						esc_url( admin_url( 'tools.php?page=crontrol_admin_manage_page' ) ),
						esc_html__( 'Manage Cron Events', 'wp-crontrol' )
					);
				?>
			</p>
		</div>
		<div class="wrap narrow">
			<h2 class="title"><?php esc_html_e( 'Add Cron Schedule', 'wp-crontrol' ); ?></h2>
			<p><?php esc_html_e( 'Adding a new cron schedule will allow you to schedule events that re-occur at the given interval.', 'wp-crontrol' ); ?></p>
			<form method="post" action="options-general.php?page=crontrol_admin_options_page">
				<table class="form-table">
					<tbody>
					<tr>
						<th valign="top" scope="row"><label for="cron_internal_name"><?php esc_html_e( 'Internal name', 'wp-crontrol' ); ?></label></th>
						<td><input type="text" class="regular-text" value="" id="cron_internal_name" name="internal_name" required/></td>
					</tr>
					<tr>
						<th valign="top" scope="row"><label for="cron_interval"><?php esc_html_e( 'Interval (seconds)', 'wp-crontrol' ); ?></label></th>
						<td><input type="number" class="regular-text" value="" id="cron_interval" name="interval" min="1" step="1" required/></td>
					</tr>
					<tr>
						<th valign="top" scope="row"><label for="cron_display_name"><?php esc_html_e( 'Display name', 'wp-crontrol' ); ?></label></th>
						<td><input type="text" class="regular-text" value="" id="cron_display_name" name="display_name" required/></td>
					</tr>
				</tbody></table>
				<p class="submit"><input id="schedadd-submit" type="submit" class="button-primary" value="<?php esc_attr_e( 'Add Cron Schedule', 'wp-crontrol' ); ?>" name="new_schedule"/></p>
				<?php wp_nonce_field( 'new-sched' ); ?>
			</form>
		</div>
		<?php
	}

	/**
	 * Gets a sorted (according to interval) list of the cron schedules
	 */
	public function get_schedules() {
		$schedules = wp_get_schedules();
		uasort( $schedules, array( $this, 'sort_schedules' ) );
		return $schedules;
	}

	/**
	 * Internal sorting comparison callback function. Compares schedules by their interval.
	 *
	 * @param array $a The first schedule to compare for sorting.
	 * @param array $b The second schedule to compare for sorting.
	 * @return int An integer less than, equal to, or greater than zero if the first argument is considered to be
	 *             respectively less than, equal to, or greater than the second.
	 */
	protected function sort_schedules( $a, $b ) {
		return ( $a['interval'] - $b['interval'] );
	}

	/**
	 * Displays a dropdown filled with the possible schedules, including non-repeating.
	 *
	 * @param bool $current The currently selected schedule.
	 */
	public function schedules_dropdown( $current = false ) {
		$schedules = $this->get_schedules();
		?>
		<select class="postform" name="schedule" id="schedule" required>
		<option <?php selected( $current, '_oneoff' ); ?> value="_oneoff"><?php esc_html_e( 'Non-repeating', 'wp-crontrol' ); ?></option>
		<?php foreach ( $schedules as $sched_name => $sched_data ) { ?>
			<option <?php selected( $current, $sched_name ); ?> value="<?php echo esc_attr( $sched_name ); ?>">
				<?php
				printf(
					'%s (%s)',
					esc_html( $sched_data['display'] ),
					esc_html( $this->interval( $sched_data['interval'] ) )
				);
				?>
			</option>
		<?php } ?>
		</select>
		<?php
	}

	/**
	 * Gets the status of WP-Cron functionality on the site by performing a test spawn. Cached for one hour when all is well.
	 *
	 * @param bool $cache Whether to use the cached result from previous calls.
	 * @return true|WP_Error Boolean true if the cron spawner is working as expected, or a WP_Error object if not.
	 */
	public function test_cron_spawn( $cache = true ) {
		global $wp_version;

		if ( defined( 'DISABLE_WP_CRON' ) && DISABLE_WP_CRON ) {
			/* translators: 1: The name of the PHP constant that is set. */
			return new WP_Error( 'crontrol_info', sprintf( __( 'The %s constant is set to true. WP-Cron spawning is disabled.', 'wp-crontrol' ), 'DISABLE_WP_CRON' ) );
		}

		if ( defined( 'ALTERNATE_WP_CRON' ) && ALTERNATE_WP_CRON ) {
			/* translators: 1: The name of the PHP constant that is set. */
			return new WP_Error( 'crontrol_info', sprintf( __( 'The %s constant is set to true.', 'wp-crontrol' ), 'ALTERNATE_WP_CRON' ) );
		}

		$cached_status = get_transient( 'wp-cron-test-ok' );

		if ( $cache && $cached_status ) {
			return true;
		}

		$sslverify     = version_compare( $wp_version, 4.0, '<' );
		$doing_wp_cron = sprintf( '%.22F', microtime( true ) );

		$cron_request = apply_filters( 'cron_request', array(
			'url'  => site_url( 'wp-cron.php?doing_wp_cron=' . $doing_wp_cron ),
			'key'  => $doing_wp_cron,
			'args' => array(
				'timeout'   => 3,
				'blocking'  => true,
				'sslverify' => apply_filters( 'https_local_ssl_verify', $sslverify ),
			),
		) );

		$cron_request['args']['blocking'] = true;

		$result = wp_remote_post( $cron_request['url'], $cron_request['args'] );

		if ( is_wp_error( $result ) ) {
			return $result;
		} elseif ( wp_remote_retrieve_response_code( $result ) >= 300 ) {
			return new WP_Error( 'unexpected_http_response_code', sprintf(
				/* translators: 1: The HTTP response code. */
				__( 'Unexpected HTTP response code: %s', 'wp-crontrol' ),
				intval( wp_remote_retrieve_response_code( $result ) )
			) );
		} else {
			set_transient( 'wp-cron-test-ok', 1, 3600 );
			return true;
		}

	}

	/**
	 * Shows the status of WP-Cron functionality on the site. Only displays a message when there's a problem.
	 */
	public function show_cron_status() {

		$status = $this->test_cron_spawn();

		if ( is_wp_error( $status ) ) {
			if ( 'crontrol_info' === $status->get_error_code() ) {
				?>
				<div id="cron-status-notice" class="notice notice-info">
					<p><?php echo esc_html( $status->get_error_message() ); ?></p>
				</div>
				<?php
			} else {
				?>
				<div id="cron-status-error" class="error">
					<p>
						<?php
						printf(
							/* translators: 1: Error message text. */
							esc_html__( 'There was a problem spawning a call to the WP-Cron system on your site. This means WP-Cron events on your site may not work. The problem was: %s', 'wp-crontrol' ),
							'<br><strong>' . esc_html( $status->get_error_message() ) . '</strong>'
						);
						?>
					</p>
				</div>
				<?php
			}
		}

	}

	/**
	 * Get the display name for the site's timezone.
	 *
	 * @return string The name and UTC offset for the site's timezone.
	 */
	public function get_timezone_name() {
		$timezone_string = get_option( 'timezone_string', '' );
		$gmt_offset      = get_option( 'gmt_offset', 0 );

		if ( $gmt_offset >= 0 ) {
			$gmt_offset = '+' . $gmt_offset;
		}

		if ( '' === $timezone_string ) {
			$name = sprintf( 'UTC%s', $gmt_offset );
		} else {
			$name = sprintf( '%s (UTC%s)', str_replace( '_', ' ', $timezone_string ), $gmt_offset );
		}

		return $name;
	}

	/**
	 * Shows the form used to add/edit cron events.
	 *
	 * @param bool  $is_php   Whether this is a PHP cron event.
	 * @param mixed $existing An array of existing values for the cron event, or null.
	 */
	public function show_cron_form( $is_php, $existing ) {
		$new_tabs = array(
			'cron'     => __( 'Add Cron Event', 'wp-crontrol' ),
			'php-cron' => __( 'Add PHP Cron Event', 'wp-crontrol' ),
		);
		$modify_tabs = array(
			'cron'     => __( 'Modify Cron Event', 'wp-crontrol' ),
			'php-cron' => __( 'Modify PHP Cron Event', 'wp-crontrol' ),
		);
		$new_links = array(
			'cron'     => admin_url( 'tools.php?page=crontrol_admin_manage_page&action=new-cron' ) . '#crontrol_form',
			'php-cron' => admin_url( 'tools.php?page=crontrol_admin_manage_page&action=new-php-cron' ) . '#crontrol_form',
		);
		$display_args = '';
		if ( $is_php ) {
			$helper_text = esc_html__( 'Cron events trigger actions in your code. Enter the schedule of the event, as well as the PHP code to execute when the action is triggered.', 'wp-crontrol' );
		} else {
			$helper_text = sprintf(
				/* translators: %s: A file name */
				esc_html__( 'Cron events trigger actions in your code. A cron event needs a corresponding action hook somewhere in code, e.g. the %1$s file in your theme.', 'wp-crontrol' ),
				'<code>functions.php</code>'
			);
		}
		if ( is_array( $existing ) ) {
			$other_fields  = wp_nonce_field( "edit-cron_{$existing['hookname']}_{$existing['sig']}_{$existing['next_run']}", '_wpnonce', true, false );
			$other_fields .= sprintf( '<input name="original_hookname" type="hidden" value="%s" />',
				esc_attr( $existing['hookname'] )
			);
			$other_fields .= sprintf( '<input name="original_sig" type="hidden" value="%s" />',
				esc_attr( $existing['sig'] )
			);
			$other_fields .= sprintf( '<input name="original_next_run" type="hidden" value="%s" />',
				esc_attr( $existing['next_run'] )
			);
			if ( ! empty( $existing['args'] ) ) {
				$display_args = wp_json_encode( $existing['args'] );
			}
			$action = $is_php ? 'edit_php_cron' : 'edit_cron';
			$button = $is_php ? $modify_tabs['php-cron'] : $modify_tabs['cron'];
			$show_edit_tab = true;
			$next_run_date = get_date_from_gmt( date( 'Y-m-d H:i:s', $existing['next_run'] ), 'Y-m-d H:i:s' );
		} else {
			$other_fields = wp_nonce_field( 'new-cron', '_wpnonce', true, false );
			$existing = array(
				'hookname' => '',
				'args'     => array(),
				'next_run' => 'now',
				'schedule' => false,
			);
			$action = $is_php ? 'new_php_cron' : 'new_cron';
			$button = $is_php ? $new_tabs['php-cron'] : $new_tabs['cron'];
			$show_edit_tab = false;
			$next_run_date = '';
		}
		if ( $is_php ) {
			if ( ! isset( $existing['args']['code'] ) ) {
				$existing['args']['code'] = '';
			}
			if ( ! isset( $existing['args']['name'] ) ) {
				$existing['args']['name'] = '';
			}
		}

		$allowed = ( ! $is_php || current_user_can( 'edit_files' ) );
		?>
		<div id="crontrol_form" class="wrap narrow">
			<h2 class="nav-tab-wrapper">
				<a href="<?php echo esc_url( $new_links['cron'] ); ?>" class="nav-tab<?php echo ( ! $show_edit_tab && ! $is_php ) ? ' nav-tab-active' : ''; ?>"><?php echo esc_html( $new_tabs['cron'] ); ?></a>
				<a href="<?php echo esc_url( $new_links['php-cron'] ); ?>" class="nav-tab<?php echo ( ! $show_edit_tab && $is_php ) ? ' nav-tab-active' : ''; ?>"><?php echo esc_html( $new_tabs['php-cron'] ); ?></a>
				<?php if ( $show_edit_tab ) { ?>
					<span class="nav-tab nav-tab-active"><?php echo esc_html( $button ); ?></span>
				<?php } ?>
			</h2>
			<?php if ( $allowed ) { ?>
			<p><?php echo $helper_text; // WPCS:: XSS ok. ?></p>
			<form method="post" action="<?php echo esc_url( admin_url( 'tools.php?page=crontrol_admin_manage_page' ) ); ?>">
				<?php echo $other_fields; // WPCS:: XSS ok. ?>
				<table class="form-table"><tbody>
					<?php if ( $is_php ) : ?>
						<tr>
							<th valign="top" scope="row"><label for="hookcode"><?php esc_html_e( 'PHP Code', 'wp-crontrol' ); ?></label></th>
							<td>
								<p class="description">
									<?php
										printf(
											/* translators: The PHP tag name */
											esc_html__( 'The opening %s tag must not be included.', 'wp-crontrol' ),
											'<code>&lt;?php</code>'
										);
									?>
								</p>
								<p><textarea class="large-text code" rows="10" cols="50" id="hookcode" name="hookcode"><?php echo esc_textarea( $existing['args']['code'] ); ?></textarea></p>
							</td>
						</tr>
						<tr>
							<th valign="top" scope="row"><label for="eventname"><?php esc_html_e( 'Event Name (optional)', 'wp-crontrol' ); ?></label></th>
							<td><input type="text" class="regular-text" id="eventname" name="eventname" value="<?php echo esc_attr( $existing['args']['name'] ); ?>"/></td>
						</tr>
					<?php else : ?>
						<tr>
							<th valign="top" scope="row"><label for="hookname"><?php esc_html_e( 'Hook Name', 'wp-crontrol' ); ?></label></th>
							<td><input type="text" class="regular-text" id="hookname" name="hookname" value="<?php echo esc_attr( $existing['hookname'] ); ?>" required /></td>
						</tr>
						<tr>
							<th valign="top" scope="row"><label for="args"><?php esc_html_e( 'Arguments (optional)', 'wp-crontrol' ); ?></label></th>
							<td>
								<input type="text" class="regular-text" id="args" name="args" value="<?php echo esc_attr( $display_args ); ?>"/>
								<p class="description">
									<?php
										printf(
											/* translators: 1, 2, and 3: Example values for an input field. */
											esc_html__( 'Use a JSON encoded array, e.g. %1$s, %2$s, or %3$s', 'wp-crontrol' ),
											'<code>[25]</code>',
											'<code>["asdf"]</code>',
											'<code>["i","want",25,"cakes"]</code>'
										);
									?>
								</p>
							</td>
						</tr>
					<?php endif; ?>
					<tr>
						<th valign="top" scope="row"><label for="next_run_date"><?php esc_html_e( 'Next Run', 'wp-crontrol' ); ?></label></th>
						<td>
							<ul>
								<li>
									<label>
										<input type="radio" name="next_run_date" value="now()" checked>
										<?php esc_html_e( 'Now', 'wp-crontrol' ); ?>
									</label>
								</li>
								<li>
									<label>
										<input type="radio" name="next_run_date" value="+1 day">
										<?php esc_html_e( 'Tomorrow', 'wp-crontrol' ); ?>
									</label>
								</li>
								<li>
									<label>
										<input type="radio" name="next_run_date" value="custom" id="next_run_date_custom" <?php checked( $show_edit_tab ); ?>>
										<?php
										printf(
											/* translators: %s: An input field for specifying a date and time */
											esc_html__( 'At: %s', 'wp-crontrol' ),
											sprintf(
												'<input type="text" name="next_run_date_custom" value="%s" class="regular-text" onfocus="jQuery(\'#next_run_date_custom\').prop(\'checked\',true);" />',
												esc_attr( $next_run_date )
											)
										);
										?>
									</label>
								</li>
							</ul>

							<p class="description">
								<?php
									printf(
										/* translators: 1: Date/time format for an input field, 2: PHP function name. */
										esc_html__( 'Format: %1$s or anything accepted by %2$s', 'wp-crontrol' ),
										'<code>YYYY-MM-DD HH:MM:SS</code>',
										'<a href="https://www.php.net/manual/en/function.strtotime.php"><code>strtotime()</code></a>'
									);
								?>
							</p>
							<p class="description">
								<?php
									printf(
										/* translators: %s Timezone name. */
										esc_html__( 'Timezone: %s', 'wp-crontrol' ),
										'<code>' . esc_html( $this->get_timezone_name() ) . '</code>'
									);
								?>
							</p>
						</td>
					</tr><tr>
						<th valign="top" scope="row"><label for="schedule"><?php esc_html_e( 'Recurrence', 'wp-crontrol' ); ?></label></th>
						<td>
							<?php $this->schedules_dropdown( $existing['schedule'] ); ?>
							<p class="description">
								<?php
								printf(
									'<a href="%s">%s</a>',
									esc_url( admin_url( 'options-general.php?page=crontrol_admin_options_page' ) ),
									esc_html__( 'Manage Cron Schedules', 'wp-crontrol' )
								);
								?>
							</p>
						</td>
					</tr>
				</tbody></table>
				<p class="submit"><input type="submit" class="button-primary" value="<?php echo esc_attr( $button ); ?>" name="<?php echo esc_attr( $action ); ?>"/></p>
			</form>
			<?php } else { ?>
				<div class="error inline">
					<p><?php esc_html_e( 'You cannot add, edit, or delete PHP cron events because your user account does not have the ability to edit files.', 'wp-crontrol' ); ?></p>
				</div>
			<?php } ?>
		</div>
		<?php
	}

	/**
	 * Returns a flattened array of cron events.
	 *
	 * @return array[]|WP_Error An array of cron event arrays, or a WP_Error object if there's an error or no events.
	 */
	public function get_cron_events() {

		$crons  = _get_cron_array();
		$events = array();

		if ( empty( $crons ) ) {
			return new WP_Error(
				'no_events',
				__( 'You currently have no scheduled cron events.', 'wp-crontrol' )
			);
		}

		foreach ( $crons as $time => $cron ) {
			foreach ( $cron as $hook => $dings ) {
				foreach ( $dings as $sig => $data ) {

					// This is a prime candidate for a Crontrol_Event class but I'm not bothering currently.
					$events[ "$hook-$sig-$time" ] = (object) array(
						'hook'     => $hook,
						'time'     => $time,
						'sig'      => $sig,
						'args'     => $data['args'],
						'schedule' => $data['schedule'],
						'interval' => isset( $data['interval'] ) ? $data['interval'] : null,
					);

				}
			}
		}

		return $events;

	}

	/**
	 * Displays the manage page for the plugin.
	 */
	public function admin_manage_page() {
		$messages = array(
			/* translators: 1: The name of the cron event. */
			'1' => __( 'Successfully executed the cron event %s.', 'wp-crontrol' ),
			/* translators: 1: The name of the cron event. */
			'4' => __( 'Successfully edited the cron event %s.', 'wp-crontrol' ),
			/* translators: 1: The name of the cron event. */
			'5' => __( 'Successfully created the cron event %s.', 'wp-crontrol' ),
			/* translators: 1: The name of the cron event. */
			'6' => __( 'Successfully deleted the cron event %s.', 'wp-crontrol' ),
			/* translators: 1: The name of the cron event. */
			'7' => __( 'Failed to the delete the cron event %s.', 'wp-crontrol' ),
			/* translators: 1: The name of the cron event. */
			'8' => __( 'Failed to the execute the cron event %s.', 'wp-crontrol' ),
			'9' => __( 'Successfully deleted the selected cron events.', 'wp-crontrol' ),
		);
		if ( isset( $_GET['crontrol_name'] ) && isset( $_GET['crontrol_message'] ) && isset( $messages[ $_GET['crontrol_message'] ] ) ) {
			$hook = wp_unslash( $_GET['crontrol_name'] );
			$message = wp_unslash( $_GET['crontrol_message'] );
			$msg = sprintf( esc_html( $messages[ $message ] ), '<strong>' . esc_html( $hook ) . '</strong>' );

			printf( '<div id="message" class="updated notice is-dismissible"><p>%s</p></div>', $msg ); // WPCS:: XSS ok.
		}
		$events = $this->get_cron_events();
		$doing_edit = ( isset( $_GET['action'] ) && 'edit-cron' === $_GET['action'] ) ? wp_unslash( $_GET['id'] ) : false;
		$time_format = 'Y-m-d H:i:s';
		$can_edit_files = current_user_can( 'edit_files' );

		$core_hooks = array(
			'wp_version_check',
			'wp_update_plugins',
			'wp_update_themes',
			'wp_scheduled_delete',
			'wp_scheduled_auto_draft_delete',
			'update_network_counts',
			'delete_expired_transients',
		);

		$this->show_cron_status();

		?>
		<div class="wrap">
		<h1><?php esc_html_e( 'WP-Cron Events', 'wp-crontrol' ); ?></h1>
		<form method="post" action="tools.php?page=crontrol_admin_manage_page">
		<table class="widefat striped">
		<thead>
			<tr>
				<td id="cb" class="manage-column column-cb check-column"><label class="screen-reader-text" for="cb-select-all-1"><?php esc_html_e( 'Select All', 'wp-crontrol' ); ?></label><input id="cb-select-all-1" type="checkbox"></td>
				<th scope="col"><?php esc_html_e( 'Hook Name', 'wp-crontrol' ); ?></th>
				<th scope="col"><?php esc_html_e( 'Arguments', 'wp-crontrol' ); ?></th>
				<th scope="col"><?php esc_html_e( 'Actions', 'wp-crontrol' ); ?></th>
				<th scope="col"><?php esc_html_e( 'Next Run', 'wp-crontrol' ); ?></th>
				<th scope="col"><?php esc_html_e( 'Recurrence', 'wp-crontrol' ); ?></th>
				<th scope="col"><span class="screen-reader-text"><?php esc_html_e( 'Actions', 'wp-crontrol' ); ?></span></th>
			</tr>
		</thead>
		<tbody>
		<?php
		if ( is_wp_error( $events ) ) {
			?>
			<tr><td colspan="7"><?php echo esc_html( $events->get_error_message() ); ?></td></tr>
			<?php
		} else {
			foreach ( $events as $id => $event ) {

				if ( $doing_edit && $doing_edit == $event->hook && $event->time == $_GET['next_run'] && $event->sig == $_GET['sig'] ) {
					$doing_edit = array(
						'hookname' => $event->hook,
						'next_run' => $event->time,
						'schedule' => ( $event->schedule ? $event->schedule : '_oneoff' ),
						'sig'      => $event->sig,
						'args'     => $event->args,
					);
				}

				if ( empty( $event->args ) ) {
					$args = '<em>' . esc_html__( 'None', 'wp-crontrol' ) . '</em>';
				} else {
					$json_options = 0;

					if ( defined( 'JSON_UNESCAPED_SLASHES' ) ) {
						$json_options |= JSON_UNESCAPED_SLASHES;
					}
					if ( defined( 'JSON_PRETTY_PRINT' ) ) {
						$json_options |= JSON_PRETTY_PRINT;
					}

					$args = '<pre style="white-space:pre-wrap;margin-top:0">' . wp_json_encode( $event->args, $json_options ) . '</pre>';
				}

				echo '<tr>';

				echo '<th scope="row" class="check-column">';
				if ( ! in_array( $event->hook, $core_hooks, true ) ) {
					printf(
						'<input type="checkbox" name="delete[%1$s][%2$s]" value="%3$s">',
						esc_attr( $event->time ),
						esc_attr( rawurlencode( $event->hook ) ),
						esc_attr( $event->sig )
					);
				}
				echo '</th>';

				if ( 'crontrol_cron_job' === $event->hook ) {
					if ( ! empty( $event->args['name'] ) ) {
						/* translators: 1: The name of the PHP cron event. */
						echo '<td><em>' . esc_html( sprintf( __( 'PHP Cron (%s)', 'wp-crontrol' ), $event->args['name'] ) ) . '</em></td>';
					} else {
						echo '<td><em>' . esc_html__( 'PHP Cron', 'wp-crontrol' ) . '</em></td>';
					}
					echo '<td><em>' . esc_html__( 'PHP Code', 'wp-crontrol' ) . '</em></td>';
					echo '<td><em>' . esc_html__( 'WP Crontrol', 'wp-crontrol' ) . '</em></td>';
				} else {
					echo '<td>' . esc_html( $event->hook ) . '</td>';
					echo '<td>' . $args . '</td>'; // WPCS:: XSS ok.
					echo '<td>';
					$callbacks = array();
					foreach ( $this->get_action_callbacks( $event->hook ) as $callback ) {
						$callbacks[] = '<pre style="margin-top:0">' . self::output_callback( $callback ) . '</pre>';
					}
					echo implode( '', $callbacks ); // WPCS:: XSS ok.
					echo '</td>';
				}

				echo '<td style="white-space:nowrap">';
				printf( '%s (%s)',
					esc_html( get_date_from_gmt( date( 'Y-m-d H:i:s', $event->time ), $time_format ) ),
					esc_html( $this->time_since( time(), $event->time ) )
				);
				echo '</td>';

				echo '<td style="white-space:nowrap">';
				if ( $event->schedule ) {
					$schedule_name = $this->get_schedule_name( $event );
					if ( is_wp_error( $schedule_name ) ) {
						printf(
							'<span class="dashicons dashicons-warning"></span>%s',
							esc_html( $schedule_name->get_error_message() )
						);
					} else {
						echo esc_html( $schedule_name );
					}
				} else {
					esc_html_e( 'Non-repeating', 'wp-crontrol' );
				}
				echo '</td>';

				$links = array();

				echo '<td style="white-space:nowrap"><span class="row-actions visible">';

				if ( ( 'crontrol_cron_job' !== $event->hook ) || $can_edit_files ) {
					$link = array(
						'page'     => 'crontrol_admin_manage_page',
						'action'   => 'edit-cron',
						'id'       => rawurlencode( $event->hook ),
						'sig'      => rawurlencode( $event->sig ),
						'next_run' => rawurlencode( $event->time ),
					);
					$link = add_query_arg( $link, admin_url( 'tools.php' ) ) . '#crontrol_form';
					$links[] = "<a href='" . esc_url( $link ) . "'>" . esc_html__( 'Edit', 'wp-crontrol' ) . '</a>';
				}

				$link = array(
					'page'     => 'crontrol_admin_manage_page',
					'action'   => 'run-cron',
					'id'       => rawurlencode( $event->hook ),
					'sig'      => rawurlencode( $event->sig ),
					'next_run' => rawurlencode( $event->time ),
				);
				$link = add_query_arg( $link, admin_url( 'tools.php' ) );
				$link = wp_nonce_url( $link, "run-cron_{$event->hook}_{$event->sig}" );
				$links[] = "<a href='" . esc_url( $link ) . "'>" . esc_html__( 'Run Now', 'wp-crontrol' ) . '</a>';

				if ( ! in_array( $event->hook, $core_hooks, true ) && ( ( 'crontrol_cron_job' !== $event->hook ) || $can_edit_files ) ) {
					$link = array(
						'page'     => 'crontrol_admin_manage_page',
						'action'   => 'delete-cron',
						'id'       => rawurlencode( $event->hook ),
						'sig'      => rawurlencode( $event->sig ),
						'next_run' => rawurlencode( $event->time ),
					);
					$link = add_query_arg( $link, admin_url( 'tools.php' ) );
					$link = wp_nonce_url( $link, "delete-cron_{$event->hook}_{$event->sig}_{$event->time}" );
					$links[] = "<span class='delete'><a href='" . esc_url( $link ) . "'>" . esc_html__( 'Delete', 'wp-crontrol' ) . '</a></span>';
				}

				echo implode( ' | ', $links ); // WPCS:: XSS ok.
				echo '</span></td>';
				echo '</tr>';

			}
		}
		?>
		</tbody>
		</table>
		<p style="float:right">
			<?php
				echo esc_html( sprintf(
					/* translators: %s: The current date and time */
					__( 'Current site time: %s', 'wp-crontrol' ),
					date_i18n( 'Y-m-d H:i:s' )
				) );
			?>
		</p>
		<?php
		wp_nonce_field( 'bulk-delete-crons' );
		submit_button(
			__( 'Delete Selected Events', 'wp-crontrol' ),
			'primary large',
			'delete_crons'
		);
		?>
		</form>

		</div>
		<?php
		if ( is_array( $doing_edit ) ) {
			$this->show_cron_form( 'crontrol_cron_job' == $doing_edit['hookname'], $doing_edit );
		} else {
			$this->show_cron_form( ( isset( $_GET['action'] ) && 'new-php-cron' === $_GET['action'] ), false );
		}
	}

	/**
	 * Returns an array of the callback functions that are attached to the given hook name.
	 *
	 * @param string $name The hook name.
	 * @return array[] Array of callbacks attached to the hook.
	 */
	protected function get_action_callbacks( $name ) {
		global $wp_filter;

		$actions = array();

		if ( isset( $wp_filter[ $name ] ) ) {

			// See http://core.trac.wordpress.org/ticket/17817.
			$action = $wp_filter[ $name ];

			foreach ( $action as $priority => $callbacks ) {
				foreach ( $callbacks as $callback ) {
					$callback = self::populate_callback( $callback );

					$actions[] = array(
						'priority' => $priority,
						'callback' => $callback,
					);
				}
			}
		}

		return $actions;
	}

	/**
	 * Populates the details of the given callback function.
	 *
	 * @param array $callback A callback entry.
	 * @return The updated callback entry.
	 */
	public static function populate_callback( array $callback ) {
		// If Query Monitor is installed, use its rich callback analysis.
		if ( method_exists( 'QM_Util', 'populate_callback' ) ) {
			return QM_Util::populate_callback( $callback );
		}

		if ( is_string( $callback['function'] ) && ( false !== strpos( $callback['function'], '::' ) ) ) {
			$callback['function'] = explode( '::', $callback['function'] );
		}

		if ( is_array( $callback['function'] ) ) {
			if ( is_object( $callback['function'][0] ) ) {
				$class  = get_class( $callback['function'][0] );
				$access = '->';
			} else {
				$class  = $callback['function'][0];
				$access = '::';
			}

			$callback['name'] = $class . $access . $callback['function'][1] . '()';
		} elseif ( is_object( $callback['function'] ) ) {
			if ( is_a( $callback['function'], 'Closure' ) ) {
				$callback['name'] = 'Closure';
			} else {
				$class = get_class( $callback['function'] );
				$callback['name'] = $class . '->__invoke()';
			}
		} else {
			$callback['name'] = $callback['function'] . '()';
		}

		return $callback;

	}

	/**
	 * Returns a user-friendly representation of the callback function.
	 *
	 * @param array $callback The callback entry.
	 * @return string The displayable version of the callback name.
	 */
	public static function output_callback( array $callback ) {
		$qm   = WP_PLUGIN_DIR . '/query-monitor/query-monitor.php';
		$html = plugin_dir_path( $qm ) . 'output/Html.php';

		// If Query Monitor is installed, use its rich callback output.
		if ( class_exists( 'QueryMonitor' ) && file_exists( $html ) ) {
			require_once $html;

			if ( class_exists( 'QM_Output_Html' ) ) {
				return QM_Output_Html::output_filename(
					$callback['callback']['name'],
					$callback['callback']['file'],
					$callback['callback']['line']
				);
			}
		}

		return $callback['callback']['name'];
	}

	/**
	 * Pretty-prints the difference in two times.
	 *
	 * @param int $older_date Unix timestamp.
	 * @param int $newer_date Unix timestamp.
	 * @return string The pretty time_since value
	 * @link http://binarybonsai.com/code/timesince.txt
	 */
	public function time_since( $older_date, $newer_date ) {
		return $this->interval( $newer_date - $older_date );
	}

	/**
	 * Converts a period of time in seconds into a human-readable format representing the interval.
	 *
	 * Example:
	 *
	 *     echo self::interval( 90 );
	 *     // 1 minute 30 seconds
	 *
	 * @param  int $since A period of time in seconds.
	 * @return string An interval represented as a string.
	 */
	public function interval( $since ) {
		// Array of time period chunks.
		$chunks = array(
			/* translators: 1: The number of years in an interval of time. */
			array( 60 * 60 * 24 * 365, _n_noop( '%s year', '%s years', 'wp-crontrol' ) ),
			/* translators: 1: The number of months in an interval of time. */
			array( 60 * 60 * 24 * 30, _n_noop( '%s month', '%s months', 'wp-crontrol' ) ),
			/* translators: 1: The number of weeks in an interval of time. */
			array( 60 * 60 * 24 * 7, _n_noop( '%s week', '%s weeks', 'wp-crontrol' ) ),
			/* translators: 1: The number of days in an interval of time. */
			array( 60 * 60 * 24, _n_noop( '%s day', '%s days', 'wp-crontrol' ) ),
			/* translators: 1: The number of hours in an interval of time. */
			array( 60 * 60, _n_noop( '%s hour', '%s hours', 'wp-crontrol' ) ),
			/* translators: 1: The number of minutes in an interval of time. */
			array( 60, _n_noop( '%s minute', '%s minutes', 'wp-crontrol' ) ),
			/* translators: 1: The number of seconds in an interval of time. */
			array( 1, _n_noop( '%s second', '%s seconds', 'wp-crontrol' ) ),
		);

		if ( $since <= 0 ) {
			return __( 'now', 'wp-crontrol' );
		}

		/**
		 * We only want to output two chunks of time here, eg:
		 * x years, xx months
		 * x days, xx hours
		 * so there's only two bits of calculation below:
		 */
		$j = count( $chunks );

		// Step one: the first chunk.
		for ( $i = 0; $i < $j; $i++ ) {
			$seconds = $chunks[ $i ][0];
			$name = $chunks[ $i ][1];

			// Finding the biggest chunk (if the chunk fits, break).
			$count = floor( $since / $seconds );
			if ( $count ) {
				break;
			}
		}

		// Set output var.
		$output = sprintf( translate_nooped_plural( $name, $count, 'wp-crontrol' ), $count );

		// Step two: the second chunk.
		if ( $i + 1 < $j ) {
			$seconds2 = $chunks[ $i + 1 ][0];
			$name2 = $chunks[ $i + 1 ][1];
			$count2 = floor( ( $since - ( $seconds * $count ) ) / $seconds2 );
			if ( $count2 ) {
				// Add to output var.
				$output .= ' ' . sprintf( translate_nooped_plural( $name2, $count2, 'wp-crontrol' ), $count2 );
			}
		}

		return $output;
	}

	/**
	 * Returns the schedule display name for a given event.
	 *
	 * @param stdClass $event A WP-Cron event.
	 * @return string|WP_Error The interval display name, or a WP_Error object if no such schedule exists.
	 */
	protected function get_schedule_name( stdClass $event ) {
		$schedules = $this->get_schedules();

		if ( isset( $schedules[ $event->schedule ] ) ) {
			return $schedules[ $event->schedule ]['display'];
		}

		return new WP_Error( 'unknown_schedule', __( 'Unknown', 'wp-crontrol' ) );
	}

	/**
	 * Enqueues the editor UI that's used for the PHP cron event code editor.
	 */
	public function enqueue_code_editor() {
		if ( ! function_exists( 'wp_enqueue_code_editor' ) ) {
			return;
		}
		if ( ! current_user_can( 'edit_files' ) ) {
			return;
		}

		$settings = wp_enqueue_code_editor( array(
			'type' => 'text/x-php',
		) );

		if ( false === $settings ) {
			return;
		}

		wp_add_inline_script( 'code-editor', sprintf(
			'jQuery( function( $ ) {
				if ( $( "#hookcode" ).length ) {
					wp.codeEditor.initialize( "hookcode", %s );
				}
			} );',
			wp_json_encode( $settings )
		) );
	}

	/**
	 * Filters the list of query arguments which get removed from admin area URLs in WordPress.
	 *
	 * @link https://core.trac.wordpress.org/ticket/23367
	 *
	 * @param string[] $args List of removable query arguments.
	 * @return string[] Updated list of removable query arguments.
	 */
	public function filter_removable_query_args( array $args ) {
		return array_merge( $args, array(
			'crontrol_message',
			'crontrol_name',
		) );
	}

	/**
	 * Singleton instance instantiator.
	 */
	public static function init() {

		static $instance = null;

		if ( ! $instance ) {
			$instance = new Crontrol();
		}

		return $instance;

	}
}

// Get this show on the road.
Crontrol::init();