@nalvietnam/avatar-cli 1.20.2 → 1.20.3

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/dist/index.js CHANGED
@@ -1,95 +1,95 @@
1
1
  // @nalvietnam/avatar-cli — built with tsup
2
- var ls=Object.defineProperty;var nn=(t,e)=>()=>(t&&(e=t(t=0)),e);var us=(t,e)=>{for(var n in e)ls(t,n,{get:e[n],enumerable:!0})};import{constants as ps,promises as wt}from"fs";import{dirname as ms,join as Am,relative as Sm}from"path";async function d(t){try{return await wt.access(t,ps.F_OK),!0}catch{return!1}}async function E(t){await wt.mkdir(t,{recursive:!0})}async function D(t){return await wt.readFile(t,"utf8")}async function v(t){return JSON.parse(await D(t))}async function yt(t,e,n){await E(ms(t));let r=`${t}.tmp-${process.pid}-${Date.now()}`;await wt.writeFile(r,e,"utf8"),n!==void 0&&await wt.chmod(r,n),await wt.rename(r,t)}async function $(t,e,n){await yt(t,`${JSON.stringify(e,null,2)}
3
- `,n)}var k=nn(()=>{"use strict"});import{spawnSync as xa}from"child_process";import{chmodSync as Aa,existsSync as Sr,readFileSync as Cr,renameSync as Sa,writeFileSync as Ca}from"fs";import{join as Pa}from"path";function xt(t){return Pa(t,$a)}function $r(t){let e=t.indexOf(Pr);if(e===-1)return null;let n=t.indexOf(mn,e);return n===-1?null:t.slice(e,n+mn.length)}function Er(t){let e=new Map,n=/^export\s+([A-Z][A-Z0-9_]*)="((?:[^"\\]|\\.)*)"$/gm,r;for(;(r=n.exec(t))!==null;){let i=r[1],s=r[2];if(!i||s===void 0)continue;let a=s.replace(/\\"/g,'"').replace(/\\\\/g,"\\");e.set(i,a)}return e}function Ra(t){return t.replace(/\\/g,"\\\\").replace(/"/g,'\\"')}function Ta(t){let e=[Ea],n=[...t.keys()].sort();for(let r of n){let i=t.get(r);i!==void 0&&e.push(`export ${r}="${Ra(i)}"`)}return e.push(mn),e.join(`
4
- `)}async function at(t,e){let n=xt(t),r="";Sr(n)&&(r=Cr(n,"utf8"));let i=$r(r),s=i?Er(i):new Map,a=s.size;for(let[f,A]of Object.entries(e))A===null?s.delete(f):s.set(f,A);let c=Ta(s),l,u=!1;i?(l=r.replace(i,c),u=r.replace(i,"").trim().length>0):r.trim()?(u=!0,l=`${r.replace(/\s+$/,"")}
2
+ var rs=Object.defineProperty;var Ze=(t,e)=>()=>(t&&(e=t(t=0)),e);var is=(t,e)=>{for(var n in e)rs(t,n,{get:e[n],enumerable:!0})};import{constants as os,promises as wt}from"fs";import{dirname as ss,join as Qp,relative as tm}from"path";async function d(t){try{return await wt.access(t,os.F_OK),!0}catch{return!1}}async function E(t){await wt.mkdir(t,{recursive:!0})}async function U(t){return await wt.readFile(t,"utf8")}async function y(t){return JSON.parse(await U(t))}async function yt(t,e,n){await E(ss(t));let r=`${t}.tmp-${process.pid}-${Date.now()}`;await wt.writeFile(r,e,"utf8"),n!==void 0&&await wt.chmod(r,n),await wt.rename(r,t)}async function $(t,e,n){await yt(t,`${JSON.stringify(e,null,2)}
3
+ `,n)}var k=Ze(()=>{"use strict"});import{spawnSync as ta}from"child_process";import{chmodSync as ea,existsSync as kr,readFileSync as wr,renameSync as na,writeFileSync as ra}from"fs";import{join as ia}from"path";function xt(t){return ia(t,oa)}function vr(t){let e=t.indexOf(yr);if(e===-1)return null;let n=t.indexOf(cn,e);return n===-1?null:t.slice(e,n+cn.length)}function br(t){let e=new Map,n=/^export\s+([A-Z][A-Z0-9_]*)="((?:[^"\\]|\\.)*)"$/gm,r;for(;(r=n.exec(t))!==null;){let i=r[1],s=r[2];if(!i||s===void 0)continue;let a=s.replace(/\\"/g,'"').replace(/\\\\/g,"\\");e.set(i,a)}return e}function aa(t){return t.replace(/\\/g,"\\\\").replace(/"/g,'\\"')}function ca(t){let e=[sa],n=[...t.keys()].sort();for(let r of n){let i=t.get(r);i!==void 0&&e.push(`export ${r}="${aa(i)}"`)}return e.push(cn),e.join(`
4
+ `)}async function st(t,e){let n=xt(t),r="";kr(n)&&(r=wr(n,"utf8"));let i=vr(r),s=i?br(i):new Map,a=s.size;for(let[f,S]of Object.entries(e))S===null?s.delete(f):s.set(f,S);let c=ca(s),l,u=!1;i?(l=r.replace(i,c),u=r.replace(i,"").trim().length>0):r.trim()?(u=!0,l=`${r.replace(/\s+$/,"")}
5
5
 
6
6
  ${c}
7
7
  `):l=`${c}
8
- `;let m=`${n}.avatar-tmp-${process.pid}`;Ca(m,l,{encoding:"utf8",mode:384}),Sa(m,n),Aa(n,384);let g=null;try{g=xa("direnv",["allow",n],{encoding:"utf8",timeout:5e3}).status===0}catch{g=null}return{envrcPath:n,varsBefore:a,varsAfter:s.size,direnvAllowOk:g,hadExistingNonAvatarContent:u}}function V(t){let e=xt(t);if(!Sr(e))return new Map;let n=Cr(e,"utf8"),r=$r(n);return r?Er(r):new Map}var $a,Pr,mn,Ea,Ht=nn(()=>{"use strict";$a=".envrc",Pr="# >>> avatar-cli secrets >>>",mn="# <<< avatar-cli secrets <<<",Ea=`${Pr}
8
+ `;let m=`${n}.avatar-tmp-${process.pid}`;ra(m,l,{encoding:"utf8",mode:384}),na(m,n),ea(n,384);let g=null;try{g=ta("direnv",["allow",n],{encoding:"utf8",timeout:5e3}).status===0}catch{g=null}return{envrcPath:n,varsBefore:a,varsAfter:s.size,direnvAllowOk:g,hadExistingNonAvatarContent:u}}function V(t){let e=xt(t);if(!kr(e))return new Map;let n=wr(e,"utf8"),r=vr(n);return r?br(r):new Map}var oa,yr,cn,sa,Ut=Ze(()=>{"use strict";oa=".envrc",yr="# >>> avatar-cli secrets >>>",cn="# <<< avatar-cli secrets <<<",sa=`${yr}
9
9
  # Per-project secrets managed by avatar-cli.
10
10
  # DO NOT COMMIT \u2014 must be gitignored (avatar enforces this on setup).
11
11
  # Edit via: avatar secrets set <NAME>
12
12
  # Remove via: avatar secrets rm <NAME>
13
- # View: avatar secrets list`});var bo={};us(bo,{MIGRATABLE_SECRET_KEYS:()=>vo,detectPlaintextSecretsInSettings:()=>_t,migrateSecretsFromSettingsToEnvrc:()=>Xe});import{promises as cp}from"fs";import{join as lp}from"path";function pp(t){return lp(t,...up)}function mp(t){let e=new Date,n=`${e.getFullYear().toString().slice(-2)}${String(e.getMonth()+1).padStart(2,"0")}${String(e.getDate()).padStart(2,"0")}-${String(e.getHours()).padStart(2,"0")}${String(e.getMinutes()).padStart(2,"0")}`;return`${t}.backup-secrets-migration-${n}`}async function _t(t){let e=pp(t);if(!await d(e))return{keys:[],settingsPath:e};let n;try{n=await v(e)}catch{return{keys:[],settingsPath:e}}let r=n.env??{},i=[];for(let s of vo)typeof r[s]=="string"&&r[s].length>0&&i.push(s);return{keys:i,settingsPath:e}}async function Xe(t){let{keys:e,settingsPath:n}=await _t(t);if(e.length===0)return{migratedKeys:[],settingsJsonBackupPath:null,noOp:!0};let r=await v(n),i=r.env??{},s={};for(let c of e){let l=i[c];typeof l=="string"&&l.length>0&&(s[c]=l)}await at(t,s);let a=mp(n);await cp.copyFile(n,a);for(let c of e)delete i[c];return Object.keys(i).length===0?r.env=void 0:r.env=i,await $(n,r),{migratedKeys:e,settingsJsonBackupPath:a,noOp:!1}}var vo,up,Wn=nn(()=>{"use strict";k();Ht();vo=["ANTHROPIC_API_KEY","ANTHROPIC_AUTH_TOKEN","OPENAI_API_KEY","GEMINI_API_KEY","DEEPSEEK_API_KEY","GROQ_API_KEY","PERPLEXITY_API_KEY","ANTHROPIC_VERTEX_PROJECT_ID","GOOGLE_APPLICATION_CREDENTIALS"],up=[".claude","settings.json"]});import{Command as wm}from"commander";k();import{promises as za}from"fs";import{join as Nr}from"path";import{confirm as qa}from"@inquirer/prompts";import{existsSync as se,readFileSync as ds}from"fs";import{dirname as gs,join as ae}from"path";var fs=5;function hs(t){let e=se(ae(t,".claude")),n=se(ae(t,"CLAUDE.md"));if(!e||!n)return!1;let r=se(ae(t,"src")),i=ae(t,".gitmodules");if(r)return!0;if(se(i))try{let s=ds(i,"utf8");return s.includes("path = src")||s.includes("path = ./src")}catch{return!1}return!1}function vt(t){let e=t;for(let n=0;n<fs;n++){if(hs(e))return e;let r=gs(e);if(r===e)return null;e=r}return null}import{promises as or}from"fs";import{homedir as ws}from"os";import{join as jt}from"path";import{z as h}from"zod";var nr=h.object({email:h.string().email(),name:h.string(),access_token:h.string().min(1),refresh_token:h.string().min(1),expires_at:h.string().datetime(),id_token:h.string().min(1)}),ks=h.object({installed_tools:h.record(h.string(),h.object({version:h.string().optional(),installed_at:h.string().datetime(),install_method:h.string()})).default({}),tool_inputs:h.record(h.string(),h.unknown()).default({})}),Tm=h.object({$schema:h.string().optional(),includeCoAuthoredBy:h.boolean().optional(),env:h.record(h.string(),h.string()).default({}),permissions:h.object({allow:h.array(h.string()).default([]),deny:h.array(h.string()).default([])}).partial().optional(),hooks:h.record(h.string(),h.array(h.unknown())).optional(),statusLine:h.object({type:h.string(),command:h.string(),padding:h.number().optional()}).optional()}),Im=h.enum(["internal","client","library"]);k();var Gt=jt(ws(),".avatar"),tt=jt(Gt,"config.json"),Gm=jt(Gt,"state.json"),rn=jt(Gt,"audit.log"),Lm=jt(Gt,"backups"),ys=384;async function on(){await E(Gt)}async function et(){if(!await d(tt))return null;let t=await v(tt),e=nr.safeParse(t);return e.success?e.data:null}async function rr(t){await on(),await $(tt,t,ys)}async function ir(){if(await d(tt)){let{promises:t}=await import("fs");await t.unlink(tt)}}function nt(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}async function vs(){try{await or.chmod(rn,384)}catch{}}async function w(t,e){await on();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
14
- `;await or.appendFile(rn,r,{encoding:"utf8",mode:384}),await vs()}import{spawnSync as sn}from"child_process";import p from"chalk";import bs from"ora";var o={info:t=>process.stdout.write(`${p.blue("\u2139")} ${t}
13
+ # View: avatar secrets list`});var fo={};is(fo,{MIGRATABLE_SECRET_KEYS:()=>go,detectPlaintextSecretsInSettings:()=>It,migrateSecretsFromSettingsToEnvrc:()=>ze});import{promises as Hu}from"fs";import{join as Gu}from"path";function Du(t){return Gu(t,...Uu)}function Fu(t){let e=new Date,n=`${e.getFullYear().toString().slice(-2)}${String(e.getMonth()+1).padStart(2,"0")}${String(e.getDate()).padStart(2,"0")}-${String(e.getHours()).padStart(2,"0")}${String(e.getMinutes()).padStart(2,"0")}`;return`${t}.backup-secrets-migration-${n}`}async function It(t){let e=Du(t);if(!await d(e))return{keys:[],settingsPath:e};let n;try{n=await y(e)}catch{return{keys:[],settingsPath:e}}let r=n.env??{},i=[];for(let s of go)typeof r[s]=="string"&&r[s].length>0&&i.push(s);return{keys:i,settingsPath:e}}async function ze(t){let{keys:e,settingsPath:n}=await It(t);if(e.length===0)return{migratedKeys:[],settingsJsonBackupPath:null,noOp:!0};let r=await y(n),i=r.env??{},s={};for(let c of e){let l=i[c];typeof l=="string"&&l.length>0&&(s[c]=l)}await st(t,s);let a=Fu(n);await Hu.copyFile(n,a);for(let c of e)delete i[c];return Object.keys(i).length===0?r.env=void 0:r.env=i,await $(n,r),{migratedKeys:e,settingsJsonBackupPath:a,noOp:!1}}var go,Uu,Vn=Ze(()=>{"use strict";k();Ut();go=["ANTHROPIC_API_KEY","ANTHROPIC_AUTH_TOKEN","OPENAI_API_KEY","GEMINI_API_KEY","DEEPSEEK_API_KEY","GROQ_API_KEY","PERPLEXITY_API_KEY","ANTHROPIC_VERTEX_PROJECT_ID","GOOGLE_APPLICATION_CREDENTIALS"],Uu=[".claude","settings.json"]});import{Command as qp}from"commander";k();import{promises as Sa}from"fs";import{join as Er}from"path";import{confirm as Aa}from"@inquirer/prompts";import{existsSync as oe,readFileSync as as}from"fs";import{dirname as cs,join as se}from"path";var ls=5;function us(t){let e=oe(se(t,".claude")),n=oe(se(t,"CLAUDE.md"));if(!e||!n)return!1;let r=oe(se(t,"src")),i=se(t,".gitmodules");if(r)return!0;if(oe(i))try{let s=as(i,"utf8");return s.includes("path = src")||s.includes("path = ./src")}catch{return!1}return!1}function vt(t){let e=t;for(let n=0;n<ls;n++){if(us(e))return e;let r=cs(e);if(r===e)return null;e=r}return null}import{promises as nr}from"fs";import{homedir as ms}from"os";import{join as jt}from"path";import{z as h}from"zod";var Qn=h.object({email:h.string().email(),name:h.string(),access_token:h.string().min(1),refresh_token:h.string().min(1),expires_at:h.string().datetime(),id_token:h.string().min(1)}),ps=h.object({installed_tools:h.record(h.string(),h.object({version:h.string().optional(),installed_at:h.string().datetime(),install_method:h.string()})).default({}),tool_inputs:h.record(h.string(),h.unknown()).default({})}),sm=h.object({$schema:h.string().optional(),includeCoAuthoredBy:h.boolean().optional(),env:h.record(h.string(),h.string()).default({}),permissions:h.object({allow:h.array(h.string()).default([]),deny:h.array(h.string()).default([])}).partial().optional(),hooks:h.record(h.string(),h.array(h.unknown())).optional(),statusLine:h.object({type:h.string(),command:h.string(),padding:h.number().optional()}).optional()}),am=h.enum(["internal","client","library"]);k();var Lt=jt(ms(),".avatar"),Q=jt(Lt,"config.json"),dm=jt(Lt,"state.json"),Qe=jt(Lt,"audit.log"),gm=jt(Lt,"backups"),ds=384;async function tn(){await E(Lt)}async function tt(){if(!await d(Q))return null;let t=await y(Q),e=Qn.safeParse(t);return e.success?e.data:null}async function tr(t){await tn(),await $(Q,t,ds)}async function er(){if(await d(Q)){let{promises:t}=await import("fs");await t.unlink(Q)}}function et(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}async function gs(){try{await nr.chmod(Qe,384)}catch{}}async function v(t,e){await tn();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
14
+ `;await nr.appendFile(Qe,r,{encoding:"utf8",mode:384}),await gs()}import{spawnSync as en}from"child_process";import p from"chalk";import fs from"ora";var o={info:t=>process.stdout.write(`${p.blue("\u2139")} ${t}
15
15
  `),success:t=>process.stdout.write(`${p.green("\u2713")} ${t}
16
16
  `),warn:t=>process.stdout.write(`${p.yellow("\u26A0")} ${t}
17
17
  `),error:t=>process.stderr.write(`${p.red("\u2717")} ${t}
18
18
  `),dim:t=>process.stdout.write(`${p.dim(t)}
19
19
  `),plain:t=>process.stdout.write(`${t}
20
- `)};function F(t){return bs({text:t,spinner:"dots",isEnabled:process.stdout.isTTY??!1}).start()}function Lt(t){let e=Date.now(),n=F(`${t} (0:00)`),r=()=>{let s=Math.floor((Date.now()-e)/1e3),a=Math.floor(s/60),c=s%60;return`${a}:${String(c).padStart(2,"0")}`},i=setInterval(()=>{n.text=`${t} (${r()})`},1e3);return{succeed:s=>{clearInterval(i),n.succeed(`${s} (${r()})`)},fail:s=>{clearInterval(i),n.fail(`${s} (${r()})`)},stop:()=>{clearInterval(i),n.stop()}}}var sr=6e4,xs="ok";function an(){let t=sn("claude",["auth","status"],{encoding:"utf8"});if(t.error||t.status!==0)return{state:"not-authenticated"};let e=(t.stdout||"").trim();if(!e.startsWith("{"))return{state:"authenticated"};try{let n=JSON.parse(e);return n.loggedIn!==!0?{state:"not-authenticated"}:{state:"authenticated",email:n.email,subscriptionType:n.subscriptionType,apiProvider:n.apiProvider}}catch{return{state:"authenticated"}}}function cn(){o.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=sn("claude",["auth","login"],{stdio:"inherit"});if(t.status!==0)throw new Error(`claude auth login th\u1EA5t b\u1EA1i (exit ${t.status}). Th\u1EED 'claude auth login' tay r\u1ED3i ch\u1EA1y l\u1EA1i.`);o.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp Claude Code")}function As(t){let e=t.toLowerCase();return e.includes("credit_balance_too_low")||e.includes("credit balance too low")?"credit_balance_too_low":e.includes("insufficient_quota")||e.includes("insufficient quota")||e.includes("quota_exceeded")||e.includes("quota exceeded")||e.includes("usage limit")||e.includes("you've used all")?"insufficient_quota":e.includes("401")||e.includes("invalid authentication")||e.includes("authentication credentials")||e.includes("failed to authenticate")||e.includes("authentication failed")||e.includes("unauthorized")?"auth-expired":e.includes("invalid_api_key")||e.includes("invalid api key")?"invalid_api_key":e.includes("rate_limit")||e.includes("rate limit")||e.includes("429")?"rate_limit":"unknown"}function ar(t){switch(t){case"auth-expired":return"Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n/b\u1ECB revoke. Ch\u1EA1y: `claude auth logout && claude auth login`.";case"credit_balance_too_low":case"insufficient_quota":return"H\u1EBFt quota subscription. Upgrade plan ho\u1EB7c d\xF9ng LLMLite (avatar ai setup \u2192 ch\u1ECDn LLMLite).";case"invalid_api_key":return"API key invalid. Re-login: `claude auth login`.";case"rate_limit":return"B\u1ECB rate limit t\u1EA1m th\u1EDDi. Ch\u1EDD v\xE0i ph\xFAt r\u1ED3i ch\u1EA1y `avatar ai setup`.";case"timeout":return"Timeout 60s: (1) m\u1EA1ng VN ch\u1EADm \u2014 th\u1EED VPN, (2) Anthropic API spike \u2014 retry v\xE0i ph\xFAt, (3) token v\u1EABn auth nh\u01B0ng quota h\u1EBFt \xE2m th\u1EA7m \u2014 check t\u1EA1i claude.ai/settings/usage.";default:return"L\u1ED7i ch\u01B0a bi\u1EBFt. Xem stderr \u1EDF tr\xEAn + ch\u1EA1y `claude --print ok` tay \u0111\u1EC3 debug."}}function ln(){let t=sn("claude",["--print",xs],{encoding:"utf8",timeout:sr,stdio:["ignore","pipe","pipe"]});if(t.signal==="SIGTERM"||t.status===143||t.error?.code==="ETIMEDOUT")return{ok:!1,reason:"timeout",detail:`claude --print > ${sr/1e3}s (m\u1EA1ng ch\u1EADm / API rate limit / token revoked kh\xF4ng return error)`};let n=t.stderr||"",r=t.stdout||"";if(t.status===0)return{ok:!0};let i=r.trim(),s=n.toLowerCase();if(i.length>20&&!s.includes("error")&&!s.includes("limit")&&!s.includes("quota")&&!s.includes("401"))return o.warn(`claude --print exit=${t.status} nh\u01B0ng c\xF3 response (${i.length} chars). Accept v\u1EDBi caution.`),{ok:!0};let a=As(`${n}
21
- ${r}`);return a==="unknown"&&(o.warn(`[debug] claude --print exit=${t.status} signal=${t.signal??"none"}`),n.trim()&&o.warn(`[debug] stderr: ${n.slice(0,500)}`),r.trim()&&o.warn(`[debug] stdout: ${r.slice(0,300)}`)),{ok:!1,reason:a,detail:n.slice(0,500)||r.slice(0,500)}}import{spawnSync as cr}from"child_process";import{platform as Ss}from"os";function rt(){let t=Ss();return t==="darwin"||t==="linux"||t==="win32"?t:"unsupported"}var Cs=5e3,Ps=/(\d+\.\d+\.\d+)/;function $s(){let e=rt()==="win32"?"where":"which",n=cr(e,["claude"],{encoding:"utf8"});if(n.error||n.status!==0)return null;let r=(n.stdout||"").trim();return r?r.split(/\r?\n/)[0].trim():null}function Es(){let t=cr("claude",["--version"],{encoding:"utf8",timeout:Cs});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return Ps.exec(e)?.[1]??null}var it=null;function Ut(){if(it!==null)return it;let t=$s();return t?(it={installed:!0,version:Es(),path:t},it):(it={installed:!1,version:null,path:null},it)}function ce(){it=null}import{spawnSync as Rs}from"child_process";var lr=300*1e3,ur="@anthropic-ai/claude-code",ot=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallClaudeCodeError",this.reason=e,this.exitCode=r}};function Ts(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new ot("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${ur} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new ot("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new ot("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function pr(){o.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Rs("npm",["install","-g",ur],{stdio:["inherit","inherit","pipe"],timeout:lr,encoding:"utf8"});if(t.signal==="SIGTERM")throw new ot("timeout",`npm install timeout sau ${lr/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),Ts(t.status,t.stderr||"");ce();let e=Ut();if(!e.installed||!e.path)throw new ot("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `claude` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return o.success(`\u0110\xE3 c\xE0i Claude Code${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}import{readFileSync as Is}from"fs";import{homedir as _s}from"os";import{join as Os}from"path";import{select as mr}from"@inquirer/prompts";function Ns(){return Os(_s(),".claude","settings.json")}function un(){let t=Ns(),e;try{e=Is(t,"utf8")}catch{return{exists:!1,hasBaseUrl:!1,hasToken:!1}}let n;try{n=JSON.parse(e)}catch{return{exists:!0,hasBaseUrl:!1,hasToken:!1}}let r=n.env||{},i=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:void 0,s=typeof r.ANTHROPIC_AUTH_TOKEN=="string"&&r.ANTHROPIC_AUTH_TOKEN.length>0,a=typeof n.model=="string"?n.model:void 0;return{exists:!0,hasBaseUrl:!!i,baseUrl:i,hasToken:s,model:a,rawSettings:n}}async function dr(t=un()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await mr({message:`Ph\xE1t hi\u1EC7n AI config global (base URL: ${t.baseUrl}). D\xF9ng cho project n\xE0y?`,choices:[{name:"a. Yes \u2014 copy config global v\xE0o .claude/settings.json (per-project)",value:"use-global"},{name:"b. No \u2014 setup ri\xEAng (ch\u1ECDn provider kh\xE1c)",value:"setup-fresh"}]})==="use-global"?"use-global":await mr({message:"Ch\u1ECDn provider cho AI features:",choices:[{name:"1. Claude Code Subscription (d\xF9ng quota c\xE1 nh\xE2n Anthropic, OAuth login)",value:"subscription"},{name:"2. LLM key NAL (ai.nal.vn \u2014 gateway n\u1ED9i b\u1ED9, key sk-...)",value:"llmlite"},{name:"3. Anthropic API key tr\u1EF1c ti\u1EBFp (console.anthropic.com, key sk-ant-...)",value:"anthropic"},{name:"4. OpenAI/Codex API key (platform.openai.com, key sk-... cho GPT-5.5+)",value:"codex"},{name:"5. Google Gemini API key (aistudio.google.com, key AIza... cho Gemini 2.x+)",value:"gemini"}]})}import{password as Ms,select as js}from"@inquirer/prompts";var le="https://api.anthropic.com",Gs="2023-06-01",gr=1e4;function ue(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function Ls(t){let e=t.trim();return e.length===0?"API key b\u1EAFt bu\u1ED9c":e.startsWith("sk-ant-")?!0:"Anthropic API key th\u01B0\u1EDDng b\u1EAFt \u0111\u1EA7u b\u1EB1ng 'sk-ant-' (l\u1EA5y t\u1EEB console.anthropic.com)."}async function Us(){return await Ms({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:Ls})}async function pe(t){let e=new AbortController,n=setTimeout(()=>e.abort(),gr);try{let r=await fetch(`${le}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":Gs,Accept:"application/json"},signal:e.signal});if(r.status===401)throw new Error("API key invalid (HTTP 401). Check key tr\xEAn console.anthropic.com.");if(r.status===403)throw new Error("API key b\u1ECB reject (HTTP 403). Key c\xF3 th\u1EC3 \u0111\xE3 b\u1ECB revoke ho\u1EB7c thi\u1EBFu permission.");if(r.status===429)throw new Error("Rate limit (HTTP 429). Ch\u1EDD v\xE0i gi\xE2y r\u1ED3i th\u1EED l\u1EA1i.");if(!r.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${r.status}).`);let s=((await r.json()).data||[]).map(a=>typeof a.id=="string"?a.id:null).filter(a=>a!==null);if(s.length===0)throw new Error("Anthropic tr\u1EA3 v\u1EC1 list r\u1ED7ng. Li\xEAn h\u1EC7 support ho\u1EB7c check account.");return s}catch(r){throw r.name==="AbortError"?new Error(`Connect ${le} timeout sau ${gr/1e3}s.`):r}finally{clearTimeout(n)}}async function Hs(t){if(t.length===1){let n=t[0];return o.info(`Auto-pick model: ${n} (ch\u1EC9 1 model available)`),n}let e=[...t].sort((n,r)=>{let i=s=>{let a=s.toLowerCase();return a.includes("sonnet")?0:a.includes("opus")?1:a.includes("haiku")?2:3};return i(n)-i(r)});return await js({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function fr(){let t=await Us();o.info(`Verify key (${ue(t)}) qua ${le}/v1/models...`);let e=await pe(t);o.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await Hs(e);return{apiKey:t,baseUrl:le,model:n}}import{password as Ds,select as Fs}from"@inquirer/prompts";var me="https://generativelanguage.googleapis.com/v1beta",hr=1e4;function Ks(t){return t.length<=12?"AIza***":`${t.slice(0,6)}...${t.slice(-4)}`}function Vs(t){let e=t.trim();return e.length===0?"API key b\u1EAFt bu\u1ED9c":e.startsWith("AIza")?!0:"Google AI Studio key th\u01B0\u1EDDng b\u1EAFt \u0111\u1EA7u b\u1EB1ng 'AIza' (l\u1EA5y t\u1EEB aistudio.google.com/apikey)."}async function Bs(){return await Ds({message:"Gemini API key (AIza..., \u1EA9n input):",mask:"*",validate:Vs})}async function Ws(t){let e=new AbortController,n=setTimeout(()=>e.abort(),hr);try{let r=`${me}/models?key=${encodeURIComponent(t)}`,i=await fetch(r,{method:"GET",headers:{Accept:"application/json"},signal:e.signal});if(!i.ok){let a=await i.text().catch(()=>"");throw i.status===400||i.status===403?new Error(`Key kh\xF4ng valid (${i.status}). Ki\u1EC3m tra l\u1EA1i key t\u1EEB aistudio.google.com.`):i.status===429?new Error("Rate limit (429). Ch\u1EDD v\xE0i gi\xE2y r\u1ED3i th\u1EED l\u1EA1i."):new Error(`Gemini /v1beta/models tr\u1EA3 ${i.status}: ${a.slice(0,200)}`)}return(await i.json()).models.filter(a=>a.supportedGenerationMethods?.includes("generateContent")??!0).map(a=>a.name.replace(/^models\//,""))}catch(r){throw r instanceof Error&&r.name==="AbortError"?new Error(`Connect ${me} timeout sau ${hr/1e3}s.`):r}finally{clearTimeout(n)}}function zs(t){let e=t.filter(n=>/gemini-[23456789]/.test(n));return e.length>0?e.sort():t.sort()}async function qs(t){if(t.length===0)throw new Error("Kh\xF4ng c\xF3 model n\xE0o tr\u1EA3 v\u1EC1 t\u1EEB /v1beta/models.");return t.length===1?(o.info(`Ch\u1EC9 c\xF3 1 model kh\u1EA3 d\u1EE5ng: ${t[0]}. Auto-pick.`),t[0]):await Fs({message:"Ch\u1ECDn Gemini model:",choices:t.map(e=>({name:e,value:e}))})}async function kr(){let t=await Bs();o.info(`Verify key (${Ks(t)}) qua ${me}/models...`);let e=await Ws(t);o.success(`Key valid (${e.length} models available)`);let n=zs(e);n.length!==e.length&&o.dim(` Filter cho Gemini 2.x+: ${n.length}/${e.length}`);let r=await qs(n);return{apiKey:t,baseUrl:me,model:r}}import{input as Ys,password as Js,select as Xs}from"@inquirer/prompts";var Zs="https://ai.nal.vn",wr=1e4;function K(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function Qs(){return await Js({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function ta(t=Zs){return(await Ys({message:"LLMLite base URL:",default:t,validate:n=>/^https?:\/\//.test(n)?!0:"Ph\u1EA3i l\xE0 URL h\u1EE3p l\u1EC7 (http/https)"})).replace(/\/+$/,"")}async function ea(t,e){let n=new AbortController,r=setTimeout(()=>n.abort(),wr);try{let i=await fetch(`${t}/v1/models`,{method:"GET",headers:{Authorization:`Bearer ${e}`,Accept:"application/json"},signal:n.signal});if(i.status===401||i.status===403)throw new Error(`API key invalid (HTTP ${i.status}).`);if(i.status===404)throw new Error(`Endpoint /v1/models kh\xF4ng t\u1ED3n t\u1EA1i tr\xEAn ${t}.`);if(!i.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${i.status}).`);let a=((await i.json()).data||[]).map(c=>typeof c.id=="string"?c.id:null).filter(c=>c!==null);if(a.length===0)throw new Error("LLMLite tr\u1EA3 v\u1EC1 list r\u1ED7ng. Li\xEAn h\u1EC7 admin NAL.");return a}catch(i){if(i.name==="AbortError"){let c=t.includes("nal.vn")||t.includes("nal-vn")?"\n Hint: ai.nal.vn l\xE0 internal endpoint NAL \u2014 ki\u1EC3m tra VPN NAL \u0111\xE3 b\u1EADt ch\u01B0a, ho\u1EB7c tcp test b\u1EB1ng `curl -v https://ai.nal.vn`.":`
22
- Hint: check m\u1EA1ng / firewall / VPN, ho\u1EB7c base URL c\xF3 \u0111\xFAng kh\xF4ng.`;throw new Error(`Connect ${t} timeout sau ${wr/1e3}s.${c}`)}let s=i.message||String(i);if(s.toLowerCase().includes("fetch failed")||s.includes("ENOTFOUND")){let c=t.includes("nal.vn")?" (ai.nal.vn c\u1EA7n VPN NAL \u2014 ki\u1EC3m tra VPN \u0111\xE3 b\u1EADt ch\u01B0a)":"";throw new Error(`Network error khi connect ${t}: ${s}${c}`)}throw i}finally{clearTimeout(r)}}async function na(t){let e=t.filter(r=>r.toLowerCase().includes("claude"));if(e.length===1){let r=e[0];return o.info(`Auto-pick model: ${r} (ch\u1EC9 1 claude alias tr\xEAn endpoint)`),r}let n=e.length>0?e:t;return await Xs({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function yr(){let t=await Qs(),e=await ta();o.info(`Verify key (${K(t)}) qua ${e}/v1/models...`);let n=await ea(e,t);o.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await na(n);return{apiKey:t,baseUrl:e,model:r}}import{password as ra,select as ia}from"@inquirer/prompts";var de="https://api.openai.com/v1",vr=1e4;function oa(t){return t.length<=12?"sk-***":`${t.slice(0,7)}...${t.slice(-4)}`}function sa(t){let e=t.trim();return e.length===0?"API key b\u1EAFt bu\u1ED9c":e.startsWith("sk-")?!0:"OpenAI API key th\u01B0\u1EDDng b\u1EAFt \u0111\u1EA7u b\u1EB1ng 'sk-' (l\u1EA5y t\u1EEB platform.openai.com/api-keys)."}async function aa(){return await ra({message:"OpenAI API key (sk-..., \u1EA9n input):",mask:"*",validate:sa})}async function ca(t){let e=new AbortController,n=setTimeout(()=>e.abort(),vr);try{let r=await fetch(`${de}/models`,{method:"GET",headers:{Authorization:`Bearer ${t}`,Accept:"application/json"},signal:e.signal});if(!r.ok){let s=await r.text().catch(()=>"");throw r.status===401?new Error("Key kh\xF4ng valid (401 Unauthorized). Ki\u1EC3m tra l\u1EA1i key."):r.status===429?new Error("Rate limit (429). Ch\u1EDD v\xE0i gi\xE2y r\u1ED3i th\u1EED l\u1EA1i."):new Error(`OpenAI /v1/models tr\u1EA3 ${r.status}: ${s.slice(0,200)}`)}return(await r.json()).data.map(s=>s.id)}catch(r){throw r instanceof Error&&r.name==="AbortError"?new Error(`Connect ${de} timeout sau ${vr/1e3}s.`):r}finally{clearTimeout(n)}}function la(t){let e=t.filter(n=>{let r=n.toLowerCase();return r.startsWith("gpt-5")||r.startsWith("o1")||r.startsWith("o2")||r.startsWith("o3")||r.startsWith("o4")||r.startsWith("o5")||r==="gpt-4o"||r==="gpt-4o-mini"});return e.length>0?e.sort():t.sort()}async function ua(t){if(t.length===0)throw new Error("Kh\xF4ng c\xF3 model n\xE0o tr\u1EA3 v\u1EC1 t\u1EEB /v1/models \u2014 key c\xF3 th\u1EC3 kh\xF4ng c\xF3 quy\u1EC1n.");return t.length===1?(o.info(`Ch\u1EC9 c\xF3 1 model kh\u1EA3 d\u1EE5ng: ${t[0]}. Auto-pick.`),t[0]):await ia({message:"Ch\u1ECDn model:",choices:t.map(e=>({name:e,value:e}))})}async function br(){let t=await aa();o.info(`Verify key (${oa(t)}) qua ${de}/models...`);let e=await ca(t);o.success(`Key valid (${e.length} models available)`);let n=la(e);n.length!==e.length&&o.dim(` Filter cho coding models: ${n.length}/${e.length}`);let r=await ua(n);return{apiKey:t,baseUrl:de,model:r}}import{promises as we}from"fs";import{join as Rr}from"path";import{spawnSync as pa}from"child_process";import{existsSync as ma,readFileSync as da}from"fs";import{homedir as ga}from"os";import{join as fa}from"path";function pn(t,e){try{let n=pa(t,e,{encoding:"utf8",timeout:5e3});return{ok:n.status===0,stdout:(n.stdout??"").trim()}}catch{return{ok:!1,stdout:""}}}async function bt(){let t=pn("direnv",["--version"]);if(t.ok&&t.stdout){let n=pn("which",["direnv"]);return{installed:!0,version:t.stdout,binaryPath:n.ok?n.stdout:null,shellHookInZshrc:xr()}}let e=pn("which",["direnv"]);return e.ok&&e.stdout?{installed:!0,version:null,binaryPath:e.stdout,shellHookInZshrc:xr()}:{installed:!1,version:null,binaryPath:null,shellHookInZshrc:!1}}function xr(){let t=fa(ga(),".zshrc");if(!ma(t))return!1;try{let e=da(t,"utf8");return/\bdirenv\s+hook\s+(zsh|bash|sh)\b/.test(e)}catch{return!1}}import{spawnSync as Ar}from"child_process";import{appendFileSync as ha,existsSync as ka,readFileSync as wa}from"fs";import{homedir as ya,platform as va}from"os";import{join as ba}from"path";var st="https://direnv.net/docs/installation.html";function ge(t){return Ar("which",[t],{encoding:"utf8",timeout:3e3}).status===0}function fe(t,e){try{let n=Ar(t,e,{encoding:"utf8",timeout:12e4});return{ok:n.status===0,output:((n.stdout??"")+(n.stderr??"")).trim()}}catch(n){return{ok:!1,output:n instanceof Error?n.message:String(n)}}}async function he(){let t=va();if(t==="darwin"){if(ge("brew")){let e=fe("brew",["install","direnv"]);return{success:e.ok,method:"brew",message:e.ok?"Installed via Homebrew":`brew install failed: ${e.output.slice(0,500)}`,manualInstructionsUrl:st}}return{success:!1,method:null,message:"macOS detected but Homebrew not found. Install brew first: https://brew.sh",manualInstructionsUrl:st}}if(t==="linux"){if(ge("apt-get")){let e=fe("sudo",["apt-get","install","-y","direnv"]);return{success:e.ok,method:"apt-get",message:e.ok?"Installed via apt-get":`apt-get install failed: ${e.output.slice(0,500)}`,manualInstructionsUrl:st}}if(ge("dnf")){let e=fe("sudo",["dnf","install","-y","direnv"]);return{success:e.ok,method:"dnf",message:e.ok?"Installed via dnf":`dnf install failed: ${e.output.slice(0,500)}`,manualInstructionsUrl:st}}if(ge("pacman")){let e=fe("sudo",["pacman","-S","--noconfirm","direnv"]);return{success:e.ok,method:"pacman",message:e.ok?"Installed via pacman":`pacman install failed: ${e.output.slice(0,500)}`,manualInstructionsUrl:st}}return{success:!1,method:null,message:"Linux detected but no supported package manager (apt-get/dnf/pacman) found.",manualInstructionsUrl:st}}return{success:!1,method:null,message:`Unsupported platform: ${t}. Install direnv manually.`,manualInstructionsUrl:st}}function ke(){let t=ba(ya(),".zshrc"),e='eval "$(direnv hook zsh)"',n="# direnv shell integration (added by avatar-cli)",r="";if(ka(t))try{r=wa(t,"utf8")}catch{}if(/\bdirenv\s+hook\s+(zsh|bash|sh)\b/.test(r))return{modified:!1,zshrcPath:t};let i=`
20
+ `)};function D(t){return fs({text:t,spinner:"dots",isEnabled:process.stdout.isTTY??!1}).start()}function Ht(t){let e=Date.now(),n=D(`${t} (0:00)`),r=()=>{let s=Math.floor((Date.now()-e)/1e3),a=Math.floor(s/60),c=s%60;return`${a}:${String(c).padStart(2,"0")}`},i=setInterval(()=>{n.text=`${t} (${r()})`},1e3);return{succeed:s=>{clearInterval(i),n.succeed(`${s} (${r()})`)},fail:s=>{clearInterval(i),n.fail(`${s} (${r()})`)},stop:()=>{clearInterval(i),n.stop()}}}var rr=6e4,hs="ok";function nn(){let t=en("claude",["auth","status"],{encoding:"utf8"});if(t.error||t.status!==0)return{state:"not-authenticated"};let e=(t.stdout||"").trim();if(!e.startsWith("{"))return{state:"authenticated"};try{let n=JSON.parse(e);return n.loggedIn!==!0?{state:"not-authenticated"}:{state:"authenticated",email:n.email,subscriptionType:n.subscriptionType,apiProvider:n.apiProvider}}catch{return{state:"authenticated"}}}function rn(){o.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=en("claude",["auth","login"],{stdio:"inherit"});if(t.status!==0)throw new Error(`claude auth login th\u1EA5t b\u1EA1i (exit ${t.status}). Th\u1EED 'claude auth login' tay r\u1ED3i ch\u1EA1y l\u1EA1i.`);o.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp Claude Code")}function ks(t){let e=t.toLowerCase();return e.includes("credit_balance_too_low")||e.includes("credit balance too low")?"credit_balance_too_low":e.includes("insufficient_quota")||e.includes("insufficient quota")||e.includes("quota_exceeded")||e.includes("quota exceeded")||e.includes("usage limit")||e.includes("you've used all")?"insufficient_quota":e.includes("401")||e.includes("invalid authentication")||e.includes("authentication credentials")||e.includes("failed to authenticate")||e.includes("authentication failed")||e.includes("unauthorized")?"auth-expired":e.includes("invalid_api_key")||e.includes("invalid api key")?"invalid_api_key":e.includes("rate_limit")||e.includes("rate limit")||e.includes("429")?"rate_limit":"unknown"}function ir(t){switch(t){case"auth-expired":return"Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n/b\u1ECB revoke. Ch\u1EA1y: `claude auth logout && claude auth login`.";case"credit_balance_too_low":case"insufficient_quota":return"H\u1EBFt quota subscription. Upgrade plan ho\u1EB7c d\xF9ng LLMLite (avatar ai setup \u2192 ch\u1ECDn LLMLite).";case"invalid_api_key":return"API key invalid. Re-login: `claude auth login`.";case"rate_limit":return"B\u1ECB rate limit t\u1EA1m th\u1EDDi. Ch\u1EDD v\xE0i ph\xFAt r\u1ED3i ch\u1EA1y `avatar ai setup`.";case"timeout":return"Timeout 60s: (1) m\u1EA1ng VN ch\u1EADm \u2014 th\u1EED VPN, (2) Anthropic API spike \u2014 retry v\xE0i ph\xFAt, (3) token v\u1EABn auth nh\u01B0ng quota h\u1EBFt \xE2m th\u1EA7m \u2014 check t\u1EA1i claude.ai/settings/usage.";default:return"L\u1ED7i ch\u01B0a bi\u1EBFt. Xem stderr \u1EDF tr\xEAn + ch\u1EA1y `claude --print ok` tay \u0111\u1EC3 debug."}}function on(){let t=en("claude",["--print",hs],{encoding:"utf8",timeout:rr,stdio:["ignore","pipe","pipe"]});if(t.signal==="SIGTERM"||t.status===143||t.error?.code==="ETIMEDOUT")return{ok:!1,reason:"timeout",detail:`claude --print > ${rr/1e3}s (m\u1EA1ng ch\u1EADm / API rate limit / token revoked kh\xF4ng return error)`};let n=t.stderr||"",r=t.stdout||"";if(t.status===0)return{ok:!0};let i=r.trim(),s=n.toLowerCase();if(i.length>20&&!s.includes("error")&&!s.includes("limit")&&!s.includes("quota")&&!s.includes("401"))return o.warn(`claude --print exit=${t.status} nh\u01B0ng c\xF3 response (${i.length} chars). Accept v\u1EDBi caution.`),{ok:!0};let a=ks(`${n}
21
+ ${r}`);return a==="unknown"&&(o.warn(`[debug] claude --print exit=${t.status} signal=${t.signal??"none"}`),n.trim()&&o.warn(`[debug] stderr: ${n.slice(0,500)}`),r.trim()&&o.warn(`[debug] stdout: ${r.slice(0,300)}`)),{ok:!1,reason:a,detail:n.slice(0,500)||r.slice(0,500)}}import{spawnSync as or}from"child_process";import{platform as ws}from"os";function nt(){let t=ws();return t==="darwin"||t==="linux"||t==="win32"?t:"unsupported"}var ys=5e3,vs=/(\d+\.\d+\.\d+)/;function bs(){let e=nt()==="win32"?"where":"which",n=or(e,["claude"],{encoding:"utf8"});if(n.error||n.status!==0)return null;let r=(n.stdout||"").trim();return r?r.split(/\r?\n/)[0].trim():null}function xs(){let t=or("claude",["--version"],{encoding:"utf8",timeout:ys});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return vs.exec(e)?.[1]??null}var rt=null;function Gt(){if(rt!==null)return rt;let t=bs();return t?(rt={installed:!0,version:xs(),path:t},rt):(rt={installed:!1,version:null,path:null},rt)}function ae(){rt=null}import{spawnSync as Ss}from"child_process";var sr=300*1e3,ar="@anthropic-ai/claude-code",it=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallClaudeCodeError",this.reason=e,this.exitCode=r}};function As(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new it("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${ar} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new it("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new it("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function cr(){o.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Ss("npm",["install","-g",ar],{stdio:["inherit","inherit","pipe"],timeout:sr,encoding:"utf8"});if(t.signal==="SIGTERM")throw new it("timeout",`npm install timeout sau ${sr/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),As(t.status,t.stderr||"");ae();let e=Gt();if(!e.installed||!e.path)throw new it("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `claude` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return o.success(`\u0110\xE3 c\xE0i Claude Code${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}import{readFileSync as Cs}from"fs";import{homedir as Ps}from"os";import{join as $s}from"path";import{select as lr}from"@inquirer/prompts";function Es(){return $s(Ps(),".claude","settings.json")}function sn(){let t=Es(),e;try{e=Cs(t,"utf8")}catch{return{exists:!1,hasBaseUrl:!1,hasToken:!1}}let n;try{n=JSON.parse(e)}catch{return{exists:!0,hasBaseUrl:!1,hasToken:!1}}let r=n.env||{},i=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:void 0,s=typeof r.ANTHROPIC_AUTH_TOKEN=="string"&&r.ANTHROPIC_AUTH_TOKEN.length>0,a=typeof n.model=="string"?n.model:void 0;return{exists:!0,hasBaseUrl:!!i,baseUrl:i,hasToken:s,model:a,rawSettings:n}}async function ur(t=sn()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await lr({message:`Ph\xE1t hi\u1EC7n AI config global (base URL: ${t.baseUrl}). D\xF9ng cho project n\xE0y?`,choices:[{name:"a. Yes \u2014 copy config global v\xE0o .claude/settings.json (per-project)",value:"use-global"},{name:"b. No \u2014 setup ri\xEAng (ch\u1ECDn provider kh\xE1c)",value:"setup-fresh"}]})==="use-global"?"use-global":await lr({message:"Ch\u1ECDn provider cho AI features:",choices:[{name:"1. Claude Code Subscription (d\xF9ng quota c\xE1 nh\xE2n Anthropic, OAuth login)",value:"subscription"},{name:"2. LLM key NAL (ai.nal.vn \u2014 gateway n\u1ED9i b\u1ED9, key sk-...)",value:"llmlite"},{name:"3. Anthropic API key tr\u1EF1c ti\u1EBFp (console.anthropic.com, key sk-ant-...)",value:"anthropic"}]})}import{password as Rs,select as Ts}from"@inquirer/prompts";var ce="https://api.anthropic.com",_s="2023-06-01",pr=1e4;function le(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function Is(t){let e=t.trim();return e.length===0?"API key b\u1EAFt bu\u1ED9c":e.startsWith("sk-ant-")?!0:"Anthropic API key th\u01B0\u1EDDng b\u1EAFt \u0111\u1EA7u b\u1EB1ng 'sk-ant-' (l\u1EA5y t\u1EEB console.anthropic.com)."}async function Os(){return await Rs({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:Is})}async function ue(t){let e=new AbortController,n=setTimeout(()=>e.abort(),pr);try{let r=await fetch(`${ce}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":_s,Accept:"application/json"},signal:e.signal});if(r.status===401)throw new Error("API key invalid (HTTP 401). Check key tr\xEAn console.anthropic.com.");if(r.status===403)throw new Error("API key b\u1ECB reject (HTTP 403). Key c\xF3 th\u1EC3 \u0111\xE3 b\u1ECB revoke ho\u1EB7c thi\u1EBFu permission.");if(r.status===429)throw new Error("Rate limit (HTTP 429). Ch\u1EDD v\xE0i gi\xE2y r\u1ED3i th\u1EED l\u1EA1i.");if(!r.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${r.status}).`);let s=((await r.json()).data||[]).map(a=>typeof a.id=="string"?a.id:null).filter(a=>a!==null);if(s.length===0)throw new Error("Anthropic tr\u1EA3 v\u1EC1 list r\u1ED7ng. Li\xEAn h\u1EC7 support ho\u1EB7c check account.");return s}catch(r){throw r.name==="AbortError"?new Error(`Connect ${ce} timeout sau ${pr/1e3}s.`):r}finally{clearTimeout(n)}}async function Ns(t){if(t.length===1){let n=t[0];return o.info(`Auto-pick model: ${n} (ch\u1EC9 1 model available)`),n}let e=[...t].sort((n,r)=>{let i=s=>{let a=s.toLowerCase();return a.includes("sonnet")?0:a.includes("opus")?1:a.includes("haiku")?2:3};return i(n)-i(r)});return await Ts({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function mr(){let t=await Os();o.info(`Verify key (${le(t)}) qua ${ce}/v1/models...`);let e=await ue(t);o.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await Ns(e);return{apiKey:t,baseUrl:ce,model:n}}import{input as Ms,password as js,select as Ls}from"@inquirer/prompts";var Hs="https://ai.nal.vn",dr=1e4;function F(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function Gs(){return await js({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function Us(t=Hs){return(await Ms({message:"LLMLite base URL:",default:t,validate:n=>/^https?:\/\//.test(n)?!0:"Ph\u1EA3i l\xE0 URL h\u1EE3p l\u1EC7 (http/https)"})).replace(/\/+$/,"")}async function Ds(t,e){let n=new AbortController,r=setTimeout(()=>n.abort(),dr);try{let i=await fetch(`${t}/v1/models`,{method:"GET",headers:{Authorization:`Bearer ${e}`,Accept:"application/json"},signal:n.signal});if(i.status===401||i.status===403)throw new Error(`API key invalid (HTTP ${i.status}).`);if(i.status===404)throw new Error(`Endpoint /v1/models kh\xF4ng t\u1ED3n t\u1EA1i tr\xEAn ${t}.`);if(!i.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${i.status}).`);let a=((await i.json()).data||[]).map(c=>typeof c.id=="string"?c.id:null).filter(c=>c!==null);if(a.length===0)throw new Error("LLMLite tr\u1EA3 v\u1EC1 list r\u1ED7ng. Li\xEAn h\u1EC7 admin NAL.");return a}catch(i){if(i.name==="AbortError"){let c=t.includes("nal.vn")||t.includes("nal-vn")?"\n Hint: ai.nal.vn l\xE0 internal endpoint NAL \u2014 ki\u1EC3m tra VPN NAL \u0111\xE3 b\u1EADt ch\u01B0a, ho\u1EB7c tcp test b\u1EB1ng `curl -v https://ai.nal.vn`.":`
22
+ Hint: check m\u1EA1ng / firewall / VPN, ho\u1EB7c base URL c\xF3 \u0111\xFAng kh\xF4ng.`;throw new Error(`Connect ${t} timeout sau ${dr/1e3}s.${c}`)}let s=i.message||String(i);if(s.toLowerCase().includes("fetch failed")||s.includes("ENOTFOUND")){let c=t.includes("nal.vn")?" (ai.nal.vn c\u1EA7n VPN NAL \u2014 ki\u1EC3m tra VPN \u0111\xE3 b\u1EADt ch\u01B0a)":"";throw new Error(`Network error khi connect ${t}: ${s}${c}`)}throw i}finally{clearTimeout(r)}}async function Fs(t){let e=t.filter(r=>r.toLowerCase().includes("claude"));if(e.length===1){let r=e[0];return o.info(`Auto-pick model: ${r} (ch\u1EC9 1 claude alias tr\xEAn endpoint)`),r}let n=e.length>0?e:t;return await Ls({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function gr(){let t=await Gs(),e=await Us();o.info(`Verify key (${F(t)}) qua ${e}/v1/models...`);let n=await Ds(e,t);o.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await Fs(n);return{apiKey:t,baseUrl:e,model:r}}import{promises as fe}from"fs";import{join as xr}from"path";import{spawnSync as Vs}from"child_process";import{existsSync as Ks,readFileSync as Bs}from"fs";import{homedir as Ws}from"os";import{join as zs}from"path";function an(t,e){try{let n=Vs(t,e,{encoding:"utf8",timeout:5e3});return{ok:n.status===0,stdout:(n.stdout??"").trim()}}catch{return{ok:!1,stdout:""}}}async function bt(){let t=an("direnv",["--version"]);if(t.ok&&t.stdout){let n=an("which",["direnv"]);return{installed:!0,version:t.stdout,binaryPath:n.ok?n.stdout:null,shellHookInZshrc:fr()}}let e=an("which",["direnv"]);return e.ok&&e.stdout?{installed:!0,version:null,binaryPath:e.stdout,shellHookInZshrc:fr()}:{installed:!1,version:null,binaryPath:null,shellHookInZshrc:!1}}function fr(){let t=zs(Ws(),".zshrc");if(!Ks(t))return!1;try{let e=Bs(t,"utf8");return/\bdirenv\s+hook\s+(zsh|bash|sh)\b/.test(e)}catch{return!1}}import{spawnSync as hr}from"child_process";import{appendFileSync as qs,existsSync as Js,readFileSync as Ys}from"fs";import{homedir as Xs,platform as Zs}from"os";import{join as Qs}from"path";var ot="https://direnv.net/docs/installation.html";function pe(t){return hr("which",[t],{encoding:"utf8",timeout:3e3}).status===0}function me(t,e){try{let n=hr(t,e,{encoding:"utf8",timeout:12e4});return{ok:n.status===0,output:((n.stdout??"")+(n.stderr??"")).trim()}}catch(n){return{ok:!1,output:n instanceof Error?n.message:String(n)}}}async function de(){let t=Zs();if(t==="darwin"){if(pe("brew")){let e=me("brew",["install","direnv"]);return{success:e.ok,method:"brew",message:e.ok?"Installed via Homebrew":`brew install failed: ${e.output.slice(0,500)}`,manualInstructionsUrl:ot}}return{success:!1,method:null,message:"macOS detected but Homebrew not found. Install brew first: https://brew.sh",manualInstructionsUrl:ot}}if(t==="linux"){if(pe("apt-get")){let e=me("sudo",["apt-get","install","-y","direnv"]);return{success:e.ok,method:"apt-get",message:e.ok?"Installed via apt-get":`apt-get install failed: ${e.output.slice(0,500)}`,manualInstructionsUrl:ot}}if(pe("dnf")){let e=me("sudo",["dnf","install","-y","direnv"]);return{success:e.ok,method:"dnf",message:e.ok?"Installed via dnf":`dnf install failed: ${e.output.slice(0,500)}`,manualInstructionsUrl:ot}}if(pe("pacman")){let e=me("sudo",["pacman","-S","--noconfirm","direnv"]);return{success:e.ok,method:"pacman",message:e.ok?"Installed via pacman":`pacman install failed: ${e.output.slice(0,500)}`,manualInstructionsUrl:ot}}return{success:!1,method:null,message:"Linux detected but no supported package manager (apt-get/dnf/pacman) found.",manualInstructionsUrl:ot}}return{success:!1,method:null,message:`Unsupported platform: ${t}. Install direnv manually.`,manualInstructionsUrl:ot}}function ge(){let t=Qs(Xs(),".zshrc"),e='eval "$(direnv hook zsh)"',n="# direnv shell integration (added by avatar-cli)",r="";if(Js(t))try{r=Ys(t,"utf8")}catch{}if(/\bdirenv\s+hook\s+(zsh|bash|sh)\b/.test(r))return{modified:!1,zshrcPath:t};let i=`
23
23
  ${n}
24
24
  ${e}
25
- `;try{return ha(t,i,"utf8"),{modified:!0,zshrcPath:t}}catch{return{modified:!1,zshrcPath:t}}}Ht();var Ia=[".claude",".env"];async function _a(t,e){let n=Rr(t,...Ia);await we.mkdir(Rr(t,".claude"),{recursive:!0});let r="";try{r=await we.readFile(n,"utf8")}catch{}let i=r?r.split(`
25
+ `;try{return qs(t,i,"utf8"),{modified:!0,zshrcPath:t}}catch{return{modified:!1,zshrcPath:t}}}Ut();var la=[".claude",".env"];async function ua(t,e){let n=xr(t,...la);await fe.mkdir(xr(t,".claude"),{recursive:!0});let r="";try{r=await fe.readFile(n,"utf8")}catch{}let i=r?r.split(`
26
26
  `):[],s=new Set(Object.keys(e)),a=i.filter(l=>{let u=l.match(/^([A-Z][A-Z0-9_]*)=/);return!(u?.[1]&&s.has(u[1]))});for(let[l,u]of Object.entries(e)){let m=u.replace(/'/g,"'\\''");a.push(`${l}='${m}'`)}let c=a.filter((l,u,m)=>!(l===""&&m[u-1]==="")).join(`
27
- `);return await we.writeFile(n,c.endsWith(`
27
+ `);return await fe.writeFile(n,c.endsWith(`
28
28
  `)?c:`${c}
29
- `,{encoding:"utf8",mode:384}),await we.chmod(n,384),n}async function Dt(t,e){let n=await bt(),r=!1,i=!1;if(!n.installed){o.info("\u0110ang setup direnv \u0111\u1EC3 b\u1EA3o v\u1EC7 API key (silent)...");let a=await he();if(a.success)o.dim(` \u2713 direnv installed via ${a.method}`),r=!0,n=await bt();else{o.warn(` ! direnv install fail (${a.message}). Fallback sang .claude/.env (gitignored).`);let c=await _a(t,e);return o.warn(` \u26A0 Key \u0111\xE3 ghi v\xE0o ${c} (chmod 600) nh\u01B0ng KH\xD4NG auto-load \u2014 c\u1EA7n 'source ${c}' m\u1ED7i terminal m\u1EDBi.`),o.dim(` Manual install direnv sau: ${a.manualInstructionsUrl}`),{storageMethod:"claude-env-fallback",storagePath:c,autoLoad:!1,installedDirenvNow:!1,addedShellHook:!1}}}if(!n.shellHookInZshrc){let a=ke();a.modified&&(o.dim(` \u2713 Added direnv hook to ${a.zshrcPath}`),o.dim(" Reload shell: 'source ~/.zshrc' ho\u1EB7c m\u1EDF terminal m\u1EDBi"),i=!0)}let s=await at(t,e);return o.success(`\u2713 Key \u0111\xE3 l\u01B0u (${Object.keys(e).length} vars) v\xE0o ${s.envrcPath}`),s.direnvAllowOk===!1&&o.warn(" direnv allow fail \u2014 ch\u1EA1y 'direnv allow' trong workspace \u0111\u1EC3 activate."),{storageMethod:"envrc",storagePath:s.envrcPath,autoLoad:!0,installedDirenvNow:r,addedShellHook:i}}k();import{promises as Oa}from"fs";import{join as Na}from"path";var dn=384;function Ma(t){return Na(t,".claude","settings.json")}async function ja(t){if(!await d(t))return{};try{return await v(t)}catch(e){throw new Error(`Kh\xF4ng parse \u0111\u01B0\u1EE3c ${t} (JSON l\u1ED7i): ${e.message}. Backup file r\u1ED3i x\xF3a \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}function Ga(t,e){let{env:n,...r}=t,i={...r,model:e};if(n){let{ANTHROPIC_BASE_URL:s,ANTHROPIC_AUTH_TOKEN:a,ANTHROPIC_API_KEY:c,...l}=n;Object.keys(l).length>0&&(i.env=l)}return i}function ye(t){if(!t)return t;let{ANTHROPIC_AUTH_TOKEN:e,ANTHROPIC_API_KEY:n,OPENAI_API_KEY:r,GEMINI_API_KEY:i,...s}=t;return s}function La(t,e,n,r,i){let a={...ye(t.env),ANTHROPIC_BASE_URL:n};return i||(a.ANTHROPIC_AUTH_TOKEN=e),{...t,env:a,model:r,avatarProvider:"llmlite"}}function Ua(t,e,n,r,i){let a={...ye(t.env),ANTHROPIC_BASE_URL:n};return i||(a.ANTHROPIC_API_KEY=e),{...t,env:a,model:r,avatarProvider:"anthropic"}}function Ha(t,e,n,r,i){let a={...ye(t.env),OPENAI_BASE_URL:n};return i||(a.OPENAI_API_KEY=e),{...t,env:a,model:r,avatarProvider:"codex"}}function Da(t,e,n,r,i){let a={...ye(t.env),GEMINI_BASE_URL:n};return i||(a.GEMINI_API_KEY=e),{...t,env:a,model:r,avatarProvider:"gemini"}}function Fa(t,e){let n=e.env||{},r=typeof e.model=="string"?e.model:void 0;return{...t,env:{...t.env||{},...n},...r?{model:r}:{}}}async function j(t,e){let n=Ma(t),r=await ja(n),i;switch(e.provider){case"subscription":i=Ga(r,e.model);break;case"llmlite":i=La(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"anthropic":i=Ua(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"codex":i=Ha(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"gemini":i=Da(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"use-global":i=Fa(r,e.sourceSettings);break}await $(n,i,dn);try{await Oa.chmod(n,dn)}catch{}return{path:n,mode:dn}}var G="sonnet";async function ve(t){try{o.info("Setup AI provider cho workspace...");let e=Ut();if(e.installed)o.success(`Claude Code \u0111\xE3 c\xF3${e.version?` v${e.version}`:""}`);else if(o.info("Ch\u01B0a c\xF3 Claude Code \u2014 s\u1EBD t\u1EF1 c\xE0i qua npm."),pr(),ce(),e=Ut(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=un();switch(await dr(n)){case"subscription":{let i=an();if(i.state!=="authenticated"&&(cn(),i=an()),i.state==="authenticated"&&i.subscriptionType)return await j(t.workspacePath,{provider:"subscription",model:G}),await w("ai_setup",`provider=subscription,result=ok,plan=${i.subscriptionType},probe=skipped`),o.success(`AI ready \xB7 Subscription (${i.subscriptionType}) \xB7 model=${G}`),{ok:!0,provider:"subscription",model:G};o.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let s=ln();if(!s.ok&&s.reason==="auth-expired"&&(o.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),cn(),s=ln()),!s.ok&&(s.reason==="timeout"||s.reason==="unknown"))return o.warn(`Probe verify ${s.reason} \u2014 accept trust auth status. Ti\u1EBFp t\u1EE5c.`),s.detail?.trim()&&o.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),await j(t.workspacePath,{provider:"subscription",model:G}),await w("ai_setup",`provider=subscription,result=ok,probe=${s.reason}-soft-pass`),o.success(`AI ready \xB7 Subscription (probe ${s.reason}, soft-pass) \xB7 model=${G}`),{ok:!0,provider:"subscription",model:G};if(!s.ok){let a=s.reason??"unknown";return await w("ai_setup",`provider=subscription,result=no-quota,reason=${a}`),o.warn(`Subscription verify th\u1EA5t b\u1EA1i (${a}).`),s.detail?.trim()&&o.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),o.warn(` \u2192 ${ar(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await j(t.workspacePath,{provider:"subscription",model:G}),await w("ai_setup","provider=subscription,result=ok"),o.success(`AI ready \xB7 Subscription \xB7 model=${G}`),{ok:!0,provider:"subscription",model:G}}case"llmlite":{let i=await yr(),s=await Dt(t.workspacePath,{ANTHROPIC_AUTH_TOKEN:i.apiKey});return await j(t.workspacePath,{provider:"llmlite",apiKey:i.apiKey,baseUrl:i.baseUrl,model:i.model,skipApiKey:s.storageMethod==="envrc"}),await w("ai_setup",`provider=llmlite,result=ok,model=${i.model},base=${i.baseUrl},storage=${s.storageMethod}`),o.success(`AI ready \xB7 LLMLite (NAL) \xB7 model=${i.model} \xB7 ${i.baseUrl}`),{ok:!0,provider:"llmlite",model:i.model}}case"anthropic":{let i=await fr(),s=await Dt(t.workspacePath,{ANTHROPIC_API_KEY:i.apiKey});return await j(t.workspacePath,{provider:"anthropic",apiKey:i.apiKey,baseUrl:i.baseUrl,model:i.model,skipApiKey:s.storageMethod==="envrc"}),await w("ai_setup",`provider=anthropic,result=ok,model=${i.model},storage=${s.storageMethod}`),o.success(`AI ready \xB7 Anthropic Direct \xB7 model=${i.model} \xB7 ${i.baseUrl}`),{ok:!0,provider:"anthropic",model:i.model}}case"codex":{let i=await br(),s=await Dt(t.workspacePath,{OPENAI_API_KEY:i.apiKey});return await j(t.workspacePath,{provider:"codex",apiKey:i.apiKey,baseUrl:i.baseUrl,model:i.model,skipApiKey:s.storageMethod==="envrc"}),await w("ai_setup",`provider=codex,result=ok,model=${i.model},storage=${s.storageMethod}`),o.success(`AI ready \xB7 OpenAI/Codex Direct \xB7 model=${i.model} \xB7 ${i.baseUrl}`),o.dim(" Note: Claude Code native kh\xF4ng support OpenAI \u2014 c\u1EA7n proxy qua LiteLLM/ai.nal.vn \u0111\u1EC3 d\xF9ng l\xE0m main AI. Key \u0111\xE3 l\u01B0u \u0111\u1EC3 skills kh\xE1c d\xF9ng (vd ai-multimodal)."),{ok:!0,provider:"codex",model:i.model}}case"gemini":{let i=await kr(),s=await Dt(t.workspacePath,{GEMINI_API_KEY:i.apiKey});return await j(t.workspacePath,{provider:"gemini",apiKey:i.apiKey,baseUrl:i.baseUrl,model:i.model,skipApiKey:s.storageMethod==="envrc"}),await w("ai_setup",`provider=gemini,result=ok,model=${i.model},storage=${s.storageMethod}`),o.success(`AI ready \xB7 Google Gemini Direct \xB7 model=${i.model} \xB7 ${i.baseUrl}`),o.dim(" Note: Claude Code native kh\xF4ng support Gemini \u2014 c\u1EA7n proxy qua LiteLLM/ai.nal.vn \u0111\u1EC3 d\xF9ng l\xE0m main AI. Key \u0111\xE3 l\u01B0u \u0111\u1EC3 skills kh\xE1c d\xF9ng."),{ok:!0,provider:"gemini",model:i.model}}case"use-global":{if(!n.rawSettings)throw new Error("use-global ch\u1ECDn nh\u01B0ng kh\xF4ng \u0111\u1ECDc \u0111\u01B0\u1EE3c global settings.");return await j(t.workspacePath,{provider:"use-global",sourceSettings:n.rawSettings}),await w("ai_setup","provider=use-global,result=ok"),o.success(`AI ready \xB7 Copy t\u1EEB global config (${n.baseUrl??"subscription"})`),{ok:!0,provider:"use-global",model:n.model}}}}catch(e){let n=e instanceof Error?e.message:String(e);return o.warn(`AI setup th\u1EA5t b\u1EA1i: ${n}`),o.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup AI sau qua: avatar ai setup"),await w("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}import{spawnSync as Ka}from"child_process";var gn=1e4,Tr=3e4,_r=5,fn="say ok",Ir="2023-06-01";async function Va(t,e,n){o.info(`Testing LLMLite provider: ${t} (key: ${K(e)})`);let r=new AbortController,i=setTimeout(()=>r.abort(),gn);try{let s=await fetch(`${t}/v1/models`,{headers:{Authorization:`Bearer ${e}`},signal:r.signal});if(s.status===401||s.status===403)throw new Error(`API key invalid (HTTP ${s.status}). Re-run: avatar ai setup`);if(!s.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${s.status}).`);let c=((await s.json()).data||[]).map(f=>typeof f.id=="string"?f.id:null).filter(f=>f!==null);if(o.success(`Connectivity OK \xB7 ${c.length} models available`),c.length>0){let f=c.slice(0,5).join(", "),A=c.length>5?` ...+${c.length-5} more`:"";o.dim(` Models: ${f}${A}`)}o.info(`Testing chat completion v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/chat/completions`,{method:"POST",headers:{Authorization:`Bearer ${e}`,"Content-Type":"application/json"},body:JSON.stringify({model:n,messages:[{role:"user",content:fn}],max_tokens:_r}),signal:r.signal});if(!l.ok){let f=(await l.text()).slice(0,200);throw new Error(`Chat completion fail (HTTP ${l.status}). ${f}`)}let u=await l.json(),m=typeof u.choices?.[0]?.message?.content=="string"?u.choices[0].message.content:"(empty response)",g=u.usage?.total_tokens??"?";o.success(`Response: "${String(m).trim().slice(0,100)}"`),o.dim(` Tokens used: ${g}`)}catch(s){throw s.name==="AbortError"?new Error(`Timeout ${gn/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):s}finally{clearTimeout(i)}}function Ba(){o.info("Testing Subscription provider qua `claude --print`...");let t=Ka("claude",["--print",fn],{encoding:"utf8",timeout:Tr});if(t.signal==="SIGTERM")throw new Error(`Timeout ${Tr/1e3}s. Check m\u1EA1ng / endpoint.`);if(t.status!==0){let e=(t.stderr||"").toLowerCase();throw e.includes("401")||e.includes("invalid authentication")||e.includes("unauthorized")?new Error("Token Claude Code stale (401). Fix: `claude auth logout && claude auth login`."):new Error(`Test fail (exit ${t.status}). Stderr: ${(t.stderr||"").slice(0,200)}`)}o.success(`Response: "${(t.stdout||"").trim().slice(0,100)}"`)}async function Wa(t,e,n){o.info(`Testing Anthropic Direct provider: ${t} (key: ${K(e)})`);let r=new AbortController,i=setTimeout(()=>r.abort(),gn);try{let s=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":Ir},signal:r.signal});if(s.status===401||s.status===403)throw new Error(`API key invalid (HTTP ${s.status}). Re-run: avatar ai setup`);if(!s.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${s.status}).`);let c=((await s.json()).data||[]).map(g=>typeof g.id=="string"?g.id:null).filter(g=>g!==null);o.success(`Connectivity OK \xB7 ${c.length} models available`),o.info(`Testing message v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/messages`,{method:"POST",headers:{"x-api-key":e,"anthropic-version":Ir,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:_r,messages:[{role:"user",content:fn}]}),signal:r.signal});if(!l.ok){let g=(await l.text()).slice(0,200);throw new Error(`Message endpoint fail (HTTP ${l.status}): ${g}`)}let m=((await l.json()).content||[]).map(g=>typeof g.text=="string"?g.text:"").join("").trim().slice(0,100);o.success(`Response: "${m}"`)}finally{clearTimeout(i)}}async function Or(t){let e=t.env||{},n=typeof e.ANTHROPIC_BASE_URL=="string"?e.ANTHROPIC_BASE_URL:void 0,r=typeof e.ANTHROPIC_AUTH_TOKEN=="string"?e.ANTHROPIC_AUTH_TOKEN:void 0,i=typeof e.ANTHROPIC_API_KEY=="string"?e.ANTHROPIC_API_KEY:void 0,s=typeof t.model=="string"?t.model:"default";return i&&n?(await Wa(n,i,s),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&r?(await Va(n,r,s),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(Ba(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function be(){let t=process.cwd(),e=vt(t);return e||(o.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
29
+ `,{encoding:"utf8",mode:384}),await fe.chmod(n,384),n}async function ln(t,e){let n=await bt(),r=!1,i=!1;if(!n.installed){o.info("\u0110ang setup direnv \u0111\u1EC3 b\u1EA3o v\u1EC7 API key (silent)...");let a=await de();if(a.success)o.dim(` \u2713 direnv installed via ${a.method}`),r=!0,n=await bt();else{o.warn(` ! direnv install fail (${a.message}). Fallback sang .claude/.env (gitignored).`);let c=await ua(t,e);return o.warn(` \u26A0 Key \u0111\xE3 ghi v\xE0o ${c} (chmod 600) nh\u01B0ng KH\xD4NG auto-load \u2014 c\u1EA7n 'source ${c}' m\u1ED7i terminal m\u1EDBi.`),o.dim(` Manual install direnv sau: ${a.manualInstructionsUrl}`),{storageMethod:"claude-env-fallback",storagePath:c,autoLoad:!1,installedDirenvNow:!1,addedShellHook:!1}}}if(!n.shellHookInZshrc){let a=ge();a.modified&&(o.dim(` \u2713 Added direnv hook to ${a.zshrcPath}`),o.dim(" Reload shell: 'source ~/.zshrc' ho\u1EB7c m\u1EDF terminal m\u1EDBi"),i=!0)}let s=await st(t,e);return o.success(`\u2713 Key \u0111\xE3 l\u01B0u (${Object.keys(e).length} vars) v\xE0o ${s.envrcPath}`),s.direnvAllowOk===!1&&o.warn(" direnv allow fail \u2014 ch\u1EA1y 'direnv allow' trong workspace \u0111\u1EC3 activate."),{storageMethod:"envrc",storagePath:s.envrcPath,autoLoad:!0,installedDirenvNow:r,addedShellHook:i}}k();import{promises as pa}from"fs";import{join as ma}from"path";var un=384;function da(t){return ma(t,".claude","settings.json")}async function ga(t){if(!await d(t))return{};try{return await y(t)}catch(e){throw new Error(`Kh\xF4ng parse \u0111\u01B0\u1EE3c ${t} (JSON l\u1ED7i): ${e.message}. Backup file r\u1ED3i x\xF3a \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}function fa(t,e){let{env:n,...r}=t,i={...r,model:e};if(n){let{ANTHROPIC_BASE_URL:s,ANTHROPIC_AUTH_TOKEN:a,ANTHROPIC_API_KEY:c,...l}=n;Object.keys(l).length>0&&(i.env=l)}return i}function Sr(t){if(!t)return t;let{ANTHROPIC_AUTH_TOKEN:e,ANTHROPIC_API_KEY:n,...r}=t;return r}function ha(t,e,n,r,i){let a={...Sr(t.env),ANTHROPIC_BASE_URL:n};return i||(a.ANTHROPIC_AUTH_TOKEN=e),{...t,env:a,model:r,avatarProvider:"llmlite"}}function ka(t,e,n,r,i){let a={...Sr(t.env),ANTHROPIC_BASE_URL:n};return i||(a.ANTHROPIC_API_KEY=e),{...t,env:a,model:r,avatarProvider:"anthropic"}}function wa(t,e){let n=e.env||{},r=typeof e.model=="string"?e.model:void 0;return{...t,env:{...t.env||{},...n},...r?{model:r}:{}}}async function at(t,e){let n=da(t),r=await ga(n),i;switch(e.provider){case"subscription":i=fa(r,e.model);break;case"llmlite":i=ha(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"anthropic":i=ka(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"use-global":i=wa(r,e.sourceSettings);break}await $(n,i,un);try{await pa.chmod(n,un)}catch{}return{path:n,mode:un}}var j="sonnet";async function he(t){try{o.info("Setup AI provider cho workspace...");let e=Gt();if(e.installed)o.success(`Claude Code \u0111\xE3 c\xF3${e.version?` v${e.version}`:""}`);else if(o.info("Ch\u01B0a c\xF3 Claude Code \u2014 s\u1EBD t\u1EF1 c\xE0i qua npm."),cr(),ae(),e=Gt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=sn();switch(await ur(n)){case"subscription":{let i=nn();if(i.state!=="authenticated"&&(rn(),i=nn()),i.state==="authenticated"&&i.subscriptionType)return await at(t.workspacePath,{provider:"subscription",model:j}),await v("ai_setup",`provider=subscription,result=ok,plan=${i.subscriptionType},probe=skipped`),o.success(`AI ready \xB7 Subscription (${i.subscriptionType}) \xB7 model=${j}`),{ok:!0,provider:"subscription",model:j};o.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let s=on();if(!s.ok&&s.reason==="auth-expired"&&(o.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),rn(),s=on()),!s.ok&&(s.reason==="timeout"||s.reason==="unknown"))return o.warn(`Probe verify ${s.reason} \u2014 accept trust auth status. Ti\u1EBFp t\u1EE5c.`),s.detail?.trim()&&o.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),await at(t.workspacePath,{provider:"subscription",model:j}),await v("ai_setup",`provider=subscription,result=ok,probe=${s.reason}-soft-pass`),o.success(`AI ready \xB7 Subscription (probe ${s.reason}, soft-pass) \xB7 model=${j}`),{ok:!0,provider:"subscription",model:j};if(!s.ok){let a=s.reason??"unknown";return await v("ai_setup",`provider=subscription,result=no-quota,reason=${a}`),o.warn(`Subscription verify th\u1EA5t b\u1EA1i (${a}).`),s.detail?.trim()&&o.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),o.warn(` \u2192 ${ir(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await at(t.workspacePath,{provider:"subscription",model:j}),await v("ai_setup","provider=subscription,result=ok"),o.success(`AI ready \xB7 Subscription \xB7 model=${j}`),{ok:!0,provider:"subscription",model:j}}case"llmlite":{let i=await gr(),s=await ln(t.workspacePath,{ANTHROPIC_AUTH_TOKEN:i.apiKey});return await at(t.workspacePath,{provider:"llmlite",apiKey:i.apiKey,baseUrl:i.baseUrl,model:i.model,skipApiKey:s.storageMethod==="envrc"}),await v("ai_setup",`provider=llmlite,result=ok,model=${i.model},base=${i.baseUrl},storage=${s.storageMethod}`),o.success(`AI ready \xB7 LLMLite (NAL) \xB7 model=${i.model} \xB7 ${i.baseUrl}`),{ok:!0,provider:"llmlite",model:i.model}}case"anthropic":{let i=await mr(),s=await ln(t.workspacePath,{ANTHROPIC_API_KEY:i.apiKey});return await at(t.workspacePath,{provider:"anthropic",apiKey:i.apiKey,baseUrl:i.baseUrl,model:i.model,skipApiKey:s.storageMethod==="envrc"}),await v("ai_setup",`provider=anthropic,result=ok,model=${i.model},storage=${s.storageMethod}`),o.success(`AI ready \xB7 Anthropic Direct \xB7 model=${i.model} \xB7 ${i.baseUrl}`),{ok:!0,provider:"anthropic",model:i.model}}case"use-global":{if(!n.rawSettings)throw new Error("use-global ch\u1ECDn nh\u01B0ng kh\xF4ng \u0111\u1ECDc \u0111\u01B0\u1EE3c global settings.");return await at(t.workspacePath,{provider:"use-global",sourceSettings:n.rawSettings}),await v("ai_setup","provider=use-global,result=ok"),o.success(`AI ready \xB7 Copy t\u1EEB global config (${n.baseUrl??"subscription"})`),{ok:!0,provider:"use-global",model:n.model}}}}catch(e){let n=e instanceof Error?e.message:String(e);return o.warn(`AI setup th\u1EA5t b\u1EA1i: ${n}`),o.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup AI sau qua: avatar ai setup"),await v("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}import{spawnSync as ya}from"child_process";var pn=1e4,Ar=3e4,Pr=5,mn="say ok",Cr="2023-06-01";async function va(t,e,n){o.info(`Testing LLMLite provider: ${t} (key: ${F(e)})`);let r=new AbortController,i=setTimeout(()=>r.abort(),pn);try{let s=await fetch(`${t}/v1/models`,{headers:{Authorization:`Bearer ${e}`},signal:r.signal});if(s.status===401||s.status===403)throw new Error(`API key invalid (HTTP ${s.status}). Re-run: avatar ai setup`);if(!s.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${s.status}).`);let c=((await s.json()).data||[]).map(f=>typeof f.id=="string"?f.id:null).filter(f=>f!==null);if(o.success(`Connectivity OK \xB7 ${c.length} models available`),c.length>0){let f=c.slice(0,5).join(", "),S=c.length>5?` ...+${c.length-5} more`:"";o.dim(` Models: ${f}${S}`)}o.info(`Testing chat completion v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/chat/completions`,{method:"POST",headers:{Authorization:`Bearer ${e}`,"Content-Type":"application/json"},body:JSON.stringify({model:n,messages:[{role:"user",content:mn}],max_tokens:Pr}),signal:r.signal});if(!l.ok){let f=(await l.text()).slice(0,200);throw new Error(`Chat completion fail (HTTP ${l.status}). ${f}`)}let u=await l.json(),m=typeof u.choices?.[0]?.message?.content=="string"?u.choices[0].message.content:"(empty response)",g=u.usage?.total_tokens??"?";o.success(`Response: "${String(m).trim().slice(0,100)}"`),o.dim(` Tokens used: ${g}`)}catch(s){throw s.name==="AbortError"?new Error(`Timeout ${pn/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):s}finally{clearTimeout(i)}}function ba(){o.info("Testing Subscription provider qua `claude --print`...");let t=ya("claude",["--print",mn],{encoding:"utf8",timeout:Ar});if(t.signal==="SIGTERM")throw new Error(`Timeout ${Ar/1e3}s. Check m\u1EA1ng / endpoint.`);if(t.status!==0){let e=(t.stderr||"").toLowerCase();throw e.includes("401")||e.includes("invalid authentication")||e.includes("unauthorized")?new Error("Token Claude Code stale (401). Fix: `claude auth logout && claude auth login`."):new Error(`Test fail (exit ${t.status}). Stderr: ${(t.stderr||"").slice(0,200)}`)}o.success(`Response: "${(t.stdout||"").trim().slice(0,100)}"`)}async function xa(t,e,n){o.info(`Testing Anthropic Direct provider: ${t} (key: ${F(e)})`);let r=new AbortController,i=setTimeout(()=>r.abort(),pn);try{let s=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":Cr},signal:r.signal});if(s.status===401||s.status===403)throw new Error(`API key invalid (HTTP ${s.status}). Re-run: avatar ai setup`);if(!s.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${s.status}).`);let c=((await s.json()).data||[]).map(g=>typeof g.id=="string"?g.id:null).filter(g=>g!==null);o.success(`Connectivity OK \xB7 ${c.length} models available`),o.info(`Testing message v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/messages`,{method:"POST",headers:{"x-api-key":e,"anthropic-version":Cr,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:Pr,messages:[{role:"user",content:mn}]}),signal:r.signal});if(!l.ok){let g=(await l.text()).slice(0,200);throw new Error(`Message endpoint fail (HTTP ${l.status}): ${g}`)}let m=((await l.json()).content||[]).map(g=>typeof g.text=="string"?g.text:"").join("").trim().slice(0,100);o.success(`Response: "${m}"`)}finally{clearTimeout(i)}}async function $r(t){let e=t.env||{},n=typeof e.ANTHROPIC_BASE_URL=="string"?e.ANTHROPIC_BASE_URL:void 0,r=typeof e.ANTHROPIC_AUTH_TOKEN=="string"?e.ANTHROPIC_AUTH_TOKEN:void 0,i=typeof e.ANTHROPIC_API_KEY=="string"?e.ANTHROPIC_API_KEY:void 0,s=typeof t.model=="string"?t.model:"default";return i&&n?(await xa(n,i,s),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&r?(await va(n,r,s),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(ba(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function ke(){let t=process.cwd(),e=vt(t);return e||(o.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
30
30
  Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
31
31
  B\u1EA1n \u0111ang \u1EDF: ${t}
32
- Cd v\xE0o workspace dir (vd /path/to/<project>-workspace) r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&o.dim(`Detected workspace root: ${e}`),e}async function hn(t){let e=Nr(t,".claude","settings.json");if(!await d(e))return{};try{return await v(e)}catch{return{}}}async function Ya(){let t=await be();await ve({workspacePath:t})}async function Ja(){let t=await be(),e=await hn(t),n=e.env||{},r=typeof n.ANTHROPIC_BASE_URL=="string"?n.ANTHROPIC_BASE_URL:void 0,i=typeof n.ANTHROPIC_AUTH_TOKEN=="string"?n.ANTHROPIC_AUTH_TOKEN:void 0,s=typeof n.ANTHROPIC_API_KEY=="string"?n.ANTHROPIC_API_KEY:void 0,a=typeof e.model=="string"?e.model:void 0,c,l;s?(c=r?.includes("api.anthropic.com")||s.startsWith("sk-ant-")?"Anthropic Direct":"Custom (API key)",l=K(s)):r&&i?(c="LLMLite",l=K(i)):i?(c="Custom",l=K(i)):(c="Subscription (default)",l="(kh\xF4ng set \u2014 d\xF9ng subscription auth)"),o.info(`Project: ${t}`),o.info(`Provider: ${c}${r?` (${r})`:""}`),o.info(`Model: ${a??"(default \u2014 Claude Code ch\u1ECDn)"}`),o.info(`Token: ${l}`)}async function Xa(){let t=await be(),e=await hn(t);try{let n=await Or(e);o.success(`\u2713 ${n.message}`)}catch(n){o.error(`Test fail: ${n.message}`),process.exit(1)}}async function Za(t){let e=await be(),n=Nr(e,".claude","settings.json"),r=await hn(e);if(!t.yes&&!await qa({message:"X\xF3a AI config (v\u1EC1 d\xF9ng Claude Code Subscription default)?",default:!1})){o.dim("\u0110\xE3 h\u1EE7y.");return}let{env:i,...s}=r,a={...s};if(i){let{ANTHROPIC_BASE_URL:c,ANTHROPIC_AUTH_TOKEN:l,ANTHROPIC_API_KEY:u,...m}=i;Object.keys(m).length>0&&(a.env=m)}Object.keys(a).length===0?(await za.unlink(n).catch(()=>{}),o.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await $(n,a,384),o.success("\u0110\xE3 reset env block trong .claude/settings.json"))}function Mr(t){let e=t.command("ai").description("Qu\u1EA3n l\xFD AI provider config (M12)");e.command("setup").description("Wizard setup/re-config AI provider cho workspace hi\u1EC7n t\u1EA1i").action(async()=>{await Ya()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await Ja()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await Xa()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await Za(n)})}import{input as rc}from"@inquirer/prompts";import{spawnSync as Qa}from"child_process";import{existsSync as kn}from"fs";import{join as xe}from"path";function tc(t){let e=xe(t,"src",".git"),n=xe(t,".git"),r=xe(t,".claude");if(!kn(n))throw new Error(`Kh\xF4ng ph\u1EA3i workspace root: ${t}
33
- Ch\u1EA1y 'avatar commit' trong workspace dir (c\xF3 .git v\xE0 .claude/).`);if(!kn(r))throw new Error(`Kh\xF4ng th\u1EA5y .claude/ trong ${t}.
34
- Ch\u1EA1y 'avatar commit' trong Avatar workspace, kh\xF4ng ph\u1EA3i project b\xECnh th\u01B0\u1EDDng.`);if(!kn(e))throw new Error(`Kh\xF4ng th\u1EA5y src/.git trong ${t}.
35
- Workspace thi\u1EBFu submodule src/. Ch\u1EA1y 'avatar init' l\u1EA1i?`)}function L(t,e){let n=Qa("git",["-C",t,...e],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(n.status!==0){let r=(n.stderr||"").trim();throw new Error(`git ${e.join(" ")} (in ${t}) failed:
36
- ${r}`)}return(n.stdout||"").trim()}function jr(t){return L(t,["status","--porcelain"]).length>0}async function ec(t,e){let n=xe(t,"src");if(!jr(n))return o.dim("src/: nothing to commit (clean)"),{};o.info("Committing src/ ..."),L(n,["add","."]),L(n,["commit","-m",e.message]);let r=L(n,["rev-parse","HEAD"]);o.success(`src/ committed: ${r.slice(0,7)}`);let i=!1;return e.push&&(o.info("Pushing src/ ..."),L(n,["push"]),o.success("src/ pushed"),i=!0),{sha:r,pushed:i}}async function nc(t,e){if(!jr(t))return o.dim("workspace: nothing to commit (clean)"),{};o.info("Committing workspace root ..."),L(t,["add","."]),L(t,["commit","-m",e.message]);let n=L(t,["rev-parse","HEAD"]);o.success(`workspace committed: ${n.slice(0,7)}`);let r=!1;return e.push&&(o.info("Pushing workspace ..."),L(t,["push"]),o.success("workspace pushed"),r=!0),{sha:n,pushed:r}}async function Gr(t){tc(t.workspaceRoot);let e={target:t.target,skipped:[]};if(t.target==="src"||t.target==="all"){let n=await ec(t.workspaceRoot,t.options);e.srcCommitSha=n.sha,e.srcPushed=n.pushed,n.sha||e.skipped?.push("src")}if(t.target==="workspace"||t.target==="all"){let n=await nc(t.workspaceRoot,t.options);e.workspaceCommitSha=n.sha,e.workspacePushed=n.pushed,n.sha||e.skipped?.push("workspace")}return e}function Lr(t){t.command("commit").description("Commit code kh\xE1ch trong src/ (Avatar state do admin sync) (M07)").command("src").description("Commit src/ \u2192 push l\xEAn client repo remote").option("-m, --message <msg>","Commit message").option("--push","Auto push sau commit (default: ch\u1EC9 commit)").action(async n=>{await ic(n)})}async function ic(t){try{let e=t.message??await rc({message:"Commit message:",validate:r=>r.trim().length>0?!0:"Message b\u1EAFt bu\u1ED9c"}),n=await Gr({workspaceRoot:process.cwd(),target:"src",options:{message:e,push:t.push}});n.srcCommitSha&&o.success(`src/: ${n.srcCommitSha.slice(0,7)}${n.srcPushed?" (pushed)":""}`),n.skipped&&n.skipped.length>0&&o.dim(`Skipped (nothing to commit): ${n.skipped.join(", ")}`)}catch(e){o.error(e instanceof Error?e.message:String(e)),process.exit(1)}}import{spawnSync as xn}from"child_process";import{promises as An}from"fs";import{join as ct}from"path";import Tc from"boxen";import{join as kc}from"path";k();import{promises as uc}from"fs";import{join as Kr}from"path";k();import{promises as oc}from"fs";import Ur,{join as wn}from"path";async function sc(t,e){let r=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!r?.[2])return!0;let i=r[2];if(i.startsWith("/"))return o.warn(`Pack hook reject: absolute path "${i}" \u2014 ch\u1EC9 accept relative path t\u1EDBi workspace.`),!1;let s=wn(t,i),a=Ur.resolve(t),c=Ur.resolve(s);return!c.startsWith(`${a}/`)&&c!==a?(o.warn(`Pack hook reject: path "${i}" resolve ra ngo\xE0i workspace (path traversal).`),!1):await d(s)}function yn(t){let e=new Date,n=`${e.getFullYear().toString().slice(-2)+String(e.getMonth()+1).padStart(2,"0")+String(e.getDate()).padStart(2,"0")}-${String(e.getHours()).padStart(2,"0")}${String(e.getMinutes()).padStart(2,"0")}`;return`${t}.backup-${n}`}function Ft(t,e){let n=new Set,r=[];for(let i of[...t,...e]){let s=typeof i=="string"?i:JSON.stringify(i);n.has(s)||(n.add(s),r.push(i))}return r}function ac(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function cc(t){if(typeof t!="object"||t===null)return[];let e=t,n=Array.isArray(e.hooks)?e.hooks:[],r=[];for(let i of n)if(typeof i=="object"&&i!==null){let s=i.command;typeof s=="string"&&r.push(s)}return r}function Hr(t){let e=t.map((s,a)=>{let c=cc(s),l=c.some(u=>u.includes("${CLAUDE_PROJECT_DIR}"));return{index:a,entry:s,commands:c,hasVar:l}}),n=new Map;for(let s of e){let a=s.commands.map(ac).sort().join("|");if(!a)continue;let c=n.get(a)??[];c.push(s),n.set(a,c)}let r=new Set;for(let[,s]of n)if(!(s.length<2||!s.some(c=>c.hasVar)))for(let c of s)c.hasVar||r.add(c.index);return r.size===0?{entries:t,droppedCount:0}:{entries:t.filter((s,a)=>!r.has(a)),droppedCount:r.size}}function lc(t,e){let n=[],r={...e},i=0;for(let s of Object.keys(r)){let a=r[s]||[],{entries:c,droppedCount:l}=Hr(a);l>0&&(r[s]=c,i+=l,n.includes(s)||n.push(s))}for(let[s,a]of Object.entries(t)){let c=r[s]||[],l=Ft(c,a),{entries:u,droppedCount:m}=Hr(l);m>0&&(i+=m),u.length!==c.length&&(n.includes(s)||n.push(s)),r[s]=u}return{merged:r,touchedEvents:n,migratedCount:i}}async function Ae(t){let e=wn(t,".claude","pack","templates","settings.json.tpl"),n=wn(t,".claude","settings.json");if(!await d(e))return{action:"no-pack-template",changes:[]};let r;try{let u=await D(e);r=JSON.parse(u)}catch(u){throw new Error(`Pack settings template kh\xF4ng parse \u0111\u01B0\u1EE3c JSON: ${u.message}. Path: ${e}`)}let i={},s=!1;if(await d(n)){s=!0;try{i=await v(n)}catch(u){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${u.message}. Manual fix tr\u01B0\u1EDBc khi sync.`)}}let a=[],c={...i};if(r.statusLine&&!i.statusLine&&(await sc(t,r.statusLine.command)?(c.statusLine=r.statusLine,a.push("statusLine added")):a.push(`statusLine SKIPPED (file ref '${r.statusLine.command}' kh\xF4ng t\u1ED3n t\u1EA1i)`)),typeof r.includeCoAuthoredBy=="boolean"&&typeof i.includeCoAuthoredBy!="boolean"&&(c.includeCoAuthoredBy=r.includeCoAuthoredBy,a.push("includeCoAuthoredBy added")),r.model&&!i.model&&(c.model=r.model,a.push("model added")),r.env||i.env){let u={...i.env||{}},m=!1;for(let[g,f]of Object.entries(u))typeof f=="string"&&f.includes("llm.nal.vn")&&(u[g]=f.replace(/llm\.nal\.vn/g,"ai.nal.vn"),m=!0);if(m&&a.push("rewrote legacy llm.nal.vn \u2192 ai.nal.vn in env vars"),r.env)for(let[g,f]of Object.entries(r.env))g in u||(u[g]=f,m=!0);m&&(c.env=u,a.some(g=>g.includes("env vars"))||a.push("env vars added from pack"))}if(r.permissions){let u=i.permissions?.allow||[],m=i.permissions?.deny||[],g=r.permissions.allow||[],f=r.permissions.deny||[],A=Ft(u,g),y=Ft(m,f);(A.length!==u.length||y.length!==m.length)&&(c.permissions={allow:A,deny:y},a.push(`permissions union (+${A.length-u.length} allow, +${y.length-m.length} deny)`))}if(r.hooks){let u=i.hooks||{},{merged:m,touchedEvents:g,migratedCount:f}=lc(r.hooks,u);(g.length>0||f>0)&&(c.hooks=m,g.length>0&&a.push(`hooks added for events: ${g.join(", ")}`),f>0&&a.push(`migrated ${f} stale relative-path hook entries to use \${CLAUDE_PROJECT_DIR} version (fixes hook failure when shell cwd changes)`))}if(a.length===0)return{action:"no-change",changes:[]};let l;return s&&(l=yn(n),await oc.copyFile(n,l)),await $(n,c),{action:"merged",backupPath:l,changes:a}}function pc(t,e){return Kr(t,".claude","pack","features",e,"feature.json")}async function Kt(t,e){let n=pc(t,e);return await d(n)?await v(n):null}var mc=[".claude","settings.json"];function Vr(t){return Kr(t,...mc)}function Se(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function Br(t){let e=Vr(t);if(!await d(e))return{settings:{},existed:!1};try{return{settings:await v(e),existed:!0}}catch(n){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${n.message}. Manual fix tr\u01B0\u1EDBc khi enable/disable feature.`)}}async function Wr(t,e,n){let r=Vr(t),i;return n&&(i=yn(r),await uc.copyFile(r,i)),await $(r,e),i}function Dr(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function Fr(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function dc(t,e){let n=new Set;for(let s of e)Fr(s)&&n.add(Dr(s));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(s=>Fr(s)?!0:n.has(Dr(s))?(r++,!1):!0),droppedCount:r}}async function Ce(t,e){let{settings:n,existed:r}=await Br(t),i={...n},s=[],a=e.settings.hooks??{};if(Object.keys(a).length>0){let u=n.hooks??{},m={...u},g=[],f=0;for(let[A,y]of Object.entries(a)){let P=u[A]??[],{migratedUser:O,droppedCount:S}=dc(P,y);S>0&&(f+=S);let N=new Set(O.map(Se)),_=y.filter(ht=>!N.has(Se(ht)));(_.length>0||S>0)&&(m[A]=[...O,..._],g.push(A))}g.length>0&&(i.hooks=m,s.push(`hooks added: ${g.join(", ")}`),f>0&&s.push(`migrated ${f} stale relative-path entries \u2192 \\{CLAUDE_PROJECT_DIR\\} version`))}let c=e.settings.permissions?.deny??[];if(c.length>0){let u=n.permissions?.deny??[],m=Ft(u,c);m.length!==u.length&&(i.permissions={...n.permissions,deny:m},s.push(`deny +${m.length-u.length}`))}return s.length===0?{action:"no-change",changes:[]}:{action:"enabled",backupPath:await Wr(t,i,r),changes:s}}async function zr(t,e){let{settings:n,existed:r}=await Br(t);if(!r)return{action:"no-change",changes:[]};let{hooks:i,permissions:s,...a}=n,c={...a},l=[],u=e.settings.hooks??{},m=n.hooks??{},g={},f=[];for(let[S,N]of Object.entries(m)){let _=u[S];if(!_){g[S]=N;continue}let ht=new Set(_.map(Se)),kt=N.filter(oe=>!ht.has(Se(oe)));kt.length!==N.length&&f.push(S),kt.length>0&&(g[S]=kt)}Object.keys(g).length>0&&(c.hooks=g),f.length>0&&l.push(`hooks removed: ${f.join(", ")}`);let A=new Set(e.settings.permissions?.deny??[]),y=n.permissions?.deny??[],P=A.size>0?y.filter(S=>!A.has(S)):y;if(n.permissions){let{deny:S,...N}=n.permissions,_={...N,...P.length>0?{deny:P}:{}};Object.keys(_).length>0&&(c.permissions=_)}return P.length!==y.length&&l.push(`deny -${y.length-P.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await Wr(t,c,r),changes:l}}k();import{join as gc}from"path";var fc=".claude/avatar-features.json";function qr(){return{features:{}}}function Yr(t){return gc(t,fc)}async function At(t){let e=Yr(t);if(!await d(e))return qr();try{return{features:(await v(e)).features??{}}}catch{return qr()}}async function hc(t,e){await $(Yr(t),e)}async function Pe(t,e,n){let r=await At(t);return r.features[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await hc(t,r),r}async function St(t){let e=await At(t);return Object.entries(e.features).filter(([,n])=>n.enabled).map(([n])=>n)}k();function wc(t,e){let n=e.settings.hooks??{},r=t.hooks??{};for(let[i,s]of Object.entries(n)){let a=r[i]??[],c=new Set(a.flatMap(l=>(l.hooks??[]).map(u=>u.command??"")));for(let l of s)for(let u of l.hooks??[])if(u.command&&!c.has(u.command))return!1}return!0}async function Jr(t){let e=await St(t);if(e.length===0)return[];let n=kc(t,".claude","settings.json"),r={};if(await d(n))try{r=await v(n)}catch{r={}}let i=[];for(let s of e){let a=await Kt(t,s);if(!a){i.push({name:`Feature: ${s}`,status:"warn",detail:"state=enabled nh\u01B0ng pack thi\u1EBFu manifest (version skew). Ch\u1EA1y 'avatar sync'.",fixable:!1});continue}wc(r,a)?i.push({name:`Feature: ${s}`,status:"ok",detail:`enabled, hook active (v${a.version})`,fixable:!1}):i.push({name:`Feature: ${s}`,status:"fail",detail:"state=enabled nh\u01B0ng hook thi\u1EBFu trong settings.json \u2014 fixable (re-apply)",fixable:!0,fix:async()=>{await Ce(t,a)}})}return i}k();k();import{join as vn}from"path";import{simpleGit as yc}from"simple-git";function b(t=process.cwd()){return yc({baseDir:t,binary:"git"})}async function Ct(t=process.cwd()){return await d(vn(t,".git"))}async function Xr(t,e,n=process.cwd()){await b(n).subModule(["add",t,e])}async function Vt(t,e,n=process.cwd()){let r=vn(n,t);await b(r).fetch(["--tags"]),await b(r).checkout(e)}async function $e(t,e,n=process.cwd()){let r=vn(n,t);await b(r).fetch(["origin"]),await b(r).checkout(["-B",e,`origin/${e}`])}async function B(t=process.cwd()){return(await b(t).tags()).all}async function Ee(t=process.cwd()){try{return(await b(t).raw(["describe","--tags","--exact-match","HEAD"])).trim()||null}catch{return null}}async function Pt(t=process.cwd()){return(await b(t).revparse(["HEAD"])).trim()}k();import{promises as ni}from"fs";import{join as W}from"path";k();import{existsSync as bc}from"fs";import{dirname as Qr,join as Bt}from"path";import{fileURLToPath as xc}from"url";var vc=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g;function Zr(t,e){return t.replace(vc,(n,r)=>{let i=e[r];return i===void 0?n:String(i)})}var Ac=Qr(xc(import.meta.url)),ti=Pc(Ac),Sc=Bt(ti,"src","templates"),Cc=Bt(ti,"src","hooks");function Pc(t){let e=t;for(;;){if(bc(Bt(e,"package.json")))return e;let n=Qr(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function $c(t){return await D(Bt(Sc,`${t}.tpl`))}async function Re(t,e){let n=await $c(t);return Zr(n,e)}async function ei(t){return await D(Bt(Cc,`${t}.sh.tpl`))}async function Ec(t){if(!await d(t))return null;let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`,r=n,i=1;for(;await d(r);)if(r=`${n}-${i}`,i++,i>5)throw new Error(`Could not find free backup name for ${t}`);return await ni.rename(t,r),r}async function ri(t,e,n){let r=await Ec(t);return await yt(t,e,n),r}var Rc=["state","_pending","_backup"];async function ii(t){let e=W(t,".claude");await E(e);for(let n of Rc){let r=W(e,n);await E(r),await yt(W(r,".gitkeep"),"")}}async function oi(t,e){return[]}async function bn(t,e){let n=await Re("CLAUDE.md",e);return await ri(W(t,"CLAUDE.md"),n)}async function si(t,e){let n=await Re("settings.json",e);return await ri(W(t,".claude","settings.json"),n)}async function ai(t){let e=W(t,".gitignore"),n=await Re("gitignore",{}),r="# Avatar \u2014 git-ignored entries injected on `avatar init`",i="";if(await d(e)&&(i=await ni.readFile(e,"utf8"),i.includes(r)))return;let s=i.endsWith(`
32
+ Cd v\xE0o workspace dir (vd /path/to/<project>-workspace) r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&o.dim(`Detected workspace root: ${e}`),e}async function dn(t){let e=Er(t,".claude","settings.json");if(!await d(e))return{};try{return await y(e)}catch{return{}}}async function Ca(){let t=await ke();await he({workspacePath:t})}async function Pa(){let t=await ke(),e=await dn(t),n=e.env||{},r=typeof n.ANTHROPIC_BASE_URL=="string"?n.ANTHROPIC_BASE_URL:void 0,i=typeof n.ANTHROPIC_AUTH_TOKEN=="string"?n.ANTHROPIC_AUTH_TOKEN:void 0,s=typeof n.ANTHROPIC_API_KEY=="string"?n.ANTHROPIC_API_KEY:void 0,a=typeof e.model=="string"?e.model:void 0,c,l;s?(c=r?.includes("api.anthropic.com")||s.startsWith("sk-ant-")?"Anthropic Direct":"Custom (API key)",l=F(s)):r&&i?(c="LLMLite",l=F(i)):i?(c="Custom",l=F(i)):(c="Subscription (default)",l="(kh\xF4ng set \u2014 d\xF9ng subscription auth)"),o.info(`Project: ${t}`),o.info(`Provider: ${c}${r?` (${r})`:""}`),o.info(`Model: ${a??"(default \u2014 Claude Code ch\u1ECDn)"}`),o.info(`Token: ${l}`)}async function $a(){let t=await ke(),e=await dn(t);try{let n=await $r(e);o.success(`\u2713 ${n.message}`)}catch(n){o.error(`Test fail: ${n.message}`),process.exit(1)}}async function Ea(t){let e=await ke(),n=Er(e,".claude","settings.json"),r=await dn(e);if(!t.yes&&!await Aa({message:"X\xF3a AI config (v\u1EC1 d\xF9ng Claude Code Subscription default)?",default:!1})){o.dim("\u0110\xE3 h\u1EE7y.");return}let{env:i,...s}=r,a={...s};if(i){let{ANTHROPIC_BASE_URL:c,ANTHROPIC_AUTH_TOKEN:l,ANTHROPIC_API_KEY:u,...m}=i;Object.keys(m).length>0&&(a.env=m)}Object.keys(a).length===0?(await Sa.unlink(n).catch(()=>{}),o.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await $(n,a,384),o.success("\u0110\xE3 reset env block trong .claude/settings.json"))}function Rr(t){let e=t.command("ai").description("Qu\u1EA3n l\xFD AI provider config (M12)");e.command("setup").description("Wizard setup/re-config AI provider cho workspace hi\u1EC7n t\u1EA1i").action(async()=>{await Ca()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await Pa()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await $a()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await Ea(n)})}import{input as Oa}from"@inquirer/prompts";import{spawnSync as Ra}from"child_process";import{existsSync as gn}from"fs";import{join as we}from"path";function Ta(t){let e=we(t,"src",".git"),n=we(t,".git"),r=we(t,".claude");if(!gn(n))throw new Error(`Kh\xF4ng ph\u1EA3i workspace root: ${t}
33
+ Ch\u1EA1y 'avatar commit' trong workspace dir (c\xF3 .git v\xE0 .claude/).`);if(!gn(r))throw new Error(`Kh\xF4ng th\u1EA5y .claude/ trong ${t}.
34
+ Ch\u1EA1y 'avatar commit' trong Avatar workspace, kh\xF4ng ph\u1EA3i project b\xECnh th\u01B0\u1EDDng.`);if(!gn(e))throw new Error(`Kh\xF4ng th\u1EA5y src/.git trong ${t}.
35
+ Workspace thi\u1EBFu submodule src/. Ch\u1EA1y 'avatar init' l\u1EA1i?`)}function L(t,e){let n=Ra("git",["-C",t,...e],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(n.status!==0){let r=(n.stderr||"").trim();throw new Error(`git ${e.join(" ")} (in ${t}) failed:
36
+ ${r}`)}return(n.stdout||"").trim()}function Tr(t){return L(t,["status","--porcelain"]).length>0}async function _a(t,e){let n=we(t,"src");if(!Tr(n))return o.dim("src/: nothing to commit (clean)"),{};o.info("Committing src/ ..."),L(n,["add","."]),L(n,["commit","-m",e.message]);let r=L(n,["rev-parse","HEAD"]);o.success(`src/ committed: ${r.slice(0,7)}`);let i=!1;return e.push&&(o.info("Pushing src/ ..."),L(n,["push"]),o.success("src/ pushed"),i=!0),{sha:r,pushed:i}}async function Ia(t,e){if(!Tr(t))return o.dim("workspace: nothing to commit (clean)"),{};o.info("Committing workspace root ..."),L(t,["add","."]),L(t,["commit","-m",e.message]);let n=L(t,["rev-parse","HEAD"]);o.success(`workspace committed: ${n.slice(0,7)}`);let r=!1;return e.push&&(o.info("Pushing workspace ..."),L(t,["push"]),o.success("workspace pushed"),r=!0),{sha:n,pushed:r}}async function _r(t){Ta(t.workspaceRoot);let e={target:t.target,skipped:[]};if(t.target==="src"||t.target==="all"){let n=await _a(t.workspaceRoot,t.options);e.srcCommitSha=n.sha,e.srcPushed=n.pushed,n.sha||e.skipped?.push("src")}if(t.target==="workspace"||t.target==="all"){let n=await Ia(t.workspaceRoot,t.options);e.workspaceCommitSha=n.sha,e.workspacePushed=n.pushed,n.sha||e.skipped?.push("workspace")}return e}function Ir(t){t.command("commit").description("Commit code kh\xE1ch trong src/ (Avatar state do admin sync) (M07)").command("src").description("Commit src/ \u2192 push l\xEAn client repo remote").option("-m, --message <msg>","Commit message").option("--push","Auto push sau commit (default: ch\u1EC9 commit)").action(async n=>{await Na(n)})}async function Na(t){try{let e=t.message??await Oa({message:"Commit message:",validate:r=>r.trim().length>0?!0:"Message b\u1EAFt bu\u1ED9c"}),n=await _r({workspaceRoot:process.cwd(),target:"src",options:{message:e,push:t.push}});n.srcCommitSha&&o.success(`src/: ${n.srcCommitSha.slice(0,7)}${n.srcPushed?" (pushed)":""}`),n.skipped&&n.skipped.length>0&&o.dim(`Skipped (nothing to commit): ${n.skipped.join(", ")}`)}catch(e){o.error(e instanceof Error?e.message:String(e)),process.exit(1)}}import{spawnSync as yn}from"child_process";import{promises as vn}from"fs";import{join as ct}from"path";import sc from"boxen";import{join as za}from"path";k();import{promises as Ua}from"fs";import{join as Lr}from"path";k();import{promises as Ma}from"fs";import Or,{join as fn}from"path";async function ja(t,e){let r=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!r?.[2])return!0;let i=r[2];if(i.startsWith("/"))return o.warn(`Pack hook reject: absolute path "${i}" \u2014 ch\u1EC9 accept relative path t\u1EDBi workspace.`),!1;let s=fn(t,i),a=Or.resolve(t),c=Or.resolve(s);return!c.startsWith(`${a}/`)&&c!==a?(o.warn(`Pack hook reject: path "${i}" resolve ra ngo\xE0i workspace (path traversal).`),!1):await d(s)}function hn(t){let e=new Date,n=`${e.getFullYear().toString().slice(-2)+String(e.getMonth()+1).padStart(2,"0")+String(e.getDate()).padStart(2,"0")}-${String(e.getHours()).padStart(2,"0")}${String(e.getMinutes()).padStart(2,"0")}`;return`${t}.backup-${n}`}function Dt(t,e){let n=new Set,r=[];for(let i of[...t,...e]){let s=typeof i=="string"?i:JSON.stringify(i);n.has(s)||(n.add(s),r.push(i))}return r}function La(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function Ha(t){if(typeof t!="object"||t===null)return[];let e=t,n=Array.isArray(e.hooks)?e.hooks:[],r=[];for(let i of n)if(typeof i=="object"&&i!==null){let s=i.command;typeof s=="string"&&r.push(s)}return r}function Nr(t){let e=t.map((s,a)=>{let c=Ha(s),l=c.some(u=>u.includes("${CLAUDE_PROJECT_DIR}"));return{index:a,entry:s,commands:c,hasVar:l}}),n=new Map;for(let s of e){let a=s.commands.map(La).sort().join("|");if(!a)continue;let c=n.get(a)??[];c.push(s),n.set(a,c)}let r=new Set;for(let[,s]of n)if(!(s.length<2||!s.some(c=>c.hasVar)))for(let c of s)c.hasVar||r.add(c.index);return r.size===0?{entries:t,droppedCount:0}:{entries:t.filter((s,a)=>!r.has(a)),droppedCount:r.size}}function Ga(t,e){let n=[],r={...e},i=0;for(let s of Object.keys(r)){let a=r[s]||[],{entries:c,droppedCount:l}=Nr(a);l>0&&(r[s]=c,i+=l,n.includes(s)||n.push(s))}for(let[s,a]of Object.entries(t)){let c=r[s]||[],l=Dt(c,a),{entries:u,droppedCount:m}=Nr(l);m>0&&(i+=m),u.length!==c.length&&(n.includes(s)||n.push(s)),r[s]=u}return{merged:r,touchedEvents:n,migratedCount:i}}async function ye(t){let e=fn(t,".claude","pack","templates","settings.json.tpl"),n=fn(t,".claude","settings.json");if(!await d(e))return{action:"no-pack-template",changes:[]};let r;try{let u=await U(e);r=JSON.parse(u)}catch(u){throw new Error(`Pack settings template kh\xF4ng parse \u0111\u01B0\u1EE3c JSON: ${u.message}. Path: ${e}`)}let i={},s=!1;if(await d(n)){s=!0;try{i=await y(n)}catch(u){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${u.message}. Manual fix tr\u01B0\u1EDBc khi sync.`)}}let a=[],c={...i};if(r.statusLine&&!i.statusLine&&(await ja(t,r.statusLine.command)?(c.statusLine=r.statusLine,a.push("statusLine added")):a.push(`statusLine SKIPPED (file ref '${r.statusLine.command}' kh\xF4ng t\u1ED3n t\u1EA1i)`)),typeof r.includeCoAuthoredBy=="boolean"&&typeof i.includeCoAuthoredBy!="boolean"&&(c.includeCoAuthoredBy=r.includeCoAuthoredBy,a.push("includeCoAuthoredBy added")),r.model&&!i.model&&(c.model=r.model,a.push("model added")),r.env||i.env){let u={...i.env||{}},m=!1;for(let[g,f]of Object.entries(u))typeof f=="string"&&f.includes("llm.nal.vn")&&(u[g]=f.replace(/llm\.nal\.vn/g,"ai.nal.vn"),m=!0);if(m&&a.push("rewrote legacy llm.nal.vn \u2192 ai.nal.vn in env vars"),r.env)for(let[g,f]of Object.entries(r.env))g in u||(u[g]=f,m=!0);m&&(c.env=u,a.some(g=>g.includes("env vars"))||a.push("env vars added from pack"))}if(r.permissions){let u=i.permissions?.allow||[],m=i.permissions?.deny||[],g=r.permissions.allow||[],f=r.permissions.deny||[],S=Dt(u,g),w=Dt(m,f);(S.length!==u.length||w.length!==m.length)&&(c.permissions={allow:S,deny:w},a.push(`permissions union (+${S.length-u.length} allow, +${w.length-m.length} deny)`))}if(r.hooks){let u=i.hooks||{},{merged:m,touchedEvents:g,migratedCount:f}=Ga(r.hooks,u);(g.length>0||f>0)&&(c.hooks=m,g.length>0&&a.push(`hooks added for events: ${g.join(", ")}`),f>0&&a.push(`migrated ${f} stale relative-path hook entries to use \${CLAUDE_PROJECT_DIR} version (fixes hook failure when shell cwd changes)`))}if(a.length===0)return{action:"no-change",changes:[]};let l;return s&&(l=hn(n),await Ma.copyFile(n,l)),await $(n,c),{action:"merged",backupPath:l,changes:a}}function Da(t,e){return Lr(t,".claude","pack","features",e,"feature.json")}async function Ft(t,e){let n=Da(t,e);return await d(n)?await y(n):null}var Fa=[".claude","settings.json"];function Hr(t){return Lr(t,...Fa)}function ve(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function Gr(t){let e=Hr(t);if(!await d(e))return{settings:{},existed:!1};try{return{settings:await y(e),existed:!0}}catch(n){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${n.message}. Manual fix tr\u01B0\u1EDBc khi enable/disable feature.`)}}async function Ur(t,e,n){let r=Hr(t),i;return n&&(i=hn(r),await Ua.copyFile(r,i)),await $(r,e),i}function Mr(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function jr(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function Va(t,e){let n=new Set;for(let s of e)jr(s)&&n.add(Mr(s));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(s=>jr(s)?!0:n.has(Mr(s))?(r++,!1):!0),droppedCount:r}}async function be(t,e){let{settings:n,existed:r}=await Gr(t),i={...n},s=[],a=e.settings.hooks??{};if(Object.keys(a).length>0){let u=n.hooks??{},m={...u},g=[],f=0;for(let[S,w]of Object.entries(a)){let P=u[S]??[],{migratedUser:O,droppedCount:A}=Va(P,w);A>0&&(f+=A);let N=new Set(O.map(ve)),I=w.filter(ht=>!N.has(ve(ht)));(I.length>0||A>0)&&(m[S]=[...O,...I],g.push(S))}g.length>0&&(i.hooks=m,s.push(`hooks added: ${g.join(", ")}`),f>0&&s.push(`migrated ${f} stale relative-path entries \u2192 \\{CLAUDE_PROJECT_DIR\\} version`))}let c=e.settings.permissions?.deny??[];if(c.length>0){let u=n.permissions?.deny??[],m=Dt(u,c);m.length!==u.length&&(i.permissions={...n.permissions,deny:m},s.push(`deny +${m.length-u.length}`))}return s.length===0?{action:"no-change",changes:[]}:{action:"enabled",backupPath:await Ur(t,i,r),changes:s}}async function Dr(t,e){let{settings:n,existed:r}=await Gr(t);if(!r)return{action:"no-change",changes:[]};let{hooks:i,permissions:s,...a}=n,c={...a},l=[],u=e.settings.hooks??{},m=n.hooks??{},g={},f=[];for(let[A,N]of Object.entries(m)){let I=u[A];if(!I){g[A]=N;continue}let ht=new Set(I.map(ve)),kt=N.filter(ie=>!ht.has(ve(ie)));kt.length!==N.length&&f.push(A),kt.length>0&&(g[A]=kt)}Object.keys(g).length>0&&(c.hooks=g),f.length>0&&l.push(`hooks removed: ${f.join(", ")}`);let S=new Set(e.settings.permissions?.deny??[]),w=n.permissions?.deny??[],P=S.size>0?w.filter(A=>!S.has(A)):w;if(n.permissions){let{deny:A,...N}=n.permissions,I={...N,...P.length>0?{deny:P}:{}};Object.keys(I).length>0&&(c.permissions=I)}return P.length!==w.length&&l.push(`deny -${w.length-P.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await Ur(t,c,r),changes:l}}k();import{join as Ka}from"path";var Ba=".claude/avatar-features.json";function Fr(){return{features:{}}}function Vr(t){return Ka(t,Ba)}async function St(t){let e=Vr(t);if(!await d(e))return Fr();try{return{features:(await y(e)).features??{}}}catch{return Fr()}}async function Wa(t,e){await $(Vr(t),e)}async function xe(t,e,n){let r=await St(t);return r.features[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await Wa(t,r),r}async function At(t){let e=await St(t);return Object.entries(e.features).filter(([,n])=>n.enabled).map(([n])=>n)}k();function qa(t,e){let n=e.settings.hooks??{},r=t.hooks??{};for(let[i,s]of Object.entries(n)){let a=r[i]??[],c=new Set(a.flatMap(l=>(l.hooks??[]).map(u=>u.command??"")));for(let l of s)for(let u of l.hooks??[])if(u.command&&!c.has(u.command))return!1}return!0}async function Kr(t){let e=await At(t);if(e.length===0)return[];let n=za(t,".claude","settings.json"),r={};if(await d(n))try{r=await y(n)}catch{r={}}let i=[];for(let s of e){let a=await Ft(t,s);if(!a){i.push({name:`Feature: ${s}`,status:"warn",detail:"state=enabled nh\u01B0ng pack thi\u1EBFu manifest (version skew). Ch\u1EA1y 'avatar sync'.",fixable:!1});continue}qa(r,a)?i.push({name:`Feature: ${s}`,status:"ok",detail:`enabled, hook active (v${a.version})`,fixable:!1}):i.push({name:`Feature: ${s}`,status:"fail",detail:"state=enabled nh\u01B0ng hook thi\u1EBFu trong settings.json \u2014 fixable (re-apply)",fixable:!0,fix:async()=>{await be(t,a)}})}return i}k();k();import{join as kn}from"path";import{simpleGit as Ja}from"simple-git";function b(t=process.cwd()){return Ja({baseDir:t,binary:"git"})}async function Ct(t=process.cwd()){return await d(kn(t,".git"))}async function Br(t,e,n=process.cwd()){await b(n).subModule(["add",t,e])}async function Vt(t,e,n=process.cwd()){let r=kn(n,t);await b(r).fetch(["--tags"]),await b(r).checkout(e)}async function Se(t,e,n=process.cwd()){let r=kn(n,t);await b(r).fetch(["origin"]),await b(r).checkout(["-B",e,`origin/${e}`])}async function K(t=process.cwd()){return(await b(t).tags()).all}async function Ae(t=process.cwd()){try{return(await b(t).raw(["describe","--tags","--exact-match","HEAD"])).trim()||null}catch{return null}}async function Pt(t=process.cwd()){return(await b(t).revparse(["HEAD"])).trim()}k();import{promises as Yr}from"fs";import{join as B}from"path";k();import{existsSync as Xa}from"fs";import{dirname as zr,join as Kt}from"path";import{fileURLToPath as Za}from"url";var Ya=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g;function Wr(t,e){return t.replace(Ya,(n,r)=>{let i=e[r];return i===void 0?n:String(i)})}var Qa=zr(Za(import.meta.url)),qr=nc(Qa),tc=Kt(qr,"src","templates"),ec=Kt(qr,"src","hooks");function nc(t){let e=t;for(;;){if(Xa(Kt(e,"package.json")))return e;let n=zr(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function rc(t){return await U(Kt(tc,`${t}.tpl`))}async function Ce(t,e){let n=await rc(t);return Wr(n,e)}async function Jr(t){return await U(Kt(ec,`${t}.sh.tpl`))}async function ic(t){if(!await d(t))return null;let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`,r=n,i=1;for(;await d(r);)if(r=`${n}-${i}`,i++,i>5)throw new Error(`Could not find free backup name for ${t}`);return await Yr.rename(t,r),r}async function Xr(t,e,n){let r=await ic(t);return await yt(t,e,n),r}var oc=["state","_pending","_backup"];async function Zr(t){let e=B(t,".claude");await E(e);for(let n of oc){let r=B(e,n);await E(r),await yt(B(r,".gitkeep"),"")}}async function Qr(t,e){return[]}async function wn(t,e){let n=await Ce("CLAUDE.md",e);return await Xr(B(t,"CLAUDE.md"),n)}async function ti(t,e){let n=await Ce("settings.json",e);return await Xr(B(t,".claude","settings.json"),n)}async function ei(t){let e=B(t,".gitignore"),n=await Ce("gitignore",{}),r="# Avatar \u2014 git-ignored entries injected on `avatar init`",i="";if(await d(e)&&(i=await Yr.readFile(e,"utf8"),i.includes(r)))return;let s=i.endsWith(`
37
37
  `)||i.length===0?"":`
38
38
  `;await yt(e,`${i}${s}
39
- ${n}`)}async function Wt(t,e){let n=await ei(e),r=W(t,"hooks");await E(r);let i=W(r,e);await yt(i,n,493)}function ci(t){t.command("doctor").description("Ch\u1EA9n \u0111o\xE1n c\xE0i \u0111\u1EB7t Avatar: hooks, MCP, login, submodule, ...").option("--fix","T\u1EF1 \u0111\u1ED9ng fix c\xE1c issue c\xF3 th\u1EC3 fix t\u1EF1 \u0111\u1ED9ng").action(async e=>{try{let n=await Ic(process.cwd());_c(n),e.fix&&await Oc(n)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Ic(t){let e=[],n=process.versions.node,[r,i]=n.split(".").map(M=>Number.parseInt(M,10)),s=(r??0)>18||(r??0)===18&&(i??0)>=17;e.push({name:"Node.js version",status:s?"ok":"fail",detail:`v${n}${s?"":" (c\u1EA7n >= 18.17)"}`,fixable:!1});let a=await et();a?nt(a)?e.push({name:"Login status",status:"warn",detail:`Token h\u1EBFt h\u1EA1n (${a.email}) \u2014 ch\u1EA1y 'avatar login'`,fixable:!1}):e.push({name:"Login status",status:"ok",detail:`Logged in: ${a.email}`,fixable:!1}):e.push({name:"Login status",status:"fail",detail:"Ch\u01B0a \u0111\u0103ng nh\u1EADp \u2014 ch\u1EA1y 'avatar login'",fixable:!1});let c=ct(t,".claude","pack"),l=ct(t,"CLAUDE.md"),u=ct(t,".git","hooks","post-merge"),[m,g,f,A]=await Promise.all([Ct(t),d(c),d(l),d(u)]);e.push({name:"Git repository",status:m?"ok":"warn",detail:m?t:"Kh\xF4ng ph\u1EA3i git repo (c\u1EA7n cho 'avatar init')",fixable:!1}),e.push({name:"team-ai-pack submodule",status:g?"ok":"warn",detail:g?c:"Avatar ch\u01B0a init \u2014 ch\u1EA1y 'avatar init'",fixable:!1}),e.push({name:"CLAUDE.md",status:f?"ok":"warn",detail:f?"t\u1ED3n t\u1EA1i \u1EDF project root":"thi\u1EBFu \u2014 ch\u1EA1y 'avatar init'",fixable:!1}),m&&g&&e.push({name:"Git hook post-merge",status:A?"ok":"fail",detail:A?"installed":"missing \u2014 fixable",fixable:!A,fix:A?void 0:async()=>{await Wt(ct(t,".git"),"post-merge")}});let y=ct(t,".gitignore"),P="";if(m){let M=!1;await d(y)&&(P=await An.readFile(y,"utf8"),M=P.includes(".claude/_pending/")),e.push({name:".gitignore Avatar entries",status:M?"ok":g?"fail":"warn",detail:M?"c\xF3 .claude/_pending/, .claude/_backup/":"thi\u1EBFu entries",fixable:!1});let Q=P.includes(".claude/settings.json")||P.includes("/.claude/settings.json")||P.includes(".claude/*.json");e.push({name:"\u{1F512} settings.json gitignored",status:Q?"ok":"fail",detail:Q?"an to\xE0n \u2014 settings.json kh\xF4ng commit nh\u1EA7m":"CRITICAL: settings.json ch\u1EE9a API key \u2014 ch\u1EA1y 'avatar doctor --fix' \u0111\u1EC3 add gitignore",fixable:!Q,fix:Q?void 0:async()=>{await An.appendFile(y,`
39
+ ${n}`)}async function Bt(t,e){let n=await Jr(e),r=B(t,"hooks");await E(r);let i=B(r,e);await yt(i,n,493)}function ni(t){t.command("doctor").description("Ch\u1EA9n \u0111o\xE1n c\xE0i \u0111\u1EB7t Avatar: hooks, MCP, login, submodule, ...").option("--fix","T\u1EF1 \u0111\u1ED9ng fix c\xE1c issue c\xF3 th\u1EC3 fix t\u1EF1 \u0111\u1ED9ng").action(async e=>{try{let n=await ac(process.cwd());cc(n),e.fix&&await lc(n)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function ac(t){let e=[],n=process.versions.node,[r,i]=n.split(".").map(M=>Number.parseInt(M,10)),s=(r??0)>18||(r??0)===18&&(i??0)>=17;e.push({name:"Node.js version",status:s?"ok":"fail",detail:`v${n}${s?"":" (c\u1EA7n >= 18.17)"}`,fixable:!1});let a=await tt();a?et(a)?e.push({name:"Login status",status:"warn",detail:`Token h\u1EBFt h\u1EA1n (${a.email}) \u2014 ch\u1EA1y 'avatar login'`,fixable:!1}):e.push({name:"Login status",status:"ok",detail:`Logged in: ${a.email}`,fixable:!1}):e.push({name:"Login status",status:"fail",detail:"Ch\u01B0a \u0111\u0103ng nh\u1EADp \u2014 ch\u1EA1y 'avatar login'",fixable:!1});let c=ct(t,".claude","pack"),l=ct(t,"CLAUDE.md"),u=ct(t,".git","hooks","post-merge"),[m,g,f,S]=await Promise.all([Ct(t),d(c),d(l),d(u)]);e.push({name:"Git repository",status:m?"ok":"warn",detail:m?t:"Kh\xF4ng ph\u1EA3i git repo (c\u1EA7n cho 'avatar init')",fixable:!1}),e.push({name:"team-ai-pack submodule",status:g?"ok":"warn",detail:g?c:"Avatar ch\u01B0a init \u2014 ch\u1EA1y 'avatar init'",fixable:!1}),e.push({name:"CLAUDE.md",status:f?"ok":"warn",detail:f?"t\u1ED3n t\u1EA1i \u1EDF project root":"thi\u1EBFu \u2014 ch\u1EA1y 'avatar init'",fixable:!1}),m&&g&&e.push({name:"Git hook post-merge",status:S?"ok":"fail",detail:S?"installed":"missing \u2014 fixable",fixable:!S,fix:S?void 0:async()=>{await Bt(ct(t,".git"),"post-merge")}});let w=ct(t,".gitignore"),P="";if(m){let M=!1;await d(w)&&(P=await vn.readFile(w,"utf8"),M=P.includes(".claude/_pending/")),e.push({name:".gitignore Avatar entries",status:M?"ok":g?"fail":"warn",detail:M?"c\xF3 .claude/_pending/, .claude/_backup/":"thi\u1EBFu entries",fixable:!1});let Z=P.includes(".claude/settings.json")||P.includes("/.claude/settings.json")||P.includes(".claude/*.json");e.push({name:"\u{1F512} settings.json gitignored",status:Z?"ok":"fail",detail:Z?"an to\xE0n \u2014 settings.json kh\xF4ng commit nh\u1EA7m":"CRITICAL: settings.json ch\u1EE9a API key \u2014 ch\u1EA1y 'avatar doctor --fix' \u0111\u1EC3 add gitignore",fixable:!Z,fix:Z?void 0:async()=>{await vn.appendFile(w,`
40
40
  # Avatar v1.7.0 \u2014 Security: settings.json ch\u1EE9a raw API key, KH\xD4NG commit.
41
41
  .claude/settings.json
42
42
  .claude/settings.json.backup-*
43
- `)}})}let O=xn("which",["python"]),S=xn("which",["python3"]),N=O.status===0,_=S.status===0;_&&!N?e.push({name:"Python binary alias",status:"warn",detail:`Ch\u1EC9 c\xF3 python3 (modern macOS). Pack scripts th\u01B0\u1EDDng ref 'python' \u2192 suggest: ln -s ${S.stdout.toString().trim()} ~/.local/bin/python`,fixable:!1}):N?e.push({name:"Python binary",status:"ok",detail:`python: ${O.stdout.toString().trim()}`,fixable:!1}):_&&e.push({name:"Python binary",status:"ok",detail:`python3: ${S.stdout.toString().trim()}`,fixable:!1});let ht=ct(t,".claude","settings.json");if(await d(ht))try{let M=await An.readFile(ht,"utf8"),Q=JSON.parse(M);if(Q.statusLine?.command){let tr=Q.statusLine.command.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(tr?.[2]){let Mt=tr[2],cs=Mt.startsWith("/")?Mt:ct(t,Mt),er=await d(cs);e.push({name:"statusLine command",status:er?"ok":"fail",detail:er?`ref OK: ${Mt}`:`BROKEN: settings.json ref '${Mt}' nh\u01B0ng file kh\xF4ng t\u1ED3n t\u1EA1i. Strip field statusLine ho\u1EB7c fix path.`,fixable:!1})}}}catch{}let kt=xn("which",["claude"]),oe=kt.status===0;if(e.push({name:"Claude Code CLI",status:oe?"ok":"warn",detail:oe?kt.stdout.toString().trim():"kh\xF4ng t\xECm th\u1EA5y 'claude' tr\xEAn PATH",fixable:!1}),g){let M=await Jr(t);e.push(...M)}return e}function _c(t){let e=[p.bold("Avatar Doctor"),"\u2500".repeat(48)],n=0,r=0,i=0;for(let s of t){let a=s.status==="ok"?p.green("\u2713"):s.status==="warn"?p.yellow("\u26A0"):p.red("\u2717");e.push(`${a} ${s.name.padEnd(28)} ${p.dim(s.detail)}`),s.status==="ok"?n+=1:(r+=1,s.fixable&&(i+=1))}e.push("\u2500".repeat(48)),e.push(`${n} checks passed, ${r} issue${r===1?"":"s"}${i>0?` (${i} fixable \u2014 ch\u1EA1y 'avatar doctor --fix')`:""}`),process.stdout.write(`${Tc(e.join(`
43
+ `)}})}let O=yn("which",["python"]),A=yn("which",["python3"]),N=O.status===0,I=A.status===0;I&&!N?e.push({name:"Python binary alias",status:"warn",detail:`Ch\u1EC9 c\xF3 python3 (modern macOS). Pack scripts th\u01B0\u1EDDng ref 'python' \u2192 suggest: ln -s ${A.stdout.toString().trim()} ~/.local/bin/python`,fixable:!1}):N?e.push({name:"Python binary",status:"ok",detail:`python: ${O.stdout.toString().trim()}`,fixable:!1}):I&&e.push({name:"Python binary",status:"ok",detail:`python3: ${A.stdout.toString().trim()}`,fixable:!1});let ht=ct(t,".claude","settings.json");if(await d(ht))try{let M=await vn.readFile(ht,"utf8"),Z=JSON.parse(M);if(Z.statusLine?.command){let Xn=Z.statusLine.command.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(Xn?.[2]){let Mt=Xn[2],ns=Mt.startsWith("/")?Mt:ct(t,Mt),Zn=await d(ns);e.push({name:"statusLine command",status:Zn?"ok":"fail",detail:Zn?`ref OK: ${Mt}`:`BROKEN: settings.json ref '${Mt}' nh\u01B0ng file kh\xF4ng t\u1ED3n t\u1EA1i. Strip field statusLine ho\u1EB7c fix path.`,fixable:!1})}}}catch{}let kt=yn("which",["claude"]),ie=kt.status===0;if(e.push({name:"Claude Code CLI",status:ie?"ok":"warn",detail:ie?kt.stdout.toString().trim():"kh\xF4ng t\xECm th\u1EA5y 'claude' tr\xEAn PATH",fixable:!1}),g){let M=await Kr(t);e.push(...M)}return e}function cc(t){let e=[p.bold("Avatar Doctor"),"\u2500".repeat(48)],n=0,r=0,i=0;for(let s of t){let a=s.status==="ok"?p.green("\u2713"):s.status==="warn"?p.yellow("\u26A0"):p.red("\u2717");e.push(`${a} ${s.name.padEnd(28)} ${p.dim(s.detail)}`),s.status==="ok"?n+=1:(r+=1,s.fixable&&(i+=1))}e.push("\u2500".repeat(48)),e.push(`${n} checks passed, ${r} issue${r===1?"":"s"}${i>0?` (${i} fixable \u2014 ch\u1EA1y 'avatar doctor --fix')`:""}`),process.stdout.write(`${sc(e.join(`
44
44
  `),{padding:1,borderStyle:"round"})}
45
- `)}async function Oc(t){let e=0;for(let n of t)if(n.fixable&&n.fix)try{await n.fix(),o.success(`Fixed: ${n.name}`),e+=1}catch(r){o.error(`Failed to fix ${n.name}: ${r instanceof Error?r.message:String(r)}`)}e===0&&o.dim("Kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 fix t\u1EF1 \u0111\u1ED9ng.")}import{resolve as Lc}from"path";import{checkbox as Uc,confirm as Hc}from"@inquirer/prompts";import{promises as Nc}from"fs";import{join as Sn}from"path";k();var Mc=[".claude","pack","features"];function li(t){return Sn(t,...Mc)}async function Cn(t){let e=li(t);if(!await d(e))return[];let n=await Nc.readdir(e,{withFileTypes:!0}),r=[];for(let i of n)i.isDirectory()&&await d(Sn(e,i.name,"feature.json"))&&r.push(i.name);return r.sort()}async function ui(t){let e=Sn(li(t),"defaults.json");if(!await d(e))return[];try{return(await v(e)).defaultFeatures??[]}catch{return[]}}async function zt(t){let[e,n]=await Promise.all([Cn(t),St(t)]);return{available:e,enabled:n}}function pi(t,e){let n=new Set(e);return t.map(r=>({name:n.has(r)?`${r} (installed)`:r,value:r,checked:n.has(r)}))}function mi(t){return t.map(e=>({name:e,value:e,checked:!1}))}function di(t){return[...t]}import{spawnSync as jc}from"child_process";function Gc(t){if(!t)return;let e=jc(t,["--version"],{stdio:"ignore"});(e.status!==0||e.error)&&o.warn(`Feature y\xEAu c\u1EA7u runtime '${t}' nh\u01B0ng kh\xF4ng t\xECm th\u1EA5y tr\xEAn PATH. Hook c\xF3 th\u1EC3 fail khi ch\u1EA1y. C\xE0i '${t}' r\u1ED3i th\u1EED l\u1EA1i.`)}function gi(t,e){switch(e.action){case"enabled":o.success(`Feature '${t}' enabled (${e.changes.join("; ")}).`),e.backupPath&&o.dim(` Backup: ${e.backupPath}`);break;case"disabled":o.success(`Feature '${t}' disabled (${e.changes.join("; ")}).`),e.backupPath&&o.dim(` Backup: ${e.backupPath}`);break;case"no-change":o.dim(`Feature '${t}': settings.json \u0111\xE3 \u0111\xFAng tr\u1EA1ng th\xE1i, kh\xF4ng thay \u0111\u1ED5i.`);break;case"no-manifest":break}}async function lt(t,e,n={}){let r=await Kt(t,e);if(!r)return n.silent||o.warn(`Pack version hi\u1EC7n t\u1EA1i ch\u01B0a h\u1ED7 tr\u1EE3 feature '${e}' (thi\u1EBFu .claude/pack/features/${e}/feature.json). Ch\u1EA1y 'avatar sync' \u0111\u1EC3 pull pack m\u1EDBi, ho\u1EB7c ki\u1EC3m t\xEAn feature.`),!1;Gc(r.requires?.runtime);let i=await Ce(t,r);return gi(e,i),await Pe(t,e,{enabled:!0,version:r.version}),!0}async function Pn(t,e){let n=await Kt(t,e);if(!n){o.warn(`Kh\xF4ng t\xECm th\u1EA5y manifest feature '${e}' \u0111\u1EC3 bi\u1EBFt entry n\xE0o c\u1EA7n r\xFAt. N\u1EBFu pack \u0111\xE3 \u0111\u1ED5i, state v\u1EABn \u0111\u01B0\u1EE3c set disabled nh\u01B0ng settings.json c\xF3 th\u1EC3 c\xF2n s\xF3t entry.`);let s=(await At(t)).features[e]?.version??"unknown";return await Pe(t,e,{enabled:!1,version:s}),!1}let r=await zr(t,n);return gi(e,r),await Pe(t,e,{enabled:!1,version:n.version}),!0}function qt(t){return Lc(t.target??process.cwd())}async function fi(t,e,n,r){if(n.all===!0||!process.stdout.isTTY)return di(t);let s=r==="add"?pi(t,e):mi(t);return await Uc({message:r==="add"?"Ch\u1ECDn feature \u0111\u1EC3 c\xE0i (space ch\u1ECDn, enter x\xE1c nh\u1EADn):":"Ch\u1ECDn feature \u0111\u1EC3 g\u1EE1 (space ch\u1ECDn, enter x\xE1c nh\u1EADn):",choices:s})}async function Dc(t){let e=qt(t),{available:n,enabled:r}=await zt(e);if(n.length===0){o.dim("Kh\xF4ng c\xF3 feature available (pack ch\u01B0a c\xF3 features/, ho\u1EB7c ch\u01B0a sync).");return}let i=await fi(n,r,t,"add");if(i.length===0){o.dim("Kh\xF4ng ch\u1ECDn feature n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}for(let s of i)await lt(e,s)}async function Fc(t){let e=qt(t),{enabled:n}=await zt(e);if(n.length===0){o.dim("Kh\xF4ng c\xF3 feature n\xE0o \u0111ang b\u1EADt \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}let r=await fi(n,[],t,"remove");if(r.length===0){o.dim("Kh\xF4ng ch\u1ECDn feature n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}if(t.yes!==!0&&t.all!==!0&&process.stdout.isTTY&&!await Hc({message:`G\u1EE1 ${r.length} feature: ${r.join(", ")}?`,default:!0})){o.dim("H\u1EE7y \u2014 kh\xF4ng g\u1EE1 feature n\xE0o.");return}for(let s of r)await Pn(e,s)}async function Kc(t){let e=qt(t),n=await Cn(e),r=await At(e);if(n.length===0&&Object.keys(r.features).length===0){o.dim("Kh\xF4ng c\xF3 feature n\xE0o (pack ch\u01B0a c\xF3 features/, ho\u1EB7c ch\u01B0a sync). Ch\u1EA1y 'avatar sync'.");return}let i=[...new Set([...n,...Object.keys(r.features)])].sort();o.info("Features:"),console.log(` ${"NAME".padEnd(20)} ${"AVAILABLE".padEnd(10)} ${"ENABLED".padEnd(8)} VERSION`);for(let s of i){let a=n.includes(s)?"yes":"no",c=r.features[s],l=c?.enabled?"yes":"no",u=c?.version??"-";console.log(` ${s.padEnd(20)} ${a.padEnd(10)} ${l.padEnd(8)} ${u}`)}}function hi(t){let e=t.command("feature").description("Qu\u1EA3n l\xFD feature toggle (prompt-scoring, ...) b\u1EADt/t\u1EAFt nguy\xEAn kh\u1ED1i");e.command("enable <name>").description("B\u1EADt feature: merge manifest v\xE0o settings.json + ghi state").option("--target <path>","Workspace root (default: cwd)").action(async(n,r)=>{let i=qt(r);await lt(i,n)||process.exit(1)}),e.command("disable <name>").description("T\u1EAFt feature: r\xFAt \u0111\xFAng entry manifest ra kh\u1ECFi settings.json + ghi state").option("--target <path>","Workspace root (default: cwd)").action(async(n,r)=>{let i=qt(r);await Pn(i,n)}),e.command("list").description("Li\u1EC7t k\xEA feature: available | enabled | version").option("--target <path>","Workspace root (default: cwd)").action(async n=>{await Kc(n)}),e.command("add").description("Ch\u1ECDn (multi-select) feature available \u0111\u1EC3 c\xE0i").option("--target <path>","Workspace root (default: cwd)").option("--all","C\xE0i h\u1EBFt feature available (b\u1ECF qua prompt; auto khi non-TTY)").option("--yes","B\u1ECF x\xE1c nh\u1EADn").action(async n=>{await Dc(n)}),e.command("remove").description("Ch\u1ECDn (multi-select) feature \u0111ang b\u1EADt \u0111\u1EC3 g\u1EE1").option("--target <path>","Workspace root (default: cwd)").option("--all","G\u1EE1 h\u1EBFt feature \u0111ang b\u1EADt (b\u1ECF qua prompt; auto khi non-TTY)").option("--yes","B\u1ECF x\xE1c nh\u1EADn").action(async n=>{await Fc(n)})}k();import{spawnSync as Cl}from"child_process";import{promises as Pl}from"fs";import{join as Mi}from"path";import{spawnSync as wi}from"child_process";import{existsSync as Vc}from"fs";import{join as ki}from"path";var Bc=120*1e3,Wc=300*1e3,z=class extends Error{operation;reason;exitCode;stderr;constructor(e,n,r,i=null,s){super(r),this.name="GitnexusOperationError",this.operation=e,this.reason=n,this.exitCode=i,this.stderr=s}};function yi(t,e,n,r){if(n==="SIGTERM")return new z(t,"timeout",`gitnexus ${t} timeout. Check m\u1EA1ng / repo size.`,null,r);let i=r.toLowerCase();return i.includes("eacces")||i.includes("permission denied")?new z(t,"permission",`gitnexus ${t} fail (permission). Check write access ~/.claude ho\u1EB7c cwd.`,e,r):new z(t,"non-zero-exit",`gitnexus ${t} exit ${e??"null"}. Xem log ph\xEDa tr\xEAn.`,e,r)}function Te(t,e){return t.split(`
45
+ `)}async function lc(t){let e=0;for(let n of t)if(n.fixable&&n.fix)try{await n.fix(),o.success(`Fixed: ${n.name}`),e+=1}catch(r){o.error(`Failed to fix ${n.name}: ${r instanceof Error?r.message:String(r)}`)}e===0&&o.dim("Kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 fix t\u1EF1 \u0111\u1ED9ng.")}import{resolve as gc}from"path";import{checkbox as fc,confirm as hc}from"@inquirer/prompts";import{promises as uc}from"fs";import{join as bn}from"path";k();var pc=[".claude","pack","features"];function ri(t){return bn(t,...pc)}async function xn(t){let e=ri(t);if(!await d(e))return[];let n=await uc.readdir(e,{withFileTypes:!0}),r=[];for(let i of n)i.isDirectory()&&await d(bn(e,i.name,"feature.json"))&&r.push(i.name);return r.sort()}async function ii(t){let e=bn(ri(t),"defaults.json");if(!await d(e))return[];try{return(await y(e)).defaultFeatures??[]}catch{return[]}}async function Wt(t){let[e,n]=await Promise.all([xn(t),At(t)]);return{available:e,enabled:n}}function oi(t,e){let n=new Set(e);return t.map(r=>({name:n.has(r)?`${r} (installed)`:r,value:r,checked:n.has(r)}))}function si(t){return t.map(e=>({name:e,value:e,checked:!1}))}function ai(t){return[...t]}import{spawnSync as mc}from"child_process";function dc(t){if(!t)return;let e=mc(t,["--version"],{stdio:"ignore"});(e.status!==0||e.error)&&o.warn(`Feature y\xEAu c\u1EA7u runtime '${t}' nh\u01B0ng kh\xF4ng t\xECm th\u1EA5y tr\xEAn PATH. Hook c\xF3 th\u1EC3 fail khi ch\u1EA1y. C\xE0i '${t}' r\u1ED3i th\u1EED l\u1EA1i.`)}function ci(t,e){switch(e.action){case"enabled":o.success(`Feature '${t}' enabled (${e.changes.join("; ")}).`),e.backupPath&&o.dim(` Backup: ${e.backupPath}`);break;case"disabled":o.success(`Feature '${t}' disabled (${e.changes.join("; ")}).`),e.backupPath&&o.dim(` Backup: ${e.backupPath}`);break;case"no-change":o.dim(`Feature '${t}': settings.json \u0111\xE3 \u0111\xFAng tr\u1EA1ng th\xE1i, kh\xF4ng thay \u0111\u1ED5i.`);break;case"no-manifest":break}}async function lt(t,e,n={}){let r=await Ft(t,e);if(!r)return n.silent||o.warn(`Pack version hi\u1EC7n t\u1EA1i ch\u01B0a h\u1ED7 tr\u1EE3 feature '${e}' (thi\u1EBFu .claude/pack/features/${e}/feature.json). Ch\u1EA1y 'avatar sync' \u0111\u1EC3 pull pack m\u1EDBi, ho\u1EB7c ki\u1EC3m t\xEAn feature.`),!1;dc(r.requires?.runtime);let i=await be(t,r);return ci(e,i),await xe(t,e,{enabled:!0,version:r.version}),!0}async function Sn(t,e){let n=await Ft(t,e);if(!n){o.warn(`Kh\xF4ng t\xECm th\u1EA5y manifest feature '${e}' \u0111\u1EC3 bi\u1EBFt entry n\xE0o c\u1EA7n r\xFAt. N\u1EBFu pack \u0111\xE3 \u0111\u1ED5i, state v\u1EABn \u0111\u01B0\u1EE3c set disabled nh\u01B0ng settings.json c\xF3 th\u1EC3 c\xF2n s\xF3t entry.`);let s=(await St(t)).features[e]?.version??"unknown";return await xe(t,e,{enabled:!1,version:s}),!1}let r=await Dr(t,n);return ci(e,r),await xe(t,e,{enabled:!1,version:n.version}),!0}function zt(t){return gc(t.target??process.cwd())}async function li(t,e,n,r){if(n.all===!0||!process.stdout.isTTY)return ai(t);let s=r==="add"?oi(t,e):si(t);return await fc({message:r==="add"?"Ch\u1ECDn feature \u0111\u1EC3 c\xE0i (space ch\u1ECDn, enter x\xE1c nh\u1EADn):":"Ch\u1ECDn feature \u0111\u1EC3 g\u1EE1 (space ch\u1ECDn, enter x\xE1c nh\u1EADn):",choices:s})}async function kc(t){let e=zt(t),{available:n,enabled:r}=await Wt(e);if(n.length===0){o.dim("Kh\xF4ng c\xF3 feature available (pack ch\u01B0a c\xF3 features/, ho\u1EB7c ch\u01B0a sync).");return}let i=await li(n,r,t,"add");if(i.length===0){o.dim("Kh\xF4ng ch\u1ECDn feature n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}for(let s of i)await lt(e,s)}async function wc(t){let e=zt(t),{enabled:n}=await Wt(e);if(n.length===0){o.dim("Kh\xF4ng c\xF3 feature n\xE0o \u0111ang b\u1EADt \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}let r=await li(n,[],t,"remove");if(r.length===0){o.dim("Kh\xF4ng ch\u1ECDn feature n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}if(t.yes!==!0&&t.all!==!0&&process.stdout.isTTY&&!await hc({message:`G\u1EE1 ${r.length} feature: ${r.join(", ")}?`,default:!0})){o.dim("H\u1EE7y \u2014 kh\xF4ng g\u1EE1 feature n\xE0o.");return}for(let s of r)await Sn(e,s)}async function yc(t){let e=zt(t),n=await xn(e),r=await St(e);if(n.length===0&&Object.keys(r.features).length===0){o.dim("Kh\xF4ng c\xF3 feature n\xE0o (pack ch\u01B0a c\xF3 features/, ho\u1EB7c ch\u01B0a sync). Ch\u1EA1y 'avatar sync'.");return}let i=[...new Set([...n,...Object.keys(r.features)])].sort();o.info("Features:"),console.log(` ${"NAME".padEnd(20)} ${"AVAILABLE".padEnd(10)} ${"ENABLED".padEnd(8)} VERSION`);for(let s of i){let a=n.includes(s)?"yes":"no",c=r.features[s],l=c?.enabled?"yes":"no",u=c?.version??"-";console.log(` ${s.padEnd(20)} ${a.padEnd(10)} ${l.padEnd(8)} ${u}`)}}function ui(t){let e=t.command("feature").description("Qu\u1EA3n l\xFD feature toggle (prompt-scoring, ...) b\u1EADt/t\u1EAFt nguy\xEAn kh\u1ED1i");e.command("enable <name>").description("B\u1EADt feature: merge manifest v\xE0o settings.json + ghi state").option("--target <path>","Workspace root (default: cwd)").action(async(n,r)=>{let i=zt(r);await lt(i,n)||process.exit(1)}),e.command("disable <name>").description("T\u1EAFt feature: r\xFAt \u0111\xFAng entry manifest ra kh\u1ECFi settings.json + ghi state").option("--target <path>","Workspace root (default: cwd)").action(async(n,r)=>{let i=zt(r);await Sn(i,n)}),e.command("list").description("Li\u1EC7t k\xEA feature: available | enabled | version").option("--target <path>","Workspace root (default: cwd)").action(async n=>{await yc(n)}),e.command("add").description("Ch\u1ECDn (multi-select) feature available \u0111\u1EC3 c\xE0i").option("--target <path>","Workspace root (default: cwd)").option("--all","C\xE0i h\u1EBFt feature available (b\u1ECF qua prompt; auto khi non-TTY)").option("--yes","B\u1ECF x\xE1c nh\u1EADn").action(async n=>{await kc(n)}),e.command("remove").description("Ch\u1ECDn (multi-select) feature \u0111ang b\u1EADt \u0111\u1EC3 g\u1EE1").option("--target <path>","Workspace root (default: cwd)").option("--all","G\u1EE1 h\u1EBFt feature \u0111ang b\u1EADt (b\u1ECF qua prompt; auto khi non-TTY)").option("--yes","B\u1ECF x\xE1c nh\u1EADn").action(async n=>{await wc(n)})}k();import{spawnSync as el}from"child_process";import{promises as nl}from"fs";import{join as Ri}from"path";import{spawnSync as mi}from"child_process";import{existsSync as vc}from"fs";import{join as pi}from"path";var bc=120*1e3,xc=300*1e3,W=class extends Error{operation;reason;exitCode;stderr;constructor(e,n,r,i=null,s){super(r),this.name="GitnexusOperationError",this.operation=e,this.reason=n,this.exitCode=i,this.stderr=s}};function di(t,e,n,r){if(n==="SIGTERM")return new W(t,"timeout",`gitnexus ${t} timeout. Check m\u1EA1ng / repo size.`,null,r);let i=r.toLowerCase();return i.includes("eacces")||i.includes("permission denied")?new W(t,"permission",`gitnexus ${t} fail (permission). Check write access ~/.claude ho\u1EB7c cwd.`,e,r):new W(t,"non-zero-exit",`gitnexus ${t} exit ${e??"null"}. Xem log ph\xEDa tr\xEAn.`,e,r)}function Pe(t,e){return t.split(`
46
46
  `).slice(-e).join(`
47
- `)}function vi(){let t=Lt("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=wi("gitnexus",["setup"],{stdio:["ignore","pipe","pipe"],timeout:Bc,encoding:"utf8"});if(e.status!==0||e.signal==="SIGTERM"){t.fail("GitNexus setup failed");let n=(e.stderr||"").trim(),r=(e.stdout||"").trim();throw n?process.stderr.write(`${Te(n,30)}
48
- `):r&&process.stderr.write(`${Te(r,30)}
49
- `),yi("setup",e.status,e.signal,n)}t.succeed("GitNexus setup OK (global skills installed)")}function Ie(t){let e=Lt(`Analyze workspace ${t} (1-3 ph\xFAt)`),n=wi("gitnexus",["analyze","."],{cwd:t,stdio:["ignore","pipe","pipe"],timeout:Wc,encoding:"utf8"});if(n.status!==0||n.signal==="SIGTERM"){e.fail("Analyze failed");let i=(n.stderr||"").trim(),s=(n.stdout||"").trim();throw i?process.stderr.write(`${Te(i,30)}
50
- `):s&&process.stderr.write(`${Te(s,30)}
51
- `),yi("analyze",n.status,n.signal,i)}let r=ki(t,".gitnexus","meta.json");if(!Vc(r))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new z("analyze","missing-output",`gitnexus analyze xong nh\u01B0ng kh\xF4ng th\u1EA5y ${r}. Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus fail silent.`);e.succeed(`Analyze OK (index t\u1EA1i ${ki(t,".gitnexus")})`)}import{confirm as vl}from"@inquirer/prompts";import bl from"boxen";import{spawnSync as bi}from"child_process";var zc=5e3,qc=/(\d+\.\d+\.\d+)/;function Yc(){let e=rt()==="win32"?"where":"which",n=bi(e,["gitnexus"],{encoding:"utf8"});if(n.error||n.status!==0)return null;let r=(n.stdout||"").trim();return r?r.split(/\r?\n/)[0].trim():null}function Jc(){let t=bi("gitnexus",["--version"],{encoding:"utf8",timeout:zc});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return qc.exec(e)?.[1]??null}var ut=null;function Yt(){if(ut!==null)return ut;let t=Yc();return t?(ut={installed:!0,version:Jc(),path:t},ut):(ut={installed:!1,version:null,path:null},ut)}function _e(){ut=null}import{spawnSync as Xc}from"child_process";var xi=300*1e3,Ai="gitnexus",U=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=r}};function Zc(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new U("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${Ai} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new U("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new U("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function Si(){o.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Xc("npm",["install","-g",Ai],{stdio:["inherit","inherit","pipe"],timeout:xi,encoding:"utf8"});if(t.signal==="SIGTERM")throw new U("timeout",`npm install timeout sau ${xi/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),Zc(t.status,t.stderr||"");_e();let e=Yt();if(!e.installed||!e.path)throw new U("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `gitnexus` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return o.success(`\u0110\xE3 c\xE0i GitNexus${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}import{input as Zf,select as Qc}from"@inquirer/prompts";var x=class extends Error{constructor(e){super(e),this.name="UserAbortedRecoveryError"}};async function T(t){o.warn(`${t.taskName} th\u1EA5t b\u1EA1i: ${t.reason}`),t.hint&&o.info(t.hint);let e=[{name:"Th\u1EED l\u1EA1i (Retry)",value:"retry"}];return t.allowSkip&&e.push({name:"B\u1ECF qua b\u01B0\u1EDBc n\xE0y v\xE0 ti\u1EBFp t\u1EE5c (Skip)",value:"skip"}),e.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i sau (Abort)",value:"abort"}),await Qc({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}k();import{promises as $i}from"fs";import{homedir as tl}from"os";import{join as el}from"path";var Ci=384,Pi={command:"gitnexus",args:["mcp"]};function nl(){return el(tl(),".claude","mcp_servers.json")}function rl(t,e){return t===e?!0:typeof t!="object"||typeof e!="object"||t===null||e===null?!1:JSON.stringify(t)===JSON.stringify(e)}async function il(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await $i.copyFile(t,n),n}async function Ei(){let t=nl(),e={},n=!1;if(await d(t)){n=!0;try{e=await v(t)}catch(a){throw new Error(`MCP config corrupted (${t}): ${a.message}. Backup + x\xF3a file \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}let r=e.mcp_servers?.gitnexus;if(r&&rl(r,Pi))return o.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let i;n&&(i=await il(t),o.dim(`Backup ${t} \u2192 ${i}`));let s={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:Pi}};await $(t,s,Ci);try{await $i.chmod(t,Ci)}catch{}return o.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:i}}import{spawnSync as pl}from"child_process";import{existsSync as ml}from"fs";import{join as Oi}from"path";import{confirm as dl}from"@inquirer/prompts";var ol=[/^claude-(opus|sonnet)-4/i,/^claude-(opus|sonnet|haiku)-[5-9]/i,/^o1(-|$)/i,/^o3(-|$)/i,/^o4(-|$)/i,/nal-claude-(opus|sonnet)-?4/i];function Ri(t){return t?ol.some(e=>e.test(t)):!1}k();Ht();import{promises as Jt}from"fs";import{homedir as sl}from"os";import{dirname as al,join as Ti}from"path";var cl=Ti(sl(),".gitnexus"),Xt=Ti(cl,"config.json");async function ll(){try{let t=await Jt.readFile(Xt,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function ul(t){await Jt.mkdir(al(Xt),{recursive:!0});let e=`${Xt}.tmp-${process.pid}`;await Jt.writeFile(e,t,{mode:384}),await Jt.rename(e,Xt)}async function Ii(t){let n={...await ll(),apiKey:t.apiKey,baseUrl:t.baseUrl,model:t.model,isReasoningModel:t.isReasoningModel??!1};await ul(JSON.stringify(n,null,2)),await Jt.chmod(Xt,384).catch(()=>{})}var gl=900*1e3,fl="nal-claude",hl="claude-sonnet-4-5";function kl(t){let e=t.replace(/\/+$/,"");return e.endsWith("/v1")?`${e}/`:e.endsWith("/v1/")?e:`${e}/v1/`}async function wl(t){let e=Oi(t,".claude","settings.json");if(!await d(e))return null;try{let n=await v(e),r=n.env||{},i=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:null;if(!i)return null;let s=typeof n.model=="string"?n.model:"",a=typeof r.ANTHROPIC_MODEL=="string"?r.ANTHROPIC_MODEL:"",c=s.length>0?s:a,l=null,u=()=>{if(l!==null)return l;try{l=V(t)}catch{l=new Map}return l},m=y=>{let P=process.env[y];if(typeof P=="string"&&P.length>0)return P;let O=r[y];if(typeof O=="string"&&O.length>0)return O;let S=u().get(y);return typeof S=="string"&&S.length>0?S:null},g=typeof n.avatarProvider=="string"?n.avatarProvider:null,f=!1;try{f=new URL(i).hostname==="api.anthropic.com"}catch{f=!1}if((g==="anthropic"?"anthropic":g==="llmlite"?"llmlite":f?"anthropic":"llmlite")==="anthropic"){let y=m("ANTHROPIC_API_KEY");if(y)return{provider:"anthropic",apiKey:y,baseUrl:kl(i),model:c.length>0?c:hl}}else{let y=m("ANTHROPIC_AUTH_TOKEN");if(y)return{provider:"llmlite",apiKey:y,baseUrl:i,model:c.length>0?c:fl}}return null}catch{return null}}async function yl(t,e){return await dl({message:`Generate wiki cho workspace? (~$0.50 qua ${t} model=${e}, 2-5 ph\xFAt). Continue?`,default:!0})}function _i(t,e){return t.split(`
47
+ `)}function gi(){let t=Ht("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=mi("gitnexus",["setup"],{stdio:["ignore","pipe","pipe"],timeout:bc,encoding:"utf8"});if(e.status!==0||e.signal==="SIGTERM"){t.fail("GitNexus setup failed");let n=(e.stderr||"").trim(),r=(e.stdout||"").trim();throw n?process.stderr.write(`${Pe(n,30)}
48
+ `):r&&process.stderr.write(`${Pe(r,30)}
49
+ `),di("setup",e.status,e.signal,n)}t.succeed("GitNexus setup OK (global skills installed)")}function $e(t){let e=Ht(`Analyze workspace ${t} (1-3 ph\xFAt)`),n=mi("gitnexus",["analyze","."],{cwd:t,stdio:["ignore","pipe","pipe"],timeout:xc,encoding:"utf8"});if(n.status!==0||n.signal==="SIGTERM"){e.fail("Analyze failed");let i=(n.stderr||"").trim(),s=(n.stdout||"").trim();throw i?process.stderr.write(`${Pe(i,30)}
50
+ `):s&&process.stderr.write(`${Pe(s,30)}
51
+ `),di("analyze",n.status,n.signal,i)}let r=pi(t,".gitnexus","meta.json");if(!vc(r))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new W("analyze","missing-output",`gitnexus analyze xong nh\u01B0ng kh\xF4ng th\u1EA5y ${r}. Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus fail silent.`);e.succeed(`Analyze OK (index t\u1EA1i ${pi(t,".gitnexus")})`)}import{confirm as Yc}from"@inquirer/prompts";import Xc from"boxen";import{spawnSync as fi}from"child_process";var Sc=5e3,Ac=/(\d+\.\d+\.\d+)/;function Cc(){let e=nt()==="win32"?"where":"which",n=fi(e,["gitnexus"],{encoding:"utf8"});if(n.error||n.status!==0)return null;let r=(n.stdout||"").trim();return r?r.split(/\r?\n/)[0].trim():null}function Pc(){let t=fi("gitnexus",["--version"],{encoding:"utf8",timeout:Sc});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return Ac.exec(e)?.[1]??null}var ut=null;function qt(){if(ut!==null)return ut;let t=Cc();return t?(ut={installed:!0,version:Pc(),path:t},ut):(ut={installed:!1,version:null,path:null},ut)}function Ee(){ut=null}import{spawnSync as $c}from"child_process";var hi=300*1e3,ki="gitnexus",H=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=r}};function Ec(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new H("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${ki} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new H("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new H("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function wi(){o.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=$c("npm",["install","-g",ki],{stdio:["inherit","inherit","pipe"],timeout:hi,encoding:"utf8"});if(t.signal==="SIGTERM")throw new H("timeout",`npm install timeout sau ${hi/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),Ec(t.status,t.stderr||"");Ee();let e=qt();if(!e.installed||!e.path)throw new H("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `gitnexus` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return o.success(`\u0110\xE3 c\xE0i GitNexus${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}import{input as vf,select as Rc}from"@inquirer/prompts";var x=class extends Error{constructor(e){super(e),this.name="UserAbortedRecoveryError"}};async function T(t){o.warn(`${t.taskName} th\u1EA5t b\u1EA1i: ${t.reason}`),t.hint&&o.info(t.hint);let e=[{name:"Th\u1EED l\u1EA1i (Retry)",value:"retry"}];return t.allowSkip&&e.push({name:"B\u1ECF qua b\u01B0\u1EDBc n\xE0y v\xE0 ti\u1EBFp t\u1EE5c (Skip)",value:"skip"}),e.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i sau (Abort)",value:"abort"}),await Rc({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}k();import{promises as bi}from"fs";import{homedir as Tc}from"os";import{join as _c}from"path";var yi=384,vi={command:"gitnexus",args:["mcp"]};function Ic(){return _c(Tc(),".claude","mcp_servers.json")}function Oc(t,e){return t===e?!0:typeof t!="object"||typeof e!="object"||t===null||e===null?!1:JSON.stringify(t)===JSON.stringify(e)}async function Nc(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await bi.copyFile(t,n),n}async function xi(){let t=Ic(),e={},n=!1;if(await d(t)){n=!0;try{e=await y(t)}catch(a){throw new Error(`MCP config corrupted (${t}): ${a.message}. Backup + x\xF3a file \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}let r=e.mcp_servers?.gitnexus;if(r&&Oc(r,vi))return o.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let i;n&&(i=await Nc(t),o.dim(`Backup ${t} \u2192 ${i}`));let s={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:vi}};await $(t,s,yi);try{await bi.chmod(t,yi)}catch{}return o.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:i}}import{spawnSync as Dc}from"child_process";import{existsSync as Fc}from"fs";import{join as $i}from"path";import{confirm as Vc}from"@inquirer/prompts";var Mc=[/^claude-(opus|sonnet)-4/i,/^claude-(opus|sonnet|haiku)-[5-9]/i,/^o1(-|$)/i,/^o3(-|$)/i,/^o4(-|$)/i,/nal-claude-(opus|sonnet)-?4/i];function Si(t){return t?Mc.some(e=>e.test(t)):!1}k();Ut();import{promises as Jt}from"fs";import{homedir as jc}from"os";import{dirname as Lc,join as Ai}from"path";var Hc=Ai(jc(),".gitnexus"),Yt=Ai(Hc,"config.json");async function Gc(){try{let t=await Jt.readFile(Yt,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function Uc(t){await Jt.mkdir(Lc(Yt),{recursive:!0});let e=`${Yt}.tmp-${process.pid}`;await Jt.writeFile(e,t,{mode:384}),await Jt.rename(e,Yt)}async function Ci(t){let n={...await Gc(),apiKey:t.apiKey,baseUrl:t.baseUrl,model:t.model,isReasoningModel:t.isReasoningModel??!1};await Uc(JSON.stringify(n,null,2)),await Jt.chmod(Yt,384).catch(()=>{})}var Kc=900*1e3,Bc="nal-claude",Wc="claude-sonnet-4-5";function zc(t){let e=t.replace(/\/+$/,"");return e.endsWith("/v1")?`${e}/`:e.endsWith("/v1/")?e:`${e}/v1/`}async function qc(t){let e=$i(t,".claude","settings.json");if(!await d(e))return null;try{let n=await y(e),r=n.env||{},i=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:null;if(!i)return null;let s=typeof n.model=="string"?n.model:"",a=typeof r.ANTHROPIC_MODEL=="string"?r.ANTHROPIC_MODEL:"",c=s.length>0?s:a,l=null,u=()=>{if(l!==null)return l;try{l=V(t)}catch{l=new Map}return l},m=w=>{let P=process.env[w];if(typeof P=="string"&&P.length>0)return P;let O=r[w];if(typeof O=="string"&&O.length>0)return O;let A=u().get(w);return typeof A=="string"&&A.length>0?A:null},g=typeof n.avatarProvider=="string"?n.avatarProvider:null,f=!1;try{f=new URL(i).hostname==="api.anthropic.com"}catch{f=!1}if((g==="anthropic"?"anthropic":g==="llmlite"?"llmlite":f?"anthropic":"llmlite")==="anthropic"){let w=m("ANTHROPIC_API_KEY");if(w)return{provider:"anthropic",apiKey:w,baseUrl:zc(i),model:c.length>0?c:Wc}}else{let w=m("ANTHROPIC_AUTH_TOKEN");if(w)return{provider:"llmlite",apiKey:w,baseUrl:i,model:c.length>0?c:Bc}}return null}catch{return null}}async function Jc(t,e){return await Vc({message:`Generate wiki cho workspace? (~$0.50 qua ${t} model=${e}, 2-5 ph\xFAt). Continue?`,default:!0})}function Pi(t,e){return t.split(`
52
52
  `).slice(-e).join(`
53
- `)}async function Ni(t){let e=await wl(t);if(!e)return o.warn(`Kh\xF4ng resolve \u0111\u01B0\u1EE3c API key cho wiki gen.
53
+ `)}async function Ei(t){let e=await qc(t);if(!e)return o.warn(`Kh\xF4ng resolve \u0111\u01B0\u1EE3c API key cho wiki gen.
54
54
  \u0110\xE3 ki\u1EC3m tra: process.env \u2192 .envrc Avatar block \u2192 settings.json.env
55
55
  Nguy\xEAn nh\xE2n c\xF3 th\u1EC3:
56
56
  \u2022 Subscription mode (OAuth, kh\xF4ng c\xF3 key)
57
57
  \u2022 Key ch\u01B0a setup \u2014 ch\u1EA1y 'avatar ai setup' ho\u1EB7c 'avatar secrets set ANTHROPIC_API_KEY'
58
58
  \u2022 settings.json.env.ANTHROPIC_BASE_URL b\u1ECB thi\u1EBFu (provider ch\u01B0a configured)`),o.dim(`\u0110\u1EC3 gen wiki sau khi \u0111\xE3 c\xF3 key, ch\u1EA1y manual:
59
- gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await yl(e.baseUrl,e.model))return o.dim("User decline wiki gen \u2014 workspace OK kh\xF4ng c\xF3 wiki. Ch\u1EA1y `gitnexus wiki` manual sau khi c\u1EA7n."),{ran:!1,skipped:!0,reason:"user-declined"};let r=Ri(e.model);await Ii({apiKey:e.apiKey,baseUrl:e.baseUrl,model:e.model,isReasoningModel:r});let i=["wiki",".","--base-url",e.baseUrl,"--model",e.model];r&&i.push("--reasoning-model");let s=Lt(`Generating wiki via ${e.baseUrl} (${e.provider}) model=${e.model}${r?" [reasoning]":""}`),a=pl("gitnexus",i,{cwd:t,stdio:["ignore","pipe","pipe"],timeout:gl,encoding:"utf8"});if(a.status!==0||a.signal==="SIGTERM"){let l=a.signal==="SIGTERM"?"timeout":"non-zero-exit";s.fail(`Wiki gen ${l} (exit ${a.status??"null"})`);let u=(a.stderr||"").trim(),m=(a.stdout||"").trim();return u?process.stderr.write(`${_i(u,30)}
60
- `):m&&process.stderr.write(`${_i(m,30)}
61
- `),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${l} (exit ${a.status??"null"})`}}let c=Oi(t,".gitnexus","wiki","index.html");return ml(c)?(s.succeed(`Wiki ready: ${c}`),{ran:!0,skipped:!1,wikiPath:c}):(s.fail("Wiki exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y index.html"),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y ${c}`})}async function xl(){let t=[p.bold("\u{1F9E0} GitNexus ch\u01B0a c\xE0i"),"","GitNexus = code intelligence layer cho Claude Code:"," \u2022 Architectural awareness (impact analysis)"," \u2022 Call chain debug + blast radius tr\u01B0\u1EDBc refactor"," \u2022 Wiki HTML t\u1EF1 gen m\xF4 t\u1EA3 codebase","",`S\u1EBD c\xE0i: ${p.cyan("npm install -g gitnexus")} (global)`];return process.stdout.write(`${bl(t.join(`
59
+ gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await Jc(e.baseUrl,e.model))return o.dim("User decline wiki gen \u2014 workspace OK kh\xF4ng c\xF3 wiki. Ch\u1EA1y `gitnexus wiki` manual sau khi c\u1EA7n."),{ran:!1,skipped:!0,reason:"user-declined"};let r=Si(e.model);await Ci({apiKey:e.apiKey,baseUrl:e.baseUrl,model:e.model,isReasoningModel:r});let i=["wiki",".","--base-url",e.baseUrl,"--model",e.model];r&&i.push("--reasoning-model");let s=Ht(`Generating wiki via ${e.baseUrl} (${e.provider}) model=${e.model}${r?" [reasoning]":""}`),a=Dc("gitnexus",i,{cwd:t,stdio:["ignore","pipe","pipe"],timeout:Kc,encoding:"utf8"});if(a.status!==0||a.signal==="SIGTERM"){let l=a.signal==="SIGTERM"?"timeout":"non-zero-exit";s.fail(`Wiki gen ${l} (exit ${a.status??"null"})`);let u=(a.stderr||"").trim(),m=(a.stdout||"").trim();return u?process.stderr.write(`${Pi(u,30)}
60
+ `):m&&process.stderr.write(`${Pi(m,30)}
61
+ `),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${l} (exit ${a.status??"null"})`}}let c=$i(t,".gitnexus","wiki","index.html");return Fc(c)?(s.succeed(`Wiki ready: ${c}`),{ran:!0,skipped:!1,wikiPath:c}):(s.fail("Wiki exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y index.html"),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y ${c}`})}async function Zc(){let t=[p.bold("\u{1F9E0} GitNexus ch\u01B0a c\xE0i"),"","GitNexus = code intelligence layer cho Claude Code:"," \u2022 Architectural awareness (impact analysis)"," \u2022 Call chain debug + blast radius tr\u01B0\u1EDBc refactor"," \u2022 Wiki HTML t\u1EF1 gen m\xF4 t\u1EA3 codebase","",`S\u1EBD c\xE0i: ${p.cyan("npm install -g gitnexus")} (global)`];return process.stdout.write(`${Xc(t.join(`
62
62
  `),{padding:1,borderStyle:"round",borderColor:"cyan"})}
63
- `),await vl({message:"C\xE0i GitNexus global?",default:!0})}async function Al(){for(;;)try{return Si(),!0}catch(t){let e=t instanceof Error?t.message:String(t),n=t instanceof U&&t.reason==="permission-denied"?"Th\u1EED l\u1EA1i v\u1EDBi sudo, ho\u1EB7c fix npm prefix: npm config set prefix ~/.npm-global":"Check log npm ph\xEDa tr\xEAn + th\u1EED l\u1EA1i.",r=await T({taskName:"C\xE0i GitNexus qua npm",reason:e,allowSkip:!0,hint:n});if(r==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i GitNexus.");if(r==="skip")return!1}}async function Sl(t){for(;;)try{return Ie(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),r=e instanceof z&&e.reason==="missing-output"?"Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus version mismatch. Check `gitnexus --version`.":"Network glitch? Retry th\u01B0\u1EDDng work.",i=await T({taskName:"GitNexus analyze workspace",reason:n,allowSkip:!0,hint:r});if(i==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc GitNexus analyze.");if(i==="skip")return!1}}async function Oe(t){let e={ok:!1,installed:!1,analyzed:!1,wikiGenerated:!1,mcpRegistered:!1};try{o.info("=== Phase 10: GitNexus Setup ===");let n=Yt();if(!n.installed){if(!await xl())return await w("gitnexus_setup","result=skipped,reason=user-declined"),o.dim("Skip GitNexus. C\xE0i sau qua `avatar gitnexus install`."),e.reason="user-declined",e;if(!await Al())return await w("gitnexus_setup","result=skipped,reason=install-skipped"),o.dim("Skip GitNexus install. Workspace OK kh\xF4ng c\xF3 codebase intelligence."),e.reason="install-skipped",e;if(_e(),n=Yt(),!n.installed)throw new Error("C\xE0i xong nh\u01B0ng kh\xF4ng detect \u0111\u01B0\u1EE3c binary (PATH issue).")}e.installed=!0,o.success(`GitNexus available${n.version?` v${n.version}`:""}`);try{vi()}catch(s){o.warn(`gitnexus setup fail: ${s.message}`),o.dim("Skip global skills install. Workspace v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c.")}if(!await Sl(t.workspacePath))return await w("gitnexus_setup","result=skipped,reason=analyze-skipped"),o.dim("Skip analyze. GitNexus installed nh\u01B0ng ch\u01B0a index."),e.reason="analyze-skipped",e;e.analyzed=!0;let i=await Ni(t.workspacePath);e.wikiGenerated=i.ran,i.skipped&&i.reason==="fail"&&o.warn(`Wiki gen fail (workspace v\u1EABn OK): ${i.detail??"unknown"}`);try{let s=await Ei();e.mcpRegistered=!0,s.wasUpdated||o.dim("MCP server gitnexus \u0111\xE3 registered tr\u01B0\u1EDBc \u0111\xF3.")}catch(s){o.warn(`MCP server register fail: ${s.message}`),o.dim("Workspace OK nh\u01B0ng Claude Code kh\xF4ng t\u1EF1 attach MCP server. Manual add v\xE0o ~/.claude/mcp_servers.json.")}return e.ok=!0,await w("gitnexus_setup",`result=ok,analyzed=${e.analyzed},wiki=${e.wikiGenerated},mcp=${e.mcpRegistered}`),o.success("GitNexus ready"),e}catch(n){if(n instanceof x)throw n;let r=n instanceof Error?n.message:String(n);return o.warn(`GitNexus setup th\u1EA5t b\u1EA1i: ${r}`),o.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup sau qua `avatar gitnexus install`."),await w("gitnexus_setup",`result=failed,error=${r.slice(0,200)}`),e.reason=r,e}}function $n(){let t=process.cwd(),e=vt(t);return e||(o.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
63
+ `),await Yc({message:"C\xE0i GitNexus global?",default:!0})}async function Qc(){for(;;)try{return wi(),!0}catch(t){let e=t instanceof Error?t.message:String(t),n=t instanceof H&&t.reason==="permission-denied"?"Th\u1EED l\u1EA1i v\u1EDBi sudo, ho\u1EB7c fix npm prefix: npm config set prefix ~/.npm-global":"Check log npm ph\xEDa tr\xEAn + th\u1EED l\u1EA1i.",r=await T({taskName:"C\xE0i GitNexus qua npm",reason:e,allowSkip:!0,hint:n});if(r==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i GitNexus.");if(r==="skip")return!1}}async function tl(t){for(;;)try{return $e(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),r=e instanceof W&&e.reason==="missing-output"?"Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus version mismatch. Check `gitnexus --version`.":"Network glitch? Retry th\u01B0\u1EDDng work.",i=await T({taskName:"GitNexus analyze workspace",reason:n,allowSkip:!0,hint:r});if(i==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc GitNexus analyze.");if(i==="skip")return!1}}async function Re(t){let e={ok:!1,installed:!1,analyzed:!1,wikiGenerated:!1,mcpRegistered:!1};try{o.info("=== Phase 10: GitNexus Setup ===");let n=qt();if(!n.installed){if(!await Zc())return await v("gitnexus_setup","result=skipped,reason=user-declined"),o.dim("Skip GitNexus. C\xE0i sau qua `avatar gitnexus install`."),e.reason="user-declined",e;if(!await Qc())return await v("gitnexus_setup","result=skipped,reason=install-skipped"),o.dim("Skip GitNexus install. Workspace OK kh\xF4ng c\xF3 codebase intelligence."),e.reason="install-skipped",e;if(Ee(),n=qt(),!n.installed)throw new Error("C\xE0i xong nh\u01B0ng kh\xF4ng detect \u0111\u01B0\u1EE3c binary (PATH issue).")}e.installed=!0,o.success(`GitNexus available${n.version?` v${n.version}`:""}`);try{gi()}catch(s){o.warn(`gitnexus setup fail: ${s.message}`),o.dim("Skip global skills install. Workspace v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c.")}if(!await tl(t.workspacePath))return await v("gitnexus_setup","result=skipped,reason=analyze-skipped"),o.dim("Skip analyze. GitNexus installed nh\u01B0ng ch\u01B0a index."),e.reason="analyze-skipped",e;e.analyzed=!0;let i=await Ei(t.workspacePath);e.wikiGenerated=i.ran,i.skipped&&i.reason==="fail"&&o.warn(`Wiki gen fail (workspace v\u1EABn OK): ${i.detail??"unknown"}`);try{let s=await xi();e.mcpRegistered=!0,s.wasUpdated||o.dim("MCP server gitnexus \u0111\xE3 registered tr\u01B0\u1EDBc \u0111\xF3.")}catch(s){o.warn(`MCP server register fail: ${s.message}`),o.dim("Workspace OK nh\u01B0ng Claude Code kh\xF4ng t\u1EF1 attach MCP server. Manual add v\xE0o ~/.claude/mcp_servers.json.")}return e.ok=!0,await v("gitnexus_setup",`result=ok,analyzed=${e.analyzed},wiki=${e.wikiGenerated},mcp=${e.mcpRegistered}`),o.success("GitNexus ready"),e}catch(n){if(n instanceof x)throw n;let r=n instanceof Error?n.message:String(n);return o.warn(`GitNexus setup th\u1EA5t b\u1EA1i: ${r}`),o.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup sau qua `avatar gitnexus install`."),await v("gitnexus_setup",`result=failed,error=${r.slice(0,200)}`),e.reason=r,e}}function An(){let t=process.cwd(),e=vt(t);return e||(o.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
64
64
  Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
65
65
  B\u1EA1n \u0111ang \u1EDF: ${t}
66
- Cd v\xE0o workspace dir r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&o.dim(`Detected workspace root: ${e}`),e}async function $l(){let t=$n(),e=await Oe({workspacePath:t});e.ok?(o.success("GitNexus setup complete"),o.dim("Update CLAUDE.md \u0111\u1EC3 re-render section GitNexus: re-run avatar init ho\u1EB7c ch\u1EC9nh tay.")):o.warn(`Setup kh\xF4ng complete: ${e.reason??"unknown"}`)}async function El(){let t=$n(),e=Mi(t,".gitnexus","meta.json");if(!await d(e)){o.warn(`Ch\u01B0a c\xF3 ${e}. Ch\u1EA1y: avatar gitnexus install`);return}try{let n=await v(e);if(o.info(`Project: ${t}`),o.info(`Last commit: ${n.lastCommit?.slice(0,7)??"(unknown)"}`),o.info(`Indexed at: ${n.indexedAt??"(unknown)"}`),n.stats&&o.info(`Stats: ${n.stats.files??"?"} files \xB7 ${n.stats.nodes??"?"} nodes \xB7 ${n.stats.edges??"?"} edges`),n.lastCommit){let i=Cl("git",["rev-parse","HEAD"],{cwd:t,encoding:"utf8"});i.status===0&&(i.stdout.trim()!==n.lastCommit?o.warn("\u26A0 Index stale \u2014 HEAD \u0111\xE3 ti\u1EBFn t\u1EEB lastCommit. Ch\u1EA1y: avatar gitnexus analyze"):o.dim("Index fresh (HEAD === lastCommit)"))}let r=Mi(t,".gitnexus","wiki","index.html");if(await d(r)){let i=await Pl.stat(r);o.info(`Wiki: ${i.mtime.toISOString()}`)}else o.dim("Wiki: ch\u01B0a generate (ch\u1EA1y gitnexus wiki manual)")}catch(n){o.error(`Read meta.json fail: ${n.message}`),process.exit(1)}}async function Rl(){let t=$n();try{Ie(t),o.success("Index refreshed. Ch\u1EA1y `avatar gitnexus status` xem chi ti\u1EBFt.")}catch(e){o.error(`Analyze fail: ${e.message}`),process.exit(1)}}function ji(t){let e=t.command("gitnexus").description("Qu\u1EA3n l\xFD GitNexus code intelligence (M10)");e.command("install").description("C\xE0i + setup GitNexus cho workspace hi\u1EC7n t\u1EA1i").action(async()=>{await $l()}),e.command("status").description("Show index info + staleness warning").action(async()=>{await El()}),e.command("analyze").description("Re-run analyze refresh index (no wiki)").action(async()=>{await Rl()})}import{select as Tp}from"@inquirer/prompts";import En from"chalk";var Ne=[" \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2557 \u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2588\u2557 ","\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u255A\u2550\u2550\u2588\u2588\u2554\u2550\u2550\u255D\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557","\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551\u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551\u2588\u2588\u2588\u2588\u2588\u2588\u2554\u255D","\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551\u255A\u2588\u2588\u2557 \u2588\u2588\u2554\u255D\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557","\u2588\u2588\u2551 \u2588\u2588\u2551 \u255A\u2588\u2588\u2588\u2588\u2554\u255D \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2551 \u2588\u2588\u2551","\u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u2550\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D\u255A\u2550\u255D \u255A\u2550\u255D"],Me=[[217,79,30],[200,70,80],[170,70,140],[125,88,217]];function Rn(t,e,n){return Math.round(t+(e-t)*n)}function Tl(t){let n=Math.max(0,Math.min(1,t))*(Me.length-1),r=Math.floor(n),i=Math.min(Me.length-1,r+1),s=n-r,a=Me[r],c=Me[i];return[Rn(a[0],c[0],s),Rn(a[1],c[1],s),Rn(a[2],c[2],s)]}function Tn(t){if(!((process.stdout.isTTY??!1)&&En.level>0))return[...Ne,...t?.tagline?["",t.tagline]:[]].join(`
67
- `);let r=Ne.map((i,s)=>{let a=Ne.length===1?0:s/(Ne.length-1),[c,l,u]=Tl(a);return En.rgb(c,l,u).bold(i)});return t?.tagline&&(r.push(""),r.push(En.dim(t.tagline))),r.join(`
66
+ Cd v\xE0o workspace dir r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&o.dim(`Detected workspace root: ${e}`),e}async function rl(){let t=An(),e=await Re({workspacePath:t});e.ok?(o.success("GitNexus setup complete"),o.dim("Update CLAUDE.md \u0111\u1EC3 re-render section GitNexus: re-run avatar init ho\u1EB7c ch\u1EC9nh tay.")):o.warn(`Setup kh\xF4ng complete: ${e.reason??"unknown"}`)}async function il(){let t=An(),e=Ri(t,".gitnexus","meta.json");if(!await d(e)){o.warn(`Ch\u01B0a c\xF3 ${e}. Ch\u1EA1y: avatar gitnexus install`);return}try{let n=await y(e);if(o.info(`Project: ${t}`),o.info(`Last commit: ${n.lastCommit?.slice(0,7)??"(unknown)"}`),o.info(`Indexed at: ${n.indexedAt??"(unknown)"}`),n.stats&&o.info(`Stats: ${n.stats.files??"?"} files \xB7 ${n.stats.nodes??"?"} nodes \xB7 ${n.stats.edges??"?"} edges`),n.lastCommit){let i=el("git",["rev-parse","HEAD"],{cwd:t,encoding:"utf8"});i.status===0&&(i.stdout.trim()!==n.lastCommit?o.warn("\u26A0 Index stale \u2014 HEAD \u0111\xE3 ti\u1EBFn t\u1EEB lastCommit. Ch\u1EA1y: avatar gitnexus analyze"):o.dim("Index fresh (HEAD === lastCommit)"))}let r=Ri(t,".gitnexus","wiki","index.html");if(await d(r)){let i=await nl.stat(r);o.info(`Wiki: ${i.mtime.toISOString()}`)}else o.dim("Wiki: ch\u01B0a generate (ch\u1EA1y gitnexus wiki manual)")}catch(n){o.error(`Read meta.json fail: ${n.message}`),process.exit(1)}}async function ol(){let t=An();try{$e(t),o.success("Index refreshed. Ch\u1EA1y `avatar gitnexus status` xem chi ti\u1EBFt.")}catch(e){o.error(`Analyze fail: ${e.message}`),process.exit(1)}}function Ti(t){let e=t.command("gitnexus").description("Qu\u1EA3n l\xFD GitNexus code intelligence (M10)");e.command("install").description("C\xE0i + setup GitNexus cho workspace hi\u1EC7n t\u1EA1i").action(async()=>{await rl()}),e.command("status").description("Show index info + staleness warning").action(async()=>{await il()}),e.command("analyze").description("Re-run analyze refresh index (no wiki)").action(async()=>{await ol()})}import{select as sp}from"@inquirer/prompts";import Cn from"chalk";var Te=[" \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2557 \u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2588\u2557 ","\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u255A\u2550\u2550\u2588\u2588\u2554\u2550\u2550\u255D\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557","\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551\u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551\u2588\u2588\u2588\u2588\u2588\u2588\u2554\u255D","\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551\u255A\u2588\u2588\u2557 \u2588\u2588\u2554\u255D\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557","\u2588\u2588\u2551 \u2588\u2588\u2551 \u255A\u2588\u2588\u2588\u2588\u2554\u255D \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2551 \u2588\u2588\u2551","\u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u2550\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D\u255A\u2550\u255D \u255A\u2550\u255D"],_e=[[217,79,30],[200,70,80],[170,70,140],[125,88,217]];function Pn(t,e,n){return Math.round(t+(e-t)*n)}function sl(t){let n=Math.max(0,Math.min(1,t))*(_e.length-1),r=Math.floor(n),i=Math.min(_e.length-1,r+1),s=n-r,a=_e[r],c=_e[i];return[Pn(a[0],c[0],s),Pn(a[1],c[1],s),Pn(a[2],c[2],s)]}function $n(t){if(!((process.stdout.isTTY??!1)&&Cn.level>0))return[...Te,...t?.tagline?["",t.tagline]:[]].join(`
67
+ `);let r=Te.map((i,s)=>{let a=Te.length===1?0:s/(Te.length-1),[c,l,u]=sl(a);return Cn.rgb(c,l,u).bold(i)});return t?.tagline&&(r.push(""),r.push(Cn.dim(t.tagline))),r.join(`
68
68
  `)}function $t(t){process.stdout.write(`
69
- ${Tn(t)}
69
+ ${$n(t)}
70
70
 
71
- `)}import{spawnSync as Hi}from"child_process";import{input as Kl,select as Vl}from"@inquirer/prompts";import{spawnSync as On}from"child_process";import{promises as Nl}from"fs";import{basename as Ml,join as Li}from"path";import{confirm as jl,select as Gl}from"@inquirer/prompts";import{spawnSync as Il}from"child_process";var In=class extends Error{constructor(e){super(`Repo "${e}" \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub. \u0110\u1ED5i t\xEAn ho\u1EB7c x\xF3a repo c\u0169.`),this.name="RepoAlreadyExistsError"}};function Gi(t){let e=`${t.org}/${t.name}`,n=["repo","create",e,`--${t.visibility}`,"--source",t.folder,"--remote","origin","--push"],r=Il("gh",n,{stdio:"inherit"});if(r.status!==0)throw r.status===1?new In(e):new Error(`gh repo create th\u1EA5t b\u1EA1i (exit ${r.status})`);return{sshUrl:`git@github.com:${e}.git`,httpsUrl:`https://github.com/${e}.git`}}import{spawnSync as _l}from"child_process";function je(){let t=_l("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(t.status!==0)throw new Error(`Kh\xF4ng l\u1EA5y \u0111\u01B0\u1EE3c GitHub username: ${t.stderr?.trim()}`);return t.stdout.trim()}var Ol=/^[a-zA-Z0-9._-]{1,100}$/,_n=class extends Error{constructor(e){super(`T\xEAn repo "${e}" kh\xF4ng h\u1EE3p l\u1EC7. Ch\u1EC9 d\xF9ng ch\u1EEF/s\u1ED1/d\u1EA5u ch\u1EA5m/g\u1EA1ch/underscore, d\xE0i 1-100 k\xFD t\u1EF1.`),this.name="InvalidRepoNameError"}};function Ge(t){if(!Ol.test(t))throw new _n(t)}function Le(t){if(t!=="private"&&t!=="public")throw new Error(`Visibility ph\u1EA3i l\xE0 "private" ho\u1EB7c "public", nh\u1EADn: "${t}"`)}function Et(t){Ge(t.name),Le(t.visibility);let e=t.org??je();o.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=Gi({folder:t.folder,org:e,name:t.name,visibility:t.visibility});return o.success(`\u0110\xE3 t\u1EA1o: ${n.sshUrl}`),n}k();function Ll(){let t=new Date;return`${t.getFullYear().toString().slice(-2)}${String(t.getMonth()+1).padStart(2,"0")}${String(t.getDate()).padStart(2,"0")}-${String(t.getHours()).padStart(2,"0")}${String(t.getMinutes()).padStart(2,"0")}`}async function Ul(t){let e=Li(t,".git");if(!await d(e))throw new Error(`.git kh\xF4ng t\u1ED3n t\u1EA1i \u1EDF ${t} \u2014 kh\xF4ng c\u1EA7n reset.`);let n=`.git.backup-${Ll()}`,r=Li(t,n);return await Nl.rename(e,r),o.success(`Backup .git \u2192 ${n}`),r}function Hl(t){let e=On("git",["-C",t,"init","-b","main"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(e.status!==0)throw new Error(`git init th\u1EA5t b\u1EA1i: ${e.stderr||e.stdout}`);o.success("Git init m\u1EDBi (branch main)"),On("git",["-C",t,"add","-A"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});let n=On("git",["-C",t,"commit","-m","chore: initial commit from existing folder"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});n.status!==0?o.warn(`git commit th\u1EA5t b\u1EA1i (folder c\xF3 th\u1EC3 r\u1ED7ng): ${(n.stderr||"").slice(0,200)}`):o.success("Initial commit")}async function Ui(t){let e=Ml(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await jl({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
71
+ `)}import{spawnSync as Ni}from"child_process";import{input as yl,select as vl}from"@inquirer/prompts";import{spawnSync as Tn}from"child_process";import{promises as ul}from"fs";import{basename as pl,join as Ii}from"path";import{confirm as ml,select as dl}from"@inquirer/prompts";import{spawnSync as al}from"child_process";var En=class extends Error{constructor(e){super(`Repo "${e}" \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub. \u0110\u1ED5i t\xEAn ho\u1EB7c x\xF3a repo c\u0169.`),this.name="RepoAlreadyExistsError"}};function _i(t){let e=`${t.org}/${t.name}`,n=["repo","create",e,`--${t.visibility}`,"--source",t.folder,"--remote","origin","--push"],r=al("gh",n,{stdio:"inherit"});if(r.status!==0)throw r.status===1?new En(e):new Error(`gh repo create th\u1EA5t b\u1EA1i (exit ${r.status})`);return{sshUrl:`git@github.com:${e}.git`,httpsUrl:`https://github.com/${e}.git`}}import{spawnSync as cl}from"child_process";function Ie(){let t=cl("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(t.status!==0)throw new Error(`Kh\xF4ng l\u1EA5y \u0111\u01B0\u1EE3c GitHub username: ${t.stderr?.trim()}`);return t.stdout.trim()}var ll=/^[a-zA-Z0-9._-]{1,100}$/,Rn=class extends Error{constructor(e){super(`T\xEAn repo "${e}" kh\xF4ng h\u1EE3p l\u1EC7. Ch\u1EC9 d\xF9ng ch\u1EEF/s\u1ED1/d\u1EA5u ch\u1EA5m/g\u1EA1ch/underscore, d\xE0i 1-100 k\xFD t\u1EF1.`),this.name="InvalidRepoNameError"}};function Oe(t){if(!ll.test(t))throw new Rn(t)}function Ne(t){if(t!=="private"&&t!=="public")throw new Error(`Visibility ph\u1EA3i l\xE0 "private" ho\u1EB7c "public", nh\u1EADn: "${t}"`)}function Et(t){Oe(t.name),Ne(t.visibility);let e=t.org??Ie();o.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=_i({folder:t.folder,org:e,name:t.name,visibility:t.visibility});return o.success(`\u0110\xE3 t\u1EA1o: ${n.sshUrl}`),n}k();function gl(){let t=new Date;return`${t.getFullYear().toString().slice(-2)}${String(t.getMonth()+1).padStart(2,"0")}${String(t.getDate()).padStart(2,"0")}-${String(t.getHours()).padStart(2,"0")}${String(t.getMinutes()).padStart(2,"0")}`}async function fl(t){let e=Ii(t,".git");if(!await d(e))throw new Error(`.git kh\xF4ng t\u1ED3n t\u1EA1i \u1EDF ${t} \u2014 kh\xF4ng c\u1EA7n reset.`);let n=`.git.backup-${gl()}`,r=Ii(t,n);return await ul.rename(e,r),o.success(`Backup .git \u2192 ${n}`),r}function hl(t){let e=Tn("git",["-C",t,"init","-b","main"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(e.status!==0)throw new Error(`git init th\u1EA5t b\u1EA1i: ${e.stderr||e.stdout}`);o.success("Git init m\u1EDBi (branch main)"),Tn("git",["-C",t,"add","-A"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});let n=Tn("git",["-C",t,"commit","-m","chore: initial commit from existing folder"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});n.status!==0?o.warn(`git commit th\u1EA5t b\u1EA1i (folder c\xF3 th\u1EC3 r\u1ED7ng): ${(n.stderr||"").slice(0,200)}`):o.success("Initial commit")}async function Oi(t){let e=pl(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await ml({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
72
72
  1. Backup .git \u2192 .git.backup-{timestamp}
73
73
  2. Git init m\u1EDBi (branch main, initial commit)
74
74
  3. T\u1EA1o GitHub repo m\u1EDBi '${n}' d\u01B0\u1EDBi account c\u1EE7a b\u1EA1n
75
- Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let r=t.visibility??(t.autoYes?"private":await Gl({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),i=await Ul(t.folderPath);return Hl(t.folderPath),{newRemoteUrl:Et({folder:t.folderPath,name:n,visibility:r,org:t.org}).httpsUrl,backupPath:i}}import{spawnSync as Dl}from"child_process";var Nn=5e3;function Fl(t){let e=t.toLowerCase();return e.includes("authentication")||e.includes("could not read username")||e.includes("permission denied")||e.includes("403")||e.includes("access denied")||e.includes("repository not found")?"no-access":e.includes("404")||e.includes("does not exist")?"not-found":e.includes("could not resolve host")||e.includes("network")||e.includes("connection refused")||e.includes("connection timed out")?"network":"unknown"}function Rt(t){let e=Dl("git",["ls-remote","--exit-code",t,"HEAD"],{encoding:"utf8",timeout:Nn,stdio:["ignore","pipe","pipe"]});if(e.error){let i=e.error;return i.code==="ENOENT"?{ok:!1,reason:"network",detail:"git binary kh\xF4ng t\xECm th\u1EA5y \u2014 c\xE0i git r\u1ED3i retry"}:i.code==="ETIMEDOUT"?{ok:!1,reason:"timeout",detail:`git ls-remote > ${Nn/1e3}s`}:{ok:!1,reason:"unknown",detail:i.message.slice(0,300)}}if(e.status===0)return{ok:!0};if(e.signal==="SIGTERM")return{ok:!1,reason:"timeout",detail:`git ls-remote > ${Nn/1e3}s`};let n=(e.stderr||"").trim();return{ok:!1,reason:Fl(n),detail:n.slice(0,300)}}var Zt=class extends Error{constructor(e){super(e),this.name="RemoteAccessAbortedError"}};function Bl(){let t=Hi("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function Wl(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=Hi("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&o.warn(`gh auth login exit ${t.status}. B\u1EA1n c\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function zl(t,e,n){switch(t){case"no-access":return n?`Repo c\xF3 th\u1EC3 private v\xE0 account '${n}' kh\xF4ng c\xF3 quy\u1EC1n access. Switch sang account c\xF3 quy\u1EC1n, ho\u1EB7c xin invite t\u1EEB owner repo.`:"Repo c\xF3 th\u1EC3 private v\xE0 gh CLI ch\u01B0a login. Login tr\u01B0\u1EDBc r\u1ED3i retry.";case"not-found":return`URL c\xF3 th\u1EC3 sai ch\xEDnh t\u1EA3, ho\u1EB7c repo \u0111\xE3 b\u1ECB x\xF3a/rename. Nh\u1EADp l\u1EA1i URL \u0111\xFAng. Check: ${e}`;case"network":return"Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c GitHub. Check m\u1EA1ng / VPN / firewall.";case"timeout":return"Network ch\u1EADm > 5s. Check m\u1EA1ng r\u1ED3i retry.";default:return"L\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh. URL c\xF3 th\u1EC3 sai \u2014 nh\u1EADp l\u1EA1i, ho\u1EB7c check gh auth status."}}function ql(t){let e=t.trim();return e?/^https?:\/\/[\w.@/-]+$/.test(e)||/^git@[\w.-]+:[\w./-]+\.git$/.test(e)||/^[\w.-]+\/[\w.-]+$/.test(e):!1}async function Ue(t){let e=t.url,n=t.initialReason,r=t.initialDetail;for(;;){let i=Bl();o.warn(`Kh\xF4ng truy c\u1EADp \u0111\u01B0\u1EE3c ${e}`),o.dim(` L\xFD do: ${n}${r?` \u2014 ${r.slice(0,150)}`:""}`),o.info(zl(n,e,i)),i&&o.dim(` gh CLI hi\u1EC7n \u0111ang login: ${i}`);let s=[{name:"Nh\u1EADp l\u1EA1i URL \u0111\xFAng (recommended khi URL sai ch\xEDnh t\u1EA3)",value:"re-input-url"},{name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch"},{name:"T\xF4i v\u1EEBa fix (accept invite / s\u1EEDa permission) \u2014 retry verify",value:"retry"}];t.folderPath&&s.push({name:"Reset folder & t\u1EA1o repo M\u1EDAI d\u01B0\u1EDBi account c\u1EE7a t\xF4i (backup .git c\u0169)",value:"reset-folder"}),s.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i 'avatar init' sau",value:"abort"});let a=await Vl({message:"C\xE1ch x\u1EED l\xFD?",choices:s});if(a==="abort")throw new Zt(`User ch\u1ECDn t\u1EA1m ng\u01B0ng. Fix access ${e} r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar init'.`);if(a==="reset-folder"){if(!t.folderPath){o.warn("Reset folder c\u1EA7n folderPath \u2014 internal error.");continue}try{let l=await Ui({folderPath:t.folderPath,visibility:t.defaultVisibility});return o.success(`Folder \u0111\xE3 reset. Backup t\u1EA1i: ${l.backupPath}`),o.success(`Remote m\u1EDBi: ${l.newRemoteUrl}`),{resolvedUrl:l.newRemoteUrl}}catch(l){o.warn(`Reset folder th\u1EA5t b\u1EA1i: ${l.message}`);continue}}a==="re-input-url"&&(e=(await Kl({message:"URL git remote (https://github.com/owner/repo.git ho\u1EB7c git@github.com:owner/repo.git):",default:e,validate:u=>ql(u)||"URL kh\xF4ng \u0111\xFAng format git remote"})).trim()),a==="switch"&&Wl(),o.info(`Verify remote l\u1EA1i: ${e}...`);let c=Rt(e);if(c.ok)return o.success(`Remote accessible: ${e}`),{resolvedUrl:e};n=c.reason??"unknown",r=c.detail}}import{readdirSync as uu}from"fs";import{select as pu}from"@inquirer/prompts";import{simpleGit as Bi}from"simple-git";import{existsSync as Yl,statSync as Jl}from"fs";import{join as Xl}from"path";function He(t){let e=Xl(t,".git");if(!Yl(e))return!1;let n=Jl(e);return n.isDirectory()||n.isFile()}import{simpleGit as Zl}from"simple-git";var Di="chore: initial commit";async function Qt(t){let e=Zl({baseDir:t});await e.checkIsRepo().catch(()=>!1)||await e.init();try{await e.branch(["-M","main"])}catch{}await e.add(".");let r=await e.status();(await e.raw(["rev-list","-n","1","--all"]).catch(()=>"")).trim()||(r.files.length===0?await e.commit(Di,void 0,{"--allow-empty":null}):await e.commit(Di))}import{existsSync as Ql}from"fs";import{join as tu}from"path";var eu={node:["package.json"],python:["pyproject.toml","requirements.txt","setup.py","Pipfile"],go:["go.mod"],rust:["Cargo.toml"],java:["pom.xml","build.gradle","build.gradle.kts"],ruby:["Gemfile"]};function Fi(t){let e=[];for(let[n,r]of Object.entries(eu))r.some(i=>Ql(tu(t,i)))&&e.push(n);return e.length>0?e:["generic"]}import{readFileSync as nu}from"fs";import{dirname as ru,join as te}from"path";import{fileURLToPath as iu}from"url";var De=ru(iu(import.meta.url)),ou=[te(De,"templates","gitignore"),te(De,"..","templates","gitignore"),te(De,"..","..","src","templates","gitignore"),te(De,"..","src","templates","gitignore")],ee="# === avatar ===",pt="# === /avatar ===";function su(t){for(let e of ou)try{return nu(te(e,`${t}.txt`),"utf8")}catch{}throw new Error(`Kh\xF4ng t\xECm th\u1EA5y template gitignore cho stack "${t}"`)}function Ki(t){let n=["generic",...t.filter(r=>r!=="generic")].map(r=>`# --- ${r} ---
76
- ${su(r).trim()}`);return[ee,...n,pt,""].join(`
77
- `)}import{existsSync as au,readFileSync as cu,writeFileSync as Mn}from"fs";import{join as lu}from"path";function Vi(t,e){let n=lu(t,".gitignore");if(!au(n)){Mn(n,e,"utf8");return}let r=cu(n,"utf8"),i=r.indexOf(ee),s=r.indexOf(pt);if(i!==-1&&s!==-1&&s>i){let a=r.slice(0,i),c=r.slice(s+pt.length);Mn(n,`${a.trimEnd()}
75
+ Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let r=t.visibility??(t.autoYes?"private":await dl({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),i=await fl(t.folderPath);return hl(t.folderPath),{newRemoteUrl:Et({folder:t.folderPath,name:n,visibility:r,org:t.org}).httpsUrl,backupPath:i}}import{spawnSync as kl}from"child_process";var _n=5e3;function wl(t){let e=t.toLowerCase();return e.includes("authentication")||e.includes("could not read username")||e.includes("permission denied")||e.includes("403")||e.includes("access denied")||e.includes("repository not found")?"no-access":e.includes("404")||e.includes("does not exist")?"not-found":e.includes("could not resolve host")||e.includes("network")||e.includes("connection refused")||e.includes("connection timed out")?"network":"unknown"}function Rt(t){let e=kl("git",["ls-remote","--exit-code",t,"HEAD"],{encoding:"utf8",timeout:_n,stdio:["ignore","pipe","pipe"]});if(e.error){let i=e.error;return i.code==="ENOENT"?{ok:!1,reason:"network",detail:"git binary kh\xF4ng t\xECm th\u1EA5y \u2014 c\xE0i git r\u1ED3i retry"}:i.code==="ETIMEDOUT"?{ok:!1,reason:"timeout",detail:`git ls-remote > ${_n/1e3}s`}:{ok:!1,reason:"unknown",detail:i.message.slice(0,300)}}if(e.status===0)return{ok:!0};if(e.signal==="SIGTERM")return{ok:!1,reason:"timeout",detail:`git ls-remote > ${_n/1e3}s`};let n=(e.stderr||"").trim();return{ok:!1,reason:wl(n),detail:n.slice(0,300)}}var Xt=class extends Error{constructor(e){super(e),this.name="RemoteAccessAbortedError"}};function bl(){let t=Ni("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function xl(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=Ni("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&o.warn(`gh auth login exit ${t.status}. B\u1EA1n c\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function Sl(t,e,n){switch(t){case"no-access":return n?`Repo c\xF3 th\u1EC3 private v\xE0 account '${n}' kh\xF4ng c\xF3 quy\u1EC1n access. Switch sang account c\xF3 quy\u1EC1n, ho\u1EB7c xin invite t\u1EEB owner repo.`:"Repo c\xF3 th\u1EC3 private v\xE0 gh CLI ch\u01B0a login. Login tr\u01B0\u1EDBc r\u1ED3i retry.";case"not-found":return`URL c\xF3 th\u1EC3 sai ch\xEDnh t\u1EA3, ho\u1EB7c repo \u0111\xE3 b\u1ECB x\xF3a/rename. Nh\u1EADp l\u1EA1i URL \u0111\xFAng. Check: ${e}`;case"network":return"Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c GitHub. Check m\u1EA1ng / VPN / firewall.";case"timeout":return"Network ch\u1EADm > 5s. Check m\u1EA1ng r\u1ED3i retry.";default:return"L\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh. URL c\xF3 th\u1EC3 sai \u2014 nh\u1EADp l\u1EA1i, ho\u1EB7c check gh auth status."}}function Al(t){let e=t.trim();return e?/^https?:\/\/[\w.@/-]+$/.test(e)||/^git@[\w.-]+:[\w./-]+\.git$/.test(e)||/^[\w.-]+\/[\w.-]+$/.test(e):!1}async function Me(t){let e=t.url,n=t.initialReason,r=t.initialDetail;for(;;){let i=bl();o.warn(`Kh\xF4ng truy c\u1EADp \u0111\u01B0\u1EE3c ${e}`),o.dim(` L\xFD do: ${n}${r?` \u2014 ${r.slice(0,150)}`:""}`),o.info(Sl(n,e,i)),i&&o.dim(` gh CLI hi\u1EC7n \u0111ang login: ${i}`);let s=[{name:"Nh\u1EADp l\u1EA1i URL \u0111\xFAng (recommended khi URL sai ch\xEDnh t\u1EA3)",value:"re-input-url"},{name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch"},{name:"T\xF4i v\u1EEBa fix (accept invite / s\u1EEDa permission) \u2014 retry verify",value:"retry"}];t.folderPath&&s.push({name:"Reset folder & t\u1EA1o repo M\u1EDAI d\u01B0\u1EDBi account c\u1EE7a t\xF4i (backup .git c\u0169)",value:"reset-folder"}),s.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i 'avatar init' sau",value:"abort"});let a=await vl({message:"C\xE1ch x\u1EED l\xFD?",choices:s});if(a==="abort")throw new Xt(`User ch\u1ECDn t\u1EA1m ng\u01B0ng. Fix access ${e} r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar init'.`);if(a==="reset-folder"){if(!t.folderPath){o.warn("Reset folder c\u1EA7n folderPath \u2014 internal error.");continue}try{let l=await Oi({folderPath:t.folderPath,visibility:t.defaultVisibility});return o.success(`Folder \u0111\xE3 reset. Backup t\u1EA1i: ${l.backupPath}`),o.success(`Remote m\u1EDBi: ${l.newRemoteUrl}`),{resolvedUrl:l.newRemoteUrl}}catch(l){o.warn(`Reset folder th\u1EA5t b\u1EA1i: ${l.message}`);continue}}a==="re-input-url"&&(e=(await yl({message:"URL git remote (https://github.com/owner/repo.git ho\u1EB7c git@github.com:owner/repo.git):",default:e,validate:u=>Al(u)||"URL kh\xF4ng \u0111\xFAng format git remote"})).trim()),a==="switch"&&xl(),o.info(`Verify remote l\u1EA1i: ${e}...`);let c=Rt(e);if(c.ok)return o.success(`Remote accessible: ${e}`),{resolvedUrl:e};n=c.reason??"unknown",r=c.detail}}import{readdirSync as Ul}from"fs";import{select as Dl}from"@inquirer/prompts";import{simpleGit as Gi}from"simple-git";import{existsSync as Cl,statSync as Pl}from"fs";import{join as $l}from"path";function je(t){let e=$l(t,".git");if(!Cl(e))return!1;let n=Pl(e);return n.isDirectory()||n.isFile()}import{simpleGit as El}from"simple-git";var Mi="chore: initial commit";async function Zt(t){let e=El({baseDir:t});await e.checkIsRepo().catch(()=>!1)||await e.init();try{await e.branch(["-M","main"])}catch{}await e.add(".");let r=await e.status();(await e.raw(["rev-list","-n","1","--all"]).catch(()=>"")).trim()||(r.files.length===0?await e.commit(Mi,void 0,{"--allow-empty":null}):await e.commit(Mi))}import{existsSync as Rl}from"fs";import{join as Tl}from"path";var _l={node:["package.json"],python:["pyproject.toml","requirements.txt","setup.py","Pipfile"],go:["go.mod"],rust:["Cargo.toml"],java:["pom.xml","build.gradle","build.gradle.kts"],ruby:["Gemfile"]};function ji(t){let e=[];for(let[n,r]of Object.entries(_l))r.some(i=>Rl(Tl(t,i)))&&e.push(n);return e.length>0?e:["generic"]}import{readFileSync as Il}from"fs";import{dirname as Ol,join as Qt}from"path";import{fileURLToPath as Nl}from"url";var Le=Ol(Nl(import.meta.url)),Ml=[Qt(Le,"templates","gitignore"),Qt(Le,"..","templates","gitignore"),Qt(Le,"..","..","src","templates","gitignore"),Qt(Le,"..","src","templates","gitignore")],te="# === avatar ===",pt="# === /avatar ===";function jl(t){for(let e of Ml)try{return Il(Qt(e,`${t}.txt`),"utf8")}catch{}throw new Error(`Kh\xF4ng t\xECm th\u1EA5y template gitignore cho stack "${t}"`)}function Li(t){let n=["generic",...t.filter(r=>r!=="generic")].map(r=>`# --- ${r} ---
76
+ ${jl(r).trim()}`);return[te,...n,pt,""].join(`
77
+ `)}import{existsSync as Ll,readFileSync as Hl,writeFileSync as In}from"fs";import{join as Gl}from"path";function Hi(t,e){let n=Gl(t,".gitignore");if(!Ll(n)){In(n,e,"utf8");return}let r=Hl(n,"utf8"),i=r.indexOf(te),s=r.indexOf(pt);if(i!==-1&&s!==-1&&s>i){let a=r.slice(0,i),c=r.slice(s+pt.length);In(n,`${a.trimEnd()}
78
78
 
79
- ${e}${c.trimStart()}`,"utf8");return}Mn(n,`${r.trimEnd()}
79
+ ${e}${c.trimStart()}`,"utf8");return}In(n,`${r.trimEnd()}
80
80
 
81
- ${e}`,"utf8")}var ne=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};async function mu(t){return He(t)?(await Bi({baseDir:t}).status()).isClean()?"clean":"dirty":uu(t).filter(s=>s!==".git").length===0?"empty":"untracked-only"}async function du(t,e){return e.presetStrategy?e.presetStrategy:e.autoYes?"stash":t==="empty"||t==="clean"?"commit-all":await pu({message:t==="dirty"?"Folder c\xF3 changes ch\u01B0a commit. C\xE1ch x\u1EED l\xFD:":"Folder c\xF3 file ch\u01B0a version. C\xE1ch x\u1EED l\xFD:",choices:[{value:"stash",name:"1. Stash changes \u2192 bootstrap \u2192 restore (KHUY\u1EBEN NGH\u1ECA)"},{value:"commit-all",name:"2. Commit to\xE0n b\u1ED9 v\xE0o initial commit (legacy v1.1.6)"},{value:"skip",name:"3. Skip \u2014 t\xF4i commit th\u1EE7 c\xF4ng r\u1ED3i ch\u1EA1y l\u1EA1i"},{value:"branch",name:"4. Commit v\xE0o branch ri\xEAng `avatar/init` (main gi\u1EEF s\u1EA1ch)"}],default:"stash"})}async function gu(t,e){let n=await t.status();return n.isClean()&&n.not_added.length===0?!1:(await t.stash(["push","--include-untracked","-m",e]),o.info(`Stashed changes: ${e}`),!0)}async function fu(t,e){try{await t.stash(["pop"]),o.success(`Restored stash: ${e}`)}catch(n){o.warn("Restore stash conflict \u2014 files c\xF3 Avatar t\u1EA1o v\xE0 stash c\u1EE7a user xung \u0111\u1ED9t. Stash gi\u1EEF t\u1EA1i ref stash@{0}."),o.warn("Resolve: git stash show -p stash@{0} \u2192 fix conflict \u2192 git stash drop"),o.dim(`Detail: ${n.message}`)}}async function hu(t){try{let n=(await t.revparse(["--abbrev-ref","HEAD"])).trim();return n==="HEAD"?"main":n}catch{return"main"}}async function Fe(t){let e=Fi(t);o.info(`Tech stack: ${e.join(", ")}`),Vi(t,Ki(e)),o.success(".gitignore \u0111\xE3 ghi (Avatar block)")}async function ku(t,e){let n=Bi({baseDir:t});switch(e){case"skip":throw new ne("Init aborted. Commit th\u1EE7 c\xF4ng changes hi\u1EC7n t\u1EA1i r\u1ED3i ch\u1EA1y l\u1EA1i `avatar init`.");case"stash":{let r=`avatar-init-backup-${Date.now()}`;He(t)||(await n.init(),await n.branch(["-M","main"]).catch(()=>{})),(await n.raw(["rev-list","-n","1","--all"]).catch(()=>"")).trim()||await n.commit("chore: avatar baseline (pre-stash)",void 0,{"--allow-empty":null});let a=await gu(n,r);try{await Fe(t),await Qt(t)}finally{a&&await fu(n,r)}break}case"commit-all":{await Fe(t),await Qt(t);break}case"branch":{He(t)||(await n.init(),await n.branch(["-M","main"]));let i=await hu(n);try{await n.checkoutLocalBranch("avatar/init")}catch{await n.checkout("avatar/init")}await Fe(t),await Qt(t);try{await n.checkout(i),o.info(`Avatar init committed \u1EDF branch 'avatar/init'. Switch back v\u1EC1 '${i}'. Merge khi s\u1EB5n s\xE0ng: git merge avatar/init`)}catch{o.warn(`Kh\xF4ng switch v\u1EC1 '${i}' \u0111\u01B0\u1EE3c \u2014 \u1EDF l\u1EA1i branch 'avatar/init'. Switch tay sau.`)}break}}}async function jn(t,e={}){let n=await mu(t);if(o.info(`Folder state: ${n}`),n==="empty"||n==="clean"){await Fe(t),n==="empty"&&await Qt(t),await w("bootstrap",`state=${n},strategy=auto`);return}let r=await du(n,e);await ku(t,r),await w("bootstrap",`state=${n},strategy=${r}`)}import{join as Zi}from"path";import{spawnSync as Ln}from"child_process";import{confirm as wu,select as yu}from"@inquirer/prompts";import vu from"boxen";function zi(t){return t.match(/github\.com[/:]([^/]+\/[^/]+?)(?:\.git)?$/)?.[1]??null}function Wi(t){return Ln("gh",["api",`repos/${t}`],{stdio:"ignore"}).status===0}function Gn(){let t=Ln("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function bu(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=Ln("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&o.warn(`gh auth login exit ${t.status}. C\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}async function xu(t){if(await wu({message:"Copy th\xF4ng tin (GitHub username + email) v\xE0o clipboard \u0111\u1EC3 d\xE1n v\xE0o Slack/email?",default:!0}))try{let{default:n}=await import("clipboardy");await n.write(t),o.success("\u0110\xE3 copy v\xE0o clipboard")}catch(n){o.dim(`Copy clipboard fail: ${n.message}`)}}function Au(t,e,n){let r=[`${p.red("\u26D4 KH\xD4NG C\xD3 QUY\u1EC0N ACCESS")}`,"",`Repo: ${p.bold(t)}`,"","B\u1EA1n c\u1EA7n \u0111\u01B0\u1EE3c admin add v\xE0o org \u0111\u1EC3 pull team-ai-pack.","",`${p.dim("Th\xF4ng tin g\u1EEDi admin:")}`,` GitHub username: ${p.cyan(e??"(ch\u01B0a gh auth \u2014 ch\u1EA1y: gh auth login)")}`,` NAL email: ${p.cyan(n??"(ch\u01B0a avatar login \u2014 ch\u1EA1y: avatar login)")}`,` Date: ${new Date().toISOString().split("T")[0]}`,"",`${p.dim("Li\xEAn h\u1EC7:")} luke@nal.vn (Slack #avatar-setup)`];process.stdout.write(`${vu(r.join(`
81
+ ${e}`,"utf8")}var ee=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};async function Fl(t){return je(t)?(await Gi({baseDir:t}).status()).isClean()?"clean":"dirty":Ul(t).filter(s=>s!==".git").length===0?"empty":"untracked-only"}async function Vl(t,e){return e.presetStrategy?e.presetStrategy:e.autoYes?"stash":t==="empty"||t==="clean"?"commit-all":await Dl({message:t==="dirty"?"Folder c\xF3 changes ch\u01B0a commit. C\xE1ch x\u1EED l\xFD:":"Folder c\xF3 file ch\u01B0a version. C\xE1ch x\u1EED l\xFD:",choices:[{value:"stash",name:"1. Stash changes \u2192 bootstrap \u2192 restore (KHUY\u1EBEN NGH\u1ECA)"},{value:"commit-all",name:"2. Commit to\xE0n b\u1ED9 v\xE0o initial commit (legacy v1.1.6)"},{value:"skip",name:"3. Skip \u2014 t\xF4i commit th\u1EE7 c\xF4ng r\u1ED3i ch\u1EA1y l\u1EA1i"},{value:"branch",name:"4. Commit v\xE0o branch ri\xEAng `avatar/init` (main gi\u1EEF s\u1EA1ch)"}],default:"stash"})}async function Kl(t,e){let n=await t.status();return n.isClean()&&n.not_added.length===0?!1:(await t.stash(["push","--include-untracked","-m",e]),o.info(`Stashed changes: ${e}`),!0)}async function Bl(t,e){try{await t.stash(["pop"]),o.success(`Restored stash: ${e}`)}catch(n){o.warn("Restore stash conflict \u2014 files c\xF3 Avatar t\u1EA1o v\xE0 stash c\u1EE7a user xung \u0111\u1ED9t. Stash gi\u1EEF t\u1EA1i ref stash@{0}."),o.warn("Resolve: git stash show -p stash@{0} \u2192 fix conflict \u2192 git stash drop"),o.dim(`Detail: ${n.message}`)}}async function Wl(t){try{let n=(await t.revparse(["--abbrev-ref","HEAD"])).trim();return n==="HEAD"?"main":n}catch{return"main"}}async function He(t){let e=ji(t);o.info(`Tech stack: ${e.join(", ")}`),Hi(t,Li(e)),o.success(".gitignore \u0111\xE3 ghi (Avatar block)")}async function zl(t,e){let n=Gi({baseDir:t});switch(e){case"skip":throw new ee("Init aborted. Commit th\u1EE7 c\xF4ng changes hi\u1EC7n t\u1EA1i r\u1ED3i ch\u1EA1y l\u1EA1i `avatar init`.");case"stash":{let r=`avatar-init-backup-${Date.now()}`;je(t)||(await n.init(),await n.branch(["-M","main"]).catch(()=>{})),(await n.raw(["rev-list","-n","1","--all"]).catch(()=>"")).trim()||await n.commit("chore: avatar baseline (pre-stash)",void 0,{"--allow-empty":null});let a=await Kl(n,r);try{await He(t),await Zt(t)}finally{a&&await Bl(n,r)}break}case"commit-all":{await He(t),await Zt(t);break}case"branch":{je(t)||(await n.init(),await n.branch(["-M","main"]));let i=await Wl(n);try{await n.checkoutLocalBranch("avatar/init")}catch{await n.checkout("avatar/init")}await He(t),await Zt(t);try{await n.checkout(i),o.info(`Avatar init committed \u1EDF branch 'avatar/init'. Switch back v\u1EC1 '${i}'. Merge khi s\u1EB5n s\xE0ng: git merge avatar/init`)}catch{o.warn(`Kh\xF4ng switch v\u1EC1 '${i}' \u0111\u01B0\u1EE3c \u2014 \u1EDF l\u1EA1i branch 'avatar/init'. Switch tay sau.`)}break}}}async function On(t,e={}){let n=await Fl(t);if(o.info(`Folder state: ${n}`),n==="empty"||n==="clean"){await He(t),n==="empty"&&await Zt(t),await v("bootstrap",`state=${n},strategy=auto`);return}let r=await Vl(n,e);await zl(t,r),await v("bootstrap",`state=${n},strategy=${r}`)}import{join as Wi}from"path";import{spawnSync as Mn}from"child_process";import{confirm as ql,select as Jl}from"@inquirer/prompts";import Yl from"boxen";function Di(t){return t.match(/github\.com[/:]([^/]+\/[^/]+?)(?:\.git)?$/)?.[1]??null}function Ui(t){return Mn("gh",["api",`repos/${t}`],{stdio:"ignore"}).status===0}function Nn(){let t=Mn("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function Xl(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=Mn("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&o.warn(`gh auth login exit ${t.status}. C\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}async function Zl(t){if(await ql({message:"Copy th\xF4ng tin (GitHub username + email) v\xE0o clipboard \u0111\u1EC3 d\xE1n v\xE0o Slack/email?",default:!0}))try{let{default:n}=await import("clipboardy");await n.write(t),o.success("\u0110\xE3 copy v\xE0o clipboard")}catch(n){o.dim(`Copy clipboard fail: ${n.message}`)}}function Ql(t,e,n){let r=[`${p.red("\u26D4 KH\xD4NG C\xD3 QUY\u1EC0N ACCESS")}`,"",`Repo: ${p.bold(t)}`,"","B\u1EA1n c\u1EA7n \u0111\u01B0\u1EE3c admin add v\xE0o org \u0111\u1EC3 pull team-ai-pack.","",`${p.dim("Th\xF4ng tin g\u1EEDi admin:")}`,` GitHub username: ${p.cyan(e??"(ch\u01B0a gh auth \u2014 ch\u1EA1y: gh auth login)")}`,` NAL email: ${p.cyan(n??"(ch\u01B0a avatar login \u2014 ch\u1EA1y: avatar login)")}`,` Date: ${new Date().toISOString().split("T")[0]}`,"",`${p.dim("Li\xEAn h\u1EC7:")} luke@nal.vn (Slack #avatar-setup)`];process.stdout.write(`${Yl(r.join(`
82
82
  `),{padding:1,borderColor:"red",borderStyle:"round"})}
83
- `)}function Su(t,e,n){return[`Request access ${t} (NAL)`,"",`GitHub username: ${e??"(ch\u01B0a gh auth \u2014 ch\u1EA1y: gh auth login)"}`,`NAL email: ${n??"(ch\u01B0a avatar login \u2014 ch\u1EA1y: avatar login)"}`,`Date: ${new Date().toISOString().split("T")[0]}`].join(`
84
- `)}async function qi(t){if(Wi(t.repoSlug))return!0;let e=Gn();for(Au(t.repoSlug,e,t.ssoEmail??null),await xu(Su(t.repoSlug,e,t.ssoEmail??null));;){let n=Gn(),i=await yu({message:"C\xE1ch x\u1EED l\xFD?",choices:[{name:`\u0110\xE3 \u0111\u01B0\u1EE3c grant access v\u1EDBi GitHub username '${n??"(ch\u01B0a gh auth)"}' \u2014 ki\u1EC3m tra l\u1EA1i`,value:"retry-same"},{name:"\u0110\xE3 grant v\u1EDBi GitHub account kh\xE1c \u2014 switch gh (m\u1EDF browser)",value:"switch-account"},{name:"T\u1EA1m ng\u01B0ng \u2014 ch\u1EA1y l\u1EA1i 'avatar init' sau",value:"abort"}]});if(i==="abort")return o.dim("T\u1EA1m ng\u01B0ng. Ch\u1EA1y l\u1EA1i 'avatar init' sau khi \u0111\xE3 accept invite t\u1EEB GitHub."),!1;if(i==="switch-account"&&bu(),o.info("Ki\u1EC3m tra access..."),Wi(t.repoSlug)){let s=Gn();return o.success(`\u0110\xE3 c\xF3 access v\u1EDBi '${s??"(unknown)"}' \u2014 ti\u1EBFp t\u1EE5c.`),!0}o.warn(`V\u1EABn ch\u01B0a c\xF3 access v\u1EDBi account '${n??"(unknown)"}'. \u0110\u1EA3m b\u1EA3o \u0111\xE3 accept email invite ho\u1EB7c switch \u0111\xFAng account.`)}}var Cu=/^v?(\d+)\.(\d+)\.(\d+)(?:-([0-9A-Za-z.-]+))?$/;function Pu(t){let e=t.match(Cu);if(!e)return null;let[,n,r,i,s]=e;return{raw:t,major:Number.parseInt(n??"0",10),minor:Number.parseInt(r??"0",10),patch:Number.parseInt(i??"0",10),prerelease:s??null}}function q(t,e=!1){let n=t.map(Pu).filter(r=>r!==null).filter(r=>e||r.prerelease===null);return n.length===0?null:(n.sort((r,i)=>r.major!==i.major?r.major-i.major:r.minor!==i.minor?r.minor-i.minor:r.patch-i.patch),n[n.length-1]?.raw??null)}var Yi="git@github.com:nalvn/team-ai-pack.git",$u=/^(git@github\.com:|https:\/\/github\.com\/)[A-Za-z0-9._-]+\/[A-Za-z0-9._-]+\.git$/,Un=class extends Error{url;constructor(e){super(`AVATAR_TEAM_PACK_REPO_URL kh\xF4ng h\u1EE3p l\u1EC7: ${e}
83
+ `)}function tu(t,e,n){return[`Request access ${t} (NAL)`,"",`GitHub username: ${e??"(ch\u01B0a gh auth \u2014 ch\u1EA1y: gh auth login)"}`,`NAL email: ${n??"(ch\u01B0a avatar login \u2014 ch\u1EA1y: avatar login)"}`,`Date: ${new Date().toISOString().split("T")[0]}`].join(`
84
+ `)}async function Fi(t){if(Ui(t.repoSlug))return!0;let e=Nn();for(Ql(t.repoSlug,e,t.ssoEmail??null),await Zl(tu(t.repoSlug,e,t.ssoEmail??null));;){let n=Nn(),i=await Jl({message:"C\xE1ch x\u1EED l\xFD?",choices:[{name:`\u0110\xE3 \u0111\u01B0\u1EE3c grant access v\u1EDBi GitHub username '${n??"(ch\u01B0a gh auth)"}' \u2014 ki\u1EC3m tra l\u1EA1i`,value:"retry-same"},{name:"\u0110\xE3 grant v\u1EDBi GitHub account kh\xE1c \u2014 switch gh (m\u1EDF browser)",value:"switch-account"},{name:"T\u1EA1m ng\u01B0ng \u2014 ch\u1EA1y l\u1EA1i 'avatar init' sau",value:"abort"}]});if(i==="abort")return o.dim("T\u1EA1m ng\u01B0ng. Ch\u1EA1y l\u1EA1i 'avatar init' sau khi \u0111\xE3 accept invite t\u1EEB GitHub."),!1;if(i==="switch-account"&&Xl(),o.info("Ki\u1EC3m tra access..."),Ui(t.repoSlug)){let s=Nn();return o.success(`\u0110\xE3 c\xF3 access v\u1EDBi '${s??"(unknown)"}' \u2014 ti\u1EBFp t\u1EE5c.`),!0}o.warn(`V\u1EABn ch\u01B0a c\xF3 access v\u1EDBi account '${n??"(unknown)"}'. \u0110\u1EA3m b\u1EA3o \u0111\xE3 accept email invite ho\u1EB7c switch \u0111\xFAng account.`)}}var eu=/^v?(\d+)\.(\d+)\.(\d+)(?:-([0-9A-Za-z.-]+))?$/;function nu(t){let e=t.match(eu);if(!e)return null;let[,n,r,i,s]=e;return{raw:t,major:Number.parseInt(n??"0",10),minor:Number.parseInt(r??"0",10),patch:Number.parseInt(i??"0",10),prerelease:s??null}}function z(t,e=!1){let n=t.map(nu).filter(r=>r!==null).filter(r=>e||r.prerelease===null);return n.length===0?null:(n.sort((r,i)=>r.major!==i.major?r.major-i.major:r.minor!==i.minor?r.minor-i.minor:r.patch-i.patch),n[n.length-1]?.raw??null)}var Vi="git@github.com:nalvn/team-ai-pack.git",ru=/^(git@github\.com:|https:\/\/github\.com\/)[A-Za-z0-9._-]+\/[A-Za-z0-9._-]+\.git$/,jn=class extends Error{url;constructor(e){super(`AVATAR_TEAM_PACK_REPO_URL kh\xF4ng h\u1EE3p l\u1EC7: ${e}
85
85
  Y\xEAu c\u1EA7u: GitHub URL c\xF3 .git suffix, vd:
86
86
  git@github.com:owner/repo.git
87
87
  https://github.com/owner/repo.git
88
- L\xFD do block: ng\u0103n file:// (filesystem exfil) + URL malicious (git hook RCE).`),this.name="InvalidTeamPackUrlError",this.url=e}};function Ji(t){if(!$u.test(t))throw new Un(t)}function Hn(){let t=process.env.AVATAR_TEAM_PACK_REPO_URL;return t?(Ji(t),t):(Ji(Yi),Yi)}var nw=Hn(),R=".claude/pack",mt=class extends Error{constructor(e){super(e),this.name="TeamPackAccessAbortedError"}},Xi="main";async function Qi(t,e,n,r=!1){let i=Hn(),s=zi(i);if(s&&!await qi({repoSlug:s,ssoEmail:n}))throw new mt("User ch\u1ECDn t\u1EA1m ng\u01B0ng. Ch\u1EA1y l\u1EA1i 'avatar init' sau khi \u0111\u01B0\u1EE3c add v\xE0o org.");try{await Xr(i,R,t)}catch(u){let m=u instanceof Error?u.message:String(u);throw(m.includes("Repository not found")||m.includes("not found"))&&o.error(`Repo team-ai-pack kh\xF4ng t\u1ED3n t\u1EA1i: ${i}
88
+ L\xFD do block: ng\u0103n file:// (filesystem exfil) + URL malicious (git hook RCE).`),this.name="InvalidTeamPackUrlError",this.url=e}};function Ki(t){if(!ru.test(t))throw new jn(t)}function Ln(){let t=process.env.AVATAR_TEAM_PACK_REPO_URL;return t?(Ki(t),t):(Ki(Vi),Vi)}var Ak=Ln(),R=".claude/pack",mt=class extends Error{constructor(e){super(e),this.name="TeamPackAccessAbortedError"}},Bi="main";async function zi(t,e,n,r=!1){let i=Ln(),s=Di(i);if(s&&!await Fi({repoSlug:s,ssoEmail:n}))throw new mt("User ch\u1ECDn t\u1EA1m ng\u01B0ng. Ch\u1EA1y l\u1EA1i 'avatar init' sau khi \u0111\u01B0\u1EE3c add v\xE0o org.");try{await Br(i,R,t)}catch(u){let m=u instanceof Error?u.message:String(u);throw(m.includes("Repository not found")||m.includes("not found"))&&o.error(`Repo team-ai-pack kh\xF4ng t\u1ED3n t\u1EA1i: ${i}
89
89
  C\xE1ch fix:
90
90
  1. T\u1EA1o repo: gh repo create <owner>/team-ai-pack --private --add-readme
91
91
  2. Ho\u1EB7c override URL: export AVATAR_TEAM_PACK_REPO_URL=<url-repo-c\u1EE7a-b\u1EA1n>
92
- 3. Ho\u1EB7c d\xF9ng flag --skip-team-pack`),u}if(e)return await Vt(R,e,t),{pinnedTag:e};if(r)return await $e(R,Xi,t),{pinnedTag:`${Xi} (HEAD)`};let a=Zi(t,R),c=await B(a),l=q(c);return l&&await Vt(R,l,t),{pinnedTag:l}}async function Ke(t){let e=Zi(t,R),n=await Ee(e);return n||(await Pt(e)).slice(0,7)}import{basename as hp,join as kp,resolve as Ze}from"path";import{input as ie,select as wp}from"@inquirer/prompts";import{spawnSync as to}from"child_process";import{select as Eu}from"@inquirer/prompts";function Ru(t){let e=t.toLowerCase();return e.includes("permission denied (publickey)")||e.includes("publickey)")||e.includes("ssh: could not resolve")||e.includes("host key verification failed")}function Tu(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=to("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&o.warn(`gh auth login exit ${t.status}. C\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function Iu(){o.info("M\u1EDF trang GitHub Settings \u2192 SSH Keys..."),to("open",["https://github.com/settings/keys"],{stdio:"ignore"}).status!==0&&o.info("URL: https://github.com/settings/keys")}async function _u(){return await Eu({message:"SSH permission denied. C\xE1ch x\u1EED l\xFD?",choices:[{name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch"},{name:"D\xF9ng HTTPS thay SSH (override URL b\u1EB1ng env AVATAR_TEAM_PACK_REPO_URL)",value:"https"},{name:"T\xF4i v\u1EEBa add SSH key l\xEAn GitHub \u2014 retry",value:"retry"},{name:"B\u1ECF qua team-ai-pack (d\xF9ng avatar sync sau)",value:"skip"},{name:"T\u1EA1m ng\u01B0ng init \u2014 fix SSH key tay r\u1ED3i ch\u1EA1y l\u1EA1i",value:"abort"}]})}async function Ve(t,e,n,r=!1){for(;;)try{return{pinnedTag:(await Qi(t,e,n,r)).pinnedTag,skipped:!1}}catch(i){if(i instanceof mt)throw i;let s=i instanceof Error?i.message:String(i);if(Ru(s)){o.warn("Pull team-ai-pack th\u1EA5t b\u1EA1i: SSH permission denied (publickey)."),o.dim(" \u2192 M\xE1y n\xE0y ch\u01B0a c\xF3 SSH key \u0111\u01B0\u1EE3c register tr\xEAn GitHub, ho\u1EB7c key thu\u1ED9c account kh\xE1c.");let c=await _u();if(c==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc pull team-ai-pack. Fix SSH key (https://github.com/settings/keys) r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar init'.");if(c==="skip")return o.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng kh\xF4ng c\xF3 shared knowledge. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};if(c==="switch"){Tu();continue}if(c==="https"){process.env.AVATAR_TEAM_PACK_REPO_URL="https://github.com/nalvn/team-ai-pack.git",o.info("Override URL sang HTTPS. L\u01B0u \xFD: HTTPS c\xF3 th\u1EC3 fail 403 n\u1EBFu read-only role."),Iu();continue}continue}let a=await T({taskName:"Pull team-ai-pack submodule",reason:s,allowSkip:!0,hint:"Network glitch? Retry th\u01B0\u1EDDng work. N\u1EBFu skip, d\xF9ng `avatar sync` sau \u0111\u1EC3 pull pack."});if(a==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc pull team-ai-pack.");if(a==="skip")return o.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng kh\xF4ng c\xF3 shared knowledge. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0}}}k();import{spawnSync as Ou}from"child_process";function Be(){let t=Ou("gh",["auth","status"],{stdio:"ignore"});return t.error&&t.error.code==="ENOENT"?"not-installed":t.status===0?"authenticated":"not-authenticated"}import{spawnSync as Nu}from"child_process";function Mu(t){let e=rt();return Nu(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function eo(){let t=rt(),e=t==="darwin"?["brew"]:t==="win32"?["winget"]:t==="linux"?["apt","dnf","pacman"]:[];for(let n of e)if(Mu(n))return n;return null}import{spawnSync as ju}from"child_process";var Gu={brew:{cmd:"brew",args:["install","gh"]},apt:{cmd:"sudo",args:["apt-get","install","-y","gh"]},dnf:{cmd:"sudo",args:["dnf","install","-y","gh"]},pacman:{cmd:"sudo",args:["pacman","-S","--noconfirm","github-cli"]},winget:{cmd:"winget",args:["install","--id","GitHub.cli","-e","--silent"]}};function no(t){let e=Gu[t];o.info(`\u0110ang c\xE0i gh CLI qua ${t}...`);let n=ju(e.cmd,e.args,{stdio:"inherit"});if(n.status!==0)throw new Error(`C\xE0i gh CLI th\u1EA5t b\u1EA1i qua ${t} (exit ${n.status}). C\xE0i tay r\u1ED3i ch\u1EA1y l\u1EA1i.`);o.success("\u0110\xE3 c\xE0i gh CLI")}import{spawnSync as Lu}from"child_process";function ro(){if(Lu("gh",["auth","setup-git"],{stdio:"ignore"}).status!==0){o.warn("gh auth setup-git fail (non-fatal). N\u1EBFu git clone l\u1ED7i 128 \u2192 ch\u1EA1y th\u1EE7 c\xF4ng.");return}o.dim("Git credential helper \u0111\xE3 link v\u1EDBi gh token.")}import{spawnSync as Uu}from"child_process";function io(){o.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp GitHub qua gh CLI (browser s\u1EBD m\u1EDF)...");let t=Uu("gh",["auth","login","--hostname","github.com","--web","--git-protocol","ssh"],{stdio:"inherit"});if(t.status!==0)throw new Error(`gh auth login th\u1EA5t b\u1EA1i (exit ${t.status}). Th\u1EED 'gh auth login' tay.`);o.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp GitHub")}async function dt(t){for(;Be()==="not-installed";){o.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=eo();if(!e){if(await T({taskName:"Ph\xE1t hi\u1EC7n package manager",reason:"Kh\xF4ng t\xECm th\u1EA5y brew/apt/dnf/pacman/winget tr\xEAn m\xE1y.",allowSkip:!1,hint:"C\xE0i gh CLI tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{no(e)}catch(n){if(await T({taskName:`C\xE0i gh CLI qua ${e}`,reason:n.message,allowSkip:!1,hint:"C\xE0i tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry, ho\u1EB7c Abort."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;Be()==="not-authenticated";){o.warn("Ch\u01B0a \u0111\u0103ng nh\u1EADp GitHub.");try{io()}catch(e){if(await T({taskName:"\u0110\u0103ng nh\u1EADp GitHub qua gh",reason:e.message,allowSkip:!1,hint:"Th\u1EED l\u1EA1i \u2014 ch\u1ECDn c\xE1ch login kh\xE1c (browser vs token) khi gh prompt."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(Be()!=="authenticated"&&await T({taskName:"Verify gh auth",reason:"Sau gh auth login v\u1EABn b\xE1o not-authenticated.",allowSkip:!1,hint:"C\xF3 th\u1EC3 user cancel browser. Th\u1EED l\u1EA1i ho\u1EB7c abort."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(o.success("gh CLI s\u1EB5n s\xE0ng"),ro(),t){let e=Rt(t);return e.ok?(o.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await Ue({url:t,initialReason:e.reason??"unknown",initialDetail:e.detail})).resolvedUrl}}return{}}function Dn(t){if(t.preserveUncommitted)return"stash";if(!t.bootstrapStrategy)return;let e=["stash","commit-all","skip","branch"];if(e.includes(t.bootstrapStrategy))return t.bootstrapStrategy;throw new Error(`--bootstrap-strategy kh\xF4ng h\u1EE3p l\u1EC7: ${t.bootstrapStrategy}. Ch\u1ECDn: ${e.join(" | ")}`)}import{readFileSync as Hu}from"fs";import{dirname as Du,resolve as Fu}from"path";import{fileURLToPath as Ku}from"url";var Tt=null;function H(){if(Tt!==null)return Tt;let t=Du(Ku(import.meta.url));for(let e=0;e<5;e++){let n=Fu(t,...Array(e).fill(".."),"package.json");try{let r=Hu(n,"utf8"),i=JSON.parse(r);if(i.name==="@nalvietnam/avatar-cli"&&typeof i.version=="string")return Tt=i.version,Tt}catch{}}return Tt="unknown",Tt}function oo(t){let r=(t.trim().replace(/\/+$/,"").split(/[/:]/).pop()??"").replace(/\.git$/,"");return r?`avatar-${r.replace(/^avatar-/,"")}-workspace`:"avatar-client-workspace"}function Vu(t){return t?`
92
+ 3. Ho\u1EB7c d\xF9ng flag --skip-team-pack`),u}if(e)return await Vt(R,e,t),{pinnedTag:e};if(r)return await Se(R,Bi,t),{pinnedTag:`${Bi} (HEAD)`};let a=Wi(t,R),c=await K(a),l=z(c);return l&&await Vt(R,l,t),{pinnedTag:l}}async function Ge(t){let e=Wi(t,R),n=await Ae(e);return n||(await Pt(e)).slice(0,7)}import{basename as Wu,join as zu,resolve as qe}from"path";import{input as re,select as qu}from"@inquirer/prompts";import{spawnSync as qi}from"child_process";import{select as iu}from"@inquirer/prompts";function ou(t){let e=t.toLowerCase();return e.includes("permission denied (publickey)")||e.includes("publickey)")||e.includes("ssh: could not resolve")||e.includes("host key verification failed")}function su(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=qi("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&o.warn(`gh auth login exit ${t.status}. C\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function au(){o.info("M\u1EDF trang GitHub Settings \u2192 SSH Keys..."),qi("open",["https://github.com/settings/keys"],{stdio:"ignore"}).status!==0&&o.info("URL: https://github.com/settings/keys")}async function cu(){return await iu({message:"SSH permission denied. C\xE1ch x\u1EED l\xFD?",choices:[{name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch"},{name:"D\xF9ng HTTPS thay SSH (override URL b\u1EB1ng env AVATAR_TEAM_PACK_REPO_URL)",value:"https"},{name:"T\xF4i v\u1EEBa add SSH key l\xEAn GitHub \u2014 retry",value:"retry"},{name:"B\u1ECF qua team-ai-pack (d\xF9ng avatar sync sau)",value:"skip"},{name:"T\u1EA1m ng\u01B0ng init \u2014 fix SSH key tay r\u1ED3i ch\u1EA1y l\u1EA1i",value:"abort"}]})}async function Ue(t,e,n,r=!1){for(;;)try{return{pinnedTag:(await zi(t,e,n,r)).pinnedTag,skipped:!1}}catch(i){if(i instanceof mt)throw i;let s=i instanceof Error?i.message:String(i);if(ou(s)){o.warn("Pull team-ai-pack th\u1EA5t b\u1EA1i: SSH permission denied (publickey)."),o.dim(" \u2192 M\xE1y n\xE0y ch\u01B0a c\xF3 SSH key \u0111\u01B0\u1EE3c register tr\xEAn GitHub, ho\u1EB7c key thu\u1ED9c account kh\xE1c.");let c=await cu();if(c==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc pull team-ai-pack. Fix SSH key (https://github.com/settings/keys) r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar init'.");if(c==="skip")return o.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng kh\xF4ng c\xF3 shared knowledge. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};if(c==="switch"){su();continue}if(c==="https"){process.env.AVATAR_TEAM_PACK_REPO_URL="https://github.com/nalvn/team-ai-pack.git",o.info("Override URL sang HTTPS. L\u01B0u \xFD: HTTPS c\xF3 th\u1EC3 fail 403 n\u1EBFu read-only role."),au();continue}continue}let a=await T({taskName:"Pull team-ai-pack submodule",reason:s,allowSkip:!0,hint:"Network glitch? Retry th\u01B0\u1EDDng work. N\u1EBFu skip, d\xF9ng `avatar sync` sau \u0111\u1EC3 pull pack."});if(a==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc pull team-ai-pack.");if(a==="skip")return o.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng kh\xF4ng c\xF3 shared knowledge. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0}}}k();import{spawnSync as lu}from"child_process";function De(){let t=lu("gh",["auth","status"],{stdio:"ignore"});return t.error&&t.error.code==="ENOENT"?"not-installed":t.status===0?"authenticated":"not-authenticated"}import{spawnSync as uu}from"child_process";function pu(t){let e=nt();return uu(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function Ji(){let t=nt(),e=t==="darwin"?["brew"]:t==="win32"?["winget"]:t==="linux"?["apt","dnf","pacman"]:[];for(let n of e)if(pu(n))return n;return null}import{spawnSync as mu}from"child_process";var du={brew:{cmd:"brew",args:["install","gh"]},apt:{cmd:"sudo",args:["apt-get","install","-y","gh"]},dnf:{cmd:"sudo",args:["dnf","install","-y","gh"]},pacman:{cmd:"sudo",args:["pacman","-S","--noconfirm","github-cli"]},winget:{cmd:"winget",args:["install","--id","GitHub.cli","-e","--silent"]}};function Yi(t){let e=du[t];o.info(`\u0110ang c\xE0i gh CLI qua ${t}...`);let n=mu(e.cmd,e.args,{stdio:"inherit"});if(n.status!==0)throw new Error(`C\xE0i gh CLI th\u1EA5t b\u1EA1i qua ${t} (exit ${n.status}). C\xE0i tay r\u1ED3i ch\u1EA1y l\u1EA1i.`);o.success("\u0110\xE3 c\xE0i gh CLI")}import{spawnSync as gu}from"child_process";function Xi(){if(gu("gh",["auth","setup-git"],{stdio:"ignore"}).status!==0){o.warn("gh auth setup-git fail (non-fatal). N\u1EBFu git clone l\u1ED7i 128 \u2192 ch\u1EA1y th\u1EE7 c\xF4ng.");return}o.dim("Git credential helper \u0111\xE3 link v\u1EDBi gh token.")}import{spawnSync as fu}from"child_process";function Zi(){o.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp GitHub qua gh CLI (browser s\u1EBD m\u1EDF)...");let t=fu("gh",["auth","login","--hostname","github.com","--web","--git-protocol","ssh"],{stdio:"inherit"});if(t.status!==0)throw new Error(`gh auth login th\u1EA5t b\u1EA1i (exit ${t.status}). Th\u1EED 'gh auth login' tay.`);o.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp GitHub")}async function dt(t){for(;De()==="not-installed";){o.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=Ji();if(!e){if(await T({taskName:"Ph\xE1t hi\u1EC7n package manager",reason:"Kh\xF4ng t\xECm th\u1EA5y brew/apt/dnf/pacman/winget tr\xEAn m\xE1y.",allowSkip:!1,hint:"C\xE0i gh CLI tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{Yi(e)}catch(n){if(await T({taskName:`C\xE0i gh CLI qua ${e}`,reason:n.message,allowSkip:!1,hint:"C\xE0i tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry, ho\u1EB7c Abort."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;De()==="not-authenticated";){o.warn("Ch\u01B0a \u0111\u0103ng nh\u1EADp GitHub.");try{Zi()}catch(e){if(await T({taskName:"\u0110\u0103ng nh\u1EADp GitHub qua gh",reason:e.message,allowSkip:!1,hint:"Th\u1EED l\u1EA1i \u2014 ch\u1ECDn c\xE1ch login kh\xE1c (browser vs token) khi gh prompt."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(De()!=="authenticated"&&await T({taskName:"Verify gh auth",reason:"Sau gh auth login v\u1EABn b\xE1o not-authenticated.",allowSkip:!1,hint:"C\xF3 th\u1EC3 user cancel browser. Th\u1EED l\u1EA1i ho\u1EB7c abort."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(o.success("gh CLI s\u1EB5n s\xE0ng"),Xi(),t){let e=Rt(t);return e.ok?(o.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await Me({url:t,initialReason:e.reason??"unknown",initialDetail:e.detail})).resolvedUrl}}return{}}function Hn(t){if(t.preserveUncommitted)return"stash";if(!t.bootstrapStrategy)return;let e=["stash","commit-all","skip","branch"];if(e.includes(t.bootstrapStrategy))return t.bootstrapStrategy;throw new Error(`--bootstrap-strategy kh\xF4ng h\u1EE3p l\u1EC7: ${t.bootstrapStrategy}. Ch\u1ECDn: ${e.join(" | ")}`)}import{readFileSync as hu}from"fs";import{dirname as ku,resolve as wu}from"path";import{fileURLToPath as yu}from"url";var Tt=null;function G(){if(Tt!==null)return Tt;let t=ku(yu(import.meta.url));for(let e=0;e<5;e++){let n=wu(t,...Array(e).fill(".."),"package.json");try{let r=hu(n,"utf8"),i=JSON.parse(r);if(i.name==="@nalvietnam/avatar-cli"&&typeof i.version=="string")return Tt=i.version,Tt}catch{}}return Tt="unknown",Tt}function Qi(t){let r=(t.trim().replace(/\/+$/,"").split(/[/:]/).pop()??"").replace(/\.git$/,"");return r?`avatar-${r.replace(/^avatar-/,"")}-workspace`:"avatar-client-workspace"}function vu(t){return t?`
93
93
  ### \u{1F9E0} CODEBASE INTELLIGENCE \u2014 GitNexus
94
94
 
95
95
  Workspace c\xF3 GitNexus index t\u1EA1i \`.gitnexus/\` cung c\u1EA5p architectural awareness
@@ -117,47 +117,47 @@ Khi user c\u1EA7n regenerate wiki sau refactor l\u1EDBn \u2014 ch\u1EA1y:
117
117
  \`\`\`bash
118
118
  gitnexus wiki . --api-key <key> --base-url <url>
119
119
  \`\`\`
120
- `:""}function Fn(t){return{projectName:t.projectName,projectDescription:t.projectDescription,teamOwner:t.teamOwner,avatarVersion:H(),packVersion:t.packVersion,lastScan:new Date().toISOString(),mode:t.mode,gitnexusSection:Vu(t.gitnexusReady??!1)}}k();import{join as ze,relative as qu,resolve as lo}from"path";import{input as po,select as Yu}from"@inquirer/prompts";import Ju from"boxen";import Bu from"boxen";var so=[{cmd:"/avatar:help",desc:"H\u01B0\u1EDBng d\u1EABn s\u1EED d\u1EE5ng Avatar \u2014 ch\u1EC9 c\u1EA7n g\xF5 t\u1EF1 nhi\xEAn"},{cmd:"/avatar:brainstorm",desc:"Brainstorm \xFD t\u01B0\u1EDFng cho m\u1ED9t feature"},{cmd:"/avatar:plan",desc:"T\u1EA1o plan th\xF4ng minh v\u1EDBi prompt enhancement"},{cmd:"/avatar:scout",desc:"T\xECm file li\xEAn quan trong codebase, ti\u1EBFt ki\u1EC7m token"},{cmd:"/avatar:implement",desc:"B\u1EAFt \u0111\u1EA7u code & test theo plan c\xF3 s\u1EB5n"},{cmd:"/avatar:build-full-flow",desc:"Implement m\u1ED9t feature t\u1EEBng b\u01B0\u1EDBc (end-to-end)"},{cmd:"/avatar:fix",desc:"Ph\xE2n t\xEDch v\xE0 fix v\u1EA5n \u0111\u1EC1 t\u1EF1 \u0111i\u1EC1u h\u01B0\u1EDBng"},{cmd:"/avatar:debug",desc:"Debug v\u1EA5n \u0111\u1EC1 k\u1EF9 thu\u1EADt + \u0111\u01B0a ra gi\u1EA3i ph\xE1p"},{cmd:"/avatar:test",desc:"Ch\u1EA1y test tr\xEAn m\xE1y + ph\xE2n t\xEDch b\xE1o c\xE1o t\u1ED5ng h\u1EE3p"},{cmd:"/avatar:design:good",desc:"T\u1EA1o thi\u1EBFt k\u1EBF ch\u1EC9n chu, s\u1ED1ng \u0111\u1ED9ng"},{cmd:"/avatar:docs:init",desc:"Ph\xE2n t\xEDch codebase + t\u1EA1o t\xE0i li\u1EC7u kh\u1EDFi \u0111\u1EA7u"},{cmd:"/avatar:status",desc:"Xem l\u1EA1i thay \u0111\u1ED5i g\u1EA7n \u0111\xE2y + t\u1ED5ng k\u1EBFt c\xF4ng vi\u1EC7c"},{cmd:"/avatar:journal",desc:"Ghi nh\u1EADt k\xFD session"}];function ao(){let t=Math.max(...so.map(a=>a.cmd.length)),e=p.bold("\u{1F3AF} Slash commands t\u1EEB team-ai-pack"),n=p.dim("G\xF5 trong Claude Code session \u0111\u1EC3 g\u1ECDi capability c\u1EE7a pack:"),r=so.map(a=>` ${p.cyan(a.cmd.padEnd(t))} ${p.dim(a.desc)}`),i=p.dim("Catalog \u0111\u1EA7y \u0111\u1EE7 46 commands: cat .claude/pack/scripts/commands_data.yaml"),s=[e,n,"",...r,"",i].join(`
121
- `);return Bu(s,{padding:1,borderStyle:"round",borderColor:"cyan"})}k();import{readdir as Wu}from"fs/promises";import{join as zu}from"path";async function We(t){if(!await d(t))return!0;try{return(await Wu(t)).filter(r=>!r.startsWith(".")&&r!=="Thumbs.db").length===0}catch{return!1}}async function co(t,e,n=10){for(let r=2;r<n;r++){let i=zu(t,`${e}-${r}`);if(await We(i))return i}return null}function uo(t,e){if(/[\/\\]/.test(e)||e==="."||e===".."||e.includes(".."))throw new Error(`T\xEAn workspace kh\xF4ng an to\xE0n (ch\u1EE9a path separator ho\u1EB7c '..'): "${e}".
122
- Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=lo(ze(t,e)),r=lo(t);if(!n.startsWith(`${r}/`)&&n!==r)throw new Error(`T\xEAn workspace "${e}" resolve ra ngo\xE0i parent dir (path traversal). Block.`)}async function re(t,e,n){uo(t,e);let r=ze(t,e);if(await We(r))return r;for(o.warn(`Workspace path "${r}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let i=await co(t,e);if(n&&i)return o.info(`--force: d\xF9ng ${i}`),i;let s=[];i&&s.push({name:`D\xF9ng "${i}" (suggest)`,value:"use-alt"}),s.push({name:"Nh\u1EADp t\xEAn workspace kh\xE1c (manual)",value:"manual"}),s.push({name:"T\u1EA1m ng\u01B0ng init",value:"abort"});let a=await Yu({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:s});if(a==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc resolve workspace path. Ch\u1EA1y l\u1EA1i v\u1EDBi --workspace-name kh\xE1c.");if(a==="use-alt"&&i)return i;let c=await po({message:"T\xEAn workspace m\u1EDBi:",validate:u=>{let m=u.trim();return m.length===0?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng":/[\/\\]/.test(m)||m==="."||m===".."||m.includes("..")?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c ch\u1EE9a '/', '\\', '..' (path traversal)":!0}});uo(t,c.trim());let l=ze(t,c.trim());if(await We(l))return l;o.warn(`"${l}" c\u0169ng \u0111\xE3 c\xF3 n\u1ED9i dung. Th\u1EED t\xEAn kh\xE1c.`)}}async function qe(t){return await po({message:"Team owner email:",default:t})}async function mo(t,e){if(e){o.warn("Skip commit (--no-commit). Ch\u1EA1y 'git status' + commit th\u1EE7 c\xF4ng sau.");return}let n=b(t);await n.add(["CLAUDE.md",".claude/",".gitignore",".gitmodules","notes/","scripts/"]),await n.commit("chore: initialize Avatar workspace"),o.success("\u0110\xE3 commit workspace")}function Xu(t){if(t===null)return` ${p.yellow("AI:")} skipped \xB7 ${p.cyan("avatar ai setup")} \u0111\u1EC3 config sau`;if(t.ok){let e=t.model?` \xB7 model=${t.model}`:"";return` ${p.green("AI:")} ready \xB7 ${t.provider}${e}`}return` ${p.yellow("AI:")} failed (${t.reason.slice(0,60)}) \xB7 th\u1EED ${p.cyan("avatar ai setup")}`}function Zu(t){if(t===null)return` ${p.yellow("GitNexus:")} skipped \xB7 ${p.cyan("avatar gitnexus install")} \u0111\u1EC3 setup sau`;if(t.ok){let e=["ready"];return t.analyzed&&e.push("indexed"),t.wikiGenerated&&e.push("wiki"),t.mcpRegistered&&e.push("mcp"),` ${p.green("GitNexus:")} ${e.join(" \xB7 ")}`}return` ${p.yellow("GitNexus:")} skipped (${(t.reason??"unknown").slice(0,40)}) \xB7 th\u1EED ${p.cyan("avatar gitnexus install")}`}async function go(t,e,n=null,r=null){let i=[`${p.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${qu(process.cwd(),t)||t}`,` ${p.dim(`(flow: ${e})`)}`,Xu(n),Zu(r),"",` ${p.cyan(`cd ${t}`)}`,` ${p.cyan("claude")} M\u1EDF Claude Code \u1EDF workspace root`,"",` ${p.cyan("avatar commit src")} Commit code l\xEAn client remote`,` ${p.cyan("avatar sync")} Pull team-ai-pack m\u1EDBi`,` ${p.cyan("avatar uninstall")} G\u1EE1 Avatar (gi\u1EEF code)`];process.stdout.write(`${Ju(i.join(`
120
+ `:""}function Gn(t){return{projectName:t.projectName,projectDescription:t.projectDescription,teamOwner:t.teamOwner,avatarVersion:G(),packVersion:t.packVersion,lastScan:new Date().toISOString(),mode:t.mode,gitnexusSection:vu(t.gitnexusReady??!1)}}k();import{join as Ve,relative as Au,resolve as ro}from"path";import{input as oo,select as Cu}from"@inquirer/prompts";import Pu from"boxen";import bu from"boxen";var to=[{cmd:"/avatar:help",desc:"H\u01B0\u1EDBng d\u1EABn s\u1EED d\u1EE5ng Avatar \u2014 ch\u1EC9 c\u1EA7n g\xF5 t\u1EF1 nhi\xEAn"},{cmd:"/avatar:brainstorm",desc:"Brainstorm \xFD t\u01B0\u1EDFng cho m\u1ED9t feature"},{cmd:"/avatar:plan",desc:"T\u1EA1o plan th\xF4ng minh v\u1EDBi prompt enhancement"},{cmd:"/avatar:scout",desc:"T\xECm file li\xEAn quan trong codebase, ti\u1EBFt ki\u1EC7m token"},{cmd:"/avatar:implement",desc:"B\u1EAFt \u0111\u1EA7u code & test theo plan c\xF3 s\u1EB5n"},{cmd:"/avatar:build-full-flow",desc:"Implement m\u1ED9t feature t\u1EEBng b\u01B0\u1EDBc (end-to-end)"},{cmd:"/avatar:fix",desc:"Ph\xE2n t\xEDch v\xE0 fix v\u1EA5n \u0111\u1EC1 t\u1EF1 \u0111i\u1EC1u h\u01B0\u1EDBng"},{cmd:"/avatar:debug",desc:"Debug v\u1EA5n \u0111\u1EC1 k\u1EF9 thu\u1EADt + \u0111\u01B0a ra gi\u1EA3i ph\xE1p"},{cmd:"/avatar:test",desc:"Ch\u1EA1y test tr\xEAn m\xE1y + ph\xE2n t\xEDch b\xE1o c\xE1o t\u1ED5ng h\u1EE3p"},{cmd:"/avatar:design:good",desc:"T\u1EA1o thi\u1EBFt k\u1EBF ch\u1EC9n chu, s\u1ED1ng \u0111\u1ED9ng"},{cmd:"/avatar:docs:init",desc:"Ph\xE2n t\xEDch codebase + t\u1EA1o t\xE0i li\u1EC7u kh\u1EDFi \u0111\u1EA7u"},{cmd:"/avatar:status",desc:"Xem l\u1EA1i thay \u0111\u1ED5i g\u1EA7n \u0111\xE2y + t\u1ED5ng k\u1EBFt c\xF4ng vi\u1EC7c"},{cmd:"/avatar:journal",desc:"Ghi nh\u1EADt k\xFD session"}];function eo(){let t=Math.max(...to.map(a=>a.cmd.length)),e=p.bold("\u{1F3AF} Slash commands t\u1EEB team-ai-pack"),n=p.dim("G\xF5 trong Claude Code session \u0111\u1EC3 g\u1ECDi capability c\u1EE7a pack:"),r=to.map(a=>` ${p.cyan(a.cmd.padEnd(t))} ${p.dim(a.desc)}`),i=p.dim("Catalog \u0111\u1EA7y \u0111\u1EE7 46 commands: cat .claude/pack/scripts/commands_data.yaml"),s=[e,n,"",...r,"",i].join(`
121
+ `);return bu(s,{padding:1,borderStyle:"round",borderColor:"cyan"})}k();import{readdir as xu}from"fs/promises";import{join as Su}from"path";async function Fe(t){if(!await d(t))return!0;try{return(await xu(t)).filter(r=>!r.startsWith(".")&&r!=="Thumbs.db").length===0}catch{return!1}}async function no(t,e,n=10){for(let r=2;r<n;r++){let i=Su(t,`${e}-${r}`);if(await Fe(i))return i}return null}function io(t,e){if(/[\/\\]/.test(e)||e==="."||e===".."||e.includes(".."))throw new Error(`T\xEAn workspace kh\xF4ng an to\xE0n (ch\u1EE9a path separator ho\u1EB7c '..'): "${e}".
122
+ Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=ro(Ve(t,e)),r=ro(t);if(!n.startsWith(`${r}/`)&&n!==r)throw new Error(`T\xEAn workspace "${e}" resolve ra ngo\xE0i parent dir (path traversal). Block.`)}async function ne(t,e,n){io(t,e);let r=Ve(t,e);if(await Fe(r))return r;for(o.warn(`Workspace path "${r}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let i=await no(t,e);if(n&&i)return o.info(`--force: d\xF9ng ${i}`),i;let s=[];i&&s.push({name:`D\xF9ng "${i}" (suggest)`,value:"use-alt"}),s.push({name:"Nh\u1EADp t\xEAn workspace kh\xE1c (manual)",value:"manual"}),s.push({name:"T\u1EA1m ng\u01B0ng init",value:"abort"});let a=await Cu({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:s});if(a==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc resolve workspace path. Ch\u1EA1y l\u1EA1i v\u1EDBi --workspace-name kh\xE1c.");if(a==="use-alt"&&i)return i;let c=await oo({message:"T\xEAn workspace m\u1EDBi:",validate:u=>{let m=u.trim();return m.length===0?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng":/[\/\\]/.test(m)||m==="."||m===".."||m.includes("..")?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c ch\u1EE9a '/', '\\', '..' (path traversal)":!0}});io(t,c.trim());let l=Ve(t,c.trim());if(await Fe(l))return l;o.warn(`"${l}" c\u0169ng \u0111\xE3 c\xF3 n\u1ED9i dung. Th\u1EED t\xEAn kh\xE1c.`)}}async function Ke(t){return await oo({message:"Team owner email:",default:t})}async function so(t,e){if(e){o.warn("Skip commit (--no-commit). Ch\u1EA1y 'git status' + commit th\u1EE7 c\xF4ng sau.");return}let n=b(t);await n.add(["CLAUDE.md",".claude/",".gitignore",".gitmodules","notes/","scripts/"]),await n.commit("chore: initialize Avatar workspace"),o.success("\u0110\xE3 commit workspace")}function $u(t){if(t===null)return` ${p.yellow("AI:")} skipped \xB7 ${p.cyan("avatar ai setup")} \u0111\u1EC3 config sau`;if(t.ok){let e=t.model?` \xB7 model=${t.model}`:"";return` ${p.green("AI:")} ready \xB7 ${t.provider}${e}`}return` ${p.yellow("AI:")} failed (${t.reason.slice(0,60)}) \xB7 th\u1EED ${p.cyan("avatar ai setup")}`}function Eu(t){if(t===null)return` ${p.yellow("GitNexus:")} skipped \xB7 ${p.cyan("avatar gitnexus install")} \u0111\u1EC3 setup sau`;if(t.ok){let e=["ready"];return t.analyzed&&e.push("indexed"),t.wikiGenerated&&e.push("wiki"),t.mcpRegistered&&e.push("mcp"),` ${p.green("GitNexus:")} ${e.join(" \xB7 ")}`}return` ${p.yellow("GitNexus:")} skipped (${(t.reason??"unknown").slice(0,40)}) \xB7 th\u1EED ${p.cyan("avatar gitnexus install")}`}async function ao(t,e,n=null,r=null){let i=[`${p.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${Au(process.cwd(),t)||t}`,` ${p.dim(`(flow: ${e})`)}`,$u(n),Eu(r),"",` ${p.cyan(`cd ${t}`)}`,` ${p.cyan("claude")} M\u1EDF Claude Code \u1EDF workspace root`,"",` ${p.cyan("avatar commit src")} Commit code l\xEAn client remote`,` ${p.cyan("avatar sync")} Pull team-ai-pack m\u1EDBi`,` ${p.cyan("avatar uninstall")} G\u1EE1 Avatar (gi\u1EEF code)`];process.stdout.write(`${Pu(i.join(`
123
123
  `),{padding:1,borderStyle:"round"})}
124
- `);let s=ze(t,R);await d(s)&&process.stdout.write(`
125
- ${ao()}
126
- `)}import{join as Ot}from"path";import{basename as dp}from"path";import{confirm as xo,input as Ao,select as zn}from"@inquirer/prompts";import{spawnSync as gt}from"child_process";var It=class extends Error{reason;fullName;stderr;constructor(e,n,r,i){super(r),this.name="CreateWorkspaceRemoteError",this.reason=e,this.fullName=n,this.stderr=i}};function Qu(t){let e=t.toLowerCase();return e.includes("name already exists")||e.includes("already exists on this account")||e.includes("repository already exists")?"repo-exists":e.includes("403")||e.includes("permission")||e.includes("not authorized")||e.includes("forbidden")?"no-permission":e.includes("invalid")&&e.includes("name")?"name-invalid":e.includes("could not resolve")||e.includes("network")||e.includes("connection refused")?"network":"unknown"}function tp(t){return gt("gh",["repo","view",t,"--json","name"],{stdio:"ignore"}).status===0}function ep(t,e){return t.toLowerCase()===e.toLowerCase()?{ok:!0}:gt("gh",["api",`orgs/${t}/members/${e}`,"--silent"],{stdio:"ignore"}).status===0?{ok:!0}:gt("gh",["api",`orgs/${t}`,"--silent"],{stdio:"ignore"}).status!==0?gt("gh",["api",`users/${t}`,"--silent"],{stdio:"ignore"}).status===0?{ok:!1,reason:`'${t}' l\xE0 personal account kh\xE1c (kh\xF4ng ph\u1EA3i org). B\u1EA1n (${e}) kh\xF4ng th\u1EC3 t\u1EA1o repo d\u01B0\u1EDBi account c\u1EE7a user kh\xE1c. Pass --repo-org=${e} ho\u1EB7c switch gh: gh auth login`}:{ok:!1,reason:`'${t}' kh\xF4ng t\u1ED3n t\u1EA1i tr\xEAn GitHub. Check ch\xEDnh t\u1EA3 ho\u1EB7c t\u1EA1o org tr\u01B0\u1EDBc.`}:{ok:!1,reason:`'${e}' kh\xF4ng ph\u1EA3i member c\u1EE7a org '${t}'. Li\xEAn h\u1EC7 admin org \u0111\u1EC3 \u0111\u01B0\u1EE3c invite, ho\u1EB7c pass --repo-org=${e} t\u1EA1o d\u01B0\u1EDBi personal account.`}}async function fo(t){Ge(t.workspaceName),Le(t.visibility),await dt();let e=je(),n=t.org??e,r=ep(n,e);if(!r.ok)throw new Error(`Kh\xF4ng th\u1EC3 t\u1EA1o repo d\u01B0\u1EDBi '${n}/': ${r.reason}`);let i=`${n}/${t.workspaceName}`;if(tp(i))throw new It("repo-exists",i,`Repo '${i}' \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub. C\xF3 th\u1EC3 b\u1EA1n \u0111\xE3 init workspace n\xE0y tr\u01B0\u1EDBc \u0111\xF3.`);o.info(`T\u1EA1o GitHub repo cho workspace: ${i} (${t.visibility})...`);let s=gt("gh",["repo","create",i,`--${t.visibility}`,"--source",t.workspacePath,"--remote","origin","--push"],{stdio:["ignore","pipe","pipe"],encoding:"utf8"});if(s.status!==0){let l=(s.stderr||"").trim(),u=(s.stdout||"").trim(),m=[l,u].filter(Boolean).join(`
127
- `),g=Qu(m);throw m&&process.stderr.write(`
124
+ `);let s=Ve(t,R);await d(s)&&process.stdout.write(`
125
+ ${eo()}
126
+ `)}import{join as Ot}from"path";import{basename as Vu}from"path";import{confirm as ho,input as ko,select as Kn}from"@inquirer/prompts";import{spawnSync as gt}from"child_process";var _t=class extends Error{reason;fullName;stderr;constructor(e,n,r,i){super(r),this.name="CreateWorkspaceRemoteError",this.reason=e,this.fullName=n,this.stderr=i}};function Ru(t){let e=t.toLowerCase();return e.includes("name already exists")||e.includes("already exists on this account")||e.includes("repository already exists")?"repo-exists":e.includes("403")||e.includes("permission")||e.includes("not authorized")||e.includes("forbidden")?"no-permission":e.includes("invalid")&&e.includes("name")?"name-invalid":e.includes("could not resolve")||e.includes("network")||e.includes("connection refused")?"network":"unknown"}function Tu(t){return gt("gh",["repo","view",t,"--json","name"],{stdio:"ignore"}).status===0}function _u(t,e){return t.toLowerCase()===e.toLowerCase()?{ok:!0}:gt("gh",["api",`orgs/${t}/members/${e}`,"--silent"],{stdio:"ignore"}).status===0?{ok:!0}:gt("gh",["api",`orgs/${t}`,"--silent"],{stdio:"ignore"}).status!==0?gt("gh",["api",`users/${t}`,"--silent"],{stdio:"ignore"}).status===0?{ok:!1,reason:`'${t}' l\xE0 personal account kh\xE1c (kh\xF4ng ph\u1EA3i org). B\u1EA1n (${e}) kh\xF4ng th\u1EC3 t\u1EA1o repo d\u01B0\u1EDBi account c\u1EE7a user kh\xE1c. Pass --repo-org=${e} ho\u1EB7c switch gh: gh auth login`}:{ok:!1,reason:`'${t}' kh\xF4ng t\u1ED3n t\u1EA1i tr\xEAn GitHub. Check ch\xEDnh t\u1EA3 ho\u1EB7c t\u1EA1o org tr\u01B0\u1EDBc.`}:{ok:!1,reason:`'${e}' kh\xF4ng ph\u1EA3i member c\u1EE7a org '${t}'. Li\xEAn h\u1EC7 admin org \u0111\u1EC3 \u0111\u01B0\u1EE3c invite, ho\u1EB7c pass --repo-org=${e} t\u1EA1o d\u01B0\u1EDBi personal account.`}}async function co(t){Oe(t.workspaceName),Ne(t.visibility),await dt();let e=Ie(),n=t.org??e,r=_u(n,e);if(!r.ok)throw new Error(`Kh\xF4ng th\u1EC3 t\u1EA1o repo d\u01B0\u1EDBi '${n}/': ${r.reason}`);let i=`${n}/${t.workspaceName}`;if(Tu(i))throw new _t("repo-exists",i,`Repo '${i}' \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub. C\xF3 th\u1EC3 b\u1EA1n \u0111\xE3 init workspace n\xE0y tr\u01B0\u1EDBc \u0111\xF3.`);o.info(`T\u1EA1o GitHub repo cho workspace: ${i} (${t.visibility})...`);let s=gt("gh",["repo","create",i,`--${t.visibility}`,"--source",t.workspacePath,"--remote","origin","--push"],{stdio:["ignore","pipe","pipe"],encoding:"utf8"});if(s.status!==0){let l=(s.stderr||"").trim(),u=(s.stdout||"").trim(),m=[l,u].filter(Boolean).join(`
127
+ `),g=Ru(m);throw m&&process.stderr.write(`
128
128
  ${m}
129
129
 
130
- `),new It(g,i,`T\u1EA1o workspace remote th\u1EA5t b\u1EA1i (exit ${s.status}): ${g}.`,m)}let a=`git@github.com:${i}.git`,c=`https://github.com/${i}.git`;return o.success(`Workspace remote: ${a}`),{sshUrl:a,httpsUrl:c}}function ho(t){let e=`git@github.com:${t.fullName}.git`,n=`https://github.com/${t.fullName}.git`;return gt("git",["-C",t.workspacePath,"remote","add","origin",e],{encoding:"utf8",stdio:["ignore","pipe","pipe"]}).status!==0&&gt("git",["-C",t.workspacePath,"remote","set-url","origin",e],{stdio:"ignore"}),o.success(`Linked existing remote: ${e}`),{sshUrl:e,httpsUrl:n}}k();import{confirm as np}from"@inquirer/prompts";var rp={"prompt-scoring":"prompt scoring"};async function ko(t,e={}){let n=await ui(t);if(n.length!==0)for(let r of n){let i=rp[r]??r,s;if(e.autoYes?s=!0:s=await np({message:`Do you want to setup ${i} (y/n)`,default:!0}),!s){o.dim(` Skip feature '${r}' (user opt-out).`);continue}await lt(t,r,{silent:!0})}}import{promises as Ye}from"fs";import{dirname as Kn,join as yo,relative as Vn}from"path";import{promises as ip}from"fs";function op(){let t=new Date,e=n=>n.toString().padStart(2,"0");return`${t.getFullYear()}-${e(t.getMonth()+1)}-${e(t.getDate())}-${e(t.getHours())}${e(t.getMinutes())}`}async function wo(t){let e=`${t}.backup-${op()}`;return await ip.rename(t,e),e}k();var Bn=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function sp(t){try{return(await Ye.lstat(t)).isSymbolicLink()}catch{return!1}}async function ap(t,e,n){let r=Vn(Kn(e),e)||e;if(!await d(t))return{dir:r,action:"source-missing"};if(await d(e))if(await sp(e))await Ye.unlink(e);else if(n){let s=await wo(e),a=Vn(Kn(e),t);return await Ye.symlink(a,e),{dir:r,action:"backed-up-and-linked",backupPath:s}}else return{dir:r,action:"skipped-conflict"};let i=Vn(Kn(e),t);return await Ye.symlink(i,e),{dir:r,action:"created"}}async function Je(t,e,n){let r=[];for(let i of Bn){let s=yo(t,i),a=yo(e,i);r.push(await ap(s,a,n))}return r}async function So(t,e){let r=(await b(t).getRemotes(!0)).find(l=>l.name==="origin");if(r?.refs.push)return o.success(`Folder \u0111\xE3 c\xF3 remote origin: ${r.refs.push}`),r.refs.push;if(!(e.createRemote??await xo({message:"Folder ch\u01B0a c\xF3 remote. T\u1EA1o GitHub repo ngay \u0111\u1EC3 share team?",default:!0}))){o.warn("Ti\u1EBFp t\u1EE5c v\u1EDBi local path. Workspace ch\u1EC9 ch\u1EA1y \u0111\u01B0\u1EE3c tr\xEAn m\xE1y b\u1EA1n.");return}await dt();let s=e.repoVisibility??await zn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),a=await Ao({message:"T\xEAn repo:",default:dp(t)});return Et({folder:t,name:a,visibility:s,org:e.repoOrg}).sshUrl}async function qn(t){await E(t.workspacePath),await b(t.workspacePath).init();let e=F(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await b(t.workspacePath).subModule(["add",t.srcRemoteUrl,"src"]);let n="HEAD";t.skipTeamPack?e.succeed("Skip team-ai-pack (--skip-team-pack)"):(e.stop(),n=(await Ve(t.workspacePath,t.packVersion,t.ssoEmail,t.packLatest===!0&&!t.packVersion)).pinnedTag??"HEAD",e.succeed(`Pin team-ai-pack v\xE0o ${n}`)),await Yn({workspacePath:t.workspacePath,workspaceName:t.workspaceName,teamOwner:t.teamOwner,description:t.description,packVersion:n,autoYes:t.autoYes,skipCommit:t.skipCommit,createWorkspaceRemote:t.createWorkspaceRemote,repoVisibility:t.repoVisibility,repoOrg:t.repoOrg,flow:t.flow,aiSkip:t.aiSkip,gitnexusSkip:t.gitnexusSkip})}catch(n){throw e.fail("Init workspace th\u1EA5t b\u1EA1i"),n}}async function Yn(t){let e=Fn({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client"});await ii(t.workspacePath),await oi(t.workspacePath,e),await bn(t.workspacePath,e),await si(t.workspacePath,e),await ai(t.workspacePath),await E(Ot(t.workspacePath,"notes")),await E(Ot(t.workspacePath,"scripts")),await Wt(Ot(t.workspacePath,".git"),"post-merge"),await Wt(Ot(t.workspacePath,".git","modules","src"),"pre-push"),o.success("C\xE0i post-merge (workspace) + pre-push (src/)"),await gp(t.workspacePath,t.autoYes),await w("init",`flow=${t.flow},workspace=${t.workspaceName}`),await mo(t.workspacePath,t.skipCommit),await fp(t);let n=null;t.aiSkip?o.dim("B\u1ECF qua AI setup (--ai-skip). Setup sau qua: avatar ai setup"):n=await ve({workspacePath:t.workspacePath});let r=null;if(t.aiSkip||t.gitnexusSkip?t.gitnexusSkip?o.dim("B\u1ECF qua GitNexus setup (--gitnexus-skip). Setup sau: avatar gitnexus install"):o.dim("B\u1ECF qua GitNexus setup (auto-skip do --ai-skip)."):r=await Oe({workspacePath:t.workspacePath}),r?.ok){let s=Fn({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client",gitnexusReady:!0});await bn(t.workspacePath,s),o.dim("Updated CLAUDE.md v\u1EDBi GitNexus section")}await go(t.workspacePath,t.flow,n,r)}async function gp(t,e){let n=Ot(t,R);if(!await d(n)){o.dim("Pack submodule kh\xF4ng t\u1ED3n t\u1EA1i (skip auto-sync). C\xF3 th\u1EC3 ch\u1EA1y `avatar sync` sau.");return}let r=Ot(t,".claude");o.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings merge)...");try{let i=await Je(n,r,!1),s=i.filter(l=>l.action==="created"||l.action==="updated").length,a=i.filter(l=>l.action==="source-missing").length;o.success(` \u2713 Symlinks: ${s} created${a>0?`, ${a} source-missing (pack thi\u1EBFu dir)`:""}`);let c=await Ae(t);switch(c.action){case"merged":o.success(` \u2713 settings.json merged (${c.changes.join("; ")})`);break;case"no-change":o.dim(" - settings.json \u0111\xE3 sync, kh\xF4ng c\u1EA7n thay \u0111\u1ED5i.");break;case"no-pack-template":o.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip merge.");break}await ko(t,{autoYes:e});try{let{detectPlaintextSecretsInSettings:l,migrateSecretsFromSettingsToEnvrc:u}=await Promise.resolve().then(()=>(Wn(),bo)),m=await l(t);m.keys.length>0&&(o.info(`Auto-migrating ${m.keys.length} plaintext secrets t\u1EEB settings.json \u2192 .envrc...`),await u(t))}catch(l){o.warn(`Auto-migrate secrets fail: ${l instanceof Error?l.message:l}.`)}}catch(i){o.warn(`Auto-sync pack fail: ${i instanceof Error?i.message:i}. Ch\u1EA1y \`avatar sync\` th\u1EE7 c\xF4ng \u0111\u1EC3 retry.`)}}async function fp(t){if(t.skipCommit){o.dim("Skip workspace remote (ch\u01B0a commit). Setup sau qua: gh repo create ...");return}let e=t.createWorkspaceRemote;if(e===void 0){if(t.autoYes)return;e=await xo({message:"T\u1EA1o remote GitHub cho workspace \u0111\u1EC3 share team? (Avatar state)",default:!1})}if(!e)return;let n=t.repoVisibility??(t.autoYes?"private":await zn({message:"Workspace visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh, an to\xE0n)",value:"private"},{name:"public",value:"public"}]}));for(;;)try{await fo({workspacePath:t.workspacePath,workspaceName:t.workspaceName,visibility:n,org:t.repoOrg});return}catch(r){if(r instanceof It&&r.reason==="repo-exists"){let s=r.fullName,a=await zn({message:`Repo '${s}' \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub. C\xE1ch x\u1EED l\xFD?`,choices:[{name:"D\xF9ng remote \u0111\xE3 c\xF3 (link workspace local v\xE0o repo n\xE0y)",value:"reuse"},{name:"Nh\u1EADp t\xEAn workspace kh\xE1c (t\u1EA1o repo m\u1EDBi)",value:"rename"},{name:"B\u1ECF qua (workspace local-only)",value:"skip"},{name:"T\u1EA1m ng\u01B0ng init",value:"abort"}]});if(a==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA1o workspace remote.");if(a==="skip"){o.warn("Workspace v\u1EABn s\u1EB5n s\xE0ng local-only. Setup remote sau khi c\u1EA7n.");return}if(a==="reuse"){ho({workspacePath:t.workspacePath,fullName:s});return}let c=await Ao({message:"T\xEAn workspace m\u1EDBi (s\u1EBD t\u1EA1o repo new):",validate:l=>l.trim().length>0?!0:"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng"});t.workspaceName=c.trim();continue}let i=await T({taskName:"T\u1EA1o workspace remote tr\xEAn GitHub",reason:r instanceof Error?r.message:String(r),allowSkip:!0,hint:"Tip: sai org? Pass --repo-org=<your-gh-user>. Ho\u1EB7c switch gh: gh auth login."});if(i==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA1o workspace remote.");if(i==="skip"){o.warn("Workspace v\u1EABn s\u1EB5n s\xE0ng local-only. Setup remote sau khi c\u1EA7n.");return}}}async function Co(t,e){let n=t.clientRepo??await ie({message:"URL git c\u1EE7a repo:",validate:m=>m.length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),{resolvedRemoteUrl:r}=await dt(n),i=r??n,s=t.teamOwner??await qe(e),a=oo(i),c=t.workspaceName??await ie({message:"T\xEAn workspace:",default:a}),l=Ze(t.workspaceParent??"."),u=await re(l,c,t.force);await qn({workspacePath:u,workspaceName:c,srcRemoteUrl:i,teamOwner:s,skipTeamPack:t.skipTeamPack,description:t.description??`Avatar workspace cho ${i}`,packVersion:t.packVersion,packLatest:t.latest,autoYes:t.yes,skipCommit:t.commit===!1,createWorkspaceRemote:t.workspaceRemote,repoVisibility:t.repoVisibility,repoOrg:t.repoOrg,flow:"existing-remote",aiSkip:t.aiSkip,gitnexusSkip:t.gitnexusSkip,ssoEmail:e})}async function Po(t,e){let n=Ze(t.folderPath??await ie({message:"\u0110\u01B0\u1EDDng d\u1EABn folder hi\u1EC7n c\xF3:",validate:u=>u.length>0?!0:"Path b\u1EAFt bu\u1ED9c"}));await jn(n,{presetStrategy:Dn(t),autoYes:t.yes});let r=await So(n,t);if(r){let u=Rt(r);u.ok||(o.warn(`Remote ${r} kh\xF4ng accessible (${u.reason??"unknown"}).`),r=(await Ue({url:r,initialReason:u.reason??"unknown",initialDetail:u.detail,folderPath:n,defaultVisibility:t.repoVisibility})).resolvedUrl)}let i=t.teamOwner??await qe(e),s=t.workspaceName??`${hp(n)}-avatar-workspace`,a=t.workspaceName??await ie({message:"T\xEAn workspace:",default:s}),c=Ze(t.workspaceParent??"."),l=await re(c,a,t.force);await qn({workspacePath:l,workspaceName:a,srcRemoteUrl:r??n,teamOwner:i,skipTeamPack:t.skipTeamPack,description:t.description??`Avatar workspace cho folder ${n}`,packVersion:t.packVersion,packLatest:t.latest,autoYes:t.yes,skipCommit:t.commit===!1,createWorkspaceRemote:t.workspaceRemote,repoVisibility:t.repoVisibility,repoOrg:t.repoOrg,flow:"existing-folder",aiSkip:t.aiSkip,gitnexusSkip:t.gitnexusSkip,ssoEmail:e})}async function $o(t,e){await dt();let n=t.workspaceName??await ie({message:"T\xEAn d\u1EF1 \xE1n:",validate:m=>m.length>0?!0:"T\xEAn b\u1EAFt bu\u1ED9c"}),r=t.repoVisibility??await wp({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),i=t.teamOwner??await qe(e),s=Ze(t.workspaceParent??"."),a=await re(s,n,t.force),c=kp(a,"src");await E(a),await E(c),await jn(c,{autoYes:!0});let l=Et({folder:c,name:n,visibility:r,org:t.repoOrg});await b(a).init();let u=F(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await b(a).subModule(["add",l.sshUrl,"src"]);let m="HEAD";t.skipTeamPack?u.succeed("Skip team-ai-pack (--skip-team-pack)"):(u.stop(),m=(await Ve(a,t.packVersion,e,t.latest===!0&&!t.packVersion)).pinnedTag??"HEAD",u.succeed(`Pin team-ai-pack v\xE0o ${m}`)),await Yn({workspacePath:a,workspaceName:n,teamOwner:i,description:t.description??`D\u1EF1 \xE1n m\u1EDBi: ${n}`,packVersion:m,autoYes:t.yes,skipCommit:t.commit===!1,createWorkspaceRemote:t.workspaceRemote,repoVisibility:t.repoVisibility,repoOrg:t.repoOrg,flow:"new-project",aiSkip:t.aiSkip,gitnexusSkip:t.gitnexusSkip})}catch(m){throw u.fail("Init workspace th\u1EA5t b\u1EA1i"),m}}import $p from"boxen";import Ep from"open";var Xn="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",yp="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",Jn="nal.vn",vp=["openid","email","profile"],bp="https://oauth2.googleapis.com/device/code",xp="https://oauth2.googleapis.com/token",Ap="https://oauth2.googleapis.com/revoke";async function Eo(){let t=new URLSearchParams({client_id:Xn,scope:vp.join(" ")}),e=await fetch(bp,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:t});if(!e.ok){let n=await e.text();throw new Error(`Device code request failed (${e.status}): ${n}`)}return await e.json()}async function Ro(t){let e=new URLSearchParams({client_id:Xn,client_secret:yp,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(xp,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e});if(n.ok)return await n.json();let r="";try{r=(await n.json()).error??""}catch{r=""}if(r==="authorization_pending"||r==="slow_down")return null;throw r==="access_denied"?new Error("User t\u1EEB ch\u1ED1i quy\u1EC1n truy c\u1EADp"):r==="expired_token"?new Error("H\u1EBFt h\u1EA1n ch\u1EDD (5 ph\xFAt). Ch\u1EA1y l\u1EA1i 'avatar login'."):new Error(`OAuth token endpoint tr\u1EA3 l\u1ED7i: ${r||n.status}`)}function To(t){let e=t.split(".");if(e.length!==3)throw new Error("id_token format kh\xF4ng h\u1EE3p l\u1EC7");let n=e[1];if(!n)throw new Error("id_token thi\u1EBFu payload");let r=n.replace(/-/g,"+").replace(/_/g,"/"),i=Buffer.from(r,"base64").toString("utf8");return JSON.parse(i)}var Sp=new Set(["https://accounts.google.com","accounts.google.com"]),Cp=60;function Pp(t){if(!Sp.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==Xn)throw new Error("id_token audience kh\xF4ng kh\u1EDBp client ID Avatar. Token c\xF3 th\u1EC3 \u0111\u01B0\u1EE3c sign cho app kh\xE1c.");let e=Math.floor(Date.now()/1e3);if(t.exp+Cp<e){let n=e-t.exp;throw new Error(`id_token \u0111\xE3 h\u1EBFt h\u1EA1n ${n}s tr\u01B0\u1EDBc. Login l\u1EA1i \u0111\u1EC3 l\u1EA5y token m\u1EDBi.`)}if(t.hd!==Jn)throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u @${Jn}). Nh\u1EADn: ${t.email}`);if(!t.email_verified)throw new Error("Email ch\u01B0a \u0111\u01B0\u1EE3c Google verify")}var Io=Pp;function _o(t,e){let n=new Date(Date.now()+t.expires_in*1e3).toISOString();return{email:e.email,name:e.name??e.email,access_token:t.access_token,refresh_token:t.refresh_token,expires_at:n,id_token:t.id_token}}async function Oo(t){let e=new URLSearchParams({token:t});await fetch(Ap,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e}).catch(()=>{})}function No(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.searchParams.set("hd",Jn),e.toString()}function Mo(t){t.command("login").description("\u0110\u0103ng nh\u1EADp Google SSO (workspace @nal.vn)").option("--reset","X\xF3a credential c\u0169 v\xE0 \u0111\u0103ng nh\u1EADp l\u1EA1i").action(async e=>{try{await Zn(e)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Zn(t){if($t({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await ir(),await w("login_reset");else{let g=await et();if(g&&!nt(g)){o.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=F("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await Eo(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let r=No(n),i=[`1. Truy c\u1EADp: ${p.cyan(n.verification_url)}`,`2. Nh\u1EADp code: ${p.bold.yellow(n.user_code)}`,"",`Ho\u1EB7c Avatar t\u1EF1 m\u1EDF browser, click ${p.green("Allow")}...`].join(`
131
- `);process.stdout.write(`${$p(i,{padding:1,borderStyle:"round"})}
132
- `),Ep(r).catch(()=>{o.dim("(Kh\xF4ng m\u1EDF \u0111\u01B0\u1EE3c browser t\u1EF1 \u0111\u1ED9ng \u2014 copy URL \u1EDF tr\xEAn)")});let s=F("\u0110ang ch\u1EDD x\xE1c nh\u1EADn trong browser..."),a=n.interval*1e3,c=Date.now()+n.expires_in*1e3,l=null;for(;Date.now()<c;){await Rp(a);try{if(l=await Ro(n.device_code),l)break}catch(g){throw s.fail("X\xE1c th\u1EF1c th\u1EA5t b\u1EA1i"),g}}l||(s.fail("H\u1EBFt h\u1EA1n ch\u1EDD (5 ph\xFAt). Ch\u1EA1y l\u1EA1i 'avatar login'."),process.exit(1)),s.succeed("\u0110\xE3 nh\u1EADn token t\u1EEB Google");let u=To(l.id_token);try{Io(u)}catch(g){throw await Oo(l.access_token),g}let m=_o(l,u);await rr(m),await w("login",m.email),o.success(`X\xE1c th\u1EF1c th\xE0nh c\xF4ng: ${m.email}`),o.success(`Verify hosted domain: ${u.hd} \u2713`),o.success(`L\u01B0u credential v\xE0o ${tt} (chmod 600)`)}function Rp(t){return new Promise(e=>setTimeout(e,t))}function jo(t){t.command("init").description("Kh\u1EDFi t\u1EA1o Avatar \u2014 3 flow t\u1EF1 nh\u1EADn di\u1EC7n (repo / folder / new)").option("--project-status <val>","existing-remote | existing-folder | new-project").option("--folder-path <path>","\u0110\u01B0\u1EDDng d\u1EABn folder hi\u1EC7n c\xF3 (flow existing-folder)").option("--create-remote","Force t\u1EA1o remote qua gh (flow existing-folder ho\u1EB7c new-project)").option("--repo-visibility <val>","private (m\u1EB7c \u0111\u1ECBnh) | public").option("--repo-org <name>","GitHub org/owner cho repo m\u1EDBi").option("--client-repo <url>","URL git remote (flow existing-remote)").option("--workspace-name <name>","T\xEAn workspace").option("--workspace-parent <path>","Th\u01B0 m\u1EE5c cha t\u1EA1o workspace (m\u1EB7c \u0111\u1ECBnh . \u2014 CWD)").option("--pack-version <tag>","Pin team-ai-pack v\xE0o tag c\u1EE5 th\u1EC3").option("--latest","Pull HEAD c\u1EE7a team-ai-pack main branch (b\u1ECF qua tag SemVer, bleeding-edge)").option("--team-owner <email>","Email team owner (b\u1ECF qua prompt)").option("--description <text>","M\xF4 t\u1EA3 1 d\xF2ng c\u1EE7a d\u1EF1 \xE1n").option("--skip-scan","B\u1ECF qua project-scanner sau scaffold").option("--skip-team-pack","B\u1ECF qua submodule team-ai-pack (test mode)").option("--force","B\u1ECF qua prompt khi workspace path \u0111\xE3 t\u1ED3n t\u1EA1i").option("--yes","Auto-confirm t\u1EA5t c\u1EA3 prompt").option("--no-commit","Skip commit workspace initial state (m\u1EB7c \u0111\u1ECBnh LU\xD4N commit)").option("--workspace-remote","T\u1EA1o GitHub remote cho workspace root (default: prompt)").option("--ai-skip","B\u1ECF qua phase AI setup (CI/test mode \u2014 ch\u1EA1y `avatar ai setup` sau)").option("--gitnexus-skip","B\u1ECF qua phase GitNexus setup (M10 \u2014 ch\u1EA1y `avatar gitnexus install` sau)").option("--bootstrap-strategy <s>","X\u1EED l\xFD folder dirty: stash | commit-all | skip | branch (default: prompt)").option("--preserve-uncommitted","Alias cho --bootstrap-strategy=stash (gi\u1EEF changes user)").option("--mode <mode>","[DEPRECATED] D\xF9ng --project-status thay th\u1EBF").action(async e=>{try{await Ip(e)}catch(n){n instanceof ne&&(o.dim(n.message),process.exit(0)),n instanceof mt&&(o.dim(n.message),process.exit(0)),n instanceof Zt&&(o.dim(n.message),process.exit(0)),n instanceof x&&(o.dim(n.message),process.exit(0)),o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Ip(t){t.yes||$t({tagline:"Kh\u1EDFi t\u1EA1o Avatar trong d\u1EF1 \xE1n c\u1EE7a b\u1EA1n"}),t.mode&&o.warn("Flag --mode \u0111\xE3 deprecated t\u1EEB v1.1. D\xF9ng --project-status thay th\u1EBF.");let e=await et();for(;!e||nt(e);){o.info("Ch\u01B0a \u0111\u0103ng nh\u1EADp \u2014 ch\u1EA1y login flow tr\u01B0\u1EDBc khi init...");try{await Zn({})}catch(i){o.warn(`Login fail: ${i.message}`)}if(e=await et(),e&&!nt(e))break;if(await T({taskName:"\u0110\u0103ng nh\u1EADp SSO Google",reason:"Token ch\u01B0a \u0111\u01B0\u1EE3c t\u1EA1o ho\u1EB7c \u0111\xE3 h\u1EBFt h\u1EA1n.",allowSkip:!1,hint:"\u0110\u1EA3m b\u1EA3o b\u1EA1n ch\u1ECDn 'Allow' tr\xEAn browser v\xE0 d\xF9ng email @nal.vn."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc login. Ch\u1EA1y 'avatar login' tay r\u1ED3i 'avatar init' l\u1EA1i.")}switch(t.projectStatus??await _p()){case"existing-remote":await Co(t,e.email);break;case"existing-folder":await Po(t,e.email);break;case"new-project":await $o(t,e.email);break}}async function _p(){return await Tp({message:"T\xECnh tr\u1EA1ng d\u1EF1 \xE1n c\u1EE7a b\u1EA1n?",choices:[{name:"1. \u0110\xE3 c\xF3 repo git remote (URL c\xF3 s\u1EB5n)",value:"existing-remote"},{name:"2. \u0110\xE3 c\xF3 folder code local",value:"existing-folder"},{name:"3. D\u1EF1 \xE1n m\u1EDBi ho\xE0n to\xE0n",value:"new-project"}]})}function I(t,e){return()=>{process.stdout.write(`${p.yellow("\u23F3")} ${p.bold(`avatar ${t}`)} \u2014 ch\u01B0a implement \u1EDF milestone hi\u1EC7n t\u1EA1i.
130
+ `),new _t(g,i,`T\u1EA1o workspace remote th\u1EA5t b\u1EA1i (exit ${s.status}): ${g}.`,m)}let a=`git@github.com:${i}.git`,c=`https://github.com/${i}.git`;return o.success(`Workspace remote: ${a}`),{sshUrl:a,httpsUrl:c}}function lo(t){let e=`git@github.com:${t.fullName}.git`,n=`https://github.com/${t.fullName}.git`;return gt("git",["-C",t.workspacePath,"remote","add","origin",e],{encoding:"utf8",stdio:["ignore","pipe","pipe"]}).status!==0&&gt("git",["-C",t.workspacePath,"remote","set-url","origin",e],{stdio:"ignore"}),o.success(`Linked existing remote: ${e}`),{sshUrl:e,httpsUrl:n}}k();import{confirm as Iu}from"@inquirer/prompts";var Ou={"prompt-scoring":"prompt scoring"};async function uo(t,e={}){let n=await ii(t);if(n.length!==0)for(let r of n){let i=Ou[r]??r,s;if(e.autoYes?s=!0:s=await Iu({message:`Do you want to setup ${i} (y/n)`,default:!0}),!s){o.dim(` Skip feature '${r}' (user opt-out).`);continue}await lt(t,r,{silent:!0})}}import{promises as Be}from"fs";import{dirname as Un,join as mo,relative as Dn}from"path";import{promises as Nu}from"fs";function Mu(){let t=new Date,e=n=>n.toString().padStart(2,"0");return`${t.getFullYear()}-${e(t.getMonth()+1)}-${e(t.getDate())}-${e(t.getHours())}${e(t.getMinutes())}`}async function po(t){let e=`${t}.backup-${Mu()}`;return await Nu.rename(t,e),e}k();var Fn=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function ju(t){try{return(await Be.lstat(t)).isSymbolicLink()}catch{return!1}}async function Lu(t,e,n){let r=Dn(Un(e),e)||e;if(!await d(t))return{dir:r,action:"source-missing"};if(await d(e))if(await ju(e))await Be.unlink(e);else if(n){let s=await po(e),a=Dn(Un(e),t);return await Be.symlink(a,e),{dir:r,action:"backed-up-and-linked",backupPath:s}}else return{dir:r,action:"skipped-conflict"};let i=Dn(Un(e),t);return await Be.symlink(i,e),{dir:r,action:"created"}}async function We(t,e,n){let r=[];for(let i of Fn){let s=mo(t,i),a=mo(e,i);r.push(await Lu(s,a,n))}return r}async function wo(t,e){let r=(await b(t).getRemotes(!0)).find(l=>l.name==="origin");if(r?.refs.push)return o.success(`Folder \u0111\xE3 c\xF3 remote origin: ${r.refs.push}`),r.refs.push;if(!(e.createRemote??await ho({message:"Folder ch\u01B0a c\xF3 remote. T\u1EA1o GitHub repo ngay \u0111\u1EC3 share team?",default:!0}))){o.warn("Ti\u1EBFp t\u1EE5c v\u1EDBi local path. Workspace ch\u1EC9 ch\u1EA1y \u0111\u01B0\u1EE3c tr\xEAn m\xE1y b\u1EA1n.");return}await dt();let s=e.repoVisibility??await Kn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),a=await ko({message:"T\xEAn repo:",default:Vu(t)});return Et({folder:t,name:a,visibility:s,org:e.repoOrg}).sshUrl}async function Bn(t){await E(t.workspacePath),await b(t.workspacePath).init();let e=D(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await b(t.workspacePath).subModule(["add",t.srcRemoteUrl,"src"]);let n="HEAD";t.skipTeamPack?e.succeed("Skip team-ai-pack (--skip-team-pack)"):(e.stop(),n=(await Ue(t.workspacePath,t.packVersion,t.ssoEmail,t.packLatest===!0&&!t.packVersion)).pinnedTag??"HEAD",e.succeed(`Pin team-ai-pack v\xE0o ${n}`)),await Wn({workspacePath:t.workspacePath,workspaceName:t.workspaceName,teamOwner:t.teamOwner,description:t.description,packVersion:n,autoYes:t.autoYes,skipCommit:t.skipCommit,createWorkspaceRemote:t.createWorkspaceRemote,repoVisibility:t.repoVisibility,repoOrg:t.repoOrg,flow:t.flow,aiSkip:t.aiSkip,gitnexusSkip:t.gitnexusSkip})}catch(n){throw e.fail("Init workspace th\u1EA5t b\u1EA1i"),n}}async function Wn(t){let e=Gn({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client"});await Zr(t.workspacePath),await Qr(t.workspacePath,e),await wn(t.workspacePath,e),await ti(t.workspacePath,e),await ei(t.workspacePath),await E(Ot(t.workspacePath,"notes")),await E(Ot(t.workspacePath,"scripts")),await Bt(Ot(t.workspacePath,".git"),"post-merge"),await Bt(Ot(t.workspacePath,".git","modules","src"),"pre-push"),o.success("C\xE0i post-merge (workspace) + pre-push (src/)"),await Ku(t.workspacePath,t.autoYes),await v("init",`flow=${t.flow},workspace=${t.workspaceName}`),await so(t.workspacePath,t.skipCommit),await Bu(t);let n=null;t.aiSkip?o.dim("B\u1ECF qua AI setup (--ai-skip). Setup sau qua: avatar ai setup"):n=await he({workspacePath:t.workspacePath});let r=null;if(t.aiSkip||t.gitnexusSkip?t.gitnexusSkip?o.dim("B\u1ECF qua GitNexus setup (--gitnexus-skip). Setup sau: avatar gitnexus install"):o.dim("B\u1ECF qua GitNexus setup (auto-skip do --ai-skip)."):r=await Re({workspacePath:t.workspacePath}),r?.ok){let s=Gn({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client",gitnexusReady:!0});await wn(t.workspacePath,s),o.dim("Updated CLAUDE.md v\u1EDBi GitNexus section")}await ao(t.workspacePath,t.flow,n,r)}async function Ku(t,e){let n=Ot(t,R);if(!await d(n)){o.dim("Pack submodule kh\xF4ng t\u1ED3n t\u1EA1i (skip auto-sync). C\xF3 th\u1EC3 ch\u1EA1y `avatar sync` sau.");return}let r=Ot(t,".claude");o.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings merge)...");try{let i=await We(n,r,!1),s=i.filter(l=>l.action==="created"||l.action==="updated").length,a=i.filter(l=>l.action==="source-missing").length;o.success(` \u2713 Symlinks: ${s} created${a>0?`, ${a} source-missing (pack thi\u1EBFu dir)`:""}`);let c=await ye(t);switch(c.action){case"merged":o.success(` \u2713 settings.json merged (${c.changes.join("; ")})`);break;case"no-change":o.dim(" - settings.json \u0111\xE3 sync, kh\xF4ng c\u1EA7n thay \u0111\u1ED5i.");break;case"no-pack-template":o.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip merge.");break}await uo(t,{autoYes:e});try{let{detectPlaintextSecretsInSettings:l,migrateSecretsFromSettingsToEnvrc:u}=await Promise.resolve().then(()=>(Vn(),fo)),m=await l(t);m.keys.length>0&&(o.info(`Auto-migrating ${m.keys.length} plaintext secrets t\u1EEB settings.json \u2192 .envrc...`),await u(t))}catch(l){o.warn(`Auto-migrate secrets fail: ${l instanceof Error?l.message:l}.`)}}catch(i){o.warn(`Auto-sync pack fail: ${i instanceof Error?i.message:i}. Ch\u1EA1y \`avatar sync\` th\u1EE7 c\xF4ng \u0111\u1EC3 retry.`)}}async function Bu(t){if(t.skipCommit){o.dim("Skip workspace remote (ch\u01B0a commit). Setup sau qua: gh repo create ...");return}let e=t.createWorkspaceRemote;if(e===void 0){if(t.autoYes)return;e=await ho({message:"T\u1EA1o remote GitHub cho workspace \u0111\u1EC3 share team? (Avatar state)",default:!1})}if(!e)return;let n=t.repoVisibility??(t.autoYes?"private":await Kn({message:"Workspace visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh, an to\xE0n)",value:"private"},{name:"public",value:"public"}]}));for(;;)try{await co({workspacePath:t.workspacePath,workspaceName:t.workspaceName,visibility:n,org:t.repoOrg});return}catch(r){if(r instanceof _t&&r.reason==="repo-exists"){let s=r.fullName,a=await Kn({message:`Repo '${s}' \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub. C\xE1ch x\u1EED l\xFD?`,choices:[{name:"D\xF9ng remote \u0111\xE3 c\xF3 (link workspace local v\xE0o repo n\xE0y)",value:"reuse"},{name:"Nh\u1EADp t\xEAn workspace kh\xE1c (t\u1EA1o repo m\u1EDBi)",value:"rename"},{name:"B\u1ECF qua (workspace local-only)",value:"skip"},{name:"T\u1EA1m ng\u01B0ng init",value:"abort"}]});if(a==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA1o workspace remote.");if(a==="skip"){o.warn("Workspace v\u1EABn s\u1EB5n s\xE0ng local-only. Setup remote sau khi c\u1EA7n.");return}if(a==="reuse"){lo({workspacePath:t.workspacePath,fullName:s});return}let c=await ko({message:"T\xEAn workspace m\u1EDBi (s\u1EBD t\u1EA1o repo new):",validate:l=>l.trim().length>0?!0:"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng"});t.workspaceName=c.trim();continue}let i=await T({taskName:"T\u1EA1o workspace remote tr\xEAn GitHub",reason:r instanceof Error?r.message:String(r),allowSkip:!0,hint:"Tip: sai org? Pass --repo-org=<your-gh-user>. Ho\u1EB7c switch gh: gh auth login."});if(i==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA1o workspace remote.");if(i==="skip"){o.warn("Workspace v\u1EABn s\u1EB5n s\xE0ng local-only. Setup remote sau khi c\u1EA7n.");return}}}async function yo(t,e){let n=t.clientRepo??await re({message:"URL git c\u1EE7a repo:",validate:m=>m.length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),{resolvedRemoteUrl:r}=await dt(n),i=r??n,s=t.teamOwner??await Ke(e),a=Qi(i),c=t.workspaceName??await re({message:"T\xEAn workspace:",default:a}),l=qe(t.workspaceParent??"."),u=await ne(l,c,t.force);await Bn({workspacePath:u,workspaceName:c,srcRemoteUrl:i,teamOwner:s,skipTeamPack:t.skipTeamPack,description:t.description??`Avatar workspace cho ${i}`,packVersion:t.packVersion,packLatest:t.latest,autoYes:t.yes,skipCommit:t.commit===!1,createWorkspaceRemote:t.workspaceRemote,repoVisibility:t.repoVisibility,repoOrg:t.repoOrg,flow:"existing-remote",aiSkip:t.aiSkip,gitnexusSkip:t.gitnexusSkip,ssoEmail:e})}async function vo(t,e){let n=qe(t.folderPath??await re({message:"\u0110\u01B0\u1EDDng d\u1EABn folder hi\u1EC7n c\xF3:",validate:u=>u.length>0?!0:"Path b\u1EAFt bu\u1ED9c"}));await On(n,{presetStrategy:Hn(t),autoYes:t.yes});let r=await wo(n,t);if(r){let u=Rt(r);u.ok||(o.warn(`Remote ${r} kh\xF4ng accessible (${u.reason??"unknown"}).`),r=(await Me({url:r,initialReason:u.reason??"unknown",initialDetail:u.detail,folderPath:n,defaultVisibility:t.repoVisibility})).resolvedUrl)}let i=t.teamOwner??await Ke(e),s=t.workspaceName??`${Wu(n)}-avatar-workspace`,a=t.workspaceName??await re({message:"T\xEAn workspace:",default:s}),c=qe(t.workspaceParent??"."),l=await ne(c,a,t.force);await Bn({workspacePath:l,workspaceName:a,srcRemoteUrl:r??n,teamOwner:i,skipTeamPack:t.skipTeamPack,description:t.description??`Avatar workspace cho folder ${n}`,packVersion:t.packVersion,packLatest:t.latest,autoYes:t.yes,skipCommit:t.commit===!1,createWorkspaceRemote:t.workspaceRemote,repoVisibility:t.repoVisibility,repoOrg:t.repoOrg,flow:"existing-folder",aiSkip:t.aiSkip,gitnexusSkip:t.gitnexusSkip,ssoEmail:e})}async function bo(t,e){await dt();let n=t.workspaceName??await re({message:"T\xEAn d\u1EF1 \xE1n:",validate:m=>m.length>0?!0:"T\xEAn b\u1EAFt bu\u1ED9c"}),r=t.repoVisibility??await qu({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),i=t.teamOwner??await Ke(e),s=qe(t.workspaceParent??"."),a=await ne(s,n,t.force),c=zu(a,"src");await E(a),await E(c),await On(c,{autoYes:!0});let l=Et({folder:c,name:n,visibility:r,org:t.repoOrg});await b(a).init();let u=D(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await b(a).subModule(["add",l.sshUrl,"src"]);let m="HEAD";t.skipTeamPack?u.succeed("Skip team-ai-pack (--skip-team-pack)"):(u.stop(),m=(await Ue(a,t.packVersion,e,t.latest===!0&&!t.packVersion)).pinnedTag??"HEAD",u.succeed(`Pin team-ai-pack v\xE0o ${m}`)),await Wn({workspacePath:a,workspaceName:n,teamOwner:i,description:t.description??`D\u1EF1 \xE1n m\u1EDBi: ${n}`,packVersion:m,autoYes:t.yes,skipCommit:t.commit===!1,createWorkspaceRemote:t.workspaceRemote,repoVisibility:t.repoVisibility,repoOrg:t.repoOrg,flow:"new-project",aiSkip:t.aiSkip,gitnexusSkip:t.gitnexusSkip})}catch(m){throw u.fail("Init workspace th\u1EA5t b\u1EA1i"),m}}import rp from"boxen";import ip from"open";var qn="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",Ju="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",zn="nal.vn",Yu=["openid","email","profile"],Xu="https://oauth2.googleapis.com/device/code",Zu="https://oauth2.googleapis.com/token",Qu="https://oauth2.googleapis.com/revoke";async function xo(){let t=new URLSearchParams({client_id:qn,scope:Yu.join(" ")}),e=await fetch(Xu,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:t});if(!e.ok){let n=await e.text();throw new Error(`Device code request failed (${e.status}): ${n}`)}return await e.json()}async function So(t){let e=new URLSearchParams({client_id:qn,client_secret:Ju,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(Zu,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e});if(n.ok)return await n.json();let r="";try{r=(await n.json()).error??""}catch{r=""}if(r==="authorization_pending"||r==="slow_down")return null;throw r==="access_denied"?new Error("User t\u1EEB ch\u1ED1i quy\u1EC1n truy c\u1EADp"):r==="expired_token"?new Error("H\u1EBFt h\u1EA1n ch\u1EDD (5 ph\xFAt). Ch\u1EA1y l\u1EA1i 'avatar login'."):new Error(`OAuth token endpoint tr\u1EA3 l\u1ED7i: ${r||n.status}`)}function Ao(t){let e=t.split(".");if(e.length!==3)throw new Error("id_token format kh\xF4ng h\u1EE3p l\u1EC7");let n=e[1];if(!n)throw new Error("id_token thi\u1EBFu payload");let r=n.replace(/-/g,"+").replace(/_/g,"/"),i=Buffer.from(r,"base64").toString("utf8");return JSON.parse(i)}var tp=new Set(["https://accounts.google.com","accounts.google.com"]),ep=60;function np(t){if(!tp.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==qn)throw new Error("id_token audience kh\xF4ng kh\u1EDBp client ID Avatar. Token c\xF3 th\u1EC3 \u0111\u01B0\u1EE3c sign cho app kh\xE1c.");let e=Math.floor(Date.now()/1e3);if(t.exp+ep<e){let n=e-t.exp;throw new Error(`id_token \u0111\xE3 h\u1EBFt h\u1EA1n ${n}s tr\u01B0\u1EDBc. Login l\u1EA1i \u0111\u1EC3 l\u1EA5y token m\u1EDBi.`)}if(t.hd!==zn)throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u @${zn}). Nh\u1EADn: ${t.email}`);if(!t.email_verified)throw new Error("Email ch\u01B0a \u0111\u01B0\u1EE3c Google verify")}var Co=np;function Po(t,e){let n=new Date(Date.now()+t.expires_in*1e3).toISOString();return{email:e.email,name:e.name??e.email,access_token:t.access_token,refresh_token:t.refresh_token,expires_at:n,id_token:t.id_token}}async function $o(t){let e=new URLSearchParams({token:t});await fetch(Qu,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e}).catch(()=>{})}function Eo(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.searchParams.set("hd",zn),e.toString()}function Ro(t){t.command("login").description("\u0110\u0103ng nh\u1EADp Google SSO (workspace @nal.vn)").option("--reset","X\xF3a credential c\u0169 v\xE0 \u0111\u0103ng nh\u1EADp l\u1EA1i").action(async e=>{try{await Jn(e)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Jn(t){if($t({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await er(),await v("login_reset");else{let g=await tt();if(g&&!et(g)){o.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=D("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await xo(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let r=Eo(n),i=[`1. Truy c\u1EADp: ${p.cyan(n.verification_url)}`,`2. Nh\u1EADp code: ${p.bold.yellow(n.user_code)}`,"",`Ho\u1EB7c Avatar t\u1EF1 m\u1EDF browser, click ${p.green("Allow")}...`].join(`
131
+ `);process.stdout.write(`${rp(i,{padding:1,borderStyle:"round"})}
132
+ `),ip(r).catch(()=>{o.dim("(Kh\xF4ng m\u1EDF \u0111\u01B0\u1EE3c browser t\u1EF1 \u0111\u1ED9ng \u2014 copy URL \u1EDF tr\xEAn)")});let s=D("\u0110ang ch\u1EDD x\xE1c nh\u1EADn trong browser..."),a=n.interval*1e3,c=Date.now()+n.expires_in*1e3,l=null;for(;Date.now()<c;){await op(a);try{if(l=await So(n.device_code),l)break}catch(g){throw s.fail("X\xE1c th\u1EF1c th\u1EA5t b\u1EA1i"),g}}l||(s.fail("H\u1EBFt h\u1EA1n ch\u1EDD (5 ph\xFAt). Ch\u1EA1y l\u1EA1i 'avatar login'."),process.exit(1)),s.succeed("\u0110\xE3 nh\u1EADn token t\u1EEB Google");let u=Ao(l.id_token);try{Co(u)}catch(g){throw await $o(l.access_token),g}let m=Po(l,u);await tr(m),await v("login",m.email),o.success(`X\xE1c th\u1EF1c th\xE0nh c\xF4ng: ${m.email}`),o.success(`Verify hosted domain: ${u.hd} \u2713`),o.success(`L\u01B0u credential v\xE0o ${Q} (chmod 600)`)}function op(t){return new Promise(e=>setTimeout(e,t))}function To(t){t.command("init").description("Kh\u1EDFi t\u1EA1o Avatar \u2014 3 flow t\u1EF1 nh\u1EADn di\u1EC7n (repo / folder / new)").option("--project-status <val>","existing-remote | existing-folder | new-project").option("--folder-path <path>","\u0110\u01B0\u1EDDng d\u1EABn folder hi\u1EC7n c\xF3 (flow existing-folder)").option("--create-remote","Force t\u1EA1o remote qua gh (flow existing-folder ho\u1EB7c new-project)").option("--repo-visibility <val>","private (m\u1EB7c \u0111\u1ECBnh) | public").option("--repo-org <name>","GitHub org/owner cho repo m\u1EDBi").option("--client-repo <url>","URL git remote (flow existing-remote)").option("--workspace-name <name>","T\xEAn workspace").option("--workspace-parent <path>","Th\u01B0 m\u1EE5c cha t\u1EA1o workspace (m\u1EB7c \u0111\u1ECBnh . \u2014 CWD)").option("--pack-version <tag>","Pin team-ai-pack v\xE0o tag c\u1EE5 th\u1EC3").option("--latest","Pull HEAD c\u1EE7a team-ai-pack main branch (b\u1ECF qua tag SemVer, bleeding-edge)").option("--team-owner <email>","Email team owner (b\u1ECF qua prompt)").option("--description <text>","M\xF4 t\u1EA3 1 d\xF2ng c\u1EE7a d\u1EF1 \xE1n").option("--skip-scan","B\u1ECF qua project-scanner sau scaffold").option("--skip-team-pack","B\u1ECF qua submodule team-ai-pack (test mode)").option("--force","B\u1ECF qua prompt khi workspace path \u0111\xE3 t\u1ED3n t\u1EA1i").option("--yes","Auto-confirm t\u1EA5t c\u1EA3 prompt").option("--no-commit","Skip commit workspace initial state (m\u1EB7c \u0111\u1ECBnh LU\xD4N commit)").option("--workspace-remote","T\u1EA1o GitHub remote cho workspace root (default: prompt)").option("--ai-skip","B\u1ECF qua phase AI setup (CI/test mode \u2014 ch\u1EA1y `avatar ai setup` sau)").option("--gitnexus-skip","B\u1ECF qua phase GitNexus setup (M10 \u2014 ch\u1EA1y `avatar gitnexus install` sau)").option("--bootstrap-strategy <s>","X\u1EED l\xFD folder dirty: stash | commit-all | skip | branch (default: prompt)").option("--preserve-uncommitted","Alias cho --bootstrap-strategy=stash (gi\u1EEF changes user)").option("--mode <mode>","[DEPRECATED] D\xF9ng --project-status thay th\u1EBF").action(async e=>{try{await ap(e)}catch(n){n instanceof ee&&(o.dim(n.message),process.exit(0)),n instanceof mt&&(o.dim(n.message),process.exit(0)),n instanceof Xt&&(o.dim(n.message),process.exit(0)),n instanceof x&&(o.dim(n.message),process.exit(0)),o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function ap(t){t.yes||$t({tagline:"Kh\u1EDFi t\u1EA1o Avatar trong d\u1EF1 \xE1n c\u1EE7a b\u1EA1n"}),t.mode&&o.warn("Flag --mode \u0111\xE3 deprecated t\u1EEB v1.1. D\xF9ng --project-status thay th\u1EBF.");let e=await tt();for(;!e||et(e);){o.info("Ch\u01B0a \u0111\u0103ng nh\u1EADp \u2014 ch\u1EA1y login flow tr\u01B0\u1EDBc khi init...");try{await Jn({})}catch(i){o.warn(`Login fail: ${i.message}`)}if(e=await tt(),e&&!et(e))break;if(await T({taskName:"\u0110\u0103ng nh\u1EADp SSO Google",reason:"Token ch\u01B0a \u0111\u01B0\u1EE3c t\u1EA1o ho\u1EB7c \u0111\xE3 h\u1EBFt h\u1EA1n.",allowSkip:!1,hint:"\u0110\u1EA3m b\u1EA3o b\u1EA1n ch\u1ECDn 'Allow' tr\xEAn browser v\xE0 d\xF9ng email @nal.vn."})==="abort")throw new x("User abort t\u1EA1i b\u01B0\u1EDBc login. Ch\u1EA1y 'avatar login' tay r\u1ED3i 'avatar init' l\u1EA1i.")}switch(t.projectStatus??await cp()){case"existing-remote":await yo(t,e.email);break;case"existing-folder":await vo(t,e.email);break;case"new-project":await bo(t,e.email);break}}async function cp(){return await sp({message:"T\xECnh tr\u1EA1ng d\u1EF1 \xE1n c\u1EE7a b\u1EA1n?",choices:[{name:"1. \u0110\xE3 c\xF3 repo git remote (URL c\xF3 s\u1EB5n)",value:"existing-remote"},{name:"2. \u0110\xE3 c\xF3 folder code local",value:"existing-folder"},{name:"3. D\u1EF1 \xE1n m\u1EDBi ho\xE0n to\xE0n",value:"new-project"}]})}function _(t,e){return()=>{process.stdout.write(`${p.yellow("\u23F3")} ${p.bold(`avatar ${t}`)} \u2014 ch\u01B0a implement \u1EDF milestone hi\u1EC7n t\u1EA1i.
133
133
  `),e&&process.stdout.write(` D\u1EF1 ki\u1EBFn: ${p.cyan(e)}
134
134
  `),process.stdout.write(` Spec \u0111\xE3 c\xF3 trong avatar-cli-implementation_4.html.
135
- `),process.exit(0)}}function Go(t){t.command("mcp-run <tool-id>",{hidden:!0}).description("[internal] Spawn MCP v\u1EDBi secrets injected (M09)").action(I("mcp-run","Milestone 09"))}k();import{join as Op}from"path";import Lo from"boxen";var Np=".claude/pack";function Uo(t){t.command("pack").description("Qu\u1EA3n l\xFD team-ai-pack submodule (status, ...)").command("status").description("Hi\u1EC3n th\u1ECB tag pack \u0111ang pin (default offline; --fetch \u0111\u1EC3 check remote)").option("--json","Output JSON cho script").option("--fetch","Fetch remote tags tr\u01B0\u1EDBc khi so s\xE1nh (ch\u1EADm th\xEAm 1-5s)").action(async n=>{try{let r=await Mp(process.cwd(),n.fetch===!0);n.json?process.stdout.write(`${JSON.stringify(r,null,2)}
136
- `):jp(r)}catch(r){o.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function Mp(t,e){let n=Op(t,Np);if(!await d(n)||!await Ct(n))return{installed:!1,currentTag:null,latestTag:null,upToDate:null,fetched:!1};let r=await Ke(t).catch(()=>null),i=!1;if(e)try{await b(n).fetch(["--tags","origin"]),i=!0}catch{i=!1}let s=await B(n).catch(()=>[]),a=q(s);return{installed:!0,currentTag:r,latestTag:a,upToDate:r&&a?r===a:null,fetched:i}}function jp(t){if(!t.installed){let s=[`${p.bold("team-ai-pack")} \xB7 ${p.yellow("ch\u01B0a c\xE0i")}`,"\u2500".repeat(48),p.dim("Ch\u1EA1y `avatar init` ho\u1EB7c `avatar sync` \u0111\u1EC3 c\xE0i pack.")];process.stdout.write(`${Lo(s.join(`
135
+ `),process.exit(0)}}function _o(t){t.command("mcp-run <tool-id>",{hidden:!0}).description("[internal] Spawn MCP v\u1EDBi secrets injected (M09)").action(_("mcp-run","Milestone 09"))}k();import{join as lp}from"path";import Io from"boxen";var up=".claude/pack";function Oo(t){t.command("pack").description("Qu\u1EA3n l\xFD team-ai-pack submodule (status, ...)").command("status").description("Hi\u1EC3n th\u1ECB tag pack \u0111ang pin (default offline; --fetch \u0111\u1EC3 check remote)").option("--json","Output JSON cho script").option("--fetch","Fetch remote tags tr\u01B0\u1EDBc khi so s\xE1nh (ch\u1EADm th\xEAm 1-5s)").action(async n=>{try{let r=await pp(process.cwd(),n.fetch===!0);n.json?process.stdout.write(`${JSON.stringify(r,null,2)}
136
+ `):mp(r)}catch(r){o.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function pp(t,e){let n=lp(t,up);if(!await d(n)||!await Ct(n))return{installed:!1,currentTag:null,latestTag:null,upToDate:null,fetched:!1};let r=await Ge(t).catch(()=>null),i=!1;if(e)try{await b(n).fetch(["--tags","origin"]),i=!0}catch{i=!1}let s=await K(n).catch(()=>[]),a=z(s);return{installed:!0,currentTag:r,latestTag:a,upToDate:r&&a?r===a:null,fetched:i}}function mp(t){if(!t.installed){let s=[`${p.bold("team-ai-pack")} \xB7 ${p.yellow("ch\u01B0a c\xE0i")}`,"\u2500".repeat(48),p.dim("Ch\u1EA1y `avatar init` ho\u1EB7c `avatar sync` \u0111\u1EC3 c\xE0i pack.")];process.stdout.write(`${Io(s.join(`
137
137
  `),{padding:1,borderStyle:"round"})}
138
- `);return}let e=t.currentTag??p.yellow("(unknown)"),n=t.latestTag??p.dim(t.fetched?"(no tags)":"(kh\xF4ng fetch)"),r;t.upToDate===!0?r=p.green("\u2713 \u0110ang d\xF9ng tag m\u1EDBi nh\u1EA5t"):t.upToDate===!1?r=`${p.yellow("\u26A0 C\xF3 version m\u1EDBi")} ${p.dim("\u2192 avatar sync")}`:r=p.dim("(kh\xF4ng so s\xE1nh \u0111\u01B0\u1EE3c)");let i=[`${p.bold("team-ai-pack status")}`,"\u2500".repeat(48),`${p.dim("Tag hi\u1EC7n t\u1EA1i:")} ${e}`,`${p.dim("Tag m\u1EDBi nh\u1EA5t:")} ${n}`,`${p.dim("Tr\u1EA1ng th\xE1i:")} ${r}`];process.stdout.write(`${Lo(i.join(`
138
+ `);return}let e=t.currentTag??p.yellow("(unknown)"),n=t.latestTag??p.dim(t.fetched?"(no tags)":"(kh\xF4ng fetch)"),r;t.upToDate===!0?r=p.green("\u2713 \u0110ang d\xF9ng tag m\u1EDBi nh\u1EA5t"):t.upToDate===!1?r=`${p.yellow("\u26A0 C\xF3 version m\u1EDBi")} ${p.dim("\u2192 avatar sync")}`:r=p.dim("(kh\xF4ng so s\xE1nh \u0111\u01B0\u1EE3c)");let i=[`${p.bold("team-ai-pack status")}`,"\u2500".repeat(48),`${p.dim("Tag hi\u1EC7n t\u1EA1i:")} ${e}`,`${p.dim("Tag m\u1EDBi nh\u1EA5t:")} ${n}`,`${p.dim("Tr\u1EA1ng th\xE1i:")} ${r}`];process.stdout.write(`${Io(i.join(`
139
139
  `),{padding:1,borderStyle:"round"})}
140
- `)}function Ho(t){t.command("restore").description("Kh\xF4i ph\u1EE5c .claude/pack/ t\u1EEB backup (M08)").option("--backup <name>","T\xEAn backup folder trong .claude/_backup/").option("--list","Li\u1EC7t k\xEA c\xE1c backup hi\u1EC7n c\xF3").action(I("restore","Milestone 08"))}function Do(t){t.command("review").description("Review pending proposals t\u1EEB avatar scan (M08)").option("--accept-all","Approve m\u1ECDi pending kh\xF4ng h\u1ECFi (CI mode)").option("--reject-all","X\xF3a m\u1ECDi pending kh\xF4ng h\u1ECFi").action(I("review","Milestone 08"))}function Fo(t){t.command("scan").description("Ch\u1EA1y project scanner v\xE0 \u0111\u1EC1 xu\u1EA5t knowledge update (M06)").option("--incremental","Ch\u1EC9 scan c\xE1c file thay \u0111\u1ED5i t\u1EEB commit cu\u1ED1i").option("--full","Scan to\xE0n b\u1ED9 d\u1EF1 \xE1n (default)").option("--scanners <list>","tech-stack,conventions,architecture,domain,git-pattern").option("--quiet","Ch\u1EA1y ng\u1EA7m, \xEDt output (d\xF9ng cho git hook)").action(I("scan","Milestone 06"))}import{confirm as Nt,password as Gp}from"@inquirer/prompts";Wn();Ht();async function ft(){let t=vt(process.cwd());return t||(o.error("Kh\xF4ng t\xECm th\u1EA5y Avatar workspace. Ch\u1EA1y 'avatar init' tr\u01B0\u1EDBc, ho\u1EB7c cd v\xE0o workspace."),process.exit(1)),t}function Ko(t){return t.length<=8?"****":`${t.slice(0,4)}...${t.slice(-4)}`}async function Lp(t){o.info(`Workspace: ${t}`);let e=await bt();if(e.installed)o.success(`\u2713 direnv ${e.version??"(installed)"}`);else if(o.warn("direnv ch\u01B0a c\xE0i. Avatar secrets d\xF9ng direnv \u0111\u1EC3 auto-load .envrc khi cd v\xE0o project."),await Nt({message:"C\xE0i direnv ngay? (qua brew/apt/dnf/pacman t\xF9y OS)",default:!0})){o.info("\u0110ang c\xE0i direnv...");let s=await he();s.success?o.success(`\u2713 ${s.message}`):(o.warn(`! ${s.message}`),o.info(`Manual install: ${s.manualInstructionsUrl}`),o.info("Ti\u1EBFp t\u1EE5c \u2014 .envrc s\u1EBD ghi nh\u01B0ng c\u1EA7n direnv \u0111\u1EC3 auto-load."))}if(e.shellHookInZshrc)o.dim("\u2713 direnv shell hook \u0111\xE3 trong ~/.zshrc");else if(o.warn("~/.zshrc ch\u01B0a c\xF3 direnv shell hook \u2192 .envrc s\u1EBD kh\xF4ng t\u1EF1 load."),await Nt({message:'Add eval "$(direnv hook zsh)" v\xE0o ~/.zshrc?',default:!0})){let s=ke();s.modified?(o.success(`\u2713 Added direnv hook to ${s.zshrcPath}`),o.info("Reload shell: 'source ~/.zshrc' ho\u1EB7c m\u1EDF terminal m\u1EDBi")):o.dim("Skip \u2014 hook \u0111\xE3 t\u1ED3n t\u1EA1i ho\u1EB7c kh\xF4ng write \u0111\u01B0\u1EE3c")}let n=await _t(t);if(n.keys.length>0&&(o.warn(`Ph\xE1t hi\u1EC7n ${n.keys.length} secret plaintext trong .claude/settings.json: ${n.keys.join(", ")}`),o.warn("Risk: leak n\u1EBFu file commit l\xEAn git."),await Nt({message:"Migrate sang .envrc (direnv) ngay?",default:!0}))){let s=await Xe(t);o.success(`\u2713 Migrated ${s.migratedKeys.length} secrets: ${s.migratedKeys.join(", ")}`),s.settingsJsonBackupPath&&o.dim(` settings.json backup: ${s.settingsJsonBackupPath}`)}V(t).has("ANTHROPIC_API_KEY")||await Nt({message:"Set ANTHROPIC_API_KEY cho project n\xE0y ngay?",default:!0})&&await Vo(t,"ANTHROPIC_API_KEY",{validateAnthropic:!0}),o.success("\u2713 Secrets setup done."),o.dim(` .envrc: ${xt(t)}`),o.dim(" View: avatar secrets list")}async function Up(){let t=await ft();await Lp(t)}async function Vo(t,e,n={}){let r=await Gp({message:`Nh\u1EADp value cho ${e} (input \u1EA9n):`,mask:"*",validate:s=>s.trim().length>0?!0:"Value b\u1EAFt bu\u1ED9c"});if(n.validateAnthropic&&e==="ANTHROPIC_API_KEY"){o.info(`Verify key (${ue(r)}) qua Anthropic /v1/models...`);try{let s=await pe(r);o.success(`\u2713 Key valid (${s.length} models available)`)}catch(s){if(o.error(`Key kh\xF4ng valid: ${s instanceof Error?s.message:s}`),!await Nt({message:"V\u1EABn ghi key (b\u1ECF qua validation)?",default:!1}))return}}let i=await at(t,{[e]:r});o.success(`\u2713 Set ${e} (.envrc: ${i.varsBefore} \u2192 ${i.varsAfter} vars)`),i.direnvAllowOk===!0?o.dim(" direnv allow ok \u2014 s\u1EBD auto-load l\u1EA7n cd k\u1EBF ti\u1EBFp"):i.direnvAllowOk===null?o.warn(" direnv missing \u2192 file ghi OK nh\u01B0ng kh\xF4ng auto-load. C\xE0i direnv ho\u1EB7c source th\u1EE7 c\xF4ng."):o.warn(" direnv allow failed \u2014 ch\u1EA1y 'direnv allow' th\u1EE7 c\xF4ng trong workspace")}async function Hp(t){let e=await ft();await Vo(e,t,{validateAnthropic:t==="ANTHROPIC_API_KEY"})}async function Dp(t){let e=await ft(),r=V(e).get(t);r===void 0&&(o.error(`Secret '${t}' kh\xF4ng t\u1ED3n t\u1EA1i trong .envrc.`),process.exit(1)),process.stdout.isTTY?(console.log(Ko(r)),o.dim(`(masked. \u0110\u1EC3 in raw: 'avatar secrets get ${t} | cat')`)):console.log(r)}async function Fp(){let t=await ft(),e=V(t);if(e.size===0){o.info("Ch\u01B0a c\xF3 secret n\xE0o trong .envrc Avatar block."),o.dim("Set: avatar secrets set <NAME>");return}o.info(`${e.size} secrets in ${xt(t)}:`);for(let[n,r]of[...e.entries()].sort())console.log(` ${n.padEnd(30)} = ${Ko(r)}`)}async function Kp(t){let e=await ft();if(!V(e).has(t)){o.warn(`Secret '${t}' kh\xF4ng t\u1ED3n t\u1EA1i \u2014 no-op.`);return}if(!await Nt({message:`X\xF3a secret '${t}' kh\u1ECFi .envrc?`,default:!1})){o.dim("H\u1EE7y.");return}let i=await at(e,{[t]:null});o.success(`\u2713 Removed ${t} (.envrc: ${i.varsBefore} \u2192 ${i.varsAfter} vars)`)}async function Vp(){let t=await ft();o.info(`Workspace: ${t}
141
- `);let e=await bt();o.info(`direnv installed: ${e.installed?`\u2713 ${e.version??""}`:"\u2717"}`),o.info(`direnv shell hook (~/.zshrc): ${e.shellHookInZshrc?"\u2713":"\u2717"}`);let n=xt(t),r=V(t);o.info(`.envrc: ${r.size>0?`\u2713 ${r.size} secrets at ${n}`:"\u2717 (no Avatar block)"}`);let i=await _t(t);i.keys.length>0&&o.warn(`\u26A0 Plaintext secrets in settings.json: ${i.keys.join(", ")} \u2014 run 'avatar secrets migrate'`);let s=r.get("ANTHROPIC_API_KEY");if(s){o.info(`Verifying ANTHROPIC_API_KEY (${ue(s)})...`);try{let a=await pe(s);o.success(`\u2713 Anthropic key valid (${a.length} models)`)}catch(a){o.error(`\u2717 Anthropic key invalid: ${a instanceof Error?a.message:a}`)}}}async function Bp(){let t=await ft(),e=await _t(t);if(e.keys.length===0){o.info("Kh\xF4ng c\xF3 plaintext secret trong settings.json \u2014 nothing to migrate.");return}o.warn(`Migrating ${e.keys.length} secrets: ${e.keys.join(", ")}`);let n=await Xe(t);o.success(`\u2713 Migrated ${n.migratedKeys.length} secrets to .envrc`),n.settingsJsonBackupPath&&o.dim(` Backup: ${n.settingsJsonBackupPath}`)}function Bo(t){let e=t.command("secrets").description("Qu\u1EA3n l\xFD per-project secrets via direnv .envrc (v1.19.0)"),n=i=>async()=>{try{await i()}catch(s){o.error(`secrets failed: ${s instanceof Error?s.message:s}`),process.exit(1)}},r=i=>async s=>{try{await i(s)}catch(a){o.error(`secrets failed: ${a instanceof Error?a.message:a}`),process.exit(1)}};e.command("setup").description("Interactive wizard: install direnv + migrate + prompt new key").action(n(Up)),e.command("set <NAME>").description("Set/update one secret in .envrc (hidden input prompt)").action(r(Hp)),e.command("get <NAME>").description("Print one secret value (masked on TTY, raw when piped)").action(r(Dp)),e.command("list").description("List Avatar-managed secret names (values masked)").action(n(Fp)),e.command("rm <NAME>").description("Remove one secret from .envrc").action(r(Kp)),e.command("check").description("Health: direnv installed, .envrc exists, Anthropic key validates").action(n(Vp)),e.command("migrate").description("Migrate plaintext secrets from settings.json \u2192 .envrc").action(n(Bp))}import{promises as Yp}from"fs";import{join as Qe}from"path";import Jp from"boxen";k();k();import{promises as Wp}from"fs";import{join as zp}from"path";var qp="_backup";async function Wo(t){let e=zp(t,".claude",qp);return await d(e)?(await Wp.readdir(e,{withFileTypes:!0})).filter(r=>r.isDirectory()).map(r=>r.name).sort().reverse():[]}function zo(t){t.command("status").description("Snapshot t\u1EE9c th\xEC: project, pack version, pending, backup").option("--json","Output JSON cho script").action(async e=>{try{let n=await Xp(process.cwd());e.json?process.stdout.write(`${JSON.stringify(n,null,2)}
142
- `):tm(n)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Xp(t){let e=t.split("/").filter(Boolean).pop()??"unknown",n=Qe(t,".claude");if(!await d(n))return{projectName:e,cliVersion:H(),packVersion:null,pendingCount:0,backupCount:0,techStackSummary:"(Avatar ch\u01B0a init)",hasAvatar:!1,featuresEnabled:[],featuresAvailableCount:0};let i=Qe(n,"_pending"),[s,a,c,l,u]=await Promise.all([(async()=>await Ct(Qe(n,"pack"))?Ke(t).catch(()=>null):null)(),(async()=>await d(i)?(await Yp.readdir(i)).filter(g=>g.endsWith(".diff.md")).length:0)(),Wo(t).then(m=>m.length).catch(()=>0),Zp(n).catch(()=>"(read error)"),zt(t).catch(()=>({available:[],enabled:[]}))]);return{projectName:e,cliVersion:H(),packVersion:s,pendingCount:a,backupCount:c,techStackSummary:l,hasAvatar:!0,featuresEnabled:u.enabled,featuresAvailableCount:u.available.length}}async function Zp(t){let e=Qe(t,"project","tech-stack.md");return await d(e)?(await D(e)).split(`
143
- `).find(i=>i.trim()&&!i.startsWith("#")&&!i.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function Qp(t){return t.featuresEnabled.length>0?`${t.featuresEnabled.join(", ")} (${t.featuresEnabled.length}/${t.featuresAvailableCount} available)`:t.featuresAvailableCount>0?`none enabled (${t.featuresAvailableCount} available \u2014 avatar feature list)`:"none"}function tm(t){let e=[`${p.bold("Avatar Status")} \xB7 ${p.cyan(t.projectName)}`,"\u2500".repeat(48),`${p.dim("CLI version:")} ${t.cliVersion}`,`${p.dim("Pack version:")} ${t.packVersion??p.yellow("not installed")}`,`${p.dim("Pending changes:")} ${t.pendingCount}${t.pendingCount>0?p.dim(" (avatar review)"):""}`,`${p.dim("Backups:")} ${t.backupCount}`,`${p.dim("Tech stack:")} ${t.techStackSummary}`,`${p.dim("Features:")} ${Qp(t)}`];process.stdout.write(`${Jp(e.join(`
140
+ `)}function No(t){t.command("restore").description("Kh\xF4i ph\u1EE5c .claude/pack/ t\u1EEB backup (M08)").option("--backup <name>","T\xEAn backup folder trong .claude/_backup/").option("--list","Li\u1EC7t k\xEA c\xE1c backup hi\u1EC7n c\xF3").action(_("restore","Milestone 08"))}function Mo(t){t.command("review").description("Review pending proposals t\u1EEB avatar scan (M08)").option("--accept-all","Approve m\u1ECDi pending kh\xF4ng h\u1ECFi (CI mode)").option("--reject-all","X\xF3a m\u1ECDi pending kh\xF4ng h\u1ECFi").action(_("review","Milestone 08"))}function jo(t){t.command("scan").description("Ch\u1EA1y project scanner v\xE0 \u0111\u1EC1 xu\u1EA5t knowledge update (M06)").option("--incremental","Ch\u1EC9 scan c\xE1c file thay \u0111\u1ED5i t\u1EEB commit cu\u1ED1i").option("--full","Scan to\xE0n b\u1ED9 d\u1EF1 \xE1n (default)").option("--scanners <list>","tech-stack,conventions,architecture,domain,git-pattern").option("--quiet","Ch\u1EA1y ng\u1EA7m, \xEDt output (d\xF9ng cho git hook)").action(_("scan","Milestone 06"))}import{confirm as Nt,password as dp}from"@inquirer/prompts";Vn();Ut();async function ft(){let t=vt(process.cwd());return t||(o.error("Kh\xF4ng t\xECm th\u1EA5y Avatar workspace. Ch\u1EA1y 'avatar init' tr\u01B0\u1EDBc, ho\u1EB7c cd v\xE0o workspace."),process.exit(1)),t}function Lo(t){return t.length<=8?"****":`${t.slice(0,4)}...${t.slice(-4)}`}async function gp(t){o.info(`Workspace: ${t}`);let e=await bt();if(e.installed)o.success(`\u2713 direnv ${e.version??"(installed)"}`);else if(o.warn("direnv ch\u01B0a c\xE0i. Avatar secrets d\xF9ng direnv \u0111\u1EC3 auto-load .envrc khi cd v\xE0o project."),await Nt({message:"C\xE0i direnv ngay? (qua brew/apt/dnf/pacman t\xF9y OS)",default:!0})){o.info("\u0110ang c\xE0i direnv...");let s=await de();s.success?o.success(`\u2713 ${s.message}`):(o.warn(`! ${s.message}`),o.info(`Manual install: ${s.manualInstructionsUrl}`),o.info("Ti\u1EBFp t\u1EE5c \u2014 .envrc s\u1EBD ghi nh\u01B0ng c\u1EA7n direnv \u0111\u1EC3 auto-load."))}if(e.shellHookInZshrc)o.dim("\u2713 direnv shell hook \u0111\xE3 trong ~/.zshrc");else if(o.warn("~/.zshrc ch\u01B0a c\xF3 direnv shell hook \u2192 .envrc s\u1EBD kh\xF4ng t\u1EF1 load."),await Nt({message:'Add eval "$(direnv hook zsh)" v\xE0o ~/.zshrc?',default:!0})){let s=ge();s.modified?(o.success(`\u2713 Added direnv hook to ${s.zshrcPath}`),o.info("Reload shell: 'source ~/.zshrc' ho\u1EB7c m\u1EDF terminal m\u1EDBi")):o.dim("Skip \u2014 hook \u0111\xE3 t\u1ED3n t\u1EA1i ho\u1EB7c kh\xF4ng write \u0111\u01B0\u1EE3c")}let n=await It(t);if(n.keys.length>0&&(o.warn(`Ph\xE1t hi\u1EC7n ${n.keys.length} secret plaintext trong .claude/settings.json: ${n.keys.join(", ")}`),o.warn("Risk: leak n\u1EBFu file commit l\xEAn git."),await Nt({message:"Migrate sang .envrc (direnv) ngay?",default:!0}))){let s=await ze(t);o.success(`\u2713 Migrated ${s.migratedKeys.length} secrets: ${s.migratedKeys.join(", ")}`),s.settingsJsonBackupPath&&o.dim(` settings.json backup: ${s.settingsJsonBackupPath}`)}V(t).has("ANTHROPIC_API_KEY")||await Nt({message:"Set ANTHROPIC_API_KEY cho project n\xE0y ngay?",default:!0})&&await Ho(t,"ANTHROPIC_API_KEY",{validateAnthropic:!0}),o.success("\u2713 Secrets setup done."),o.dim(` .envrc: ${xt(t)}`),o.dim(" View: avatar secrets list")}async function fp(){let t=await ft();await gp(t)}async function Ho(t,e,n={}){let r=await dp({message:`Nh\u1EADp value cho ${e} (input \u1EA9n):`,mask:"*",validate:s=>s.trim().length>0?!0:"Value b\u1EAFt bu\u1ED9c"});if(n.validateAnthropic&&e==="ANTHROPIC_API_KEY"){o.info(`Verify key (${le(r)}) qua Anthropic /v1/models...`);try{let s=await ue(r);o.success(`\u2713 Key valid (${s.length} models available)`)}catch(s){if(o.error(`Key kh\xF4ng valid: ${s instanceof Error?s.message:s}`),!await Nt({message:"V\u1EABn ghi key (b\u1ECF qua validation)?",default:!1}))return}}let i=await st(t,{[e]:r});o.success(`\u2713 Set ${e} (.envrc: ${i.varsBefore} \u2192 ${i.varsAfter} vars)`),i.direnvAllowOk===!0?o.dim(" direnv allow ok \u2014 s\u1EBD auto-load l\u1EA7n cd k\u1EBF ti\u1EBFp"):i.direnvAllowOk===null?o.warn(" direnv missing \u2192 file ghi OK nh\u01B0ng kh\xF4ng auto-load. C\xE0i direnv ho\u1EB7c source th\u1EE7 c\xF4ng."):o.warn(" direnv allow failed \u2014 ch\u1EA1y 'direnv allow' th\u1EE7 c\xF4ng trong workspace")}async function hp(t){let e=await ft();await Ho(e,t,{validateAnthropic:t==="ANTHROPIC_API_KEY"})}async function kp(t){let e=await ft(),r=V(e).get(t);r===void 0&&(o.error(`Secret '${t}' kh\xF4ng t\u1ED3n t\u1EA1i trong .envrc.`),process.exit(1)),process.stdout.isTTY?(console.log(Lo(r)),o.dim(`(masked. \u0110\u1EC3 in raw: 'avatar secrets get ${t} | cat')`)):console.log(r)}async function wp(){let t=await ft(),e=V(t);if(e.size===0){o.info("Ch\u01B0a c\xF3 secret n\xE0o trong .envrc Avatar block."),o.dim("Set: avatar secrets set <NAME>");return}o.info(`${e.size} secrets in ${xt(t)}:`);for(let[n,r]of[...e.entries()].sort())console.log(` ${n.padEnd(30)} = ${Lo(r)}`)}async function yp(t){let e=await ft();if(!V(e).has(t)){o.warn(`Secret '${t}' kh\xF4ng t\u1ED3n t\u1EA1i \u2014 no-op.`);return}if(!await Nt({message:`X\xF3a secret '${t}' kh\u1ECFi .envrc?`,default:!1})){o.dim("H\u1EE7y.");return}let i=await st(e,{[t]:null});o.success(`\u2713 Removed ${t} (.envrc: ${i.varsBefore} \u2192 ${i.varsAfter} vars)`)}async function vp(){let t=await ft();o.info(`Workspace: ${t}
141
+ `);let e=await bt();o.info(`direnv installed: ${e.installed?`\u2713 ${e.version??""}`:"\u2717"}`),o.info(`direnv shell hook (~/.zshrc): ${e.shellHookInZshrc?"\u2713":"\u2717"}`);let n=xt(t),r=V(t);o.info(`.envrc: ${r.size>0?`\u2713 ${r.size} secrets at ${n}`:"\u2717 (no Avatar block)"}`);let i=await It(t);i.keys.length>0&&o.warn(`\u26A0 Plaintext secrets in settings.json: ${i.keys.join(", ")} \u2014 run 'avatar secrets migrate'`);let s=r.get("ANTHROPIC_API_KEY");if(s){o.info(`Verifying ANTHROPIC_API_KEY (${le(s)})...`);try{let a=await ue(s);o.success(`\u2713 Anthropic key valid (${a.length} models)`)}catch(a){o.error(`\u2717 Anthropic key invalid: ${a instanceof Error?a.message:a}`)}}}async function bp(){let t=await ft(),e=await It(t);if(e.keys.length===0){o.info("Kh\xF4ng c\xF3 plaintext secret trong settings.json \u2014 nothing to migrate.");return}o.warn(`Migrating ${e.keys.length} secrets: ${e.keys.join(", ")}`);let n=await ze(t);o.success(`\u2713 Migrated ${n.migratedKeys.length} secrets to .envrc`),n.settingsJsonBackupPath&&o.dim(` Backup: ${n.settingsJsonBackupPath}`)}function Go(t){let e=t.command("secrets").description("Qu\u1EA3n l\xFD per-project secrets via direnv .envrc (v1.19.0)"),n=i=>async()=>{try{await i()}catch(s){o.error(`secrets failed: ${s instanceof Error?s.message:s}`),process.exit(1)}},r=i=>async s=>{try{await i(s)}catch(a){o.error(`secrets failed: ${a instanceof Error?a.message:a}`),process.exit(1)}};e.command("setup").description("Interactive wizard: install direnv + migrate + prompt new key").action(n(fp)),e.command("set <NAME>").description("Set/update one secret in .envrc (hidden input prompt)").action(r(hp)),e.command("get <NAME>").description("Print one secret value (masked on TTY, raw when piped)").action(r(kp)),e.command("list").description("List Avatar-managed secret names (values masked)").action(n(wp)),e.command("rm <NAME>").description("Remove one secret from .envrc").action(r(yp)),e.command("check").description("Health: direnv installed, .envrc exists, Anthropic key validates").action(n(vp)),e.command("migrate").description("Migrate plaintext secrets from settings.json \u2192 .envrc").action(n(bp))}import{promises as Cp}from"fs";import{join as Je}from"path";import Pp from"boxen";k();k();import{promises as xp}from"fs";import{join as Sp}from"path";var Ap="_backup";async function Uo(t){let e=Sp(t,".claude",Ap);return await d(e)?(await xp.readdir(e,{withFileTypes:!0})).filter(r=>r.isDirectory()).map(r=>r.name).sort().reverse():[]}function Do(t){t.command("status").description("Snapshot t\u1EE9c th\xEC: project, pack version, pending, backup").option("--json","Output JSON cho script").action(async e=>{try{let n=await $p(process.cwd());e.json?process.stdout.write(`${JSON.stringify(n,null,2)}
142
+ `):Tp(n)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function $p(t){let e=t.split("/").filter(Boolean).pop()??"unknown",n=Je(t,".claude");if(!await d(n))return{projectName:e,cliVersion:G(),packVersion:null,pendingCount:0,backupCount:0,techStackSummary:"(Avatar ch\u01B0a init)",hasAvatar:!1,featuresEnabled:[],featuresAvailableCount:0};let i=Je(n,"_pending"),[s,a,c,l,u]=await Promise.all([(async()=>await Ct(Je(n,"pack"))?Ge(t).catch(()=>null):null)(),(async()=>await d(i)?(await Cp.readdir(i)).filter(g=>g.endsWith(".diff.md")).length:0)(),Uo(t).then(m=>m.length).catch(()=>0),Ep(n).catch(()=>"(read error)"),Wt(t).catch(()=>({available:[],enabled:[]}))]);return{projectName:e,cliVersion:G(),packVersion:s,pendingCount:a,backupCount:c,techStackSummary:l,hasAvatar:!0,featuresEnabled:u.enabled,featuresAvailableCount:u.available.length}}async function Ep(t){let e=Je(t,"project","tech-stack.md");return await d(e)?(await U(e)).split(`
143
+ `).find(i=>i.trim()&&!i.startsWith("#")&&!i.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function Rp(t){return t.featuresEnabled.length>0?`${t.featuresEnabled.join(", ")} (${t.featuresEnabled.length}/${t.featuresAvailableCount} available)`:t.featuresAvailableCount>0?`none enabled (${t.featuresAvailableCount} available \u2014 avatar feature list)`:"none"}function Tp(t){let e=[`${p.bold("Avatar Status")} \xB7 ${p.cyan(t.projectName)}`,"\u2500".repeat(48),`${p.dim("CLI version:")} ${t.cliVersion}`,`${p.dim("Pack version:")} ${t.packVersion??p.yellow("not installed")}`,`${p.dim("Pending changes:")} ${t.pendingCount}${t.pendingCount>0?p.dim(" (avatar review)"):""}`,`${p.dim("Backups:")} ${t.backupCount}`,`${p.dim("Tech stack:")} ${t.techStackSummary}`,`${p.dim("Features:")} ${Rp(t)}`];process.stdout.write(`${Pp(e.join(`
144
144
  `),{padding:1,borderStyle:"round"})}
145
- `)}k();import{join as Xo}from"path";k();import{join as qo}from"path";async function em(t,e,n){let r=qo(t,n),i=qo(e,n);if(!await d(r))return"source-missing";if(!await d(i))return"needs-creation";let{promises:s}=await import("fs");return(await s.lstat(i)).isSymbolicLink()?"already-linked":"conflict-real-dir"}async function nm(t,e,n){try{return(await b(t).raw(["log","--oneline",`${e}..${n}`])).split(`
146
- `).map(i=>i.trim()).filter(i=>i.length>0)}catch{return[]}}async function Yo(t,e,n){let r=await Ee(t),i=await Pt(t),s=r??i.slice(0,7),a=await B(t),c=n??q(a)??"HEAD",l=await nm(t,i,c),u=[];for(let m of Bn)u.push({dir:m,status:await em(t,e,m)});return{currentVersion:s,targetVersion:c,commitsBehind:l,mountDirStatuses:u}}async function Jo(t){let e=await St(t);if(e.length!==0){o.info(`Re-apply ${e.length} feature(s) \u0111ang b\u1EADt v\xE0o settings.json...`);for(let n of e)await lt(t,n)}}var tn="main";async function rm(t){let e=process.cwd(),n=Xo(e,".claude"),r=Xo(e,R);await d(r)||(o.error(`team-ai-pack submodule ch\u01B0a \u0111\u01B0\u1EE3c kh\u1EDFi t\u1EA1o \u1EDF ${R}/.
147
- Ch\u1EA1y 'avatar init' \u0111\u1EC3 add submodule, ho\u1EB7c 'git submodule update --init' n\u1EBFu \u0111\xE3 clone repo.`),process.exit(1));try{await b(r).fetch(["--tags","origin"])}catch(l){o.warn(`Kh\xF4ng fetch \u0111\u01B0\u1EE3c tags t\u1EEB origin (${l instanceof Error?l.message:l}). S\u1EBD d\xF9ng tag local hi\u1EC7n c\xF3.`)}let i=t.latest===!0&&!t.version,s=await B(r),a;if(i)a=`${tn} (HEAD)`;else{let l=t.version??q(s);l||(o.error(`Kh\xF4ng t\xECm th\u1EA5y stable SemVer tag (vMAJOR.MINOR.PATCH) trong team-ai-pack submodule.
145
+ `)}k();import{join as Bo}from"path";k();import{join as Fo}from"path";async function _p(t,e,n){let r=Fo(t,n),i=Fo(e,n);if(!await d(r))return"source-missing";if(!await d(i))return"needs-creation";let{promises:s}=await import("fs");return(await s.lstat(i)).isSymbolicLink()?"already-linked":"conflict-real-dir"}async function Ip(t,e,n){try{return(await b(t).raw(["log","--oneline",`${e}..${n}`])).split(`
146
+ `).map(i=>i.trim()).filter(i=>i.length>0)}catch{return[]}}async function Vo(t,e,n){let r=await Ae(t),i=await Pt(t),s=r??i.slice(0,7),a=await K(t),c=n??z(a)??"HEAD",l=await Ip(t,i,c),u=[];for(let m of Fn)u.push({dir:m,status:await _p(t,e,m)});return{currentVersion:s,targetVersion:c,commitsBehind:l,mountDirStatuses:u}}async function Ko(t){let e=await At(t);if(e.length!==0){o.info(`Re-apply ${e.length} feature(s) \u0111ang b\u1EADt v\xE0o settings.json...`);for(let n of e)await lt(t,n)}}var Ye="main";async function Op(t){let e=process.cwd(),n=Bo(e,".claude"),r=Bo(e,R);await d(r)||(o.error(`team-ai-pack submodule ch\u01B0a \u0111\u01B0\u1EE3c kh\u1EDFi t\u1EA1o \u1EDF ${R}/.
147
+ Ch\u1EA1y 'avatar init' \u0111\u1EC3 add submodule, ho\u1EB7c 'git submodule update --init' n\u1EBFu \u0111\xE3 clone repo.`),process.exit(1));try{await b(r).fetch(["--tags","origin"])}catch(l){o.warn(`Kh\xF4ng fetch \u0111\u01B0\u1EE3c tags t\u1EEB origin (${l instanceof Error?l.message:l}). S\u1EBD d\xF9ng tag local hi\u1EC7n c\xF3.`)}let i=t.latest===!0&&!t.version,s=await K(r),a;if(i)a=`${Ye} (HEAD)`;else{let l=t.version??z(s);l||(o.error(`Kh\xF4ng t\xECm th\u1EA5y stable SemVer tag (vMAJOR.MINOR.PATCH) trong team-ai-pack submodule.
148
148
  Tags hi\u1EC7n c\xF3: ${s.length>0?s.join(", "):"(none)"}
149
- Pass --version <tag> r\xF5 r\xE0ng, ho\u1EB7c d\xF9ng --latest \u0111\u1EC3 pull HEAD branch ${tn}.`),process.exit(1)),a=l}if(t.dryRun){let l=await Yo(r,n,a);if(o.info(`Pack version hi\u1EC7n t\u1EA1i: ${l.currentVersion}`),o.info(`Target version: ${l.targetVersion}`),l.commitsBehind.length===0)o.info("\u0110\xE3 \u1EDF target version, kh\xF4ng c\xF3 commit m\u1EDBi.");else{o.info(`Commits c\u1EA7n pull (${l.commitsBehind.length}):`);for(let u of l.commitsBehind.slice(0,20))console.log(` ${u}`);l.commitsBehind.length>20&&console.log(` ... v\xE0 ${l.commitsBehind.length-20} commits kh\xE1c`)}o.info(`
149
+ Pass --version <tag> r\xF5 r\xE0ng, ho\u1EB7c d\xF9ng --latest \u0111\u1EC3 pull HEAD branch ${Ye}.`),process.exit(1)),a=l}if(t.dryRun){let l=await Vo(r,n,a);if(o.info(`Pack version hi\u1EC7n t\u1EA1i: ${l.currentVersion}`),o.info(`Target version: ${l.targetVersion}`),l.commitsBehind.length===0)o.info("\u0110\xE3 \u1EDF target version, kh\xF4ng c\xF3 commit m\u1EDBi.");else{o.info(`Commits c\u1EA7n pull (${l.commitsBehind.length}):`);for(let u of l.commitsBehind.slice(0,20))console.log(` ${u}`);l.commitsBehind.length>20&&console.log(` ... v\xE0 ${l.commitsBehind.length-20} commits kh\xE1c`)}o.info(`
150
150
  Mount dir statuses:`);for(let u of l.mountDirStatuses)console.log(` ${u.dir.padEnd(12)} ${u.status}`);o.info(`
151
- Dry-run done. Kh\xF4ng apply thay \u0111\u1ED5i. B\u1ECF --dry-run \u0111\u1EC3 th\u1EF1c thi.`);return}if(i){o.info(`Pulling HEAD c\u1EE7a branch ${tn} (bleeding-edge mode)...`),await $e(R,tn,e);let l=await Pt(r);o.dim(` HEAD = ${l.slice(0,7)}`),o.warn("\u26A0 --latest mode: workspace pin v\xE0o commit floating, kh\xF4ng reproducible. Chuy\u1EC3n v\u1EC1 tag stable: avatar sync (no flag).")}else o.info(`Checking out ${a} trong submodule...`),await Vt(R,a,e);o.info("Creating symlink farm...");let c=await Je(r,n,t.force===!0);im(c,t.force===!0),o.info("Merging pack settings.json template into project settings.json...");try{let l=await Ae(e);switch(l.action){case"merged":o.success(` \u2713 settings.json merged (${l.changes.join("; ")}). Backup: ${l.backupPath??"n/a"}`);break;case"no-change":o.info(" - settings.json \u0111\xE3 sync v\u1EDBi pack, kh\xF4ng c\xF3 thay \u0111\u1ED5i.");break;case"no-pack-template":o.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip merge.");break}}catch(l){o.warn(` ! Merge settings.json fail: ${l instanceof Error?l.message:l}. Symlinks \u0111\xE3 t\u1EA1o OK, hooks c\xF3 th\u1EC3 ch\u01B0a active. Manual merge n\u1EBFu c\u1EA7n.`)}try{await Jo(e)}catch(l){o.warn(` ! Re-apply features fail: ${l instanceof Error?l.message:l}. Ch\u1EA1y 'avatar feature enable <name>' th\u1EE7 c\xF4ng n\u1EBFu c\u1EA7n.`)}o.success(`Synced team-ai-pack to ${a}.`)}function im(t,e){for(let r of t)switch(r.action){case"created":o.info(` \u2713 ${r.dir} \u2192 symlinked (new)`);break;case"updated":o.info(` \u2713 ${r.dir} \u2192 symlink refreshed`);break;case"backed-up-and-linked":o.info(` \u2713 ${r.dir} \u2192 symlinked (backup: ${r.backupPath})`);break;case"source-missing":o.warn(` - ${r.dir} \u2192 pack kh\xF4ng c\xF3 dir n\xE0y, skip`);break;case"skipped-conflict":o.warn(` ! ${r.dir} \u2192 CONFLICT: existing real dir. D\xF9ng --force \u0111\u1EC3 backup + override (s\u1EBD gi\u1EEF data \u1EDF ${r.dir}.backup-<timestamp>/).`);break}let n=t.filter(r=>r.action==="skipped-conflict").length;n>0&&!e&&o.warn(`${n} mount dir(s) b\u1ECB skip do conflict. Ch\u1EA1y l\u1EA1i v\u1EDBi --force n\u1EBFu mu\u1ED1n override (backup t\u1EF1 \u0111\u1ED9ng).`)}function Zo(t){t.command("sync").description("Pull team-ai-pack m\u1EDBi nh\u1EA5t + t\u1EA1o symlink farm v\xE0o .claude/").option("--force","Override .claude/<dir>/ n\u1EBFu l\xE0 real dir (backup tr\u01B0\u1EDBc)").option("--version <tag>","Pin v\xE0o version c\u1EE5 th\u1EC3 (vd: v0.2.0)").option("--latest","Bleeding-edge: pull HEAD c\u1EE7a main branch (b\u1ECF qua tag SemVer)").option("--dry-run","Hi\u1EC3n th\u1ECB preview, kh\xF4ng apply thay \u0111\u1ED5i").action(rm)}function Qo(t){let e=t.command("tools").description("Qu\u1EA3n l\xFD system tools + MCP servers (M09)");e.command("list").description("Li\u1EC7t k\xEA tool \u0111\xE3 c\xE0i / c\xF2n thi\u1EBFu").option("--installed","Ch\u1EC9 li\u1EC7t k\xEA tool \u0111\xE3 c\xE0i").option("--missing","Ch\u1EC9 li\u1EC7t k\xEA tool c\xF2n thi\u1EBFu").option("--json","Output JSON cho script").action(I("tools list","Milestone 09")),e.command("install [tool-ids...]").description("C\xE0i tool v\xE0 \u0111\u0103ng k\xFD v\xE0o ~/.claude.json").option("--all-recommended","C\xE0i m\u1ECDi MCP \u0111\u01B0\u1EE3c recommend cho project type").option("--verify","Ch\u1EA1y MCP th\u1EED \u0111\u1EC3 verify (m\u1EA5t ~30s/tool)").option("--no-secrets","Skip prompt secrets, set sau qua 'avatar secrets'").action(I("tools install","Milestone 09")),e.command("remove <tool-id>").description("G\u1EE1 tool kh\u1ECFi ~/.claude.json (optional uninstall binary)").option("--keep-secrets","Kh\xF4ng x\xF3a secrets kh\u1ECFi keychain").option("--keep-binary","Kh\xF4ng uninstall npm global binary").action(I("tools remove","Milestone 09"))}import{relative as mm}from"path";import{confirm as dm}from"@inquirer/prompts";import gm from"boxen";import{cp as en,mkdir as ts,writeFile as om}from"fs/promises";import{homedir as sm}from"os";import{basename as am,join as Y}from"path";var cm=Y(sm(),".avatar","uninstall-backups");async function es(t,e,n){let r=am(t),i=new Date().toISOString().replace(/[:.]/g,"-"),s=Y(cm,`${r}-${i}`);if(await ts(s,{recursive:!0,mode:448}),e.claudeDir&&await en(e.claudeDir,Y(s,".claude"),{recursive:!0}),e.claudeMd&&await en(e.claudeMd,Y(s,"CLAUDE.md")),e.postMergeHook||e.prePushHook){let c=Y(s,"hooks");await ts(c,{recursive:!0}),e.postMergeHook&&await en(e.postMergeHook,Y(c,"post-merge")),e.prePushHook&&await en(e.prePushHook,Y(c,"pre-push"))}let a={projectName:r,projectPath:t,timestamp:i,avatarVersion:n,artifacts:{claudeDir:!!e.claudeDir,claudeMd:!!e.claudeMd,postMergeHook:!!e.postMergeHook,prePushHook:!!e.prePushHook}};return await om(Y(s,"manifest.json"),JSON.stringify(a,null,2),"utf8"),s}import{existsSync as lm}from"fs";import{join as J}from"path";function X(t){return lm(t)?t:null}function ns(t){let e=X(J(t,".claude")),n=X(J(t,"CLAUDE.md")),r=X(J(t,".git","hooks","post-merge")),i=X(J(t,".git","modules","src","hooks","pre-push")),s=X(J(t,".gitignore")),a=X(J(t,".gitmodules")),c=X(J(t,"notes")),l=X(J(t,"scripts"));return{hasAnyArtifact:!!(e||n||r||i),claudeDir:e,claudeMd:n,postMergeHook:r,prePushHook:i,gitignorePath:s,gitmodulesPath:a,notesDir:c,scriptsDir:l}}import{readFile as rs,rm as Z,writeFile as is}from"fs/promises";async function os(t,e){if(t.claudeDir)if(e.keepSubmodule){let{readdir:n}=await import("fs/promises"),{join:r}=await import("path"),i=await n(t.claudeDir);for(let s of i)s!=="pack"&&await Z(r(t.claudeDir,s),{recursive:!0,force:!0})}else await Z(t.claudeDir,{recursive:!0,force:!0});t.claudeMd&&await Z(t.claudeMd,{force:!0}),e.keepHooks||(t.postMergeHook&&await Z(t.postMergeHook,{force:!0}),t.prePushHook&&await Z(t.prePushHook,{force:!0})),t.gitignorePath&&await um(t.gitignorePath),t.gitmodulesPath&&!e.keepSubmodule&&await pm(t.gitmodulesPath,".claude/pack");for(let n of[t.notesDir,t.scriptsDir]){if(!n)continue;let{readdir:r}=await import("fs/promises");(await r(n)).length===0&&await Z(n,{recursive:!0,force:!0})}}async function um(t){let e=await rs(t,"utf8"),n=e.indexOf(ee),r=e.indexOf(pt);if(n===-1||r===-1)return;let i=e.slice(0,n),s=e.slice(r+pt.length),a=`${i.trimEnd()}
152
- ${s.trimStart()}`.trim();a.length===0?await Z(t,{force:!0}):await is(t,`${a}
153
- `,"utf8")}async function pm(t,e){let r=(await rs(t,"utf8")).split(`
151
+ Dry-run done. Kh\xF4ng apply thay \u0111\u1ED5i. B\u1ECF --dry-run \u0111\u1EC3 th\u1EF1c thi.`);return}if(i){o.info(`Pulling HEAD c\u1EE7a branch ${Ye} (bleeding-edge mode)...`),await Se(R,Ye,e);let l=await Pt(r);o.dim(` HEAD = ${l.slice(0,7)}`),o.warn("\u26A0 --latest mode: workspace pin v\xE0o commit floating, kh\xF4ng reproducible. Chuy\u1EC3n v\u1EC1 tag stable: avatar sync (no flag).")}else o.info(`Checking out ${a} trong submodule...`),await Vt(R,a,e);o.info("Creating symlink farm...");let c=await We(r,n,t.force===!0);Np(c,t.force===!0),o.info("Merging pack settings.json template into project settings.json...");try{let l=await ye(e);switch(l.action){case"merged":o.success(` \u2713 settings.json merged (${l.changes.join("; ")}). Backup: ${l.backupPath??"n/a"}`);break;case"no-change":o.info(" - settings.json \u0111\xE3 sync v\u1EDBi pack, kh\xF4ng c\xF3 thay \u0111\u1ED5i.");break;case"no-pack-template":o.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip merge.");break}}catch(l){o.warn(` ! Merge settings.json fail: ${l instanceof Error?l.message:l}. Symlinks \u0111\xE3 t\u1EA1o OK, hooks c\xF3 th\u1EC3 ch\u01B0a active. Manual merge n\u1EBFu c\u1EA7n.`)}try{await Ko(e)}catch(l){o.warn(` ! Re-apply features fail: ${l instanceof Error?l.message:l}. Ch\u1EA1y 'avatar feature enable <name>' th\u1EE7 c\xF4ng n\u1EBFu c\u1EA7n.`)}o.success(`Synced team-ai-pack to ${a}.`)}function Np(t,e){for(let r of t)switch(r.action){case"created":o.info(` \u2713 ${r.dir} \u2192 symlinked (new)`);break;case"updated":o.info(` \u2713 ${r.dir} \u2192 symlink refreshed`);break;case"backed-up-and-linked":o.info(` \u2713 ${r.dir} \u2192 symlinked (backup: ${r.backupPath})`);break;case"source-missing":o.warn(` - ${r.dir} \u2192 pack kh\xF4ng c\xF3 dir n\xE0y, skip`);break;case"skipped-conflict":o.warn(` ! ${r.dir} \u2192 CONFLICT: existing real dir. D\xF9ng --force \u0111\u1EC3 backup + override (s\u1EBD gi\u1EEF data \u1EDF ${r.dir}.backup-<timestamp>/).`);break}let n=t.filter(r=>r.action==="skipped-conflict").length;n>0&&!e&&o.warn(`${n} mount dir(s) b\u1ECB skip do conflict. Ch\u1EA1y l\u1EA1i v\u1EDBi --force n\u1EBFu mu\u1ED1n override (backup t\u1EF1 \u0111\u1ED9ng).`)}function Wo(t){t.command("sync").description("Pull team-ai-pack m\u1EDBi nh\u1EA5t + t\u1EA1o symlink farm v\xE0o .claude/").option("--force","Override .claude/<dir>/ n\u1EBFu l\xE0 real dir (backup tr\u01B0\u1EDBc)").option("--version <tag>","Pin v\xE0o version c\u1EE5 th\u1EC3 (vd: v0.2.0)").option("--latest","Bleeding-edge: pull HEAD c\u1EE7a main branch (b\u1ECF qua tag SemVer)").option("--dry-run","Hi\u1EC3n th\u1ECB preview, kh\xF4ng apply thay \u0111\u1ED5i").action(Op)}function zo(t){let e=t.command("tools").description("Qu\u1EA3n l\xFD system tools + MCP servers (M09)");e.command("list").description("Li\u1EC7t k\xEA tool \u0111\xE3 c\xE0i / c\xF2n thi\u1EBFu").option("--installed","Ch\u1EC9 li\u1EC7t k\xEA tool \u0111\xE3 c\xE0i").option("--missing","Ch\u1EC9 li\u1EC7t k\xEA tool c\xF2n thi\u1EBFu").option("--json","Output JSON cho script").action(_("tools list","Milestone 09")),e.command("install [tool-ids...]").description("C\xE0i tool v\xE0 \u0111\u0103ng k\xFD v\xE0o ~/.claude.json").option("--all-recommended","C\xE0i m\u1ECDi MCP \u0111\u01B0\u1EE3c recommend cho project type").option("--verify","Ch\u1EA1y MCP th\u1EED \u0111\u1EC3 verify (m\u1EA5t ~30s/tool)").option("--no-secrets","Skip prompt secrets, set sau qua 'avatar secrets'").action(_("tools install","Milestone 09")),e.command("remove <tool-id>").description("G\u1EE1 tool kh\u1ECFi ~/.claude.json (optional uninstall binary)").option("--keep-secrets","Kh\xF4ng x\xF3a secrets kh\u1ECFi keychain").option("--keep-binary","Kh\xF4ng uninstall npm global binary").action(_("tools remove","Milestone 09"))}import{relative as Fp}from"path";import{confirm as Vp}from"@inquirer/prompts";import Kp from"boxen";import{cp as Xe,mkdir as qo,writeFile as Mp}from"fs/promises";import{homedir as jp}from"os";import{basename as Lp,join as q}from"path";var Hp=q(jp(),".avatar","uninstall-backups");async function Jo(t,e,n){let r=Lp(t),i=new Date().toISOString().replace(/[:.]/g,"-"),s=q(Hp,`${r}-${i}`);if(await qo(s,{recursive:!0,mode:448}),e.claudeDir&&await Xe(e.claudeDir,q(s,".claude"),{recursive:!0}),e.claudeMd&&await Xe(e.claudeMd,q(s,"CLAUDE.md")),e.postMergeHook||e.prePushHook){let c=q(s,"hooks");await qo(c,{recursive:!0}),e.postMergeHook&&await Xe(e.postMergeHook,q(c,"post-merge")),e.prePushHook&&await Xe(e.prePushHook,q(c,"pre-push"))}let a={projectName:r,projectPath:t,timestamp:i,avatarVersion:n,artifacts:{claudeDir:!!e.claudeDir,claudeMd:!!e.claudeMd,postMergeHook:!!e.postMergeHook,prePushHook:!!e.prePushHook}};return await Mp(q(s,"manifest.json"),JSON.stringify(a,null,2),"utf8"),s}import{existsSync as Gp}from"fs";import{join as J}from"path";function Y(t){return Gp(t)?t:null}function Yo(t){let e=Y(J(t,".claude")),n=Y(J(t,"CLAUDE.md")),r=Y(J(t,".git","hooks","post-merge")),i=Y(J(t,".git","modules","src","hooks","pre-push")),s=Y(J(t,".gitignore")),a=Y(J(t,".gitmodules")),c=Y(J(t,"notes")),l=Y(J(t,"scripts"));return{hasAnyArtifact:!!(e||n||r||i),claudeDir:e,claudeMd:n,postMergeHook:r,prePushHook:i,gitignorePath:s,gitmodulesPath:a,notesDir:c,scriptsDir:l}}import{readFile as Xo,rm as X,writeFile as Zo}from"fs/promises";async function Qo(t,e){if(t.claudeDir)if(e.keepSubmodule){let{readdir:n}=await import("fs/promises"),{join:r}=await import("path"),i=await n(t.claudeDir);for(let s of i)s!=="pack"&&await X(r(t.claudeDir,s),{recursive:!0,force:!0})}else await X(t.claudeDir,{recursive:!0,force:!0});t.claudeMd&&await X(t.claudeMd,{force:!0}),e.keepHooks||(t.postMergeHook&&await X(t.postMergeHook,{force:!0}),t.prePushHook&&await X(t.prePushHook,{force:!0})),t.gitignorePath&&await Up(t.gitignorePath),t.gitmodulesPath&&!e.keepSubmodule&&await Dp(t.gitmodulesPath,".claude/pack");for(let n of[t.notesDir,t.scriptsDir]){if(!n)continue;let{readdir:r}=await import("fs/promises");(await r(n)).length===0&&await X(n,{recursive:!0,force:!0})}}async function Up(t){let e=await Xo(t,"utf8"),n=e.indexOf(te),r=e.indexOf(pt);if(n===-1||r===-1)return;let i=e.slice(0,n),s=e.slice(r+pt.length),a=`${i.trimEnd()}
152
+ ${s.trimStart()}`.trim();a.length===0?await X(t,{force:!0}):await Zo(t,`${a}
153
+ `,"utf8")}async function Dp(t,e){let r=(await Xo(t,"utf8")).split(`
154
154
  `),i=[],s=!1;for(let c of r){if(c.trim().startsWith("[submodule")&&c.includes(e)){s=!0;continue}s&&c.trim().startsWith("[submodule")&&(s=!1),s||i.push(c)}let a=i.join(`
155
- `).trim();a.length===0?await Z(t,{force:!0}):await is(t,`${a}
156
- `,"utf8")}function ss(t){t.command("uninstall").description("G\u1EE1 Avatar kh\u1ECFi project \u2014 backup t\u1EF1 \u0111\u1ED9ng (M11)").option("--yes","Skip confirm prompt").option("--no-backup","Kh\xF4ng t\u1EA1o backup tr\u01B0\u1EDBc khi x\xF3a (nguy hi\u1EC3m)").option("--keep-submodule","Gi\u1EEF submodule .claude/pack/").option("--keep-hooks","Gi\u1EEF git hooks post-merge, pre-push").option("--dry-run","Hi\u1EC3n th\u1ECB danh s\xE1ch s\u1EBD x\xF3a, kh\xF4ng th\u1EF1c thi").action(async e=>{try{await fm(e)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function fm(t){let e=process.cwd(),n=ns(e);if(!n.hasAnyArtifact){o.info("Project ch\u01B0a c\xE0i Avatar \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}if(hm(e,n,t),t.dryRun){o.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await dm({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){o.info("\u0110\xE3 h\u1EE7y.");return}let r=null;t.noBackup||(r=await es(e,n,H()),o.success(`Backup t\u1EA1o t\u1EA1i: ${r}`)),await os(n,{keepSubmodule:t.keepSubmodule,keepHooks:t.keepHooks}),await w("uninstall",`project=${e},backup=${r??"skipped"}`),km(r)}function hm(t,e,n){o.info(`Project: ${t}`),o.plain(""),o.plain("C\xE1c artifact s\u1EBD g\u1EE1:"),e.claudeDir&&o.plain(` ${p.red("\u2717")} ${mm(t,e.claudeDir)||".claude/"}`),e.claudeMd&&o.plain(` ${p.red("\u2717")} CLAUDE.md`),e.postMergeHook&&!n.keepHooks&&o.plain(` ${p.red("\u2717")} .git/hooks/post-merge`),e.prePushHook&&!n.keepHooks&&o.plain(` ${p.red("\u2717")} .git/modules/src/hooks/pre-push`),e.gitignorePath&&o.plain(` ${p.yellow("\u270E")} .gitignore (g\u1EE1 Avatar block)`),e.gitmodulesPath&&!n.keepSubmodule&&o.plain(` ${p.yellow("\u270E")} .gitmodules (g\u1EE1 entry .claude/pack)`),o.plain(""),o.plain("Kh\xF4ng \u0111\u1EE5ng:"),o.plain(` ${p.green("\u2713")} src/ (code kh\xE1ch)`),o.plain(` ${p.green("\u2713")} Git history`),o.plain(` ${p.green("\u2713")} ~/.avatar/config.json (token SSO)`),o.plain(` ${p.green("\u2713")} Secrets trong keychain`),o.plain("")}function km(t){let e=[`${p.green("\u2713")} Avatar \u0111\xE3 \u0111\u01B0\u1EE3c g\u1EE1 kh\u1ECFi project`];t&&(e.push(""),e.push(` ${p.dim("Backup:")} ${t}`),e.push(` ${p.dim("Restore:")} ${p.cyan(`cp -r "${t}"/* .`)}`)),process.stdout.write(`${gm(e.join(`
155
+ `).trim();a.length===0?await X(t,{force:!0}):await Zo(t,`${a}
156
+ `,"utf8")}function ts(t){t.command("uninstall").description("G\u1EE1 Avatar kh\u1ECFi project \u2014 backup t\u1EF1 \u0111\u1ED9ng (M11)").option("--yes","Skip confirm prompt").option("--no-backup","Kh\xF4ng t\u1EA1o backup tr\u01B0\u1EDBc khi x\xF3a (nguy hi\u1EC3m)").option("--keep-submodule","Gi\u1EEF submodule .claude/pack/").option("--keep-hooks","Gi\u1EEF git hooks post-merge, pre-push").option("--dry-run","Hi\u1EC3n th\u1ECB danh s\xE1ch s\u1EBD x\xF3a, kh\xF4ng th\u1EF1c thi").action(async e=>{try{await Bp(e)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Bp(t){let e=process.cwd(),n=Yo(e);if(!n.hasAnyArtifact){o.info("Project ch\u01B0a c\xE0i Avatar \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}if(Wp(e,n,t),t.dryRun){o.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await Vp({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){o.info("\u0110\xE3 h\u1EE7y.");return}let r=null;t.noBackup||(r=await Jo(e,n,G()),o.success(`Backup t\u1EA1o t\u1EA1i: ${r}`)),await Qo(n,{keepSubmodule:t.keepSubmodule,keepHooks:t.keepHooks}),await v("uninstall",`project=${e},backup=${r??"skipped"}`),zp(r)}function Wp(t,e,n){o.info(`Project: ${t}`),o.plain(""),o.plain("C\xE1c artifact s\u1EBD g\u1EE1:"),e.claudeDir&&o.plain(` ${p.red("\u2717")} ${Fp(t,e.claudeDir)||".claude/"}`),e.claudeMd&&o.plain(` ${p.red("\u2717")} CLAUDE.md`),e.postMergeHook&&!n.keepHooks&&o.plain(` ${p.red("\u2717")} .git/hooks/post-merge`),e.prePushHook&&!n.keepHooks&&o.plain(` ${p.red("\u2717")} .git/modules/src/hooks/pre-push`),e.gitignorePath&&o.plain(` ${p.yellow("\u270E")} .gitignore (g\u1EE1 Avatar block)`),e.gitmodulesPath&&!n.keepSubmodule&&o.plain(` ${p.yellow("\u270E")} .gitmodules (g\u1EE1 entry .claude/pack)`),o.plain(""),o.plain("Kh\xF4ng \u0111\u1EE5ng:"),o.plain(` ${p.green("\u2713")} src/ (code kh\xE1ch)`),o.plain(` ${p.green("\u2713")} Git history`),o.plain(` ${p.green("\u2713")} ~/.avatar/config.json (token SSO)`),o.plain(` ${p.green("\u2713")} Secrets trong keychain`),o.plain("")}function zp(t){let e=[`${p.green("\u2713")} Avatar \u0111\xE3 \u0111\u01B0\u1EE3c g\u1EE1 kh\u1ECFi project`];t&&(e.push(""),e.push(` ${p.dim("Backup:")} ${t}`),e.push(` ${p.dim("Restore:")} ${p.cyan(`cp -r "${t}"/* .`)}`)),process.stdout.write(`${Kp(e.join(`
157
157
  `),{padding:1,borderStyle:"round"})}
158
- `)}var Qn=H(),C=new wm;C.name("avatar").description("AI harness CLI for NAL Vietnam engineering").version(Qn,"-v, --version","Hi\u1EC3n th\u1ECB phi\xEAn b\u1EA3n Avatar CLI").addHelpText("beforeAll",()=>`
159
- ${Tn({tagline:`v${Qn} \xB7 AI harness CLI for NAL Vietnam`})}
158
+ `)}var Yn=G(),C=new qp;C.name("avatar").description("AI harness CLI for NAL Vietnam engineering").version(Yn,"-v, --version","Hi\u1EC3n th\u1ECB phi\xEAn b\u1EA3n Avatar CLI").addHelpText("beforeAll",()=>`
159
+ ${$n({tagline:`v${Yn} \xB7 AI harness CLI for NAL Vietnam`})}
160
160
 
161
- `);var ym=process.argv.includes("-v")||process.argv.includes("--version");ym&&($t({tagline:`v${Qn} \xB7 AI harness CLI for NAL Vietnam`}),process.exit(0));Mo(C);jo(C);Zo(C);Fo(C);Do(C);zo(C);ci(C);Ho(C);Lr(C);Qo(C);Bo(C);Go(C);Mr(C);ji(C);hi(C);Uo(C);ss(C);C.parseAsync(process.argv).catch(t=>{let e=t instanceof Error?t.message:String(t);process.stderr.write(`\u2717 L\u1ED7i kh\xF4ng x\u1EED l\xFD \u0111\u01B0\u1EE3c: ${e}
161
+ `);var Jp=process.argv.includes("-v")||process.argv.includes("--version");Jp&&($t({tagline:`v${Yn} \xB7 AI harness CLI for NAL Vietnam`}),process.exit(0));Ro(C);To(C);Wo(C);jo(C);Mo(C);Do(C);ni(C);No(C);Ir(C);zo(C);Go(C);_o(C);Rr(C);Ti(C);ui(C);Oo(C);ts(C);C.parseAsync(process.argv).catch(t=>{let e=t instanceof Error?t.message:String(t);process.stderr.write(`\u2717 L\u1ED7i kh\xF4ng x\u1EED l\xFD \u0111\u01B0\u1EE3c: ${e}
162
162
  `),process.exit(1)});
163
163
  //# sourceMappingURL=index.js.map