@ -1,11 +1,11 @@
/ *
* Copyright 2012 - 20 17 the original author or authors .
* Copyright 2012 - 20 2 1 the original author or authors .
*
* 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
* http s : //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 ,
@ -16,64 +16,695 @@
package org.springframework.boot.actuate.quartz ;
import java.time.Duration ;
import java.time.Instant ;
import java.time.LocalTime ;
import java.time.temporal.ChronoUnit ;
import java.util.ArrayList ;
import java.util.Arrays ;
import java.util.Calendar ;
import java.util.Collections ;
import java.util.Date ;
import java.util.LinkedHashSet ;
import java.util.List ;
import java.util.Map ;
import java.util.Map.Entry ;
import java.util.TimeZone ;
import java.util.stream.Stream ;
import org.junit.Test ;
import org.assertj.core.api.InstanceOfAssertFactories ;
import org.assertj.core.api.InstanceOfAssertFactory ;
import org.assertj.core.api.MapAssert ;
import org.junit.jupiter.api.Test ;
import org.junit.jupiter.params.ParameterizedTest ;
import org.junit.jupiter.params.provider.Arguments ;
import org.junit.jupiter.params.provider.MethodSource ;
import org.quartz.CalendarIntervalScheduleBuilder ;
import org.quartz.CalendarIntervalTrigger ;
import org.quartz.CronScheduleBuilder ;
import org.quartz.CronTrigger ;
import org.quartz.DailyTimeIntervalScheduleBuilder ;
import org.quartz.DailyTimeIntervalTrigger ;
import org.quartz.DateBuilder.IntervalUnit ;
import org.quartz.Job ;
import org.quartz.JobBuilder ;
import org.quartz.JobDetail ;
import org.quartz.JobKey ;
import org.quartz.Scheduler ;
import org.quartz.SchedulerException ;
import org.quartz.SimpleScheduleBuilder ;
import org.quartz.SimpleTrigger ;
import org.quartz.TimeOfDay ;
import org.quartz.Trigger ;
import org.quartz.Trigger.TriggerState ;
import org.quartz.TriggerBuilder ;
import org.quartz.TriggerKey ;
import org.quartz.impl.matchers.GroupMatcher ;
import org.quartz.spi.OperableTrigger ;
import org.springframework.boot.actuate.quartz.QuartzEndpoint.QuartzJob ;
import org.springframework.boot.actuate.endpoint.Sanitizer ;
import org.springframework.boot.actuate.quartz.QuartzEndpoint.QuartzJobDetails ;
import org.springframework.boot.actuate.quartz.QuartzEndpoint.QuartzJobGroupSummary ;
import org.springframework.boot.actuate.quartz.QuartzEndpoint.QuartzJobSummary ;
import org.springframework.boot.actuate.quartz.QuartzEndpoint.QuartzReport ;
import org.springframework.boot.actuate.quartz.QuartzEndpoint.QuartzTriggerGroupSummary ;
import org.springframework.scheduling.quartz.DelegatingJob ;
import org.springframework.util.LinkedMultiValueMap ;
import org.springframework.util.MultiValueMap ;
import static org.assertj.core.api.Assertions.assertThat ;
import static org.assertj.core.api.Assertions.entry ;
import static org.mockito.BDDMockito.given ;
import static org.mockito.Mockito.mock ;
import static org.mockito.Mockito.verify ;
import static org.mockito.Mockito.verifyNoMoreInteractions ;
/ * *
* Tests for { @link QuartzEndpoint } .
*
* @author Vedran Pavic
* @author Stephane Nicoll
* /
public class QuartzEndpointTests {
class QuartzEndpointTests {
private final Scheduler scheduler = mock ( Scheduler . class ) ;
private static final JobDetail jobOne = JobBuilder . newJob ( Job . class ) . withIdentity ( "jobOne" ) . build ( ) ;
private final QuartzEndpoint endpoint = new QuartzEndpoint ( this . scheduler ) ;
private static final JobDetail jobTwo = JobBuilder . newJob ( DelegatingJob . class ) . withIdentity ( "jobTwo" ) . build ( ) ;
private final JobDetail jobDetail = JobBuilder . newJob ( Job . class ) . withIdentity ( " testJob ") . build ( ) ;
private static final JobDetail jobThree = JobBuilder . newJob ( Job . class ) . withIdentity ( " jobThree", "samples ") . build ( ) ;
private final Trigger trigger = TriggerBuilder . newTrigger ( ) . forJob ( this . jobDetail ) . withIdentity ( "t estT rigger")
private static final Trigger trigger One = TriggerBuilder . newTrigger ( ) . forJob ( jobOne ) . withIdentity ( "t riggerOne ")
. build ( ) ;
private static final Trigger triggerTwo = TriggerBuilder . newTrigger ( ) . forJob ( jobOne ) . withIdentity ( "triggerTwo" )
. build ( ) ;
private static final Trigger triggerThree = TriggerBuilder . newTrigger ( ) . forJob ( jobThree )
. withIdentity ( "triggerThree" , "samples" ) . build ( ) ;
private final Scheduler scheduler ;
private final QuartzEndpoint endpoint ;
QuartzEndpointTests ( ) {
this . scheduler = mock ( Scheduler . class ) ;
this . endpoint = new QuartzEndpoint ( this . scheduler ) ;
}
@Test
void quartzReport ( ) throws SchedulerException {
given ( this . scheduler . getJobGroupNames ( ) ) . willReturn ( Arrays . asList ( "jobSamples" , "DEFAULT" ) ) ;
given ( this . scheduler . getTriggerGroupNames ( ) ) . willReturn ( Collections . singletonList ( "triggerSamples" ) ) ;
QuartzReport quartzReport = this . endpoint . quartzReport ( ) ;
assertThat ( quartzReport . getJobs ( ) . getGroups ( ) ) . containsOnly ( "jobSamples" , "DEFAULT" ) ;
assertThat ( quartzReport . getTriggers ( ) . getGroups ( ) ) . containsOnly ( "triggerSamples" ) ;
verify ( this . scheduler ) . getJobGroupNames ( ) ;
verify ( this . scheduler ) . getTriggerGroupNames ( ) ;
verifyNoMoreInteractions ( this . scheduler ) ;
}
@Test
void quartzReportWithNoJob ( ) throws SchedulerException {
given ( this . scheduler . getJobGroupNames ( ) ) . willReturn ( Collections . emptyList ( ) ) ;
given ( this . scheduler . getTriggerGroupNames ( ) ) . willReturn ( Arrays . asList ( "triggerSamples" , "DEFAULT" ) ) ;
QuartzReport quartzReport = this . endpoint . quartzReport ( ) ;
assertThat ( quartzReport . getJobs ( ) . getGroups ( ) ) . isEmpty ( ) ;
assertThat ( quartzReport . getTriggers ( ) . getGroups ( ) ) . containsOnly ( "triggerSamples" , "DEFAULT" ) ;
}
@Test
void quartzReportWithNoTrigger ( ) throws SchedulerException {
given ( this . scheduler . getJobGroupNames ( ) ) . willReturn ( Collections . singletonList ( "jobSamples" ) ) ;
given ( this . scheduler . getTriggerGroupNames ( ) ) . willReturn ( Collections . emptyList ( ) ) ;
QuartzReport quartzReport = this . endpoint . quartzReport ( ) ;
assertThat ( quartzReport . getJobs ( ) . getGroups ( ) ) . containsOnly ( "jobSamples" ) ;
assertThat ( quartzReport . getTriggers ( ) . getGroups ( ) ) . isEmpty ( ) ;
}
@Test
void quartzJobGroupsWithExistingGroups ( ) throws SchedulerException {
mockJobs ( jobOne , jobTwo , jobThree ) ;
Map < String , Object > jobGroups = this . endpoint . quartzJobGroups ( ) . getGroups ( ) ;
assertThat ( jobGroups ) . containsOnlyKeys ( "DEFAULT" , "samples" ) ;
assertThat ( jobGroups ) . extractingByKey ( "DEFAULT" , nestedMap ( ) )
. containsOnly ( entry ( "jobs" , Arrays . asList ( "jobOne" , "jobTwo" ) ) ) ;
assertThat ( jobGroups ) . extractingByKey ( "samples" , nestedMap ( ) )
. containsOnly ( entry ( "jobs" , Collections . singletonList ( "jobThree" ) ) ) ;
}
@Test
void quartzJobGroupsWithNoGroup ( ) throws SchedulerException {
given ( this . scheduler . getJobGroupNames ( ) ) . willReturn ( Collections . emptyList ( ) ) ;
Map < String , Object > jobGroups = this . endpoint . quartzJobGroups ( ) . getGroups ( ) ;
assertThat ( jobGroups ) . isEmpty ( ) ;
}
@Test
void quartzTriggerGroupsWithExistingGroups ( ) throws SchedulerException {
mockTriggers ( triggerOne , triggerTwo , triggerThree ) ;
given ( this . scheduler . getPausedTriggerGroups ( ) ) . willReturn ( Collections . singleton ( "samples" ) ) ;
Map < String , Object > triggerGroups = this . endpoint . quartzTriggerGroups ( ) . getGroups ( ) ;
assertThat ( triggerGroups ) . containsOnlyKeys ( "DEFAULT" , "samples" ) ;
assertThat ( triggerGroups ) . extractingByKey ( "DEFAULT" , nestedMap ( ) ) . containsOnly ( entry ( "paused" , false ) ,
entry ( "triggers" , Arrays . asList ( "triggerOne" , "triggerTwo" ) ) ) ;
assertThat ( triggerGroups ) . extractingByKey ( "samples" , nestedMap ( ) ) . containsOnly ( entry ( "paused" , true ) ,
entry ( "triggers" , Collections . singletonList ( "triggerThree" ) ) ) ;
}
@Test
void quartzTriggerGroupsWithNoGroup ( ) throws SchedulerException {
given ( this . scheduler . getTriggerGroupNames ( ) ) . willReturn ( Collections . emptyList ( ) ) ;
Map < String , Object > triggerGroups = this . endpoint . quartzTriggerGroups ( ) . getGroups ( ) ;
assertThat ( triggerGroups ) . isEmpty ( ) ;
}
@Test
void quartzJobGroupSummaryWithInvalidGroup ( ) throws SchedulerException {
given ( this . scheduler . getJobGroupNames ( ) ) . willReturn ( Collections . singletonList ( "DEFAULT" ) ) ;
QuartzJobGroupSummary summary = this . endpoint . quartzJobGroupSummary ( "unknown" ) ;
assertThat ( summary ) . isNull ( ) ;
}
@Test
void quartzJobGroupSummaryWithEmptyGroup ( ) throws SchedulerException {
given ( this . scheduler . getJobGroupNames ( ) ) . willReturn ( Collections . singletonList ( "samples" ) ) ;
given ( this . scheduler . getJobKeys ( GroupMatcher . jobGroupEquals ( "samples" ) ) ) . willReturn ( Collections . emptySet ( ) ) ;
QuartzJobGroupSummary summary = this . endpoint . quartzJobGroupSummary ( "samples" ) ;
assertThat ( summary ) . isNotNull ( ) ;
assertThat ( summary . getGroup ( ) ) . isEqualTo ( "samples" ) ;
assertThat ( summary . getJobs ( ) ) . isEmpty ( ) ;
}
@Test
void quartzJobGroupSummaryWithJobs ( ) throws SchedulerException {
mockJobs ( jobOne , jobTwo ) ;
QuartzJobGroupSummary summary = this . endpoint . quartzJobGroupSummary ( "DEFAULT" ) ;
assertThat ( summary ) . isNotNull ( ) ;
assertThat ( summary . getGroup ( ) ) . isEqualTo ( "DEFAULT" ) ;
Map < String , QuartzJobSummary > jobSummaries = summary . getJobs ( ) ;
assertThat ( jobSummaries ) . containsOnlyKeys ( "jobOne" , "jobTwo" ) ;
assertThat ( jobSummaries . get ( "jobOne" ) . getClassName ( ) ) . isEqualTo ( Job . class . getName ( ) ) ;
assertThat ( jobSummaries . get ( "jobTwo" ) . getClassName ( ) ) . isEqualTo ( DelegatingJob . class . getName ( ) ) ;
}
@Test
void quartzTriggerGroupSummaryWithInvalidGroup ( ) throws SchedulerException {
given ( this . scheduler . getTriggerGroupNames ( ) ) . willReturn ( Collections . singletonList ( "DEFAULT" ) ) ;
QuartzTriggerGroupSummary summary = this . endpoint . quartzTriggerGroupSummary ( "unknown" ) ;
assertThat ( summary ) . isNull ( ) ;
}
@Test
void quartzTriggerGroupSummaryWithEmptyGroup ( ) throws SchedulerException {
given ( this . scheduler . getTriggerGroupNames ( ) ) . willReturn ( Collections . singletonList ( "samples" ) ) ;
given ( this . scheduler . getTriggerKeys ( GroupMatcher . triggerGroupEquals ( "samples" ) ) )
. willReturn ( Collections . emptySet ( ) ) ;
QuartzTriggerGroupSummary summary = this . endpoint . quartzTriggerGroupSummary ( "samples" ) ;
assertThat ( summary ) . isNotNull ( ) ;
assertThat ( summary . getGroup ( ) ) . isEqualTo ( "samples" ) ;
assertThat ( summary . isPaused ( ) ) . isFalse ( ) ;
assertThat ( summary . getTriggers ( ) . getCron ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getSimple ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getDailyTimeInterval ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getCalendarInterval ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getCustom ( ) ) . isEmpty ( ) ;
}
@Test
void quartzTriggerGroupSummaryWithCronTrigger ( ) throws SchedulerException {
CronTrigger cronTrigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "3am-every-day" , "samples" )
. withSchedule ( CronScheduleBuilder . dailyAtHourAndMinute ( 3 , 0 ) ) . build ( ) ;
mockTriggers ( cronTrigger ) ;
QuartzTriggerGroupSummary summary = this . endpoint . quartzTriggerGroupSummary ( "samples" ) ;
assertThat ( summary . getGroup ( ) ) . isEqualTo ( "samples" ) ;
assertThat ( summary . isPaused ( ) ) . isFalse ( ) ;
assertThat ( summary . getTriggers ( ) . getCron ( ) ) . containsOnlyKeys ( "3am-every-day" ) ;
assertThat ( summary . getTriggers ( ) . getSimple ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getDailyTimeInterval ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getCalendarInterval ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getCustom ( ) ) . isEmpty ( ) ;
}
@Test
void quartzTriggerGroupSummaryWithCronTriggerDetails ( ) throws SchedulerException {
Date previousFireTime = Date . from ( Instant . parse ( "2020-11-30T03:00:00Z" ) ) ;
Date nextFireTime = Date . from ( Instant . parse ( "2020-12-01T03:00:00Z" ) ) ;
TimeZone timeZone = TimeZone . getTimeZone ( "Europe/Paris" ) ;
CronTrigger cronTrigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "3am-every-day" , "samples" ) . withPriority ( 3 )
. withSchedule ( CronScheduleBuilder . dailyAtHourAndMinute ( 3 , 0 ) . inTimeZone ( timeZone ) ) . build ( ) ;
( ( OperableTrigger ) cronTrigger ) . setPreviousFireTime ( previousFireTime ) ;
( ( OperableTrigger ) cronTrigger ) . setNextFireTime ( nextFireTime ) ;
mockTriggers ( cronTrigger ) ;
QuartzTriggerGroupSummary summary = this . endpoint . quartzTriggerGroupSummary ( "samples" ) ;
Map < String , Object > triggers = summary . getTriggers ( ) . getCron ( ) ;
assertThat ( triggers ) . containsOnlyKeys ( "3am-every-day" ) ;
assertThat ( triggers ) . extractingByKey ( "3am-every-day" , nestedMap ( ) ) . containsOnly (
entry ( "previousFireTime" , previousFireTime ) , entry ( "nextFireTime" , nextFireTime ) , entry ( "priority" , 3 ) ,
entry ( "expression" , "0 0 3 ? * *" ) , entry ( "timeZone" , timeZone ) ) ;
}
@Test
void quartzTriggerGroupSummaryWithSimpleTrigger ( ) throws SchedulerException {
SimpleTrigger simpleTrigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "every-hour" , "samples" )
. withSchedule ( SimpleScheduleBuilder . repeatHourlyForever ( 1 ) ) . build ( ) ;
mockTriggers ( simpleTrigger ) ;
QuartzTriggerGroupSummary summary = this . endpoint . quartzTriggerGroupSummary ( "samples" ) ;
assertThat ( summary . getGroup ( ) ) . isEqualTo ( "samples" ) ;
assertThat ( summary . isPaused ( ) ) . isFalse ( ) ;
assertThat ( summary . getTriggers ( ) . getCron ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getSimple ( ) ) . containsOnlyKeys ( "every-hour" ) ;
assertThat ( summary . getTriggers ( ) . getDailyTimeInterval ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getCalendarInterval ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getCustom ( ) ) . isEmpty ( ) ;
}
@Test
void quartzTriggerGroupSummaryWithSimpleTriggerDetails ( ) throws SchedulerException {
Date previousFireTime = Date . from ( Instant . parse ( "2020-11-30T03:00:00Z" ) ) ;
Date nextFireTime = Date . from ( Instant . parse ( "2020-12-01T03:00:00Z" ) ) ;
SimpleTrigger simpleTrigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "every-hour" , "samples" ) . withPriority ( 7 )
. withSchedule ( SimpleScheduleBuilder . repeatHourlyForever ( 1 ) ) . build ( ) ;
( ( OperableTrigger ) simpleTrigger ) . setPreviousFireTime ( previousFireTime ) ;
( ( OperableTrigger ) simpleTrigger ) . setNextFireTime ( nextFireTime ) ;
mockTriggers ( simpleTrigger ) ;
QuartzTriggerGroupSummary summary = this . endpoint . quartzTriggerGroupSummary ( "samples" ) ;
Map < String , Object > triggers = summary . getTriggers ( ) . getSimple ( ) ;
assertThat ( triggers ) . containsOnlyKeys ( "every-hour" ) ;
assertThat ( triggers ) . extractingByKey ( "every-hour" , nestedMap ( ) ) . containsOnly (
entry ( "previousFireTime" , previousFireTime ) , entry ( "nextFireTime" , nextFireTime ) , entry ( "priority" , 7 ) ,
entry ( "interval" , 3600000L ) ) ;
}
@Test
void quartzTriggerGroupSummaryWithDailyIntervalTrigger ( ) throws SchedulerException {
DailyTimeIntervalTrigger trigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "every-hour-9am" , "samples" )
. withSchedule ( DailyTimeIntervalScheduleBuilder . dailyTimeIntervalSchedule ( )
. startingDailyAt ( TimeOfDay . hourAndMinuteOfDay ( 9 , 0 ) ) . withInterval ( 1 , IntervalUnit . HOUR ) )
. build ( ) ;
mockTriggers ( trigger ) ;
QuartzTriggerGroupSummary summary = this . endpoint . quartzTriggerGroupSummary ( "samples" ) ;
assertThat ( summary . getGroup ( ) ) . isEqualTo ( "samples" ) ;
assertThat ( summary . isPaused ( ) ) . isFalse ( ) ;
assertThat ( summary . getTriggers ( ) . getCron ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getSimple ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getDailyTimeInterval ( ) ) . containsOnlyKeys ( "every-hour-9am" ) ;
assertThat ( summary . getTriggers ( ) . getCalendarInterval ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getCustom ( ) ) . isEmpty ( ) ;
}
@Test
public void quartzReport ( ) throws Exception {
String jobGroup = this . jobDetail . getKey ( ) . getGroup ( ) ;
given ( this . scheduler . getJobGroupNames ( ) ) . willReturn ( Collections . singletonList ( jobGroup ) ) ;
given ( this . scheduler . getJobKeys ( GroupMatcher . jobGroupEquals ( jobGroup ) ) )
. willReturn ( Collections . singleton ( this . jobDetail . getKey ( ) ) ) ;
Map < String , Object > quartzReport = this . endpoint . quartzReport ( ) ;
assertThat ( quartzReport ) . hasSize ( 1 ) ;
}
@Test
public void quartzJob ( ) throws Exception {
JobKey jobKey = this . jobDetail . getKey ( ) ;
given ( this . scheduler . getJobDetail ( jobKey ) ) . willReturn ( this . jobDetail ) ;
given ( this . scheduler . getTriggersOfJob ( jobKey ) )
. willAnswer ( invocation - > Collections . singletonList ( this . trigger ) ) ;
QuartzJob quartzJob = this . endpoint . quartzJob ( jobKey . getGroup ( ) , jobKey . getName ( ) ) ;
assertThat ( quartzJob . getJobGroup ( ) ) . isEqualTo ( jobKey . getGroup ( ) ) ;
assertThat ( quartzJob . getJobName ( ) ) . isEqualTo ( jobKey . getName ( ) ) ;
assertThat ( quartzJob . getClassName ( ) ) . isEqualTo ( this . jobDetail . getJobClass ( ) . getName ( ) ) ;
assertThat ( quartzJob . getTriggers ( ) ) . hasSize ( 1 ) ;
assertThat ( quartzJob . getTriggers ( ) . get ( 0 ) . getTriggerGroup ( ) ) . isEqualTo ( this . trigger . getKey ( ) . getGroup ( ) ) ;
assertThat ( quartzJob . getTriggers ( ) . get ( 0 ) . getTriggerName ( ) ) . isEqualTo ( this . trigger . getKey ( ) . getName ( ) ) ;
void quartzTriggerGroupSummaryWithDailyIntervalTriggerDetails ( ) throws SchedulerException {
Date previousFireTime = Date . from ( Instant . parse ( "2020-11-30T03:00:00Z" ) ) ;
Date nextFireTime = Date . from ( Instant . parse ( "2020-12-01T03:00:00Z" ) ) ;
DailyTimeIntervalTrigger trigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "every-hour-tue-thu" , "samples" )
. withPriority ( 4 )
. withSchedule ( DailyTimeIntervalScheduleBuilder . dailyTimeIntervalSchedule ( )
. onDaysOfTheWeek ( Calendar . TUESDAY , Calendar . THURSDAY )
. startingDailyAt ( TimeOfDay . hourAndMinuteOfDay ( 9 , 0 ) )
. endingDailyAt ( TimeOfDay . hourAndMinuteOfDay ( 18 , 0 ) ) . withInterval ( 1 , IntervalUnit . HOUR ) )
. build ( ) ;
( ( OperableTrigger ) trigger ) . setPreviousFireTime ( previousFireTime ) ;
( ( OperableTrigger ) trigger ) . setNextFireTime ( nextFireTime ) ;
mockTriggers ( trigger ) ;
QuartzTriggerGroupSummary summary = this . endpoint . quartzTriggerGroupSummary ( "samples" ) ;
Map < String , Object > triggers = summary . getTriggers ( ) . getDailyTimeInterval ( ) ;
assertThat ( triggers ) . containsOnlyKeys ( "every-hour-tue-thu" ) ;
assertThat ( triggers ) . extractingByKey ( "every-hour-tue-thu" , nestedMap ( ) ) . containsOnly (
entry ( "previousFireTime" , previousFireTime ) , entry ( "nextFireTime" , nextFireTime ) , entry ( "priority" , 4 ) ,
entry ( "interval" , 3600000L ) , entry ( "startTimeOfDay" , LocalTime . of ( 9 , 0 ) ) ,
entry ( "endTimeOfDay" , LocalTime . of ( 18 , 0 ) ) ,
entry ( "daysOfWeek" , new LinkedHashSet < > ( Arrays . asList ( 3 , 5 ) ) ) ) ;
}
@Test
void quartzTriggerGroupSummaryWithCalendarIntervalTrigger ( ) throws SchedulerException {
CalendarIntervalTrigger trigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "once-a-week" , "samples" )
. withSchedule ( CalendarIntervalScheduleBuilder . calendarIntervalSchedule ( ) . withIntervalInWeeks ( 1 ) )
. build ( ) ;
mockTriggers ( trigger ) ;
QuartzTriggerGroupSummary summary = this . endpoint . quartzTriggerGroupSummary ( "samples" ) ;
assertThat ( summary . getGroup ( ) ) . isEqualTo ( "samples" ) ;
assertThat ( summary . isPaused ( ) ) . isFalse ( ) ;
assertThat ( summary . getTriggers ( ) . getCron ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getSimple ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getDailyTimeInterval ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getCalendarInterval ( ) ) . containsOnlyKeys ( "once-a-week" ) ;
assertThat ( summary . getTriggers ( ) . getCustom ( ) ) . isEmpty ( ) ;
}
@Test
void quartzTriggerGroupSummaryWithCalendarIntervalTriggerDetails ( ) throws SchedulerException {
TimeZone timeZone = TimeZone . getTimeZone ( "Europe/Paris" ) ;
Date previousFireTime = Date . from ( Instant . parse ( "2020-11-30T03:00:00Z" ) ) ;
Date nextFireTime = Date . from ( Instant . parse ( "2020-12-01T03:00:00Z" ) ) ;
CalendarIntervalTrigger trigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "once-a-week" , "samples" )
. withPriority ( 8 ) . withSchedule ( CalendarIntervalScheduleBuilder . calendarIntervalSchedule ( )
. withIntervalInWeeks ( 1 ) . inTimeZone ( timeZone ) )
. build ( ) ;
( ( OperableTrigger ) trigger ) . setPreviousFireTime ( previousFireTime ) ;
( ( OperableTrigger ) trigger ) . setNextFireTime ( nextFireTime ) ;
mockTriggers ( trigger ) ;
QuartzTriggerGroupSummary summary = this . endpoint . quartzTriggerGroupSummary ( "samples" ) ;
Map < String , Object > triggers = summary . getTriggers ( ) . getCalendarInterval ( ) ;
assertThat ( triggers ) . containsOnlyKeys ( "once-a-week" ) ;
assertThat ( triggers ) . extractingByKey ( "once-a-week" , nestedMap ( ) ) . containsOnly (
entry ( "previousFireTime" , previousFireTime ) , entry ( "nextFireTime" , nextFireTime ) , entry ( "priority" , 8 ) ,
entry ( "interval" , 604800000L ) , entry ( "timeZone" , timeZone ) ) ;
}
@Test
void quartzTriggerGroupSummaryWithCustomTrigger ( ) throws SchedulerException {
Trigger trigger = mock ( Trigger . class ) ;
given ( trigger . getKey ( ) ) . willReturn ( TriggerKey . triggerKey ( "custom" , "samples" ) ) ;
mockTriggers ( trigger ) ;
QuartzTriggerGroupSummary summary = this . endpoint . quartzTriggerGroupSummary ( "samples" ) ;
assertThat ( summary . getGroup ( ) ) . isEqualTo ( "samples" ) ;
assertThat ( summary . isPaused ( ) ) . isFalse ( ) ;
assertThat ( summary . getTriggers ( ) . getCron ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getSimple ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getDailyTimeInterval ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getCalendarInterval ( ) ) . isEmpty ( ) ;
assertThat ( summary . getTriggers ( ) . getCustom ( ) ) . containsOnlyKeys ( "custom" ) ;
}
@Test
void quartzTriggerGroupSummaryWithCustomTriggerDetails ( ) throws SchedulerException {
Date previousFireTime = Date . from ( Instant . parse ( "2020-11-30T03:00:00Z" ) ) ;
Date nextFireTime = Date . from ( Instant . parse ( "2020-12-01T03:00:00Z" ) ) ;
Trigger trigger = mock ( Trigger . class ) ;
given ( trigger . getKey ( ) ) . willReturn ( TriggerKey . triggerKey ( "custom" , "samples" ) ) ;
given ( trigger . getPreviousFireTime ( ) ) . willReturn ( previousFireTime ) ;
given ( trigger . getNextFireTime ( ) ) . willReturn ( nextFireTime ) ;
given ( trigger . getPriority ( ) ) . willReturn ( 9 ) ;
mockTriggers ( trigger ) ;
QuartzTriggerGroupSummary summary = this . endpoint . quartzTriggerGroupSummary ( "samples" ) ;
Map < String , Object > triggers = summary . getTriggers ( ) . getCustom ( ) ;
assertThat ( triggers ) . containsOnlyKeys ( "custom" ) ;
assertThat ( triggers ) . extractingByKey ( "custom" , nestedMap ( ) ) . containsOnly (
entry ( "previousFireTime" , previousFireTime ) , entry ( "nextFireTime" , nextFireTime ) , entry ( "priority" , 9 ) ,
entry ( "trigger" , trigger . toString ( ) ) ) ;
}
@Test
void quartzTriggerWithCronTrigger ( ) throws SchedulerException {
Date previousFireTime = Date . from ( Instant . parse ( "2020-11-30T03:00:00Z" ) ) ;
Date nextFireTime = Date . from ( Instant . parse ( "2020-12-01T03:00:00Z" ) ) ;
TimeZone timeZone = TimeZone . getTimeZone ( "Europe/Paris" ) ;
CronTrigger trigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "3am-every-day" , "samples" ) . withPriority ( 3 )
. withDescription ( "Sample description" )
. withSchedule ( CronScheduleBuilder . dailyAtHourAndMinute ( 3 , 0 ) . inTimeZone ( timeZone ) ) . build ( ) ;
( ( OperableTrigger ) trigger ) . setPreviousFireTime ( previousFireTime ) ;
( ( OperableTrigger ) trigger ) . setNextFireTime ( nextFireTime ) ;
mockTriggers ( trigger ) ;
given ( this . scheduler . getTriggerState ( TriggerKey . triggerKey ( "3am-every-day" , "samples" ) ) )
. willReturn ( TriggerState . NORMAL ) ;
Map < String , Object > triggerDetails = this . endpoint . quartzTrigger ( "samples" , "3am-every-day" ) ;
assertThat ( triggerDetails ) . contains ( entry ( "group" , "samples" ) , entry ( "name" , "3am-every-day" ) ,
entry ( "description" , "Sample description" ) , entry ( "type" , "cron" ) , entry ( "state" , TriggerState . NORMAL ) ,
entry ( "priority" , 3 ) ) ;
assertThat ( triggerDetails ) . contains ( entry ( "previousFireTime" , previousFireTime ) ,
entry ( "nextFireTime" , nextFireTime ) ) ;
assertThat ( triggerDetails ) . doesNotContainKeys ( "simple" , "dailyTimeInterval" , "calendarInterval" , "custom" ) ;
assertThat ( triggerDetails ) . extractingByKey ( "cron" , nestedMap ( ) ) . containsOnly ( entry ( "expression" , "0 0 3 ? * *" ) ,
entry ( "timeZone" , timeZone ) ) ;
}
@Test
void quartzTriggerWithSimpleTrigger ( ) throws SchedulerException {
Date startTime = Date . from ( Instant . parse ( "2020-01-01T09:00:00Z" ) ) ;
Date previousFireTime = Date . from ( Instant . parse ( "2020-11-30T03:00:00Z" ) ) ;
Date nextFireTime = Date . from ( Instant . parse ( "2020-12-01T03:00:00Z" ) ) ;
Date endTime = Date . from ( Instant . parse ( "2020-01-31T09:00:00Z" ) ) ;
SimpleTrigger trigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "every-hour" , "samples" ) . withPriority ( 20 )
. withDescription ( "Every hour" ) . startAt ( startTime ) . endAt ( endTime )
. withSchedule ( SimpleScheduleBuilder . simpleSchedule ( ) . withIntervalInHours ( 1 ) . withRepeatCount ( 2000 ) )
. build ( ) ;
( ( OperableTrigger ) trigger ) . setPreviousFireTime ( previousFireTime ) ;
( ( OperableTrigger ) trigger ) . setNextFireTime ( nextFireTime ) ;
mockTriggers ( trigger ) ;
given ( this . scheduler . getTriggerState ( TriggerKey . triggerKey ( "every-hour" , "samples" ) ) )
. willReturn ( TriggerState . COMPLETE ) ;
Map < String , Object > triggerDetails = this . endpoint . quartzTrigger ( "samples" , "every-hour" ) ;
assertThat ( triggerDetails ) . contains ( entry ( "group" , "samples" ) , entry ( "name" , "every-hour" ) ,
entry ( "description" , "Every hour" ) , entry ( "type" , "simple" ) , entry ( "state" , TriggerState . COMPLETE ) ,
entry ( "priority" , 20 ) ) ;
assertThat ( triggerDetails ) . contains ( entry ( "startTime" , startTime ) , entry ( "previousFireTime" , previousFireTime ) ,
entry ( "nextFireTime" , nextFireTime ) , entry ( "endTime" , endTime ) ) ;
assertThat ( triggerDetails ) . doesNotContainKeys ( "cron" , "dailyTimeInterval" , "calendarInterval" , "custom" ) ;
assertThat ( triggerDetails ) . extractingByKey ( "simple" , nestedMap ( ) ) . containsOnly ( entry ( "interval" , 3600000L ) ,
entry ( "repeatCount" , 2000 ) , entry ( "timesTriggered" , 0 ) ) ;
}
@Test
void quartzTriggerWithDailyTimeIntervalTrigger ( ) throws SchedulerException {
Date previousFireTime = Date . from ( Instant . parse ( "2020-11-30T03:00:00Z" ) ) ;
Date nextFireTime = Date . from ( Instant . parse ( "2020-12-01T03:00:00Z" ) ) ;
DailyTimeIntervalTrigger trigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "every-hour-mon-wed" , "samples" )
. withDescription ( "Every working hour Mon Wed" ) . withPriority ( 4 )
. withSchedule ( DailyTimeIntervalScheduleBuilder . dailyTimeIntervalSchedule ( )
. onDaysOfTheWeek ( Calendar . MONDAY , Calendar . WEDNESDAY )
. startingDailyAt ( TimeOfDay . hourAndMinuteOfDay ( 9 , 0 ) )
. endingDailyAt ( TimeOfDay . hourAndMinuteOfDay ( 18 , 0 ) ) . withInterval ( 1 , IntervalUnit . HOUR ) )
. build ( ) ;
( ( OperableTrigger ) trigger ) . setPreviousFireTime ( previousFireTime ) ;
( ( OperableTrigger ) trigger ) . setNextFireTime ( nextFireTime ) ;
mockTriggers ( trigger ) ;
given ( this . scheduler . getTriggerState ( TriggerKey . triggerKey ( "every-hour-mon-wed" , "samples" ) ) )
. willReturn ( TriggerState . NORMAL ) ;
Map < String , Object > triggerDetails = this . endpoint . quartzTrigger ( "samples" , "every-hour-mon-wed" ) ;
assertThat ( triggerDetails ) . contains ( entry ( "group" , "samples" ) , entry ( "name" , "every-hour-mon-wed" ) ,
entry ( "description" , "Every working hour Mon Wed" ) , entry ( "type" , "dailyTimeInterval" ) ,
entry ( "state" , TriggerState . NORMAL ) , entry ( "priority" , 4 ) ) ;
assertThat ( triggerDetails ) . contains ( entry ( "previousFireTime" , previousFireTime ) ,
entry ( "nextFireTime" , nextFireTime ) ) ;
assertThat ( triggerDetails ) . doesNotContainKeys ( "cron" , "simple" , "calendarInterval" , "custom" ) ;
assertThat ( triggerDetails ) . extractingByKey ( "dailyTimeInterval" , nestedMap ( ) ) . containsOnly (
entry ( "interval" , 3600000L ) , entry ( "startTimeOfDay" , LocalTime . of ( 9 , 0 ) ) ,
entry ( "endTimeOfDay" , LocalTime . of ( 18 , 0 ) ) ,
entry ( "daysOfWeek" , new LinkedHashSet < > ( Arrays . asList ( 2 , 4 ) ) ) , entry ( "repeatCount" , - 1 ) ,
entry ( "timesTriggered" , 0 ) ) ;
}
@Test
void quartzTriggerWithCalendarTimeIntervalTrigger ( ) throws SchedulerException {
TimeZone timeZone = TimeZone . getTimeZone ( "Europe/Paris" ) ;
Date previousFireTime = Date . from ( Instant . parse ( "2020-11-30T03:00:00Z" ) ) ;
Date nextFireTime = Date . from ( Instant . parse ( "2020-12-01T03:00:00Z" ) ) ;
CalendarIntervalTrigger trigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "once-a-week" , "samples" )
. withDescription ( "Once a week" ) . withPriority ( 8 )
. withSchedule ( CalendarIntervalScheduleBuilder . calendarIntervalSchedule ( ) . withIntervalInWeeks ( 1 )
. inTimeZone ( timeZone ) . preserveHourOfDayAcrossDaylightSavings ( true ) )
. build ( ) ;
( ( OperableTrigger ) trigger ) . setPreviousFireTime ( previousFireTime ) ;
( ( OperableTrigger ) trigger ) . setNextFireTime ( nextFireTime ) ;
mockTriggers ( trigger ) ;
given ( this . scheduler . getTriggerState ( TriggerKey . triggerKey ( "once-a-week" , "samples" ) ) )
. willReturn ( TriggerState . BLOCKED ) ;
Map < String , Object > triggerDetails = this . endpoint . quartzTrigger ( "samples" , "once-a-week" ) ;
assertThat ( triggerDetails ) . contains ( entry ( "group" , "samples" ) , entry ( "name" , "once-a-week" ) ,
entry ( "description" , "Once a week" ) , entry ( "type" , "calendarInterval" ) ,
entry ( "state" , TriggerState . BLOCKED ) , entry ( "priority" , 8 ) ) ;
assertThat ( triggerDetails ) . contains ( entry ( "previousFireTime" , previousFireTime ) ,
entry ( "nextFireTime" , nextFireTime ) ) ;
assertThat ( triggerDetails ) . doesNotContainKeys ( "cron" , "simple" , "dailyTimeInterval" , "custom" ) ;
assertThat ( triggerDetails ) . extractingByKey ( "calendarInterval" , nestedMap ( ) ) . containsOnly (
entry ( "interval" , 604800000L ) , entry ( "timeZone" , timeZone ) ,
entry ( "preserveHourOfDayAcrossDaylightSavings" , true ) , entry ( "skipDayIfHourDoesNotExist" , false ) ,
entry ( "timesTriggered" , 0 ) ) ;
}
@Test
void quartzTriggerWithCustomTrigger ( ) throws SchedulerException {
Date previousFireTime = Date . from ( Instant . parse ( "2020-11-30T03:00:00Z" ) ) ;
Date nextFireTime = Date . from ( Instant . parse ( "2020-12-01T03:00:00Z" ) ) ;
Trigger trigger = mock ( Trigger . class ) ;
given ( trigger . getKey ( ) ) . willReturn ( TriggerKey . triggerKey ( "custom" , "samples" ) ) ;
given ( trigger . getPreviousFireTime ( ) ) . willReturn ( previousFireTime ) ;
given ( trigger . getNextFireTime ( ) ) . willReturn ( nextFireTime ) ;
given ( trigger . getPriority ( ) ) . willReturn ( 9 ) ;
mockTriggers ( trigger ) ;
given ( this . scheduler . getTriggerState ( TriggerKey . triggerKey ( "custom" , "samples" ) ) )
. willReturn ( TriggerState . ERROR ) ;
Map < String , Object > triggerDetails = this . endpoint . quartzTrigger ( "samples" , "custom" ) ;
assertThat ( triggerDetails ) . contains ( entry ( "group" , "samples" ) , entry ( "name" , "custom" ) , entry ( "type" , "custom" ) ,
entry ( "state" , TriggerState . ERROR ) , entry ( "priority" , 9 ) ) ;
assertThat ( triggerDetails ) . contains ( entry ( "previousFireTime" , previousFireTime ) ,
entry ( "nextFireTime" , nextFireTime ) ) ;
assertThat ( triggerDetails ) . doesNotContainKeys ( "cron" , "simple" , "calendarInterval" , "dailyTimeInterval" ) ;
assertThat ( triggerDetails ) . extractingByKey ( "custom" , nestedMap ( ) )
. containsOnly ( entry ( "trigger" , trigger . toString ( ) ) ) ;
}
@Test
void quartzTriggerWithDataMap ( ) throws SchedulerException {
CronTrigger trigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "3am-every-day" , "samples" )
. withSchedule ( CronScheduleBuilder . dailyAtHourAndMinute ( 3 , 0 ) ) . usingJobData ( "user" , "user" )
. usingJobData ( "password" , "secret" ) . usingJobData ( "url" , "https://user:secret@example.com" ) . build ( ) ;
mockTriggers ( trigger ) ;
given ( this . scheduler . getTriggerState ( TriggerKey . triggerKey ( "3am-every-day" , "samples" ) ) )
. willReturn ( TriggerState . NORMAL ) ;
Map < String , Object > triggerDetails = this . endpoint . quartzTrigger ( "samples" , "3am-every-day" ) ;
assertThat ( triggerDetails ) . extractingByKey ( "data" , nestedMap ( ) ) . containsOnly ( entry ( "user" , "user" ) ,
entry ( "password" , "******" ) , entry ( "url" , "https://user:******@example.com" ) ) ;
}
@ParameterizedTest ( name = "unit {1}" )
@MethodSource ( "intervalUnitParameters" )
void canConvertIntervalUnit ( int amount , IntervalUnit unit , Duration expectedDuration ) throws SchedulerException {
CalendarIntervalTrigger trigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "trigger" , "samples" )
. withSchedule ( CalendarIntervalScheduleBuilder . calendarIntervalSchedule ( ) . withInterval ( amount , unit ) )
. build ( ) ;
mockTriggers ( trigger ) ;
Map < String , Object > triggerDetails = this . endpoint . quartzTrigger ( "samples" , "trigger" ) ;
assertThat ( triggerDetails ) . extractingByKey ( "calendarInterval" , nestedMap ( ) )
. contains ( entry ( "interval" , expectedDuration . toMillis ( ) ) ) ;
}
static Stream < Arguments > intervalUnitParameters ( ) {
return Stream . of ( Arguments . of ( 3 , IntervalUnit . DAY , Duration . ofDays ( 3 ) ) ,
Arguments . of ( 2 , IntervalUnit . HOUR , Duration . ofHours ( 2 ) ) ,
Arguments . of ( 5 , IntervalUnit . MINUTE , Duration . ofMinutes ( 5 ) ) ,
Arguments . of ( 1 , IntervalUnit . MONTH , ChronoUnit . MONTHS . getDuration ( ) ) ,
Arguments . of ( 30 , IntervalUnit . SECOND , Duration . ofSeconds ( 30 ) ) ,
Arguments . of ( 100 , IntervalUnit . MILLISECOND , Duration . ofMillis ( 100 ) ) ,
Arguments . of ( 1 , IntervalUnit . WEEK , ChronoUnit . WEEKS . getDuration ( ) ) ,
Arguments . of ( 1 , IntervalUnit . YEAR , ChronoUnit . YEARS . getDuration ( ) ) ) ;
}
@Test
void quartzJobWithoutTrigger ( ) throws SchedulerException {
JobDetail job = JobBuilder . newJob ( Job . class ) . withIdentity ( "hello" , "samples" ) . withDescription ( "A sample job" )
. storeDurably ( ) . requestRecovery ( false ) . build ( ) ;
mockJobs ( job ) ;
QuartzJobDetails jobDetails = this . endpoint . quartzJob ( "samples" , "hello" ) ;
assertThat ( jobDetails . getGroup ( ) ) . isEqualTo ( "samples" ) ;
assertThat ( jobDetails . getName ( ) ) . isEqualTo ( "hello" ) ;
assertThat ( jobDetails . getDescription ( ) ) . isEqualTo ( "A sample job" ) ;
assertThat ( jobDetails . getClassName ( ) ) . isEqualTo ( Job . class . getName ( ) ) ;
assertThat ( jobDetails . isDurable ( ) ) . isTrue ( ) ;
assertThat ( jobDetails . isRequestRecovery ( ) ) . isFalse ( ) ;
assertThat ( jobDetails . getData ( ) ) . isEmpty ( ) ;
assertThat ( jobDetails . getTriggers ( ) ) . isEmpty ( ) ;
}
@Test
void quartzJobWithTrigger ( ) throws SchedulerException {
Date previousFireTime = Date . from ( Instant . parse ( "2020-11-30T03:00:00Z" ) ) ;
Date nextFireTime = Date . from ( Instant . parse ( "2020-12-01T03:00:00Z" ) ) ;
JobDetail job = JobBuilder . newJob ( Job . class ) . withIdentity ( "hello" , "samples" ) . build ( ) ;
TimeZone timeZone = TimeZone . getTimeZone ( "Europe/Paris" ) ;
Trigger trigger = TriggerBuilder . newTrigger ( ) . withIdentity ( "3am-every-day" , "samples" ) . withPriority ( 4 )
. withSchedule ( CronScheduleBuilder . dailyAtHourAndMinute ( 3 , 0 ) . inTimeZone ( timeZone ) ) . build ( ) ;
( ( OperableTrigger ) trigger ) . setPreviousFireTime ( previousFireTime ) ;
( ( OperableTrigger ) trigger ) . setNextFireTime ( nextFireTime ) ;
mockJobs ( job ) ;
mockTriggers ( trigger ) ;
given ( this . scheduler . getTriggersOfJob ( JobKey . jobKey ( "hello" , "samples" ) ) )
. willAnswer ( ( invocation ) - > Collections . singletonList ( trigger ) ) ;
QuartzJobDetails jobDetails = this . endpoint . quartzJob ( "samples" , "hello" ) ;
assertThat ( jobDetails . getTriggers ( ) ) . hasSize ( 1 ) ;
Map < String , Object > triggerDetails = jobDetails . getTriggers ( ) . get ( 0 ) ;
assertThat ( triggerDetails ) . containsOnly ( entry ( "group" , "samples" ) , entry ( "name" , "3am-every-day" ) ,
entry ( "previousFireTime" , previousFireTime ) , entry ( "nextFireTime" , nextFireTime ) , entry ( "priority" , 4 ) ) ;
}
@Test
void quartzJobOrdersTriggersAccordingToNextFireTime ( ) throws SchedulerException {
JobDetail job = JobBuilder . newJob ( Job . class ) . withIdentity ( "hello" , "samples" ) . build ( ) ;
mockJobs ( job ) ;
Date triggerOneNextFireTime = Date . from ( Instant . parse ( "2020-12-01T03:00:00Z" ) ) ;
CronTrigger triggerOne = TriggerBuilder . newTrigger ( ) . withIdentity ( "one" , "samples" ) . withPriority ( 5 )
. withSchedule ( CronScheduleBuilder . dailyAtHourAndMinute ( 3 , 0 ) ) . build ( ) ;
( ( OperableTrigger ) triggerOne ) . setNextFireTime ( triggerOneNextFireTime ) ;
Date triggerTwoNextFireTime = Date . from ( Instant . parse ( "2020-12-01T02:00:00Z" ) ) ;
CronTrigger triggerTwo = TriggerBuilder . newTrigger ( ) . withIdentity ( "two" , "samples" ) . withPriority ( 10 )
. withSchedule ( CronScheduleBuilder . dailyAtHourAndMinute ( 2 , 0 ) ) . build ( ) ;
( ( OperableTrigger ) triggerTwo ) . setNextFireTime ( triggerTwoNextFireTime ) ;
mockTriggers ( triggerOne , triggerTwo ) ;
given ( this . scheduler . getTriggersOfJob ( JobKey . jobKey ( "hello" , "samples" ) ) )
. willAnswer ( ( invocation ) - > Arrays . asList ( triggerOne , triggerTwo ) ) ;
QuartzJobDetails jobDetails = this . endpoint . quartzJob ( "samples" , "hello" ) ;
assertThat ( jobDetails . getTriggers ( ) ) . hasSize ( 2 ) ;
assertThat ( jobDetails . getTriggers ( ) . get ( 0 ) ) . containsEntry ( "name" , "two" ) ;
assertThat ( jobDetails . getTriggers ( ) . get ( 1 ) ) . containsEntry ( "name" , "one" ) ;
}
@Test
void quartzJobOrdersTriggersAccordingNextFireTimeAndPriority ( ) throws SchedulerException {
JobDetail job = JobBuilder . newJob ( Job . class ) . withIdentity ( "hello" , "samples" ) . build ( ) ;
mockJobs ( job ) ;
Date nextFireTime = Date . from ( Instant . parse ( "2020-12-01T03:00:00Z" ) ) ;
CronTrigger triggerOne = TriggerBuilder . newTrigger ( ) . withIdentity ( "one" , "samples" ) . withPriority ( 3 )
. withSchedule ( CronScheduleBuilder . dailyAtHourAndMinute ( 3 , 0 ) ) . build ( ) ;
( ( OperableTrigger ) triggerOne ) . setNextFireTime ( nextFireTime ) ;
CronTrigger triggerTwo = TriggerBuilder . newTrigger ( ) . withIdentity ( "two" , "samples" ) . withPriority ( 7 )
. withSchedule ( CronScheduleBuilder . dailyAtHourAndMinute ( 3 , 0 ) ) . build ( ) ;
( ( OperableTrigger ) triggerTwo ) . setNextFireTime ( nextFireTime ) ;
mockTriggers ( triggerOne , triggerTwo ) ;
given ( this . scheduler . getTriggersOfJob ( JobKey . jobKey ( "hello" , "samples" ) ) )
. willAnswer ( ( invocation ) - > Arrays . asList ( triggerOne , triggerTwo ) ) ;
QuartzJobDetails jobDetails = this . endpoint . quartzJob ( "samples" , "hello" ) ;
assertThat ( jobDetails . getTriggers ( ) ) . hasSize ( 2 ) ;
assertThat ( jobDetails . getTriggers ( ) . get ( 0 ) ) . containsEntry ( "name" , "two" ) ;
assertThat ( jobDetails . getTriggers ( ) . get ( 1 ) ) . containsEntry ( "name" , "one" ) ;
}
@Test
void quartzJobWithSensitiveDataMap ( ) throws SchedulerException {
JobDetail job = JobBuilder . newJob ( Job . class ) . withIdentity ( "hello" , "samples" ) . usingJobData ( "user" , "user" )
. usingJobData ( "password" , "secret" ) . usingJobData ( "url" , "https://user:secret@example.com" ) . build ( ) ;
mockJobs ( job ) ;
QuartzJobDetails jobDetails = this . endpoint . quartzJob ( "samples" , "hello" ) ;
assertThat ( jobDetails . getData ( ) ) . containsOnly ( entry ( "user" , "user" ) , entry ( "password" , "******" ) ,
entry ( "url" , "https://user:******@example.com" ) ) ;
}
@Test
void quartzJobWithSensitiveDataMapAndCustomSanitizier ( ) throws SchedulerException {
JobDetail job = JobBuilder . newJob ( Job . class ) . withIdentity ( "hello" , "samples" ) . usingJobData ( "test" , "value" )
. usingJobData ( "secret" , "value" ) . build ( ) ;
mockJobs ( job ) ;
Sanitizer sanitizer = mock ( Sanitizer . class ) ;
given ( sanitizer . sanitize ( "test" , "value" ) ) . willReturn ( "value" ) ;
given ( sanitizer . sanitize ( "secret" , "value" ) ) . willReturn ( "----" ) ;
QuartzJobDetails jobDetails = new QuartzEndpoint ( this . scheduler , sanitizer ) . quartzJob ( "samples" , "hello" ) ;
assertThat ( jobDetails . getData ( ) ) . containsOnly ( entry ( "test" , "value" ) , entry ( "secret" , "----" ) ) ;
verify ( sanitizer ) . sanitize ( "test" , "value" ) ;
verify ( sanitizer ) . sanitize ( "secret" , "value" ) ;
verifyNoMoreInteractions ( sanitizer ) ;
}
private void mockJobs ( JobDetail . . . jobs ) throws SchedulerException {
MultiValueMap < String , JobKey > jobKeys = new LinkedMultiValueMap < > ( ) ;
for ( JobDetail jobDetail : jobs ) {
JobKey key = jobDetail . getKey ( ) ;
given ( this . scheduler . getJobDetail ( key ) ) . willReturn ( jobDetail ) ;
jobKeys . add ( key . getGroup ( ) , key ) ;
}
given ( this . scheduler . getJobGroupNames ( ) ) . willReturn ( new ArrayList < > ( jobKeys . keySet ( ) ) ) ;
for ( Entry < String , List < JobKey > > entry : jobKeys . entrySet ( ) ) {
given ( this . scheduler . getJobKeys ( GroupMatcher . jobGroupEquals ( entry . getKey ( ) ) ) )
. willReturn ( new LinkedHashSet < > ( entry . getValue ( ) ) ) ;
}
}
private void mockTriggers ( Trigger . . . triggers ) throws SchedulerException {
MultiValueMap < String , TriggerKey > triggerKeys = new LinkedMultiValueMap < > ( ) ;
for ( Trigger trigger : triggers ) {
TriggerKey key = trigger . getKey ( ) ;
given ( this . scheduler . getTrigger ( key ) ) . willReturn ( trigger ) ;
triggerKeys . add ( key . getGroup ( ) , key ) ;
}
given ( this . scheduler . getTriggerGroupNames ( ) ) . willReturn ( new ArrayList < > ( triggerKeys . keySet ( ) ) ) ;
for ( Entry < String , List < TriggerKey > > entry : triggerKeys . entrySet ( ) ) {
given ( this . scheduler . getTriggerKeys ( GroupMatcher . triggerGroupEquals ( entry . getKey ( ) ) ) )
. willReturn ( new LinkedHashSet < > ( entry . getValue ( ) ) ) ;
}
}
@SuppressWarnings ( "rawtypes" )
private static InstanceOfAssertFactory < Map , MapAssert < String , Object > > nestedMap ( ) {
return InstanceOfAssertFactories . map ( String . class , Object . class ) ;
}
}