Skip to content

Commit b40021c

Browse files
fix: Fix long running jobs getting executed multiple times (#14)
--------- Signed-off-by: starry-shivam <[email protected]>
1 parent 76561b8 commit b40021c

File tree

2 files changed

+34
-2
lines changed

2 files changed

+34
-2
lines changed

src/main/kotlin/dev/starry/ktscheduler/scheduler/KtScheduler.kt

+6-2
Original file line numberDiff line numberDiff line change
@@ -277,6 +277,11 @@ class KtScheduler(
277277

278278
dueJobs.forEach { job ->
279279
logger.info("Processing due jobs...")
280+
// Set the next run time for the job or remove it if it has no next run time.
281+
// We do this before executing the job to ensure that the job is not executed again
282+
// if its task is long-running and the next run time is already due.
283+
setNextRunTimeOrRemoveJob(job, now)
284+
// Execute the job.
280285
executor.execute(
281286
job = job,
282287
onSuccess = { handleJobCompletion(job, now) },
@@ -287,14 +292,13 @@ class KtScheduler(
287292

288293
// Handles the completion of a job by updating the next run time or removing the job.
289294
private fun handleJobCompletion(job: Job, now: ZonedDateTime) {
290-
setNextRunTimeOrRemoveJob(job, now)
295+
logger.info("Job ${job.jobId} completed successfully")
291296
notifyJobComplete(job.jobId)
292297
}
293298

294299
// Handles an error encountered while executing a job.
295300
private fun handleJobError(job: Job, now: ZonedDateTime, exception: Exception) {
296301
logger.severe("Error executing job ${job.jobId}: $exception")
297-
setNextRunTimeOrRemoveJob(job, now)
298302
notifyJobError(job.jobId, exception)
299303
}
300304

src/test/kotlin/dev/starry/ktscheduler/KtSchedulerTest.kt

+28
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@ import dev.starry.ktscheduler.triggers.OneTimeTrigger
2626
import junit.framework.TestCase.assertFalse
2727
import junit.framework.TestCase.assertTrue
2828
import kotlinx.coroutines.ExperimentalCoroutinesApi
29+
import kotlinx.coroutines.delay
2930
import kotlinx.coroutines.test.runTest
3031
import org.junit.Test
3132
import java.time.ZonedDateTime
@@ -289,6 +290,33 @@ class KtSchedulerTest {
289290
scheduler.shutdown()
290291
}
291292

293+
@Test
294+
fun `scheduler should not execute job multiple times if it is still running`(): Unit = runTest {
295+
val scheduler = KtScheduler()
296+
297+
// Create a job that takes 2 seconds to execute
298+
val job = Job(
299+
jobId = "longRunningJob",
300+
trigger = IntervalTrigger(intervalSeconds = 1),
301+
nextRunTime = ZonedDateTime.now().plusSeconds(1),
302+
callback = { delay(2000) }
303+
)
304+
val eventListener = TestJobEventListener()
305+
306+
scheduler.addJob(job)
307+
scheduler.addEventListener(eventListener)
308+
scheduler.start()
309+
Thread.sleep(200)
310+
// Job should not be completed yet
311+
assertEquals(0, eventListener.completedJobs.size)
312+
// Wait for enough time to ensure job has run
313+
Thread.sleep(3000)
314+
scheduler.shutdown()
315+
// Assert that the job was only executed once
316+
assertEquals(1, eventListener.completedJobs.size)
317+
assertEquals("longRunningJob", eventListener.completedJobs[0])
318+
}
319+
292320
private fun createTestJob(
293321
jobId: String,
294322
runAt: ZonedDateTime = ZonedDateTime.now().plusSeconds(1),

0 commit comments

Comments
 (0)