dabke 0.78.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 (194) hide show
  1. package/CHANGELOG.md +120 -0
  2. package/LICENSE +21 -0
  3. package/README.md +187 -0
  4. package/dist/client.d.ts +14 -0
  5. package/dist/client.d.ts.map +1 -0
  6. package/dist/client.js +42 -0
  7. package/dist/client.js.map +1 -0
  8. package/dist/client.schemas.d.ts +250 -0
  9. package/dist/client.schemas.d.ts.map +1 -0
  10. package/dist/client.schemas.js +137 -0
  11. package/dist/client.schemas.js.map +1 -0
  12. package/dist/client.types.d.ts +34 -0
  13. package/dist/client.types.d.ts.map +1 -0
  14. package/dist/client.types.js +18 -0
  15. package/dist/client.types.js.map +1 -0
  16. package/dist/cpsat/model-builder.d.ts +128 -0
  17. package/dist/cpsat/model-builder.d.ts.map +1 -0
  18. package/dist/cpsat/model-builder.js +640 -0
  19. package/dist/cpsat/model-builder.js.map +1 -0
  20. package/dist/cpsat/response.d.ts +74 -0
  21. package/dist/cpsat/response.d.ts.map +1 -0
  22. package/dist/cpsat/response.js +92 -0
  23. package/dist/cpsat/response.js.map +1 -0
  24. package/dist/cpsat/rules/assign-together.d.ts +23 -0
  25. package/dist/cpsat/rules/assign-together.d.ts.map +1 -0
  26. package/dist/cpsat/rules/assign-together.js +78 -0
  27. package/dist/cpsat/rules/assign-together.js.map +1 -0
  28. package/dist/cpsat/rules/employee-assignment-priority.d.ts +64 -0
  29. package/dist/cpsat/rules/employee-assignment-priority.d.ts.map +1 -0
  30. package/dist/cpsat/rules/employee-assignment-priority.js +151 -0
  31. package/dist/cpsat/rules/employee-assignment-priority.js.map +1 -0
  32. package/dist/cpsat/rules/index.d.ts +13 -0
  33. package/dist/cpsat/rules/index.d.ts.map +1 -0
  34. package/dist/cpsat/rules/index.js +13 -0
  35. package/dist/cpsat/rules/index.js.map +1 -0
  36. package/dist/cpsat/rules/location-preference.d.ts +29 -0
  37. package/dist/cpsat/rules/location-preference.d.ts.map +1 -0
  38. package/dist/cpsat/rules/location-preference.js +59 -0
  39. package/dist/cpsat/rules/location-preference.js.map +1 -0
  40. package/dist/cpsat/rules/max-consecutive-days.d.ts +28 -0
  41. package/dist/cpsat/rules/max-consecutive-days.d.ts.map +1 -0
  42. package/dist/cpsat/rules/max-consecutive-days.js +70 -0
  43. package/dist/cpsat/rules/max-consecutive-days.js.map +1 -0
  44. package/dist/cpsat/rules/max-hours-day.d.ts +57 -0
  45. package/dist/cpsat/rules/max-hours-day.d.ts.map +1 -0
  46. package/dist/cpsat/rules/max-hours-day.js +159 -0
  47. package/dist/cpsat/rules/max-hours-day.js.map +1 -0
  48. package/dist/cpsat/rules/max-hours-week.d.ts +62 -0
  49. package/dist/cpsat/rules/max-hours-week.d.ts.map +1 -0
  50. package/dist/cpsat/rules/max-hours-week.js +169 -0
  51. package/dist/cpsat/rules/max-hours-week.js.map +1 -0
  52. package/dist/cpsat/rules/max-shifts-day.d.ts +69 -0
  53. package/dist/cpsat/rules/max-shifts-day.d.ts.map +1 -0
  54. package/dist/cpsat/rules/max-shifts-day.js +170 -0
  55. package/dist/cpsat/rules/max-shifts-day.js.map +1 -0
  56. package/dist/cpsat/rules/min-consecutive-days.d.ts +29 -0
  57. package/dist/cpsat/rules/min-consecutive-days.d.ts.map +1 -0
  58. package/dist/cpsat/rules/min-consecutive-days.js +104 -0
  59. package/dist/cpsat/rules/min-consecutive-days.js.map +1 -0
  60. package/dist/cpsat/rules/min-hours-day.d.ts +28 -0
  61. package/dist/cpsat/rules/min-hours-day.d.ts.map +1 -0
  62. package/dist/cpsat/rules/min-hours-day.js +61 -0
  63. package/dist/cpsat/rules/min-hours-day.js.map +1 -0
  64. package/dist/cpsat/rules/min-hours-week.d.ts +29 -0
  65. package/dist/cpsat/rules/min-hours-week.d.ts.map +1 -0
  66. package/dist/cpsat/rules/min-hours-week.js +68 -0
  67. package/dist/cpsat/rules/min-hours-week.js.map +1 -0
  68. package/dist/cpsat/rules/min-rest-between-shifts.d.ts +28 -0
  69. package/dist/cpsat/rules/min-rest-between-shifts.d.ts.map +1 -0
  70. package/dist/cpsat/rules/min-rest-between-shifts.js +95 -0
  71. package/dist/cpsat/rules/min-rest-between-shifts.js.map +1 -0
  72. package/dist/cpsat/rules/registry.d.ts +7 -0
  73. package/dist/cpsat/rules/registry.d.ts.map +1 -0
  74. package/dist/cpsat/rules/registry.js +28 -0
  75. package/dist/cpsat/rules/registry.js.map +1 -0
  76. package/dist/cpsat/rules/resolver.d.ts +31 -0
  77. package/dist/cpsat/rules/resolver.d.ts.map +1 -0
  78. package/dist/cpsat/rules/resolver.js +124 -0
  79. package/dist/cpsat/rules/resolver.js.map +1 -0
  80. package/dist/cpsat/rules/rules.types.d.ts +32 -0
  81. package/dist/cpsat/rules/rules.types.d.ts.map +1 -0
  82. package/dist/cpsat/rules/rules.types.js +2 -0
  83. package/dist/cpsat/rules/rules.types.js.map +1 -0
  84. package/dist/cpsat/rules/scoping.d.ts +129 -0
  85. package/dist/cpsat/rules/scoping.d.ts.map +1 -0
  86. package/dist/cpsat/rules/scoping.js +190 -0
  87. package/dist/cpsat/rules/scoping.js.map +1 -0
  88. package/dist/cpsat/rules/time-off.d.ts +78 -0
  89. package/dist/cpsat/rules/time-off.d.ts.map +1 -0
  90. package/dist/cpsat/rules/time-off.js +261 -0
  91. package/dist/cpsat/rules/time-off.js.map +1 -0
  92. package/dist/cpsat/rules.d.ts +5 -0
  93. package/dist/cpsat/rules.d.ts.map +1 -0
  94. package/dist/cpsat/rules.js +4 -0
  95. package/dist/cpsat/rules.js.map +1 -0
  96. package/dist/cpsat/semantic-time.d.ts +198 -0
  97. package/dist/cpsat/semantic-time.d.ts.map +1 -0
  98. package/dist/cpsat/semantic-time.js +222 -0
  99. package/dist/cpsat/semantic-time.js.map +1 -0
  100. package/dist/cpsat/types.d.ts +180 -0
  101. package/dist/cpsat/types.d.ts.map +1 -0
  102. package/dist/cpsat/types.js +2 -0
  103. package/dist/cpsat/types.js.map +1 -0
  104. package/dist/cpsat/utils.d.ts +47 -0
  105. package/dist/cpsat/utils.d.ts.map +1 -0
  106. package/dist/cpsat/utils.js +92 -0
  107. package/dist/cpsat/utils.js.map +1 -0
  108. package/dist/cpsat/validation-reporter.d.ts +54 -0
  109. package/dist/cpsat/validation-reporter.d.ts.map +1 -0
  110. package/dist/cpsat/validation-reporter.js +261 -0
  111. package/dist/cpsat/validation-reporter.js.map +1 -0
  112. package/dist/cpsat/validation.types.d.ts +141 -0
  113. package/dist/cpsat/validation.types.d.ts.map +1 -0
  114. package/dist/cpsat/validation.types.js +14 -0
  115. package/dist/cpsat/validation.types.js.map +1 -0
  116. package/dist/datetime.utils.d.ts +245 -0
  117. package/dist/datetime.utils.d.ts.map +1 -0
  118. package/dist/datetime.utils.js +372 -0
  119. package/dist/datetime.utils.js.map +1 -0
  120. package/dist/errors.d.ts +12 -0
  121. package/dist/errors.d.ts.map +1 -0
  122. package/dist/errors.js +17 -0
  123. package/dist/errors.js.map +1 -0
  124. package/dist/index.d.ts +112 -0
  125. package/dist/index.d.ts.map +1 -0
  126. package/dist/index.js +116 -0
  127. package/dist/index.js.map +1 -0
  128. package/dist/llms.d.ts +5 -0
  129. package/dist/llms.d.ts.map +1 -0
  130. package/dist/llms.js +8 -0
  131. package/dist/llms.js.map +1 -0
  132. package/dist/testing/index.d.ts +12 -0
  133. package/dist/testing/index.d.ts.map +1 -0
  134. package/dist/testing/index.js +11 -0
  135. package/dist/testing/index.js.map +1 -0
  136. package/dist/testing/solver-container.d.ts +49 -0
  137. package/dist/testing/solver-container.d.ts.map +1 -0
  138. package/dist/testing/solver-container.js +127 -0
  139. package/dist/testing/solver-container.js.map +1 -0
  140. package/dist/types.d.ts +155 -0
  141. package/dist/types.d.ts.map +1 -0
  142. package/dist/types.js +20 -0
  143. package/dist/types.js.map +1 -0
  144. package/dist/validation.d.ts +105 -0
  145. package/dist/validation.d.ts.map +1 -0
  146. package/dist/validation.js +130 -0
  147. package/dist/validation.js.map +1 -0
  148. package/llms.txt +2188 -0
  149. package/package.json +76 -0
  150. package/solver/Dockerfile +31 -0
  151. package/solver/README.md +23 -0
  152. package/solver/pyproject.toml +28 -0
  153. package/solver/src/solver/__init__.py +1 -0
  154. package/solver/src/solver/app.py +24 -0
  155. package/solver/src/solver/models.py +120 -0
  156. package/solver/src/solver/solver.py +359 -0
  157. package/solver/tests/test_solver.py +156 -0
  158. package/solver/uv.lock +661 -0
  159. package/src/client.schemas.ts +163 -0
  160. package/src/client.ts +67 -0
  161. package/src/client.types.ts +66 -0
  162. package/src/cpsat/model-builder.ts +858 -0
  163. package/src/cpsat/response.ts +130 -0
  164. package/src/cpsat/rules/assign-together.ts +96 -0
  165. package/src/cpsat/rules/employee-assignment-priority.ts +182 -0
  166. package/src/cpsat/rules/index.ts +12 -0
  167. package/src/cpsat/rules/location-preference.ts +68 -0
  168. package/src/cpsat/rules/max-consecutive-days.ts +98 -0
  169. package/src/cpsat/rules/max-hours-day.ts +187 -0
  170. package/src/cpsat/rules/max-hours-week.ts +197 -0
  171. package/src/cpsat/rules/max-shifts-day.ts +198 -0
  172. package/src/cpsat/rules/min-consecutive-days.ts +140 -0
  173. package/src/cpsat/rules/min-hours-day.ts +69 -0
  174. package/src/cpsat/rules/min-hours-week.ts +77 -0
  175. package/src/cpsat/rules/min-rest-between-shifts.ts +121 -0
  176. package/src/cpsat/rules/registry.ts +49 -0
  177. package/src/cpsat/rules/resolver.ts +181 -0
  178. package/src/cpsat/rules/rules.types.ts +41 -0
  179. package/src/cpsat/rules/scoping.ts +340 -0
  180. package/src/cpsat/rules/time-off.ts +336 -0
  181. package/src/cpsat/rules.ts +27 -0
  182. package/src/cpsat/semantic-time.ts +463 -0
  183. package/src/cpsat/types.ts +194 -0
  184. package/src/cpsat/utils.ts +105 -0
  185. package/src/cpsat/validation-reporter.ts +366 -0
  186. package/src/cpsat/validation.types.ts +185 -0
  187. package/src/datetime.utils.ts +426 -0
  188. package/src/errors.ts +17 -0
  189. package/src/index.ts +289 -0
  190. package/src/llms.ts +9 -0
  191. package/src/testing/index.ts +12 -0
  192. package/src/testing/solver-container.ts +172 -0
  193. package/src/types.ts +191 -0
  194. package/src/validation.ts +188 -0
@@ -0,0 +1,69 @@
1
+ import * as z from "zod";
2
+ import type { CompilationRule } from "../model-builder.js";
3
+ declare const MaxShiftsDaySchema: z.ZodObject<{
4
+ shifts: z.ZodNumber;
5
+ priority: z.ZodUnion<readonly [z.ZodLiteral<"LOW">, z.ZodLiteral<"MEDIUM">, z.ZodLiteral<"HIGH">, z.ZodLiteral<"MANDATORY">]>;
6
+ } & {
7
+ employeeIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
8
+ } & {
9
+ roleIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
10
+ } & {
11
+ skillIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
12
+ } & {
13
+ dateRange: z.ZodOptional<z.ZodObject<{
14
+ start: z.ZodString;
15
+ end: z.ZodString;
16
+ }, z.core.$strip>>;
17
+ } & {
18
+ specificDates: z.ZodOptional<z.ZodArray<z.ZodString>>;
19
+ } & {
20
+ dayOfWeek: z.ZodOptional<z.ZodArray<z.ZodUnion<readonly [z.ZodLiteral<"monday">, z.ZodLiteral<"tuesday">, z.ZodLiteral<"wednesday">, z.ZodLiteral<"thursday">, z.ZodLiteral<"friday">, z.ZodLiteral<"saturday">, z.ZodLiteral<"sunday">]>>>;
21
+ } & {
22
+ recurringPeriods: z.ZodOptional<z.ZodArray<z.ZodObject<{
23
+ name: z.ZodString;
24
+ startMonth: z.ZodNumber;
25
+ startDay: z.ZodNumber;
26
+ endMonth: z.ZodNumber;
27
+ endDay: z.ZodNumber;
28
+ }, z.core.$strip>>>;
29
+ }, z.core.$strip>;
30
+ export type MaxShiftsDayConfig = z.infer<typeof MaxShiftsDaySchema>;
31
+ /**
32
+ * Limits how many shifts a person can work in a single day.
33
+ *
34
+ * This rule controls the maximum number of distinct shift assignments per day,
35
+ * regardless of shift duration. For limiting total hours worked, use `max-hours-day`.
36
+ *
37
+ * Supports entity scoping (people, roles, skills) and time scoping
38
+ * (date ranges, specific dates, days of week, recurring periods).
39
+ *
40
+ * @example Limit to one shift per day (common for most schedules)
41
+ * ```ts
42
+ * createMaxShiftsDayRule({
43
+ * shifts: 1,
44
+ * priority: "MANDATORY",
45
+ * });
46
+ * ```
47
+ *
48
+ * @example Allow up to two shifts per day for part-time workers
49
+ * ```ts
50
+ * createMaxShiftsDayRule({
51
+ * roleIds: ["part-time"],
52
+ * shifts: 2,
53
+ * priority: "HIGH",
54
+ * });
55
+ * ```
56
+ *
57
+ * @example Students can work 2 shifts on weekends only
58
+ * ```ts
59
+ * createMaxShiftsDayRule({
60
+ * roleIds: ["student"],
61
+ * shifts: 2,
62
+ * dayOfWeek: ["saturday", "sunday"],
63
+ * priority: "MANDATORY",
64
+ * });
65
+ * ```
66
+ */
67
+ export declare function createMaxShiftsDayRule(config: MaxShiftsDayConfig): CompilationRule;
68
+ export {};
69
+ //# sourceMappingURL=max-shifts-day.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"max-shifts-day.d.ts","sourceRoot":"","sources":["../../../src/cpsat/rules/max-shifts-day.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,CAAC,MAAM,KAAK,CAAC;AACzB,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AAM3D,QAAA,MAAM,kBAAkB;;;;;;;;;;;;;;;;;;;;;;;;;;iBAcvB,CAAC;AAEF,MAAM,MAAM,kBAAkB,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,kBAAkB,CAAC,CAAC;AAEpE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAmCG;AACH,wBAAgB,sBAAsB,CAAC,MAAM,EAAE,kBAAkB,GAAG,eAAe,CAmClF"}
@@ -0,0 +1,170 @@
1
+ import * as z from "zod";
2
+ import { parseDayString, priorityToPenalty } from "../utils.js";
3
+ import { normalizeScope, withScopes } from "./scoping.js";
4
+ const MaxShiftsDaySchema = withScopes(z.object({
5
+ shifts: z.number().int().min(1),
6
+ priority: z.union([
7
+ z.literal("LOW"),
8
+ z.literal("MEDIUM"),
9
+ z.literal("HIGH"),
10
+ z.literal("MANDATORY"),
11
+ ]),
12
+ }), {
13
+ entities: ["employees", "roles", "skills"],
14
+ times: ["dateRange", "specificDates", "dayOfWeek", "recurring"],
15
+ });
16
+ /**
17
+ * Limits how many shifts a person can work in a single day.
18
+ *
19
+ * This rule controls the maximum number of distinct shift assignments per day,
20
+ * regardless of shift duration. For limiting total hours worked, use `max-hours-day`.
21
+ *
22
+ * Supports entity scoping (people, roles, skills) and time scoping
23
+ * (date ranges, specific dates, days of week, recurring periods).
24
+ *
25
+ * @example Limit to one shift per day (common for most schedules)
26
+ * ```ts
27
+ * createMaxShiftsDayRule({
28
+ * shifts: 1,
29
+ * priority: "MANDATORY",
30
+ * });
31
+ * ```
32
+ *
33
+ * @example Allow up to two shifts per day for part-time workers
34
+ * ```ts
35
+ * createMaxShiftsDayRule({
36
+ * roleIds: ["part-time"],
37
+ * shifts: 2,
38
+ * priority: "HIGH",
39
+ * });
40
+ * ```
41
+ *
42
+ * @example Students can work 2 shifts on weekends only
43
+ * ```ts
44
+ * createMaxShiftsDayRule({
45
+ * roleIds: ["student"],
46
+ * shifts: 2,
47
+ * dayOfWeek: ["saturday", "sunday"],
48
+ * priority: "MANDATORY",
49
+ * });
50
+ * ```
51
+ */
52
+ export function createMaxShiftsDayRule(config) {
53
+ const parsed = MaxShiftsDaySchema.parse(config);
54
+ const { shifts, priority } = parsed;
55
+ return {
56
+ compile(b) {
57
+ const scope = normalizeScope(parsed, b.employees);
58
+ const targetEmployees = resolveEmployees(scope, b.employees);
59
+ const activeDays = resolveActiveDays(scope, b.days);
60
+ if (targetEmployees.length === 0 || activeDays.length === 0)
61
+ return;
62
+ for (const emp of targetEmployees) {
63
+ for (const day of activeDays) {
64
+ const terms = [];
65
+ for (const pattern of b.shiftPatterns) {
66
+ if (!b.canAssign(emp, pattern))
67
+ continue;
68
+ if (!b.patternAvailableOnDay(pattern, day))
69
+ continue;
70
+ terms.push({
71
+ var: b.assignment(emp.id, pattern.id, day),
72
+ coeff: 1,
73
+ });
74
+ }
75
+ if (terms.length === 0)
76
+ continue;
77
+ if (priority === "MANDATORY") {
78
+ b.addLinear(terms, "<=", shifts);
79
+ }
80
+ else {
81
+ b.addSoftLinear(terms, "<=", shifts, priorityToPenalty(priority));
82
+ }
83
+ }
84
+ }
85
+ },
86
+ };
87
+ }
88
+ function resolveEmployees(scope, employees) {
89
+ const entity = scope.entity;
90
+ switch (entity.type) {
91
+ case "employees":
92
+ return employees.filter((e) => entity.employeeIds.includes(e.id));
93
+ case "roles":
94
+ return employees.filter((e) => e.roleIds.some((r) => entity.roleIds.includes(r)));
95
+ case "skills":
96
+ return employees.filter((e) => e.skillIds?.some((s) => entity.skillIds.includes(s)));
97
+ case "global":
98
+ default:
99
+ return employees;
100
+ }
101
+ }
102
+ function getDayOfWeekName(dayIndex) {
103
+ const names = {
104
+ 0: "sunday",
105
+ 1: "monday",
106
+ 2: "tuesday",
107
+ 3: "wednesday",
108
+ 4: "thursday",
109
+ 5: "friday",
110
+ 6: "saturday",
111
+ };
112
+ return names[dayIndex % 7] ?? "sunday";
113
+ }
114
+ function resolveActiveDays(scope, allDays) {
115
+ const timeScope = scope.time;
116
+ if (!timeScope) {
117
+ return allDays;
118
+ }
119
+ switch (timeScope.type) {
120
+ case "always":
121
+ return allDays;
122
+ case "dateRange": {
123
+ const start = timeScope.start;
124
+ const end = timeScope.end;
125
+ return allDays.filter((day) => day >= start && day <= end);
126
+ }
127
+ case "specificDates":
128
+ return allDays.filter((day) => timeScope.dates.includes(day));
129
+ case "dayOfWeek": {
130
+ const targetDays = new Set(timeScope.days);
131
+ return allDays.filter((day) => {
132
+ const date = parseDayString(day);
133
+ const dayName = getDayOfWeekName(date.getUTCDay());
134
+ return targetDays.has(dayName);
135
+ });
136
+ }
137
+ case "recurring": {
138
+ return allDays.filter((day) => {
139
+ const date = parseDayString(day);
140
+ const month = date.getUTCMonth() + 1;
141
+ const dayOfMonth = date.getUTCDate();
142
+ return timeScope.periods.some((period) => isDateInRecurringPeriod(month, dayOfMonth, period));
143
+ });
144
+ }
145
+ default:
146
+ return allDays;
147
+ }
148
+ }
149
+ function isDateInRecurringPeriod(month, dayOfMonth, period) {
150
+ const { startMonth, startDay, endMonth, endDay } = period;
151
+ if (startMonth <= endMonth) {
152
+ if (month < startMonth || month > endMonth)
153
+ return false;
154
+ if (month === startMonth && dayOfMonth < startDay)
155
+ return false;
156
+ if (month === endMonth && dayOfMonth > endDay)
157
+ return false;
158
+ return true;
159
+ }
160
+ else {
161
+ if (month > endMonth && month < startMonth)
162
+ return false;
163
+ if (month === startMonth && dayOfMonth < startDay)
164
+ return false;
165
+ if (month === endMonth && dayOfMonth > endDay)
166
+ return false;
167
+ return true;
168
+ }
169
+ }
170
+ //# sourceMappingURL=max-shifts-day.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"max-shifts-day.js","sourceRoot":"","sources":["../../../src/cpsat/rules/max-shifts-day.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,29 @@
1
+ import * as z from "zod";
2
+ import type { CompilationRule } from "../model-builder.js";
3
+ declare const MinConsecutiveDaysSchema: z.ZodObject<{
4
+ days: z.ZodNumber;
5
+ priority: z.ZodUnion<readonly [z.ZodLiteral<"LOW">, z.ZodLiteral<"MEDIUM">, z.ZodLiteral<"HIGH">, z.ZodLiteral<"MANDATORY">]>;
6
+ } & {
7
+ employeeIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
8
+ } & {
9
+ roleIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
10
+ } & {
11
+ skillIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
12
+ }, z.core.$strip>;
13
+ export type MinConsecutiveDaysConfig = z.infer<typeof MinConsecutiveDaysSchema>;
14
+ /**
15
+ * Requires that once a person starts working, they continue for a minimum
16
+ * number of consecutive days.
17
+ *
18
+ * @example
19
+ * ```ts
20
+ * const rule = createMinConsecutiveDaysRule({
21
+ * days: 3,
22
+ * priority: "MANDATORY",
23
+ * });
24
+ * builder = new ModelBuilder({ ...config, rules: [rule] });
25
+ * ```
26
+ */
27
+ export declare function createMinConsecutiveDaysRule(config: MinConsecutiveDaysConfig): CompilationRule;
28
+ export {};
29
+ //# sourceMappingURL=min-consecutive-days.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"min-consecutive-days.d.ts","sourceRoot":"","sources":["../../../src/cpsat/rules/min-consecutive-days.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,CAAC,MAAM,KAAK,CAAC;AACzB,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AAI3D,QAAA,MAAM,wBAAwB;;;;;;;;;iBAW7B,CAAC;AAEF,MAAM,MAAM,wBAAwB,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,wBAAwB,CAAC,CAAC;AAEhF;;;;;;;;;;;;GAYG;AACH,wBAAgB,4BAA4B,CAAC,MAAM,EAAE,wBAAwB,GAAG,eAAe,CA0G9F"}
@@ -0,0 +1,104 @@
1
+ import * as z from "zod";
2
+ import { priorityToPenalty } from "../utils.js";
3
+ import { withScopes } from "./scoping.js";
4
+ const MinConsecutiveDaysSchema = withScopes(z.object({
5
+ days: z.number().min(0),
6
+ priority: z.union([
7
+ z.literal("LOW"),
8
+ z.literal("MEDIUM"),
9
+ z.literal("HIGH"),
10
+ z.literal("MANDATORY"),
11
+ ]),
12
+ }), { entities: ["employees", "roles", "skills"], times: [] });
13
+ /**
14
+ * Requires that once a person starts working, they continue for a minimum
15
+ * number of consecutive days.
16
+ *
17
+ * @example
18
+ * ```ts
19
+ * const rule = createMinConsecutiveDaysRule({
20
+ * days: 3,
21
+ * priority: "MANDATORY",
22
+ * });
23
+ * builder = new ModelBuilder({ ...config, rules: [rule] });
24
+ * ```
25
+ */
26
+ export function createMinConsecutiveDaysRule(config) {
27
+ const { days, priority, employeeIds } = MinConsecutiveDaysSchema.parse(config);
28
+ return {
29
+ compile(b) {
30
+ if (days <= 1)
31
+ return;
32
+ const employees = employeeIds
33
+ ? b.employees.filter((e) => employeeIds.includes(e.id))
34
+ : b.employees;
35
+ for (const emp of employees) {
36
+ const worksByDay = [];
37
+ for (const day of b.days) {
38
+ const worksVar = b.boolVar(`works_${emp.id}_${day}`);
39
+ worksByDay.push(worksVar);
40
+ const dayAssignments = b.shiftPatterns
41
+ .filter((p) => b.canAssign(emp, p) && b.patternAvailableOnDay(p, day))
42
+ .map((p) => b.assignment(emp.id, p.id, day));
43
+ if (dayAssignments.length === 0) {
44
+ b.addLinear([{ var: worksVar, coeff: 1 }], "==", 0);
45
+ }
46
+ else {
47
+ for (const assignVar of dayAssignments) {
48
+ b.addLinear([
49
+ { var: worksVar, coeff: 1 },
50
+ { var: assignVar, coeff: -1 },
51
+ ], ">=", 0);
52
+ }
53
+ b.addLinear([{ var: worksVar, coeff: 1 }, ...dayAssignments.map((v) => ({ var: v, coeff: -1 }))], "<=", 0);
54
+ }
55
+ }
56
+ for (let i = 0; i < b.days.length; i++) {
57
+ const dayLabel = b.days[i];
58
+ if (!dayLabel)
59
+ continue;
60
+ const worksToday = worksByDay[i];
61
+ if (!worksToday)
62
+ continue;
63
+ const worksYesterday = i > 0 ? worksByDay[i - 1] : undefined;
64
+ const startVar = b.boolVar(`work_start_${emp.id}_${dayLabel}`);
65
+ b.addLinear([
66
+ { var: startVar, coeff: 1 },
67
+ { var: worksToday, coeff: -1 },
68
+ ], "<=", 0);
69
+ if (worksYesterday) {
70
+ b.addLinear([
71
+ { var: startVar, coeff: 1 },
72
+ { var: worksYesterday, coeff: 1 },
73
+ ], "<=", 1);
74
+ b.addLinear([
75
+ { var: startVar, coeff: 1 },
76
+ { var: worksYesterday, coeff: 1 },
77
+ { var: worksToday, coeff: -1 },
78
+ ], ">=", 0);
79
+ }
80
+ else {
81
+ b.addLinear([
82
+ { var: startVar, coeff: 1 },
83
+ { var: worksToday, coeff: -1 },
84
+ ], ">=", 0);
85
+ }
86
+ const window = worksByDay.slice(i, i + days).filter(Boolean);
87
+ if (window.length === 0)
88
+ continue;
89
+ const terms = [
90
+ ...window.map((v) => ({ var: v, coeff: 1 })),
91
+ { var: startVar, coeff: -days },
92
+ ];
93
+ if (priority === "MANDATORY") {
94
+ b.addLinear(terms, ">=", 0);
95
+ }
96
+ else {
97
+ b.addSoftLinear(terms, ">=", 0, priorityToPenalty(priority));
98
+ }
99
+ }
100
+ }
101
+ },
102
+ };
103
+ }
104
+ //# sourceMappingURL=min-consecutive-days.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"min-consecutive-days.js","sourceRoot":"","sources":["../../../src/cpsat/rules/min-consecutive-days.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,28 @@
1
+ import * as z from "zod";
2
+ import type { CompilationRule } from "../model-builder.js";
3
+ declare const MinHoursDaySchema: z.ZodObject<{
4
+ hours: z.ZodNumber;
5
+ priority: z.ZodUnion<readonly [z.ZodLiteral<"LOW">, z.ZodLiteral<"MEDIUM">, z.ZodLiteral<"HIGH">, z.ZodLiteral<"MANDATORY">]>;
6
+ } & {
7
+ employeeIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
8
+ } & {
9
+ roleIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
10
+ } & {
11
+ skillIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
12
+ }, z.core.$strip>;
13
+ export type MinHoursDayConfig = z.infer<typeof MinHoursDaySchema>;
14
+ /**
15
+ * Ensures a person works at least a minimum number of hours per day.
16
+ *
17
+ * @example
18
+ * ```ts
19
+ * const rule = createMinHoursDayRule({
20
+ * hours: 6,
21
+ * priority: "MANDATORY",
22
+ * });
23
+ * builder = new ModelBuilder({ ...config, rules: [rule] });
24
+ * ```
25
+ */
26
+ export declare function createMinHoursDayRule(config: MinHoursDayConfig): CompilationRule;
27
+ export {};
28
+ //# sourceMappingURL=min-hours-day.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"min-hours-day.d.ts","sourceRoot":"","sources":["../../../src/cpsat/rules/min-hours-day.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,CAAC,MAAM,KAAK,CAAC;AACzB,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AAK3D,QAAA,MAAM,iBAAiB;;;;;;;;;iBAWtB,CAAC;AAEF,MAAM,MAAM,iBAAiB,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,iBAAiB,CAAC,CAAC;AAElE;;;;;;;;;;;GAWG;AACH,wBAAgB,qBAAqB,CAAC,MAAM,EAAE,iBAAiB,GAAG,eAAe,CAmChF"}
@@ -0,0 +1,61 @@
1
+ import * as z from "zod";
2
+ import { priorityToPenalty } from "../utils.js";
3
+ import { withScopes } from "./scoping.js";
4
+ const MinHoursDaySchema = withScopes(z.object({
5
+ hours: z.number().min(0),
6
+ priority: z.union([
7
+ z.literal("LOW"),
8
+ z.literal("MEDIUM"),
9
+ z.literal("HIGH"),
10
+ z.literal("MANDATORY"),
11
+ ]),
12
+ }), { entities: ["employees", "roles", "skills"], times: [] });
13
+ /**
14
+ * Ensures a person works at least a minimum number of hours per day.
15
+ *
16
+ * @example
17
+ * ```ts
18
+ * const rule = createMinHoursDayRule({
19
+ * hours: 6,
20
+ * priority: "MANDATORY",
21
+ * });
22
+ * builder = new ModelBuilder({ ...config, rules: [rule] });
23
+ * ```
24
+ */
25
+ export function createMinHoursDayRule(config) {
26
+ const { hours, priority, employeeIds } = MinHoursDaySchema.parse(config);
27
+ const minMinutes = hours * 60;
28
+ return {
29
+ compile(b) {
30
+ if (hours <= 0)
31
+ return;
32
+ const employees = employeeIds
33
+ ? b.employees.filter((e) => employeeIds.includes(e.id))
34
+ : b.employees;
35
+ for (const emp of employees) {
36
+ for (const day of b.days) {
37
+ const terms = [];
38
+ for (const pattern of b.shiftPatterns) {
39
+ if (!b.canAssign(emp, pattern))
40
+ continue;
41
+ if (!b.patternAvailableOnDay(pattern, day))
42
+ continue;
43
+ terms.push({
44
+ var: b.assignment(emp.id, pattern.id, day),
45
+ coeff: b.patternDuration(pattern.id),
46
+ });
47
+ }
48
+ if (terms.length === 0)
49
+ continue;
50
+ if (priority === "MANDATORY") {
51
+ b.addLinear(terms, ">=", minMinutes);
52
+ }
53
+ else {
54
+ b.addSoftLinear(terms, ">=", minMinutes, priorityToPenalty(priority));
55
+ }
56
+ }
57
+ }
58
+ },
59
+ };
60
+ }
61
+ //# sourceMappingURL=min-hours-day.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"min-hours-day.js","sourceRoot":"","sources":["../../../src/cpsat/rules/min-hours-day.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,CAAC,MAAM,KAAK,CAAC;AAGzB,OAAO,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAChD,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAE1C,MAAM,iBAAiB,GAAG,UAAU,CAClC,CAAC,CAAC,MAAM,CAAC;IACP,KAAK,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;IACxB,QAAQ,EAAE,CAAC,CAAC,KAAK,CAAC;QAChB,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC;QAChB,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC;QACnB,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC;QACjB,CAAC,CAAC,OAAO,CAAC,WAAW,CAAC;KACvB,CAAC;CACH,CAAC,EACF,EAAE,QAAQ,EAAE,CAAC,WAAW,EAAE,OAAO,EAAE,QAAQ,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAC1D,CAAC;AAIF;;;;;;;;;;;GAWG;AACH,MAAM,UAAU,qBAAqB,CAAC,MAAyB;IAC7D,MAAM,EAAE,KAAK,EAAE,QAAQ,EAAE,WAAW,EAAE,GAAG,iBAAiB,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;IACzE,MAAM,UAAU,GAAG,KAAK,GAAG,EAAE,CAAC;IAE9B,OAAO;QACL,OAAO,CAAC,CAAC;YACP,IAAI,KAAK,IAAI,CAAC;gBAAE,OAAO;YAEvB,MAAM,SAAS,GAAG,WAAW;gBAC3B,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;gBACvD,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;YAEhB,KAAK,MAAM,GAAG,IAAI,SAAS,EAAE,CAAC;gBAC5B,KAAK,MAAM,GAAG,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC;oBACzB,MAAM,KAAK,GAAW,EAAE,CAAC;oBACzB,KAAK,MAAM,OAAO,IAAI,CAAC,CAAC,aAAa,EAAE,CAAC;wBACtC,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC,GAAG,EAAE,OAAO,CAAC;4BAAE,SAAS;wBACzC,IAAI,CAAC,CAAC,CAAC,qBAAqB,CAAC,OAAO,EAAE,GAAG,CAAC;4BAAE,SAAS;wBACrD,KAAK,CAAC,IAAI,CAAC;4BACT,GAAG,EAAE,CAAC,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,EAAE,OAAO,CAAC,EAAE,EAAE,GAAG,CAAC;4BAC1C,KAAK,EAAE,CAAC,CAAC,eAAe,CAAC,OAAO,CAAC,EAAE,CAAC;yBACrC,CAAC,CAAC;oBACL,CAAC;oBAED,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;wBAAE,SAAS;oBAEjC,IAAI,QAAQ,KAAK,WAAW,EAAE,CAAC;wBAC7B,CAAC,CAAC,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,UAAU,CAAC,CAAC;oBACvC,CAAC;yBAAM,CAAC;wBACN,CAAC,CAAC,aAAa,CAAC,KAAK,EAAE,IAAI,EAAE,UAAU,EAAE,iBAAiB,CAAC,QAAQ,CAAC,CAAC,CAAC;oBACxE,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;KACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,29 @@
1
+ import * as z from "zod";
2
+ import type { CompilationRule } from "../model-builder.js";
3
+ declare const MinHoursWeekSchema: z.ZodObject<{
4
+ hours: z.ZodNumber;
5
+ priority: z.ZodUnion<readonly [z.ZodLiteral<"LOW">, z.ZodLiteral<"MEDIUM">, z.ZodLiteral<"HIGH">, z.ZodLiteral<"MANDATORY">]>;
6
+ weekStartsOn: z.ZodOptional<z.ZodUnion<readonly [z.ZodLiteral<"monday">, z.ZodLiteral<"tuesday">, z.ZodLiteral<"wednesday">, z.ZodLiteral<"thursday">, z.ZodLiteral<"friday">, z.ZodLiteral<"saturday">, z.ZodLiteral<"sunday">]>>;
7
+ } & {
8
+ employeeIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
9
+ } & {
10
+ roleIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
11
+ } & {
12
+ skillIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
13
+ }, z.core.$strip>;
14
+ export type MinHoursWeekConfig = z.infer<typeof MinHoursWeekSchema>;
15
+ /**
16
+ * Enforces a minimum total number of hours per scheduling week.
17
+ *
18
+ * @example
19
+ * ```ts
20
+ * const rule = createMinHoursWeekRule({
21
+ * hours: 30,
22
+ * priority: "HIGH",
23
+ * });
24
+ * builder = new ModelBuilder({ ...config, rules: [rule] });
25
+ * ```
26
+ */
27
+ export declare function createMinHoursWeekRule(config: MinHoursWeekConfig): CompilationRule;
28
+ export {};
29
+ //# sourceMappingURL=min-hours-week.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"min-hours-week.d.ts","sourceRoot":"","sources":["../../../src/cpsat/rules/min-hours-week.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,CAAC,MAAM,KAAK,CAAC;AAEzB,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AAK3D,QAAA,MAAM,kBAAkB;;;;;;;;;;iBAavB,CAAC;AAEF,MAAM,MAAM,kBAAkB,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,kBAAkB,CAAC,CAAC;AAEpE;;;;;;;;;;;GAWG;AACH,wBAAgB,sBAAsB,CAAC,MAAM,EAAE,kBAAkB,GAAG,eAAe,CAwClF"}
@@ -0,0 +1,68 @@
1
+ import * as z from "zod";
2
+ import { DayOfWeekSchema } from "../../types.js";
3
+ import { priorityToPenalty, splitIntoWeeks } from "../utils.js";
4
+ import { withScopes } from "./scoping.js";
5
+ const MinHoursWeekSchema = withScopes(z.object({
6
+ hours: z.number().min(0),
7
+ priority: z.union([
8
+ z.literal("LOW"),
9
+ z.literal("MEDIUM"),
10
+ z.literal("HIGH"),
11
+ z.literal("MANDATORY"),
12
+ ]),
13
+ // Optional override; defaults to ModelBuilder.weekStartsOn
14
+ weekStartsOn: DayOfWeekSchema.optional(),
15
+ }), { entities: ["employees", "roles", "skills"], times: [] });
16
+ /**
17
+ * Enforces a minimum total number of hours per scheduling week.
18
+ *
19
+ * @example
20
+ * ```ts
21
+ * const rule = createMinHoursWeekRule({
22
+ * hours: 30,
23
+ * priority: "HIGH",
24
+ * });
25
+ * builder = new ModelBuilder({ ...config, rules: [rule] });
26
+ * ```
27
+ */
28
+ export function createMinHoursWeekRule(config) {
29
+ const parsed = MinHoursWeekSchema.parse(config);
30
+ const { hours, priority, employeeIds } = parsed;
31
+ const minMinutes = hours * 60;
32
+ return {
33
+ compile(b) {
34
+ if (hours <= 0)
35
+ return;
36
+ const employees = employeeIds
37
+ ? b.employees.filter((e) => employeeIds.includes(e.id))
38
+ : b.employees;
39
+ const weeks = splitIntoWeeks(b.days, parsed.weekStartsOn ?? b.weekStartsOn);
40
+ for (const emp of employees) {
41
+ for (const weekDays of weeks) {
42
+ const terms = [];
43
+ for (const day of weekDays) {
44
+ for (const pattern of b.shiftPatterns) {
45
+ if (!b.canAssign(emp, pattern))
46
+ continue;
47
+ if (!b.patternAvailableOnDay(pattern, day))
48
+ continue;
49
+ terms.push({
50
+ var: b.assignment(emp.id, pattern.id, day),
51
+ coeff: b.patternDuration(pattern.id),
52
+ });
53
+ }
54
+ }
55
+ if (terms.length === 0)
56
+ continue;
57
+ if (priority === "MANDATORY") {
58
+ b.addLinear(terms, ">=", minMinutes);
59
+ }
60
+ else {
61
+ b.addSoftLinear(terms, ">=", minMinutes, priorityToPenalty(priority));
62
+ }
63
+ }
64
+ }
65
+ },
66
+ };
67
+ }
68
+ //# sourceMappingURL=min-hours-week.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"min-hours-week.js","sourceRoot":"","sources":["../../../src/cpsat/rules/min-hours-week.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,28 @@
1
+ import * as z from "zod";
2
+ import type { CompilationRule } from "../model-builder.js";
3
+ export type MinRestBetweenShiftsConfig = z.infer<typeof MinRestBetweenShiftsSchema>;
4
+ declare const MinRestBetweenShiftsSchema: z.ZodObject<{
5
+ hours: z.ZodNumber;
6
+ priority: z.ZodUnion<readonly [z.ZodLiteral<"LOW">, z.ZodLiteral<"MEDIUM">, z.ZodLiteral<"HIGH">, z.ZodLiteral<"MANDATORY">]>;
7
+ } & {
8
+ employeeIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
9
+ } & {
10
+ roleIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
11
+ } & {
12
+ skillIds: z.ZodOptional<z.ZodArray<z.ZodString>>;
13
+ }, z.core.$strip>;
14
+ /**
15
+ * Enforces a minimum rest period between any two shifts a person works.
16
+ *
17
+ * @example
18
+ * ```ts
19
+ * const rule = createMinRestBetweenShiftsRule({
20
+ * hours: 10,
21
+ * priority: "MANDATORY",
22
+ * });
23
+ * builder = new ModelBuilder({ ...config, rules: [rule] });
24
+ * ```
25
+ */
26
+ export declare function createMinRestBetweenShiftsRule(config: MinRestBetweenShiftsConfig): CompilationRule;
27
+ export {};
28
+ //# sourceMappingURL=min-rest-between-shifts.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"min-rest-between-shifts.d.ts","sourceRoot":"","sources":["../../../src/cpsat/rules/min-rest-between-shifts.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,CAAC,MAAM,KAAK,CAAC;AACzB,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AAI3D,MAAM,MAAM,0BAA0B,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,0BAA0B,CAAC,CAAC;AAEpF,QAAA,MAAM,0BAA0B;;;;;;;;;iBAW/B,CAAC;AAEF;;;;;;;;;;;GAWG;AACH,wBAAgB,8BAA8B,CAC5C,MAAM,EAAE,0BAA0B,GACjC,eAAe,CAsFjB"}