spec_test.go 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  1. package cron
  2. import (
  3. "testing"
  4. "time"
  5. )
  6. func TestActivation(t *testing.T) {
  7. tests := []struct {
  8. time, spec string
  9. expected bool
  10. }{
  11. // Every fifteen minutes.
  12. {"Mon Jul 9 15:00 2012", "0/15 * * * *", true},
  13. {"Mon Jul 9 15:45 2012", "0/15 * * * *", true},
  14. {"Mon Jul 9 15:40 2012", "0/15 * * * *", false},
  15. // Every fifteen minutes, starting at 5 minutes.
  16. {"Mon Jul 9 15:05 2012", "5/15 * * * *", true},
  17. {"Mon Jul 9 15:20 2012", "5/15 * * * *", true},
  18. {"Mon Jul 9 15:50 2012", "5/15 * * * *", true},
  19. // Named months
  20. {"Sun Jul 15 15:00 2012", "0/15 * * Jul *", true},
  21. {"Sun Jul 15 15:00 2012", "0/15 * * Jun *", false},
  22. // Everything set.
  23. {"Sun Jul 15 08:30 2012", "30 08 ? Jul Sun", true},
  24. {"Sun Jul 15 08:30 2012", "30 08 15 Jul ?", true},
  25. {"Mon Jul 16 08:30 2012", "30 08 ? Jul Sun", false},
  26. {"Mon Jul 16 08:30 2012", "30 08 15 Jul ?", false},
  27. // Predefined schedules
  28. {"Mon Jul 9 15:00 2012", "@hourly", true},
  29. {"Mon Jul 9 15:04 2012", "@hourly", false},
  30. {"Mon Jul 9 15:00 2012", "@daily", false},
  31. {"Mon Jul 9 00:00 2012", "@daily", true},
  32. {"Mon Jul 9 00:00 2012", "@weekly", false},
  33. {"Sun Jul 8 00:00 2012", "@weekly", true},
  34. {"Sun Jul 8 01:00 2012", "@weekly", false},
  35. {"Sun Jul 8 00:00 2012", "@monthly", false},
  36. {"Sun Jul 1 00:00 2012", "@monthly", true},
  37. // Test interaction of DOW and DOM.
  38. // If both are specified, then only one needs to match.
  39. {"Sun Jul 15 00:00 2012", "* * 1,15 * Sun", true},
  40. {"Fri Jun 15 00:00 2012", "* * 1,15 * Sun", true},
  41. {"Wed Aug 1 00:00 2012", "* * 1,15 * Sun", true},
  42. // However, if one has a star, then both need to match.
  43. {"Sun Jul 15 00:00 2012", "* * * * Mon", false},
  44. {"Sun Jul 15 00:00 2012", "* * */10 * Sun", false},
  45. {"Mon Jul 9 00:00 2012", "* * 1,15 * *", false},
  46. {"Sun Jul 15 00:00 2012", "* * 1,15 * *", true},
  47. {"Sun Jul 15 00:00 2012", "* * */2 * Sun", true},
  48. }
  49. for _, test := range tests {
  50. sched, err := ParseStandard(test.spec)
  51. if err != nil {
  52. t.Error(err)
  53. continue
  54. }
  55. actual := sched.Next(getTime(test.time).Add(-1 * time.Second))
  56. expected := getTime(test.time)
  57. if test.expected && expected != actual || !test.expected && expected == actual {
  58. t.Errorf("Fail evaluating %s on %s: (expected) %s != %s (actual)",
  59. test.spec, test.time, expected, actual)
  60. }
  61. }
  62. }
  63. func TestNext(t *testing.T) {
  64. runs := []struct {
  65. time, spec string
  66. expected string
  67. }{
  68. // Simple cases
  69. {"Mon Jul 9 14:45 2012", "0 0/15 * * * *", "Mon Jul 9 15:00 2012"},
  70. {"Mon Jul 9 14:59 2012", "0 0/15 * * * *", "Mon Jul 9 15:00 2012"},
  71. {"Mon Jul 9 14:59:59 2012", "0 0/15 * * * *", "Mon Jul 9 15:00 2012"},
  72. // Wrap around hours
  73. {"Mon Jul 9 15:45 2012", "0 20-35/15 * * * *", "Mon Jul 9 16:20 2012"},
  74. // Wrap around days
  75. {"Mon Jul 9 23:46 2012", "0 */15 * * * *", "Tue Jul 10 00:00 2012"},
  76. {"Mon Jul 9 23:45 2012", "0 20-35/15 * * * *", "Tue Jul 10 00:20 2012"},
  77. {"Mon Jul 9 23:35:51 2012", "15/35 20-35/15 * * * *", "Tue Jul 10 00:20:15 2012"},
  78. {"Mon Jul 9 23:35:51 2012", "15/35 20-35/15 1/2 * * *", "Tue Jul 10 01:20:15 2012"},
  79. {"Mon Jul 9 23:35:51 2012", "15/35 20-35/15 10-12 * * *", "Tue Jul 10 10:20:15 2012"},
  80. {"Mon Jul 9 23:35:51 2012", "15/35 20-35/15 1/2 */2 * *", "Thu Jul 11 01:20:15 2012"},
  81. {"Mon Jul 9 23:35:51 2012", "15/35 20-35/15 * 9-20 * *", "Wed Jul 10 00:20:15 2012"},
  82. {"Mon Jul 9 23:35:51 2012", "15/35 20-35/15 * 9-20 Jul *", "Wed Jul 10 00:20:15 2012"},
  83. // Wrap around months
  84. {"Mon Jul 9 23:35 2012", "0 0 0 9 Apr-Oct ?", "Thu Aug 9 00:00 2012"},
  85. {"Mon Jul 9 23:35 2012", "0 0 0 */5 Apr,Aug,Oct Mon", "Mon Aug 6 00:00 2012"},
  86. {"Mon Jul 9 23:35 2012", "0 0 0 */5 Oct Mon", "Mon Oct 1 00:00 2012"},
  87. // Wrap around years
  88. {"Mon Jul 9 23:35 2012", "0 0 0 * Feb Mon", "Mon Feb 4 00:00 2013"},
  89. {"Mon Jul 9 23:35 2012", "0 0 0 * Feb Mon/2", "Fri Feb 1 00:00 2013"},
  90. // Wrap around minute, hour, day, month, and year
  91. {"Mon Dec 31 23:59:45 2012", "0 * * * * *", "Tue Jan 1 00:00:00 2013"},
  92. // Leap year
  93. {"Mon Jul 9 23:35 2012", "0 0 0 29 Feb ?", "Mon Feb 29 00:00 2016"},
  94. // Daylight savings time 2am EST (-5) -> 3am EDT (-4)
  95. {"2012-03-11T00:00:00-0500", "TZ=America/New_York 0 30 2 11 Mar ?", "2013-03-11T02:30:00-0400"},
  96. // hourly job
  97. {"2012-03-11T00:00:00-0500", "TZ=America/New_York 0 0 * * * ?", "2012-03-11T01:00:00-0500"},
  98. {"2012-03-11T01:00:00-0500", "TZ=America/New_York 0 0 * * * ?", "2012-03-11T03:00:00-0400"},
  99. {"2012-03-11T03:00:00-0400", "TZ=America/New_York 0 0 * * * ?", "2012-03-11T04:00:00-0400"},
  100. {"2012-03-11T04:00:00-0400", "TZ=America/New_York 0 0 * * * ?", "2012-03-11T05:00:00-0400"},
  101. // 1am nightly job
  102. {"2012-03-11T00:00:00-0500", "TZ=America/New_York 0 0 1 * * ?", "2012-03-11T01:00:00-0500"},
  103. {"2012-03-11T01:00:00-0500", "TZ=America/New_York 0 0 1 * * ?", "2012-03-12T01:00:00-0400"},
  104. // 2am nightly job (skipped)
  105. {"2012-03-11T00:00:00-0500", "TZ=America/New_York 0 0 2 * * ?", "2012-03-12T02:00:00-0400"},
  106. // Daylight savings time 2am EDT (-4) => 1am EST (-5)
  107. {"2012-11-04T00:00:00-0400", "TZ=America/New_York 0 30 2 04 Nov ?", "2012-11-04T02:30:00-0500"},
  108. {"2012-11-04T01:45:00-0400", "TZ=America/New_York 0 30 1 04 Nov ?", "2012-11-04T01:30:00-0500"},
  109. // hourly job
  110. {"2012-11-04T00:00:00-0400", "TZ=America/New_York 0 0 * * * ?", "2012-11-04T01:00:00-0400"},
  111. {"2012-11-04T01:00:00-0400", "TZ=America/New_York 0 0 * * * ?", "2012-11-04T01:00:00-0500"},
  112. {"2012-11-04T01:00:00-0500", "TZ=America/New_York 0 0 * * * ?", "2012-11-04T02:00:00-0500"},
  113. // 1am nightly job (runs twice)
  114. {"2012-11-04T00:00:00-0400", "TZ=America/New_York 0 0 1 * * ?", "2012-11-04T01:00:00-0400"},
  115. {"2012-11-04T01:00:00-0400", "TZ=America/New_York 0 0 1 * * ?", "2012-11-04T01:00:00-0500"},
  116. {"2012-11-04T01:00:00-0500", "TZ=America/New_York 0 0 1 * * ?", "2012-11-05T01:00:00-0500"},
  117. // 2am nightly job
  118. {"2012-11-04T00:00:00-0400", "TZ=America/New_York 0 0 2 * * ?", "2012-11-04T02:00:00-0500"},
  119. {"2012-11-04T02:00:00-0500", "TZ=America/New_York 0 0 2 * * ?", "2012-11-05T02:00:00-0500"},
  120. // 3am nightly job
  121. {"2012-11-04T00:00:00-0400", "TZ=America/New_York 0 0 3 * * ?", "2012-11-04T03:00:00-0500"},
  122. {"2012-11-04T03:00:00-0500", "TZ=America/New_York 0 0 3 * * ?", "2012-11-05T03:00:00-0500"},
  123. // hourly job
  124. {"2012-11-04T00:00:00-0400", "0 0 * * * ?", "2012-11-04T01:00:00-0400"},
  125. {"2012-11-04T01:00:00-0400", "0 0 * * * ?", "2012-11-04T01:00:00-0500"},
  126. {"2012-11-04T01:00:00-0500", "0 0 * * * ?", "2012-11-04T02:00:00-0500"},
  127. // 1am nightly job (runs twice)
  128. {"2012-11-04T00:00:00-0400", "0 0 1 * * ?", "2012-11-04T01:00:00-0400"},
  129. {"2012-11-04T01:00:00-0400", "0 0 1 * * ?", "2012-11-04T01:00:00-0500"},
  130. {"2012-11-04T01:00:00-0500", "0 0 1 * * ?", "2012-11-05T01:00:00-0500"},
  131. // 2am nightly job
  132. {"2012-11-04T00:00:00-0400", "0 0 2 * * ?", "2012-11-04T02:00:00-0500"},
  133. {"2012-11-04T02:00:00-0500", "0 0 2 * * ?", "2012-11-05T02:00:00-0500"},
  134. // 3am nightly job
  135. {"2012-11-04T00:00:00-0400", "0 0 3 * * ?", "2012-11-04T03:00:00-0500"},
  136. {"2012-11-04T03:00:00-0500", "0 0 3 * * ?", "2012-11-05T03:00:00-0500"},
  137. // Unsatisfiable
  138. {"Mon Jul 9 23:35 2012", "0 0 0 30 Feb ?", ""},
  139. {"Mon Jul 9 23:35 2012", "0 0 0 31 Apr ?", ""},
  140. }
  141. for _, c := range runs {
  142. sched, err := secondParser.Parse(c.spec)
  143. if err != nil {
  144. t.Error(err)
  145. continue
  146. }
  147. actual := sched.Next(getTime(c.time))
  148. expected := getTime(c.expected)
  149. if !actual.Equal(expected) {
  150. t.Errorf("%s, \"%s\": (expected) %v != %v (actual)", c.time, c.spec, expected, actual)
  151. }
  152. }
  153. }
  154. func TestErrors(t *testing.T) {
  155. invalidSpecs := []string{
  156. "xyz",
  157. "60 0 * * *",
  158. "0 60 * * *",
  159. "0 0 * * XYZ",
  160. }
  161. for _, spec := range invalidSpecs {
  162. _, err := ParseStandard(spec)
  163. if err == nil {
  164. t.Error("expected an error parsing: ", spec)
  165. }
  166. }
  167. }
  168. func getTime(value string) time.Time {
  169. if value == "" {
  170. return time.Time{}
  171. }
  172. var layouts = []string{
  173. "Mon Jan 2 15:04 2006",
  174. "Mon Jan 2 15:04:05 2006",
  175. }
  176. for _, layout := range layouts {
  177. if t, err := time.ParseInLocation(layout, value, time.Local); err == nil {
  178. return t
  179. }
  180. }
  181. if t, err := time.Parse("2006-01-02T15:04:05-0700", value); err == nil {
  182. return t
  183. }
  184. panic("could not parse time value " + value)
  185. }
  186. func TestNextWithTz(t *testing.T) {
  187. runs := []struct {
  188. time, spec string
  189. expected string
  190. }{
  191. // Failing tests
  192. {"2016-01-03T13:09:03+0530", "14 14 * * *", "2016-01-03T14:14:00+0530"},
  193. {"2016-01-03T04:09:03+0530", "14 14 * * ?", "2016-01-03T14:14:00+0530"},
  194. // Passing tests
  195. {"2016-01-03T14:09:03+0530", "14 14 * * *", "2016-01-03T14:14:00+0530"},
  196. {"2016-01-03T14:00:00+0530", "14 14 * * ?", "2016-01-03T14:14:00+0530"},
  197. }
  198. for _, c := range runs {
  199. sched, err := ParseStandard(c.spec)
  200. if err != nil {
  201. t.Error(err)
  202. continue
  203. }
  204. actual := sched.Next(getTimeTZ(c.time))
  205. expected := getTimeTZ(c.expected)
  206. if !actual.Equal(expected) {
  207. t.Errorf("%s, \"%s\": (expected) %v != %v (actual)", c.time, c.spec, expected, actual)
  208. }
  209. }
  210. }
  211. func getTimeTZ(value string) time.Time {
  212. if value == "" {
  213. return time.Time{}
  214. }
  215. t, err := time.Parse("Mon Jan 2 15:04 2006", value)
  216. if err != nil {
  217. t, err = time.Parse("Mon Jan 2 15:04:05 2006", value)
  218. if err != nil {
  219. t, err = time.Parse("2006-01-02T15:04:05-0700", value)
  220. if err != nil {
  221. panic(err)
  222. }
  223. }
  224. }
  225. return t
  226. }