7#include "greedyscheduler.h"
9#include <ekos_scheduler_debug.h>
14#include "ui_scheduler.h"
15#include "schedulerjob.h"
16#include "schedulerutils.h"
18#define TEST_PRINT if (false) fprintf
21constexpr int SCHEDULE_RESOLUTION_MINUTES = 2;
26GreedyScheduler::GreedyScheduler()
30void GreedyScheduler::setParams(
bool restartImmediately,
bool restartQueue,
31 bool rescheduleErrors,
int abortDelay,
32 int errorHandlingDelay)
34 setRescheduleAbortsImmediate(restartImmediately);
35 setRescheduleAbortsQueue(restartQueue);
36 setRescheduleErrors(rescheduleErrors);
37 setAbortDelaySeconds(abortDelay);
38 setErrorDelaySeconds(errorHandlingDelay);
47void GreedyScheduler::scheduleJobs(
const QList<SchedulerJob *> &jobs,
49 const QMap<QString, uint16_t> &capturedFramesCount,
58 scheduledJob =
nullptr;
61 prepareJobsForEvaluation(jobs, now, capturedFramesCount, logger);
64 const QList<SchedulerJob *> leadJobs = SchedulerUtils::filterLeadJobs(jobs);
66 scheduledJob = selectNextJob(leadJobs, now,
nullptr, SIMULATE, &when,
nullptr,
nullptr, &capturedFramesCount);
67 auto schedule = getSchedule();
68 if (logger !=
nullptr)
70 if (!schedule.empty())
75 for (
int i = schedule.size() - 1; i >= 0; i--)
76 logger->appendLogText(GreedyScheduler::jobScheduleString(schedule[i]));
77 logger->appendLogText(QString(
"Greedy Scheduler plan for the next 48 hours starting %1 (%2)s:")
80 else logger->appendLogText(QString(
"Greedy Scheduler: empty plan (%1s)").arg(timer.
elapsed() / 1000.0));
82 if (scheduledJob !=
nullptr)
84 qCDebug(KSTARS_EKOS_SCHEDULER)
85 << QString(
"Greedy Scheduler scheduling next job %1 at %2")
86 .arg(scheduledJob->getName(), when.
toString(
"hh:mm"));
88 scheduledJob->setStartupTime(when);
98bool GreedyScheduler::checkJob(
const QList<SchedulerJob *> &jobs,
100 const SchedulerJob *
const currentJob)
103 if (currentJob && currentJob->getStateTime().secsTo(now) < 5)
110 SimulationType simType = SIMULATE_EACH_JOB_ONCE;
111 if (m_SimSeconds > 0.5 ||
112 (m_LastCheckJobSim.isValid() && m_LastCheckJobSim.secsTo(now) < 60))
113 simType = DONT_SIMULATE;
115 const SchedulerJob *
next = selectNextJob(jobs, now, currentJob, simType, &startTime);
116 if (next == currentJob && now.
secsTo(startTime) <= 1)
118 if (simType != DONT_SIMULATE)
119 m_LastCheckJobSim = now;
126 qCDebug(KSTARS_EKOS_SCHEDULER)
127 << QString(
"Greedy Scheduler bumping current job %1 for %2 at %3")
128 .arg(currentJob->getName(), next ?
next->getName() :
"---", now.
toString(
"hh:mm"));
147void GreedyScheduler::prepareJobsForEvaluation(
148 const QList<SchedulerJob *> &jobs,
const QDateTime &now,
149 const QMap<QString, uint16_t> &capturedFramesCount, ModuleLogger *logger,
bool reestimateJobTimes)
const
152 foreach (SchedulerJob *job, jobs)
154 job->clearSimulatedSchedule();
155 switch (job->getCompletionCondition())
159 if (job->getFinishAtTime().isValid() && job->getFinishAtTime() < now)
169 if (job->getRepeatsRemaining() == 0)
171 if (logger !=
nullptr)
logger->appendLogText(
i18n(
"Job '%1' has no more batches remaining.", job->getName()));
173 job->setEstimatedTime(0);
184 foreach (SchedulerJob *job, jobs)
186 switch (job->getState())
208 foreach (SchedulerJob *job, jobs)
216 if (reestimateJobTimes)
218 job->setEstimatedTime(-1);
219 if (SchedulerUtils::estimateJobTime(job, capturedFramesCount, logger) ==
false)
225 if (job->getEstimatedTime() == 0)
227 job->setRepeatsRemaining(0);
232 TEST_PRINT(stderr,
"JOB %s estimated time: %ld state %d\n", job->getName().toLatin1().data(), job->getEstimatedTime(),
242bool allowJob(
const SchedulerJob *job,
bool rescheduleAbortsImmediate,
bool rescheduleAbortsQueue,
bool rescheduleErrors)
246 if (job->getState() ==
SCHEDJOB_ABORTED && !rescheduleAbortsImmediate && !rescheduleAbortsQueue)
256QDateTime firstPossibleStart(
const SchedulerJob *job,
const QDateTime &now,
257 bool rescheduleAbortsQueue,
int abortDelaySeconds,
258 bool rescheduleErrors,
int errorDelaySeconds)
260 QDateTime possibleStart = now;
261 const QDateTime &abortTime = job->getLastAbortTime();
262 const QDateTime &errorTime = job->getLastErrorTime();
264 if (abortTime.
isValid() && rescheduleAbortsQueue)
266 auto abortStartTime = abortTime.
addSecs(abortDelaySeconds);
267 if (abortStartTime > now)
268 possibleStart = abortStartTime;
272 if (errorTime.
isValid() && rescheduleErrors)
274 auto errorStartTime = errorTime.
addSecs(errorDelaySeconds);
275 if (errorStartTime > now)
276 possibleStart = errorStartTime;
279 if (!possibleStart.
isValid() || possibleStart < now)
281 return possibleStart;
303SchedulerJob *GreedyScheduler::selectNextJob(
const QList<SchedulerJob *> &jobs,
const QDateTime &now,
304 const SchedulerJob *
const currentJob, SimulationType simType, QDateTime *when,
305 QDateTime *nextInterruption, QString *interruptReason,
306 const QMap<QString, uint16_t> *capturedFramesCount)
310 constexpr int MIN_RUN_SECS = 10 * 60;
313 constexpr int MAX_INTERRUPT_SECS = 30;
316 bool currentJobIsStartAt = (currentJob && currentJob->getFileStartupCondition() == START_AT &&
317 currentJob->getStartAtTime().isValid());
319 SchedulerJob * nextJob =
nullptr;
320 QString interruptStr;
322 for (
int i = 0; i < jobs.
size(); ++i)
324 SchedulerJob *
const job = jobs[i];
325 const bool evaluatingCurrentJob = (currentJob && (job == currentJob));
327 TEST_PRINT(stderr,
" considering %s (%s)\n", job->getName().toLatin1().data(), evaluatingCurrentJob ?
"evaluating" :
"");
329 if (!allowJob(job, rescheduleAbortsImmediate, rescheduleAbortsQueue, rescheduleErrors))
331 TEST_PRINT(stderr,
" not allowed\n");
336 QDateTime startSearchingtAt = firstPossibleStart(
337 job, now, rescheduleAbortsQueue, abortDelaySeconds, rescheduleErrors, errorDelaySeconds);
343 const QDateTime startTime = job->getNextPossibleStartTime(startSearchingtAt, SCHEDULE_RESOLUTION_MINUTES,
344 evaluatingCurrentJob);
349 if (nextJob ==
nullptr)
352 nextStart = startTime;
354 if (nextInterruption) *nextInterruption = QDateTime();
357 else if (Options::greedyScheduling())
361 const int runSecs = evaluatingCurrentJob ? MAX_INTERRUPT_SECS : MIN_RUN_SECS;
364 if (evaluatingCurrentJob && currentJobIsStartAt)
366 if (nextInterruption) *nextInterruption = QDateTime();
367 nextStart = startTime;
371 else if (startTime.
secsTo(nextStart) > runSecs)
375 if (nextInterruption) *nextInterruption = nextStart;
376 interruptStr = QString(
"interrupted by %1").
arg(nextJob->getName());
377 nextStart = startTime;
383 if (!currentJob && nextStart.
isValid() && now.
secsTo(nextStart) < MIN_RUN_SECS)
386 else if (evaluatingCurrentJob)
394 if (evaluatingCurrentJob)
break;
396 if (nextJob !=
nullptr)
402 for (
int i = 0; i < jobs.
size(); ++i)
404 SchedulerJob *
const atJob = jobs[i];
405 if (atJob == nextJob)
407 const QDateTime atTime = atJob->getStartAtTime();
408 if (atJob->getFileStartupCondition() == START_AT && atTime.
isValid())
410 if (!allowJob(atJob, rescheduleAbortsImmediate, rescheduleAbortsQueue, rescheduleErrors))
413 QDateTime startSearchingtAt = firstPossibleStart(
414 atJob, now, rescheduleAbortsQueue, abortDelaySeconds, rescheduleErrors,
418 const QDateTime atJobStartTime = atJob->getNextPossibleStartTime(startSearchingtAt, SCHEDULE_RESOLUTION_MINUTES, currentJob
419 && (atJob == currentJob));
423 const double startDelta = atJobStartTime.
secsTo(atTime);
424 if (fabs(startDelta) < 20 * 60)
430 const int gap = currentJob ==
nullptr ? MIN_RUN_SECS : 30;
431 if (nextStart.
secsTo(atJobStartTime) <= gap)
434 nextStart = atJobStartTime;
435 if (nextInterruption) *nextInterruption = QDateTime();
437 else if (nextInterruption)
441 if (!nextInterruption->
isValid() ||
442 atJobStartTime.
secsTo(*nextInterruption) < 0)
444 *nextInterruption = atJobStartTime;
445 interruptStr = QString(
"interrupted by %1").
arg(atJob->getName());
457 if (nextJob && !nextJob->getGroup().isEmpty() && Options::greedyScheduling() && nextJob->getCompletedIterations() > 0)
459 TEST_PRINT(stderr,
" Considering GROUPS (%d jobs) selected %s\n", jobs.
size(), nextJob->getName().toLatin1().data());
461 bool foundSelectedJob =
false;
462 for (
int i = 0; i < jobs.
size(); ++i)
464 SchedulerJob *
const job = jobs[i];
467 foundSelectedJob =
true;
471 TEST_PRINT(stderr,
" Job %s (group %s) %s (%d vs %d iterations) %s\n",
472 job->getName().toLatin1().data(), (job->getGroup() != nextJob->getGroup()) ?
"Different" :
"Same",
473 foundSelectedJob ?
"Found" :
"not found yet",
474 job->getCompletedIterations(), nextJob->getCompletedIterations(),
475 allowJob(job, rescheduleAbortsImmediate, rescheduleAbortsQueue, rescheduleErrors) ?
"allowed" :
"not allowed");
480 if (!foundSelectedJob ||
481 (job->getGroup() != nextJob->getGroup()) ||
482 (job->getCompletedIterations() >= nextJob->getCompletedIterations()) ||
483 !allowJob(job, rescheduleAbortsImmediate, rescheduleAbortsQueue, rescheduleErrors))
486 const bool evaluatingCurrentJob = (currentJob && (job == currentJob));
489 QDateTime startSearchingtAt = firstPossibleStart(
490 job, now, rescheduleAbortsQueue, abortDelaySeconds, rescheduleErrors, errorDelaySeconds);
493 const QDateTime startTime = job->getNextPossibleStartTime(startSearchingtAt, SCHEDULE_RESOLUTION_MINUTES,
494 evaluatingCurrentJob);
497 if (!startTime.
isValid() || startTime.
secsTo(nextStart) > MAX_INTERRUPT_SECS)
501 if (evaluatingCurrentJob && currentJobIsStartAt)
503 if (nextInterruption) *nextInterruption = QDateTime();
504 nextStart = startTime;
508 else if (startTime.
secsTo(nextStart) >= -MAX_INTERRUPT_SECS)
511 nextStart = startTime;
517 if (when !=
nullptr) *when = nextStart;
518 if (interruptReason !=
nullptr) *interruptReason = interruptStr;
525 unsetEvaluation(jobs);
527 QElapsedTimer simTimer;
529 const int simDays = SIM_HOURS * 3600;
530 if (simType != DONT_SIMULATE && nextJob !=
nullptr)
532 QDateTime simulationLimit = now.
addSecs(simDays);
534 QDateTime simEnd = simulate(jobs, now, simulationLimit, capturedFramesCount, simType);
538 if (!Options::rememberJobProgress() && Options::schedulerRepeatEverything())
540 int repeats = 0, maxRepeats = 5;
541 while (simEnd.
isValid() && simEnd.
secsTo(simulationLimit) > 0 && ++repeats < maxRepeats)
544 simEnd = simulate(jobs, simEnd, simulationLimit,
nullptr, simType);
547 m_SimSeconds = simTimer.
elapsed() / 1000.0;
548 TEST_PRINT(stderr,
"********************************* simulate(%s,%d) took %.3fs\n",
549 simType == SIMULATE ?
"SIM" :
"ONLY_1", SIM_HOURS, m_SimSeconds);
556QDateTime GreedyScheduler::simulate(
const QList<SchedulerJob *> &jobs,
const QDateTime &time,
const QDateTime &endTime,
557 const QMap<QString, uint16_t> *capturedFramesCount, SimulationType simType)
559 TEST_PRINT(stderr,
"%d simulate()\n", __LINE__);
561 QList<SchedulerJob *> copiedJobs;
562 QList<SchedulerJob *> scheduledJobs;
563 QDateTime simEndTime;
565 foreach (SchedulerJob *job, jobs)
567 SchedulerJob *newJob =
new SchedulerJob();
571 newJob->followerJobs().clear();
572 newJob->clearSimulatedSchedule();
573 copiedJobs.
append(newJob);
574 job->setStopTime(QDateTime());
579 int numStartupCandidates = 0, numStartups = 0;
581 foreach (SchedulerJob *job, copiedJobs)
583 job->setStartupTime(QDateTime());
584 const auto state = job->getState();
587 numStartupCandidates++;
590 QMap<QString, uint16_t> capturedFramesCopy;
591 if (capturedFramesCount !=
nullptr)
592 capturedFramesCopy = *capturedFramesCount;
593 QList<SchedulerJob *>simJobs = copiedJobs;
594 prepareJobsForEvaluation(copiedJobs, time, capturedFramesCopy,
nullptr,
false);
596 QDateTime simTime = time;
598 bool exceededIterations =
false;
599 QHash<SchedulerJob*, int> workDone;
600 QHash<SchedulerJob*, int> originalIteration, originalSecsLeftIteration;
602 for(
int i = 0; i < simJobs.
size(); ++i)
603 workDone[simJobs[i]] = 0.0;
607 QDateTime jobStartTime;
608 QDateTime jobInterruptTime;
609 QString interruptReason;
613 SchedulerJob *selectedJob =
614 selectNextJob(simJobs, simTime,
nullptr, DONT_SIMULATE, &jobStartTime, &jobInterruptTime, &interruptReason);
615 if (selectedJob ==
nullptr)
618 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
"%1 starting at %2 interrupted at \"%3\" reason \"%4\"")
619 .arg(selectedJob->getName()).arg(jobStartTime.
toString(
"MM/dd hh:mm"))
620 .arg(jobInterruptTime.
toString(
"MM/dd hh:mm")).arg(interruptReason).toLatin1().data());
622 if (endTime.
isValid() && jobStartTime.
secsTo(endTime) < 0)
break;
627 QDateTime nextStartAtTime;
628 foreach (SchedulerJob *job, simJobs)
630 if (job != selectedJob &&
631 job->getStartupCondition() == START_AT &&
632 jobStartTime.
secsTo(job->getStartupTime()) > 0 &&
636 QDateTime startAtTime = job->getStartupTime();
637 if (!nextStartAtTime.
isValid() || nextStartAtTime.
secsTo(startAtTime) < 0)
638 nextStartAtTime = startAtTime;
642 QDateTime constraintStopTime = jobInterruptTime;
643 if (nextStartAtTime.
isValid() &&
644 (!constraintStopTime.
isValid() ||
645 nextStartAtTime.
secsTo(constraintStopTime) < 0))
647 constraintStopTime = nextStartAtTime;
648 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
" job will be interrupted by a START_AT job").toLatin1().data());
651 QString constraintReason;
653 QDateTime jobConstraintTime = selectedJob->getNextEndTime(jobStartTime, SCHEDULE_RESOLUTION_MINUTES, &constraintReason,
656 std::abs(jobConstraintTime.
secsTo(nextStartAtTime)) < 2 * SCHEDULE_RESOLUTION_MINUTES)
657 constraintReason =
"interrupted by start-at job";
658 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
" constraint \"%1\" reason \"%2\"")
659 .arg(jobConstraintTime.
toString(
"MM/dd hh:mm")).arg(constraintReason).toLatin1().data());
660 QDateTime jobCompletionTime;
661 TEST_PRINT(stderr,
"%d %s\n", __LINE__,
662 QString(
" estimated time = %1").arg(selectedJob->getEstimatedTime()).toLatin1().data());
663 if (selectedJob->getEstimatedTime() > 0)
666 const int timeLeft = selectedJob->getEstimatedTime() - workDone[selectedJob];
667 jobCompletionTime = jobStartTime.
addSecs(timeLeft);
668 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
" completion \"%1\" time left %2s")
669 .arg(jobCompletionTime.
toString(
"MM/dd hh:mm")).arg(timeLeft).toLatin1().data());
673 QDateTime jobStopTime = jobInterruptTime;
674 QString stopReason = jobStopTime.
isValid() ? interruptReason :
"";
675 if (jobConstraintTime.
isValid() && (!jobStopTime.
isValid() || jobStopTime.
secsTo(jobConstraintTime) < 0))
677 stopReason = constraintReason;
678 jobStopTime = jobConstraintTime;
679 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
" picked constraint").toLatin1().data());
681 if (jobCompletionTime.
isValid() && (!jobStopTime.
isValid() || jobStopTime.
secsTo(jobCompletionTime) < 0))
683 stopReason =
"job completion";
684 jobStopTime = jobCompletionTime;
685 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
" picked completion").toLatin1().data());
690 if (!selectedJob->getGroup().isEmpty() &&
691 (selectedJob->getCompletionCondition() == FINISH_LOOP ||
692 selectedJob->getCompletionCondition() == FINISH_REPEAT ||
693 selectedJob->getCompletionCondition() == FINISH_AT))
695 if (originalIteration.
find(selectedJob) == originalIteration.
end())
696 originalIteration[selectedJob] = selectedJob->getCompletedIterations();
697 if (originalSecsLeftIteration.
find(selectedJob) == originalSecsLeftIteration.
end())
698 originalSecsLeftIteration[selectedJob] = selectedJob->getEstimatedTimeLeftThisRepeat();
701 int leftThisRepeat = selectedJob->getEstimatedTimeLeftThisRepeat();
702 int secsPerRepeat = selectedJob->getEstimatedTimePerRepeat();
703 int secsLeftThisRepeat = (workDone[selectedJob] < leftThisRepeat) ?
704 leftThisRepeat - workDone[selectedJob] : secsPerRepeat;
706 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
" sec per repeat %1 sec left this repeat %2")
707 .arg(secsPerRepeat).arg(secsLeftThisRepeat).toLatin1().data());
709 if (workDone[selectedJob] == 0)
711 secsLeftThisRepeat += selectedJob->getEstimatedStartupTime();
712 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
" adding %1 to secsLeftThisRepeat")
713 .arg(selectedJob->getEstimatedStartupTime()).arg(secsLeftThisRepeat).toLatin1().data());
717 if (secsLeftThisRepeat > 0 &&
718 (!jobStopTime.
isValid() || secsLeftThisRepeat < jobStartTime.
secsTo(jobStopTime)))
720 auto tempStart = jobStartTime;
721 auto tempInterrupt = jobInterruptTime;
722 auto tempReason = stopReason;
723 SchedulerJob keepJob = *selectedJob;
725 auto t = jobStartTime.
addSecs(secsLeftThisRepeat);
726 int iteration = selectedJob->getCompletedIterations();
727 int iters = 0, maxIters = 20;
728 while ((!jobStopTime.
isValid() || t.secsTo(jobStopTime) > 0) && iters++ < maxIters)
730 selectedJob->setCompletedIterations(++iteration);
731 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
" iteration=%1").arg(iteration).toLatin1().data());
732 SchedulerJob *
next = selectNextJob(simJobs, t,
nullptr, DONT_SIMULATE, &tempStart, &tempInterrupt, &tempReason);
733 if (next != selectedJob)
735 stopReason =
"interrupted for group member";
737 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
" switched to group member %1 at %2")
738 .arg(next ==
nullptr ?
"null" :
next->getName()).arg(t.toString(
"MM/dd hh:mm")).toLatin1().data());
742 t = t.addSecs(secsPerRepeat);
744 *selectedJob = keepJob;
751 const int secondsRun = jobStartTime.
secsTo(jobStopTime);
752 workDone[selectedJob] += secondsRun;
754 if ((originalIteration.
find(selectedJob) != originalIteration.
end()) &&
755 (originalSecsLeftIteration.
find(selectedJob) != originalSecsLeftIteration.
end()))
757 int completedIterations = originalIteration[selectedJob];
758 if (workDone[selectedJob] >= originalSecsLeftIteration[selectedJob] &&
759 selectedJob->getEstimatedTimePerRepeat() > 0)
760 completedIterations +=
761 1 + (workDone[selectedJob] - originalSecsLeftIteration[selectedJob]) / selectedJob->getEstimatedTimePerRepeat();
762 TEST_PRINT(stderr,
"%d %s\n", __LINE__,
763 QString(
" work sets interations=%1").arg(completedIterations).toLatin1().data());
764 selectedJob->setCompletedIterations(completedIterations);
770 if (!selectedJob->getStartupTime().isValid())
773 selectedJob->setStartupTime(jobStartTime);
774 selectedJob->setStopTime(jobStopTime);
775 selectedJob->setStopReason(stopReason);
777 scheduledJobs.
append(selectedJob);
778 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
" Scheduled: %1 %2 -> %3 %4 work done %5s")
779 .arg(selectedJob->getName()).arg(selectedJob->getStartupTime().toString(
"MM/dd hh:mm"))
780 .arg(selectedJob->getStopTime().toString(
"MM/dd hh:mm")).arg(selectedJob->getStopReason())
781 .arg(workDone[selectedJob]).toLatin1().data());
785 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
" Added: %1 %2 -> %3 %4 work done %5s")
786 .arg(selectedJob->getName()).arg(jobStartTime.
toString(
"MM/dd hh:mm"))
787 .arg(jobStopTime.
toString(
"MM/dd hh:mm")).arg(stopReason)
788 .arg(workDone[selectedJob]).toLatin1().data());
792 if (selectedJob->getEstimatedTime() >= 0 &&
793 workDone[selectedJob] >= selectedJob->getEstimatedTime())
796 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
" job %1 is complete")
797 .arg(selectedJob->getName()).toLatin1().data());
799 selectedJob->appendSimulatedSchedule(JobSchedule(
nullptr, jobStartTime, jobStopTime, stopReason));
800 schedule.
append(JobSchedule(jobs[copiedJobs.
indexOf(selectedJob)], jobStartTime, jobStopTime, stopReason));
801 simEndTime = jobStopTime;
802 simTime = jobStopTime.
addSecs(60);
809 if (++iterations > std::max(20, numStartupCandidates))
811 exceededIterations =
true;
812 TEST_PRINT(stderr,
"%d %s\n", __LINE__, QString(
"ending simulation after %1 iterations")
813 .arg(iterations).toLatin1().data());
817 if (simType == SIMULATE_EACH_JOB_ONCE)
819 bool allJobsProcessedOnce =
true;
820 for (
const auto job : simJobs)
822 if (allowJob(job, rescheduleAbortsImmediate, rescheduleAbortsQueue, rescheduleErrors) &&
823 !job->getStartupTime().isValid())
825 allJobsProcessedOnce =
false;
829 if (allJobsProcessedOnce)
831 TEST_PRINT(stderr,
"%d ending simulation, all jobs processed once\n", __LINE__);
840 for (
int i = 0; i < jobs.
size(); ++i)
842 if (scheduledJobs.
indexOf(copiedJobs[i]) >= 0)
848 jobs[i]->setStartupTime(copiedJobs[i]->getStartupTime());
851 jobs[i]->setStopTime(copiedJobs[i]->getStopTime());
852 jobs[i]->setStopReason(copiedJobs[i]->getStopReason());
853 if (simType == SIMULATE)
854 jobs[i]->setSimulatedSchedule(copiedJobs[i]->getSimulatedSchedule());
859 unsetEvaluation(jobs);
861 return exceededIterations ? QDateTime() : simEndTime;
864void GreedyScheduler::unsetEvaluation(
const QList<SchedulerJob *> &jobs)
const
866 for (
int i = 0; i < jobs.
size(); ++i)
873QString GreedyScheduler::jobScheduleString(
const JobSchedule &jobSchedule)
875 return QString(
"%1\t%2 --> %3 \t%4")
876 .arg(jobSchedule.job->getName(), -10)
877 .arg(jobSchedule.startTime.toString(
"MM/dd hh:mm"),
878 jobSchedule.stopTime.toString(
"hh:mm"), jobSchedule.stopReason);
881void GreedyScheduler::printSchedule(
const QList<JobSchedule> &schedule)
883 foreach (
auto &line, schedule)
885 fprintf(stderr,
"%s\n", QString(
"%1 %2 --> %3 (%4)")
886 .arg(jobScheduleString(line)).toLatin1().data());
QString i18n(const char *text, const TYPE &arg...)
Ekos is an advanced Astrophotography tool for Linux.
@ SCHEDJOB_ABORTED
Job encountered a transitory issue while processing, and will be rescheduled.
@ SCHEDJOB_INVALID
Job has an incorrect configuration, and cannot proceed.
@ SCHEDJOB_ERROR
Job encountered a fatal issue while processing, and must be reset manually.
@ SCHEDJOB_COMPLETE
Job finished all required captures.
@ SCHEDJOB_EVALUATION
Job is being evaluated.
@ SCHEDJOB_SCHEDULED
Job was evaluated, and has a schedule.
@ SCHEDJOB_BUSY
Job is being processed.
@ SCHEDJOB_IDLE
Job was just created, and is not evaluated yet.
const QList< QKeySequence > & next()
QCA_EXPORT Logger * logger()
QDateTime addSecs(qint64 s) const const
bool isValid() const const
qint64 secsTo(const QDateTime &other) const const
QString toString(QStringView format, QCalendar cal) const const
qint64 elapsed() const const
iterator find(const Key &key)
void append(QList< T > &&value)
qsizetype indexOf(const AT &value, qsizetype from) const const
qsizetype size() const const
QString & append(QChar ch)
QString arg(Args &&... args) const const
QByteArray toLatin1() const const