com.github.kagkarlsson.scheduler.FetchCandidates Maven / Gradle / Ivy
/*
* 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.CandidateEventType;
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.Optional;
import java.util.concurrent.Callable;
import java.util.concurrent.atomic.AtomicInteger;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class FetchCandidates implements PollStrategy {
private static final Logger LOG = LoggerFactory.getLogger(FetchCandidates.class);
private final Executor executor;
private final TaskRepository taskRepository;
private final SchedulerClient schedulerClient;
private final SchedulerListeners schedulerListeners;
private final List executionInterceptors;
private final SchedulerState schedulerState;
private final ConfigurableLogger failureLogger;
private final TaskResolver taskResolver;
private final Clock clock;
private final PollingStrategyConfig pollingStrategyConfig;
private final Runnable triggerCheckForNewExecutions;
private HeartbeatConfig heartbeatConfig;
AtomicInteger currentGenerationNumber = new AtomicInteger(0);
private final int lowerLimit;
private final int upperLimit;
public FetchCandidates(
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 heartbeatConfig) {
this.executor = executor;
this.taskRepository = taskRepository;
this.schedulerClient = schedulerClient;
this.schedulerListeners = schedulerListeners;
this.executionInterceptors = executionInterceptors;
this.schedulerState = schedulerState;
this.failureLogger = failureLogger;
this.taskResolver = taskResolver;
this.clock = clock;
this.pollingStrategyConfig = pollingStrategyConfig;
this.triggerCheckForNewExecutions = triggerCheckForNewExecutions;
this.heartbeatConfig = heartbeatConfig;
lowerLimit = pollingStrategyConfig.getLowerLimit(threadpoolSize);
// FIXLATER: this is not "upper limit", but rather nr of executions to get. those already in
// queue will become stale
upperLimit = pollingStrategyConfig.getUpperLimit(threadpoolSize);
}
@Override
public void run() {
Instant now = clock.now();
// Fetch new candidates for execution. Old ones still in ExecutorService will become stale and
// be discarded
final int executionsToFetch = upperLimit;
List fetchedDueExecutions = taskRepository.getDue(now, executionsToFetch);
LOG.trace(
"Fetched {} task instances due for execution at {}", fetchedDueExecutions.size(), now);
currentGenerationNumber.incrementAndGet();
DueExecutionsBatch newDueBatch =
new DueExecutionsBatch(
currentGenerationNumber.get(),
fetchedDueExecutions.size(),
executionsToFetch == fetchedDueExecutions.size(),
(Integer leftInBatch) -> leftInBatch <= lowerLimit);
for (Execution e : fetchedDueExecutions) {
executor.addToQueue(
() -> {
final Optional candidate = new PickDue(e, newDueBatch).call();
candidate.ifPresent(
picked ->
new ExecutePicked(
executor,
taskRepository,
schedulerClient,
schedulerListeners,
executionInterceptors,
taskResolver,
schedulerState,
failureLogger,
clock,
heartbeatConfig,
picked)
.run());
},
() -> {
newDueBatch.oneExecutionDone(triggerCheckForNewExecutions::run);
});
}
schedulerListeners.onSchedulerEvent(SchedulerEventType.RAN_EXECUTE_DUE);
}
private class PickDue implements Callable> {
private final Execution candidate;
private final DueExecutionsBatch addedDueExecutionsBatch;
public PickDue(Execution candidate, DueExecutionsBatch dueExecutionsBatch) {
this.candidate = candidate;
this.addedDueExecutionsBatch = dueExecutionsBatch;
}
@Override
public Optional call() {
if (schedulerState.isShuttingDown()) {
LOG.info(
"Scheduler has been shutdown. Skipping fetched due execution: "
+ candidate.taskInstance.getTaskAndInstance());
return Optional.empty();
}
if (addedDueExecutionsBatch.isOlderGenerationThan(currentGenerationNumber.get())) {
// skipping execution due to it being stale
addedDueExecutionsBatch.markBatchAsStale();
schedulerListeners.onCandidateEvent(CandidateEventType.STALE);
LOG.trace(
"Skipping queued execution (current generationNumber: {}, execution generationNumber: {})",
currentGenerationNumber,
addedDueExecutionsBatch.getGenerationNumber());
return Optional.empty();
}
final Optional pickedExecution = taskRepository.pick(candidate, clock.now());
if (!pickedExecution.isPresent()) {
// someone else picked id
LOG.debug("Execution picked by another scheduler. Continuing to next due execution.");
schedulerListeners.onCandidateEvent(CandidateEventType.ALREADY_PICKED);
return Optional.empty();
}
return pickedExecution;
}
}
}