typescript-virtual-container 1.5.3 → 1.5.5

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