@fctc/interface-logic 1.0.1 → 1.0.2
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/dist/base-model-type-DvO53Lwi.d.mts +7 -0
- package/dist/base-model-type-DvO53Lwi.d.ts +7 -0
- package/dist/config.d.mts +16 -0
- package/dist/config.d.ts +16 -0
- package/dist/config.js +2355 -0
- package/dist/config.mjs +2318 -0
- package/dist/constants.d.mts +128 -0
- package/dist/constants.d.ts +128 -0
- package/dist/constants.js +202 -0
- package/dist/constants.mjs +163 -0
- package/dist/context-type-D5XefoL-.d.mts +8 -0
- package/dist/context-type-D5XefoL-.d.ts +8 -0
- package/dist/environment.d.mts +41 -0
- package/dist/environment.d.ts +41 -0
- package/dist/environment.js +3061 -0
- package/dist/environment.mjs +3021 -0
- package/dist/hook.d.mts +292 -0
- package/dist/hook.d.ts +292 -0
- package/dist/hook.js +5683 -0
- package/dist/hook.mjs +5587 -0
- package/dist/index-C_nK1Mii.d.mts +19 -0
- package/dist/index-C_nK1Mii.d.ts +19 -0
- package/dist/index.d.mts +1 -1389
- package/dist/index.d.ts +1 -1389
- package/dist/index.js +0 -6856
- package/dist/index.mjs +0 -6629
- package/dist/model.d.mts +35 -0
- package/dist/model.d.ts +35 -0
- package/dist/model.js +3320 -0
- package/dist/model.mjs +3281 -0
- package/dist/provider.d.mts +15 -0
- package/dist/provider.d.ts +15 -0
- package/dist/provider.js +6566 -0
- package/dist/provider.mjs +6550 -0
- package/dist/services.d.mts +224 -0
- package/dist/services.d.ts +224 -0
- package/dist/services.js +4477 -0
- package/dist/services.mjs +4432 -0
- package/dist/session-storage-CxkkEmQh.d.mts +15 -0
- package/dist/session-storage-CxkkEmQh.d.ts +15 -0
- package/dist/store.d.mts +643 -0
- package/dist/store.d.ts +643 -0
- package/dist/store.js +814 -0
- package/dist/store.mjs +709 -0
- package/dist/types.d.mts +12 -0
- package/dist/types.d.ts +12 -0
- package/dist/types.js +18 -0
- package/dist/types.mjs +0 -0
- package/dist/use-get-selection-QZu1jKqa.d.mts +15 -0
- package/dist/use-get-selection-QZu1jKqa.d.ts +15 -0
- package/dist/utils.d.mts +87 -0
- package/dist/utils.d.ts +87 -0
- package/dist/utils.js +2947 -0
- package/dist/utils.mjs +2881 -0
- package/dist/view-type-y6vtF3wg.d.mts +106 -0
- package/dist/view-type-y6vtF3wg.d.ts +106 -0
- package/package.json +53 -12
package/dist/utils.mjs
ADDED
@@ -0,0 +1,2881 @@
|
|
1
|
+
// src/utils/error-handler.ts
|
2
|
+
var WesapError = class extends Error {
|
3
|
+
code;
|
4
|
+
constructor(message, code) {
|
5
|
+
super(message);
|
6
|
+
this.code = code;
|
7
|
+
}
|
8
|
+
};
|
9
|
+
function handleError(error, env) {
|
10
|
+
if (error instanceof WesapError) {
|
11
|
+
env.services.notification.error(error.message);
|
12
|
+
} else {
|
13
|
+
env.services.notification.error("An unexpected error occurred");
|
14
|
+
}
|
15
|
+
}
|
16
|
+
|
17
|
+
// src/utils/format.ts
|
18
|
+
import moment from "moment";
|
19
|
+
var formatCurrency = (amount, currency = "USD") => {
|
20
|
+
const formatter = new Intl.NumberFormat("vi-VN", {
|
21
|
+
style: "currency",
|
22
|
+
currency,
|
23
|
+
minimumFractionDigits: 0
|
24
|
+
});
|
25
|
+
return formatter.format(amount).replaceAll(".", ",");
|
26
|
+
};
|
27
|
+
var formatDate = (date, locale = "en-US") => {
|
28
|
+
return new Intl.DateTimeFormat(locale).format(new Date(date));
|
29
|
+
};
|
30
|
+
var validateAndParseDate = (input, isDateTime = false) => {
|
31
|
+
if (!input || typeof input !== "string") return null;
|
32
|
+
const cleanInput = input.replace(/[^0-9-\/:\s]/g, "");
|
33
|
+
const dateFormat = "YYYY-MM-DD";
|
34
|
+
const dateTimeFormat = "YYYY-MM-DD HH:mm:ss";
|
35
|
+
const currentDay = moment().format("DD");
|
36
|
+
const currentMonth = moment().format("MM");
|
37
|
+
const currentYear = moment().format("YYYY");
|
38
|
+
const defaultTime = "00:00:00";
|
39
|
+
const maxYear = parseInt(currentYear) + 10;
|
40
|
+
const isValidDate = (day, month, year) => {
|
41
|
+
const date = moment(`${day}-${month}-${year}`, "DD-MM-YYYY", true);
|
42
|
+
return date.isValid();
|
43
|
+
};
|
44
|
+
const isValidTime = (hour, minute = "00", second = "00") => {
|
45
|
+
const h = parseInt(hour, 10);
|
46
|
+
const m = parseInt(minute, 10);
|
47
|
+
const s = parseInt(second, 10);
|
48
|
+
return h >= 0 && h <= 23 && m >= 0 && m <= 59 && s >= 0 && s <= 59;
|
49
|
+
};
|
50
|
+
const formatOutput = (day, month, year, time = defaultTime) => {
|
51
|
+
let result = moment(
|
52
|
+
`${day}-${month}-${year} ${time}`,
|
53
|
+
"DD-MM-YYYY HH:mm:ss"
|
54
|
+
);
|
55
|
+
if (!result.isValid()) return null;
|
56
|
+
if (isDateTime) {
|
57
|
+
result = result.subtract(7, "hours");
|
58
|
+
return result.format(dateTimeFormat);
|
59
|
+
}
|
60
|
+
return result.format(dateFormat);
|
61
|
+
};
|
62
|
+
if (isDateTime && input.match(
|
63
|
+
/^\d{1,2}[\/-]\d{1,2}[\/-]\d{2,4}\s+\d{1,2}(:\d{1,2}(:\d{1,2})?)?$/
|
64
|
+
)) {
|
65
|
+
const [datePart, timePart] = input.split(/\s+/);
|
66
|
+
const dateParts = datePart.split(/[\/-]/);
|
67
|
+
const timeParts = timePart.split(":");
|
68
|
+
const day = dateParts[0].padStart(2, "0");
|
69
|
+
const month = dateParts[1].padStart(2, "0");
|
70
|
+
const year = dateParts[2].length <= 2 ? `20${dateParts[2].padStart(2, "0")}` : dateParts[2].padStart(4, "0");
|
71
|
+
const hour = timeParts[0].padStart(2, "0");
|
72
|
+
const minute = timeParts[1] ? timeParts[1].padStart(2, "0") : "00";
|
73
|
+
const second = timeParts[2] ? timeParts[2].padStart(2, "0") : "00";
|
74
|
+
if (isValidDate(day, month, year) && isValidTime(hour, minute, second)) {
|
75
|
+
let result = moment(
|
76
|
+
`${day}-${month}-${year} ${hour}:${minute}:${second}`,
|
77
|
+
"DD-MM-YYYY HH:mm:ss"
|
78
|
+
);
|
79
|
+
if (!result.isValid()) return null;
|
80
|
+
result = result.subtract(7, "hours");
|
81
|
+
return result.format(dateTimeFormat);
|
82
|
+
}
|
83
|
+
return null;
|
84
|
+
}
|
85
|
+
if (cleanInput.match(/^\d{4}-\d{2}-\d{2}$/)) {
|
86
|
+
const [year, month, day] = cleanInput.split("-");
|
87
|
+
if (isValidDate(day, month, year)) {
|
88
|
+
return formatOutput(day, month, year);
|
89
|
+
}
|
90
|
+
return null;
|
91
|
+
}
|
92
|
+
if (cleanInput.match(/^\d{1,2}\/\d{1,2}\/\d{2,4}$/)) {
|
93
|
+
const [day, month, year] = cleanInput.split("/");
|
94
|
+
const paddedDay = day.padStart(2, "0");
|
95
|
+
const paddedMonth = month.padStart(2, "0");
|
96
|
+
const fullYear = year.length <= 2 ? `20${year.padStart(2, "0")}` : year.padStart(4, "0");
|
97
|
+
if (isValidDate(paddedDay, paddedMonth, fullYear)) {
|
98
|
+
return formatOutput(paddedDay, paddedMonth, fullYear);
|
99
|
+
}
|
100
|
+
return null;
|
101
|
+
}
|
102
|
+
if (cleanInput.match(/^\d{1,2}-\d{1,2}-\d{2,4}$/)) {
|
103
|
+
const [day, month, year] = cleanInput.split("-");
|
104
|
+
const paddedDay = day.padStart(2, "0");
|
105
|
+
const paddedMonth = month.padStart(2, "0");
|
106
|
+
const fullYear = year.length <= 2 ? `20${year.padStart(2, "0")}` : year.padStart(4, "0");
|
107
|
+
if (isValidDate(paddedDay, paddedMonth, fullYear)) {
|
108
|
+
return formatOutput(paddedDay, paddedMonth, fullYear);
|
109
|
+
}
|
110
|
+
return null;
|
111
|
+
}
|
112
|
+
if (cleanInput.match(/^\d{1,2}[\/-]\d{1,2}$/)) {
|
113
|
+
const [day, month] = cleanInput.split(/[\/-]/);
|
114
|
+
const paddedDay = day.padStart(2, "0");
|
115
|
+
const paddedMonth = month.padStart(2, "0");
|
116
|
+
if (isValidDate(paddedDay, paddedMonth, currentYear)) {
|
117
|
+
return formatOutput(paddedDay, paddedMonth, currentYear);
|
118
|
+
}
|
119
|
+
return null;
|
120
|
+
}
|
121
|
+
if (cleanInput.match(/^\d{4}$/)) {
|
122
|
+
const num = parseInt(cleanInput, 10);
|
123
|
+
if (num >= 2e3 && num <= maxYear) {
|
124
|
+
if (isValidDate(currentDay, currentMonth, num.toString())) {
|
125
|
+
return formatOutput(currentDay, currentMonth, num.toString());
|
126
|
+
}
|
127
|
+
return null;
|
128
|
+
}
|
129
|
+
const day = cleanInput.slice(0, 2);
|
130
|
+
const month = cleanInput.slice(2, 4);
|
131
|
+
if (isValidDate(day, month, currentYear)) {
|
132
|
+
return formatOutput(day, month, currentYear);
|
133
|
+
}
|
134
|
+
return null;
|
135
|
+
}
|
136
|
+
if (cleanInput.startsWith("-") && /^\-\d+$/.test(cleanInput)) {
|
137
|
+
const daysToSubtract = Math.abs(parseInt(cleanInput, 10));
|
138
|
+
let result = moment().subtract(daysToSubtract, "days");
|
139
|
+
if (isDateTime) {
|
140
|
+
result = result.subtract(7, "hours");
|
141
|
+
}
|
142
|
+
if (result.isValid()) {
|
143
|
+
return isDateTime ? result.format(dateTimeFormat) : result.format(dateFormat);
|
144
|
+
}
|
145
|
+
return null;
|
146
|
+
}
|
147
|
+
if (input.match(/^\d{1,2}[^0-9-\/]+\d{1,2}[^0-9-\/]+\d{2,4}.*$/)) {
|
148
|
+
const parts = input.split(/[^0-9-\/]+/).filter(Boolean);
|
149
|
+
const day = parts[0].padStart(2, "0");
|
150
|
+
const month = parts[1].padStart(2, "0");
|
151
|
+
let year = parts[2];
|
152
|
+
year = year.length === 2 ? `20${year}` : year.padStart(4, "0");
|
153
|
+
if (isValidDate(day, month, year)) {
|
154
|
+
return formatOutput(day, month, year);
|
155
|
+
}
|
156
|
+
return null;
|
157
|
+
}
|
158
|
+
if (isDateTime) {
|
159
|
+
if (cleanInput.length === 9) {
|
160
|
+
const day = cleanInput.slice(0, 2);
|
161
|
+
const month = cleanInput.slice(2, 4);
|
162
|
+
const year = cleanInput.slice(4, 8);
|
163
|
+
const hour = cleanInput.slice(8, 9).padStart(2, "0");
|
164
|
+
if (isValidDate(day, month, year) && isValidTime(hour)) {
|
165
|
+
let result = moment(
|
166
|
+
`${day}-${month}-${year} ${hour}:00:00`,
|
167
|
+
"DD-MM-YYYY HH:mm:ss"
|
168
|
+
);
|
169
|
+
if (!result.isValid()) return null;
|
170
|
+
result = result.subtract(7, "hours");
|
171
|
+
return result.format(dateTimeFormat);
|
172
|
+
}
|
173
|
+
return null;
|
174
|
+
}
|
175
|
+
if (cleanInput.length === 10) {
|
176
|
+
const day = cleanInput.slice(0, 2);
|
177
|
+
const month = cleanInput.slice(2, 4);
|
178
|
+
const year = cleanInput.slice(4, 8);
|
179
|
+
const hour = cleanInput.slice(8, 10);
|
180
|
+
if (isValidDate(day, month, year) && isValidTime(hour)) {
|
181
|
+
let result = moment(
|
182
|
+
`${day}-${month}-${year} ${hour}:00:00`,
|
183
|
+
"DD-MM-YYYY HH:mm:ss"
|
184
|
+
);
|
185
|
+
if (!result.isValid()) return null;
|
186
|
+
result = result.subtract(7, "hours");
|
187
|
+
return result.format(dateTimeFormat);
|
188
|
+
}
|
189
|
+
return null;
|
190
|
+
}
|
191
|
+
if (cleanInput.length === 11) {
|
192
|
+
const day = cleanInput.slice(0, 2);
|
193
|
+
const month = cleanInput.slice(2, 4);
|
194
|
+
const year = cleanInput.slice(4, 8);
|
195
|
+
const hour = cleanInput.slice(8, 10);
|
196
|
+
const minute = cleanInput.slice(10, 11).padStart(2, "0");
|
197
|
+
if (isValidDate(day, month, year) && isValidTime(hour, minute)) {
|
198
|
+
let result = moment(
|
199
|
+
`${day}-${month}-${year} ${hour}:${minute}:00`,
|
200
|
+
"DD-MM-YYYY HH:mm:ss"
|
201
|
+
);
|
202
|
+
if (!result.isValid()) return null;
|
203
|
+
result = result.subtract(7, "hours");
|
204
|
+
return result.format(dateTimeFormat);
|
205
|
+
}
|
206
|
+
return null;
|
207
|
+
}
|
208
|
+
if (cleanInput.length === 12) {
|
209
|
+
const day = cleanInput.slice(0, 2);
|
210
|
+
const month = cleanInput.slice(2, 4);
|
211
|
+
const year = cleanInput.slice(4, 8);
|
212
|
+
const hour = cleanInput.slice(8, 10);
|
213
|
+
const minute = cleanInput.slice(10, 12);
|
214
|
+
if (isValidDate(day, month, year) && isValidTime(hour, minute)) {
|
215
|
+
let result = moment(
|
216
|
+
`${day}-${month}-${year} ${hour}:${minute}:00`,
|
217
|
+
"DD-MM-YYYY HH:mm:ss"
|
218
|
+
);
|
219
|
+
if (!result.isValid()) return null;
|
220
|
+
result = result.subtract(7, "hours");
|
221
|
+
return result.format(dateTimeFormat);
|
222
|
+
}
|
223
|
+
return null;
|
224
|
+
}
|
225
|
+
if (cleanInput.length === 13) {
|
226
|
+
const day = cleanInput.slice(0, 2);
|
227
|
+
const month = cleanInput.slice(2, 4);
|
228
|
+
const year = cleanInput.slice(4, 8);
|
229
|
+
const hour = cleanInput.slice(8, 10);
|
230
|
+
const minute = cleanInput.slice(10, 12);
|
231
|
+
const second = cleanInput.slice(12, 13).padStart(2, "0");
|
232
|
+
if (isValidDate(day, month, year) && isValidTime(hour, minute, second)) {
|
233
|
+
let result = moment(
|
234
|
+
`${day}-${month}-${year} ${hour}:${minute}:${second}`,
|
235
|
+
"DD-MM-YYYY HH:mm:ss"
|
236
|
+
);
|
237
|
+
if (!result.isValid()) return null;
|
238
|
+
result = result.subtract(7, "hours");
|
239
|
+
return result.format(dateTimeFormat);
|
240
|
+
}
|
241
|
+
return null;
|
242
|
+
}
|
243
|
+
if (cleanInput.length === 14) {
|
244
|
+
const day = cleanInput.slice(0, 2);
|
245
|
+
const month = cleanInput.slice(2, 4);
|
246
|
+
const year = cleanInput.slice(4, 8);
|
247
|
+
const hour = cleanInput.slice(8, 10);
|
248
|
+
const minute = cleanInput.slice(10, 12);
|
249
|
+
const second = cleanInput.slice(12, 14);
|
250
|
+
if (isValidDate(day, month, year) && isValidTime(hour, minute, second)) {
|
251
|
+
let result = moment(
|
252
|
+
`${day}-${month}-${year} ${hour}:${minute}:${second}`,
|
253
|
+
"DD-MM-YYYY HH:mm:ss"
|
254
|
+
);
|
255
|
+
if (!result.isValid()) return null;
|
256
|
+
result = result.subtract(7, "hours");
|
257
|
+
return result.format(dateTimeFormat);
|
258
|
+
}
|
259
|
+
return null;
|
260
|
+
}
|
261
|
+
}
|
262
|
+
const len = cleanInput.length;
|
263
|
+
if (len === 1 || len === 2) {
|
264
|
+
const paddedDay = cleanInput.padStart(2, "0");
|
265
|
+
if (isValidDate(paddedDay, currentMonth, currentYear)) {
|
266
|
+
return formatOutput(paddedDay, currentMonth, currentYear);
|
267
|
+
}
|
268
|
+
return null;
|
269
|
+
}
|
270
|
+
if (len === 3) {
|
271
|
+
const day = cleanInput.slice(0, 2);
|
272
|
+
const month = cleanInput.slice(2, 3).padStart(2, "0");
|
273
|
+
if (isValidDate(day, month, currentYear)) {
|
274
|
+
return formatOutput(day, month, currentYear);
|
275
|
+
}
|
276
|
+
return null;
|
277
|
+
}
|
278
|
+
if (len === 6) {
|
279
|
+
const day = cleanInput.slice(0, 2);
|
280
|
+
const month = cleanInput.slice(2, 4);
|
281
|
+
let year = cleanInput.slice(4, 6);
|
282
|
+
year = `20${year}`;
|
283
|
+
if (parseInt(month) > 12) {
|
284
|
+
if (isValidDate(day, currentMonth, currentYear)) {
|
285
|
+
return formatOutput(day, currentMonth, currentYear);
|
286
|
+
}
|
287
|
+
return null;
|
288
|
+
}
|
289
|
+
if (isValidDate(day, month, year)) {
|
290
|
+
return formatOutput(day, month, year);
|
291
|
+
}
|
292
|
+
return null;
|
293
|
+
}
|
294
|
+
if (len === 7) {
|
295
|
+
return null;
|
296
|
+
}
|
297
|
+
if (len === 8) {
|
298
|
+
const day = cleanInput.slice(0, 2);
|
299
|
+
const month = cleanInput.slice(2, 4);
|
300
|
+
const year = cleanInput.slice(4, 8);
|
301
|
+
if (isValidDate(day, month, year)) {
|
302
|
+
return formatOutput(day, month, year);
|
303
|
+
}
|
304
|
+
return null;
|
305
|
+
}
|
306
|
+
if (len > 8 && !isDateTime) {
|
307
|
+
return null;
|
308
|
+
}
|
309
|
+
return null;
|
310
|
+
};
|
311
|
+
|
312
|
+
// src/utils/domain/py_tokenizer.ts
|
313
|
+
var TokenizerError = class extends Error {
|
314
|
+
};
|
315
|
+
var directMap = {
|
316
|
+
"\\": "\\",
|
317
|
+
'"': '"',
|
318
|
+
"'": "'",
|
319
|
+
a: "\x07",
|
320
|
+
b: "\b",
|
321
|
+
f: "\f",
|
322
|
+
n: "\n",
|
323
|
+
r: "\r",
|
324
|
+
t: " ",
|
325
|
+
v: "\v"
|
326
|
+
};
|
327
|
+
function decodeStringLiteral(str, unicode) {
|
328
|
+
const out = [];
|
329
|
+
let code;
|
330
|
+
for (let i = 0; i < str.length; ++i) {
|
331
|
+
if (str[i] !== "\\") {
|
332
|
+
out.push(str[i]);
|
333
|
+
continue;
|
334
|
+
}
|
335
|
+
const escape = str[i + 1];
|
336
|
+
if (escape in directMap) {
|
337
|
+
out.push(directMap[escape]);
|
338
|
+
++i;
|
339
|
+
continue;
|
340
|
+
}
|
341
|
+
switch (escape) {
|
342
|
+
case "\n":
|
343
|
+
++i;
|
344
|
+
continue;
|
345
|
+
case "N":
|
346
|
+
if (!unicode) {
|
347
|
+
break;
|
348
|
+
}
|
349
|
+
throw new TokenizerError("SyntaxError: \\N{} escape not implemented");
|
350
|
+
case "u":
|
351
|
+
if (!unicode) {
|
352
|
+
break;
|
353
|
+
}
|
354
|
+
const uni = str.slice(i + 2, i + 6);
|
355
|
+
if (!/[0-9a-f]{4}/i.test(uni)) {
|
356
|
+
throw new TokenizerError(
|
357
|
+
[
|
358
|
+
"SyntaxError: (unicode error) 'unicodeescape' codec",
|
359
|
+
" can't decode bytes in position ",
|
360
|
+
i,
|
361
|
+
"-",
|
362
|
+
i + 4,
|
363
|
+
": truncated \\uXXXX escape"
|
364
|
+
].join("")
|
365
|
+
);
|
366
|
+
}
|
367
|
+
code = parseInt(uni, 16);
|
368
|
+
out.push(String.fromCharCode(code));
|
369
|
+
i += 5;
|
370
|
+
continue;
|
371
|
+
case "U":
|
372
|
+
if (!unicode) {
|
373
|
+
break;
|
374
|
+
}
|
375
|
+
throw new TokenizerError("SyntaxError: \\U escape not implemented");
|
376
|
+
case "x":
|
377
|
+
const hex = str.slice(i + 2, i + 4);
|
378
|
+
if (!/[0-9a-f]{2}/i.test(hex)) {
|
379
|
+
if (!unicode) {
|
380
|
+
throw new TokenizerError("ValueError: invalid \\x escape");
|
381
|
+
}
|
382
|
+
throw new TokenizerError(
|
383
|
+
[
|
384
|
+
"SyntaxError: (unicode error) 'unicodeescape'",
|
385
|
+
" codec can't decode bytes in position ",
|
386
|
+
i,
|
387
|
+
"-",
|
388
|
+
i + 2,
|
389
|
+
": truncated \\xXX escape"
|
390
|
+
].join("")
|
391
|
+
);
|
392
|
+
}
|
393
|
+
code = parseInt(hex, 16);
|
394
|
+
out.push(String.fromCharCode(code));
|
395
|
+
i += 3;
|
396
|
+
continue;
|
397
|
+
default:
|
398
|
+
if (!/[0-8]/.test(escape)) {
|
399
|
+
break;
|
400
|
+
}
|
401
|
+
const r = /[0-8]{1,3}/g;
|
402
|
+
r.lastIndex = i + 1;
|
403
|
+
const m = r.exec(str);
|
404
|
+
if (!m) break;
|
405
|
+
const oct = m[0];
|
406
|
+
code = parseInt(oct, 8);
|
407
|
+
out.push(String.fromCharCode(code));
|
408
|
+
i += oct.length;
|
409
|
+
continue;
|
410
|
+
}
|
411
|
+
out.push("\\");
|
412
|
+
}
|
413
|
+
return out.join("");
|
414
|
+
}
|
415
|
+
var constants = /* @__PURE__ */ new Set(["None", "False", "True"]);
|
416
|
+
var comparators = [
|
417
|
+
"in",
|
418
|
+
"not",
|
419
|
+
"not in",
|
420
|
+
"is",
|
421
|
+
"is not",
|
422
|
+
"<",
|
423
|
+
"<=",
|
424
|
+
">",
|
425
|
+
">=",
|
426
|
+
"<>",
|
427
|
+
"!=",
|
428
|
+
"=="
|
429
|
+
];
|
430
|
+
var binaryOperators = [
|
431
|
+
"or",
|
432
|
+
"and",
|
433
|
+
"|",
|
434
|
+
"^",
|
435
|
+
"&",
|
436
|
+
"<<",
|
437
|
+
">>",
|
438
|
+
"+",
|
439
|
+
"-",
|
440
|
+
"*",
|
441
|
+
"/",
|
442
|
+
"//",
|
443
|
+
"%",
|
444
|
+
"~",
|
445
|
+
"**",
|
446
|
+
"."
|
447
|
+
];
|
448
|
+
var unaryOperators = ["-"];
|
449
|
+
var symbols = /* @__PURE__ */ new Set([
|
450
|
+
...["(", ")", "[", "]", "{", "}", ":", ","],
|
451
|
+
...["if", "else", "lambda", "="],
|
452
|
+
...comparators,
|
453
|
+
...binaryOperators,
|
454
|
+
...unaryOperators
|
455
|
+
]);
|
456
|
+
function group(...args) {
|
457
|
+
return "(" + args.join("|") + ")";
|
458
|
+
}
|
459
|
+
var Name = "[a-zA-Z_]\\w*";
|
460
|
+
var Whitespace = "[ \\f\\t]*";
|
461
|
+
var DecNumber = "\\d+(L|l)?";
|
462
|
+
var IntNumber = DecNumber;
|
463
|
+
var Exponent = "[eE][+-]?\\d+";
|
464
|
+
var PointFloat = group(`\\d+\\.\\d*(${Exponent})?`, `\\.\\d+(${Exponent})?`);
|
465
|
+
var FloatNumber = group(PointFloat, `\\d+${Exponent}`);
|
466
|
+
var Number2 = group(FloatNumber, IntNumber);
|
467
|
+
var Operator = group(
|
468
|
+
"\\*\\*=?",
|
469
|
+
">>=?",
|
470
|
+
"<<=?",
|
471
|
+
"<>",
|
472
|
+
"!=",
|
473
|
+
"//=?",
|
474
|
+
"[+\\-*/%&|^=<>]=?",
|
475
|
+
"~"
|
476
|
+
);
|
477
|
+
var Bracket = "[\\[\\]\\(\\)\\{\\}]";
|
478
|
+
var Special = "[:;.,`@]";
|
479
|
+
var Funny = group(Operator, Bracket, Special);
|
480
|
+
var ContStr = group(
|
481
|
+
"([uU])?'([^\\n'\\\\]*(?:\\\\.[^\\n'\\\\]*)*)'",
|
482
|
+
'([uU])?"([^\\n"\\\\]*(?:\\\\.[^\\n"\\\\]*)*)"'
|
483
|
+
);
|
484
|
+
var PseudoToken = Whitespace + group(Number2, Funny, ContStr, Name);
|
485
|
+
var NumberPattern = new RegExp("^" + Number2 + "$");
|
486
|
+
var StringPattern = new RegExp("^" + ContStr + "$");
|
487
|
+
var NamePattern = new RegExp("^" + Name + "$");
|
488
|
+
var strip = new RegExp("^" + Whitespace);
|
489
|
+
function tokenize(str) {
|
490
|
+
const tokens = [];
|
491
|
+
const max = str.length;
|
492
|
+
let start = 0;
|
493
|
+
let end = 0;
|
494
|
+
const pseudoprog = new RegExp(PseudoToken, "g");
|
495
|
+
while (pseudoprog.lastIndex < max) {
|
496
|
+
const pseudomatch = pseudoprog.exec(str);
|
497
|
+
if (!pseudomatch) {
|
498
|
+
if (/^\s+$/.test(str.slice(end))) {
|
499
|
+
break;
|
500
|
+
}
|
501
|
+
throw new TokenizerError(
|
502
|
+
"Failed to tokenize <<" + str + ">> at index " + (end || 0) + "; parsed so far: " + tokens
|
503
|
+
);
|
504
|
+
}
|
505
|
+
if (pseudomatch.index > end) {
|
506
|
+
if (str.slice(end, pseudomatch.index).trim()) {
|
507
|
+
throw new TokenizerError("Invalid expression");
|
508
|
+
}
|
509
|
+
}
|
510
|
+
start = pseudomatch.index;
|
511
|
+
end = pseudoprog.lastIndex;
|
512
|
+
let token = str.slice(start, end).replace(strip, "");
|
513
|
+
if (NumberPattern.test(token)) {
|
514
|
+
tokens.push({
|
515
|
+
type: 0,
|
516
|
+
value: parseFloat(token)
|
517
|
+
});
|
518
|
+
} else if (StringPattern.test(token)) {
|
519
|
+
const m = StringPattern.exec(token);
|
520
|
+
if (!m) throw new TokenizerError("Invalid string match");
|
521
|
+
tokens.push({
|
522
|
+
type: 1,
|
523
|
+
value: decodeStringLiteral(
|
524
|
+
m[3] !== void 0 ? m[3] : m[5],
|
525
|
+
!!(m[2] || m[4])
|
526
|
+
)
|
527
|
+
});
|
528
|
+
} else if (symbols.has(token)) {
|
529
|
+
if (token === "in" && tokens.length > 0 && tokens[tokens.length - 1].value === "not") {
|
530
|
+
token = "not in";
|
531
|
+
tokens.pop();
|
532
|
+
} else if (token === "not" && tokens.length > 0 && tokens[tokens.length - 1].value === "is") {
|
533
|
+
token = "is not";
|
534
|
+
tokens.pop();
|
535
|
+
}
|
536
|
+
tokens.push({
|
537
|
+
type: 2,
|
538
|
+
value: token
|
539
|
+
});
|
540
|
+
} else if (constants.has(token)) {
|
541
|
+
tokens.push({
|
542
|
+
type: 4,
|
543
|
+
value: token
|
544
|
+
});
|
545
|
+
} else if (NamePattern.test(token)) {
|
546
|
+
tokens.push({
|
547
|
+
type: 3,
|
548
|
+
value: token
|
549
|
+
});
|
550
|
+
} else {
|
551
|
+
throw new TokenizerError("Invalid expression");
|
552
|
+
}
|
553
|
+
}
|
554
|
+
return tokens;
|
555
|
+
}
|
556
|
+
|
557
|
+
// src/utils/domain/py_parser.ts
|
558
|
+
var ParserError = class extends Error {
|
559
|
+
};
|
560
|
+
var chainedOperators = new Set(comparators);
|
561
|
+
var infixOperators = /* @__PURE__ */ new Set([...binaryOperators, ...comparators]);
|
562
|
+
function bp(symbol) {
|
563
|
+
switch (symbol) {
|
564
|
+
case "=":
|
565
|
+
return 10;
|
566
|
+
case "if":
|
567
|
+
return 20;
|
568
|
+
case "in":
|
569
|
+
case "not in":
|
570
|
+
case "is":
|
571
|
+
case "is not":
|
572
|
+
case "<":
|
573
|
+
case "<=":
|
574
|
+
case ">":
|
575
|
+
case ">=":
|
576
|
+
case "<>":
|
577
|
+
case "==":
|
578
|
+
case "!=":
|
579
|
+
return 60;
|
580
|
+
case "or":
|
581
|
+
return 30;
|
582
|
+
case "and":
|
583
|
+
return 40;
|
584
|
+
case "not":
|
585
|
+
return 50;
|
586
|
+
case "|":
|
587
|
+
return 70;
|
588
|
+
case "^":
|
589
|
+
return 80;
|
590
|
+
case "&":
|
591
|
+
return 90;
|
592
|
+
case "<<":
|
593
|
+
case ">>":
|
594
|
+
return 100;
|
595
|
+
case "+":
|
596
|
+
case "-":
|
597
|
+
return 110;
|
598
|
+
case "*":
|
599
|
+
case "/":
|
600
|
+
case "//":
|
601
|
+
case "%":
|
602
|
+
return 120;
|
603
|
+
case "**":
|
604
|
+
return 140;
|
605
|
+
case ".":
|
606
|
+
case "(":
|
607
|
+
case "[":
|
608
|
+
return 150;
|
609
|
+
default:
|
610
|
+
return 0;
|
611
|
+
}
|
612
|
+
}
|
613
|
+
function bindingPower(token) {
|
614
|
+
return token.type === 2 ? bp(token.value) : 0;
|
615
|
+
}
|
616
|
+
function isSymbol(token, value) {
|
617
|
+
return token.type === 2 && token.value === value;
|
618
|
+
}
|
619
|
+
function parsePrefix(current, tokens) {
|
620
|
+
switch (current.type) {
|
621
|
+
case 0:
|
622
|
+
return { type: 0, value: current.value };
|
623
|
+
case 1:
|
624
|
+
return { type: 1, value: current.value };
|
625
|
+
case 4:
|
626
|
+
if (current.value === "None") {
|
627
|
+
return {
|
628
|
+
type: 3
|
629
|
+
/* None */
|
630
|
+
};
|
631
|
+
} else {
|
632
|
+
return { type: 2, value: current.value === "True" };
|
633
|
+
}
|
634
|
+
case 3:
|
635
|
+
return { type: 5, value: current.value };
|
636
|
+
case 2:
|
637
|
+
switch (current.value) {
|
638
|
+
case "-":
|
639
|
+
case "+":
|
640
|
+
case "~":
|
641
|
+
return {
|
642
|
+
type: 6,
|
643
|
+
op: current.value,
|
644
|
+
right: _parse(tokens, 130)
|
645
|
+
};
|
646
|
+
case "not":
|
647
|
+
return {
|
648
|
+
type: 6,
|
649
|
+
op: current.value,
|
650
|
+
right: _parse(tokens, 50)
|
651
|
+
};
|
652
|
+
case "(":
|
653
|
+
const content = [];
|
654
|
+
let isTuple = false;
|
655
|
+
while (tokens[0] && !isSymbol(tokens[0], ")")) {
|
656
|
+
content.push(_parse(tokens, 0));
|
657
|
+
if (tokens[0]) {
|
658
|
+
if (tokens[0] && isSymbol(tokens[0], ",")) {
|
659
|
+
isTuple = true;
|
660
|
+
tokens.shift();
|
661
|
+
} else if (!isSymbol(tokens[0], ")")) {
|
662
|
+
throw new ParserError("parsing error");
|
663
|
+
}
|
664
|
+
} else {
|
665
|
+
throw new ParserError("parsing error");
|
666
|
+
}
|
667
|
+
}
|
668
|
+
if (!tokens[0] || !isSymbol(tokens[0], ")")) {
|
669
|
+
throw new ParserError("parsing error");
|
670
|
+
}
|
671
|
+
tokens.shift();
|
672
|
+
isTuple = isTuple || content.length === 0;
|
673
|
+
return isTuple ? { type: 10, value: content } : content[0];
|
674
|
+
case "[":
|
675
|
+
const value = [];
|
676
|
+
while (tokens[0] && !isSymbol(tokens[0], "]")) {
|
677
|
+
value.push(_parse(tokens, 0));
|
678
|
+
if (tokens[0]) {
|
679
|
+
if (isSymbol(tokens[0], ",")) {
|
680
|
+
tokens.shift();
|
681
|
+
} else if (!isSymbol(tokens[0], "]")) {
|
682
|
+
throw new ParserError("parsing error");
|
683
|
+
}
|
684
|
+
}
|
685
|
+
}
|
686
|
+
if (!tokens[0] || !isSymbol(tokens[0], "]")) {
|
687
|
+
throw new ParserError("parsing error");
|
688
|
+
}
|
689
|
+
tokens.shift();
|
690
|
+
return { type: 4, value };
|
691
|
+
case "{":
|
692
|
+
const dict = {};
|
693
|
+
while (tokens[0] && !isSymbol(tokens[0], "}")) {
|
694
|
+
const key = _parse(tokens, 0);
|
695
|
+
if (key.type !== 1 && key.type !== 0 || !tokens[0] || !isSymbol(tokens[0], ":")) {
|
696
|
+
throw new ParserError("parsing error");
|
697
|
+
}
|
698
|
+
tokens.shift();
|
699
|
+
const val = _parse(tokens, 0);
|
700
|
+
dict[key.value] = val;
|
701
|
+
if (isSymbol(tokens[0], ",")) {
|
702
|
+
tokens.shift();
|
703
|
+
}
|
704
|
+
}
|
705
|
+
if (!tokens.shift()) {
|
706
|
+
throw new ParserError("parsing error");
|
707
|
+
}
|
708
|
+
return { type: 11, value: dict };
|
709
|
+
default:
|
710
|
+
throw new ParserError("Token cannot be parsed");
|
711
|
+
}
|
712
|
+
default:
|
713
|
+
throw new ParserError("Token cannot be parsed");
|
714
|
+
}
|
715
|
+
}
|
716
|
+
function parseInfix(left, current, tokens) {
|
717
|
+
switch (current.type) {
|
718
|
+
case 2:
|
719
|
+
if (infixOperators.has(current.value)) {
|
720
|
+
let right = _parse(tokens, bindingPower(current));
|
721
|
+
if (current.value === "and" || current.value === "or") {
|
722
|
+
return {
|
723
|
+
type: 14,
|
724
|
+
op: current.value,
|
725
|
+
left,
|
726
|
+
right
|
727
|
+
};
|
728
|
+
} else if (current.value === ".") {
|
729
|
+
if (right.type === 5) {
|
730
|
+
return {
|
731
|
+
type: 15,
|
732
|
+
obj: left,
|
733
|
+
key: right.value
|
734
|
+
};
|
735
|
+
} else {
|
736
|
+
throw new ParserError("invalid obj lookup");
|
737
|
+
}
|
738
|
+
}
|
739
|
+
let op = {
|
740
|
+
type: 7,
|
741
|
+
op: current.value,
|
742
|
+
left,
|
743
|
+
right
|
744
|
+
};
|
745
|
+
while (chainedOperators.has(current.value) && tokens[0] && tokens[0].type === 2 && chainedOperators.has(tokens[0].value)) {
|
746
|
+
const nextToken = tokens.shift();
|
747
|
+
op = {
|
748
|
+
type: 14,
|
749
|
+
op: "and",
|
750
|
+
left: op,
|
751
|
+
right: {
|
752
|
+
type: 7,
|
753
|
+
op: nextToken.value,
|
754
|
+
left: right,
|
755
|
+
right: _parse(tokens, bindingPower(nextToken))
|
756
|
+
}
|
757
|
+
};
|
758
|
+
right = op.right;
|
759
|
+
}
|
760
|
+
return op;
|
761
|
+
}
|
762
|
+
switch (current.value) {
|
763
|
+
case "(":
|
764
|
+
const args = [];
|
765
|
+
const kwargs = {};
|
766
|
+
while (tokens[0] && !isSymbol(tokens[0], ")")) {
|
767
|
+
const arg = _parse(tokens, 0);
|
768
|
+
if (arg.type === 9) {
|
769
|
+
kwargs[arg.name.value] = arg.value;
|
770
|
+
} else {
|
771
|
+
args.push(arg);
|
772
|
+
}
|
773
|
+
if (tokens[0] && isSymbol(tokens[0], ",")) {
|
774
|
+
tokens.shift();
|
775
|
+
}
|
776
|
+
}
|
777
|
+
if (!tokens[0] || !isSymbol(tokens[0], ")")) {
|
778
|
+
throw new ParserError("parsing error");
|
779
|
+
}
|
780
|
+
tokens.shift();
|
781
|
+
return { type: 8, fn: left, args, kwargs };
|
782
|
+
case "=":
|
783
|
+
if (left.type === 5) {
|
784
|
+
return {
|
785
|
+
type: 9,
|
786
|
+
name: left,
|
787
|
+
value: _parse(tokens, 10)
|
788
|
+
};
|
789
|
+
}
|
790
|
+
break;
|
791
|
+
case "[":
|
792
|
+
const key = _parse(tokens);
|
793
|
+
if (!tokens[0] || !isSymbol(tokens[0], "]")) {
|
794
|
+
throw new ParserError("parsing error");
|
795
|
+
}
|
796
|
+
tokens.shift();
|
797
|
+
return {
|
798
|
+
type: 12,
|
799
|
+
target: left,
|
800
|
+
key
|
801
|
+
};
|
802
|
+
case "if":
|
803
|
+
const condition = _parse(tokens);
|
804
|
+
if (!tokens[0] || !isSymbol(tokens[0], "else")) {
|
805
|
+
throw new ParserError("parsing error");
|
806
|
+
}
|
807
|
+
tokens.shift();
|
808
|
+
const ifFalse = _parse(tokens);
|
809
|
+
return {
|
810
|
+
type: 13,
|
811
|
+
condition,
|
812
|
+
ifTrue: left,
|
813
|
+
ifFalse
|
814
|
+
};
|
815
|
+
default:
|
816
|
+
break;
|
817
|
+
}
|
818
|
+
}
|
819
|
+
throw new ParserError("Token cannot be parsed");
|
820
|
+
}
|
821
|
+
function _parse(tokens, bp2 = 0) {
|
822
|
+
const token = tokens.shift();
|
823
|
+
if (!token) {
|
824
|
+
throw new ParserError("Unexpected end of input");
|
825
|
+
}
|
826
|
+
let expr = parsePrefix(token, tokens);
|
827
|
+
while (tokens[0] && bindingPower(tokens[0]) > bp2) {
|
828
|
+
expr = parseInfix(expr, tokens.shift(), tokens);
|
829
|
+
}
|
830
|
+
return expr;
|
831
|
+
}
|
832
|
+
function parse(tokens) {
|
833
|
+
if (tokens.length) {
|
834
|
+
return _parse(tokens, 0);
|
835
|
+
}
|
836
|
+
throw new ParserError("Missing token");
|
837
|
+
}
|
838
|
+
function parseArgs(args, spec) {
|
839
|
+
const last = args[args.length - 1];
|
840
|
+
const unnamedArgs = typeof last === "object" ? args.slice(0, -1) : args;
|
841
|
+
const kwargs = typeof last === "object" ? last : {};
|
842
|
+
for (const [index, val] of unnamedArgs.entries()) {
|
843
|
+
kwargs[spec[index]] = val;
|
844
|
+
}
|
845
|
+
return kwargs;
|
846
|
+
}
|
847
|
+
|
848
|
+
// src/utils/domain/py_date.ts
|
849
|
+
var AssertionError = class extends Error {
|
850
|
+
};
|
851
|
+
var ValueError = class extends Error {
|
852
|
+
};
|
853
|
+
var NotSupportedError = class extends Error {
|
854
|
+
};
|
855
|
+
function fmt2(n) {
|
856
|
+
return String(n).padStart(2, "0");
|
857
|
+
}
|
858
|
+
function fmt4(n) {
|
859
|
+
return String(n).padStart(4, "0");
|
860
|
+
}
|
861
|
+
function divmod(a, b, fn) {
|
862
|
+
let mod = a % b;
|
863
|
+
if (mod > 0 && b < 0 || mod < 0 && b > 0) {
|
864
|
+
mod += b;
|
865
|
+
}
|
866
|
+
return fn(Math.floor(a / b), mod);
|
867
|
+
}
|
868
|
+
function assert(bool, message = "AssertionError") {
|
869
|
+
if (!bool) {
|
870
|
+
throw new AssertionError(message);
|
871
|
+
}
|
872
|
+
}
|
873
|
+
var DAYS_IN_MONTH = [
|
874
|
+
null,
|
875
|
+
31,
|
876
|
+
28,
|
877
|
+
31,
|
878
|
+
30,
|
879
|
+
31,
|
880
|
+
30,
|
881
|
+
31,
|
882
|
+
31,
|
883
|
+
30,
|
884
|
+
31,
|
885
|
+
30,
|
886
|
+
31
|
887
|
+
];
|
888
|
+
var DAYS_BEFORE_MONTH = [null];
|
889
|
+
for (let dbm = 0, i = 1; i < DAYS_IN_MONTH.length; ++i) {
|
890
|
+
DAYS_BEFORE_MONTH.push(dbm);
|
891
|
+
dbm += DAYS_IN_MONTH[i];
|
892
|
+
}
|
893
|
+
function daysInMonth(year, month) {
|
894
|
+
if (month === 2 && isLeap(year)) {
|
895
|
+
return 29;
|
896
|
+
}
|
897
|
+
return DAYS_IN_MONTH[month];
|
898
|
+
}
|
899
|
+
function isLeap(year) {
|
900
|
+
return year % 4 === 0 && (year % 100 !== 0 || year % 400 === 0);
|
901
|
+
}
|
902
|
+
function daysBeforeYear(year) {
|
903
|
+
const y = year - 1;
|
904
|
+
return y * 365 + Math.floor(y / 4) - Math.floor(y / 100) + Math.floor(y / 400);
|
905
|
+
}
|
906
|
+
function daysBeforeMonth(year, month) {
|
907
|
+
const postLeapFeb = month > 2 && isLeap(year);
|
908
|
+
return DAYS_BEFORE_MONTH[month] + (postLeapFeb ? 1 : 0);
|
909
|
+
}
|
910
|
+
function ymd2ord(year, month, day) {
|
911
|
+
const dim = daysInMonth(year, month);
|
912
|
+
if (!(1 <= day && day <= dim)) {
|
913
|
+
throw new ValueError(`day must be in 1..${dim}`);
|
914
|
+
}
|
915
|
+
return daysBeforeYear(year) + daysBeforeMonth(year, month) + day;
|
916
|
+
}
|
917
|
+
var DI400Y = daysBeforeYear(401);
|
918
|
+
var DI100Y = daysBeforeYear(101);
|
919
|
+
var DI4Y = daysBeforeYear(5);
|
920
|
+
function ord2ymd(n) {
|
921
|
+
--n;
|
922
|
+
let n400 = 0, n100 = 0, n4 = 0, n1 = 0, n0 = 0;
|
923
|
+
divmod(n, DI400Y, (_n400, n2) => {
|
924
|
+
n400 = _n400;
|
925
|
+
divmod(n2, DI100Y, (_n100, n3) => {
|
926
|
+
n100 = _n100;
|
927
|
+
divmod(n3, DI4Y, (_n4, n5) => {
|
928
|
+
n4 = _n4;
|
929
|
+
divmod(n5, 365, (_n1, n6) => {
|
930
|
+
n1 = _n1;
|
931
|
+
n0 = n6;
|
932
|
+
});
|
933
|
+
});
|
934
|
+
});
|
935
|
+
});
|
936
|
+
n = n0;
|
937
|
+
const year = n400 * 400 + 1 + n100 * 100 + n4 * 4 + n1;
|
938
|
+
if (n1 === 4 || n100 === 100) {
|
939
|
+
assert(n0 === 0);
|
940
|
+
return {
|
941
|
+
year: year - 1,
|
942
|
+
month: 12,
|
943
|
+
day: 31
|
944
|
+
};
|
945
|
+
}
|
946
|
+
const leapyear = n1 === 3 && (n4 !== 24 || n100 === 3);
|
947
|
+
assert(leapyear === isLeap(year));
|
948
|
+
let month = n + 50 >> 5;
|
949
|
+
let preceding = DAYS_BEFORE_MONTH[month] + (month > 2 && leapyear ? 1 : 0);
|
950
|
+
if (preceding > n) {
|
951
|
+
--month;
|
952
|
+
preceding -= DAYS_IN_MONTH[month] + (month === 2 && leapyear ? 1 : 0);
|
953
|
+
}
|
954
|
+
n -= preceding;
|
955
|
+
return {
|
956
|
+
year,
|
957
|
+
month,
|
958
|
+
day: n + 1
|
959
|
+
};
|
960
|
+
}
|
961
|
+
function tmxxx(year, month, day, hour, minute, second, microsecond) {
|
962
|
+
hour = hour || 0;
|
963
|
+
minute = minute || 0;
|
964
|
+
second = second || 0;
|
965
|
+
microsecond = microsecond || 0;
|
966
|
+
if (microsecond < 0 || microsecond > 999999) {
|
967
|
+
divmod(microsecond, 1e6, (carry, ms) => {
|
968
|
+
microsecond = ms;
|
969
|
+
second += carry;
|
970
|
+
});
|
971
|
+
}
|
972
|
+
if (second < 0 || second > 59) {
|
973
|
+
divmod(second, 60, (carry, s) => {
|
974
|
+
second = s;
|
975
|
+
minute += carry;
|
976
|
+
});
|
977
|
+
}
|
978
|
+
if (minute < 0 || minute > 59) {
|
979
|
+
divmod(minute, 60, (carry, m) => {
|
980
|
+
minute = m;
|
981
|
+
hour += carry;
|
982
|
+
});
|
983
|
+
}
|
984
|
+
if (hour < 0 || hour > 23) {
|
985
|
+
divmod(hour, 24, (carry, h) => {
|
986
|
+
hour = h;
|
987
|
+
day += carry;
|
988
|
+
});
|
989
|
+
}
|
990
|
+
if (month < 1 || month > 12) {
|
991
|
+
divmod(month - 1, 12, (carry, m) => {
|
992
|
+
month = m + 1;
|
993
|
+
year += carry;
|
994
|
+
});
|
995
|
+
}
|
996
|
+
const dim = daysInMonth(year, month);
|
997
|
+
if (day < 1 || day > dim) {
|
998
|
+
if (day === 0) {
|
999
|
+
--month;
|
1000
|
+
if (month > 0) {
|
1001
|
+
day = daysInMonth(year, month);
|
1002
|
+
} else {
|
1003
|
+
--year;
|
1004
|
+
month = 12;
|
1005
|
+
day = 31;
|
1006
|
+
}
|
1007
|
+
} else if (day === dim + 1) {
|
1008
|
+
++month;
|
1009
|
+
day = 1;
|
1010
|
+
if (month > 12) {
|
1011
|
+
month = 1;
|
1012
|
+
++year;
|
1013
|
+
}
|
1014
|
+
} else {
|
1015
|
+
const r = ord2ymd(ymd2ord(year, month, 1) + (day - 1));
|
1016
|
+
year = r.year;
|
1017
|
+
month = r.month;
|
1018
|
+
day = r.day;
|
1019
|
+
}
|
1020
|
+
}
|
1021
|
+
return {
|
1022
|
+
year,
|
1023
|
+
month,
|
1024
|
+
day,
|
1025
|
+
hour,
|
1026
|
+
minute,
|
1027
|
+
second,
|
1028
|
+
microsecond
|
1029
|
+
};
|
1030
|
+
}
|
1031
|
+
var PyDate = class _PyDate {
|
1032
|
+
constructor(year, month, day) {
|
1033
|
+
this.year = year;
|
1034
|
+
this.month = month;
|
1035
|
+
this.day = day;
|
1036
|
+
}
|
1037
|
+
static today() {
|
1038
|
+
return this.convertDate(/* @__PURE__ */ new Date());
|
1039
|
+
}
|
1040
|
+
static convertDate(date) {
|
1041
|
+
const year = date.getFullYear();
|
1042
|
+
const month = date.getMonth() + 1;
|
1043
|
+
const day = date.getDate();
|
1044
|
+
return new _PyDate(year, month, day);
|
1045
|
+
}
|
1046
|
+
static create(...args) {
|
1047
|
+
const { year, month, day } = parseArgs(args, ["year", "month", "day"]);
|
1048
|
+
return new _PyDate(year, month, day);
|
1049
|
+
}
|
1050
|
+
add(timedelta) {
|
1051
|
+
const s = tmxxx(this.year, this.month, this.day + timedelta.days, 0, 0, 0);
|
1052
|
+
return new _PyDate(s.year, s.month, s.day);
|
1053
|
+
}
|
1054
|
+
isEqual(other) {
|
1055
|
+
if (!(other instanceof _PyDate)) {
|
1056
|
+
return false;
|
1057
|
+
}
|
1058
|
+
return this.year === other.year && this.month === other.month && this.day === other.day;
|
1059
|
+
}
|
1060
|
+
strftime(format) {
|
1061
|
+
return format.replace(/%([A-Za-z])/g, (m, c) => {
|
1062
|
+
switch (c) {
|
1063
|
+
case "Y":
|
1064
|
+
return fmt4(this.year);
|
1065
|
+
case "m":
|
1066
|
+
return fmt2(this.month);
|
1067
|
+
case "d":
|
1068
|
+
return fmt2(this.day);
|
1069
|
+
default:
|
1070
|
+
throw new ValueError(`No known conversion for ${m}`);
|
1071
|
+
}
|
1072
|
+
});
|
1073
|
+
}
|
1074
|
+
substract(other) {
|
1075
|
+
if (other instanceof PyTimeDelta) {
|
1076
|
+
return this.add(other.negate());
|
1077
|
+
}
|
1078
|
+
if (other instanceof _PyDate) {
|
1079
|
+
return PyTimeDelta.create(this.toordinal() - other.toordinal());
|
1080
|
+
}
|
1081
|
+
throw new NotSupportedError();
|
1082
|
+
}
|
1083
|
+
toJSON() {
|
1084
|
+
return this.strftime("%Y-%m-%d");
|
1085
|
+
}
|
1086
|
+
toordinal() {
|
1087
|
+
return ymd2ord(this.year, this.month, this.day);
|
1088
|
+
}
|
1089
|
+
};
|
1090
|
+
var PyDateTime = class _PyDateTime {
|
1091
|
+
constructor(year, month, day, hour, minute, second, microsecond) {
|
1092
|
+
this.year = year;
|
1093
|
+
this.month = month;
|
1094
|
+
this.day = day;
|
1095
|
+
this.hour = hour;
|
1096
|
+
this.minute = minute;
|
1097
|
+
this.second = second;
|
1098
|
+
this.microsecond = microsecond;
|
1099
|
+
}
|
1100
|
+
static now() {
|
1101
|
+
return this.convertDate(/* @__PURE__ */ new Date());
|
1102
|
+
}
|
1103
|
+
static convertDate(date) {
|
1104
|
+
const year = date.getFullYear();
|
1105
|
+
const month = date.getMonth() + 1;
|
1106
|
+
const day = date.getDate();
|
1107
|
+
const hour = date.getHours();
|
1108
|
+
const minute = date.getMinutes();
|
1109
|
+
const second = date.getSeconds();
|
1110
|
+
return new _PyDateTime(year, month, day, hour, minute, second, 0);
|
1111
|
+
}
|
1112
|
+
static create(...args) {
|
1113
|
+
const namedArgs = parseArgs(args, [
|
1114
|
+
"year",
|
1115
|
+
"month",
|
1116
|
+
"day",
|
1117
|
+
"hour",
|
1118
|
+
"minute",
|
1119
|
+
"second",
|
1120
|
+
"microsecond"
|
1121
|
+
]);
|
1122
|
+
const year = namedArgs.year;
|
1123
|
+
const month = namedArgs.month;
|
1124
|
+
const day = namedArgs.day;
|
1125
|
+
const hour = namedArgs.hour || 0;
|
1126
|
+
const minute = namedArgs.minute || 0;
|
1127
|
+
const second = namedArgs.second || 0;
|
1128
|
+
const ms = namedArgs.microsecond / 1e3 || 0;
|
1129
|
+
return new _PyDateTime(year, month, day, hour, minute, second, ms);
|
1130
|
+
}
|
1131
|
+
static combine(...args) {
|
1132
|
+
const { date, time } = parseArgs(args, ["date", "time"]);
|
1133
|
+
return _PyDateTime.create(
|
1134
|
+
date.year,
|
1135
|
+
date.month,
|
1136
|
+
date.day,
|
1137
|
+
time.hour,
|
1138
|
+
time.minute,
|
1139
|
+
time.second
|
1140
|
+
);
|
1141
|
+
}
|
1142
|
+
add(timedelta) {
|
1143
|
+
const s = tmxxx(
|
1144
|
+
this.year,
|
1145
|
+
this.month,
|
1146
|
+
this.day + timedelta.days,
|
1147
|
+
this.hour,
|
1148
|
+
this.minute,
|
1149
|
+
this.second + timedelta.seconds,
|
1150
|
+
this.microsecond + timedelta.microseconds
|
1151
|
+
);
|
1152
|
+
return new _PyDateTime(
|
1153
|
+
s.year,
|
1154
|
+
s.month,
|
1155
|
+
s.day,
|
1156
|
+
s.hour,
|
1157
|
+
s.minute,
|
1158
|
+
s.second,
|
1159
|
+
s.microsecond
|
1160
|
+
);
|
1161
|
+
}
|
1162
|
+
isEqual(other) {
|
1163
|
+
if (!(other instanceof _PyDateTime)) {
|
1164
|
+
return false;
|
1165
|
+
}
|
1166
|
+
return this.year === other.year && this.month === other.month && this.day === other.day && this.hour === other.hour && this.minute === other.minute && this.second === other.second && this.microsecond === other.microsecond;
|
1167
|
+
}
|
1168
|
+
strftime(format) {
|
1169
|
+
return format.replace(/%([A-Za-z])/g, (m, c) => {
|
1170
|
+
switch (c) {
|
1171
|
+
case "Y":
|
1172
|
+
return fmt4(this.year);
|
1173
|
+
case "m":
|
1174
|
+
return fmt2(this.month);
|
1175
|
+
case "d":
|
1176
|
+
return fmt2(this.day);
|
1177
|
+
case "H":
|
1178
|
+
return fmt2(this.hour);
|
1179
|
+
case "M":
|
1180
|
+
return fmt2(this.minute);
|
1181
|
+
case "S":
|
1182
|
+
return fmt2(this.second);
|
1183
|
+
default:
|
1184
|
+
throw new ValueError(`No known conversion for ${m}`);
|
1185
|
+
}
|
1186
|
+
});
|
1187
|
+
}
|
1188
|
+
substract(timedelta) {
|
1189
|
+
return this.add(timedelta.negate());
|
1190
|
+
}
|
1191
|
+
toJSON() {
|
1192
|
+
return this.strftime("%Y-%m-%d %H:%M:%S");
|
1193
|
+
}
|
1194
|
+
to_utc() {
|
1195
|
+
const d = new Date(
|
1196
|
+
this.year,
|
1197
|
+
this.month - 1,
|
1198
|
+
this.day,
|
1199
|
+
this.hour,
|
1200
|
+
this.minute,
|
1201
|
+
this.second
|
1202
|
+
);
|
1203
|
+
const timedelta = PyTimeDelta.create({ minutes: d.getTimezoneOffset() });
|
1204
|
+
return this.add(timedelta);
|
1205
|
+
}
|
1206
|
+
};
|
1207
|
+
var PyTime = class _PyTime extends PyDate {
|
1208
|
+
constructor(hour, minute, second) {
|
1209
|
+
const now = /* @__PURE__ */ new Date();
|
1210
|
+
const year = now.getFullYear();
|
1211
|
+
const month = now.getMonth() + 1;
|
1212
|
+
const day = now.getDate();
|
1213
|
+
super(year, month, day);
|
1214
|
+
this.hour = hour;
|
1215
|
+
this.minute = minute;
|
1216
|
+
this.second = second;
|
1217
|
+
this.hour = hour;
|
1218
|
+
this.minute = minute;
|
1219
|
+
this.second = second;
|
1220
|
+
}
|
1221
|
+
static create(...args) {
|
1222
|
+
const namedArgs = parseArgs(args, ["hour", "minute", "second"]);
|
1223
|
+
const hour = namedArgs.hour || 0;
|
1224
|
+
const minute = namedArgs.minute || 0;
|
1225
|
+
const second = namedArgs.second || 0;
|
1226
|
+
return new _PyTime(hour, minute, second);
|
1227
|
+
}
|
1228
|
+
strftime(format) {
|
1229
|
+
return format.replace(/%([A-Za-z])/g, (m, c) => {
|
1230
|
+
switch (c) {
|
1231
|
+
case "Y":
|
1232
|
+
return fmt4(this.year);
|
1233
|
+
case "m":
|
1234
|
+
return fmt2(this.month);
|
1235
|
+
case "d":
|
1236
|
+
return fmt2(this.day);
|
1237
|
+
case "H":
|
1238
|
+
return fmt2(this.hour);
|
1239
|
+
case "M":
|
1240
|
+
return fmt2(this.minute);
|
1241
|
+
case "S":
|
1242
|
+
return fmt2(this.second);
|
1243
|
+
default:
|
1244
|
+
throw new ValueError(`No known conversion for ${m}`);
|
1245
|
+
}
|
1246
|
+
});
|
1247
|
+
}
|
1248
|
+
toJSON() {
|
1249
|
+
return this.strftime("%H:%M:%S");
|
1250
|
+
}
|
1251
|
+
};
|
1252
|
+
var DAYS_IN_YEAR = [
|
1253
|
+
31,
|
1254
|
+
59,
|
1255
|
+
90,
|
1256
|
+
120,
|
1257
|
+
151,
|
1258
|
+
181,
|
1259
|
+
212,
|
1260
|
+
243,
|
1261
|
+
273,
|
1262
|
+
304,
|
1263
|
+
334,
|
1264
|
+
366
|
1265
|
+
];
|
1266
|
+
var TIME_PERIODS = ["hour", "minute", "second"];
|
1267
|
+
var PERIODS = ["year", "month", "day", ...TIME_PERIODS];
|
1268
|
+
var RELATIVE_KEYS = "years months weeks days hours minutes seconds microseconds leapdays".split(
|
1269
|
+
" "
|
1270
|
+
);
|
1271
|
+
var ABSOLUTE_KEYS = "year month day hour minute second microsecond weekday nlyearday yearday".split(
|
1272
|
+
" "
|
1273
|
+
);
|
1274
|
+
var argsSpec = ["dt1", "dt2"];
|
1275
|
+
var PyRelativeDelta = class _PyRelativeDelta {
|
1276
|
+
static create(...args) {
|
1277
|
+
const params = parseArgs(args, argsSpec);
|
1278
|
+
if ("dt1" in params) {
|
1279
|
+
throw new Error("relativedelta(dt1, dt2) is not supported for now");
|
1280
|
+
}
|
1281
|
+
for (const period of PERIODS) {
|
1282
|
+
if (period in params) {
|
1283
|
+
const val = params[period];
|
1284
|
+
assert(val >= 0, `${period} ${val} is out of range`);
|
1285
|
+
}
|
1286
|
+
}
|
1287
|
+
for (const key of RELATIVE_KEYS) {
|
1288
|
+
params[key] = params[key] || 0;
|
1289
|
+
}
|
1290
|
+
for (const key of ABSOLUTE_KEYS) {
|
1291
|
+
params[key] = key in params ? params[key] : null;
|
1292
|
+
}
|
1293
|
+
params.days += 7 * params.weeks;
|
1294
|
+
let yearDay = 0;
|
1295
|
+
if (params.nlyearday) {
|
1296
|
+
yearDay = params.nlyearday;
|
1297
|
+
} else if (params.yearday) {
|
1298
|
+
yearDay = params.yearday;
|
1299
|
+
if (yearDay > 59) {
|
1300
|
+
params.leapDays = -1;
|
1301
|
+
}
|
1302
|
+
}
|
1303
|
+
if (yearDay) {
|
1304
|
+
for (let monthIndex = 0; monthIndex < DAYS_IN_YEAR.length; monthIndex++) {
|
1305
|
+
if (yearDay <= DAYS_IN_YEAR[monthIndex]) {
|
1306
|
+
params.month = monthIndex + 1;
|
1307
|
+
if (monthIndex === 0) {
|
1308
|
+
params.day = yearDay;
|
1309
|
+
} else {
|
1310
|
+
params.day = yearDay - DAYS_IN_YEAR[monthIndex - 1];
|
1311
|
+
}
|
1312
|
+
break;
|
1313
|
+
}
|
1314
|
+
}
|
1315
|
+
}
|
1316
|
+
return new _PyRelativeDelta(params);
|
1317
|
+
}
|
1318
|
+
static add(date, delta) {
|
1319
|
+
if (!(date instanceof PyDate || date instanceof PyDateTime)) {
|
1320
|
+
throw new NotSupportedError();
|
1321
|
+
}
|
1322
|
+
const s = tmxxx(
|
1323
|
+
(delta.year || date.year) + delta.years,
|
1324
|
+
(delta.month || date.month) + delta.months,
|
1325
|
+
delta.day || date.day,
|
1326
|
+
delta.hour || (date instanceof PyDateTime ? date.hour : 0),
|
1327
|
+
delta.minute || (date instanceof PyDateTime ? date.minute : 0),
|
1328
|
+
delta.second || (date instanceof PyDateTime ? date.second : 0),
|
1329
|
+
delta.microseconds || (date instanceof PyDateTime ? date.microsecond : 0)
|
1330
|
+
);
|
1331
|
+
const newDateTime = new PyDateTime(
|
1332
|
+
s.year,
|
1333
|
+
s.month,
|
1334
|
+
s.day,
|
1335
|
+
s.hour,
|
1336
|
+
s.minute,
|
1337
|
+
s.second,
|
1338
|
+
s.microsecond
|
1339
|
+
);
|
1340
|
+
let leapDays = 0;
|
1341
|
+
if (delta.leapDays && newDateTime.month > 2 && isLeap(newDateTime.year)) {
|
1342
|
+
leapDays = delta.leapDays;
|
1343
|
+
}
|
1344
|
+
const temp = newDateTime.add(
|
1345
|
+
PyTimeDelta.create({
|
1346
|
+
days: delta.days + leapDays,
|
1347
|
+
hours: delta.hours,
|
1348
|
+
minutes: delta.minutes,
|
1349
|
+
seconds: delta.seconds,
|
1350
|
+
microseconds: delta.microseconds
|
1351
|
+
})
|
1352
|
+
);
|
1353
|
+
const hasTime = Boolean(
|
1354
|
+
temp.hour || temp.minute || temp.second || temp.microsecond
|
1355
|
+
);
|
1356
|
+
const returnDate = !hasTime && date instanceof PyDate ? new PyDate(temp.year, temp.month, temp.day) : temp;
|
1357
|
+
if (delta.weekday !== null) {
|
1358
|
+
const wantedDow = delta.weekday + 1;
|
1359
|
+
const _date = new Date(
|
1360
|
+
returnDate.year,
|
1361
|
+
returnDate.month - 1,
|
1362
|
+
returnDate.day
|
1363
|
+
);
|
1364
|
+
const days = (7 - _date.getDay() + wantedDow) % 7;
|
1365
|
+
return returnDate.add(new PyTimeDelta(days, 0, 0));
|
1366
|
+
}
|
1367
|
+
return returnDate;
|
1368
|
+
}
|
1369
|
+
static substract(date, delta) {
|
1370
|
+
return _PyRelativeDelta.add(date, delta.negate());
|
1371
|
+
}
|
1372
|
+
constructor(params = {}, sign = 1) {
|
1373
|
+
this.years = sign * params.years;
|
1374
|
+
this.months = sign * params.months;
|
1375
|
+
this.days = sign * params.days;
|
1376
|
+
this.hours = sign * params.hours;
|
1377
|
+
this.minutes = sign * params.minutes;
|
1378
|
+
this.seconds = sign * params.seconds;
|
1379
|
+
this.microseconds = sign * params.microseconds;
|
1380
|
+
this.leapDays = params.leapDays;
|
1381
|
+
this.year = params.year;
|
1382
|
+
this.month = params.month;
|
1383
|
+
this.day = params.day;
|
1384
|
+
this.hour = params.hour;
|
1385
|
+
this.minute = params.minute;
|
1386
|
+
this.second = params.second;
|
1387
|
+
this.microsecond = params.microsecond;
|
1388
|
+
this.weekday = params.weekday;
|
1389
|
+
}
|
1390
|
+
years;
|
1391
|
+
months;
|
1392
|
+
days;
|
1393
|
+
hours;
|
1394
|
+
minutes;
|
1395
|
+
seconds;
|
1396
|
+
microseconds;
|
1397
|
+
leapDays;
|
1398
|
+
year;
|
1399
|
+
month;
|
1400
|
+
day;
|
1401
|
+
hour;
|
1402
|
+
minute;
|
1403
|
+
second;
|
1404
|
+
microsecond;
|
1405
|
+
weekday;
|
1406
|
+
negate() {
|
1407
|
+
return new _PyRelativeDelta(this, -1);
|
1408
|
+
}
|
1409
|
+
isEqual() {
|
1410
|
+
throw new NotSupportedError();
|
1411
|
+
}
|
1412
|
+
};
|
1413
|
+
var TIME_DELTA_KEYS = "weeks days hours minutes seconds milliseconds microseconds".split(" ");
|
1414
|
+
function modf(x) {
|
1415
|
+
const mod = x % 1;
|
1416
|
+
return [mod < 0 ? mod + 1 : mod, Math.floor(x)];
|
1417
|
+
}
|
1418
|
+
var PyTimeDelta = class _PyTimeDelta {
|
1419
|
+
constructor(days, seconds, microseconds) {
|
1420
|
+
this.days = days;
|
1421
|
+
this.seconds = seconds;
|
1422
|
+
this.microseconds = microseconds;
|
1423
|
+
}
|
1424
|
+
static create(...args) {
|
1425
|
+
const namedArgs = parseArgs(args, ["days", "seconds", "microseconds"]);
|
1426
|
+
for (const key of TIME_DELTA_KEYS) {
|
1427
|
+
namedArgs[key] = namedArgs[key] || 0;
|
1428
|
+
}
|
1429
|
+
let d = 0;
|
1430
|
+
let s = 0;
|
1431
|
+
let us = 0;
|
1432
|
+
const days = namedArgs.days + namedArgs.weeks * 7;
|
1433
|
+
let seconds = namedArgs.seconds + 60 * namedArgs.minutes + 3600 * namedArgs.hours;
|
1434
|
+
let microseconds = namedArgs.microseconds + 1e3 * namedArgs.milliseconds;
|
1435
|
+
const [dFrac, dInt] = modf(days);
|
1436
|
+
d = dInt;
|
1437
|
+
let daysecondsfrac = 0;
|
1438
|
+
if (dFrac) {
|
1439
|
+
const [dsFrac, dsInt] = modf(dFrac * 24 * 3600);
|
1440
|
+
s = dsInt;
|
1441
|
+
daysecondsfrac = dsFrac;
|
1442
|
+
}
|
1443
|
+
const [sFrac, sInt] = modf(seconds);
|
1444
|
+
seconds = sInt;
|
1445
|
+
const secondsfrac = sFrac + daysecondsfrac;
|
1446
|
+
divmod(seconds, 24 * 3600, (days2, seconds2) => {
|
1447
|
+
d += days2;
|
1448
|
+
s += seconds2;
|
1449
|
+
});
|
1450
|
+
microseconds += secondsfrac * 1e6;
|
1451
|
+
divmod(microseconds, 1e6, (seconds2, microseconds2) => {
|
1452
|
+
divmod(seconds2, 24 * 3600, (days2, seconds3) => {
|
1453
|
+
d += days2;
|
1454
|
+
s += seconds3;
|
1455
|
+
us += Math.round(microseconds2);
|
1456
|
+
});
|
1457
|
+
});
|
1458
|
+
return new _PyTimeDelta(d, s, us);
|
1459
|
+
}
|
1460
|
+
add(other) {
|
1461
|
+
return _PyTimeDelta.create({
|
1462
|
+
days: this.days + other.days,
|
1463
|
+
seconds: this.seconds + other.seconds,
|
1464
|
+
microseconds: this.microseconds + other.microseconds
|
1465
|
+
});
|
1466
|
+
}
|
1467
|
+
divide(n) {
|
1468
|
+
const us = (this.days * 24 * 3600 + this.seconds) * 1e6 + this.microseconds;
|
1469
|
+
return _PyTimeDelta.create({ microseconds: Math.floor(us / n) });
|
1470
|
+
}
|
1471
|
+
isEqual(other) {
|
1472
|
+
if (!(other instanceof _PyTimeDelta)) {
|
1473
|
+
return false;
|
1474
|
+
}
|
1475
|
+
return this.days === other.days && this.seconds === other.seconds && this.microseconds === other.microseconds;
|
1476
|
+
}
|
1477
|
+
isTrue() {
|
1478
|
+
return this.days !== 0 || this.seconds !== 0 || this.microseconds !== 0;
|
1479
|
+
}
|
1480
|
+
multiply(n) {
|
1481
|
+
return _PyTimeDelta.create({
|
1482
|
+
days: n * this.days,
|
1483
|
+
seconds: n * this.seconds,
|
1484
|
+
microseconds: n * this.microseconds
|
1485
|
+
});
|
1486
|
+
}
|
1487
|
+
negate() {
|
1488
|
+
return _PyTimeDelta.create({
|
1489
|
+
days: -this.days,
|
1490
|
+
seconds: -this.seconds,
|
1491
|
+
microseconds: -this.microseconds
|
1492
|
+
});
|
1493
|
+
}
|
1494
|
+
substract(other) {
|
1495
|
+
return _PyTimeDelta.create({
|
1496
|
+
days: this.days - other.days,
|
1497
|
+
seconds: this.seconds - other.seconds,
|
1498
|
+
microseconds: this.microseconds - other.microseconds
|
1499
|
+
});
|
1500
|
+
}
|
1501
|
+
total_seconds() {
|
1502
|
+
return this.days * 86400 + this.seconds + this.microseconds / 1e6;
|
1503
|
+
}
|
1504
|
+
};
|
1505
|
+
|
1506
|
+
// src/utils/domain/py_builtin.ts
|
1507
|
+
var EvaluationError = class extends Error {
|
1508
|
+
constructor(message) {
|
1509
|
+
super(message);
|
1510
|
+
this.name = "EvaluationError";
|
1511
|
+
}
|
1512
|
+
};
|
1513
|
+
function execOnIterable(iterable, func) {
|
1514
|
+
if (iterable === null) {
|
1515
|
+
throw new EvaluationError("value not iterable");
|
1516
|
+
}
|
1517
|
+
if (typeof iterable === "object" && !Array.isArray(iterable) && !(iterable instanceof Set)) {
|
1518
|
+
iterable = Object.keys(iterable);
|
1519
|
+
}
|
1520
|
+
if (typeof iterable?.[Symbol.iterator] !== "function") {
|
1521
|
+
throw new EvaluationError("value not iterable");
|
1522
|
+
}
|
1523
|
+
return func(iterable);
|
1524
|
+
}
|
1525
|
+
var BUILTINS = {
|
1526
|
+
/**
|
1527
|
+
* @param {any} value
|
1528
|
+
* @returns {boolean}
|
1529
|
+
*/
|
1530
|
+
bool(value) {
|
1531
|
+
switch (typeof value) {
|
1532
|
+
case "number":
|
1533
|
+
return value !== 0;
|
1534
|
+
case "string":
|
1535
|
+
return value !== "";
|
1536
|
+
case "boolean":
|
1537
|
+
return value;
|
1538
|
+
case "object":
|
1539
|
+
if (value === null || value === void 0) {
|
1540
|
+
return false;
|
1541
|
+
}
|
1542
|
+
if ("isTrue" in value && typeof value.isTrue === "function") {
|
1543
|
+
return value.isTrue();
|
1544
|
+
}
|
1545
|
+
if (value instanceof Array) {
|
1546
|
+
return !!value.length;
|
1547
|
+
}
|
1548
|
+
if (value instanceof Set) {
|
1549
|
+
return !!value.size;
|
1550
|
+
}
|
1551
|
+
return Object.keys(value).length !== 0;
|
1552
|
+
default:
|
1553
|
+
return true;
|
1554
|
+
}
|
1555
|
+
},
|
1556
|
+
set(iterable) {
|
1557
|
+
if (arguments.length > 2) {
|
1558
|
+
throw new EvaluationError(
|
1559
|
+
`set expected at most 1 argument, got (${arguments.length - 1})`
|
1560
|
+
);
|
1561
|
+
}
|
1562
|
+
return execOnIterable(
|
1563
|
+
iterable,
|
1564
|
+
(iterable2) => new Set(iterable2)
|
1565
|
+
);
|
1566
|
+
},
|
1567
|
+
time: {
|
1568
|
+
strftime(format) {
|
1569
|
+
return PyDateTime.now().strftime(format);
|
1570
|
+
}
|
1571
|
+
},
|
1572
|
+
context_today() {
|
1573
|
+
return PyDate.today();
|
1574
|
+
},
|
1575
|
+
get current_date() {
|
1576
|
+
return this.today;
|
1577
|
+
},
|
1578
|
+
get today() {
|
1579
|
+
return PyDate.today().strftime("%Y-%m-%d");
|
1580
|
+
},
|
1581
|
+
get now() {
|
1582
|
+
return PyDateTime.now().strftime("%Y-%m-%d %H:%M:%S");
|
1583
|
+
},
|
1584
|
+
datetime: {
|
1585
|
+
time: PyTime,
|
1586
|
+
timedelta: PyTimeDelta,
|
1587
|
+
datetime: PyDateTime,
|
1588
|
+
date: PyDate
|
1589
|
+
},
|
1590
|
+
relativedelta: PyRelativeDelta,
|
1591
|
+
true: true,
|
1592
|
+
false: false
|
1593
|
+
};
|
1594
|
+
|
1595
|
+
// src/utils/domain/py_utils.ts
|
1596
|
+
function toPyValue(value) {
|
1597
|
+
switch (typeof value) {
|
1598
|
+
case "string":
|
1599
|
+
return { type: 1, value };
|
1600
|
+
case "number":
|
1601
|
+
return { type: 0, value };
|
1602
|
+
case "boolean":
|
1603
|
+
return { type: 2, value };
|
1604
|
+
case "object":
|
1605
|
+
if (Array.isArray(value)) {
|
1606
|
+
return { type: 4, value: value.map(toPyValue) };
|
1607
|
+
} else if (value === null) {
|
1608
|
+
return {
|
1609
|
+
type: 3
|
1610
|
+
/* None */
|
1611
|
+
};
|
1612
|
+
} else if (value instanceof Date) {
|
1613
|
+
return {
|
1614
|
+
type: 1,
|
1615
|
+
value: String(PyDateTime.convertDate(value))
|
1616
|
+
};
|
1617
|
+
} else if (value instanceof PyDate || value instanceof PyDateTime) {
|
1618
|
+
return { type: 1, value };
|
1619
|
+
} else {
|
1620
|
+
const content = {};
|
1621
|
+
for (const key in value) {
|
1622
|
+
content[key] = toPyValue(value[key]);
|
1623
|
+
}
|
1624
|
+
return { type: 11, value: content };
|
1625
|
+
}
|
1626
|
+
default:
|
1627
|
+
throw new Error("Invalid type");
|
1628
|
+
}
|
1629
|
+
}
|
1630
|
+
function formatAST(ast, lbp = 0) {
|
1631
|
+
switch (ast.type) {
|
1632
|
+
case 3:
|
1633
|
+
return "None";
|
1634
|
+
case 1:
|
1635
|
+
return JSON.stringify(ast.value);
|
1636
|
+
case 0:
|
1637
|
+
return String(ast.value);
|
1638
|
+
case 2:
|
1639
|
+
return ast.value ? "True" : "False";
|
1640
|
+
case 4:
|
1641
|
+
return `[${ast.value.map(formatAST).join(", ")}]`;
|
1642
|
+
case 6:
|
1643
|
+
if (ast.op === "not") {
|
1644
|
+
return `not ${formatAST(ast.right, 50)}`;
|
1645
|
+
}
|
1646
|
+
return `${ast.op}${formatAST(ast.right, 130)}`;
|
1647
|
+
case 7:
|
1648
|
+
const abp = bp(ast.op);
|
1649
|
+
const binaryStr = `${formatAST(ast.left, abp)} ${ast.op} ${formatAST(ast.right, abp)}`;
|
1650
|
+
return abp < lbp ? `(${binaryStr})` : binaryStr;
|
1651
|
+
case 11:
|
1652
|
+
const pairs = [];
|
1653
|
+
for (const k in ast.value) {
|
1654
|
+
pairs.push(`"${k}": ${formatAST(ast.value[k])}`);
|
1655
|
+
}
|
1656
|
+
return `{${pairs.join(", ")}}`;
|
1657
|
+
case 10:
|
1658
|
+
return `(${ast.value.map(formatAST).join(", ")})`;
|
1659
|
+
case 5:
|
1660
|
+
return ast.value;
|
1661
|
+
case 12:
|
1662
|
+
return `${formatAST(ast.target)}[${formatAST(ast.key)}]`;
|
1663
|
+
case 13:
|
1664
|
+
const { ifTrue, condition, ifFalse } = ast;
|
1665
|
+
return `${formatAST(ifTrue)} if ${formatAST(condition)} else ${formatAST(ifFalse)}`;
|
1666
|
+
case 14:
|
1667
|
+
const boolAbp = bp(ast.op);
|
1668
|
+
const boolStr = `${formatAST(ast.left, boolAbp)} ${ast.op} ${formatAST(ast.right, boolAbp)}`;
|
1669
|
+
return boolAbp < lbp ? `(${boolStr})` : boolStr;
|
1670
|
+
case 15:
|
1671
|
+
return `${formatAST(ast.obj, 150)}.${ast.key}`;
|
1672
|
+
case 8:
|
1673
|
+
const args = ast.args.map(formatAST);
|
1674
|
+
const kwargs = [];
|
1675
|
+
for (const kwarg in ast.kwargs) {
|
1676
|
+
kwargs.push(`${kwarg} = ${formatAST(ast.kwargs[kwarg])}`);
|
1677
|
+
}
|
1678
|
+
const argStr = args.concat(kwargs).join(", ");
|
1679
|
+
return `${formatAST(ast.fn)}(${argStr})`;
|
1680
|
+
default:
|
1681
|
+
throw new Error("invalid expression: " + JSON.stringify(ast));
|
1682
|
+
}
|
1683
|
+
}
|
1684
|
+
var PY_DICT = /* @__PURE__ */ Object.create(null);
|
1685
|
+
function toPyDict(obj) {
|
1686
|
+
return new Proxy(obj, {
|
1687
|
+
getPrototypeOf() {
|
1688
|
+
return PY_DICT;
|
1689
|
+
}
|
1690
|
+
});
|
1691
|
+
}
|
1692
|
+
|
1693
|
+
// src/utils/domain/py_interpreter.ts
|
1694
|
+
var isTrue = BUILTINS.bool;
|
1695
|
+
function applyUnaryOp(ast, context) {
|
1696
|
+
const value = evaluate(ast.right, context);
|
1697
|
+
switch (ast.op) {
|
1698
|
+
case "-":
|
1699
|
+
if (value instanceof Object && "negate" in value) {
|
1700
|
+
return value.negate();
|
1701
|
+
}
|
1702
|
+
return -value;
|
1703
|
+
case "+":
|
1704
|
+
return value;
|
1705
|
+
case "not":
|
1706
|
+
return !isTrue(value);
|
1707
|
+
default:
|
1708
|
+
throw new EvaluationError(`Unknown unary operator: ${ast.op}`);
|
1709
|
+
}
|
1710
|
+
}
|
1711
|
+
function pytypeIndex(val) {
|
1712
|
+
switch (typeof val) {
|
1713
|
+
case "object":
|
1714
|
+
return val === null ? 1 : Array.isArray(val) ? 5 : 3;
|
1715
|
+
case "number":
|
1716
|
+
return 2;
|
1717
|
+
case "string":
|
1718
|
+
return 4;
|
1719
|
+
default:
|
1720
|
+
throw new EvaluationError(`Unknown type: ${typeof val}`);
|
1721
|
+
}
|
1722
|
+
}
|
1723
|
+
function isLess(left, right) {
|
1724
|
+
if (typeof left === "number" && typeof right === "number") {
|
1725
|
+
return left < right;
|
1726
|
+
}
|
1727
|
+
if (typeof left === "boolean") {
|
1728
|
+
left = left ? 1 : 0;
|
1729
|
+
}
|
1730
|
+
if (typeof right === "boolean") {
|
1731
|
+
right = right ? 1 : 0;
|
1732
|
+
}
|
1733
|
+
const leftIndex = pytypeIndex(left);
|
1734
|
+
const rightIndex = pytypeIndex(right);
|
1735
|
+
if (leftIndex === rightIndex) {
|
1736
|
+
return left < right;
|
1737
|
+
}
|
1738
|
+
return leftIndex < rightIndex;
|
1739
|
+
}
|
1740
|
+
function isEqual(left, right) {
|
1741
|
+
if (typeof left !== typeof right) {
|
1742
|
+
if (typeof left === "boolean" && typeof right === "number") {
|
1743
|
+
return right === (left ? 1 : 0);
|
1744
|
+
}
|
1745
|
+
if (typeof left === "number" && typeof right === "boolean") {
|
1746
|
+
return left === (right ? 1 : 0);
|
1747
|
+
}
|
1748
|
+
return false;
|
1749
|
+
}
|
1750
|
+
if (left instanceof Object && "isEqual" in left) {
|
1751
|
+
return left.isEqual(right);
|
1752
|
+
}
|
1753
|
+
return left === right;
|
1754
|
+
}
|
1755
|
+
function isIn(left, right) {
|
1756
|
+
if (Array.isArray(right)) {
|
1757
|
+
return right.includes(left);
|
1758
|
+
}
|
1759
|
+
if (typeof right === "string" && typeof left === "string") {
|
1760
|
+
return right.includes(left);
|
1761
|
+
}
|
1762
|
+
if (typeof right === "object") {
|
1763
|
+
return left in right;
|
1764
|
+
}
|
1765
|
+
return false;
|
1766
|
+
}
|
1767
|
+
function applyBinaryOp(ast, context) {
|
1768
|
+
const left = evaluate(ast.left, context);
|
1769
|
+
const right = evaluate(ast.right, context);
|
1770
|
+
switch (ast.op) {
|
1771
|
+
case "+": {
|
1772
|
+
const relativeDeltaOnLeft = left instanceof PyRelativeDelta;
|
1773
|
+
const relativeDeltaOnRight = right instanceof PyRelativeDelta;
|
1774
|
+
if (relativeDeltaOnLeft || relativeDeltaOnRight) {
|
1775
|
+
const date = relativeDeltaOnLeft ? right : left;
|
1776
|
+
const delta = relativeDeltaOnLeft ? left : right;
|
1777
|
+
return PyRelativeDelta.add(date, delta);
|
1778
|
+
}
|
1779
|
+
const timeDeltaOnLeft = left instanceof PyTimeDelta;
|
1780
|
+
const timeDeltaOnRight = right instanceof PyTimeDelta;
|
1781
|
+
if (timeDeltaOnLeft && timeDeltaOnRight) {
|
1782
|
+
return left.add(right);
|
1783
|
+
}
|
1784
|
+
if (timeDeltaOnLeft) {
|
1785
|
+
if (right instanceof PyDate || right instanceof PyDateTime) {
|
1786
|
+
return right.add(left);
|
1787
|
+
} else {
|
1788
|
+
throw new NotSupportedError();
|
1789
|
+
}
|
1790
|
+
}
|
1791
|
+
if (timeDeltaOnRight) {
|
1792
|
+
if (left instanceof PyDate || left instanceof PyDateTime) {
|
1793
|
+
return left.add(right);
|
1794
|
+
} else {
|
1795
|
+
throw new NotSupportedError();
|
1796
|
+
}
|
1797
|
+
}
|
1798
|
+
if (left instanceof Array && right instanceof Array) {
|
1799
|
+
return [...left, ...right];
|
1800
|
+
}
|
1801
|
+
return left + right;
|
1802
|
+
}
|
1803
|
+
case "-": {
|
1804
|
+
const isRightDelta = right instanceof PyRelativeDelta;
|
1805
|
+
if (isRightDelta) {
|
1806
|
+
return PyRelativeDelta.substract(left, right);
|
1807
|
+
}
|
1808
|
+
const timeDeltaOnRight = right instanceof PyTimeDelta;
|
1809
|
+
if (timeDeltaOnRight) {
|
1810
|
+
if (left instanceof PyTimeDelta) {
|
1811
|
+
return left.substract(right);
|
1812
|
+
} else if (left instanceof PyDate || left instanceof PyDateTime) {
|
1813
|
+
return left.substract(right);
|
1814
|
+
} else {
|
1815
|
+
throw new NotSupportedError();
|
1816
|
+
}
|
1817
|
+
}
|
1818
|
+
if (left instanceof PyDate) {
|
1819
|
+
return left.substract(right);
|
1820
|
+
}
|
1821
|
+
return left - right;
|
1822
|
+
}
|
1823
|
+
case "*": {
|
1824
|
+
const timeDeltaOnLeft = left instanceof PyTimeDelta;
|
1825
|
+
const timeDeltaOnRight = right instanceof PyTimeDelta;
|
1826
|
+
if (timeDeltaOnLeft || timeDeltaOnRight) {
|
1827
|
+
const number = timeDeltaOnLeft ? right : left;
|
1828
|
+
const delta = timeDeltaOnLeft ? left : right;
|
1829
|
+
return delta.multiply(number);
|
1830
|
+
}
|
1831
|
+
return left * right;
|
1832
|
+
}
|
1833
|
+
case "/":
|
1834
|
+
return left / right;
|
1835
|
+
case "%":
|
1836
|
+
return left % right;
|
1837
|
+
case "//":
|
1838
|
+
if (left instanceof PyTimeDelta) {
|
1839
|
+
return left.divide(right);
|
1840
|
+
}
|
1841
|
+
return Math.floor(left / right);
|
1842
|
+
case "**":
|
1843
|
+
return left ** right;
|
1844
|
+
case "==":
|
1845
|
+
return isEqual(left, right);
|
1846
|
+
case "<>":
|
1847
|
+
case "!=":
|
1848
|
+
return !isEqual(left, right);
|
1849
|
+
case "<":
|
1850
|
+
return isLess(left, right);
|
1851
|
+
case ">":
|
1852
|
+
return isLess(right, left);
|
1853
|
+
case ">=":
|
1854
|
+
return isEqual(left, right) || isLess(right, left);
|
1855
|
+
case "<=":
|
1856
|
+
return isEqual(left, right) || isLess(left, right);
|
1857
|
+
case "in":
|
1858
|
+
return isIn(left, right);
|
1859
|
+
case "not in":
|
1860
|
+
return !isIn(left, right);
|
1861
|
+
default:
|
1862
|
+
throw new EvaluationError(`Unknown binary operator: ${ast.op}`);
|
1863
|
+
}
|
1864
|
+
}
|
1865
|
+
var DICT = {
|
1866
|
+
get(...args) {
|
1867
|
+
const { key, defValue } = parseArgs(args, ["key", "defValue"]);
|
1868
|
+
const self = this;
|
1869
|
+
if (key in self) {
|
1870
|
+
return self[key];
|
1871
|
+
} else if (defValue !== void 0) {
|
1872
|
+
return defValue;
|
1873
|
+
}
|
1874
|
+
return null;
|
1875
|
+
}
|
1876
|
+
};
|
1877
|
+
var STRING = {
|
1878
|
+
lower() {
|
1879
|
+
return this.toLowerCase();
|
1880
|
+
},
|
1881
|
+
upper() {
|
1882
|
+
return this.toUpperCase();
|
1883
|
+
}
|
1884
|
+
};
|
1885
|
+
function applyFunc(key, func, set, ...args) {
|
1886
|
+
if (args.length === 1) {
|
1887
|
+
return new Set(set);
|
1888
|
+
}
|
1889
|
+
if (args.length > 2) {
|
1890
|
+
throw new EvaluationError(
|
1891
|
+
`${key}: py_js supports at most 1 argument, got (${args.length - 1})`
|
1892
|
+
);
|
1893
|
+
}
|
1894
|
+
return execOnIterable(args[0], func);
|
1895
|
+
}
|
1896
|
+
var SET = {
|
1897
|
+
intersection(...args) {
|
1898
|
+
return applyFunc(
|
1899
|
+
"intersection",
|
1900
|
+
(iterable) => {
|
1901
|
+
const intersection = /* @__PURE__ */ new Set();
|
1902
|
+
for (const i of iterable) {
|
1903
|
+
if (this.has(i)) {
|
1904
|
+
intersection.add(i);
|
1905
|
+
}
|
1906
|
+
}
|
1907
|
+
return intersection;
|
1908
|
+
},
|
1909
|
+
this,
|
1910
|
+
...args
|
1911
|
+
);
|
1912
|
+
},
|
1913
|
+
difference(...args) {
|
1914
|
+
return applyFunc(
|
1915
|
+
"difference",
|
1916
|
+
(iterable) => {
|
1917
|
+
iterable = new Set(iterable);
|
1918
|
+
const difference = /* @__PURE__ */ new Set();
|
1919
|
+
for (const e of this) {
|
1920
|
+
if (!iterable.has(e)) {
|
1921
|
+
difference.add(e);
|
1922
|
+
}
|
1923
|
+
}
|
1924
|
+
return difference;
|
1925
|
+
},
|
1926
|
+
this,
|
1927
|
+
...args
|
1928
|
+
);
|
1929
|
+
},
|
1930
|
+
union(...args) {
|
1931
|
+
return applyFunc(
|
1932
|
+
"union",
|
1933
|
+
(iterable) => {
|
1934
|
+
return /* @__PURE__ */ new Set([...this, ...iterable]);
|
1935
|
+
},
|
1936
|
+
this,
|
1937
|
+
...args
|
1938
|
+
);
|
1939
|
+
}
|
1940
|
+
};
|
1941
|
+
function methods(_class) {
|
1942
|
+
return Object.getOwnPropertyNames(_class.prototype).map(
|
1943
|
+
(prop) => _class.prototype[prop]
|
1944
|
+
);
|
1945
|
+
}
|
1946
|
+
var allowedFns = /* @__PURE__ */ new Set([
|
1947
|
+
BUILTINS.time.strftime,
|
1948
|
+
BUILTINS.set,
|
1949
|
+
BUILTINS.bool,
|
1950
|
+
BUILTINS.context_today,
|
1951
|
+
BUILTINS.datetime.datetime.now,
|
1952
|
+
BUILTINS.datetime.datetime.combine,
|
1953
|
+
BUILTINS.datetime.date.today,
|
1954
|
+
...methods(BUILTINS.relativedelta),
|
1955
|
+
...Object.values(BUILTINS.datetime).flatMap((obj) => methods(obj)),
|
1956
|
+
...Object.values(SET),
|
1957
|
+
...Object.values(DICT),
|
1958
|
+
...Object.values(STRING)
|
1959
|
+
]);
|
1960
|
+
var unboundFn = Symbol("unbound function");
|
1961
|
+
function evaluate(ast, context = {}) {
|
1962
|
+
const dicts = /* @__PURE__ */ new Set();
|
1963
|
+
let pyContext;
|
1964
|
+
const evalContext = Object.create(context);
|
1965
|
+
if (!evalContext?.context) {
|
1966
|
+
Object.defineProperty(evalContext, "context", {
|
1967
|
+
get() {
|
1968
|
+
if (!pyContext) {
|
1969
|
+
pyContext = toPyDict(context);
|
1970
|
+
}
|
1971
|
+
return pyContext;
|
1972
|
+
}
|
1973
|
+
});
|
1974
|
+
}
|
1975
|
+
function _innerEvaluate(ast2) {
|
1976
|
+
switch (ast2?.type) {
|
1977
|
+
case 0:
|
1978
|
+
// Number
|
1979
|
+
case 1:
|
1980
|
+
return ast2.value;
|
1981
|
+
case 5:
|
1982
|
+
if (ast2.value in evalContext) {
|
1983
|
+
if (typeof evalContext[ast2.value] === "object" && evalContext[ast2.value]?.id) {
|
1984
|
+
return evalContext[ast2.value]?.id;
|
1985
|
+
}
|
1986
|
+
return evalContext[ast2.value] ?? false;
|
1987
|
+
} else if (ast2.value in BUILTINS) {
|
1988
|
+
return BUILTINS[ast2.value];
|
1989
|
+
} else {
|
1990
|
+
return false;
|
1991
|
+
}
|
1992
|
+
case 3:
|
1993
|
+
return null;
|
1994
|
+
case 2:
|
1995
|
+
return ast2.value;
|
1996
|
+
case 6:
|
1997
|
+
return applyUnaryOp(ast2, evalContext);
|
1998
|
+
case 7:
|
1999
|
+
return applyBinaryOp(ast2, evalContext);
|
2000
|
+
case 14:
|
2001
|
+
const left = _evaluate(ast2.left);
|
2002
|
+
if (ast2.op === "and") {
|
2003
|
+
return isTrue(left) ? _evaluate(ast2.right) : left;
|
2004
|
+
} else {
|
2005
|
+
return isTrue(left) ? left : _evaluate(ast2.right);
|
2006
|
+
}
|
2007
|
+
case 4:
|
2008
|
+
// List
|
2009
|
+
case 10:
|
2010
|
+
return ast2.value.map(_evaluate);
|
2011
|
+
case 11:
|
2012
|
+
const dict = {};
|
2013
|
+
for (const key2 in ast2.value) {
|
2014
|
+
dict[key2] = _evaluate(ast2.value[key2]);
|
2015
|
+
}
|
2016
|
+
dicts.add(dict);
|
2017
|
+
return dict;
|
2018
|
+
case 8:
|
2019
|
+
const fnValue = _evaluate(ast2.fn);
|
2020
|
+
const args = ast2.args.map(_evaluate);
|
2021
|
+
const kwargs = {};
|
2022
|
+
for (const kwarg in ast2.kwargs) {
|
2023
|
+
kwargs[kwarg] = _evaluate(ast2?.kwargs[kwarg]);
|
2024
|
+
}
|
2025
|
+
if (fnValue === PyDate || fnValue === PyDateTime || fnValue === PyTime || fnValue === PyRelativeDelta || fnValue === PyTimeDelta) {
|
2026
|
+
return fnValue.create(...args, kwargs);
|
2027
|
+
}
|
2028
|
+
return fnValue(...args, kwargs);
|
2029
|
+
case 12:
|
2030
|
+
const dictVal = _evaluate(ast2.target);
|
2031
|
+
const key = _evaluate(ast2.key);
|
2032
|
+
return dictVal[key];
|
2033
|
+
case 13:
|
2034
|
+
if (isTrue(_evaluate(ast2.condition))) {
|
2035
|
+
return _evaluate(ast2.ifTrue);
|
2036
|
+
} else {
|
2037
|
+
return _evaluate(ast2.ifFalse);
|
2038
|
+
}
|
2039
|
+
case 15:
|
2040
|
+
let leftVal = _evaluate(ast2.obj);
|
2041
|
+
let result;
|
2042
|
+
if (dicts.has(leftVal) || Object.isPrototypeOf.call(PY_DICT, leftVal)) {
|
2043
|
+
result = DICT[ast2.key];
|
2044
|
+
} else if (typeof leftVal === "string") {
|
2045
|
+
result = STRING[ast2.key];
|
2046
|
+
} else if (leftVal instanceof Set) {
|
2047
|
+
result = SET[ast2.key];
|
2048
|
+
} else if (ast2.key === "get" && typeof leftVal === "object") {
|
2049
|
+
result = DICT[ast2.key];
|
2050
|
+
leftVal = toPyDict(leftVal);
|
2051
|
+
} else {
|
2052
|
+
result = leftVal[ast2.key];
|
2053
|
+
}
|
2054
|
+
if (typeof result === "function") {
|
2055
|
+
const bound = result.bind(leftVal);
|
2056
|
+
bound[unboundFn] = result;
|
2057
|
+
return bound;
|
2058
|
+
}
|
2059
|
+
return result;
|
2060
|
+
default:
|
2061
|
+
throw new EvaluationError(`AST of type ${ast2.type} cannot be evaluated`);
|
2062
|
+
}
|
2063
|
+
}
|
2064
|
+
function _evaluate(ast2) {
|
2065
|
+
const val = _innerEvaluate(ast2);
|
2066
|
+
if (typeof val === "function" && !allowedFns.has(val) && !allowedFns.has(val[unboundFn])) {
|
2067
|
+
throw new Error("Invalid Function Call");
|
2068
|
+
}
|
2069
|
+
return val;
|
2070
|
+
}
|
2071
|
+
return _evaluate(ast);
|
2072
|
+
}
|
2073
|
+
|
2074
|
+
// src/utils/domain/py.ts
|
2075
|
+
function parseExpr(expr) {
|
2076
|
+
const tokens = tokenize(expr);
|
2077
|
+
return parse(tokens);
|
2078
|
+
}
|
2079
|
+
function evaluateExpr(expr, context = {}) {
|
2080
|
+
let ast;
|
2081
|
+
try {
|
2082
|
+
ast = parseExpr(expr);
|
2083
|
+
} catch (error) {
|
2084
|
+
throw new EvalError(
|
2085
|
+
`Can not parse python expression: (${expr})
|
2086
|
+
Error: ${error.message}`
|
2087
|
+
);
|
2088
|
+
}
|
2089
|
+
try {
|
2090
|
+
return evaluate(ast, context);
|
2091
|
+
} catch (error) {
|
2092
|
+
throw new EvalError(
|
2093
|
+
`Can not evaluate python expression: (${expr})
|
2094
|
+
Error: ${error.message}`
|
2095
|
+
);
|
2096
|
+
}
|
2097
|
+
}
|
2098
|
+
function evaluateBooleanExpr(expr, context = {}) {
|
2099
|
+
if (!expr || expr === "False" || expr === "0") {
|
2100
|
+
return false;
|
2101
|
+
}
|
2102
|
+
if (expr === "True" || expr === "1") {
|
2103
|
+
return true;
|
2104
|
+
}
|
2105
|
+
return evaluateExpr(`bool(${expr})`, context);
|
2106
|
+
}
|
2107
|
+
|
2108
|
+
// src/utils/domain/context.ts
|
2109
|
+
function evalPartialContext(_context, evaluationContext = {}) {
|
2110
|
+
const ast = parseExpr(_context);
|
2111
|
+
const context = {};
|
2112
|
+
for (const key in ast.value) {
|
2113
|
+
const value = ast.value[key];
|
2114
|
+
try {
|
2115
|
+
context[key] = evaluate(value, evaluationContext);
|
2116
|
+
} catch {
|
2117
|
+
}
|
2118
|
+
}
|
2119
|
+
return context;
|
2120
|
+
}
|
2121
|
+
|
2122
|
+
// src/utils/domain/objects.ts
|
2123
|
+
function shallowEqual(obj1, obj2, comparisonFn = (a, b) => a === b) {
|
2124
|
+
if (!obj1 || !obj2 || typeof obj1 !== "object" || typeof obj2 !== "object") {
|
2125
|
+
return obj1 === obj2;
|
2126
|
+
}
|
2127
|
+
const obj1Keys = Object.keys(obj1);
|
2128
|
+
return obj1Keys.length === Object.keys(obj2).length && obj1Keys.every((key) => comparisonFn(obj1[key], obj2[key]));
|
2129
|
+
}
|
2130
|
+
|
2131
|
+
// src/utils/domain/arrays.ts
|
2132
|
+
var shallowEqual2 = shallowEqual;
|
2133
|
+
|
2134
|
+
// src/utils/domain/strings.ts
|
2135
|
+
var escapeMethod = Symbol("html");
|
2136
|
+
function escapeRegExp(str) {
|
2137
|
+
return str.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
|
2138
|
+
}
|
2139
|
+
|
2140
|
+
// src/utils/domain/domain.ts
|
2141
|
+
var InvalidDomainError = class extends Error {
|
2142
|
+
};
|
2143
|
+
var Domain = class _Domain {
|
2144
|
+
ast = { type: -1, value: null };
|
2145
|
+
static TRUE;
|
2146
|
+
static FALSE;
|
2147
|
+
static combine(domains, operator) {
|
2148
|
+
if (domains.length === 0) {
|
2149
|
+
return new _Domain([]);
|
2150
|
+
}
|
2151
|
+
const domain1 = domains[0] instanceof _Domain ? domains[0] : new _Domain(domains[0]);
|
2152
|
+
if (domains.length === 1) {
|
2153
|
+
return domain1;
|
2154
|
+
}
|
2155
|
+
const domain2 = _Domain.combine(domains.slice(1), operator);
|
2156
|
+
const result = new _Domain([]);
|
2157
|
+
const astValues1 = domain1.ast.value;
|
2158
|
+
const astValues2 = domain2.ast.value;
|
2159
|
+
const op = operator === "AND" ? "&" : "|";
|
2160
|
+
const combinedAST = {
|
2161
|
+
type: 4,
|
2162
|
+
value: astValues1.concat(astValues2)
|
2163
|
+
};
|
2164
|
+
result.ast = normalizeDomainAST(combinedAST, op);
|
2165
|
+
return result;
|
2166
|
+
}
|
2167
|
+
static and(domains) {
|
2168
|
+
return _Domain.combine(domains, "AND");
|
2169
|
+
}
|
2170
|
+
static or(domains) {
|
2171
|
+
return _Domain.combine(domains, "OR");
|
2172
|
+
}
|
2173
|
+
static not(domain) {
|
2174
|
+
const result = new _Domain(domain);
|
2175
|
+
result.ast.value.unshift({ type: 1, value: "!" });
|
2176
|
+
return result;
|
2177
|
+
}
|
2178
|
+
static removeDomainLeaves(domain, keysToRemove) {
|
2179
|
+
function processLeaf(elements, idx, operatorCtx, newDomain2) {
|
2180
|
+
const leaf = elements[idx];
|
2181
|
+
if (leaf.type === 10) {
|
2182
|
+
if (keysToRemove.includes(leaf.value[0].value)) {
|
2183
|
+
if (operatorCtx === "&") {
|
2184
|
+
newDomain2.ast.value.push(..._Domain.TRUE.ast.value);
|
2185
|
+
} else if (operatorCtx === "|") {
|
2186
|
+
newDomain2.ast.value.push(..._Domain.FALSE.ast.value);
|
2187
|
+
}
|
2188
|
+
} else {
|
2189
|
+
newDomain2.ast.value.push(leaf);
|
2190
|
+
}
|
2191
|
+
return 1;
|
2192
|
+
} else if (leaf.type === 1) {
|
2193
|
+
if (leaf.value === "|" && elements[idx + 1].type === 10 && elements[idx + 2].type === 10 && keysToRemove.includes(elements[idx + 1].value[0].value) && keysToRemove.includes(elements[idx + 2].value[0].value)) {
|
2194
|
+
newDomain2.ast.value.push(..._Domain.TRUE.ast.value);
|
2195
|
+
return 3;
|
2196
|
+
}
|
2197
|
+
newDomain2.ast.value.push(leaf);
|
2198
|
+
if (leaf.value === "!") {
|
2199
|
+
return 1 + processLeaf(elements, idx + 1, "&", newDomain2);
|
2200
|
+
}
|
2201
|
+
const firstLeafSkip = processLeaf(
|
2202
|
+
elements,
|
2203
|
+
idx + 1,
|
2204
|
+
leaf.value,
|
2205
|
+
newDomain2
|
2206
|
+
);
|
2207
|
+
const secondLeafSkip = processLeaf(
|
2208
|
+
elements,
|
2209
|
+
idx + 1 + firstLeafSkip,
|
2210
|
+
leaf.value,
|
2211
|
+
newDomain2
|
2212
|
+
);
|
2213
|
+
return 1 + firstLeafSkip + secondLeafSkip;
|
2214
|
+
}
|
2215
|
+
return 0;
|
2216
|
+
}
|
2217
|
+
const d = new _Domain(domain);
|
2218
|
+
if (d.ast.value.length === 0) {
|
2219
|
+
return d;
|
2220
|
+
}
|
2221
|
+
const newDomain = new _Domain([]);
|
2222
|
+
processLeaf(d.ast.value, 0, "&", newDomain);
|
2223
|
+
return newDomain;
|
2224
|
+
}
|
2225
|
+
constructor(descr = []) {
|
2226
|
+
if (descr instanceof _Domain) {
|
2227
|
+
return new _Domain(descr.toString());
|
2228
|
+
} else {
|
2229
|
+
let rawAST;
|
2230
|
+
try {
|
2231
|
+
rawAST = typeof descr === "string" ? parseExpr(descr) : toAST(descr);
|
2232
|
+
} catch (error) {
|
2233
|
+
throw new InvalidDomainError(
|
2234
|
+
`Invalid domain representation: ${descr}`,
|
2235
|
+
{
|
2236
|
+
cause: error
|
2237
|
+
}
|
2238
|
+
);
|
2239
|
+
}
|
2240
|
+
this.ast = normalizeDomainAST(rawAST);
|
2241
|
+
}
|
2242
|
+
}
|
2243
|
+
contains(record) {
|
2244
|
+
const expr = evaluate(this.ast, record);
|
2245
|
+
return matchDomain(record, expr);
|
2246
|
+
}
|
2247
|
+
toString() {
|
2248
|
+
return formatAST(this.ast);
|
2249
|
+
}
|
2250
|
+
toList(context) {
|
2251
|
+
return evaluate(this.ast, context);
|
2252
|
+
}
|
2253
|
+
toJson() {
|
2254
|
+
try {
|
2255
|
+
const evaluatedAsList = this.toList({});
|
2256
|
+
const evaluatedDomain = new _Domain(evaluatedAsList);
|
2257
|
+
if (evaluatedDomain.toString() === this.toString()) {
|
2258
|
+
return evaluatedAsList;
|
2259
|
+
}
|
2260
|
+
return this.toString();
|
2261
|
+
} catch {
|
2262
|
+
return this.toString();
|
2263
|
+
}
|
2264
|
+
}
|
2265
|
+
};
|
2266
|
+
var TRUE_LEAF = [1, "=", 1];
|
2267
|
+
var FALSE_LEAF = [0, "=", 1];
|
2268
|
+
var TRUE_DOMAIN = new Domain([TRUE_LEAF]);
|
2269
|
+
var FALSE_DOMAIN = new Domain([FALSE_LEAF]);
|
2270
|
+
Domain.TRUE = TRUE_DOMAIN;
|
2271
|
+
Domain.FALSE = FALSE_DOMAIN;
|
2272
|
+
function toAST(domain) {
|
2273
|
+
const elems = domain.map((elem) => {
|
2274
|
+
switch (elem) {
|
2275
|
+
case "!":
|
2276
|
+
case "&":
|
2277
|
+
case "|":
|
2278
|
+
return { type: 1, value: elem };
|
2279
|
+
default:
|
2280
|
+
return {
|
2281
|
+
type: 10,
|
2282
|
+
value: elem.map(toPyValue)
|
2283
|
+
};
|
2284
|
+
}
|
2285
|
+
});
|
2286
|
+
return { type: 4, value: elems };
|
2287
|
+
}
|
2288
|
+
function normalizeDomainAST(domain, op = "&") {
|
2289
|
+
if (domain.type !== 4) {
|
2290
|
+
if (domain.type === 10) {
|
2291
|
+
const value = domain.value;
|
2292
|
+
if (value.findIndex((e) => e.type === 10) === -1 || !value.every((e) => e.type === 10 || e.type === 1)) {
|
2293
|
+
throw new InvalidDomainError("Invalid domain AST");
|
2294
|
+
}
|
2295
|
+
} else {
|
2296
|
+
throw new InvalidDomainError("Invalid domain AST");
|
2297
|
+
}
|
2298
|
+
}
|
2299
|
+
if (domain.value.length === 0) {
|
2300
|
+
return domain;
|
2301
|
+
}
|
2302
|
+
let expected = 1;
|
2303
|
+
for (const child of domain.value) {
|
2304
|
+
switch (child.type) {
|
2305
|
+
case 1:
|
2306
|
+
if (child.value === "&" || child.value === "|") {
|
2307
|
+
expected++;
|
2308
|
+
} else if (child.value !== "!") {
|
2309
|
+
throw new InvalidDomainError("Invalid domain AST");
|
2310
|
+
}
|
2311
|
+
break;
|
2312
|
+
case 4:
|
2313
|
+
/* list */
|
2314
|
+
case 10:
|
2315
|
+
if (child.value.length === 3) {
|
2316
|
+
expected--;
|
2317
|
+
break;
|
2318
|
+
}
|
2319
|
+
throw new InvalidDomainError("Invalid domain AST");
|
2320
|
+
default:
|
2321
|
+
throw new InvalidDomainError("Invalid domain AST");
|
2322
|
+
}
|
2323
|
+
}
|
2324
|
+
const values = domain.value.slice();
|
2325
|
+
while (expected < 0) {
|
2326
|
+
expected++;
|
2327
|
+
values.unshift({ type: 1, value: op });
|
2328
|
+
}
|
2329
|
+
if (expected > 0) {
|
2330
|
+
throw new InvalidDomainError(
|
2331
|
+
`invalid domain ${formatAST(domain)} (missing ${expected} segment(s))`
|
2332
|
+
);
|
2333
|
+
}
|
2334
|
+
return { type: 4, value: values };
|
2335
|
+
}
|
2336
|
+
function matchCondition(record, condition) {
|
2337
|
+
if (typeof condition === "boolean") {
|
2338
|
+
return condition;
|
2339
|
+
}
|
2340
|
+
const [field, operator, value] = condition;
|
2341
|
+
if (typeof field === "string") {
|
2342
|
+
const names = field.split(".");
|
2343
|
+
if (names.length >= 2) {
|
2344
|
+
return matchCondition(record[names[0]], [
|
2345
|
+
names.slice(1).join("."),
|
2346
|
+
operator,
|
2347
|
+
value
|
2348
|
+
]);
|
2349
|
+
}
|
2350
|
+
}
|
2351
|
+
let likeRegexp, ilikeRegexp;
|
2352
|
+
if (["like", "not like", "ilike", "not ilike"].includes(operator)) {
|
2353
|
+
likeRegexp = new RegExp(
|
2354
|
+
`(.*)${escapeRegExp(value).replaceAll("%", "(.*)")}(.*)`,
|
2355
|
+
"g"
|
2356
|
+
);
|
2357
|
+
ilikeRegexp = new RegExp(
|
2358
|
+
`(.*)${escapeRegExp(value).replaceAll("%", "(.*)")}(.*)`,
|
2359
|
+
"gi"
|
2360
|
+
);
|
2361
|
+
}
|
2362
|
+
const fieldValue = typeof field === "number" ? field : record[field];
|
2363
|
+
switch (operator) {
|
2364
|
+
case "=?":
|
2365
|
+
if ([false, null].includes(value)) {
|
2366
|
+
return true;
|
2367
|
+
}
|
2368
|
+
// eslint-disable-next-line no-fallthrough
|
2369
|
+
case "=":
|
2370
|
+
case "==":
|
2371
|
+
if (Array.isArray(fieldValue) && Array.isArray(value)) {
|
2372
|
+
return shallowEqual2(fieldValue, value);
|
2373
|
+
}
|
2374
|
+
return fieldValue === value;
|
2375
|
+
case "!=":
|
2376
|
+
case "<>":
|
2377
|
+
return !matchCondition(record, [field, "==", value]);
|
2378
|
+
case "<":
|
2379
|
+
return fieldValue < value;
|
2380
|
+
case "<=":
|
2381
|
+
return fieldValue <= value;
|
2382
|
+
case ">":
|
2383
|
+
return fieldValue > value;
|
2384
|
+
case ">=":
|
2385
|
+
return fieldValue >= value;
|
2386
|
+
case "in": {
|
2387
|
+
const val = Array.isArray(value) ? value : [value];
|
2388
|
+
const fieldVal = Array.isArray(fieldValue) ? fieldValue : [fieldValue];
|
2389
|
+
return fieldVal.some((fv) => val.includes(fv));
|
2390
|
+
}
|
2391
|
+
case "not in": {
|
2392
|
+
const val = Array.isArray(value) ? value : [value];
|
2393
|
+
const fieldVal = Array.isArray(fieldValue) ? fieldValue : [fieldValue];
|
2394
|
+
return !fieldVal.some((fv) => val.includes(fv));
|
2395
|
+
}
|
2396
|
+
case "like":
|
2397
|
+
if (fieldValue === false) {
|
2398
|
+
return false;
|
2399
|
+
}
|
2400
|
+
return Boolean(fieldValue.match(likeRegexp));
|
2401
|
+
case "not like":
|
2402
|
+
if (fieldValue === false) {
|
2403
|
+
return false;
|
2404
|
+
}
|
2405
|
+
return Boolean(!fieldValue.match(likeRegexp));
|
2406
|
+
case "=like":
|
2407
|
+
if (fieldValue === false) {
|
2408
|
+
return false;
|
2409
|
+
}
|
2410
|
+
return new RegExp(escapeRegExp(value).replace(/%/g, ".*")).test(
|
2411
|
+
fieldValue
|
2412
|
+
);
|
2413
|
+
case "ilike":
|
2414
|
+
if (fieldValue === false) {
|
2415
|
+
return false;
|
2416
|
+
}
|
2417
|
+
return Boolean(fieldValue.match(ilikeRegexp));
|
2418
|
+
case "not ilike":
|
2419
|
+
if (fieldValue === false) {
|
2420
|
+
return false;
|
2421
|
+
}
|
2422
|
+
return Boolean(!fieldValue.match(ilikeRegexp));
|
2423
|
+
case "=ilike":
|
2424
|
+
if (fieldValue === false) {
|
2425
|
+
return false;
|
2426
|
+
}
|
2427
|
+
return new RegExp(escapeRegExp(value).replace(/%/g, ".*"), "i").test(
|
2428
|
+
fieldValue
|
2429
|
+
);
|
2430
|
+
}
|
2431
|
+
throw new InvalidDomainError("could not match domain");
|
2432
|
+
}
|
2433
|
+
function makeOperators(record) {
|
2434
|
+
const match = matchCondition.bind(null, record);
|
2435
|
+
return {
|
2436
|
+
"!": (x) => !match(x),
|
2437
|
+
"&": (a, b) => match(a) && match(b),
|
2438
|
+
"|": (a, b) => match(a) || match(b)
|
2439
|
+
};
|
2440
|
+
}
|
2441
|
+
function matchDomain(record, domain) {
|
2442
|
+
if (domain.length === 0) {
|
2443
|
+
return true;
|
2444
|
+
}
|
2445
|
+
const operators = makeOperators(record);
|
2446
|
+
const reversedDomain = Array.from(domain).reverse();
|
2447
|
+
const condStack = [];
|
2448
|
+
for (const item of reversedDomain) {
|
2449
|
+
const operator = typeof item === "string" && operators[item];
|
2450
|
+
if (operator) {
|
2451
|
+
const operands = condStack.splice(-operator.length);
|
2452
|
+
condStack.push(operator(...operands));
|
2453
|
+
} else {
|
2454
|
+
condStack.push(item);
|
2455
|
+
}
|
2456
|
+
}
|
2457
|
+
return matchCondition(record, condStack.pop());
|
2458
|
+
}
|
2459
|
+
var checkDomain = (context, domain) => {
|
2460
|
+
try {
|
2461
|
+
if (domain === void 0 || domain === "0" || domain === "False" || domain === false) {
|
2462
|
+
return false;
|
2463
|
+
} else if (domain === "1" || domain === "True" || domain === true) {
|
2464
|
+
return true;
|
2465
|
+
}
|
2466
|
+
try {
|
2467
|
+
if (context && domain) {
|
2468
|
+
const d = new Domain(domain);
|
2469
|
+
return d.contains(context);
|
2470
|
+
}
|
2471
|
+
} catch (error) {
|
2472
|
+
if (context && domain) {
|
2473
|
+
const domainEval = evaluateBooleanExpr(domain, context);
|
2474
|
+
return domainEval;
|
2475
|
+
}
|
2476
|
+
return false;
|
2477
|
+
}
|
2478
|
+
return false;
|
2479
|
+
} catch (e) {
|
2480
|
+
return false;
|
2481
|
+
}
|
2482
|
+
};
|
2483
|
+
var matchDomains = (context, domains) => {
|
2484
|
+
if (Array.isArray(domains)) {
|
2485
|
+
if (domains?.length > 0) {
|
2486
|
+
return domains && domains.some((domain) => checkDomain(context, domain));
|
2487
|
+
}
|
2488
|
+
} else return checkDomain(context, domains);
|
2489
|
+
return false;
|
2490
|
+
};
|
2491
|
+
|
2492
|
+
// src/utils/function.ts
|
2493
|
+
var evalJSONContext = (_context, context = {}) => {
|
2494
|
+
try {
|
2495
|
+
return evalPartialContext(_context, context);
|
2496
|
+
} catch (err) {
|
2497
|
+
return null;
|
2498
|
+
}
|
2499
|
+
};
|
2500
|
+
var evalJSONDomain = (domain, context) => {
|
2501
|
+
try {
|
2502
|
+
if (context) {
|
2503
|
+
Object.keys(context)?.forEach((key) => {
|
2504
|
+
if (Array.isArray(context[key])) {
|
2505
|
+
const isTypeObject = context[key]?.every(
|
2506
|
+
(item) => typeof item === "object" && item !== null && item?.id !== void 0
|
2507
|
+
);
|
2508
|
+
if (isTypeObject) {
|
2509
|
+
context[key] = context[key]?.map((item) => item?.id);
|
2510
|
+
}
|
2511
|
+
}
|
2512
|
+
});
|
2513
|
+
}
|
2514
|
+
const value = evaluateExpr(domain, context);
|
2515
|
+
return value;
|
2516
|
+
} catch (err) {
|
2517
|
+
try {
|
2518
|
+
const domainObject = new Domain(domain).toList(context);
|
2519
|
+
return domainObject;
|
2520
|
+
} catch (err2) {
|
2521
|
+
return [];
|
2522
|
+
}
|
2523
|
+
}
|
2524
|
+
};
|
2525
|
+
var formatSortingString = (input) => {
|
2526
|
+
if (!input) return null;
|
2527
|
+
return input.split(",").map((field) => {
|
2528
|
+
const [key, order] = field.trim().split(/\s+/);
|
2529
|
+
const sortOrder = order?.toUpperCase() === "DESC" ? "DESC" : "ASC";
|
2530
|
+
return `${key} ${sortOrder}`;
|
2531
|
+
}).join(", ");
|
2532
|
+
};
|
2533
|
+
var domainHelper = {
|
2534
|
+
checkDomain,
|
2535
|
+
matchDomains,
|
2536
|
+
Domain
|
2537
|
+
};
|
2538
|
+
var toQueryString = (params) => {
|
2539
|
+
return Object.keys(params).map(
|
2540
|
+
(key) => encodeURIComponent(key) + "=" + encodeURIComponent(params[key].toString())
|
2541
|
+
).join("&");
|
2542
|
+
};
|
2543
|
+
var convertFloatToTime = (floatValue) => {
|
2544
|
+
const hours = Math.floor(floatValue);
|
2545
|
+
const minutes = Math.round((floatValue - hours) * 60);
|
2546
|
+
const formattedHours = String(hours).padStart(2, "0");
|
2547
|
+
const formattedMinutes = String(minutes).padStart(2, "0");
|
2548
|
+
return `${formattedHours}:${formattedMinutes}`;
|
2549
|
+
};
|
2550
|
+
var convertTimeToFloat = (timeString) => {
|
2551
|
+
const [hours, minutes] = timeString.split(":").map(Number);
|
2552
|
+
return hours + minutes / 60;
|
2553
|
+
};
|
2554
|
+
var stringToColor = (name, id) => {
|
2555
|
+
const combined = name + id / 2;
|
2556
|
+
let hash = 0;
|
2557
|
+
for (let i = 0; i < combined.length; i++) {
|
2558
|
+
hash = combined.charCodeAt(i) + ((hash << 5) - hash);
|
2559
|
+
}
|
2560
|
+
const r = hash >> 16 & 255;
|
2561
|
+
const g = hash >> 8 & 255;
|
2562
|
+
const b = hash & 255;
|
2563
|
+
const adjustedR = 120 + r % 61;
|
2564
|
+
const adjustedG = 120 + g % 61;
|
2565
|
+
const adjustedB = 120 + b % 61;
|
2566
|
+
return `#${adjustedR.toString(16).padStart(2, "0")}${adjustedG.toString(16).padStart(2, "0")}${adjustedB.toString(16).padStart(2, "0")}`;
|
2567
|
+
};
|
2568
|
+
var getFieldsOnChange = (fields) => {
|
2569
|
+
const result = [];
|
2570
|
+
function traverse(items) {
|
2571
|
+
for (const item of items) {
|
2572
|
+
if (item) {
|
2573
|
+
if (item?.type_co === "field" && matchDomains(fields, item?.on_change)) {
|
2574
|
+
result.push(item.name);
|
2575
|
+
}
|
2576
|
+
if (item?.fields && Array.isArray(item?.fields)) {
|
2577
|
+
traverse(item?.fields);
|
2578
|
+
}
|
2579
|
+
}
|
2580
|
+
}
|
2581
|
+
}
|
2582
|
+
traverse(fields);
|
2583
|
+
return result;
|
2584
|
+
};
|
2585
|
+
var filterFieldDirty = ({
|
2586
|
+
id,
|
2587
|
+
viewData,
|
2588
|
+
formValues,
|
2589
|
+
dirtyFields,
|
2590
|
+
model,
|
2591
|
+
defaultData
|
2592
|
+
}) => {
|
2593
|
+
const data = id ? { ...dirtyFields } : { ...formValues };
|
2594
|
+
for (const key in data) {
|
2595
|
+
if (viewData?.models?.[model]?.[key]?.type === "one2many" /* ONE2MANY */) {
|
2596
|
+
const lineData = [];
|
2597
|
+
(formValues[key] ?? []).forEach((itemData, index) => {
|
2598
|
+
if (typeof itemData?.id === "string" && itemData?.id.includes("virtual")) {
|
2599
|
+
delete itemData?.id;
|
2600
|
+
}
|
2601
|
+
if (!itemData?.id) {
|
2602
|
+
lineData.push([
|
2603
|
+
0 /* CREATE */,
|
2604
|
+
`virtual_${index}`,
|
2605
|
+
filterFieldDirty({
|
2606
|
+
id: itemData?.id,
|
2607
|
+
viewData,
|
2608
|
+
formValues: itemData,
|
2609
|
+
dirtyFields: {},
|
2610
|
+
model: viewData?.models?.[model]?.[key]?.relation,
|
2611
|
+
defaultData
|
2612
|
+
})
|
2613
|
+
]);
|
2614
|
+
} else if (dirtyFields[key]?.length) {
|
2615
|
+
dirtyFields[key].forEach((itemDirty, indexDirty) => {
|
2616
|
+
if (Object.values(itemDirty).includes(true) && indexDirty === index) {
|
2617
|
+
lineData.push([
|
2618
|
+
1 /* UPDATE */,
|
2619
|
+
itemData?.id,
|
2620
|
+
filterFieldDirty({
|
2621
|
+
id: itemData?.id,
|
2622
|
+
viewData,
|
2623
|
+
formValues: itemData,
|
2624
|
+
dirtyFields: itemDirty,
|
2625
|
+
model: viewData?.models?.[model]?.[key]?.relation,
|
2626
|
+
defaultData: {}
|
2627
|
+
})
|
2628
|
+
]);
|
2629
|
+
}
|
2630
|
+
});
|
2631
|
+
}
|
2632
|
+
});
|
2633
|
+
(defaultData[key] ?? []).forEach((item) => {
|
2634
|
+
if (!(formValues[key] ?? []).find(
|
2635
|
+
(itemData) => itemData?.id === item?.id
|
2636
|
+
)) {
|
2637
|
+
lineData.push([2 /* DELETE */, item?.id, item]);
|
2638
|
+
}
|
2639
|
+
});
|
2640
|
+
data[key] = lineData;
|
2641
|
+
} else if (viewData?.models?.[model]?.[key]?.type === "many2many" /* MANY2MANY */) {
|
2642
|
+
const lineData = [];
|
2643
|
+
(formValues[key] || []).forEach((itemData) => {
|
2644
|
+
if (itemData?.id) {
|
2645
|
+
lineData.push([4 /* NO_CHANGE */, itemData?.id]);
|
2646
|
+
}
|
2647
|
+
});
|
2648
|
+
(defaultData[key] ?? []).forEach((item) => {
|
2649
|
+
if (!(formValues[key] ?? []).find(
|
2650
|
+
(itemData) => itemData?.id === item?.id
|
2651
|
+
)) {
|
2652
|
+
lineData.push([3 /* UNLINK */, item?.id]);
|
2653
|
+
}
|
2654
|
+
});
|
2655
|
+
data[key] = lineData;
|
2656
|
+
} else {
|
2657
|
+
if (id && (typeof dirtyFields?.[key] === "object" && !dirtyFields?.[key]?.id || typeof dirtyFields[key] !== "object" && !dirtyFields[key])) {
|
2658
|
+
delete data[key];
|
2659
|
+
} else {
|
2660
|
+
if (!data[key]) {
|
2661
|
+
delete data[key];
|
2662
|
+
} else {
|
2663
|
+
data[key] = formValues?.[key]?.display_name ? formValues?.[key]?.id : formValues?.[key];
|
2664
|
+
}
|
2665
|
+
}
|
2666
|
+
}
|
2667
|
+
}
|
2668
|
+
return data;
|
2669
|
+
};
|
2670
|
+
var mergeObjects = (object1, object2) => {
|
2671
|
+
if (!object1 || !object2) return void 0;
|
2672
|
+
const mergedObject = { ...object2 };
|
2673
|
+
Object.keys(object1).forEach((key) => {
|
2674
|
+
if (Array.isArray(object1[key]) && Array.isArray(object2[key])) {
|
2675
|
+
mergedObject[key] = object2[key].map((item, index) => {
|
2676
|
+
if (object1[key][index]) {
|
2677
|
+
return {
|
2678
|
+
...item,
|
2679
|
+
...object1[key][index]
|
2680
|
+
};
|
2681
|
+
}
|
2682
|
+
return item;
|
2683
|
+
});
|
2684
|
+
} else if (typeof object1[key] === "object" && typeof object2[key] === "object" && object1[key] !== null && object2[key] !== null) {
|
2685
|
+
mergedObject[key] = mergeObjects(object1[key], object2[key]);
|
2686
|
+
} else {
|
2687
|
+
mergedObject[key] = object1[key] !== void 0 ? object1[key] : object2[key];
|
2688
|
+
}
|
2689
|
+
});
|
2690
|
+
if (object2) {
|
2691
|
+
Object.keys(object2).forEach((key) => {
|
2692
|
+
if (!mergedObject.hasOwnProperty(key)) {
|
2693
|
+
mergedObject[key] = object2[key];
|
2694
|
+
}
|
2695
|
+
});
|
2696
|
+
}
|
2697
|
+
return mergedObject;
|
2698
|
+
};
|
2699
|
+
var formatUrlPath = ({
|
2700
|
+
viewType,
|
2701
|
+
aid,
|
2702
|
+
model,
|
2703
|
+
id,
|
2704
|
+
actionPath
|
2705
|
+
}) => {
|
2706
|
+
let _url = `/${viewType}/${actionPath}?aid=${aid}&model=${model}`;
|
2707
|
+
if (id) {
|
2708
|
+
_url += `&id=${id}`;
|
2709
|
+
}
|
2710
|
+
return _url;
|
2711
|
+
};
|
2712
|
+
var removeUndefinedFields = (obj) => {
|
2713
|
+
const newObj = {};
|
2714
|
+
for (const key in obj) {
|
2715
|
+
if (obj[key] !== void 0) {
|
2716
|
+
newObj[key] = obj[key];
|
2717
|
+
}
|
2718
|
+
}
|
2719
|
+
return newObj;
|
2720
|
+
};
|
2721
|
+
var useTabModel = (viewData, onchangeData) => {
|
2722
|
+
const tabsData = viewData?.views?.form?.tabs?.filter((val) => {
|
2723
|
+
if (!val) return null;
|
2724
|
+
const hide = checkDomain(onchangeData, val.invisible);
|
2725
|
+
if (!hide) {
|
2726
|
+
return val;
|
2727
|
+
}
|
2728
|
+
return false;
|
2729
|
+
}) || [];
|
2730
|
+
return tabsData;
|
2731
|
+
};
|
2732
|
+
var isBase64File = (str) => {
|
2733
|
+
try {
|
2734
|
+
const dataUriPattern = /^data:([a-zA-Z]+\/[a-zA-Z0-9-.+]+)?;base64,/;
|
2735
|
+
if (dataUriPattern.test(str)) {
|
2736
|
+
return true;
|
2737
|
+
}
|
2738
|
+
const base64Pattern = (
|
2739
|
+
// eslint-disable-next-line no-useless-escape
|
2740
|
+
/^(?:[A-Za-z0-9+\/]{4})*(?:[A-Za-z0-9+\/]{3}=|[A-Za-z0-9+\/]{2}==)?$/
|
2741
|
+
);
|
2742
|
+
return base64Pattern.test(str);
|
2743
|
+
} catch (e) {
|
2744
|
+
return false;
|
2745
|
+
}
|
2746
|
+
};
|
2747
|
+
var isBase64Image = (str) => {
|
2748
|
+
const base64Regex = /^data:image\/(png|jpeg|jpg|gif|webp);base64,/;
|
2749
|
+
if (!base64Regex.test(str)) {
|
2750
|
+
return false;
|
2751
|
+
}
|
2752
|
+
try {
|
2753
|
+
const base64Data = str.split(",")[1];
|
2754
|
+
return !!base64Data && atob(base64Data).length > 0;
|
2755
|
+
} catch (error) {
|
2756
|
+
return false;
|
2757
|
+
}
|
2758
|
+
};
|
2759
|
+
var checkIsImageLink = (url) => {
|
2760
|
+
const imageExtensions = /\.(jpg|jpeg|png|gif|bmp|webp|svg|tiff|ico)$/i;
|
2761
|
+
return imageExtensions.test(url) || isBase64Image(url);
|
2762
|
+
};
|
2763
|
+
var formatFileSize = (size) => {
|
2764
|
+
if (size < 1024) return `${size} B`;
|
2765
|
+
const i = Math.floor(Math.log(size) / Math.log(1024));
|
2766
|
+
const sizes = ["B", "KB", "MB", "GB", "TB"];
|
2767
|
+
return `${(size / Math.pow(1024, i)).toFixed(2)} ${sizes[i]}`;
|
2768
|
+
};
|
2769
|
+
var getSubdomain = (url = window.location.href) => {
|
2770
|
+
const parts = url?.split(".");
|
2771
|
+
if (parts.length > 2) {
|
2772
|
+
return parts[0].replace("https://", "").replace("http://", "");
|
2773
|
+
}
|
2774
|
+
return null;
|
2775
|
+
};
|
2776
|
+
var resequence = (arr, start, end) => {
|
2777
|
+
if (start < 0 || start >= arr.length || end < 0 || end >= arr.length) {
|
2778
|
+
return [];
|
2779
|
+
}
|
2780
|
+
const [element] = arr.splice(start, 1);
|
2781
|
+
arr.splice(end, 0, element);
|
2782
|
+
return arr.slice(Math.min(start, end), Math.max(start, end) + 1);
|
2783
|
+
};
|
2784
|
+
var getOffSet = (arr, start, end) => {
|
2785
|
+
if (start < 0 || start >= arr.length || end < 0 || end >= arr.length) {
|
2786
|
+
return 0;
|
2787
|
+
}
|
2788
|
+
if (start > end) {
|
2789
|
+
return end;
|
2790
|
+
}
|
2791
|
+
return arr.slice(0, start).length;
|
2792
|
+
};
|
2793
|
+
var copyTextToClipboard = async (text) => {
|
2794
|
+
if ("clipboard" in navigator) {
|
2795
|
+
return await navigator.clipboard.writeText(text);
|
2796
|
+
} else {
|
2797
|
+
const textArea = document.createElement("textarea");
|
2798
|
+
textArea.value = text;
|
2799
|
+
textArea.style.position = "fixed";
|
2800
|
+
document.body.appendChild(textArea);
|
2801
|
+
textArea.focus();
|
2802
|
+
textArea.select();
|
2803
|
+
try {
|
2804
|
+
document.execCommand("copy");
|
2805
|
+
} finally {
|
2806
|
+
document.body.removeChild(textArea);
|
2807
|
+
}
|
2808
|
+
}
|
2809
|
+
};
|
2810
|
+
var isObjectEmpty = (obj) => {
|
2811
|
+
return Object.keys(obj).length === 0;
|
2812
|
+
};
|
2813
|
+
|
2814
|
+
// src/utils/storage/local-storage.ts
|
2815
|
+
var localStorageUtils = () => {
|
2816
|
+
const setToken = async (access_token) => {
|
2817
|
+
localStorage.setItem("accessToken", access_token);
|
2818
|
+
};
|
2819
|
+
const setRefreshToken = async (refresh_token) => {
|
2820
|
+
localStorage.setItem("refreshToken", refresh_token);
|
2821
|
+
};
|
2822
|
+
const getAccessToken = async () => {
|
2823
|
+
return localStorage.getItem("accessToken");
|
2824
|
+
};
|
2825
|
+
const getRefreshToken = async () => {
|
2826
|
+
return localStorage.getItem("refreshToken");
|
2827
|
+
};
|
2828
|
+
const clearToken = async () => {
|
2829
|
+
localStorage.removeItem("accessToken");
|
2830
|
+
localStorage.removeItem("refreshToken");
|
2831
|
+
};
|
2832
|
+
return {
|
2833
|
+
setToken,
|
2834
|
+
setRefreshToken,
|
2835
|
+
getAccessToken,
|
2836
|
+
getRefreshToken,
|
2837
|
+
clearToken
|
2838
|
+
};
|
2839
|
+
};
|
2840
|
+
|
2841
|
+
// src/utils/storage/session-storage.ts
|
2842
|
+
var sessionStorageUtils = () => {
|
2843
|
+
const getBrowserSession = async () => {
|
2844
|
+
return sessionStorage.getItem("browserSession");
|
2845
|
+
};
|
2846
|
+
return {
|
2847
|
+
getBrowserSession
|
2848
|
+
};
|
2849
|
+
};
|
2850
|
+
export {
|
2851
|
+
WesapError,
|
2852
|
+
checkIsImageLink,
|
2853
|
+
convertFloatToTime,
|
2854
|
+
convertTimeToFloat,
|
2855
|
+
copyTextToClipboard,
|
2856
|
+
domainHelper,
|
2857
|
+
evalJSONContext,
|
2858
|
+
evalJSONDomain,
|
2859
|
+
filterFieldDirty,
|
2860
|
+
formatCurrency,
|
2861
|
+
formatDate,
|
2862
|
+
formatFileSize,
|
2863
|
+
formatSortingString,
|
2864
|
+
formatUrlPath,
|
2865
|
+
getFieldsOnChange,
|
2866
|
+
getOffSet,
|
2867
|
+
getSubdomain,
|
2868
|
+
handleError,
|
2869
|
+
isBase64File,
|
2870
|
+
isBase64Image,
|
2871
|
+
isObjectEmpty,
|
2872
|
+
localStorageUtils,
|
2873
|
+
mergeObjects,
|
2874
|
+
removeUndefinedFields,
|
2875
|
+
resequence,
|
2876
|
+
sessionStorageUtils,
|
2877
|
+
stringToColor,
|
2878
|
+
toQueryString,
|
2879
|
+
useTabModel,
|
2880
|
+
validateAndParseDate
|
2881
|
+
};
|