moodle/calendar/classes/rrule_manager.php
Jun Pataleta c3b1178d58 MDL-58810 calendar: Fix rrule_manager issues
* Set the correct repeat ID for the recurring event
* Unset UUID for 'child' events of recurring events
2017-05-08 09:49:13 +08:00

1358 lines
49 KiB
PHP

<?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/>.
/**
* 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 <ankit.agrr@gmail.com>
* @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,
];
}
}