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,49 +0,0 @@
1
- import type { ShellModule } from "../types/commands";
2
- import { ifFlag } from "./command-helpers";
3
-
4
- /**
5
- * Declare variables and give them attributes (integer, readonly, export, array).
6
- * @category shell
7
- * @params ["[-i] [-r] [-x] [-a] [name[=value]...]"]
8
- */
9
- export const declareCommand: ShellModule = {
10
- name: "declare",
11
- aliases: ["local", "typeset"],
12
- description: "Declare variables and give them attributes",
13
- category: "shell",
14
- params: ["[-i] [-r] [-x] [-a] [name[=value]...]"],
15
- run: ({ args, env }) => {
16
- if (!env) return { exitCode: 0 };
17
-
18
- const integer = ifFlag(args, ["-i"]);
19
- const _readonly = ifFlag(args, ["-r"]);
20
- const _export_ = ifFlag(args, ["-x"]);
21
- const printAll = args.filter((a) => !a.startsWith("-")).length === 0;
22
-
23
- if (printAll) {
24
- const lines = Object.entries(env.vars).map(
25
- ([k, v]) => `declare -- ${k}="${v}"`,
26
- );
27
- return { stdout: lines.join("\n"), exitCode: 0 };
28
- }
29
-
30
- const assignments = args.filter((a) => !a.startsWith("-"));
31
- for (const token of assignments) {
32
- const eq = token.indexOf("=");
33
- if (eq === -1) {
34
- // Just declare (no value)
35
- if (!(token in env.vars)) env.vars[token] = "";
36
- } else {
37
- const name = token.slice(0, eq);
38
- let val = token.slice(eq + 1);
39
- if (integer) {
40
- const n = parseInt(val, 10);
41
- val = Number.isNaN(n) ? "0" : String(n);
42
- }
43
- env.vars[name] = val;
44
- }
45
- }
46
-
47
- return { exitCode: 0 };
48
- },
49
- };
@@ -1,98 +0,0 @@
1
- import type { CommandResult, ShellModule } from "../types/commands";
2
- import type { VirtualShell } from "../VirtualShell";
3
-
4
- /**
5
- * Delete a user. Root-only.
6
- *
7
- * Without `-f`: prompts for confirmation — the user must type the exact
8
- * username to proceed.
9
- *
10
- * With `-f` / `--force`: deletes without confirmation.
11
- *
12
- * Usage:
13
- * deluser <username>
14
- * deluser -f <username>
15
- */
16
- export const deluserCommand: ShellModule = {
17
- name: "deluser",
18
- description: "Delete a user",
19
- category: "users",
20
- params: ["[-f] <username>"],
21
- run: async ({ authUser, args, shell }) => {
22
- if (authUser !== "root") {
23
- return { stderr: "deluser: permission denied\n", exitCode: 1 };
24
- }
25
-
26
- const force =
27
- args.includes("-f") ||
28
- args.includes("--force") ||
29
- args.includes("-y");
30
- const username = args.find((a) => !a.startsWith("-"));
31
-
32
- if (!username) {
33
- return {
34
- stderr: "Usage: deluser [-f] <username>\n",
35
- exitCode: 1,
36
- };
37
- }
38
-
39
- if (!shell.users.listUsers().includes(username)) {
40
- return {
41
- stderr: `deluser: user '${username}' does not exist\n`,
42
- exitCode: 1,
43
- };
44
- }
45
-
46
- if (username === "root") {
47
- return {
48
- stderr: "deluser: cannot remove the root account\n",
49
- exitCode: 1,
50
- };
51
- }
52
-
53
- // Force mode — delete without confirmation
54
- if (force) {
55
- await shell.users.deleteUser(username);
56
- return {
57
- stdout: `Removing user '${username}' ...\ndeluser: done.\n`,
58
- exitCode: 0,
59
- };
60
- }
61
-
62
- // Interactive confirmation
63
- const onPassword = async (
64
- input: string,
65
- sh: VirtualShell,
66
- ): Promise<{ result: CommandResult | null; nextPrompt?: string }> => {
67
- if (input.trim() !== username) {
68
- return {
69
- result: {
70
- stderr: "deluser: confirmation did not match — user not deleted\n",
71
- exitCode: 1,
72
- },
73
- };
74
- }
75
-
76
- await sh.users.deleteUser(username);
77
- return {
78
- result: {
79
- stdout: `Removing user '${username}' ...\ndeluser: done.\n`,
80
- exitCode: 0,
81
- },
82
- };
83
- };
84
-
85
- return {
86
- sudoChallenge: {
87
- username,
88
- targetUser: username,
89
- commandLine: null,
90
- loginShell: false,
91
- prompt: `Warning: deleting user '${username}'.\nType the username to confirm: `,
92
- mode: "confirm",
93
- onPassword,
94
- },
95
- exitCode: 0,
96
- };
97
- },
98
- };
@@ -1,23 +0,0 @@
1
- import type { ShellModule } from "../types/commands";
2
-
3
- /**
4
- * Report filesystem disk space usage.
5
- * @category system
6
- * @params ["[-h]"]
7
- */
8
- export const dfCommand: ShellModule = {
9
- name: "df",
10
- description: "Report filesystem disk space usage",
11
- category: "system",
12
- params: ["[-h]"],
13
- run: ({ shell }) => {
14
- const bytes = shell.vfs.getUsageBytes();
15
- const used = (bytes / 1024).toFixed(0);
16
- const total = "1048576"; // 1GB virtual
17
- const avail = String(Number(total) - Number(used));
18
- const pct = Math.round((Number(used) / Number(total)) * 100);
19
- const hdr = "Filesystem 1K-blocks Used Available Use% Mounted on";
20
- const row = `virtual-fs ${total.padStart(9)} ${used.padStart(7)} ${avail.padStart(9)} ${pct}% /`;
21
- return { stdout: `${hdr}\n${row}`, exitCode: 0 };
22
- },
23
- };
@@ -1,43 +0,0 @@
1
- import type { ShellModule } from "../types/commands";
2
- import { resolvePath } from "./helpers";
3
-
4
- /**
5
- * Compare files line-by-line and print differing lines.
6
- * @category text
7
- * @params ["<file1> <file2>"]
8
- */
9
- export const diffCommand: ShellModule = {
10
- name: "diff",
11
- description: "Compare files line by line",
12
- category: "text",
13
- params: ["<file1> <file2>"],
14
- run: ({ shell, cwd, args }) => {
15
- const [f1, f2] = args;
16
- if (!f1 || !f2) return { stderr: "diff: missing operand", exitCode: 1 };
17
- const p1 = resolvePath(cwd, f1);
18
- const p2 = resolvePath(cwd, f2);
19
- let a: string[], b: string[];
20
- try {
21
- a = shell.vfs.readFile(p1).split("\n");
22
- } catch {
23
- return { stderr: `diff: ${f1}: No such file or directory`, exitCode: 2 };
24
- }
25
- try {
26
- b = shell.vfs.readFile(p2).split("\n");
27
- } catch {
28
- return { stderr: `diff: ${f2}: No such file or directory`, exitCode: 2 };
29
- }
30
-
31
- const out: string[] = [];
32
- const max = Math.max(a.length, b.length);
33
- for (let i = 0; i < max; i++) {
34
- const la = a[i];
35
- const lb = b[i];
36
- if (la !== lb) {
37
- if (la !== undefined) out.push(`< ${la}`);
38
- if (lb !== undefined) out.push(`> ${lb}`);
39
- }
40
- }
41
- return { stdout: out.join("\n"), exitCode: out.length > 0 ? 1 : 0 };
42
- },
43
- };
@@ -1,180 +0,0 @@
1
- import type { ShellModule } from "../types/commands";
2
- import { ifFlag, parseArgs } from "./command-helpers";
3
- import { getPackageManager } from "./helpers";
4
-
5
- /**
6
- * dpkg compatibility command (query/remove/list) backed by the virtual package manager.
7
- * @category package
8
- * @params ["[-l] [-s pkg] [-L pkg] [-i pkg] [--remove pkg]"]
9
- */
10
- export const dpkgCommand: ShellModule = {
11
- name: "dpkg",
12
- description: "Debian package manager low-level tool",
13
- category: "package",
14
- params: ["[-l] [-s pkg] [-L pkg] [-i pkg] [--remove pkg]"],
15
- run: ({ args, authUser, shell }) => {
16
- const pm = getPackageManager(shell);
17
- if (!pm)
18
- return { stderr: "dpkg: package manager not initialised", exitCode: 1 };
19
-
20
- const listFlag = ifFlag(args, ["-l", "--list"]);
21
- const statusFlag = ifFlag(args, ["-s", "--status"]);
22
- const listFilesFlag = ifFlag(args, ["-L", "--listfiles"]);
23
- const removeFlag = ifFlag(args, ["-r", "--remove"]);
24
- const purgeFlag = ifFlag(args, ["-P", "--purge"]);
25
-
26
- const { positionals } = parseArgs(args, {
27
- flags: [
28
- "-l",
29
- "--list",
30
- "-s",
31
- "--status",
32
- "-L",
33
- "--listfiles",
34
- "-r",
35
- "--remove",
36
- "-P",
37
- "--purge",
38
- ],
39
- });
40
-
41
- if (listFlag) {
42
- const pkgList = pm.listInstalled();
43
- if (pkgList.length === 0) {
44
- return {
45
- stdout: [
46
- "Desired=Unknown/Install/Remove/Purge/Hold",
47
- "|Status=Not/Inst/Conf-files/Unpacked/halF-conf/Half-inst/trig-aWait/Trig-pend",
48
- "|/ Err?=(none)/Reinst-required (Status,Err: uppercase=bad)",
49
- "||/ Name Version Architecture Description",
50
- "+++-==============-===============-============-========================================",
51
- "(no packages installed)",
52
- ].join("\n"),
53
- exitCode: 0,
54
- };
55
- }
56
-
57
- const header = [
58
- "Desired=Unknown/Install/Remove/Purge/Hold",
59
- "|Status=Not/Inst/Conf-files/Unpacked/halF-conf/Half-inst/trig-aWait/Trig-pend",
60
- "|/ Err?=(none)/Reinst-required (Status,Err: uppercase=bad)",
61
- "||/ Name Version Architecture Description",
62
- "+++-==============-===============-============-========================================",
63
- ];
64
-
65
- const rows = pkgList.map((p) => {
66
- const name = p.name.padEnd(14).slice(0, 14);
67
- const ver = p.version.padEnd(15).slice(0, 15);
68
- const arch = p.architecture.padEnd(12).slice(0, 12);
69
- const desc = (p.description || "").slice(0, 40);
70
- return `ii ${name} ${ver} ${arch} ${desc}`;
71
- });
72
-
73
- return { stdout: [...header, ...rows].join("\n"), exitCode: 0 };
74
- }
75
-
76
- if (statusFlag) {
77
- const pkgName = positionals[0];
78
- if (!pkgName)
79
- return { stderr: "dpkg: -s needs a package name", exitCode: 1 };
80
- const info = pm.show(pkgName);
81
- if (!info)
82
- return {
83
- stderr: `dpkg-query: package '${pkgName}' is not installed and no information is available`,
84
- exitCode: 1,
85
- };
86
- return { stdout: info, exitCode: 0 };
87
- }
88
-
89
- if (listFilesFlag) {
90
- const pkgName = positionals[0];
91
- if (!pkgName)
92
- return { stderr: "dpkg: -L needs a package name", exitCode: 1 };
93
- const installed = pm.listInstalled().find((p) => p.name === pkgName);
94
- if (!installed)
95
- return {
96
- stderr: `dpkg-query: package '${pkgName}' is not installed`,
97
- exitCode: 1,
98
- };
99
- if (installed.files.length === 0)
100
- return { stdout: "/.keep", exitCode: 0 };
101
- return { stdout: installed.files.join("\n"), exitCode: 0 };
102
- }
103
-
104
- if (removeFlag || purgeFlag) {
105
- if (authUser !== "root")
106
- return {
107
- stderr:
108
- "dpkg: error: requested operation requires superuser privilege",
109
- exitCode: 2,
110
- };
111
- if (positionals.length === 0)
112
- return { stderr: "dpkg: error: need an action option", exitCode: 2 };
113
- const { output, exitCode } = pm.remove(positionals, { purge: purgeFlag });
114
- return { stdout: output || undefined, exitCode };
115
- }
116
-
117
- // Default: show help
118
- return {
119
- stdout: [
120
- "Usage: dpkg [<option>...] <command>",
121
- "",
122
- "Commands:",
123
- " -l, --list List packages matching given pattern",
124
- " -s, --status <pkg>... Report status of specified package",
125
- " -L, --listfiles <pkg>... List files owned by package",
126
- " -r, --remove <pkg>... Remove <pkg> but leave its configuration",
127
- " -P, --purge <pkg>... Remove <pkg> and its configuration",
128
- ].join("\n"),
129
- exitCode: 0,
130
- };
131
- },
132
- };
133
-
134
- export const dpkgQueryCommand: ShellModule = {
135
- name: "dpkg-query",
136
- description: "Show information about installed packages",
137
- category: "package",
138
- params: ["-W [pkg] | -l [pattern]"],
139
- run: ({ args, shell }) => {
140
- const pm = getPackageManager(shell);
141
- if (!pm)
142
- return {
143
- stderr: "dpkg-query: package manager not initialised",
144
- exitCode: 1,
145
- };
146
-
147
- const listFlag = ifFlag(args, ["-l"]);
148
- const showFlag = ifFlag(args, ["-W", "--show"]);
149
- const { positionals } = parseArgs(args, {
150
- flags: ["-l", "-W", "--show"],
151
- });
152
-
153
- if (listFlag || showFlag) {
154
- const pkgList = pm.listInstalled();
155
- const pattern = positionals[0];
156
- const filtered = pattern
157
- ? pkgList.filter((p) => p.name.includes(pattern))
158
- : pkgList;
159
-
160
- if (showFlag) {
161
- return {
162
- stdout: filtered.map((p) => `${p.name}\t${p.version}`).join("\n"),
163
- exitCode: 0,
164
- };
165
- }
166
-
167
- const rows = filtered.map((p) => {
168
- const name = p.name.padEnd(14).slice(0, 14);
169
- const ver = p.version.padEnd(15).slice(0, 15);
170
- return `ii ${name} ${ver} amd64 ${(p.description || "").slice(0, 40)}`;
171
- });
172
- return {
173
- stdout: rows.join("\n") || "(no packages match)",
174
- exitCode: 0,
175
- };
176
- }
177
-
178
- return { stderr: "dpkg-query: need a flag (-l, -W)", exitCode: 1 };
179
- },
180
- };
@@ -1,56 +0,0 @@
1
- import type { ShellModule } from "../types/commands";
2
- import { ifFlag } from "./command-helpers";
3
- import { resolvePath } from "./helpers";
4
-
5
- /**
6
- * Estimate file and directory space usage.
7
- * @category system
8
- * @params ["[-h] [-s] [path]"]
9
- */
10
- export const duCommand: ShellModule = {
11
- name: "du",
12
- description: "Estimate file space usage",
13
- category: "system",
14
- params: ["[-h] [-s] [path]"],
15
- run: ({ shell, cwd, args }) => {
16
- const human = ifFlag(args, ["-h"]);
17
- const summary = ifFlag(args, ["-s"]);
18
- const target = args.find((a) => !a.startsWith("-")) ?? ".";
19
- const p = resolvePath(cwd, target);
20
-
21
- const fmt = (b: number) =>
22
- human ? `${(b / 1024).toFixed(1)}K` : String(Math.ceil(b / 1024));
23
-
24
- if (!shell.vfs.exists(p))
25
- return {
26
- stderr: `du: ${target}: No such file or directory`,
27
- exitCode: 1,
28
- };
29
-
30
- if (summary || shell.vfs.stat(p).type === "file") {
31
- return {
32
- stdout: `${fmt(shell.vfs.getUsageBytes(p))}\t${target}`,
33
- exitCode: 0,
34
- };
35
- }
36
-
37
- const lines: string[] = [];
38
- const walk = (dir: string, rel: string) => {
39
- let total = 0;
40
- for (const e of shell.vfs.list(dir)) {
41
- const full = `${dir}/${e}`,
42
- r = `${rel}/${e}`;
43
- const st = shell.vfs.stat(full);
44
- if (st.type === "directory") total += walk(full, r);
45
- else {
46
- total += st.size;
47
- if (!summary) lines.push(`${fmt(st.size)}\t${r}`);
48
- }
49
- }
50
- lines.push(`${fmt(total)}\t${rel}`);
51
- return total;
52
- };
53
- walk(p, target);
54
- return { stdout: lines.join("\n"), exitCode: 0 };
55
- },
56
- };
@@ -1,58 +0,0 @@
1
- import type { ShellModule } from "../types/commands";
2
- import { parseArgs } from "./command-helpers";
3
- import { expandSync } from "../utils/expand";
4
-
5
- /**
6
- * Expand escape sequences for `echo -e`.
7
- * Handles \n \t \r \\ \a \b \f \v and \0NNN (octal).
8
- */
9
- function expandEscapes(text: string): string {
10
- return text
11
- .replace(/\\n/g, "\n")
12
- .replace(/\\t/g, "\t")
13
- .replace(/\\r/g, "\r")
14
- .replace(/\\\\/g, "\\")
15
- .replace(/\\a/g, "\x07")
16
- .replace(/\\b/g, "\x08")
17
- .replace(/\\f/g, "\x0C")
18
- .replace(/\\v/g, "\x0B")
19
- .replace(/\\0(\d{1,3})/g, (_, oct) =>
20
- String.fromCharCode(parseInt(oct, 8)),
21
- );
22
- }
23
-
24
- /**
25
- * Echo text to stdout with shell-style expansion and escape support.
26
- * @category shell
27
- * @params ["[-n] [-e] [text...]"]
28
- */
29
- export const echoCommand: ShellModule = {
30
- name: "echo",
31
- description: "Display text",
32
- category: "shell",
33
- params: ["[-n] [-e] [text...]"],
34
- run: ({ args, stdin, env }) => {
35
- const { flags, positionals } = parseArgs(args, {
36
- flags: ["-n", "-e", "-E"],
37
- });
38
- const noNewline = flags.has("-n");
39
- const escapes = flags.has("-e");
40
-
41
- const rawText =
42
- positionals.length > 0 ? positionals.join(" ") : (stdin ?? "");
43
-
44
- // Full expansion: $? ${#VAR} $((expr)) ~ ${VAR:-def} $VAR etc.
45
- // $(cmd) is already resolved upstream by runCommand before echo.run is called.
46
- const expanded = expandSync(
47
- rawText,
48
- env?.vars ?? {},
49
- env?.lastExitCode ?? 0,
50
- );
51
- const text = escapes ? expandEscapes(expanded) : expanded;
52
-
53
- return {
54
- stdout: noNewline ? text : `${text}\n`,
55
- exitCode: 0,
56
- };
57
- },
58
- };
@@ -1,23 +0,0 @@
1
- /** biome-ignore-all lint/style/useNamingConvention: ENV VARS */
2
- import type { ShellModule } from "../types/commands";
3
-
4
- /**
5
- * Print environment variables for the current session.
6
- * @category shell
7
- * @params []
8
- */
9
- export const envCommand: ShellModule = {
10
- name: "env",
11
- description: "Print environment variables",
12
- category: "shell",
13
- params: [],
14
- run: ({ env, authUser }) => {
15
- const vars = { ...env.vars, USER: authUser, HOME: `/home/${authUser}` };
16
- return {
17
- stdout: Object.entries(vars)
18
- .map(([k, v]) => `${k}=${v}`)
19
- .join("\n"),
20
- exitCode: 0,
21
- };
22
- },
23
- };
@@ -1,18 +0,0 @@
1
- import type { ShellModule } from "../types/commands";
2
-
3
- /**
4
- * Exit the current shell session (closeSession flag).
5
- * @category shell
6
- * @params ["[code]"]
7
- */
8
- export const exitCommand: ShellModule = {
9
- name: "exit",
10
- aliases: ["bye"],
11
- description: "Exit the shell session",
12
- category: "shell",
13
- params: ["[code]"],
14
- run: ({ args }) => ({
15
- closeSession: true,
16
- exitCode: parseInt(args[0] ?? "0", 10) || 0,
17
- }),
18
- };
@@ -1,34 +0,0 @@
1
- import type { ShellModule } from "../types/commands";
2
-
3
- /**
4
- * Set or display shell environment variables for child processes.
5
- * @category shell
6
- * @params ["[VAR=value]"]
7
- */
8
- export const exportCommand: ShellModule = {
9
- name: "export",
10
- description: "Set shell environment variable",
11
- category: "shell",
12
- params: ["[VAR=value]"],
13
- run: ({ args, env }) => {
14
- // export -p or export with no args — list all exported vars
15
- if (args.length === 0 || (args.length === 1 && args[0] === "-p")) {
16
- const out = Object.entries(env.vars)
17
- .filter(([k]) => k && /^[A-Za-z_][A-Za-z0-9_]*$/.test(k))
18
- .map(([k, v]) => `declare -x ${k}="${v}"`)
19
- .join("\n");
20
- return { stdout: out ? `${out}\n` : "", exitCode: 0 };
21
- }
22
- for (const arg of args.filter((a) => a !== "-p")) {
23
- if (arg.includes("=")) {
24
- const eq = arg.indexOf("=");
25
- const name = arg.slice(0, eq);
26
- const value = arg.slice(eq + 1);
27
- env.vars[name] = value;
28
- } else {
29
- // mark existing as exported (already is)
30
- }
31
- }
32
- return { exitCode: 0 };
33
- },
34
- };
@@ -1,68 +0,0 @@
1
- import type { ShellModule } from "../types/commands";
2
- import { getFlag } from "./command-helpers";
3
- import { assertPathAccess, resolvePath } from "./helpers";
4
-
5
- /**
6
- * Find files and directories by name and type with minimal pattern support.
7
- * @category files
8
- * @params ["[path] [-name <pattern>] [-type f|d]"]
9
- */
10
- export const findCommand: ShellModule = {
11
- name: "find",
12
- description: "Search for files",
13
- category: "files",
14
- params: ["[path] [-name <pattern>] [-type f|d]"],
15
- run: ({ authUser, shell, cwd, args }) => {
16
- const namePattern = getFlag(args, ["-name"]);
17
- const typeFilter = getFlag(args, ["-type"]);
18
- const positionals = args.filter(
19
- (a) => !a.startsWith("-") && a !== namePattern && a !== typeFilter,
20
- );
21
- const rootArg = positionals[0] ?? ".";
22
- const rootPath = resolvePath(cwd, rootArg);
23
-
24
- try {
25
- assertPathAccess(authUser, rootPath, "find");
26
- if (!shell.vfs.exists(rootPath)) {
27
- return {
28
- stderr: `find: ${rootArg}: No such file or directory`,
29
- exitCode: 1,
30
- };
31
- }
32
- } catch (err) {
33
- const msg = err instanceof Error ? err.message : String(err);
34
- return { stderr: `find: ${msg}`, exitCode: 1 };
35
- }
36
-
37
- const nameRegex = namePattern
38
- ? new RegExp(
39
- `^${(namePattern as string).replace(/\./g, "\\.").replace(/\*/g, ".*").replace(/\?/g, ".")}$`,
40
- )
41
- : null;
42
-
43
- const results: string[] = [];
44
- const walk = (currentPath: string, display: string) => {
45
- const stat = shell.vfs.stat(currentPath);
46
-
47
- const matchesType =
48
- !typeFilter ||
49
- (typeFilter === "f" && stat.type === "file") ||
50
- (typeFilter === "d" && stat.type === "directory");
51
- const matchesName =
52
- !nameRegex || nameRegex.test(currentPath.split("/").pop() ?? "");
53
-
54
- if (matchesType && matchesName) results.push(display);
55
-
56
- if (stat.type === "directory") {
57
- for (const entry of shell.vfs.list(currentPath)) {
58
- const full = `${currentPath}/${entry}`;
59
- const disp = `${display}/${entry}`;
60
- walk(full, disp);
61
- }
62
- }
63
- };
64
-
65
- walk(rootPath, rootArg);
66
- return { stdout: results.join("\n"), exitCode: 0 };
67
- },
68
- };