typescript-virtual-container 1.5.3 → 1.5.5
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 +44 -532
- package/dist/.tsbuildinfo +1 -0
- package/dist/SSHMimic/executor.js +23 -5
- package/dist/VirtualPackageManager/index.js +10 -0
- package/dist/VirtualShell/shell.js +158 -11
- package/dist/commands/basename.d.ts +13 -0
- package/dist/commands/basename.js +45 -0
- package/dist/commands/bc.d.ts +2 -0
- package/dist/commands/bc.js +28 -0
- package/dist/commands/file.d.ts +8 -0
- package/dist/commands/file.js +57 -0
- package/dist/commands/fun.d.ts +32 -0
- package/dist/commands/fun.js +172 -0
- package/dist/commands/ip.d.ts +7 -0
- package/dist/commands/ip.js +52 -0
- package/dist/commands/jobs.d.ts +4 -0
- package/dist/commands/jobs.js +27 -0
- package/dist/commands/last.d.ts +13 -0
- package/dist/commands/last.js +68 -0
- package/dist/commands/manuals-bundle.js +598 -6
- package/dist/commands/registry.js +30 -2
- package/dist/commands/runtime.js +24 -3
- package/dist/commands/set.js +20 -0
- package/dist/commands/sh.js +74 -1
- package/dist/commands/tput.d.ts +13 -0
- package/dist/commands/tput.js +76 -0
- package/dist/commands/w.d.ts +7 -0
- package/dist/commands/w.js +38 -0
- package/dist/utils/expand.d.ts +12 -0
- package/dist/utils/expand.js +87 -1
- package/package.json +9 -3
- package/.github/ISSUE_TEMPLATE/bug_report.yml +0 -50
- package/.github/ISSUE_TEMPLATE/feature_request.yml +0 -31
- package/.github/dependabot.yml +0 -27
- package/.github/pull_request_template.md +0 -21
- package/.github/workflows/create-pull-request.yml +0 -85
- package/.github/workflows/publish.yml +0 -25
- package/.github/workflows/test-battery.yml +0 -102
- package/.vscode/settings.json +0 -20
- package/CODE_OF_CONDUCT.md +0 -39
- package/CONTRIBUTING.md +0 -59
- package/HONEYPOT.md +0 -358
- package/SECURITY.md +0 -33
- package/benchmark-results.txt +0 -40
- package/benchmark-virtualshell.ts +0 -88
- package/biome.json +0 -37
- package/build.js +0 -22
- package/builds/fortune-nyx-v1.5.3-directbash-k6.1.0.mjs +0 -1764
- package/builds/fortune-nyx-v1.5.3-ssh-nosftp.js +0 -1764
- package/builds/fortune-nyx-v1.5.3-ssh.cjs +0 -1765
- package/builds/fortune-nyx-v1.5.3-web.min.js +0 -17036
- package/bun.lock +0 -244
- package/docs/.nojekyll +0 -1
- package/docs/app.js +0 -1751
- package/docs/assets/hierarchy.js +0 -1
- package/docs/assets/highlight.css +0 -162
- package/docs/assets/icons.js +0 -18
- package/docs/assets/icons.svg +0 -1
- package/docs/assets/main.js +0 -60
- package/docs/assets/navigation.js +0 -1
- package/docs/assets/search.js +0 -1
- package/docs/assets/style.css +0 -1633
- package/docs/classes/HoneyPot.html +0 -31
- package/docs/classes/IdleManager.html +0 -162
- package/docs/classes/SshClient.html +0 -66
- package/docs/classes/VirtualFileSystem.html +0 -279
- package/docs/classes/VirtualPackageManager.html +0 -63
- package/docs/classes/VirtualSftpServer.html +0 -169
- package/docs/classes/VirtualShell.html +0 -285
- package/docs/classes/VirtualSshServer.html +0 -182
- package/docs/classes/VirtualUserManager.html +0 -276
- package/docs/demo.html +0 -82
- package/docs/functions/assertDiff.html +0 -6
- package/docs/functions/diffSnapshots.html +0 -7
- package/docs/functions/formatDiff.html +0 -6
- package/docs/functions/getArg.html +0 -13
- package/docs/functions/getFlag.html +0 -15
- package/docs/functions/ifFlag.html +0 -11
- package/docs/hierarchy.html +0 -1
- package/docs/index.html +0 -1869
- package/docs/interfaces/AuditLogEntry.html +0 -6
- package/docs/interfaces/CommandContext.html +0 -22
- package/docs/interfaces/CommandResult.html +0 -26
- package/docs/interfaces/ExecStream.html +0 -11
- package/docs/interfaces/HoneyPotStats.html +0 -16
- package/docs/interfaces/IdleManagerOptions.html +0 -7
- package/docs/interfaces/InstalledPackage.html +0 -20
- package/docs/interfaces/NanoEditorSession.html +0 -8
- package/docs/interfaces/PackageDefinition.html +0 -30
- package/docs/interfaces/PackageFile.html +0 -8
- package/docs/interfaces/PasswordChallenge.html +0 -16
- package/docs/interfaces/RemoveOptions.html +0 -4
- package/docs/interfaces/ShellEnv.html +0 -6
- package/docs/interfaces/ShellModule.html +0 -14
- package/docs/interfaces/ShellProperties.html +0 -14
- package/docs/interfaces/ShellStream.html +0 -11
- package/docs/interfaces/SudoChallenge.html +0 -24
- package/docs/interfaces/VfsBaseNode.html +0 -12
- package/docs/interfaces/VfsDiff.html +0 -10
- package/docs/interfaces/VfsDiffEntry.html +0 -6
- package/docs/interfaces/VfsDiffModified.html +0 -10
- package/docs/interfaces/VfsDirectoryNode.html +0 -15
- package/docs/interfaces/VfsFileNode.html +0 -17
- package/docs/interfaces/VfsOptions.html +0 -26
- package/docs/interfaces/VfsSnapshot.html +0 -3
- package/docs/interfaces/VfsSnapshotBaseNode.html +0 -8
- package/docs/interfaces/VfsSnapshotDirectoryNode.html +0 -10
- package/docs/interfaces/VfsSnapshotFileNode.html +0 -12
- package/docs/interfaces/VirtualActiveSession.html +0 -12
- package/docs/interfaces/VirtualSftpServerOptions.html +0 -7
- package/docs/interfaces/VirtualShellVfsLike.html +0 -15
- package/docs/interfaces/VirtualShellVfsOptions.html +0 -3
- package/docs/interfaces/WriteFileOptions.html +0 -6
- package/docs/media/LICENSE +0 -21
- package/docs/modules.html +0 -1
- package/docs/types/ArgParseOptions.html +0 -4
- package/docs/types/CommandMode.html +0 -2
- package/docs/types/CommandOutcome.html +0 -2
- package/docs/types/IdleState.html +0 -1
- package/docs/types/VfsNodeStats.html +0 -2
- package/docs/types/VfsNodeType.html +0 -2
- package/docs/types/VfsPersistenceMode.html +0 -5
- package/docs/types/VfsSnapshotNode.html +0 -2
- package/examples/README.md +0 -288
- package/examples/app.js +0 -1751
- package/examples/app.ts +0 -299
- package/examples/build.js +0 -27
- package/examples/demo.html +0 -33
- package/examples/honeypot-audit.ts +0 -180
- package/examples/honeypot-export.ts +0 -253
- package/examples/honeypot-quickstart.ts +0 -110
- package/examples/index.html +0 -82
- package/examples/server.js +0 -55
- package/polyfills/buffer.js +0 -117
- package/polyfills/node_child_process/index.js +0 -2
- package/polyfills/node_crypto/index.js +0 -167
- package/polyfills/node_events/index.js +0 -9
- package/polyfills/node_fs/index.js +0 -202
- package/polyfills/node_fs/promises.js +0 -4
- package/polyfills/node_os/index.js +0 -9
- package/polyfills/node_path/index.js +0 -28
- package/polyfills/node_vm/index.js +0 -7
- package/polyfills/node_zlib/index.js +0 -3
- package/polyfills/process.js +0 -14
- package/polyfills/ssh2/index.js +0 -75
- package/scripts/build-all.mjs +0 -226
- package/scripts/build-names.mjs +0 -43
- package/scripts/generate-manuals-bundle.mjs +0 -49
- package/scripts/postinstall.js +0 -42
- package/scripts/publish-package.sh +0 -70
- package/src/Honeypot/index.ts +0 -457
- package/src/SSHClient/index.ts +0 -270
- package/src/SSHMimic/exec.ts +0 -49
- package/src/SSHMimic/executor.ts +0 -251
- package/src/SSHMimic/hostKey.ts +0 -21
- package/src/SSHMimic/index.ts +0 -337
- package/src/SSHMimic/loginBanner.ts +0 -36
- package/src/SSHMimic/loginFormat.ts +0 -10
- package/src/SSHMimic/prompt.ts +0 -14
- package/src/SSHMimic/sftp.ts +0 -883
- package/src/VirtualFileSystem/binaryPack.ts +0 -258
- package/src/VirtualFileSystem/index.ts +0 -1193
- package/src/VirtualFileSystem/internalTypes.ts +0 -43
- package/src/VirtualFileSystem/journal.ts +0 -171
- package/src/VirtualFileSystem/path.ts +0 -74
- package/src/VirtualPackageManager/index.ts +0 -996
- package/src/VirtualShell/idleManager.ts +0 -137
- package/src/VirtualShell/index.ts +0 -475
- package/src/VirtualShell/shell.ts +0 -700
- package/src/VirtualShell/shellParser.ts +0 -285
- package/src/VirtualUserManager/index.ts +0 -758
- package/src/bun.d.ts +0 -1
- package/src/commands/adduser.ts +0 -103
- package/src/commands/alias.ts +0 -69
- package/src/commands/apt.ts +0 -233
- package/src/commands/awk.ts +0 -168
- package/src/commands/base64.ts +0 -29
- package/src/commands/cat.ts +0 -52
- package/src/commands/cd.ts +0 -25
- package/src/commands/chmod.ts +0 -85
- package/src/commands/clear.ts +0 -15
- package/src/commands/command-helpers.ts +0 -286
- package/src/commands/cp.ts +0 -83
- package/src/commands/curl.ts +0 -147
- package/src/commands/cut.ts +0 -36
- package/src/commands/date.ts +0 -30
- package/src/commands/declare.ts +0 -49
- package/src/commands/deluser.ts +0 -98
- package/src/commands/df.ts +0 -23
- package/src/commands/diff.ts +0 -43
- package/src/commands/dpkg.ts +0 -180
- package/src/commands/du.ts +0 -56
- package/src/commands/echo.ts +0 -58
- package/src/commands/env.ts +0 -23
- package/src/commands/exit.ts +0 -18
- package/src/commands/export.ts +0 -34
- package/src/commands/find.ts +0 -68
- package/src/commands/free.ts +0 -47
- package/src/commands/grep.ts +0 -116
- package/src/commands/groups.ts +0 -19
- package/src/commands/gzip.ts +0 -88
- package/src/commands/head.ts +0 -52
- package/src/commands/help.ts +0 -152
- package/src/commands/helpers.ts +0 -234
- package/src/commands/history.ts +0 -34
- package/src/commands/hostname.ts +0 -14
- package/src/commands/htop.ts +0 -20
- package/src/commands/id.ts +0 -19
- package/src/commands/index.ts +0 -9
- package/src/commands/kill.ts +0 -19
- package/src/commands/ln.ts +0 -71
- package/src/commands/ls.ts +0 -243
- package/src/commands/lsb-release.ts +0 -63
- package/src/commands/man.ts +0 -31
- package/src/commands/manuals/adduser.txt +0 -11
- package/src/commands/manuals/apt-cache.txt +0 -12
- package/src/commands/manuals/apt.txt +0 -20
- package/src/commands/manuals/awk.txt +0 -13
- package/src/commands/manuals/cat.txt +0 -14
- package/src/commands/manuals/cd.txt +0 -16
- package/src/commands/manuals/chmod.txt +0 -16
- package/src/commands/manuals/clear.txt +0 -10
- package/src/commands/manuals/cp.txt +0 -10
- package/src/commands/manuals/curl.txt +0 -20
- package/src/commands/manuals/date.txt +0 -14
- package/src/commands/manuals/declare.txt +0 -12
- package/src/commands/manuals/deluser.txt +0 -10
- package/src/commands/manuals/df.txt +0 -10
- package/src/commands/manuals/dpkg-query.txt +0 -11
- package/src/commands/manuals/dpkg.txt +0 -14
- package/src/commands/manuals/du.txt +0 -11
- package/src/commands/manuals/echo.txt +0 -11
- package/src/commands/manuals/false.txt +0 -10
- package/src/commands/manuals/find.txt +0 -11
- package/src/commands/manuals/free.txt +0 -12
- package/src/commands/manuals/grep.txt +0 -13
- package/src/commands/manuals/groups.txt +0 -10
- package/src/commands/manuals/gzip.txt +0 -11
- package/src/commands/manuals/head.txt +0 -10
- package/src/commands/manuals/help.txt +0 -11
- package/src/commands/manuals/history.txt +0 -11
- package/src/commands/manuals/hostname.txt +0 -10
- package/src/commands/manuals/id.txt +0 -10
- package/src/commands/manuals/kill.txt +0 -13
- package/src/commands/manuals/ls.txt +0 -20
- package/src/commands/manuals/lsb_release.txt +0 -14
- package/src/commands/manuals/mkdir.txt +0 -10
- package/src/commands/manuals/mv.txt +0 -10
- package/src/commands/manuals/nano.txt +0 -11
- package/src/commands/manuals/neofetch.txt +0 -10
- package/src/commands/manuals/node.txt +0 -13
- package/src/commands/manuals/npm.txt +0 -13
- package/src/commands/manuals/npx.txt +0 -13
- package/src/commands/manuals/passwd.txt +0 -11
- package/src/commands/manuals/ping.txt +0 -10
- package/src/commands/manuals/printf.txt +0 -11
- package/src/commands/manuals/ps.txt +0 -10
- package/src/commands/manuals/pwd.txt +0 -10
- package/src/commands/manuals/python3.txt +0 -13
- package/src/commands/manuals/readlink.txt +0 -10
- package/src/commands/manuals/return.txt +0 -10
- package/src/commands/manuals/rm.txt +0 -10
- package/src/commands/manuals/sed.txt +0 -11
- package/src/commands/manuals/set.txt +0 -11
- package/src/commands/manuals/shift.txt +0 -10
- package/src/commands/manuals/sleep.txt +0 -10
- package/src/commands/manuals/sort.txt +0 -12
- package/src/commands/manuals/source.txt +0 -11
- package/src/commands/manuals/ssh.txt +0 -11
- package/src/commands/manuals/stat.txt +0 -10
- package/src/commands/manuals/su.txt +0 -13
- package/src/commands/manuals/sudo.txt +0 -11
- package/src/commands/manuals/tail.txt +0 -10
- package/src/commands/manuals/tar.txt +0 -19
- package/src/commands/manuals/tee.txt +0 -10
- package/src/commands/manuals/test.txt +0 -11
- package/src/commands/manuals/touch.txt +0 -11
- package/src/commands/manuals/tr.txt +0 -10
- package/src/commands/manuals/trap.txt +0 -10
- package/src/commands/manuals/true.txt +0 -10
- package/src/commands/manuals/type.txt +0 -10
- package/src/commands/manuals/uname.txt +0 -12
- package/src/commands/manuals/uniq.txt +0 -12
- package/src/commands/manuals/unset.txt +0 -10
- package/src/commands/manuals/uptime.txt +0 -11
- package/src/commands/manuals/wc.txt +0 -12
- package/src/commands/manuals/wget.txt +0 -12
- package/src/commands/manuals/which.txt +0 -10
- package/src/commands/manuals/whoami.txt +0 -10
- package/src/commands/manuals/xargs.txt +0 -10
- package/src/commands/manuals-bundle.ts +0 -898
- package/src/commands/mkdir.ts +0 -31
- package/src/commands/mv.ts +0 -50
- package/src/commands/nano.ts +0 -38
- package/src/commands/neofetch.ts +0 -53
- package/src/commands/node.ts +0 -341
- package/src/commands/npm.ts +0 -132
- package/src/commands/passwd.ts +0 -50
- package/src/commands/ping.ts +0 -32
- package/src/commands/printf.ts +0 -129
- package/src/commands/ps.ts +0 -58
- package/src/commands/pwd.ts +0 -9
- package/src/commands/python.ts +0 -2229
- package/src/commands/read.ts +0 -46
- package/src/commands/registry.ts +0 -249
- package/src/commands/rm.ts +0 -42
- package/src/commands/runtime.ts +0 -421
- package/src/commands/sed.ts +0 -68
- package/src/commands/seq.ts +0 -43
- package/src/commands/set.ts +0 -29
- package/src/commands/sh.ts +0 -467
- package/src/commands/shift.ts +0 -63
- package/src/commands/sleep.ts +0 -20
- package/src/commands/sort.ts +0 -46
- package/src/commands/source.ts +0 -52
- package/src/commands/stat.ts +0 -61
- package/src/commands/su.ts +0 -72
- package/src/commands/sudo.ts +0 -76
- package/src/commands/tail.ts +0 -53
- package/src/commands/tar.ts +0 -102
- package/src/commands/tee.ts +0 -36
- package/src/commands/test.ts +0 -137
- package/src/commands/touch.ts +0 -28
- package/src/commands/tr.ts +0 -70
- package/src/commands/tree.ts +0 -20
- package/src/commands/true.ts +0 -27
- package/src/commands/type.ts +0 -48
- package/src/commands/uname.ts +0 -29
- package/src/commands/uniq.ts +0 -39
- package/src/commands/unset.ts +0 -17
- package/src/commands/uptime.ts +0 -54
- package/src/commands/wc.ts +0 -55
- package/src/commands/wget.ts +0 -148
- package/src/commands/which.ts +0 -37
- package/src/commands/who.ts +0 -25
- package/src/commands/whoami.ts +0 -14
- package/src/commands/xargs.ts +0 -31
- package/src/index.ts +0 -67
- package/src/modules/linuxRootfs.ts +0 -1961
- package/src/modules/neofetch.ts +0 -358
- package/src/modules/shellInteractive.ts +0 -57
- package/src/modules/shellRuntime.ts +0 -76
- package/src/self-standalone.ts +0 -542
- package/src/standalone-wo-sftp.ts +0 -38
- package/src/standalone.ts +0 -72
- package/src/types/commands.ts +0 -146
- package/src/types/pipeline.ts +0 -52
- package/src/types/streams.ts +0 -32
- package/src/types/tar-stream.d.ts +0 -38
- package/src/types/vfs.ts +0 -98
- package/src/utils/expand.ts +0 -491
- package/src/utils/perfLogger.ts +0 -72
- package/src/utils/tokenize.ts +0 -98
- package/src/utils/vfsDiff.ts +0 -275
- package/tests/command-helpers.test.ts +0 -116
- package/tests/commands-admin-net.test.ts +0 -441
- package/tests/commands-advanced.test.ts +0 -456
- package/tests/commands-core.test.ts +0 -562
- package/tests/commands-missing.test.ts +0 -570
- package/tests/commands-specific-units.test.ts +0 -327
- package/tests/commands-text-sys.test.ts +0 -445
- package/tests/expand.test.ts +0 -170
- package/tests/helpers.test.ts +0 -97
- package/tests/new-features.test.ts +0 -1036
- package/tests/parser-executor.test.ts +0 -37
- package/tests/sftp.test.ts +0 -323
- package/tests/ssh-exec.test.ts +0 -45
- package/tests/test-helper.ts +0 -79
- package/tests/users.test.ts +0 -86
- package/tsconfig.json +0 -49
- package/typedoc.json +0 -47
package/src/utils/expand.ts
DELETED
|
@@ -1,491 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* expand.ts
|
|
3
|
-
*
|
|
4
|
-
* Centralised shell variable and expression expansion.
|
|
5
|
-
* Used by `runCommand` (index.ts), `echo`, and `sh.ts`.
|
|
6
|
-
*
|
|
7
|
-
* Handles (in order):
|
|
8
|
-
* ~ tilde to $HOME
|
|
9
|
-
* $? last exit code
|
|
10
|
-
* $$ mock PID
|
|
11
|
-
* $# argument count (0 outside scripts)
|
|
12
|
-
* ${#VAR} string length
|
|
13
|
-
* ${VAR:-def} default if unset/empty
|
|
14
|
-
* ${VAR:=def} assign default if unset/empty
|
|
15
|
-
* ${VAR:+val} alternate value if set
|
|
16
|
-
* ${VAR} simple braced reference
|
|
17
|
-
* $VAR simple reference
|
|
18
|
-
* $((expr)) arithmetic (integer)
|
|
19
|
-
*/
|
|
20
|
-
|
|
21
|
-
// ─── arithmetic evaluator ────────────────────────────────────────────────────
|
|
22
|
-
|
|
23
|
-
type ArithToken =
|
|
24
|
-
| { type: "number"; value: number }
|
|
25
|
-
| { type: "plus" | "minus" | "mul" | "div" | "mod" | "pow" | "lparen" | "rparen" };
|
|
26
|
-
|
|
27
|
-
function tokenizeArith(expr: string, env: Record<string, string>): ArithToken[] {
|
|
28
|
-
const tokens: ArithToken[] = [];
|
|
29
|
-
let i = 0;
|
|
30
|
-
while (i < expr.length) {
|
|
31
|
-
const ch = expr[i]!;
|
|
32
|
-
if (/\s/.test(ch)) {
|
|
33
|
-
i++;
|
|
34
|
-
continue;
|
|
35
|
-
}
|
|
36
|
-
if (ch === "+") { tokens.push({ type: "plus" }); i++; continue; }
|
|
37
|
-
if (ch === "-") { tokens.push({ type: "minus" }); i++; continue; }
|
|
38
|
-
if (ch === "*") {
|
|
39
|
-
if (expr[i + 1] === "*") { tokens.push({ type: "pow" }); i += 2; continue; }
|
|
40
|
-
tokens.push({ type: "mul" });
|
|
41
|
-
i++;
|
|
42
|
-
continue;
|
|
43
|
-
}
|
|
44
|
-
if (ch === "/") { tokens.push({ type: "div" }); i++; continue; }
|
|
45
|
-
if (ch === "%") { tokens.push({ type: "mod" }); i++; continue; }
|
|
46
|
-
if (ch === "(") { tokens.push({ type: "lparen" }); i++; continue; }
|
|
47
|
-
if (ch === ")") { tokens.push({ type: "rparen" }); i++; continue; }
|
|
48
|
-
if (/\d/.test(ch)) {
|
|
49
|
-
let j = i + 1;
|
|
50
|
-
while (j < expr.length && /\d/.test(expr[j]!)) j++;
|
|
51
|
-
tokens.push({ type: "number", value: Number(expr.slice(i, j)) });
|
|
52
|
-
i = j;
|
|
53
|
-
continue;
|
|
54
|
-
}
|
|
55
|
-
if (/[A-Za-z_]/.test(ch)) {
|
|
56
|
-
let j = i + 1;
|
|
57
|
-
while (j < expr.length && /[A-Za-z0-9_]/.test(expr[j]!)) j++;
|
|
58
|
-
const name = expr.slice(i, j);
|
|
59
|
-
const raw = env[name];
|
|
60
|
-
const value = raw === undefined || raw === "" ? 0 : Number(raw);
|
|
61
|
-
tokens.push({ type: "number", value: Number.isFinite(value) ? value : 0 });
|
|
62
|
-
i = j;
|
|
63
|
-
continue;
|
|
64
|
-
}
|
|
65
|
-
return [];
|
|
66
|
-
}
|
|
67
|
-
return tokens;
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
/**
|
|
71
|
-
* Evaluate a simple integer arithmetic expression with a bounded parser.
|
|
72
|
-
* Supports: + - * / % ** unary- ( )
|
|
73
|
-
* Variables are resolved from `env`.
|
|
74
|
-
* Returns NaN on syntax error.
|
|
75
|
-
*/
|
|
76
|
-
export function evalArith(expr: string, env: Record<string, string>): number {
|
|
77
|
-
const trimmed = expr.trim();
|
|
78
|
-
if (trimmed.length === 0 || trimmed.length > 1024) return NaN;
|
|
79
|
-
const tokens = tokenizeArith(trimmed, env);
|
|
80
|
-
if (tokens.length === 0) return NaN;
|
|
81
|
-
|
|
82
|
-
let index = 0;
|
|
83
|
-
|
|
84
|
-
const peek = () => tokens[index];
|
|
85
|
-
const consume = () => tokens[index++];
|
|
86
|
-
|
|
87
|
-
const parsePrimary = (): number => {
|
|
88
|
-
const token = consume();
|
|
89
|
-
if (!token) return NaN;
|
|
90
|
-
if (token.type === "number") return token.value;
|
|
91
|
-
if (token.type === "lparen") {
|
|
92
|
-
const value = parseExpression();
|
|
93
|
-
if (tokens[index]?.type !== "rparen") return NaN;
|
|
94
|
-
index++;
|
|
95
|
-
return value;
|
|
96
|
-
}
|
|
97
|
-
return NaN;
|
|
98
|
-
};
|
|
99
|
-
|
|
100
|
-
const parseUnary = (): number => {
|
|
101
|
-
const token = peek();
|
|
102
|
-
if (token?.type === "plus") {
|
|
103
|
-
consume();
|
|
104
|
-
return parseUnary();
|
|
105
|
-
}
|
|
106
|
-
if (token?.type === "minus") {
|
|
107
|
-
consume();
|
|
108
|
-
return -parseUnary();
|
|
109
|
-
}
|
|
110
|
-
return parsePrimary();
|
|
111
|
-
};
|
|
112
|
-
|
|
113
|
-
const parsePower = (): number => {
|
|
114
|
-
let left = parseUnary();
|
|
115
|
-
while (peek()?.type === "pow") {
|
|
116
|
-
consume();
|
|
117
|
-
const right = parseUnary();
|
|
118
|
-
left = left ** right;
|
|
119
|
-
}
|
|
120
|
-
return left;
|
|
121
|
-
};
|
|
122
|
-
|
|
123
|
-
const parseTerm = (): number => {
|
|
124
|
-
let left = parsePower();
|
|
125
|
-
while (true) {
|
|
126
|
-
const token = peek();
|
|
127
|
-
if (token?.type === "mul") {
|
|
128
|
-
consume();
|
|
129
|
-
left *= parsePower();
|
|
130
|
-
continue;
|
|
131
|
-
}
|
|
132
|
-
if (token?.type === "div") {
|
|
133
|
-
consume();
|
|
134
|
-
const right = parsePower();
|
|
135
|
-
left = right === 0 ? NaN : left / right;
|
|
136
|
-
continue;
|
|
137
|
-
}
|
|
138
|
-
if (token?.type === "mod") {
|
|
139
|
-
consume();
|
|
140
|
-
const right = parsePower();
|
|
141
|
-
left = right === 0 ? NaN : left % right;
|
|
142
|
-
continue;
|
|
143
|
-
}
|
|
144
|
-
return left;
|
|
145
|
-
}
|
|
146
|
-
};
|
|
147
|
-
|
|
148
|
-
const parseExpression = (): number => {
|
|
149
|
-
let left = parseTerm();
|
|
150
|
-
while (true) {
|
|
151
|
-
const token = peek();
|
|
152
|
-
if (token?.type === "plus") {
|
|
153
|
-
consume();
|
|
154
|
-
left += parseTerm();
|
|
155
|
-
continue;
|
|
156
|
-
}
|
|
157
|
-
if (token?.type === "minus") {
|
|
158
|
-
consume();
|
|
159
|
-
left -= parseTerm();
|
|
160
|
-
continue;
|
|
161
|
-
}
|
|
162
|
-
return left;
|
|
163
|
-
}
|
|
164
|
-
};
|
|
165
|
-
|
|
166
|
-
const result = parseExpression();
|
|
167
|
-
if (!Number.isFinite(result) || index !== tokens.length) return NaN;
|
|
168
|
-
return Math.trunc(result);
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
// ─── synchronous expansion ───────────────────────────────────────────────────
|
|
172
|
-
|
|
173
|
-
/**
|
|
174
|
-
* Apply a replacer only to portions of `input` that are NOT inside single quotes.
|
|
175
|
-
* Single-quoted content is passed through verbatim (POSIX sh behaviour).
|
|
176
|
-
*/
|
|
177
|
-
function outsideSingleQuotes(
|
|
178
|
-
input: string,
|
|
179
|
-
replacer: (chunk: string) => string,
|
|
180
|
-
): string {
|
|
181
|
-
const parts: string[] = [];
|
|
182
|
-
let i = 0;
|
|
183
|
-
while (i < input.length) {
|
|
184
|
-
const sqIdx = input.indexOf("'", i);
|
|
185
|
-
if (sqIdx === -1) {
|
|
186
|
-
// No more single quotes — expand the rest
|
|
187
|
-
parts.push(replacer(input.slice(i)));
|
|
188
|
-
break;
|
|
189
|
-
}
|
|
190
|
-
// Expand the part before the single quote
|
|
191
|
-
parts.push(replacer(input.slice(i, sqIdx)));
|
|
192
|
-
// Find closing single quote — everything inside is literal
|
|
193
|
-
const closeIdx = input.indexOf("'", sqIdx + 1);
|
|
194
|
-
if (closeIdx === -1) {
|
|
195
|
-
// Unclosed quote — treat rest as literal
|
|
196
|
-
parts.push(input.slice(sqIdx));
|
|
197
|
-
break;
|
|
198
|
-
}
|
|
199
|
-
parts.push(input.slice(sqIdx, closeIdx + 1)); // include quotes
|
|
200
|
-
i = closeIdx + 1;
|
|
201
|
-
}
|
|
202
|
-
return parts.join("");
|
|
203
|
-
}
|
|
204
|
-
|
|
205
|
-
/**
|
|
206
|
-
* Expand all shell variable and expression forms synchronously.
|
|
207
|
-
* Does NOT handle `$(cmd)` — that requires async; see `expandAsync`.
|
|
208
|
-
* Content inside single quotes is left verbatim per POSIX sh rules.
|
|
209
|
-
*
|
|
210
|
-
* @param input Raw string possibly containing `$VAR`, `${...}`, `$((...))`.
|
|
211
|
-
* @param env Current session env vars.
|
|
212
|
-
* @param lastExit Last command exit code (for `$?`).
|
|
213
|
-
* @param home Home directory path (for `~`).
|
|
214
|
-
*/
|
|
215
|
-
|
|
216
|
-
/**
|
|
217
|
-
* Expand brace expressions in a single token.
|
|
218
|
-
* - `{a,b,c}` → `["a", "b", "c"]`
|
|
219
|
-
* - `{1..5}` → `["1", "2", "3", "4", "5"]`
|
|
220
|
-
* - `{a..e}` → `["a", "b", "c", "d", "e"]`
|
|
221
|
-
* - `prefix{a,b}suffix` → `["prefixasuffix", "prefixbsuffix"]`
|
|
222
|
-
* Returns a single-element array when no brace expansion applies.
|
|
223
|
-
*/
|
|
224
|
-
export function expandBraces(token: string): string[] {
|
|
225
|
-
const MaxBraceDepth = 8;
|
|
226
|
-
const MaxBraceExpansions = 256;
|
|
227
|
-
|
|
228
|
-
function expandBracesInternal(value: string, depth: number): string[] {
|
|
229
|
-
if (depth > MaxBraceDepth) return [value];
|
|
230
|
-
// Find the first { not preceded by $
|
|
231
|
-
let braceDepth = 0;
|
|
232
|
-
let start = -1;
|
|
233
|
-
for (let i = 0; i < value.length; i++) {
|
|
234
|
-
const ch = value[i]!;
|
|
235
|
-
if (ch === "{" && value[i - 1] !== "$") {
|
|
236
|
-
if (braceDepth === 0) start = i;
|
|
237
|
-
braceDepth++;
|
|
238
|
-
} else if (ch === "}") {
|
|
239
|
-
braceDepth--;
|
|
240
|
-
if (braceDepth === 0 && start !== -1) {
|
|
241
|
-
const prefix = value.slice(0, start);
|
|
242
|
-
const inner = value.slice(start + 1, i);
|
|
243
|
-
const suffix = value.slice(i + 1);
|
|
244
|
-
|
|
245
|
-
// Range: {1..5} or {a..e}
|
|
246
|
-
const rangeMatch = inner.match(/^(-?\d+)\.\.(-?\d+)(?:\.\.-?(\d+))?$/) ||
|
|
247
|
-
inner.match(/^([a-z])\.\.([a-z])$/);
|
|
248
|
-
if (rangeMatch) {
|
|
249
|
-
const items: string[] = [];
|
|
250
|
-
if (/\d/.test(rangeMatch[1]!)) {
|
|
251
|
-
const from = parseInt(rangeMatch[1]!, 10);
|
|
252
|
-
const to = parseInt(rangeMatch[2]!, 10);
|
|
253
|
-
const step = rangeMatch[3] ? parseInt(rangeMatch[3], 10) : 1;
|
|
254
|
-
const inc = from <= to ? step : -step;
|
|
255
|
-
for (let n = from; from <= to ? n <= to : n >= to; n += inc) {
|
|
256
|
-
items.push(String(n));
|
|
257
|
-
}
|
|
258
|
-
} else {
|
|
259
|
-
const from = rangeMatch[1]!.charCodeAt(0);
|
|
260
|
-
const to = rangeMatch[2]!.charCodeAt(0);
|
|
261
|
-
const inc = from <= to ? 1 : -1;
|
|
262
|
-
for (let c = from; from <= to ? c <= to : c >= to; c += inc) {
|
|
263
|
-
items.push(String.fromCharCode(c));
|
|
264
|
-
}
|
|
265
|
-
}
|
|
266
|
-
|
|
267
|
-
const expanded = items.map((v) => `${prefix}${v}${suffix}`);
|
|
268
|
-
const output: string[] = [];
|
|
269
|
-
for (const item of expanded) {
|
|
270
|
-
output.push(...expandBracesInternal(item, depth + 1));
|
|
271
|
-
if (output.length > MaxBraceExpansions) return [value];
|
|
272
|
-
}
|
|
273
|
-
return output;
|
|
274
|
-
}
|
|
275
|
-
|
|
276
|
-
// Comma list: {a,b,c} — split respecting nested braces
|
|
277
|
-
const parts: string[] = [];
|
|
278
|
-
let cur = "";
|
|
279
|
-
let innerDepth = 0;
|
|
280
|
-
for (const ch2 of inner) {
|
|
281
|
-
if (ch2 === "{") { innerDepth++; cur += ch2; }
|
|
282
|
-
else if (ch2 === "}") { innerDepth--; cur += ch2; }
|
|
283
|
-
else if (ch2 === "," && innerDepth === 0) { parts.push(cur); cur = ""; }
|
|
284
|
-
else { cur += ch2; }
|
|
285
|
-
}
|
|
286
|
-
parts.push(cur);
|
|
287
|
-
|
|
288
|
-
if (parts.length > 1) {
|
|
289
|
-
const output: string[] = [];
|
|
290
|
-
for (const part of parts) {
|
|
291
|
-
output.push(...expandBracesInternal(`${prefix}${part}${suffix}`, depth + 1));
|
|
292
|
-
if (output.length > MaxBraceExpansions) return [value];
|
|
293
|
-
}
|
|
294
|
-
return output;
|
|
295
|
-
}
|
|
296
|
-
break;
|
|
297
|
-
}
|
|
298
|
-
}
|
|
299
|
-
}
|
|
300
|
-
return [value];
|
|
301
|
-
}
|
|
302
|
-
|
|
303
|
-
return expandBracesInternal(token, 0);
|
|
304
|
-
}
|
|
305
|
-
|
|
306
|
-
function expandArithmeticChunks(input: string, env: Record<string, string>): string {
|
|
307
|
-
let result = "";
|
|
308
|
-
let index = 0;
|
|
309
|
-
while (index < input.length) {
|
|
310
|
-
if (input[index] === "$" && input[index + 1] === "(" && input[index + 2] === "(") {
|
|
311
|
-
let scan = index + 3;
|
|
312
|
-
let depth = 0;
|
|
313
|
-
while (scan < input.length) {
|
|
314
|
-
const ch = input[scan]!;
|
|
315
|
-
if (ch === "(") {
|
|
316
|
-
depth++;
|
|
317
|
-
} else if (ch === ")") {
|
|
318
|
-
if (depth > 0) {
|
|
319
|
-
depth--;
|
|
320
|
-
} else if (input[scan + 1] === ")") {
|
|
321
|
-
const expr = input.slice(index + 3, scan);
|
|
322
|
-
const value = evalArith(expr, env);
|
|
323
|
-
result += Number.isNaN(value) ? "0" : String(value);
|
|
324
|
-
index = scan + 2;
|
|
325
|
-
break;
|
|
326
|
-
}
|
|
327
|
-
}
|
|
328
|
-
scan++;
|
|
329
|
-
}
|
|
330
|
-
if (scan >= input.length) {
|
|
331
|
-
result += input.slice(index);
|
|
332
|
-
break;
|
|
333
|
-
}
|
|
334
|
-
continue;
|
|
335
|
-
}
|
|
336
|
-
result += input[index]!
|
|
337
|
-
index++;
|
|
338
|
-
}
|
|
339
|
-
return result;
|
|
340
|
-
}
|
|
341
|
-
|
|
342
|
-
export function expandSync(
|
|
343
|
-
input: string,
|
|
344
|
-
env: Record<string, string>,
|
|
345
|
-
lastExit = 0,
|
|
346
|
-
home?: string,
|
|
347
|
-
): string {
|
|
348
|
-
const homePath = home ?? env.HOME ?? "/home/user";
|
|
349
|
-
|
|
350
|
-
return outsideSingleQuotes(input, (chunk) => {
|
|
351
|
-
let s = chunk;
|
|
352
|
-
|
|
353
|
-
// Tilde expansion — only at start of token or after `:` or whitespace
|
|
354
|
-
s = s.replace(
|
|
355
|
-
/(^|[\s:])~(\/|$)/g,
|
|
356
|
-
(_, pre, post) => `${pre}${homePath}${post}`,
|
|
357
|
-
);
|
|
358
|
-
|
|
359
|
-
// $? $$ $#
|
|
360
|
-
s = s.replace(/\$\?/g, String(lastExit));
|
|
361
|
-
s = s.replace(/\$\$/g, "1");
|
|
362
|
-
s = s.replace(/\$#/g, "0");
|
|
363
|
-
|
|
364
|
-
// $(( arithmetic )) — must come before ${ and $VAR to avoid conflicts
|
|
365
|
-
s = expandArithmeticChunks(s, env);
|
|
366
|
-
|
|
367
|
-
// ${#VAR} — string length
|
|
368
|
-
s = s.replace(/\$\{#([A-Za-z_][A-Za-z0-9_]*)\}/g, (_, name) =>
|
|
369
|
-
String((env[name] ?? "").length),
|
|
370
|
-
);
|
|
371
|
-
|
|
372
|
-
// ${VAR:-default}
|
|
373
|
-
s = s.replace(/\$\{([A-Za-z_][A-Za-z0-9_]*):-([^}]*)\}/g, (_, name, def) =>
|
|
374
|
-
env[name] !== undefined && env[name] !== "" ? (env[name] as string) : def,
|
|
375
|
-
);
|
|
376
|
-
|
|
377
|
-
// ${VAR:=default} — also assigns to env
|
|
378
|
-
s = s.replace(
|
|
379
|
-
/\$\{([A-Za-z_][A-Za-z0-9_]*):=([^}]*)\}/g,
|
|
380
|
-
(_, name, def) => {
|
|
381
|
-
if (env[name] === undefined || env[name] === "") env[name] = def;
|
|
382
|
-
return env[name] as string;
|
|
383
|
-
},
|
|
384
|
-
);
|
|
385
|
-
|
|
386
|
-
// ${VAR:+alternate}
|
|
387
|
-
s = s.replace(
|
|
388
|
-
/\$\{([A-Za-z_][A-Za-z0-9_]*):\+([^}]*)\}/g,
|
|
389
|
-
(_, name, alt) =>
|
|
390
|
-
env[name] !== undefined && env[name] !== "" ? alt : "",
|
|
391
|
-
);
|
|
392
|
-
|
|
393
|
-
// ${VAR}
|
|
394
|
-
s = s.replace(
|
|
395
|
-
/\$\{([A-Za-z_][A-Za-z0-9_]*)\}/g,
|
|
396
|
-
(_, name) => env[name] ?? "",
|
|
397
|
-
);
|
|
398
|
-
|
|
399
|
-
// $VAR and positional params $1 $2 ...
|
|
400
|
-
s = s.replace(/\$([A-Za-z_][A-Za-z0-9_]*|\d+)/g, (_, name) => env[name] ?? "");
|
|
401
|
-
|
|
402
|
-
return s;
|
|
403
|
-
});
|
|
404
|
-
}
|
|
405
|
-
|
|
406
|
-
// ─── async expansion (includes $(cmd)) ──────────────────────────────────────
|
|
407
|
-
|
|
408
|
-
/**
|
|
409
|
-
* Expand all shell forms including `$(cmd)` command substitution.
|
|
410
|
-
*
|
|
411
|
-
* Processes `$(...)` blocks depth-first, respecting single-quote boundaries.
|
|
412
|
-
* Then delegates to `expandSync` for the remaining forms.
|
|
413
|
-
*
|
|
414
|
-
* @param input Raw string.
|
|
415
|
-
* @param env Current session env vars.
|
|
416
|
-
* @param lastExit Last exit code.
|
|
417
|
-
* @param runCmd Async callback to execute a command and return its stdout.
|
|
418
|
-
*/
|
|
419
|
-
export async function expandAsync(
|
|
420
|
-
input: string,
|
|
421
|
-
env: Record<string, string>,
|
|
422
|
-
lastExit: number,
|
|
423
|
-
runCmd: (cmd: string) => Promise<string>,
|
|
424
|
-
): Promise<string> {
|
|
425
|
-
const depthKey = "__shellExpandDepth";
|
|
426
|
-
const maxDepth = 8;
|
|
427
|
-
const currentDepth = Number(env[depthKey] ?? "0");
|
|
428
|
-
if (currentDepth >= maxDepth) {
|
|
429
|
-
return expandSync(input, env, lastExit);
|
|
430
|
-
}
|
|
431
|
-
env[depthKey] = String(currentDepth + 1);
|
|
432
|
-
try {
|
|
433
|
-
// $(cmd) substitution — skip content inside single quotes
|
|
434
|
-
if (input.includes("$(")) {
|
|
435
|
-
let result = "";
|
|
436
|
-
let inSingle = false;
|
|
437
|
-
let i = 0;
|
|
438
|
-
|
|
439
|
-
while (i < input.length) {
|
|
440
|
-
const ch = input[i]!;
|
|
441
|
-
|
|
442
|
-
if (ch === "'" && !inSingle) {
|
|
443
|
-
inSingle = true;
|
|
444
|
-
result += ch;
|
|
445
|
-
i++;
|
|
446
|
-
continue;
|
|
447
|
-
}
|
|
448
|
-
if (ch === "'" && inSingle) {
|
|
449
|
-
inSingle = false;
|
|
450
|
-
result += ch;
|
|
451
|
-
i++;
|
|
452
|
-
continue;
|
|
453
|
-
}
|
|
454
|
-
|
|
455
|
-
if (!inSingle && ch === "$" && input[i + 1] === "(") {
|
|
456
|
-
// $((expr)) arithmetic — NOT a $(cmd) substitution, skip it
|
|
457
|
-
if (input[i + 2] === "(") {
|
|
458
|
-
result += ch;
|
|
459
|
-
i++;
|
|
460
|
-
continue;
|
|
461
|
-
}
|
|
462
|
-
// Find matching ) with depth tracking
|
|
463
|
-
let depth = 0;
|
|
464
|
-
let j = i + 1;
|
|
465
|
-
while (j < input.length) {
|
|
466
|
-
if (input[j] === "(") depth++;
|
|
467
|
-
else if (input[j] === ")") {
|
|
468
|
-
depth--;
|
|
469
|
-
if (depth === 0) break;
|
|
470
|
-
}
|
|
471
|
-
j++;
|
|
472
|
-
}
|
|
473
|
-
const sub = input.slice(i + 2, j).trim();
|
|
474
|
-
const out = (await runCmd(sub)).replace(/\n$/, "");
|
|
475
|
-
result += out;
|
|
476
|
-
i = j + 1;
|
|
477
|
-
continue;
|
|
478
|
-
}
|
|
479
|
-
|
|
480
|
-
result += ch;
|
|
481
|
-
i++;
|
|
482
|
-
}
|
|
483
|
-
input = result;
|
|
484
|
-
}
|
|
485
|
-
|
|
486
|
-
return expandSync(input, env, lastExit);
|
|
487
|
-
} finally {
|
|
488
|
-
if (currentDepth <= 0) delete env[depthKey];
|
|
489
|
-
else env[depthKey] = String(currentDepth);
|
|
490
|
-
}
|
|
491
|
-
}
|
package/src/utils/perfLogger.ts
DELETED
|
@@ -1,72 +0,0 @@
|
|
|
1
|
-
export type PerfLogger = {
|
|
2
|
-
enabled: boolean;
|
|
3
|
-
mark: (label: string) => void;
|
|
4
|
-
done: (label?: string) => void;
|
|
5
|
-
};
|
|
6
|
-
|
|
7
|
-
function isTruthyEnv(value: string | undefined): boolean {
|
|
8
|
-
return value === "1" || value === "true";
|
|
9
|
-
}
|
|
10
|
-
|
|
11
|
-
function nowMs(): number {
|
|
12
|
-
if (
|
|
13
|
-
typeof performance !== "undefined" &&
|
|
14
|
-
typeof performance.now === "function"
|
|
15
|
-
) {
|
|
16
|
-
return performance.now();
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
return Date.now();
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
export function isPerfLoggingEnabled(): boolean {
|
|
23
|
-
return (
|
|
24
|
-
isTruthyEnv(process.env.DEV_MODE) || isTruthyEnv(process.env.RENDER_PERF)
|
|
25
|
-
);
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
export function createPerfLogger(scope: string): PerfLogger {
|
|
29
|
-
const enabled = isPerfLoggingEnabled();
|
|
30
|
-
if (!enabled) {
|
|
31
|
-
return {
|
|
32
|
-
enabled,
|
|
33
|
-
mark: () => undefined,
|
|
34
|
-
done: () => undefined,
|
|
35
|
-
};
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
const startedAt = nowMs();
|
|
39
|
-
|
|
40
|
-
const mark = (label: string): void => {
|
|
41
|
-
const elapsedMs = nowMs() - startedAt;
|
|
42
|
-
console.log(`[perf][${scope}] ${label}: ${elapsedMs.toFixed(1)}ms`);
|
|
43
|
-
};
|
|
44
|
-
|
|
45
|
-
const done = (label = "done"): void => {
|
|
46
|
-
mark(label);
|
|
47
|
-
};
|
|
48
|
-
|
|
49
|
-
return {
|
|
50
|
-
enabled,
|
|
51
|
-
mark,
|
|
52
|
-
done,
|
|
53
|
-
};
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
export async function withPerf<T>(
|
|
57
|
-
scope: string,
|
|
58
|
-
label: string,
|
|
59
|
-
work: () => Promise<T>,
|
|
60
|
-
): Promise<T> {
|
|
61
|
-
const perf = createPerfLogger(scope);
|
|
62
|
-
if (!perf.enabled) {
|
|
63
|
-
return work();
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
perf.mark(`${label}:start`);
|
|
67
|
-
try {
|
|
68
|
-
return await work();
|
|
69
|
-
} finally {
|
|
70
|
-
perf.done(`${label}:done`);
|
|
71
|
-
}
|
|
72
|
-
}
|
package/src/utils/tokenize.ts
DELETED
|
@@ -1,98 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* tokenize.ts
|
|
3
|
-
*
|
|
4
|
-
* Shared shell tokenizer used by `shellParser.ts` and `runtime.ts`.
|
|
5
|
-
* Splits a shell input string into tokens respecting single and double
|
|
6
|
-
* quotes, and separates `>`, `>>`, `<` as standalone redirect tokens.
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
/**
|
|
10
|
-
* Tokenize a shell command line respecting quoted strings and redirect
|
|
11
|
-
* operators.
|
|
12
|
-
*
|
|
13
|
-
* - Single-quoted content is preserved verbatim.
|
|
14
|
-
* - Double-quoted content is preserved (expansion happens later).
|
|
15
|
-
* - `>`, `>>`, and `<` are emitted as standalone tokens.
|
|
16
|
-
*
|
|
17
|
-
* @param input Raw shell command string.
|
|
18
|
-
* @returns Array of string tokens.
|
|
19
|
-
*/
|
|
20
|
-
export function tokenizeCommand(input: string): string[] {
|
|
21
|
-
const tokens: string[] = [];
|
|
22
|
-
let current = "";
|
|
23
|
-
let inQ = false;
|
|
24
|
-
let qChar = "";
|
|
25
|
-
let i = 0;
|
|
26
|
-
|
|
27
|
-
while (i < input.length) {
|
|
28
|
-
const ch = input[i]!;
|
|
29
|
-
const next = input[i + 1];
|
|
30
|
-
|
|
31
|
-
if ((ch === '"' || ch === "'") && !inQ) {
|
|
32
|
-
inQ = true;
|
|
33
|
-
qChar = ch;
|
|
34
|
-
i++;
|
|
35
|
-
continue;
|
|
36
|
-
}
|
|
37
|
-
if (inQ && ch === qChar) {
|
|
38
|
-
inQ = false;
|
|
39
|
-
qChar = "";
|
|
40
|
-
i++;
|
|
41
|
-
continue;
|
|
42
|
-
}
|
|
43
|
-
if (inQ) {
|
|
44
|
-
current += ch;
|
|
45
|
-
i++;
|
|
46
|
-
continue;
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
if (ch === " ") {
|
|
50
|
-
if (current) {
|
|
51
|
-
tokens.push(current);
|
|
52
|
-
current = "";
|
|
53
|
-
}
|
|
54
|
-
i++;
|
|
55
|
-
continue;
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
// Handle 2>&1, 2>>, 2>, >&, >>
|
|
59
|
-
if (!inQ && ch === "2" && (next === ">" )) {
|
|
60
|
-
const rest = input.slice(i + 1);
|
|
61
|
-
if (rest.startsWith(">>&1") || rest.startsWith(">> &1")) {
|
|
62
|
-
if (current) { tokens.push(current); current = ""; }
|
|
63
|
-
tokens.push("2>>&1"); i += 5; continue;
|
|
64
|
-
}
|
|
65
|
-
if (rest.startsWith(">&1")) {
|
|
66
|
-
if (current) { tokens.push(current); current = ""; }
|
|
67
|
-
tokens.push("2>&1"); i += 4; continue;
|
|
68
|
-
}
|
|
69
|
-
if (rest.startsWith(">>")) {
|
|
70
|
-
if (current) { tokens.push(current); current = ""; }
|
|
71
|
-
tokens.push("2>>"); i += 3; continue;
|
|
72
|
-
}
|
|
73
|
-
if (rest.startsWith(">")) {
|
|
74
|
-
if (current) { tokens.push(current); current = ""; }
|
|
75
|
-
tokens.push("2>"); i += 2; continue;
|
|
76
|
-
}
|
|
77
|
-
}
|
|
78
|
-
if ((ch === ">" || ch === "<") && !inQ) {
|
|
79
|
-
if (current) {
|
|
80
|
-
tokens.push(current);
|
|
81
|
-
current = "";
|
|
82
|
-
}
|
|
83
|
-
if (ch === ">" && next === ">") {
|
|
84
|
-
tokens.push(">>");
|
|
85
|
-
i += 2;
|
|
86
|
-
} else {
|
|
87
|
-
tokens.push(ch);
|
|
88
|
-
i++;
|
|
89
|
-
}
|
|
90
|
-
continue;
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
current += ch;
|
|
94
|
-
i++;
|
|
95
|
-
}
|
|
96
|
-
if (current) tokens.push(current);
|
|
97
|
-
return tokens;
|
|
98
|
-
}
|