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,700 +0,0 @@
1
- import type { ChildProcessWithoutNullStreams } from "node:child_process";
2
- import { readFile, unlink, writeFile } from "node:fs/promises";
3
- import * as path from "node:path";
4
- import type { ShellProperties, VirtualShell } from ".";
5
- import { getCommandNames, makeDefaultEnv, runCommand, userHome } from "../commands";
6
- import {
7
- spawnHtopProcess,
8
- spawnNanoEditorProcess,
9
- } from "../modules/shellInteractive";
10
- import {
11
- getVisibleHtopPidList,
12
- resolvePath,
13
- type TerminalSize,
14
- toTtyLines,
15
- } from "../modules/shellRuntime";
16
- import { buildLoginBanner } from "../SSHMimic/loginBanner";
17
- import { buildPrompt } from "../SSHMimic/prompt";
18
- import type { CommandResult, ShellEnv } from "../types/commands";
19
- import type { ShellStream } from "../types/streams";
20
- import type VirtualFileSystem from "../VirtualFileSystem";
21
-
22
- interface NanoSession {
23
- kind: "nano" | "htop";
24
- targetPath: string;
25
- tempPath: string;
26
- process: ChildProcessWithoutNullStreams;
27
- }
28
-
29
- interface PendingSudo {
30
- username: string;
31
- targetUser: string;
32
- commandLine: string | null;
33
- loginShell: boolean;
34
- prompt: string;
35
- buffer: string;
36
- mode?: "sudo" | "passwd" | "confirm";
37
- onPassword?: (input: string, shell: VirtualShell) => Promise<{
38
- result: CommandResult | null;
39
- nextPrompt?: string;
40
- }>;
41
- }
42
-
43
- export function startShell(
44
- properties: ShellProperties,
45
- stream: ShellStream,
46
- authUser: string,
47
- hostname: string,
48
- sessionId: string | null,
49
- remoteAddress = "unknown",
50
- terminalSize: TerminalSize = { cols: 80, rows: 24 },
51
- shell: VirtualShell,
52
- ): void {
53
- let lineBuffer = "";
54
- let cursorPos = 0;
55
- let history = loadHistory(shell.vfs, authUser);
56
- let historyIndex: number | null = null;
57
- let historyDraft = "";
58
- let cwd = userHome(authUser);
59
- const shellEnv: ShellEnv = makeDefaultEnv(authUser, hostname);
60
- let nanoSession: NanoSession | null = null;
61
- let pendingSudo: PendingSudo | null = null;
62
- const buildCurrentPrompt = (): string => {
63
- const homePath = userHome(authUser);
64
- const cwdLabel = cwd === homePath ? "~" : path.posix.basename(cwd) || "/";
65
- return buildPrompt(authUser, hostname, cwdLabel);
66
- };
67
- const commandNames = Array.from(new Set(getCommandNames())).sort();
68
- console.log(
69
- `[${sessionId}] Shell started for user '${authUser}' at ${remoteAddress}`,
70
- );
71
-
72
- // Load .bashrc if it exists
73
- void (async () => {
74
- const bashrcPath = `${userHome(authUser)}/.bashrc`;
75
- if (shell.vfs.exists(bashrcPath)) {
76
- try {
77
- const bashrc = shell.vfs.readFile(bashrcPath);
78
- for (const line of bashrc.split("\n")) {
79
- const l = line.trim();
80
- if (!l || l.startsWith("#")) continue;
81
- await runCommand(
82
- l,
83
- authUser,
84
- hostname,
85
- "shell",
86
- cwd,
87
- shell,
88
- undefined,
89
- shellEnv,
90
- );
91
- }
92
- } catch {
93
- /* ignore bashrc errors */
94
- }
95
- }
96
- })();
97
-
98
- function renderLine(): void {
99
- const prompt = buildCurrentPrompt();
100
- stream.write(`\r${prompt}${lineBuffer}\u001b[K`);
101
-
102
- const moveLeft = lineBuffer.length - cursorPos;
103
- if (moveLeft > 0) {
104
- stream.write(`\u001b[${moveLeft}D`);
105
- }
106
- }
107
-
108
- function clearCurrentLine(): void {
109
- stream.write("\r\u001b[K");
110
- }
111
-
112
- function startSudoPrompt(challenge: {
113
- username: string;
114
- targetUser: string;
115
- commandLine: string | null;
116
- loginShell: boolean;
117
- prompt: string;
118
- mode?: "sudo" | "passwd" | "confirm";
119
- onPassword?: (input: string, shell: VirtualShell) => Promise<{
120
- result: CommandResult | null;
121
- nextPrompt?: string;
122
- }>;
123
- }): void {
124
- pendingSudo = {
125
- ...challenge,
126
- buffer: "",
127
- };
128
- clearCurrentLine();
129
- stream.write(challenge.prompt);
130
- }
131
-
132
- async function finishSudoPrompt(success: boolean): Promise<void> {
133
- if (!pendingSudo) {
134
- return;
135
- }
136
-
137
- const challenge = pendingSudo;
138
- pendingSudo = null;
139
-
140
- if (!success) {
141
- stream.write("\r\nSorry, try again.\r\n");
142
- renderLine();
143
- return;
144
- }
145
-
146
- if (!challenge.commandLine) {
147
- authUser = challenge.targetUser;
148
- if (challenge.loginShell) {
149
- cwd = userHome(authUser);
150
- }
151
- shell.users.updateSession(sessionId, authUser, remoteAddress);
152
- stream.write("\r\n");
153
- renderLine();
154
- return;
155
- }
156
-
157
- const runCwd = challenge.loginShell ? userHome(challenge.targetUser) : cwd;
158
- const result = await Promise.resolve(
159
- runCommand(
160
- challenge.commandLine,
161
- challenge.targetUser,
162
- hostname,
163
- "shell",
164
- runCwd,
165
- shell,
166
- ),
167
- );
168
-
169
- stream.write("\r\n");
170
-
171
- if (result.openEditor) {
172
- await startNanoEditor(
173
- result.openEditor.targetPath,
174
- result.openEditor.initialContent,
175
- result.openEditor.tempPath,
176
- );
177
- return;
178
- }
179
-
180
- if (result.openHtop) {
181
- await startHtop();
182
- return;
183
- }
184
-
185
- if (result.clearScreen) {
186
- stream.write("\u001b[2J\u001b[H");
187
- }
188
-
189
- if (result.stdout) {
190
- stream.write(`${toTtyLines(result.stdout)}\r\n`);
191
- }
192
-
193
- if (result.stderr) {
194
- stream.write(`${toTtyLines(result.stderr)}\r\n`);
195
- }
196
-
197
- if (result.switchUser) {
198
- authUser = result.switchUser;
199
- cwd = result.nextCwd ?? userHome(authUser);
200
- shell.users.updateSession(sessionId, authUser, remoteAddress);
201
- } else if (result.nextCwd) {
202
- cwd = result.nextCwd;
203
- }
204
-
205
- // WAL: checkpoint handled by auto-flush timer
206
- renderLine();
207
- }
208
-
209
- async function finishNanoEditor(): Promise<void> {
210
- if (!nanoSession) {
211
- return;
212
- }
213
-
214
- const activeSession = nanoSession;
215
-
216
- if (activeSession.kind === "nano") {
217
- try {
218
- const updatedContent = await readFile(activeSession.tempPath, "utf8");
219
- shell.writeFileAsUser(
220
- authUser,
221
- activeSession.targetPath,
222
- updatedContent,
223
- );
224
- // WAL: checkpoint handled by auto-flush timer
225
- } catch {
226
- // If temp file does not exist, nano exited without writing.
227
- }
228
-
229
- await unlink(activeSession.tempPath).catch(() => undefined);
230
- }
231
-
232
- nanoSession = null;
233
- lineBuffer = "";
234
- cursorPos = 0;
235
- stream.write("\r\n");
236
- renderLine();
237
- }
238
-
239
- async function startNanoEditor(
240
- targetPath: string,
241
- initialContent: string,
242
- tempPath: string,
243
- ): Promise<void> {
244
- if (shell.vfs.exists(targetPath)) {
245
- await writeFile(tempPath, initialContent, "utf8");
246
- }
247
-
248
- const editor = spawnNanoEditorProcess(tempPath, terminalSize, stream);
249
-
250
- editor.on("error", (error: Error) => {
251
- stream.write(`nano: ${error.message}\r\n`);
252
- void finishNanoEditor();
253
- });
254
-
255
- editor.on("close", () => {
256
- void finishNanoEditor();
257
- });
258
-
259
- nanoSession = {
260
- kind: "nano",
261
- targetPath,
262
- tempPath,
263
- process: editor,
264
- };
265
- }
266
-
267
- async function startHtop(): Promise<void> {
268
- const pidList = await getVisibleHtopPidList();
269
- if (!pidList) {
270
- stream.write("htop: no child_process processes to display\r\n");
271
- return;
272
- }
273
-
274
- const monitor = spawnHtopProcess(pidList, terminalSize, stream);
275
-
276
- monitor.on("error", (error: Error) => {
277
- stream.write(`htop: ${error.message}\r\n`);
278
- void finishNanoEditor();
279
- });
280
-
281
- monitor.on("close", () => {
282
- void finishNanoEditor();
283
- });
284
-
285
- nanoSession = {
286
- kind: "htop",
287
- targetPath: "",
288
- tempPath: "",
289
- process: monitor,
290
- };
291
- }
292
-
293
- function applyHistoryLine(nextLine: string): void {
294
- lineBuffer = nextLine;
295
- cursorPos = lineBuffer.length;
296
- renderLine();
297
- }
298
-
299
- function insertText(text: string): void {
300
- lineBuffer = `${lineBuffer.slice(0, cursorPos)}${text}${lineBuffer.slice(cursorPos)}`;
301
- cursorPos += text.length;
302
- renderLine();
303
- }
304
-
305
- function getTokenRange(
306
- line: string,
307
- cursor: number,
308
- ): { start: number; end: number } {
309
- let start = cursor;
310
- while (start > 0 && !/\s/.test(line[start - 1]!)) {
311
- start -= 1;
312
- }
313
-
314
- let end = cursor;
315
- while (end < line.length && !/\s/.test(line[end]!)) {
316
- end += 1;
317
- }
318
-
319
- return { start, end };
320
- }
321
-
322
- function listPathCompletions(prefix: string): string[] {
323
- const slashIndex = prefix.lastIndexOf("/");
324
- const dirPart = slashIndex >= 0 ? prefix.slice(0, slashIndex + 1) : "";
325
- const namePart = slashIndex >= 0 ? prefix.slice(slashIndex + 1) : prefix;
326
- const basePath = resolvePath(cwd, dirPart || ".");
327
-
328
- try {
329
- return shell.vfs
330
- .list(basePath)
331
- .filter((entry) => !entry.startsWith("."))
332
- .filter((entry) => entry.startsWith(namePart))
333
- .map((entry) => {
334
- const fullPath = path.posix.join(basePath, entry);
335
- const st = shell.vfs.stat(fullPath);
336
- const suffix = st.type === "directory" ? "/" : "";
337
- return `${dirPart}${entry}${suffix}`;
338
- })
339
- .sort();
340
- } catch {
341
- return [];
342
- }
343
- }
344
-
345
- function handleTabCompletion(): void {
346
- const { start, end } = getTokenRange(lineBuffer, cursorPos);
347
- const token = lineBuffer.slice(start, cursorPos);
348
-
349
- if (token.length === 0) {
350
- return;
351
- }
352
-
353
- const firstToken = lineBuffer.slice(0, start).trim().length === 0;
354
- const commandCandidates = firstToken
355
- ? commandNames.filter((name) => name.startsWith(token))
356
- : [];
357
- const pathCandidates = listPathCompletions(token);
358
- const candidates = Array.from(
359
- new Set([...commandCandidates, ...pathCandidates]),
360
- ).sort();
361
-
362
- if (candidates.length === 0) {
363
- return;
364
- }
365
-
366
- if (candidates.length === 1) {
367
- const completed = candidates[0]!;
368
- const suffix = completed.endsWith("/") ? "" : " ";
369
- lineBuffer = `${lineBuffer.slice(0, start)}${completed}${suffix}${lineBuffer.slice(end)}`;
370
- cursorPos = start + completed.length + suffix.length;
371
- renderLine();
372
- return;
373
- }
374
-
375
- stream.write("\r\n");
376
- stream.write(`${candidates.join(" ")}\r\n`);
377
- renderLine();
378
- }
379
-
380
- function pushHistory(cmd: string): void {
381
- if (cmd.length === 0) {
382
- return;
383
- }
384
-
385
- history.push(cmd);
386
- if (history.length > 500) {
387
- history = history.slice(history.length - 500);
388
- }
389
-
390
- const data = history.length > 0 ? `${history.join("\n")}\n` : "";
391
- shell.vfs.writeFile(`${userHome(authUser)}/.bash_history`, data);
392
- }
393
-
394
- function readLastLogin(): { at: string; from: string } | null {
395
- const lastlogPath = `${userHome(authUser)}/.lastlog.json`;
396
- if (!shell.vfs.exists(lastlogPath)) {
397
- return null;
398
- }
399
-
400
- try {
401
- return JSON.parse(shell.vfs.readFile(lastlogPath)) as {
402
- at: string;
403
- from: string;
404
- };
405
- } catch {
406
- return null;
407
- }
408
- }
409
-
410
- function writeLastLogin(nowIso: string): void {
411
- const lastlogPath = `${userHome(authUser)}/.lastlog`;
412
- shell.vfs.writeFile(
413
- lastlogPath,
414
- JSON.stringify({ at: nowIso, from: remoteAddress }),
415
- );
416
- }
417
-
418
- function renderLoginBanner(): void {
419
- const last = readLastLogin();
420
- const nowIso = new Date().toISOString();
421
- stream.write(buildLoginBanner(hostname, properties, last));
422
- writeLastLogin(nowIso);
423
- }
424
-
425
- renderLoginBanner();
426
- renderLine();
427
-
428
- stream.on("data", async (chunk: Buffer) => {
429
- if (nanoSession) {
430
- nanoSession.process.stdin.write(chunk);
431
- return;
432
- }
433
-
434
- if (pendingSudo) {
435
- const input = chunk.toString("utf8");
436
-
437
- for (let i = 0; i < input.length; i += 1) {
438
- const ch = input[i]!;
439
-
440
- if (ch === "\u0003") {
441
- pendingSudo = null;
442
- stream.write("^C\r\n");
443
- renderLine();
444
- return;
445
- }
446
-
447
- if (ch === "\u007f" || ch === "\b") {
448
- pendingSudo.buffer = pendingSudo.buffer.slice(0, -1);
449
- continue;
450
- }
451
-
452
- if (ch === "\r" || ch === "\n") {
453
- const typed = pendingSudo.buffer;
454
- pendingSudo.buffer = "";
455
-
456
- // ── Generic onPassword handler (passwd / confirm modes) ────
457
- if (pendingSudo.onPassword) {
458
- const { result, nextPrompt } = await pendingSudo.onPassword(typed, shell);
459
- stream.write("\r\n");
460
- if (result !== null) {
461
- pendingSudo = null;
462
- if (result.stdout) stream.write(result.stdout.replace(/\n/g, "\r\n"));
463
- if (result.stderr) stream.write(result.stderr.replace(/\n/g, "\r\n"));
464
- renderLine();
465
- } else {
466
- if (nextPrompt) pendingSudo.prompt = nextPrompt;
467
- stream.write(pendingSudo.prompt);
468
- }
469
- return;
470
- }
471
-
472
- // ── Default sudo mode — verify current user's password ─────
473
- const valid = shell.users.verifyPassword(
474
- pendingSudo.username,
475
- typed,
476
- );
477
- await finishSudoPrompt(valid);
478
- return;
479
- }
480
-
481
- if (ch >= " ") {
482
- pendingSudo.buffer += ch;
483
- }
484
- }
485
-
486
- return;
487
- }
488
-
489
- const input = chunk.toString("utf8");
490
-
491
- for (let i = 0; i < input.length; i += 1) {
492
- const ch = input[i]!;
493
-
494
- if (ch === "\u0004") {
495
- lineBuffer = "";
496
- cursorPos = 0;
497
- historyIndex = null;
498
- historyDraft = "";
499
- stream.write("bye\r\n");
500
- pushHistory("bye");
501
- // WAL: checkpoint handled by auto-flush timer
502
- stream.write("logout\r\n");
503
- stream.exit(0);
504
- stream.end();
505
- return;
506
- }
507
-
508
- if (ch === "\t") {
509
- handleTabCompletion();
510
- continue;
511
- }
512
-
513
- if (ch === "\u001b") {
514
- const next = input[i + 1];
515
- const third = input[i + 2];
516
- const fourth = input[i + 3];
517
-
518
- if (next === "[" && third) {
519
- if (third === "A") {
520
- i += 2;
521
- if (history.length > 0) {
522
- if (historyIndex === null) {
523
- historyDraft = lineBuffer;
524
- historyIndex = history.length - 1;
525
- } else if (historyIndex > 0) {
526
- historyIndex -= 1;
527
- }
528
- applyHistoryLine(history[historyIndex] ?? "");
529
- }
530
- continue;
531
- }
532
-
533
- if (third === "B") {
534
- i += 2;
535
- if (historyIndex !== null) {
536
- if (historyIndex < history.length - 1) {
537
- historyIndex += 1;
538
- applyHistoryLine(history[historyIndex] ?? "");
539
- } else {
540
- historyIndex = null;
541
- applyHistoryLine(historyDraft);
542
- }
543
- }
544
- continue;
545
- }
546
-
547
- if (third === "C") {
548
- i += 2;
549
- if (cursorPos < lineBuffer.length) {
550
- cursorPos += 1;
551
- stream.write("\u001b[C");
552
- }
553
- continue;
554
- }
555
-
556
- if (third === "D") {
557
- i += 2;
558
- if (cursorPos > 0) {
559
- cursorPos -= 1;
560
- stream.write("\u001b[D");
561
- }
562
- continue;
563
- }
564
-
565
- if (third === "3" && fourth === "~") {
566
- i += 3;
567
- if (cursorPos < lineBuffer.length) {
568
- lineBuffer = `${lineBuffer.slice(0, cursorPos)}${lineBuffer.slice(cursorPos + 1)}`;
569
- renderLine();
570
- }
571
- continue;
572
- }
573
- }
574
- }
575
-
576
- if (ch === "\u0003") {
577
- lineBuffer = "";
578
- cursorPos = 0;
579
- historyIndex = null;
580
- historyDraft = "";
581
- stream.write("^C\r\n");
582
- renderLine();
583
- continue;
584
- }
585
-
586
- if (ch === "\r" || ch === "\n") {
587
- const line = lineBuffer.trim();
588
- lineBuffer = "";
589
- cursorPos = 0;
590
- historyIndex = null;
591
- historyDraft = "";
592
- stream.write("\r\n");
593
-
594
- if (line.length > 0) {
595
- const result = await Promise.resolve(
596
- runCommand(
597
- line,
598
- authUser,
599
- hostname,
600
- "shell",
601
- cwd,
602
- shell,
603
- undefined,
604
- shellEnv,
605
- ),
606
- );
607
-
608
- pushHistory(line);
609
-
610
- if (result.openEditor) {
611
- await startNanoEditor(
612
- result.openEditor.targetPath,
613
- result.openEditor.initialContent,
614
- result.openEditor.tempPath,
615
- );
616
- return;
617
- }
618
-
619
- if (result.openHtop) {
620
- await startHtop();
621
- return;
622
- }
623
-
624
- if (result.sudoChallenge) {
625
- startSudoPrompt(result.sudoChallenge);
626
- return;
627
- }
628
-
629
- if (result.clearScreen) {
630
- stream.write("\u001b[2J\u001b[H");
631
- }
632
-
633
- if (result.stdout) {
634
- stream.write(`${toTtyLines(result.stdout)}\r\n`);
635
- }
636
-
637
- if (result.stderr) {
638
- stream.write(`${toTtyLines(result.stderr)}\r\n`);
639
- }
640
-
641
- if (result.closeSession) {
642
- stream.write("logout\r\n");
643
- stream.exit(result.exitCode ?? 0);
644
- stream.end();
645
- return;
646
- }
647
-
648
- if (result.nextCwd) {
649
- cwd = result.nextCwd;
650
- }
651
-
652
- if (result.switchUser) {
653
- authUser = result.switchUser;
654
- cwd = result.nextCwd ?? userHome(authUser);
655
- shell.users.updateSession(sessionId, authUser, remoteAddress);
656
- lineBuffer = "";
657
- cursorPos = 0;
658
- }
659
-
660
- // WAL: checkpoint handled by auto-flush timer
661
- }
662
-
663
- renderLine();
664
- continue;
665
- }
666
-
667
- if (ch === "\u007f" || ch === "\b") {
668
- if (cursorPos > 0) {
669
- lineBuffer = `${lineBuffer.slice(0, cursorPos - 1)}${lineBuffer.slice(cursorPos)}`;
670
- cursorPos -= 1;
671
- renderLine();
672
- }
673
- continue;
674
- }
675
-
676
- insertText(ch);
677
- }
678
- });
679
-
680
- stream.on("close", () => {
681
- if (nanoSession) {
682
- nanoSession.process.kill("SIGTERM");
683
- nanoSession = null;
684
- }
685
- });
686
- }
687
-
688
- function loadHistory(vfs: VirtualFileSystem, authUser: string): string[] {
689
- const historyPath = `${userHome(authUser)}/.bash_history`;
690
- if (!vfs.exists(historyPath)) {
691
- vfs.writeFile(historyPath, "");
692
- return [];
693
- }
694
-
695
- const raw = vfs.readFile(historyPath);
696
- return raw
697
- .split("\n")
698
- .map((line) => line.trim())
699
- .filter((line) => line.length > 0);
700
- }