. /** * Defines calendar class to manage recurrence rule (rrule) during ical imports. * * @package core_calendar * @copyright 2014 onwards Ankit Agarwal * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ namespace core_calendar; use calendar_event; use DateInterval; use DateTime; use moodle_exception; use stdClass; defined('MOODLE_INTERNAL') || die(); require_once($CFG->dirroot . '/calendar/lib.php'); /** * Defines calendar class to manage recurrence rule (rrule) during ical imports. * * Please refer to RFC 2445 {@link http://www.ietf.org/rfc/rfc2445.txt} for detail explanation of the logic. * Here is a basic extract from it to explain various params:- * recur = "FREQ"=freq *( * ; either UNTIL or COUNT may appear in a 'recur', * ; but UNTIL and COUNT MUST NOT occur in the same 'recur' * ( ";" "UNTIL" "=" enddate ) / * ( ";" "COUNT" "=" 1*DIGIT ) / * ; the rest of these keywords are optional, * ; but MUST NOT occur more than once * ( ";" "INTERVAL" "=" 1*DIGIT ) / * ( ";" "BYSECOND" "=" byseclist ) / * ( ";" "BYMINUTE" "=" byminlist ) / * ( ";" "BYHOUR" "=" byhrlist ) / * ( ";" "BYDAY" "=" bywdaylist ) / * ( ";" "BYMONTHDAY" "=" bymodaylist ) / * ( ";" "BYYEARDAY" "=" byyrdaylist ) / * ( ";" "BYWEEKNO" "=" bywknolist ) / * ( ";" "BYMONTH" "=" bymolist ) / * ( ";" "BYSETPOS" "=" bysplist ) / * ( ";" "WKST" "=" weekday ) / * ( ";" x-name "=" text ) * ) * * freq = "SECONDLY" / "MINUTELY" / "HOURLY" / "DAILY" * / "WEEKLY" / "MONTHLY" / "YEARLY" * enddate = date * enddate =/ date-time ;An UTC value * byseclist = seconds / ( seconds *("," seconds) ) * seconds = 1DIGIT / 2DIGIT ;0 to 59 * byminlist = minutes / ( minutes *("," minutes) ) * minutes = 1DIGIT / 2DIGIT ;0 to 59 * byhrlist = hour / ( hour *("," hour) ) * hour = 1DIGIT / 2DIGIT ;0 to 23 * bywdaylist = weekdaynum / ( weekdaynum *("," weekdaynum) ) * weekdaynum = [([plus] ordwk / minus ordwk)] weekday * plus = "+" * minus = "-" * ordwk = 1DIGIT / 2DIGIT ;1 to 53 * weekday = "SU" / "MO" / "TU" / "WE" / "TH" / "FR" / "SA" * ;Corresponding to SUNDAY, MONDAY, TUESDAY, WEDNESDAY, THURSDAY, * ;FRIDAY, SATURDAY and SUNDAY days of the week. * bymodaylist = monthdaynum / ( monthdaynum *("," monthdaynum) ) * monthdaynum = ([plus] ordmoday) / (minus ordmoday) * ordmoday = 1DIGIT / 2DIGIT ;1 to 31 * byyrdaylist = yeardaynum / ( yeardaynum *("," yeardaynum) ) * yeardaynum = ([plus] ordyrday) / (minus ordyrday) * ordyrday = 1DIGIT / 2DIGIT / 3DIGIT ;1 to 366 * bywknolist = weeknum / ( weeknum *("," weeknum) ) * weeknum = ([plus] ordwk) / (minus ordwk) * bymolist = monthnum / ( monthnum *("," monthnum) ) * monthnum = 1DIGIT / 2DIGIT ;1 to 12 * bysplist = setposday / ( setposday *("," setposday) ) * setposday = yeardaynum * * @package core_calendar * @copyright 2014 onwards Ankit Agarwal * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later */ class rrule_manager { /** const string Frequency constant */ const FREQ_YEARLY = 'yearly'; /** const string Frequency constant */ const FREQ_MONTHLY = 'monthly'; /** const string Frequency constant */ const FREQ_WEEKLY = 'weekly'; /** const string Frequency constant */ const FREQ_DAILY = 'daily'; /** const string Frequency constant */ const FREQ_HOURLY = 'hourly'; /** const string Frequency constant */ const FREQ_MINUTELY = 'everyminute'; /** const string Frequency constant */ const FREQ_SECONDLY = 'everysecond'; /** const string Day constant */ const DAY_MONDAY = 'Monday'; /** const string Day constant */ const DAY_TUESDAY = 'Tuesday'; /** const string Day constant */ const DAY_WEDNESDAY = 'Wednesday'; /** const string Day constant */ const DAY_THURSDAY = 'Thursday'; /** const string Day constant */ const DAY_FRIDAY = 'Friday'; /** const string Day constant */ const DAY_SATURDAY = 'Saturday'; /** const string Day constant */ const DAY_SUNDAY = 'Sunday'; /** const int For forever repeating events, repeat for this many years */ const TIME_UNLIMITED_YEARS = 10; /** const array Array of days in a week. */ const DAYS_OF_WEEK = [ 'MO' => self::DAY_MONDAY, 'TU' => self::DAY_TUESDAY, 'WE' => self::DAY_WEDNESDAY, 'TH' => self::DAY_THURSDAY, 'FR' => self::DAY_FRIDAY, 'SA' => self::DAY_SATURDAY, 'SU' => self::DAY_SUNDAY, ]; /** @var string string representing the recurrence rule */ protected $rrule; /** @var string Frequency of event */ protected $freq; /** @var int defines a timestamp value which bounds the recurrence rule in an inclusive manner.*/ protected $until = 0; /** @var int Defines the number of occurrences at which to range-bound the recurrence */ protected $count = 0; /** @var int This rule part contains a positive integer representing how often the recurrence rule repeats */ protected $interval = 1; /** @var array List of second rules */ protected $bysecond = array(); /** @var array List of Minute rules */ protected $byminute = array(); /** @var array List of hour rules */ protected $byhour = array(); /** @var array List of day rules */ protected $byday = array(); /** @var array List of monthday rules */ protected $bymonthday = array(); /** @var array List of yearday rules */ protected $byyearday = array(); /** @var array List of weekno rules */ protected $byweekno = array(); /** @var array List of month rules */ protected $bymonth = array(); /** @var array List of setpos rules */ protected $bysetpos = array(); /** @var string Week start rule. Default is Monday. */ protected $wkst = self::DAY_MONDAY; /** * Constructor for the class * * @param string $rrule Recurrence rule */ public function __construct($rrule) { $this->rrule = $rrule; } /** * Parse the recurrence rule and setup all properties. */ public function parse_rrule() { $rules = explode(';', $this->rrule); if (empty($rules)) { return; } foreach ($rules as $rule) { $this->parse_rrule_property($rule); } // Validate the rules as a whole. $this->validate_rules(); } /** * Create events for specified rrule. * * @param calendar_event $passedevent Properties of event to create. * @throws moodle_exception */ public function create_events($passedevent) { global $DB; $event = clone($passedevent); // If Frequency is not set, there is nothing to do. if (empty($this->freq)) { return; } // Delete all child events in case of an update. This should be faster than verifying if the event exists and updating it. $where = "repeatid = ? AND id != ?"; $DB->delete_records_select('event', $where, array($event->id, $event->id)); $eventrec = $event->properties(); // Generate timestamps that obey the rrule. $eventtimes = $this->generate_recurring_event_times($eventrec); // Update the parent event. Make sure that its repeat ID is the same as its ID. $calevent = new calendar_event($eventrec); $updatedata = new stdClass(); $updatedata->repeatid = $event->id; // Also, adjust the parent event's timestart, if necessary. if (count($eventtimes) > 0 && !in_array($eventrec->timestart, $eventtimes)) { $updatedata->timestart = reset($eventtimes); } $calevent->update($updatedata, false); $eventrec->timestart = $calevent->timestart; // Create the recurring calendar events. $this->create_recurring_events($eventrec, $eventtimes); } /** * Parse a property of the recurrence rule. * * @param string $prop property string with type-value pair * @throws moodle_exception */ protected function parse_rrule_property($prop) { list($property, $value) = explode('=', $prop); switch ($property) { case 'FREQ' : $this->set_frequency($value); break; case 'UNTIL' : $this->set_until($value); break; CASE 'COUNT' : $this->set_count($value); break; CASE 'INTERVAL' : $this->set_interval($value); break; CASE 'BYSECOND' : $this->set_bysecond($value); break; CASE 'BYMINUTE' : $this->set_byminute($value); break; CASE 'BYHOUR' : $this->set_byhour($value); break; CASE 'BYDAY' : $this->set_byday($value); break; CASE 'BYMONTHDAY' : $this->set_bymonthday($value); break; CASE 'BYYEARDAY' : $this->set_byyearday($value); break; CASE 'BYWEEKNO' : $this->set_byweekno($value); break; CASE 'BYMONTH' : $this->set_bymonth($value); break; CASE 'BYSETPOS' : $this->set_bysetpos($value); break; CASE 'WKST' : $this->wkst = $this->get_day($value); break; default: // We should never get here, something is very wrong. throw new moodle_exception('errorrrule', 'calendar'); } } /** * Sets Frequency property. * * @param string $freq Frequency of event * @throws moodle_exception */ protected function set_frequency($freq) { switch ($freq) { case 'YEARLY': $this->freq = self::FREQ_YEARLY; break; case 'MONTHLY': $this->freq = self::FREQ_MONTHLY; break; case 'WEEKLY': $this->freq = self::FREQ_WEEKLY; break; case 'DAILY': $this->freq = self::FREQ_DAILY; break; case 'HOURLY': $this->freq = self::FREQ_HOURLY; break; case 'MINUTELY': $this->freq = self::FREQ_MINUTELY; break; case 'SECONDLY': $this->freq = self::FREQ_SECONDLY; break; default: // We should never get here, something is very wrong. throw new moodle_exception('errorrrulefreq', 'calendar'); } } /** * Gets the day from day string. * * @param string $daystring Day string (MO, TU, etc) * @throws moodle_exception * * @return string Day represented by the parameter. */ protected function get_day($daystring) { switch ($daystring) { case 'MO': return self::DAY_MONDAY; break; case 'TU': return self::DAY_TUESDAY; break; case 'WE': return self::DAY_WEDNESDAY; break; case 'TH': return self::DAY_THURSDAY; break; case 'FR': return self::DAY_FRIDAY; break; case 'SA': return self::DAY_SATURDAY; break; case 'SU': return self::DAY_SUNDAY; break; default: // We should never get here, something is very wrong. throw new moodle_exception('errorrruleday', 'calendar'); } } /** * Sets the UNTIL rule. * * @param string $until The date string representation of the UNTIL rule. * @throws moodle_exception */ protected function set_until($until) { $this->until = strtotime($until); } /** * Sets the COUNT rule. * * @param string $count The count value. * @throws moodle_exception */ protected function set_count($count) { $this->count = intval($count); } /** * Sets the INTERVAL rule. * * The INTERVAL rule part contains a positive integer representing how often the recurrence rule repeats. * The default value is "1", meaning: * - every second for a SECONDLY rule, or * - every minute for a MINUTELY rule, * - every hour for an HOURLY rule, * - every day for a DAILY rule, * - every week for a WEEKLY rule, * - every month for a MONTHLY rule and * - every year for a YEARLY rule. * * @param string $intervalstr The value for the interval rule. * @throws moodle_exception */ protected function set_interval($intervalstr) { $interval = intval($intervalstr); if ($interval < 1) { throw new moodle_exception('errorinvalidinterval', 'calendar'); } $this->interval = $interval; } /** * Sets the BYSECOND rule. * * The BYSECOND rule part specifies a comma-separated list of seconds within a minute. * Valid values are 0 to 59. * * @param string $bysecond Comma-separated list of seconds within a minute. * @throws moodle_exception */ protected function set_bysecond($bysecond) { $seconds = explode(',', $bysecond); $bysecondrules = []; foreach ($seconds as $second) { if ($second < 0 || $second > 59) { throw new moodle_exception('errorinvalidbysecond', 'calendar'); } $bysecondrules[] = (int)$second; } $this->bysecond = $bysecondrules; } /** * Sets the BYMINUTE rule. * * The BYMINUTE rule part specifies a comma-separated list of seconds within an hour. * Valid values are 0 to 59. * * @param string $byminute Comma-separated list of minutes within an hour. * @throws moodle_exception */ protected function set_byminute($byminute) { $minutes = explode(',', $byminute); $byminuterules = []; foreach ($minutes as $minute) { if ($minute < 0 || $minute > 59) { throw new moodle_exception('errorinvalidbyminute', 'calendar'); } $byminuterules[] = (int)$minute; } $this->byminute = $byminuterules; } /** * Sets the BYHOUR rule. * * The BYHOUR rule part specifies a comma-separated list of hours of the day. * Valid values are 0 to 23. * * @param string $byhour Comma-separated list of hours of the day. * @throws moodle_exception */ protected function set_byhour($byhour) { $hours = explode(',', $byhour); $byhourrules = []; foreach ($hours as $hour) { if ($hour < 0 || $hour > 23) { throw new moodle_exception('errorinvalidbyhour', 'calendar'); } $byhourrules[] = (int)$hour; } $this->byhour = $byhourrules; } /** * Sets the BYDAY rule. * * The BYDAY rule part specifies a comma-separated list of days of the week; * - MO indicates Monday; * - TU indicates Tuesday; * - WE indicates Wednesday; * - TH indicates Thursday; * - FR indicates Friday; * - SA indicates Saturday; * - SU indicates Sunday. * * Each BYDAY value can also be preceded by a positive (+n) or negative (-n) integer. * If present, this indicates the nth occurrence of the specific day within the MONTHLY or YEARLY RRULE. * For example, within a MONTHLY rule, +1MO (or simply 1MO) represents the first Monday within the month, * whereas -1MO represents the last Monday of the month. * If an integer modifier is not present, it means all days of this type within the specified frequency. * For example, within a MONTHLY rule, MO represents all Mondays within the month. * * @param string $byday Comma-separated list of days of the week. * @throws moodle_exception */ protected function set_byday($byday) { $weekdays = array_keys(self::DAYS_OF_WEEK); $days = explode(',', $byday); $bydayrules = []; foreach ($days as $day) { $suffix = substr($day, -2); if (!in_array($suffix, $weekdays)) { throw new moodle_exception('errorinvalidbydaysuffix', 'calendar'); } $bydayrule = new stdClass(); $bydayrule->day = substr($suffix, -2); $bydayrule->value = (int)str_replace($suffix, '', $day); $bydayrules[] = $bydayrule; } $this->byday = $bydayrules; } /** * Sets the BYMONTHDAY rule. * * The BYMONTHDAY rule part specifies a comma-separated list of days of the month. * Valid values are 1 to 31 or -31 to -1. For example, -10 represents the tenth to the last day of the month. * * @param string $bymonthday Comma-separated list of days of the month. * @throws moodle_exception */ protected function set_bymonthday($bymonthday) { $monthdays = explode(',', $bymonthday); $bymonthdayrules = []; foreach ($monthdays as $day) { // Valid values are 1 to 31 or -31 to -1. if ($day < -31 || $day > 31 || $day == 0) { throw new moodle_exception('errorinvalidbymonthday', 'calendar'); } $bymonthdayrules[] = (int)$day; } // Sort these MONTHDAY rules in ascending order. sort($bymonthdayrules); $this->bymonthday = $bymonthdayrules; } /** * Sets the BYYEARDAY rule. * * The BYYEARDAY rule part specifies a comma-separated list of days of the year. * Valid values are 1 to 366 or -366 to -1. For example, -1 represents the last day of the year (December 31st) * and -306 represents the 306th to the last day of the year (March 1st). * * @param string $byyearday Comma-separated list of days of the year. * @throws moodle_exception */ protected function set_byyearday($byyearday) { $yeardays = explode(',', $byyearday); $byyeardayrules = []; foreach ($yeardays as $day) { // Valid values are 1 to 366 or -366 to -1. if ($day < -366 || $day > 366 || $day == 0) { throw new moodle_exception('errorinvalidbyyearday', 'calendar'); } $byyeardayrules[] = (int)$day; } $this->byyearday = $byyeardayrules; } /** * Sets the BYWEEKNO rule. * * The BYWEEKNO rule part specifies a comma-separated list of ordinals specifying weeks of the year. * Valid values are 1 to 53 or -53 to -1. This corresponds to weeks according to week numbering as defined in [ISO 8601]. * A week is defined as a seven day period, starting on the day of the week defined to be the week start (see WKST). * Week number one of the calendar year is the first week which contains at least four (4) days in that calendar year. * This rule part is only valid for YEARLY rules. For example, 3 represents the third week of the year. * * Note: Assuming a Monday week start, week 53 can only occur when Thursday is January 1 or if it is a leap year and Wednesday * is January 1. * * @param string $byweekno Comma-separated list of number of weeks. * @throws moodle_exception */ protected function set_byweekno($byweekno) { $weeknumbers = explode(',', $byweekno); $byweeknorules = []; foreach ($weeknumbers as $week) { // Valid values are 1 to 53 or -53 to -1. if ($week < -53 || $week > 53 || $week == 0) { throw new moodle_exception('errorinvalidbyweekno', 'calendar'); } $byweeknorules[] = (int)$week; } $this->byweekno = $byweeknorules; } /** * Sets the BYMONTH rule. * * The BYMONTH rule part specifies a comma-separated list of months of the year. * Valid values are 1 to 12. * * @param string $bymonth Comma-separated list of months of the year. * @throws moodle_exception */ protected function set_bymonth($bymonth) { $months = explode(',', $bymonth); $bymonthrules = []; foreach ($months as $month) { // Valid values are 1 to 12. if ($month < 1 || $month > 12) { throw new moodle_exception('errorinvalidbymonth', 'calendar'); } $bymonthrules[] = (int)$month; } $this->bymonth = $bymonthrules; } /** * Sets the BYSETPOS rule. * * The BYSETPOS rule part specifies a comma-separated list of values which corresponds to the nth occurrence within the set of * events specified by the rule. Valid values are 1 to 366 or -366 to -1. * It MUST only be used in conjunction with another BYxxx rule part. * * For example "the last work day of the month" could be represented as: RRULE:FREQ=MONTHLY;BYDAY=MO,TU,WE,TH,FR;BYSETPOS=-1 * * @param string $bysetpos Comma-separated list of values. * @throws moodle_exception */ protected function set_bysetpos($bysetpos) { $setposes = explode(',', $bysetpos); $bysetposrules = []; foreach ($setposes as $pos) { // Valid values are 1 to 366 or -366 to -1. if ($pos < -366 || $pos > 366 || $pos == 0) { throw new moodle_exception('errorinvalidbysetpos', 'calendar'); } $bysetposrules[] = (int)$pos; } $this->bysetpos = $bysetposrules; } /** * Validate the rules as a whole. * * @throws moodle_exception */ protected function validate_rules() { // UNTIL and COUNT cannot be in the same recurrence rule. if (!empty($this->until) && !empty($this->count)) { throw new moodle_exception('errorhasuntilandcount', 'calendar'); } // BYSETPOS only be used in conjunction with another BYxxx rule part. if (!empty($this->bysetpos) && empty($this->bymonth) && empty($this->bymonthday) && empty($this->bysecond) && empty($this->byday) && empty($this->byweekno) && empty($this->byhour) && empty($this->byminute) && empty($this->byyearday)) { throw new moodle_exception('errormustbeusedwithotherbyrule', 'calendar'); } // Integer values preceding BYDAY rules can only be present for MONTHLY or YEARLY RRULE. foreach ($this->byday as $bydayrule) { if (!empty($bydayrule->value) && $this->freq != self::FREQ_MONTHLY && $this->freq != self::FREQ_YEARLY) { throw new moodle_exception('errorinvalidbydayprefix', 'calendar'); } } // The BYWEEKNO rule is only valid for YEARLY rules. if (!empty($this->byweekno) && $this->freq != self::FREQ_YEARLY) { throw new moodle_exception('errornonyearlyfreqwithbyweekno', 'calendar'); } } /** * Creates calendar events for the recurring events. * * @param stdClass $event The parent event. * @param int[] $eventtimes The timestamps of the recurring events. */ protected function create_recurring_events($event, $eventtimes) { $count = false; if ($this->count) { $count = $this->count; } foreach ($eventtimes as $time) { // Skip if time is the same time with the parent event's timestamp. if ($time == $event->timestart) { continue; } // Decrement count, if set. if ($count !== false) { $count--; if ($count == 0) { break; } } // Create the recurring event. $cloneevent = clone($event); $cloneevent->repeatid = $event->id; $cloneevent->timestart = $time; unset($cloneevent->id); // UUID should only be set on the first instance of the recurring events. unset($cloneevent->uuid); calendar_event::create($cloneevent, false); } // If COUNT rule is defined and the number of the generated event times is less than the the COUNT rule, // repeat the processing until the COUNT rule is satisfied. if ($count !== false && $count > 0) { // Set count to the remaining counts. $this->count = $count; // Clone the original event, but set the timestart to the last generated event time. $tmpevent = clone($event); $tmpevent->timestart = end($eventtimes); // Generate the additional event times. $additionaleventtimes = $this->generate_recurring_event_times($tmpevent); // Create the additional events. $this->create_recurring_events($event, $additionaleventtimes); } } /** * Generates recurring events based on the parent event and the RRULE set. * * If multiple BYxxx rule parts are specified, then after evaluating the specified FREQ and INTERVAL rule parts, * the BYxxx rule parts are applied to the current set of evaluated occurrences in the following order: * BYMONTH, BYWEEKNO, BYYEARDAY, BYMONTHDAY, BYDAY, BYHOUR, BYMINUTE, BYSECOND and BYSETPOS; * then COUNT and UNTIL are evaluated. * * @param stdClass $event The event object. * @return array The list of timestamps that obey the given RRULE. */ protected function generate_recurring_event_times($event) { $interval = $this->get_interval(); // Candidate event times. $eventtimes = []; $eventdatetime = new DateTime(date('Y-m-d H:i:s', $event->timestart)); $until = null; if (empty($this->count)) { if ($this->until) { $until = $this->until; } else { // Forever event. However, since there's no such thing as 'forever' (at least not in Moodle), // we only repeat the events until 10 years from the current time. $untildate = new DateTime(); $foreverinterval = new DateInterval('P' . self::TIME_UNLIMITED_YEARS . 'Y'); $untildate->add($foreverinterval); $until = $untildate->getTimestamp(); } } else { // If count is defined, let's define a tentative until date. We'll just trim the number of events later. $untildate = clone($eventdatetime); $count = $this->count; while ($count >= 0) { $untildate->add($interval); $count--; } $until = $untildate->getTimestamp(); } // No filters applied. Generate recurring events right away. if (!$this->has_by_rules()) { // Get initial list of prospective events. $tmpstart = clone($eventdatetime); while ($tmpstart->getTimestamp() <= $until) { $eventtimes[] = $tmpstart->getTimestamp(); $tmpstart->add($interval); } return $eventtimes; } // Get all of potential dates covered by the periods from the event's start date until the last. $dailyinterval = new DateInterval('P1D'); $boundslist = $this->get_period_bounds_list($eventdatetime->getTimestamp(), $until); foreach ($boundslist as $bounds) { $tmpdate = new DateTime(date('Y-m-d H:i:s', $bounds->start)); while ($tmpdate->getTimestamp() >= $bounds->start && $tmpdate->getTimestamp() < $bounds->next) { $eventtimes[] = $tmpdate->getTimestamp(); $tmpdate->add($dailyinterval); } } // Evaluate BYMONTH rules. $eventtimes = $this->filter_by_month($eventtimes); // Evaluate BYWEEKNO rules. $eventtimes = $this->filter_by_weekno($eventtimes); // Evaluate BYYEARDAY rules. $eventtimes = $this->filter_by_yearday($eventtimes); // If BYYEARDAY, BYMONTHDAY and BYDAY are not set, default to BYMONTHDAY based on the DTSTART's day. if ($this->freq != self::FREQ_DAILY && empty($this->byyearday) && empty($this->bymonthday) && empty($this->byday)) { $this->bymonthday = [$eventdatetime->format('j')]; } // Evaluate BYMONTHDAY rules. $eventtimes = $this->filter_by_monthday($eventtimes); // Evaluate BYDAY rules. $eventtimes = $this->filter_by_day($event, $eventtimes, $until); // Evaluate BYHOUR rules. $eventtimes = $this->apply_hour_minute_second_rules($eventdatetime, $eventtimes); // Evaluate BYSETPOS rules. $eventtimes = $this->filter_by_setpos($event, $eventtimes, $until); // Sort event times in ascending order. sort($eventtimes); // Finally, filter candidate event times to make sure they are within the DTSTART and UNTIL/tentative until boundaries. $results = []; foreach ($eventtimes as $time) { // Skip out-of-range events. if ($time < $eventdatetime->getTimestamp()) { continue; } // End if event time is beyond the until limit. if ($time > $until) { break; } $results[] = $time; } return $results; } /** * Generates a DateInterval object based on the FREQ and INTERVAL rules. * * @return DateInterval * @throws moodle_exception */ protected function get_interval() { $intervalspec = null; switch ($this->freq) { case self::FREQ_YEARLY: $intervalspec = 'P' . $this->interval . 'Y'; break; case self::FREQ_MONTHLY: $intervalspec = 'P' . $this->interval . 'M'; break; case self::FREQ_WEEKLY: $intervalspec = 'P' . $this->interval . 'W'; break; case self::FREQ_DAILY: $intervalspec = 'P' . $this->interval . 'D'; break; case self::FREQ_HOURLY: $intervalspec = 'PT' . $this->interval . 'H'; break; case self::FREQ_MINUTELY: $intervalspec = 'PT' . $this->interval . 'M'; break; case self::FREQ_SECONDLY: $intervalspec = 'PT' . $this->interval . 'S'; break; default: // We should never get here, something is very wrong. throw new moodle_exception('errorrrulefreq', 'calendar'); } return new DateInterval($intervalspec); } /** * Determines whether the RRULE has BYxxx rules or not. * * @return bool True if there is one or more BYxxx rules to process. False, otherwise. */ protected function has_by_rules() { return !empty($this->bymonth) || !empty($this->bymonthday) || !empty($this->bysecond) || !empty($this->byday) || !empty($this->byweekno) || !empty($this->byhour) || !empty($this->byminute) || !empty($this->byyearday); } /** * Filter event times based on the BYMONTH rule. * * @param int[] $eventdates Timestamps of event times to be filtered. * @return int[] Array of filtered timestamps. */ protected function filter_by_month($eventdates) { if (empty($this->bymonth)) { return $eventdates; } $filteredbymonth = []; foreach ($eventdates as $time) { foreach ($this->bymonth as $month) { $prospectmonth = date('n', $time); if ($month == $prospectmonth) { $filteredbymonth[] = $time; break; } } } return $filteredbymonth; } /** * Filter event times based on the BYWEEKNO rule. * * @param int[] $eventdates Timestamps of event times to be filtered. * @return int[] Array of filtered timestamps. */ protected function filter_by_weekno($eventdates) { if (empty($this->byweekno)) { return $eventdates; } $filteredbyweekno = []; $weeklyinterval = null; foreach ($eventdates as $time) { $tmpdate = new DateTime(date('Y-m-d H:i:s', $time)); foreach ($this->byweekno as $weekno) { if ($weekno > 0) { if ($tmpdate->format('W') == $weekno) { $filteredbyweekno[] = $time; break; } } else if ($weekno < 0) { if ($weeklyinterval === null) { $weeklyinterval = new DateInterval('P1W'); } $weekstart = new DateTime(); $weekstart->setISODate($tmpdate->format('Y'), $weekno); $weeknext = clone($weekstart); $weeknext->add($weeklyinterval); $tmptimestamp = $tmpdate->getTimestamp(); if ($tmptimestamp >= $weekstart->getTimestamp() && $tmptimestamp < $weeknext->getTimestamp()) { $filteredbyweekno[] = $time; break; } } } } return $filteredbyweekno; } /** * Filter event times based on the BYYEARDAY rule. * * @param int[] $eventdates Timestamps of event times to be filtered. * @return int[] Array of filtered timestamps. */ protected function filter_by_yearday($eventdates) { if (empty($this->byyearday)) { return $eventdates; } $filteredbyyearday = []; foreach ($eventdates as $time) { $tmpdate = new DateTime(date('Y-m-d', $time)); foreach ($this->byyearday as $yearday) { $dayoffset = abs($yearday) - 1; $dayoffsetinterval = new DateInterval("P{$dayoffset}D"); if ($yearday > 0) { $tmpyearday = (int)$tmpdate->format('z') + 1; if ($tmpyearday == $yearday) { $filteredbyyearday[] = $time; break; } } else if ($yearday < 0) { $yeardaydate = new DateTime('last day of ' . $tmpdate->format('Y')); $yeardaydate->sub($dayoffsetinterval); $tmpdate->getTimestamp(); if ($yeardaydate->format('z') == $tmpdate->format('z')) { $filteredbyyearday[] = $time; break; } } } } return $filteredbyyearday; } /** * Filter event times based on the BYMONTHDAY rule. * * @param int[] $eventdates The event times to be filtered. * @return int[] Array of filtered timestamps. */ protected function filter_by_monthday($eventdates) { if (empty($this->bymonthday)) { return $eventdates; } $filteredbymonthday = []; foreach ($eventdates as $time) { $eventdatetime = new DateTime(date('Y-m-d', $time)); foreach ($this->bymonthday as $monthday) { // Days to add/subtract. $daysoffset = abs($monthday) - 1; $dayinterval = new DateInterval("P{$daysoffset}D"); if ($monthday > 0) { if ($eventdatetime->format('j') == $monthday) { $filteredbymonthday[] = $time; break; } } else if ($monthday < 0) { $tmpdate = clone($eventdatetime); // Reset to the first day of the month. $tmpdate->modify('first day of this month'); // Then go to last day of the month. $tmpdate->modify('last day of this month'); if ($daysoffset > 0) { // Then subtract the monthday value. $tmpdate->sub($dayinterval); } if ($eventdatetime->format('j') == $tmpdate->format('j')) { $filteredbymonthday[] = $time; break; } } } } return $filteredbymonthday; } /** * Filter event times based on the BYDAY rule. * * @param stdClass $event The parent event. * @param int[] $eventdates The event times to be filtered. * @param int $until Event times generation limit date. * @return int[] Array of filtered timestamps. */ protected function filter_by_day($event, $eventdates, $until) { if (empty($this->byday)) { return $eventdates; } $filteredbyday = []; $bounds = $this->get_period_bounds_list($event->timestart, $until); $nextmonthinterval = new DateInterval('P1M'); foreach ($eventdates as $time) { $tmpdatetime = new DateTime(date('Y-m-d', $time)); foreach ($this->byday as $day) { $dayname = self::DAYS_OF_WEEK[$day->day]; // Skip if they day name of the event time does not match the day part of the BYDAY rule. if ($tmpdatetime->format('l') !== $dayname) { continue; } if (empty($day->value)) { // No modifier value. Applies to all weekdays of the given period. $filteredbyday[] = $time; break; } else if ($day->value > 0) { // Positive value. if ($this->freq === self::FREQ_YEARLY && empty($this->bymonth)) { // Get the first day of the year. $firstdaydate = $tmpdatetime->format('Y') . '-01-01'; } else { // Get the first day of the month. $firstdaydate = $tmpdatetime->format('Y-m') . '-01'; } $expecteddate = new DateTime($firstdaydate); $count = $day->value; // Get the nth week day of the year/month. $expecteddate->modify("+$count $dayname"); if ($expecteddate->format('Y-m-d') === $tmpdatetime->format('Y-m-d')) { $filteredbyday[] = $time; break; } } else { // Negative value. $count = $day->value; if ($this->freq === self::FREQ_YEARLY && empty($this->bymonth)) { // The -Nth week day of the year. $eventyear = (int)$tmpdatetime->format('Y'); // Get temporary DateTime object starting from the first day of the next year. $expecteddate = new DateTime((++$eventyear) . '-01-01'); while ($count < 0) { // Get the start of the previous week. $expecteddate->modify('last ' . $this->wkst); $tmpexpecteddate = clone($expecteddate); if ($tmpexpecteddate->format('l') !== $dayname) { $tmpexpecteddate->modify('next ' . $dayname); } if ($this->in_bounds($tmpexpecteddate->getTimestamp(), $bounds)) { $expecteddate = $tmpexpecteddate; $count++; } } if ($expecteddate->format('l') !== $dayname) { $expecteddate->modify('next ' . $dayname); } if ($expecteddate->getTimestamp() == $time) { $filteredbyday[] = $time; break; } } else { // The -Nth week day of the month. $expectedmonthyear = $tmpdatetime->format('F Y'); $expecteddate = new DateTime("first day of $expectedmonthyear"); $expecteddate->add($nextmonthinterval); while ($count < 0) { // Get the start of the previous week. $expecteddate->modify('last ' . $this->wkst); $tmpexpecteddate = clone($expecteddate); if ($tmpexpecteddate->format('l') !== $dayname) { $tmpexpecteddate->modify('next ' . $dayname); } if ($this->in_bounds($tmpexpecteddate->getTimestamp(), $bounds)) { $expecteddate = $tmpexpecteddate; $count++; } } // Compare the expected date with the event's timestamp. if ($expecteddate->getTimestamp() == $time) { $filteredbyday[] = $time; break; } } } } } return $filteredbyday; } /** * Applies BYHOUR, BYMINUTE and BYSECOND rules to the calculated event dates. * Defaults to the DTSTART's hour/minute/second component when not defined. * * @param DateTime $eventdatetime The parent event DateTime object pertaining to the DTSTART. * @param int[] $eventdates Array of candidate event date timestamps. * @return array List of updated event timestamps that contain the time component of the event times. */ protected function apply_hour_minute_second_rules(DateTime $eventdatetime, $eventdates) { // If BYHOUR rules are not set, set the hour of the events from the DTSTART's hour component. if (empty($this->byhour)) { $this->byhour = [$eventdatetime->format('G')]; } // If BYMINUTE rules are not set, set the hour of the events from the DTSTART's minute component. if (empty($this->byminute)) { $this->byminute = [(int)$eventdatetime->format('i')]; } // If BYSECOND rules are not set, set the hour of the events from the DTSTART's second component. if (empty($this->bysecond)) { $this->bysecond = [(int)$eventdatetime->format('s')]; } $results = []; foreach ($eventdates as $time) { $datetime = new DateTime(date('Y-m-d', $time)); foreach ($this->byhour as $hour) { foreach ($this->byminute as $minute) { foreach ($this->bysecond as $second) { $datetime->setTime($hour, $minute, $second); $results[] = $datetime->getTimestamp(); } } } } return $results; } /** * Filter event times based on the BYSETPOS rule. * * @param stdClass $event The parent event. * @param int[] $eventtimes The event times to be filtered. * @param int $until Event times generation limit date. * @return int[] Array of filtered timestamps. */ protected function filter_by_setpos($event, $eventtimes, $until) { if (empty($this->bysetpos)) { return $eventtimes; } $filteredbysetpos = []; $boundslist = $this->get_period_bounds_list($event->timestart, $until); sort($eventtimes); foreach ($boundslist as $bounds) { // Generate a list of candidate event times based that are covered in a period's bounds. $prospecttimes = []; foreach ($eventtimes as $time) { if ($time >= $bounds->start && $time < $bounds->next) { $prospecttimes[] = $time; } } if (empty($prospecttimes)) { continue; } // Add the event times that correspond to the set position rule into the filtered results. foreach ($this->bysetpos as $pos) { $tmptimes = $prospecttimes; if ($pos < 0) { rsort($tmptimes); } $index = abs($pos) - 1; if (isset($tmptimes[$index])) { $filteredbysetpos[] = $tmptimes[$index]; } } } return $filteredbysetpos; } /** * Gets the list of period boundaries covered by the recurring events. * * @param int $eventtime The event timestamp. * @param int $until The end timestamp. * @return array List of period bounds, with start and next properties. */ protected function get_period_bounds_list($eventtime, $until) { $interval = $this->get_interval(); $periodbounds = $this->get_period_boundaries($eventtime); $periodstart = $periodbounds['start']; $periodafter = $periodbounds['next']; $bounds = []; if ($until !== null) { while ($periodstart->getTimestamp() < $until) { $bounds[] = (object)[ 'start' => $periodstart->getTimestamp(), 'next' => $periodafter->getTimestamp() ]; $periodstart->add($interval); $periodafter->add($interval); } } else { $count = $this->count; while ($count > 0) { $bounds[] = (object)[ 'start' => $periodstart->getTimestamp(), 'next' => $periodafter->getTimestamp() ]; $periodstart->add($interval); $periodafter->add($interval); $count--; } } return $bounds; } /** * Determine whether the date-time in question is within the bounds of the periods that are covered by the RRULE. * * @param int $time The timestamp to be evaluated. * @param array $bounds Array of period boundaries covered by the RRULE. * @return bool */ protected function in_bounds($time, $bounds) { foreach ($bounds as $bound) { if ($time >= $bound->start && $time < $bound->next) { return true; } } return false; } /** * Determines the start and end DateTime objects that serve as references to determine whether a calculated event timestamp * falls on the period defined by these DateTimes objects. * * @param int $eventtime Unix timestamp of the event time. * @return DateTime[] * @throws moodle_exception */ protected function get_period_boundaries($eventtime) { $nextintervalspec = null; switch ($this->freq) { case self::FREQ_YEARLY: $nextintervalspec = 'P1Y'; $timestart = date('Y-01-01', $eventtime); break; case self::FREQ_MONTHLY: $nextintervalspec = 'P1M'; $timestart = date('Y-m-01', $eventtime); break; case self::FREQ_WEEKLY: $nextintervalspec = 'P1W'; if (date('l', $eventtime) === $this->wkst) { $weekstarttime = $eventtime; } else { $weekstarttime = strtotime('last ' . $this->wkst, $eventtime); } $timestart = date('Y-m-d', $weekstarttime); break; case self::FREQ_DAILY: $nextintervalspec = 'P1D'; $timestart = date('Y-m-d', $eventtime); break; case self::FREQ_HOURLY: $nextintervalspec = 'PT1H'; $timestart = date('Y-m-d H:00:00', $eventtime); break; case self::FREQ_MINUTELY: $nextintervalspec = 'PT1M'; $timestart = date('Y-m-d H:i:00', $eventtime); break; case self::FREQ_SECONDLY: $nextintervalspec = 'PT1S'; $timestart = date('Y-m-d H:i:s', $eventtime); break; default: // We should never get here, something is very wrong. throw new moodle_exception('errorrrulefreq', 'calendar'); } $eventstart = new DateTime($timestart); $eventnext = clone($eventstart); $nextinterval = new DateInterval($nextintervalspec); $eventnext->add($nextinterval); return [ 'start' => $eventstart, 'next' => $eventnext, ]; } }