timezoneConverter = $timezoneConverter ?: ObjectManager::getInstance()->get(TimezoneInterface::class); } /** * @inheritdoc */ public function _construct() { $this->_init(\Magento\Cron\Model\ResourceModel\Schedule::class); } /** * Set cron expression. * * @param string $expr * @return $this * @throws \Magento\Framework\Exception\CronException */ public function setCronExpr($expr) { $e = preg_split('#\s+#', $expr, null, PREG_SPLIT_NO_EMPTY); if (sizeof($e) < 5 || sizeof($e) > 6) { throw new CronException(__('Invalid cron expression: %1', $expr)); } $this->setCronExprArr($e); return $this; } /** * Checks the observer's cron expression against time. * * Supports $this->setCronExpr('* 0-5,10-59/5 2-10,15-25 january-june/2 mon-fri') * * @return bool */ public function trySchedule() { $time = $this->getScheduledAt(); $e = $this->getCronExprArr(); if (!$e || !$time) { return false; } if (!is_numeric($time)) { //convert time from UTC to admin store timezone //we assume that all schedules in configuration (crontab.xml and DB tables) are in admin store timezone $time = $this->timezoneConverter->date($time)->format('Y-m-d H:i'); $time = strtotime($time); } $match = $this->matchCronExpression($e[0], strftime('%M', $time)) && $this->matchCronExpression($e[1], strftime('%H', $time)) && $this->matchCronExpression($e[2], strftime('%d', $time)) && $this->matchCronExpression($e[3], strftime('%m', $time)) && $this->matchCronExpression($e[4], strftime('%w', $time)); return $match; } /** * Match cron expression. * * @param string $expr * @param int $num * @return bool * @throws \Magento\Framework\Exception\CronException * @SuppressWarnings(PHPMD.CyclomaticComplexity) * @SuppressWarnings(PHPMD.NPathComplexity) */ public function matchCronExpression($expr, $num) { // handle ALL match if ($expr === '*') { return true; } // handle multiple options if (strpos($expr, ',') !== false) { foreach (explode(',', $expr) as $e) { if ($this->matchCronExpression($e, $num)) { return true; } } return false; } // handle modulus if (strpos($expr, '/') !== false) { $e = explode('/', $expr); if (sizeof($e) !== 2) { throw new CronException(__('Invalid cron expression, expecting \'match/modulus\': %1', $expr)); } if (!is_numeric($e[1])) { throw new CronException(__('Invalid cron expression, expecting numeric modulus: %1', $expr)); } $expr = $e[0]; $mod = $e[1]; } else { $mod = 1; } // handle all match by modulus if ($expr === '*') { $from = 0; $to = 60; } elseif (strpos($expr, '-') !== false) { // handle range $e = explode('-', $expr); if (sizeof($e) !== 2) { throw new CronException(__('Invalid cron expression, expecting \'from-to\' structure: %1', $expr)); } $from = $this->getNumeric($e[0]); $to = $this->getNumeric($e[1]); } else { // handle regular token $from = $this->getNumeric($expr); $to = $from; } if ($from === false || $to === false) { throw new CronException(__('Invalid cron expression: %1', $expr)); } return $num >= $from && $num <= $to && $num % $mod === 0; } /** * Get number of a month. * * @param int|string $value * @return bool|int|string */ public function getNumeric($value) { static $data = [ 'jan' => 1, 'feb' => 2, 'mar' => 3, 'apr' => 4, 'may' => 5, 'jun' => 6, 'jul' => 7, 'aug' => 8, 'sep' => 9, 'oct' => 10, 'nov' => 11, 'dec' => 12, 'sun' => 0, 'mon' => 1, 'tue' => 2, 'wed' => 3, 'thu' => 4, 'fri' => 5, 'sat' => 6, ]; if (is_numeric($value)) { return $value; } if (is_string($value)) { $value = strtolower(substr($value, 0, 3)); if (isset($data[$value])) { return $data[$value]; } } return false; } /** * Lock the cron job so no other scheduled instances run simultaneously. * * Sets a job to STATUS_RUNNING only if it is currently in STATUS_PENDING * and no other jobs of the same code are currently in STATUS_RUNNING. * Returns true if status was changed and false otherwise. * * @return boolean */ public function tryLockJob() { if ($this->_getResource()->trySetJobUniqueStatusAtomic( $this->getId(), self::STATUS_RUNNING, self::STATUS_PENDING )) { $this->setStatus(self::STATUS_RUNNING); return true; } return false; } }