dabke 0.81.1 → 0.83.0
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/CHANGELOG.md +58 -0
- package/README.md +45 -27
- package/dist/client.d.ts +20 -2
- package/dist/client.d.ts.map +1 -1
- package/dist/client.js +4 -1
- package/dist/client.js.map +1 -1
- package/dist/client.types.d.ts +9 -0
- package/dist/client.types.d.ts.map +1 -1
- package/dist/client.types.js +1 -0
- package/dist/client.types.js.map +1 -1
- package/dist/cpsat/model-builder.d.ts +9 -0
- package/dist/cpsat/model-builder.d.ts.map +1 -1
- package/dist/cpsat/model-builder.js +36 -34
- package/dist/cpsat/model-builder.js.map +1 -1
- package/dist/cpsat/response.d.ts +13 -1
- package/dist/cpsat/response.d.ts.map +1 -1
- package/dist/cpsat/response.js +4 -0
- package/dist/cpsat/response.js.map +1 -1
- package/dist/cpsat/rules/cost-utils.d.ts +11 -0
- package/dist/cpsat/rules/cost-utils.d.ts.map +1 -0
- package/dist/cpsat/rules/cost-utils.js +24 -0
- package/dist/cpsat/rules/cost-utils.js.map +1 -0
- package/dist/cpsat/rules/day-cost-multiplier.d.ts.map +1 -1
- package/dist/cpsat/rules/day-cost-multiplier.js +3 -14
- package/dist/cpsat/rules/day-cost-multiplier.js.map +1 -1
- package/dist/cpsat/rules/day-cost-surcharge.d.ts.map +1 -1
- package/dist/cpsat/rules/day-cost-surcharge.js +3 -7
- package/dist/cpsat/rules/day-cost-surcharge.js.map +1 -1
- package/dist/cpsat/rules/index.d.ts +3 -0
- package/dist/cpsat/rules/index.d.ts.map +1 -1
- package/dist/cpsat/rules/index.js +3 -0
- package/dist/cpsat/rules/index.js.map +1 -1
- package/dist/cpsat/rules/max-consecutive-days.d.ts.map +1 -1
- package/dist/cpsat/rules/max-consecutive-days.js +16 -2
- package/dist/cpsat/rules/max-consecutive-days.js.map +1 -1
- package/dist/cpsat/rules/max-days-week.d.ts +44 -0
- package/dist/cpsat/rules/max-days-week.d.ts.map +1 -0
- package/dist/cpsat/rules/max-days-week.js +95 -0
- package/dist/cpsat/rules/max-days-week.js.map +1 -0
- package/dist/cpsat/rules/max-hours-day.d.ts.map +1 -1
- package/dist/cpsat/rules/max-hours-day.js +15 -2
- package/dist/cpsat/rules/max-hours-day.js.map +1 -1
- package/dist/cpsat/rules/max-hours-week.d.ts.map +1 -1
- package/dist/cpsat/rules/max-hours-week.js +16 -2
- package/dist/cpsat/rules/max-hours-week.js.map +1 -1
- package/dist/cpsat/rules/max-shifts-day.d.ts.map +1 -1
- package/dist/cpsat/rules/max-shifts-day.js +15 -2
- package/dist/cpsat/rules/max-shifts-day.js.map +1 -1
- package/dist/cpsat/rules/min-consecutive-days.d.ts.map +1 -1
- package/dist/cpsat/rules/min-consecutive-days.js +15 -2
- package/dist/cpsat/rules/min-consecutive-days.js.map +1 -1
- package/dist/cpsat/rules/min-days-week.d.ts +34 -0
- package/dist/cpsat/rules/min-days-week.d.ts.map +1 -0
- package/dist/cpsat/rules/min-days-week.js +84 -0
- package/dist/cpsat/rules/min-days-week.js.map +1 -0
- package/dist/cpsat/rules/min-hours-day.d.ts.map +1 -1
- package/dist/cpsat/rules/min-hours-day.js +15 -2
- package/dist/cpsat/rules/min-hours-day.js.map +1 -1
- package/dist/cpsat/rules/min-hours-week.d.ts.map +1 -1
- package/dist/cpsat/rules/min-hours-week.js +16 -2
- package/dist/cpsat/rules/min-hours-week.js.map +1 -1
- package/dist/cpsat/rules/min-rest-between-shifts.d.ts.map +1 -1
- package/dist/cpsat/rules/min-rest-between-shifts.js +72 -2
- package/dist/cpsat/rules/min-rest-between-shifts.js.map +1 -1
- package/dist/cpsat/rules/minimize-cost.d.ts.map +1 -1
- package/dist/cpsat/rules/minimize-cost.js +2 -23
- package/dist/cpsat/rules/minimize-cost.js.map +1 -1
- package/dist/cpsat/rules/must-assign.d.ts +49 -0
- package/dist/cpsat/rules/must-assign.d.ts.map +1 -0
- package/dist/cpsat/rules/must-assign.js +86 -0
- package/dist/cpsat/rules/must-assign.js.map +1 -0
- package/dist/cpsat/rules/overtime-daily-multiplier.d.ts.map +1 -1
- package/dist/cpsat/rules/overtime-daily-multiplier.js +1 -12
- package/dist/cpsat/rules/overtime-daily-multiplier.js.map +1 -1
- package/dist/cpsat/rules/overtime-daily-surcharge.d.ts.map +1 -1
- package/dist/cpsat/rules/overtime-daily-surcharge.js +1 -5
- package/dist/cpsat/rules/overtime-daily-surcharge.js.map +1 -1
- package/dist/cpsat/rules/overtime-tiered-multiplier.d.ts +5 -1
- package/dist/cpsat/rules/overtime-tiered-multiplier.d.ts.map +1 -1
- package/dist/cpsat/rules/overtime-tiered-multiplier.js +1 -12
- package/dist/cpsat/rules/overtime-tiered-multiplier.js.map +1 -1
- package/dist/cpsat/rules/overtime-weekly-multiplier.d.ts.map +1 -1
- package/dist/cpsat/rules/overtime-weekly-multiplier.js +1 -12
- package/dist/cpsat/rules/overtime-weekly-multiplier.js.map +1 -1
- package/dist/cpsat/rules/overtime-weekly-surcharge.d.ts.map +1 -1
- package/dist/cpsat/rules/overtime-weekly-surcharge.js +1 -5
- package/dist/cpsat/rules/overtime-weekly-surcharge.js.map +1 -1
- package/dist/cpsat/rules/registry.d.ts +28 -2
- package/dist/cpsat/rules/registry.d.ts.map +1 -1
- package/dist/cpsat/rules/registry.js +4 -1
- package/dist/cpsat/rules/registry.js.map +1 -1
- package/dist/cpsat/rules/resolver.js +2 -2
- package/dist/cpsat/rules/resolver.js.map +1 -1
- package/dist/cpsat/rules/rules.types.d.ts +3 -0
- package/dist/cpsat/rules/rules.types.d.ts.map +1 -1
- package/dist/cpsat/rules/scope.types.d.ts +18 -1
- package/dist/cpsat/rules/scope.types.d.ts.map +1 -1
- package/dist/cpsat/rules/scope.types.js +59 -16
- package/dist/cpsat/rules/scope.types.js.map +1 -1
- package/dist/cpsat/rules/time-cost-surcharge.d.ts.map +1 -1
- package/dist/cpsat/rules/time-cost-surcharge.js +2 -1
- package/dist/cpsat/rules/time-cost-surcharge.js.map +1 -1
- package/dist/cpsat/rules/time-off.d.ts.map +1 -1
- package/dist/cpsat/rules/time-off.js +6 -3
- package/dist/cpsat/rules/time-off.js.map +1 -1
- package/dist/cpsat/semantic-time.d.ts +44 -42
- package/dist/cpsat/semantic-time.d.ts.map +1 -1
- package/dist/cpsat/semantic-time.js +64 -46
- package/dist/cpsat/semantic-time.js.map +1 -1
- package/dist/cpsat/types.d.ts +37 -27
- package/dist/cpsat/types.d.ts.map +1 -1
- package/dist/cpsat/utils.d.ts.map +1 -1
- package/dist/cpsat/utils.js +7 -12
- package/dist/cpsat/utils.js.map +1 -1
- package/dist/cpsat/validation-reporter.d.ts +10 -7
- package/dist/cpsat/validation-reporter.d.ts.map +1 -1
- package/dist/cpsat/validation-reporter.js +44 -72
- package/dist/cpsat/validation-reporter.js.map +1 -1
- package/dist/cpsat/validation.types.d.ts +54 -44
- package/dist/cpsat/validation.types.d.ts.map +1 -1
- package/dist/cpsat/validation.types.js +15 -10
- package/dist/cpsat/validation.types.js.map +1 -1
- package/dist/datetime.utils.d.ts +3 -203
- package/dist/datetime.utils.d.ts.map +1 -1
- package/dist/datetime.utils.js +1 -288
- package/dist/datetime.utils.js.map +1 -1
- package/dist/index.d.ts +14 -83
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +11 -83
- package/dist/index.js.map +1 -1
- package/dist/schedule/cost.d.ts +204 -0
- package/dist/schedule/cost.d.ts.map +1 -0
- package/dist/schedule/cost.js +187 -0
- package/dist/schedule/cost.js.map +1 -0
- package/dist/schedule/coverage.d.ts +85 -0
- package/dist/schedule/coverage.d.ts.map +1 -0
- package/dist/schedule/coverage.js +33 -0
- package/dist/schedule/coverage.js.map +1 -0
- package/dist/schedule/definition.d.ts +227 -0
- package/dist/schedule/definition.d.ts.map +1 -0
- package/dist/schedule/definition.js +659 -0
- package/dist/schedule/definition.js.map +1 -0
- package/dist/schedule/index.d.ts +67 -0
- package/dist/schedule/index.d.ts.map +1 -0
- package/dist/schedule/index.js +69 -0
- package/dist/schedule/index.js.map +1 -0
- package/dist/schedule/rules.d.ts +353 -0
- package/dist/schedule/rules.d.ts.map +1 -0
- package/dist/schedule/rules.js +352 -0
- package/dist/schedule/rules.js.map +1 -0
- package/dist/schedule/shift-patterns.d.ts +34 -0
- package/dist/schedule/shift-patterns.d.ts.map +1 -0
- package/dist/schedule/shift-patterns.js +41 -0
- package/dist/schedule/shift-patterns.js.map +1 -0
- package/dist/schedule/time-periods.d.ts +69 -0
- package/dist/schedule/time-periods.d.ts.map +1 -0
- package/dist/schedule/time-periods.js +91 -0
- package/dist/schedule/time-periods.js.map +1 -0
- package/dist/types.d.ts +14 -78
- package/dist/types.d.ts.map +1 -1
- package/dist/types.js.map +1 -1
- package/package.json +4 -9
- package/solver/src/solver/app.py +1 -1
- package/solver/src/solver/solver.py +7 -4
- package/src/client.ts +6 -8
- package/src/client.types.ts +9 -0
- package/src/cpsat/model-builder.ts +44 -35
- package/src/cpsat/response.ts +13 -1
- package/src/cpsat/rules/cost-utils.ts +25 -0
- package/src/cpsat/rules/day-cost-multiplier.ts +3 -14
- package/src/cpsat/rules/day-cost-surcharge.ts +3 -8
- package/src/cpsat/rules/index.ts +3 -0
- package/src/cpsat/rules/max-consecutive-days.ts +17 -0
- package/src/cpsat/rules/max-days-week.ts +143 -0
- package/src/cpsat/rules/max-hours-day.ts +21 -1
- package/src/cpsat/rules/max-hours-week.ts +22 -1
- package/src/cpsat/rules/max-shifts-day.ts +21 -1
- package/src/cpsat/rules/min-consecutive-days.ts +16 -1
- package/src/cpsat/rules/min-days-week.ts +120 -0
- package/src/cpsat/rules/min-hours-day.ts +16 -1
- package/src/cpsat/rules/min-hours-week.ts +17 -1
- package/src/cpsat/rules/min-rest-between-shifts.ts +92 -2
- package/src/cpsat/rules/minimize-cost.ts +2 -29
- package/src/cpsat/rules/must-assign.ts +108 -0
- package/src/cpsat/rules/overtime-daily-multiplier.ts +1 -12
- package/src/cpsat/rules/overtime-daily-surcharge.ts +1 -6
- package/src/cpsat/rules/overtime-tiered-multiplier.ts +6 -13
- package/src/cpsat/rules/overtime-weekly-multiplier.ts +1 -12
- package/src/cpsat/rules/overtime-weekly-surcharge.ts +1 -6
- package/src/cpsat/rules/registry.ts +8 -2
- package/src/cpsat/rules/resolver.ts +2 -2
- package/src/cpsat/rules/rules.types.ts +3 -0
- package/src/cpsat/rules/scope.types.ts +73 -20
- package/src/cpsat/rules/time-cost-surcharge.ts +2 -1
- package/src/cpsat/rules/time-off.ts +6 -2
- package/src/cpsat/semantic-time.ts +115 -91
- package/src/cpsat/types.ts +37 -27
- package/src/cpsat/utils.ts +8 -12
- package/src/cpsat/validation-reporter.ts +51 -82
- package/src/cpsat/validation.types.ts +72 -47
- package/src/datetime.utils.ts +3 -334
- package/src/index.ts +35 -107
- package/src/schedule/cost.ts +242 -0
- package/src/schedule/coverage.ts +135 -0
- package/src/schedule/definition.ts +958 -0
- package/src/schedule/index.ts +112 -0
- package/src/schedule/rules.ts +529 -0
- package/src/schedule/shift-patterns.ts +46 -0
- package/src/schedule/time-periods.ts +110 -0
- package/src/types.ts +14 -88
- package/dist/errors.d.ts +0 -12
- package/dist/errors.d.ts.map +0 -1
- package/dist/errors.js +0 -17
- package/dist/errors.js.map +0 -1
- package/dist/llms.d.ts +0 -2
- package/dist/llms.d.ts.map +0 -1
- package/dist/llms.js +0 -3
- package/dist/llms.js.map +0 -1
- package/dist/schedule.d.ts +0 -724
- package/dist/schedule.d.ts.map +0 -1
- package/dist/schedule.js +0 -899
- package/dist/schedule.js.map +0 -1
- package/dist/validation.d.ts +0 -105
- package/dist/validation.d.ts.map +0 -1
- package/dist/validation.js +0 -130
- package/dist/validation.js.map +0 -1
- package/llms.txt +0 -925
- package/src/errors.ts +0 -17
- package/src/llms.ts +0 -3
- package/src/schedule.ts +0 -1419
- package/src/validation.ts +0 -188
|
@@ -0,0 +1,659 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Schedule definition, compilation, and solving.
|
|
3
|
+
*
|
|
4
|
+
* @module
|
|
5
|
+
*/
|
|
6
|
+
import { defineSemanticTimes } from "../cpsat/semantic-time.js";
|
|
7
|
+
import { resolveDaysFromPeriod } from "../datetime.utils.js";
|
|
8
|
+
import { ModelBuilder } from "../cpsat/model-builder.js";
|
|
9
|
+
import { builtInCpsatRuleFactories } from "../cpsat/rules/registry.js";
|
|
10
|
+
import { parseSolverResponse, resolveAssignments } from "../cpsat/response.js";
|
|
11
|
+
import { calculateScheduleCost } from "../cpsat/cost.js";
|
|
12
|
+
import { resolveAppliesTo } from "./rules.js";
|
|
13
|
+
// ============================================================================
|
|
14
|
+
// Schedule class
|
|
15
|
+
// ============================================================================
|
|
16
|
+
/**
|
|
17
|
+
* An immutable schedule definition.
|
|
18
|
+
*
|
|
19
|
+
* Created by {@link schedule}, composed via {@link Schedule.with},
|
|
20
|
+
* and solved via {@link Schedule.solve}.
|
|
21
|
+
*
|
|
22
|
+
* @category Schedule Definition
|
|
23
|
+
*/
|
|
24
|
+
export class Schedule {
|
|
25
|
+
#config;
|
|
26
|
+
/** @internal */
|
|
27
|
+
constructor(config) {
|
|
28
|
+
this.#config = config;
|
|
29
|
+
}
|
|
30
|
+
/** @internal Returns a defensive copy of the config for merging. */
|
|
31
|
+
_getConfig() {
|
|
32
|
+
return {
|
|
33
|
+
...this.#config,
|
|
34
|
+
roleIds: [...this.#config.roleIds],
|
|
35
|
+
skillIds: [...this.#config.skillIds],
|
|
36
|
+
times: { ...this.#config.times },
|
|
37
|
+
coverage: [...this.#config.coverage],
|
|
38
|
+
shiftPatterns: [...this.#config.shiftPatterns],
|
|
39
|
+
rules: [...this.#config.rules],
|
|
40
|
+
ruleFactories: { ...this.#config.ruleFactories },
|
|
41
|
+
members: [...this.#config.members],
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
// --------------------------------------------------------------------------
|
|
45
|
+
// Inspection
|
|
46
|
+
// --------------------------------------------------------------------------
|
|
47
|
+
/** Declared role IDs. */
|
|
48
|
+
get roleIds() {
|
|
49
|
+
return this.#config.roleIds;
|
|
50
|
+
}
|
|
51
|
+
/** Declared skill IDs. */
|
|
52
|
+
get skillIds() {
|
|
53
|
+
return this.#config.skillIds;
|
|
54
|
+
}
|
|
55
|
+
/** Names of declared semantic times. */
|
|
56
|
+
get timeNames() {
|
|
57
|
+
return Object.keys(this.#config.times);
|
|
58
|
+
}
|
|
59
|
+
/** Shift pattern IDs. */
|
|
60
|
+
get shiftPatternIds() {
|
|
61
|
+
return this.#config.shiftPatterns.map((sp) => sp.id);
|
|
62
|
+
}
|
|
63
|
+
/** Internal rule identifiers in kebab-case. */
|
|
64
|
+
get ruleNames() {
|
|
65
|
+
return this.#config.rules.map((r) => r._rule);
|
|
66
|
+
}
|
|
67
|
+
// --------------------------------------------------------------------------
|
|
68
|
+
// Composition
|
|
69
|
+
// --------------------------------------------------------------------------
|
|
70
|
+
/**
|
|
71
|
+
* Merges schedules or members onto this schedule, returning a new
|
|
72
|
+
* immutable `Schedule`. The original is untouched.
|
|
73
|
+
*
|
|
74
|
+
* Accepts any mix of `Schedule` instances and `SchedulingMember[]` arrays.
|
|
75
|
+
*
|
|
76
|
+
* Merge semantics (when merging schedules):
|
|
77
|
+
* - Roles: union (additive)
|
|
78
|
+
* - Skills: union (additive)
|
|
79
|
+
* - Times: additive; error on name collision
|
|
80
|
+
* - Coverage: additive
|
|
81
|
+
* - Shift patterns: additive; error on ID collision
|
|
82
|
+
* - Rules: additive
|
|
83
|
+
* - Members: additive; error on duplicate ID
|
|
84
|
+
*
|
|
85
|
+
* Validation runs eagerly: role/skill disjointness, coverage targets
|
|
86
|
+
* referencing declared roles/skills, member role references, etc.
|
|
87
|
+
*/
|
|
88
|
+
with(...args) {
|
|
89
|
+
const merged = mergeConfig(this.#config, args);
|
|
90
|
+
return new Schedule(merged);
|
|
91
|
+
}
|
|
92
|
+
// --------------------------------------------------------------------------
|
|
93
|
+
// Solve / compile
|
|
94
|
+
// --------------------------------------------------------------------------
|
|
95
|
+
/**
|
|
96
|
+
* Compiles, validates, solves, and parses in one call.
|
|
97
|
+
*
|
|
98
|
+
* @param client - Solver client (e.g., `new HttpSolverClient(fetch, url)`)
|
|
99
|
+
* @param options - Date range and optional pinned assignments
|
|
100
|
+
*/
|
|
101
|
+
async solve(client, options) {
|
|
102
|
+
const compiled = this.compile(options);
|
|
103
|
+
if (!compiled.canSolve) {
|
|
104
|
+
return {
|
|
105
|
+
status: "infeasible",
|
|
106
|
+
assignments: [],
|
|
107
|
+
validation: compiled.validation,
|
|
108
|
+
};
|
|
109
|
+
}
|
|
110
|
+
const response = await client.solve(compiled.request);
|
|
111
|
+
return buildSolveResult(response, compiled, this.#config);
|
|
112
|
+
}
|
|
113
|
+
/**
|
|
114
|
+
* Diagnostic escape hatch. Compiles the schedule without solving.
|
|
115
|
+
*
|
|
116
|
+
* @param options - Date range and optional pinned assignments
|
|
117
|
+
*/
|
|
118
|
+
compile(options) {
|
|
119
|
+
if (options.pinned && options.pinned.length > 0) {
|
|
120
|
+
throw new Error("Pinned assignments are not yet supported.");
|
|
121
|
+
}
|
|
122
|
+
const modelConfig = resolveToModelConfig(this.#config, options);
|
|
123
|
+
const builder = new ModelBuilder(modelConfig);
|
|
124
|
+
const result = builder.compile();
|
|
125
|
+
return { ...result, builder };
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
// ============================================================================
|
|
129
|
+
// schedule() factory
|
|
130
|
+
// ============================================================================
|
|
131
|
+
/**
|
|
132
|
+
* Create a schedule definition.
|
|
133
|
+
*
|
|
134
|
+
* Returns an immutable {@link Schedule} that can be composed via `.with()`
|
|
135
|
+
* and solved via `.solve()`.
|
|
136
|
+
*
|
|
137
|
+
* @example
|
|
138
|
+
* ```typescript
|
|
139
|
+
* const venue = schedule({
|
|
140
|
+
* roleIds: ["waiter", "runner", "manager"],
|
|
141
|
+
* skillIds: ["senior"],
|
|
142
|
+
* times: {
|
|
143
|
+
* lunch: time({ startTime: t(12), endTime: t(15) }),
|
|
144
|
+
* dinner: time(
|
|
145
|
+
* { startTime: t(17), endTime: t(21) },
|
|
146
|
+
* { startTime: t(18), endTime: t(22), dayOfWeek: weekend },
|
|
147
|
+
* ),
|
|
148
|
+
* },
|
|
149
|
+
* coverage: [
|
|
150
|
+
* cover("lunch", "waiter", 2),
|
|
151
|
+
* cover("dinner", "waiter", 4, { dayOfWeek: weekdays }),
|
|
152
|
+
* cover("dinner", "waiter", 5, { dayOfWeek: weekend }),
|
|
153
|
+
* cover("dinner", "manager", 1),
|
|
154
|
+
* ],
|
|
155
|
+
* shiftPatterns: [
|
|
156
|
+
* shift("lunch_shift", t(11, 30), t(15)),
|
|
157
|
+
* shift("evening", t(17), t(22)),
|
|
158
|
+
* ],
|
|
159
|
+
* rules: [
|
|
160
|
+
* maxHoursPerDay(10),
|
|
161
|
+
* maxHoursPerWeek(48),
|
|
162
|
+
* minRestBetweenShifts(11),
|
|
163
|
+
* ],
|
|
164
|
+
* });
|
|
165
|
+
* ```
|
|
166
|
+
*
|
|
167
|
+
* @category Schedule Definition
|
|
168
|
+
*/
|
|
169
|
+
export function schedule(config) {
|
|
170
|
+
const merged = buildMergedConfig(config);
|
|
171
|
+
validateConfig(merged);
|
|
172
|
+
return new Schedule(merged);
|
|
173
|
+
}
|
|
174
|
+
/**
|
|
175
|
+
* Create a partial schedule for composition via `.with()`.
|
|
176
|
+
*
|
|
177
|
+
* Unlike {@link schedule}, all fields are optional. Use this for
|
|
178
|
+
* schedules that layer rules, coverage, or other config onto a
|
|
179
|
+
* complete base schedule.
|
|
180
|
+
*
|
|
181
|
+
* @example
|
|
182
|
+
* ```typescript
|
|
183
|
+
* const companyPolicy = partialSchedule({
|
|
184
|
+
* rules: [maxHoursPerWeek(40), minRestBetweenShifts(11)],
|
|
185
|
+
* });
|
|
186
|
+
*
|
|
187
|
+
* const ready = venue.with(companyPolicy, teamMembers);
|
|
188
|
+
* ```
|
|
189
|
+
*
|
|
190
|
+
* @category Schedule Definition
|
|
191
|
+
*/
|
|
192
|
+
export function partialSchedule(config) {
|
|
193
|
+
const merged = buildMergedConfig({
|
|
194
|
+
roleIds: [],
|
|
195
|
+
times: {},
|
|
196
|
+
coverage: [],
|
|
197
|
+
shiftPatterns: [],
|
|
198
|
+
...config,
|
|
199
|
+
});
|
|
200
|
+
validateConfig(merged);
|
|
201
|
+
return new Schedule(merged);
|
|
202
|
+
}
|
|
203
|
+
// ============================================================================
|
|
204
|
+
// Internal: Build merged config from user input
|
|
205
|
+
// ============================================================================
|
|
206
|
+
function buildMergedConfig(config) {
|
|
207
|
+
return {
|
|
208
|
+
roleIds: [...config.roleIds],
|
|
209
|
+
skillIds: [...(config.skillIds ?? [])],
|
|
210
|
+
times: { ...config.times },
|
|
211
|
+
coverage: [...config.coverage],
|
|
212
|
+
shiftPatterns: [...config.shiftPatterns],
|
|
213
|
+
rules: [...(config.rules ?? [])],
|
|
214
|
+
ruleFactories: config.ruleFactories ? { ...config.ruleFactories } : {},
|
|
215
|
+
members: [...(config.members ?? [])],
|
|
216
|
+
dayOfWeek: config.dayOfWeek,
|
|
217
|
+
weekStartsOn: config.weekStartsOn,
|
|
218
|
+
};
|
|
219
|
+
}
|
|
220
|
+
// ============================================================================
|
|
221
|
+
// Internal: Validate merged config
|
|
222
|
+
// ============================================================================
|
|
223
|
+
function validateConfig(config) {
|
|
224
|
+
const roles = new Set(config.roleIds);
|
|
225
|
+
const skills = new Set(config.skillIds);
|
|
226
|
+
// Validate custom rule factories don't override built-in names
|
|
227
|
+
for (const name of Object.keys(config.ruleFactories)) {
|
|
228
|
+
if (name in builtInCpsatRuleFactories) {
|
|
229
|
+
throw new Error(`Custom rule factory "${name}" conflicts with a built-in rule. Choose a different name.`);
|
|
230
|
+
}
|
|
231
|
+
}
|
|
232
|
+
// Validate role/skill disjointness
|
|
233
|
+
for (const skill of skills) {
|
|
234
|
+
if (roles.has(skill)) {
|
|
235
|
+
throw new Error(`"${skill}" is declared as both a role and a skill. Roles and skills must be disjoint.`);
|
|
236
|
+
}
|
|
237
|
+
}
|
|
238
|
+
// Validate shift pattern role references
|
|
239
|
+
for (const sp of config.shiftPatterns) {
|
|
240
|
+
if (sp.roleIds) {
|
|
241
|
+
for (const role of sp.roleIds) {
|
|
242
|
+
if (!roles.has(role)) {
|
|
243
|
+
throw new Error(`Shift pattern "${sp.id}" references unknown role "${role}". ` +
|
|
244
|
+
`Declared roles: ${[...roles].join(", ")}`);
|
|
245
|
+
}
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
// Validate coverage entries
|
|
250
|
+
for (const entry of config.coverage) {
|
|
251
|
+
validateCoverageEntry(entry, roles, skills);
|
|
252
|
+
}
|
|
253
|
+
// Validate member references
|
|
254
|
+
const memberIds = new Set();
|
|
255
|
+
for (const member of config.members) {
|
|
256
|
+
if (memberIds.has(member.id)) {
|
|
257
|
+
throw new Error(`Duplicate member ID "${member.id}".`);
|
|
258
|
+
}
|
|
259
|
+
memberIds.add(member.id);
|
|
260
|
+
if (roles.has(member.id)) {
|
|
261
|
+
throw new Error(`Member ID "${member.id}" collides with a declared role name.`);
|
|
262
|
+
}
|
|
263
|
+
if (skills.has(member.id)) {
|
|
264
|
+
throw new Error(`Member ID "${member.id}" collides with a declared skill name.`);
|
|
265
|
+
}
|
|
266
|
+
for (const role of member.roleIds) {
|
|
267
|
+
if (!roles.has(role)) {
|
|
268
|
+
throw new Error(`Member "${member.id}" references unknown role "${role}". ` +
|
|
269
|
+
`Declared roles: ${[...roles].join(", ")}`);
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
if (member.skillIds) {
|
|
273
|
+
for (const skill of member.skillIds) {
|
|
274
|
+
if (!skills.has(skill)) {
|
|
275
|
+
throw new Error(`Member "${member.id}" references unknown skill "${skill}". ` +
|
|
276
|
+
`Declared skills: ${[...skills].join(", ")}`);
|
|
277
|
+
}
|
|
278
|
+
}
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
function validateCoverageEntry(entry, roles, skills) {
|
|
283
|
+
const targets = Array.isArray(entry.target) ? entry.target : [entry.target];
|
|
284
|
+
if (Array.isArray(entry.target)) {
|
|
285
|
+
for (const target of targets) {
|
|
286
|
+
if (!roles.has(target)) {
|
|
287
|
+
throw new Error(`Coverage for "${entry.timeName}" references "${target}" in a role OR group, ` +
|
|
288
|
+
`but it is not a declared role. Declared roles: ${[...roles].join(", ")}`);
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
else {
|
|
293
|
+
if (!roles.has(entry.target) && !skills.has(entry.target)) {
|
|
294
|
+
throw new Error(`Coverage for "${entry.timeName}" references unknown target "${entry.target}". ` +
|
|
295
|
+
`Declared roles: ${[...roles].join(", ")}. ` +
|
|
296
|
+
`Declared skills: ${[...skills].join(", ")}`);
|
|
297
|
+
}
|
|
298
|
+
}
|
|
299
|
+
if (entry.options.skillIds) {
|
|
300
|
+
for (const s of entry.options.skillIds) {
|
|
301
|
+
if (!skills.has(s)) {
|
|
302
|
+
throw new Error(`Coverage for "${entry.timeName}" uses skill filter "${s}" ` +
|
|
303
|
+
`which is not a declared skill. Declared skills: ${[...skills].join(", ")}`);
|
|
304
|
+
}
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
}
|
|
308
|
+
// ============================================================================
|
|
309
|
+
// Internal: Merge logic
|
|
310
|
+
// ============================================================================
|
|
311
|
+
function mergeConfig(base, args) {
|
|
312
|
+
const result = {
|
|
313
|
+
roleIds: [...base.roleIds],
|
|
314
|
+
skillIds: [...base.skillIds],
|
|
315
|
+
times: { ...base.times },
|
|
316
|
+
coverage: [...base.coverage],
|
|
317
|
+
shiftPatterns: [...base.shiftPatterns],
|
|
318
|
+
rules: [...base.rules],
|
|
319
|
+
ruleFactories: { ...base.ruleFactories },
|
|
320
|
+
members: [...base.members],
|
|
321
|
+
dayOfWeek: base.dayOfWeek,
|
|
322
|
+
weekStartsOn: base.weekStartsOn,
|
|
323
|
+
};
|
|
324
|
+
for (const arg of args) {
|
|
325
|
+
if (arg instanceof Schedule) {
|
|
326
|
+
mergeScheduleFragment(result, arg);
|
|
327
|
+
}
|
|
328
|
+
else if (Array.isArray(arg)) {
|
|
329
|
+
mergeMembers(result, arg);
|
|
330
|
+
}
|
|
331
|
+
else {
|
|
332
|
+
throw new Error(`Unexpected argument passed to .with(): expected Schedule or SchedulingMember[], got ${typeof arg}`);
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
// Validate the merged result
|
|
336
|
+
validateConfig(result);
|
|
337
|
+
return result;
|
|
338
|
+
}
|
|
339
|
+
function mergeScheduleFragment(result, s) {
|
|
340
|
+
const other = s._getConfig();
|
|
341
|
+
// dayOfWeek: error on conflict (semantics of union vs intersection are ambiguous)
|
|
342
|
+
if (other.dayOfWeek !== undefined) {
|
|
343
|
+
if (result.dayOfWeek !== undefined) {
|
|
344
|
+
const baseSet = new Set(result.dayOfWeek);
|
|
345
|
+
const same = result.dayOfWeek.length === other.dayOfWeek.length &&
|
|
346
|
+
other.dayOfWeek.every((d) => baseSet.has(d));
|
|
347
|
+
if (!same) {
|
|
348
|
+
throw new Error("Cannot merge schedules with different dayOfWeek filters. " +
|
|
349
|
+
`Base has [${result.dayOfWeek.join(", ")}], ` +
|
|
350
|
+
`incoming has [${other.dayOfWeek.join(", ")}].`);
|
|
351
|
+
}
|
|
352
|
+
}
|
|
353
|
+
else {
|
|
354
|
+
result.dayOfWeek = other.dayOfWeek;
|
|
355
|
+
}
|
|
356
|
+
}
|
|
357
|
+
// weekStartsOn: error on conflict (only one week boundary for weekly rules)
|
|
358
|
+
if (other.weekStartsOn !== undefined) {
|
|
359
|
+
if (result.weekStartsOn !== undefined && result.weekStartsOn !== other.weekStartsOn) {
|
|
360
|
+
throw new Error("Cannot merge schedules with different weekStartsOn values. " +
|
|
361
|
+
`Base has "${result.weekStartsOn}", incoming has "${other.weekStartsOn}".`);
|
|
362
|
+
}
|
|
363
|
+
result.weekStartsOn = other.weekStartsOn;
|
|
364
|
+
}
|
|
365
|
+
// Roles: union
|
|
366
|
+
for (const role of other.roleIds) {
|
|
367
|
+
if (!result.roleIds.includes(role)) {
|
|
368
|
+
result.roleIds.push(role);
|
|
369
|
+
}
|
|
370
|
+
}
|
|
371
|
+
// Skills: union
|
|
372
|
+
for (const skill of other.skillIds) {
|
|
373
|
+
if (!result.skillIds.includes(skill)) {
|
|
374
|
+
result.skillIds.push(skill);
|
|
375
|
+
}
|
|
376
|
+
}
|
|
377
|
+
// Times: additive, error on collision
|
|
378
|
+
for (const [name, entry] of Object.entries(other.times)) {
|
|
379
|
+
if (name in result.times) {
|
|
380
|
+
throw new Error(`Time name "${name}" already exists. Cannot merge schedules with colliding time names.`);
|
|
381
|
+
}
|
|
382
|
+
result.times[name] = entry;
|
|
383
|
+
}
|
|
384
|
+
// Coverage: additive
|
|
385
|
+
result.coverage.push(...other.coverage);
|
|
386
|
+
// Shift patterns: additive, error on ID collision
|
|
387
|
+
const existingIds = new Set(result.shiftPatterns.map((sp) => sp.id));
|
|
388
|
+
for (const sp of other.shiftPatterns) {
|
|
389
|
+
if (existingIds.has(sp.id)) {
|
|
390
|
+
throw new Error(`Shift pattern ID "${sp.id}" already exists. Cannot merge schedules with colliding shift pattern IDs.`);
|
|
391
|
+
}
|
|
392
|
+
result.shiftPatterns.push(sp);
|
|
393
|
+
existingIds.add(sp.id);
|
|
394
|
+
}
|
|
395
|
+
// Rules: additive
|
|
396
|
+
result.rules.push(...other.rules);
|
|
397
|
+
// Rule factories: merge, error on collision
|
|
398
|
+
for (const [name, factory] of Object.entries(other.ruleFactories)) {
|
|
399
|
+
if (name in result.ruleFactories && result.ruleFactories[name] !== factory) {
|
|
400
|
+
throw new Error(`Rule factory "${name}" already registered. Cannot merge schedules with colliding rule factories.`);
|
|
401
|
+
}
|
|
402
|
+
result.ruleFactories[name] = factory;
|
|
403
|
+
}
|
|
404
|
+
// Members: additive, error on duplicate ID
|
|
405
|
+
const existingMemberIds = new Set(result.members.map((m) => m.id));
|
|
406
|
+
for (const member of other.members) {
|
|
407
|
+
if (existingMemberIds.has(member.id)) {
|
|
408
|
+
throw new Error(`Duplicate member ID "${member.id}". Cannot merge schedules with colliding member IDs.`);
|
|
409
|
+
}
|
|
410
|
+
result.members.push(member);
|
|
411
|
+
existingMemberIds.add(member.id);
|
|
412
|
+
}
|
|
413
|
+
}
|
|
414
|
+
function mergeMembers(result, incoming) {
|
|
415
|
+
const existingIds = new Set(result.members.map((m) => m.id));
|
|
416
|
+
for (const member of incoming) {
|
|
417
|
+
if (existingIds.has(member.id)) {
|
|
418
|
+
throw new Error(`Duplicate member ID "${member.id}". Cannot merge members with colliding IDs.`);
|
|
419
|
+
}
|
|
420
|
+
result.members.push(member);
|
|
421
|
+
existingIds.add(member.id);
|
|
422
|
+
}
|
|
423
|
+
}
|
|
424
|
+
// ============================================================================
|
|
425
|
+
// Internal: Resolve to ModelBuilderConfig
|
|
426
|
+
// ============================================================================
|
|
427
|
+
function resolveToModelConfig(config, options) {
|
|
428
|
+
const roles = new Set(config.roleIds);
|
|
429
|
+
const skills = new Set(config.skillIds);
|
|
430
|
+
const memberIds = new Set(config.members.map((m) => m.id));
|
|
431
|
+
// Build semantic time context
|
|
432
|
+
const semanticTimes = defineSemanticTimes(config.times);
|
|
433
|
+
// Convert coverage entries to semantic coverage requirements
|
|
434
|
+
const coverageReqs = buildCoverageRequirements(config.coverage, roles, skills);
|
|
435
|
+
// Resolve scheduling period with dayOfWeek filter
|
|
436
|
+
const schedulingPeriod = {
|
|
437
|
+
dateRange: options.dateRange,
|
|
438
|
+
};
|
|
439
|
+
const resolvedPeriod = applyDaysFilter(schedulingPeriod, config.dayOfWeek);
|
|
440
|
+
const days = resolveDaysFromPeriod(resolvedPeriod);
|
|
441
|
+
// Resolve coverage
|
|
442
|
+
const resolvedCoverage = semanticTimes.resolve(coverageReqs, days);
|
|
443
|
+
// Resolve rules
|
|
444
|
+
const allRules = [...config.rules];
|
|
445
|
+
// Validate pay data when cost rules are present
|
|
446
|
+
const costRuleNames = new Set([
|
|
447
|
+
"minimize-cost",
|
|
448
|
+
"day-cost-multiplier",
|
|
449
|
+
"day-cost-surcharge",
|
|
450
|
+
"time-cost-surcharge",
|
|
451
|
+
"overtime-weekly-multiplier",
|
|
452
|
+
"overtime-weekly-surcharge",
|
|
453
|
+
"overtime-daily-multiplier",
|
|
454
|
+
"overtime-daily-surcharge",
|
|
455
|
+
"overtime-tiered-multiplier",
|
|
456
|
+
]);
|
|
457
|
+
const hasCostRules = allRules.some((r) => costRuleNames.has(r._rule));
|
|
458
|
+
if (hasCostRules) {
|
|
459
|
+
const missingPay = config.members.filter((m) => !m.pay).map((m) => m.id);
|
|
460
|
+
if (missingPay.length > 0) {
|
|
461
|
+
throw new Error(`Cost rules require pay data on all members. Missing pay: ${missingPay.join(", ")}`);
|
|
462
|
+
}
|
|
463
|
+
}
|
|
464
|
+
// Sort rules so minimize-cost compiles before modifier rules
|
|
465
|
+
const sortedRules = sortCostRulesFirst(allRules);
|
|
466
|
+
const ruleConfigs = resolveRules(sortedRules, roles, skills, memberIds);
|
|
467
|
+
return {
|
|
468
|
+
members: config.members,
|
|
469
|
+
shiftPatterns: config.shiftPatterns,
|
|
470
|
+
schedulingPeriod: resolvedPeriod,
|
|
471
|
+
coverage: resolvedCoverage,
|
|
472
|
+
ruleConfigs,
|
|
473
|
+
ruleFactories: Object.keys(config.ruleFactories).length > 0
|
|
474
|
+
? { ...builtInCpsatRuleFactories, ...config.ruleFactories }
|
|
475
|
+
: undefined,
|
|
476
|
+
weekStartsOn: config.weekStartsOn,
|
|
477
|
+
};
|
|
478
|
+
}
|
|
479
|
+
// ============================================================================
|
|
480
|
+
// Internal: Build SolveResult from solver response
|
|
481
|
+
// ============================================================================
|
|
482
|
+
function mapSolverStatus(solverStatus) {
|
|
483
|
+
switch (solverStatus) {
|
|
484
|
+
case "OPTIMAL":
|
|
485
|
+
return "optimal";
|
|
486
|
+
case "FEASIBLE":
|
|
487
|
+
return "feasible";
|
|
488
|
+
case "INFEASIBLE":
|
|
489
|
+
return "infeasible";
|
|
490
|
+
case "TIMEOUT":
|
|
491
|
+
case "ERROR":
|
|
492
|
+
return "no_solution";
|
|
493
|
+
default:
|
|
494
|
+
return "no_solution";
|
|
495
|
+
}
|
|
496
|
+
}
|
|
497
|
+
function buildSolveResult(response, compiled, config) {
|
|
498
|
+
const status = mapSolverStatus(response.status);
|
|
499
|
+
const parsed = parseSolverResponse(response);
|
|
500
|
+
// Run post-solve validation when a solution exists
|
|
501
|
+
if (parsed.assignments.length > 0 && (status === "optimal" || status === "feasible")) {
|
|
502
|
+
const resolved = resolveAssignments(parsed.assignments, compiled.builder.shiftPatterns);
|
|
503
|
+
compiled.builder.reporter.analyzeSolution(response);
|
|
504
|
+
compiled.builder.validateSolution(resolved);
|
|
505
|
+
}
|
|
506
|
+
const validation = compiled.builder.reporter.getValidation();
|
|
507
|
+
const result = {
|
|
508
|
+
status,
|
|
509
|
+
assignments: parsed.assignments,
|
|
510
|
+
validation,
|
|
511
|
+
};
|
|
512
|
+
// Compute cost breakdown when cost rules are present and a solution was found
|
|
513
|
+
if (parsed.assignments.length > 0 && (status === "optimal" || status === "feasible")) {
|
|
514
|
+
const hasCostRules = config.rules.some((r) => r._rule === "minimize-cost");
|
|
515
|
+
if (hasCostRules) {
|
|
516
|
+
result.cost = calculateScheduleCost(parsed.assignments, {
|
|
517
|
+
members: config.members,
|
|
518
|
+
shiftPatterns: config.shiftPatterns,
|
|
519
|
+
rules: compiled.builder.rules,
|
|
520
|
+
});
|
|
521
|
+
}
|
|
522
|
+
}
|
|
523
|
+
return result;
|
|
524
|
+
}
|
|
525
|
+
// ============================================================================
|
|
526
|
+
// Internal: Coverage Translation
|
|
527
|
+
// ============================================================================
|
|
528
|
+
function buildCoverageRequirements(entries, roles, skills) {
|
|
529
|
+
return entries.map((entry) => {
|
|
530
|
+
// Variant form: produce a VariantCoverageRequirement
|
|
531
|
+
if (entry.variants) {
|
|
532
|
+
return buildVariantCoverageRequirement(entry, roles, skills);
|
|
533
|
+
}
|
|
534
|
+
// Simple form: produce a SemanticCoverageRequirement
|
|
535
|
+
const base = {
|
|
536
|
+
semanticTime: entry.timeName,
|
|
537
|
+
targetCount: entry.count,
|
|
538
|
+
};
|
|
539
|
+
if (entry.options.priority)
|
|
540
|
+
base.priority = entry.options.priority;
|
|
541
|
+
if (entry.options.dayOfWeek && entry.options.dayOfWeek.length > 0) {
|
|
542
|
+
base.dayOfWeek = entry.options.dayOfWeek;
|
|
543
|
+
}
|
|
544
|
+
if (entry.options.dates)
|
|
545
|
+
base.dates = entry.options.dates;
|
|
546
|
+
return buildSimpleCoverageTarget(entry, base, roles, skills);
|
|
547
|
+
});
|
|
548
|
+
}
|
|
549
|
+
/**
|
|
550
|
+
* Resolve the target (role/skill) for a simple coverage entry.
|
|
551
|
+
*/
|
|
552
|
+
function buildSimpleCoverageTarget(entry, base, roles, skills) {
|
|
553
|
+
if (Array.isArray(entry.target)) {
|
|
554
|
+
return {
|
|
555
|
+
...base,
|
|
556
|
+
roleIds: entry.target,
|
|
557
|
+
};
|
|
558
|
+
}
|
|
559
|
+
const singleTarget = entry.target;
|
|
560
|
+
if (roles.has(singleTarget)) {
|
|
561
|
+
if (entry.options.skillIds) {
|
|
562
|
+
return {
|
|
563
|
+
...base,
|
|
564
|
+
roleIds: [singleTarget],
|
|
565
|
+
skillIds: entry.options.skillIds,
|
|
566
|
+
};
|
|
567
|
+
}
|
|
568
|
+
return {
|
|
569
|
+
...base,
|
|
570
|
+
roleIds: [singleTarget],
|
|
571
|
+
};
|
|
572
|
+
}
|
|
573
|
+
if (skills.has(singleTarget)) {
|
|
574
|
+
return {
|
|
575
|
+
...base,
|
|
576
|
+
skillIds: [singleTarget],
|
|
577
|
+
};
|
|
578
|
+
}
|
|
579
|
+
throw new Error(`Coverage target "${singleTarget}" is not a declared role or skill.`);
|
|
580
|
+
}
|
|
581
|
+
/**
|
|
582
|
+
* Build a VariantCoverageRequirement from a variant-form CoverageEntry.
|
|
583
|
+
*/
|
|
584
|
+
function buildVariantCoverageRequirement(entry, roles, skills) {
|
|
585
|
+
const variants = entry.variants;
|
|
586
|
+
const resolveTarget = () => {
|
|
587
|
+
if (Array.isArray(entry.target)) {
|
|
588
|
+
return { roleIds: entry.target };
|
|
589
|
+
}
|
|
590
|
+
const singleTarget = entry.target;
|
|
591
|
+
if (roles.has(singleTarget)) {
|
|
592
|
+
return { roleIds: [singleTarget] };
|
|
593
|
+
}
|
|
594
|
+
if (skills.has(singleTarget)) {
|
|
595
|
+
return { skillIds: [singleTarget] };
|
|
596
|
+
}
|
|
597
|
+
throw new Error(`Coverage target "${singleTarget}" is not a declared role or skill.`);
|
|
598
|
+
};
|
|
599
|
+
return {
|
|
600
|
+
semanticTime: entry.timeName,
|
|
601
|
+
variants,
|
|
602
|
+
...resolveTarget(),
|
|
603
|
+
};
|
|
604
|
+
}
|
|
605
|
+
// ============================================================================
|
|
606
|
+
// Internal: Rule Translation
|
|
607
|
+
// ============================================================================
|
|
608
|
+
function resolveRules(rules, roles, skills, memberIds) {
|
|
609
|
+
const ctx = { roles, skills, memberIds };
|
|
610
|
+
return rules.map((rule) => {
|
|
611
|
+
// Rules with custom resolvers handle their own translation
|
|
612
|
+
if (rule._resolve) {
|
|
613
|
+
return rule._resolve(ctx);
|
|
614
|
+
}
|
|
615
|
+
// Default resolution: appliesTo → entity scope, dates → specificDates
|
|
616
|
+
const { _type, _rule, _resolve, appliesTo, dates, ...passthrough } = rule;
|
|
617
|
+
const entityScope = resolveAppliesTo(appliesTo, roles, skills, memberIds);
|
|
618
|
+
const resolvedDates = dates ? { specificDates: dates } : {};
|
|
619
|
+
return {
|
|
620
|
+
name: _rule,
|
|
621
|
+
...passthrough,
|
|
622
|
+
...entityScope,
|
|
623
|
+
...resolvedDates,
|
|
624
|
+
};
|
|
625
|
+
});
|
|
626
|
+
}
|
|
627
|
+
// ============================================================================
|
|
628
|
+
// Internal: Cost Rule Ordering
|
|
629
|
+
// ============================================================================
|
|
630
|
+
/**
|
|
631
|
+
* Sorts rules so that `minimize-cost` compiles before cost modifier rules.
|
|
632
|
+
*
|
|
633
|
+
* The `minimize-cost` rule must be compiled first because modifier rules
|
|
634
|
+
* (multipliers, surcharges) reference cost variables it creates.
|
|
635
|
+
* Non-cost rules retain their original relative order.
|
|
636
|
+
*/
|
|
637
|
+
function sortCostRulesFirst(rules) {
|
|
638
|
+
return rules.toSorted((a, b) => {
|
|
639
|
+
const aIsCostBase = a._rule === "minimize-cost" ? 0 : 1;
|
|
640
|
+
const bIsCostBase = b._rule === "minimize-cost" ? 0 : 1;
|
|
641
|
+
return aIsCostBase - bIsCostBase;
|
|
642
|
+
});
|
|
643
|
+
}
|
|
644
|
+
// ============================================================================
|
|
645
|
+
// Internal: Scheduling Period Helpers
|
|
646
|
+
// ============================================================================
|
|
647
|
+
function applyDaysFilter(schedulingPeriod, dayOfWeek) {
|
|
648
|
+
if (!dayOfWeek || dayOfWeek.length === 0) {
|
|
649
|
+
return schedulingPeriod;
|
|
650
|
+
}
|
|
651
|
+
const existingDays = schedulingPeriod.dayOfWeek;
|
|
652
|
+
if (!existingDays || existingDays.length === 0) {
|
|
653
|
+
return { ...schedulingPeriod, dayOfWeek: [...dayOfWeek] };
|
|
654
|
+
}
|
|
655
|
+
const existingSet = new Set(existingDays);
|
|
656
|
+
const intersected = dayOfWeek.filter((day) => existingSet.has(day));
|
|
657
|
+
return { ...schedulingPeriod, dayOfWeek: intersected };
|
|
658
|
+
}
|
|
659
|
+
//# sourceMappingURL=definition.js.map
|