velocious 1.0.317 → 1.0.319

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1630,6 +1630,37 @@ Supported schedule syntax:
1630
1630
  - `every: ["1h", {first_in: "30s"}]`
1631
1631
  - `every: ["1 day", {firstIn: "5 minutes"}]`
1632
1632
 
1633
+ Or a 5-field POSIX crontab expression via `cron`:
1634
+
1635
+ ```js
1636
+ scheduledBackgroundJobs: {
1637
+ jobs: {
1638
+ nightlyDigest: {
1639
+ class: NightlyDigestJob,
1640
+ cron: "0 3 * * *" // every day at 03:00 server-local time
1641
+ },
1642
+ weekdayMornings: {
1643
+ class: WeekdayMorningJob,
1644
+ cron: "0 9 * * 1-5" // 09:00 Mon–Fri
1645
+ },
1646
+ everyHour: {
1647
+ class: HourlyCleanupJob,
1648
+ cron: "@hourly"
1649
+ }
1650
+ }
1651
+ }
1652
+ ```
1653
+
1654
+ Cron fields are: `minute hour day-of-month month day-of-week`. Supported syntax:
1655
+
1656
+ - `*` (any), single values (`5`), ranges (`1-5`), lists (`1,3,5`).
1657
+ - Step expressions: `*/15` (every 15 minutes), `0-30/5` (every 5 between 0 and 30).
1658
+ - Month and weekday names: `jan`-`dec`, `sun`-`sat` (case-insensitive). Both `0` and `7` mean Sunday.
1659
+ - POSIX shortcuts: `@hourly`, `@daily` / `@midnight`, `@weekly`, `@monthly`, `@yearly` / `@annually`.
1660
+ - Day-of-month and day-of-week interaction follows POSIX/Vixie cron: when both are restricted (neither `*`), the job fires when **either** matches.
1661
+
1662
+ Each job must define exactly one of `every` or `cron`. Cron times are evaluated in the **server's local timezone**, at minute granularity.
1663
+
1633
1664
  `background-jobs-main` owns the schedule and enqueues the configured jobs into the normal Velocious background-jobs queue. The HTTP server does not run scheduled jobs itself.
1634
1665
 
1635
1666
  ## Persistence and retries
@@ -0,0 +1,62 @@
1
+ /**
2
+ * @typedef {object} ParsedCron
3
+ * @property {Set<number>} minute - Allowed minute values (0-59).
4
+ * @property {Set<number>} hour - Allowed hour values (0-23).
5
+ * @property {Set<number>} dayOfMonth - Allowed day-of-month values (1-31).
6
+ * @property {Set<number>} month - Allowed month values (1-12).
7
+ * @property {Set<number>} dayOfWeek - Allowed day-of-week values (0-6, 0=Sun).
8
+ * @property {boolean} dayOfMonthRestricted - True when the dayOfMonth field is not `*`.
9
+ * @property {boolean} dayOfWeekRestricted - True when the dayOfWeek field is not `*`.
10
+ * @property {string} expression - Original expression for diagnostics.
11
+ */
12
+ /**
13
+ * @param {string} expression - Cron expression or shortcut.
14
+ * @returns {ParsedCron}
15
+ */
16
+ export function parseCronExpression(expression: string): ParsedCron;
17
+ /**
18
+ * Returns the next Date strictly after `from` that satisfies `parsed`.
19
+ * Operates at minute granularity. Bails out with an error after five
20
+ * years of search, which only happens if the expression matches no
21
+ * real time (e.g., `0 0 31 2 *` — Feb 31st).
22
+ *
23
+ * @param {ParsedCron} parsed - Parsed cron expression.
24
+ * @param {Date} from - Reference Date — the next match is strictly after this.
25
+ * @returns {Date}
26
+ */
27
+ export function nextCronFireDate(parsed: ParsedCron, from: Date): Date;
28
+ export type ParsedCron = {
29
+ /**
30
+ * - Allowed minute values (0-59).
31
+ */
32
+ minute: Set<number>;
33
+ /**
34
+ * - Allowed hour values (0-23).
35
+ */
36
+ hour: Set<number>;
37
+ /**
38
+ * - Allowed day-of-month values (1-31).
39
+ */
40
+ dayOfMonth: Set<number>;
41
+ /**
42
+ * - Allowed month values (1-12).
43
+ */
44
+ month: Set<number>;
45
+ /**
46
+ * - Allowed day-of-week values (0-6, 0=Sun).
47
+ */
48
+ dayOfWeek: Set<number>;
49
+ /**
50
+ * - True when the dayOfMonth field is not `*`.
51
+ */
52
+ dayOfMonthRestricted: boolean;
53
+ /**
54
+ * - True when the dayOfWeek field is not `*`.
55
+ */
56
+ dayOfWeekRestricted: boolean;
57
+ /**
58
+ * - Original expression for diagnostics.
59
+ */
60
+ expression: string;
61
+ };
62
+ //# sourceMappingURL=cron-expression.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cron-expression.d.ts","sourceRoot":"","sources":["../../../src/background-jobs/cron-expression.js"],"names":[],"mappings":"AAsCA;;;;;;;;;;GAUG;AAEH;;;GAGG;AACH,gDAHW,MAAM,GACJ,UAAU,CA+BtB;AAsID;;;;;;;;;GASG;AACH,yCAJW,UAAU,QACV,IAAI,GACF,IAAI,CAehB;;;;;YAxMa,GAAG,CAAC,MAAM,CAAC;;;;UACX,GAAG,CAAC,MAAM,CAAC;;;;gBACX,GAAG,CAAC,MAAM,CAAC;;;;WACX,GAAG,CAAC,MAAM,CAAC;;;;eACX,GAAG,CAAC,MAAM,CAAC;;;;0BACX,OAAO;;;;yBACP,OAAO;;;;gBACP,MAAM"}
@@ -0,0 +1,228 @@
1
+ // @ts-check
2
+ /**
3
+ * Minimal POSIX-style 5-field cron parser used by the background-job
4
+ * scheduler. Supports `*`, single values, ranges (`N-M`), steps
5
+ * (`*\/N` or `N-M/N`), comma-separated lists, and the common
6
+ * `@hourly`/`@daily`/`@weekly`/`@monthly`/`@yearly`/`@midnight`
7
+ * shortcuts. Month and day-of-week names (`jan`-`dec`, `sun`-`sat`,
8
+ * case-insensitive) are also accepted.
9
+ *
10
+ * For day-of-month + day-of-week interaction, follows POSIX/Vixie
11
+ * cron semantics: when both fields are restricted (neither `*`), the
12
+ * job fires when EITHER matches. When one is `*` it has no effect.
13
+ */
14
+ const MONTH_NAMES = ["jan", "feb", "mar", "apr", "may", "jun", "jul", "aug", "sep", "oct", "nov", "dec"];
15
+ const DAY_NAMES = ["sun", "mon", "tue", "wed", "thu", "fri", "sat"];
16
+ const SHORTCUTS = {
17
+ "@hourly": "0 * * * *",
18
+ "@daily": "0 0 * * *",
19
+ "@midnight": "0 0 * * *",
20
+ "@weekly": "0 0 * * 0",
21
+ "@monthly": "0 0 1 * *",
22
+ "@yearly": "0 0 1 1 *",
23
+ "@annually": "0 0 1 1 *"
24
+ };
25
+ const FIELDS = [
26
+ { name: "minute", min: 0, max: 59 },
27
+ { name: "hour", min: 0, max: 23 },
28
+ { name: "dayOfMonth", min: 1, max: 31 },
29
+ { name: "month", min: 1, max: 12, names: MONTH_NAMES },
30
+ // Accept 0-7 so ranges like `5-7` (Fri-Sun) work; we normalize 7
31
+ // down to 0 after parsing in `normalizeDayOfWeek` below.
32
+ { name: "dayOfWeek", min: 0, max: 7, names: DAY_NAMES }
33
+ ];
34
+ /**
35
+ * @typedef {object} ParsedCron
36
+ * @property {Set<number>} minute - Allowed minute values (0-59).
37
+ * @property {Set<number>} hour - Allowed hour values (0-23).
38
+ * @property {Set<number>} dayOfMonth - Allowed day-of-month values (1-31).
39
+ * @property {Set<number>} month - Allowed month values (1-12).
40
+ * @property {Set<number>} dayOfWeek - Allowed day-of-week values (0-6, 0=Sun).
41
+ * @property {boolean} dayOfMonthRestricted - True when the dayOfMonth field is not `*`.
42
+ * @property {boolean} dayOfWeekRestricted - True when the dayOfWeek field is not `*`.
43
+ * @property {string} expression - Original expression for diagnostics.
44
+ */
45
+ /**
46
+ * @param {string} expression - Cron expression or shortcut.
47
+ * @returns {ParsedCron}
48
+ */
49
+ export function parseCronExpression(expression) {
50
+ if (typeof expression !== "string" || !expression.trim()) {
51
+ throw new Error(`Invalid cron expression: ${expression}`);
52
+ }
53
+ const trimmed = expression.trim().toLowerCase();
54
+ const expanded = SHORTCUTS[ /** @type {keyof typeof SHORTCUTS} */(trimmed)] || trimmed;
55
+ const fields = expanded.split(/\s+/);
56
+ if (fields.length !== 5) {
57
+ throw new Error(`Invalid cron expression "${expression}": expected 5 fields, got ${fields.length}`);
58
+ }
59
+ const [minuteField, hourField, dayOfMonthField, monthField, dayOfWeekField] = fields;
60
+ const parsed = {
61
+ minute: parseField(minuteField, FIELDS[0], expression),
62
+ hour: parseField(hourField, FIELDS[1], expression),
63
+ dayOfMonth: parseField(dayOfMonthField, FIELDS[2], expression),
64
+ month: parseField(monthField, FIELDS[3], expression),
65
+ // Cron treats both 0 and 7 as Sunday. We accept 7 throughout the
66
+ // parse pass (so `5-7` for Fri-Sun works) and then normalize any
67
+ // 7s down to 0 so the matcher only deals with 0-6.
68
+ dayOfWeek: normalizeDayOfWeek(parseField(dayOfWeekField, FIELDS[4], expression)),
69
+ dayOfMonthRestricted: dayOfMonthField !== "*",
70
+ dayOfWeekRestricted: dayOfWeekField !== "*",
71
+ expression
72
+ };
73
+ return parsed;
74
+ }
75
+ /**
76
+ * @param {Set<number>} dayOfWeek
77
+ * @returns {Set<number>}
78
+ */
79
+ function normalizeDayOfWeek(dayOfWeek) {
80
+ if (dayOfWeek.has(7)) {
81
+ dayOfWeek.delete(7);
82
+ dayOfWeek.add(0);
83
+ }
84
+ return dayOfWeek;
85
+ }
86
+ /**
87
+ * @param {string} field - Field expression.
88
+ * @param {{name: string, min: number, max: number, names?: string[]}} fieldSpec - Field spec.
89
+ * @param {string} expression - Whole cron expression for error messages.
90
+ * @returns {Set<number>}
91
+ */
92
+ function parseField(field, fieldSpec, expression) {
93
+ const result = new Set();
94
+ for (const part of field.split(",")) {
95
+ addPartValues(part, fieldSpec, expression, result);
96
+ }
97
+ return result;
98
+ }
99
+ /**
100
+ * @param {string} part - Single comma-separated chunk.
101
+ * @param {{name: string, min: number, max: number, names?: string[]}} fieldSpec - Field spec.
102
+ * @param {string} expression - Original expression for errors.
103
+ * @param {Set<number>} result - Accumulator.
104
+ * @returns {void}
105
+ */
106
+ function addPartValues(part, fieldSpec, expression, result) {
107
+ if (!part) {
108
+ throw new Error(`Invalid ${fieldSpec.name} field in cron expression "${expression}"`);
109
+ }
110
+ const [rangePart, stepPart] = part.split("/");
111
+ const step = stepPart === undefined ? 1 : parseStep(stepPart, fieldSpec, expression);
112
+ const [start, end] = parseRange(rangePart, fieldSpec, expression, stepPart !== undefined);
113
+ for (let value = start; value <= end; value += step) {
114
+ if (value < fieldSpec.min || value > fieldSpec.max) {
115
+ throw new Error(`Value ${value} out of range for ${fieldSpec.name} in cron expression "${expression}"`);
116
+ }
117
+ result.add(value);
118
+ }
119
+ }
120
+ /**
121
+ * @param {string} value - Step value.
122
+ * @param {{name: string, min: number, max: number}} fieldSpec - Field spec.
123
+ * @param {string} expression - Original expression for errors.
124
+ * @returns {number}
125
+ */
126
+ function parseStep(value, fieldSpec, expression) {
127
+ const step = Number(value);
128
+ if (!Number.isInteger(step) || step <= 0) {
129
+ throw new Error(`Invalid step "${value}" for ${fieldSpec.name} in cron expression "${expression}"`);
130
+ }
131
+ return step;
132
+ }
133
+ /**
134
+ * @param {string} rangePart - Range portion (before any `/`).
135
+ * @param {{name: string, min: number, max: number, names?: string[]}} fieldSpec - Field spec.
136
+ * @param {string} expression - Original expression for errors.
137
+ * @param {boolean} hasStep - Whether the part had a `/step` suffix.
138
+ * @returns {[number, number]}
139
+ */
140
+ function parseRange(rangePart, fieldSpec, expression, hasStep) {
141
+ if (rangePart === "*") {
142
+ return [fieldSpec.min, fieldSpec.max];
143
+ }
144
+ const dashIndex = rangePart.indexOf("-");
145
+ if (dashIndex === -1) {
146
+ const value = parseValue(rangePart, fieldSpec, expression);
147
+ // `N/step` is shorthand for `N-max/step` (Vixie cron).
148
+ return [value, hasStep ? fieldSpec.max : value];
149
+ }
150
+ const start = parseValue(rangePart.slice(0, dashIndex), fieldSpec, expression);
151
+ const end = parseValue(rangePart.slice(dashIndex + 1), fieldSpec, expression);
152
+ if (start > end) {
153
+ throw new Error(`Range start ${start} > end ${end} for ${fieldSpec.name} in cron expression "${expression}"`);
154
+ }
155
+ return [start, end];
156
+ }
157
+ /**
158
+ * @param {string} rawValue - Raw value (may be a name).
159
+ * @param {{name: string, min: number, max: number, names?: string[]}} fieldSpec - Field spec.
160
+ * @param {string} expression - Original expression for errors.
161
+ * @returns {number}
162
+ */
163
+ function parseValue(rawValue, fieldSpec, expression) {
164
+ if (!rawValue) {
165
+ throw new Error(`Invalid ${fieldSpec.name} value in cron expression "${expression}"`);
166
+ }
167
+ const namedIndex = fieldSpec.names?.indexOf(rawValue);
168
+ if (typeof namedIndex === "number" && namedIndex !== -1) {
169
+ return namedIndex + fieldSpec.min;
170
+ }
171
+ const value = Number(rawValue);
172
+ if (!Number.isInteger(value)) {
173
+ throw new Error(`Invalid ${fieldSpec.name} value "${rawValue}" in cron expression "${expression}"`);
174
+ }
175
+ return value;
176
+ }
177
+ // 5 years of minutes — covers the worst-case legitimate gap, the
178
+ // `0 0 29 2 *` (Feb 29) leap-year-only schedule, with a one-year
179
+ // buffer so we never report a real cron pattern as "never matches".
180
+ const MAX_NEXT_FIRE_ITERATIONS = 5 * 366 * 24 * 60;
181
+ /**
182
+ * Returns the next Date strictly after `from` that satisfies `parsed`.
183
+ * Operates at minute granularity. Bails out with an error after five
184
+ * years of search, which only happens if the expression matches no
185
+ * real time (e.g., `0 0 31 2 *` — Feb 31st).
186
+ *
187
+ * @param {ParsedCron} parsed - Parsed cron expression.
188
+ * @param {Date} from - Reference Date — the next match is strictly after this.
189
+ * @returns {Date}
190
+ */
191
+ export function nextCronFireDate(parsed, from) {
192
+ const candidate = new Date(from.getTime());
193
+ candidate.setSeconds(0, 0);
194
+ candidate.setMinutes(candidate.getMinutes() + 1);
195
+ for (let iterations = 0; iterations < MAX_NEXT_FIRE_ITERATIONS; iterations += 1) {
196
+ if (candidateMatches(candidate, parsed))
197
+ return candidate;
198
+ candidate.setMinutes(candidate.getMinutes() + 1);
199
+ }
200
+ throw new Error(`Cron expression "${parsed.expression}" never matches`);
201
+ }
202
+ /**
203
+ * @param {Date} candidate - Candidate Date (in local time).
204
+ * @param {ParsedCron} parsed - Parsed expression.
205
+ * @returns {boolean}
206
+ */
207
+ function candidateMatches(candidate, parsed) {
208
+ if (!parsed.minute.has(candidate.getMinutes()))
209
+ return false;
210
+ if (!parsed.hour.has(candidate.getHours()))
211
+ return false;
212
+ if (!parsed.month.has(candidate.getMonth() + 1))
213
+ return false;
214
+ const dayOfMonthMatch = parsed.dayOfMonth.has(candidate.getDate());
215
+ const dayOfWeekMatch = parsed.dayOfWeek.has(candidate.getDay());
216
+ // POSIX/Vixie cron OR semantics: when both day fields are
217
+ // restricted, fire when EITHER matches. When only one is
218
+ // restricted, only that one applies.
219
+ if (parsed.dayOfMonthRestricted && parsed.dayOfWeekRestricted) {
220
+ return dayOfMonthMatch || dayOfWeekMatch;
221
+ }
222
+ if (parsed.dayOfMonthRestricted)
223
+ return dayOfMonthMatch;
224
+ if (parsed.dayOfWeekRestricted)
225
+ return dayOfWeekMatch;
226
+ return true;
227
+ }
228
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cron-expression.js","sourceRoot":"","sources":["../../../src/background-jobs/cron-expression.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\n/**\n * Minimal POSIX-style 5-field cron parser used by the background-job\n * scheduler. Supports `*`, single values, ranges (`N-M`), steps\n * (`*\\/N` or `N-M/N`), comma-separated lists, and the common\n * `@hourly`/`@daily`/`@weekly`/`@monthly`/`@yearly`/`@midnight`\n * shortcuts. Month and day-of-week names (`jan`-`dec`, `sun`-`sat`,\n * case-insensitive) are also accepted.\n *\n * For day-of-month + day-of-week interaction, follows POSIX/Vixie\n * cron semantics: when both fields are restricted (neither `*`), the\n * job fires when EITHER matches. When one is `*` it has no effect.\n */\n\nconst MONTH_NAMES = [\"jan\", \"feb\", \"mar\", \"apr\", \"may\", \"jun\", \"jul\", \"aug\", \"sep\", \"oct\", \"nov\", \"dec\"]\nconst DAY_NAMES = [\"sun\", \"mon\", \"tue\", \"wed\", \"thu\", \"fri\", \"sat\"]\n\nconst SHORTCUTS = {\n  \"@hourly\": \"0 * * * *\",\n  \"@daily\": \"0 0 * * *\",\n  \"@midnight\": \"0 0 * * *\",\n  \"@weekly\": \"0 0 * * 0\",\n  \"@monthly\": \"0 0 1 * *\",\n  \"@yearly\": \"0 0 1 1 *\",\n  \"@annually\": \"0 0 1 1 *\"\n}\n\nconst FIELDS = [\n  {name: \"minute\", min: 0, max: 59},\n  {name: \"hour\", min: 0, max: 23},\n  {name: \"dayOfMonth\", min: 1, max: 31},\n  {name: \"month\", min: 1, max: 12, names: MONTH_NAMES},\n  // Accept 0-7 so ranges like `5-7` (Fri-Sun) work; we normalize 7\n  // down to 0 after parsing in `normalizeDayOfWeek` below.\n  {name: \"dayOfWeek\", min: 0, max: 7, names: DAY_NAMES}\n]\n\n/**\n * @typedef {object} ParsedCron\n * @property {Set<number>} minute - Allowed minute values (0-59).\n * @property {Set<number>} hour - Allowed hour values (0-23).\n * @property {Set<number>} dayOfMonth - Allowed day-of-month values (1-31).\n * @property {Set<number>} month - Allowed month values (1-12).\n * @property {Set<number>} dayOfWeek - Allowed day-of-week values (0-6, 0=Sun).\n * @property {boolean} dayOfMonthRestricted - True when the dayOfMonth field is not `*`.\n * @property {boolean} dayOfWeekRestricted - True when the dayOfWeek field is not `*`.\n * @property {string} expression - Original expression for diagnostics.\n */\n\n/**\n * @param {string} expression - Cron expression or shortcut.\n * @returns {ParsedCron}\n */\nexport function parseCronExpression(expression) {\n  if (typeof expression !== \"string\" || !expression.trim()) {\n    throw new Error(`Invalid cron expression: ${expression}`)\n  }\n\n  const trimmed = expression.trim().toLowerCase()\n  const expanded = SHORTCUTS[/** @type {keyof typeof SHORTCUTS} */ (trimmed)] || trimmed\n  const fields = expanded.split(/\\s+/)\n\n  if (fields.length !== 5) {\n    throw new Error(`Invalid cron expression \"${expression}\": expected 5 fields, got ${fields.length}`)\n  }\n\n  const [minuteField, hourField, dayOfMonthField, monthField, dayOfWeekField] = fields\n  const parsed = {\n    minute: parseField(minuteField, FIELDS[0], expression),\n    hour: parseField(hourField, FIELDS[1], expression),\n    dayOfMonth: parseField(dayOfMonthField, FIELDS[2], expression),\n    month: parseField(monthField, FIELDS[3], expression),\n    // Cron treats both 0 and 7 as Sunday. We accept 7 throughout the\n    // parse pass (so `5-7` for Fri-Sun works) and then normalize any\n    // 7s down to 0 so the matcher only deals with 0-6.\n    dayOfWeek: normalizeDayOfWeek(parseField(dayOfWeekField, FIELDS[4], expression)),\n    dayOfMonthRestricted: dayOfMonthField !== \"*\",\n    dayOfWeekRestricted: dayOfWeekField !== \"*\",\n    expression\n  }\n\n  return parsed\n}\n\n/**\n * @param {Set<number>} dayOfWeek\n * @returns {Set<number>}\n */\nfunction normalizeDayOfWeek(dayOfWeek) {\n  if (dayOfWeek.has(7)) {\n    dayOfWeek.delete(7)\n    dayOfWeek.add(0)\n  }\n\n  return dayOfWeek\n}\n\n/**\n * @param {string} field - Field expression.\n * @param {{name: string, min: number, max: number, names?: string[]}} fieldSpec - Field spec.\n * @param {string} expression - Whole cron expression for error messages.\n * @returns {Set<number>}\n */\nfunction parseField(field, fieldSpec, expression) {\n  const result = new Set()\n\n  for (const part of field.split(\",\")) {\n    addPartValues(part, fieldSpec, expression, result)\n  }\n\n  return result\n}\n\n/**\n * @param {string} part - Single comma-separated chunk.\n * @param {{name: string, min: number, max: number, names?: string[]}} fieldSpec - Field spec.\n * @param {string} expression - Original expression for errors.\n * @param {Set<number>} result - Accumulator.\n * @returns {void}\n */\nfunction addPartValues(part, fieldSpec, expression, result) {\n  if (!part) {\n    throw new Error(`Invalid ${fieldSpec.name} field in cron expression \"${expression}\"`)\n  }\n\n  const [rangePart, stepPart] = part.split(\"/\")\n  const step = stepPart === undefined ? 1 : parseStep(stepPart, fieldSpec, expression)\n  const [start, end] = parseRange(rangePart, fieldSpec, expression, stepPart !== undefined)\n\n  for (let value = start; value <= end; value += step) {\n    if (value < fieldSpec.min || value > fieldSpec.max) {\n      throw new Error(`Value ${value} out of range for ${fieldSpec.name} in cron expression \"${expression}\"`)\n    }\n\n    result.add(value)\n  }\n}\n\n/**\n * @param {string} value - Step value.\n * @param {{name: string, min: number, max: number}} fieldSpec - Field spec.\n * @param {string} expression - Original expression for errors.\n * @returns {number}\n */\nfunction parseStep(value, fieldSpec, expression) {\n  const step = Number(value)\n\n  if (!Number.isInteger(step) || step <= 0) {\n    throw new Error(`Invalid step \"${value}\" for ${fieldSpec.name} in cron expression \"${expression}\"`)\n  }\n\n  return step\n}\n\n/**\n * @param {string} rangePart - Range portion (before any `/`).\n * @param {{name: string, min: number, max: number, names?: string[]}} fieldSpec - Field spec.\n * @param {string} expression - Original expression for errors.\n * @param {boolean} hasStep - Whether the part had a `/step` suffix.\n * @returns {[number, number]}\n */\nfunction parseRange(rangePart, fieldSpec, expression, hasStep) {\n  if (rangePart === \"*\") {\n    return [fieldSpec.min, fieldSpec.max]\n  }\n\n  const dashIndex = rangePart.indexOf(\"-\")\n\n  if (dashIndex === -1) {\n    const value = parseValue(rangePart, fieldSpec, expression)\n\n    // `N/step` is shorthand for `N-max/step` (Vixie cron).\n    return [value, hasStep ? fieldSpec.max : value]\n  }\n\n  const start = parseValue(rangePart.slice(0, dashIndex), fieldSpec, expression)\n  const end = parseValue(rangePart.slice(dashIndex + 1), fieldSpec, expression)\n\n  if (start > end) {\n    throw new Error(`Range start ${start} > end ${end} for ${fieldSpec.name} in cron expression \"${expression}\"`)\n  }\n\n  return [start, end]\n}\n\n/**\n * @param {string} rawValue - Raw value (may be a name).\n * @param {{name: string, min: number, max: number, names?: string[]}} fieldSpec - Field spec.\n * @param {string} expression - Original expression for errors.\n * @returns {number}\n */\nfunction parseValue(rawValue, fieldSpec, expression) {\n  if (!rawValue) {\n    throw new Error(`Invalid ${fieldSpec.name} value in cron expression \"${expression}\"`)\n  }\n\n  const namedIndex = fieldSpec.names?.indexOf(rawValue)\n\n  if (typeof namedIndex === \"number\" && namedIndex !== -1) {\n    return namedIndex + fieldSpec.min\n  }\n\n  const value = Number(rawValue)\n\n  if (!Number.isInteger(value)) {\n    throw new Error(`Invalid ${fieldSpec.name} value \"${rawValue}\" in cron expression \"${expression}\"`)\n  }\n\n  return value\n}\n\n// 5 years of minutes — covers the worst-case legitimate gap, the\n// `0 0 29 2 *` (Feb 29) leap-year-only schedule, with a one-year\n// buffer so we never report a real cron pattern as \"never matches\".\nconst MAX_NEXT_FIRE_ITERATIONS = 5 * 366 * 24 * 60\n\n/**\n * Returns the next Date strictly after `from` that satisfies `parsed`.\n * Operates at minute granularity. Bails out with an error after five\n * years of search, which only happens if the expression matches no\n * real time (e.g., `0 0 31 2 *` — Feb 31st).\n *\n * @param {ParsedCron} parsed - Parsed cron expression.\n * @param {Date} from - Reference Date — the next match is strictly after this.\n * @returns {Date}\n */\nexport function nextCronFireDate(parsed, from) {\n  const candidate = new Date(from.getTime())\n\n  candidate.setSeconds(0, 0)\n  candidate.setMinutes(candidate.getMinutes() + 1)\n\n  for (let iterations = 0; iterations < MAX_NEXT_FIRE_ITERATIONS; iterations += 1) {\n    if (candidateMatches(candidate, parsed)) return candidate\n\n    candidate.setMinutes(candidate.getMinutes() + 1)\n  }\n\n  throw new Error(`Cron expression \"${parsed.expression}\" never matches`)\n}\n\n/**\n * @param {Date} candidate - Candidate Date (in local time).\n * @param {ParsedCron} parsed - Parsed expression.\n * @returns {boolean}\n */\nfunction candidateMatches(candidate, parsed) {\n  if (!parsed.minute.has(candidate.getMinutes())) return false\n  if (!parsed.hour.has(candidate.getHours())) return false\n  if (!parsed.month.has(candidate.getMonth() + 1)) return false\n\n  const dayOfMonthMatch = parsed.dayOfMonth.has(candidate.getDate())\n  const dayOfWeekMatch = parsed.dayOfWeek.has(candidate.getDay())\n\n  // POSIX/Vixie cron OR semantics: when both day fields are\n  // restricted, fire when EITHER matches. When only one is\n  // restricted, only that one applies.\n  if (parsed.dayOfMonthRestricted && parsed.dayOfWeekRestricted) {\n    return dayOfMonthMatch || dayOfWeekMatch\n  }\n\n  if (parsed.dayOfMonthRestricted) return dayOfMonthMatch\n  if (parsed.dayOfWeekRestricted) return dayOfWeekMatch\n\n  return true\n}\n"]}
@@ -33,6 +33,8 @@ export default class BackgroundJobsScheduler {
33
33
  intervalIds: Array<ReturnType<typeof setInterval>>;
34
34
  /** @type {Array<ReturnType<typeof setTimeout>>} */
35
35
  timeoutIds: Array<ReturnType<typeof setTimeout>>;
36
+ /** @type {boolean} - True between stop() and the next start(); cron self-rescheduler checks this so a stop() during an in-flight enqueue doesn't immediately re-arm. */
37
+ stopped: boolean;
36
38
  /** @returns {Promise<void>} */
37
39
  start(): Promise<void>;
38
40
  /** @returns {void} */
@@ -47,6 +49,31 @@ export default class BackgroundJobsScheduler {
47
49
  jobConfiguration: import("../configuration-types.js").ScheduledBackgroundJobConfiguration;
48
50
  jobKey: string;
49
51
  }): void;
52
+ /**
53
+ * @param {object} args - Options.
54
+ * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.
55
+ * @param {string} args.jobKey - Job key.
56
+ * @returns {void}
57
+ */
58
+ scheduleEveryJob({ jobConfiguration, jobKey }: {
59
+ jobConfiguration: import("../configuration-types.js").ScheduledBackgroundJobConfiguration;
60
+ jobKey: string;
61
+ }): void;
62
+ /**
63
+ * Crontab schedules don't have a constant interval (`0 9 * * 1-5`
64
+ * fires once per weekday at 9 AM, with gaps of varying length), so
65
+ * we self-reschedule with `setTimeout` after every fire instead of
66
+ * using `setInterval`.
67
+ *
68
+ * @param {object} args - Options.
69
+ * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.
70
+ * @param {string} args.jobKey - Job key.
71
+ * @returns {void}
72
+ */
73
+ scheduleCronJob({ jobConfiguration, jobKey }: {
74
+ jobConfiguration: import("../configuration-types.js").ScheduledBackgroundJobConfiguration;
75
+ jobKey: string;
76
+ }): void;
50
77
  /**
51
78
  * @param {object} args - Options.
52
79
  * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.
@@ -58,10 +85,10 @@ export default class BackgroundJobsScheduler {
58
85
  jobKey: string;
59
86
  }): Promise<void>;
60
87
  /**
61
- * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration["every"]} every - Every config.
88
+ * @param {NonNullable<import("../configuration-types.js").ScheduledBackgroundJobConfiguration["every"]>} every - Every config (caller must guarantee not undefined).
62
89
  * @returns {{everyValue: number | string, firstInValue?: number | string}} - Normalized interval and first-run delay values.
63
90
  */
64
- normalizeEvery(every: import("../configuration-types.js").ScheduledBackgroundJobConfiguration["every"]): {
91
+ normalizeEvery(every: NonNullable<import("../configuration-types.js").ScheduledBackgroundJobConfiguration["every"]>): {
65
92
  everyValue: number | string;
66
93
  firstInValue?: number | string;
67
94
  };
@@ -1 +1 @@
1
- {"version":3,"file":"scheduler.d.ts","sourceRoot":"","sources":["../../../src/background-jobs/scheduler.js"],"names":[],"mappings":"AAsBA,gEAAgE;AAEhE;;;;GAIG;AACH,8CAJW,MAAM,GAAG,MAAM,aACf,MAAM,GACJ,MAAM,CA8BlB;AAED,0DAA0D;AAC1D;IACE;;;;OAIG;IACH,2CAHG;QAAoD,aAAa,EAAzD,OAAO,qBAAqB,EAAE,OAAO;QAC0H,UAAU,EAAzK,CAAS,IAA8H,EAA9H;YAAC,IAAI,EAAE,GAAG,EAAE,CAAC;YAAC,QAAQ,EAAE,cAAc,UAAU,EAAE,OAAO,CAAC;YAAC,MAAM,EAAE,MAAM,CAAC;YAAC,OAAO,EAAE,OAAO,YAAY,EAAE,oBAAoB,CAAA;SAAC,KAAI,OAAO,CAAC,IAAI,CAAC;KAClK,EASA;IAPC,qDAAkC;IAClC,mBAJkB;QAAC,IAAI,EAAE,GAAG,EAAE,CAAC;QAAC,QAAQ,EAAE,cAAc,UAAU,EAAE,OAAO,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,OAAO,YAAY,EAAE,oBAAoB,CAAA;KAAC,KAAI,OAAO,CAAC,IAAI,CAAC,CAIrI;IAC5B,eAA8B;IAC9B,oDAAoD;IACpD,aADW,KAAK,CAAC,UAAU,CAAC,OAAO,WAAW,CAAC,CAAC,CAC3B;IACrB,mDAAmD;IACnD,YADW,KAAK,CAAC,UAAU,CAAC,OAAO,UAAU,CAAC,CAAC,CAC3B;IAGtB,+BAA+B;IAC/B,SADc,OAAO,CAAC,IAAI,CAAC,CAiB1B;IAED,sBAAsB;IACtB,QADc,IAAI,CAYjB;IAED;;;;;OAKG;IACH,0CAJG;QAAsF,gBAAgB,EAA9F,OAAO,2BAA2B,EAAE,mCAAmC;QAC1D,MAAM,EAAnB,MAAM;KACd,GAAU,IAAI,CA0BhB;IAED;;;;;OAKG;IACH,kDAJG;QAAsF,gBAAgB,EAA9F,OAAO,2BAA2B,EAAE,mCAAmC;QAC1D,MAAM,EAAnB,MAAM;KACd,GAAU,OAAO,CAAC,IAAI,CAAC,CAazB;IAED;;;OAGG;IACH,sBAHW,OAAO,2BAA2B,EAAE,mCAAmC,CAAC,OAAO,CAAC,GAC9E;QAAC,UAAU,EAAE,MAAM,GAAG,MAAM,CAAC;QAAC,YAAY,CAAC,EAAE,MAAM,GAAG,MAAM,CAAA;KAAC,CAczE;CACF;2BA3Ja,MAAM,OAAO,oBAAoB;mBApB5B,cAAc"}
1
+ {"version":3,"file":"scheduler.d.ts","sourceRoot":"","sources":["../../../src/background-jobs/scheduler.js"],"names":[],"mappings":"AAuBA,gEAAgE;AAEhE;;;;GAIG;AACH,8CAJW,MAAM,GAAG,MAAM,aACf,MAAM,GACJ,MAAM,CA8BlB;AAED,0DAA0D;AAC1D;IACE;;;;OAIG;IACH,2CAHG;QAAoD,aAAa,EAAzD,OAAO,qBAAqB,EAAE,OAAO;QAC0H,UAAU,EAAzK,CAAS,IAA8H,EAA9H;YAAC,IAAI,EAAE,GAAG,EAAE,CAAC;YAAC,QAAQ,EAAE,cAAc,UAAU,EAAE,OAAO,CAAC;YAAC,MAAM,EAAE,MAAM,CAAC;YAAC,OAAO,EAAE,OAAO,YAAY,EAAE,oBAAoB,CAAA;SAAC,KAAI,OAAO,CAAC,IAAI,CAAC;KAClK,EAWA;IATC,qDAAkC;IAClC,mBAJkB;QAAC,IAAI,EAAE,GAAG,EAAE,CAAC;QAAC,QAAQ,EAAE,cAAc,UAAU,EAAE,OAAO,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;QAAC,OAAO,EAAE,OAAO,YAAY,EAAE,oBAAoB,CAAA;KAAC,KAAI,OAAO,CAAC,IAAI,CAAC,CAIrI;IAC5B,eAA8B;IAC9B,oDAAoD;IACpD,aADW,KAAK,CAAC,UAAU,CAAC,OAAO,WAAW,CAAC,CAAC,CAC3B;IACrB,mDAAmD;IACnD,YADW,KAAK,CAAC,UAAU,CAAC,OAAO,UAAU,CAAC,CAAC,CAC3B;IACpB,wKAAwK;IACxK,SADW,OAAO,CACE;IAGtB,+BAA+B;IAC/B,SADc,OAAO,CAAC,IAAI,CAAC,CAmB1B;IAED,sBAAsB;IACtB,QADc,IAAI,CAcjB;IAED;;;;;OAKG;IACH,0CAJG;QAAsF,gBAAgB,EAA9F,OAAO,2BAA2B,EAAE,mCAAmC;QAC1D,MAAM,EAAnB,MAAM;KACd,GAAU,IAAI,CAsBhB;IAED;;;;;OAKG;IACH,+CAJG;QAAsF,gBAAgB,EAA9F,OAAO,2BAA2B,EAAE,mCAAmC;QAC1D,MAAM,EAAnB,MAAM;KACd,GAAU,IAAI,CAuBhB;IAED;;;;;;;;;;OAUG;IACH,8CAJG;QAAsF,gBAAgB,EAA9F,OAAO,2BAA2B,EAAE,mCAAmC;QAC1D,MAAM,EAAnB,MAAM;KACd,GAAU,IAAI,CA+BhB;IAED;;;;;OAKG;IACH,kDAJG;QAAsF,gBAAgB,EAA9F,OAAO,2BAA2B,EAAE,mCAAmC;QAC1D,MAAM,EAAnB,MAAM;KACd,GAAU,OAAO,CAAC,IAAI,CAAC,CAazB;IAED;;;OAGG;IACH,sBAHW,WAAW,CAAC,OAAO,2BAA2B,EAAE,mCAAmC,CAAC,OAAO,CAAC,CAAC,GAC3F;QAAC,UAAU,EAAE,MAAM,GAAG,MAAM,CAAC;QAAC,YAAY,CAAC,EAAE,MAAM,GAAG,MAAM,CAAA;KAAC,CAczE;CACF;2BApOa,MAAM,OAAO,oBAAoB;mBArB5B,cAAc"}
@@ -1,5 +1,6 @@
1
1
  // @ts-check
2
2
  import Logger from "../logger.js";
3
+ import { nextCronFireDate, parseCronExpression } from "./cron-expression.js";
3
4
  const DURATION_MULTIPLIERS = {
4
5
  d: 24 * 60 * 60 * 1000,
5
6
  day: 24 * 60 * 60 * 1000,
@@ -61,9 +62,12 @@ export default class BackgroundJobsScheduler {
61
62
  this.intervalIds = [];
62
63
  /** @type {Array<ReturnType<typeof setTimeout>>} */
63
64
  this.timeoutIds = [];
65
+ /** @type {boolean} - True between stop() and the next start(); cron self-rescheduler checks this so a stop() during an in-flight enqueue doesn't immediately re-arm. */
66
+ this.stopped = false;
64
67
  }
65
68
  /** @returns {Promise<void>} */
66
69
  async start() {
70
+ this.stopped = false;
67
71
  const scheduledBackgroundJobsConfig = await this.configuration.getScheduledBackgroundJobsConfig();
68
72
  if (!scheduledBackgroundJobsConfig?.jobs) {
69
73
  return;
@@ -78,6 +82,7 @@ export default class BackgroundJobsScheduler {
78
82
  }
79
83
  /** @returns {void} */
80
84
  stop() {
85
+ this.stopped = true;
81
86
  for (const intervalId of this.intervalIds) {
82
87
  clearInterval(intervalId);
83
88
  }
@@ -94,15 +99,35 @@ export default class BackgroundJobsScheduler {
94
99
  * @returns {void}
95
100
  */
96
101
  scheduleJob({ jobConfiguration, jobKey }) {
97
- const { everyValue, firstInValue } = this.normalizeEvery(jobConfiguration.every);
102
+ if (!jobConfiguration.class || typeof jobConfiguration.class.performLaterWithOptions !== "function") {
103
+ throw new Error(`Scheduled background job ${jobKey} must define a job class.`);
104
+ }
105
+ if (jobConfiguration.cron !== undefined && jobConfiguration.every !== undefined) {
106
+ throw new Error(`Scheduled background job ${jobKey} must define either "every" or "cron", not both.`);
107
+ }
108
+ if (jobConfiguration.cron !== undefined) {
109
+ this.scheduleCronJob({ jobConfiguration, jobKey });
110
+ return;
111
+ }
112
+ if (jobConfiguration.every === undefined) {
113
+ throw new Error(`Scheduled background job ${jobKey} must define either "every" or "cron".`);
114
+ }
115
+ this.scheduleEveryJob({ jobConfiguration, jobKey });
116
+ }
117
+ /**
118
+ * @param {object} args - Options.
119
+ * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.
120
+ * @param {string} args.jobKey - Job key.
121
+ * @returns {void}
122
+ */
123
+ scheduleEveryJob({ jobConfiguration, jobKey }) {
124
+ const everyConfig = /** @type {NonNullable<typeof jobConfiguration.every>} */ (jobConfiguration.every);
125
+ const { everyValue, firstInValue } = this.normalizeEvery(everyConfig);
98
126
  const intervalMs = parseScheduledDuration(everyValue, `${jobKey}.every`);
99
127
  const firstInMs = firstInValue !== undefined ? parseScheduledDuration(firstInValue, `${jobKey}.first_in`) : intervalMs;
100
128
  if (intervalMs < 1) {
101
129
  throw new Error(`Scheduled background job ${jobKey}.every must be at least 1 millisecond.`);
102
130
  }
103
- if (!jobConfiguration.class || typeof jobConfiguration.class.performLaterWithOptions !== "function") {
104
- throw new Error(`Scheduled background job ${jobKey} must define a job class.`);
105
- }
106
131
  const timeoutId = setTimeout(() => {
107
132
  void this.enqueueScheduledJob({ jobConfiguration, jobKey });
108
133
  const intervalId = setInterval(() => {
@@ -112,6 +137,42 @@ export default class BackgroundJobsScheduler {
112
137
  }, firstInMs);
113
138
  this.timeoutIds.push(timeoutId);
114
139
  }
140
+ /**
141
+ * Crontab schedules don't have a constant interval (`0 9 * * 1-5`
142
+ * fires once per weekday at 9 AM, with gaps of varying length), so
143
+ * we self-reschedule with `setTimeout` after every fire instead of
144
+ * using `setInterval`.
145
+ *
146
+ * @param {object} args - Options.
147
+ * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.
148
+ * @param {string} args.jobKey - Job key.
149
+ * @returns {void}
150
+ */
151
+ scheduleCronJob({ jobConfiguration, jobKey }) {
152
+ const cronExpression = jobConfiguration.cron;
153
+ if (typeof cronExpression !== "string") {
154
+ throw new Error(`Scheduled background job ${jobKey}.cron must be a string.`);
155
+ }
156
+ const parsed = parseCronExpression(cronExpression);
157
+ const scheduleNext = () => {
158
+ if (this.stopped)
159
+ return;
160
+ const nextDate = nextCronFireDate(parsed, new Date());
161
+ const delayMs = Math.max(1, nextDate.getTime() - Date.now());
162
+ const timeoutId = setTimeout(async () => {
163
+ if (this.stopped)
164
+ return;
165
+ await this.enqueueScheduledJob({ jobConfiguration, jobKey });
166
+ // The await above can yield to a stop() call. Re-check before
167
+ // re-arming so we don't keep firing after shutdown.
168
+ if (this.stopped)
169
+ return;
170
+ scheduleNext();
171
+ }, delayMs);
172
+ this.timeoutIds.push(timeoutId);
173
+ };
174
+ scheduleNext();
175
+ }
115
176
  /**
116
177
  * @param {object} args - Options.
117
178
  * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.
@@ -132,7 +193,7 @@ export default class BackgroundJobsScheduler {
132
193
  }
133
194
  }
134
195
  /**
135
- * @param {import("../configuration-types.js").ScheduledBackgroundJobConfiguration["every"]} every - Every config.
196
+ * @param {NonNullable<import("../configuration-types.js").ScheduledBackgroundJobConfiguration["every"]>} every - Every config (caller must guarantee not undefined).
136
197
  * @returns {{everyValue: number | string, firstInValue?: number | string}} - Normalized interval and first-run delay values.
137
198
  */
138
199
  normalizeEvery(every) {
@@ -146,4 +207,4 @@ export default class BackgroundJobsScheduler {
146
207
  return { everyValue: every };
147
208
  }
148
209
  }
149
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scheduler.js","sourceRoot":"","sources":["../../../src/background-jobs/scheduler.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport Logger from \"../logger.js\"\n\nconst DURATION_MULTIPLIERS = {\n  d: 24 * 60 * 60 * 1000,\n  day: 24 * 60 * 60 * 1000,\n  days: 24 * 60 * 60 * 1000,\n  h: 60 * 60 * 1000,\n  hour: 60 * 60 * 1000,\n  hours: 60 * 60 * 1000,\n  m: 60 * 1000,\n  minute: 60 * 1000,\n  minutes: 60 * 1000,\n  ms: 1,\n  s: 1000,\n  second: 1000,\n  seconds: 1000,\n  w: 7 * 24 * 60 * 60 * 1000,\n  week: 7 * 24 * 60 * 60 * 1000,\n  weeks: 7 * 24 * 60 * 60 * 1000\n}\n/** @typedef {keyof typeof DURATION_MULTIPLIERS} DurationUnit */\n\n/**\n * @param {number | string} value - Duration value.\n * @param {string} fieldName - Field name for errors.\n * @returns {number} - Duration in milliseconds.\n */\nexport function parseScheduledDuration(value, fieldName) {\n  if (typeof value === \"number\") {\n    if (!Number.isFinite(value) || value < 1) {\n      throw new Error(`Scheduled background job ${fieldName} must be a positive number of milliseconds.`)\n    }\n\n    return value\n  }\n\n  if (typeof value !== \"string\" || !value.trim()) {\n    throw new Error(`Scheduled background job ${fieldName} must be a non-empty string or number.`)\n  }\n\n  const normalizedValue = value.trim().toLowerCase()\n  const match = normalizedValue.match(/^(\\d+(?:\\.\\d+)?)\\s*(ms|s|m|h|d|w|second|seconds|minute|minutes|hour|hours|day|days|week|weeks)$/)\n\n  if (!match) {\n    throw new Error(`Invalid scheduled background job ${fieldName}: ${value}`)\n  }\n\n  const numericValue = Number(match[1])\n  const multiplier = DURATION_MULTIPLIERS[/** @type {DurationUnit} */ (match[2])]\n\n  if (!multiplier) {\n    throw new Error(`Invalid scheduled background job ${fieldName}: ${value}`)\n  }\n\n  return Math.round(numericValue * multiplier)\n}\n\n/** Runs configured recurring background job schedules. */\nexport default class BackgroundJobsScheduler {\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration.\n   * @param {function({args: any[], jobClass: typeof import(\"./job.js\").default, jobKey: string, options: import(\"./types.js\").BackgroundJobOptions}) : Promise<void>} args.enqueueJob - Enqueue callback.\n   */\n  constructor({configuration, enqueueJob}) {\n    this.configuration = configuration\n    this.enqueueJob = enqueueJob\n    this.logger = new Logger(this)\n    /** @type {Array<ReturnType<typeof setInterval>>} */\n    this.intervalIds = []\n    /** @type {Array<ReturnType<typeof setTimeout>>} */\n    this.timeoutIds = []\n  }\n\n  /** @returns {Promise<void>} */\n  async start() {\n    const scheduledBackgroundJobsConfig = await this.configuration.getScheduledBackgroundJobsConfig()\n\n    if (!scheduledBackgroundJobsConfig?.jobs) {\n      return\n    }\n\n    for (const jobKey of Object.keys(scheduledBackgroundJobsConfig.jobs)) {\n      const jobConfiguration = scheduledBackgroundJobsConfig.jobs[jobKey]\n\n      if (!jobConfiguration || jobConfiguration.enabled === false) {\n        continue\n      }\n\n      this.scheduleJob({jobConfiguration, jobKey})\n    }\n  }\n\n  /** @returns {void} */\n  stop() {\n    for (const intervalId of this.intervalIds) {\n      clearInterval(intervalId)\n    }\n\n    for (const timeoutId of this.timeoutIds) {\n      clearTimeout(timeoutId)\n    }\n\n    this.intervalIds = []\n    this.timeoutIds = []\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration-types.js\").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.\n   * @param {string} args.jobKey - Job key.\n   * @returns {void}\n   */\n  scheduleJob({jobConfiguration, jobKey}) {\n    const {everyValue, firstInValue} = this.normalizeEvery(jobConfiguration.every)\n    const intervalMs = parseScheduledDuration(everyValue, `${jobKey}.every`)\n    const firstInMs = firstInValue !== undefined ? parseScheduledDuration(firstInValue, `${jobKey}.first_in`) : intervalMs\n\n    if (intervalMs < 1) {\n      throw new Error(`Scheduled background job ${jobKey}.every must be at least 1 millisecond.`)\n    }\n\n    if (!jobConfiguration.class || typeof jobConfiguration.class.performLaterWithOptions !== \"function\") {\n      throw new Error(`Scheduled background job ${jobKey} must define a job class.`)\n    }\n\n    const timeoutId = setTimeout(() => {\n      void this.enqueueScheduledJob({jobConfiguration, jobKey})\n\n      const intervalId = setInterval(() => {\n        void this.enqueueScheduledJob({jobConfiguration, jobKey})\n      }, intervalMs)\n\n      this.intervalIds.push(intervalId)\n    }, firstInMs)\n\n    this.timeoutIds.push(timeoutId)\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration-types.js\").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.\n   * @param {string} args.jobKey - Job key.\n   * @returns {Promise<void>}\n   */\n  async enqueueScheduledJob({jobConfiguration, jobKey}) {\n    try {\n      await this.enqueueJob({\n        args: Array.isArray(jobConfiguration.args) ? jobConfiguration.args : [],\n        jobClass: jobConfiguration.class,\n        jobKey,\n        options: jobConfiguration.options || {}\n      })\n    } catch (error) {\n      await this.logger.error(() => [\"Failed to enqueue scheduled background job\", {jobKey, jobName: jobConfiguration.class.jobName()}, error])\n    }\n  }\n\n  /**\n   * @param {import(\"../configuration-types.js\").ScheduledBackgroundJobConfiguration[\"every\"]} every - Every config.\n   * @returns {{everyValue: number | string, firstInValue?: number | string}} - Normalized interval and first-run delay values.\n   */\n  normalizeEvery(every) {\n    if (Array.isArray(every)) {\n      const [everyValue, everyOptions] = every\n\n      if (!everyOptions || typeof everyOptions !== \"object\" || Array.isArray(everyOptions)) {\n        return {everyValue}\n      }\n\n      return {everyValue, firstInValue: everyOptions.firstIn ?? everyOptions.first_in}\n    }\n\n    return {everyValue: every}\n  }\n}\n"]}
210
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scheduler.js","sourceRoot":"","sources":["../../../src/background-jobs/scheduler.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport Logger from \"../logger.js\"\nimport {nextCronFireDate, parseCronExpression} from \"./cron-expression.js\"\n\nconst DURATION_MULTIPLIERS = {\n  d: 24 * 60 * 60 * 1000,\n  day: 24 * 60 * 60 * 1000,\n  days: 24 * 60 * 60 * 1000,\n  h: 60 * 60 * 1000,\n  hour: 60 * 60 * 1000,\n  hours: 60 * 60 * 1000,\n  m: 60 * 1000,\n  minute: 60 * 1000,\n  minutes: 60 * 1000,\n  ms: 1,\n  s: 1000,\n  second: 1000,\n  seconds: 1000,\n  w: 7 * 24 * 60 * 60 * 1000,\n  week: 7 * 24 * 60 * 60 * 1000,\n  weeks: 7 * 24 * 60 * 60 * 1000\n}\n/** @typedef {keyof typeof DURATION_MULTIPLIERS} DurationUnit */\n\n/**\n * @param {number | string} value - Duration value.\n * @param {string} fieldName - Field name for errors.\n * @returns {number} - Duration in milliseconds.\n */\nexport function parseScheduledDuration(value, fieldName) {\n  if (typeof value === \"number\") {\n    if (!Number.isFinite(value) || value < 1) {\n      throw new Error(`Scheduled background job ${fieldName} must be a positive number of milliseconds.`)\n    }\n\n    return value\n  }\n\n  if (typeof value !== \"string\" || !value.trim()) {\n    throw new Error(`Scheduled background job ${fieldName} must be a non-empty string or number.`)\n  }\n\n  const normalizedValue = value.trim().toLowerCase()\n  const match = normalizedValue.match(/^(\\d+(?:\\.\\d+)?)\\s*(ms|s|m|h|d|w|second|seconds|minute|minutes|hour|hours|day|days|week|weeks)$/)\n\n  if (!match) {\n    throw new Error(`Invalid scheduled background job ${fieldName}: ${value}`)\n  }\n\n  const numericValue = Number(match[1])\n  const multiplier = DURATION_MULTIPLIERS[/** @type {DurationUnit} */ (match[2])]\n\n  if (!multiplier) {\n    throw new Error(`Invalid scheduled background job ${fieldName}: ${value}`)\n  }\n\n  return Math.round(numericValue * multiplier)\n}\n\n/** Runs configured recurring background job schedules. */\nexport default class BackgroundJobsScheduler {\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration.\n   * @param {function({args: any[], jobClass: typeof import(\"./job.js\").default, jobKey: string, options: import(\"./types.js\").BackgroundJobOptions}) : Promise<void>} args.enqueueJob - Enqueue callback.\n   */\n  constructor({configuration, enqueueJob}) {\n    this.configuration = configuration\n    this.enqueueJob = enqueueJob\n    this.logger = new Logger(this)\n    /** @type {Array<ReturnType<typeof setInterval>>} */\n    this.intervalIds = []\n    /** @type {Array<ReturnType<typeof setTimeout>>} */\n    this.timeoutIds = []\n    /** @type {boolean} - True between stop() and the next start(); cron self-rescheduler checks this so a stop() during an in-flight enqueue doesn't immediately re-arm. */\n    this.stopped = false\n  }\n\n  /** @returns {Promise<void>} */\n  async start() {\n    this.stopped = false\n\n    const scheduledBackgroundJobsConfig = await this.configuration.getScheduledBackgroundJobsConfig()\n\n    if (!scheduledBackgroundJobsConfig?.jobs) {\n      return\n    }\n\n    for (const jobKey of Object.keys(scheduledBackgroundJobsConfig.jobs)) {\n      const jobConfiguration = scheduledBackgroundJobsConfig.jobs[jobKey]\n\n      if (!jobConfiguration || jobConfiguration.enabled === false) {\n        continue\n      }\n\n      this.scheduleJob({jobConfiguration, jobKey})\n    }\n  }\n\n  /** @returns {void} */\n  stop() {\n    this.stopped = true\n\n    for (const intervalId of this.intervalIds) {\n      clearInterval(intervalId)\n    }\n\n    for (const timeoutId of this.timeoutIds) {\n      clearTimeout(timeoutId)\n    }\n\n    this.intervalIds = []\n    this.timeoutIds = []\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration-types.js\").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.\n   * @param {string} args.jobKey - Job key.\n   * @returns {void}\n   */\n  scheduleJob({jobConfiguration, jobKey}) {\n    if (!jobConfiguration.class || typeof jobConfiguration.class.performLaterWithOptions !== \"function\") {\n      throw new Error(`Scheduled background job ${jobKey} must define a job class.`)\n    }\n\n    if (jobConfiguration.cron !== undefined && jobConfiguration.every !== undefined) {\n      throw new Error(`Scheduled background job ${jobKey} must define either \"every\" or \"cron\", not both.`)\n    }\n\n    if (jobConfiguration.cron !== undefined) {\n      this.scheduleCronJob({jobConfiguration, jobKey})\n\n      return\n    }\n\n    if (jobConfiguration.every === undefined) {\n      throw new Error(`Scheduled background job ${jobKey} must define either \"every\" or \"cron\".`)\n    }\n\n    this.scheduleEveryJob({jobConfiguration, jobKey})\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration-types.js\").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.\n   * @param {string} args.jobKey - Job key.\n   * @returns {void}\n   */\n  scheduleEveryJob({jobConfiguration, jobKey}) {\n    const everyConfig = /** @type {NonNullable<typeof jobConfiguration.every>} */ (jobConfiguration.every)\n    const {everyValue, firstInValue} = this.normalizeEvery(everyConfig)\n    const intervalMs = parseScheduledDuration(everyValue, `${jobKey}.every`)\n    const firstInMs = firstInValue !== undefined ? parseScheduledDuration(firstInValue, `${jobKey}.first_in`) : intervalMs\n\n    if (intervalMs < 1) {\n      throw new Error(`Scheduled background job ${jobKey}.every must be at least 1 millisecond.`)\n    }\n\n    const timeoutId = setTimeout(() => {\n      void this.enqueueScheduledJob({jobConfiguration, jobKey})\n\n      const intervalId = setInterval(() => {\n        void this.enqueueScheduledJob({jobConfiguration, jobKey})\n      }, intervalMs)\n\n      this.intervalIds.push(intervalId)\n    }, firstInMs)\n\n    this.timeoutIds.push(timeoutId)\n  }\n\n  /**\n   * Crontab schedules don't have a constant interval (`0 9 * * 1-5`\n   * fires once per weekday at 9 AM, with gaps of varying length), so\n   * we self-reschedule with `setTimeout` after every fire instead of\n   * using `setInterval`.\n   *\n   * @param {object} args - Options.\n   * @param {import(\"../configuration-types.js\").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.\n   * @param {string} args.jobKey - Job key.\n   * @returns {void}\n   */\n  scheduleCronJob({jobConfiguration, jobKey}) {\n    const cronExpression = jobConfiguration.cron\n\n    if (typeof cronExpression !== \"string\") {\n      throw new Error(`Scheduled background job ${jobKey}.cron must be a string.`)\n    }\n\n    const parsed = parseCronExpression(cronExpression)\n    const scheduleNext = () => {\n      if (this.stopped) return\n\n      const nextDate = nextCronFireDate(parsed, new Date())\n      const delayMs = Math.max(1, nextDate.getTime() - Date.now())\n      const timeoutId = setTimeout(async () => {\n        if (this.stopped) return\n\n        await this.enqueueScheduledJob({jobConfiguration, jobKey})\n\n        // The await above can yield to a stop() call. Re-check before\n        // re-arming so we don't keep firing after shutdown.\n        if (this.stopped) return\n\n        scheduleNext()\n      }, delayMs)\n\n      this.timeoutIds.push(timeoutId)\n    }\n\n    scheduleNext()\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration-types.js\").ScheduledBackgroundJobConfiguration} args.jobConfiguration - Job configuration.\n   * @param {string} args.jobKey - Job key.\n   * @returns {Promise<void>}\n   */\n  async enqueueScheduledJob({jobConfiguration, jobKey}) {\n    try {\n      await this.enqueueJob({\n        args: Array.isArray(jobConfiguration.args) ? jobConfiguration.args : [],\n        jobClass: jobConfiguration.class,\n        jobKey,\n        options: jobConfiguration.options || {}\n      })\n    } catch (error) {\n      await this.logger.error(() => [\"Failed to enqueue scheduled background job\", {jobKey, jobName: jobConfiguration.class.jobName()}, error])\n    }\n  }\n\n  /**\n   * @param {NonNullable<import(\"../configuration-types.js\").ScheduledBackgroundJobConfiguration[\"every\"]>} every - Every config (caller must guarantee not undefined).\n   * @returns {{everyValue: number | string, firstInValue?: number | string}} - Normalized interval and first-run delay values.\n   */\n  normalizeEvery(every) {\n    if (Array.isArray(every)) {\n      const [everyValue, everyOptions] = every\n\n      if (!everyOptions || typeof everyOptions !== \"object\" || Array.isArray(everyOptions)) {\n        return {everyValue}\n      }\n\n      return {everyValue, firstInValue: everyOptions.firstIn ?? everyOptions.first_in}\n    }\n\n    return {everyValue: every}\n  }\n}\n"]}