progy 0.1.0

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/README.md ADDED
@@ -0,0 +1,15 @@
1
+ # app
2
+
3
+ To install dependencies:
4
+
5
+ ```bash
6
+ bun install
7
+ ```
8
+
9
+ To run:
10
+
11
+ ```bash
12
+ bun run index.ts
13
+ ```
14
+
15
+ This project was created using `bun init` in bun v1.3.6. [Bun](https://bun.com) is a fast all-in-one JavaScript runtime.
@@ -0,0 +1,33 @@
1
+ // @bun
2
+ var{serve:o}=globalThis.Bun;import{readdir as p,readFile as E,writeFile as x,mkdir as y,exists as M,stat as t}from"fs/promises";import{join as v}from"path";import{spawn as u}from"child_process";import{homedir as s}from"os";var A=process.env.PROG_CWD||process.cwd(),l=v(s(),".progy"),N=v(l,"config.json"),m=v(A,"course.json"),O=v(A,".progy"),r=v(O,"exercises.json"),_=v(O,"progress.json"),P=v(import.meta.dir,import.meta.file.endsWith(".ts")?"../../public":"../public");console.log("[INFO] Server starting...");console.log(`[INFO] Working Directory: ${A}`);var e={stats:{totalXp:0,currentStreak:0,longestStreak:0,lastActiveDate:null},exercises:{},quizzes:{},achievements:[]};async function G(){try{if(await M(m)){let V=await E(m,"utf-8");return JSON.parse(V)}return null}catch(V){return console.warn(`[WARN] Failed to read course.json: ${V}`),null}}async function f(){let V=null;try{if(await M(_)){let Y=await E(_,"utf-8");V=JSON.parse(Y),console.log(`[PROGRESS] Loaded local progress. XP: ${V?.stats.totalXp}`)}else console.log(`[PROGRESS] No local progress file found at ${_}`)}catch(Y){console.warn(`[WARN] Failed to read progress.json: ${Y}`)}let X=await n();if(X?.token&&W?.id){console.log(`[PROGRESS] Checking cloud for course ${W.id}`);let Y=await XV(W.id,X.token);if(Y)if(console.log(`[SYNC] Found cloud progress. Cloud XP: ${Y.stats.totalXp}, Local XP: ${V?.stats.totalXp||0}`),!V||Y.stats.totalXp>(V?.stats.totalXp||0))console.log("[SYNC] Updating local progress with cloud data."),V=Y,await y(O,{recursive:!0}),await x(_,JSON.stringify(V,null,2));else console.log("[SYNC] Local progress is ahead or equal. Keeping local.");else console.log("[SYNC] No cloud progress found.")}return V||JSON.parse(JSON.stringify(e))}async function d(V){console.log(`[PROGRESS] Saving progress... XP: ${V.stats.totalXp}`);try{await y(O,{recursive:!0}),await x(_,JSON.stringify(V,null,2)),console.log(`[PROGRESS] Saved to ${_}`);let X=await n();if(X?.token&&W?.id)console.log("[PROGRESS] Triggering background cloud sync..."),VV(W.id,V,X.token)}catch(X){console.error(`[ERROR] Failed to save progress.json: ${X}`)}}async function n(){if(await M(N))return JSON.parse(await E(N,"utf-8"));return null}async function VV(V,X,Y){let Q=process.env.PROGY_API_URL||"https://progy.francy.workers.dev";try{console.log(`[SYNC] Syncing ${V} to cloud...`);let J=await fetch(`${Q}/api/progress/sync`,{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${Y}`},body:JSON.stringify({courseId:V,data:X})});if(J.ok)console.log(`[SYNC] Successfully synced ${V} progress.`);else console.warn(`[SYNC] Cloud sync failed: ${J.status}`)}catch(J){console.error(`[SYNC] Connection error during sync: ${J}`)}}async function XV(V,X){let Y=process.env.PROGY_API_URL||"https://progy.francy.workers.dev";try{let Q=new AbortController,J=setTimeout(()=>Q.abort(),3000),K=await fetch(`${Y}/api/progress/get?courseId=${V}`,{signal:Q.signal,headers:{Authorization:`Bearer ${X}`}});if(clearTimeout(J),K.ok)return await K.json()}catch(Q){if(Q.name==="AbortError")console.warn("[SYNC] Cloud fetch timed out after 3s.");else console.error(`[SYNC] Failed to fetch progress from cloud: ${Q}`)}return null}function c(V){let Y=new Date().toISOString().split("T")[0];if(V.lastActiveDate===Y)return V;let Q=new Date;Q.setDate(Q.getDate()-1);let J=Q.toISOString().split("T")[0];if(V.lastActiveDate===J)V.currentStreak+=1;else V.currentStreak=1;if(V.currentStreak>V.longestStreak)V.longestStreak=V.currentStreak;return V.lastActiveDate=Y,V}async function i(V){console.log("[INFO] Scanning exercises...");try{if(g&&Date.now()-a<5000)return g;let X=V.content.exercises,Y=v(A,X);if(!await M(Y))return console.warn(`[WARN] Exercises directory not found at ${Y}`),{};let J=(await p(Y)).filter(($)=>!$.startsWith(".")&&$!=="README.md"&&$!=="mod.rs"&&$!=="practice");J.sort(($,Z)=>{let z=parseInt($.split("_")[0]||"999"),k=parseInt(Z.split("_")[0]||"999");return z-k});let K={},q=($)=>$.replace(/_/g," ").replace(/([a-zA-Z])(\d+)/g,"$1 $2").replace(/\b\w/g,(Z)=>Z.toUpperCase());async function b($,Z,z,k,H,B,S){let T=q(z);if(H[z]?.title)T=H[z].title;let R=v(B,Z.name);if(Z.isDirectory()){let j=["exercise.rs","main.rs","index.ts","main.go","index.js"];for(let L of j){let U=v(R,L);if(await M(U)){R=U;break}}}if(await M(R)&&(await Bun.file(R).stat()).isFile())try{let L=(await E(R,"utf-8")).match(/\/\/\s*(?:Title|title):\s*(.+)/);if(L&&L[1])T=L[1].trim()}catch(j){}let C={id:`${$}/${Z.name}`,module:$,moduleTitle:k,name:Z.name,exerciseName:z,friendlyName:T,path:v(B,Z.name)};if(Z.isDirectory()){let j=v(B,Z.name,"quiz.json"),L=await M(j);if(L)console.log(`[DEBUG] Found quiz at ${j}`);S[$].push({...C,markdownPath:v(B,Z.name,"README.md"),hasQuiz:L,type:"directory"})}else if(Z.isFile()){if(Z.name.endsWith(".test.ts")||Z.name==="package.json")return;S[$].push({...C,markdownPath:null,type:"file"})}}for(let $ of J){let Z=v(Y,$);if((await Bun.file(Z).stat()).isDirectory()){K[$]=[];let k=await p(Z,{withFileTypes:!0}),H=q($),B={},S=v(Z,"info.toml");if(await M(S))try{let U=await E(S,"utf-8"),D=Bun.TOML.parse(U);if(D.module?.message)H=D.module.message;if(Array.isArray(D.exercises)){for(let I of D.exercises)if(I.name)B[I.name]=I}else if(D.exercises&&typeof D.exercises==="object")for(let[I,F]of Object.entries(D.exercises))B[I]=typeof F==="string"?{title:F}:F}catch(U){console.warn(`[WARN] Failed to parse info.toml in ${$}: ${U}`)}let T=(U)=>{let D=U.match(/^(\d+)_/);return D?parseInt(D[1]||"0"):9999},R=new Map;for(let U of k){if(U.name.startsWith(".")||U.name==="README.md"||U.name==="mod.rs"||U.name==="info.toml")continue;let D=U.name.split(".")[0]||"";R.set(D,U)}let C=Object.keys(B),j=new Set;for(let U of C){let D=R.get(U);if(!D)continue;j.add(U),await b($,D,U,H,B,Z,K)}let L=k.filter((U)=>{let D=U.name.split(".")[0]||"";return!j.has(D)&&!U.name.startsWith(".")&&U.name!=="README.md"&&U.name!=="mod.rs"&&U.name!=="info.toml"});L.sort((U,D)=>{let I=T(U.name||""),F=T(D.name||"");if(I!==F)return I-F;return(U.name||"").localeCompare(D.name||"")});for(let U of L){let D=U.name.split(".")[0]||"";await b($,U,D,H,B,Z,K)}}}let w=v(Y,"practice");if(await M(w)){K.practice=[];let $=await p(w);for(let Z of $)if(Z.endsWith(".rs")||Z.endsWith(".ts")||Z.endsWith(".js")||Z.endsWith(".go"))K.practice.push({id:`practice/${Z}`,module:"practice",name:Z,exerciseName:Z.split(".")[0],path:v(w,Z),markdownPath:null,type:"file"})}return await y(O,{recursive:!0}),await x(r,JSON.stringify(K,null,2)),g=K,a=Date.now(),K}catch(X){return console.error("[ERROR] Manifest generation failed:",X),{}}}async function YV(V){let X=[];for(let Y of V.checks)if(Y.type==="command")try{let Q=Y.command.split(" "),J=Q[0];if(!J)continue;let K=Q.slice(1),q=u(J,K,{stdio:"ignore"}),b=await new Promise((w)=>{q.on("close",($)=>w($===0)),q.on("error",()=>w(!1))});X.push({name:Y.name,status:b?"pass":"fail",message:b?"Found":"Not found or failed"})}catch(Q){X.push({name:Y.name,status:"fail",message:String(Q)})}return X}var h=null,W=null,g=null,a=0;G().then(async(V)=>{if(W=V,V&&V.id&&V.content&&V.content.exercises)console.log(`[INIT] Loaded config for ${V.id}`),h=await i(V);else if(V)console.error("[INIT] course.json is missing required fields (id, content.exercises)");else console.warn("[INIT] No valid course.json found")});function ZV(V,X){try{let J=null,K=V.match(/__SRP_BEGIN__\s*([\s\S]*?)\s*__SRP_END__/);if(K&&K[1])J=K[1].trim();else{let q=V.match(/\{[\s\S]*\}/g);if(q&&q.length>0)J=q[q.length-1]?.trim()??null}if(J){let q=J.indexOf("{"),b=J.lastIndexOf("}");if(q!==-1&&b!==-1){let w=J.substring(q,b+1),$=JSON.parse(w),Z=`## ${$.success?"\u2705 Success":"\u274C Failed"}
3
+
4
+ `;if(Z+=`> ${$.summary||($.success?"All checks passed":"Some issues were found")}
5
+
6
+ `,$.diagnostics&&$.diagnostics.length>0){Z+=`### \uD83D\uDCCD Diagnostics
7
+
8
+ `;for(let z of $.diagnostics){let k=z.severity==="error"?"\u274C":z.severity==="warning"?"\u26A0\uFE0F":"\u2139\uFE0F";if(Z+=`#### ${k} ${z.severity.toUpperCase()}
9
+ `,Z+=`**${z.message}**
10
+ `,z.file)Z+=`\`${z.file}:${z.line||0}:${z.column||0}\`
11
+
12
+ `;if(z.snippet)Z+=`\`\`\`rust
13
+ ${z.snippet}
14
+ \`\`\`
15
+
16
+ `;if(z.suggestion)Z+=`> \uD83D\uDCA1 **Suggestion:**
17
+ `,Z+=`> \`\`\`rust
18
+ > ${z.suggestion}
19
+ > \`\`\`
20
+
21
+ `;Z+=`---
22
+
23
+ `}}if($.tests&&$.tests.length>0){Z+=`### \uD83E\uDDEA Tests
24
+
25
+ `;for(let z of $.tests){let k=z.status==="pass"?"\u2705":"\u274C";if(Z+=`- ${k} **${z.name}**
26
+ `,z.message)Z+=` > ${z.message.replace(/\n/g,`
27
+ > `)}
28
+
29
+ `}}return{success:$.success,output:$.raw.trim(),friendlyOutput:Z}}}}catch(J){}let Y=X===0&&!V.includes("\u274C"),Q=Y?`\u2705 All tests passed!
30
+
31
+ `:`\u274C Issues detected during execution.
32
+
33
+ `;return Q+=V,{success:Y,output:V,friendlyOutput:Q}}var $V=o({port:3001,routes:{"/":()=>new Response(Bun.file(v(P,"index.html"))),"/main.js":()=>new Response(Bun.file(v(P,"main.js"))),"/main.css":()=>new Response(Bun.file(v(P,"main.css"))),"/api/progress":{async GET(){return Response.json(await f())}},"/api/progress/update":{async POST(V){try{let{type:X,id:Y,success:Q}=await V.json();if(!Y)return Response.json({success:!1,error:"Missing ID"});let J=await f(),K=new Date().toISOString();if(X==="quiz"&&Q){if(!J.quizzes[Y])J.quizzes[Y]={passed:!0,xpEarned:10,completedAt:K},J.stats.totalXp+=10,J.stats=c(J.stats),await d(J)}return Response.json({success:!0,progress:J})}catch(X){return Response.json({success:!1,error:String(X)})}}},"/api/config":{async GET(){if(!W)W=await G();return Response.json({...W||{},remoteApiUrl:process.env.PROGY_API_URL||"https://progy.francy.workers.dev"})}},"/api/exercises":{async GET(){if(!W)W=await G();if(!W)return Response.json({error:"No config"});if(h=await i(W),Array.isArray(h))return Response.json({});return Response.json(h)}},"/api/exercises/quiz":{async GET(V){let Y=new URL(V.url).searchParams.get("path");if(!Y)return new Response("Missing path",{status:400});let Q=v(Y,"quiz.json");try{if(await M(Q)){let J=await E(Q,"utf-8");return Response.json(JSON.parse(J))}return Response.json({error:"Quiz not found"},{status:404})}catch(J){return Response.json({error:"Invalid quiz file"},{status:500})}}},"/api/exercises/code":{async GET(V){let X=new URL(V.url),Y=X.searchParams.get("path"),Q=X.searchParams.get("markdownPath");if(!Y)return new Response("Missing path",{status:400});try{let J="";if((await Bun.file(Y).stat()).isDirectory()){let b=["exercise.rs","main.rs","index.ts","main.go","index.js"];for(let w of b){let $=v(Y,w);if(await M($)){J=await E($,"utf-8");break}}if(!J)J="// No entry file found"}else J=await E(Y,"utf-8");let q=null;if(Q&&await M(Q))q=await E(Q,"utf-8");return Response.json({code:J,markdown:q})}catch(J){return Response.json({error:"File not found"},{status:404})}}},"/api/exercises/run":{async POST(V){try{if(!W)W=await G();let X=await V.json(),{exerciseName:Y,id:Q}=X,K=(Q?.split("/")||[])[0]||"",q=W.runner.command,b=W.runner.args.map((Z)=>Z.replace("{{exercise}}",Y).replace("{{id}}",Q||"").replace("{{module}}",K)),$=(W.runner.cwd?v(A,W.runner.cwd):A).replace("{{exercise}}",Y).replace("{{id}}",Q||"").replace("{{module}}",K);return new Promise((Z)=>{let z=u(q,b,{cwd:$,stdio:["ignore","pipe","pipe"],env:{...process.env,FORCE_COLOR:"1"}}),k="";if(z.stdout)z.stdout.on("data",(H)=>k+=H.toString());if(z.stderr)z.stderr.on("data",(H)=>k+=H.toString());z.on("close",async(H)=>{let B=ZV(k,H||0);if(B.success&&X.id){let S=await f();if(!S.exercises[X.id])S.exercises[X.id]={status:"pass",xpEarned:20,completedAt:new Date().toISOString()},S.stats.totalXp+=20,S.stats=c(S.stats),await d(S)}Z(Response.json({success:B.success,output:B.output||"No output",friendlyOutput:B.friendlyOutput}))}),z.on("error",(H)=>Z(Response.json({success:!1,output:H.message})))})}catch(X){return Response.json({success:!1,output:String(X)})}}},"/api/ai/hint":{async POST(V){return Response.json({hint:"Thinking..."})}},"/api/setup/status":{async GET(){if(!W)W=await G();if(!W||!W.setup)return Response.json({success:!0,checks:[]});let V=await YV(W.setup),X=V.every((Y)=>Y.status==="pass");return Response.json({success:X,checks:V})}},"/api/setup/guide":{async GET(){if(!W)W=await G();if(!W||!W.setup?.guide)return Response.json({markdown:"# No setup guide available"});let V=v(A,W.setup.guide);if(await M(V)){let X=await E(V,"utf-8");return Response.json({markdown:X})}return Response.json({markdown:"# Setup guide not found"})}},"/api/auth/token":{async GET(){try{if(await M(N)){let V=JSON.parse(await E(N,"utf-8"));return console.log(`[AUTH] Local token found: ${V.token?"Yes":"No"}`),Response.json({token:V.token||null})}}catch(V){console.error(`[AUTH] Error reading global config: ${V}`)}return Response.json({token:null})},async POST(){try{if(await M(N)){let V=JSON.parse(await E(N,"utf-8"));delete V.token,await x(N,JSON.stringify(V,null,2)),console.log("[AUTH] Local token cleared.")}return Response.json({success:!0})}catch(V){return Response.json({success:!1,error:String(V)},{status:500})}}},"/api/local-settings":{async GET(){try{if(await M(N)){let V=JSON.parse(await E(N,"utf-8")),{token:X,...Y}=V;return Response.json(Y)}}catch(V){}return Response.json({})},async POST(V){try{let X=await V.json(),Y={};if(await M(N))Y=JSON.parse(await E(N,"utf-8"));else if(!await M(l))await y(l,{recursive:!0});return Object.assign(Y,X),await x(N,JSON.stringify(Y,null,2)),Response.json({success:!0})}catch(X){return Response.json({success:!1,error:String(X)},{status:500})}}},"/api/ide/open":{async POST(V){try{let{path:X}=await V.json();if(!X)return Response.json({success:!1,error:"Missing path"},{status:400});try{if((await t(X)).isDirectory()){let b=["exercise.rs","main.rs","main.go","index.ts","index.js","App.tsx"];for(let w of b){let $=v(X,w);if(await M($)){X=$;break}}}}catch(q){}let Y="vs-code";if(await M(N))try{let q=JSON.parse(await E(N,"utf-8"));if(q.ide)Y=q.ide}catch(q){}let Q="code",J=[X];switch(Y){case"vs-code":Q="code";break;case"cursor":Q="cursor";break;case"zed":Q="zed";break;case"antigravity":Q="antigravity";break;case"vim":Q="vim";break}return console.log(`[IDE] Opening ${X} with ${Q}`),u(Q,J,{detached:!0,stdio:"ignore",shell:!0}).unref(),Response.json({success:!0})}catch(X){return console.error(`[IDE] Failed to open: ${X}`),Response.json({success:!1,error:String(X)},{status:500})}}}},development:{hmr:!0},fetch(V){return new Response("Not Found",{status:404})}});console.log(`\uD83D\uDE80 Progy Server running on ${$V.url}`);
package/dist/cli.js ADDED
@@ -0,0 +1,140 @@
1
+ #!/usr/bin/env bun
2
+ // @bun
3
+ var $q=Object.create;var{getPrototypeOf:zq,defineProperty:u,getOwnPropertyNames:Rq}=Object;var Bq=Object.prototype.hasOwnProperty;var Tq=(q,J,Q)=>{Q=q!=null?$q(zq(q)):{};let X=J||!q||!q.__esModule?u(Q,"default",{value:q,enumerable:!0}):Q;for(let Y of Rq(q))if(!Bq.call(X,Y))u(X,Y,{get:()=>q[Y],enumerable:!0});return X};var S=(q,J)=>()=>(J||q((J={exports:{}}).exports,J),J.exports);var P=import.meta.require;var j=S((Uq)=>{class A extends Error{constructor(q,J,Q){super(Q);Error.captureStackTrace(this,this.constructor),this.name=this.constructor.name,this.code=J,this.exitCode=q,this.nestedError=void 0}}class g extends A{constructor(q){super(1,"commander.invalidArgument",q);Error.captureStackTrace(this,this.constructor),this.name=this.constructor.name}}Uq.CommanderError=A;Uq.InvalidArgumentError=g});var y=S((Lq)=>{var{InvalidArgumentError:Gq}=j();class c{constructor(q,J){switch(this.description=J||"",this.variadic=!1,this.parseArg=void 0,this.defaultValue=void 0,this.defaultValueDescription=void 0,this.argChoices=void 0,q[0]){case"<":this.required=!0,this._name=q.slice(1,-1);break;case"[":this.required=!1,this._name=q.slice(1,-1);break;default:this.required=!0,this._name=q;break}if(this._name.endsWith("..."))this.variadic=!0,this._name=this._name.slice(0,-3)}name(){return this._name}_collectValue(q,J){if(J===this.defaultValue||!Array.isArray(J))return[q];return J.push(q),J}default(q,J){return this.defaultValue=q,this.defaultValueDescription=J,this}argParser(q){return this.parseArg=q,this}choices(q){return this.argChoices=q.slice(),this.parseArg=(J,Q)=>{if(!this.argChoices.includes(J))throw new Gq(`Allowed choices are ${this.argChoices.join(", ")}.`);if(this.variadic)return this._collectValue(J,Q);return J},this}argRequired(){return this.required=!0,this}argOptional(){return this.required=!1,this}}function Kq(q){let J=q.name()+(q.variadic===!0?"...":"");return q.required?"<"+J+">":"["+J+"]"}Lq.Argument=c;Lq.humanReadableArgName=Kq});var k=S((Pq)=>{var{humanReadableArgName:Sq}=y();class l{constructor(){this.helpWidth=void 0,this.minWidthToWrap=40,this.sortSubcommands=!1,this.sortOptions=!1,this.showGlobalOptions=!1}prepareContext(q){this.helpWidth=this.helpWidth??q.helpWidth??80}visibleCommands(q){let J=q.commands.filter((X)=>!X._hidden),Q=q._getHelpCommand();if(Q&&!Q._hidden)J.push(Q);if(this.sortSubcommands)J.sort((X,Y)=>{return X.name().localeCompare(Y.name())});return J}compareOptions(q,J){let Q=(X)=>{return X.short?X.short.replace(/^-/,""):X.long.replace(/^--/,"")};return Q(q).localeCompare(Q(J))}visibleOptions(q){let J=q.options.filter((X)=>!X.hidden),Q=q._getHelpOption();if(Q&&!Q.hidden){let X=Q.short&&q._findOption(Q.short),Y=Q.long&&q._findOption(Q.long);if(!X&&!Y)J.push(Q);else if(Q.long&&!Y)J.push(q.createOption(Q.long,Q.description));else if(Q.short&&!X)J.push(q.createOption(Q.short,Q.description))}if(this.sortOptions)J.sort(this.compareOptions);return J}visibleGlobalOptions(q){if(!this.showGlobalOptions)return[];let J=[];for(let Q=q.parent;Q;Q=Q.parent){let X=Q.options.filter((Y)=>!Y.hidden);J.push(...X)}if(this.sortOptions)J.sort(this.compareOptions);return J}visibleArguments(q){if(q._argsDescription)q.registeredArguments.forEach((J)=>{J.description=J.description||q._argsDescription[J.name()]||""});if(q.registeredArguments.find((J)=>J.description))return q.registeredArguments;return[]}subcommandTerm(q){let J=q.registeredArguments.map((Q)=>Sq(Q)).join(" ");return q._name+(q._aliases[0]?"|"+q._aliases[0]:"")+(q.options.length?" [options]":"")+(J?" "+J:"")}optionTerm(q){return q.flags}argumentTerm(q){return q.name()}longestSubcommandTermLength(q,J){return J.visibleCommands(q).reduce((Q,X)=>{return Math.max(Q,this.displayWidth(J.styleSubcommandTerm(J.subcommandTerm(X))))},0)}longestOptionTermLength(q,J){return J.visibleOptions(q).reduce((Q,X)=>{return Math.max(Q,this.displayWidth(J.styleOptionTerm(J.optionTerm(X))))},0)}longestGlobalOptionTermLength(q,J){return J.visibleGlobalOptions(q).reduce((Q,X)=>{return Math.max(Q,this.displayWidth(J.styleOptionTerm(J.optionTerm(X))))},0)}longestArgumentTermLength(q,J){return J.visibleArguments(q).reduce((Q,X)=>{return Math.max(Q,this.displayWidth(J.styleArgumentTerm(J.argumentTerm(X))))},0)}commandUsage(q){let J=q._name;if(q._aliases[0])J=J+"|"+q._aliases[0];let Q="";for(let X=q.parent;X;X=X.parent)Q=X.name()+" "+Q;return Q+J+" "+q.usage()}commandDescription(q){return q.description()}subcommandDescription(q){return q.summary()||q.description()}optionDescription(q){let J=[];if(q.argChoices)J.push(`choices: ${q.argChoices.map((Q)=>JSON.stringify(Q)).join(", ")}`);if(q.defaultValue!==void 0){if(q.required||q.optional||q.isBoolean()&&typeof q.defaultValue==="boolean")J.push(`default: ${q.defaultValueDescription||JSON.stringify(q.defaultValue)}`)}if(q.presetArg!==void 0&&q.optional)J.push(`preset: ${JSON.stringify(q.presetArg)}`);if(q.envVar!==void 0)J.push(`env: ${q.envVar}`);if(J.length>0){let Q=`(${J.join(", ")})`;if(q.description)return`${q.description} ${Q}`;return Q}return q.description}argumentDescription(q){let J=[];if(q.argChoices)J.push(`choices: ${q.argChoices.map((Q)=>JSON.stringify(Q)).join(", ")}`);if(q.defaultValue!==void 0)J.push(`default: ${q.defaultValueDescription||JSON.stringify(q.defaultValue)}`);if(J.length>0){let Q=`(${J.join(", ")})`;if(q.description)return`${q.description} ${Q}`;return Q}return q.description}formatItemList(q,J,Q){if(J.length===0)return[];return[Q.styleTitle(q),...J,""]}groupItems(q,J,Q){let X=new Map;return q.forEach((Y)=>{let Z=Q(Y);if(!X.has(Z))X.set(Z,[])}),J.forEach((Y)=>{let Z=Q(Y);if(!X.has(Z))X.set(Z,[]);X.get(Z).push(Y)}),X}formatHelp(q,J){let Q=J.padWidth(q,J),X=J.helpWidth??80;function Y(B,U){return J.formatItem(B,Q,U,J)}let Z=[`${J.styleTitle("Usage:")} ${J.styleUsage(J.commandUsage(q))}`,""],$=J.commandDescription(q);if($.length>0)Z=Z.concat([J.boxWrap(J.styleCommandDescription($),X),""]);let R=J.visibleArguments(q).map((B)=>{return Y(J.styleArgumentTerm(J.argumentTerm(B)),J.styleArgumentDescription(J.argumentDescription(B)))});if(Z=Z.concat(this.formatItemList("Arguments:",R,J)),this.groupItems(q.options,J.visibleOptions(q),(B)=>B.helpGroupHeading??"Options:").forEach((B,U)=>{let K=B.map((G)=>{return Y(J.styleOptionTerm(J.optionTerm(G)),J.styleOptionDescription(J.optionDescription(G)))});Z=Z.concat(this.formatItemList(U,K,J))}),J.showGlobalOptions){let B=J.visibleGlobalOptions(q).map((U)=>{return Y(J.styleOptionTerm(J.optionTerm(U)),J.styleOptionDescription(J.optionDescription(U)))});Z=Z.concat(this.formatItemList("Global Options:",B,J))}return this.groupItems(q.commands,J.visibleCommands(q),(B)=>B.helpGroup()||"Commands:").forEach((B,U)=>{let K=B.map((G)=>{return Y(J.styleSubcommandTerm(J.subcommandTerm(G)),J.styleSubcommandDescription(J.subcommandDescription(G)))});Z=Z.concat(this.formatItemList(U,K,J))}),Z.join(`
4
+ `)}displayWidth(q){return m(q).length}styleTitle(q){return q}styleUsage(q){return q.split(" ").map((J)=>{if(J==="[options]")return this.styleOptionText(J);if(J==="[command]")return this.styleSubcommandText(J);if(J[0]==="["||J[0]==="<")return this.styleArgumentText(J);return this.styleCommandText(J)}).join(" ")}styleCommandDescription(q){return this.styleDescriptionText(q)}styleOptionDescription(q){return this.styleDescriptionText(q)}styleSubcommandDescription(q){return this.styleDescriptionText(q)}styleArgumentDescription(q){return this.styleDescriptionText(q)}styleDescriptionText(q){return q}styleOptionTerm(q){return this.styleOptionText(q)}styleSubcommandTerm(q){return q.split(" ").map((J)=>{if(J==="[options]")return this.styleOptionText(J);if(J[0]==="["||J[0]==="<")return this.styleArgumentText(J);return this.styleSubcommandText(J)}).join(" ")}styleArgumentTerm(q){return this.styleArgumentText(q)}styleOptionText(q){return q}styleArgumentText(q){return q}styleSubcommandText(q){return q}styleCommandText(q){return q}padWidth(q,J){return Math.max(J.longestOptionTermLength(q,J),J.longestGlobalOptionTermLength(q,J),J.longestSubcommandTermLength(q,J),J.longestArgumentTermLength(q,J))}preformatted(q){return/\n[^\S\r\n]/.test(q)}formatItem(q,J,Q,X){let Z=" ".repeat(2);if(!Q)return Z+q;let $=q.padEnd(J+q.length-X.displayWidth(q)),R=2,z=(this.helpWidth??80)-J-R-2,B;if(z<this.minWidthToWrap||X.preformatted(Q))B=Q;else B=X.boxWrap(Q,z).replace(/\n/g,`
5
+ `+" ".repeat(J+R));return Z+$+" ".repeat(R)+B.replace(/\n/g,`
6
+ ${Z}`)}boxWrap(q,J){if(J<this.minWidthToWrap)return q;let Q=q.split(/\r\n|\n/),X=/[\s]*[^\s]+/g,Y=[];return Q.forEach((Z)=>{let $=Z.match(X);if($===null){Y.push("");return}let R=[$.shift()],T=this.displayWidth(R[0]);$.forEach((z)=>{let B=this.displayWidth(z);if(T+B<=J){R.push(z),T+=B;return}Y.push(R.join(""));let U=z.trimStart();R=[U],T=this.displayWidth(U)}),Y.push(R.join(""))}),Y.join(`
7
+ `)}}function m(q){let J=/\x1b\[\d*(;\d*)*m/g;return q.replace(J,"")}Pq.Help=l;Pq.stripColor=m});var w=S((yq)=>{var{InvalidArgumentError:jq}=j();class p{constructor(q,J){this.flags=q,this.description=J||"",this.required=q.includes("<"),this.optional=q.includes("["),this.variadic=/\w\.\.\.[>\]]$/.test(q),this.mandatory=!1;let Q=Eq(q);if(this.short=Q.shortFlag,this.long=Q.longFlag,this.negate=!1,this.long)this.negate=this.long.startsWith("--no-");this.defaultValue=void 0,this.defaultValueDescription=void 0,this.presetArg=void 0,this.envVar=void 0,this.parseArg=void 0,this.hidden=!1,this.argChoices=void 0,this.conflictsWith=[],this.implied=void 0,this.helpGroupHeading=void 0}default(q,J){return this.defaultValue=q,this.defaultValueDescription=J,this}preset(q){return this.presetArg=q,this}conflicts(q){return this.conflictsWith=this.conflictsWith.concat(q),this}implies(q){let J=q;if(typeof q==="string")J={[q]:!0};return this.implied=Object.assign(this.implied||{},J),this}env(q){return this.envVar=q,this}argParser(q){return this.parseArg=q,this}makeOptionMandatory(q=!0){return this.mandatory=!!q,this}hideHelp(q=!0){return this.hidden=!!q,this}_collectValue(q,J){if(J===this.defaultValue||!Array.isArray(J))return[q];return J.push(q),J}choices(q){return this.argChoices=q.slice(),this.parseArg=(J,Q)=>{if(!this.argChoices.includes(J))throw new jq(`Allowed choices are ${this.argChoices.join(", ")}.`);if(this.variadic)return this._collectValue(J,Q);return J},this}name(){if(this.long)return this.long.replace(/^--/,"");return this.short.replace(/^-/,"")}attributeName(){if(this.negate)return d(this.name().replace(/^no-/,""));return d(this.name())}helpGroup(q){return this.helpGroupHeading=q,this}is(q){return this.short===q||this.long===q}isBoolean(){return!this.required&&!this.optional&&!this.negate}}class s{constructor(q){this.positiveOptions=new Map,this.negativeOptions=new Map,this.dualOptions=new Set,q.forEach((J)=>{if(J.negate)this.negativeOptions.set(J.attributeName(),J);else this.positiveOptions.set(J.attributeName(),J)}),this.negativeOptions.forEach((J,Q)=>{if(this.positiveOptions.has(Q))this.dualOptions.add(Q)})}valueFromOption(q,J){let Q=J.attributeName();if(!this.dualOptions.has(Q))return!0;let X=this.negativeOptions.get(Q).presetArg,Y=X!==void 0?X:!1;return J.negate===(Y===q)}}function d(q){return q.split("-").reduce((J,Q)=>{return J+Q[0].toUpperCase()+Q.slice(1)})}function Eq(q){let J,Q,X=/^-[^-]$/,Y=/^--[^-]/,Z=q.split(/[ |,]+/).concat("guard");if(X.test(Z[0]))J=Z.shift();if(Y.test(Z[0]))Q=Z.shift();if(!J&&X.test(Z[0]))J=Z.shift();if(!J&&Y.test(Z[0]))J=Q,Q=Z.shift();if(Z[0].startsWith("-")){let $=Z[0],R=`option creation failed due to '${$}' in option flags '${q}'`;if(/^-[^-][^-]/.test($))throw Error(`${R}
8
+ - a short flag is a single dash and a single character
9
+ - either use a single dash and a single character (for a short flag)
10
+ - or use a double dash for a long option (and can have two, like '--ws, --workspace')`);if(X.test($))throw Error(`${R}
11
+ - too many short flags`);if(Y.test($))throw Error(`${R}
12
+ - too many long flags`);throw Error(`${R}
13
+ - unrecognised flag format`)}if(J===void 0&&Q===void 0)throw Error(`option creation failed due to no flags found in '${q}'.`);return{shortFlag:J,longFlag:Q}}yq.Option=p;yq.DualOptions=s});var i=S((kq)=>{function Fq(q,J){if(Math.abs(q.length-J.length)>3)return Math.max(q.length,J.length);let Q=[];for(let X=0;X<=q.length;X++)Q[X]=[X];for(let X=0;X<=J.length;X++)Q[0][X]=X;for(let X=1;X<=J.length;X++)for(let Y=1;Y<=q.length;Y++){let Z=1;if(q[Y-1]===J[X-1])Z=0;else Z=1;if(Q[Y][X]=Math.min(Q[Y-1][X]+1,Q[Y][X-1]+1,Q[Y-1][X-1]+Z),Y>1&&X>1&&q[Y-1]===J[X-2]&&q[Y-2]===J[X-1])Q[Y][X]=Math.min(Q[Y][X],Q[Y-2][X-2]+1)}return Q[q.length][J.length]}function Aq(q,J){if(!J||J.length===0)return"";J=Array.from(new Set(J));let Q=q.startsWith("--");if(Q)q=q.slice(2),J=J.map(($)=>$.slice(2));let X=[],Y=3,Z=0.4;if(J.forEach(($)=>{if($.length<=1)return;let R=Fq(q,$),T=Math.max(q.length,$.length);if((T-R)/T>Z){if(R<Y)Y=R,X=[$];else if(R===Y)X.push($)}}),X.sort(($,R)=>$.localeCompare(R)),Q)X=X.map(($)=>`--${$}`);if(X.length>1)return`
14
+ (Did you mean one of ${X.join(", ")}?)`;if(X.length===1)return`
15
+ (Did you mean ${X[0]}?)`;return""}kq.suggestSimilar=Aq});var a=S((hq)=>{var xq=P("events").EventEmitter,x=P("child_process"),L=P("path"),N=P("fs"),H=P("process"),{Argument:Oq,humanReadableArgName:Cq}=y(),{CommanderError:O}=j(),{Help:bq,stripColor:fq}=k(),{Option:r,DualOptions:vq}=w(),{suggestSimilar:t}=i();class b extends xq{constructor(q){super();this.commands=[],this.options=[],this.parent=null,this._allowUnknownOption=!1,this._allowExcessArguments=!1,this.registeredArguments=[],this._args=this.registeredArguments,this.args=[],this.rawArgs=[],this.processedArgs=[],this._scriptPath=null,this._name=q||"",this._optionValues={},this._optionValueSources={},this._storeOptionsAsProperties=!1,this._actionHandler=null,this._executableHandler=!1,this._executableFile=null,this._executableDir=null,this._defaultCommandName=null,this._exitCallback=null,this._aliases=[],this._combineFlagAndOptionalValue=!0,this._description="",this._summary="",this._argsDescription=void 0,this._enablePositionalOptions=!1,this._passThroughOptions=!1,this._lifeCycleHooks={},this._showHelpAfterError=!1,this._showSuggestionAfterError=!0,this._savedState=null,this._outputConfiguration={writeOut:(J)=>H.stdout.write(J),writeErr:(J)=>H.stderr.write(J),outputError:(J,Q)=>Q(J),getOutHelpWidth:()=>H.stdout.isTTY?H.stdout.columns:void 0,getErrHelpWidth:()=>H.stderr.isTTY?H.stderr.columns:void 0,getOutHasColors:()=>C()??(H.stdout.isTTY&&H.stdout.hasColors?.()),getErrHasColors:()=>C()??(H.stderr.isTTY&&H.stderr.hasColors?.()),stripColor:(J)=>fq(J)},this._hidden=!1,this._helpOption=void 0,this._addImplicitHelpCommand=void 0,this._helpCommand=void 0,this._helpConfiguration={},this._helpGroupHeading=void 0,this._defaultCommandGroup=void 0,this._defaultOptionGroup=void 0}copyInheritedSettings(q){return this._outputConfiguration=q._outputConfiguration,this._helpOption=q._helpOption,this._helpCommand=q._helpCommand,this._helpConfiguration=q._helpConfiguration,this._exitCallback=q._exitCallback,this._storeOptionsAsProperties=q._storeOptionsAsProperties,this._combineFlagAndOptionalValue=q._combineFlagAndOptionalValue,this._allowExcessArguments=q._allowExcessArguments,this._enablePositionalOptions=q._enablePositionalOptions,this._showHelpAfterError=q._showHelpAfterError,this._showSuggestionAfterError=q._showSuggestionAfterError,this}_getCommandAndAncestors(){let q=[];for(let J=this;J;J=J.parent)q.push(J);return q}command(q,J,Q){let X=J,Y=Q;if(typeof X==="object"&&X!==null)Y=X,X=null;Y=Y||{};let[,Z,$]=q.match(/([^ ]+) *(.*)/),R=this.createCommand(Z);if(X)R.description(X),R._executableHandler=!0;if(Y.isDefault)this._defaultCommandName=R._name;if(R._hidden=!!(Y.noHelp||Y.hidden),R._executableFile=Y.executableFile||null,$)R.arguments($);if(this._registerCommand(R),R.parent=this,R.copyInheritedSettings(this),X)return this;return R}createCommand(q){return new b(q)}createHelp(){return Object.assign(new bq,this.configureHelp())}configureHelp(q){if(q===void 0)return this._helpConfiguration;return this._helpConfiguration=q,this}configureOutput(q){if(q===void 0)return this._outputConfiguration;return this._outputConfiguration={...this._outputConfiguration,...q},this}showHelpAfterError(q=!0){if(typeof q!=="string")q=!!q;return this._showHelpAfterError=q,this}showSuggestionAfterError(q=!0){return this._showSuggestionAfterError=!!q,this}addCommand(q,J){if(!q._name)throw Error(`Command passed to .addCommand() must have a name
16
+ - specify the name in Command constructor or using .name()`);if(J=J||{},J.isDefault)this._defaultCommandName=q._name;if(J.noHelp||J.hidden)q._hidden=!0;return this._registerCommand(q),q.parent=this,q._checkForBrokenPassThrough(),this}createArgument(q,J){return new Oq(q,J)}argument(q,J,Q,X){let Y=this.createArgument(q,J);if(typeof Q==="function")Y.default(X).argParser(Q);else Y.default(Q);return this.addArgument(Y),this}arguments(q){return q.trim().split(/ +/).forEach((J)=>{this.argument(J)}),this}addArgument(q){let J=this.registeredArguments.slice(-1)[0];if(J?.variadic)throw Error(`only the last argument can be variadic '${J.name()}'`);if(q.required&&q.defaultValue!==void 0&&q.parseArg===void 0)throw Error(`a default value for a required argument is never used: '${q.name()}'`);return this.registeredArguments.push(q),this}helpCommand(q,J){if(typeof q==="boolean"){if(this._addImplicitHelpCommand=q,q&&this._defaultCommandGroup)this._initCommandGroup(this._getHelpCommand());return this}let Q=q??"help [command]",[,X,Y]=Q.match(/([^ ]+) *(.*)/),Z=J??"display help for command",$=this.createCommand(X);if($.helpOption(!1),Y)$.arguments(Y);if(Z)$.description(Z);if(this._addImplicitHelpCommand=!0,this._helpCommand=$,q||J)this._initCommandGroup($);return this}addHelpCommand(q,J){if(typeof q!=="object")return this.helpCommand(q,J),this;return this._addImplicitHelpCommand=!0,this._helpCommand=q,this._initCommandGroup(q),this}_getHelpCommand(){if(this._addImplicitHelpCommand??(this.commands.length&&!this._actionHandler&&!this._findCommand("help"))){if(this._helpCommand===void 0)this.helpCommand(void 0,void 0);return this._helpCommand}return null}hook(q,J){let Q=["preSubcommand","preAction","postAction"];if(!Q.includes(q))throw Error(`Unexpected value for event passed to hook : '${q}'.
17
+ Expecting one of '${Q.join("', '")}'`);if(this._lifeCycleHooks[q])this._lifeCycleHooks[q].push(J);else this._lifeCycleHooks[q]=[J];return this}exitOverride(q){if(q)this._exitCallback=q;else this._exitCallback=(J)=>{if(J.code!=="commander.executeSubCommandAsync")throw J};return this}_exit(q,J,Q){if(this._exitCallback)this._exitCallback(new O(q,J,Q));H.exit(q)}action(q){let J=(Q)=>{let X=this.registeredArguments.length,Y=Q.slice(0,X);if(this._storeOptionsAsProperties)Y[X]=this;else Y[X]=this.opts();return Y.push(this),q.apply(this,Y)};return this._actionHandler=J,this}createOption(q,J){return new r(q,J)}_callParseArg(q,J,Q,X){try{return q.parseArg(J,Q)}catch(Y){if(Y.code==="commander.invalidArgument"){let Z=`${X} ${Y.message}`;this.error(Z,{exitCode:Y.exitCode,code:Y.code})}throw Y}}_registerOption(q){let J=q.short&&this._findOption(q.short)||q.long&&this._findOption(q.long);if(J){let Q=q.long&&this._findOption(q.long)?q.long:q.short;throw Error(`Cannot add option '${q.flags}'${this._name&&` to command '${this._name}'`} due to conflicting flag '${Q}'
18
+ - already used by option '${J.flags}'`)}this._initOptionGroup(q),this.options.push(q)}_registerCommand(q){let J=(X)=>{return[X.name()].concat(X.aliases())},Q=J(q).find((X)=>this._findCommand(X));if(Q){let X=J(this._findCommand(Q)).join("|"),Y=J(q).join("|");throw Error(`cannot add command '${Y}' as already have command '${X}'`)}this._initCommandGroup(q),this.commands.push(q)}addOption(q){this._registerOption(q);let J=q.name(),Q=q.attributeName();if(q.negate){let Y=q.long.replace(/^--no-/,"--");if(!this._findOption(Y))this.setOptionValueWithSource(Q,q.defaultValue===void 0?!0:q.defaultValue,"default")}else if(q.defaultValue!==void 0)this.setOptionValueWithSource(Q,q.defaultValue,"default");let X=(Y,Z,$)=>{if(Y==null&&q.presetArg!==void 0)Y=q.presetArg;let R=this.getOptionValue(Q);if(Y!==null&&q.parseArg)Y=this._callParseArg(q,Y,R,Z);else if(Y!==null&&q.variadic)Y=q._collectValue(Y,R);if(Y==null)if(q.negate)Y=!1;else if(q.isBoolean()||q.optional)Y=!0;else Y="";this.setOptionValueWithSource(Q,Y,$)};if(this.on("option:"+J,(Y)=>{let Z=`error: option '${q.flags}' argument '${Y}' is invalid.`;X(Y,Z,"cli")}),q.envVar)this.on("optionEnv:"+J,(Y)=>{let Z=`error: option '${q.flags}' value '${Y}' from env '${q.envVar}' is invalid.`;X(Y,Z,"env")});return this}_optionEx(q,J,Q,X,Y){if(typeof J==="object"&&J instanceof r)throw Error("To add an Option object use addOption() instead of option() or requiredOption()");let Z=this.createOption(J,Q);if(Z.makeOptionMandatory(!!q.mandatory),typeof X==="function")Z.default(Y).argParser(X);else if(X instanceof RegExp){let $=X;X=(R,T)=>{let z=$.exec(R);return z?z[0]:T},Z.default(Y).argParser(X)}else Z.default(X);return this.addOption(Z)}option(q,J,Q,X){return this._optionEx({},q,J,Q,X)}requiredOption(q,J,Q,X){return this._optionEx({mandatory:!0},q,J,Q,X)}combineFlagAndOptionalValue(q=!0){return this._combineFlagAndOptionalValue=!!q,this}allowUnknownOption(q=!0){return this._allowUnknownOption=!!q,this}allowExcessArguments(q=!0){return this._allowExcessArguments=!!q,this}enablePositionalOptions(q=!0){return this._enablePositionalOptions=!!q,this}passThroughOptions(q=!0){return this._passThroughOptions=!!q,this._checkForBrokenPassThrough(),this}_checkForBrokenPassThrough(){if(this.parent&&this._passThroughOptions&&!this.parent._enablePositionalOptions)throw Error(`passThroughOptions cannot be used for '${this._name}' without turning on enablePositionalOptions for parent command(s)`)}storeOptionsAsProperties(q=!0){if(this.options.length)throw Error("call .storeOptionsAsProperties() before adding options");if(Object.keys(this._optionValues).length)throw Error("call .storeOptionsAsProperties() before setting option values");return this._storeOptionsAsProperties=!!q,this}getOptionValue(q){if(this._storeOptionsAsProperties)return this[q];return this._optionValues[q]}setOptionValue(q,J){return this.setOptionValueWithSource(q,J,void 0)}setOptionValueWithSource(q,J,Q){if(this._storeOptionsAsProperties)this[q]=J;else this._optionValues[q]=J;return this._optionValueSources[q]=Q,this}getOptionValueSource(q){return this._optionValueSources[q]}getOptionValueSourceWithGlobals(q){let J;return this._getCommandAndAncestors().forEach((Q)=>{if(Q.getOptionValueSource(q)!==void 0)J=Q.getOptionValueSource(q)}),J}_prepareUserArgs(q,J){if(q!==void 0&&!Array.isArray(q))throw Error("first parameter to parse must be array or undefined");if(J=J||{},q===void 0&&J.from===void 0){if(H.versions?.electron)J.from="electron";let X=H.execArgv??[];if(X.includes("-e")||X.includes("--eval")||X.includes("-p")||X.includes("--print"))J.from="eval"}if(q===void 0)q=H.argv;this.rawArgs=q.slice();let Q;switch(J.from){case void 0:case"node":this._scriptPath=q[1],Q=q.slice(2);break;case"electron":if(H.defaultApp)this._scriptPath=q[1],Q=q.slice(2);else Q=q.slice(1);break;case"user":Q=q.slice(0);break;case"eval":Q=q.slice(1);break;default:throw Error(`unexpected parse option { from: '${J.from}' }`)}if(!this._name&&this._scriptPath)this.nameFromFilename(this._scriptPath);return this._name=this._name||"program",Q}parse(q,J){this._prepareForParse();let Q=this._prepareUserArgs(q,J);return this._parseCommand([],Q),this}async parseAsync(q,J){this._prepareForParse();let Q=this._prepareUserArgs(q,J);return await this._parseCommand([],Q),this}_prepareForParse(){if(this._savedState===null)this.saveStateBeforeParse();else this.restoreStateBeforeParse()}saveStateBeforeParse(){this._savedState={_name:this._name,_optionValues:{...this._optionValues},_optionValueSources:{...this._optionValueSources}}}restoreStateBeforeParse(){if(this._storeOptionsAsProperties)throw Error(`Can not call parse again when storeOptionsAsProperties is true.
19
+ - either make a new Command for each call to parse, or stop storing options as properties`);this._name=this._savedState._name,this._scriptPath=null,this.rawArgs=[],this._optionValues={...this._savedState._optionValues},this._optionValueSources={...this._savedState._optionValueSources},this.args=[],this.processedArgs=[]}_checkForMissingExecutable(q,J,Q){if(N.existsSync(q))return;let X=J?`searched for local subcommand relative to directory '${J}'`:"no directory for search for local subcommand, use .executableDir() to supply a custom directory",Y=`'${q}' does not exist
20
+ - if '${Q}' is not meant to be an executable command, remove description parameter from '.command()' and use '.description()' instead
21
+ - if the default executable name is not suitable, use the executableFile option to supply a custom name or path
22
+ - ${X}`;throw Error(Y)}_executeSubCommand(q,J){J=J.slice();let Q=!1,X=[".js",".ts",".tsx",".mjs",".cjs"];function Y(z,B){let U=L.resolve(z,B);if(N.existsSync(U))return U;if(X.includes(L.extname(B)))return;let K=X.find((G)=>N.existsSync(`${U}${G}`));if(K)return`${U}${K}`;return}this._checkForMissingMandatoryOptions(),this._checkForConflictingOptions();let Z=q._executableFile||`${this._name}-${q._name}`,$=this._executableDir||"";if(this._scriptPath){let z;try{z=N.realpathSync(this._scriptPath)}catch{z=this._scriptPath}$=L.resolve(L.dirname(z),$)}if($){let z=Y($,Z);if(!z&&!q._executableFile&&this._scriptPath){let B=L.basename(this._scriptPath,L.extname(this._scriptPath));if(B!==this._name)z=Y($,`${B}-${q._name}`)}Z=z||Z}Q=X.includes(L.extname(Z));let R;if(H.platform!=="win32")if(Q)J.unshift(Z),J=n(H.execArgv).concat(J),R=x.spawn(H.argv[0],J,{stdio:"inherit"});else R=x.spawn(Z,J,{stdio:"inherit"});else this._checkForMissingExecutable(Z,$,q._name),J.unshift(Z),J=n(H.execArgv).concat(J),R=x.spawn(H.execPath,J,{stdio:"inherit"});if(!R.killed)["SIGUSR1","SIGUSR2","SIGTERM","SIGINT","SIGHUP"].forEach((B)=>{H.on(B,()=>{if(R.killed===!1&&R.exitCode===null)R.kill(B)})});let T=this._exitCallback;R.on("close",(z)=>{if(z=z??1,!T)H.exit(z);else T(new O(z,"commander.executeSubCommandAsync","(close)"))}),R.on("error",(z)=>{if(z.code==="ENOENT")this._checkForMissingExecutable(Z,$,q._name);else if(z.code==="EACCES")throw Error(`'${Z}' not executable`);if(!T)H.exit(1);else{let B=new O(1,"commander.executeSubCommandAsync","(error)");B.nestedError=z,T(B)}}),this.runningCommand=R}_dispatchSubcommand(q,J,Q){let X=this._findCommand(q);if(!X)this.help({error:!0});X._prepareForParse();let Y;return Y=this._chainOrCallSubCommandHook(Y,X,"preSubcommand"),Y=this._chainOrCall(Y,()=>{if(X._executableHandler)this._executeSubCommand(X,J.concat(Q));else return X._parseCommand(J,Q)}),Y}_dispatchHelpCommand(q){if(!q)this.help();let J=this._findCommand(q);if(J&&!J._executableHandler)J.help();return this._dispatchSubcommand(q,[],[this._getHelpOption()?.long??this._getHelpOption()?.short??"--help"])}_checkNumberOfArguments(){if(this.registeredArguments.forEach((q,J)=>{if(q.required&&this.args[J]==null)this.missingArgument(q.name())}),this.registeredArguments.length>0&&this.registeredArguments[this.registeredArguments.length-1].variadic)return;if(this.args.length>this.registeredArguments.length)this._excessArguments(this.args)}_processArguments(){let q=(Q,X,Y)=>{let Z=X;if(X!==null&&Q.parseArg){let $=`error: command-argument value '${X}' is invalid for argument '${Q.name()}'.`;Z=this._callParseArg(Q,X,Y,$)}return Z};this._checkNumberOfArguments();let J=[];this.registeredArguments.forEach((Q,X)=>{let Y=Q.defaultValue;if(Q.variadic){if(X<this.args.length){if(Y=this.args.slice(X),Q.parseArg)Y=Y.reduce((Z,$)=>{return q(Q,$,Z)},Q.defaultValue)}else if(Y===void 0)Y=[]}else if(X<this.args.length){if(Y=this.args[X],Q.parseArg)Y=q(Q,Y,Q.defaultValue)}J[X]=Y}),this.processedArgs=J}_chainOrCall(q,J){if(q?.then&&typeof q.then==="function")return q.then(()=>J());return J()}_chainOrCallHooks(q,J){let Q=q,X=[];if(this._getCommandAndAncestors().reverse().filter((Y)=>Y._lifeCycleHooks[J]!==void 0).forEach((Y)=>{Y._lifeCycleHooks[J].forEach((Z)=>{X.push({hookedCommand:Y,callback:Z})})}),J==="postAction")X.reverse();return X.forEach((Y)=>{Q=this._chainOrCall(Q,()=>{return Y.callback(Y.hookedCommand,this)})}),Q}_chainOrCallSubCommandHook(q,J,Q){let X=q;if(this._lifeCycleHooks[Q]!==void 0)this._lifeCycleHooks[Q].forEach((Y)=>{X=this._chainOrCall(X,()=>{return Y(this,J)})});return X}_parseCommand(q,J){let Q=this.parseOptions(J);if(this._parseOptionsEnv(),this._parseOptionsImplied(),q=q.concat(Q.operands),J=Q.unknown,this.args=q.concat(J),q&&this._findCommand(q[0]))return this._dispatchSubcommand(q[0],q.slice(1),J);if(this._getHelpCommand()&&q[0]===this._getHelpCommand().name())return this._dispatchHelpCommand(q[1]);if(this._defaultCommandName)return this._outputHelpIfRequested(J),this._dispatchSubcommand(this._defaultCommandName,q,J);if(this.commands.length&&this.args.length===0&&!this._actionHandler&&!this._defaultCommandName)this.help({error:!0});this._outputHelpIfRequested(Q.unknown),this._checkForMissingMandatoryOptions(),this._checkForConflictingOptions();let X=()=>{if(Q.unknown.length>0)this.unknownOption(Q.unknown[0])},Y=`command:${this.name()}`;if(this._actionHandler){X(),this._processArguments();let Z;if(Z=this._chainOrCallHooks(Z,"preAction"),Z=this._chainOrCall(Z,()=>this._actionHandler(this.processedArgs)),this.parent)Z=this._chainOrCall(Z,()=>{this.parent.emit(Y,q,J)});return Z=this._chainOrCallHooks(Z,"postAction"),Z}if(this.parent?.listenerCount(Y))X(),this._processArguments(),this.parent.emit(Y,q,J);else if(q.length){if(this._findCommand("*"))return this._dispatchSubcommand("*",q,J);if(this.listenerCount("command:*"))this.emit("command:*",q,J);else if(this.commands.length)this.unknownCommand();else X(),this._processArguments()}else if(this.commands.length)X(),this.help({error:!0});else X(),this._processArguments()}_findCommand(q){if(!q)return;return this.commands.find((J)=>J._name===q||J._aliases.includes(q))}_findOption(q){return this.options.find((J)=>J.is(q))}_checkForMissingMandatoryOptions(){this._getCommandAndAncestors().forEach((q)=>{q.options.forEach((J)=>{if(J.mandatory&&q.getOptionValue(J.attributeName())===void 0)q.missingMandatoryOptionValue(J)})})}_checkForConflictingLocalOptions(){let q=this.options.filter((Q)=>{let X=Q.attributeName();if(this.getOptionValue(X)===void 0)return!1;return this.getOptionValueSource(X)!=="default"});q.filter((Q)=>Q.conflictsWith.length>0).forEach((Q)=>{let X=q.find((Y)=>Q.conflictsWith.includes(Y.attributeName()));if(X)this._conflictingOption(Q,X)})}_checkForConflictingOptions(){this._getCommandAndAncestors().forEach((q)=>{q._checkForConflictingLocalOptions()})}parseOptions(q){let J=[],Q=[],X=J;function Y(z){return z.length>1&&z[0]==="-"}let Z=(z)=>{if(!/^-(\d+|\d*\.\d+)(e[+-]?\d+)?$/.test(z))return!1;return!this._getCommandAndAncestors().some((B)=>B.options.map((U)=>U.short).some((U)=>/^-\d$/.test(U)))},$=null,R=null,T=0;while(T<q.length||R){let z=R??q[T++];if(R=null,z==="--"){if(X===Q)X.push(z);X.push(...q.slice(T));break}if($&&(!Y(z)||Z(z))){this.emit(`option:${$.name()}`,z);continue}if($=null,Y(z)){let B=this._findOption(z);if(B){if(B.required){let U=q[T++];if(U===void 0)this.optionMissingArgument(B);this.emit(`option:${B.name()}`,U)}else if(B.optional){let U=null;if(T<q.length&&(!Y(q[T])||Z(q[T])))U=q[T++];this.emit(`option:${B.name()}`,U)}else this.emit(`option:${B.name()}`);$=B.variadic?B:null;continue}}if(z.length>2&&z[0]==="-"&&z[1]!=="-"){let B=this._findOption(`-${z[1]}`);if(B){if(B.required||B.optional&&this._combineFlagAndOptionalValue)this.emit(`option:${B.name()}`,z.slice(2));else this.emit(`option:${B.name()}`),R=`-${z.slice(2)}`;continue}}if(/^--[^=]+=/.test(z)){let B=z.indexOf("="),U=this._findOption(z.slice(0,B));if(U&&(U.required||U.optional)){this.emit(`option:${U.name()}`,z.slice(B+1));continue}}if(X===J&&Y(z)&&!(this.commands.length===0&&Z(z)))X=Q;if((this._enablePositionalOptions||this._passThroughOptions)&&J.length===0&&Q.length===0){if(this._findCommand(z)){J.push(z),Q.push(...q.slice(T));break}else if(this._getHelpCommand()&&z===this._getHelpCommand().name()){J.push(z,...q.slice(T));break}else if(this._defaultCommandName){Q.push(z,...q.slice(T));break}}if(this._passThroughOptions){X.push(z,...q.slice(T));break}X.push(z)}return{operands:J,unknown:Q}}opts(){if(this._storeOptionsAsProperties){let q={},J=this.options.length;for(let Q=0;Q<J;Q++){let X=this.options[Q].attributeName();q[X]=X===this._versionOptionName?this._version:this[X]}return q}return this._optionValues}optsWithGlobals(){return this._getCommandAndAncestors().reduce((q,J)=>Object.assign(q,J.opts()),{})}error(q,J){if(this._outputConfiguration.outputError(`${q}
23
+ `,this._outputConfiguration.writeErr),typeof this._showHelpAfterError==="string")this._outputConfiguration.writeErr(`${this._showHelpAfterError}
24
+ `);else if(this._showHelpAfterError)this._outputConfiguration.writeErr(`
25
+ `),this.outputHelp({error:!0});let Q=J||{},X=Q.exitCode||1,Y=Q.code||"commander.error";this._exit(X,Y,q)}_parseOptionsEnv(){this.options.forEach((q)=>{if(q.envVar&&q.envVar in H.env){let J=q.attributeName();if(this.getOptionValue(J)===void 0||["default","config","env"].includes(this.getOptionValueSource(J)))if(q.required||q.optional)this.emit(`optionEnv:${q.name()}`,H.env[q.envVar]);else this.emit(`optionEnv:${q.name()}`)}})}_parseOptionsImplied(){let q=new vq(this.options),J=(Q)=>{return this.getOptionValue(Q)!==void 0&&!["default","implied"].includes(this.getOptionValueSource(Q))};this.options.filter((Q)=>Q.implied!==void 0&&J(Q.attributeName())&&q.valueFromOption(this.getOptionValue(Q.attributeName()),Q)).forEach((Q)=>{Object.keys(Q.implied).filter((X)=>!J(X)).forEach((X)=>{this.setOptionValueWithSource(X,Q.implied[X],"implied")})})}missingArgument(q){let J=`error: missing required argument '${q}'`;this.error(J,{code:"commander.missingArgument"})}optionMissingArgument(q){let J=`error: option '${q.flags}' argument missing`;this.error(J,{code:"commander.optionMissingArgument"})}missingMandatoryOptionValue(q){let J=`error: required option '${q.flags}' not specified`;this.error(J,{code:"commander.missingMandatoryOptionValue"})}_conflictingOption(q,J){let Q=(Z)=>{let $=Z.attributeName(),R=this.getOptionValue($),T=this.options.find((B)=>B.negate&&$===B.attributeName()),z=this.options.find((B)=>!B.negate&&$===B.attributeName());if(T&&(T.presetArg===void 0&&R===!1||T.presetArg!==void 0&&R===T.presetArg))return T;return z||Z},X=(Z)=>{let $=Q(Z),R=$.attributeName();if(this.getOptionValueSource(R)==="env")return`environment variable '${$.envVar}'`;return`option '${$.flags}'`},Y=`error: ${X(q)} cannot be used with ${X(J)}`;this.error(Y,{code:"commander.conflictingOption"})}unknownOption(q){if(this._allowUnknownOption)return;let J="";if(q.startsWith("--")&&this._showSuggestionAfterError){let X=[],Y=this;do{let Z=Y.createHelp().visibleOptions(Y).filter(($)=>$.long).map(($)=>$.long);X=X.concat(Z),Y=Y.parent}while(Y&&!Y._enablePositionalOptions);J=t(q,X)}let Q=`error: unknown option '${q}'${J}`;this.error(Q,{code:"commander.unknownOption"})}_excessArguments(q){if(this._allowExcessArguments)return;let J=this.registeredArguments.length,Q=J===1?"":"s",Y=`error: too many arguments${this.parent?` for '${this.name()}'`:""}. Expected ${J} argument${Q} but got ${q.length}.`;this.error(Y,{code:"commander.excessArguments"})}unknownCommand(){let q=this.args[0],J="";if(this._showSuggestionAfterError){let X=[];this.createHelp().visibleCommands(this).forEach((Y)=>{if(X.push(Y.name()),Y.alias())X.push(Y.alias())}),J=t(q,X)}let Q=`error: unknown command '${q}'${J}`;this.error(Q,{code:"commander.unknownCommand"})}version(q,J,Q){if(q===void 0)return this._version;this._version=q,J=J||"-V, --version",Q=Q||"output the version number";let X=this.createOption(J,Q);return this._versionOptionName=X.attributeName(),this._registerOption(X),this.on("option:"+X.name(),()=>{this._outputConfiguration.writeOut(`${q}
26
+ `),this._exit(0,"commander.version",q)}),this}description(q,J){if(q===void 0&&J===void 0)return this._description;if(this._description=q,J)this._argsDescription=J;return this}summary(q){if(q===void 0)return this._summary;return this._summary=q,this}alias(q){if(q===void 0)return this._aliases[0];let J=this;if(this.commands.length!==0&&this.commands[this.commands.length-1]._executableHandler)J=this.commands[this.commands.length-1];if(q===J._name)throw Error("Command alias can't be the same as its name");let Q=this.parent?._findCommand(q);if(Q){let X=[Q.name()].concat(Q.aliases()).join("|");throw Error(`cannot add alias '${q}' to command '${this.name()}' as already have command '${X}'`)}return J._aliases.push(q),this}aliases(q){if(q===void 0)return this._aliases;return q.forEach((J)=>this.alias(J)),this}usage(q){if(q===void 0){if(this._usage)return this._usage;let J=this.registeredArguments.map((Q)=>{return Cq(Q)});return[].concat(this.options.length||this._helpOption!==null?"[options]":[],this.commands.length?"[command]":[],this.registeredArguments.length?J:[]).join(" ")}return this._usage=q,this}name(q){if(q===void 0)return this._name;return this._name=q,this}helpGroup(q){if(q===void 0)return this._helpGroupHeading??"";return this._helpGroupHeading=q,this}commandsGroup(q){if(q===void 0)return this._defaultCommandGroup??"";return this._defaultCommandGroup=q,this}optionsGroup(q){if(q===void 0)return this._defaultOptionGroup??"";return this._defaultOptionGroup=q,this}_initOptionGroup(q){if(this._defaultOptionGroup&&!q.helpGroupHeading)q.helpGroup(this._defaultOptionGroup)}_initCommandGroup(q){if(this._defaultCommandGroup&&!q.helpGroup())q.helpGroup(this._defaultCommandGroup)}nameFromFilename(q){return this._name=L.basename(q,L.extname(q)),this}executableDir(q){if(q===void 0)return this._executableDir;return this._executableDir=q,this}helpInformation(q){let J=this.createHelp(),Q=this._getOutputContext(q);J.prepareContext({error:Q.error,helpWidth:Q.helpWidth,outputHasColors:Q.hasColors});let X=J.formatHelp(this,J);if(Q.hasColors)return X;return this._outputConfiguration.stripColor(X)}_getOutputContext(q){q=q||{};let J=!!q.error,Q,X,Y;if(J)Q=($)=>this._outputConfiguration.writeErr($),X=this._outputConfiguration.getErrHasColors(),Y=this._outputConfiguration.getErrHelpWidth();else Q=($)=>this._outputConfiguration.writeOut($),X=this._outputConfiguration.getOutHasColors(),Y=this._outputConfiguration.getOutHelpWidth();return{error:J,write:($)=>{if(!X)$=this._outputConfiguration.stripColor($);return Q($)},hasColors:X,helpWidth:Y}}outputHelp(q){let J;if(typeof q==="function")J=q,q=void 0;let Q=this._getOutputContext(q),X={error:Q.error,write:Q.write,command:this};this._getCommandAndAncestors().reverse().forEach((Z)=>Z.emit("beforeAllHelp",X)),this.emit("beforeHelp",X);let Y=this.helpInformation({error:Q.error});if(J){if(Y=J(Y),typeof Y!=="string"&&!Buffer.isBuffer(Y))throw Error("outputHelp callback must return a string or a Buffer")}if(Q.write(Y),this._getHelpOption()?.long)this.emit(this._getHelpOption().long);this.emit("afterHelp",X),this._getCommandAndAncestors().forEach((Z)=>Z.emit("afterAllHelp",X))}helpOption(q,J){if(typeof q==="boolean"){if(q){if(this._helpOption===null)this._helpOption=void 0;if(this._defaultOptionGroup)this._initOptionGroup(this._getHelpOption())}else this._helpOption=null;return this}if(this._helpOption=this.createOption(q??"-h, --help",J??"display help for command"),q||J)this._initOptionGroup(this._helpOption);return this}_getHelpOption(){if(this._helpOption===void 0)this.helpOption(void 0,void 0);return this._helpOption}addHelpOption(q){return this._helpOption=q,this._initOptionGroup(q),this}help(q){this.outputHelp(q);let J=Number(H.exitCode??0);if(J===0&&q&&typeof q!=="function"&&q.error)J=1;this._exit(J,"commander.help","(outputHelp)")}addHelpText(q,J){let Q=["beforeAll","before","after","afterAll"];if(!Q.includes(q))throw Error(`Unexpected value for position to addHelpText.
27
+ Expecting one of '${Q.join("', '")}'`);let X=`${q}Help`;return this.on(X,(Y)=>{let Z;if(typeof J==="function")Z=J({error:Y.error,command:Y.command});else Z=J;if(Z)Y.write(`${Z}
28
+ `)}),this}_outputHelpIfRequested(q){let J=this._getHelpOption();if(J&&q.find((X)=>J.is(X)))this.outputHelp(),this._exit(0,"commander.helpDisplayed","(outputHelp)")}}function n(q){return q.map((J)=>{if(!J.startsWith("--inspect"))return J;let Q,X="127.0.0.1",Y="9229",Z;if((Z=J.match(/^(--inspect(-brk)?)$/))!==null)Q=Z[1];else if((Z=J.match(/^(--inspect(-brk|-port)?)=([^:]+)$/))!==null)if(Q=Z[1],/^\d+$/.test(Z[3]))Y=Z[3];else X=Z[3];else if((Z=J.match(/^(--inspect(-brk|-port)?)=([^:]+):(\d+)$/))!==null)Q=Z[1],X=Z[3],Y=Z[4];if(Q&&Y!=="0")return`${Q}=${X}:${parseInt(Y)+1}`;return J})}function C(){if(H.env.NO_COLOR||H.env.FORCE_COLOR==="0"||H.env.FORCE_COLOR==="false")return!1;if(H.env.FORCE_COLOR||H.env.CLICOLOR_FORCE!==void 0)return!0;return}hq.Command=b;hq.useColor=C});var Jq=S((mq)=>{var{Argument:o}=y(),{Command:f}=a(),{CommanderError:cq,InvalidArgumentError:e}=j(),{Help:lq}=k(),{Option:qq}=w();mq.program=new f;mq.createCommand=(q)=>new f(q);mq.createOption=(q,J)=>new qq(q,J);mq.createArgument=(q,J)=>new o(q,J);mq.Command=f;mq.Option=qq;mq.Argument=o;mq.Help=lq;mq.CommanderError=cq;mq.InvalidArgumentError=e;mq.InvalidOptionArgumentError=e});var Qq=Tq(Jq(),1),{program:I,createCommand:LJ,createArgument:_J,createOption:WJ,CommanderError:SJ,InvalidArgumentError:PJ,InvalidOptionArgumentError:VJ,Command:IJ,Argument:jJ,Option:EJ,Help:yJ}=Qq.default;import{cp as Xq,exists as _,mkdir as D,writeFile as V,readFile as kJ}from"fs/promises";import{join as M}from"path";import{spawn as F}from"child_process";import{homedir as JJ}from"os";var v={rust:{courseJson:{id:"{{id}}",name:"{{name}}",runner:{command:"cargo",args:["test","--quiet","--manifest-path","./content/{{id}}/Cargo.toml"],cwd:"."},content:{root:".",exercises:"content"},setup:{checks:[{name:"Rust Compiler",type:"command",command:"rustc --version"},{name:"Cargo Package Manager",type:"command",command:"cargo --version"}],guide:"SETUP.md"}},setupMd:`# \uD83E\uDD80 Rust Setup Guide
29
+
30
+ To run the exercises in this course, you need to have **Rust** installed on your system.
31
+
32
+ ## \uD83D\uDEE0\uFE0F Installation Steps
33
+
34
+ ### 1. Install Rust via Rustup
35
+ Go to [rust-lang.org/tools/install](https://www.rust-lang.org/tools/install) and follow the instructions.
36
+
37
+ ### 2. Verify Installation
38
+ Open a terminal and run:
39
+ \`\`\`bash
40
+ rustc --version
41
+ cargo --version
42
+ \`\`\`
43
+ `,introReadme:`# Hello Rust
44
+
45
+ This is your first exercise.
46
+
47
+ ## Goal
48
+ Make the code compile and print "Hello".
49
+ `,introCode:`fn main() {
50
+ println!("Hello, world!");
51
+ }
52
+
53
+ #[cfg(test)]
54
+ mod tests {
55
+ #[test]
56
+ fn test_hello() {
57
+ assert!(true);
58
+ }
59
+ }
60
+ `,introFilename:"main.rs"},go:{courseJson:{id:"{{id}}",name:"{{name}}",runner:{command:"go",args:["run","./runner/main.go","test","{{id}}"],cwd:"."},content:{root:".",exercises:"content"},setup:{checks:[{name:"Go Compiler",type:"command",command:"go version"}],guide:"SETUP.md"}},setupMd:`# \uD83D\uDC39 Go Setup Guide
61
+
62
+ To run the exercises in this course, you need to have **Go** installed on your system.
63
+
64
+ ## \uD83D\uDEE0\uFE0F Installation Steps
65
+
66
+ ### 1. Install Go
67
+ Go to [go.dev/dl/](https://go.dev/dl/) and download the installer.
68
+
69
+ ### 2. Verify Installation
70
+ Open a terminal and run:
71
+ \`\`\`bash
72
+ go version
73
+ \`\`\`
74
+ `,introReadme:`# Hello Go
75
+
76
+ This is your first exercise.
77
+
78
+ ## Goal
79
+ Make the code pass the test.
80
+ `,introCode:`package main
81
+
82
+ import "fmt"
83
+
84
+ func Greeting() string {
85
+ return "Hello World"
86
+ }
87
+
88
+ func main() {
89
+ fmt.Println(Greeting())
90
+ }
91
+ `,introFilename:"main.go"}};var h=M(JJ(),".progy"),QJ=M(h,"config.json"),Yq=process.env.PROGY_API_URL||"https://progy.francy.workers.dev",XJ=process.env.PROGY_FRONTEND_URL||Yq,E="course.json";async function YJ(){let q=M(import.meta.dir,"../../../courses");if(await _(q))return q;return null}async function ZJ(q){if(!await _(h))await D(h,{recursive:!0});await V(QJ,JSON.stringify({token:q}))}function $J(q){let J=process.platform==="win32"?"start":process.platform==="darwin"?"open":"xdg-open";F(J,[q],{shell:!0}).unref()}async function zJ(q,J){console.log(`[INFO] Local courses not found. Attempting to download ${q} from https://github.com/fhorray/rust-flow.git...`);let X=M(process.cwd(),".prog-temp-"+Date.now());try{console.log("[GIT] Cloning repository...");let Y=F("git",["clone","--depth","1","https://github.com/fhorray/rust-flow.git",X],{stdio:"inherit"});await new Promise((R,T)=>{Y.on("close",(z)=>z===0?R(null):T("Git clone failed"))});let Z=M(X,"courses",q);if(!await _(Z))throw Error(`Course '${q}' not found in repository.`);let $=["content","runner","Cargo.toml","go.mod","SETUP.md",E];for(let R of $){let T=M(Z,R),z=M(J,R);if(await _(T))console.log(`[COPY] ${R}...`),await Xq(T,z,{recursive:!0})}}finally{try{if(await _(X))F("rm",["-rf",X],{stdio:"ignore"}).unref()}catch(Y){}}}I.name("progy").description("Universal programming course runner").version("0.0.1");I.command("init").description("Initialize a new course in the current directory").option("-c, --course <course>","Language/Course to initialize (e.g., rust)").action(async(q)=>{let J=process.cwd(),Q=M(J,E),X=await _(Q);if(!q.course)if(X)console.log(`[INFO] Detected '${E}'. Starting progy...`);else console.error(`[ERROR] No '${E}' found. Please specify a course to initialize:`),console.error(" progy init --course <rust|go|cloudflare...>"),process.exit(1);else{let T=q.course;console.log(`[INFO] Initializing ${T} course in ${J}...`);let z=await YJ(),B=!1;if(z){console.log(`[INFO] Found local courses directory: ${z}`);let U=M(z,T);if(await _(U)){let K=["content","runner","Cargo.toml","go.mod","SETUP.md",E];for(let G of K){let W=M(U,G),Zq=M(J,G);if(await _(W))console.log(`[COPY] ${G}...`),await Xq(W,Zq,{recursive:!0})}B=!0}else console.warn(`[WARN] Course ${T} not found locally.`)}if(!B)try{await zJ(T,J)}catch(U){console.error(`[ERROR] Failed to initialize course: ${U}`),process.exit(1)}console.log("[INFO] Initialization complete!")}console.log("[INFO] Starting UI...");let Z=import.meta.file.endsWith(".ts")?"ts":"js",$=M(import.meta.dir,"backend",`server.${Z}`);F("bun",["run","--hot",$],{stdio:"inherit",env:{...process.env,PROG_CWD:J}}).on("close",(T)=>process.exit(T??0))});I.command("create-course").description("Scaffold a new course with standard directory structure").requiredOption("--name <name>","Name of the course (e.g., rust-advanced)").requiredOption("-c, --course <course>","Programming language template (rust, go)").action(async(q)=>{let J=process.cwd(),Q=q.name,X=q.course.toLowerCase(),Y=M(J,Q);if(await _(Y))console.error(`[ERROR] Directory '${Q}' already exists.`),process.exit(1);let Z=v[X];if(!Z)console.error(`[ERROR] Unsupported language '${X}'. Supported: ${Object.keys(v).join(", ")}`),process.exit(1);console.log(`[INFO] Creating course '${Q}' with template '${X}'...`),await D(Y,{recursive:!0}),await D(M(Y,"content","01_intro"),{recursive:!0});let $=JSON.stringify(Z.courseJson,null,2).replace(/{{id}}/g,Q).replace(/{{name}}/g,Q);if(await V(M(Y,"course.json"),$),await V(M(Y,"SETUP.md"),Z.setupMd),await V(M(Y,"content","01_intro","README.md"),Z.introReadme),await V(M(Y,"content","01_intro",Z.introFilename),Z.introCode),X==="go"){let R=M(Y,"runner");await D(R,{recursive:!0});let T=`package main
92
+ import (
93
+ "bufio"
94
+ "encoding/json"
95
+ "fmt"
96
+ "os"
97
+ "os/exec"
98
+ "strings"
99
+ )
100
+ type SRPDiagnostic struct {
101
+ Severity string \`json:"severity"\`
102
+ Message string \`json:"message"\`
103
+ File string \`json:"file,omitempty"\`
104
+ Line int \`json:"line,omitempty"\`
105
+ }
106
+ type SRPTest struct {
107
+ Name string \`json:"name"\`
108
+ Status string \`json:"status"\`
109
+ Message string \`json:"message,omitempty"\`
110
+ }
111
+ type SRPOutput struct {
112
+ Success bool \`json:"success"\`
113
+ Summary string \`json:"summary"\`
114
+ Diagnostics []SRPDiagnostic \`json:"diagnostics,omitempty"\`
115
+ Tests []SRPTest \`json:"tests,omitempty"\`
116
+ Raw string \`json:"raw"\`
117
+ }
118
+ func main() {
119
+ if len(os.Args) < 3 { os.Exit(1) }
120
+ id := os.Args[2]
121
+ cmd := exec.Command("go", "test", "-v", "./content/"+id)
122
+ output, _ := cmd.CombinedOutput()
123
+ raw := string(output)
124
+ srp := SRPOutput{ Success: cmd.ProcessState.Success(), Raw: raw, Summary: "Test execution completed" }
125
+ // ... Simplified parsing for scaffold ...
126
+ json, _ := json.MarshalIndent(srp, "", " ")
127
+ fmt.Println("__SRP_BEGIN__")
128
+ fmt.Println(string(json))
129
+ fmt.Println("__SRP_END__")
130
+ if !srp.Success { os.Exit(0) }
131
+ }
132
+ `;await V(M(R,"main.go"),T),await V(M(Y,"go.mod"),`module ${Q}
133
+
134
+ go 1.21
135
+ `)}console.log("[SUCCESS] Course created!"),console.log(`
136
+ To get started:
137
+ cd ${Q}
138
+ bunx progy init`)});I.command("login").description("Authenticate with Progy").action(async()=>{let{createAuthClient:q}=await import("better-auth/client"),{deviceAuthorizationClient:J}=await import("better-auth/client/plugins"),Q=q({baseURL:Yq,plugins:[J()]});try{console.log("[INFO] Requesting login session...");let{data:X,error:Y}=await Q.device.code({client_id:"progy-cli"});if(Y)throw Error(Y.error_description||"Failed to initiate device authorization");let{device_code:Z,user_code:$,verification_uri:R,interval:T}=X,z=R.startsWith("http")?R:`${XJ}${R}`;console.log(`
139
+ Please authenticate in your browser:`),console.log(`\x1B[36m${z}\x1B[0m`),console.log(`Code: \x1B[33m${$}\x1B[0m
140
+ `),$J(z),console.log("[WAIT] Waiting for authorization...");let U=await(async()=>{while(!0){let{data:K,error:G}=await Q.device.token({grant_type:"urn:ietf:params:oauth:grant-type:device_code",device_code:Z,client_id:"progy-cli"});if(K?.access_token)return K.access_token;if(G){let W=G.error;if(W==="access_denied"||W==="expired_token")throw Error(G.error_description||W)}await new Promise((W)=>setTimeout(W,(T||5)*1000))}})();if(U)await ZJ(U),console.log("[SUCCESS] Logged in successfully!")}catch(X){console.error(`[ERROR] Login failed: ${X.message||X}`),process.exit(1)}});I.parse();
@@ -0,0 +1,26 @@
1
+ <!DOCTYPE html>
2
+ <html lang="en">
3
+
4
+ <head>
5
+ <meta charset="UTF-8">
6
+ <meta name="viewport" content="width=device-width, initial-scale=1.0">
7
+ <title>prog - Learning Rust with Flow</title>
8
+
9
+ <script src="https://cdn.jsdelivr.net/npm/@tailwindcss/browser@4"></script>
10
+ <style type="text/tailwindcss">
11
+ @theme {
12
+ --color-rust: #ce412b;
13
+ --color-rust-dark: #8a2b1d;
14
+ --color-rust-light: #dea584;
15
+ }
16
+ </style>
17
+
18
+ <link rel="stylesheet" crossorigin href="../prog/index-cqhmgv8p.css"><script type="module" crossorigin src="../prog/index-p4er7fk9.js"></script><link rel="stylesheet" href="/main.css">
19
+ </head>
20
+
21
+ <body class="bg-zinc-950 text-zinc-100">
22
+ <div id="root"></div>
23
+
24
+ </body>
25
+
26
+ </html>
@@ -0,0 +1 @@
1
+ pre code.hljs{display:block;overflow-x:auto;padding:1em}code.hljs{padding:3px 5px}.hljs{color:#c9d1d9;background:#0d1117}.hljs-doctag,.hljs-keyword,.hljs-meta .hljs-keyword,.hljs-template-tag,.hljs-template-variable,.hljs-type,.hljs-variable.language_{color:#ff7b72}.hljs-title,.hljs-title.class_,.hljs-title.class_.inherited__,.hljs-title.function_{color:#d2a8ff}.hljs-attr,.hljs-attribute,.hljs-literal,.hljs-meta,.hljs-number,.hljs-operator,.hljs-variable,.hljs-selector-attr,.hljs-selector-class,.hljs-selector-id{color:#79c0ff}.hljs-regexp,.hljs-string,.hljs-meta .hljs-string{color:#a5d6ff}.hljs-built_in,.hljs-symbol{color:#ffa657}.hljs-comment,.hljs-code,.hljs-formula{color:#8b949e}.hljs-name,.hljs-quote,.hljs-selector-tag,.hljs-selector-pseudo{color:#7ee787}.hljs-subst{color:#c9d1d9}.hljs-section{color:#1f6feb;font-weight:700}.hljs-bullet{color:#f2cc60}.hljs-emphasis{color:#c9d1d9;font-style:italic}.hljs-strong{color:#c9d1d9;font-weight:700}.hljs-addition{color:#aff5b4;background-color:#033a16}.hljs-deletion{color:#ffdcd7;background-color:#67060c}