mirror of
https://github.com/rlanvin/php-rrule.git
synced 2025-04-05 21:40:27 +02:00
Implementing BYSETPOS with freq > DAILY
Needs more tests
This commit is contained in:
parent
066a97d517
commit
66e56cd4f9
121
src/RRule.php
121
src/RRule.php
@ -482,6 +482,9 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
|
||||
$this->bysecond = array();
|
||||
foreach ( $parts['BYSECOND'] as $value ) {
|
||||
// yes, "60" is a valid value, in (very rare) cases on leap seconds
|
||||
// December 31, 2005 23:59:60 UTC is a valid date...
|
||||
// so is 2012-06-30T23:59:60UTC
|
||||
if ( $value < 0 || $value > 60 ) {
|
||||
throw new \InvalidArgumentException('Invalid BYSECOND value: '.$value);
|
||||
}
|
||||
@ -497,10 +500,11 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
}
|
||||
|
||||
if ( $this->freq < self::HOURLY ) {
|
||||
// for frequencies DAILY, WEEKLY, MONTHLY AND YEARLY, we build
|
||||
// for frequencies DAILY, WEEKLY, MONTHLY AND YEARLY, we can build
|
||||
// an array of every time of the day at which there should be an
|
||||
// occurrence - default, if no BYHOUR/BYMINUTE/BYSECOND are provided
|
||||
// is only one time, and it's the DTSTART time.
|
||||
// is only one time, and it's the DTSTART time. This is a cached version
|
||||
// if you will, since it'll never change at these frequencies
|
||||
$this->timeset = array();
|
||||
foreach ( $this->byhour as $hour ) {
|
||||
foreach ( $this->byminute as $minute ) {
|
||||
@ -510,7 +514,8 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
}
|
||||
}
|
||||
}
|
||||
sort($this->timeset);
|
||||
// FIXME sort ??
|
||||
// sort($this->timeset);
|
||||
}
|
||||
}
|
||||
|
||||
@ -704,6 +709,7 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
break;
|
||||
case self::SECONDLY:
|
||||
$diff = $date->diff($this->dtstart);
|
||||
// XXX does not account for leap second (should it?)
|
||||
$diff = $diff->s + $diff->i * 60 + $diff->h * 3600 + $diff->days * 86400;
|
||||
if ( $diff % $this->interval !== 0 ) {
|
||||
return false;
|
||||
@ -1013,7 +1019,13 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
|
||||
|
||||
/**
|
||||
* Not sure what it does yet
|
||||
* This builds an array of every time of the day that matches the BYXXX time
|
||||
* criteria. It will only process $this->frequency at one time. So:
|
||||
* - for HOURLY frequencies it builds the minutes and second of the given hour
|
||||
* - for MINUTELY frequencies it builds the seconds of the given minute
|
||||
* - for SECONDLY frequencies, it returns an array with one element
|
||||
*
|
||||
* This method is called everytime an increment of at least one hour is made.
|
||||
*/
|
||||
protected function getTimeSet($hour, $minute, $second)
|
||||
{
|
||||
@ -1070,7 +1082,8 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
*
|
||||
* Another quirk of this approach is that because the granularity is by day,
|
||||
* higher frequencies (hourly, minutely and secondly) have to have
|
||||
* their own special loops within the main loop.
|
||||
* their own special loops within the main loop, making the all thing quite
|
||||
* convoluted.
|
||||
* Moreover, at such frequencies, the brute-force approach starts to really
|
||||
* suck. For example, a rule like
|
||||
* "Every minute, every Jan 1st between 10:00 and 10:59, for 10 years"
|
||||
@ -1094,13 +1107,13 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
// at every call of the method (to emulate a generator)
|
||||
static $year = null, $month = null, $day = null;
|
||||
static $hour = null, $minute = null, $second = null;
|
||||
static $current_set = null, $masks = null, $timeset = null;
|
||||
static $dayset = null, $masks = null, $timeset = null;
|
||||
static $total = 0;
|
||||
|
||||
if ( $reset ) {
|
||||
$year = $month = $day = null;
|
||||
$hour = $minute = $second = null;
|
||||
$current_set = $masks = $timeset = null;
|
||||
$dayset = $masks = $timeset = null;
|
||||
$total = 0;
|
||||
}
|
||||
|
||||
@ -1128,8 +1141,7 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
$second = (int) $second;
|
||||
}
|
||||
|
||||
// todo, not sure when this should be rebuilt
|
||||
// and not sure what this does anyway
|
||||
// we initialize the timeset
|
||||
if ( $timeset == null ) {
|
||||
if ( $this->freq < self::HOURLY ) {
|
||||
// daily, weekly, monthly or yearly
|
||||
@ -1137,6 +1149,7 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
$timeset = $this->timeset;
|
||||
}
|
||||
else {
|
||||
// initialize empty if it's not going to occurs on the first iteration
|
||||
if (
|
||||
($this->freq >= self::HOURLY && $this->byhour && ! in_array($hour, $this->byhour))
|
||||
|| ($this->freq >= self::MINUTELY && $this->byminute && ! in_array($minute, $this->byminute))
|
||||
@ -1149,8 +1162,6 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
}
|
||||
}
|
||||
}
|
||||
// echo json_encode($timeset),"\n";
|
||||
// fgets(STDIN);
|
||||
|
||||
// while (true) {
|
||||
$max_cycles = self::$REPEAT_CYCLES[$this->freq <= self::DAILY ? $this->freq : self::DAILY];
|
||||
@ -1158,7 +1169,7 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
// 1. get an array of all days in the next interval (day, month, week, etc.)
|
||||
// we filter out from this array all days that do not match the BYXXX conditions
|
||||
// to speed things up, we use days of the year (day numbers) instead of date
|
||||
if ( $current_set === null ) {
|
||||
if ( $dayset === null ) {
|
||||
// rebuild the various masks and converters
|
||||
// these arrays will allow fast date operations
|
||||
// without relying on date() methods
|
||||
@ -1196,13 +1207,13 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
}
|
||||
|
||||
// calculate the current set
|
||||
$current_set = $this->getDaySet($year, $month, $day, $masks);
|
||||
// echo"\tWorking with $year-$month-$day set=".json_encode($current_set)."\n";
|
||||
$dayset = $this->getDaySet($year, $month, $day, $masks);
|
||||
// echo"\tWorking with $year-$month-$day set=".json_encode($dayset)."\n";
|
||||
// print_r(json_encode($masks));
|
||||
// fgets(STDIN);
|
||||
$filtered_set = array();
|
||||
|
||||
foreach ( $current_set as $yearday ) {
|
||||
foreach ( $dayset as $yearday ) {
|
||||
if ( $this->bymonth && ! in_array($masks['yearday_to_month'][$yearday], $this->bymonth) ) {
|
||||
continue;
|
||||
}
|
||||
@ -1242,51 +1253,91 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
}
|
||||
// echo "\tFiltered set (before BYSETPOS)=".json_encode($filtered_set)."\n";
|
||||
|
||||
$current_set = $filtered_set;
|
||||
$dayset = $filtered_set;
|
||||
|
||||
// XXX this needs to be applied after expanding the timeset
|
||||
if ( $this->bysetpos ) {
|
||||
// if BYSETPOS is set, we need to expand the timeset to filter by pos
|
||||
// so we make a special loop to return while generating
|
||||
if ( $this->bysetpos && $timeset ) {
|
||||
$filtered_set = array();
|
||||
$n = sizeof($current_set);
|
||||
foreach ( $this->bysetpos as $pos ) {
|
||||
// echo "pos = $pos => ";
|
||||
$n = count($timeset);
|
||||
if ( $pos < 0 ) {
|
||||
$pos = $n + $pos;
|
||||
$pos = $n * count($dayset) + $pos;
|
||||
}
|
||||
else {
|
||||
$pos = $pos - 1;
|
||||
}
|
||||
if ( isset($current_set[$pos]) ) {
|
||||
$filtered_set[] = $current_set[$pos];
|
||||
// echo "$pos => ";
|
||||
$div = (int) ($pos / $n); // daypos
|
||||
$mod = $pos % $n; // timepos
|
||||
// echo "array index $div / $mod\n";
|
||||
// echo "div = $div, mod = $mod\n";
|
||||
// echo "dayset[$div] = ",$dayset[$div]," timeset[$mod] = ",json_encode($timeset[$mod]),"\n";
|
||||
// fgets(STDIN);
|
||||
|
||||
if ( isset($dayset[$div]) && isset($timeset[$mod]) ) {
|
||||
$yearday = $dayset[$div];
|
||||
$time = $timeset[$mod];
|
||||
// used as array key to ensure uniqueness
|
||||
$tmp = $year.':'.$yearday.':'.$time[0].':'.$time[1].':'.$time[2];
|
||||
if ( ! isset($filtered_set[$tmp]) ) {
|
||||
$occurrence = \DateTime::createFromFormat(
|
||||
'Y z',
|
||||
"$year $yearday"
|
||||
);
|
||||
$occurrence->setTime($time[0], $time[1], $time[2]);
|
||||
$filtered_set[$tmp] = $occurrence;
|
||||
}
|
||||
}
|
||||
}
|
||||
$current_set = array_unique($filtered_set);
|
||||
sort($filtered_set);
|
||||
$dayset = $filtered_set;
|
||||
}
|
||||
|
||||
// echo "\tFiltered set (after BYSETPOS)=".json_encode($filtered_set)."\n";
|
||||
}
|
||||
|
||||
// 2. loop, generate a valid date, and return the result (fake "yield")
|
||||
// at the same time, we check the end condition and return null if
|
||||
// we need to stop
|
||||
while ( ($yearday = current($current_set)) !== false ) {
|
||||
$occurrence = \DateTime::createFromFormat('Y z', "$year $yearday");
|
||||
if ( $this->bysetpos && $timeset ) {
|
||||
while ( ($occurrence = current($dayset)) !== false ) {
|
||||
|
||||
while ( ($time = current($timeset)) !== false ) {
|
||||
$occurrence->setTime($time[0], $time[1], $time[2]);
|
||||
// consider end conditions
|
||||
if ( $this->until && $occurrence > $this->until ) {
|
||||
// $this->length = $total (?)
|
||||
return null;
|
||||
}
|
||||
|
||||
next($timeset);
|
||||
next($dayset);
|
||||
if ( $occurrence >= $this->dtstart ) { // ignore occurrences before DTSTART
|
||||
$total += 1;
|
||||
return $occurrence; // yield
|
||||
}
|
||||
}
|
||||
reset($timeset);
|
||||
next($current_set);
|
||||
}
|
||||
else {
|
||||
// normal loop, without BYSETPOS
|
||||
while ( ($yearday = current($dayset)) !== false ) {
|
||||
$occurrence = \DateTime::createFromFormat('Y z', "$year $yearday");
|
||||
|
||||
while ( ($time = current($timeset)) !== false ) {
|
||||
$occurrence->setTime($time[0], $time[1], $time[2]);
|
||||
// consider end conditions
|
||||
if ( $this->until && $occurrence > $this->until ) {
|
||||
// $this->length = $total (?)
|
||||
return null;
|
||||
}
|
||||
|
||||
next($timeset);
|
||||
if ( $occurrence >= $this->dtstart ) { // ignore occurrences before DTSTART
|
||||
$total += 1;
|
||||
return $occurrence; // yield
|
||||
}
|
||||
}
|
||||
reset($timeset);
|
||||
next($dayset);
|
||||
}
|
||||
}
|
||||
|
||||
// 3. we reset the loop to the next interval
|
||||
@ -1327,7 +1378,7 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
// call the DateTime method at the very end.
|
||||
|
||||
case self::HOURLY:
|
||||
if ( empty($current_set) ) {
|
||||
if ( empty($dayset) ) {
|
||||
// an empty set means that this day has been filtered out
|
||||
// by one of the BYXXX rule. So there is no need to
|
||||
// examine it any further, we know nothing is going to
|
||||
@ -1358,7 +1409,7 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
$timeset = $this->getTimeSet($hour, $minute, $second);
|
||||
break;
|
||||
case self::MINUTELY:
|
||||
if ( empty($current_set) ) {
|
||||
if ( empty($dayset) ) {
|
||||
$minute += ((int) ((1439 - ($hour*60+$minute)) / $this->interval)) * $this->interval;
|
||||
}
|
||||
|
||||
@ -1391,7 +1442,7 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
$timeset = $this->getTimeSet($hour, $minute, $second);
|
||||
break;
|
||||
case self::SECONDLY:
|
||||
if ( empty($current_set) ) {
|
||||
if ( empty($dayset) ) {
|
||||
$second += ((int) ((86399 - ($hour*3600 + $minute*60 + $second)) / $this->interval)) * $this->interval;
|
||||
}
|
||||
|
||||
@ -1435,7 +1486,7 @@ class RRule implements \Iterator, \ArrayAccess
|
||||
if ( $days_increment ) {
|
||||
list($year,$month,$day) = explode('-',date_create("$year-$month-$day")->modify("+ $days_increment days")->format('Y-n-j'));
|
||||
}
|
||||
$current_set = null; // reset the loop
|
||||
$dayset = null; // reset the loop
|
||||
}
|
||||
return null; // stop the iterator
|
||||
}
|
||||
|
@ -112,7 +112,7 @@ class RRuleTest extends PHPUnit_Framework_TestCase
|
||||
array(array('BYWEEKNO' => 53, 'BYDAY' => 'MO'), array(
|
||||
date_create('1998-12-28'), date_create('2004-12-27'), date_create('2009-12-28'))),
|
||||
|
||||
// TODO BYSETPOS
|
||||
// todo bysetpos
|
||||
|
||||
array(array('BYHOUR' => array(6, 18)),array(
|
||||
date_create('1997-09-02 06:00:00'),
|
||||
@ -142,10 +142,10 @@ class RRuleTest extends PHPUnit_Framework_TestCase
|
||||
date_create('1997-09-02 06:15:10'),
|
||||
date_create('1997-09-02 06:15:20'),
|
||||
date_create('1997-09-02 06:30:10'))),
|
||||
// array(array('BYMONTHDAY'=>15,'BYHOUR'=>array(6, 18),'BYSETPOS'=>array(3, -3)),array(
|
||||
// date_create('1997-11-15 18:00:00'),
|
||||
// date_create('1998-02-15 06:00:00'),
|
||||
// date_create('1998-11-15 18:00:00')))
|
||||
array(array('BYMONTHDAY'=>15,'BYHOUR'=>array(6, 18),'BYSETPOS'=>array(3, -3)),array(
|
||||
date_create('1997-11-15 18:00:00'),
|
||||
date_create('1998-02-15 06:00:00'),
|
||||
date_create('1998-11-15 18:00:00')))
|
||||
);
|
||||
}
|
||||
|
||||
@ -204,7 +204,11 @@ class RRuleTest extends PHPUnit_Framework_TestCase
|
||||
array(array('BYMONTH' => array(1, 3), 'BYMONTHDAY' => array(1, 3), 'BYDAY' => array('TU', 'TH')),array(
|
||||
date_create('1998-01-01'),date_create('1998-03-03'),date_create('2001-03-01'))),
|
||||
|
||||
// TODO BYSETPOS
|
||||
// last workday of the month
|
||||
array(array('BYDAY'=>'MO,TU,WE,TH,FR','BYSETPOS'=>-1), array(
|
||||
date_create('1997-09-30'),
|
||||
date_create('1997-10-31'),
|
||||
date_create('1997-11-28'))),
|
||||
|
||||
array(array('BYHOUR'=> array(6, 18)),array(
|
||||
date_create('1997-09-02 06:00:00'),date_create('1997-09-02 18:00:00'),date_create('1997-10-02 06:00:00'))),
|
||||
@ -220,8 +224,13 @@ class RRuleTest extends PHPUnit_Framework_TestCase
|
||||
date_create('1997-09-02 00:06:06'),date_create('1997-09-02 00:06:18'),date_create('1997-09-02 00:18:06'))),
|
||||
array(array('BYHOUR'=>array(6, 18),'BYMINUTE'=>array(6, 18),'BYSECOND'=>array(6, 18)),array(
|
||||
date_create('1997-09-02 06:06:06'),date_create('1997-09-02 06:06:18'),date_create('1997-09-02 06:18:06'))),
|
||||
// array(array('BYMONTHDAY'=>array(13, 17),'BYHOUR'=>array(6, 18),'BYSETPOS'=>array(3, -3)),array(
|
||||
// date_create('1997-09-13 06:00'),date_create('1997-09-17'),date_create('1997-10-13')))
|
||||
array(array('BYMONTHDAY'=>array(13, 17),'BYHOUR'=>array(6, 18),'BYSETPOS'=>array(3, -3)),array(
|
||||
date_create('1997-09-13 18:00'),date_create('1997-09-17 06:00'),date_create('1997-10-13 18:00'))),
|
||||
// avoid duplicates
|
||||
array(array('BYMONTHDAY'=>array(13, 17),'BYHOUR'=>array(6, 18),'BYSETPOS'=>array(3, 3, -3)),array(
|
||||
date_create('1997-09-13 18:00'),date_create('1997-09-17 06:00'),date_create('1997-10-13 18:00'))),
|
||||
array(array('BYMONTHDAY'=>array(13, 17),'BYHOUR'=>array(6, 18),'BYSETPOS'=>array(4, -1)),array(
|
||||
date_create('1997-09-17 18:00'),date_create('1997-10-17 18:00'),date_create('1997-11-17 18:00')))
|
||||
);
|
||||
}
|
||||
|
||||
@ -1300,7 +1309,7 @@ class RRuleTest extends PHPUnit_Framework_TestCase
|
||||
array(array(
|
||||
'freq' => 'monthly',
|
||||
'interval' => 2,
|
||||
'bymonth' => [1,3,5,7,9,11],
|
||||
'bymonth' => '1,3,5,7,9,11',
|
||||
'dtstart' => '1997-02-02 09:00:00',
|
||||
'count' => 1
|
||||
)),
|
||||
@ -1406,7 +1415,7 @@ class RRuleTest extends PHPUnit_Framework_TestCase
|
||||
|
||||
array(
|
||||
array('freq' => 'hourly', 'dtstart' => '1999-09-02 09:00:00', 'INTERVAL' => 2),
|
||||
array('1999-09-02 10:00:00', '1999-09-02 12:00:00')
|
||||
array('1999-09-02 10:00:00', '1999-09-02 09:01:01','1999-09-02 12:00:00')
|
||||
),
|
||||
array(
|
||||
array('freq' => 'hourly', 'dtstart' => '1999-09-02 09:00:00', 'INTERVAL' => 5),
|
||||
|
Loading…
x
Reference in New Issue
Block a user