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

com.github.kagkarlsson.scheduler.LockAndFetchCandidates Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) Gustav Karlsson
 *
 * 

Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of the License at * *

http://www.apache.org/licenses/LICENSE-2.0 * *

Unless required by applicable law or agreed to in writing, software distributed under the * License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either * express or implied. See the License for the specific language governing permissions and * limitations under the License. */ package com.github.kagkarlsson.scheduler; import com.github.kagkarlsson.scheduler.event.ExecutionInterceptor; import com.github.kagkarlsson.scheduler.event.SchedulerListener.SchedulerEventType; import com.github.kagkarlsson.scheduler.event.SchedulerListeners; import com.github.kagkarlsson.scheduler.logging.ConfigurableLogger; import com.github.kagkarlsson.scheduler.task.Execution; import java.time.Instant; import java.util.List; import java.util.concurrent.atomic.AtomicBoolean; import org.slf4j.Logger; import org.slf4j.LoggerFactory; public class LockAndFetchCandidates implements PollStrategy { private static final Logger LOG = LoggerFactory.getLogger(LockAndFetchCandidates.class); private final Executor executor; private final TaskRepository taskRepository; private final SchedulerClient schedulerClient; private final SchedulerListeners schedulerListeners; private final List executionInterceptors; private final TaskResolver taskResolver; private final SchedulerState schedulerState; private final ConfigurableLogger failureLogger; private final Clock clock; private final PollingStrategyConfig pollingStrategyConfig; private final Runnable triggerCheckForNewExecutions; private HeartbeatConfig maxAgeBeforeConsideredDead; private final int lowerLimit; private final int upperLimit; private AtomicBoolean moreExecutionsInDatabase = new AtomicBoolean(false); public LockAndFetchCandidates( Executor executor, TaskRepository taskRepository, SchedulerClient schedulerClient, int threadpoolSize, SchedulerListeners schedulerListeners, List executionInterceptors, SchedulerState schedulerState, ConfigurableLogger failureLogger, TaskResolver taskResolver, Clock clock, PollingStrategyConfig pollingStrategyConfig, Runnable triggerCheckForNewExecutions, HeartbeatConfig maxAgeBeforeConsideredDead) { this.executor = executor; this.taskRepository = taskRepository; this.schedulerClient = schedulerClient; this.schedulerListeners = schedulerListeners; this.executionInterceptors = executionInterceptors; this.taskResolver = taskResolver; this.schedulerState = schedulerState; this.failureLogger = failureLogger; this.clock = clock; this.pollingStrategyConfig = pollingStrategyConfig; this.triggerCheckForNewExecutions = triggerCheckForNewExecutions; this.maxAgeBeforeConsideredDead = maxAgeBeforeConsideredDead; lowerLimit = pollingStrategyConfig.getLowerLimit(threadpoolSize); upperLimit = pollingStrategyConfig.getUpperLimit(threadpoolSize); } @Override public void run() { Instant now = clock.now(); int executionsToFetch = upperLimit - executor.getNumberInQueueOrProcessing(); // Might happen if upperLimit == threads and all threads are busy if (executionsToFetch <= 0) { LOG.trace("No executions to fetch."); return; } // FIXLATER: should it fetch here if not under lowerLimit? probably List pickedExecutions = taskRepository.lockAndGetDue(now, executionsToFetch); LOG.trace("Picked {} taskinstances due for execution", pickedExecutions.size()); // Shared indicator for if there are more due executions in the database. // As soon as we know there are not more executions in the database, we can stop triggering // checks for more (and vice versa) moreExecutionsInDatabase.set(pickedExecutions.size() == executionsToFetch); if (pickedExecutions.size() == 0) { // No picked executions to execute LOG.trace("No executions due."); return; } for (Execution picked : pickedExecutions) { executor.addToQueue( new ExecutePicked( executor, taskRepository, schedulerClient, schedulerListeners, executionInterceptors, taskResolver, schedulerState, failureLogger, clock, maxAgeBeforeConsideredDead, picked), () -> { if (moreExecutionsInDatabase.get() && executor.getNumberInQueueOrProcessing() <= lowerLimit) { triggerCheckForNewExecutions.run(); } }); } schedulerListeners.onSchedulerEvent(SchedulerEventType.RAN_EXECUTE_DUE); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy