All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.nemerosa.ontrack.job.support.DefaultJobScheduler Maven / Gradle / Ivy

package net.nemerosa.ontrack.job.support;

import net.nemerosa.ontrack.common.Time;
import net.nemerosa.ontrack.job.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.time.LocalDateTime;
import java.time.temporal.ChronoUnit;
import java.util.*;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.AtomicReference;
import java.util.stream.Collectors;

public class DefaultJobScheduler implements JobScheduler {

    private final Logger logger = LoggerFactory.getLogger(JobScheduler.class);

    private final JobDecorator jobDecorator;
    private final ScheduledExecutorService scheduledExecutorService;
    private final JobListener jobListener;

    private final Map services = new ConcurrentHashMap<>(new TreeMap<>());
    private final AtomicBoolean schedulerPaused = new AtomicBoolean(false);

    private final AtomicLong idGenerator = new AtomicLong();

    public DefaultJobScheduler(JobDecorator jobDecorator, ScheduledExecutorService scheduledExecutorService, JobListener jobListener) {
        this.jobDecorator = jobDecorator;
        this.scheduledExecutorService = scheduledExecutorService;
        this.jobListener = jobListener;
    }

    @Override
    public void schedule(Job job, Schedule schedule) {
        logger.info("[job]{} Scheduling with {}", job.getKey(), schedule);
        // Manages existing schedule
        JobScheduledService existingService = services.remove(job.getKey());
        if (existingService != null) {
            logger.info("[job]{} Stopping existing schedule", job.getKey());
            existingService.cancel(false);
        }
        // Creates and starts the scheduled service
        logger.info("[job]{} Starting service", job.getKey());
        // Copy stats from old schedule
        JobScheduledService jobScheduledService = new JobScheduledService(
                job,
                schedule,
                scheduledExecutorService,
                existingService,
                jobListener.isPausedAtStartup(job.getKey())
        );
        // Registration
        services.put(job.getKey(), jobScheduledService);
    }

    @Override
    public boolean unschedule(JobKey key) {
        return unschedule(key, true);
    }

    protected boolean unschedule(JobKey key, boolean forceStop) {
        JobScheduledService existingService = services.remove(key);
        if (existingService != null) {
            existingService.cancel(forceStop);
            return true;
        } else {
            return false;
        }
    }

    @Override
    public void pause() {
        schedulerPaused.set(true);
    }

    @Override
    public void resume() {
        schedulerPaused.set(false);
    }

    @Override
    public boolean pause(JobKey key) {
        JobScheduledService existingService = services.get(key);
        if (existingService != null) {
            existingService.pause();
            return true;
        } else {
            throw new JobNotScheduledException(key);
        }
    }

    @Override
    public boolean resume(JobKey key) {
        JobScheduledService existingService = services.get(key);
        if (existingService != null) {
            existingService.resume();
            return true;
        } else {
            throw new JobNotScheduledException(key);
        }
    }

    @Override
    public Optional getJobStatus(JobKey key) {
        JobScheduledService existingService = services.get(key);
        if (existingService != null) {
            return Optional.of(existingService.getJobStatus());
        } else {
            return Optional.empty();
        }
    }

    @Override
    public Optional getJobKey(long id) {
        return services.values().stream()
                .filter(service -> service.getId() == id)
                .map(JobScheduledService::getJobKey)
                .findFirst();
    }

    @Override
    public Collection getAllJobKeys() {
        return services.keySet();
    }

    @Override
    public Collection getJobKeysOfType(JobType type) {
        return getAllJobKeys().stream()
                .filter(key -> key.sameType(type))
                .collect(Collectors.toSet());
    }

    @Override
    public Collection getJobKeysOfCategory(JobCategory category) {
        return getAllJobKeys().stream()
                .filter(key -> key.sameCategory(category))
                .collect(Collectors.toSet());
    }

    @Override
    public Collection getJobStatuses() {
        return services.values().stream()
                .map(JobScheduledService::getJobStatus)
                .collect(Collectors.toList());
    }

    @Override
    public Future fireImmediately(JobKey jobKey) {
        return fireImmediately(jobKey, Collections.emptyMap());
    }

    @Override
    public Future fireImmediately(JobKey jobKey, Map parameters) {
        // Gets the existing scheduled service
        JobScheduledService jobScheduledService = services.get(jobKey);
        if (jobScheduledService == null) {
            throw new JobNotScheduledException(jobKey);
        }
        // Fires the job immediately
        return jobScheduledService.fireImmediately(true, parameters);
    }

    private class JobScheduledService implements Runnable {

        private final long id;
        private final Job job;
        private final Schedule schedule;
        private final ScheduledFuture scheduledFuture;

        private final AtomicBoolean paused;

        private final AtomicReference> runParameters = new AtomicReference<>();
        private final AtomicReference> completableFuture = new AtomicReference<>();

        private final AtomicReference runProgress = new AtomicReference<>();
        private final AtomicLong runCount = new AtomicLong();
        private final AtomicReference lastRunDate = new AtomicReference<>();
        private final AtomicLong lastRunDurationMs = new AtomicLong();
        private final AtomicLong lastErrorCount = new AtomicLong();
        private final AtomicReference lastError = new AtomicReference<>(null);

        private JobScheduledService(Job job, Schedule schedule, ScheduledExecutorService scheduledExecutorService, JobScheduledService old, boolean pausedAtStartup) {
            this.id = idGenerator.incrementAndGet();
            this.job = job;
            this.schedule = schedule;
            // Paused at startup
            this.paused = new AtomicBoolean(pausedAtStartup);
            if (pausedAtStartup) {
                logger.debug("[job]{} Job paused at startup", job.getKey());
            }
            // Copies stats from old service
            if (old != null) {
                runCount.set(old.runCount.get());
                lastRunDate.set(old.lastRunDate.get());
                lastRunDurationMs.set(old.lastRunDurationMs.get());
                lastErrorCount.set(old.lastErrorCount.get());
                lastError.set(old.lastError.get());
            }
            // Scheduling now
            if (schedule.getPeriod() > 0) {
                scheduledFuture = scheduledExecutorService.scheduleWithFixedDelay(
                        this,
                        schedule.getInitialPeriod(),
                        schedule.getPeriod(),
                        schedule.getUnit()
                );
            } else {
                logger.debug("[job]{} Job not scheduled since period = 0", job.getKey());
                scheduledFuture = null;
            }
        }

        public long getId() {
            return id;
        }

        public JobKey getJobKey() {
            return job.getKey();
        }

        @Override
        public void run() {
            fireImmediately(false, Collections.emptyMap());
        }

        public void cancel(boolean forceStop) {
            if (scheduledFuture != null) {
                scheduledFuture.cancel(false);
            }
            // The decorated task might still run
            CompletableFuture future = this.completableFuture.get();
            if (forceStop && future != null && !future.isDone() && !future.isCancelled()) {
                future.cancel(true);
            }
        }

        public Future fireImmediately(boolean force, Map parameters) {
            return completableFuture.updateAndGet(cf -> optionallyFireTask(cf, force, parameters));
        }

        protected CompletableFuture optionallyFireTask(CompletableFuture runningCompletableFuture, boolean force, Map parameters) {
            if (runningCompletableFuture != null) {
                /*
                 * If the task is already running, we do not run it in concurrency,
                 * even if force is set to true
                 */
                return runningCompletableFuture;
            } else {
                return fireTask(force, parameters);
            }
        }

        protected CompletableFuture fireTask(boolean force, Map parameters) {
            runParameters.set(parameters);
            return CompletableFuture
                    .runAsync(jobDecorator.decorate(job, new MonitoredTask(force)), scheduledExecutorService)
                    .whenComplete((ignored, ex) -> completableFuture.set(null));
        }

        public JobStatus getJobStatus() {
            boolean valid = job.isValid();
            return new JobStatus(
                    id,
                    job.getKey(),
                    schedule,
                    job.getDescription(),
                    completableFuture.get() != null,
                    valid,
                    paused.get(),
                    job.isDisabled(),
                    runParameters.get(),
                    runProgress.get(),
                    runCount.get(),
                    lastRunDate.get(),
                    lastRunDurationMs.get(),
                    getNextRunDate(valid),
                    lastErrorCount.get(),
                    lastError.get()
            );
        }

        private LocalDateTime getNextRunDate(boolean valid) {
            if (valid && scheduledFuture != null) {
                return Time.now().plus(
                        scheduledFuture.getDelay(TimeUnit.SECONDS),
                        ChronoUnit.SECONDS
                );
            } else {
                return null;
            }
        }

        public void pause() {
            if (scheduledFuture != null) {
                paused.set(true);
                jobListener.onJobPaused(job.getKey());
            }
        }

        public void resume() {
            if (scheduledFuture != null) {
                paused.set(false);
                jobListener.onJobResumed(job.getKey());
            }
        }

        private class DefaultJobRunListener implements JobRunListener {

            @Override
            public void progress(JobRunProgress progress) {
                jobListener.onJobProgress(job.getKey(), progress);
                logger.debug("[job]{} {}",
                        job.getKey(),
                        progress.getText()
                );
                runProgress.set(progress);
            }

            @Override
            public  Optional getParam(String key) {
                Map parameters = runParameters.get();
                if (parameters != null) {
                    @SuppressWarnings("unchecked")
                    T t = (T) parameters.get(key);
                    return Optional.ofNullable(t);
                } else {
                    return Optional.empty();
                }
            }
        }

        private class MonitoredTask implements Runnable {

            private final boolean force;

            private MonitoredTask(boolean force) {
                this.force = force;
            }

            @Override
            public void run() {
                logger.debug("[job]{} Trying to run now - forced = {}", job.getKey(), force);
                if (job.isValid()) {
                    if (canRunNow(force)) {
                        try {
                            logger.debug("[job]{} Running now", job.getKey());
                            lastRunDate.set(Time.now());
                            runCount.incrementAndGet();
                            // Starting
                            jobListener.onJobStart(job.getKey());
                            // Runs the job
                            long _start = System.currentTimeMillis();
                            job.getTask().run(new DefaultJobRunListener());
                            // No error, counting time
                            long _end = System.currentTimeMillis();
                            lastRunDurationMs.set(_end - _start);
                            logger.debug("[job]{} Ran in {} ms", job.getKey(), lastRunDurationMs.get());
                            // Starting
                            jobListener.onJobEnd(job.getKey(), lastRunDurationMs.get());
                            // No error - resetting the counters
                            lastErrorCount.set(0);
                            lastError.set(null);
                        } catch (Exception ex) {
                            lastErrorCount.incrementAndGet();
                            lastError.set(ex.getMessage());
                            logger.error("[job]{} Error: {}", job.getKey(), ex.getMessage());
                            // Reporter
                            jobListener.onJobError(getJobStatus(), ex);
                            // Rethrows the error
                            throw ex;
                        } finally {
                            runProgress.set(null);
                            // Removes any parameter
                            runParameters.set(null);
                            // Starting
                            jobListener.onJobComplete(job.getKey());
                        }
                    } else {
                        logger.debug("[job]{} Not allowed to run now", job.getKey());
                    }
                } else {
                    logger.debug("[job]{} Not valid - removing from schedule", job.getKey());
                    unschedule(job.getKey(), false);
                }
            }
        }

        /**
         * A job can run if:
         * 

* * not disabled * * AND (forced OR not paused) * * AND schedule NOT paused */ private boolean canRunNow(boolean force) { return !job.isDisabled() && (!paused.get() || force) && !schedulerPaused.get(); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy