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