dispatcher = $dispatcher; $this->scheduleRepository = $scheduleRepository; $this->taskRepository = $taskRepository; } /** * Set the time that this schedule should be run at. This will override the time * defined on the schedule itself. Useful for triggering one-off task runs. * * @param \DateTimeInterface $time * @return $this */ public function setRunTimeOverride(DateTimeInterface $time) { $this->runTimeOverride = $time; return $this; } /** * Process a schedule and push the first task onto the queue worker. * * @param \Pterodactyl\Models\Schedule $schedule * * @throws \Pterodactyl\Exceptions\Model\DataValidationException * @throws \Pterodactyl\Exceptions\Repository\RecordNotFoundException */ public function handle(Schedule $schedule) { $this->scheduleRepository->loadTasks($schedule); /** @var \Pterodactyl\Models\Task $task */ $task = $schedule->getRelation('tasks')->where('sequence_id', 1)->first(); $formattedCron = sprintf('%s %s %s * %s', $schedule->cron_minute, $schedule->cron_hour, $schedule->cron_day_of_month, $schedule->cron_day_of_week ); $this->scheduleRepository->update($schedule->id, [ 'is_processing' => true, 'next_run_at' => $this->getRunAtTime($formattedCron), ]); $this->taskRepository->update($task->id, ['is_queued' => true]); $this->dispatcher->dispatch( (new RunTaskJob($task->id, $schedule->id))->delay($task->time_offset) ); } /** * Get the timestamp to store in the database as the next_run time for a schedule. * * @param string $formatted * @return \Cake\Chronos\ChronosInterface */ private function getRunAtTime(string $formatted): ChronosInterface { if (! is_null($this->runTimeOverride)) { return $this->runTimeOverride instanceof ChronosInterface ? $this->runTimeOverride : Chronos::instance($this->runTimeOverride); } return Chronos::instance(CronExpression::factory($formatted)->getNextRunDate()); } }