typescript-virtual-container 1.4.0 → 1.4.1
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.
- package/builds/self-standalone.js +184 -300
- package/builds/self-standalone.js.map +3 -3
- package/builds/standalone-wo-sftp.js +155 -271
- package/builds/standalone-wo-sftp.js.map +3 -3
- package/builds/standalone.js +149 -265
- package/builds/standalone.js.map +3 -3
- package/dist/VirtualPackageManager/index.d.ts.map +1 -1
- package/dist/VirtualPackageManager/index.js +29 -1
- package/dist/commands/curl.d.ts.map +1 -1
- package/dist/commands/curl.js +2 -1
- package/dist/commands/gzip.d.ts.map +1 -1
- package/dist/commands/gzip.js +6 -0
- package/dist/commands/man.d.ts.map +1 -1
- package/dist/commands/man.js +30 -136
- package/dist/commands/neofetch.d.ts.map +1 -1
- package/dist/commands/neofetch.js +6 -0
- package/dist/commands/wget.d.ts.map +1 -1
- package/dist/commands/wget.js +11 -1
- package/package.json +2 -2
- package/src/VirtualPackageManager/index.ts +29 -1
- package/src/commands/curl.ts +2 -1
- package/src/commands/gzip.ts +7 -0
- package/src/commands/man.ts +38 -143
- package/src/commands/manuals/adduser.txt +11 -0
- package/src/commands/manuals/apt-cache.txt +12 -0
- package/src/commands/manuals/apt.txt +20 -0
- package/src/commands/manuals/awk.txt +13 -0
- package/src/commands/manuals/cat.txt +14 -0
- package/src/commands/manuals/cd.txt +16 -0
- package/src/commands/manuals/chmod.txt +16 -0
- package/src/commands/manuals/clear.txt +10 -0
- package/src/commands/manuals/cp.txt +10 -0
- package/src/commands/manuals/curl.txt +20 -0
- package/src/commands/manuals/date.txt +14 -0
- package/src/commands/manuals/declare.txt +12 -0
- package/src/commands/manuals/deluser.txt +10 -0
- package/src/commands/manuals/df.txt +10 -0
- package/src/commands/manuals/dpkg-query.txt +11 -0
- package/src/commands/manuals/dpkg.txt +14 -0
- package/src/commands/manuals/du.txt +11 -0
- package/src/commands/manuals/echo.txt +11 -0
- package/src/commands/manuals/false.txt +10 -0
- package/src/commands/manuals/find.txt +11 -0
- package/src/commands/manuals/free.txt +12 -0
- package/src/commands/manuals/grep.txt +13 -0
- package/src/commands/manuals/groups.txt +10 -0
- package/src/commands/manuals/gzip.txt +11 -0
- package/src/commands/manuals/head.txt +10 -0
- package/src/commands/manuals/help.txt +11 -0
- package/src/commands/manuals/history.txt +11 -0
- package/src/commands/manuals/hostname.txt +10 -0
- package/src/commands/manuals/id.txt +10 -0
- package/src/commands/manuals/kill.txt +13 -0
- package/src/commands/manuals/ls.txt +20 -0
- package/src/commands/manuals/lsb_release.txt +14 -0
- package/src/commands/manuals/mkdir.txt +10 -0
- package/src/commands/manuals/mv.txt +10 -0
- package/src/commands/manuals/nano.txt +11 -0
- package/src/commands/manuals/neofetch.txt +10 -0
- package/src/commands/manuals/node.txt +13 -0
- package/src/commands/manuals/npm.txt +13 -0
- package/src/commands/manuals/npx.txt +13 -0
- package/src/commands/manuals/passwd.txt +11 -0
- package/src/commands/manuals/ping.txt +10 -0
- package/src/commands/manuals/printf.txt +11 -0
- package/src/commands/manuals/ps.txt +10 -0
- package/src/commands/manuals/pwd.txt +10 -0
- package/src/commands/manuals/python3.txt +13 -0
- package/src/commands/manuals/readlink.txt +10 -0
- package/src/commands/manuals/return.txt +10 -0
- package/src/commands/manuals/rm.txt +10 -0
- package/src/commands/manuals/sed.txt +11 -0
- package/src/commands/manuals/set.txt +11 -0
- package/src/commands/manuals/shift.txt +10 -0
- package/src/commands/manuals/sleep.txt +10 -0
- package/src/commands/manuals/sort.txt +12 -0
- package/src/commands/manuals/source.txt +11 -0
- package/src/commands/manuals/ssh.txt +11 -0
- package/src/commands/manuals/stat.txt +10 -0
- package/src/commands/manuals/su.txt +13 -0
- package/src/commands/manuals/sudo.txt +11 -0
- package/src/commands/manuals/tail.txt +10 -0
- package/src/commands/manuals/tar.txt +19 -0
- package/src/commands/manuals/tee.txt +10 -0
- package/src/commands/manuals/test.txt +11 -0
- package/src/commands/manuals/touch.txt +11 -0
- package/src/commands/manuals/tr.txt +10 -0
- package/src/commands/manuals/trap.txt +10 -0
- package/src/commands/manuals/true.txt +10 -0
- package/src/commands/manuals/type.txt +10 -0
- package/src/commands/manuals/uname.txt +12 -0
- package/src/commands/manuals/uniq.txt +12 -0
- package/src/commands/manuals/unset.txt +10 -0
- package/src/commands/manuals/uptime.txt +11 -0
- package/src/commands/manuals/wc.txt +12 -0
- package/src/commands/manuals/wget.txt +12 -0
- package/src/commands/manuals/which.txt +10 -0
- package/src/commands/manuals/whoami.txt +10 -0
- package/src/commands/manuals/xargs.txt +10 -0
- package/src/commands/neofetch.ts +7 -0
- package/src/commands/wget.ts +12 -1
- package/tests/new-features.test.ts +2 -2
|
@@ -1,14 +1,14 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
import{readFile as
|
|
2
|
+
import{readFile as Ki,unlink as Gi,writeFile as Zi}from"node:fs/promises";import{basename as Ji}from"node:path";import{stdin as it,stdout as rt}from"node:process";import{createInterface as Qi}from"node:readline";var De={name:"adduser",description:"Add a new user",category:"users",params:["<username>"],run:({authUser:e,shell:t,args:r})=>{if(e!=="root")return{stderr:`adduser: permission denied
|
|
3
3
|
`,exitCode:1};let n=r[0];if(!n)return{stderr:`Usage: adduser <username>
|
|
4
4
|
`,exitCode:1};if(t.users.listUsers().includes(n))return{stderr:`adduser: user '${n}' already exists
|
|
5
5
|
`,exitCode:1};let i="",s="new";return{sudoChallenge:{username:n,targetUser:n,commandLine:null,loginShell:!1,prompt:"New password: ",mode:"passwd",onPassword:async(a,l)=>s==="new"?a.length<1?{result:{stderr:`adduser: password cannot be empty
|
|
6
6
|
`,exitCode:1}}:(i=a,s="retype",{result:null,nextPrompt:"Retype new password: "}):a!==i?{result:{stderr:`adduser: passwords do not match \u2014 user not created
|
|
7
7
|
`,exitCode:1}}:(await l.users.addUser(n,i),{result:{stdout:`${[`Adding user '${n}' ...`,`Adding new group '${n}' (1001) ...`,`Adding new user '${n}' (1001) with group '${n}' ...`,`Creating home directory '/home/${n}' ...`,`passwd: password set for '${n}'`,"adduser: done."].join(`
|
|
8
8
|
`)}
|
|
9
|
-
`,exitCode:0}})},exitCode:0}}};function
|
|
9
|
+
`,exitCode:0}})},exitCode:0}}};function _e(e){return Array.isArray(e)?e:[e]}function jt(e,t){if(e===t)return{matched:!0,inlineValue:null};let r=`${t}=`;return e.startsWith(r)?{matched:!0,inlineValue:e.slice(r.length)}:t.length===2&&t.startsWith("-")&&!t.startsWith("--")&&e.startsWith(t)&&e.length>t.length?{matched:!0,inlineValue:e.slice(t.length)}:{matched:!1,inlineValue:null}}function xs(e,t={}){let r=new Set(t.flags??[]),n=new Set(t.flagsWithValue??[]),i=[],s=!1;for(let o=0;o<e.length;o+=1){let a=e[o];if(s){i.push(a);continue}if(a==="--"){s=!0;continue}let l=!1;for(let c of r){let{matched:u}=jt(a,c);if(u){l=!0;break}}if(!l){for(let c of n){let u=jt(a,c);if(u.matched){l=!0,u.inlineValue===null&&o+1<e.length&&(o+=1);break}}l||i.push(a)}}return i}function y(e,t){let r=_e(t);for(let n of e)for(let i of r)if(jt(n,i).matched)return!0;return!1}function st(e,t){let r=_e(t);for(let n=0;n<e.length;n+=1){let i=e[n];for(let s of r){let o=jt(i,s);if(!o.matched)continue;if(o.inlineValue!==null)return o.inlineValue;let a=e[n+1];return a!==void 0&&a!=="--"?a:!0}}}function xt(e,t,r={}){return xs(e,r)[t]}function nt(e,t={}){let r=new Set,n=new Map,i=[],s=new Set(t.flags??[]),o=new Set(t.flagsWithValue??[]),a=!1;for(let l=0;l<e.length;l+=1){let c=e[l];if(a){i.push(c);continue}if(c==="--"){a=!0;continue}if(s.has(c)){r.add(c);continue}if(o.has(c)){let d=e[l+1];d&&!d.startsWith("-")?(n.set(c,d),l+=1):n.set(c,"");continue}let u=Array.from(o).find(d=>c.startsWith(`${d}=`));if(u){n.set(u,c.slice(u.length+1));continue}i.push(c)}return{flags:r,flagsWithValues:n,positionals:i}}var Le={name:"alias",description:"Define or display aliases",category:"shell",params:["[name[=value] ...]"],run:({args:e,env:t})=>{if(!t)return{exitCode:0};if(e.length===0)return{stdout:Object.entries(t.vars).filter(([i])=>i.startsWith("__alias_")).map(([i,s])=>`alias ${i.slice(8)}='${s}'`).join(`
|
|
10
10
|
`)||"",exitCode:0};let r=[];for(let n of e){let i=n.indexOf("=");if(i===-1){let s=t.vars[`__alias_${n}`];if(s)r.push(`alias ${n}='${s}'`);else return{stderr:`alias: ${n}: not found`,exitCode:1}}else{let s=n.slice(0,i),o=n.slice(i+1).replace(/^['"]|['"]$/g,"");t.vars[`__alias_${s}`]=o}}return{stdout:r.join(`
|
|
11
|
-
`)||void 0,exitCode:0}}},
|
|
11
|
+
`)||void 0,exitCode:0}}},ze={name:"unalias",description:"Remove alias definitions",category:"shell",params:["<name...> | -a"],run:({args:e,env:t})=>{if(!t)return{exitCode:0};if(y(e,["-a"])){for(let r of Object.keys(t.vars))r.startsWith("__alias_")&&delete t.vars[r];return{exitCode:0}}for(let r of e)delete t.vars[`__alias_${r}`];return{exitCode:0}}};import*as ut from"node:path";var Ss=["/virtual-env-js/.auth"];function S(e,t){return!t||t.trim()===""?e:t.startsWith("/")?ut.posix.normalize(t):ut.posix.normalize(ut.posix.join(e,t))}function bs(e){let t=e.startsWith("/")?ut.posix.normalize(e):ut.posix.normalize(`/${e}`);return Ss.some(r=>t===r||t.startsWith(`${r}/`))}function L(e,t,r){if(e!=="root"&&bs(t))throw new Error(`${r}: permission denied: ${t}`)}function Ue(e){let r=(e.split("?")[0]?.split("#")[0]??e).split("/").filter(Boolean).pop();return r&&r.length>0?r:"index.html"}function vs(e,t){let r=Array.from({length:e.length+1},()=>Array(t.length+1).fill(0));for(let n=0;n<=e.length;n+=1)r[n][0]=n;for(let n=0;n<=t.length;n+=1)r[0][n]=n;for(let n=1;n<=e.length;n+=1)for(let i=1;i<=t.length;i+=1){let s=e[n-1]===t[i-1]?0:1;r[n][i]=Math.min(r[n-1][i]+1,r[n][i-1]+1,r[n-1][i-1]+s)}return r[e.length][t.length]}function Te(e,t,r){let n=S(t,r);if(e.exists(n))return n;let i=ut.posix.dirname(n),s=ut.posix.basename(n),o=e.list(i),a=o.filter(c=>c.toLowerCase()===s.toLowerCase());if(a.length===1)return ut.posix.join(i,a[0]);let l=o.filter(c=>vs(c.toLowerCase(),s.toLowerCase())<=1);return l.length===1?ut.posix.join(i,l[0]):n}function Be(e,t,r){return t.map(n=>{let i=S(e,n);return r(i).type==="directory"?`${n}/`:n}).join(" ")}function Pt(e){return e.packageManager}var Oe={name:"apt",aliases:["apt-get"],description:"Package manager",category:"package",params:["<install|remove|update|upgrade|search|show|list> [pkg...]"],run:({args:e,shell:t,authUser:r})=>{let n=Pt(t);if(!n)return{stderr:"apt: package manager not initialised",exitCode:1};let i=e[0]?.toLowerCase(),s=e.slice(1),o=y(s,["-q","--quiet","-qq"]),a=y(s,["--purge"]),l=s.filter(u=>!u.startsWith("-"));if(["install","remove","purge","upgrade","update"].includes(i??"")&&r!=="root")return{stderr:`E: Could not open lock file /var/lib/dpkg/lock-frontend - open (13: Permission denied)
|
|
12
12
|
E: Unable to acquire the dpkg frontend lock, are you root?`,exitCode:100};switch(i){case"install":{if(l.length===0)return{stderr:"apt: no packages specified",exitCode:1};let{output:u,exitCode:d}=n.install(l,{quiet:o});return{stdout:u||void 0,exitCode:d}}case"remove":case"purge":{if(l.length===0)return{stderr:"apt: no packages specified",exitCode:1};let{output:u,exitCode:d}=n.remove(l,{purge:i==="purge"||a,quiet:o});return{stdout:u||void 0,exitCode:d}}case"update":return{stdout:["Hit:1 fortune://packages.fortune.local aurora InRelease","Hit:2 fortune://security.fortune.local aurora-security InRelease","Reading package lists... Done","Building dependency tree... Done","Reading state information... Done","All packages are up to date."].join(`
|
|
13
13
|
`),exitCode:0};case"upgrade":return{stdout:["Reading package lists... Done","Building dependency tree... Done","Reading state information... Done","Calculating upgrade... Done","0 upgraded, 0 newly installed, 0 to remove and 0 not upgraded."].join(`
|
|
14
14
|
`),exitCode:0};case"search":{let u=l[0];if(!u)return{stderr:"apt: search requires a term",exitCode:1};let d=n.search(u);return d.length===0?{stdout:`Sorting... Done
|
|
@@ -23,23 +23,23 @@ ${p.map(g=>`${g.name}/${g.section} ${g.version} ${g.architecture} [installed]`).
|
|
|
23
23
|
`)}`,exitCode:0}}return{stdout:`Listing... Done
|
|
24
24
|
${n.listAvailable().map(p=>`${p.name}/${p.section??"misc"} ${p.version} amd64`).join(`
|
|
25
25
|
`)}`,exitCode:0}}default:return{stdout:["Usage: apt [options] command","","Commands:"," install <pkg...> Install packages"," remove <pkg...> Remove packages"," purge <pkg...> Remove packages and config files"," update Refresh package index"," upgrade Upgrade all packages"," search <term> Search in package descriptions"," show <pkg> Show package details"," list [--installed] List packages"].join(`
|
|
26
|
-
`),exitCode:0}}}},
|
|
26
|
+
`),exitCode:0}}}},We={name:"apt-cache",description:"Query the package cache",category:"package",params:["<search|show|policy> [pkg]"],run:({args:e,shell:t})=>{let r=Pt(t);if(!r)return{stderr:"apt-cache: package manager not initialised",exitCode:1};let n=e[0]?.toLowerCase(),i=e[1];switch(n){case"search":return i?{stdout:r.search(i).map(o=>`${o.name} - ${o.shortDesc??o.description}`).join(`
|
|
27
27
|
`)||"(no results)",exitCode:0}:{stderr:"Need a search term",exitCode:1};case"show":{if(!i)return{stderr:"Need a package name",exitCode:1};let s=r.show(i);return s?{stdout:s,exitCode:0}:{stderr:`N: Unable to locate package ${i}`,exitCode:100}}case"policy":{if(!i)return{stderr:"Need a package name",exitCode:1};let s=r.findInRegistry(i);if(!s)return{stderr:`N: Unable to locate package ${i}`,exitCode:100};let o=r.isInstalled(i);return{stdout:[`${i}:`,` Installed: ${o?s.version:"(none)"}`,` Candidate: ${s.version}`," Version table:",` ${s.version} 500`," 500 fortune://packages.fortune.local aurora/main amd64 Packages"].join(`
|
|
28
|
-
`),exitCode:0}}default:return{stderr:`apt-cache: unknown command '${n??""}'`,exitCode:1}}}};var
|
|
29
|
-
`);u[u.length-1]===""&&u.pop();let d=[],m=a.trim();if(!m.startsWith("{")&&!m.includes("{"))d.push({pattern:m,action:"print $0"});else{let
|
|
28
|
+
`),exitCode:0}}default:return{stderr:`apt-cache: unknown command '${n??""}'`,exitCode:1}}}};var je={name:"awk",description:"Pattern scanning and processing language",category:"text",params:["[-F <sep>] '<program>' [file]"],run:({authUser:e,args:t,stdin:r,cwd:n,shell:i})=>{let s=st(t,["-F"])??" ",o=t.filter(x=>!x.startsWith("-")&&x!==s),a=o[0],l=o[1];if(!a)return{stderr:"awk: no program",exitCode:1};let c=r??"";if(l){let x=S(n,l);try{L(e,x,"awk"),c=i.vfs.readFile(x)}catch{return{stderr:`awk: ${l}: No such file or directory`,exitCode:1}}}let u=c.split(`
|
|
29
|
+
`);u[u.length-1]===""&&u.pop();let d=[],m=a.trim();if(!m.startsWith("{")&&!m.includes("{"))d.push({pattern:m,action:"print $0"});else{let x=/([^{]*)\{([^}]*)\}/g,k=x.exec(m);for(;k!==null;)d.push({pattern:k[1].trim(),action:k[2].trim()}),k=x.exec(m);d.length===0&&d.push({pattern:"",action:m.replace(/[{}]/g,"").trim()})}let p=[],w=d.find(x=>x.pattern==="BEGIN"),g=d.find(x=>x.pattern==="END"),f=d.filter(x=>x.pattern!=="BEGIN"&&x.pattern!=="END");function $(x){return s===" "?x.trim().split(/\s+/).filter(Boolean):x.split(s)}function E(x,k,V){let b=$(k),P=b.length,R=v=>{if(v=v.trim(),v==="NR")return String(V);if(v==="NF")return String(P);if(v==="$0")return k;if(v==="$NF")return b[P-1]??"";if(/^\$\d+$/.test(v))return b[parseInt(v.slice(1),10)-1]??"";let N=v.replace(/\bNR\b/g,String(V)).replace(/\bNF\b/g,String(P));if(/^[\d\s+\-*/()]+$/.test(N))try{return String(Function(`"use strict"; return (${N});`)())}catch{}return v.replace(/"/g,"")},C=x.split(";").map(v=>v.trim()).filter(Boolean);for(let v of C)if(v==="print"||v==="print $0")p.push(k);else if(v.startsWith("print ")){let N=v.slice(6).split(/\s*,\s*/);p.push(N.map(R).join(" "))}}function M(x,k,V){if(!x||x==="1")return!0;let b=x.match(/^NR\s*([=!<>]=?|==)\s*(\d+)$/);if(b){let C=b[1],v=parseInt(b[2],10);switch(C){case"==":return V===v;case"!=":return V!==v;case">":return V>v;case">=":return V>=v;case"<":return V<v;case"<=":return V<=v}}let P=x.match(/^NR%(\d+)==(\d+)$/);if(P)return V%parseInt(P[1],10)===parseInt(P[2],10);if(x.startsWith("/")&&x.endsWith("/"))try{return new RegExp(x.slice(1,-1)).test(k)}catch{return!1}let R=x.match(/^\$(\d+)~\/(.*)\/$/);if(R){let v=$(k)[parseInt(R[1],10)-1]??"";try{return new RegExp(R[2]).test(v)}catch{return!1}}return!1}w&&E(w.action,"",0);for(let x=1;x<=u.length;x++){let k=u[x-1];for(let V of f)M(V.pattern,k,x)&&E(V.action,k,x)}return g&&E(g.action,"",u.length+1),{stdout:p.join(`
|
|
30
30
|
`)+(p.length>0?`
|
|
31
|
-
`:""),exitCode:0}}};var
|
|
31
|
+
`:""),exitCode:0}}};var He={name:"base64",description:"Encode/decode base64",category:"text",params:["[-d] [file]"],run:({args:e,stdin:t})=>{let r=y(e,["-d","--decode"]),n=t??"";if(r)try{return{stdout:Buffer.from(n.trim(),"base64").toString("utf8"),exitCode:0}}catch{return{stderr:"base64: invalid input",exitCode:1}}return{stdout:Buffer.from(n).toString("base64"),exitCode:0}}};var qe={name:"cat",description:"Concatenate and print files",category:"files",params:["[-n] [-b] <file...>"],run:({authUser:e,shell:t,cwd:r,args:n,stdin:i})=>{let s=y(n,["-n","--number"]),o=y(n,["-b","--number-nonblank"]),a=n.filter(m=>!m.startsWith("-"));if(a.length===0&&i!==void 0)return{stdout:i,exitCode:0};if(a.length===0)return{stderr:"cat: missing file operand",exitCode:1};let l=[];for(let m of a){let p=Te(t.vfs,r,m);L(e,p,"cat"),l.push(t.vfs.readFile(p))}let c=l.join("");if(!s&&!o)return{stdout:c,exitCode:0};let u=1;return{stdout:c.split(`
|
|
32
32
|
`).map(m=>o&&m.trim()===""?m:`${String(u++).padStart(6)} ${m}`).join(`
|
|
33
|
-
`),exitCode:0}}};var
|
|
34
|
-
`),exitCode:0};let o=s[0];if(!o)return{stderr:"curl: no URL specified",exitCode:1};let a=i.get("-o")??i.get("--output")??null,l=(i.get("-X")??i.get("--request")??"GET").toUpperCase(),c=i.get("-d")??i.get("--data")??null,u=i.get("-H")??i.get("--header")??null,d=y(r,["-s","--silent"]),m=y(r,["-I","--head"]),p=y(r,["-L","--location"]),
|
|
33
|
+
`),exitCode:0}}};var Ke={name:"cd",description:"Change directory",category:"navigation",params:["[path]"],run:({authUser:e,shell:t,cwd:r,args:n,mode:i})=>{let s=S(r,n[0]??"/virtual-env-js");return L(e,s,"cd"),t.vfs.stat(s).type!=="directory"?{stderr:`cd: not a directory: ${s}`,exitCode:1}:{nextCwd:s,exitCode:0}}};function Cs(e,t){let r=/^([ugoa]*)([+\-=])([rwx]*)$/,n=t.split(","),i=e;for(let s of n){let o=s.trim().match(r);if(!o)return null;let[,a="a",l,c=""]=o,u=a===""||a==="a"?["u","g","o"]:a.split(""),d={u:{r:256,w:128,x:64},g:{r:32,w:16,x:8},o:{r:4,w:2,x:1}};for(let m of u)for(let p of c.split("")){let w=d[m]?.[p];if(w!==void 0){if(l==="+")i|=w;else if(l==="-")i&=~w;else if(l==="="){let g=Object.values(d[m]??{}).reduce((f,$)=>f|$,0);i=i&~g|w}}}}return i}var Ge={name:"chmod",description:"Change file permissions",category:"files",params:["<mode> <file>"],run:({authUser:e,shell:t,cwd:r,args:n})=>{let[i,s]=n;if(!i||!s)return{stderr:"chmod: missing operand",exitCode:1};let o=S(r,s);try{if(L(e,o,"chmod"),!t.vfs.exists(o))return{stderr:`chmod: ${s}: No such file or directory`,exitCode:1};let a,l=parseInt(i,8);if(!Number.isNaN(l)&&/^[0-7]+$/.test(i))a=l;else{let c=t.vfs.stat(o).mode,u=Cs(c,i);if(u===null)return{stderr:`chmod: invalid mode: ${i}`,exitCode:1};a=u}return t.vfs.chmod(o,a),{exitCode:0}}catch(a){return{stderr:`chmod: ${a instanceof Error?a.message:String(a)}`,exitCode:1}}}};var Ze={name:"clear",description:"Clear the terminal screen",category:"shell",params:[],run:()=>({clearScreen:!0,stdout:"",exitCode:0})};var Je={name:"cp",description:"Copy files or directories",category:"files",params:["[-r] <source> <dest>"],run:({authUser:e,shell:t,cwd:r,args:n})=>{let i=y(n,["-r","-R","--recursive"]),s=n.filter(u=>!u.startsWith("-")),[o,a]=s;if(!o||!a)return{stderr:"cp: missing operand",exitCode:1};let l=S(r,o),c=S(r,a);try{if(L(e,l,"cp"),L(e,c,"cp"),!t.vfs.exists(l))return{stderr:`cp: ${o}: No such file or directory`,exitCode:1};if(t.vfs.stat(l).type==="directory"){if(!i)return{stderr:`cp: ${o}: is a directory (use -r)`,exitCode:1};let d=(p,w)=>{t.vfs.mkdir(w,493);for(let g of t.vfs.list(p)){let f=`${p}/${g}`,$=`${w}/${g}`;if(t.vfs.stat(f).type==="directory")d(f,$);else{let M=t.vfs.readFileRaw(f);t.writeFileAsUser(e,$,M)}}},m=t.vfs.exists(c)&&t.vfs.stat(c).type==="directory"?`${c}/${o.split("/").pop()}`:c;d(l,m)}else{let d=t.vfs.exists(c)&&t.vfs.stat(c).type==="directory"?`${c}/${o.split("/").pop()}`:c,m=t.vfs.readFileRaw(l);t.writeFileAsUser(e,d,m)}return{exitCode:0}}catch(u){return{stderr:`cp: ${u instanceof Error?u.message:String(u)}`,exitCode:1}}}};var Qe={name:"curl",description:"Transfer data from or to a server (pure fetch)",category:"network",params:["[options] <url>"],run:async({authUser:e,cwd:t,args:r,shell:n})=>{let{flagsWithValues:i,positionals:s}=nt(r,{flagsWithValue:["-o","--output","-X","--request","-d","--data","-H","--header","-u","--user"]});if(y(r,["--help","-h"]))return{stdout:["Usage: curl [options] <url>"," -o, --output <file> Write to file"," -X, --request <method> HTTP method"," -d, --data <data> POST data"," -H, --header <hdr> Extra header"," -s, --silent Silent mode"," -I, --head Fetch headers only"," -L, --location Follow redirects"," -v, --verbose Verbose"].join(`
|
|
34
|
+
`),exitCode:0};let o=s[0];if(!o)return{stderr:"curl: no URL specified",exitCode:1};let a=i.get("-o")??i.get("--output")??null,l=(i.get("-X")??i.get("--request")??"GET").toUpperCase(),c=i.get("-d")??i.get("--data")??null,u=i.get("-H")??i.get("--header")??null,d=y(r,["-s","--silent"]),m=y(r,["-I","--head"]),p=y(r,["-L","--location"]),w=y(r,["-v","--verbose"]),g={"User-Agent":"curl/7.88.1"};if(u){let k=u.indexOf(":");k!==-1&&(g[u.slice(0,k).trim()]=u.slice(k+1).trim())}let f=c&&l==="GET"?"POST":l,$={method:f,headers:g,redirect:p?"follow":"manual"};c&&(g["Content-Type"]??="application/x-www-form-urlencoded",$.body=c);let E=[];w&&(E.push(`* Trying ${o}...`,"* Connected"),E.push(`> ${f} / HTTP/1.1`,`> Host: ${new URL(o).host}`));let M;try{let k=o.startsWith("http://")||o.startsWith("https://")?o:`http://${o}`;M=await fetch(k,$)}catch(k){return{stderr:`curl: (6) Could not resolve host: ${k instanceof Error?k.message:String(k)}`,exitCode:6}}if(w&&E.push(`< HTTP/1.1 ${M.status} ${M.statusText}`),m){let k=[`HTTP/1.1 ${M.status} ${M.statusText}`];for(let[V,b]of M.headers.entries())k.push(`${V}: ${b}`);return{stdout:`${k.join(`\r
|
|
35
35
|
`)}\r
|
|
36
|
-
`,exitCode:0}}let
|
|
37
|
-
100 ${
|
|
38
|
-
`)||void 0,exitCode:M.ok?0:22}}return{stdout:
|
|
36
|
+
`,exitCode:0}}let x;try{x=await M.text()}catch{return{stderr:"curl: failed to read response body",exitCode:1}}if(a){let k=S(t,a);return L(e,k,"curl"),n.writeFileAsUser(e,k,x),d||E.push(` % Total % Received
|
|
37
|
+
100 ${x.length} 100 ${x.length}`),{stderr:E.join(`
|
|
38
|
+
`)||void 0,exitCode:M.ok?0:22}}return{stdout:x,stderr:E.length>0?E.join(`
|
|
39
39
|
`):void 0,exitCode:M.ok?0:22}}};var Ye={name:"cut",description:"Remove sections from lines",category:"text",params:["-d <delim> -f <fields> [file]"],run:({args:e,stdin:t})=>{let r=st(e,["-d"])??" ",i=(st(e,["-f"])??"1").split(",").map(a=>{let[l,c]=a.split("-").map(Number);return c!==void 0?{from:(l??1)-1,to:c-1}:{from:(l??1)-1,to:(l??1)-1}});return{stdout:(t??"").split(`
|
|
40
40
|
`).map(a=>{let l=a.split(r),c=[];for(let u of i)for(let d=u.from;d<=Math.min(u.to,l.length-1);d++)c.push(l[d]??"");return c.join(r)}).join(`
|
|
41
|
-
`),exitCode:0}}};var
|
|
42
|
-
`),exitCode:0};let o=e.filter(a=>!a.startsWith("-"));for(let a of o){let l=a.indexOf("=");if(l===-1)a in t.vars||(t.vars[a]="");else{let c=a.slice(0,l),u=a.slice(l+1);if(r){let d=parseInt(u,10);u=Number.isNaN(d)?"0":String(d)}t.vars[c]=u}}return{exitCode:0}}};var
|
|
41
|
+
`),exitCode:0}}};var Xe={name:"date",description:"Print current date and time",category:"system",params:["[+format]"],run:({args:e})=>{let t=new Date,r=e[0];return r?.startsWith("+")?{stdout:r.slice(1).replace("%Y",String(t.getFullYear())).replace("%m",String(t.getMonth()+1).padStart(2,"0")).replace("%d",String(t.getDate()).padStart(2,"0")).replace("%H",String(t.getHours()).padStart(2,"0")).replace("%M",String(t.getMinutes()).padStart(2,"0")).replace("%S",String(t.getSeconds()).padStart(2,"0")).replace("%s",String(Math.floor(t.getTime()/1e3))),exitCode:0}:{stdout:t.toString(),exitCode:0}}};var tr={name:"declare",aliases:["local","typeset"],description:"Declare variables and give them attributes",category:"shell",params:["[-i] [-r] [-x] [-a] [name[=value]...]"],run:({args:e,env:t})=>{if(!t)return{exitCode:0};let r=y(e,["-i"]),n=y(e,["-r"]),i=y(e,["-x"]);if(e.filter(a=>!a.startsWith("-")).length===0)return{stdout:Object.entries(t.vars).map(([l,c])=>`declare -- ${l}="${c}"`).join(`
|
|
42
|
+
`),exitCode:0};let o=e.filter(a=>!a.startsWith("-"));for(let a of o){let l=a.indexOf("=");if(l===-1)a in t.vars||(t.vars[a]="");else{let c=a.slice(0,l),u=a.slice(l+1);if(r){let d=parseInt(u,10);u=Number.isNaN(d)?"0":String(d)}t.vars[c]=u}}return{exitCode:0}}};var er={name:"deluser",description:"Delete a user",category:"users",params:["[-f] <username>"],run:async({authUser:e,args:t,shell:r})=>{if(e!=="root")return{stderr:`deluser: permission denied
|
|
43
43
|
`,exitCode:1};let n=t.includes("-f")||t.includes("--force")||t.includes("-y"),i=t.find(o=>!o.startsWith("-"));if(!i)return{stderr:`Usage: deluser [-f] <username>
|
|
44
44
|
`,exitCode:1};if(!r.users.listUsers().includes(i))return{stderr:`deluser: user '${i}' does not exist
|
|
45
45
|
`,exitCode:1};if(i==="root")return{stderr:`deluser: cannot remove the root account
|
|
@@ -49,346 +49,226 @@ deluser: done.
|
|
|
49
49
|
`,exitCode:1}}:(await a.users.deleteUser(i),{result:{stdout:`Removing user '${i}' ...
|
|
50
50
|
deluser: done.
|
|
51
51
|
`,exitCode:0}});return{sudoChallenge:{username:i,targetUser:i,commandLine:null,loginShell:!1,prompt:`Warning: deleting user '${i}'.
|
|
52
|
-
Type the username to confirm: `,mode:"confirm",onPassword:s},exitCode:0}}};var
|
|
53
|
-
${a}`,exitCode:0}}};var
|
|
52
|
+
Type the username to confirm: `,mode:"confirm",onPassword:s},exitCode:0}}};var rr={name:"df",description:"Report filesystem disk space usage",category:"system",params:["[-h]"],run:({shell:e})=>{let r=(e.vfs.getUsageBytes()/1024).toFixed(0),n="1048576",i=String(Number(n)-Number(r)),s=Math.round(Number(r)/Number(n)*100),o="Filesystem 1K-blocks Used Available Use% Mounted on",a=`virtual-fs ${n.padStart(9)} ${r.padStart(7)} ${i.padStart(9)} ${s}% /`;return{stdout:`${o}
|
|
53
|
+
${a}`,exitCode:0}}};var nr={name:"diff",description:"Compare files line by line",category:"text",params:["<file1> <file2>"],run:({shell:e,cwd:t,args:r})=>{let[n,i]=r;if(!n||!i)return{stderr:"diff: missing operand",exitCode:1};let s=S(t,n),o=S(t,i),a,l;try{a=e.vfs.readFile(s).split(`
|
|
54
54
|
`)}catch{return{stderr:`diff: ${n}: No such file or directory`,exitCode:2}}try{l=e.vfs.readFile(o).split(`
|
|
55
55
|
`)}catch{return{stderr:`diff: ${i}: No such file or directory`,exitCode:2}}let c=[],u=Math.max(a.length,l.length);for(let d=0;d<u;d++){let m=a[d],p=l[d];m!==p&&(m!==void 0&&c.push(`< ${m}`),p!==void 0&&c.push(`> ${p}`))}return{stdout:c.join(`
|
|
56
|
-
`),exitCode:c.length>0?1:0}}};var
|
|
57
|
-
`),exitCode:0};let d=["Desired=Unknown/Install/Remove/Purge/Hold","|Status=Not/Inst/Conf-files/Unpacked/halF-conf/Half-inst/trig-aWait/Trig-pend","|/ Err?=(none)/Reinst-required (Status,Err: uppercase=bad)","||/ Name Version Architecture Description","+++-==============-===============-============-========================================"],m=u.map(p=>{let
|
|
56
|
+
`),exitCode:c.length>0?1:0}}};var sr={name:"dpkg",description:"Debian package manager low-level tool",category:"package",params:["[-l] [-s pkg] [-L pkg] [-i pkg] [--remove pkg]"],run:({args:e,authUser:t,shell:r})=>{let n=Pt(r);if(!n)return{stderr:"dpkg: package manager not initialised",exitCode:1};let i=y(e,["-l","--list"]),s=y(e,["-s","--status"]),o=y(e,["-L","--listfiles"]),a=y(e,["-r","--remove"]),l=y(e,["-P","--purge"]),{positionals:c}=nt(e,{flags:["-l","--list","-s","--status","-L","--listfiles","-r","--remove","-P","--purge"]});if(i){let u=n.listInstalled();if(u.length===0)return{stdout:["Desired=Unknown/Install/Remove/Purge/Hold","|Status=Not/Inst/Conf-files/Unpacked/halF-conf/Half-inst/trig-aWait/Trig-pend","|/ Err?=(none)/Reinst-required (Status,Err: uppercase=bad)","||/ Name Version Architecture Description","+++-==============-===============-============-========================================","(no packages installed)"].join(`
|
|
57
|
+
`),exitCode:0};let d=["Desired=Unknown/Install/Remove/Purge/Hold","|Status=Not/Inst/Conf-files/Unpacked/halF-conf/Half-inst/trig-aWait/Trig-pend","|/ Err?=(none)/Reinst-required (Status,Err: uppercase=bad)","||/ Name Version Architecture Description","+++-==============-===============-============-========================================"],m=u.map(p=>{let w=p.name.padEnd(14).slice(0,14),g=p.version.padEnd(15).slice(0,15),f=p.architecture.padEnd(12).slice(0,12),$=(p.description||"").slice(0,40);return`ii ${w} ${g} ${f} ${$}`});return{stdout:[...d,...m].join(`
|
|
58
58
|
`),exitCode:0}}if(s){let u=c[0];if(!u)return{stderr:"dpkg: -s needs a package name",exitCode:1};let d=n.show(u);return d?{stdout:d,exitCode:0}:{stderr:`dpkg-query: package '${u}' is not installed and no information is available`,exitCode:1}}if(o){let u=c[0];if(!u)return{stderr:"dpkg: -L needs a package name",exitCode:1};let d=n.listInstalled().find(m=>m.name===u);return d?d.files.length===0?{stdout:"/.keep",exitCode:0}:{stdout:d.files.join(`
|
|
59
59
|
`),exitCode:0}:{stderr:`dpkg-query: package '${u}' is not installed`,exitCode:1}}if(a||l){if(t!=="root")return{stderr:"dpkg: error: requested operation requires superuser privilege",exitCode:2};if(c.length===0)return{stderr:"dpkg: error: need an action option",exitCode:2};let{output:u,exitCode:d}=n.remove(c,{purge:l});return{stdout:u||void 0,exitCode:d}}return{stdout:["Usage: dpkg [<option>...] <command>","","Commands:"," -l, --list List packages matching given pattern"," -s, --status <pkg>... Report status of specified package"," -L, --listfiles <pkg>... List files owned by package"," -r, --remove <pkg>... Remove <pkg> but leave its configuration"," -P, --purge <pkg>... Remove <pkg> and its configuration"].join(`
|
|
60
|
-
`),exitCode:0}}},
|
|
60
|
+
`),exitCode:0}}},ir={name:"dpkg-query",description:"Show information about installed packages",category:"package",params:["-W [pkg] | -l [pattern]"],run:({args:e,shell:t})=>{let r=Pt(t);if(!r)return{stderr:"dpkg-query: package manager not initialised",exitCode:1};let n=y(e,["-l"]),i=y(e,["-W","--show"]),{positionals:s}=nt(e,{flags:["-l","-W","--show"]});if(n||i){let o=r.listInstalled(),a=s[0],l=a?o.filter(u=>u.name.includes(a)):o;return i?{stdout:l.map(u=>`${u.name} ${u.version}`).join(`
|
|
61
61
|
`),exitCode:0}:{stdout:l.map(u=>{let d=u.name.padEnd(14).slice(0,14),m=u.version.padEnd(15).slice(0,15);return`ii ${d} ${m} amd64 ${(u.description||"").slice(0,40)}`}).join(`
|
|
62
|
-
`)||"(no packages match)",exitCode:0}}return{stderr:"dpkg-query: need a flag (-l, -W)",exitCode:1}}};var
|
|
63
|
-
`),exitCode:0}}};function
|
|
64
|
-
`).replace(/\\t/g," ").replace(/\\r/g,"\r").replace(/\\\\/g,"\\").replace(/\\a/g,"\x07").replace(/\\b/g,"\b").replace(/\\f/g,"\f").replace(/\\v/g,"\v").replace(/\\0(\d{1,3})/g,(t,r)=>String.fromCharCode(parseInt(r,8)))}var
|
|
65
|
-
`,exitCode:0}}};var
|
|
66
|
-
`),exitCode:0}}};var
|
|
67
|
-
`),exitCode:0};for(let r of e)if(r.includes("=")){let n=r.indexOf("="),i=r.slice(0,n),s=r.slice(n+1);t.vars[i]=s}return{exitCode:0}}};var
|
|
68
|
-
`),exitCode:0}}};import*as qt from"node:os";var
|
|
69
|
-
`),exitCode:0}}};var
|
|
70
|
-
`),b=[];for(let P=0;P<
|
|
71
|
-
`,exitCode:
|
|
62
|
+
`)||"(no packages match)",exitCode:0}}return{stderr:"dpkg-query: need a flag (-l, -W)",exitCode:1}}};var or={name:"du",description:"Estimate file space usage",category:"system",params:["[-h] [-s] [path]"],run:({shell:e,cwd:t,args:r})=>{let n=y(r,["-h"]),i=y(r,["-s"]),s=r.find(u=>!u.startsWith("-"))??".",o=S(t,s),a=u=>n?`${(u/1024).toFixed(1)}K`:String(Math.ceil(u/1024));if(!e.vfs.exists(o))return{stderr:`du: ${s}: No such file or directory`,exitCode:1};if(i||e.vfs.stat(o).type==="file")return{stdout:`${a(e.vfs.getUsageBytes(o))} ${s}`,exitCode:0};let l=[],c=(u,d)=>{let m=0;for(let p of e.vfs.list(u)){let w=`${u}/${p}`,g=`${d}/${p}`,f=e.vfs.stat(w);f.type==="directory"?m+=c(w,g):(m+=f.size,i||l.push(`${a(f.size)} ${g}`))}return l.push(`${a(m)} ${d}`),m};return c(o,s),{stdout:l.join(`
|
|
63
|
+
`),exitCode:0}}};function $s(e,t){let r=e.replace(/\b([A-Za-z_][A-Za-z0-9_]*)\b/g,(n,i)=>{let s=t[i];return s!==void 0&&s!==""?s:"0"});if(!/^[\d\s+\-*/%()^!&|<>=,. ]+$/.test(r))return NaN;try{let n=Function(`"use strict"; return (${r.replace(/\*\*/g,"**")});`)();return typeof n=="number"?Math.trunc(n):NaN}catch{return NaN}}function Ps(e,t){let r=[],n=0;for(;n<e.length;){let i=e.indexOf("'",n);if(i===-1){r.push(t(e.slice(n)));break}r.push(t(e.slice(n,i)));let s=e.indexOf("'",i+1);if(s===-1){r.push(e.slice(i));break}r.push(e.slice(i,s+1)),n=s+1}return r.join("")}function pe(e,t,r=0,n){let i=n??t.HOME??"/home/user";return Ps(e,s=>{let o=s;return o=o.replace(/(^|[\s:])~(\/|$)/g,(a,l,c)=>`${l}${i}${c}`),o=o.replace(/\$\?/g,String(r)),o=o.replace(/\$\$/g,"1"),o=o.replace(/\$#/g,"0"),o=o.replace(/\$\(\(([^)]+(?:\([^)]*\)[^)]*)*)\)\)/g,(a,l)=>{let c=$s(l,t);return Number.isNaN(c)?"0":String(c)}),o=o.replace(/\$\{#([A-Za-z_][A-Za-z0-9_]*)\}/g,(a,l)=>String((t[l]??"").length)),o=o.replace(/\$\{([A-Za-z_][A-Za-z0-9_]*):-([^}]*)\}/g,(a,l,c)=>t[l]!==void 0&&t[l]!==""?t[l]:c),o=o.replace(/\$\{([A-Za-z_][A-Za-z0-9_]*):=([^}]*)\}/g,(a,l,c)=>((t[l]===void 0||t[l]==="")&&(t[l]=c),t[l])),o=o.replace(/\$\{([A-Za-z_][A-Za-z0-9_]*):\+([^}]*)\}/g,(a,l,c)=>t[l]!==void 0&&t[l]!==""?c:""),o=o.replace(/\$\{([A-Za-z_][A-Za-z0-9_]*)\}/g,(a,l)=>t[l]??""),o=o.replace(/\$([A-Za-z_][A-Za-z0-9_]*)/g,(a,l)=>t[l]??""),o})}async function Ht(e,t,r,n){if(e.includes("$(")){let i="",s=!1,o=0;for(;o<e.length;){let a=e[o];if(a==="'"&&!s){s=!0,i+=a,o++;continue}if(a==="'"&&s){s=!1,i+=a,o++;continue}if(!s&&a==="$"&&e[o+1]==="("){if(e[o+2]==="("){i+=a,o++;continue}let l=0,c=o+1;for(;c<e.length;){if(e[c]==="(")l++;else if(e[c]===")"&&(l--,l===0))break;c++}let u=e.slice(o+2,c).trim(),d=(await n(u)).replace(/\n$/,"");i+=d,o=c+1;continue}i+=a,o++}e=i}return pe(e,t,r)}function ks(e){return e.replace(/\\n/g,`
|
|
64
|
+
`).replace(/\\t/g," ").replace(/\\r/g,"\r").replace(/\\\\/g,"\\").replace(/\\a/g,"\x07").replace(/\\b/g,"\b").replace(/\\f/g,"\f").replace(/\\v/g,"\v").replace(/\\0(\d{1,3})/g,(t,r)=>String.fromCharCode(parseInt(r,8)))}var ar={name:"echo",description:"Display text",category:"shell",params:["[-n] [-e] [text...]"],run:({args:e,stdin:t,env:r})=>{let{flags:n,positionals:i}=nt(e,{flags:["-n","-e","-E"]}),s=n.has("-n"),o=n.has("-e"),a=i.length>0?i.join(" "):t??"",l=pe(a,r?.vars??{},r?.lastExitCode??0),c=o?ks(l):l;return{stdout:s?c:`${c}
|
|
65
|
+
`,exitCode:0}}};var lr={name:"env",description:"Print environment variables",category:"shell",params:[],run:({env:e,authUser:t})=>{let r={...e.vars,USER:t,HOME:`/home/${t}`};return{stdout:Object.entries(r).map(([n,i])=>`${n}=${i}`).join(`
|
|
66
|
+
`),exitCode:0}}};var cr={name:"exit",aliases:["bye"],description:"Exit the shell session",category:"shell",params:["[code]"],run:({args:e})=>({closeSession:!0,exitCode:parseInt(e[0]??"0",10)||0})};var ur={name:"export",description:"Set shell environment variable",category:"shell",params:["[VAR=value]"],run:({args:e,env:t})=>{if(e.length===0)return{stdout:Object.entries(t.vars).map(([n,i])=>`declare -x ${n}="${i}"`).join(`
|
|
67
|
+
`),exitCode:0};for(let r of e)if(r.includes("=")){let n=r.indexOf("="),i=r.slice(0,n),s=r.slice(n+1);t.vars[i]=s}return{exitCode:0}}};var dr={name:"find",description:"Search for files",category:"files",params:["[path] [-name <pattern>] [-type f|d]"],run:({authUser:e,shell:t,cwd:r,args:n})=>{let i=st(n,["-name"]),s=st(n,["-type"]),a=n.filter(m=>!m.startsWith("-")&&m!==i&&m!==s)[0]??".",l=S(r,a);try{if(L(e,l,"find"),!t.vfs.exists(l))return{stderr:`find: ${a}: No such file or directory`,exitCode:1}}catch(m){return{stderr:`find: ${m instanceof Error?m.message:String(m)}`,exitCode:1}}let c=i?new RegExp(`^${i.replace(/\./g,"\\.").replace(/\*/g,".*").replace(/\?/g,".")}$`):null,u=[],d=(m,p)=>{let w=t.vfs.stat(m),g=!s||s==="f"&&w.type==="file"||s==="d"&&w.type==="directory",f=!c||c.test(m.split("/").pop()??"");if(g&&f&&u.push(p),w.type==="directory")for(let $ of t.vfs.list(m)){let E=`${m}/${$}`,M=`${p}/${$}`;d(E,M)}};return d(l,a),{stdout:u.join(`
|
|
68
|
+
`),exitCode:0}}};import*as qt from"node:os";var mr={name:"free",description:"Display amount of free and used memory",category:"system",params:["[-h] [-m] [-g]"],run:({args:e})=>{let t=y(e,["-h","--human"]),r=y(e,["-m"]),n=y(e,["-g"]),i=qt.totalmem(),s=qt.freemem(),o=i-s,a=Math.floor(i*.02),l=Math.floor(i*.05),c=Math.floor(s*.95),u=Math.floor(i*.5),d=g=>t?g>=1024*1024*1024?`${(g/(1024*1024*1024)).toFixed(1)}G`:g>=1024*1024?`${(g/(1024*1024)).toFixed(1)}M`:`${(g/1024).toFixed(1)}K`:String(Math.floor(n?g/(1024*1024*1024):r?g/(1024*1024):g/1024)),m=" total used free shared buff/cache available",p=`Mem: ${d(i).padStart(12)} ${d(o).padStart(11)} ${d(s).padStart(11)} ${d(a).padStart(11)} ${d(l).padStart(11)} ${d(c).padStart(11)}`,w=`Swap: ${d(u).padStart(12)} ${d(0).padStart(11)} ${d(u).padStart(11)}`;return{stdout:[m,p,w].join(`
|
|
69
|
+
`),exitCode:0}}};var pr={name:"grep",description:"Search text patterns",category:"text",params:["[-i] [-v] [-n] [-r] <pattern> [file...]"],run:({authUser:e,shell:t,cwd:r,args:n,stdin:i})=>{let{flags:s,positionals:o}=nt(n,{flags:["-i","-v","-n","-r","-c","-l","-L","-q","--quiet","--silent"]}),a=s.has("-i"),l=s.has("-v"),c=s.has("-n"),u=s.has("-r"),d=s.has("-c"),m=s.has("-l"),p=s.has("-q")||s.has("--quiet")||s.has("--silent"),w=o[0],g=o.slice(1);if(!w)return{stderr:"grep: no pattern specified",exitCode:1};let f;try{let x=a?"mi":"m";f=new RegExp(w,x)}catch{return{stderr:`grep: invalid regex: ${w}`,exitCode:1}}let $=(x,k="")=>{let V=x.split(`
|
|
70
|
+
`),b=[];for(let P=0;P<V.length;P++){let R=V[P]??"",C=f.test(R);if(l?!C:C){let N=c?`${P+1}:`:"";b.push(`${k}${N}${R}`)}}return b},E=x=>{if(!t.vfs.exists(x))return[];if(t.vfs.stat(x).type==="file")return[x];if(!u)return[];let V=[],b=P=>{for(let R of t.vfs.list(P)){let C=`${P}/${R}`;t.vfs.stat(C).type==="file"?V.push(C):b(C)}};return b(x),V},M=[];if(g.length===0){if(!i)return{stdout:"",exitCode:1};let x=$(i);if(d)return{stdout:`${x.length}
|
|
71
|
+
`,exitCode:x.length>0?0:1};if(p)return{exitCode:x.length>0?0:1};M.push(...x)}else{let x=g.flatMap(k=>{let V=S(r,k);return E(V).map(b=>({file:k,path:b}))});for(let{file:k,path:V}of x)try{L(e,V,"grep");let b=t.vfs.readFile(V),P=x.length>1?`${k}:`:"",R=$(b,P);d?M.push(x.length>1?`${k}:${R.length}`:String(R.length)):m?R.length>0&&M.push(k):M.push(...R)}catch{return{stderr:`grep: ${k}: No such file or directory`,exitCode:1}}}return{stdout:M.length>0?`${M.join(`
|
|
72
72
|
`)}
|
|
73
|
-
`:"",exitCode:M.length>0?0:1}}};var
|
|
74
|
-
|
|
73
|
+
`:"",exitCode:M.length>0?0:1}}};var fr={name:"groups",description:"Print group memberships",category:"system",params:["[user]"],run:({authUser:e,shell:t,args:r})=>{let n=r[0]??e;return{stdout:t.users.isSudoer(n)?`${n} sudo root`:n,exitCode:0}}};var hr={name:"gzip",description:"Compress files",category:"archive",params:["[-k] [-d] <file>"],run:({shell:e,cwd:t,args:r})=>{if(!e.packageManager.isInstalled("gzip"))return{stderr:`bash: gzip: command not found
|
|
74
|
+
Hint: install it with: apt install gzip
|
|
75
|
+
`,exitCode:127};let n=r.includes("-k")||r.includes("--keep"),i=r.includes("-d"),s=r.find(c=>!c.startsWith("-"));if(!s)return{stderr:`gzip: no file specified
|
|
76
|
+
`,exitCode:1};let o=S(t,s);if(i){if(!s.endsWith(".gz"))return{stderr:`gzip: ${s}: unknown suffix -- ignored
|
|
75
77
|
`,exitCode:1};if(!e.vfs.exists(o))return{stderr:`gzip: ${s}: No such file or directory
|
|
76
78
|
`,exitCode:1};let c=e.vfs.readFile(o),u=o.slice(0,-3);return e.vfs.writeFile(u,c),n||e.vfs.remove(o),{exitCode:0}}if(!e.vfs.exists(o))return{stderr:`gzip: ${s}: No such file or directory
|
|
77
79
|
`,exitCode:1};if(s.endsWith(".gz"))return{stderr:`gzip: ${s}: already has .gz suffix -- unchanged
|
|
78
|
-
`,exitCode:1};let a=e.vfs.readFileRaw(o),l=`${o}.gz`;return e.vfs.writeFile(l,a,{compress:!0}),n||e.vfs.remove(o),{exitCode:0}}},
|
|
79
|
-
`,exitCode:1};let s=
|
|
80
|
+
`,exitCode:1};let a=e.vfs.readFileRaw(o),l=`${o}.gz`;return e.vfs.writeFile(l,a,{compress:!0}),n||e.vfs.remove(o),{exitCode:0}}},gr={name:"gunzip",description:"Decompress files",category:"archive",aliases:["zcat"],params:["[-k] <file>"],run:({shell:e,cwd:t,args:r})=>{let n=r.includes("-k")||r.includes("--keep"),i=r.find(l=>!l.startsWith("-"));if(!i)return{stderr:`gunzip: no file specified
|
|
81
|
+
`,exitCode:1};let s=S(t,i);if(!e.vfs.exists(s))return{stderr:`gunzip: ${i}: No such file or directory
|
|
80
82
|
`,exitCode:1};if(!i.endsWith(".gz"))return{stderr:`gunzip: ${i}: unknown suffix -- ignored
|
|
81
|
-
`,exitCode:1};let o=e.vfs.readFile(s),a=s.slice(0,-3);return e.vfs.writeFile(a,o),n||e.vfs.remove(s),{exitCode:0}}};var
|
|
83
|
+
`,exitCode:1};let o=e.vfs.readFile(s),a=s.slice(0,-3);return e.vfs.writeFile(a,o),n||e.vfs.remove(s),{exitCode:0}}};var yr={name:"head",description:"Output first lines",category:"text",params:["[-n <lines>] [file...]"],run:({authUser:e,shell:t,cwd:r,args:n,stdin:i})=>{let s=st(n,["-n"]),o=n.find(d=>/^-\d+$/.test(d)),a=typeof s=="string"?parseInt(s,10):o?parseInt(o.slice(1),10):10,l=n.filter(d=>!d.startsWith("-")&&d!==s&&d!==String(a)),c=d=>{let m=d.split(`
|
|
82
84
|
`),p=m.slice(0,a);return p.join(`
|
|
83
85
|
`)+(d.endsWith(`
|
|
84
86
|
`)&&p.length===m.slice(0,a).length?`
|
|
85
|
-
`:"")};if(l.length===0)return{stdout:c(i??""),exitCode:0};let u=[];for(let d of l){let m=
|
|
86
|
-
`),exitCode:0}}};var
|
|
87
|
-
`)}function
|
|
88
|
-
`)}function
|
|
87
|
+
`:"")};if(l.length===0)return{stdout:c(i??""),exitCode:0};let u=[];for(let d of l){let m=S(r,d);try{L(e,m,"head"),u.push(c(t.vfs.readFile(m)))}catch{return{stderr:`head: ${d}: No such file or directory`,exitCode:1}}}return{stdout:u.join(`
|
|
88
|
+
`),exitCode:0}}};var wr=["navigation","files","text","archive","system","package","network","shell","users","misc"],Sr={navigation:"Navigation",files:"Files & Filesystem",text:"Text Processing",archive:"Archive & Compression",system:"System",package:"Package Management",network:"Network",shell:"Shell & Scripting",users:"Users & Permissions",misc:"Miscellaneous"},br="\x1B[1m",dt="\x1B[0m",Ms="\x1B[36m",Fs="\x1B[33m",Vt="\x1B[2m",As="\x1B[32m";function xr(e,t){return e.length>=t?e:e+" ".repeat(t-e.length)}function Es(e){let t=e.aliases?.length?` ${Vt}(${e.aliases.join(", ")})${dt}`:"";return` ${Ms}${xr(e.name,16)}${dt}${t}${xr("",(e.aliases?.length,0))} ${e.description??""}`}function Ns(e){let t={};for(let s of e){let o=s.category??"misc";t[o]||(t[o]=[]),t[o].push(s)}let r=[`${br}Available commands${dt}`,`${Vt}Type 'help <command>' for detailed usage.${dt}`,""],n=[...wr.filter(s=>t[s]),...Object.keys(t).filter(s=>!wr.includes(s)).sort()];for(let s of n){let o=t[s];if(!o?.length)continue;r.push(`${Fs}${Sr[s]??s}${dt}`);let a=[...o].sort((l,c)=>l.name.localeCompare(c.name));for(let l of a)r.push(Es(l));r.push("")}let i=e.length;return r.push(`${Vt}${i} commands available.${dt}`),r.join(`
|
|
89
|
+
`)}function Is(e){let t=[];if(t.push(`${br}${e.name}${dt} \u2014 ${e.description??"no description"}`),e.aliases?.length&&t.push(`${Vt}Aliases: ${e.aliases.join(", ")}${dt}`),t.push(""),t.push(`${As}Usage:${dt}`),e.params.length)for(let n of e.params)t.push(` ${e.name} ${n}`);else t.push(` ${e.name}`);let r=Sr[e.category??"misc"]??e.category??"misc";return t.push(""),t.push(`${Vt}Category: ${r}${dt}`),t.join(`
|
|
90
|
+
`)}function vr(e){return{name:"help",description:"List all commands, or show usage for a specific command",category:"shell",params:["[command]"],run:({args:t})=>{let r=fe();if(t[0]){let n=t[0].toLowerCase(),i=r.find(s=>s.name===n||s.aliases?.includes(n));return i?{stdout:Is(i),exitCode:0}:{stderr:`help: no help entry for '${t[0]}'`,exitCode:1}}return{stdout:Ns(r),exitCode:0}}}}var Cr={name:"history",description:"Display command history",category:"shell",params:["[n]"],run:({args:e,shell:t})=>{let r="/virtual-env-js/.bash_history";if(!t.vfs.exists(r))return{stdout:"",exitCode:0};let i=t.vfs.readFile(r).split(`
|
|
89
91
|
`).filter(Boolean),s=e[0],o=s?parseInt(s,10):null,a=o&&!Number.isNaN(o)?i.slice(-o):i,l=i.length-a.length+1;return{stdout:a.map((u,d)=>`${String(l+d).padStart(5)} ${u}`).join(`
|
|
90
|
-
`),exitCode:0}}};var
|
|
91
|
-
`,exitCode:1};let s=
|
|
92
|
+
`),exitCode:0}}};var $r={name:"hostname",description:"Print hostname",category:"system",params:[],run:({hostname:e})=>({stdout:e,exitCode:0})};var Pr={name:"htop",description:"System monitor",category:"system",params:[],run:({mode:e})=>e==="exec"?{stderr:"htop: interactive terminal required",exitCode:1}:{openHtop:!0,exitCode:0}};var kr={name:"id",description:"Print user identity",category:"system",params:["[user]"],run:({authUser:e,shell:t,args:r})=>{let n=r[0]??e,i=n==="root"?0:1e3,s=i,a=t.users.isSudoer(n)?`${s}(${n}),0(root)`:`${s}(${n})`;return{stdout:`uid=${i}(${n}) gid=${s}(${n}) groups=${a}`,exitCode:0}}};var Mr={name:"kill",description:"Send signal to process",category:"system",params:["[-9] <pid>"],run:({args:e})=>e.find(r=>!r.startsWith("-"))?{stdout:"",exitCode:0}:{stderr:"kill: no pid specified",exitCode:1}};var Fr={name:"ln",description:"Create links",category:"files",params:["[-s] <target> <link_name>"],run:({authUser:e,shell:t,cwd:r,args:n})=>{let i=y(n,["-s","--symbolic"]),s=n.filter(u=>!u.startsWith("-")),[o,a]=s;if(!o||!a)return{stderr:"ln: missing operand",exitCode:1};let l=S(r,a),c=i?o:S(r,o);try{if(L(e,l,"ln"),i)t.vfs.symlink(c,l);else{let u=S(r,o);if(L(e,u,"ln"),!t.vfs.exists(u))return{stderr:`ln: ${o}: No such file or directory`,exitCode:1};let d=t.vfs.readFile(u);t.writeFileAsUser(e,l,d)}return{exitCode:0}}catch(u){return{stderr:`ln: ${u instanceof Error?u.message:String(u)}`,exitCode:1}}}},Ar={name:"readlink",description:"Print resolved path of symbolic link",category:"files",params:["[-f] <path>"],run:({shell:e,cwd:t,args:r})=>{let n=r.includes("-f")||r.includes("-e"),i=r.find(a=>!a.startsWith("-"));if(!i)return{stderr:`readlink: missing operand
|
|
93
|
+
`,exitCode:1};let s=S(t,i);return e.vfs.exists(s)?e.vfs.isSymlink(s)?{stdout:`${e.vfs.resolveSymlink(s)}
|
|
92
94
|
`,exitCode:0}:{stderr:`readlink: ${i}: not a symbolic link
|
|
93
95
|
`,exitCode:1}:{stderr:`readlink: ${i}: No such file or directory
|
|
94
|
-
`,exitCode:1}}};var
|
|
95
|
-
`,exitCode:1};let o=
|
|
96
|
-
`,exitCode:1};let a=e.vfs.stat(o),l=a.type==="directory",c=e.vfs.isSymlink(o),u=e.vfs.isSymlink(o),d=$=>{let
|
|
97
|
-
`,exitCode:0}:{stdout:`${[` File: ${s}${u?` -> ${e.vfs.resolveSymlink(o)}`:""}`,` Size: ${
|
|
96
|
+
`,exitCode:1}}};var Er={name:"stat",description:"Display file status",category:"files",params:["[-c <format>] <file>"],run:({shell:e,cwd:t,args:r})=>{let n=r.findIndex($=>$==="-c"||$==="--format"),i=n!==-1?r[n+1]:void 0,s=r.find($=>!$.startsWith("-")&&$!==i);if(!s)return{stderr:`stat: missing operand
|
|
97
|
+
`,exitCode:1};let o=S(t,s);if(!e.vfs.exists(o))return{stderr:`stat: cannot stat '${s}': No such file or directory
|
|
98
|
+
`,exitCode:1};let a=e.vfs.stat(o),l=a.type==="directory",c=e.vfs.isSymlink(o),u=e.vfs.isSymlink(o),d=$=>{let E=[256,128,64,32,16,8,4,2,1],M=["r","w","x","r","w","x","r","w","x"];return(l?"d":u?"l":"-")+E.map((x,k)=>$&x?M[k]:"-").join("")},m=a.mode.toString(8).padStart(4,"0"),p=d(a.mode),w="size"in a?a.size:0,g=$=>$.toISOString().replace("T"," ").replace(/\.\d+Z$/," +0000");return i?{stdout:`${i.replace("%n",s).replace("%s",String(w)).replace("%a",m.slice(1)).replace("%A",p).replace("%F",u?"symbolic link":l?"directory":"regular file").replace("%y",g(a.updatedAt)).replace("%z",g(a.updatedAt))}
|
|
99
|
+
`,exitCode:0}:{stdout:`${[` File: ${s}${u?` -> ${e.vfs.resolveSymlink(o)}`:""}`,` Size: ${w}${" ".repeat(3)}${u?"symbolic link":l?"directory":"regular file"}`,`Access: (${m}/${p}) Uid: ( 0/ root) Gid: ( 0/ root)`,`Modify: ${g(a.updatedAt)}`,`Change: ${g(a.updatedAt)}`].join(`
|
|
98
100
|
`)}
|
|
99
|
-
`,exitCode:0}}};function
|
|
100
|
-
`,exitCode:0}}return{stdout:a.split("/").pop()??a,exitCode:0}}}let l=t.vfs.list(a).filter(u=>s||!u.startsWith("."));return{stdout:i?l.map(u=>{let d=
|
|
101
|
-
`):
|
|
101
|
+
`,exitCode:0}}};function Nr(e,t){let r=t?"d":"-",i=[[256,"r"],[128,"w"],[64,"x"],[32,"r"],[16,"w"],[8,"x"],[4,"r"],[2,"w"],[1,"x"]].map(([s,o])=>e&s?o:"-").join("");return`${r}${i}`}function Ir(e){return e.toISOString().replace("T"," ").slice(0,16)}var Vr={name:"ls",description:"List directory contents",category:"navigation",params:["[-la] [path]"],run:({authUser:e,shell:t,cwd:r,args:n})=>{let i=y(n,["-l","--long"]),s=y(n,["-a","--all"]),o=xt(n,0,{flags:["-l","--long","-a","--all","-la","-al"]}),a=S(r,o??r);if(L(e,a,"ls"),t.vfs.exists(a)){let u=t.vfs.stat(a);if(u.type==="file"||t.vfs.isSymlink(a)){if(i){let d=a.split("/").pop()??a,m=u.type==="file"?u.size:0;return{stdout:`${Nr(u.mode,!1)} 1 root root ${m} ${Ir(u.updatedAt)} ${d}
|
|
102
|
+
`,exitCode:0}}return{stdout:a.split("/").pop()??a,exitCode:0}}}let l=t.vfs.list(a).filter(u=>s||!u.startsWith("."));return{stdout:i?l.map(u=>{let d=S(a,u),m=t.vfs.stat(d),p=m.type==="file"?m.size:m.childrenCount;return`${Nr(m.mode,m.type==="directory")} 1 ${p} ${Ir(m.updatedAt)} ${u}${m.type==="directory"?"/":""}`}).join(`
|
|
103
|
+
`):Be(a,l,u=>t.vfs.stat(u)),exitCode:0}}};var Rr={name:"lsb_release",description:"Print distribution-specific information",category:"system",params:["[-a] [-i] [-d] [-r] [-c]"],run:({args:e,shell:t})=>{let r=t.properties?.os??"Fortune GNU/Linux x64",n="aurora",i="1.0";try{let d=t.vfs.readFile("/etc/os-release");for(let m of d.split(`
|
|
102
104
|
`))m.startsWith("PRETTY_NAME=")&&(r=m.slice(12).replace(/^"|"$/g,"").trim()),m.startsWith("VERSION_CODENAME=")&&(n=m.slice(17).trim()),m.startsWith("VERSION_ID=")&&(i=m.slice(11).replace(/^"|"$/g,"").trim())}catch{}let s=y(e,["-a","--all"]),o=y(e,["-i","--id"]),a=y(e,["-d","--description"]),l=y(e,["-r","--release"]),c=y(e,["-c","--codename"]);if(s||e.length===0)return{stdout:["Distributor ID: Fortune",`Description: ${r}`,`Release: ${i}`,`Codename: ${n}`].join(`
|
|
103
105
|
`),exitCode:0};let u=[];return o&&u.push("Distributor ID: Fortune"),a&&u.push(`Description: ${r}`),l&&u.push(`Release: ${i}`),c&&u.push(`Codename: ${n}`),{stdout:u.join(`
|
|
104
|
-
`),exitCode:0}}};var
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
DESCRIPTION
|
|
113
|
-
List information about the FILEs (the current directory by default).
|
|
114
|
-
|
|
115
|
-
OPTIONS
|
|
116
|
-
-l use a long listing format
|
|
117
|
-
-a do not ignore entries starting with .
|
|
118
|
-
-h with -l, print human readable sizes
|
|
119
|
-
-r reverse order while sorting
|
|
120
|
-
-t sort by modification time
|
|
121
|
-
|
|
122
|
-
AUTHOR
|
|
123
|
-
Written by Richard M. Stallman and David MacKenzie.`,cat:`CAT(1) User Commands CAT(1)
|
|
124
|
-
|
|
125
|
-
NAME
|
|
126
|
-
cat - concatenate files and print on the standard output
|
|
127
|
-
|
|
128
|
-
SYNOPSIS
|
|
129
|
-
cat [OPTION]... [FILE]...
|
|
130
|
-
|
|
131
|
-
DESCRIPTION
|
|
132
|
-
Concatenate FILE(s) to standard output.
|
|
133
|
-
|
|
134
|
-
OPTIONS
|
|
135
|
-
-n, --number number all output lines
|
|
136
|
-
-b, --number-nonblank number nonempty output lines`,grep:`GREP(1) User Commands GREP(1)
|
|
137
|
-
|
|
138
|
-
NAME
|
|
139
|
-
grep, egrep, fgrep - print lines that match patterns
|
|
140
|
-
|
|
141
|
-
SYNOPSIS
|
|
142
|
-
grep [OPTION]... PATTERNS [FILE]...
|
|
143
|
-
|
|
144
|
-
OPTIONS
|
|
145
|
-
-i, --ignore-case ignore case distinctions in patterns and data
|
|
146
|
-
-v, --invert-match select non-matching lines
|
|
147
|
-
-n, --line-number print line number with output lines
|
|
148
|
-
-r, --recursive read all files under each directory, recursively`,apt:`APT(8) APT APT(8)
|
|
149
|
-
|
|
150
|
-
NAME
|
|
151
|
-
apt - command-line interface
|
|
152
|
-
|
|
153
|
-
SYNOPSIS
|
|
154
|
-
apt [options] command
|
|
155
|
-
|
|
156
|
-
DESCRIPTION
|
|
157
|
-
apt provides a high-level commandline interface for the package
|
|
158
|
-
management system.
|
|
159
|
-
|
|
160
|
-
COMMANDS
|
|
161
|
-
install pkg... Install packages
|
|
162
|
-
remove pkg... Remove packages
|
|
163
|
-
update Download package information
|
|
164
|
-
upgrade Upgrade installed packages
|
|
165
|
-
search term Search in package descriptions
|
|
166
|
-
show pkg Show package information
|
|
167
|
-
list List packages`,ssh:`SSH(1) OpenSSH SSH(1)
|
|
168
|
-
|
|
169
|
-
NAME
|
|
170
|
-
ssh - OpenSSH remote login client
|
|
171
|
-
|
|
172
|
-
SYNOPSIS
|
|
173
|
-
ssh [-p port] [user@]hostname [command]
|
|
174
|
-
|
|
175
|
-
DESCRIPTION
|
|
176
|
-
ssh (SSH client) is a program for logging into a remote machine and
|
|
177
|
-
for executing commands on a remote machine.`,curl:`CURL(1) User Commands CURL(1)
|
|
178
|
-
|
|
179
|
-
NAME
|
|
180
|
-
curl - transfer a URL
|
|
181
|
-
|
|
182
|
-
SYNOPSIS
|
|
183
|
-
curl [options / URLs]
|
|
184
|
-
|
|
185
|
-
DESCRIPTION
|
|
186
|
-
curl is a tool for transferring data with URL syntax.
|
|
187
|
-
|
|
188
|
-
OPTIONS
|
|
189
|
-
-o, --output <file> Write output to <file>
|
|
190
|
-
-X, --request <method> Specify request method
|
|
191
|
-
-d, --data <data> HTTP POST data
|
|
192
|
-
-H, --header <header> Pass custom header
|
|
193
|
-
-s, --silent Silent mode
|
|
194
|
-
-I, --head Show document info only
|
|
195
|
-
-L, --location Follow redirects
|
|
196
|
-
-v, --verbose Make the operation more talkative`,chmod:`CHMOD(1) User Commands CHMOD(1)
|
|
197
|
-
|
|
198
|
-
NAME
|
|
199
|
-
chmod - change file mode bits
|
|
200
|
-
|
|
201
|
-
SYNOPSIS
|
|
202
|
-
chmod [OPTION]... MODE[,MODE]... FILE...
|
|
203
|
-
chmod [OPTION]... OCTAL-MODE FILE...
|
|
204
|
-
|
|
205
|
-
DESCRIPTION
|
|
206
|
-
Change the file mode bits of each given file according to MODE.
|
|
207
|
-
|
|
208
|
-
EXAMPLES
|
|
209
|
-
chmod 755 script.sh rwxr-xr-x
|
|
210
|
-
chmod 644 file.txt rw-r--r--
|
|
211
|
-
chmod +x script.sh add execute permission`,tar:`TAR(1) GNU tar Manual TAR(1)
|
|
212
|
-
|
|
213
|
-
NAME
|
|
214
|
-
tar - an archiving utility
|
|
215
|
-
|
|
216
|
-
SYNOPSIS
|
|
217
|
-
tar [OPTION...] [FILE]...
|
|
218
|
-
|
|
219
|
-
DESCRIPTION
|
|
220
|
-
tar saves many files together into a single tape or disk archive,
|
|
221
|
-
and can restore individual files from the archive.
|
|
222
|
-
|
|
223
|
-
OPTIONS
|
|
224
|
-
-c, --create create a new archive
|
|
225
|
-
-x, --extract extract files from an archive
|
|
226
|
-
-z, --gzip filter the archive through gzip
|
|
227
|
-
-f, --file=ARCHIVE use archive file or device ARCHIVE
|
|
228
|
-
-v, --verbose verbosely list files processed
|
|
229
|
-
-t, --list list the contents of an archive`},Vr={name:"man",description:"Interface to the system reference manuals",category:"shell",params:["<command>"],run:({args:e,shell:t})=>{let r=e[0];if(!r)return{stderr:"What manual page do you want?",exitCode:1};let n=`/usr/share/man/man1/${r}.1`;if(t.vfs.exists(n))return{stdout:t.vfs.readFile(n),exitCode:0};let i=Is[r.toLowerCase()];return i?{stdout:i,exitCode:0}:{stderr:`No manual entry for ${r}`,exitCode:16}}};var Dr={name:"mkdir",description:"Make directories",category:"files",params:["<dir>"],run:({authUser:e,shell:t,cwd:r,args:n})=>{if(n.length===0)return{stderr:"mkdir: missing operand",exitCode:1};for(let i=0;i<n.length;i++){let s=wt(n,i);if(!s)return{stderr:"mkdir: missing operand",exitCode:1};let o=x(r,s);L(e,o,"mkdir"),t.vfs.mkdir(o)}return{exitCode:0}}};var _r={name:"mv",description:"Move or rename files",category:"files",params:["<source> <dest>"],run:({authUser:e,shell:t,cwd:r,args:n})=>{let i=n.filter(c=>!c.startsWith("-")),[s,o]=i;if(!s||!o)return{stderr:"mv: missing operand",exitCode:1};let a=x(r,s),l=x(r,o);try{if(L(e,a,"mv"),L(e,l,"mv"),!t.vfs.exists(a))return{stderr:`mv: ${s}: No such file or directory`,exitCode:1};let c=t.vfs.exists(l)&&t.vfs.stat(l).type==="directory"?`${l}/${s.split("/").pop()}`:l;return t.vfs.move(a,c),{exitCode:0}}catch(c){return{stderr:`mv: ${c instanceof Error?c.message:String(c)}`,exitCode:1}}}};import*as Lr from"node:path";var Tr={name:"nano",description:"Text editor",category:"files",params:["<file>"],run:({authUser:e,shell:t,cwd:r,args:n})=>{let i=n[0];if(!i)return{stderr:"nano: missing file operand",exitCode:1};let s=x(r,i);L(e,s,"nano");let o=t.vfs.exists(s)?t.vfs.readFile(s):"",a=Lr.posix.basename(s)||"buffer",l=`/tmp/sshmimic-nano-${Date.now()}-${a}.tmp`;return{openEditor:{targetPath:s,tempPath:l,initialContent:o},exitCode:0}}};import{existsSync as Hr,readdirSync as Rs,readFileSync as fe}from"node:fs";import*as tt from"node:os";import*as qr from"node:path";function Vs(e){let t=Math.max(1,Math.floor(e/60)),r=Math.floor(t/1440),n=Math.floor(t%1440/60),i=t%60,s=[];return r>0&&s.push(`${r} day${r>1?"s":""}`),n>0&&s.push(`${n} hour${n>1?"s":""}`),(i>0||s.length===0)&&s.push(`${i} min${i>1?"s":""}`),s.join(", ")}function zr(e){return`\x1B[${e}m \x1B[0m`}function Ds(){let e=[40,41,42,43,44,45,46,47].map(zr).join(""),t=[100,101,102,103,104,105,106,107].map(zr).join("");return[e,t]}function Or(e,t,r){if(e.trim().length===0)return e;let n={r:255,g:255,b:255},i={r:168,g:85,b:247},s=r<=1?0:t/(r-1),o=Math.round(n.r+(i.r-n.r)*s),a=Math.round(n.g+(i.g-n.g)*s),l=Math.round(n.b+(i.b-n.b)*s);return`\x1B[38;2;${o};${a};${l}m${e}\x1B[0m`}function _s(e){if(e.trim().length===0)return e;let t=e.indexOf(":");if(t===-1)return e.includes("@")?Ur(e):e;let r=e.substring(0,t+1),n=e.substring(t+1);return Ur(r)+n}function Ur(e){let t=new RegExp("\x1B\\[[\\d;]*m","g"),r=e.replace(t,"");if(r.trim().length===0)return e;let n={r:255,g:255,b:255},i={r:168,g:85,b:247},s="";for(let o=0;o<r.length;o+=1){let a=r.length<=1?0:o/(r.length-1),l=Math.round(n.r+(i.r-n.r)*a),c=Math.round(n.g+(i.g-n.g)*a),u=Math.round(n.b+(i.b-n.b)*a);s+=`\x1B[38;2;${l};${c};${u}m${r[o]}\x1B[0m`}return s}function Br(e){return Math.max(0,Math.round(e/(1024*1024)))}function Wr(){try{let e=fe("/etc/os-release","utf8");for(let t of e.split(`
|
|
230
|
-
`)){if(!t.startsWith("PRETTY_NAME="))continue;return t.slice(12).trim().replace(/^"|"$/g,"")}}catch{return}}function jr(e){try{let t=fe(e,"utf8").split(`
|
|
231
|
-
`)[0]?.trim();return!t||t.length===0?void 0:t}catch{return}}function Ls(e){let t=jr("/sys/devices/virtual/dmi/id/sys_vendor"),r=jr("/sys/devices/virtual/dmi/id/product_name");return t&&r?`${t} ${r}`:r||e}function Ts(){let e=["/var/lib/dpkg/status","/usr/local/var/lib/dpkg/status"];for(let t of e)if(Hr(t))try{return fe(t,"utf8").match(/^Package:\s+/gm)?.length??0}catch{}}function zs(){let e=["/snap","/var/lib/snapd/snaps"];for(let t of e)if(Hr(t))try{return Rs(t,{withFileTypes:!0}).filter(i=>i.isDirectory()).length}catch{}}function Os(){let e=Ts(),t=zs();return e!==void 0&&t!==void 0?`${e} (dpkg), ${t} (snap)`:e!==void 0?`${e} (dpkg)`:t!==void 0?`${t} (snap)`:"n/a"}function Us(){let e=tt.cpus();if(e.length===0)return"unknown";let t=e[0];if(!t)return"unknown";let r=(t.speed/1e3).toFixed(2);return`${t.model} (${e.length}) @ ${r}GHz`}function Bs(e){return!e||e.trim().length===0?"unknown":qr.posix.basename(e.trim())}function Ws(e){let t=tt.totalmem(),r=tt.freemem(),n=Math.max(0,t-r),i=e.shellProps,s=process.uptime();return e.uptimeSeconds===void 0&&(e.uptimeSeconds=Math.round(s)),{user:e.user,host:e.host,osName:i?.os??e.osName??`${Wr()??tt.type()} ${tt.arch()}`,kernel:i?.kernel??e.kernel??tt.release(),uptimeSeconds:e.uptimeSeconds??tt.uptime(),packages:e.packages??Os(),shell:Bs(e.shell),shellProps:e.shellProps??{kernel:e.kernel??tt.release(),os:e.osName??`${Wr()??tt.type()} ${tt.arch()}`,arch:tt.arch()},resolution:e.resolution??"n/a (ssh)",terminal:e.terminal??"unknown",cpu:e.cpu??Us(),gpu:e.gpu??"n/a",memoryUsedMiB:e.memoryUsedMiB??Br(n),memoryTotalMiB:e.memoryTotalMiB??Br(t)}}function Kr(e){let t=Ws(e),r=Vs(t.uptimeSeconds),n=Ds(),i=[" .. .:. "," .::.. .. .. ",". .... ... .. ",": .... .:. .. ",": .:.:........:. .. ",": .. ",". : ",". : ",".. : "," :. .. "," .. .. "," :-. :: "," .:. :. "," ..: ... "," ..: :.. "," :... :...."," .. ...."," . .. "," .:. .: "," :. .. "," ::. .. ","..... ..:... ","...:. .. ",".:...:. ::. .. "," ... ..:::::.. ..:::::::.. "],s=[`${t.user}@${t.host}`,"-------------------------",`OS: ${t.osName}`,`Host: ${Ls(t.host)}`,`Kernel: ${t.kernel}`,`Uptime: ${r}`,`Packages: ${t.packages}`,`Shell: ${t.shell}`,`Resolution: ${t.resolution}`,`Terminal: ${t.terminal}`,`CPU: ${t.cpu}`,`GPU: ${t.gpu}`,`Memory: ${t.memoryUsedMiB}MiB / ${t.memoryTotalMiB}MiB`,"",n[0],n[1]],o=Math.max(i.length,s.length),a=[];for(let l=0;l<o;l+=1){let c=i[l]??"",u=s[l]??"";if(u.length>0){let d=Or(c.padEnd(31," "),l,i.length),m=_s(u);a.push(`${d} ${m}`);continue}a.push(Or(c,l,i.length))}return a.join(`
|
|
232
|
-
`)}var Gr={name:"neofetch",description:"System info display",category:"system",params:["[--off]"],run:({args:e,authUser:t,hostname:r,shell:n,env:i})=>y(e,"--help")?{stdout:"Usage: neofetch [--off]",exitCode:0}:y(e,"--off")?{stdout:`${t}@${r}`,exitCode:0}:{stdout:Kr({user:t,host:r,shell:i.vars.SHELL,shellProps:n.properties,terminal:i.vars.TERM,uptimeSeconds:Math.floor((Date.now()-n.startTime)/1e3),packages:`${n.packageManager?.installedCount()??0} (dpkg)`}),exitCode:0}};import Zr from"node:vm";var Kt="v18.19.0",Jr={node:Kt,npm:"9.2.0",v8:"10.2.154.26-node.22"};function js(e,t){let r={version:Kt,versions:Jr,platform:"linux",arch:"x64",env:{NODE_ENV:"production",HOME:"/root",PATH:"/usr/local/bin:/usr/bin:/bin"},argv:["node"],stdout:{write:s=>(e.push(s),!0)},stderr:{write:s=>(t.push(s),!0)},exit:(s=0)=>{throw new Gt(s)},cwd:()=>"/root",hrtime:()=>[0,0]},n={log:(...s)=>e.push(s.map(mt).join(" ")),error:(...s)=>t.push(s.map(mt).join(" ")),warn:(...s)=>t.push(s.map(mt).join(" ")),info:(...s)=>e.push(s.map(mt).join(" ")),dir:s=>e.push(mt(s))},i=s=>{switch(s){case"path":return{join:(...o)=>o.join("/").replace(/\/+/g,"/"),resolve:(...o)=>`/${o.join("/").replace(/^\/+/,"")}`,dirname:o=>o.split("/").slice(0,-1).join("/")||"/",basename:o=>o.split("/").pop()??"",extname:o=>{let a=o.split("/").pop()??"",l=a.lastIndexOf(".");return l>0?a.slice(l):""},sep:"/",delimiter:":"};case"os":return{platform:()=>"linux",arch:()=>"x64",type:()=>"Linux",hostname:()=>"fortune-vm",homedir:()=>"/root",tmpdir:()=>"/tmp",EOL:`
|
|
233
|
-
`};case"util":return{format:(...o)=>o.map(mt).join(" "),inspect:o=>mt(o)};case"fs":case"fs/promises":throw new Error(`Cannot require '${s}': filesystem access not available in virtual runtime`);case"child_process":case"net":case"http":case"https":throw new Error(`Cannot require '${s}': not available in virtual runtime`);default:throw new Error(`Cannot find module '${s}'`)}};return i.resolve=s=>{throw new Error(`Cannot resolve '${s}'`)},i.cache={},i.extensions={},Zr.createContext({console:n,process:r,require:i,Math,JSON,Object,Array,String,Number,Boolean,Symbol,Date,RegExp,Error,TypeError,RangeError,SyntaxError,Promise,Map,Set,WeakMap,WeakSet,parseInt,parseFloat,isNaN,isFinite,encodeURIComponent,decodeURIComponent,encodeURI,decodeURI,setTimeout:()=>{},clearTimeout:()=>{},setInterval:()=>{},clearInterval:()=>{},queueMicrotask:()=>{},globalThis:void 0,undefined:void 0,Infinity:1/0,NaN:NaN})}var Gt=class{constructor(t){this.code=t}code};function mt(e){if(e===null)return"null";if(e===void 0)return"undefined";if(typeof e=="string")return e;if(typeof e=="function")return`[Function: ${e.name||"(anonymous)"}]`;if(Array.isArray(e))return`[ ${e.map(mt).join(", ")} ]`;if(e instanceof Error)return`${e.name}: ${e.message}`;if(typeof e=="object")try{return`{ ${Object.entries(e).map(([r,n])=>`${r}: ${mt(n)}`).join(", ")} }`}catch{return"[Object]"}return String(e)}function Zt(e){let t=[],r=[],n=js(t,r),i=0;try{let s=Zr.runInContext(e,n,{timeout:5e3});s!==void 0&&t.length===0&&t.push(mt(s))}catch(s){s instanceof Gt?i=s.code:s instanceof Error?(r.push(`${s.name}: ${s.message}`),i=1):(r.push(String(s)),i=1)}return{stdout:t.length?`${t.join(`
|
|
106
|
+
`),exitCode:0}}};var Vs={gunzip:"gzip"},Kt=new Map,Rs=new URL("./manuals/",import.meta.url);async function Ds(e){return new Function("moduleName","return import(moduleName)")(e)}async function _s(e){let t=e.toLowerCase(),r=Vs[t]??t,n=`builtin:${r}`;if(Kt.has(n))return Kt.get(n)??null;try{let i=await Ds("node:fs/promises"),s=new URL(`${r}.txt`,Rs),a=(await i.readFile(s,"utf8")).replace(/\n$/,"");return Kt.set(n,a),a}catch{return Kt.set(n,null),null}}var Dr={name:"man",description:"Interface to the system reference manuals",category:"shell",params:["<command>"],run:async({args:e,shell:t})=>{let r=e[0];if(!r)return{stderr:"What manual page do you want?",exitCode:1};let n=`/usr/share/man/man1/${r}.1`;if(t.vfs.exists(n))return{stdout:t.vfs.readFile(n),exitCode:0};let i=await _s(r);return i?{stdout:i,exitCode:0}:{stderr:`No manual entry for ${r}`,exitCode:16}}};var _r={name:"mkdir",description:"Make directories",category:"files",params:["<dir>"],run:({authUser:e,shell:t,cwd:r,args:n})=>{if(n.length===0)return{stderr:"mkdir: missing operand",exitCode:1};for(let i=0;i<n.length;i++){let s=xt(n,i);if(!s)return{stderr:"mkdir: missing operand",exitCode:1};let o=S(r,s);L(e,o,"mkdir"),t.vfs.mkdir(o)}return{exitCode:0}}};var Lr={name:"mv",description:"Move or rename files",category:"files",params:["<source> <dest>"],run:({authUser:e,shell:t,cwd:r,args:n})=>{let i=n.filter(c=>!c.startsWith("-")),[s,o]=i;if(!s||!o)return{stderr:"mv: missing operand",exitCode:1};let a=S(r,s),l=S(r,o);try{if(L(e,a,"mv"),L(e,l,"mv"),!t.vfs.exists(a))return{stderr:`mv: ${s}: No such file or directory`,exitCode:1};let c=t.vfs.exists(l)&&t.vfs.stat(l).type==="directory"?`${l}/${s.split("/").pop()}`:l;return t.vfs.move(a,c),{exitCode:0}}catch(c){return{stderr:`mv: ${c instanceof Error?c.message:String(c)}`,exitCode:1}}}};import*as zr from"node:path";var Ur={name:"nano",description:"Text editor",category:"files",params:["<file>"],run:({authUser:e,shell:t,cwd:r,args:n})=>{let i=n[0];if(!i)return{stderr:"nano: missing file operand",exitCode:1};let s=S(r,i);L(e,s,"nano");let o=t.vfs.exists(s)?t.vfs.readFile(s):"",a=zr.posix.basename(s)||"buffer",l=`/tmp/sshmimic-nano-${Date.now()}-${a}.tmp`;return{openEditor:{targetPath:s,tempPath:l,initialContent:o},exitCode:0}}};import{existsSync as qr,readdirSync as Ls,readFileSync as he}from"node:fs";import*as tt from"node:os";import*as Kr from"node:path";function zs(e){let t=Math.max(1,Math.floor(e/60)),r=Math.floor(t/1440),n=Math.floor(t%1440/60),i=t%60,s=[];return r>0&&s.push(`${r} day${r>1?"s":""}`),n>0&&s.push(`${n} hour${n>1?"s":""}`),(i>0||s.length===0)&&s.push(`${i} min${i>1?"s":""}`),s.join(", ")}function Tr(e){return`\x1B[${e}m \x1B[0m`}function Us(){let e=[40,41,42,43,44,45,46,47].map(Tr).join(""),t=[100,101,102,103,104,105,106,107].map(Tr).join("");return[e,t]}function Br(e,t,r){if(e.trim().length===0)return e;let n={r:255,g:255,b:255},i={r:168,g:85,b:247},s=r<=1?0:t/(r-1),o=Math.round(n.r+(i.r-n.r)*s),a=Math.round(n.g+(i.g-n.g)*s),l=Math.round(n.b+(i.b-n.b)*s);return`\x1B[38;2;${o};${a};${l}m${e}\x1B[0m`}function Ts(e){if(e.trim().length===0)return e;let t=e.indexOf(":");if(t===-1)return e.includes("@")?Or(e):e;let r=e.substring(0,t+1),n=e.substring(t+1);return Or(r)+n}function Or(e){let t=new RegExp("\x1B\\[[\\d;]*m","g"),r=e.replace(t,"");if(r.trim().length===0)return e;let n={r:255,g:255,b:255},i={r:168,g:85,b:247},s="";for(let o=0;o<r.length;o+=1){let a=r.length<=1?0:o/(r.length-1),l=Math.round(n.r+(i.r-n.r)*a),c=Math.round(n.g+(i.g-n.g)*a),u=Math.round(n.b+(i.b-n.b)*a);s+=`\x1B[38;2;${l};${c};${u}m${r[o]}\x1B[0m`}return s}function Wr(e){return Math.max(0,Math.round(e/(1024*1024)))}function jr(){try{let e=he("/etc/os-release","utf8");for(let t of e.split(`
|
|
107
|
+
`)){if(!t.startsWith("PRETTY_NAME="))continue;return t.slice(12).trim().replace(/^"|"$/g,"")}}catch{return}}function Hr(e){try{let t=he(e,"utf8").split(`
|
|
108
|
+
`)[0]?.trim();return!t||t.length===0?void 0:t}catch{return}}function Bs(e){let t=Hr("/sys/devices/virtual/dmi/id/sys_vendor"),r=Hr("/sys/devices/virtual/dmi/id/product_name");return t&&r?`${t} ${r}`:r||e}function Os(){let e=["/var/lib/dpkg/status","/usr/local/var/lib/dpkg/status"];for(let t of e)if(qr(t))try{return he(t,"utf8").match(/^Package:\s+/gm)?.length??0}catch{}}function Ws(){let e=["/snap","/var/lib/snapd/snaps"];for(let t of e)if(qr(t))try{return Ls(t,{withFileTypes:!0}).filter(i=>i.isDirectory()).length}catch{}}function js(){let e=Os(),t=Ws();return e!==void 0&&t!==void 0?`${e} (dpkg), ${t} (snap)`:e!==void 0?`${e} (dpkg)`:t!==void 0?`${t} (snap)`:"n/a"}function Hs(){let e=tt.cpus();if(e.length===0)return"unknown";let t=e[0];if(!t)return"unknown";let r=(t.speed/1e3).toFixed(2);return`${t.model} (${e.length}) @ ${r}GHz`}function qs(e){return!e||e.trim().length===0?"unknown":Kr.posix.basename(e.trim())}function Ks(e){let t=tt.totalmem(),r=tt.freemem(),n=Math.max(0,t-r),i=e.shellProps,s=process.uptime();return e.uptimeSeconds===void 0&&(e.uptimeSeconds=Math.round(s)),{user:e.user,host:e.host,osName:i?.os??e.osName??`${jr()??tt.type()} ${tt.arch()}`,kernel:i?.kernel??e.kernel??tt.release(),uptimeSeconds:e.uptimeSeconds??tt.uptime(),packages:e.packages??js(),shell:qs(e.shell),shellProps:e.shellProps??{kernel:e.kernel??tt.release(),os:e.osName??`${jr()??tt.type()} ${tt.arch()}`,arch:tt.arch()},resolution:e.resolution??"n/a (ssh)",terminal:e.terminal??"unknown",cpu:e.cpu??Hs(),gpu:e.gpu??"n/a",memoryUsedMiB:e.memoryUsedMiB??Wr(n),memoryTotalMiB:e.memoryTotalMiB??Wr(t)}}function Gr(e){let t=Ks(e),r=zs(t.uptimeSeconds),n=Us(),i=[" .. .:. "," .::.. .. .. ",". .... ... .. ",": .... .:. .. ",": .:.:........:. .. ",": .. ",". : ",". : ",".. : "," :. .. "," .. .. "," :-. :: "," .:. :. "," ..: ... "," ..: :.. "," :... :...."," .. ...."," . .. "," .:. .: "," :. .. "," ::. .. ","..... ..:... ","...:. .. ",".:...:. ::. .. "," ... ..:::::.. ..:::::::.. "],s=[`${t.user}@${t.host}`,"-------------------------",`OS: ${t.osName}`,`Host: ${Bs(t.host)}`,`Kernel: ${t.kernel}`,`Uptime: ${r}`,`Packages: ${t.packages}`,`Shell: ${t.shell}`,`Resolution: ${t.resolution}`,`Terminal: ${t.terminal}`,`CPU: ${t.cpu}`,`GPU: ${t.gpu}`,`Memory: ${t.memoryUsedMiB}MiB / ${t.memoryTotalMiB}MiB`,"",n[0],n[1]],o=Math.max(i.length,s.length),a=[];for(let l=0;l<o;l+=1){let c=i[l]??"",u=s[l]??"";if(u.length>0){let d=Br(c.padEnd(31," "),l,i.length),m=Ts(u);a.push(`${d} ${m}`);continue}a.push(Br(c,l,i.length))}return a.join(`
|
|
109
|
+
`)}var Zr={name:"neofetch",description:"System info display",category:"system",params:["[--off]"],run:({args:e,authUser:t,hostname:r,shell:n,env:i})=>n.packageManager.isInstalled("neofetch")?y(e,"--help")?{stdout:"Usage: neofetch [--off]",exitCode:0}:y(e,"--off")?{stdout:`${t}@${r}`,exitCode:0}:{stdout:Gr({user:t,host:r,shell:i.vars.SHELL,shellProps:n.properties,terminal:i.vars.TERM,uptimeSeconds:Math.floor((Date.now()-n.startTime)/1e3),packages:`${n.packageManager?.installedCount()??0} (dpkg)`}),exitCode:0}:{stderr:`bash: neofetch: command not found
|
|
110
|
+
Hint: install it with: apt install neofetch
|
|
111
|
+
`,exitCode:127}};import Jr from"node:vm";var Gt="v18.19.0",Qr={node:Gt,npm:"9.2.0",v8:"10.2.154.26-node.22"};function Gs(e,t){let r={version:Gt,versions:Qr,platform:"linux",arch:"x64",env:{NODE_ENV:"production",HOME:"/root",PATH:"/usr/local/bin:/usr/bin:/bin"},argv:["node"],stdout:{write:s=>(e.push(s),!0)},stderr:{write:s=>(t.push(s),!0)},exit:(s=0)=>{throw new Zt(s)},cwd:()=>"/root",hrtime:()=>[0,0]},n={log:(...s)=>e.push(s.map(mt).join(" ")),error:(...s)=>t.push(s.map(mt).join(" ")),warn:(...s)=>t.push(s.map(mt).join(" ")),info:(...s)=>e.push(s.map(mt).join(" ")),dir:s=>e.push(mt(s))},i=s=>{switch(s){case"path":return{join:(...o)=>o.join("/").replace(/\/+/g,"/"),resolve:(...o)=>`/${o.join("/").replace(/^\/+/,"")}`,dirname:o=>o.split("/").slice(0,-1).join("/")||"/",basename:o=>o.split("/").pop()??"",extname:o=>{let a=o.split("/").pop()??"",l=a.lastIndexOf(".");return l>0?a.slice(l):""},sep:"/",delimiter:":"};case"os":return{platform:()=>"linux",arch:()=>"x64",type:()=>"Linux",hostname:()=>"fortune-vm",homedir:()=>"/root",tmpdir:()=>"/tmp",EOL:`
|
|
112
|
+
`};case"util":return{format:(...o)=>o.map(mt).join(" "),inspect:o=>mt(o)};case"fs":case"fs/promises":throw new Error(`Cannot require '${s}': filesystem access not available in virtual runtime`);case"child_process":case"net":case"http":case"https":throw new Error(`Cannot require '${s}': not available in virtual runtime`);default:throw new Error(`Cannot find module '${s}'`)}};return i.resolve=s=>{throw new Error(`Cannot resolve '${s}'`)},i.cache={},i.extensions={},Jr.createContext({console:n,process:r,require:i,Math,JSON,Object,Array,String,Number,Boolean,Symbol,Date,RegExp,Error,TypeError,RangeError,SyntaxError,Promise,Map,Set,WeakMap,WeakSet,parseInt,parseFloat,isNaN,isFinite,encodeURIComponent,decodeURIComponent,encodeURI,decodeURI,setTimeout:()=>{},clearTimeout:()=>{},setInterval:()=>{},clearInterval:()=>{},queueMicrotask:()=>{},globalThis:void 0,undefined:void 0,Infinity:1/0,NaN:NaN})}var Zt=class{constructor(t){this.code=t}code};function mt(e){if(e===null)return"null";if(e===void 0)return"undefined";if(typeof e=="string")return e;if(typeof e=="function")return`[Function: ${e.name||"(anonymous)"}]`;if(Array.isArray(e))return`[ ${e.map(mt).join(", ")} ]`;if(e instanceof Error)return`${e.name}: ${e.message}`;if(typeof e=="object")try{return`{ ${Object.entries(e).map(([r,n])=>`${r}: ${mt(n)}`).join(", ")} }`}catch{return"[Object]"}return String(e)}function Jt(e){let t=[],r=[],n=Gs(t,r),i=0;try{let s=Jr.runInContext(e,n,{timeout:5e3});s!==void 0&&t.length===0&&t.push(mt(s))}catch(s){s instanceof Zt?i=s.code:s instanceof Error?(r.push(`${s.name}: ${s.message}`),i=1):(r.push(String(s)),i=1)}return{stdout:t.length?`${t.join(`
|
|
234
113
|
`)}
|
|
235
114
|
`:"",stderr:r.length?`${r.join(`
|
|
236
115
|
`)}
|
|
237
|
-
`:"",exitCode:i}}function
|
|
238
|
-
`)&&!t.startsWith("const ")&&!t.startsWith("let ")&&!t.startsWith("var ")&&!t.startsWith("function ")&&!t.startsWith("class ")&&!t.startsWith("if ")&&!t.startsWith("for ")&&!t.startsWith("while ")&&!t.startsWith("import ")&&!t.startsWith("//")?
|
|
116
|
+
`:"",exitCode:i}}function Zs(e){let t=e.trim();return!t.includes(`
|
|
117
|
+
`)&&!t.startsWith("const ")&&!t.startsWith("let ")&&!t.startsWith("var ")&&!t.startsWith("function ")&&!t.startsWith("class ")&&!t.startsWith("if ")&&!t.startsWith("for ")&&!t.startsWith("while ")&&!t.startsWith("import ")&&!t.startsWith("//")?Jt(t):Jt(`(async () => { ${e} })()`)}var Yr={name:"node",description:"JavaScript runtime (virtual)",category:"system",params:["[--version] [-e <expr>] [-p <expr>] [file]"],run:({args:e,shell:t,cwd:r})=>{if(!t.packageManager.isInstalled("nodejs"))return{stderr:`bash: node: command not found
|
|
239
118
|
Hint: install it with: apt install nodejs
|
|
240
|
-
`,exitCode:127};if(y(e,["--version","-v"]))return{stdout:`${
|
|
241
|
-
`,exitCode:0};if(y(e,["--versions"]))return{stdout:`${JSON.stringify(
|
|
119
|
+
`,exitCode:127};if(y(e,["--version","-v"]))return{stdout:`${Gt}
|
|
120
|
+
`,exitCode:0};if(y(e,["--versions"]))return{stdout:`${JSON.stringify(Qr,null,2)}
|
|
242
121
|
`,exitCode:0};let n=e.findIndex(o=>o==="-e"||o==="--eval");if(n!==-1){let o=e[n+1];if(!o)return{stderr:`node: -e requires an argument
|
|
243
|
-
`,exitCode:1};let{stdout:a,stderr:l,exitCode:c}=
|
|
244
|
-
`,exitCode:1};let{stdout:a,stderr:l,exitCode:c}=
|
|
245
|
-
`:void 0),stderr:l||void 0,exitCode:c}}let s=e.find(o=>!o.startsWith("-"));if(s){let o=
|
|
246
|
-
`,exitCode:1};let a=t.vfs.readFile(o),{stdout:l,stderr:c,exitCode:u}=
|
|
247
|
-
`),exitCode:0}}};var
|
|
122
|
+
`,exitCode:1};let{stdout:a,stderr:l,exitCode:c}=Jt(o);return{stdout:a||void 0,stderr:l||void 0,exitCode:c}}let i=e.findIndex(o=>o==="-p"||o==="--print");if(i!==-1){let o=e[i+1];if(!o)return{stderr:`node: -p requires an argument
|
|
123
|
+
`,exitCode:1};let{stdout:a,stderr:l,exitCode:c}=Jt(o);return{stdout:a||(c===0?`
|
|
124
|
+
`:void 0),stderr:l||void 0,exitCode:c}}let s=e.find(o=>!o.startsWith("-"));if(s){let o=S(r,s);if(!t.vfs.exists(o))return{stderr:`node: cannot open file '${s}': No such file or directory
|
|
125
|
+
`,exitCode:1};let a=t.vfs.readFile(o),{stdout:l,stderr:c,exitCode:u}=Zs(a);return{stdout:l||void 0,stderr:c||void 0,exitCode:u}}return{stdout:[`Welcome to Node.js ${Gt}.`,'Type ".exit" to exit the REPL.',"> "].join(`
|
|
126
|
+
`),exitCode:0}}};var Qt="9.2.0",Js="18.19.0",Xr={name:"npm",description:"Node.js package manager (virtual)",category:"system",params:["<command> [args]"],run:({args:e,shell:t})=>{if(!t.packageManager.isInstalled("npm"))return{stderr:`bash: npm: command not found
|
|
248
127
|
Hint: install it with: apt install npm
|
|
249
|
-
`,exitCode:127};if(y(e,["--version","-v"]))return{stdout:`${
|
|
250
|
-
`,exitCode:0};let r=e[0]?.toLowerCase();switch(r){case"version":case"-version":return{stdout:`{ npm: '${
|
|
128
|
+
`,exitCode:127};if(y(e,["--version","-v"]))return{stdout:`${Qt}
|
|
129
|
+
`,exitCode:0};let r=e[0]?.toLowerCase();switch(r){case"version":case"-version":return{stdout:`{ npm: '${Qt}', node: '${Js}', v8: '10.2.154.26' }
|
|
251
130
|
`,exitCode:0};case"install":case"i":case"add":return{stderr:`npm warn: package installation is not available in the virtual runtime.
|
|
252
131
|
npm warn: This environment simulates npm CLI behaviour only.
|
|
253
132
|
`,exitCode:1};case"run":case"exec":case"x":return{stderr:`npm error: script execution is not available in the virtual runtime.
|
|
254
133
|
`,exitCode:1};case"init":return{stdout:`Wrote to /home/user/package.json
|
|
255
134
|
`,exitCode:0};case"list":case"ls":return{stdout:`${r==="ls"||r==="list"?"virtual-env@1.0.0":""}
|
|
256
135
|
\u2514\u2500\u2500 (empty)
|
|
257
|
-
`,exitCode:0};case"help":case void 0:return{stdout:`${[`npm ${
|
|
136
|
+
`,exitCode:0};case"help":case void 0:return{stdout:`${[`npm ${Qt}`,"","Usage: npm <command>","","Commands:"," install (not available in virtual runtime)"," run (not available in virtual runtime)"," exec (not available in virtual runtime)"," list List installed packages"," version Print versions"," --version Print npm version"].join(`
|
|
258
137
|
`)}
|
|
259
138
|
`,exitCode:0};default:return{stderr:`npm error: unknown command: ${r}
|
|
260
|
-
`,exitCode:1}}}},
|
|
139
|
+
`,exitCode:1}}}},tn={name:"npx",description:"Node.js package runner (virtual)",category:"system",params:["<package> [args]"],run:({args:e,shell:t})=>t.packageManager.isInstalled("npm")?y(e,["--version"])?{stdout:`${Qt}
|
|
261
140
|
`,exitCode:0}:{stderr:`npx: package execution is not available in the virtual runtime.
|
|
262
141
|
`,exitCode:1}:{stderr:`bash: npx: command not found
|
|
263
142
|
Hint: install it with: apt install npm
|
|
264
|
-
`,exitCode:127}};var
|
|
143
|
+
`,exitCode:127}};var en={name:"passwd",description:"Change user password",category:"users",params:["[username]"],run:async({authUser:e,args:t,shell:r,stdin:n})=>{let i=t[0]??e;if(e!=="root"&&e!==i)return{stderr:"passwd: permission denied",exitCode:1};if(!r.users.listUsers().includes(i))return{stderr:`passwd: user '${i}' does not exist`,exitCode:1};if(n!==void 0&&n.trim().length>0){let s=n.trim().split(`
|
|
265
144
|
`)[0];return await r.users.setPassword(i,s),{stdout:`passwd: password updated successfully
|
|
266
|
-
`,exitCode:0}}return{passwordChallenge:{prompt:"New password: ",confirmPrompt:"Retype new password: ",action:"passwd",targetUsername:i},exitCode:0}}};var
|
|
267
|
-
`),exitCode:0}}};function
|
|
268
|
-
`,i+=2;continue;case"t":n+=" ",i+=2;continue;case"r":n+="\r",i+=2;continue;case"\\":n+="\\",i+=2;continue;case"a":n+="\x07",i+=2;continue;case"b":n+="\b",i+=2;continue;case"f":n+="\f",i+=2;continue;case"v":n+="\v",i+=2;continue;default:n+=e[i],i++;continue}if(e[i]==="%"&&i+1<e.length){let s=i+1,o=!1;e[s]==="-"&&(o=!0,s++);let a=!1;e[s]==="0"&&(a=!0,s++);let l=0;for(;s<e.length&&/\d/.test(e[s]);)l=l*10+parseInt(e[s],10),s++;let c=-1;if(e[s]===".")for(s++,c=0;s<e.length&&/\d/.test(e[s]);)c=c*10+parseInt(e[s],10),s++;let u=e[s],d=t[r++]??"",m=(p,
|
|
145
|
+
`,exitCode:0}}return{passwordChallenge:{prompt:"New password: ",confirmPrompt:"Retype new password: ",action:"passwd",targetUsername:i},exitCode:0}}};var rn={name:"ping",description:"Send ICMP ECHO_REQUEST (mock)",category:"network",params:["[-c <count>] <host>"],run:({args:e})=>{let{flagsWithValues:t,positionals:r}=nt(e,{flagsWithValue:["-c","-i","-W"]}),n=r[0]??"localhost",i=t.get("-c"),s=i?Math.max(1,parseInt(i,10)||4):4,o=[`PING ${n}: 56 data bytes`];for(let a=0;a<s;a++){let l=(Math.random()*10+1).toFixed(3);o.push(`64 bytes from ${n}: icmp_seq=${a} ttl=64 time=${l} ms`)}return o.push(`--- ${n} ping statistics ---`),o.push(`${s} packets transmitted, ${s} received, 0% packet loss`),{stdout:o.join(`
|
|
146
|
+
`),exitCode:0}}};function Qs(e,t){let r=0,n="",i=0;for(;i<e.length;){if(e[i]==="\\"&&i+1<e.length)switch(e[i+1]){case"n":n+=`
|
|
147
|
+
`,i+=2;continue;case"t":n+=" ",i+=2;continue;case"r":n+="\r",i+=2;continue;case"\\":n+="\\",i+=2;continue;case"a":n+="\x07",i+=2;continue;case"b":n+="\b",i+=2;continue;case"f":n+="\f",i+=2;continue;case"v":n+="\v",i+=2;continue;default:n+=e[i],i++;continue}if(e[i]==="%"&&i+1<e.length){let s=i+1,o=!1;e[s]==="-"&&(o=!0,s++);let a=!1;e[s]==="0"&&(a=!0,s++);let l=0;for(;s<e.length&&/\d/.test(e[s]);)l=l*10+parseInt(e[s],10),s++;let c=-1;if(e[s]===".")for(s++,c=0;s<e.length&&/\d/.test(e[s]);)c=c*10+parseInt(e[s],10),s++;let u=e[s],d=t[r++]??"",m=(p,w=" ")=>{if(l<=0||p.length>=l)return p;let g=w.repeat(l-p.length);return o?p+g:g+p};switch(u){case"s":{let p=String(d);c>=0&&(p=p.slice(0,c)),n+=m(p);break}case"d":case"i":n+=m(String(parseInt(d,10)||0),a?"0":" ");break;case"f":{let p=c>=0?c:6;n+=m((parseFloat(d)||0).toFixed(p));break}case"o":n+=m((parseInt(d,10)||0).toString(8),a?"0":" ");break;case"x":n+=m((parseInt(d,10)||0).toString(16),a?"0":" ");break;case"X":n+=m((parseInt(d,10)||0).toString(16).toUpperCase(),a?"0":" ");break;case"%":n+="%",r--;break;default:n+=e[i],i++;continue}i=s+1;continue}n+=e[i],i++}return n}var nn={name:"printf",description:"Format and print data",category:"shell",params:["<format> [args...]"],run:({args:e})=>{let t=e[0];return t?{stdout:Qs(t,e.slice(1)),exitCode:0}:{stderr:"printf: missing format string",exitCode:1}}};var sn={name:"ps",description:"Report process status",category:"system",params:["[-a] [-u] [-x] [aux]"],run:({authUser:e,shell:t,args:r})=>{let n=t.users.listActiveSessions(),i=y(r,["-u"])||r.includes("u")||r.includes("aux")||r.includes("au"),s=y(r,["-a","-x"])||r.includes("a")||r.includes("aux");if(i){let u=["USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND"],d=1e3;for(let m of n){let p=m.username.padEnd(10).slice(0,10),w=(Math.random()*.5).toFixed(1),g=Math.floor(Math.random()*2e4+5e3),f=Math.floor(Math.random()*5e3+1e3);u.push(`${p} ${String(d).padStart(6)} 0.0 ${w.padStart(4)} ${String(g).padStart(6)} ${String(f).padStart(5)} ${m.tty.padEnd(8)} Ss 00:00 0:00 bash`),d++}return u.push(`root ${String(d).padStart(6)} 0.0 0.0 0 0 ? S 00:00 0:00 ps`),{stdout:u.join(`
|
|
269
148
|
`),exitCode:0}}let a=[" PID TTY TIME CMD"],l=1e3;for(let c of n)!s&&c.username!==e||(a.push(`${String(l).padStart(5)} ${c.tty.padEnd(12)} 00:00:00 ${c.username===e?"bash":`bash (${c.username})`}`),l++);return a.push(`${String(l).padStart(5)} pts/0 00:00:00 ps`),{stdout:a.join(`
|
|
270
|
-
`),exitCode:0}}};var
|
|
271
|
-
`],["curdir","."],["pardir",".."]]);return r.__methods__={getcwd:()=>e,getenv:n=>typeof n=="string"?process.env[n]??h:h,path:j([["join",h],["exists",h],["dirname",h],["basename",h]]),listdir:()=>[]},t.set("__builtins__",h),t.set("__name__","__main__"),t.set("__cwd__",e),t}function
|
|
272
|
-
`],["name","posix"]]);return r._cwd=e,t._cwd=e,r.path=t,r}function
|
|
149
|
+
`),exitCode:0}}};var on={name:"pwd",description:"Print working directory",category:"navigation",params:[],run:({cwd:e})=>({stdout:e,exitCode:0})};var Ys="Python 3.11.2";var Yt="3.11.2 (default, Mar 13 2023, 12:18:29) [GCC 12.2.0]",h={__pytype__:"none"};function j(e=[]){return{__pytype__:"dict",data:new Map(e)}}function ge(e,t,r=1){return{__pytype__:"range",start:e,stop:t,step:r}}function O(e){return!!e&&typeof e=="object"&&!Array.isArray(e)&&e.__pytype__==="dict"}function Mt(e){return!!e&&typeof e=="object"&&!Array.isArray(e)&&e.__pytype__==="range"}function pt(e){return!!e&&typeof e=="object"&&!Array.isArray(e)&&e.__pytype__==="func"}function ye(e){return!!e&&typeof e=="object"&&!Array.isArray(e)&&e.__pytype__==="class"}function Rt(e){return!!e&&typeof e=="object"&&!Array.isArray(e)&&e.__pytype__==="instance"}function wt(e){return!!e&&typeof e=="object"&&!Array.isArray(e)&&e.__pytype__==="none"}function Q(e){return e===null||wt(e)?"None":e===!0?"True":e===!1?"False":typeof e=="number"?Number.isInteger(e)?String(e):e.toPrecision(12).replace(/\.?0+$/,""):typeof e=="string"?`'${e.replace(/'/g,"\\'")}'`:Array.isArray(e)?`[${e.map(Q).join(", ")}]`:O(e)?`{${[...e.data.entries()].map(([t,r])=>`'${t}': ${Q(r)}`).join(", ")}}`:Mt(e)?`range(${e.start}, ${e.stop}${e.step!==1?`, ${e.step}`:""})`:pt(e)?`<function ${e.name} at 0x...>`:ye(e)?`<class '${e.name}'>`:Rt(e)?`<${e.cls.name} object at 0x...>`:String(e)}function I(e){return e===null||wt(e)?"None":e===!0?"True":e===!1?"False":typeof e=="number"?Number.isInteger(e)?String(e):e.toPrecision(12).replace(/\.?0+$/,""):typeof e=="string"?e:Array.isArray(e)?`[${e.map(Q).join(", ")}]`:O(e)?`{${[...e.data.entries()].map(([t,r])=>`'${t}': ${Q(r)}`).join(", ")}}`:Mt(e)?`range(${e.start}, ${e.stop}${e.step!==1?`, ${e.step}`:""})`:Q(e)}function ot(e){return e===null||wt(e)?!1:typeof e=="boolean"?e:typeof e=="number"?e!==0:typeof e=="string"||Array.isArray(e)?e.length>0:O(e)?e.data.size>0:Mt(e)?ln(e)>0:!0}function ln(e){if(e.step===0)return 0;let t=Math.ceil((e.stop-e.start)/e.step);return Math.max(0,t)}function Xs(e){let t=[];for(let r=e.start;(e.step>0?r<e.stop:r>e.stop)&&(t.push(r),!(t.length>1e4));r+=e.step);return t}function J(e){if(Array.isArray(e))return e;if(typeof e=="string")return[...e];if(Mt(e))return Xs(e);if(O(e))return[...e.data.keys()];throw new W("TypeError",`'${vt(e)}' object is not iterable`)}function vt(e){return e===null||wt(e)?"NoneType":typeof e=="boolean"?"bool":typeof e=="number"?Number.isInteger(e)?"int":"float":typeof e=="string"?"str":Array.isArray(e)?"list":O(e)?"dict":Mt(e)?"range":pt(e)?"function":ye(e)?"type":Rt(e)?e.cls.name:"object"}var W=class{constructor(t,r){this.type=t;this.message=r}type;message;toString(){return`${this.type}: ${this.message}`}},kt=class{constructor(t){this.value=t}value},Dt=class{},_t=class{},Lt=class{constructor(t){this.code=t}code};function ti(e){let t=new Map,r=j([["sep","/"],["linesep",`
|
|
150
|
+
`],["curdir","."],["pardir",".."]]);return r.__methods__={getcwd:()=>e,getenv:n=>typeof n=="string"?process.env[n]??h:h,path:j([["join",h],["exists",h],["dirname",h],["basename",h]]),listdir:()=>[]},t.set("__builtins__",h),t.set("__name__","__main__"),t.set("__cwd__",e),t}function ei(e){let t=j([["sep","/"],["curdir","."]]),r=j([["sep","/"],["linesep",`
|
|
151
|
+
`],["name","posix"]]);return r._cwd=e,t._cwd=e,r.path=t,r}function ri(){return j([["version",Yt],["version_info",j([["major",3],["minor",11],["micro",2]].map(([e,t])=>[e,t]))],["platform","linux"],["executable","/usr/bin/python3"],["prefix","/usr"],["path",["/usr/lib/python3.11","/usr/lib/python3.11/lib-dynload"]],["argv",[""]],["maxsize",9007199254740991]])}function ni(){return j([["pi",Math.PI],["e",Math.E],["tau",Math.PI*2],["inf",1/0],["nan",NaN],["sqrt",h],["floor",h],["ceil",h],["log",h],["pow",h],["sin",h],["cos",h],["tan",h],["fabs",h],["factorial",h]])}function si(){return j([["dumps",h],["loads",h]])}function ii(){return j([["match",h],["search",h],["findall",h],["sub",h],["split",h],["compile",h]])}var an={os:ei,sys:()=>ri(),math:()=>ni(),json:()=>si(),re:()=>ii(),random:()=>j([["random",h],["randint",h],["choice",h],["shuffle",h]]),time:()=>j([["time",h],["sleep",h],["ctime",h]]),datetime:()=>j([["datetime",h],["date",h],["timedelta",h]]),collections:()=>j([["Counter",h],["defaultdict",h],["OrderedDict",h]]),itertools:()=>j([["chain",h],["product",h],["combinations",h],["permutations",h]]),functools:()=>j([["reduce",h],["partial",h],["lru_cache",h]]),string:()=>j([["ascii_letters","abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"],["digits","0123456789"],["punctuation","!\"#$%&'()*+,-./:;<=>?@[\\]^_`{|}~"]])},Xt=class{constructor(t){this.cwd=t}cwd;output=[];stderr=[];modules=new Map;getOutput(){return this.output.join(`
|
|
273
152
|
`)+(this.output.length?`
|
|
274
153
|
`:"")}getStderr(){return this.stderr.join(`
|
|
275
154
|
`)+(this.stderr.length?`
|
|
276
155
|
`:"")}splitArgs(t){let r=[],n=0,i="",s=!1,o="";for(let a=0;a<t.length;a++){let l=t[a];s?(i+=l,l===o&&t[a-1]!=="\\"&&(s=!1)):l==='"'||l==="'"?(s=!0,o=l,i+=l):"([{".includes(l)?(n++,i+=l):")]}".includes(l)?(n--,i+=l):l===","&&n===0?(r.push(i.trim()),i=""):i+=l}return i.trim()&&r.push(i.trim()),r}pyEval(t,r){if(t=t.trim(),!t||t==="None")return h;if(t==="True")return!0;if(t==="False")return!1;if(t==="...")return h;if(/^-?\d+$/.test(t))return parseInt(t,10);if(/^-?\d+\.\d*$/.test(t))return parseFloat(t);if(/^0x[0-9a-fA-F]+$/.test(t))return parseInt(t,16);if(/^0o[0-7]+$/.test(t))return parseInt(t.slice(2),8);if(/^('''[\s\S]*'''|"""[\s\S]*""")$/.test(t))return t.slice(3,-3);if(/^(['"])(.*)\1$/s.test(t))return t.slice(1,-1).replace(/\\n/g,`
|
|
277
|
-
`).replace(/\\t/g," ").replace(/\\r/g,"\r").replace(/\\\\/g,"\\").replace(/\\'/g,"'").replace(/\\"/g,'"');let n=t.match(/^f(['"])([\s\S]*)\1$/);if(n){let c=n[2];return c=c.replace(/\{([^{}]+)\}/g,(u,d)=>{try{return I(this.pyEval(d.trim(),r))}catch{return`{${d}}`}}),c}let i=t.match(/^b(['"])(.*)\1$/s);if(i)return i[2];if(t.startsWith("[")&&t.endsWith("]")){let c=t.slice(1,-1).trim();if(!c)return[];let u=c.match(/^(.+?)\s+for\s+(\w+)\s+in\s+(.+?)(?:\s+if\s+(.+))?$/);if(u){let[,d,m,p,
|
|
278
|
-
`);case"join":return J(n[0]??[]).map(I).join(t);case"replace":return t.replaceAll(I(n[0]??""),I(n[1]??""));case"startswith":return t.startsWith(I(n[0]??""));case"endswith":return t.endsWith(I(n[0]??""));case"find":return t.indexOf(I(n[0]??""));case"index":{let s=t.indexOf(I(n[0]??""));if(s===-1)throw new W("ValueError","substring not found");return s}case"count":return t.split(I(n[0]??"")).length-1;case"format":return this.pyStringFormat(t,n);case"encode":return t;case"decode":return t;case"isdigit":return/^\d+$/.test(t);case"isalpha":return/^[a-zA-Z]+$/.test(t);case"isalnum":return/^[a-zA-Z0-9]+$/.test(t);case"isspace":return/^\s+$/.test(t);case"isupper":return t===t.toUpperCase()&&t!==t.toLowerCase();case"islower":return t===t.toLowerCase()&&t!==t.toUpperCase();case"center":{let s=n[0]??0,o=I(n[1]??" ");return t.padStart(Math.floor((s+t.length)/2),o).padEnd(s,o)}case"ljust":return t.padEnd(n[0]??0,I(n[1]??" "));case"rjust":return t.padStart(n[0]??0,I(n[1]??" "));case"zfill":return t.padStart(n[0]??0,"0");case"title":return t.replace(/\b\w/g,s=>s.toUpperCase());case"capitalize":return t[0]?.toUpperCase()+t.slice(1).toLowerCase();case"swapcase":return[...t].map(s=>s===s.toUpperCase()?s.toLowerCase():s.toUpperCase()).join("")}if(Array.isArray(t))switch(r){case"append":return t.push(n[0]??h),h;case"extend":for(let s of J(n[0]??[]))t.push(s);return h;case"insert":return t.splice(n[0]??0,0,n[1]??h),h;case"pop":{let s=n[0]!==void 0?n[0]:-1,o=s<0?t.length+s:s;return t.splice(o,1)[0]??h}case"remove":{let s=t.findIndex(o=>
|
|
279
|
-
`.replace(/\\n/g,"")),h;case"input":return this.output.push(I(r[0]??"")),"";case"int":{if(r.length===0)return 0;let i=r[1]??10,s=parseInt(I(r[0]??0),i);return Number.isNaN(s)?(()=>{throw new W("ValueError","invalid literal for int()")})():s}case"float":{if(r.length===0)return 0;let i=parseFloat(I(r[0]??0));return Number.isNaN(i)?(()=>{throw new W("ValueError","could not convert to float")})():i}case"str":return r.length===0?"":I(r[0]??h);case"bool":return r.length===0?!1:ot(r[0]??h);case"list":return r.length===0?[]:J(r[0]??[]);case"tuple":return r.length===0?[]:J(r[0]??[]);case"set":return r.length===0?[]:[...new Set(J(r[0]??[]).map(
|
|
280
|
-
`);this.execLines(n,0,r)}execLines(t,r,n){let i=r;for(;i<t.length;){let s=t[i];if(!s.trim()||s.trim().startsWith("#")){i++;continue}i=this.execStatement(t,i,n)}return i}execBlock(t,r){try{this.execLines(t,0,r)}catch(n){if(n instanceof kt)return n.value;throw n}return h}getIndent(t){let r=0;for(let n of t)if(n===" ")r++;else if(n===" ")r+=4;else break;return r}collectBlock(t,r,n){let i=[];for(let s=r;s<t.length;s++){let o=t[s];if(!o.trim()){i.push("");continue}if(this.getIndent(o)<=n)break;i.push(o.slice(n+4))}return i}execStatement(t,r,n){let i=t[r],s=i.trim(),o=this.getIndent(i);if(s==="pass")return r+1;if(s==="break")throw new Dt;if(s==="continue")throw new _t;let a=s.match(/^return(?:\s+(.+))?$/);if(a)throw new kt(a[1]?this.pyEval(a[1],n):h);let l=s.match(/^raise(?:\s+(.+))?$/);if(l){if(l[1]){let b=this.pyEval(l[1],n);throw new W(typeof b=="string"?b:vt(b),I(b))}throw new W("RuntimeError","")}let c=s.match(/^assert\s+(.+?)(?:,\s*(.+))?$/);if(c){if(!ot(this.pyEval(c[1],n)))throw new W("AssertionError",c[2]?I(this.pyEval(c[2],n)):"");return r+1}let u=s.match(/^del\s+(.+)$/);if(u)return n.delete(u[1].trim()),r+1;let d=s.match(/^import\s+(\w+)(?:\s+as\s+(\w+))?$/);if(d){let[,b,P]=d,
|
|
156
|
+
`).replace(/\\t/g," ").replace(/\\r/g,"\r").replace(/\\\\/g,"\\").replace(/\\'/g,"'").replace(/\\"/g,'"');let n=t.match(/^f(['"])([\s\S]*)\1$/);if(n){let c=n[2];return c=c.replace(/\{([^{}]+)\}/g,(u,d)=>{try{return I(this.pyEval(d.trim(),r))}catch{return`{${d}}`}}),c}let i=t.match(/^b(['"])(.*)\1$/s);if(i)return i[2];if(t.startsWith("[")&&t.endsWith("]")){let c=t.slice(1,-1).trim();if(!c)return[];let u=c.match(/^(.+?)\s+for\s+(\w+)\s+in\s+(.+?)(?:\s+if\s+(.+))?$/);if(u){let[,d,m,p,w]=u,g=J(this.pyEval(p.trim(),r)),f=[];for(let $ of g){let E=new Map(r);E.set(m,$),!(w&&!ot(this.pyEval(w,E)))&&f.push(this.pyEval(d.trim(),E))}return f}return this.splitArgs(c).map(d=>this.pyEval(d,r))}if(t.startsWith("(")&&t.endsWith(")")){let c=t.slice(1,-1).trim();if(!c)return[];let u=this.splitArgs(c);return u.length===1&&!c.endsWith(",")?this.pyEval(u[0],r):u.map(d=>this.pyEval(d,r))}if(t.startsWith("{")&&t.endsWith("}")){let c=t.slice(1,-1).trim();if(!c)return j();let u=j();for(let d of this.splitArgs(c)){let m=d.indexOf(":");if(m===-1)continue;let p=I(this.pyEval(d.slice(0,m).trim(),r)),w=this.pyEval(d.slice(m+1).trim(),r);u.data.set(p,w)}return u}let s=t.match(/^not\s+(.+)$/);if(s)return!ot(this.pyEval(s[1],r));let o=[["or"],["and"],["in","not in","is not","is","==","!=","<=",">=","<",">"],["+","-"],["**"],["*","//","/","%"]];for(let c of o){let u=this.tryBinaryOp(t,c,r);if(u!==void 0)return u}if(t.startsWith("-")){let c=this.pyEval(t.slice(1),r);if(typeof c=="number")return-c}if(process.env.PY_DEBUG&&console.error("eval:",JSON.stringify(t)),t.endsWith("]")&&!t.startsWith("[")){let c=this.findMatchingBracket(t,"[");if(c!==-1){let u=this.pyEval(t.slice(0,c),r),d=t.slice(c+1,-1);return this.subscript(u,d,r)}}let a=t.match(/^([A-Za-z_][A-Za-z0-9_]*)\s*\(([\s\S]*)\)$/);if(a){let[,c,u]=a,d=(u?.trim()?this.splitArgs(u):[]).map(m=>this.pyEval(m,r));return this.callBuiltin(c,d,r)}let l=this.findDotAccess(t);if(l){let{objExpr:c,attr:u,callPart:d}=l,m=this.pyEval(c,r);if(d!==void 0){let p=d.slice(1,-1),w=p.trim()?this.splitArgs(p).map(g=>this.pyEval(g,r)):[];return this.callMethod(m,u,w,r)}return this.getAttr(m,u,r)}if(/^[A-Za-z_][A-Za-z0-9_]*$/.test(t)){if(r.has(t))return r.get(t);throw new W("NameError",`name '${t}' is not defined`)}if(/^[A-Za-z_][A-Za-z0-9_.]+$/.test(t)){let c=t.split("."),u=r.get(c[0])??(()=>{throw new W("NameError",`name '${c[0]}' is not defined`)})();for(let d of c.slice(1))u=this.getAttr(u,d,r);return u}return h}findMatchingBracket(t,r){let n=r==="["?"]":r==="("?")":"}",i=0;for(let s=t.length-1;s>=0;s--)if(t[s]===n&&i++,t[s]===r&&(i--,i===0))return s;return-1}findDotAccess(t){let r=0,n=!1,i="";for(let s=t.length-1;s>0;s--){let o=t[s];if(n){o===i&&t[s-1]!=="\\"&&(n=!1);continue}if(o==='"'||o==="'"){n=!0,i=o;continue}if(")]}".includes(o)){r++;continue}if("([{".includes(o)){r--;continue}if(r!==0||o!==".")continue;let a=t.slice(0,s).trim(),c=t.slice(s+1).match(/^(\w+)(\([\s\S]*\))?$/);if(c&&!/^-?\d+$/.test(a))return{objExpr:a,attr:c[1],callPart:c[2]}}return null}tryBinaryOp(t,r,n){let i=0,s=!1,o="";for(let a=t.length-1;a>=0;a--){let l=t[a];if(s){l===o&&t[a-1]!=="\\"&&(s=!1);continue}if(l==='"'||l==="'"){s=!0,o=l;continue}if(")]}".includes(l)){i++;continue}if("([{".includes(l)){i--;continue}if(i===0){for(let c of r)if(t.slice(a,a+c.length)===c){if(c==="*"&&(t[a+1]==="*"||t[a-1]==="*"))continue;let u=t[a-1],d=t[a+c.length];if(/^[a-z]/.test(c)&&(u&&/\w/.test(u)||d&&/\w/.test(d)))continue;let p=t.slice(0,a).trim(),w=t.slice(a+c.length).trim();if(!p||!w)continue;return this.applyBinaryOp(c,p,w,n)}}}}applyBinaryOp(t,r,n,i){if(t==="and"){let a=this.pyEval(r,i);return ot(a)?this.pyEval(n,i):a}if(t==="or"){let a=this.pyEval(r,i);return ot(a)?a:this.pyEval(n,i)}let s=this.pyEval(r,i),o=this.pyEval(n,i);switch(t){case"+":return typeof s=="string"&&typeof o=="string"?s+o:Array.isArray(s)&&Array.isArray(o)?[...s,...o]:s+o;case"-":return s-o;case"*":if(typeof s=="string"&&typeof o=="number")return s.repeat(o);if(Array.isArray(s)&&typeof o=="number"){let a=[];for(let l=0;l<o;l++)a.push(...s);return a}return s*o;case"/":{if(o===0)throw new W("ZeroDivisionError","division by zero");return s/o}case"//":{if(o===0)throw new W("ZeroDivisionError","integer division or modulo by zero");return Math.floor(s/o)}case"%":{if(typeof s=="string")return this.pyStringFormat(s,Array.isArray(o)?o:[o]);if(o===0)throw new W("ZeroDivisionError","integer division or modulo by zero");return s%o}case"**":return s**o;case"==":return Q(s)===Q(o)||s===o;case"!=":return Q(s)!==Q(o)&&s!==o;case"<":return s<o;case"<=":return s<=o;case">":return s>o;case">=":return s>=o;case"in":return this.pyIn(o,s);case"not in":return!this.pyIn(o,s);case"is":return s===o||wt(s)&&wt(o);case"is not":return!(s===o||wt(s)&&wt(o))}return h}pyIn(t,r){return typeof t=="string"?typeof r=="string"&&t.includes(r):Array.isArray(t)?t.some(n=>Q(n)===Q(r)):O(t)?t.data.has(I(r)):!1}subscript(t,r,n){if(r.includes(":")){let s=r.split(":").map(l=>l.trim()),o=s[0]?this.pyEval(s[0],n):void 0,a=s[1]?this.pyEval(s[1],n):void 0;return typeof t=="string"||Array.isArray(t)?t.slice(o,a):h}let i=this.pyEval(r,n);if(Array.isArray(t)){let s=i;return s<0&&(s=t.length+s),t[s]??h}if(typeof t=="string"){let s=i;return s<0&&(s=t.length+s),t[s]??h}if(O(t))return t.data.get(I(i))??h;throw new W("TypeError",`'${vt(t)}' is not subscriptable`)}getAttr(t,r,n){return O(t)?t.data.has(r)?t.data.get(r):r==="path"&&t.path?t.path:h:Rt(t)?t.attrs.get(r)??h:typeof t=="string"?{__class__:{__pytype__:"class",name:"str"}}[r]??h:h}callMethod(t,r,n,i){if(typeof t=="string")switch(r){case"upper":return t.toUpperCase();case"lower":return t.toLowerCase();case"strip":return(n[0]?t.replace(new RegExp(`[${n[0]}]+`,"g"),""):t).trim();case"lstrip":return t.trimStart();case"rstrip":return t.trimEnd();case"split":return t.split(typeof n[0]=="string"?n[0]:/\s+/).filter((s,o)=>o>0||s!=="");case"splitlines":return t.split(`
|
|
157
|
+
`);case"join":return J(n[0]??[]).map(I).join(t);case"replace":return t.replaceAll(I(n[0]??""),I(n[1]??""));case"startswith":return t.startsWith(I(n[0]??""));case"endswith":return t.endsWith(I(n[0]??""));case"find":return t.indexOf(I(n[0]??""));case"index":{let s=t.indexOf(I(n[0]??""));if(s===-1)throw new W("ValueError","substring not found");return s}case"count":return t.split(I(n[0]??"")).length-1;case"format":return this.pyStringFormat(t,n);case"encode":return t;case"decode":return t;case"isdigit":return/^\d+$/.test(t);case"isalpha":return/^[a-zA-Z]+$/.test(t);case"isalnum":return/^[a-zA-Z0-9]+$/.test(t);case"isspace":return/^\s+$/.test(t);case"isupper":return t===t.toUpperCase()&&t!==t.toLowerCase();case"islower":return t===t.toLowerCase()&&t!==t.toUpperCase();case"center":{let s=n[0]??0,o=I(n[1]??" ");return t.padStart(Math.floor((s+t.length)/2),o).padEnd(s,o)}case"ljust":return t.padEnd(n[0]??0,I(n[1]??" "));case"rjust":return t.padStart(n[0]??0,I(n[1]??" "));case"zfill":return t.padStart(n[0]??0,"0");case"title":return t.replace(/\b\w/g,s=>s.toUpperCase());case"capitalize":return t[0]?.toUpperCase()+t.slice(1).toLowerCase();case"swapcase":return[...t].map(s=>s===s.toUpperCase()?s.toLowerCase():s.toUpperCase()).join("")}if(Array.isArray(t))switch(r){case"append":return t.push(n[0]??h),h;case"extend":for(let s of J(n[0]??[]))t.push(s);return h;case"insert":return t.splice(n[0]??0,0,n[1]??h),h;case"pop":{let s=n[0]!==void 0?n[0]:-1,o=s<0?t.length+s:s;return t.splice(o,1)[0]??h}case"remove":{let s=t.findIndex(o=>Q(o)===Q(n[0]??h));return s!==-1&&t.splice(s,1),h}case"index":{let s=t.findIndex(o=>Q(o)===Q(n[0]??h));if(s===-1)throw new W("ValueError","is not in list");return s}case"count":return t.filter(s=>Q(s)===Q(n[0]??h)).length;case"sort":return t.sort((s,o)=>typeof s=="number"&&typeof o=="number"?s-o:I(s).localeCompare(I(o))),h;case"reverse":return t.reverse(),h;case"copy":return[...t];case"clear":return t.splice(0),h}if(O(t))switch(r){case"keys":return[...t.data.keys()];case"values":return[...t.data.values()];case"items":return[...t.data.entries()].map(([s,o])=>[s,o]);case"get":return t.data.get(I(n[0]??""))??n[1]??h;case"update":{if(O(n[0]??h))for(let[s,o]of n[0].data)t.data.set(s,o);return h}case"pop":{let s=I(n[0]??""),o=t.data.get(s)??n[1]??h;return t.data.delete(s),o}case"clear":return t.data.clear(),h;case"copy":return j([...t.data.entries()]);case"setdefault":{let s=I(n[0]??"");return t.data.has(s)||t.data.set(s,n[1]??h),t.data.get(s)??h}}if(O(t)&&t.data.has("name")&&t.data.get("name")==="posix")switch(r){case"getcwd":return this.cwd;case"getenv":return typeof n[0]=="string"?process.env[n[0]]??n[1]??h:h;case"listdir":return[];case"path":return t}if(O(t))switch(r){case"join":return n.map(I).join("/").replace(/\/+/g,"/");case"exists":return!1;case"dirname":return I(n[0]??"").split("/").slice(0,-1).join("/")||"/";case"basename":return I(n[0]??"").split("/").pop()??"";case"abspath":return I(n[0]??"");case"splitext":{let s=I(n[0]??""),o=s.lastIndexOf(".");return o>0?[s.slice(0,o),s.slice(o)]:[s,""]}case"isfile":return!1;case"isdir":return!1}if(O(t)&&t.data.has("version")&&t.data.get("version")===Yt&&r==="exit")throw new Lt(n[0]??0);if(O(t)){let s={sqrt:Math.sqrt,floor:Math.floor,ceil:Math.ceil,fabs:Math.abs,log:Math.log,log2:Math.log2,log10:Math.log10,sin:Math.sin,cos:Math.cos,tan:Math.tan,asin:Math.asin,acos:Math.acos,atan:Math.atan,atan2:Math.atan2,pow:Math.pow,exp:Math.exp,hypot:Math.hypot};if(r in s){let o=s[r];return o(...n.map(a=>a))}if(r==="factorial"){let o=n[0]??0,a=1;for(;o>1;)a*=o--;return a}if(r==="gcd"){let o=Math.abs(n[0]??0),a=Math.abs(n[1]??0);for(;a;)[o,a]=[a,o%a];return o}}if(O(t)){if(r==="dumps"){let s=O(n[1]??h)?n[1]:void 0,o=s?s.data.get("indent"):void 0;return JSON.stringify(this.pyToJs(n[0]??h),null,o)}if(r==="loads")return this.jsToPy(JSON.parse(I(n[0]??"")))}if(Rt(t)){let s=t.attrs.get(r)??t.cls.methods.get(r)??h;if(pt(s)){let o=new Map(s.closure);return o.set("self",t),s.params.slice(1).forEach((a,l)=>o.set(a,n[l]??h)),this.execBlock(s.body,o)}}throw new W("AttributeError",`'${vt(t)}' object has no attribute '${r}'`)}pyStringFormat(t,r){let n=0;return t.replace(/%([diouxXeEfFgGcrs%])/g,(i,s)=>{if(s==="%")return"%";let o=r[n++];switch(s){case"d":case"i":return String(Math.trunc(o));case"f":return o.toFixed(6);case"s":return I(o??h);case"r":return Q(o??h);default:return String(o)}})}pyToJs(t){return wt(t)?null:O(t)?Object.fromEntries([...t.data.entries()].map(([r,n])=>[r,this.pyToJs(n)])):Array.isArray(t)?t.map(r=>this.pyToJs(r)):t}jsToPy(t){return t==null?h:typeof t=="boolean"||typeof t=="number"||typeof t=="string"?t:Array.isArray(t)?t.map(r=>this.jsToPy(r)):typeof t=="object"?j(Object.entries(t).map(([r,n])=>[r,this.jsToPy(n)])):h}callBuiltin(t,r,n){if(n.has(t)){let i=n.get(t)??h;return pt(i)?this.callFunc(i,r,n):ye(i)?this.instantiate(i,r,n):i}switch(t){case"print":return this.output.push(r.map(I).join(" ")+`
|
|
158
|
+
`.replace(/\\n/g,"")),h;case"input":return this.output.push(I(r[0]??"")),"";case"int":{if(r.length===0)return 0;let i=r[1]??10,s=parseInt(I(r[0]??0),i);return Number.isNaN(s)?(()=>{throw new W("ValueError","invalid literal for int()")})():s}case"float":{if(r.length===0)return 0;let i=parseFloat(I(r[0]??0));return Number.isNaN(i)?(()=>{throw new W("ValueError","could not convert to float")})():i}case"str":return r.length===0?"":I(r[0]??h);case"bool":return r.length===0?!1:ot(r[0]??h);case"list":return r.length===0?[]:J(r[0]??[]);case"tuple":return r.length===0?[]:J(r[0]??[]);case"set":return r.length===0?[]:[...new Set(J(r[0]??[]).map(Q))].map(i=>J(r[0]??[]).find(o=>Q(o)===i)??h);case"dict":return r.length===0?j():O(r[0]??h)?r[0]:j();case"bytes":return typeof r[0]=="string"?r[0]:I(r[0]??"");case"bytearray":return r.length===0?"":I(r[0]??"");case"type":return r.length===1?`<class '${vt(r[0]??h)}'>`:h;case"isinstance":return vt(r[0]??h)===I(r[1]??"");case"issubclass":return!1;case"callable":return pt(r[0]??h);case"hasattr":return O(r[0]??h)?r[0].data.has(I(r[1]??"")):!1;case"getattr":return O(r[0]??h)?r[0].data.get(I(r[1]??""))??r[2]??h:r[2]??h;case"setattr":return O(r[0]??h)&&r[0].data.set(I(r[1]??""),r[2]??h),h;case"len":{let i=r[0]??h;if(typeof i=="string"||Array.isArray(i))return i.length;if(O(i))return i.data.size;if(Mt(i))return ln(i);throw new W("TypeError",`object of type '${vt(i)}' has no len()`)}case"range":return r.length===1?ge(0,r[0]):r.length===2?ge(r[0],r[1]):ge(r[0],r[1],r[2]);case"enumerate":{let i=r[1]??0;return J(r[0]??[]).map((s,o)=>[o+i,s])}case"zip":{let i=r.map(J),s=Math.min(...i.map(o=>o.length));return Array.from({length:s},(o,a)=>i.map(l=>l[a]??h))}case"map":{let i=r[0]??h;return J(r[1]??[]).map(s=>pt(i)?this.callFunc(i,[s],n):h)}case"filter":{let i=r[0]??h;return J(r[1]??[]).filter(s=>pt(i)?ot(this.callFunc(i,[s],n)):ot(s))}case"reduce":{let i=r[0]??h,s=J(r[1]??[]);if(s.length===0)return r[2]??h;let o=r[2]!==void 0?r[2]:s[0];for(let a of r[2]!==void 0?s:s.slice(1))o=pt(i)?this.callFunc(i,[o,a],n):h;return o}case"sorted":{let i=[...J(r[0]??[])],s=r[1]??h,o=O(s)?s.data.get("key")??h:s;return i.sort((a,l)=>{let c=pt(o)?this.callFunc(o,[a],n):a,u=pt(o)?this.callFunc(o,[l],n):l;return typeof c=="number"&&typeof u=="number"?c-u:I(c).localeCompare(I(u))}),i}case"reversed":return[...J(r[0]??[])].reverse();case"any":return J(r[0]??[]).some(ot);case"all":return J(r[0]??[]).every(ot);case"sum":return J(r[0]??[]).reduce((i,s)=>i+s,r[1]??0);case"max":return(r.length===1?J(r[0]??[]):r).reduce((s,o)=>s>=o?s:o);case"min":return(r.length===1?J(r[0]??[]):r).reduce((s,o)=>s<=o?s:o);case"abs":return Math.abs(r[0]??0);case"round":return r[1]!==void 0?parseFloat(r[0].toFixed(r[1])):Math.round(r[0]??0);case"divmod":{let i=r[0],s=r[1];return[Math.floor(i/s),i%s]}case"pow":return r[0]**r[1];case"hex":return`0x${r[0].toString(16)}`;case"oct":return`0o${r[0].toString(8)}`;case"bin":return`0b${r[0].toString(2)}`;case"ord":return I(r[0]??"").charCodeAt(0);case"chr":return String.fromCharCode(r[0]??0);case"id":return Math.floor(Math.random()*4294967295);case"hash":return typeof r[0]=="number"?r[0]:I(r[0]??"").split("").reduce((i,s)=>i*31+s.charCodeAt(0)|0,0);case"open":throw new W("PermissionError","open() not available in virtual runtime");case"repr":return Q(r[0]??h);case"iter":return r[0]??h;case"next":return Array.isArray(r[0])&&r[0].length>0?r[0].shift():r[1]??(()=>{throw new W("StopIteration","")})();case"vars":return j([...n.entries()].map(([i,s])=>[i,s]));case"globals":return j([...n.entries()].map(([i,s])=>[i,s]));case"locals":return j([...n.entries()].map(([i,s])=>[i,s]));case"dir":{if(r.length===0)return[...n.keys()];let i=r[0]??h;return typeof i=="string"?["upper","lower","strip","split","join","replace","find","format","encode","startswith","endswith","count","isdigit","isalpha","title","capitalize"]:Array.isArray(i)?["append","extend","insert","pop","remove","index","count","sort","reverse","copy","clear"]:O(i)?["keys","values","items","get","update","pop","clear","copy","setdefault"]:[]}case"Exception":case"ValueError":case"TypeError":case"KeyError":case"IndexError":case"AttributeError":case"NameError":case"RuntimeError":case"StopIteration":case"NotImplementedError":case"OSError":case"IOError":throw new W(t,I(r[0]??""));case"exec":return this.execScript(I(r[0]??""),n),h;case"eval":return this.pyEval(I(r[0]??""),n);default:throw new W("NameError",`name '${t}' is not defined`)}}callFunc(t,r,n){let i=new Map(t.closure);t.params.forEach((s,o)=>{if(s.startsWith("*")){i.set(s.slice(1),r.slice(o));return}i.set(s,r[o]??h)});try{return this.execBlock(t.body,i)}catch(s){if(s instanceof kt)return s.value;throw s}}instantiate(t,r,n){let i={__pytype__:"instance",cls:t,attrs:new Map};return t.methods.get("__init__")&&this.callMethod(i,"__init__",r,n),i}execScript(t,r){let n=t.split(`
|
|
159
|
+
`);this.execLines(n,0,r)}execLines(t,r,n){let i=r;for(;i<t.length;){let s=t[i];if(!s.trim()||s.trim().startsWith("#")){i++;continue}i=this.execStatement(t,i,n)}return i}execBlock(t,r){try{this.execLines(t,0,r)}catch(n){if(n instanceof kt)return n.value;throw n}return h}getIndent(t){let r=0;for(let n of t)if(n===" ")r++;else if(n===" ")r+=4;else break;return r}collectBlock(t,r,n){let i=[];for(let s=r;s<t.length;s++){let o=t[s];if(!o.trim()){i.push("");continue}if(this.getIndent(o)<=n)break;i.push(o.slice(n+4))}return i}execStatement(t,r,n){let i=t[r],s=i.trim(),o=this.getIndent(i);if(s==="pass")return r+1;if(s==="break")throw new Dt;if(s==="continue")throw new _t;let a=s.match(/^return(?:\s+(.+))?$/);if(a)throw new kt(a[1]?this.pyEval(a[1],n):h);let l=s.match(/^raise(?:\s+(.+))?$/);if(l){if(l[1]){let b=this.pyEval(l[1],n);throw new W(typeof b=="string"?b:vt(b),I(b))}throw new W("RuntimeError","")}let c=s.match(/^assert\s+(.+?)(?:,\s*(.+))?$/);if(c){if(!ot(this.pyEval(c[1],n)))throw new W("AssertionError",c[2]?I(this.pyEval(c[2],n)):"");return r+1}let u=s.match(/^del\s+(.+)$/);if(u)return n.delete(u[1].trim()),r+1;let d=s.match(/^import\s+(\w+)(?:\s+as\s+(\w+))?$/);if(d){let[,b,P]=d,R=an[b];if(R){let C=R(this.cwd);this.modules.set(b,C),n.set(P??b,C)}return r+1}let m=s.match(/^from\s+(\w+)\s+import\s+(.+)$/);if(m){let[,b,P]=m,R=an[b];if(R){let C=R(this.cwd);if(P?.trim()==="*")for(let[v,N]of C.data)n.set(v,N);else for(let v of P.split(",").map(N=>N.trim()))n.set(v,C.data.get(v)??h)}return r+1}let p=s.match(/^def\s+(\w+)\s*\(([^)]*)\)\s*:$/);if(p){let[,b,P]=p,R=P.split(",").map(N=>N.trim()).filter(Boolean),C=this.collectBlock(t,r+1,o),v={__pytype__:"func",name:b,params:R,body:C,closure:new Map(n)};return n.set(b,v),r+1+C.length}let w=s.match(/^class\s+(\w+)(?:\(([^)]*)\))?\s*:$/);if(w){let[,b,P]=w,R=P?P.split(",").map(G=>G.trim()):[],C=this.collectBlock(t,r+1,o),v={__pytype__:"class",name:b,methods:new Map,bases:R},N=0;for(;N<C.length;){let B=C[N].trim().match(/^def\s+(\w+)\s*\(([^)]*)\)\s*:$/);if(B){let[,Z,gt]=B,lt=gt.split(",").map(A=>A.trim()).filter(Boolean),yt=this.collectBlock(C,N+1,0);v.methods.set(Z,{__pytype__:"func",name:Z,params:lt,body:yt,closure:new Map(n)}),N+=1+yt.length}else N++}return n.set(b,v),r+1+C.length}if(s.startsWith("if ")&&s.endsWith(":")){let b=s.slice(3,-1).trim(),P=this.collectBlock(t,r+1,o),R=P.length+1;if(ot(this.pyEval(b,n))){this.execBlock(P,new Map(n).also?.(N=>{for(let[G,B]of n)N.set(G,B)})??n),this.runBlockInScope(P,n);let v=r+1+P.length;for(;v<t.length;){let N=t[v].trim();if(this.getIndent(t[v])<o||!N.startsWith("elif")&&!N.startsWith("else"))break;let G=this.collectBlock(t,v+1,o);v+=1+G.length}return v}let C=r+1+P.length;for(;C<t.length;){let v=t[C],N=v.trim();if(this.getIndent(v)!==o)break;let G=N.match(/^elif\s+(.+):$/);if(G){let B=this.collectBlock(t,C+1,o);if(ot(this.pyEval(G[1],n))){for(this.runBlockInScope(B,n),C+=1+B.length;C<t.length;){let Z=t[C].trim();if(this.getIndent(t[C])!==o||!Z.startsWith("elif")&&!Z.startsWith("else"))break;let gt=this.collectBlock(t,C+1,o);C+=1+gt.length}return C}C+=1+B.length;continue}if(N==="else:"){let B=this.collectBlock(t,C+1,o);return this.runBlockInScope(B,n),C+1+B.length}break}return C}let g=s.match(/^for\s+(.+?)\s+in\s+(.+?)\s*:$/);if(g){let[,b,P]=g,R=J(this.pyEval(P.trim(),n)),C=this.collectBlock(t,r+1,o),v=[],N=r+1+C.length;N<t.length&&t[N]?.trim()==="else:"&&(v=this.collectBlock(t,N+1,o),N+=1+v.length);let G=!1;for(let B of R){if(b.includes(",")){let Z=b.split(",").map(lt=>lt.trim()),gt=Array.isArray(B)?B:[B];Z.forEach((lt,yt)=>n.set(lt,gt[yt]??h))}else n.set(b.trim(),B);try{this.runBlockInScope(C,n)}catch(Z){if(Z instanceof Dt){G=!0;break}if(Z instanceof _t)continue;throw Z}}return!G&&v.length&&this.runBlockInScope(v,n),N}let f=s.match(/^while\s+(.+?)\s*:$/);if(f){let b=f[1],P=this.collectBlock(t,r+1,o),R=0;for(;ot(this.pyEval(b,n))&&R++<1e5;)try{this.runBlockInScope(P,n)}catch(C){if(C instanceof Dt)break;if(C instanceof _t)continue;throw C}return r+1+P.length}if(s==="try:"){let b=this.collectBlock(t,r+1,o),P=r+1+b.length,R=[],C=[],v=[];for(;P<t.length;){let G=t[P],B=G.trim();if(this.getIndent(G)!==o)break;if(B.startsWith("except")){let Z=B.match(/^except(?:\s+(\w+)(?:\s+as\s+(\w+))?)?\s*:$/),gt=Z?.[1]??null,lt=Z?.[2],yt=this.collectBlock(t,P+1,o);R.push({exc:gt,body:yt}),lt&&n.set(lt,""),P+=1+yt.length}else if(B==="else:")v=this.collectBlock(t,P+1,o),P+=1+v.length;else if(B==="finally:")C=this.collectBlock(t,P+1,o),P+=1+C.length;else break}let N=null;try{this.runBlockInScope(b,n),v.length&&this.runBlockInScope(v,n)}catch(G){if(G instanceof W){N=G;let B=!1;for(let Z of R)if(Z.exc===null||Z.exc===G.type||Z.exc==="Exception"){this.runBlockInScope(Z.body,n),B=!0;break}if(!B)throw G}else throw G}finally{C.length&&this.runBlockInScope(C,n)}return P}let $=s.match(/^with\s+(.+?)\s+as\s+(\w+)\s*:$/);if($){let b=this.collectBlock(t,r+1,o);return n.set($[2],h),this.runBlockInScope(b,n),r+1+b.length}let E=s.match(/^([A-Za-z_][A-Za-z0-9_]*)\s*(\+=|-=|\*=|\/\/=|\/=|%=|\*\*=|&=|\|=)\s*(.+)$/);if(E){let[,b,P,R]=E,C=n.get(b)??0,v=this.pyEval(R,n),N;switch(P){case"+=":N=typeof C=="string"?C+I(v):C+v;break;case"-=":N=C-v;break;case"*=":N=C*v;break;case"/=":N=C/v;break;case"//=":N=Math.floor(C/v);break;case"%=":N=C%v;break;case"**=":N=C**v;break;default:N=v}return n.set(b,N),r+1}let M=s.match(/^([A-Za-z_][A-Za-z0-9_]*)\[(.+)\]\s*=\s*(.+)$/);if(M){let[,b,P,R]=M,C=n.get(b)??h,v=this.pyEval(R,n)??h,N=this.pyEval(P,n)??h;return Array.isArray(C)?C[N]=v:O(C)&&C.data.set(I(N),v),r+1}let x=s.match(/^([A-Za-z_][A-Za-z0-9_.]+)\s*=\s*(.+)$/);if(x){let b=x[1].lastIndexOf(".");if(b!==-1){let P=x[1].slice(0,b),R=x[1].slice(b+1),C=this.pyEval(x[2],n),v=this.pyEval(P,n);return O(v)?v.data.set(R,C):Rt(v)&&v.attrs.set(R,C),r+1}}let k=s.match(/^([A-Za-z_][A-Za-z0-9_,\s]*),\s*([A-Za-z_][A-Za-z0-9_]*)\s*=\s*(.+)$/);if(k){let b=this.pyEval(k[3],n),P=s.split("=")[0].split(",").map(C=>C.trim()),R=J(b);return P.forEach((C,v)=>n.set(C,R[v]??h)),r+1}let V=s.match(/^([A-Za-z_][A-Za-z0-9_]*)\s*(?::[^=]+)?\s*=\s*(.+)$/);if(V){let[,b,P]=V;return n.set(b,this.pyEval(P,n)),r+1}try{this.pyEval(s,n)}catch(b){if(b instanceof W||b instanceof Lt)throw b}return r+1}runBlockInScope(t,r){this.execLines(t,0,r)}run(t){let r=ti(this.cwd);try{this.execScript(t,r)}catch(n){return n instanceof Lt?{stdout:this.getOutput(),stderr:this.getStderr(),exitCode:n.code}:n instanceof W?(this.stderr.push(n.toString()),{stdout:this.getOutput(),stderr:this.getStderr(),exitCode:1}):n instanceof kt?{stdout:this.getOutput(),stderr:this.getStderr(),exitCode:0}:(this.stderr.push(`RuntimeError: ${n}`),{stdout:this.getOutput(),stderr:this.getStderr(),exitCode:1})}return{stdout:this.getOutput(),stderr:this.getStderr(),exitCode:0}}},cn={name:"python3",aliases:["python"],description:"Python 3 interpreter (virtual)",category:"system",params:["[--version] [-c <code>] [-V] [file]"],run:({args:e,shell:t,cwd:r})=>{if(!t.packageManager.isInstalled("python3"))return{stderr:`bash: python3: command not found
|
|
281
160
|
Hint: install it with: apt install python3
|
|
282
|
-
`,exitCode:127};if(y(e,["--version","-V"]))return{stdout:`${
|
|
161
|
+
`,exitCode:127};if(y(e,["--version","-V"]))return{stdout:`${Ys}
|
|
283
162
|
`,exitCode:0};if(y(e,["--version-full"]))return{stdout:`${Yt}
|
|
284
163
|
`,exitCode:0};let n=e.indexOf("-c");if(n!==-1){let s=e[n+1];if(!s)return{stderr:`python3: -c requires a code argument
|
|
285
164
|
`,exitCode:1};let o=s.replace(/\\n/g,`
|
|
286
|
-
`).replace(/\\t/g," "),a=new
|
|
287
|
-
`,exitCode:2};let o=t.vfs.readFile(s),a=new
|
|
165
|
+
`).replace(/\\t/g," "),a=new Xt(r),{stdout:l,stderr:c,exitCode:u}=a.run(o);return{stdout:l||void 0,stderr:c||void 0,exitCode:u}}let i=e.find(s=>!s.startsWith("-"));if(i){let s=S(r,i);if(!t.vfs.exists(s))return{stderr:`python3: can't open file '${i}': [Errno 2] No such file or directory
|
|
166
|
+
`,exitCode:2};let o=t.vfs.readFile(s),a=new Xt(r),{stdout:l,stderr:c,exitCode:u}=a.run(o);return{stdout:l||void 0,stderr:c||void 0,exitCode:u}}return{stdout:`${Yt}
|
|
288
167
|
Type "help", "copyright", "credits" or "license" for more information.
|
|
289
|
-
>>> `,exitCode:0}}};var
|
|
168
|
+
>>> `,exitCode:0}}};var un={name:"read",description:"Read a line from stdin into variables",category:"shell",params:["[-r] [-p prompt] <var...>"],run:({args:e,stdin:t,env:r})=>{let n=e.indexOf("-p"),i=e.filter((a,l)=>a!=="-r"&&a!=="-p"&&e[l-1]!=="-p"),s=(t??"").split(`
|
|
290
169
|
`)[0]??"",o=y(e,["-r"])?s:s.replace(/\\(?:\r?\n|.)/g,a=>a[1]===`
|
|
291
|
-
`||a[1]==="\r"?"":a[1]);if(!r)return{exitCode:0};if(i.length===0)r.vars.REPLY=o;else if(i.length===1)r.vars[i[0]]=o;else{let a=o.split(/\s+/);for(let l=0;l<i.length;l++)r.vars[i[l]]=l<i.length-1?a[l]??"":a.slice(l).join(" ")}return{exitCode:0}}};var
|
|
292
|
-
`),exitCode:0};for(let r of e)if(r.includes("=")){let n=r.indexOf("=");t.vars[r.slice(0,n)]=r.slice(n+1)}return{exitCode:0}}};async function
|
|
293
|
-
`),l.stderr)return{...l,stdout:n.trim()};r=l}else if(i.type==="if"){let s=!1;if(await
|
|
294
|
-
`),s=!0}else{for(let o of i.elif)if(await
|
|
170
|
+
`||a[1]==="\r"?"":a[1]);if(!r)return{exitCode:0};if(i.length===0)r.vars.REPLY=o;else if(i.length===1)r.vars[i[0]]=o;else{let a=o.split(/\s+/);for(let l=0;l<i.length;l++)r.vars[i[l]]=l<i.length-1?a[l]??"":a.slice(l).join(" ")}return{exitCode:0}}};var dn={name:"rm",description:"Remove files or directories",category:"files",params:["[-r|-rf] <path>"],run:({authUser:e,shell:t,cwd:r,args:n})=>{if(n.length===0)return{stderr:"rm: missing operand",exitCode:1};let i=y(n,["-r","-rf","-fr"]),s=[];for(let o=0;;o+=1){let a=xt(n,o,{flags:["-r","-rf","-fr"]});if(!a)break;s.push(a)}if(s.length===0)return{stderr:"rm: missing operand",exitCode:1};for(let o of s){let a=S(r,o);L(e,a,"rm"),t.vfs.remove(a,{recursive:i})}return{exitCode:0}}};var mn={name:"sed",description:"Stream editor for filtering and transforming text",category:"text",params:["-e <expr> [file]","s/pattern/replace/[g]"],run:({authUser:e,shell:t,cwd:r,args:n,stdin:i})=>{let s=y(n,["-i"]),o=st(n,["-e"])??n.find(f=>!f.startsWith("-")),a=n.filter(f=>!f.startsWith("-")&&f!==o).pop();if(!o)return{stderr:"sed: no expression",exitCode:1};let l=i??"";if(a){let f=S(r,a);try{l=t.vfs.readFile(f)}catch{return{stderr:`sed: ${a}: No such file or directory`,exitCode:1}}}let c=o.match(/^s([^a-zA-Z0-9])(.+?)\1(.*?)\1([gi]*)$/);if(!c)return{stderr:`sed: unrecognized command: ${o}`,exitCode:1};let[,,u,d,m]=c,p=(m??"").includes("i")?"gi":(m??"").includes("g")?"g":"",w;try{w=new RegExp(u,p||"")}catch{return{stderr:`sed: invalid regex: ${u}`,exitCode:1}}let g=((m??"").includes("g")||p.includes("g"),l.replace(w,d??""));if(s&&a){let f=S(r,a);return t.writeFileAsUser(e,f,g),{exitCode:0}}return{stdout:g,exitCode:0}}};var pn={name:"set",description:"Display or set shell variables",category:"shell",params:["[VAR=value]"],run:({args:e,env:t})=>{if(e.length===0)return{stdout:Object.entries(t.vars).map(([n,i])=>`${n}=${i}`).join(`
|
|
171
|
+
`),exitCode:0};for(let r of e)if(r.includes("=")){let n=r.indexOf("=");t.vars[r.slice(0,n)]=r.slice(n+1)}return{exitCode:0}}};async function xe(e,t,r,n){return Ht(e,t,r,i=>q(i,n.authUser,n.hostname,n.mode,n.cwd,n.shell,void 0,n.env).then(s=>s.stdout??""))}function Ct(e){let t=[],r=0;for(;r<e.length;){let n=e[r].trim();if(!n||n.startsWith("#")){r++;continue}if(n.startsWith("if ")||n==="if"){let i=n.replace(/^if\s+/,"").replace(/;\s*then\s*$/,"").trim(),s=[],o=[],a=[],l="then",c="";for(r++;r<e.length&&e[r]?.trim()!=="fi";){let u=e[r].trim();u.startsWith("elif ")?(l="elif",c=u.replace(/^elif\s+/,"").replace(/;\s*then\s*$/,"").trim(),o.push({cond:c,body:[]})):u==="else"?l="else":u!=="then"&&(l==="then"?s.push(u):l==="elif"&&o.length>0?o[o.length-1].body.push(u):a.push(u)),r++}t.push({type:"if",cond:i,then_:s,elif:o,else_:a})}else if(n.startsWith("for ")){let i=n.match(/^for\s+(\w+)\s+in\s+(.+?)(?:\s*;\s*do)?$/);if(i){let s=[];for(r++;r<e.length&&e[r]?.trim()!=="done";){let o=e[r].trim().replace(/^do\s+/,"");o&&o!=="do"&&s.push(o),r++}t.push({type:"for",var:i[1],list:i[2],body:s})}else t.push({type:"cmd",line:n})}else if(n.startsWith("while ")){let i=n.replace(/^while\s+/,"").replace(/;\s*do\s*$/,"").trim(),s=[];for(r++;r<e.length&&e[r]?.trim()!=="done";){let o=e[r].trim().replace(/^do\s+/,"");o&&o!=="do"&&s.push(o),r++}t.push({type:"while",cond:i,body:s})}else t.push({type:"cmd",line:n});r++}return t}async function we(e,t){let r=await xe(e,t.env.vars,t.env.lastExitCode,t),n=r.match(/^\[?\s*(.+?)\s*\]?$/);if(n){let s=n[1],o=s.match(/^-([fdeznr])\s+(.+)$/);if(o){let[,c,u]=o,d=S(t.cwd,u);if(c==="f")return t.shell.vfs.exists(d)&&t.shell.vfs.stat(d).type==="file";if(c==="d")return t.shell.vfs.exists(d)&&t.shell.vfs.stat(d).type==="directory";if(c==="e")return t.shell.vfs.exists(d);if(c==="z")return(u??"").length===0;if(c==="n")return(u??"").length>0}let a=s.match(/^"?([^"]*)"?\s*(==|!=|=|<|>)\s*"?([^"]*)"?$/);if(a){let[,c,u,d]=a;if(u==="=="||u==="=")return c===d;if(u==="!=")return c!==d}let l=s.match(/^(\S+)\s+(-eq|-ne|-lt|-le|-gt|-ge)\s+(\S+)$/);if(l){let[,c,u,d]=l,m=Number(c),p=Number(d);if(u==="-eq")return m===p;if(u==="-ne")return m!==p;if(u==="-lt")return m<p;if(u==="-le")return m<=p;if(u==="-gt")return m>p;if(u==="-ge")return m>=p}}return((await q(r,t.authUser,t.hostname,t.mode,t.cwd,t.shell,void 0,t.env)).exitCode??0)===0}async function $t(e,t){let r={exitCode:0},n="";for(let i of e)if(i.type==="cmd"){let s=await xe(i.line,t.env.vars,t.env.lastExitCode,t),o=/^([A-Za-z_][A-Za-z0-9_]*)=(.*)/,a=s.trim().split(/\s+/);if(a.length>0&&o.test(a[0])&&a.every(u=>o.test(u))){for(let u of a){let d=u.match(o);t.env.vars[d[1]]=d[2]}t.env.lastExitCode=0;continue}let l=await q(s,t.authUser,t.hostname,t.mode,t.cwd,t.shell,void 0,t.env);if(t.env.lastExitCode=l.exitCode??0,l.stdout&&(n+=`${l.stdout}
|
|
172
|
+
`),l.stderr)return{...l,stdout:n.trim()};r=l}else if(i.type==="if"){let s=!1;if(await we(i.cond,t)){let o=await $t(Ct(i.then_),t);o.stdout&&(n+=`${o.stdout}
|
|
173
|
+
`),s=!0}else{for(let o of i.elif)if(await we(o.cond,t)){let a=await $t(Ct(o.body),t);a.stdout&&(n+=`${a.stdout}
|
|
295
174
|
`),s=!0;break}if(!s&&i.else_.length>0){let o=await $t(Ct(i.else_),t);o.stdout&&(n+=`${o.stdout}
|
|
296
|
-
`)}}}else if(i.type==="for"){let o=(await
|
|
297
|
-
`),l.closeSession)return l}}else if(i.type==="while"){let s=0;for(;s<1e3&&await
|
|
298
|
-
`),o.closeSession)return o;s++}}return{...r,stdout:n.trim()||r.stdout}}var
|
|
299
|
-
`).map(c=>c.trim()).filter(c=>c&&!c.startsWith("#")),l=Ct(a);return $t(l,e)}return{stderr:"sh: invalid usage. Use: sh -c 'cmd' or sh <file>",exitCode:1}}};var
|
|
175
|
+
`)}}}else if(i.type==="for"){let o=(await xe(i.list,t.env.vars,t.env.lastExitCode,t)).trim().split(/\s+/);for(let a of o){t.env.vars[i.var]=a;let l=await $t(Ct(i.body),t);if(l.stdout&&(n+=`${l.stdout}
|
|
176
|
+
`),l.closeSession)return l}}else if(i.type==="while"){let s=0;for(;s<1e3&&await we(i.cond,t);){let o=await $t(Ct(i.body),t);if(o.stdout&&(n+=`${o.stdout}
|
|
177
|
+
`),o.closeSession)return o;s++}}return{...r,stdout:n.trim()||r.stdout}}var fn={name:"sh",aliases:["bash"],description:"Execute shell script or command",category:"shell",params:["-c <script>","[<file>]"],run:async e=>{let{args:t,shell:r,cwd:n}=e;if(y(t,"-c")){let s=t[t.indexOf("-c")+1]??"";if(!s)return{stderr:"sh: -c requires a script",exitCode:1};let o=s.split(/[;\n]/).map(l=>l.trim()).filter(l=>l&&!l.startsWith("#")),a=Ct(o);return $t(a,e)}let i=t[0];if(i){let s=S(n,i);if(!r.vfs.exists(s))return{stderr:`sh: ${i}: No such file or directory`,exitCode:1};let a=r.vfs.readFile(s).split(`
|
|
178
|
+
`).map(c=>c.trim()).filter(c=>c&&!c.startsWith("#")),l=Ct(a);return $t(l,e)}return{stderr:"sh: invalid usage. Use: sh -c 'cmd' or sh <file>",exitCode:1}}};var hn={name:"shift",description:"Shift positional parameters",category:"shell",params:["[n]"],run:({args:e,env:t})=>{if(!t)return{exitCode:0};let r=parseInt(e[0]??"1",10)||1,n=t.vars.__argv?.split("\0").filter(Boolean)??[];t.vars.__argv=n.slice(r).join("\0");let i=n.slice(r);for(let s=1;s<=9;s++)t.vars[String(s)]=i[s-1]??"";return{exitCode:0}}},gn={name:"trap",description:"Trap signals and events",category:"shell",params:["[action] [signal...]"],run:({args:e,env:t})=>{if(!t||e.length===0)return{exitCode:0};let r=e[0]??"",n=e.slice(1);for(let i of n)t.vars[`__trap_${i.toUpperCase()}`]=r;return{exitCode:0}}},yn={name:"return",description:"Return from a shell function",category:"shell",params:["[n]"],run:({args:e,env:t})=>{let r=parseInt(e[0]??"0",10);return t&&(t.lastExitCode=r),{exitCode:r}}};var wn={name:"sleep",description:"Delay execution",category:"system",params:["<seconds>"],run:async({args:e})=>{let t=parseFloat(e[0]??"1");return Number.isNaN(t)||t<0?{stderr:"sleep: invalid time",exitCode:1}:(await new Promise(r=>setTimeout(r,t*1e3)),{exitCode:0})}};var xn={name:"sort",description:"Sort lines of text",category:"text",params:["[-r] [-n] [-u] [-k <col>] [file...]"],run:({authUser:e,shell:t,cwd:r,args:n,stdin:i})=>{let s=y(n,["-r"]),o=y(n,["-n"]),a=y(n,["-u"]),l=n.filter(w=>!w.startsWith("-")),d=[...(l.length>0?l.map(w=>{try{return L(e,S(r,w),"sort"),t.vfs.readFile(S(r,w))}catch{return""}}).join(`
|
|
300
179
|
`):i??"").split(`
|
|
301
|
-
`).filter(Boolean)].sort((
|
|
302
|
-
`),exitCode:0}}};var
|
|
303
|
-
`)){let d=u.trim();if(!d||d.startsWith("#"))continue;let m=await q(d,t,r,"shell",n,i,void 0,s);if(c=m.exitCode??0,m.closeSession||m.switchUser)return m}return{exitCode:c}}};var
|
|
180
|
+
`).filter(Boolean)].sort((w,g)=>o?Number(w)-Number(g):w.localeCompare(g)),m=s?d.reverse():d;return{stdout:(a?[...new Set(m)]:m).join(`
|
|
181
|
+
`),exitCode:0}}};var Sn={name:"source",aliases:["."],description:"Execute commands from a file in the current shell environment",category:"shell",params:["<file> [args...]"],run:async({args:e,authUser:t,hostname:r,cwd:n,shell:i,env:s})=>{let o=e[0];if(!o)return{stderr:"source: missing filename",exitCode:1};let a=S(n,o);if(!i.vfs.exists(a))return{stderr:`source: ${o}: No such file or directory`,exitCode:1};let l=i.vfs.readFile(a),c=0;for(let u of l.split(`
|
|
182
|
+
`)){let d=u.trim();if(!d||d.startsWith("#"))continue;let m=await q(d,t,r,"shell",n,i,void 0,s);if(c=m.exitCode??0,m.closeSession||m.switchUser)return m}return{exitCode:c}}};var bn={name:"su",description:"Switch user",category:"users",params:["[-] [-c <cmd>] [username]"],run:async({authUser:e,shell:t,args:r,hostname:n,mode:i,cwd:s})=>{let o=r.includes("-")||r.includes("-l")||r.includes("--login"),a=r.indexOf("-c"),l=a!==-1?r[a+1]:void 0,u=r.filter((d,m)=>m!==a&&m!==a+1).filter(d=>d!=="-"&&d!=="-l"&&d!=="--login").find(d=>!d.startsWith("-"))??"root";return t.users.listUsers().includes(u)?e==="root"?l?q(l,u,n,i,o?`/home/${u}`:s,t):{switchUser:u,nextCwd:o?`/home/${u}`:void 0,exitCode:0}:t.users.isSudoer(e)?{sudoChallenge:{username:u,targetUser:u,commandLine:l??null,loginShell:o,prompt:"Password: "},exitCode:0}:{stderr:`su: permission denied
|
|
304
183
|
`,exitCode:1}:{stderr:`su: user '${u}' does not exist
|
|
305
|
-
`,exitCode:1}}};function
|
|
184
|
+
`,exitCode:1}}};function oi(e){let{flags:t,flagsWithValues:r,positionals:n}=nt(e,{flags:["-i","-S"],flagsWithValue:["-u","--user"]}),i=t.has("-i"),s=r.get("-u")||r.get("--user")||"root",o=n.length>0?n.join(" "):null;return{targetUser:s,loginShell:i,commandLine:o}}var vn={name:"sudo",description:"Execute as superuser",category:"users",params:["<command...>"],run:async({authUser:e,hostname:t,mode:r,cwd:n,shell:i,args:s})=>{let{targetUser:o,loginShell:a,commandLine:l}=oi(s);if(e!=="root"&&!i.users.isSudoer(e))return{stderr:"sudo: permission denied",exitCode:1};let c=o||"root",u=`[sudo] password for ${e}: `;return e==="root"?!l&&a?{switchUser:c,nextCwd:`/home/${c}`,exitCode:0}:l?q(l,c,t,r,a?`/home/${c}`:n,i):{stderr:"sudo: missing command",exitCode:1}:{sudoChallenge:{username:e,targetUser:c,commandLine:l,loginShell:a,prompt:u},exitCode:0}}};var Cn={name:"tail",description:"Output last lines",category:"text",params:["[-n <lines>] [file...]"],run:({authUser:e,shell:t,cwd:r,args:n,stdin:i})=>{let s=st(n,["-n"]),o=n.find(d=>/^-\d+$/.test(d)),a=typeof s=="string"?parseInt(s,10):o?parseInt(o.slice(1),10):10,l=n.filter(d=>!d.startsWith("-")&&d!==s&&d!==String(a)),c=d=>{let m=d.split(`
|
|
306
185
|
`),p=d.endsWith(`
|
|
307
|
-
`),
|
|
186
|
+
`),w=p?m.slice(0,-1):m;return w.slice(Math.max(0,w.length-a)).join(`
|
|
308
187
|
`)+(p?`
|
|
309
|
-
`:"")};if(l.length===0)return{stdout:c(i??""),exitCode:0};let u=[];for(let d of l){let m=
|
|
310
|
-
`),exitCode:0}}};var
|
|
188
|
+
`:"")};if(l.length===0)return{stdout:c(i??""),exitCode:0};let u=[];for(let d of l){let m=S(r,d);try{L(e,m,"tail"),u.push(c(t.vfs.readFile(m)))}catch{return{stderr:`tail: ${d}: No such file or directory`,exitCode:1}}}return{stdout:u.join(`
|
|
189
|
+
`),exitCode:0}}};var $n={name:"tar",description:"Archive utility",category:"archive",params:["[-czf|-xzf|-tf] <archive> [files...]"],run:({authUser:e,shell:t,cwd:r,args:n})=>{let i=[],s=!1;for(let m of n)if(/^-[a-zA-Z]{2,}$/.test(m))for(let p of m.slice(1))i.push(`-${p}`);else if(!s&&/^[cxtdru]{1,}[a-zA-Z]*$/.test(m)&&!m.includes("/")&&!m.startsWith("-")){s=!0;for(let p of m)i.push(`-${p}`)}else i.push(m);let o=i.includes("-c"),a=i.includes("-x"),l=i.includes("-t"),c=i.indexOf("-f"),u=c!==-1?i[c+1]:i.find(m=>m.endsWith(".tar")||m.endsWith(".tar.gz")||m.endsWith(".tgz"));if(!o&&!a&&!l)return{stderr:`tar: must specify -c, -x, or -t
|
|
311
190
|
`,exitCode:1};if(!u)return{stderr:`tar: no archive specified
|
|
312
|
-
`,exitCode:1};let d=
|
|
313
|
-
`),exitCode:0};for(let[p,
|
|
314
|
-
`).replace(/\\t/g," ").replace(/\\r/g,"\r").replace(/\\\\/g,"\\")}function
|
|
315
|
-
`),exitCode:s}}};var Rn={name:"uname",description:"Print system information",category:"system",params:["[-a] [-s] [-r] [-m]"],run:({shell:e,args:t})=>{let r=y(t,["-a"]),n="Linux",i=e.properties?.kernel??"5.15.0",s=e.properties?.arch??"x86_64",o=e.hostname;return r?{stdout:`${n} ${o} ${i} #1 SMP ${s} GNU/Linux`,exitCode:0}:y(t,["-r"])?{stdout:i,exitCode:0}:y(t,["-m"])?{stdout:s,exitCode:0}:{stdout:n,exitCode:0}}};var
|
|
191
|
+
`,exitCode:1};let d=S(r,u);if(o){let m=new Set;c!==-1&&m.add(c+1);let p=i.filter((g,f)=>!g.startsWith("-")&&g!==u&&!m.has(f)),w={};for(let g of p){let f=S(r,g);try{if(t.vfs.stat(f).type==="file")w[g]=t.vfs.readFile(f);else{let E=(M,x)=>{for(let k of t.vfs.list(M)){let V=`${M}/${k}`,b=`${x}/${k}`;t.vfs.stat(V).type==="file"?w[b]=t.vfs.readFile(V):E(V,b)}};E(f,g)}}catch{return{stderr:`tar: ${g}: No such file or directory`,exitCode:1}}}return t.writeFileAsUser(e,d,JSON.stringify(w)),{exitCode:0}}if(l||a){let m;try{m=JSON.parse(t.vfs.readFile(d))}catch{return{stderr:`tar: ${u}: cannot open archive`,exitCode:1}}if(l)return{stdout:Object.keys(m).join(`
|
|
192
|
+
`),exitCode:0};for(let[p,w]of Object.entries(m))t.writeFileAsUser(e,S(r,p),w);return{exitCode:0}}return{stderr:"tar: must specify -c, -x, or -t",exitCode:1}}};var Pn={name:"tee",description:"Read stdin, write to stdout and files",category:"text",params:["[-a] <file...>"],run:({authUser:e,shell:t,cwd:r,args:n,stdin:i})=>{let s=y(n,["-a"]),o=n.filter(l=>!l.startsWith("-")),a=i??"";for(let l of o){let c=S(r,l);if(s){let u=(()=>{try{return t.vfs.readFile(c)}catch{return""}})();t.writeFileAsUser(e,c,u+a)}else t.writeFileAsUser(e,c,a)}return{stdout:a,exitCode:0}}};function Ft(e,t,r){if(e[e.length-1]==="]"&&(e=e.slice(0,-1)),e[0]==="["&&(e=e.slice(1)),e.length===0)return!1;if(e[0]==="!")return!Ft(e.slice(1),t,r);let n=e.indexOf("-a");if(n!==-1)return Ft(e.slice(0,n),t,r)&&Ft(e.slice(n+1),t,r);let i=e.indexOf("-o");if(i!==-1)return Ft(e.slice(0,i),t,r)||Ft(e.slice(i+1),t,r);if(e.length===2){let[s,o=""]=e,l=(c=>c.startsWith("/")?c:`${r}/${c}`.replace(/\/+/g,"/"))(o);switch(s){case"-e":return t.vfs.exists(l);case"-f":return t.vfs.exists(l)&&t.vfs.stat(l).type==="file";case"-d":return t.vfs.exists(l)&&t.vfs.stat(l).type==="directory";case"-r":return t.vfs.exists(l);case"-w":return t.vfs.exists(l);case"-x":return t.vfs.exists(l)&&!!(t.vfs.stat(l).mode&73);case"-s":return t.vfs.exists(l)&&t.vfs.stat(l).type==="file"&&t.vfs.stat(l).size>0;case"-z":return o.length===0;case"-n":return o.length>0;case"-L":return t.vfs.isSymlink(l)}}if(e.length===3){let[s="",o,a=""]=e,l=Number(s),c=Number(a);switch(o){case"=":case"==":return s===a;case"!=":return s!==a;case"<":return s<a;case">":return s>a;case"-eq":return l===c;case"-ne":return l!==c;case"-lt":return l<c;case"-le":return l<=c;case"-gt":return l>c;case"-ge":return l>=c}}return e.length===1?(e[0]??"").length>0:!1}var kn={name:"test",aliases:["["],description:"Evaluate conditional expression",category:"shell",params:["<expression>"],run:({args:e,shell:t,cwd:r})=>{try{return{exitCode:Ft([...e],t,r)?0:1}}catch{return{stderr:"test: malformed expression",exitCode:2}}}};var Mn={name:"touch",description:"Create or update files",category:"files",params:["<file>"],run:({authUser:e,shell:t,cwd:r,args:n})=>{if(n.length===0)return{stderr:"touch: missing file operand",exitCode:1};for(let i of n){let s=S(r,i);L(e,s,"touch"),t.vfs.exists(s)||t.writeFileAsUser(e,s,"")}return{exitCode:0}}};function ai(e){return e.replace(/\\n/g,`
|
|
193
|
+
`).replace(/\\t/g," ").replace(/\\r/g,"\r").replace(/\\\\/g,"\\")}function Fn(e){let t=[],r=ai(e),n=0;for(;n<r.length;){if(n+2<r.length&&r[n+1]==="-"){let i=r.charCodeAt(n),s=r.charCodeAt(n+2);if(i<=s){for(let o=i;o<=s;o++)t.push(String.fromCharCode(o));n+=3;continue}}t.push(r[n]),n++}return t}var An={name:"tr",description:"Translate or delete characters",category:"text",params:["[-d] [-s] <set1> [set2]"],run:({args:e,stdin:t})=>{let r=y(e,["-d"]),n=y(e,["-s"]),i=e.filter(l=>!l.startsWith("-")),s=Fn(i[0]??""),o=Fn(i[1]??""),a=t??"";if(r){let l=new Set(s);a=[...a].filter(c=>!l.has(c)).join("")}else if(o.length>0){let l=new Map;for(let c=0;c<s.length;c++)l.set(s[c],o[c]??o[o.length-1]??"");a=[...a].map(c=>l.get(c)??c).join("")}if(n&&o.length>0){let l=new Set(o);a=a.replace(/(.)\1+/g,(c,u)=>l.has(u)?u:c)}return{stdout:a,exitCode:0}}};var En={name:"tree",description:"Display directory tree",category:"navigation",params:["[path]"],run:({authUser:e,shell:t,cwd:r,args:n})=>{let i=S(r,xt(n,0)??r);return L(e,i,"tree"),{stdout:t.vfs.tree(i),exitCode:0}}};var Nn={name:"true",description:"Return success exit code",category:"shell",params:[],run:()=>({exitCode:0})},In={name:"false",description:"Return failure exit code",category:"shell",params:[],run:()=>({exitCode:1})};var Vn={name:"type",description:"Describe how a command would be interpreted",category:"shell",params:["<command...>"],run:({args:e,shell:t,env:r})=>{if(e.length===0)return{stderr:"type: missing argument",exitCode:1};let n=(r?.vars?.PATH??"/usr/local/bin:/usr/bin:/bin").split(":"),i=[],s=0;for(let o of e){if(ft(o)){i.push(`${o} is a shell builtin`);continue}let a=!1;for(let l of n){let c=`${l}/${o}`;if(t.vfs.exists(c)){i.push(`${o} is ${c}`),a=!0;break}}a||(i.push(`${o}: not found`),s=1)}return{stdout:i.join(`
|
|
194
|
+
`),exitCode:s}}};var Rn={name:"uname",description:"Print system information",category:"system",params:["[-a] [-s] [-r] [-m]"],run:({shell:e,args:t})=>{let r=y(t,["-a"]),n="Linux",i=e.properties?.kernel??"5.15.0",s=e.properties?.arch??"x86_64",o=e.hostname;return r?{stdout:`${n} ${o} ${i} #1 SMP ${s} GNU/Linux`,exitCode:0}:y(t,["-r"])?{stdout:i,exitCode:0}:y(t,["-m"])?{stdout:s,exitCode:0}:{stdout:n,exitCode:0}}};var Dn={name:"uniq",description:"Report or filter out repeated lines",category:"text",params:["[-c] [-d] [-u] [file]"],run:({args:e,stdin:t})=>{let r=y(e,["-c"]),n=y(e,["-d"]),i=y(e,["-u"]),s=(t??"").split(`
|
|
316
195
|
`),o=[],a=0;for(;a<s.length;){let l=a;for(;l<s.length&&s[l]===s[a];)l++;let c=l-a,u=s[a];if(n&&c===1){a=l;continue}if(i&&c>1){a=l;continue}o.push(r?`${String(c).padStart(4)} ${u}`:u),a=l}return{stdout:o.join(`
|
|
317
|
-
`),exitCode:0}}};var
|
|
196
|
+
`),exitCode:0}}};var _n={name:"unset",description:"Remove shell variable",category:"shell",params:["<VAR>"],run:({args:e,env:t})=>{for(let r of e)delete t.vars[r];return{exitCode:0}}};var Ln={name:"uptime",description:"Tell how long the system has been running",category:"system",params:["[-p] [-s]"],run:({args:e,shell:t})=>{let r=y(e,["-p"]),n=y(e,["-s"]),i=Math.floor((Date.now()-t.startTime)/1e3),s=Math.floor(i/86400),o=Math.floor(i%86400/3600),a=Math.floor(i%3600/60);if(n)return{stdout:new Date(t.startTime).toISOString().slice(0,19).replace("T"," "),exitCode:0};if(r){let m=[];return s>0&&m.push(`${s} day${s>1?"s":""}`),o>0&&m.push(`${o} hour${o>1?"s":""}`),m.push(`${a} minute${a!==1?"s":""}`),{stdout:`up ${m.join(", ")}`,exitCode:0}}let l=new Date().toTimeString().slice(0,8),c=s>0?`${s} day${s>1?"s":""}, ${String(o).padStart(2)}:${String(a).padStart(2,"0")}`:`${String(o).padStart(2)}:${String(a).padStart(2,"0")}`,u=t.users.listActiveSessions().length,d=(Math.random()*.5).toFixed(2);return{stdout:` ${l} up ${c}, ${u} user${u!==1?"s":""}, load average: ${d}, ${d}, ${d}`,exitCode:0}}};var zn={name:"wc",description:"Count words/lines/bytes",category:"text",params:["[-l] [-w] [-c] [file...]"],run:({authUser:e,shell:t,cwd:r,args:n,stdin:i})=>{let s=y(n,["-l"]),o=y(n,["-w"]),a=y(n,["-c"]),l=!s&&!o&&!a,c=n.filter(m=>!m.startsWith("-")),u=(m,p)=>{let w=m.split(`
|
|
318
197
|
`).length-(m.endsWith(`
|
|
319
|
-
`)?1:0),g=m.trim().split(/\s+/).filter(Boolean).length,f=Buffer.byteLength(m,"utf8"),$=[];return(l||s)&&$.push(String(
|
|
320
|
-
`),exitCode:0}}};var
|
|
198
|
+
`)?1:0),g=m.trim().split(/\s+/).filter(Boolean).length,f=Buffer.byteLength(m,"utf8"),$=[];return(l||s)&&$.push(String(w).padStart(7)),(l||o)&&$.push(String(g).padStart(7)),(l||a)&&$.push(String(f).padStart(7)),p&&$.push(` ${p}`),$.join("")};if(c.length===0)return{stdout:u(i??"",""),exitCode:0};let d=[];for(let m of c){let p=S(r,m);try{L(e,p,"wc");let w=t.vfs.readFile(p);d.push(u(w,m))}catch{return{stderr:`wc: ${m}: No such file or directory`,exitCode:1}}}return{stdout:d.join(`
|
|
199
|
+
`),exitCode:0}}};var Un={name:"wget",description:"File downloader (pure fetch)",category:"network",params:["[options] <url>"],run:async({authUser:e,cwd:t,args:r,shell:n})=>{let{flagsWithValues:i,positionals:s}=nt(r,{flagsWithValue:["-O","--output-document","-o","--output-file","-P","--directory-prefix","--tries","--timeout"]});if(y(r,["-h","--help"]))return{stdout:["Usage: wget [option]... [URL]..."," -O, --output-document=FILE Write to FILE ('-' for stdout)"," -P, --directory-prefix=DIR Save files in DIR"," -q, --quiet Quiet mode"," -v, --verbose Verbose output (default)"," -c, --continue Continue partial download"," --tries=N Retry N times"," --timeout=N Timeout in seconds"].join(`
|
|
321
200
|
`),exitCode:0};if(y(r,["-V","--version"]))return{stdout:"GNU Wget 1.21.3 (virtual) built on Fortune GNU/Linux.",exitCode:0};let o=s[0];if(!o)return{stderr:`wget: missing URL
|
|
322
|
-
Usage: wget [OPTION]... [URL]...`,exitCode:1};let a=
|
|
323
|
-
|
|
324
|
-
`),exitCode:
|
|
325
|
-
`)
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
`),exitCode:
|
|
329
|
-
`),exitCode:
|
|
201
|
+
Usage: wget [OPTION]... [URL]...`,exitCode:1};let a=o.startsWith("http://")||o.startsWith("https://")?o:`http://${o}`;if(!a)return{stderr:`wget: missing URL
|
|
202
|
+
Usage: wget [OPTION]... [URL]...`,exitCode:1};let l=i.get("-O")??i.get("--output-document")??null,c=i.get("-P")??i.get("--directory-prefix")??null,u=y(r,["-q","--quiet"]),d=l==="-"?null:l??Ue(a),m=d?S(t,c?`${c}/${d}`:d):null;m&&L(e,m,"wget");let p=[];u||(p.push(`--${new Date().toISOString()}-- ${a}`),p.push(`Resolving ${new URL(a).host}...`),p.push(`Connecting to ${new URL(a).host}...`));let w;try{w=await fetch(a,{headers:{"User-Agent":"Wget/1.21.3 (Fortune GNU/Linux)"}})}catch(f){let $=f instanceof Error?f.message:String(f);return p.push(`wget: unable to resolve host: ${$}`),{stderr:p.join(`
|
|
203
|
+
`),exitCode:4}}if(!w.ok)return p.push(`ERROR ${w.status}: ${w.statusText}`),{stderr:p.join(`
|
|
204
|
+
`),exitCode:8};let g;try{g=await w.text()}catch{return{stderr:"wget: failed to read response",exitCode:1}}if(!u){let f=w.headers.get("content-type")??"application/octet-stream";p.push(`HTTP request sent, awaiting response... ${w.status} ${w.statusText}`),p.push(`Length: ${g.length} [${f}]`)}return l==="-"?{stdout:g,stderr:p.join(`
|
|
205
|
+
`)||void 0,exitCode:0}:m?(n.writeFileAsUser(e,m,g),u||p.push(`Saving to: '${m}'
|
|
206
|
+
${m} 100%[==================>] ${g.length} B`),{stderr:p.join(`
|
|
207
|
+
`)||void 0,exitCode:0}):{stdout:g,exitCode:0}}};var Tn={name:"which",description:"Locate a command in PATH",category:"shell",params:["<command...>"],run:({args:e,shell:t,env:r})=>{if(e.length===0)return{stderr:"which: missing argument",exitCode:1};let n=(r?.vars?.PATH??"/usr/local/bin:/usr/bin:/bin").split(":"),i=[],s=!1;for(let o of e){let a=!1;for(let l of n){let c=`${l}/${o}`;if(t.vfs.exists(c)&&t.vfs.stat(c).type==="file"){i.push(c),a=!0;break}}a||(s=!0)}return i.length===0?{exitCode:1}:{stdout:i.join(`
|
|
208
|
+
`),exitCode:s?1:0}}};function te(e){let t=e.toLocaleString("en-US",{weekday:"short"}),r=e.toLocaleString("en-US",{month:"short"}),n=e.getDate().toString().padStart(2,"0"),i=e.getHours().toString().padStart(2,"0"),s=e.getMinutes().toString().padStart(2,"0"),o=e.getSeconds().toString().padStart(2,"0"),a=e.getFullYear();return`${t} ${r} ${n} ${i}:${s}:${o} ${a}`}var Bn={name:"who",description:"Show active sessions",category:"system",params:[],run:({shell:e})=>({stdout:e.users.listActiveSessions().map(r=>{let n=new Date(r.startedAt),i=Number.isNaN(n.getTime())?r.startedAt:te(n);return`${r.username} ${r.tty} ${i} (${r.remoteAddress||"unknown"})`}).join(`
|
|
209
|
+
`),exitCode:0})};var On={name:"whoami",description:"Print current user",category:"system",params:[],run:({authUser:e})=>({stdout:e,exitCode:0})};var Wn={name:"xargs",description:"Build and execute command lines from stdin",category:"text",params:["[command] [args...]"],run:async({authUser:e,hostname:t,mode:r,cwd:n,args:i,stdin:s,shell:o,env:a})=>{let l=i[0]??"echo",c=i.slice(1),u=(s??"").trim().split(/\s+/).filter(Boolean);if(u.length===0)return{exitCode:0};let d=[l,...c,...u].join(" ");return q(d,e,t,r,n,o,void 0,a)}};var li=[on,Ke,Vr,En,qe,Mn,dn,_r,Je,Lr,Fr,Ar,Ge,Er,dr,pr,mn,je,xn,Dn,zn,yr,Cn,Ye,An,Pn,Wn,nr,$n,hr,gr,He,On,Bn,$r,kr,fr,Rn,sn,Mr,rr,or,Xe,wn,rn,ar,lr,ur,pn,_n,fn,Ze,cr,Ur,Pr,Qe,Un,De,en,er,vn,bn,Zr,Oe,We,sr,ir,Tn,Vn,Dr,Le,ze,kn,Sn,Cr,nn,un,tr,hn,gn,yn,Nn,In,Xr,tn,Yr,cn,Ln,mr,Rr],jn=[],zt=new Map,ee=null,ci=vr(()=>be().map(e=>e.name));function Se(){zt.clear();for(let e of be()){zt.set(e.name,e);for(let t of e.aliases??[])zt.set(t,e)}ee=Array.from(zt.keys()).sort()}function be(){return[...li,...jn,ci]}function ve(e){let t={...e,name:e.name.trim().toLowerCase(),aliases:e.aliases?.map(n=>n.trim().toLowerCase())};if([t.name,...t.aliases??[]].some(n=>n.length===0||/\s/.test(n)))throw new Error("Command names must be non-empty and contain no spaces");jn.push(t),Se()}function Ce(e,t,r){return{name:e,params:t,run:r}}function $e(){return ee||Se(),ee}function fe(){return be()}function ft(e){return ee||Se(),zt.get(e.toLowerCase())}async function Hn(e,t,r,n,i,s,o){let a={exitCode:0},l=[],c=i,u=0;for(;u<e.length;){let m=e[u];if(a=await ui(m.pipeline,t,r,n,c,s,o),o.lastExitCode=a.exitCode??0,a.nextCwd&&(a.exitCode??0)===0&&(c=a.nextCwd),a.stdout&&l.push(a.stdout),a.closeSession||a.switchUser)return{...a,stdout:l.join("")||a.stdout};let p=m.op;if(!(!p||p===";")){if(p==="&&"){if((a.exitCode??0)!==0)for(;u<e.length&&e[u]?.op==="&&";)u++}else if(p==="||"&&(a.exitCode??0)===0)for(;u<e.length&&e[u]?.op==="||";)u++}u++}let d=l.join("");return{...a,stdout:d||a.stdout,nextCwd:c!==i?c:void 0}}async function ui(e,t,r,n,i,s,o){if(!e.isValid)return{stderr:e.error||"Syntax error",exitCode:1};if(e.commands.length===0)return{exitCode:0};let a=o??{vars:{},lastExitCode:0};return e.commands.length===1?di(e.commands[0],t,r,n,i,s,a):mi(e.commands,t,r,n,i,s,a)}async function di(e,t,r,n,i,s,o){let a;if(e.inputFile){let c=S(i,e.inputFile);try{a=s.vfs.readFile(c)}catch{return{stderr:`${e.inputFile}: No such file or directory`,exitCode:1}}}let l=await re(e.name,e.args,t,r,n,i,s,a,o);if(e.outputFile){let c=S(i,e.outputFile),u=l.stdout||"";try{if(e.appendOutput){let d=(()=>{try{return s.vfs.readFile(c)}catch{return""}})();s.writeFileAsUser(t,c,d+u)}else s.writeFileAsUser(t,c,u);return{...l,stdout:""}}catch{return{...l,stderr:`Failed to write to ${e.outputFile}`,exitCode:1}}}return l}async function mi(e,t,r,n,i,s,o){let a="",l=0;for(let c=0;c<e.length;c++){let u=e[c];if(c===0&&u.inputFile){let m=S(i,u.inputFile);try{a=s.vfs.readFile(m)}catch{return{stderr:`${u.inputFile}: No such file or directory`,exitCode:1}}}let d=await re(u.name,u.args,t,r,n,i,s,a,o);if(l=d.exitCode??0,c===e.length-1&&u.outputFile){let m=S(i,u.outputFile),p=d.stdout||"";try{if(u.appendOutput){let w=(()=>{try{return s.vfs.readFile(m)}catch{return""}})();s.writeFileAsUser(t,m,w+p)}else s.writeFileAsUser(t,m,p);a=""}catch{return{stderr:`Failed to write to ${u.outputFile}`,exitCode:1}}}else a=d.stdout||"";if(d.stderr&&l!==0)return{stderr:d.stderr,exitCode:l};if(d.closeSession||d.switchUser)return d}return{stdout:a,exitCode:l}}function Ut(e){let t=[],r="",n=!1,i="",s=0;for(;s<e.length;){let o=e[s],a=e[s+1];if((o==='"'||o==="'")&&!n){n=!0,i=o,s++;continue}if(n&&o===i){n=!1,i="",s++;continue}if(n){r+=o,s++;continue}if(o===" "){r&&(t.push(r),r=""),s++;continue}if((o===">"||o==="<")&&!n){r&&(t.push(r),r=""),o===">"&&a===">"?(t.push(">>"),s+=2):(t.push(o),s++);continue}r+=o,s++}return r&&t.push(r),t}function qn(e){let t=e.trim();if(!t)return{statements:[],isValid:!0};try{return{statements:pi(t),isValid:!0}}catch(r){return{statements:[],isValid:!1,error:r.message}}}function pi(e){let t=fi(e),r=[];for(let n of t){let s={pipeline:{commands:hi(n.text.trim()),isValid:!0}};n.op&&(s.op=n.op),r.push(s)}return r}function fi(e){let t=[],r="",n=0,i=!1,s="",o=0,a=l=>{r.trim()&&t.push({text:r,op:l}),r=""};for(;o<e.length;){let l=e[o],c=e.slice(o,o+2);if((l==='"'||l==="'")&&!i){i=!0,s=l,r+=l,o++;continue}if(i&&l===s){i=!1,r+=l,o++;continue}if(i){r+=l,o++;continue}if(l==="("){n++,r+=l,o++;continue}if(l===")"){n--,r+=l,o++;continue}if(n>0){r+=l,o++;continue}if(c==="&&"){a("&&"),o+=2;continue}if(c==="||"){a("||"),o+=2;continue}if(l===";"){a(";"),o++;continue}r+=l,o++}return a(),t}function hi(e){return gi(e).map(yi)}function gi(e){let t=[],r="",n=!1,i="";for(let o=0;o<e.length;o++){let a=e[o];if((a==='"'||a==="'")&&!n){n=!0,i=a,r+=a;continue}if(n&&a===i){n=!1,r+=a;continue}if(n){r+=a;continue}if(a==="|"&&e[o+1]!=="|"){if(!r.trim())throw new Error("Syntax error near unexpected token '|'");t.push(r.trim()),r=""}else r+=a}let s=r.trim();if(!s&&t.length>0)throw new Error("Syntax error near unexpected token '|'");return s&&t.push(s),t}function yi(e){let t=Ut(e);if(t.length===0)return{name:"",args:[]};let r=[],n,i,s=!1,o=0;for(;o<t.length;){let l=t[o];if(l==="<"){if(o++,o>=t.length)throw new Error("Syntax error: expected filename after <");n=t[o],o++}else if(l===">>"){if(o++,o>=t.length)throw new Error("Syntax error: expected filename after >>");i=t[o],s=!0,o++}else if(l===">"){if(o++,o>=t.length)throw new Error("Syntax error: expected filename after >");i=t[o],s=!1,o++}else r.push(l),o++}return{name:(r[0]??"").toLowerCase(),args:r.slice(1),inputFile:n,outputFile:i,appendOutput:s}}function At(e,t){return{vars:{PATH:"/usr/local/bin:/usr/bin:/bin",HOME:`/home/${e}`,USER:e,LOGNAME:e,SHELL:"/bin/sh",TERM:"xterm-256color",HOSTNAME:t,PS1:"\\u@\\h:\\w\\$ "},lastExitCode:0}}function Kn(e,t,r,n){if(e.startsWith("/")){if(!r.vfs.exists(e))return null;try{let s=r.vfs.stat(e);return s.type!=="file"||!(s.mode&73)||(e.startsWith("/sbin/")||e.startsWith("/usr/sbin/"))&&n!=="root"?null:e}catch{return null}}let i=(t.vars.PATH??"/usr/local/bin:/usr/bin:/bin").split(":");for(let s of i){if((s==="/sbin"||s==="/usr/sbin")&&n!=="root")continue;let o=`${s}/${e}`;if(r.vfs.exists(o))try{let a=r.vfs.stat(o);if(a.type!=="file"||!(a.mode&73))continue;return o}catch{}}return null}async function re(e,t,r,n,i,s,o,a,l){let c=l.vars[`__alias_${e}`];if(c)return q(`${c} ${t.join(" ")}`,r,n,i,s,o,a,l);let u=ft(e);if(!u){let d=Kn(e,l,o,r);if(d){let m=o.vfs.readFile(d),p=m.match(/exec\s+builtin\s+(\S+)/);if(p){let g=ft(p[1]);if(g)return await g.run({authUser:r,hostname:n,activeSessions:o.users.listActiveSessions(),rawInput:[e,...t].join(" "),mode:i,args:t,stdin:a,cwd:s,shell:o,env:l})}let w=ft("sh");if(w)return await w.run({authUser:r,hostname:n,activeSessions:o.users.listActiveSessions(),rawInput:`sh -c ${JSON.stringify(m)}`,mode:i,args:["-c",m,"--",...t],stdin:a,cwd:s,shell:o,env:l})}return{stderr:`${e}: command not found`,exitCode:127}}try{return await u.run({authUser:r,hostname:n,activeSessions:o.users.listActiveSessions(),rawInput:[e,...t].join(" "),mode:i,args:t,stdin:a,cwd:s,shell:o,env:l})}catch(d){return{stderr:d instanceof Error?d.message:"Command failed",exitCode:1}}}async function q(e,t,r,n,i,s,o,a){let l=e.trim();if(l.length===0)return{exitCode:0};let c=a??At(t,r),d=Ut(l)[0]?.toLowerCase()??"",m=c.vars[`__alias_${d}`],p=m?l.replace(d,m):l;if(/(?<![|&])[|](?![|])/.test(p)||p.includes(">")||p.includes("<")||p.includes("&&")||p.includes("||")||p.includes(";")){let x=qn(p);if(!x.isValid)return{stderr:x.error||"Syntax error",exitCode:1};try{return await Hn(x.statements,t,r,n,i,s,c)}catch(k){return{stderr:k instanceof Error?k.message:"Execution failed",exitCode:1}}}let g=await Ht(p,c.vars,c.lastExitCode,x=>q(x,t,r,n,i,s,void 0,c).then(k=>k.stdout??"")),f=Ut(g.trim()),$=f[0]?.toLowerCase()??"",E=f.slice(1),M=ft($);if(!M){let x=Kn($,c,s,t);if(x){let k=s.vfs.readFile(x),V=k.match(/exec\s+builtin\s+(\S+)/);if(V){let P=V[1],R=ft(P);if(R)return await R.run({authUser:t,hostname:r,activeSessions:s.users.listActiveSessions(),rawInput:[$,...E].join(" "),mode:n,args:E,stdin:o,cwd:i,shell:s,env:c})}let b=ft("sh");if(b)return await b.run({authUser:t,hostname:r,activeSessions:s.users.listActiveSessions(),rawInput:`sh -c ${JSON.stringify(k)}`,mode:n,args:["-c",k,"--",...E],stdin:o,cwd:i,shell:s,env:c})}return{stderr:`${$}: command not found`,exitCode:127}}try{return await M.run({authUser:t,hostname:r,activeSessions:s.users.listActiveSessions(),rawInput:g,mode:n,args:E,stdin:o,cwd:i,shell:s,env:c})}catch(x){return{stderr:x instanceof Error?x.message:"Command failed",exitCode:1}}}import{spawn as xi}from"node:child_process";import{readFile as wi}from"node:fs/promises";import*as ne from"node:path";function Pe(e){return`'${e.replace(/'/g,"'\\''")}'`}function Tt(e){return e.replace(/\r\n/g,`
|
|
330
210
|
`).replace(/\r/g,`
|
|
331
211
|
`).replace(/\n/g,`\r
|
|
332
|
-
`)}function
|
|
333
|
-
`).join("")}`}function
|
|
212
|
+
`)}function Gn(e,t){let r=Number.isFinite(t.cols)&&t.cols>0?Math.floor(t.cols):80,n=Number.isFinite(t.rows)&&t.rows>0?Math.floor(t.rows):24;return`stty cols ${r} rows ${n} 2>/dev/null; ${e}`}function Zn(e,t){return!t||t.trim()===""||t==="."?e:t.startsWith("/")?ne.posix.normalize(t):ne.posix.normalize(ne.posix.join(e,t))}async function Jn(e){try{let r=(await wi(`/proc/${e}/task/${e}/children`,"utf8")).trim().split(/\s+/).filter(Boolean).map(i=>Number.parseInt(i,10)).filter(i=>Number.isInteger(i)&&i>0),n=await Promise.all(r.map(i=>Jn(i)));return[...r,...n.flat()]}catch{return[]}}async function Qn(e=process.pid){let t=await Jn(e),r=Array.from(new Set(t)).sort((n,i)=>n-i);return r.length===0?null:r.join(",")}function Yn(e,t,r){let n=Gn(e,t),i=xi("script",["-qfec",n,"/dev/null"],{stdio:["pipe","pipe","pipe"],env:{...process.env,TERM:process.env.TERM??"xterm-256color"}});return i.stdout.on("data",s=>{r.write(s.toString("utf8"))}),i.stderr.on("data",s=>{r.write(s.toString("utf8"))}),i}function se(e,t,r){return Yn(`nano -- ${Pe(e)}`,t,r)}function Xn(e,t,r){return Yn(`htop -p ${Pe(e)}`,t,r)}function ie(e,t,r){let n=[`Linux ${e} ${t.kernel} ${t.arch}`,"","The programs included with the Fortune GNU/Linux system are free software;","the exact distribution terms for each program are described in the","individual files in /usr/share/doc/*/copyright.","","Fortune GNU/Linux comes with ABSOLUTELY NO WARRANTY, to the extent","permitted by applicable law."];if(r){let i=new Date(r.at),s=Number.isNaN(i.getTime())?r.at:te(i);n.push(`Last login: ${s} from ${r.from||"unknown"}`)}return n.push(""),`${n.map(i=>`${i}\r
|
|
213
|
+
`).join("")}`}function oe(e,t,r){let n=e==="root",i=n?"\x1B[31;1m":"\x1B[35;1m",s="\x1B[37;1m",o="\x1B[34;1m",a="\x1B[0m";return`${s}[${i}${e}${s}@${o}${t}${a} ${r}${s}]${a}${n?"#":"$"} `}import{EventEmitter as Wi}from"node:events";import*as Et from"node:os";function F(e,t,r=493){e.exists(t)||e.mkdir(t,r)}function T(e,t,r,n=420){e.exists(t)||e.writeFile(t,r,{mode:n})}function Si(e,t,r){F(e,"/etc"),T(e,"/etc/os-release",`${['NAME="Fortune GNU/Linux"',`PRETTY_NAME="${r.os}"`,"ID=fortune","ID_LIKE=debian",'HOME_URL="https://github.com/itsrealfortune/typescript-virtual-container"',"VERSION_CODENAME=aurora",'VERSION_ID="1.0"'].join(`
|
|
334
214
|
`)}
|
|
335
|
-
`),
|
|
336
|
-
`),
|
|
337
|
-
`),
|
|
215
|
+
`),T(e,"/etc/debian_version",`12.0
|
|
216
|
+
`),T(e,"/etc/hostname",`${t}
|
|
217
|
+
`),T(e,"/etc/shells",`/bin/sh
|
|
338
218
|
/bin/bash
|
|
339
219
|
/usr/bin/bash
|
|
340
|
-
`),
|
|
220
|
+
`),T(e,"/etc/profile",`${["export PATH=/usr/local/bin:/usr/bin:/bin","export PS1='\\u@\\h:\\w\\$ '"].join(`
|
|
341
221
|
`)}
|
|
342
|
-
`),
|
|
343
|
-
`),
|
|
344
|
-
`)),
|
|
222
|
+
`),T(e,"/etc/issue",`Fortune GNU/Linux 1.0 \\n \\l
|
|
223
|
+
`),T(e,"/etc/motd",["",`Welcome to ${r.os}`,`Kernel: ${r.kernel}`,""].join(`
|
|
224
|
+
`)),F(e,"/etc/apt"),F(e,"/etc/apt/sources.list.d"),T(e,"/etc/apt/sources.list",`${["# Fortune GNU/Linux package sources","deb [virtual] fortune://packages.fortune.local aurora main contrib","deb [virtual] fortune://security.fortune.local aurora-security main"].join(`
|
|
345
225
|
`)}
|
|
346
|
-
`),
|
|
226
|
+
`),F(e,"/etc/network"),T(e,"/etc/network/interfaces",`${["auto lo","iface lo inet loopback","","auto eth0","iface eth0 inet dhcp"].join(`
|
|
347
227
|
`)}
|
|
348
|
-
`),
|
|
228
|
+
`),T(e,"/etc/resolv.conf",`nameserver 1.1.1.1
|
|
349
229
|
nameserver 8.8.8.8
|
|
350
|
-
`),
|
|
230
|
+
`),T(e,"/etc/hosts",`${["127.0.0.1 localhost",`127.0.1.1 ${t}`,"::1 localhost ip6-localhost ip6-loopback"].join(`
|
|
351
231
|
`)}
|
|
352
|
-
`),
|
|
232
|
+
`),F(e,"/etc/cron.d"),F(e,"/etc/init.d"),F(e,"/etc/systemd"),F(e,"/etc/systemd/system")}function ke(e,t){let r=t.listUsers(),n=["root:x:0:0:root:/root:/bin/bash","daemon:x:1:1:daemon:/usr/sbin:/usr/sbin/nologin","www-data:x:33:33:www-data:/var/www:/usr/sbin/nologin","nobody:x:65534:65534:nobody:/nonexistent:/usr/sbin/nologin"],i=1e3;for(let a of r)a!=="root"&&(n.push(`${a}:x:${i}:${i}::/home/${a}:/bin/bash`),i++);e.writeFile("/etc/passwd",`${n.join(`
|
|
353
233
|
`)}
|
|
354
234
|
`);let s=["root:x:0:","daemon:x:1:",`sudo:x:27:${r.filter(a=>t.isSudoer(a)).join(",")}`,`users:x:100:${r.filter(a=>a!=="root").join(",")}`,"nogroup:x:65534:"];e.writeFile("/etc/group",`${s.join(`
|
|
355
235
|
`)}
|
|
356
236
|
`);let o=["root:*:19000:0:99999:7:::","daemon:*:19000:0:99999:7:::"];for(let a of r)a!=="root"&&o.push(`${a}:!:19000:0:99999:7:::`);e.writeFile("/etc/shadow",`${o.join(`
|
|
357
237
|
`)}
|
|
358
|
-
`,{mode:416})}function
|
|
238
|
+
`,{mode:416})}function ts(e){let t=e.match(/(\d+)$/);return 1e3+(t?.[1]?parseInt(t[1],10):0)}function es(e,t,r,n,i,s,o){let a=`/proc/${t}`;F(e,a),F(e,`${a}/fd`),F(e,`${a}/fdinfo`);let l=Math.floor((Date.now()-new Date(s).getTime())/1e3);e.writeFile(`${a}/cmdline`,`${i.replace(/\s+/g,"\0")}\0`),e.writeFile(`${a}/comm`,i.split(/\s+/)[0]??"bash"),e.writeFile(`${a}/status`,`${[`Name: ${i.split(/\s+/)[0]??"bash"}`,"State: S (sleeping)",`Pid: ${t}`,"PPid: 1","Uid: 0 0 0 0","Gid: 0 0 0 0","VmRSS: 4096 kB","VmSize: 16384 kB","Threads: 1"].join(`
|
|
359
239
|
`)}
|
|
360
240
|
`),e.writeFile(`${a}/stat`,`${t} (${i.split(/\s+/)[0]??"bash"}) S 1 ${t} ${t} 0 -1 4194304 0 0 0 0 ${l} 0 0 0 20 0 1 0 0 16384 4096 0
|
|
361
|
-
`),e.writeFile(`${a}/environ`,`${Object.entries(o).map(([c,u])=>`${c}=${u}`).join("\0")}\0`),e.writeFile(`${a}/cwd`,`/home/${r}\0`),e.writeFile(`${a}/exe`,"/bin/bash\0"),e.writeFile(`${a}/fd/0`,""),e.writeFile(`${a}/fd/1`,""),e.writeFile(`${a}/fd/2`,"")}function
|
|
362
|
-
`);let o=Math.floor(
|
|
241
|
+
`),e.writeFile(`${a}/environ`,`${Object.entries(o).map(([c,u])=>`${c}=${u}`).join("\0")}\0`),e.writeFile(`${a}/cwd`,`/home/${r}\0`),e.writeFile(`${a}/exe`,"/bin/bash\0"),e.writeFile(`${a}/fd/0`,""),e.writeFile(`${a}/fd/1`,""),e.writeFile(`${a}/fd/2`,"")}function ae(e,t,r,n,i){F(e,"/proc");let s=Math.floor((Date.now()-n)/1e3);e.writeFile("/proc/uptime",`${s}.00 ${Math.floor(s*.9)}.00
|
|
242
|
+
`);let o=Math.floor(Et.totalmem()/1024),a=Math.floor(Et.freemem()/1024),l=Math.floor(a*.95);e.writeFile("/proc/meminfo",`${[`MemTotal: ${String(o).padStart(10)} kB`,`MemFree: ${String(a).padStart(10)} kB`,`MemAvailable: ${String(l).padStart(10)} kB`,`Buffers: ${String(Math.floor(o*.02)).padStart(10)} kB`,`Cached: ${String(Math.floor(o*.15)).padStart(10)} kB`,`SwapTotal: ${String(Math.floor(o*.5)).padStart(10)} kB`,`SwapFree: ${String(Math.floor(o*.5)).padStart(10)} kB`].join(`
|
|
363
243
|
`)}
|
|
364
|
-
`);let c=
|
|
244
|
+
`);let c=Et.cpus(),u=[];for(let f=0;f<c.length;f++){let $=c[f];if(!$)continue;let E=$.speed.toFixed(3);u.push(`processor : ${f}`,`model name : ${$.model}`,`cpu MHz : ${E}`,"cache size : 8192 KB","")}e.writeFile("/proc/cpuinfo",`${u.join(`
|
|
365
245
|
`)}
|
|
366
246
|
`),e.writeFile("/proc/version",`Linux version ${t.kernel} (fortune@build) (gcc version 12.2.0) #1 SMP
|
|
367
247
|
`),e.writeFile("/proc/hostname",`${r}
|
|
368
248
|
`);let d=(Math.random()*.5).toFixed(2),m=1+(i?.length??0);e.writeFile("/proc/loadavg",`${d} ${d} ${d} ${m}/${m} 1
|
|
369
|
-
`),
|
|
249
|
+
`),F(e,"/proc/net"),T(e,"/proc/net/dev",`${["Inter-| Receive | Transmit"," face |bytes packets errs drop fifo frame compressed multicast|bytes packets errs drop fifo colls carrier compressed"," lo: 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0"," eth0: 131072 1024 0 0 0 0 0 0 65536 512 0 0 0 0 0 0"].join(`
|
|
370
250
|
`)}
|
|
371
|
-
`),
|
|
251
|
+
`),es(e,1,"root","pts/0","/sbin/init",new Date(n).toISOString(),{});let p=i??[];for(let f of p){let $=ts(f.tty);es(e,$,f.username,f.tty,"bash",f.startedAt,{USER:f.username,HOME:`/home/${f.username}`,TERM:"xterm-256color",SHELL:"/bin/bash"})}let w=p.length>0?ts(p[p.length-1].tty):1;if(e.exists("/proc/self"))try{e.remove("/proc/self")}catch{}let g=`/proc/${w}`;if(e.exists(g)){F(e,"/proc/self"),F(e,"/proc/self/fd");for(let f of e.list(g)){let $=`${g}/${f}`,E=`/proc/self/${f}`;try{e.stat($).type==="file"&&e.writeFile(E,e.readFile($))}catch{}}e.writeFile("/proc/self/status",e.exists(`${g}/status`)?e.readFile(`${g}/status`):"")}else F(e,"/proc/self"),e.writeFile("/proc/self/cmdline","bash\0"),e.writeFile("/proc/self/comm","bash"),e.writeFile("/proc/self/status",`Name: bash
|
|
372
252
|
State: S (sleeping)
|
|
373
253
|
Pid: 1
|
|
374
254
|
PPid: 0
|
|
375
|
-
`),e.writeFile("/proc/self/environ",""),e.writeFile("/proc/self/cwd","/root\0"),e.writeFile("/proc/self/exe","/bin/bash\0")}function
|
|
376
|
-
`),
|
|
377
|
-
`),
|
|
378
|
-
`),
|
|
379
|
-
`),
|
|
380
|
-
`),
|
|
381
|
-
`)}function
|
|
255
|
+
`),e.writeFile("/proc/self/environ",""),e.writeFile("/proc/self/cwd","/root\0"),e.writeFile("/proc/self/exe","/bin/bash\0")}function bi(e,t){F(e,"/sys"),F(e,"/sys/devices"),F(e,"/sys/devices/virtual"),F(e,"/sys/devices/virtual/dmi"),F(e,"/sys/devices/virtual/dmi/id"),T(e,"/sys/devices/virtual/dmi/id/sys_vendor",`Fortune Systems
|
|
256
|
+
`),T(e,"/sys/devices/virtual/dmi/id/product_name",`VirtualContainer v1
|
|
257
|
+
`),T(e,"/sys/devices/virtual/dmi/id/board_name",`fortune-board
|
|
258
|
+
`),F(e,"/sys/class"),F(e,"/sys/class/net"),F(e,"/sys/kernel"),T(e,"/sys/kernel/hostname",`fortune-vm
|
|
259
|
+
`),T(e,"/sys/kernel/osrelease",`${t.kernel}
|
|
260
|
+
`),T(e,"/sys/kernel/ostype",`Linux
|
|
261
|
+
`)}function vi(e){F(e,"/dev"),T(e,"/dev/null","",438),T(e,"/dev/zero","",438),T(e,"/dev/random","",292),T(e,"/dev/urandom","",292),F(e,"/dev/pts"),F(e,"/dev/shm")}function Ci(e){F(e,"/usr"),F(e,"/usr/bin"),F(e,"/usr/sbin"),F(e,"/usr/local"),F(e,"/usr/local/bin"),F(e,"/usr/local/lib"),F(e,"/usr/local/share"),F(e,"/usr/share"),F(e,"/usr/share/doc"),F(e,"/usr/share/man"),F(e,"/usr/share/man/man1"),F(e,"/usr/lib");let t=["sh","bash","ls","cat","echo","grep","find","sort","head","tail","cut","tr","sed","awk","wc","tee","tar","gzip","gunzip","touch","mkdir","rm","mv","cp","chmod","ln","pwd","env","date","sleep","id","whoami","hostname","uname","ps","kill","df","du","curl","wget","nano","diff","uniq","xargs","base64"];for(let r of t)T(e,`/usr/bin/${r}`,`#!/bin/sh
|
|
382
262
|
exec builtin ${r} "$@"
|
|
383
|
-
`,493);
|
|
263
|
+
`,493);T(e,"/usr/bin/lsb_release",`#!/bin/sh
|
|
384
264
|
exec lsb_release "$@"
|
|
385
|
-
`,493)}function
|
|
386
|
-
`),
|
|
265
|
+
`,493)}function $i(e){F(e,"/var"),F(e,"/var/log"),F(e,"/var/tmp"),F(e,"/var/run"),F(e,"/var/cache"),F(e,"/var/cache/apt"),F(e,"/var/cache/apt/archives"),F(e,"/var/lib"),F(e,"/var/lib/apt"),F(e,"/var/lib/apt/lists"),F(e,"/var/lib/dpkg"),F(e,"/var/lib/dpkg/info"),T(e,"/var/lib/dpkg/status",""),T(e,"/var/lib/dpkg/available",""),T(e,"/var/log/syslog",`${new Date().toUTCString()} fortune kernel: Virtual container started
|
|
266
|
+
`),T(e,"/var/log/auth.log",""),T(e,"/var/log/dpkg.log",""),T(e,"/var/log/apt/history.log",""),T(e,"/var/log/apt/term.log","")}function Pi(e){e.exists("/bin")||e.symlink("/usr/bin","/bin"),e.exists("/sbin")||e.symlink("/usr/sbin","/sbin"),e.exists("/lib")||F(e,"/lib"),e.exists("/lib64")||F(e,"/lib64")}function ki(e){F(e,"/tmp",1023)}function Mi(e){F(e,"/root",448),T(e,"/root/.bashrc",`${["# root .bashrc","export PS1='\\[\\033[0;31m\\]\\u@\\h\\[\\033[0m\\]:\\[\\033[0;34m\\]\\w\\[\\033[0m\\]# '","export PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin","alias ll='ls -la'","alias la='ls -A'"].join(`
|
|
387
267
|
`)}
|
|
388
|
-
`),
|
|
389
|
-
`),e.exists("/home/root")||e.symlink("/root","/home/root")}function
|
|
268
|
+
`),T(e,"/root/.profile",`[ -f ~/.bashrc ] && . ~/.bashrc
|
|
269
|
+
`),e.exists("/home/root")||e.symlink("/root","/home/root")}function Fi(e){F(e,"/opt"),F(e,"/srv"),F(e,"/mnt"),F(e,"/media")}function rs(e,t,r,n,i){Si(e,r,n),bi(e,n),vi(e),Ci(e),$i(e),Pi(e),ki(e),Mi(e),Fi(e),ae(e,n,r,i,[]),ke(e,t)}function ns(e){return e==="1"||e==="true"}function ss(){return typeof performance<"u"&&typeof performance.now=="function"?performance.now():Date.now()}function Ai(){return ns(process.env.DEV_MODE)||ns(process.env.RENDER_PERF)}function le(e){let t=Ai();if(!t)return{enabled:t,mark:()=>{},done:()=>{}};let r=ss(),n=s=>{let o=ss()-r;console.log(`[perf][${e}] ${s}: ${o.toFixed(1)}ms`)};return{enabled:t,mark:n,done:(s="done")=>{n(s)}}}import{EventEmitter as Ii}from"node:events";import*as St from"node:fs";import*as ht from"node:path";import{gunzipSync as Ee,gzipSync as ms}from"node:zlib";var Ae=Buffer.from([86,70,83,33]),Ei=1,is=1,os=2,Me=class{chunks=[];write(t){this.chunks.push(t)}writeUint8(t){let r=Buffer.allocUnsafe(1);r.writeUInt8(t,0),this.chunks.push(r)}writeUint16(t){let r=Buffer.allocUnsafe(2);r.writeUInt16LE(t,0),this.chunks.push(r)}writeUint32(t){let r=Buffer.allocUnsafe(4);r.writeUInt32LE(t,0),this.chunks.push(r)}writeFloat64(t){let r=Buffer.allocUnsafe(8);r.writeDoubleBE(t,0),this.chunks.push(r)}writeString(t){let r=Buffer.from(t,"utf8");this.writeUint16(r.length),this.chunks.push(r)}writeBytes(t){this.writeUint32(t.length),this.chunks.push(t)}toBuffer(){return Buffer.concat(this.chunks)}};function as(e,t){if(t.type==="file"){let r=t;e.writeUint8(is),e.writeString(r.name),e.writeUint32(r.mode),e.writeFloat64(r.createdAt.getTime()),e.writeFloat64(r.updatedAt.getTime()),e.writeUint8(r.compressed?1:0),e.writeBytes(r.content)}else{let r=t;e.writeUint8(os),e.writeString(r.name),e.writeUint32(r.mode),e.writeFloat64(r.createdAt.getTime()),e.writeFloat64(r.updatedAt.getTime());let n=Array.from(r.children.values());e.writeUint32(n.length);for(let i of n)as(e,i)}}function ls(e){let t=new Me;return t.write(Ae),t.writeUint8(Ei),as(t,e),t.toBuffer()}var Fe=class{constructor(t){this.buf=t}buf;pos=0;readUint8(){return this.buf.readUInt8(this.pos++)}readUint16(){let t=this.buf.readUInt16LE(this.pos);return this.pos+=2,t}readUint32(){let t=this.buf.readUInt32LE(this.pos);return this.pos+=4,t}readFloat64(){let t=this.buf.readDoubleBE(this.pos);return this.pos+=8,t}readString(){let t=this.readUint16(),r=this.buf.toString("utf8",this.pos,this.pos+t);return this.pos+=t,r}readBytes(){let t=this.readUint32(),r=this.buf.slice(this.pos,this.pos+t);return this.pos+=t,r}remaining(){return this.buf.length-this.pos}};function cs(e){let t=e.readUint8(),r=e.readString(),n=e.readUint32(),i=new Date(e.readFloat64()),s=new Date(e.readFloat64());if(t===is){let o=e.readUint8()===1,a=e.readBytes();return{type:"file",name:r,mode:n,createdAt:i,updatedAt:s,compressed:o,content:a}}if(t===os){let o=e.readUint32(),a=new Map;for(let l=0;l<o;l++){let c=cs(e);a.set(c.name,c)}return{type:"directory",name:r,mode:n,createdAt:i,updatedAt:s,children:a}}throw new Error(`[VFS binary] Unknown node type: 0x${t.toString(16)}`)}function us(e){if(e.length<5)throw new Error("[VFS binary] Buffer too short");if(!e.slice(0,4).equals(Ae))throw new Error("[VFS binary] Invalid magic \u2014 not a VFS binary snapshot");let r=new Fe(e);for(let i=0;i<5;i++)r.readUint8();let n=cs(r);if(n.type!=="directory")throw new Error("[VFS binary] Root node must be a directory");return n}function ds(e){return e.length>=4&&e.slice(0,4).equals(Ae)}import*as ce from"node:path";function K(e){if(!e||e.trim()==="")return"/";let t=ce.posix.normalize(e.startsWith("/")?e:`/${e}`);return t===""?"/":t}function Ni(e){return e.split("/").filter(Boolean)}function et(e,t){let r=K(t);if(r==="/")return e;let n=Ni(r),i=e;for(let s of n){if(i.type!=="directory")throw new Error(`Path '${r}' does not exist.`);let o=i.children.get(s);if(!o)throw new Error(`Path '${r}' does not exist.`);i=o}return i}function Nt(e,t,r,n){let i=K(t);if(i==="/")throw new Error("Root path has no parent directory.");let s=ce.posix.dirname(i),o=ce.posix.basename(i);if(!o)throw new Error(`Invalid path '${t}'.`);r&&n(s);let a=et(e,s);if(a.type!=="directory")throw new Error(`Parent path '${s}' is not a directory.`);return{parent:a,name:o}}var Ne=class e extends Ii{root;mode;snapshotFile;constructor(t={}){if(super(),this.mode=t.mode??"memory",this.mode==="fs"){if(!t.snapshotPath)throw new Error('VirtualFileSystem: "snapshotPath" is required when mode is "fs".');this.snapshotFile=ht.resolve(t.snapshotPath,"vfs-snapshot.vfsb")}else this.snapshotFile=null;this.root=this.makeDir("",493)}makeDir(t,r){let n=new Date;return{type:"directory",name:t,mode:r,createdAt:n,updatedAt:n,children:new Map}}makeFile(t,r,n,i){let s=new Date;return{type:"file",name:t,content:r,mode:n,compressed:i,createdAt:s,updatedAt:s}}mkdirRecursive(t,r){let n=K(t);if(n==="/")return;let i=n.split("/").filter(Boolean),s=this.root,o="";for(let a of i){o+=`/${a}`;let l=s.children.get(a);if(!l)l=this.makeDir(a,r),s.children.set(a,l),this.emit("dir:create",{path:o,mode:r});else if(l.type!=="directory")throw new Error(`Cannot create directory '${o}': path is a file.`);s=l}}async restoreMirror(){if(!(this.mode!=="fs"||!this.snapshotFile)&&St.existsSync(this.snapshotFile))try{let t=St.readFileSync(this.snapshotFile);if(ds(t))this.root=us(t);else{let r=JSON.parse(t.toString("utf8"));this.root=this.deserializeDir(r.root,""),console.info("[VirtualFileSystem] Migrating legacy JSON snapshot to binary format.")}this.emit("snapshot:restore",{path:this.snapshotFile})}catch(t){console.warn(`[VirtualFileSystem] Could not restore snapshot from ${this.snapshotFile}:`,t instanceof Error?t.message:String(t))}}async flushMirror(){if(this.mode!=="fs"||!this.snapshotFile){this.emit("mirror:flush");return}let t=ht.dirname(this.snapshotFile);St.mkdirSync(t,{recursive:!0});let r=ls(this.root);St.writeFileSync(this.snapshotFile,r),this.emit("mirror:flush",{path:this.snapshotFile})}getMode(){return this.mode}getSnapshotPath(){return this.snapshotFile}mkdir(t,r=493){let n=K(t),i=(()=>{try{return et(this.root,n)}catch{return null}})();if(i&&i.type!=="directory")throw new Error(`Cannot create directory '${n}': path is a file.`);this.mkdirRecursive(n,r)}writeFile(t,r,n={}){let i=K(t),{parent:s,name:o}=Nt(this.root,i,!0,m=>this.mkdirRecursive(m,493)),a=s.children.get(o);if(a?.type==="directory")throw new Error(`Cannot write file '${i}': path is a directory.`);let l=Buffer.isBuffer(r)?r:Buffer.from(r,"utf8"),c=n.compress??!1,u=c?ms(l):l,d=n.mode??420;if(a){let m=a;m.content=u,m.compressed=c,m.mode=d,m.updatedAt=new Date}else s.children.set(o,this.makeFile(o,u,d,c));this.emit("file:write",{path:i,size:u.length})}readFile(t){let r=K(t),n=et(this.root,r);if(n.type!=="file")throw new Error(`Cannot read '${t}': not a file.`);let i=n,s=i.compressed?Ee(i.content):i.content;return this.emit("file:read",{path:r,size:s.length}),s.toString("utf8")}readFileRaw(t){let r=K(t),n=et(this.root,r);if(n.type!=="file")throw new Error(`Cannot read '${t}': not a file.`);let i=n,s=i.compressed?Ee(i.content):i.content;return this.emit("file:read",{path:r,size:s.length}),s}exists(t){try{return et(this.root,K(t)),!0}catch{return!1}}chmod(t,r){et(this.root,K(t)).mode=r}stat(t){let r=K(t),n=et(this.root,r),i=r==="/"?"":ht.posix.basename(r);if(n.type==="file"){let o=n;return{type:"file",name:i,path:r,mode:o.mode,createdAt:o.createdAt,updatedAt:o.updatedAt,compressed:o.compressed,size:o.content.length}}let s=n;return{type:"directory",name:i,path:r,mode:s.mode,createdAt:s.createdAt,updatedAt:s.updatedAt,childrenCount:s.children.size}}list(t="/"){let r=K(t),n=et(this.root,r);if(n.type!=="directory")throw new Error(`Cannot list '${t}': not a directory.`);return Array.from(n.children.keys()).sort()}tree(t="/"){let r=K(t),n=et(this.root,r);if(n.type!=="directory")throw new Error(`Cannot render tree for '${t}': not a directory.`);let i=t==="/"?"/":ht.posix.basename(r);return this.renderTreeLines(n,i)}renderTreeLines(t,r){let n=[r],i=Array.from(t.children.keys()).sort();for(let s=0;s<i.length;s++){let o=i[s],a=t.children.get(o),l=s===i.length-1,c=l?"\u2514\u2500\u2500 ":"\u251C\u2500\u2500 ",u=l?" ":"\u2502 ";if(n.push(`${c}${o}`),a.type==="directory"){let d=this.renderTreeLines(a,"").split(`
|
|
390
270
|
`).slice(1).map(m=>`${u}${m}`);n.push(...d)}}return n.join(`
|
|
391
|
-
`)}getUsageBytes(t="/"){return this.computeUsage(et(this.root,K(t)))}computeUsage(t){if(t.type==="file")return t.content.length;let r=0;for(let n of t.children.values())r+=this.computeUsage(n);return r}compressFile(t){let r=et(this.root,K(t));if(r.type!=="file")throw new Error(`Cannot compress '${t}': not a file.`);let n=r;n.compressed||(n.content=
|
|
271
|
+
`)}getUsageBytes(t="/"){return this.computeUsage(et(this.root,K(t)))}computeUsage(t){if(t.type==="file")return t.content.length;let r=0;for(let n of t.children.values())r+=this.computeUsage(n);return r}compressFile(t){let r=et(this.root,K(t));if(r.type!=="file")throw new Error(`Cannot compress '${t}': not a file.`);let n=r;n.compressed||(n.content=ms(n.content),n.compressed=!0,n.updatedAt=new Date)}decompressFile(t){let r=et(this.root,K(t));if(r.type!=="file")throw new Error(`Cannot decompress '${t}': not a file.`);let n=r;n.compressed&&(n.content=Ee(n.content),n.compressed=!1,n.updatedAt=new Date)}symlink(t,r){let n=K(r),i=t.startsWith("/")?K(t):t,{parent:s,name:o}=Nt(this.root,n,!0,l=>this.mkdirRecursive(l,493)),a={type:"file",name:o,content:Buffer.from(i,"utf8"),mode:41471,compressed:!1,createdAt:new Date,updatedAt:new Date};s.children.set(o,a),this.emit("symlink:create",{link:n,target:i})}isSymlink(t){try{let r=et(this.root,K(t));return r.type==="file"&&r.mode===41471}catch{return!1}}resolveSymlink(t,r=8){let n=K(t);for(let i=0;i<r;i++){try{let s=et(this.root,n);if(s.type==="file"&&s.mode===41471){let o=s.content.toString("utf8");n=o.startsWith("/")?o:K(ht.posix.join(ht.posix.dirname(n),o));continue}}catch{break}return n}throw new Error(`Too many levels of symbolic links: ${t}`)}remove(t,r={}){let n=K(t);if(n==="/")throw new Error("Cannot remove root directory.");let i=et(this.root,n);if(i.type==="directory"){let a=i;if(!r.recursive&&a.children.size>0)throw new Error(`Directory '${n}' is not empty. Use recursive option.`)}let{parent:s,name:o}=Nt(this.root,n,!1,()=>{});s.children.delete(o),this.emit("node:remove",{path:n})}move(t,r){let n=K(t),i=K(r);if(n==="/"||i==="/")throw new Error("Cannot move root directory.");let s=et(this.root,n);if(this.exists(i))throw new Error(`Destination '${i}' already exists.`);this.mkdirRecursive(ht.posix.dirname(i),493);let{parent:o,name:a}=Nt(this.root,i,!1,()=>{}),{parent:l,name:c}=Nt(this.root,n,!1,()=>{});l.children.delete(c),s.name=a,o.children.set(a,s)}toSnapshot(){return{root:this.serializeDir(this.root)}}serializeDir(t){let r=[];for(let n of t.children.values())r.push(n.type==="file"?this.serializeFile(n):this.serializeDir(n));return{type:"directory",name:t.name,mode:t.mode,createdAt:t.createdAt.toISOString(),updatedAt:t.updatedAt.toISOString(),children:r}}serializeFile(t){return{type:"file",name:t.name,mode:t.mode,createdAt:t.createdAt.toISOString(),updatedAt:t.updatedAt.toISOString(),compressed:t.compressed,contentBase64:t.content.toString("base64")}}static fromSnapshot(t){let r=new e;return r.root=r.deserializeDir(t.root,""),r}importSnapshot(t){this.root=this.deserializeDir(t.root,""),this.emit("snapshot:import")}deserializeDir(t,r){let n={type:"directory",name:r,mode:t.mode,createdAt:new Date(t.createdAt),updatedAt:new Date(t.updatedAt),children:new Map};for(let i of t.children)if(i.type==="file"){let s=i;n.children.set(s.name,{type:"file",name:s.name,mode:s.mode,createdAt:new Date(s.createdAt),updatedAt:new Date(s.updatedAt),compressed:s.compressed,content:Buffer.from(s.contentBase64,"base64")})}else{let s=this.deserializeDir(i,i.name);n.children.set(i.name,s)}return n}},ps=Ne;var Ie=[{name:"vim",version:"2:9.0.1378-2",section:"editors",description:"Vi IMproved - enhanced vi editor",shortDesc:"Vi IMproved",installedSizeKb:3812,files:[{path:"/usr/bin/vim",content:`#!/bin/sh
|
|
392
272
|
echo 'vim: use nano for editing in this environment'
|
|
393
273
|
`,mode:493},{path:"/usr/bin/vi",content:`#!/bin/sh
|
|
394
274
|
exec vim "$@"
|
|
@@ -473,19 +353,23 @@ echo 'systemd is not running in this virtual container.'
|
|
|
473
353
|
exit 1
|
|
474
354
|
`,mode:493},{path:"/usr/bin/journalctl",content:`#!/bin/sh
|
|
475
355
|
echo 'journalctl: virtual stub'
|
|
476
|
-
`,mode:493}]}
|
|
356
|
+
`,mode:493}]},{name:"gzip",version:"1.12-2",section:"utils",description:"GNU compression utility",shortDesc:"compression utility",installedSizeKb:128,files:[{path:"/usr/bin/gzip",content:`#!/bin/sh
|
|
357
|
+
echo 'gzip: virtual stub'
|
|
358
|
+
`,mode:493}]},{name:"neofetch",version:"7.1.0-1",section:"utils",description:"A command-line system information tool written in bash 3.2+",shortDesc:"command-line system information tool",installedSizeKb:256,files:[{path:"/usr/bin/neofetch",content:`#!/bin/sh
|
|
359
|
+
echo 'neofetch: virtual stub'
|
|
360
|
+
`,mode:493}]}],ue=class{constructor(t,r){this.vfs=t;this.users=r}vfs;users;installed=new Map;registryPath="/var/lib/dpkg/status";logPath="/var/log/dpkg.log";aptLogPath="/var/log/apt/history.log";load(){if(!this.vfs.exists(this.registryPath))return;let t=this.vfs.readFile(this.registryPath);if(!t.trim())return;let r=t.split(/\n\n+/);for(let n of r){if(!n.trim())continue;let i=this.parseFields(n),s=i.Package;s&&this.installed.set(s,{name:s,version:i.Version??"unknown",architecture:i.Architecture??"amd64",maintainer:i.Maintainer??"Fortune Maintainers",description:i.Description??"",section:i.Section??"misc",installedSizeKb:Number(i["Installed-Size"]??0),installedAt:i["X-Installed-At"]??new Date().toISOString(),files:(i["X-Files"]??"").split("|").filter(Boolean)})}}persist(){let t=[];for(let r of this.installed.values())t.push([`Package: ${r.name}`,"Status: install ok installed","Priority: optional",`Section: ${r.section}`,`Installed-Size: ${r.installedSizeKb}`,`Maintainer: ${r.maintainer}`,`Architecture: ${r.architecture}`,`Version: ${r.version}`,`Description: ${r.description}`,`X-Installed-At: ${r.installedAt}`,`X-Files: ${r.files.join("|")}`].join(`
|
|
477
361
|
`));this.vfs.writeFile(this.registryPath,`${t.join(`
|
|
478
362
|
|
|
479
363
|
`)}
|
|
480
364
|
`)}parseFields(t){let r={};for(let n of t.split(`
|
|
481
365
|
`)){let i=n.indexOf(": ");i!==-1&&(r[n.slice(0,i)]=n.slice(i+2))}return r}log(t){let n=`${new Date().toISOString().replace("T"," ").slice(0,19)} ${t}
|
|
482
366
|
`,i=this.vfs.exists(this.logPath)?this.vfs.readFile(this.logPath):"";this.vfs.writeFile(this.logPath,i+n)}aptLog(t,r){let n=new Date().toISOString(),i=this.vfs.exists(this.aptLogPath)?this.vfs.readFile(this.aptLogPath):"",s=[`Start-Date: ${n}`,`Commandline: apt-get ${t} ${r.join(" ")}`,`${t==="install"?"Install":"Remove"}: ${r.join(", ")}`,`End-Date: ${n}`,""].join(`
|
|
483
|
-
`);this.vfs.writeFile(this.aptLogPath,i+s)}findInRegistry(t){return
|
|
367
|
+
`);this.vfs.writeFile(this.aptLogPath,i+s)}findInRegistry(t){return Ie.find(r=>r.name.toLowerCase()===t.toLowerCase())}listAvailable(){return[...Ie].sort((t,r)=>t.name.localeCompare(r.name))}listInstalled(){return[...this.installed.values()].sort((t,r)=>t.name.localeCompare(r.name))}isInstalled(t){return this.installed.has(t.toLowerCase())}installedCount(){return this.installed.size}install(t,r={}){let n=[],i=[],s=[],o=(l,c=new Set)=>{if(c.has(l)||(c.add(l),this.isInstalled(l)))return;let u=this.findInRegistry(l);if(!u){s.push(l);return}for(let d of u.depends??[])o(d,c);i.find(d=>d.name===u.name)||i.push(u)};for(let l of t)o(l);if(s.length>0)return{output:`E: Unable to locate package ${s.join(", ")}`,exitCode:100};if(i.length===0)return{output:t.map(l=>`${l} is already the newest version.`).join(`
|
|
484
368
|
`),exitCode:0};let a=i.reduce((l,c)=>l+(c.installedSizeKb??0),0);r.quiet||n.push("Reading package lists... Done","Building dependency tree... Done","Reading state information... Done","The following NEW packages will be installed:",` ${i.map(l=>l.name).join(" ")}`,`0 upgraded, ${i.length} newly installed, 0 to remove and 0 not upgraded.`,`Need to get 0 B/${a} kB of archives.`,`After this operation, ${a} kB of additional disk space will be used.`,"");for(let l of i){r.quiet||(n.push(`Selecting previously unselected package ${l.name}.`),n.push("(Reading database ... 12345 files and directories currently installed.)"),n.push(`Preparing to unpack .../archives/${l.name}_${l.version}_amd64.deb ...`),n.push(`Unpacking ${l.name} (${l.version}) ...`));for(let u of l.files??[]){let d=u.path.slice(0,u.path.lastIndexOf("/"));d&&!this.vfs.exists(d)&&this.vfs.mkdir(d,493),this.vfs.writeFile(u.path,u.content,{mode:u.mode??420})}l.onInstall?.(this.vfs,this.users),r.quiet||n.push(`Setting up ${l.name} (${l.version}) ...`);let c=new Date().toISOString();this.installed.set(l.name,{name:l.name,version:l.version,architecture:l.architecture??"amd64",maintainer:l.maintainer??"Fortune Maintainers <pkg@fortune.local>",description:l.description,section:l.section??"misc",installedSizeKb:l.installedSizeKb??0,installedAt:c,files:(l.files??[]).map(u=>u.path)}),this.log(`install ${l.name} ${l.version}`)}return this.aptLog("install",i.map(l=>l.name)),this.persist(),r.quiet||n.push("Processing triggers for man-db (2.11.2-2) ..."),{output:n.join(`
|
|
485
369
|
`),exitCode:0}}remove(t,r={}){let n=[],i=[];for(let s of t){let o=this.installed.get(s.toLowerCase());o?i.push(o):n.push(`Package '${s}' is not installed, so not removed`)}if(i.length===0)return{output:n.join(`
|
|
486
370
|
`)||"Nothing to remove.",exitCode:0};r.quiet||n.push("Reading package lists... Done","Building dependency tree... Done","The following packages will be REMOVED:",` ${i.map(s=>s.name).join(" ")}`,`0 upgraded, 0 newly installed, ${i.length} to remove and 0 not upgraded.`);for(let s of i){r.quiet||n.push(`Removing ${s.name} (${s.version}) ...`);for(let a of s.files)if(!(!r.purge&&(a.startsWith("/etc/")||a.endsWith(".conf"))))try{this.vfs.exists(a)&&this.vfs.remove(a)}catch{}this.findInRegistry(s.name)?.onRemove?.(this.vfs),this.installed.delete(s.name),this.log(`remove ${s.name} ${s.version}`)}return this.aptLog("remove",i.map(s=>s.name)),this.persist(),{output:n.join(`
|
|
487
|
-
`),exitCode:0}}search(t){let r=t.toLowerCase();return
|
|
488
|
-
`)}};import{createHash as
|
|
371
|
+
`),exitCode:0}}search(t){let r=t.toLowerCase();return Ie.filter(n=>n.name.includes(r)||n.description.toLowerCase().includes(r)||(n.shortDesc??"").toLowerCase().includes(r)).sort((n,i)=>n.name.localeCompare(i.name))}show(t){let r=this.findInRegistry(t);if(!r)return null;let n=this.installed.get(t);return[`Package: ${r.name}`,`Version: ${r.version}`,`Architecture: ${r.architecture??"amd64"}`,`Maintainer: ${r.maintainer??"Fortune Maintainers <pkg@fortune.local>"}`,`Installed-Size: ${r.installedSizeKb??0}`,`Depends: ${(r.depends??[]).join(", ")||"(none)"}`,`Section: ${r.section??"misc"}`,"Priority: optional",`Description: ${r.description}`,`Status: ${n?"install ok installed":"install ok not-installed"}`].join(`
|
|
372
|
+
`)}};import{createHash as fs,randomBytes as Vi,randomUUID as Ri,scryptSync as Di,timingSafeEqual as _i}from"node:crypto";import{EventEmitter as Li}from"node:events";import*as gs from"node:path";function zi(){let e=process.env.SSH_MIMIC_FAST_PASSWORD_HASH;return!!e&&!["0","false","no","off"].includes(e.toLowerCase())}var Y=le("VirtualUserManager"),de=class e extends Li{constructor(r,n=!0){super();this.vfs=r;this.autoSudoForNewUsers=n;Y.mark("constructor")}vfs;autoSudoForNewUsers;static recordCache=new Map;static fastPasswordHash=zi();usersPath="/virtual-env-js/.auth/htpasswd";sudoersPath="/virtual-env-js/.auth/sudoers";quotasPath="/virtual-env-js/.auth/quotas";authDirPath="/virtual-env-js/.auth";users=new Map;sudoers=new Set;quotas=new Map;activeSessions=new Map;nextTty=0;async initialize(){Y.mark("initialize"),this.loadFromVfs(),this.loadSudoersFromVfs(),this.loadQuotasFromVfs();let r=!1;this.users.has("root")||(this.users.set("root",this.createRecord("root","")),r=!0),this.sudoers.add("root"),r&&await this.persist(),this.emit("initialized")}async setQuotaBytes(r,n){if(Y.mark("setQuotaBytes"),this.validateUsername(r),!this.users.has(r))throw new Error(`quota: user '${r}' does not exist`);if(!Number.isFinite(n)||n<0)throw new Error("quota: maxBytes must be a non-negative number");this.quotas.set(r,Math.floor(n)),await this.persist()}async clearQuota(r){Y.mark("clearQuota"),this.validateUsername(r),this.quotas.delete(r),await this.persist()}getQuotaBytes(r){return Y.mark("getQuotaBytes"),this.quotas.get(r)??null}getUsageBytes(r){Y.mark("getUsageBytes");let n=`/home/${r}`;return this.vfs.exists(n)?this.vfs.getUsageBytes(n):0}assertWriteWithinQuota(r,n,i){Y.mark("assertWriteWithinQuota");let s=this.quotas.get(r);if(s===void 0)return;let o=hs(n),a=hs(`/home/${r}`);if(!(o===a||o.startsWith(`${a}/`)))return;let c=this.getUsageBytes(r),u=0;if(this.vfs.exists(o)){let p=this.vfs.stat(o);p.type==="file"&&(u=p.size)}let d=Buffer.isBuffer(i)?i.length:Buffer.byteLength(i,"utf8"),m=c-u+d;if(m>s)throw new Error(`quota exceeded for '${r}': ${m}/${s} bytes`)}verifyPassword(r,n){Y.mark("verifyPassword");let i=this.users.get(r);if(!i)return this.hashPassword(n,""),!1;let s=this.hashPassword(n,i.salt),o=i.passwordHash;try{let a=Buffer.from(s,"hex"),l=Buffer.from(o,"hex");return a.length!==l.length?!1:_i(a,l)}catch{return s===o}}async addUser(r,n){if(Y.mark("addUser"),this.validateUsername(r),this.validatePassword(n),this.users.has(r))return;this.users.set(r,this.createRecord(r,n)),this.autoSudoForNewUsers&&this.sudoers.add(r);let i=`/home/${r}`;this.vfs.exists(i)||(this.vfs.mkdir(i,493),this.vfs.writeFile(`${i}/README.txt`,`Welcome to the virtual environment, ${r}`)),await this.persist(),this.emit("user:add",{username:r})}getPasswordHash(r){Y.mark("getPasswordHash");let n=this.users.get(r);return n?n.passwordHash:null}async setPassword(r,n){if(Y.mark("setPassword"),this.validateUsername(r),this.validatePassword(n),!this.users.has(r))throw new Error(`passwd: user '${r}' does not exist`);this.users.set(r,this.createRecord(r,n)),await this.persist()}async deleteUser(r){if(Y.mark("deleteUser"),this.validateUsername(r),r==="root")throw new Error("deluser: cannot delete root");if(!this.users.delete(r))throw new Error(`deluser: user '${r}' does not exist`);this.sudoers.delete(r),this.emit("user:delete",{username:r}),await this.persist()}isSudoer(r){return Y.mark("isSudoer"),this.sudoers.has(r)}async addSudoer(r){if(Y.mark("addSudoer"),this.validateUsername(r),!this.users.has(r))throw new Error(`sudoers: user '${r}' does not exist`);this.sudoers.add(r),await this.persist()}async removeSudoer(r){if(Y.mark("removeSudoer"),this.validateUsername(r),r==="root")throw new Error("sudoers: cannot remove root");this.sudoers.delete(r),await this.persist()}registerSession(r,n){Y.mark("registerSession");let i={id:Ri(),username:r,tty:`pts/${this.nextTty++}`,remoteAddress:n,startedAt:new Date().toISOString()};return this.activeSessions.set(i.id,i),this.emit("session:register",{sessionId:i.id,username:r,remoteAddress:n}),i}unregisterSession(r){if(Y.mark("unregisterSession"),!r)return;let n=this.activeSessions.get(r);this.activeSessions.delete(r),n&&this.emit("session:unregister",{sessionId:r,username:n.username}),this.activeSessions.delete(r)}updateSession(r,n,i){if(Y.mark("updateSession"),!r)return;let s=this.activeSessions.get(r);s&&this.activeSessions.set(r,{...s,username:n,remoteAddress:i})}listActiveSessions(){return Y.mark("listActiveSessions"),Array.from(this.activeSessions.values()).sort((r,n)=>r.startedAt.localeCompare(n.startedAt))}listUsers(){return Array.from(this.users.keys()).sort()}loadFromVfs(){if(this.users.clear(),!this.vfs.exists(this.usersPath))return;let r=this.vfs.readFile(this.usersPath);for(let n of r.split(`
|
|
489
373
|
`)){let i=n.trim();if(i.length===0)continue;let s=i.split(":");if(s.length<3)continue;let[o,a,l]=s;!o||!a||!l||this.users.set(o,{username:o,salt:a,passwordHash:l})}}loadSudoersFromVfs(){if(this.sudoers.clear(),!this.vfs.exists(this.sudoersPath))return;let r=this.vfs.readFile(this.sudoersPath);for(let n of r.split(`
|
|
490
374
|
`)){let i=n.trim();i.length>0&&this.sudoers.add(i)}}loadQuotasFromVfs(){if(this.quotas.clear(),!this.vfs.exists(this.quotasPath))return;let r=this.vfs.readFile(this.quotasPath);for(let n of r.split(`
|
|
491
375
|
`)){let i=n.trim();if(i.length===0)continue;let[s,o]=i.split(":"),a=Number.parseInt(o??"",10);!s||!Number.isFinite(a)||a<0||this.quotas.set(s,a)}}async persist(){this.vfs.exists(this.authDirPath)||this.vfs.mkdir(this.authDirPath,448);let r=Array.from(this.users.values()).sort((o,a)=>o.username.localeCompare(a.username)).map(o=>[o.username,o.salt,o.passwordHash].join(":")).join(`
|
|
@@ -494,45 +378,45 @@ echo 'journalctl: virtual stub'
|
|
|
494
378
|
`),s=!1;s=this.writeIfChanged(this.usersPath,r.length>0?`${r}
|
|
495
379
|
`:"",384)||s,s=this.writeIfChanged(this.sudoersPath,n.length>0?`${n}
|
|
496
380
|
`:"",384)||s,s=this.writeIfChanged(this.quotasPath,i.length>0?`${i}
|
|
497
|
-
`:"",384)||s,s&&await this.vfs.flushMirror()}writeIfChanged(r,n,i){return this.vfs.exists(r)&&this.vfs.readFile(r)===n?(this.vfs.chmod(r,i),!1):(this.vfs.writeFile(r,n,{mode:i}),!0)}createRecord(r,n){let i=
|
|
498
|
-
`)){let _=
|
|
381
|
+
`:"",384)||s,s&&await this.vfs.flushMirror()}writeIfChanged(r,n,i){return this.vfs.exists(r)&&this.vfs.readFile(r)===n?(this.vfs.chmod(r,i),!1):(this.vfs.writeFile(r,n,{mode:i}),!0)}createRecord(r,n){let i=fs("sha256").update(r).update(":").update(n).digest("hex"),s=e.recordCache.get(i);if(s)return s;let o=Vi(16).toString("hex"),a={username:r,salt:o,passwordHash:this.hashPassword(n,o)};return e.recordCache.set(i,a),a}hasPassword(r){Y.mark("hasPassword");let n=this.users.get(r);if(!n)return!1;let i=this.hashPassword("",n.salt);return n.passwordHash===i?!1:!!n.passwordHash}hashPassword(r,n=""){return e.fastPasswordHash?fs("sha256").update(n).update(r).digest("hex"):Di(r,n||"",32).toString("hex")}validateUsername(r){if(!r||r.trim()==="")throw new Error("invalid username");if(!/^[a-z_][a-z0-9_-]{0,31}$/i.test(r))throw new Error("invalid username")}validatePassword(r){if(!r||r.trim()==="")throw new Error("invalid password")}authorizedKeys=new Map;addAuthorizedKey(r,n,i){Y.mark("addAuthorizedKey");let s=this.authorizedKeys.get(r)??[];s.push({algo:n,data:i}),this.authorizedKeys.set(r,s),this.emit("key:add",{username:r,algo:n})}removeAuthorizedKeys(r){this.authorizedKeys.delete(r),this.emit("key:remove",{username:r})}getAuthorizedKeys(r){return this.authorizedKeys.get(r)??[]}};function hs(e){let t=gs.posix.normalize(e);return t.startsWith("/")?t:`/${t}`}import{readFile as Ui,unlink as Ti,writeFile as Bi}from"node:fs/promises";import*as Ve from"node:path";function ys(e,t,r,n,i,s="unknown",o={cols:80,rows:24},a){let l="",c=0,u=Oi(a.vfs),d=null,m="",p=`/home/${r}`,w=At(r,n),g=null,f=null,$=()=>{let A=`/home/${r}`,D=p===A?"~":Ve.posix.basename(p)||"/";return oe(r,n,D)},E=Array.from(new Set($e())).sort();console.log(`[${i}] Shell started for user '${r}' at ${s}`),(async()=>{let A=`/home/${r}/.bashrc`;if(a.vfs.exists(A))try{let D=a.vfs.readFile(A);for(let z of D.split(`
|
|
382
|
+
`)){let _=z.trim();!_||_.startsWith("#")||await q(_,r,n,"shell",p,a,void 0,w)}}catch{}})();function M(){let A=$();t.write(`\r${A}${l}\x1B[K`);let D=l.length-c;D>0&&t.write(`\x1B[${D}D`)}function x(){t.write("\r\x1B[K")}function k(A){f={...A,buffer:""},x(),t.write(A.prompt)}async function V(A){if(!f)return;let D=f;if(f=null,!A){t.write(`\r
|
|
499
383
|
Sorry, try again.\r
|
|
500
384
|
`),M();return}if(!D.commandLine){r=D.targetUser,D.loginShell&&(p=`/home/${r}`),a.users.updateSession(i,r,s),t.write(`\r
|
|
501
|
-
`),M();return}let
|
|
502
|
-
`),_.openEditor){await P(_.openEditor.targetPath,_.openEditor.initialContent,_.openEditor.tempPath);return}if(_.openHtop){await
|
|
503
|
-
`),_.stderr&&t.write(`${
|
|
504
|
-
`),_.switchUser?(r=_.switchUser,p=_.nextCwd??`/home/${r}`,a.users.updateSession(i,r,s)):_.nextCwd&&(p=_.nextCwd),await a.vfs.flushMirror(),M()}async function b(){if(!g)return;let A=g;if(A.kind==="nano"){try{let D=await
|
|
505
|
-
`),M()}async function P(A,D,
|
|
506
|
-
`),b()}),_.on("close",()=>{b()}),g={kind:"nano",targetPath:A,tempPath:
|
|
507
|
-
`);return}let D=
|
|
508
|
-
`),b()}),D.on("close",()=>{b()}),g={kind:"htop",targetPath:"",tempPath:"",process:D}}function C(A){l=A,c=l.length,M()}function v(A){l=`${l.slice(0,c)}${A}${l.slice(c)}`,c+=A.length,M()}function N(A,D){let
|
|
385
|
+
`),M();return}let z=D.loginShell?`/home/${D.targetUser}`:p,_=await Promise.resolve(q(D.commandLine,D.targetUser,n,"shell",z,a));if(t.write(`\r
|
|
386
|
+
`),_.openEditor){await P(_.openEditor.targetPath,_.openEditor.initialContent,_.openEditor.tempPath);return}if(_.openHtop){await R();return}_.clearScreen&&t.write("\x1B[2J\x1B[H"),_.stdout&&t.write(`${Tt(_.stdout)}\r
|
|
387
|
+
`),_.stderr&&t.write(`${Tt(_.stderr)}\r
|
|
388
|
+
`),_.switchUser?(r=_.switchUser,p=_.nextCwd??`/home/${r}`,a.users.updateSession(i,r,s)):_.nextCwd&&(p=_.nextCwd),await a.vfs.flushMirror(),M()}async function b(){if(!g)return;let A=g;if(A.kind==="nano"){try{let D=await Ui(A.tempPath,"utf8");a.writeFileAsUser(r,A.targetPath,D),await a.vfs.flushMirror()}catch{}await Ti(A.tempPath).catch(()=>{})}g=null,l="",c=0,t.write(`\r
|
|
389
|
+
`),M()}async function P(A,D,z){a.vfs.exists(A)&&await Bi(z,D,"utf8");let _=se(z,o,t);_.on("error",X=>{t.write(`nano: ${X.message}\r
|
|
390
|
+
`),b()}),_.on("close",()=>{b()}),g={kind:"nano",targetPath:A,tempPath:z,process:_}}async function R(){let A=await Qn();if(!A){t.write(`htop: no child_process processes to display\r
|
|
391
|
+
`);return}let D=Xn(A,o,t);D.on("error",z=>{t.write(`htop: ${z.message}\r
|
|
392
|
+
`),b()}),D.on("close",()=>{b()}),g={kind:"htop",targetPath:"",tempPath:"",process:D}}function C(A){l=A,c=l.length,M()}function v(A){l=`${l.slice(0,c)}${A}${l.slice(c)}`,c+=A.length,M()}function N(A,D){let z=D;for(;z>0&&!/\s/.test(A[z-1]);)z-=1;let _=D;for(;_<A.length&&!/\s/.test(A[_]);)_+=1;return{start:z,end:_}}function G(A){let D=A.lastIndexOf("/"),z=D>=0?A.slice(0,D+1):"",_=D>=0?A.slice(D+1):A,X=Zn(p,z||".");try{return a.vfs.list(X).filter(U=>!U.startsWith(".")).filter(U=>U.startsWith(_)).map(U=>{let ct=Ve.posix.join(X,U),bt=a.vfs.stat(ct).type==="directory"?"/":"";return`${z}${U}${bt}`}).sort()}catch{return[]}}function B(){let{start:A,end:D}=N(l,c),z=l.slice(A,c);if(z.length===0)return;let X=l.slice(0,A).trim().length===0?E.filter(at=>at.startsWith(z)):[],U=G(z),ct=Array.from(new Set([...X,...U])).sort();if(ct.length!==0){if(ct.length===1){let at=ct[0],bt=at.endsWith("/")?"":" ";l=`${l.slice(0,A)}${at}${bt}${l.slice(D)}`,c=A+at.length+bt.length,M();return}t.write(`\r
|
|
509
393
|
`),t.write(`${ct.join(" ")}\r
|
|
510
394
|
`),M()}}function Z(A){if(A.length===0)return;u.push(A),u.length>500&&(u=u.slice(u.length-500));let D=u.length>0?`${u.join(`
|
|
511
395
|
`)}
|
|
512
|
-
`:"";a.vfs.writeFile("/virtual-env-js/.bash_history",D)}function gt(){let A=`/virtual-env-js/.lastlog/${r}.json`;if(!a.vfs.exists(A))return null;try{return JSON.parse(a.vfs.readFile(A))}catch{return null}}function lt(A){let D="/virtual-env-js/.lastlog";a.vfs.exists(D)||a.vfs.mkdir(D,448);let
|
|
396
|
+
`:"";a.vfs.writeFile("/virtual-env-js/.bash_history",D)}function gt(){let A=`/virtual-env-js/.lastlog/${r}.json`;if(!a.vfs.exists(A))return null;try{return JSON.parse(a.vfs.readFile(A))}catch{return null}}function lt(A){let D="/virtual-env-js/.lastlog";a.vfs.exists(D)||a.vfs.mkdir(D,448);let z=`${D}/${r}.json`;a.vfs.writeFile(z,JSON.stringify({at:A,from:s}))}function yt(){let A=gt(),D=new Date().toISOString();t.write(ie(n,e,A)),lt(D)}yt(),M(),t.on("data",async A=>{if(g){g.process.stdin.write(A);return}if(f){let z=A.toString("utf8");for(let _=0;_<z.length;_+=1){let X=z[_];if(X===""){f=null,t.write(`^C\r
|
|
513
397
|
`),M();return}if(X==="\x7F"||X==="\b"){f.buffer=f.buffer.slice(0,-1);continue}if(X==="\r"||X===`
|
|
514
|
-
`){let
|
|
398
|
+
`){let U=f.buffer;if(f.buffer="",f.onPassword){let{result:at,nextPrompt:bt}=await f.onPassword(U,a);t.write(`\r
|
|
515
399
|
`),at!==null?(f=null,at.stdout&&t.write(at.stdout.replace(/\n/g,`\r
|
|
516
400
|
`)),at.stderr&&t.write(at.stderr.replace(/\n/g,`\r
|
|
517
|
-
`)),M()):(bt&&(f.prompt=bt),t.write(f.prompt));return}let ct=a.users.verifyPassword(f.username,
|
|
401
|
+
`)),M()):(bt&&(f.prompt=bt),t.write(f.prompt));return}let ct=a.users.verifyPassword(f.username,U);await V(ct);return}X>=" "&&(f.buffer+=X)}return}let D=A.toString("utf8");for(let z=0;z<D.length;z+=1){let _=D[z];if(_===""){l="",c=0,d=null,m="",t.write(`bye\r
|
|
518
402
|
`),Z("bye"),await a.vfs.flushMirror(),t.write(`logout\r
|
|
519
|
-
`),t.exit(0),t.end();return}if(_===" "){
|
|
403
|
+
`),t.exit(0),t.end();return}if(_===" "){B();continue}if(_==="\x1B"){let X=D[z+1],U=D[z+2],ct=D[z+3];if(X==="["&&U){if(U==="A"){z+=2,u.length>0&&(d===null?(m=l,d=u.length-1):d>0&&(d-=1),C(u[d]??""));continue}if(U==="B"){z+=2,d!==null&&(d<u.length-1?(d+=1,C(u[d]??"")):(d=null,C(m)));continue}if(U==="C"){z+=2,c<l.length&&(c+=1,t.write("\x1B[C"));continue}if(U==="D"){z+=2,c>0&&(c-=1,t.write("\x1B[D"));continue}if(U==="3"&&ct==="~"){z+=3,c<l.length&&(l=`${l.slice(0,c)}${l.slice(c+1)}`,M());continue}}}if(_===""){l="",c=0,d=null,m="",t.write(`^C\r
|
|
520
404
|
`),M();continue}if(_==="\r"||_===`
|
|
521
405
|
`){let X=l.trim();if(l="",c=0,d=null,m="",t.write(`\r
|
|
522
|
-
`),X.length>0){let
|
|
523
|
-
`),
|
|
524
|
-
`),
|
|
525
|
-
`),t.exit(
|
|
526
|
-
`).map(n=>n.trim()).filter(n=>n.length>0):(e.writeFile(t,""),[])}function
|
|
406
|
+
`),X.length>0){let U=await Promise.resolve(q(X,r,n,"shell",p,a,void 0,w));if(Z(X),U.openEditor){await P(U.openEditor.targetPath,U.openEditor.initialContent,U.openEditor.tempPath);return}if(U.openHtop){await R();return}if(U.sudoChallenge){k(U.sudoChallenge);return}if(U.clearScreen&&t.write("\x1B[2J\x1B[H"),U.stdout&&t.write(`${Tt(U.stdout)}\r
|
|
407
|
+
`),U.stderr&&t.write(`${Tt(U.stderr)}\r
|
|
408
|
+
`),U.closeSession){t.write(`logout\r
|
|
409
|
+
`),t.exit(U.exitCode??0),t.end();return}U.nextCwd&&(p=U.nextCwd),U.switchUser&&(r=U.switchUser,p=U.nextCwd??`/home/${r}`,a.users.updateSession(i,r,s),l="",c=0),await a.vfs.flushMirror()}M();continue}if(_==="\x7F"||_==="\b"){c>0&&(l=`${l.slice(0,c-1)}${l.slice(c)}`,c-=1,M());continue}v(_)}}),t.on("close",()=>{g&&(g.process.kill("SIGTERM"),g=null)})}function Oi(e){let t="/virtual-env-js/.bash_history";return e.exists(t)?e.readFile(t).split(`
|
|
410
|
+
`).map(n=>n.trim()).filter(n=>n.length>0):(e.writeFile(t,""),[])}function ji(e){return typeof e=="object"&&e!==null&&"vfsInstance"in e&&ws(e.vfsInstance)}function ws(e){if(typeof e!="object"||e===null)return!1;let t=e;return typeof t.restoreMirror=="function"&&typeof t.flushMirror=="function"&&typeof t.writeFile=="function"&&typeof t.readFile=="function"&&typeof t.mkdir=="function"&&typeof t.exists=="function"&&typeof t.stat=="function"&&typeof t.list=="function"&&typeof t.remove=="function"&&typeof t.copy=="function"&&typeof t.move=="function"&&typeof t.touch=="function"}var Hi={kernel:"1.0.0+itsrealfortune+1-amd64",os:"Fortune GNU/Linux x64",arch:"x86_64"},Bt=le("VirtualShell");function qi(){let e=process.env.SSH_MIMIC_AUTO_SUDO_NEW_USERS;return e?!["0","false","no","off"].includes(e.toLowerCase()):!0}var me=class extends Wi{vfs;users;packageManager;hostname;properties;startTime;initialized;constructor(t,r,n){super(),Bt.mark("constructor"),this.hostname=t,this.properties=r||Hi,this.startTime=Date.now(),ws(n)?this.vfs=n:ji(n)?this.vfs=n.vfsInstance:this.vfs=new ps(n??{}),this.users=new de(this.vfs,qi()),this.packageManager=new ue(this.vfs,this.users);let i=this.vfs,s=this.users,o=this.packageManager,a=this.properties,l=this.hostname,c=this.startTime;this.initialized=(async()=>{await i.restoreMirror(),await s.initialize(),rs(i,s,l,a,c),o.load(),this.emit("initialized")})()}async ensureInitialized(){Bt.mark("ensureInitialized"),await this.initialized}addCommand(t,r,n){let i=t.trim().toLowerCase();if(i.length===0||/\s/.test(i))throw new Error("Command name must be non-empty and contain no spaces");ve(Ce(i,r,n))}executeCommand(t,r,n){Bt.mark("executeCommand"),q(t,r,this.hostname,"shell",n,this),this.emit("command",{command:t,user:r,cwd:n})}startInteractiveSession(t,r,n,i,s){Bt.mark("startInteractiveSession"),this.emit("session:start",{user:r,sessionId:n,remoteAddress:i}),ys(this.properties,t,r,this.hostname,n,i,s,this),this.refreshProcSessions()}refreshProcFs(){ae(this.vfs,this.properties,this.hostname,this.startTime,this.users.listActiveSessions())}refreshProcSessions(){ae(this.vfs,this.properties,this.hostname,this.startTime,this.users.listActiveSessions())}syncPasswd(){ke(this.vfs,this.users)}getVfs(){return this?.vfs??null}getUsers(){return this?.users??null}getHostname(){return this?.hostname}writeFileAsUser(t,r,n){Bt.mark("writeFileAsUser"),this.users.assertWriteWithinQuota(t,r,n),this.vfs.writeFile(r,n)}};var It=process.env.SSH_MIMIC_HOSTNAME??"typescript-vm",Re=process.argv.slice(2);function Yi(){for(let e=0;e<Re.length;e+=1){let t=Re[e];if(t==="--user"){let r=Re[e+1];if(!r||r.startsWith("--"))throw new Error("self-standalone: --user requires a value");return r}if(t?.startsWith("--user="))return t.slice(7)||"root"}return"root"}var Xi=Yi(),H=new me(It,void 0,{mode:"fs",snapshotPath:".vfs"});function to(e){let t=`/virtual-env-js/.lastlog/${e}.json`;if(!H.vfs.exists(t))return null;try{return JSON.parse(H.vfs.readFile(t))}catch{return null}}function Ot(e,t){return new Promise(r=>{if(!it.isTTY||!rt.isTTY){e.question(t,r);return}let n=!!it.isRaw,i="",s=()=>{it.off("data",a),n||it.setRawMode(!1),e.resume()},o=l=>{s(),rt.write(`
|
|
527
411
|
`),r(l)},a=l=>{let c=l.toString("utf8");for(let u=0;u<c.length;u+=1){let d=c[u];if(d==="\r"||d===`
|
|
528
|
-
`){o(i);return}if(d==="\x7F"||d==="\b"){i=i.slice(0,-1);continue}d>=" "&&(i+=d)}};e.pause(),rt.write(t),n||it.setRawMode(!0),it.resume(),it.on("data",a)})}function
|
|
529
|
-
`).map(t=>t.trim()).filter(t=>t.length>0):(H.vfs.writeFile(e,""),[])}function
|
|
412
|
+
`){o(i);return}if(d==="\x7F"||d==="\b"){i=i.slice(0,-1);continue}d>=" "&&(i+=d)}};e.pause(),rt.write(t),n||it.setRawMode(!0),it.resume(),it.on("data",a)})}function eo(e,t){let r="/virtual-env-js/.lastlog";H.vfs.exists(r)||H.vfs.mkdir(r,448),H.vfs.writeFile(`/virtual-env-js/.lastlog/${e}.json`,JSON.stringify({at:new Date().toISOString(),from:t}))}async function Wt(){await H.vfs.flushMirror()}function ro(){let e="/virtual-env-js/.bash_history";return H.vfs.exists(e)?H.vfs.readFile(e).split(`
|
|
413
|
+
`).map(t=>t.trim()).filter(t=>t.length>0):(H.vfs.writeFile(e,""),[])}function no(e){let t=e.length>0?`${e.join(`
|
|
530
414
|
`)}
|
|
531
|
-
`:"";H.vfs.writeFile("/virtual-env-js/.bash_history",t)}function
|
|
532
|
-
`),process.exit(1));let s=At(n,It),o=n,a=`/home/${o}`;s.vars.PWD=a;let l="localhost",c={cols:rt.columns??80,rows:rt.rows??24};async function u(f,$,
|
|
533
|
-
`),
|
|
534
|
-
`),
|
|
535
|
-
`);return}if(!f.commandLine){o=f.targetUser,a=`/home/${o}`,s.vars.USER=o,s.vars.LOGNAME=o,s.vars.HOME=`/home/${o}`,s.vars.PWD=a;return}let
|
|
415
|
+
`:"";H.vfs.writeFile("/virtual-env-js/.bash_history",t)}function so(e,t,r,n){let i=e,s=t;return r.switchUser?(i=r.switchUser,s=r.nextCwd??`/home/${i}`,n.vars.USER=i,n.vars.LOGNAME=i,n.vars.HOME=`/home/${i}`,n.vars.PWD=s):r.nextCwd&&(s=r.nextCwd,n.vars.PWD=s),{authUser:i,cwd:s}}H.addCommand("demo",[],()=>({stdout:"This is a demo command. It does nothing useful.",exitCode:0}));async function io(){let e=Qi({input:it,output:rt,terminal:!0});await H.ensureInitialized();let t=ro(),r=e;r.history=[...t].reverse();let n=Xi.trim()||"root";H.users.getPasswordHash(n)!==null||(process.stderr.write(`self-standalone: user '${n}' does not exist
|
|
416
|
+
`),process.exit(1));let s=At(n,It),o=n,a=`/home/${o}`;s.vars.PWD=a;let l="localhost",c={cols:rt.columns??80,rows:rt.rows??24};async function u(f,$,E){H.vfs.exists(f)&&await Zi(E,$,"utf8"),e.pause();let M=se(E,c,{write:rt.write.bind(rt),exit:()=>{},end:()=>{}}),x=!!it.isRaw,k=V=>{M.stdin.write(V)};it.resume(),x||it.setRawMode(!0),it.on("data",k),await new Promise(V=>{let b=()=>{it.off("data",k),x||it.setRawMode(!1),e.resume()};M.on("error",P=>{b(),rt.write(`nano: ${P.message}\r
|
|
417
|
+
`),V()}),M.on("close",async()=>{b(),e.write("",{ctrl:!0,name:"u"});try{let P=await Ki(E,"utf8");H.writeFileAsUser(o,f,P),await Wt()}catch{}await Gi(E).catch(()=>{}),rt.write(`\r
|
|
418
|
+
`),V()})})}async function d(f){if(f.onPassword){let x=f.prompt;for(;;){let k=await Ot(e,x),V=await f.onPassword(k,H);if(V.result===null){x=V.nextPrompt??x;continue}await p(V.result);return}}let $=await Ot(e,f.prompt);if(!H.users.verifyPassword(f.username,$)){process.stderr.write(`Sorry, try again.
|
|
419
|
+
`);return}if(!f.commandLine){o=f.targetUser,a=`/home/${o}`,s.vars.USER=o,s.vars.LOGNAME=o,s.vars.HOME=`/home/${o}`,s.vars.PWD=a;return}let E=f.loginShell?`/home/${f.targetUser}`:a,M=await q(f.commandLine,f.targetUser,It,"shell",E,H,void 0,s);await p(M)}async function m(f){let $=await Ot(e,f.prompt);if(f.confirmPrompt&&await Ot(e,f.confirmPrompt)!==$){process.stderr.write(`passwords do not match
|
|
536
420
|
`);return}switch(f.action){case"passwd":await H.users.setPassword(f.targetUsername,$),rt.write(`passwd: password updated successfully
|
|
537
421
|
`);break;case"adduser":if(!f.newUsername){process.stderr.write(`adduser: missing username
|
|
538
422
|
`);return}await H.users.addUser(f.newUsername,$),rt.write(`adduser: user '${f.newUsername}' created
|
|
@@ -542,7 +426,7 @@ deluser: done.
|
|
|
542
426
|
`)?f.stdout:`${f.stdout}
|
|
543
427
|
`),f.stderr&&process.stderr.write(f.stderr.endsWith(`
|
|
544
428
|
`)?f.stderr:`${f.stderr}
|
|
545
|
-
`),f.clearScreen&&(rt.write("\x1B[2J\x1B[H"),console.clear());let $=
|
|
546
|
-
`),process.exit(1))}let
|
|
547
|
-
`),e.write("",{ctrl:!0,name:"u"}),g()}),e.on("close",()=>{(async()=>(await Wt(),console.log(""),process.exit(0)))()}),rt.write(
|
|
429
|
+
`),f.clearScreen&&(rt.write("\x1B[2J\x1B[H"),console.clear());let $=so(o,a,f,s);o=$.authUser,a=$.cwd,f.closeSession&&(await Wt(),e.close(),process.exit(f.exitCode??0))}if(process.env.USER!=="root"&&H.users.hasPassword(o)){let f=await Ot(e,`Password for ${o}: `);H.users.verifyPassword(o,f)||(process.stderr.write(`self-standalone: authentication failed
|
|
430
|
+
`),process.exit(1))}let w=()=>{let f=a===`/home/${o}`?"~":Ji(a)||"/";return oe(o,It,f)},g=()=>{e.setPrompt(w()),e.prompt()};for(e.on("SIGINT",()=>{rt.write(`^C
|
|
431
|
+
`),e.write("",{ctrl:!0,name:"u"}),g()}),e.on("close",()=>{(async()=>(await Wt(),console.log(""),process.exit(0)))()}),rt.write(ie(It,H.properties,to(o))),eo(o,l),await Wt(),g();;){let f=await new Promise(E=>{e.once("line",M=>E(M))});e.pause(),f.trim().length>0&&(t.push(f),t.length>500&&(t=t.slice(t.length-500)),no(t),r.history=[...t].reverse());let $=await q(f,o,It,"shell",a,H,void 0,s);await p($),await Wt(),g(),e.resume()}}io().catch(e=>{console.error("Failed to start readline SSH emulation:",e),process.exit(1)});process.on("uncaughtException",e=>{console.log("Oh my god, something terrible happened: ",e)});process.on("unhandledRejection",(e,t)=>{console.log(" Oh Lord! We forgot to handle a promise rejection here: ",t),console.log(" The error was: ",e)});
|
|
548
432
|
//# sourceMappingURL=self-standalone.js.map
|