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.
Files changed (364) hide show
  1. package/README.md +43 -23
  2. package/dist/.tsbuildinfo +1 -0
  3. package/dist/SSHMimic/executor.js +23 -5
  4. package/dist/commands/basename.d.ts +13 -0
  5. package/dist/commands/basename.js +45 -0
  6. package/dist/commands/file.d.ts +8 -0
  7. package/dist/commands/file.js +57 -0
  8. package/dist/commands/fun.d.ts +32 -0
  9. package/dist/commands/fun.js +172 -0
  10. package/dist/commands/ifconfig.d.ts +7 -0
  11. package/dist/commands/ifconfig.js +52 -0
  12. package/dist/commands/last.d.ts +13 -0
  13. package/dist/commands/last.js +68 -0
  14. package/dist/commands/manuals-bundle.js +598 -6
  15. package/dist/commands/registry.js +24 -2
  16. package/dist/commands/runtime.js +159 -106
  17. package/dist/commands/sh.js +5 -0
  18. package/dist/commands/tput.d.ts +13 -0
  19. package/dist/commands/tput.js +76 -0
  20. package/dist/commands/w.d.ts +7 -0
  21. package/dist/commands/w.js +38 -0
  22. package/dist/utils/expand.d.ts +12 -0
  23. package/dist/utils/expand.js +84 -0
  24. package/package.json +9 -3
  25. package/.github/ISSUE_TEMPLATE/bug_report.yml +0 -50
  26. package/.github/ISSUE_TEMPLATE/feature_request.yml +0 -31
  27. package/.github/dependabot.yml +0 -27
  28. package/.github/pull_request_template.md +0 -21
  29. package/.github/workflows/create-pull-request.yml +0 -85
  30. package/.github/workflows/publish.yml +0 -25
  31. package/.github/workflows/test-battery.yml +0 -102
  32. package/.vscode/settings.json +0 -20
  33. package/CODE_OF_CONDUCT.md +0 -39
  34. package/CONTRIBUTING.md +0 -59
  35. package/HONEYPOT.md +0 -358
  36. package/SECURITY.md +0 -33
  37. package/benchmark-results.txt +0 -40
  38. package/benchmark-virtualshell.ts +0 -88
  39. package/biome.json +0 -37
  40. package/build.js +0 -22
  41. package/builds/fortune-nyx-v1.5.1-directbash-k6.1.0.mjs +0 -1768
  42. package/builds/fortune-nyx-v1.5.1-ssh-nosftp.js +0 -1768
  43. package/builds/fortune-nyx-v1.5.1-ssh.cjs +0 -1769
  44. package/builds/fortune-nyx-v1.5.1-web.min.js +0 -17022
  45. package/bun.lock +0 -244
  46. package/docs/.nojekyll +0 -1
  47. package/docs/app.js +0 -1755
  48. package/docs/assets/hierarchy.js +0 -1
  49. package/docs/assets/highlight.css +0 -162
  50. package/docs/assets/icons.js +0 -18
  51. package/docs/assets/icons.svg +0 -1
  52. package/docs/assets/main.js +0 -60
  53. package/docs/assets/navigation.js +0 -1
  54. package/docs/assets/search.js +0 -1
  55. package/docs/assets/style.css +0 -1633
  56. package/docs/classes/HoneyPot.html +0 -31
  57. package/docs/classes/IdleManager.html +0 -162
  58. package/docs/classes/SshClient.html +0 -66
  59. package/docs/classes/VirtualFileSystem.html +0 -279
  60. package/docs/classes/VirtualPackageManager.html +0 -63
  61. package/docs/classes/VirtualSftpServer.html +0 -169
  62. package/docs/classes/VirtualShell.html +0 -285
  63. package/docs/classes/VirtualSshServer.html +0 -182
  64. package/docs/classes/VirtualUserManager.html +0 -276
  65. package/docs/demo.html +0 -82
  66. package/docs/functions/assertDiff.html +0 -6
  67. package/docs/functions/diffSnapshots.html +0 -7
  68. package/docs/functions/formatDiff.html +0 -6
  69. package/docs/functions/getArg.html +0 -13
  70. package/docs/functions/getFlag.html +0 -15
  71. package/docs/functions/ifFlag.html +0 -11
  72. package/docs/hierarchy.html +0 -1
  73. package/docs/index.html +0 -1869
  74. package/docs/interfaces/AuditLogEntry.html +0 -6
  75. package/docs/interfaces/CommandContext.html +0 -22
  76. package/docs/interfaces/CommandResult.html +0 -26
  77. package/docs/interfaces/ExecStream.html +0 -11
  78. package/docs/interfaces/HoneyPotStats.html +0 -16
  79. package/docs/interfaces/IdleManagerOptions.html +0 -7
  80. package/docs/interfaces/InstalledPackage.html +0 -20
  81. package/docs/interfaces/NanoEditorSession.html +0 -8
  82. package/docs/interfaces/PackageDefinition.html +0 -30
  83. package/docs/interfaces/PackageFile.html +0 -8
  84. package/docs/interfaces/PasswordChallenge.html +0 -16
  85. package/docs/interfaces/RemoveOptions.html +0 -4
  86. package/docs/interfaces/ShellEnv.html +0 -6
  87. package/docs/interfaces/ShellModule.html +0 -14
  88. package/docs/interfaces/ShellProperties.html +0 -14
  89. package/docs/interfaces/ShellStream.html +0 -11
  90. package/docs/interfaces/SudoChallenge.html +0 -24
  91. package/docs/interfaces/VfsBaseNode.html +0 -12
  92. package/docs/interfaces/VfsDiff.html +0 -10
  93. package/docs/interfaces/VfsDiffEntry.html +0 -6
  94. package/docs/interfaces/VfsDiffModified.html +0 -10
  95. package/docs/interfaces/VfsDirectoryNode.html +0 -15
  96. package/docs/interfaces/VfsFileNode.html +0 -17
  97. package/docs/interfaces/VfsOptions.html +0 -26
  98. package/docs/interfaces/VfsSnapshot.html +0 -3
  99. package/docs/interfaces/VfsSnapshotBaseNode.html +0 -8
  100. package/docs/interfaces/VfsSnapshotDirectoryNode.html +0 -10
  101. package/docs/interfaces/VfsSnapshotFileNode.html +0 -12
  102. package/docs/interfaces/VirtualActiveSession.html +0 -12
  103. package/docs/interfaces/VirtualSftpServerOptions.html +0 -7
  104. package/docs/interfaces/VirtualShellVfsLike.html +0 -15
  105. package/docs/interfaces/VirtualShellVfsOptions.html +0 -3
  106. package/docs/interfaces/WriteFileOptions.html +0 -6
  107. package/docs/media/LICENSE +0 -21
  108. package/docs/modules.html +0 -1
  109. package/docs/types/ArgParseOptions.html +0 -4
  110. package/docs/types/CommandMode.html +0 -2
  111. package/docs/types/CommandOutcome.html +0 -2
  112. package/docs/types/IdleState.html +0 -1
  113. package/docs/types/VfsNodeStats.html +0 -2
  114. package/docs/types/VfsNodeType.html +0 -2
  115. package/docs/types/VfsPersistenceMode.html +0 -5
  116. package/docs/types/VfsSnapshotNode.html +0 -2
  117. package/examples/README.md +0 -288
  118. package/examples/app.js +0 -1755
  119. package/examples/app.ts +0 -299
  120. package/examples/build.js +0 -27
  121. package/examples/demo.html +0 -33
  122. package/examples/honeypot-audit.ts +0 -180
  123. package/examples/honeypot-export.ts +0 -253
  124. package/examples/honeypot-quickstart.ts +0 -110
  125. package/examples/index.html +0 -82
  126. package/examples/server.js +0 -55
  127. package/polyfills/buffer.js +0 -117
  128. package/polyfills/node_child_process/index.js +0 -2
  129. package/polyfills/node_crypto/index.js +0 -167
  130. package/polyfills/node_events/index.js +0 -9
  131. package/polyfills/node_fs/index.js +0 -202
  132. package/polyfills/node_fs/promises.js +0 -4
  133. package/polyfills/node_os/index.js +0 -9
  134. package/polyfills/node_path/index.js +0 -28
  135. package/polyfills/node_vm/index.js +0 -7
  136. package/polyfills/node_zlib/index.js +0 -3
  137. package/polyfills/process.js +0 -14
  138. package/polyfills/ssh2/index.js +0 -75
  139. package/scripts/build-all.mjs +0 -226
  140. package/scripts/build-names.mjs +0 -43
  141. package/scripts/generate-manuals-bundle.mjs +0 -49
  142. package/scripts/postinstall.js +0 -42
  143. package/scripts/publish-package.sh +0 -70
  144. package/src/Honeypot/index.ts +0 -457
  145. package/src/SSHClient/index.ts +0 -270
  146. package/src/SSHMimic/exec.ts +0 -49
  147. package/src/SSHMimic/executor.ts +0 -251
  148. package/src/SSHMimic/hostKey.ts +0 -21
  149. package/src/SSHMimic/index.ts +0 -337
  150. package/src/SSHMimic/loginBanner.ts +0 -36
  151. package/src/SSHMimic/loginFormat.ts +0 -10
  152. package/src/SSHMimic/prompt.ts +0 -14
  153. package/src/SSHMimic/sftp.ts +0 -883
  154. package/src/VirtualFileSystem/binaryPack.ts +0 -258
  155. package/src/VirtualFileSystem/index.ts +0 -1193
  156. package/src/VirtualFileSystem/internalTypes.ts +0 -43
  157. package/src/VirtualFileSystem/journal.ts +0 -171
  158. package/src/VirtualFileSystem/path.ts +0 -74
  159. package/src/VirtualPackageManager/index.ts +0 -1006
  160. package/src/VirtualShell/idleManager.ts +0 -137
  161. package/src/VirtualShell/index.ts +0 -475
  162. package/src/VirtualShell/shell.ts +0 -700
  163. package/src/VirtualShell/shellParser.ts +0 -285
  164. package/src/VirtualUserManager/index.ts +0 -758
  165. package/src/bun.d.ts +0 -1
  166. package/src/commands/adduser.ts +0 -103
  167. package/src/commands/alias.ts +0 -69
  168. package/src/commands/apt.ts +0 -233
  169. package/src/commands/awk.ts +0 -168
  170. package/src/commands/base64.ts +0 -29
  171. package/src/commands/cat.ts +0 -52
  172. package/src/commands/cd.ts +0 -25
  173. package/src/commands/chmod.ts +0 -85
  174. package/src/commands/clear.ts +0 -15
  175. package/src/commands/command-helpers.ts +0 -286
  176. package/src/commands/cp.ts +0 -83
  177. package/src/commands/curl.ts +0 -147
  178. package/src/commands/cut.ts +0 -36
  179. package/src/commands/date.ts +0 -30
  180. package/src/commands/declare.ts +0 -49
  181. package/src/commands/deluser.ts +0 -98
  182. package/src/commands/df.ts +0 -23
  183. package/src/commands/diff.ts +0 -43
  184. package/src/commands/dpkg.ts +0 -180
  185. package/src/commands/du.ts +0 -56
  186. package/src/commands/echo.ts +0 -58
  187. package/src/commands/env.ts +0 -23
  188. package/src/commands/exit.ts +0 -18
  189. package/src/commands/export.ts +0 -34
  190. package/src/commands/find.ts +0 -68
  191. package/src/commands/free.ts +0 -47
  192. package/src/commands/grep.ts +0 -116
  193. package/src/commands/groups.ts +0 -19
  194. package/src/commands/gzip.ts +0 -88
  195. package/src/commands/head.ts +0 -52
  196. package/src/commands/help.ts +0 -152
  197. package/src/commands/helpers.ts +0 -234
  198. package/src/commands/history.ts +0 -34
  199. package/src/commands/hostname.ts +0 -14
  200. package/src/commands/htop.ts +0 -20
  201. package/src/commands/id.ts +0 -19
  202. package/src/commands/index.ts +0 -9
  203. package/src/commands/kill.ts +0 -19
  204. package/src/commands/ln.ts +0 -71
  205. package/src/commands/ls.ts +0 -243
  206. package/src/commands/lsb-release.ts +0 -63
  207. package/src/commands/man.ts +0 -31
  208. package/src/commands/manuals/adduser.txt +0 -11
  209. package/src/commands/manuals/apt-cache.txt +0 -12
  210. package/src/commands/manuals/apt.txt +0 -20
  211. package/src/commands/manuals/awk.txt +0 -13
  212. package/src/commands/manuals/cat.txt +0 -14
  213. package/src/commands/manuals/cd.txt +0 -16
  214. package/src/commands/manuals/chmod.txt +0 -16
  215. package/src/commands/manuals/clear.txt +0 -10
  216. package/src/commands/manuals/cp.txt +0 -10
  217. package/src/commands/manuals/curl.txt +0 -20
  218. package/src/commands/manuals/date.txt +0 -14
  219. package/src/commands/manuals/declare.txt +0 -12
  220. package/src/commands/manuals/deluser.txt +0 -10
  221. package/src/commands/manuals/df.txt +0 -10
  222. package/src/commands/manuals/dpkg-query.txt +0 -11
  223. package/src/commands/manuals/dpkg.txt +0 -14
  224. package/src/commands/manuals/du.txt +0 -11
  225. package/src/commands/manuals/echo.txt +0 -11
  226. package/src/commands/manuals/false.txt +0 -10
  227. package/src/commands/manuals/find.txt +0 -11
  228. package/src/commands/manuals/free.txt +0 -12
  229. package/src/commands/manuals/grep.txt +0 -13
  230. package/src/commands/manuals/groups.txt +0 -10
  231. package/src/commands/manuals/gzip.txt +0 -11
  232. package/src/commands/manuals/head.txt +0 -10
  233. package/src/commands/manuals/help.txt +0 -11
  234. package/src/commands/manuals/history.txt +0 -11
  235. package/src/commands/manuals/hostname.txt +0 -10
  236. package/src/commands/manuals/id.txt +0 -10
  237. package/src/commands/manuals/kill.txt +0 -13
  238. package/src/commands/manuals/ls.txt +0 -20
  239. package/src/commands/manuals/lsb_release.txt +0 -14
  240. package/src/commands/manuals/mkdir.txt +0 -10
  241. package/src/commands/manuals/mv.txt +0 -10
  242. package/src/commands/manuals/nano.txt +0 -11
  243. package/src/commands/manuals/neofetch.txt +0 -10
  244. package/src/commands/manuals/node.txt +0 -13
  245. package/src/commands/manuals/npm.txt +0 -13
  246. package/src/commands/manuals/npx.txt +0 -13
  247. package/src/commands/manuals/passwd.txt +0 -11
  248. package/src/commands/manuals/ping.txt +0 -10
  249. package/src/commands/manuals/printf.txt +0 -11
  250. package/src/commands/manuals/ps.txt +0 -10
  251. package/src/commands/manuals/pwd.txt +0 -10
  252. package/src/commands/manuals/python3.txt +0 -13
  253. package/src/commands/manuals/readlink.txt +0 -10
  254. package/src/commands/manuals/return.txt +0 -10
  255. package/src/commands/manuals/rm.txt +0 -10
  256. package/src/commands/manuals/sed.txt +0 -11
  257. package/src/commands/manuals/set.txt +0 -11
  258. package/src/commands/manuals/shift.txt +0 -10
  259. package/src/commands/manuals/sleep.txt +0 -10
  260. package/src/commands/manuals/sort.txt +0 -12
  261. package/src/commands/manuals/source.txt +0 -11
  262. package/src/commands/manuals/ssh.txt +0 -11
  263. package/src/commands/manuals/stat.txt +0 -10
  264. package/src/commands/manuals/su.txt +0 -13
  265. package/src/commands/manuals/sudo.txt +0 -11
  266. package/src/commands/manuals/tail.txt +0 -10
  267. package/src/commands/manuals/tar.txt +0 -19
  268. package/src/commands/manuals/tee.txt +0 -10
  269. package/src/commands/manuals/test.txt +0 -11
  270. package/src/commands/manuals/touch.txt +0 -11
  271. package/src/commands/manuals/tr.txt +0 -10
  272. package/src/commands/manuals/trap.txt +0 -10
  273. package/src/commands/manuals/true.txt +0 -10
  274. package/src/commands/manuals/type.txt +0 -10
  275. package/src/commands/manuals/uname.txt +0 -12
  276. package/src/commands/manuals/uniq.txt +0 -12
  277. package/src/commands/manuals/unset.txt +0 -10
  278. package/src/commands/manuals/uptime.txt +0 -11
  279. package/src/commands/manuals/wc.txt +0 -12
  280. package/src/commands/manuals/wget.txt +0 -12
  281. package/src/commands/manuals/which.txt +0 -10
  282. package/src/commands/manuals/whoami.txt +0 -10
  283. package/src/commands/manuals/xargs.txt +0 -10
  284. package/src/commands/manuals-bundle.ts +0 -898
  285. package/src/commands/mkdir.ts +0 -31
  286. package/src/commands/mv.ts +0 -50
  287. package/src/commands/nano.ts +0 -38
  288. package/src/commands/neofetch.ts +0 -53
  289. package/src/commands/node.ts +0 -341
  290. package/src/commands/npm.ts +0 -132
  291. package/src/commands/passwd.ts +0 -50
  292. package/src/commands/ping.ts +0 -32
  293. package/src/commands/printf.ts +0 -129
  294. package/src/commands/ps.ts +0 -58
  295. package/src/commands/pwd.ts +0 -9
  296. package/src/commands/python.ts +0 -2229
  297. package/src/commands/read.ts +0 -46
  298. package/src/commands/registry.ts +0 -249
  299. package/src/commands/rm.ts +0 -42
  300. package/src/commands/runtime.ts +0 -378
  301. package/src/commands/sed.ts +0 -68
  302. package/src/commands/seq.ts +0 -43
  303. package/src/commands/set.ts +0 -29
  304. package/src/commands/sh.ts +0 -467
  305. package/src/commands/shift.ts +0 -63
  306. package/src/commands/sleep.ts +0 -20
  307. package/src/commands/sort.ts +0 -46
  308. package/src/commands/source.ts +0 -52
  309. package/src/commands/stat.ts +0 -61
  310. package/src/commands/su.ts +0 -72
  311. package/src/commands/sudo.ts +0 -76
  312. package/src/commands/tail.ts +0 -53
  313. package/src/commands/tar.ts +0 -102
  314. package/src/commands/tee.ts +0 -36
  315. package/src/commands/test.ts +0 -137
  316. package/src/commands/touch.ts +0 -28
  317. package/src/commands/tr.ts +0 -70
  318. package/src/commands/tree.ts +0 -20
  319. package/src/commands/true.ts +0 -27
  320. package/src/commands/type.ts +0 -48
  321. package/src/commands/uname.ts +0 -29
  322. package/src/commands/uniq.ts +0 -39
  323. package/src/commands/unset.ts +0 -17
  324. package/src/commands/uptime.ts +0 -54
  325. package/src/commands/wc.ts +0 -55
  326. package/src/commands/wget.ts +0 -148
  327. package/src/commands/which.ts +0 -37
  328. package/src/commands/who.ts +0 -25
  329. package/src/commands/whoami.ts +0 -14
  330. package/src/commands/xargs.ts +0 -31
  331. package/src/index.ts +0 -67
  332. package/src/modules/linuxRootfs.ts +0 -1961
  333. package/src/modules/neofetch.ts +0 -358
  334. package/src/modules/shellInteractive.ts +0 -57
  335. package/src/modules/shellRuntime.ts +0 -76
  336. package/src/self-standalone.ts +0 -542
  337. package/src/standalone-wo-sftp.ts +0 -38
  338. package/src/standalone.ts +0 -72
  339. package/src/types/commands.ts +0 -146
  340. package/src/types/pipeline.ts +0 -52
  341. package/src/types/streams.ts +0 -32
  342. package/src/types/tar-stream.d.ts +0 -38
  343. package/src/types/vfs.ts +0 -98
  344. package/src/utils/expand.ts +0 -491
  345. package/src/utils/perfLogger.ts +0 -72
  346. package/src/utils/tokenize.ts +0 -98
  347. package/src/utils/vfsDiff.ts +0 -275
  348. package/tests/command-helpers.test.ts +0 -116
  349. package/tests/commands-admin-net.test.ts +0 -441
  350. package/tests/commands-advanced.test.ts +0 -456
  351. package/tests/commands-core.test.ts +0 -562
  352. package/tests/commands-missing.test.ts +0 -570
  353. package/tests/commands-specific-units.test.ts +0 -327
  354. package/tests/commands-text-sys.test.ts +0 -445
  355. package/tests/expand.test.ts +0 -170
  356. package/tests/helpers.test.ts +0 -97
  357. package/tests/new-features.test.ts +0 -1036
  358. package/tests/parser-executor.test.ts +0 -37
  359. package/tests/sftp.test.ts +0 -323
  360. package/tests/ssh-exec.test.ts +0 -45
  361. package/tests/test-helper.ts +0 -79
  362. package/tests/users.test.ts +0 -86
  363. package/tsconfig.json +0 -49
  364. package/typedoc.json +0 -47
@@ -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
- };
@@ -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
- }
@@ -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
- };
@@ -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
- };
@@ -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
- };
@@ -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
- };