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.
Files changed (231) hide show
  1. package/CHANGELOG.md +58 -0
  2. package/README.md +45 -27
  3. package/dist/client.d.ts +20 -2
  4. package/dist/client.d.ts.map +1 -1
  5. package/dist/client.js +4 -1
  6. package/dist/client.js.map +1 -1
  7. package/dist/client.types.d.ts +9 -0
  8. package/dist/client.types.d.ts.map +1 -1
  9. package/dist/client.types.js +1 -0
  10. package/dist/client.types.js.map +1 -1
  11. package/dist/cpsat/model-builder.d.ts +9 -0
  12. package/dist/cpsat/model-builder.d.ts.map +1 -1
  13. package/dist/cpsat/model-builder.js +36 -34
  14. package/dist/cpsat/model-builder.js.map +1 -1
  15. package/dist/cpsat/response.d.ts +13 -1
  16. package/dist/cpsat/response.d.ts.map +1 -1
  17. package/dist/cpsat/response.js +4 -0
  18. package/dist/cpsat/response.js.map +1 -1
  19. package/dist/cpsat/rules/cost-utils.d.ts +11 -0
  20. package/dist/cpsat/rules/cost-utils.d.ts.map +1 -0
  21. package/dist/cpsat/rules/cost-utils.js +24 -0
  22. package/dist/cpsat/rules/cost-utils.js.map +1 -0
  23. package/dist/cpsat/rules/day-cost-multiplier.d.ts.map +1 -1
  24. package/dist/cpsat/rules/day-cost-multiplier.js +3 -14
  25. package/dist/cpsat/rules/day-cost-multiplier.js.map +1 -1
  26. package/dist/cpsat/rules/day-cost-surcharge.d.ts.map +1 -1
  27. package/dist/cpsat/rules/day-cost-surcharge.js +3 -7
  28. package/dist/cpsat/rules/day-cost-surcharge.js.map +1 -1
  29. package/dist/cpsat/rules/index.d.ts +3 -0
  30. package/dist/cpsat/rules/index.d.ts.map +1 -1
  31. package/dist/cpsat/rules/index.js +3 -0
  32. package/dist/cpsat/rules/index.js.map +1 -1
  33. package/dist/cpsat/rules/max-consecutive-days.d.ts.map +1 -1
  34. package/dist/cpsat/rules/max-consecutive-days.js +16 -2
  35. package/dist/cpsat/rules/max-consecutive-days.js.map +1 -1
  36. package/dist/cpsat/rules/max-days-week.d.ts +44 -0
  37. package/dist/cpsat/rules/max-days-week.d.ts.map +1 -0
  38. package/dist/cpsat/rules/max-days-week.js +95 -0
  39. package/dist/cpsat/rules/max-days-week.js.map +1 -0
  40. package/dist/cpsat/rules/max-hours-day.d.ts.map +1 -1
  41. package/dist/cpsat/rules/max-hours-day.js +15 -2
  42. package/dist/cpsat/rules/max-hours-day.js.map +1 -1
  43. package/dist/cpsat/rules/max-hours-week.d.ts.map +1 -1
  44. package/dist/cpsat/rules/max-hours-week.js +16 -2
  45. package/dist/cpsat/rules/max-hours-week.js.map +1 -1
  46. package/dist/cpsat/rules/max-shifts-day.d.ts.map +1 -1
  47. package/dist/cpsat/rules/max-shifts-day.js +15 -2
  48. package/dist/cpsat/rules/max-shifts-day.js.map +1 -1
  49. package/dist/cpsat/rules/min-consecutive-days.d.ts.map +1 -1
  50. package/dist/cpsat/rules/min-consecutive-days.js +15 -2
  51. package/dist/cpsat/rules/min-consecutive-days.js.map +1 -1
  52. package/dist/cpsat/rules/min-days-week.d.ts +34 -0
  53. package/dist/cpsat/rules/min-days-week.d.ts.map +1 -0
  54. package/dist/cpsat/rules/min-days-week.js +84 -0
  55. package/dist/cpsat/rules/min-days-week.js.map +1 -0
  56. package/dist/cpsat/rules/min-hours-day.d.ts.map +1 -1
  57. package/dist/cpsat/rules/min-hours-day.js +15 -2
  58. package/dist/cpsat/rules/min-hours-day.js.map +1 -1
  59. package/dist/cpsat/rules/min-hours-week.d.ts.map +1 -1
  60. package/dist/cpsat/rules/min-hours-week.js +16 -2
  61. package/dist/cpsat/rules/min-hours-week.js.map +1 -1
  62. package/dist/cpsat/rules/min-rest-between-shifts.d.ts.map +1 -1
  63. package/dist/cpsat/rules/min-rest-between-shifts.js +72 -2
  64. package/dist/cpsat/rules/min-rest-between-shifts.js.map +1 -1
  65. package/dist/cpsat/rules/minimize-cost.d.ts.map +1 -1
  66. package/dist/cpsat/rules/minimize-cost.js +2 -23
  67. package/dist/cpsat/rules/minimize-cost.js.map +1 -1
  68. package/dist/cpsat/rules/must-assign.d.ts +49 -0
  69. package/dist/cpsat/rules/must-assign.d.ts.map +1 -0
  70. package/dist/cpsat/rules/must-assign.js +86 -0
  71. package/dist/cpsat/rules/must-assign.js.map +1 -0
  72. package/dist/cpsat/rules/overtime-daily-multiplier.d.ts.map +1 -1
  73. package/dist/cpsat/rules/overtime-daily-multiplier.js +1 -12
  74. package/dist/cpsat/rules/overtime-daily-multiplier.js.map +1 -1
  75. package/dist/cpsat/rules/overtime-daily-surcharge.d.ts.map +1 -1
  76. package/dist/cpsat/rules/overtime-daily-surcharge.js +1 -5
  77. package/dist/cpsat/rules/overtime-daily-surcharge.js.map +1 -1
  78. package/dist/cpsat/rules/overtime-tiered-multiplier.d.ts +5 -1
  79. package/dist/cpsat/rules/overtime-tiered-multiplier.d.ts.map +1 -1
  80. package/dist/cpsat/rules/overtime-tiered-multiplier.js +1 -12
  81. package/dist/cpsat/rules/overtime-tiered-multiplier.js.map +1 -1
  82. package/dist/cpsat/rules/overtime-weekly-multiplier.d.ts.map +1 -1
  83. package/dist/cpsat/rules/overtime-weekly-multiplier.js +1 -12
  84. package/dist/cpsat/rules/overtime-weekly-multiplier.js.map +1 -1
  85. package/dist/cpsat/rules/overtime-weekly-surcharge.d.ts.map +1 -1
  86. package/dist/cpsat/rules/overtime-weekly-surcharge.js +1 -5
  87. package/dist/cpsat/rules/overtime-weekly-surcharge.js.map +1 -1
  88. package/dist/cpsat/rules/registry.d.ts +28 -2
  89. package/dist/cpsat/rules/registry.d.ts.map +1 -1
  90. package/dist/cpsat/rules/registry.js +4 -1
  91. package/dist/cpsat/rules/registry.js.map +1 -1
  92. package/dist/cpsat/rules/resolver.js +2 -2
  93. package/dist/cpsat/rules/resolver.js.map +1 -1
  94. package/dist/cpsat/rules/rules.types.d.ts +3 -0
  95. package/dist/cpsat/rules/rules.types.d.ts.map +1 -1
  96. package/dist/cpsat/rules/scope.types.d.ts +18 -1
  97. package/dist/cpsat/rules/scope.types.d.ts.map +1 -1
  98. package/dist/cpsat/rules/scope.types.js +59 -16
  99. package/dist/cpsat/rules/scope.types.js.map +1 -1
  100. package/dist/cpsat/rules/time-cost-surcharge.d.ts.map +1 -1
  101. package/dist/cpsat/rules/time-cost-surcharge.js +2 -1
  102. package/dist/cpsat/rules/time-cost-surcharge.js.map +1 -1
  103. package/dist/cpsat/rules/time-off.d.ts.map +1 -1
  104. package/dist/cpsat/rules/time-off.js +6 -3
  105. package/dist/cpsat/rules/time-off.js.map +1 -1
  106. package/dist/cpsat/semantic-time.d.ts +44 -42
  107. package/dist/cpsat/semantic-time.d.ts.map +1 -1
  108. package/dist/cpsat/semantic-time.js +64 -46
  109. package/dist/cpsat/semantic-time.js.map +1 -1
  110. package/dist/cpsat/types.d.ts +37 -27
  111. package/dist/cpsat/types.d.ts.map +1 -1
  112. package/dist/cpsat/utils.d.ts.map +1 -1
  113. package/dist/cpsat/utils.js +7 -12
  114. package/dist/cpsat/utils.js.map +1 -1
  115. package/dist/cpsat/validation-reporter.d.ts +10 -7
  116. package/dist/cpsat/validation-reporter.d.ts.map +1 -1
  117. package/dist/cpsat/validation-reporter.js +44 -72
  118. package/dist/cpsat/validation-reporter.js.map +1 -1
  119. package/dist/cpsat/validation.types.d.ts +54 -44
  120. package/dist/cpsat/validation.types.d.ts.map +1 -1
  121. package/dist/cpsat/validation.types.js +15 -10
  122. package/dist/cpsat/validation.types.js.map +1 -1
  123. package/dist/datetime.utils.d.ts +3 -203
  124. package/dist/datetime.utils.d.ts.map +1 -1
  125. package/dist/datetime.utils.js +1 -288
  126. package/dist/datetime.utils.js.map +1 -1
  127. package/dist/index.d.ts +14 -83
  128. package/dist/index.d.ts.map +1 -1
  129. package/dist/index.js +11 -83
  130. package/dist/index.js.map +1 -1
  131. package/dist/schedule/cost.d.ts +204 -0
  132. package/dist/schedule/cost.d.ts.map +1 -0
  133. package/dist/schedule/cost.js +187 -0
  134. package/dist/schedule/cost.js.map +1 -0
  135. package/dist/schedule/coverage.d.ts +85 -0
  136. package/dist/schedule/coverage.d.ts.map +1 -0
  137. package/dist/schedule/coverage.js +33 -0
  138. package/dist/schedule/coverage.js.map +1 -0
  139. package/dist/schedule/definition.d.ts +227 -0
  140. package/dist/schedule/definition.d.ts.map +1 -0
  141. package/dist/schedule/definition.js +659 -0
  142. package/dist/schedule/definition.js.map +1 -0
  143. package/dist/schedule/index.d.ts +67 -0
  144. package/dist/schedule/index.d.ts.map +1 -0
  145. package/dist/schedule/index.js +69 -0
  146. package/dist/schedule/index.js.map +1 -0
  147. package/dist/schedule/rules.d.ts +353 -0
  148. package/dist/schedule/rules.d.ts.map +1 -0
  149. package/dist/schedule/rules.js +352 -0
  150. package/dist/schedule/rules.js.map +1 -0
  151. package/dist/schedule/shift-patterns.d.ts +34 -0
  152. package/dist/schedule/shift-patterns.d.ts.map +1 -0
  153. package/dist/schedule/shift-patterns.js +41 -0
  154. package/dist/schedule/shift-patterns.js.map +1 -0
  155. package/dist/schedule/time-periods.d.ts +69 -0
  156. package/dist/schedule/time-periods.d.ts.map +1 -0
  157. package/dist/schedule/time-periods.js +91 -0
  158. package/dist/schedule/time-periods.js.map +1 -0
  159. package/dist/types.d.ts +14 -78
  160. package/dist/types.d.ts.map +1 -1
  161. package/dist/types.js.map +1 -1
  162. package/package.json +4 -9
  163. package/solver/src/solver/app.py +1 -1
  164. package/solver/src/solver/solver.py +7 -4
  165. package/src/client.ts +6 -8
  166. package/src/client.types.ts +9 -0
  167. package/src/cpsat/model-builder.ts +44 -35
  168. package/src/cpsat/response.ts +13 -1
  169. package/src/cpsat/rules/cost-utils.ts +25 -0
  170. package/src/cpsat/rules/day-cost-multiplier.ts +3 -14
  171. package/src/cpsat/rules/day-cost-surcharge.ts +3 -8
  172. package/src/cpsat/rules/index.ts +3 -0
  173. package/src/cpsat/rules/max-consecutive-days.ts +17 -0
  174. package/src/cpsat/rules/max-days-week.ts +143 -0
  175. package/src/cpsat/rules/max-hours-day.ts +21 -1
  176. package/src/cpsat/rules/max-hours-week.ts +22 -1
  177. package/src/cpsat/rules/max-shifts-day.ts +21 -1
  178. package/src/cpsat/rules/min-consecutive-days.ts +16 -1
  179. package/src/cpsat/rules/min-days-week.ts +120 -0
  180. package/src/cpsat/rules/min-hours-day.ts +16 -1
  181. package/src/cpsat/rules/min-hours-week.ts +17 -1
  182. package/src/cpsat/rules/min-rest-between-shifts.ts +92 -2
  183. package/src/cpsat/rules/minimize-cost.ts +2 -29
  184. package/src/cpsat/rules/must-assign.ts +108 -0
  185. package/src/cpsat/rules/overtime-daily-multiplier.ts +1 -12
  186. package/src/cpsat/rules/overtime-daily-surcharge.ts +1 -6
  187. package/src/cpsat/rules/overtime-tiered-multiplier.ts +6 -13
  188. package/src/cpsat/rules/overtime-weekly-multiplier.ts +1 -12
  189. package/src/cpsat/rules/overtime-weekly-surcharge.ts +1 -6
  190. package/src/cpsat/rules/registry.ts +8 -2
  191. package/src/cpsat/rules/resolver.ts +2 -2
  192. package/src/cpsat/rules/rules.types.ts +3 -0
  193. package/src/cpsat/rules/scope.types.ts +73 -20
  194. package/src/cpsat/rules/time-cost-surcharge.ts +2 -1
  195. package/src/cpsat/rules/time-off.ts +6 -2
  196. package/src/cpsat/semantic-time.ts +115 -91
  197. package/src/cpsat/types.ts +37 -27
  198. package/src/cpsat/utils.ts +8 -12
  199. package/src/cpsat/validation-reporter.ts +51 -82
  200. package/src/cpsat/validation.types.ts +72 -47
  201. package/src/datetime.utils.ts +3 -334
  202. package/src/index.ts +35 -107
  203. package/src/schedule/cost.ts +242 -0
  204. package/src/schedule/coverage.ts +135 -0
  205. package/src/schedule/definition.ts +958 -0
  206. package/src/schedule/index.ts +112 -0
  207. package/src/schedule/rules.ts +529 -0
  208. package/src/schedule/shift-patterns.ts +46 -0
  209. package/src/schedule/time-periods.ts +110 -0
  210. package/src/types.ts +14 -88
  211. package/dist/errors.d.ts +0 -12
  212. package/dist/errors.d.ts.map +0 -1
  213. package/dist/errors.js +0 -17
  214. package/dist/errors.js.map +0 -1
  215. package/dist/llms.d.ts +0 -2
  216. package/dist/llms.d.ts.map +0 -1
  217. package/dist/llms.js +0 -3
  218. package/dist/llms.js.map +0 -1
  219. package/dist/schedule.d.ts +0 -724
  220. package/dist/schedule.d.ts.map +0 -1
  221. package/dist/schedule.js +0 -899
  222. package/dist/schedule.js.map +0 -1
  223. package/dist/validation.d.ts +0 -105
  224. package/dist/validation.d.ts.map +0 -1
  225. package/dist/validation.js +0 -130
  226. package/dist/validation.js.map +0 -1
  227. package/llms.txt +0 -925
  228. package/src/errors.ts +0 -17
  229. package/src/llms.ts +0 -3
  230. package/src/schedule.ts +0 -1419
  231. 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