typescript-virtual-container 1.5.2 → 1.5.4
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 +43 -23
- package/dist/.tsbuildinfo +1 -0
- package/dist/SSHMimic/executor.js +23 -5
- package/dist/commands/basename.d.ts +13 -0
- package/dist/commands/basename.js +45 -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/ifconfig.d.ts +7 -0
- package/dist/commands/ifconfig.js +52 -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 +24 -2
- package/dist/commands/runtime.js +159 -106
- package/dist/commands/sh.js +5 -0
- 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 +84 -0
- 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.1-directbash-k6.1.0.mjs +0 -1768
- package/builds/fortune-nyx-v1.5.1-ssh-nosftp.js +0 -1768
- package/builds/fortune-nyx-v1.5.1-ssh.cjs +0 -1769
- package/builds/fortune-nyx-v1.5.1-web.min.js +0 -17022
- package/bun.lock +0 -244
- package/docs/.nojekyll +0 -1
- package/docs/app.js +0 -1755
- 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 -1755
- 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 -1006
- 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 -378
- 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/commands/chmod.ts
DELETED
|
@@ -1,85 +0,0 @@
|
|
|
1
|
-
import type { ShellModule } from "../types/commands";
|
|
2
|
-
import { assertPathAccess, resolvePath } from "./helpers";
|
|
3
|
-
|
|
4
|
-
/**
|
|
5
|
-
* Parse a symbolic chmod mode string (e.g. "+x", "u+x", "go-w", "a+rx")
|
|
6
|
-
* and apply it to the existing mode bits.
|
|
7
|
-
* Returns null if the string is not a valid symbolic mode.
|
|
8
|
-
*/
|
|
9
|
-
function applySymbolicMode(existing: number, modeStr: string): number | null {
|
|
10
|
-
const pattern = /^([ugoa]*)([+\-=])([rwx]*)$/;
|
|
11
|
-
const parts = modeStr.split(",");
|
|
12
|
-
let mode = existing;
|
|
13
|
-
for (const part of parts) {
|
|
14
|
-
const m = part.trim().match(pattern);
|
|
15
|
-
if (!m) return null;
|
|
16
|
-
const [, who = "a", op, perms = ""] = m;
|
|
17
|
-
const targets = who === "" || who === "a" ? ["u", "g", "o"] : who.split("");
|
|
18
|
-
const bits: Record<string, Record<string, number>> = {
|
|
19
|
-
u: { r: 0o400, w: 0o200, x: 0o100 },
|
|
20
|
-
g: { r: 0o040, w: 0o020, x: 0o010 },
|
|
21
|
-
o: { r: 0o004, w: 0o002, x: 0o001 },
|
|
22
|
-
};
|
|
23
|
-
for (const t of targets) {
|
|
24
|
-
for (const p of perms.split("")) {
|
|
25
|
-
const bit = bits[t]?.[p];
|
|
26
|
-
if (bit === undefined) continue;
|
|
27
|
-
if (op === "+") mode |= bit;
|
|
28
|
-
else if (op === "-") mode &= ~bit;
|
|
29
|
-
else if (op === "=") {
|
|
30
|
-
// clear all bits for this target, then set requested
|
|
31
|
-
const mask = Object.values(bits[t] ?? {}).reduce((a, b) => a | b, 0);
|
|
32
|
-
mode = (mode & ~mask) | bit;
|
|
33
|
-
}
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
|
-
}
|
|
37
|
-
return mode;
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
/**
|
|
41
|
-
* Change file permissions (octal or symbolic).
|
|
42
|
-
* @category files
|
|
43
|
-
* @params ["<mode> <file>"]
|
|
44
|
-
*/
|
|
45
|
-
export const chmodCommand: ShellModule = {
|
|
46
|
-
name: "chmod",
|
|
47
|
-
description: "Change file permissions",
|
|
48
|
-
category: "files",
|
|
49
|
-
params: ["<mode> <file>"],
|
|
50
|
-
run: ({ authUser, shell, cwd, args }) => {
|
|
51
|
-
const [modeArg, fileArg] = args;
|
|
52
|
-
if (!modeArg || !fileArg) {
|
|
53
|
-
return { stderr: "chmod: missing operand", exitCode: 1 };
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
const filePath = resolvePath(cwd, fileArg);
|
|
57
|
-
try {
|
|
58
|
-
assertPathAccess(authUser, filePath, "chmod");
|
|
59
|
-
if (!shell.vfs.exists(filePath)) {
|
|
60
|
-
return {
|
|
61
|
-
stderr: `chmod: ${fileArg}: No such file or directory`,
|
|
62
|
-
exitCode: 1,
|
|
63
|
-
};
|
|
64
|
-
}
|
|
65
|
-
let mode: number;
|
|
66
|
-
const octal = parseInt(modeArg, 8);
|
|
67
|
-
if (!Number.isNaN(octal) && /^[0-7]+$/.test(modeArg)) {
|
|
68
|
-
mode = octal;
|
|
69
|
-
} else {
|
|
70
|
-
// symbolic mode
|
|
71
|
-
const existing = shell.vfs.stat(filePath).mode;
|
|
72
|
-
const result = applySymbolicMode(existing, modeArg);
|
|
73
|
-
if (result === null) {
|
|
74
|
-
return { stderr: `chmod: invalid mode: ${modeArg}`, exitCode: 1 };
|
|
75
|
-
}
|
|
76
|
-
mode = result;
|
|
77
|
-
}
|
|
78
|
-
shell.vfs.chmod(filePath, mode);
|
|
79
|
-
return { exitCode: 0 };
|
|
80
|
-
} catch (err) {
|
|
81
|
-
const msg = err instanceof Error ? err.message : String(err);
|
|
82
|
-
return { stderr: `chmod: ${msg}`, exitCode: 1 };
|
|
83
|
-
}
|
|
84
|
-
},
|
|
85
|
-
};
|
package/src/commands/clear.ts
DELETED
|
@@ -1,15 +0,0 @@
|
|
|
1
|
-
import type { ShellModule } from "../types/commands";
|
|
2
|
-
|
|
3
|
-
/**
|
|
4
|
-
* Clear the terminal screen.
|
|
5
|
-
* @category shell
|
|
6
|
-
* @params []
|
|
7
|
-
*/
|
|
8
|
-
export const clearCommand: ShellModule = {
|
|
9
|
-
name: "clear",
|
|
10
|
-
description: "Clear the terminal screen",
|
|
11
|
-
category: "shell",
|
|
12
|
-
params: [],
|
|
13
|
-
// clearScreen flag triggers \x1b[2J\x1b[H in the shell layer
|
|
14
|
-
run: () => ({ clearScreen: true, stdout: "", exitCode: 0 }),
|
|
15
|
-
};
|
|
@@ -1,286 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Options for argument parsing helpers.
|
|
3
|
-
* @public
|
|
4
|
-
*/
|
|
5
|
-
export type ArgParseOptions = {
|
|
6
|
-
flags?: string[];
|
|
7
|
-
flagsWithValue?: string[];
|
|
8
|
-
};
|
|
9
|
-
|
|
10
|
-
function toFlagList(flags: string | string[]): string[] {
|
|
11
|
-
return Array.isArray(flags) ? flags : [flags];
|
|
12
|
-
}
|
|
13
|
-
|
|
14
|
-
function matchFlagToken(
|
|
15
|
-
token: string,
|
|
16
|
-
flag: string,
|
|
17
|
-
): { matched: boolean; inlineValue: string | null } {
|
|
18
|
-
if (token === flag) {
|
|
19
|
-
return { matched: true, inlineValue: null };
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
// --flag=value style
|
|
23
|
-
const prefix = `${flag}=`;
|
|
24
|
-
if (token.startsWith(prefix)) {
|
|
25
|
-
return { matched: true, inlineValue: token.slice(prefix.length) };
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
// Short flag inline value: -f2, -d: (single char flag like -f, -d, -n)
|
|
29
|
-
// Only applies to single-char flags (-X), not long flags (--flag)
|
|
30
|
-
if (flag.length === 2 && flag.startsWith("-") && !flag.startsWith("--")) {
|
|
31
|
-
if (token.startsWith(flag) && token.length > flag.length) {
|
|
32
|
-
return { matched: true, inlineValue: token.slice(flag.length) };
|
|
33
|
-
}
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
return { matched: false, inlineValue: null };
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
function collectPositionals(
|
|
40
|
-
args: string[],
|
|
41
|
-
options: ArgParseOptions = {},
|
|
42
|
-
): string[] {
|
|
43
|
-
const boolFlags = new Set(options.flags ?? []);
|
|
44
|
-
const valueFlags = new Set(options.flagsWithValue ?? []);
|
|
45
|
-
const positionals: string[] = [];
|
|
46
|
-
let passthrough = false;
|
|
47
|
-
|
|
48
|
-
for (let index = 0; index < args.length; index += 1) {
|
|
49
|
-
const arg = args[index]!;
|
|
50
|
-
|
|
51
|
-
if (passthrough) {
|
|
52
|
-
positionals.push(arg);
|
|
53
|
-
continue;
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
if (arg === "--") {
|
|
57
|
-
passthrough = true;
|
|
58
|
-
continue;
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
let consumed = false;
|
|
62
|
-
|
|
63
|
-
for (const flag of boolFlags) {
|
|
64
|
-
const { matched } = matchFlagToken(arg, flag);
|
|
65
|
-
if (matched) {
|
|
66
|
-
consumed = true;
|
|
67
|
-
break;
|
|
68
|
-
}
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
if (consumed) {
|
|
72
|
-
continue;
|
|
73
|
-
}
|
|
74
|
-
|
|
75
|
-
for (const flag of valueFlags) {
|
|
76
|
-
const match = matchFlagToken(arg, flag);
|
|
77
|
-
if (!match.matched) {
|
|
78
|
-
continue;
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
consumed = true;
|
|
82
|
-
if (match.inlineValue === null && index + 1 < args.length) {
|
|
83
|
-
index += 1;
|
|
84
|
-
}
|
|
85
|
-
break;
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
if (!consumed) {
|
|
89
|
-
positionals.push(arg);
|
|
90
|
-
}
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
return positionals;
|
|
94
|
-
}
|
|
95
|
-
|
|
96
|
-
/**
|
|
97
|
-
* Returns `true` when any of the given flags appear in `args`.
|
|
98
|
-
*
|
|
99
|
-
* Matches both standalone tokens (`-s`, `--silent`) and inline forms
|
|
100
|
-
* (`--output=file`). Useful for simple boolean flag checks inside command
|
|
101
|
-
* `run` handlers.
|
|
102
|
-
*
|
|
103
|
-
* @param args Tokenized argument array from `CommandContext.args`.
|
|
104
|
-
* @param flags Single flag string or array of equivalent flag strings.
|
|
105
|
-
* @returns `true` if at least one flag is present, otherwise `false`.
|
|
106
|
-
*
|
|
107
|
-
* @example
|
|
108
|
-
* ```ts
|
|
109
|
-
* ifFlag(args, "-r") // single flag
|
|
110
|
-
* ifFlag(args, ["-r", "--recursive"]) // aliases
|
|
111
|
-
* ```
|
|
112
|
-
*/
|
|
113
|
-
export function ifFlag(args: string[], flags: string | string[]): boolean {
|
|
114
|
-
const allFlags = toFlagList(flags);
|
|
115
|
-
|
|
116
|
-
for (const arg of args) {
|
|
117
|
-
for (const flag of allFlags) {
|
|
118
|
-
if (matchFlagToken(arg, flag).matched) {
|
|
119
|
-
return true;
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
return false;
|
|
125
|
-
}
|
|
126
|
-
|
|
127
|
-
/**
|
|
128
|
-
* Returns the value associated with a flag, or `true` if the flag is present
|
|
129
|
-
* but has no associated value, or `undefined` if the flag is absent.
|
|
130
|
-
*
|
|
131
|
-
* Handles three forms:
|
|
132
|
-
* - `--output file` → returns `"file"` (next token)
|
|
133
|
-
* - `--output=file` → returns `"file"` (inline `=` form)
|
|
134
|
-
* - `--verbose` → returns `true` (flag with no value)
|
|
135
|
-
*
|
|
136
|
-
* @param args Tokenized argument array from `CommandContext.args`.
|
|
137
|
-
* @param flags Single flag string or array of equivalent flag strings.
|
|
138
|
-
* @returns The flag value string, `true` when valueless, or `undefined`.
|
|
139
|
-
*
|
|
140
|
-
* @example
|
|
141
|
-
* ```ts
|
|
142
|
-
* const output = getFlag(args, ["-o", "--output"]);
|
|
143
|
-
* if (typeof output === "string") { /* use path *\/ }
|
|
144
|
-
* ```
|
|
145
|
-
*/
|
|
146
|
-
export function getFlag(
|
|
147
|
-
args: string[],
|
|
148
|
-
flags: string | string[],
|
|
149
|
-
): string | true | undefined {
|
|
150
|
-
const allFlags = toFlagList(flags);
|
|
151
|
-
|
|
152
|
-
for (let index = 0; index < args.length; index += 1) {
|
|
153
|
-
const arg = args[index]!;
|
|
154
|
-
|
|
155
|
-
for (const flag of allFlags) {
|
|
156
|
-
const match = matchFlagToken(arg, flag);
|
|
157
|
-
if (!match.matched) {
|
|
158
|
-
continue;
|
|
159
|
-
}
|
|
160
|
-
|
|
161
|
-
if (match.inlineValue !== null) {
|
|
162
|
-
return match.inlineValue;
|
|
163
|
-
}
|
|
164
|
-
|
|
165
|
-
const next = args[index + 1];
|
|
166
|
-
if (next !== undefined && next !== "--") {
|
|
167
|
-
return next;
|
|
168
|
-
}
|
|
169
|
-
|
|
170
|
-
return true;
|
|
171
|
-
}
|
|
172
|
-
}
|
|
173
|
-
|
|
174
|
-
return undefined;
|
|
175
|
-
}
|
|
176
|
-
|
|
177
|
-
/**
|
|
178
|
-
* Returns the positional argument at the given zero-based index, skipping
|
|
179
|
-
* known flags and their values.
|
|
180
|
-
*
|
|
181
|
-
* Flags declared in `options.flags` are treated as boolean and skipped.
|
|
182
|
-
* Flags declared in `options.flagsWithValue` consume the next token too.
|
|
183
|
-
* Tokens after `--` are always treated as positionals.
|
|
184
|
-
*
|
|
185
|
-
* @param args Tokenized argument array from `CommandContext.args`.
|
|
186
|
-
* @param index Zero-based positional index to retrieve.
|
|
187
|
-
* @param options Optional flag declarations to skip during positional collection.
|
|
188
|
-
* @returns The positional value, or `undefined` if the index is out of range.
|
|
189
|
-
*
|
|
190
|
-
* @example
|
|
191
|
-
* ```ts
|
|
192
|
-
* // args = ["-r", "src", "dest"]
|
|
193
|
-
* getArg(args, 0, { flags: ["-r"] }) // "src"
|
|
194
|
-
* getArg(args, 1, { flags: ["-r"] }) // "dest"
|
|
195
|
-
* ```
|
|
196
|
-
*/
|
|
197
|
-
export function getArg(
|
|
198
|
-
args: string[],
|
|
199
|
-
index: number,
|
|
200
|
-
options: ArgParseOptions = {},
|
|
201
|
-
): string | undefined {
|
|
202
|
-
const positionals = collectPositionals(args, options);
|
|
203
|
-
return positionals[index];
|
|
204
|
-
}
|
|
205
|
-
|
|
206
|
-
/**
|
|
207
|
-
* Parses an argument array into structured flags, flag values, and positionals.
|
|
208
|
-
*
|
|
209
|
-
* - `options.flags` — boolean flags (e.g. `["-r", "--recursive"]`); collected
|
|
210
|
-
* into a `Set<string>` and not treated as positionals.
|
|
211
|
-
* - `options.flagsWithValue` — flags that consume the next token or an inline
|
|
212
|
-
* `=value`; collected into a `Map<string, string>`.
|
|
213
|
-
* - All remaining tokens are positionals.
|
|
214
|
-
* - Tokens after `--` are always positionals, regardless of `-` prefix.
|
|
215
|
-
*
|
|
216
|
-
* @param args Tokenized argument array from `CommandContext.args`.
|
|
217
|
-
* @param options Flag declaration lists.
|
|
218
|
-
* @returns `{ flags, flagsWithValues, positionals }`.
|
|
219
|
-
*
|
|
220
|
-
* @example
|
|
221
|
-
* ```ts
|
|
222
|
-
* const { flags, flagsWithValues, positionals } = parseArgs(args, {
|
|
223
|
-
* flags: ["-r", "--recursive"],
|
|
224
|
-
* flagsWithValue: ["-o", "--output"],
|
|
225
|
-
* });
|
|
226
|
-
* const recursive = flags.has("-r");
|
|
227
|
-
* const output = flagsWithValues.get("-o");
|
|
228
|
-
* ```
|
|
229
|
-
*/
|
|
230
|
-
export function parseArgs(
|
|
231
|
-
args: string[],
|
|
232
|
-
options: { flags?: string[]; flagsWithValue?: string[] } = {},
|
|
233
|
-
): {
|
|
234
|
-
flags: Set<string>;
|
|
235
|
-
flagsWithValues: Map<string, string>;
|
|
236
|
-
positionals: string[];
|
|
237
|
-
} {
|
|
238
|
-
const flags = new Set<string>();
|
|
239
|
-
const flagsWithValues = new Map<string, string>();
|
|
240
|
-
const positionals: string[] = [];
|
|
241
|
-
const boolFlags = new Set(options.flags ?? []);
|
|
242
|
-
const valueFlags = new Set(options.flagsWithValue ?? []);
|
|
243
|
-
let passthrough = false;
|
|
244
|
-
|
|
245
|
-
for (let index = 0; index < args.length; index += 1) {
|
|
246
|
-
const arg = args[index]!;
|
|
247
|
-
|
|
248
|
-
if (passthrough) {
|
|
249
|
-
positionals.push(arg);
|
|
250
|
-
continue;
|
|
251
|
-
}
|
|
252
|
-
|
|
253
|
-
if (arg === "--") {
|
|
254
|
-
passthrough = true;
|
|
255
|
-
continue;
|
|
256
|
-
}
|
|
257
|
-
|
|
258
|
-
if (boolFlags.has(arg)) {
|
|
259
|
-
flags.add(arg);
|
|
260
|
-
continue;
|
|
261
|
-
}
|
|
262
|
-
|
|
263
|
-
if (valueFlags.has(arg)) {
|
|
264
|
-
const next = args[index + 1];
|
|
265
|
-
if (next && !next.startsWith("-")) {
|
|
266
|
-
flagsWithValues.set(arg, next);
|
|
267
|
-
index += 1;
|
|
268
|
-
} else {
|
|
269
|
-
flagsWithValues.set(arg, "");
|
|
270
|
-
}
|
|
271
|
-
continue;
|
|
272
|
-
}
|
|
273
|
-
|
|
274
|
-
const inlineFlag = Array.from(valueFlags).find((flag) =>
|
|
275
|
-
arg.startsWith(`${flag}=`),
|
|
276
|
-
);
|
|
277
|
-
if (inlineFlag) {
|
|
278
|
-
flagsWithValues.set(inlineFlag, arg.slice(inlineFlag.length + 1));
|
|
279
|
-
continue;
|
|
280
|
-
}
|
|
281
|
-
|
|
282
|
-
positionals.push(arg);
|
|
283
|
-
}
|
|
284
|
-
|
|
285
|
-
return { flags, flagsWithValues, positionals };
|
|
286
|
-
}
|
package/src/commands/cp.ts
DELETED
|
@@ -1,83 +0,0 @@
|
|
|
1
|
-
import type { ShellModule } from "../types/commands";
|
|
2
|
-
import { ifFlag } from "./command-helpers";
|
|
3
|
-
import { assertPathAccess, resolvePath } from "./helpers";
|
|
4
|
-
|
|
5
|
-
/**
|
|
6
|
-
* Copy files or directories inside the virtual filesystem.
|
|
7
|
-
* @category files
|
|
8
|
-
* @params ["[-r] <source> <dest>"]
|
|
9
|
-
*/
|
|
10
|
-
export const cpCommand: ShellModule = {
|
|
11
|
-
name: "cp",
|
|
12
|
-
description: "Copy files or directories",
|
|
13
|
-
category: "files",
|
|
14
|
-
params: ["[-r] <source> <dest>"],
|
|
15
|
-
run: ({ authUser, shell, cwd, args }) => {
|
|
16
|
-
const recursive = ifFlag(args, ["-r", "-R", "--recursive"]);
|
|
17
|
-
const positionals = args.filter((a) => !a.startsWith("-"));
|
|
18
|
-
const [srcArg, destArg] = positionals;
|
|
19
|
-
|
|
20
|
-
if (!srcArg || !destArg) {
|
|
21
|
-
return { stderr: "cp: missing operand", exitCode: 1 };
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
const srcPath = resolvePath(cwd, srcArg);
|
|
25
|
-
const destPath = resolvePath(cwd, destArg);
|
|
26
|
-
|
|
27
|
-
try {
|
|
28
|
-
assertPathAccess(authUser, srcPath, "cp");
|
|
29
|
-
assertPathAccess(authUser, destPath, "cp");
|
|
30
|
-
|
|
31
|
-
if (!shell.vfs.exists(srcPath)) {
|
|
32
|
-
return {
|
|
33
|
-
stderr: `cp: ${srcArg}: No such file or directory`,
|
|
34
|
-
exitCode: 1,
|
|
35
|
-
};
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
const srcStat = shell.vfs.stat(srcPath);
|
|
39
|
-
|
|
40
|
-
if (srcStat.type === "directory") {
|
|
41
|
-
if (!recursive) {
|
|
42
|
-
return {
|
|
43
|
-
stderr: `cp: ${srcArg}: is a directory (use -r)`,
|
|
44
|
-
exitCode: 1,
|
|
45
|
-
};
|
|
46
|
-
}
|
|
47
|
-
const copyDir = (from: string, to: string) => {
|
|
48
|
-
shell.vfs.mkdir(to, 0o755);
|
|
49
|
-
for (const entry of shell.vfs.list(from)) {
|
|
50
|
-
const fromEntry = `${from}/${entry}`;
|
|
51
|
-
const toEntry = `${to}/${entry}`;
|
|
52
|
-
const stat = shell.vfs.stat(fromEntry);
|
|
53
|
-
if (stat.type === "directory") {
|
|
54
|
-
copyDir(fromEntry, toEntry);
|
|
55
|
-
} else {
|
|
56
|
-
const content = shell.vfs.readFileRaw(fromEntry);
|
|
57
|
-
shell.writeFileAsUser(authUser, toEntry, content);
|
|
58
|
-
}
|
|
59
|
-
}
|
|
60
|
-
};
|
|
61
|
-
const finalDest =
|
|
62
|
-
shell.vfs.exists(destPath) &&
|
|
63
|
-
shell.vfs.stat(destPath).type === "directory"
|
|
64
|
-
? `${destPath}/${srcArg.split("/").pop()}`
|
|
65
|
-
: destPath;
|
|
66
|
-
copyDir(srcPath, finalDest);
|
|
67
|
-
} else {
|
|
68
|
-
const finalDest =
|
|
69
|
-
shell.vfs.exists(destPath) &&
|
|
70
|
-
shell.vfs.stat(destPath).type === "directory"
|
|
71
|
-
? `${destPath}/${srcArg.split("/").pop()}`
|
|
72
|
-
: destPath;
|
|
73
|
-
const content = shell.vfs.readFileRaw(srcPath);
|
|
74
|
-
shell.writeFileAsUser(authUser, finalDest, content);
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
return { exitCode: 0 };
|
|
78
|
-
} catch (err) {
|
|
79
|
-
const msg = err instanceof Error ? err.message : String(err);
|
|
80
|
-
return { stderr: `cp: ${msg}`, exitCode: 1 };
|
|
81
|
-
}
|
|
82
|
-
},
|
|
83
|
-
};
|
package/src/commands/curl.ts
DELETED
|
@@ -1,147 +0,0 @@
|
|
|
1
|
-
import type { ShellModule } from "../types/commands";
|
|
2
|
-
import { ifFlag, parseArgs } from "./command-helpers";
|
|
3
|
-
import { assertPathAccess, resolvePath } from "./helpers";
|
|
4
|
-
|
|
5
|
-
/**
|
|
6
|
-
* HTTP client wrapper using `fetch()` semantics (virtual curl).
|
|
7
|
-
* @category network
|
|
8
|
-
* @params ["[options] <url>"]
|
|
9
|
-
*/
|
|
10
|
-
export const curlCommand: ShellModule = {
|
|
11
|
-
name: "curl",
|
|
12
|
-
description: "Transfer data from or to a server (pure fetch)",
|
|
13
|
-
category: "network",
|
|
14
|
-
params: ["[options] <url>"],
|
|
15
|
-
run: async ({ authUser, cwd, args, shell }) => {
|
|
16
|
-
const { flagsWithValues, positionals } = parseArgs(args, {
|
|
17
|
-
flagsWithValue: [
|
|
18
|
-
"-o",
|
|
19
|
-
"--output",
|
|
20
|
-
"-X",
|
|
21
|
-
"--request",
|
|
22
|
-
"-d",
|
|
23
|
-
"--data",
|
|
24
|
-
"-H",
|
|
25
|
-
"--header",
|
|
26
|
-
"-u",
|
|
27
|
-
"--user",
|
|
28
|
-
],
|
|
29
|
-
});
|
|
30
|
-
|
|
31
|
-
if (ifFlag(args, ["--help", "-h"])) {
|
|
32
|
-
return {
|
|
33
|
-
stdout: [
|
|
34
|
-
"Usage: curl [options] <url>",
|
|
35
|
-
" -o, --output <file> Write to file",
|
|
36
|
-
" -X, --request <method> HTTP method",
|
|
37
|
-
" -d, --data <data> POST data",
|
|
38
|
-
" -H, --header <hdr> Extra header",
|
|
39
|
-
" -s, --silent Silent mode",
|
|
40
|
-
" -I, --head Fetch headers only",
|
|
41
|
-
" -L, --location Follow redirects",
|
|
42
|
-
" -v, --verbose Verbose",
|
|
43
|
-
].join("\n"),
|
|
44
|
-
exitCode: 0,
|
|
45
|
-
};
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
const url = positionals[0];
|
|
49
|
-
if (!url) return { stderr: "curl: no URL specified", exitCode: 1 };
|
|
50
|
-
|
|
51
|
-
const outputPath =
|
|
52
|
-
flagsWithValues.get("-o") ?? flagsWithValues.get("--output") ?? null;
|
|
53
|
-
const method = (
|
|
54
|
-
flagsWithValues.get("-X") ??
|
|
55
|
-
flagsWithValues.get("--request") ??
|
|
56
|
-
"GET"
|
|
57
|
-
).toUpperCase();
|
|
58
|
-
const postData =
|
|
59
|
-
flagsWithValues.get("-d") ?? flagsWithValues.get("--data") ?? null;
|
|
60
|
-
const headerRaw =
|
|
61
|
-
flagsWithValues.get("-H") ?? flagsWithValues.get("--header") ?? null;
|
|
62
|
-
const silent = ifFlag(args, ["-s", "--silent"]);
|
|
63
|
-
const headOnly = ifFlag(args, ["-I", "--head"]);
|
|
64
|
-
const followRedirects = ifFlag(args, ["-L", "--location"]);
|
|
65
|
-
const verbose = ifFlag(args, ["-v", "--verbose"]);
|
|
66
|
-
|
|
67
|
-
const extraHeaders: Record<string, string> = {
|
|
68
|
-
"User-Agent": "curl/7.88.1",
|
|
69
|
-
};
|
|
70
|
-
if (headerRaw) {
|
|
71
|
-
const idx = headerRaw.indexOf(":");
|
|
72
|
-
if (idx !== -1)
|
|
73
|
-
extraHeaders[headerRaw.slice(0, idx).trim()] = headerRaw
|
|
74
|
-
.slice(idx + 1)
|
|
75
|
-
.trim();
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
const finalMethod = postData && method === "GET" ? "POST" : method;
|
|
79
|
-
const fetchOpts: RequestInit = {
|
|
80
|
-
method: finalMethod,
|
|
81
|
-
headers: extraHeaders,
|
|
82
|
-
redirect: followRedirects ? "follow" : "manual",
|
|
83
|
-
};
|
|
84
|
-
if (postData) {
|
|
85
|
-
extraHeaders["Content-Type"] ??= "application/x-www-form-urlencoded";
|
|
86
|
-
fetchOpts.body = postData;
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
const stderrLines: string[] = [];
|
|
90
|
-
if (verbose) {
|
|
91
|
-
stderrLines.push(`* Trying ${url}...`, `* Connected`);
|
|
92
|
-
stderrLines.push(
|
|
93
|
-
`> ${finalMethod} / HTTP/1.1`,
|
|
94
|
-
`> Host: ${new URL(url).host}`,
|
|
95
|
-
);
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
let response: Response;
|
|
99
|
-
try {
|
|
100
|
-
const urlWithHttp = url.startsWith("http://") || url.startsWith("https://") ? url : `http://${url}`;
|
|
101
|
-
response = await fetch(urlWithHttp, fetchOpts);
|
|
102
|
-
} catch (err) {
|
|
103
|
-
const msg = err instanceof Error ? err.message : String(err);
|
|
104
|
-
return {
|
|
105
|
-
stderr: `curl: (6) Could not resolve host: ${msg}`,
|
|
106
|
-
exitCode: 6,
|
|
107
|
-
};
|
|
108
|
-
}
|
|
109
|
-
|
|
110
|
-
if (verbose) {
|
|
111
|
-
stderrLines.push(`< HTTP/1.1 ${response.status} ${response.statusText}`);
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
if (headOnly) {
|
|
115
|
-
const lines = [`HTTP/1.1 ${response.status} ${response.statusText}`];
|
|
116
|
-
for (const [k, v] of response.headers.entries()) lines.push(`${k}: ${v}`);
|
|
117
|
-
return { stdout: `${lines.join("\r\n")}\r\n`, exitCode: 0 };
|
|
118
|
-
}
|
|
119
|
-
|
|
120
|
-
let body: string;
|
|
121
|
-
try {
|
|
122
|
-
body = await response.text();
|
|
123
|
-
} catch {
|
|
124
|
-
return { stderr: "curl: failed to read response body", exitCode: 1 };
|
|
125
|
-
}
|
|
126
|
-
|
|
127
|
-
if (outputPath) {
|
|
128
|
-
const target = resolvePath(cwd, outputPath);
|
|
129
|
-
assertPathAccess(authUser, target, "curl");
|
|
130
|
-
shell.writeFileAsUser(authUser, target, body);
|
|
131
|
-
if (!silent)
|
|
132
|
-
stderrLines.push(
|
|
133
|
-
` % Total % Received\n100 ${body.length} 100 ${body.length}`,
|
|
134
|
-
);
|
|
135
|
-
return {
|
|
136
|
-
stderr: stderrLines.join("\n") || undefined,
|
|
137
|
-
exitCode: response.ok ? 0 : 22,
|
|
138
|
-
};
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
return {
|
|
142
|
-
stdout: body,
|
|
143
|
-
stderr: stderrLines.length > 0 ? stderrLines.join("\n") : undefined,
|
|
144
|
-
exitCode: response.ok ? 0 : 22,
|
|
145
|
-
};
|
|
146
|
-
},
|
|
147
|
-
};
|
package/src/commands/cut.ts
DELETED
|
@@ -1,36 +0,0 @@
|
|
|
1
|
-
import type { ShellModule } from "../types/commands";
|
|
2
|
-
import { getFlag } from "./command-helpers";
|
|
3
|
-
|
|
4
|
-
/**
|
|
5
|
-
* Extract selected fields from each line of input.
|
|
6
|
-
* @category text
|
|
7
|
-
* @params ["-d <delim> -f <fields> [file]"]
|
|
8
|
-
*/
|
|
9
|
-
export const cutCommand: ShellModule = {
|
|
10
|
-
name: "cut",
|
|
11
|
-
description: "Remove sections from lines",
|
|
12
|
-
category: "text",
|
|
13
|
-
params: ["-d <delim> -f <fields> [file]"],
|
|
14
|
-
run: ({ args, stdin }) => {
|
|
15
|
-
const delim = (getFlag(args, ["-d"]) as string | undefined) ?? "\t";
|
|
16
|
-
const fields = (getFlag(args, ["-f"]) as string | undefined) ?? "1";
|
|
17
|
-
const cols = fields.split(",").map((f) => {
|
|
18
|
-
const [a, b] = f.split("-").map(Number);
|
|
19
|
-
return b !== undefined
|
|
20
|
-
? { from: (a ?? 1) - 1, to: b - 1 }
|
|
21
|
-
: { from: (a ?? 1) - 1, to: (a ?? 1) - 1 };
|
|
22
|
-
});
|
|
23
|
-
const lines = (stdin ?? "").split("\n");
|
|
24
|
-
const out = lines.map((line) => {
|
|
25
|
-
const parts = line.split(delim);
|
|
26
|
-
const selected: string[] = [];
|
|
27
|
-
for (const col of cols) {
|
|
28
|
-
for (let i = col.from; i <= Math.min(col.to, parts.length - 1); i++) {
|
|
29
|
-
selected.push(parts[i] ?? "");
|
|
30
|
-
}
|
|
31
|
-
}
|
|
32
|
-
return selected.join(delim);
|
|
33
|
-
});
|
|
34
|
-
return { stdout: out.join("\n"), exitCode: 0 };
|
|
35
|
-
},
|
|
36
|
-
};
|
package/src/commands/date.ts
DELETED
|
@@ -1,30 +0,0 @@
|
|
|
1
|
-
import type { ShellModule } from "../types/commands";
|
|
2
|
-
|
|
3
|
-
/**
|
|
4
|
-
* Print the current date/time or a formatted representation.
|
|
5
|
-
* @category system
|
|
6
|
-
* @params ["[+format]"]
|
|
7
|
-
*/
|
|
8
|
-
export const dateCommand: ShellModule = {
|
|
9
|
-
name: "date",
|
|
10
|
-
description: "Print current date and time",
|
|
11
|
-
category: "system",
|
|
12
|
-
params: ["[+format]"],
|
|
13
|
-
run: ({ args }) => {
|
|
14
|
-
const now = new Date();
|
|
15
|
-
const fmt = args[0];
|
|
16
|
-
if (fmt?.startsWith("+")) {
|
|
17
|
-
const f = fmt
|
|
18
|
-
.slice(1)
|
|
19
|
-
.replace("%Y", String(now.getFullYear()))
|
|
20
|
-
.replace("%m", String(now.getMonth() + 1).padStart(2, "0"))
|
|
21
|
-
.replace("%d", String(now.getDate()).padStart(2, "0"))
|
|
22
|
-
.replace("%H", String(now.getHours()).padStart(2, "0"))
|
|
23
|
-
.replace("%M", String(now.getMinutes()).padStart(2, "0"))
|
|
24
|
-
.replace("%S", String(now.getSeconds()).padStart(2, "0"))
|
|
25
|
-
.replace("%s", String(Math.floor(now.getTime() / 1000)));
|
|
26
|
-
return { stdout: f, exitCode: 0 };
|
|
27
|
-
}
|
|
28
|
-
return { stdout: now.toString(), exitCode: 0 };
|
|
29
|
-
},
|
|
30
|
-
};
|