Search moodle.org's
Developer Documentation

See Release Notes

  • Bug fixes for general core bugs in 4.2.x will end 22 April 2024 (12 months).
  • Bug fixes for security issues in 4.2.x will end 7 October 2024 (18 months).
  • PHP version: minimum PHP 8.0.0 Note: minimum PHP version has increased since Moodle 4.1. PHP 8.1.x is supported too.
<?php
// This file is part of Moodle - http://moodle.org/
//
// Moodle 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 3 of the License, or
// (at your option) any later version.
//
// Moodle 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.
//
// You should have received a copy of the GNU General Public License
// along with Moodle.  If not, see <http://www.gnu.org/licenses/>.

/**
 * Scheduled task abstract class.
 *
 * @package    core
 * @category   task
 * @copyright  2013 Damyon Wiese
 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 */
namespace core\task;

/**
 * Abstract class defining a scheduled task.
 * @copyright  2013 Damyon Wiese
 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
 */
abstract class scheduled_task extends task_base {

    /** Minimum minute value. */
    const MINUTEMIN = 0;
    /** Maximum minute value. */
    const MINUTEMAX = 59;

    /** Minimum hour value. */
    const HOURMIN = 0;
    /** Maximum hour value. */
    const HOURMAX = 23;

    /** Minimum day of month value. */
    const DAYMIN = 1;
    /** Maximum day of month value. */
    const DAYMAX = 31;

    /** Minimum month value. */
    const MONTHMIN = 1;
    /** Maximum month value. */
    const MONTHMAX = 12;

    /** Minimum dayofweek value. */
    const DAYOFWEEKMIN = 0;
    /** Maximum dayofweek value. */
    const DAYOFWEEKMAX = 6;
    /** Maximum dayofweek value allowed in input (7 = 0). */
    const DAYOFWEEKMAXINPUT = 7;

    /**
     * Minute field identifier.
     */
    const FIELD_MINUTE = 'minute';
    /**
     * Hour field identifier.
     */
    const FIELD_HOUR = 'hour';
    /**
     * Day-of-month field identifier.
     */
    const FIELD_DAY = 'day';
    /**
     * Month field identifier.
     */
    const FIELD_MONTH = 'month';
    /**
     * Day-of-week field identifier.
     */
    const FIELD_DAYOFWEEK = 'dayofweek';

    /**
     * Time used for the next scheduled time when a task should never run. This is 2222-01-01 00:00 GMT
     * which is a large time that still fits in 10 digits.
     */
    const NEVER_RUN_TIME = 7952342400;

    /** @var string $hour - Pattern to work out the valid hours */
    private $hour = '*';

    /** @var string $minute - Pattern to work out the valid minutes */
    private $minute = '*';

    /** @var string $day - Pattern to work out the valid days */
    private $day = '*';

    /** @var string $month - Pattern to work out the valid months */
    private $month = '*';

    /** @var string $dayofweek - Pattern to work out the valid dayofweek */
    private $dayofweek = '*';

    /** @var int $lastruntime - When this task was last run */
    private $lastruntime = 0;

    /** @var boolean $customised - Has this task been changed from it's default schedule? */
    private $customised = false;

    /** @var boolean $overridden - Does the task have values set VIA config? */
    private $overridden = false;

    /** @var int $disabled - Is this task disabled in cron? */
    private $disabled = false;

    /**
     * Get the last run time for this scheduled task.
     *
     * @return int
     */
    public function get_last_run_time() {
        return $this->lastruntime;
    }

    /**
     * Set the last run time for this scheduled task.
     *
     * @param int $lastruntime
     */
    public function set_last_run_time($lastruntime) {
        $this->lastruntime = $lastruntime;
    }

    /**
     * Has this task been changed from it's default config?
     *
     * @return bool
     */
    public function is_customised() {
        return $this->customised;
    }

    /**
< * Has this task been changed from it's default config?
> * Set customised for this scheduled task.
* * @param bool */ public function set_customised($customised) { $this->customised = $customised;
> } } > > /** /** > * Determine if this task is using its default configuration changed from the default. Returns true * Has this task been changed from it's default config? > * if it is and false otherwise. Does not rely on the customised field. * > * * @return bool > * @return bool */ > */ public function is_overridden(): bool { > public function has_default_configuration(): bool { return $this->overridden; > $defaulttask = \core\task\manager::get_default_scheduled_task($this::class); } > if ($defaulttask->get_minute() !== $this->get_minute()) { > return false; /** > } * Set the overridden value. > if ($defaulttask->get_hour() != $this->get_hour()) { * > return false; * @param bool $overridden > } */ > if ($defaulttask->get_month() != $this->get_month()) { public function set_overridden(bool $overridden): void { > return false; $this->overridden = $overridden; > } } > if ($defaulttask->get_day_of_week() != $this->get_day_of_week()) { > return false; /** > } * Setter for $minute. Accepts a special 'R' value > if ($defaulttask->get_day() != $this->get_day()) { * which will be translated to a random minute. > return false; * > } * @param string $minute > if ($defaulttask->get_disabled() != $this->get_disabled()) { * @param bool $expandr - if true (default) an 'R' value in a time is expanded to an appropriate int. > return false; * If false, they are left as 'R' > } */ > return true; public function set_minute($minute, $expandr = true) { > } if ($minute === 'R' && $expandr) { > $minute = mt_rand(self::MINUTEMIN, self::MINUTEMAX); > /** } > * Disable the task. $this->minute = $minute; > */ } > public function disable(): void { > $this->set_disabled(true); /** > $this->set_customised(!$this->has_default_configuration()); * Getter for $minute. > \core\task\manager::configure_scheduled_task($this); * > } * @return string > */ > /** public function get_minute() { > * Enable the task. return $this->minute; > */ } > public function enable(): void { > $this->set_disabled(false); /** > $this->set_customised(!$this->has_default_configuration()); * Setter for $hour. Accepts a special 'R' value > \core\task\manager::configure_scheduled_task($this);
* which will be translated to a random hour. * * @param string $hour * @param bool $expandr - if true (default) an 'R' value in a time is expanded to an appropriate int. * If false, they are left as 'R' */ public function set_hour($hour, $expandr = true) { if ($hour === 'R' && $expandr) { $hour = mt_rand(self::HOURMIN, self::HOURMAX); } $this->hour = $hour; } /** * Getter for $hour. * * @return string */ public function get_hour() { return $this->hour; } /** * Setter for $month. * * @param string $month */ public function set_month($month) { $this->month = $month; } /** * Getter for $month. * * @return string */ public function get_month() { return $this->month; } /** * Setter for $day. * * @param string $day */ public function set_day($day) { $this->day = $day; } /** * Getter for $day. * * @return string */ public function get_day() { return $this->day; } /** * Setter for $dayofweek. * * @param string $dayofweek * @param bool $expandr - if true (default) an 'R' value in a time is expanded to an appropriate int. * If false, they are left as 'R' */ public function set_day_of_week($dayofweek, $expandr = true) { if ($dayofweek === 'R' && $expandr) { $dayofweek = mt_rand(self::DAYOFWEEKMIN, self::DAYOFWEEKMAX); } $this->dayofweek = $dayofweek; } /** * Getter for $dayofweek. * * @return string */ public function get_day_of_week() { return $this->dayofweek; } /** * Setter for $disabled. * * @param bool $disabled */ public function set_disabled($disabled) { $this->disabled = (bool)$disabled; } /** * Getter for $disabled. * @return bool */ public function get_disabled() { return $this->disabled; } /** * Override this function if you want this scheduled task to run, even if the component is disabled. * * @return bool */ public function get_run_if_component_disabled() { return false; } /** * Informs whether the given field is valid. * Use the constants FIELD_* to identify the field. * Have to be called after the method set_{field}(string). * * @param string $field field identifier; expected values from constants FIELD_*. * * @return bool true if given field is valid. false otherwise. */ public function is_valid(string $field): bool { return !empty($this->get_valid($field)); } /** * Calculates the list of valid values according to the given field and stored expression. * * @param string $field field identifier. Must be one of those FIELD_*. * * @return array(int) list of matching values. * * @throws \coding_exception when passed an invalid field identifier. */ private function get_valid(string $field): array { switch($field) { case self::FIELD_MINUTE: $min = self::MINUTEMIN; $max = self::MINUTEMAX; break; case self::FIELD_HOUR: $min = self::HOURMIN; $max = self::HOURMAX; break; case self::FIELD_DAY: $min = self::DAYMIN; $max = self::DAYMAX; break; case self::FIELD_MONTH: $min = self::MONTHMIN; $max = self::MONTHMAX; break; case self::FIELD_DAYOFWEEK: $min = self::DAYOFWEEKMIN; $max = self::DAYOFWEEKMAXINPUT; break; default: throw new \coding_exception("Field '$field' is not a valid crontab identifier."); } $result = $this->eval_cron_field($this->{$field}, $min, $max); if ($field === self::FIELD_DAYOFWEEK) { // For day of week, 0 and 7 both mean Sunday; if there is a 7 we set 0. The result array is sorted. if (end($result) === 7) { // Remove last element. array_pop($result); // Insert 0 as first element if it's not there already. if (reset($result) !== 0) { array_unshift($result, 0); } } } return $result; } /** * Take a cron field definition and return an array of valid numbers with the range min-max. * * @param string $field - The field definition. * @param int $min - The minimum allowable value. * @param int $max - The maximum allowable value. * @return array(int) */ public function eval_cron_field($field, $min, $max) { // Cleanse the input. $field = trim($field); // Format for a field is: // <fieldlist> := <range>(/<step>)(,<fieldlist>) // <step> := int // <range> := <any>|<int>|<min-max> // <any> := * // <min-max> := int-int // End of format BNF. // This function is complicated but is covered by unit tests. $range = array(); $matches = array(); preg_match_all('@[0-9]+|\*|,|/|-@', $field, $matches); $last = 0; $inrange = false; $instep = false; foreach ($matches[0] as $match) { if ($match == '*') { array_push($range, range($min, $max)); } else if ($match == '/') { $instep = true; } else if ($match == '-') { $inrange = true; } else if (is_numeric($match)) { if ($min > $match || $match > $max) { // This is a value error: The value lays out of the expected range of values. return []; } if ($instep) { // Normalise range property, account for "5/10". $insteprange = $range[count($range) - 1]; if (!is_array($insteprange)) { $range[count($range) - 1] = range($insteprange, $max); } for ($i = 0; $i < count($range[count($range) - 1]); $i++) { if (($i) % $match != 0) { $range[count($range) - 1][$i] = -1; } } $instep = false; } else if ($inrange) { if (count($range)) { $range[count($range) - 1] = range($last, $match); } $inrange = false; } else { array_push($range, $match); $last = $match; } } } // If inrange or instep were not processed, there is a syntax error. // Cleanup any existing values to show up the error. if ($inrange || $instep) { return []; } // Flatten the result. $result = array(); foreach ($range as $r) { if (is_array($r)) { foreach ($r as $rr) { if ($rr >= $min && $rr <= $max) { $result[$rr] = 1; } } } else if (is_numeric($r)) { if ($r >= $min && $r <= $max) { $result[$r] = 1; } } } $result = array_keys($result); sort($result, SORT_NUMERIC); return $result; } /** * Assuming $list is an ordered list of items, this function returns the item * in the list that is greater than or equal to the current value (or 0). If * no value is greater than or equal, this will return the first valid item in the list. * If list is empty, this function will return 0. * * @param int $current The current value * @param int[] $list The list of valid items. * @return int $next. */ private function next_in_list($current, $list) { foreach ($list as $l) { if ($l >= $current) { return $l; } } if (count($list)) { return $list[0]; } return 0; } /** * Calculate when this task should next be run based on the schedule. * * @param int $now Current time, for testing (leave 0 to use default time) * @return int $nextruntime. */ public function get_next_scheduled_time(int $now = 0): int { if (!$now) { $now = time(); } // We need to change to the server timezone before using php date() functions. \core_date::set_default_server_timezone(); $validminutes = $this->get_valid(self::FIELD_MINUTE); $validhours = $this->get_valid(self::FIELD_HOUR); $validdays = $this->get_valid(self::FIELD_DAY); $validdaysofweek = $this->get_valid(self::FIELD_DAYOFWEEK); $validmonths = $this->get_valid(self::FIELD_MONTH); // If any of the fields contain no valid data then the task will never run. if (!$validminutes || !$validhours || !$validdays || !$validdaysofweek || !$validmonths) { return self::NEVER_RUN_TIME; } $result = self::get_next_scheduled_time_inner($now, $validminutes, $validhours, $validdays, $validdaysofweek, $validmonths); return $result; } /** * Recursively calculate the next valid time for this task. * * @param int $now Start time * @param array $validminutes Valid minutes * @param array $validhours Valid hours * @param array $validdays Valid days * @param array $validdaysofweek Valid days of week * @param array $validmonths Valid months * @param int $originalyear Zero for first call, original year for recursive calls * @return int Next run time */ protected function get_next_scheduled_time_inner(int $now, array $validminutes, array $validhours, array $validdays, array $validdaysofweek, array $validmonths, int $originalyear = 0) { $currentyear = (int)date('Y', $now); if ($originalyear) { // In recursive calls, check we didn't go more than 8 years ahead, that indicates the // user has chosen an impossible date. 8 years is the maximum time, considering a task // set to run on 29 February over a century boundary when a leap year is skipped. if ($currentyear - $originalyear > 8) { // Use this time if it's never going to happen. return self::NEVER_RUN_TIME; } $firstyear = $originalyear; } else { $firstyear = $currentyear; } $currentmonth = (int)date('n', $now); // Evaluate month first. $nextvalidmonth = $this->next_in_list($currentmonth, $validmonths); if ($nextvalidmonth < $currentmonth) { $currentyear += 1; } // If we moved to another month, set the current time to start of month, and restart calculations. if ($nextvalidmonth !== $currentmonth) { $newtime = strtotime($currentyear . '-' . $nextvalidmonth . '-01 00:00'); return $this->get_next_scheduled_time_inner($newtime, $validminutes, $validhours, $validdays, $validdaysofweek, $validmonths, $firstyear); } // Special handling for dayofmonth vs dayofweek (see man 5 cron). If both are specified, then // it is ok to continue when either matches. If only one is specified then it must match. $currentday = (int)date("j", $now); $currentdayofweek = (int)date("w", $now); $nextvaliddayofmonth = self::next_in_list($currentday, $validdays); $nextvaliddayofweek = self::next_in_list($currentdayofweek, $validdaysofweek); $daysincrementbymonth = $nextvaliddayofmonth - $currentday; $daysinmonth = (int)date('t', $now); if ($nextvaliddayofmonth < $currentday) { $daysincrementbymonth += $daysinmonth; } $daysincrementbyweek = $nextvaliddayofweek - $currentdayofweek; if ($nextvaliddayofweek < $currentdayofweek) { $daysincrementbyweek += 7; } if ($this->dayofweek == '*') { $daysincrement = $daysincrementbymonth; } else if ($this->day == '*') { $daysincrement = $daysincrementbyweek; } else { // Take the smaller increment of days by month or week. $daysincrement = min($daysincrementbymonth, $daysincrementbyweek); } // If we moved day, recurse using new start time. if ($daysincrement != 0) { $newtime = strtotime($currentyear . '-' . $currentmonth . '-' . $currentday . ' 00:00 +' . $daysincrement . ' days'); return $this->get_next_scheduled_time_inner($newtime, $validminutes, $validhours, $validdays, $validdaysofweek, $validmonths, $firstyear); } $currenthour = (int)date('H', $now); $nextvalidhour = $this->next_in_list($currenthour, $validhours); if ($nextvalidhour != $currenthour) { if ($nextvalidhour < $currenthour) { $offset = ' +1 day'; } else { $offset = ''; } $newtime = strtotime($currentyear . '-' . $currentmonth . '-' . $currentday . ' ' . $nextvalidhour . ':00' . $offset); return $this->get_next_scheduled_time_inner($newtime, $validminutes, $validhours, $validdays, $validdaysofweek, $validmonths, $firstyear); } // Round time down to an exact minute because we need to use numeric calculations on it now. // If we construct times based on all the components, it will mess up around DST changes // (because there are two times with the same representation). $now = intdiv($now, 60) * 60; $currentminute = (int)date('i', $now); $nextvalidminute = $this->next_in_list($currentminute, $validminutes); if ($nextvalidminute == $currentminute && !$originalyear) { // This is not a recursive call so time has not moved on at all yet. We can't use the // same minute as now because it has already happened, it has to be at least one minute // later, so update time and retry. $newtime = $now + 60; return $this->get_next_scheduled_time_inner($newtime, $validminutes, $validhours, $validdays, $validdaysofweek, $validmonths, $firstyear); } if ($nextvalidminute < $currentminute) { // The time is in the next hour so we need to recurse. Don't use strtotime at this // point because it will mess up around DST changes. $minutesforward = $nextvalidminute + 60 - $currentminute; $newtime = $now + $minutesforward * 60; return $this->get_next_scheduled_time_inner($newtime, $validminutes, $validhours, $validdays, $validdaysofweek, $validmonths, $firstyear); } // The next valid minute is in the same hour so it must be valid according to all other // checks and we can finally return it. return $now + ($nextvalidminute - $currentminute) * 60; } /** * Informs whether this task can be run. * * @return bool true when this task can be run. false otherwise. */ public function can_run(): bool { return $this->is_component_enabled() || $this->get_run_if_component_disabled(); } /** * Checks whether the component and the task disabled flag enables to run this task. * This do not checks whether the task manager allows running them or if the * site allows tasks to "run now". * * @return bool true if task is enabled. false otherwise. */ public function is_enabled(): bool { return $this->can_run() && !$this->get_disabled(); } /** * Produces a valid id string to use as id attribute based on the given FQCN class name. * * @param string $classname FQCN of a task. * @return string valid string to be used as id attribute. */ public static function get_html_id(string $classname): string { return str_replace('\\', '-', ltrim($classname, '\\')); } }