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,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
- };