111 lines
3.5 KiB
PHP
111 lines
3.5 KiB
PHP
|
<?php
|
||
|
|
||
|
/**
|
||
|
* Day of month field. Allows: * , / - ? L W
|
||
|
*
|
||
|
* 'L' stands for "last" and specifies the last day of the month.
|
||
|
*
|
||
|
* The 'W' character is used to specify the weekday (Monday-Friday) nearest the
|
||
|
* given day. As an example, if you were to specify "15W" as the value for the
|
||
|
* day-of-month field, the meaning is: "the nearest weekday to the 15th of the
|
||
|
* month". So if the 15th is a Saturday, the trigger will fire on Friday the
|
||
|
* 14th. If the 15th is a Sunday, the trigger will fire on Monday the 16th. If
|
||
|
* the 15th is a Tuesday, then it will fire on Tuesday the 15th. However if you
|
||
|
* specify "1W" as the value for day-of-month, and the 1st is a Saturday, the
|
||
|
* trigger will fire on Monday the 3rd, as it will not 'jump' over the boundary
|
||
|
* of a month's days. The 'W' character can only be specified when the
|
||
|
* day-of-month is a single day, not a range or list of days.
|
||
|
*
|
||
|
* @author Michael Dowling <mtdowling@gmail.com>
|
||
|
*/
|
||
|
class CronExpression_DayOfMonthField extends CronExpression_AbstractField
|
||
|
{
|
||
|
/**
|
||
|
* Get the nearest day of the week for a given day in a month
|
||
|
*
|
||
|
* @param int $currentYear Current year
|
||
|
* @param int $currentMonth Current month
|
||
|
* @param int $targetDay Target day of the month
|
||
|
*
|
||
|
* @return DateTime Returns the nearest date
|
||
|
*/
|
||
|
private static function getNearestWeekday($currentYear, $currentMonth, $targetDay)
|
||
|
{
|
||
|
$tday = str_pad($targetDay, 2, '0', STR_PAD_LEFT);
|
||
|
$target = new DateTime("$currentYear-$currentMonth-$tday");
|
||
|
$currentWeekday = (int) $target->format('N');
|
||
|
|
||
|
if ($currentWeekday < 6) {
|
||
|
return $target;
|
||
|
}
|
||
|
|
||
|
$lastDayOfMonth = $target->format('t');
|
||
|
|
||
|
foreach (array(-1, 1, -2, 2) as $i) {
|
||
|
$adjusted = $targetDay + $i;
|
||
|
if ($adjusted > 0 && $adjusted <= $lastDayOfMonth) {
|
||
|
$target->setDate($currentYear, $currentMonth, $adjusted);
|
||
|
if ($target->format('N') < 6 && $target->format('m') == $currentMonth) {
|
||
|
return $target;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* {@inheritdoc}
|
||
|
*/
|
||
|
public function isSatisfiedBy(DateTime $date, $value)
|
||
|
{
|
||
|
// ? states that the field value is to be skipped
|
||
|
if ($value == '?') {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
$fieldValue = $date->format('d');
|
||
|
|
||
|
// Check to see if this is the last day of the month
|
||
|
if ($value == 'L') {
|
||
|
return $fieldValue == $date->format('t');
|
||
|
}
|
||
|
|
||
|
// Check to see if this is the nearest weekday to a particular value
|
||
|
if (strpos($value, 'W')) {
|
||
|
// Parse the target day
|
||
|
$targetDay = substr($value, 0, strpos($value, 'W'));
|
||
|
// Find out if the current day is the nearest day of the week
|
||
|
return $date->format('j') == self::getNearestWeekday(
|
||
|
$date->format('Y'),
|
||
|
$date->format('m'),
|
||
|
$targetDay
|
||
|
)->format('j');
|
||
|
}
|
||
|
|
||
|
return $this->isSatisfied($date->format('d'), $value);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* {@inheritdoc}
|
||
|
*/
|
||
|
public function increment(DateTime $date, $invert = false)
|
||
|
{
|
||
|
if ($invert) {
|
||
|
$date->modify('previous day');
|
||
|
$date->setTime(23, 59);
|
||
|
} else {
|
||
|
$date->modify('next day');
|
||
|
$date->setTime(0, 0);
|
||
|
}
|
||
|
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* {@inheritdoc}
|
||
|
*/
|
||
|
public function validate($value)
|
||
|
{
|
||
|
return (bool) preg_match('/[\*,\/\-\?LW0-9A-Za-z]+/', $value);
|
||
|
}
|
||
|
}
|