shell-dsl 0.0.38 → 0.0.40
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +118 -3
- package/dist/cjs/index.cjs +2 -1
- package/dist/cjs/index.cjs.map +3 -3
- package/dist/cjs/package.json +1 -1
- package/dist/cjs/src/commands/exit/exit.cjs +84 -0
- package/dist/cjs/src/commands/exit/exit.cjs.map +10 -0
- package/dist/cjs/src/commands/index.cjs +22 -2
- package/dist/cjs/src/commands/index.cjs.map +3 -3
- package/dist/cjs/src/commands/printf/printf.cjs +416 -0
- package/dist/cjs/src/commands/printf/printf.cjs.map +10 -0
- package/dist/cjs/src/commands/sh/sh.cjs +134 -0
- package/dist/cjs/src/commands/sh/sh.cjs.map +10 -0
- package/dist/cjs/src/index.cjs +2 -1
- package/dist/cjs/src/index.cjs.map +3 -3
- package/dist/cjs/src/interpreter/context.cjs +4 -1
- package/dist/cjs/src/interpreter/context.cjs.map +3 -3
- package/dist/cjs/src/interpreter/index.cjs +2 -1
- package/dist/cjs/src/interpreter/index.cjs.map +3 -3
- package/dist/cjs/src/interpreter/interpreter.cjs +301 -76
- package/dist/cjs/src/interpreter/interpreter.cjs.map +3 -3
- package/dist/cjs/src/lexer/lexer.cjs +13 -1
- package/dist/cjs/src/lexer/lexer.cjs.map +3 -3
- package/dist/cjs/src/parser/parser.cjs +11 -1
- package/dist/cjs/src/parser/parser.cjs.map +3 -3
- package/dist/cjs/src/types.cjs.map +2 -2
- package/dist/mjs/index.mjs +3 -1
- package/dist/mjs/index.mjs.map +3 -3
- package/dist/mjs/package.json +1 -1
- package/dist/mjs/src/commands/exit/exit.mjs +44 -0
- package/dist/mjs/src/commands/exit/exit.mjs.map +10 -0
- package/dist/mjs/src/commands/index.mjs +22 -2
- package/dist/mjs/src/commands/index.mjs.map +3 -3
- package/dist/mjs/src/commands/printf/printf.mjs +376 -0
- package/dist/mjs/src/commands/printf/printf.mjs.map +10 -0
- package/dist/mjs/src/commands/sh/sh.mjs +94 -0
- package/dist/mjs/src/commands/sh/sh.mjs.map +10 -0
- package/dist/mjs/src/index.mjs +3 -2
- package/dist/mjs/src/index.mjs.map +3 -3
- package/dist/mjs/src/interpreter/context.mjs +4 -1
- package/dist/mjs/src/interpreter/context.mjs.map +3 -3
- package/dist/mjs/src/interpreter/index.mjs +3 -2
- package/dist/mjs/src/interpreter/index.mjs.map +2 -2
- package/dist/mjs/src/interpreter/interpreter.mjs +301 -76
- package/dist/mjs/src/interpreter/interpreter.mjs.map +3 -3
- package/dist/mjs/src/lexer/lexer.mjs +13 -1
- package/dist/mjs/src/lexer/lexer.mjs.map +3 -3
- package/dist/mjs/src/parser/parser.mjs +11 -1
- package/dist/mjs/src/parser/parser.mjs.map +3 -3
- package/dist/mjs/src/types.mjs.map +2 -2
- package/dist/types/index.d.ts +1 -1
- package/dist/types/src/commands/exit/exit.d.ts +2 -0
- package/dist/types/src/commands/index.d.ts +3 -0
- package/dist/types/src/commands/printf/printf.d.ts +2 -0
- package/dist/types/src/commands/sh/sh.d.ts +5 -0
- package/dist/types/src/index.d.ts +2 -2
- package/dist/types/src/interpreter/context.d.ts +2 -1
- package/dist/types/src/interpreter/index.d.ts +1 -1
- package/dist/types/src/interpreter/interpreter.d.ts +24 -0
- package/dist/types/src/types.d.ts +13 -0
- package/package.json +1 -1
|
@@ -0,0 +1,376 @@
|
|
|
1
|
+
// src/commands/printf/printf.ts
|
|
2
|
+
var INTEGER_SPECIFIERS = new Set(["d", "i", "u", "o", "x", "X"]);
|
|
3
|
+
var FLOAT_SPECIFIERS = new Set(["f", "F", "e", "E", "g", "G"]);
|
|
4
|
+
var STRING_SPECIFIERS = new Set(["s", "b", "c"]);
|
|
5
|
+
var LENGTH_MODIFIERS = new Set(["h", "l", "L", "j", "z", "t"]);
|
|
6
|
+
function isOctalDigit(char) {
|
|
7
|
+
return char !== undefined && char >= "0" && char <= "7";
|
|
8
|
+
}
|
|
9
|
+
function isDigit(char) {
|
|
10
|
+
return char !== undefined && char >= "0" && char <= "9";
|
|
11
|
+
}
|
|
12
|
+
function readEscape(input, index) {
|
|
13
|
+
if (index + 1 >= input.length) {
|
|
14
|
+
return { text: "\\", nextIndex: index + 1, stop: false };
|
|
15
|
+
}
|
|
16
|
+
const char = input[index + 1];
|
|
17
|
+
if (char === "c") {
|
|
18
|
+
return { text: "", nextIndex: index + 2, stop: true };
|
|
19
|
+
}
|
|
20
|
+
if (isOctalDigit(char)) {
|
|
21
|
+
let digits = "";
|
|
22
|
+
let nextIndex = index + 1;
|
|
23
|
+
if (input[nextIndex] === "0") {
|
|
24
|
+
nextIndex++;
|
|
25
|
+
}
|
|
26
|
+
while (digits.length < 3 && isOctalDigit(input[nextIndex])) {
|
|
27
|
+
digits += input[nextIndex];
|
|
28
|
+
nextIndex++;
|
|
29
|
+
}
|
|
30
|
+
const codePoint = digits === "" ? 0 : Number.parseInt(digits, 8);
|
|
31
|
+
return { text: String.fromCharCode(codePoint), nextIndex, stop: false };
|
|
32
|
+
}
|
|
33
|
+
if (char === "x") {
|
|
34
|
+
let digits = "";
|
|
35
|
+
let nextIndex = index + 2;
|
|
36
|
+
while (digits.length < 2 && nextIndex < input.length && /[0-9a-fA-F]/.test(input[nextIndex])) {
|
|
37
|
+
digits += input[nextIndex];
|
|
38
|
+
nextIndex++;
|
|
39
|
+
}
|
|
40
|
+
if (digits.length > 0) {
|
|
41
|
+
return { text: String.fromCharCode(Number.parseInt(digits, 16)), nextIndex, stop: false };
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
switch (char) {
|
|
45
|
+
case "a":
|
|
46
|
+
return { text: "\x07", nextIndex: index + 2, stop: false };
|
|
47
|
+
case "b":
|
|
48
|
+
return { text: "\b", nextIndex: index + 2, stop: false };
|
|
49
|
+
case "f":
|
|
50
|
+
return { text: "\f", nextIndex: index + 2, stop: false };
|
|
51
|
+
case "n":
|
|
52
|
+
return { text: `
|
|
53
|
+
`, nextIndex: index + 2, stop: false };
|
|
54
|
+
case "r":
|
|
55
|
+
return { text: "\r", nextIndex: index + 2, stop: false };
|
|
56
|
+
case "t":
|
|
57
|
+
return { text: "\t", nextIndex: index + 2, stop: false };
|
|
58
|
+
case "v":
|
|
59
|
+
return { text: "\v", nextIndex: index + 2, stop: false };
|
|
60
|
+
case "\\":
|
|
61
|
+
return { text: "\\", nextIndex: index + 2, stop: false };
|
|
62
|
+
default:
|
|
63
|
+
return { text: `\\${char}`, nextIndex: index + 2, stop: false };
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
function expandPrintfEscapes(input) {
|
|
67
|
+
let text = "";
|
|
68
|
+
for (let i = 0;i < input.length; ) {
|
|
69
|
+
if (input[i] !== "\\") {
|
|
70
|
+
text += input[i];
|
|
71
|
+
i++;
|
|
72
|
+
continue;
|
|
73
|
+
}
|
|
74
|
+
const escape = readEscape(input, i);
|
|
75
|
+
text += escape.text;
|
|
76
|
+
i = escape.nextIndex;
|
|
77
|
+
if (escape.stop) {
|
|
78
|
+
return { text, stop: true };
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
return { text, stop: false };
|
|
82
|
+
}
|
|
83
|
+
function readNumber(input, index) {
|
|
84
|
+
let digits = "";
|
|
85
|
+
let nextIndex = index;
|
|
86
|
+
while (isDigit(input[nextIndex])) {
|
|
87
|
+
digits += input[nextIndex];
|
|
88
|
+
nextIndex++;
|
|
89
|
+
}
|
|
90
|
+
return {
|
|
91
|
+
value: digits === "" ? undefined : Number.parseInt(digits, 10),
|
|
92
|
+
nextIndex
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
function parseConversion(format, index) {
|
|
96
|
+
let nextIndex = index + 1;
|
|
97
|
+
let flags = "";
|
|
98
|
+
while (true) {
|
|
99
|
+
const flag = format[nextIndex];
|
|
100
|
+
if (flag === undefined || !"-+ #0".includes(flag)) {
|
|
101
|
+
break;
|
|
102
|
+
}
|
|
103
|
+
flags += flag;
|
|
104
|
+
nextIndex++;
|
|
105
|
+
}
|
|
106
|
+
const widthResult = readNumber(format, nextIndex);
|
|
107
|
+
const width = widthResult.value;
|
|
108
|
+
nextIndex = widthResult.nextIndex;
|
|
109
|
+
let precision;
|
|
110
|
+
if (format[nextIndex] === ".") {
|
|
111
|
+
const precisionResult = readNumber(format, nextIndex + 1);
|
|
112
|
+
precision = precisionResult.value ?? 0;
|
|
113
|
+
nextIndex = precisionResult.nextIndex;
|
|
114
|
+
}
|
|
115
|
+
if (LENGTH_MODIFIERS.has(format[nextIndex] ?? "")) {
|
|
116
|
+
const modifier = format[nextIndex];
|
|
117
|
+
nextIndex++;
|
|
118
|
+
if (modifier === "h" && format[nextIndex] === "h" || modifier === "l" && format[nextIndex] === "l") {
|
|
119
|
+
nextIndex++;
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
const specifier = format[nextIndex];
|
|
123
|
+
if (specifier === undefined) {
|
|
124
|
+
return { nextIndex, error: "missing format character" };
|
|
125
|
+
}
|
|
126
|
+
if (!INTEGER_SPECIFIERS.has(specifier) && !FLOAT_SPECIFIERS.has(specifier) && !STRING_SPECIFIERS.has(specifier)) {
|
|
127
|
+
return { nextIndex: nextIndex + 1, error: `invalid format character '${specifier}'` };
|
|
128
|
+
}
|
|
129
|
+
return {
|
|
130
|
+
spec: { flags, width, precision, specifier },
|
|
131
|
+
nextIndex: nextIndex + 1
|
|
132
|
+
};
|
|
133
|
+
}
|
|
134
|
+
function pad(value, spec, numeric = false) {
|
|
135
|
+
const width = spec.width ?? 0;
|
|
136
|
+
if (value.length >= width) {
|
|
137
|
+
return value;
|
|
138
|
+
}
|
|
139
|
+
const leftAlign = spec.flags.includes("-");
|
|
140
|
+
const useZeroPad = numeric && spec.flags.includes("0") && !leftAlign && spec.precision === undefined;
|
|
141
|
+
const padChar = useZeroPad ? "0" : " ";
|
|
142
|
+
const padding = padChar.repeat(width - value.length);
|
|
143
|
+
if (leftAlign) {
|
|
144
|
+
return value + padding;
|
|
145
|
+
}
|
|
146
|
+
if (useZeroPad && (value.startsWith("-") || value.startsWith("+") || value.startsWith(" "))) {
|
|
147
|
+
return value[0] + padding + value.slice(1);
|
|
148
|
+
}
|
|
149
|
+
if (useZeroPad && (value.startsWith("0x") || value.startsWith("0X"))) {
|
|
150
|
+
return value.slice(0, 2) + padding + value.slice(2);
|
|
151
|
+
}
|
|
152
|
+
return padding + value;
|
|
153
|
+
}
|
|
154
|
+
function integerFromArg(arg) {
|
|
155
|
+
const trimmed = arg.trim();
|
|
156
|
+
if (/^[+-]?0[xX][0-9a-fA-F]+$/.test(trimmed)) {
|
|
157
|
+
const sign = trimmed.startsWith("-") ? -1 : 1;
|
|
158
|
+
return sign * Number.parseInt(trimmed.replace(/^[+-]?0[xX]/, ""), 16);
|
|
159
|
+
}
|
|
160
|
+
const parsed = Number.parseInt(trimmed, 10);
|
|
161
|
+
return Number.isNaN(parsed) ? 0 : parsed;
|
|
162
|
+
}
|
|
163
|
+
function floatFromArg(arg) {
|
|
164
|
+
const parsed = Number.parseFloat(arg.trim());
|
|
165
|
+
return Number.isNaN(parsed) ? 0 : parsed;
|
|
166
|
+
}
|
|
167
|
+
function formatInteger(arg, spec) {
|
|
168
|
+
const originalValue = Math.trunc(integerFromArg(arg));
|
|
169
|
+
const unsignedValue = originalValue < 0 ? originalValue >>> 0 : originalValue;
|
|
170
|
+
let value = spec.specifier === "u" || spec.specifier === "o" || spec.specifier === "x" || spec.specifier === "X" ? unsignedValue : originalValue;
|
|
171
|
+
let sign = "";
|
|
172
|
+
if ((spec.specifier === "d" || spec.specifier === "i") && value < 0) {
|
|
173
|
+
sign = "-";
|
|
174
|
+
value = Math.abs(value);
|
|
175
|
+
} else if ((spec.specifier === "d" || spec.specifier === "i") && spec.flags.includes("+")) {
|
|
176
|
+
sign = "+";
|
|
177
|
+
} else if ((spec.specifier === "d" || spec.specifier === "i") && spec.flags.includes(" ")) {
|
|
178
|
+
sign = " ";
|
|
179
|
+
}
|
|
180
|
+
let digits;
|
|
181
|
+
if (spec.specifier === "o") {
|
|
182
|
+
digits = value.toString(8);
|
|
183
|
+
} else if (spec.specifier === "x" || spec.specifier === "X") {
|
|
184
|
+
digits = value.toString(16);
|
|
185
|
+
if (spec.specifier === "X") {
|
|
186
|
+
digits = digits.toUpperCase();
|
|
187
|
+
}
|
|
188
|
+
} else {
|
|
189
|
+
digits = value.toString(10);
|
|
190
|
+
}
|
|
191
|
+
if (spec.precision !== undefined) {
|
|
192
|
+
if (spec.precision === 0 && value === 0) {
|
|
193
|
+
digits = "";
|
|
194
|
+
} else {
|
|
195
|
+
digits = digits.padStart(spec.precision, "0");
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
let prefix = "";
|
|
199
|
+
if (spec.flags.includes("#")) {
|
|
200
|
+
if (spec.specifier === "o" && !digits.startsWith("0")) {
|
|
201
|
+
prefix = "0";
|
|
202
|
+
} else if (spec.specifier === "x" && value !== 0) {
|
|
203
|
+
prefix = "0x";
|
|
204
|
+
} else if (spec.specifier === "X" && value !== 0) {
|
|
205
|
+
prefix = "0X";
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
return pad(sign + prefix + digits, spec, true);
|
|
209
|
+
}
|
|
210
|
+
function formatFloat(arg, spec) {
|
|
211
|
+
const value = floatFromArg(arg);
|
|
212
|
+
const precision = spec.precision ?? 6;
|
|
213
|
+
let formatted;
|
|
214
|
+
switch (spec.specifier) {
|
|
215
|
+
case "e":
|
|
216
|
+
case "E":
|
|
217
|
+
formatted = value.toExponential(precision);
|
|
218
|
+
break;
|
|
219
|
+
case "g":
|
|
220
|
+
case "G":
|
|
221
|
+
formatted = value.toPrecision(precision === 0 ? 1 : precision);
|
|
222
|
+
break;
|
|
223
|
+
default:
|
|
224
|
+
formatted = value.toFixed(precision);
|
|
225
|
+
break;
|
|
226
|
+
}
|
|
227
|
+
if (spec.specifier === "E" || spec.specifier === "G" || spec.specifier === "F") {
|
|
228
|
+
formatted = formatted.toUpperCase();
|
|
229
|
+
}
|
|
230
|
+
if (value >= 0 && spec.flags.includes("+")) {
|
|
231
|
+
formatted = `+${formatted}`;
|
|
232
|
+
} else if (value >= 0 && spec.flags.includes(" ")) {
|
|
233
|
+
formatted = ` ${formatted}`;
|
|
234
|
+
}
|
|
235
|
+
return pad(formatted, spec, true);
|
|
236
|
+
}
|
|
237
|
+
function formatString(value, spec) {
|
|
238
|
+
const truncated = spec.precision === undefined ? value : value.slice(0, spec.precision);
|
|
239
|
+
return pad(truncated, spec);
|
|
240
|
+
}
|
|
241
|
+
function renderConversion(spec, args, argIndex) {
|
|
242
|
+
const hasArg = argIndex < args.length;
|
|
243
|
+
const arg = hasArg ? args[argIndex] : "";
|
|
244
|
+
const nextArgIndex = hasArg ? argIndex + 1 : argIndex;
|
|
245
|
+
if (spec.specifier === "s") {
|
|
246
|
+
return {
|
|
247
|
+
output: formatString(arg, spec),
|
|
248
|
+
nextArgIndex,
|
|
249
|
+
consumedArg: hasArg,
|
|
250
|
+
stop: false
|
|
251
|
+
};
|
|
252
|
+
}
|
|
253
|
+
if (spec.specifier === "b") {
|
|
254
|
+
const expanded = expandPrintfEscapes(arg);
|
|
255
|
+
return {
|
|
256
|
+
output: formatString(expanded.text, spec),
|
|
257
|
+
nextArgIndex,
|
|
258
|
+
consumedArg: hasArg,
|
|
259
|
+
stop: expanded.stop
|
|
260
|
+
};
|
|
261
|
+
}
|
|
262
|
+
if (spec.specifier === "c") {
|
|
263
|
+
return {
|
|
264
|
+
output: formatString(arg.slice(0, 1), spec),
|
|
265
|
+
nextArgIndex,
|
|
266
|
+
consumedArg: hasArg,
|
|
267
|
+
stop: false
|
|
268
|
+
};
|
|
269
|
+
}
|
|
270
|
+
if (INTEGER_SPECIFIERS.has(spec.specifier)) {
|
|
271
|
+
return {
|
|
272
|
+
output: formatInteger(hasArg ? arg : "0", spec),
|
|
273
|
+
nextArgIndex,
|
|
274
|
+
consumedArg: hasArg,
|
|
275
|
+
stop: false
|
|
276
|
+
};
|
|
277
|
+
}
|
|
278
|
+
return {
|
|
279
|
+
output: formatFloat(hasArg ? arg : "0", spec),
|
|
280
|
+
nextArgIndex,
|
|
281
|
+
consumedArg: hasArg,
|
|
282
|
+
stop: false
|
|
283
|
+
};
|
|
284
|
+
}
|
|
285
|
+
function renderPass(format, args, startArgIndex) {
|
|
286
|
+
let output = "";
|
|
287
|
+
let argIndex = startArgIndex;
|
|
288
|
+
let consumedArgs = 0;
|
|
289
|
+
for (let i = 0;i < format.length; ) {
|
|
290
|
+
const char = format[i];
|
|
291
|
+
if (char === "\\") {
|
|
292
|
+
const escape = readEscape(format, i);
|
|
293
|
+
output += escape.text;
|
|
294
|
+
i = escape.nextIndex;
|
|
295
|
+
if (escape.stop) {
|
|
296
|
+
return { output, nextArgIndex: argIndex, consumedArgs, stop: true };
|
|
297
|
+
}
|
|
298
|
+
continue;
|
|
299
|
+
}
|
|
300
|
+
if (char !== "%") {
|
|
301
|
+
output += char;
|
|
302
|
+
i++;
|
|
303
|
+
continue;
|
|
304
|
+
}
|
|
305
|
+
if (format[i + 1] === "%") {
|
|
306
|
+
output += "%";
|
|
307
|
+
i += 2;
|
|
308
|
+
continue;
|
|
309
|
+
}
|
|
310
|
+
const parsed = parseConversion(format, i);
|
|
311
|
+
if (parsed.error || !parsed.spec) {
|
|
312
|
+
return {
|
|
313
|
+
output,
|
|
314
|
+
nextArgIndex: argIndex,
|
|
315
|
+
consumedArgs,
|
|
316
|
+
stop: false,
|
|
317
|
+
error: parsed.error ?? "invalid format"
|
|
318
|
+
};
|
|
319
|
+
}
|
|
320
|
+
const rendered = renderConversion(parsed.spec, args, argIndex);
|
|
321
|
+
output += rendered.output;
|
|
322
|
+
argIndex = rendered.nextArgIndex;
|
|
323
|
+
if (rendered.consumedArg) {
|
|
324
|
+
consumedArgs++;
|
|
325
|
+
}
|
|
326
|
+
i = parsed.nextIndex;
|
|
327
|
+
if (rendered.stop) {
|
|
328
|
+
return { output, nextArgIndex: argIndex, consumedArgs, stop: true };
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
return { output, nextArgIndex: argIndex, consumedArgs, stop: false };
|
|
332
|
+
}
|
|
333
|
+
function formatPrintf(format, args) {
|
|
334
|
+
let output = "";
|
|
335
|
+
let argIndex = 0;
|
|
336
|
+
let renderedAtLeastOnce = false;
|
|
337
|
+
while (!renderedAtLeastOnce || argIndex < args.length) {
|
|
338
|
+
const pass = renderPass(format, args, argIndex);
|
|
339
|
+
renderedAtLeastOnce = true;
|
|
340
|
+
output += pass.output;
|
|
341
|
+
argIndex = pass.nextArgIndex;
|
|
342
|
+
if (pass.error) {
|
|
343
|
+
return { output, error: pass.error };
|
|
344
|
+
}
|
|
345
|
+
if (pass.stop) {
|
|
346
|
+
return { output };
|
|
347
|
+
}
|
|
348
|
+
if (pass.consumedArgs === 0) {
|
|
349
|
+
break;
|
|
350
|
+
}
|
|
351
|
+
}
|
|
352
|
+
return { output };
|
|
353
|
+
}
|
|
354
|
+
var printf = async (ctx) => {
|
|
355
|
+
if (ctx.args.length === 0) {
|
|
356
|
+
await ctx.stderr.writeText(`printf: missing format operand
|
|
357
|
+
`);
|
|
358
|
+
return 1;
|
|
359
|
+
}
|
|
360
|
+
const [format, ...args] = ctx.args;
|
|
361
|
+
const result = formatPrintf(format, args);
|
|
362
|
+
if (result.output.length > 0) {
|
|
363
|
+
await ctx.stdout.writeText(result.output);
|
|
364
|
+
}
|
|
365
|
+
if (result.error) {
|
|
366
|
+
await ctx.stderr.writeText(`printf: ${result.error}
|
|
367
|
+
`);
|
|
368
|
+
return 1;
|
|
369
|
+
}
|
|
370
|
+
return 0;
|
|
371
|
+
};
|
|
372
|
+
export {
|
|
373
|
+
printf
|
|
374
|
+
};
|
|
375
|
+
|
|
376
|
+
//# debugId=E3692C39E2D017E164756E2164756E21
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../../../src/commands/printf/printf.ts"],
|
|
4
|
+
"sourcesContent": [
|
|
5
|
+
"import type { Command } from \"../../types.mjs\";\n\ninterface EscapeResult {\n text: string;\n nextIndex: number;\n stop: boolean;\n}\n\ninterface FormatResult {\n output: string;\n error?: string;\n}\n\ninterface ConversionSpec {\n flags: string;\n width?: number;\n precision?: number;\n specifier: string;\n}\n\ninterface RenderPassResult {\n output: string;\n nextArgIndex: number;\n consumedArgs: number;\n stop: boolean;\n error?: string;\n}\n\nconst INTEGER_SPECIFIERS = new Set([\"d\", \"i\", \"u\", \"o\", \"x\", \"X\"]);\nconst FLOAT_SPECIFIERS = new Set([\"f\", \"F\", \"e\", \"E\", \"g\", \"G\"]);\nconst STRING_SPECIFIERS = new Set([\"s\", \"b\", \"c\"]);\nconst LENGTH_MODIFIERS = new Set([\"h\", \"l\", \"L\", \"j\", \"z\", \"t\"]);\n\nfunction isOctalDigit(char: string | undefined): boolean {\n return char !== undefined && char >= \"0\" && char <= \"7\";\n}\n\nfunction isDigit(char: string | undefined): boolean {\n return char !== undefined && char >= \"0\" && char <= \"9\";\n}\n\nfunction readEscape(input: string, index: number): EscapeResult {\n if (index + 1 >= input.length) {\n return { text: \"\\\\\", nextIndex: index + 1, stop: false };\n }\n\n const char = input[index + 1]!;\n\n if (char === \"c\") {\n return { text: \"\", nextIndex: index + 2, stop: true };\n }\n\n if (isOctalDigit(char)) {\n let digits = \"\";\n let nextIndex = index + 1;\n\n if (input[nextIndex] === \"0\") {\n nextIndex++;\n }\n\n while (digits.length < 3 && isOctalDigit(input[nextIndex])) {\n digits += input[nextIndex]!;\n nextIndex++;\n }\n\n const codePoint = digits === \"\" ? 0 : Number.parseInt(digits, 8);\n return { text: String.fromCharCode(codePoint), nextIndex, stop: false };\n }\n\n if (char === \"x\") {\n let digits = \"\";\n let nextIndex = index + 2;\n\n while (digits.length < 2 && nextIndex < input.length && /[0-9a-fA-F]/.test(input[nextIndex]!)) {\n digits += input[nextIndex]!;\n nextIndex++;\n }\n\n if (digits.length > 0) {\n return { text: String.fromCharCode(Number.parseInt(digits, 16)), nextIndex, stop: false };\n }\n }\n\n switch (char) {\n case \"a\":\n return { text: \"\\x07\", nextIndex: index + 2, stop: false };\n case \"b\":\n return { text: \"\\b\", nextIndex: index + 2, stop: false };\n case \"f\":\n return { text: \"\\f\", nextIndex: index + 2, stop: false };\n case \"n\":\n return { text: \"\\n\", nextIndex: index + 2, stop: false };\n case \"r\":\n return { text: \"\\r\", nextIndex: index + 2, stop: false };\n case \"t\":\n return { text: \"\\t\", nextIndex: index + 2, stop: false };\n case \"v\":\n return { text: \"\\v\", nextIndex: index + 2, stop: false };\n case \"\\\\\":\n return { text: \"\\\\\", nextIndex: index + 2, stop: false };\n default:\n return { text: `\\\\${char}`, nextIndex: index + 2, stop: false };\n }\n}\n\nfunction expandPrintfEscapes(input: string): { text: string; stop: boolean } {\n let text = \"\";\n\n for (let i = 0; i < input.length;) {\n if (input[i] !== \"\\\\\") {\n text += input[i]!;\n i++;\n continue;\n }\n\n const escape = readEscape(input, i);\n text += escape.text;\n i = escape.nextIndex;\n\n if (escape.stop) {\n return { text, stop: true };\n }\n }\n\n return { text, stop: false };\n}\n\nfunction readNumber(input: string, index: number): { value?: number; nextIndex: number } {\n let digits = \"\";\n let nextIndex = index;\n\n while (isDigit(input[nextIndex])) {\n digits += input[nextIndex]!;\n nextIndex++;\n }\n\n return {\n value: digits === \"\" ? undefined : Number.parseInt(digits, 10),\n nextIndex,\n };\n}\n\nfunction parseConversion(format: string, index: number): { spec?: ConversionSpec; nextIndex: number; error?: string } {\n let nextIndex = index + 1;\n let flags = \"\";\n\n while (true) {\n const flag = format[nextIndex];\n if (flag === undefined || !\"-+ #0\".includes(flag)) {\n break;\n }\n flags += flag;\n nextIndex++;\n }\n\n const widthResult = readNumber(format, nextIndex);\n const width = widthResult.value;\n nextIndex = widthResult.nextIndex;\n\n let precision: number | undefined;\n if (format[nextIndex] === \".\") {\n const precisionResult = readNumber(format, nextIndex + 1);\n precision = precisionResult.value ?? 0;\n nextIndex = precisionResult.nextIndex;\n }\n\n if (LENGTH_MODIFIERS.has(format[nextIndex] ?? \"\")) {\n const modifier = format[nextIndex]!;\n nextIndex++;\n if ((modifier === \"h\" && format[nextIndex] === \"h\") || (modifier === \"l\" && format[nextIndex] === \"l\")) {\n nextIndex++;\n }\n }\n\n const specifier = format[nextIndex];\n if (specifier === undefined) {\n return { nextIndex, error: \"missing format character\" };\n }\n\n if (\n !INTEGER_SPECIFIERS.has(specifier) &&\n !FLOAT_SPECIFIERS.has(specifier) &&\n !STRING_SPECIFIERS.has(specifier)\n ) {\n return { nextIndex: nextIndex + 1, error: `invalid format character '${specifier}'` };\n }\n\n return {\n spec: { flags, width, precision, specifier },\n nextIndex: nextIndex + 1,\n };\n}\n\nfunction pad(value: string, spec: ConversionSpec, numeric = false): string {\n const width = spec.width ?? 0;\n if (value.length >= width) {\n return value;\n }\n\n const leftAlign = spec.flags.includes(\"-\");\n const useZeroPad = numeric && spec.flags.includes(\"0\") && !leftAlign && spec.precision === undefined;\n const padChar = useZeroPad ? \"0\" : \" \";\n const padding = padChar.repeat(width - value.length);\n\n if (leftAlign) {\n return value + padding;\n }\n\n if (useZeroPad && (value.startsWith(\"-\") || value.startsWith(\"+\") || value.startsWith(\" \"))) {\n return value[0]! + padding + value.slice(1);\n }\n\n if (useZeroPad && (value.startsWith(\"0x\") || value.startsWith(\"0X\"))) {\n return value.slice(0, 2) + padding + value.slice(2);\n }\n\n return padding + value;\n}\n\nfunction integerFromArg(arg: string): number {\n const trimmed = arg.trim();\n if (/^[+-]?0[xX][0-9a-fA-F]+$/.test(trimmed)) {\n const sign = trimmed.startsWith(\"-\") ? -1 : 1;\n return sign * Number.parseInt(trimmed.replace(/^[+-]?0[xX]/, \"\"), 16);\n }\n\n const parsed = Number.parseInt(trimmed, 10);\n return Number.isNaN(parsed) ? 0 : parsed;\n}\n\nfunction floatFromArg(arg: string): number {\n const parsed = Number.parseFloat(arg.trim());\n return Number.isNaN(parsed) ? 0 : parsed;\n}\n\nfunction formatInteger(arg: string, spec: ConversionSpec): string {\n const originalValue = Math.trunc(integerFromArg(arg));\n const unsignedValue = originalValue < 0 ? originalValue >>> 0 : originalValue;\n let value = spec.specifier === \"u\" || spec.specifier === \"o\" || spec.specifier === \"x\" || spec.specifier === \"X\"\n ? unsignedValue\n : originalValue;\n\n let sign = \"\";\n if ((spec.specifier === \"d\" || spec.specifier === \"i\") && value < 0) {\n sign = \"-\";\n value = Math.abs(value);\n } else if ((spec.specifier === \"d\" || spec.specifier === \"i\") && spec.flags.includes(\"+\")) {\n sign = \"+\";\n } else if ((spec.specifier === \"d\" || spec.specifier === \"i\") && spec.flags.includes(\" \")) {\n sign = \" \";\n }\n\n let digits: string;\n if (spec.specifier === \"o\") {\n digits = value.toString(8);\n } else if (spec.specifier === \"x\" || spec.specifier === \"X\") {\n digits = value.toString(16);\n if (spec.specifier === \"X\") {\n digits = digits.toUpperCase();\n }\n } else {\n digits = value.toString(10);\n }\n\n if (spec.precision !== undefined) {\n if (spec.precision === 0 && value === 0) {\n digits = \"\";\n } else {\n digits = digits.padStart(spec.precision, \"0\");\n }\n }\n\n let prefix = \"\";\n if (spec.flags.includes(\"#\")) {\n if (spec.specifier === \"o\" && !digits.startsWith(\"0\")) {\n prefix = \"0\";\n } else if (spec.specifier === \"x\" && value !== 0) {\n prefix = \"0x\";\n } else if (spec.specifier === \"X\" && value !== 0) {\n prefix = \"0X\";\n }\n }\n\n return pad(sign + prefix + digits, spec, true);\n}\n\nfunction formatFloat(arg: string, spec: ConversionSpec): string {\n const value = floatFromArg(arg);\n const precision = spec.precision ?? 6;\n let formatted: string;\n\n switch (spec.specifier) {\n case \"e\":\n case \"E\":\n formatted = value.toExponential(precision);\n break;\n case \"g\":\n case \"G\":\n formatted = value.toPrecision(precision === 0 ? 1 : precision);\n break;\n default:\n formatted = value.toFixed(precision);\n break;\n }\n\n if (spec.specifier === \"E\" || spec.specifier === \"G\" || spec.specifier === \"F\") {\n formatted = formatted.toUpperCase();\n }\n\n if (value >= 0 && spec.flags.includes(\"+\")) {\n formatted = `+${formatted}`;\n } else if (value >= 0 && spec.flags.includes(\" \")) {\n formatted = ` ${formatted}`;\n }\n\n return pad(formatted, spec, true);\n}\n\nfunction formatString(value: string, spec: ConversionSpec): string {\n const truncated = spec.precision === undefined ? value : value.slice(0, spec.precision);\n return pad(truncated, spec);\n}\n\nfunction renderConversion(\n spec: ConversionSpec,\n args: string[],\n argIndex: number\n): { output: string; nextArgIndex: number; consumedArg: boolean; stop: boolean } {\n const hasArg = argIndex < args.length;\n const arg = hasArg ? args[argIndex]! : \"\";\n const nextArgIndex = hasArg ? argIndex + 1 : argIndex;\n\n if (spec.specifier === \"s\") {\n return {\n output: formatString(arg, spec),\n nextArgIndex,\n consumedArg: hasArg,\n stop: false,\n };\n }\n\n if (spec.specifier === \"b\") {\n const expanded = expandPrintfEscapes(arg);\n return {\n output: formatString(expanded.text, spec),\n nextArgIndex,\n consumedArg: hasArg,\n stop: expanded.stop,\n };\n }\n\n if (spec.specifier === \"c\") {\n return {\n output: formatString(arg.slice(0, 1), spec),\n nextArgIndex,\n consumedArg: hasArg,\n stop: false,\n };\n }\n\n if (INTEGER_SPECIFIERS.has(spec.specifier)) {\n return {\n output: formatInteger(hasArg ? arg : \"0\", spec),\n nextArgIndex,\n consumedArg: hasArg,\n stop: false,\n };\n }\n\n return {\n output: formatFloat(hasArg ? arg : \"0\", spec),\n nextArgIndex,\n consumedArg: hasArg,\n stop: false,\n };\n}\n\nfunction renderPass(format: string, args: string[], startArgIndex: number): RenderPassResult {\n let output = \"\";\n let argIndex = startArgIndex;\n let consumedArgs = 0;\n\n for (let i = 0; i < format.length;) {\n const char = format[i]!;\n\n if (char === \"\\\\\") {\n const escape = readEscape(format, i);\n output += escape.text;\n i = escape.nextIndex;\n\n if (escape.stop) {\n return { output, nextArgIndex: argIndex, consumedArgs, stop: true };\n }\n continue;\n }\n\n if (char !== \"%\") {\n output += char;\n i++;\n continue;\n }\n\n if (format[i + 1] === \"%\") {\n output += \"%\";\n i += 2;\n continue;\n }\n\n const parsed = parseConversion(format, i);\n if (parsed.error || !parsed.spec) {\n return {\n output,\n nextArgIndex: argIndex,\n consumedArgs,\n stop: false,\n error: parsed.error ?? \"invalid format\",\n };\n }\n\n const rendered = renderConversion(parsed.spec, args, argIndex);\n output += rendered.output;\n argIndex = rendered.nextArgIndex;\n if (rendered.consumedArg) {\n consumedArgs++;\n }\n i = parsed.nextIndex;\n\n if (rendered.stop) {\n return { output, nextArgIndex: argIndex, consumedArgs, stop: true };\n }\n }\n\n return { output, nextArgIndex: argIndex, consumedArgs, stop: false };\n}\n\nfunction formatPrintf(format: string, args: string[]): FormatResult {\n let output = \"\";\n let argIndex = 0;\n let renderedAtLeastOnce = false;\n\n while (!renderedAtLeastOnce || argIndex < args.length) {\n const pass = renderPass(format, args, argIndex);\n renderedAtLeastOnce = true;\n output += pass.output;\n argIndex = pass.nextArgIndex;\n\n if (pass.error) {\n return { output, error: pass.error };\n }\n\n if (pass.stop) {\n return { output };\n }\n\n if (pass.consumedArgs === 0) {\n break;\n }\n }\n\n return { output };\n}\n\nexport const printf: Command = async (ctx) => {\n if (ctx.args.length === 0) {\n await ctx.stderr.writeText(\"printf: missing format operand\\n\");\n return 1;\n }\n\n const [format, ...args] = ctx.args;\n const result = formatPrintf(format!, args);\n\n if (result.output.length > 0) {\n await ctx.stdout.writeText(result.output);\n }\n\n if (result.error) {\n await ctx.stderr.writeText(`printf: ${result.error}\\n`);\n return 1;\n }\n\n return 0;\n};\n"
|
|
6
|
+
],
|
|
7
|
+
"mappings": ";AA4BA,IAAM,qBAAqB,IAAI,IAAI,CAAC,KAAK,KAAK,KAAK,KAAK,KAAK,GAAG,CAAC;AACjE,IAAM,mBAAmB,IAAI,IAAI,CAAC,KAAK,KAAK,KAAK,KAAK,KAAK,GAAG,CAAC;AAC/D,IAAM,oBAAoB,IAAI,IAAI,CAAC,KAAK,KAAK,GAAG,CAAC;AACjD,IAAM,mBAAmB,IAAI,IAAI,CAAC,KAAK,KAAK,KAAK,KAAK,KAAK,GAAG,CAAC;AAE/D,SAAS,YAAY,CAAC,MAAmC;AAAA,EACvD,OAAO,SAAS,aAAa,QAAQ,OAAO,QAAQ;AAAA;AAGtD,SAAS,OAAO,CAAC,MAAmC;AAAA,EAClD,OAAO,SAAS,aAAa,QAAQ,OAAO,QAAQ;AAAA;AAGtD,SAAS,UAAU,CAAC,OAAe,OAA6B;AAAA,EAC9D,IAAI,QAAQ,KAAK,MAAM,QAAQ;AAAA,IAC7B,OAAO,EAAE,MAAM,MAAM,WAAW,QAAQ,GAAG,MAAM,MAAM;AAAA,EACzD;AAAA,EAEA,MAAM,OAAO,MAAM,QAAQ;AAAA,EAE3B,IAAI,SAAS,KAAK;AAAA,IAChB,OAAO,EAAE,MAAM,IAAI,WAAW,QAAQ,GAAG,MAAM,KAAK;AAAA,EACtD;AAAA,EAEA,IAAI,aAAa,IAAI,GAAG;AAAA,IACtB,IAAI,SAAS;AAAA,IACb,IAAI,YAAY,QAAQ;AAAA,IAExB,IAAI,MAAM,eAAe,KAAK;AAAA,MAC5B;AAAA,IACF;AAAA,IAEA,OAAO,OAAO,SAAS,KAAK,aAAa,MAAM,UAAU,GAAG;AAAA,MAC1D,UAAU,MAAM;AAAA,MAChB;AAAA,IACF;AAAA,IAEA,MAAM,YAAY,WAAW,KAAK,IAAI,OAAO,SAAS,QAAQ,CAAC;AAAA,IAC/D,OAAO,EAAE,MAAM,OAAO,aAAa,SAAS,GAAG,WAAW,MAAM,MAAM;AAAA,EACxE;AAAA,EAEA,IAAI,SAAS,KAAK;AAAA,IAChB,IAAI,SAAS;AAAA,IACb,IAAI,YAAY,QAAQ;AAAA,IAExB,OAAO,OAAO,SAAS,KAAK,YAAY,MAAM,UAAU,cAAc,KAAK,MAAM,UAAW,GAAG;AAAA,MAC7F,UAAU,MAAM;AAAA,MAChB;AAAA,IACF;AAAA,IAEA,IAAI,OAAO,SAAS,GAAG;AAAA,MACrB,OAAO,EAAE,MAAM,OAAO,aAAa,OAAO,SAAS,QAAQ,EAAE,CAAC,GAAG,WAAW,MAAM,MAAM;AAAA,IAC1F;AAAA,EACF;AAAA,EAEA,QAAQ;AAAA,SACD;AAAA,MACH,OAAO,EAAE,MAAM,QAAQ,WAAW,QAAQ,GAAG,MAAM,MAAM;AAAA,SACtD;AAAA,MACH,OAAO,EAAE,MAAM,MAAM,WAAW,QAAQ,GAAG,MAAM,MAAM;AAAA,SACpD;AAAA,MACH,OAAO,EAAE,MAAM,MAAM,WAAW,QAAQ,GAAG,MAAM,MAAM;AAAA,SACpD;AAAA,MACH,OAAO,EAAE,MAAM;AAAA,GAAM,WAAW,QAAQ,GAAG,MAAM,MAAM;AAAA,SACpD;AAAA,MACH,OAAO,EAAE,MAAM,MAAM,WAAW,QAAQ,GAAG,MAAM,MAAM;AAAA,SACpD;AAAA,MACH,OAAO,EAAE,MAAM,MAAM,WAAW,QAAQ,GAAG,MAAM,MAAM;AAAA,SACpD;AAAA,MACH,OAAO,EAAE,MAAM,MAAM,WAAW,QAAQ,GAAG,MAAM,MAAM;AAAA,SACpD;AAAA,MACH,OAAO,EAAE,MAAM,MAAM,WAAW,QAAQ,GAAG,MAAM,MAAM;AAAA;AAAA,MAEvD,OAAO,EAAE,MAAM,KAAK,QAAQ,WAAW,QAAQ,GAAG,MAAM,MAAM;AAAA;AAAA;AAIpE,SAAS,mBAAmB,CAAC,OAAgD;AAAA,EAC3E,IAAI,OAAO;AAAA,EAEX,SAAS,IAAI,EAAG,IAAI,MAAM,UAAS;AAAA,IACjC,IAAI,MAAM,OAAO,MAAM;AAAA,MACrB,QAAQ,MAAM;AAAA,MACd;AAAA,MACA;AAAA,IACF;AAAA,IAEA,MAAM,SAAS,WAAW,OAAO,CAAC;AAAA,IAClC,QAAQ,OAAO;AAAA,IACf,IAAI,OAAO;AAAA,IAEX,IAAI,OAAO,MAAM;AAAA,MACf,OAAO,EAAE,MAAM,MAAM,KAAK;AAAA,IAC5B;AAAA,EACF;AAAA,EAEA,OAAO,EAAE,MAAM,MAAM,MAAM;AAAA;AAG7B,SAAS,UAAU,CAAC,OAAe,OAAsD;AAAA,EACvF,IAAI,SAAS;AAAA,EACb,IAAI,YAAY;AAAA,EAEhB,OAAO,QAAQ,MAAM,UAAU,GAAG;AAAA,IAChC,UAAU,MAAM;AAAA,IAChB;AAAA,EACF;AAAA,EAEA,OAAO;AAAA,IACL,OAAO,WAAW,KAAK,YAAY,OAAO,SAAS,QAAQ,EAAE;AAAA,IAC7D;AAAA,EACF;AAAA;AAGF,SAAS,eAAe,CAAC,QAAgB,OAA6E;AAAA,EACpH,IAAI,YAAY,QAAQ;AAAA,EACxB,IAAI,QAAQ;AAAA,EAEZ,OAAO,MAAM;AAAA,IACX,MAAM,OAAO,OAAO;AAAA,IACpB,IAAI,SAAS,aAAa,CAAC,QAAQ,SAAS,IAAI,GAAG;AAAA,MACjD;AAAA,IACF;AAAA,IACA,SAAS;AAAA,IACT;AAAA,EACF;AAAA,EAEA,MAAM,cAAc,WAAW,QAAQ,SAAS;AAAA,EAChD,MAAM,QAAQ,YAAY;AAAA,EAC1B,YAAY,YAAY;AAAA,EAExB,IAAI;AAAA,EACJ,IAAI,OAAO,eAAe,KAAK;AAAA,IAC7B,MAAM,kBAAkB,WAAW,QAAQ,YAAY,CAAC;AAAA,IACxD,YAAY,gBAAgB,SAAS;AAAA,IACrC,YAAY,gBAAgB;AAAA,EAC9B;AAAA,EAEA,IAAI,iBAAiB,IAAI,OAAO,cAAc,EAAE,GAAG;AAAA,IACjD,MAAM,WAAW,OAAO;AAAA,IACxB;AAAA,IACA,IAAK,aAAa,OAAO,OAAO,eAAe,OAAS,aAAa,OAAO,OAAO,eAAe,KAAM;AAAA,MACtG;AAAA,IACF;AAAA,EACF;AAAA,EAEA,MAAM,YAAY,OAAO;AAAA,EACzB,IAAI,cAAc,WAAW;AAAA,IAC3B,OAAO,EAAE,WAAW,OAAO,2BAA2B;AAAA,EACxD;AAAA,EAEA,IACE,CAAC,mBAAmB,IAAI,SAAS,KACjC,CAAC,iBAAiB,IAAI,SAAS,KAC/B,CAAC,kBAAkB,IAAI,SAAS,GAChC;AAAA,IACA,OAAO,EAAE,WAAW,YAAY,GAAG,OAAO,6BAA6B,aAAa;AAAA,EACtF;AAAA,EAEA,OAAO;AAAA,IACL,MAAM,EAAE,OAAO,OAAO,WAAW,UAAU;AAAA,IAC3C,WAAW,YAAY;AAAA,EACzB;AAAA;AAGF,SAAS,GAAG,CAAC,OAAe,MAAsB,UAAU,OAAe;AAAA,EACzE,MAAM,QAAQ,KAAK,SAAS;AAAA,EAC5B,IAAI,MAAM,UAAU,OAAO;AAAA,IACzB,OAAO;AAAA,EACT;AAAA,EAEA,MAAM,YAAY,KAAK,MAAM,SAAS,GAAG;AAAA,EACzC,MAAM,aAAa,WAAW,KAAK,MAAM,SAAS,GAAG,KAAK,CAAC,aAAa,KAAK,cAAc;AAAA,EAC3F,MAAM,UAAU,aAAa,MAAM;AAAA,EACnC,MAAM,UAAU,QAAQ,OAAO,QAAQ,MAAM,MAAM;AAAA,EAEnD,IAAI,WAAW;AAAA,IACb,OAAO,QAAQ;AAAA,EACjB;AAAA,EAEA,IAAI,eAAe,MAAM,WAAW,GAAG,KAAK,MAAM,WAAW,GAAG,KAAK,MAAM,WAAW,GAAG,IAAI;AAAA,IAC3F,OAAO,MAAM,KAAM,UAAU,MAAM,MAAM,CAAC;AAAA,EAC5C;AAAA,EAEA,IAAI,eAAe,MAAM,WAAW,IAAI,KAAK,MAAM,WAAW,IAAI,IAAI;AAAA,IACpE,OAAO,MAAM,MAAM,GAAG,CAAC,IAAI,UAAU,MAAM,MAAM,CAAC;AAAA,EACpD;AAAA,EAEA,OAAO,UAAU;AAAA;AAGnB,SAAS,cAAc,CAAC,KAAqB;AAAA,EAC3C,MAAM,UAAU,IAAI,KAAK;AAAA,EACzB,IAAI,2BAA2B,KAAK,OAAO,GAAG;AAAA,IAC5C,MAAM,OAAO,QAAQ,WAAW,GAAG,IAAI,KAAK;AAAA,IAC5C,OAAO,OAAO,OAAO,SAAS,QAAQ,QAAQ,eAAe,EAAE,GAAG,EAAE;AAAA,EACtE;AAAA,EAEA,MAAM,SAAS,OAAO,SAAS,SAAS,EAAE;AAAA,EAC1C,OAAO,OAAO,MAAM,MAAM,IAAI,IAAI;AAAA;AAGpC,SAAS,YAAY,CAAC,KAAqB;AAAA,EACzC,MAAM,SAAS,OAAO,WAAW,IAAI,KAAK,CAAC;AAAA,EAC3C,OAAO,OAAO,MAAM,MAAM,IAAI,IAAI;AAAA;AAGpC,SAAS,aAAa,CAAC,KAAa,MAA8B;AAAA,EAChE,MAAM,gBAAgB,KAAK,MAAM,eAAe,GAAG,CAAC;AAAA,EACpD,MAAM,gBAAgB,gBAAgB,IAAI,kBAAkB,IAAI;AAAA,EAChE,IAAI,QAAQ,KAAK,cAAc,OAAO,KAAK,cAAc,OAAO,KAAK,cAAc,OAAO,KAAK,cAAc,MACzG,gBACA;AAAA,EAEJ,IAAI,OAAO;AAAA,EACX,KAAK,KAAK,cAAc,OAAO,KAAK,cAAc,QAAQ,QAAQ,GAAG;AAAA,IACnE,OAAO;AAAA,IACP,QAAQ,KAAK,IAAI,KAAK;AAAA,EACxB,EAAO,UAAK,KAAK,cAAc,OAAO,KAAK,cAAc,QAAQ,KAAK,MAAM,SAAS,GAAG,GAAG;AAAA,IACzF,OAAO;AAAA,EACT,EAAO,UAAK,KAAK,cAAc,OAAO,KAAK,cAAc,QAAQ,KAAK,MAAM,SAAS,GAAG,GAAG;AAAA,IACzF,OAAO;AAAA,EACT;AAAA,EAEA,IAAI;AAAA,EACJ,IAAI,KAAK,cAAc,KAAK;AAAA,IAC1B,SAAS,MAAM,SAAS,CAAC;AAAA,EAC3B,EAAO,SAAI,KAAK,cAAc,OAAO,KAAK,cAAc,KAAK;AAAA,IAC3D,SAAS,MAAM,SAAS,EAAE;AAAA,IAC1B,IAAI,KAAK,cAAc,KAAK;AAAA,MAC1B,SAAS,OAAO,YAAY;AAAA,IAC9B;AAAA,EACF,EAAO;AAAA,IACL,SAAS,MAAM,SAAS,EAAE;AAAA;AAAA,EAG5B,IAAI,KAAK,cAAc,WAAW;AAAA,IAChC,IAAI,KAAK,cAAc,KAAK,UAAU,GAAG;AAAA,MACvC,SAAS;AAAA,IACX,EAAO;AAAA,MACL,SAAS,OAAO,SAAS,KAAK,WAAW,GAAG;AAAA;AAAA,EAEhD;AAAA,EAEA,IAAI,SAAS;AAAA,EACb,IAAI,KAAK,MAAM,SAAS,GAAG,GAAG;AAAA,IAC5B,IAAI,KAAK,cAAc,OAAO,CAAC,OAAO,WAAW,GAAG,GAAG;AAAA,MACrD,SAAS;AAAA,IACX,EAAO,SAAI,KAAK,cAAc,OAAO,UAAU,GAAG;AAAA,MAChD,SAAS;AAAA,IACX,EAAO,SAAI,KAAK,cAAc,OAAO,UAAU,GAAG;AAAA,MAChD,SAAS;AAAA,IACX;AAAA,EACF;AAAA,EAEA,OAAO,IAAI,OAAO,SAAS,QAAQ,MAAM,IAAI;AAAA;AAG/C,SAAS,WAAW,CAAC,KAAa,MAA8B;AAAA,EAC9D,MAAM,QAAQ,aAAa,GAAG;AAAA,EAC9B,MAAM,YAAY,KAAK,aAAa;AAAA,EACpC,IAAI;AAAA,EAEJ,QAAQ,KAAK;AAAA,SACN;AAAA,SACA;AAAA,MACH,YAAY,MAAM,cAAc,SAAS;AAAA,MACzC;AAAA,SACG;AAAA,SACA;AAAA,MACH,YAAY,MAAM,YAAY,cAAc,IAAI,IAAI,SAAS;AAAA,MAC7D;AAAA;AAAA,MAEA,YAAY,MAAM,QAAQ,SAAS;AAAA,MACnC;AAAA;AAAA,EAGJ,IAAI,KAAK,cAAc,OAAO,KAAK,cAAc,OAAO,KAAK,cAAc,KAAK;AAAA,IAC9E,YAAY,UAAU,YAAY;AAAA,EACpC;AAAA,EAEA,IAAI,SAAS,KAAK,KAAK,MAAM,SAAS,GAAG,GAAG;AAAA,IAC1C,YAAY,IAAI;AAAA,EAClB,EAAO,SAAI,SAAS,KAAK,KAAK,MAAM,SAAS,GAAG,GAAG;AAAA,IACjD,YAAY,IAAI;AAAA,EAClB;AAAA,EAEA,OAAO,IAAI,WAAW,MAAM,IAAI;AAAA;AAGlC,SAAS,YAAY,CAAC,OAAe,MAA8B;AAAA,EACjE,MAAM,YAAY,KAAK,cAAc,YAAY,QAAQ,MAAM,MAAM,GAAG,KAAK,SAAS;AAAA,EACtF,OAAO,IAAI,WAAW,IAAI;AAAA;AAG5B,SAAS,gBAAgB,CACvB,MACA,MACA,UAC+E;AAAA,EAC/E,MAAM,SAAS,WAAW,KAAK;AAAA,EAC/B,MAAM,MAAM,SAAS,KAAK,YAAa;AAAA,EACvC,MAAM,eAAe,SAAS,WAAW,IAAI;AAAA,EAE7C,IAAI,KAAK,cAAc,KAAK;AAAA,IAC1B,OAAO;AAAA,MACL,QAAQ,aAAa,KAAK,IAAI;AAAA,MAC9B;AAAA,MACA,aAAa;AAAA,MACb,MAAM;AAAA,IACR;AAAA,EACF;AAAA,EAEA,IAAI,KAAK,cAAc,KAAK;AAAA,IAC1B,MAAM,WAAW,oBAAoB,GAAG;AAAA,IACxC,OAAO;AAAA,MACL,QAAQ,aAAa,SAAS,MAAM,IAAI;AAAA,MACxC;AAAA,MACA,aAAa;AAAA,MACb,MAAM,SAAS;AAAA,IACjB;AAAA,EACF;AAAA,EAEA,IAAI,KAAK,cAAc,KAAK;AAAA,IAC1B,OAAO;AAAA,MACL,QAAQ,aAAa,IAAI,MAAM,GAAG,CAAC,GAAG,IAAI;AAAA,MAC1C;AAAA,MACA,aAAa;AAAA,MACb,MAAM;AAAA,IACR;AAAA,EACF;AAAA,EAEA,IAAI,mBAAmB,IAAI,KAAK,SAAS,GAAG;AAAA,IAC1C,OAAO;AAAA,MACL,QAAQ,cAAc,SAAS,MAAM,KAAK,IAAI;AAAA,MAC9C;AAAA,MACA,aAAa;AAAA,MACb,MAAM;AAAA,IACR;AAAA,EACF;AAAA,EAEA,OAAO;AAAA,IACL,QAAQ,YAAY,SAAS,MAAM,KAAK,IAAI;AAAA,IAC5C;AAAA,IACA,aAAa;AAAA,IACb,MAAM;AAAA,EACR;AAAA;AAGF,SAAS,UAAU,CAAC,QAAgB,MAAgB,eAAyC;AAAA,EAC3F,IAAI,SAAS;AAAA,EACb,IAAI,WAAW;AAAA,EACf,IAAI,eAAe;AAAA,EAEnB,SAAS,IAAI,EAAG,IAAI,OAAO,UAAS;AAAA,IAClC,MAAM,OAAO,OAAO;AAAA,IAEpB,IAAI,SAAS,MAAM;AAAA,MACjB,MAAM,SAAS,WAAW,QAAQ,CAAC;AAAA,MACnC,UAAU,OAAO;AAAA,MACjB,IAAI,OAAO;AAAA,MAEX,IAAI,OAAO,MAAM;AAAA,QACf,OAAO,EAAE,QAAQ,cAAc,UAAU,cAAc,MAAM,KAAK;AAAA,MACpE;AAAA,MACA;AAAA,IACF;AAAA,IAEA,IAAI,SAAS,KAAK;AAAA,MAChB,UAAU;AAAA,MACV;AAAA,MACA;AAAA,IACF;AAAA,IAEA,IAAI,OAAO,IAAI,OAAO,KAAK;AAAA,MACzB,UAAU;AAAA,MACV,KAAK;AAAA,MACL;AAAA,IACF;AAAA,IAEA,MAAM,SAAS,gBAAgB,QAAQ,CAAC;AAAA,IACxC,IAAI,OAAO,SAAS,CAAC,OAAO,MAAM;AAAA,MAChC,OAAO;AAAA,QACL;AAAA,QACA,cAAc;AAAA,QACd;AAAA,QACA,MAAM;AAAA,QACN,OAAO,OAAO,SAAS;AAAA,MACzB;AAAA,IACF;AAAA,IAEA,MAAM,WAAW,iBAAiB,OAAO,MAAM,MAAM,QAAQ;AAAA,IAC7D,UAAU,SAAS;AAAA,IACnB,WAAW,SAAS;AAAA,IACpB,IAAI,SAAS,aAAa;AAAA,MACxB;AAAA,IACF;AAAA,IACA,IAAI,OAAO;AAAA,IAEX,IAAI,SAAS,MAAM;AAAA,MACjB,OAAO,EAAE,QAAQ,cAAc,UAAU,cAAc,MAAM,KAAK;AAAA,IACpE;AAAA,EACF;AAAA,EAEA,OAAO,EAAE,QAAQ,cAAc,UAAU,cAAc,MAAM,MAAM;AAAA;AAGrE,SAAS,YAAY,CAAC,QAAgB,MAA8B;AAAA,EAClE,IAAI,SAAS;AAAA,EACb,IAAI,WAAW;AAAA,EACf,IAAI,sBAAsB;AAAA,EAE1B,OAAO,CAAC,uBAAuB,WAAW,KAAK,QAAQ;AAAA,IACrD,MAAM,OAAO,WAAW,QAAQ,MAAM,QAAQ;AAAA,IAC9C,sBAAsB;AAAA,IACtB,UAAU,KAAK;AAAA,IACf,WAAW,KAAK;AAAA,IAEhB,IAAI,KAAK,OAAO;AAAA,MACd,OAAO,EAAE,QAAQ,OAAO,KAAK,MAAM;AAAA,IACrC;AAAA,IAEA,IAAI,KAAK,MAAM;AAAA,MACb,OAAO,EAAE,OAAO;AAAA,IAClB;AAAA,IAEA,IAAI,KAAK,iBAAiB,GAAG;AAAA,MAC3B;AAAA,IACF;AAAA,EACF;AAAA,EAEA,OAAO,EAAE,OAAO;AAAA;AAGX,IAAM,SAAkB,OAAO,QAAQ;AAAA,EAC5C,IAAI,IAAI,KAAK,WAAW,GAAG;AAAA,IACzB,MAAM,IAAI,OAAO,UAAU;AAAA,CAAkC;AAAA,IAC7D,OAAO;AAAA,EACT;AAAA,EAEA,OAAO,WAAW,QAAQ,IAAI;AAAA,EAC9B,MAAM,SAAS,aAAa,QAAS,IAAI;AAAA,EAEzC,IAAI,OAAO,OAAO,SAAS,GAAG;AAAA,IAC5B,MAAM,IAAI,OAAO,UAAU,OAAO,MAAM;AAAA,EAC1C;AAAA,EAEA,IAAI,OAAO,OAAO;AAAA,IAChB,MAAM,IAAI,OAAO,UAAU,WAAW,OAAO;AAAA,CAAS;AAAA,IACtD,OAAO;AAAA,EACT;AAAA,EAEA,OAAO;AAAA;",
|
|
8
|
+
"debugId": "E3692C39E2D017E164756E2164756E21",
|
|
9
|
+
"names": []
|
|
10
|
+
}
|
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
// src/commands/sh/sh.ts
|
|
2
|
+
async function readShellFile(fs, cwd, pathName, stderr) {
|
|
3
|
+
const path = fs.resolve(cwd, pathName);
|
|
4
|
+
if (!await fs.exists(path)) {
|
|
5
|
+
await stderr.writeText(`sh: ${pathName}: No such file or directory
|
|
6
|
+
`);
|
|
7
|
+
return { ok: false, exitCode: 127 };
|
|
8
|
+
}
|
|
9
|
+
const stat = await fs.stat(path);
|
|
10
|
+
if (stat.isDirectory()) {
|
|
11
|
+
await stderr.writeText(`sh: ${pathName}: is a directory
|
|
12
|
+
`);
|
|
13
|
+
return { ok: false, exitCode: 126 };
|
|
14
|
+
}
|
|
15
|
+
if (!stat.isFile()) {
|
|
16
|
+
await stderr.writeText(`sh: ${pathName}: not a file
|
|
17
|
+
`);
|
|
18
|
+
return { ok: false, exitCode: 126 };
|
|
19
|
+
}
|
|
20
|
+
try {
|
|
21
|
+
return { ok: true, source: await fs.readFile(path, "utf-8") };
|
|
22
|
+
} catch (err) {
|
|
23
|
+
const message = err instanceof Error ? err.message : String(err);
|
|
24
|
+
await stderr.writeText(`sh: ${pathName}: ${message}
|
|
25
|
+
`);
|
|
26
|
+
return { ok: false, exitCode: 126 };
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
var sh = async (ctx) => {
|
|
30
|
+
if (!ctx.shell) {
|
|
31
|
+
await ctx.stderr.writeText(`sh: shell evaluation not supported
|
|
32
|
+
`);
|
|
33
|
+
return 1;
|
|
34
|
+
}
|
|
35
|
+
if (ctx.args.length === 0) {
|
|
36
|
+
return ctx.shell.runShell(await ctx.stdin.text(), { argv0: "sh", args: [] });
|
|
37
|
+
}
|
|
38
|
+
const first = ctx.args[0];
|
|
39
|
+
if (first === "-c") {
|
|
40
|
+
const source = ctx.args[1];
|
|
41
|
+
if (source === undefined) {
|
|
42
|
+
await ctx.stderr.writeText(`sh: -c requires an argument
|
|
43
|
+
`);
|
|
44
|
+
return 2;
|
|
45
|
+
}
|
|
46
|
+
const argv0 = ctx.args[2] ?? "sh";
|
|
47
|
+
const args = ctx.args[2] === undefined ? [] : ctx.args.slice(3);
|
|
48
|
+
return ctx.shell.runShell(source, { argv0, args });
|
|
49
|
+
}
|
|
50
|
+
if (first.startsWith("-")) {
|
|
51
|
+
await ctx.stderr.writeText(`sh: unsupported option: ${first}
|
|
52
|
+
`);
|
|
53
|
+
return 2;
|
|
54
|
+
}
|
|
55
|
+
const loaded = await readShellFile(ctx.fs, ctx.cwd, first, ctx.stderr);
|
|
56
|
+
if (!loaded.ok) {
|
|
57
|
+
return loaded.exitCode;
|
|
58
|
+
}
|
|
59
|
+
return ctx.shell.runShell(loaded.source, { argv0: first, args: ctx.args.slice(1) });
|
|
60
|
+
};
|
|
61
|
+
var evalCmd = async (ctx) => {
|
|
62
|
+
if (!ctx.shell) {
|
|
63
|
+
await ctx.stderr.writeText(`eval: shell evaluation not supported
|
|
64
|
+
`);
|
|
65
|
+
return 1;
|
|
66
|
+
}
|
|
67
|
+
if (ctx.args.length === 0) {
|
|
68
|
+
return 0;
|
|
69
|
+
}
|
|
70
|
+
return ctx.shell.eval(ctx.args.join(" "));
|
|
71
|
+
};
|
|
72
|
+
var source = async (ctx) => {
|
|
73
|
+
if (!ctx.shell) {
|
|
74
|
+
await ctx.stderr.writeText(`source: shell evaluation not supported
|
|
75
|
+
`);
|
|
76
|
+
return 1;
|
|
77
|
+
}
|
|
78
|
+
const path = ctx.args[0];
|
|
79
|
+
if (path === undefined) {
|
|
80
|
+
await ctx.stderr.writeText(`source: filename argument required
|
|
81
|
+
`);
|
|
82
|
+
return 2;
|
|
83
|
+
}
|
|
84
|
+
return ctx.shell.source(path, ctx.args.slice(1));
|
|
85
|
+
};
|
|
86
|
+
var dot = source;
|
|
87
|
+
export {
|
|
88
|
+
source,
|
|
89
|
+
sh,
|
|
90
|
+
evalCmd,
|
|
91
|
+
dot
|
|
92
|
+
};
|
|
93
|
+
|
|
94
|
+
//# debugId=F711664330E99B5E64756E2164756E21
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../../../src/commands/sh/sh.ts"],
|
|
4
|
+
"sourcesContent": [
|
|
5
|
+
"import type { Command, Stderr, VirtualFS } from \"../../types.mjs\";\n\nasync function readShellFile(\n fs: VirtualFS,\n cwd: string,\n pathName: string,\n stderr: Stderr\n): Promise<{ ok: true; source: string } | { ok: false; exitCode: number }> {\n const path = fs.resolve(cwd, pathName);\n\n if (!(await fs.exists(path))) {\n await stderr.writeText(`sh: ${pathName}: No such file or directory\\n`);\n return { ok: false, exitCode: 127 };\n }\n\n const stat = await fs.stat(path);\n if (stat.isDirectory()) {\n await stderr.writeText(`sh: ${pathName}: is a directory\\n`);\n return { ok: false, exitCode: 126 };\n }\n if (!stat.isFile()) {\n await stderr.writeText(`sh: ${pathName}: not a file\\n`);\n return { ok: false, exitCode: 126 };\n }\n\n try {\n return { ok: true, source: await fs.readFile(path, \"utf-8\") };\n } catch (err) {\n const message = err instanceof Error ? err.message : String(err);\n await stderr.writeText(`sh: ${pathName}: ${message}\\n`);\n return { ok: false, exitCode: 126 };\n }\n}\n\nexport const sh: Command = async (ctx) => {\n if (!ctx.shell) {\n await ctx.stderr.writeText(\"sh: shell evaluation not supported\\n\");\n return 1;\n }\n\n if (ctx.args.length === 0) {\n return ctx.shell.runShell(await ctx.stdin.text(), { argv0: \"sh\", args: [] });\n }\n\n const first = ctx.args[0]!;\n if (first === \"-c\") {\n const source = ctx.args[1];\n if (source === undefined) {\n await ctx.stderr.writeText(\"sh: -c requires an argument\\n\");\n return 2;\n }\n\n const argv0 = ctx.args[2] ?? \"sh\";\n const args = ctx.args[2] === undefined ? [] : ctx.args.slice(3);\n return ctx.shell.runShell(source, { argv0, args });\n }\n\n if (first.startsWith(\"-\")) {\n await ctx.stderr.writeText(`sh: unsupported option: ${first}\\n`);\n return 2;\n }\n\n const loaded = await readShellFile(ctx.fs, ctx.cwd, first, ctx.stderr);\n if (!loaded.ok) {\n return loaded.exitCode;\n }\n\n return ctx.shell.runShell(loaded.source, { argv0: first, args: ctx.args.slice(1) });\n};\n\nexport const evalCmd: Command = async (ctx) => {\n if (!ctx.shell) {\n await ctx.stderr.writeText(\"eval: shell evaluation not supported\\n\");\n return 1;\n }\n if (ctx.args.length === 0) {\n return 0;\n }\n return ctx.shell.eval(ctx.args.join(\" \"));\n};\n\nexport const source: Command = async (ctx) => {\n if (!ctx.shell) {\n await ctx.stderr.writeText(\"source: shell evaluation not supported\\n\");\n return 1;\n }\n const path = ctx.args[0];\n if (path === undefined) {\n await ctx.stderr.writeText(\"source: filename argument required\\n\");\n return 2;\n }\n return ctx.shell.source(path, ctx.args.slice(1));\n};\n\nexport const dot = source;\n"
|
|
6
|
+
],
|
|
7
|
+
"mappings": "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",
|
|
8
|
+
"debugId": "F711664330E99B5E64756E2164756E21",
|
|
9
|
+
"names": []
|
|
10
|
+
}
|
package/dist/mjs/src/index.mjs
CHANGED
|
@@ -18,7 +18,7 @@ import {
|
|
|
18
18
|
isUntilNode,
|
|
19
19
|
isCaseNode
|
|
20
20
|
} from "./parser/index.mjs";
|
|
21
|
-
import { Interpreter, BreakException, ContinueException } from "./interpreter/index.mjs";
|
|
21
|
+
import { Interpreter, BreakException, ContinueException, ExitException } from "./interpreter/index.mjs";
|
|
22
22
|
import { createVirtualFS } from "./fs/index.mjs";
|
|
23
23
|
import {
|
|
24
24
|
FileSystem,
|
|
@@ -71,8 +71,9 @@ export {
|
|
|
71
71
|
LexError,
|
|
72
72
|
Interpreter,
|
|
73
73
|
FileSystem,
|
|
74
|
+
ExitException,
|
|
74
75
|
ContinueException,
|
|
75
76
|
BreakException
|
|
76
77
|
};
|
|
77
78
|
|
|
78
|
-
//# debugId=
|
|
79
|
+
//# debugId=D29830CA96A8F15764756E2164756E21
|
|
@@ -2,9 +2,9 @@
|
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/index.ts"],
|
|
4
4
|
"sourcesContent": [
|
|
5
|
-
"// Main class exports\nexport { ShellDSL, createShellDSL, type Program } from \"./shell-dsl.mjs\";\nexport { ShellPromise, type ShellPromiseOptions } from \"./shell-promise.mjs\";\n\n// Types\nexport type {\n VirtualFS,\n VirtualFSWritable,\n FileStat,\n Command,\n CommandContext,\n Stdin,\n Stdout,\n Stderr,\n OutputCollector,\n ExecResult,\n ShellConfig,\n RawValue,\n} from \"./types.mjs\";\nexport { isRawValue } from \"./types.mjs\";\n\n// Errors\nexport { ShellError, LexError, ParseError } from \"./errors.mjs\";\n\n// Lexer\nexport { Lexer, lex, tokenToString } from \"./lexer/index.mjs\";\nexport type { Token, RedirectMode } from \"./lexer/index.mjs\";\n\n// Parser\nexport { Parser, parse } from \"./parser/index.mjs\";\nexport type {\n ASTNode,\n Redirect,\n CommandNode,\n PipelineNode,\n AndNode,\n OrNode,\n SequenceNode,\n WordNode,\n WordPart,\n TextPart,\n VariablePart,\n SubstitutionPart,\n ArithmeticPart,\n IfNode,\n ForNode,\n WhileNode,\n UntilNode,\n CaseNode,\n CaseClause,\n} from \"./parser/index.mjs\";\nexport {\n isWordNode,\n isCommandNode,\n isPipelineNode,\n isAndNode,\n isOrNode,\n isSequenceNode,\n isIfNode,\n isForNode,\n isWhileNode,\n isUntilNode,\n isCaseNode,\n} from \"./parser/index.mjs\";\n\n// Interpreter\nexport { Interpreter, type InterpreterOptions, BreakException, ContinueException } from \"./interpreter/index.mjs\";\n\n// Filesystem\nexport { createVirtualFS } from \"./fs/index.mjs\";\nexport {\n FileSystem,\n ReadOnlyFileSystem,\n WebFileSystem,\n createWebUnderlyingFS,\n type PathOps,\n type Permission,\n type PermissionRules,\n type UnderlyingFS,\n} from \"./fs/index.mjs\";\n\n// I/O\nexport { createStdin, StdinImpl } from \"./io/index.mjs\";\nexport { createStdout, createStderr, createPipe, OutputCollectorImpl, PipeBuffer } from \"./io/index.mjs\";\n\n// Utilities\nexport { escape, escapeForInterpolation, globVirtualFS } from \"./utils/index.mjs\";\nexport type { GlobVirtualFS, GlobOptions } from \"./utils/index.mjs\";\n\n// Version Control\nexport { VersionControlSystem } from \"./vcs/index.mjs\";\nexport type {\n VCSConfig,\n VCSAttributeRule,\n VCSResolvedAttributes,\n VCSDiffMode,\n VCSPatchSuppressionReason,\n Revision,\n DiffEntry,\n TreeManifest,\n TreeEntry,\n FileEntry,\n DirectoryEntry,\n VCSIndexEntry,\n VCSIndexFile,\n CommitOptions,\n CheckoutOptions,\n LogOptions,\n LogEntry,\n BranchInfo,\n} from \"./vcs/index.mjs\";\n"
|
|
5
|
+
"// Main class exports\nexport { ShellDSL, createShellDSL, type Program } from \"./shell-dsl.mjs\";\nexport { ShellPromise, type ShellPromiseOptions } from \"./shell-promise.mjs\";\n\n// Types\nexport type {\n VirtualFS,\n VirtualFSWritable,\n FileStat,\n Command,\n CommandContext,\n Stdin,\n Stdout,\n Stderr,\n OutputCollector,\n ExecResult,\n ShellConfig,\n ShellCommandApi,\n ShellRunOptions,\n RawValue,\n} from \"./types.mjs\";\nexport { isRawValue } from \"./types.mjs\";\n\n// Errors\nexport { ShellError, LexError, ParseError } from \"./errors.mjs\";\n\n// Lexer\nexport { Lexer, lex, tokenToString } from \"./lexer/index.mjs\";\nexport type { Token, RedirectMode } from \"./lexer/index.mjs\";\n\n// Parser\nexport { Parser, parse } from \"./parser/index.mjs\";\nexport type {\n ASTNode,\n Redirect,\n CommandNode,\n PipelineNode,\n AndNode,\n OrNode,\n SequenceNode,\n WordNode,\n WordPart,\n TextPart,\n VariablePart,\n SubstitutionPart,\n ArithmeticPart,\n IfNode,\n ForNode,\n WhileNode,\n UntilNode,\n CaseNode,\n CaseClause,\n} from \"./parser/index.mjs\";\nexport {\n isWordNode,\n isCommandNode,\n isPipelineNode,\n isAndNode,\n isOrNode,\n isSequenceNode,\n isIfNode,\n isForNode,\n isWhileNode,\n isUntilNode,\n isCaseNode,\n} from \"./parser/index.mjs\";\n\n// Interpreter\nexport { Interpreter, type InterpreterOptions, BreakException, ContinueException, ExitException } from \"./interpreter/index.mjs\";\n\n// Filesystem\nexport { createVirtualFS } from \"./fs/index.mjs\";\nexport {\n FileSystem,\n ReadOnlyFileSystem,\n WebFileSystem,\n createWebUnderlyingFS,\n type PathOps,\n type Permission,\n type PermissionRules,\n type UnderlyingFS,\n} from \"./fs/index.mjs\";\n\n// I/O\nexport { createStdin, StdinImpl } from \"./io/index.mjs\";\nexport { createStdout, createStderr, createPipe, OutputCollectorImpl, PipeBuffer } from \"./io/index.mjs\";\n\n// Utilities\nexport { escape, escapeForInterpolation, globVirtualFS } from \"./utils/index.mjs\";\nexport type { GlobVirtualFS, GlobOptions } from \"./utils/index.mjs\";\n\n// Version Control\nexport { VersionControlSystem } from \"./vcs/index.mjs\";\nexport type {\n VCSConfig,\n VCSAttributeRule,\n VCSResolvedAttributes,\n VCSDiffMode,\n VCSPatchSuppressionReason,\n Revision,\n DiffEntry,\n TreeManifest,\n TreeEntry,\n FileEntry,\n DirectoryEntry,\n VCSIndexEntry,\n VCSIndexFile,\n CommitOptions,\n CheckoutOptions,\n LogOptions,\n LogEntry,\n BranchInfo,\n} from \"./vcs/index.mjs\";\n"
|
|
6
6
|
],
|
|
7
|
-
"mappings": ";AACA;AACA;
|
|
8
|
-
"debugId": "
|
|
7
|
+
"mappings": ";AACA;AACA;AAmBA;AAGA;AAGA;AAIA;AAsBA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAeA;AAGA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAYA;AACA;AAGA;AAIA;",
|
|
8
|
+
"debugId": "D29830CA96A8F15764756E2164756E21",
|
|
9
9
|
"names": []
|
|
10
10
|
}
|
|
@@ -13,10 +13,13 @@ function createCommandContext(options) {
|
|
|
13
13
|
if (options.exec) {
|
|
14
14
|
ctx.exec = options.exec;
|
|
15
15
|
}
|
|
16
|
+
if (options.shell) {
|
|
17
|
+
ctx.shell = options.shell;
|
|
18
|
+
}
|
|
16
19
|
return ctx;
|
|
17
20
|
}
|
|
18
21
|
export {
|
|
19
22
|
createCommandContext
|
|
20
23
|
};
|
|
21
24
|
|
|
22
|
-
//# debugId=
|
|
25
|
+
//# debugId=114011848381283E64756E2164756E21
|
|
@@ -2,9 +2,9 @@
|
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/interpreter/context.ts"],
|
|
4
4
|
"sourcesContent": [
|
|
5
|
-
"import type { CommandContext, VirtualFS, Stdin, Stdout, Stderr, ExecResult } from \"../types.mjs\";\n\nexport interface ContextOptions {\n args: string[];\n stdin: Stdin;\n stdout: Stdout;\n stderr: Stderr;\n fs: VirtualFS;\n cwd: string;\n env: Record<string, string>;\n setCwd: (path: string) => void;\n exec?: (name: string, args: string[]) => Promise<ExecResult>;\n}\n\nexport function createCommandContext(options: ContextOptions): CommandContext {\n const ctx: CommandContext = {\n args: options.args,\n stdin: options.stdin,\n stdout: options.stdout,\n stderr: options.stderr,\n fs: options.fs,\n cwd: options.cwd,\n env: options.env,\n setCwd: options.setCwd,\n };\n if (options.exec) {\n ctx.exec = options.exec;\n }\n return ctx;\n}\n"
|
|
5
|
+
"import type { CommandContext, VirtualFS, Stdin, Stdout, Stderr, ExecResult, ShellCommandApi } from \"../types.mjs\";\n\nexport interface ContextOptions {\n args: string[];\n stdin: Stdin;\n stdout: Stdout;\n stderr: Stderr;\n fs: VirtualFS;\n cwd: string;\n env: Record<string, string>;\n setCwd: (path: string) => void;\n exec?: (name: string, args: string[]) => Promise<ExecResult>;\n shell?: ShellCommandApi;\n}\n\nexport function createCommandContext(options: ContextOptions): CommandContext {\n const ctx: CommandContext = {\n args: options.args,\n stdin: options.stdin,\n stdout: options.stdout,\n stderr: options.stderr,\n fs: options.fs,\n cwd: options.cwd,\n env: options.env,\n setCwd: options.setCwd,\n };\n if (options.exec) {\n ctx.exec = options.exec;\n }\n if (options.shell) {\n ctx.shell = options.shell;\n }\n return ctx;\n}\n"
|
|
6
6
|
],
|
|
7
|
-
"mappings": ";
|
|
8
|
-
"debugId": "
|
|
7
|
+
"mappings": ";AAeO,SAAS,oBAAoB,CAAC,SAAyC;AAAA,EAC5E,MAAM,MAAsB;AAAA,IAC1B,MAAM,QAAQ;AAAA,IACd,OAAO,QAAQ;AAAA,IACf,QAAQ,QAAQ;AAAA,IAChB,QAAQ,QAAQ;AAAA,IAChB,IAAI,QAAQ;AAAA,IACZ,KAAK,QAAQ;AAAA,IACb,KAAK,QAAQ;AAAA,IACb,QAAQ,QAAQ;AAAA,EAClB;AAAA,EACA,IAAI,QAAQ,MAAM;AAAA,IAChB,IAAI,OAAO,QAAQ;AAAA,EACrB;AAAA,EACA,IAAI,QAAQ,OAAO;AAAA,IACjB,IAAI,QAAQ,QAAQ;AAAA,EACtB;AAAA,EACA,OAAO;AAAA;",
|
|
8
|
+
"debugId": "114011848381283E64756E2164756E21",
|
|
9
9
|
"names": []
|
|
10
10
|
}
|
|
@@ -1,11 +1,12 @@
|
|
|
1
1
|
// src/interpreter/index.ts
|
|
2
|
-
import { Interpreter, BreakException, ContinueException } from "./interpreter.mjs";
|
|
2
|
+
import { Interpreter, BreakException, ContinueException, ExitException } from "./interpreter.mjs";
|
|
3
3
|
import { createCommandContext } from "./context.mjs";
|
|
4
4
|
export {
|
|
5
5
|
createCommandContext,
|
|
6
6
|
Interpreter,
|
|
7
|
+
ExitException,
|
|
7
8
|
ContinueException,
|
|
8
9
|
BreakException
|
|
9
10
|
};
|
|
10
11
|
|
|
11
|
-
//# debugId=
|
|
12
|
+
//# debugId=3382F60717545D4664756E2164756E21
|
|
@@ -2,9 +2,9 @@
|
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../../src/interpreter/index.ts"],
|
|
4
4
|
"sourcesContent": [
|
|
5
|
-
"export { Interpreter, type InterpreterOptions, BreakException, ContinueException } from \"./interpreter.mjs\";\nexport { createCommandContext, type ContextOptions } from \"./context.mjs\";\n"
|
|
5
|
+
"export { Interpreter, type InterpreterOptions, BreakException, ContinueException, ExitException } from \"./interpreter.mjs\";\nexport { createCommandContext, type ContextOptions } from \"./context.mjs\";\n"
|
|
6
6
|
],
|
|
7
7
|
"mappings": ";AAAA;AACA;",
|
|
8
|
-
"debugId": "
|
|
8
|
+
"debugId": "3382F60717545D4664756E2164756E21",
|
|
9
9
|
"names": []
|
|
10
10
|
}
|