@nalvietnam/avatar-cli 1.18.0 → 1.19.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -1,76 +1,86 @@
1
1
  // @nalvietnam/avatar-cli — built with tsup
2
- import{Command as _l}from"commander";import{promises as bs}from"fs";import{join as Xn}from"path";import{confirm as vs}from"@inquirer/prompts";import{constants as xo,promises as dt}from"fs";import{dirname as So,join as jl,relative as Ll}from"path";async function d(t){try{return await dt.access(t,xo.F_OK),!0}catch{return!1}}async function P(t){await dt.mkdir(t,{recursive:!0})}async function M(t){return await dt.readFile(t,"utf8")}async function w(t){return JSON.parse(await M(t))}async function gt(t,e,n){await P(So(t));let i=`${t}.tmp-${process.pid}-${Date.now()}`;await dt.writeFile(i,e,"utf8"),n!==void 0&&await dt.chmod(i,n),await dt.rename(i,t)}async function R(t,e,n){await gt(t,`${JSON.stringify(e,null,2)}
3
- `,n)}import{existsSync as zt,readFileSync as Co}from"fs";import{dirname as Ao,join as Jt}from"path";var Po=5;function $o(t){let e=zt(Jt(t,".claude")),n=zt(Jt(t,"CLAUDE.md"));if(!e||!n)return!1;let i=zt(Jt(t,"src")),r=Jt(t,".gitmodules");if(i)return!0;if(zt(r))try{let s=Co(r,"utf8");return s.includes("path = src")||s.includes("path = ./src")}catch{return!1}return!1}function Yt(t){let e=t;for(let n=0;n<Po;n++){if($o(e))return e;let i=Ao(e);if(i===e)return null;e=i}return null}import{promises as On}from"fs";import{homedir as To}from"os";import{join as Pt}from"path";import{z as f}from"zod";var En=f.object({email:f.string().email(),name:f.string(),access_token:f.string().min(1),refresh_token:f.string().min(1),expires_at:f.string().datetime(),id_token:f.string().min(1)}),Ro=f.object({installed_tools:f.record(f.string(),f.object({version:f.string().optional(),installed_at:f.string().datetime(),install_method:f.string()})).default({}),tool_inputs:f.record(f.string(),f.unknown()).default({})}),Fl=f.object({$schema:f.string().optional(),includeCoAuthoredBy:f.boolean().optional(),env:f.record(f.string(),f.string()).default({}),permissions:f.object({allow:f.array(f.string()).default([]),deny:f.array(f.string()).default([])}).partial().optional(),hooks:f.record(f.string(),f.array(f.unknown())).optional(),statusLine:f.object({type:f.string(),command:f.string(),padding:f.number().optional()}).optional()}),Vl=f.enum(["internal","client","library"]);var $t=Pt(To(),".avatar"),Y=Pt($t,"config.json"),Jl=Pt($t,"state.json"),Ne=Pt($t,"audit.log"),Yl=Pt($t,"backups"),Eo=384;async function je(){await P($t)}async function X(){if(!await d(Y))return null;let t=await w(Y),e=En.safeParse(t);return e.success?e.data:null}async function _n(t){await je(),await R(Y,t,Eo)}async function In(){if(await d(Y)){let{promises:t}=await import("fs");await t.unlink(Y)}}function Q(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}async function _o(){try{await On.chmod(Ne,384)}catch{}}async function k(t,e){await je();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},i=`${JSON.stringify(n)}
4
- `;await On.appendFile(Ne,i,{encoding:"utf8",mode:384}),await _o()}import{spawnSync as Le}from"child_process";import p from"chalk";import Io from"ora";var o={info:t=>process.stdout.write(`${p.blue("\u2139")} ${t}
2
+ var es=Object.defineProperty;var ze=(t,e)=>()=>(t&&(e=t(t=0)),e);var ns=(t,e)=>{for(var n in e)es(t,n,{get:e[n],enumerable:!0})};import{constants as rs,promises as ht}from"fs";import{dirname as is,join as qp,relative as Jp}from"path";async function d(t){try{return await ht.access(t,rs.F_OK),!0}catch{return!1}}async function $(t){await ht.mkdir(t,{recursive:!0})}async function H(t){return await ht.readFile(t,"utf8")}async function w(t){return JSON.parse(await H(t))}async function kt(t,e,n){await $(is(t));let r=`${t}.tmp-${process.pid}-${Date.now()}`;await ht.writeFile(r,e,"utf8"),n!==void 0&&await ht.chmod(r,n),await ht.rename(r,t)}async function A(t,e,n){await kt(t,`${JSON.stringify(e,null,2)}
3
+ `,n)}var k=ze(()=>{"use strict"});import{spawnSync as Cu}from"child_process";import{chmodSync as Pu,existsSync as ao,readFileSync as co,renameSync as $u,writeFileSync as Eu}from"fs";import{join as Ru}from"path";function Et(t){return Ru(t,Tu)}function uo(t){let e=t.indexOf(lo);if(e===-1)return null;let n=t.indexOf(Ln,e);return n===-1?null:t.slice(e,n+Ln.length)}function po(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 Iu(t){return t.replace(/\\/g,"\\\\").replace(/"/g,'\\"')}function Ou(t){let e=[_u],n=[...t.keys()].sort();for(let r of n){let i=t.get(r);i!==void 0&&e.push(`export ${r}="${Iu(i)}"`)}return e.push(Ln),e.join(`
4
+ `)}async function te(t,e){let n=Et(t),r="";ao(n)&&(r=co(n,"utf8"));let i=uo(r),s=i?po(i):new Map,a=s.size;for(let[h,x]of Object.entries(e))x===null?s.delete(h):s.set(h,x);let c=Ou(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
+
6
+ ${c}
7
+ `):l=`${c}
8
+ `;let m=`${n}.avatar-tmp-${process.pid}`;Eu(m,l,{encoding:"utf8",mode:384}),$u(m,n),Pu(n,384);let g=null;try{g=Cu("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 Rt(t){let e=Et(t);if(!ao(e))return new Map;let n=co(e,"utf8"),r=uo(n);return r?po(r):new Map}var Tu,lo,Ln,_u,Gn=ze(()=>{"use strict";Tu=".envrc",lo="# >>> avatar-cli secrets >>>",Ln="# <<< avatar-cli secrets <<<",_u=`${lo}
9
+ # Per-project secrets managed by avatar-cli.
10
+ # DO NOT COMMIT \u2014 must be gitignored (avatar enforces this on setup).
11
+ # Edit via: avatar secrets set <NAME>
12
+ # Remove via: avatar secrets rm <NAME>
13
+ # View: avatar secrets list`});var go={};ns(go,{MIGRATABLE_SECRET_KEYS:()=>mo,detectPlaintextSecretsInSettings:()=>Tt,migrateSecretsFromSettingsToEnvrc:()=>Fe});import{promises as Nu}from"fs";import{join as ju}from"path";function Lu(t){return ju(t,...Mu)}function Gu(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 Tt(t){let e=Lu(t);if(!await d(e))return{keys:[],settingsPath:e};let n;try{n=await w(e)}catch{return{keys:[],settingsPath:e}}let r=n.env??{},i=[];for(let s of mo)typeof r[s]=="string"&&r[s].length>0&&i.push(s);return{keys:i,settingsPath:e}}async function Fe(t){let{keys:e,settingsPath:n}=await Tt(t);if(e.length===0)return{migratedKeys:[],settingsJsonBackupPath:null,noOp:!0};let r=await w(n),i=r.env??{},s={};for(let c of e){let l=i[c];typeof l=="string"&&l.length>0&&(s[c]=l)}await te(t,s);let a=Gu(n);await Nu.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 A(n,r),{migratedKeys:e,settingsJsonBackupPath:a,noOp:!1}}var mo,Mu,Hn=ze(()=>{"use strict";k();Gn();mo=["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"],Mu=[".claude","settings.json"]});import{Command as Vp}from"commander";k();import{promises as Zs}from"fs";import{join as wr}from"path";import{confirm as ta}from"@inquirer/prompts";import{existsSync as re,readFileSync as os}from"fs";import{dirname as ss,join as ie}from"path";var as=5;function cs(t){let e=re(ie(t,".claude")),n=re(ie(t,"CLAUDE.md"));if(!e||!n)return!1;let r=re(ie(t,"src")),i=ie(t,".gitmodules");if(r)return!0;if(re(i))try{let s=os(i,"utf8");return s.includes("path = src")||s.includes("path = ./src")}catch{return!1}return!1}function wt(t){let e=t;for(let n=0;n<as;n++){if(cs(e))return e;let r=ss(e);if(r===e)return null;e=r}return null}import{promises as tr}from"fs";import{homedir as us}from"os";import{join as Nt}from"path";import{z as f}from"zod";var Xn=f.object({email:f.string().email(),name:f.string(),access_token:f.string().min(1),refresh_token:f.string().min(1),expires_at:f.string().datetime(),id_token:f.string().min(1)}),ls=f.object({installed_tools:f.record(f.string(),f.object({version:f.string().optional(),installed_at:f.string().datetime(),install_method:f.string()})).default({}),tool_inputs:f.record(f.string(),f.unknown()).default({})}),em=f.object({$schema:f.string().optional(),includeCoAuthoredBy:f.boolean().optional(),env:f.record(f.string(),f.string()).default({}),permissions:f.object({allow:f.array(f.string()).default([]),deny:f.array(f.string()).default([])}).partial().optional(),hooks:f.record(f.string(),f.array(f.unknown())).optional(),statusLine:f.object({type:f.string(),command:f.string(),padding:f.number().optional()}).optional()}),nm=f.enum(["internal","client","library"]);k();var jt=Nt(us(),".avatar"),X=Nt(jt,"config.json"),cm=Nt(jt,"state.json"),qe=Nt(jt,"audit.log"),lm=Nt(jt,"backups"),ps=384;async function Je(){await $(jt)}async function Q(){if(!await d(X))return null;let t=await w(X),e=Xn.safeParse(t);return e.success?e.data:null}async function Qn(t){await Je(),await A(X,t,ps)}async function Zn(){if(await d(X)){let{promises:t}=await import("fs");await t.unlink(X)}}function Z(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}async function ms(){try{await tr.chmod(qe,384)}catch{}}async function y(t,e){await Je();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
14
+ `;await tr.appendFile(qe,r,{encoding:"utf8",mode:384}),await ms()}import{spawnSync as Ye}from"child_process";import p from"chalk";import ds from"ora";var o={info:t=>process.stdout.write(`${p.blue("\u2139")} ${t}
5
15
  `),success:t=>process.stdout.write(`${p.green("\u2713")} ${t}
6
16
  `),warn:t=>process.stdout.write(`${p.yellow("\u26A0")} ${t}
7
17
  `),error:t=>process.stderr.write(`${p.red("\u2717")} ${t}
8
18
  `),dim:t=>process.stdout.write(`${p.dim(t)}
9
19
  `),plain:t=>process.stdout.write(`${t}
10
- `)};function H(t){return Io({text:t,spinner:"dots",isEnabled:process.stdout.isTTY??!1}).start()}function Rt(t){let e=Date.now(),n=H(`${t} (0:00)`),i=()=>{let s=Math.floor((Date.now()-e)/1e3),a=Math.floor(s/60),c=s%60;return`${a}:${String(c).padStart(2,"0")}`},r=setInterval(()=>{n.text=`${t} (${i()})`},1e3);return{succeed:s=>{clearInterval(r),n.succeed(`${s} (${i()})`)},fail:s=>{clearInterval(r),n.fail(`${s} (${i()})`)},stop:()=>{clearInterval(r),n.stop()}}}var Nn=6e4,Oo="ok";function Ge(){let t=Le("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 Me(){o.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=Le("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 No(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 jn(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 He(){let t=Le("claude",["--print",Oo],{encoding:"utf8",timeout:Nn,stdio:["ignore","pipe","pipe"]});if(t.signal==="SIGTERM"||t.status===143||t.error?.code==="ETIMEDOUT")return{ok:!1,reason:"timeout",detail:`claude --print > ${Nn/1e3}s (m\u1EA1ng ch\u1EADm / API rate limit / token revoked kh\xF4ng return error)`};let n=t.stderr||"",i=t.stdout||"";if(t.status===0)return{ok:!0};let r=i.trim(),s=n.toLowerCase();if(r.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 (${r.length} chars). Accept v\u1EDBi caution.`),{ok:!0};let a=No(`${n}
11
- ${i}`);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)}`),i.trim()&&o.warn(`[debug] stdout: ${i.slice(0,300)}`)),{ok:!1,reason:a,detail:n.slice(0,500)||i.slice(0,500)}}import{spawnSync as Ln}from"child_process";import{platform as jo}from"os";function Z(){let t=jo();return t==="darwin"||t==="linux"||t==="win32"?t:"unsupported"}var Lo=5e3,Go=/(\d+\.\d+\.\d+)/;function Mo(){let e=Z()==="win32"?"where":"which",n=Ln(e,["claude"],{encoding:"utf8"});if(n.error||n.status!==0)return null;let i=(n.stdout||"").trim();return i?i.split(/\r?\n/)[0].trim():null}function Ho(){let t=Ln("claude",["--version"],{encoding:"utf8",timeout:Lo});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return Go.exec(e)?.[1]??null}var tt=null;function Tt(){if(tt!==null)return tt;let t=Mo();return t?(tt={installed:!0,version:Ho(),path:t},tt):(tt={installed:!1,version:null,path:null},tt)}function Xt(){tt=null}import{spawnSync as Uo}from"child_process";var Gn=300*1e3,Mn="@anthropic-ai/claude-code",et=class extends Error{reason;exitCode;constructor(e,n,i=null){super(n),this.name="InstallClaudeCodeError",this.reason=e,this.exitCode=i}};function Do(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new et("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${Mn} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new et("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new et("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function Hn(){o.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Uo("npm",["install","-g",Mn],{stdio:["inherit","inherit","pipe"],timeout:Gn,encoding:"utf8"});if(t.signal==="SIGTERM")throw new et("timeout",`npm install timeout sau ${Gn/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),Do(t.status,t.stderr||"");Xt();let e=Tt();if(!e.installed||!e.path)throw new et("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 Fo}from"fs";import{homedir as Vo}from"os";import{join as Bo}from"path";import{select as Un}from"@inquirer/prompts";function Ko(){return Bo(Vo(),".claude","settings.json")}function Ue(){let t=Ko(),e;try{e=Fo(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 i=n.env||{},r=typeof i.ANTHROPIC_BASE_URL=="string"?i.ANTHROPIC_BASE_URL:void 0,s=typeof i.ANTHROPIC_AUTH_TOKEN=="string"&&i.ANTHROPIC_AUTH_TOKEN.length>0,a=typeof n.model=="string"?n.model:void 0;return{exists:!0,hasBaseUrl:!!r,baseUrl:r,hasToken:s,model:a,rawSettings:n}}async function Dn(t=Ue()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await Un({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 Un({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. LLMLite API key (llm.nal.vn \u2014 NAL gateway, key sk-...)",value:"llmlite"},{name:"3. Anthropic API key tr\u1EF1c ti\u1EBFp (console.anthropic.com, key sk-ant-...)",value:"anthropic"}]})}import{password as Wo,select as qo}from"@inquirer/prompts";var Qt="https://api.anthropic.com",zo="2023-06-01",Fn=1e4;function Jo(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function Yo(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 Xo(){return await Wo({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:Yo})}async function Qo(t){let e=new AbortController,n=setTimeout(()=>e.abort(),Fn);try{let i=await fetch(`${Qt}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":zo,Accept:"application/json"},signal:e.signal});if(i.status===401)throw new Error("API key invalid (HTTP 401). Check key tr\xEAn console.anthropic.com.");if(i.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(i.status===429)throw new Error("Rate limit (HTTP 429). Ch\u1EDD v\xE0i gi\xE2y r\u1ED3i th\u1EED l\u1EA1i.");if(!i.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${i.status}).`);let s=((await i.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(i){throw i.name==="AbortError"?new Error(`Connect ${Qt} timeout sau ${Fn/1e3}s.`):i}finally{clearTimeout(n)}}async function Zo(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,i)=>{let r=s=>{let a=s.toLowerCase();return a.includes("sonnet")?0:a.includes("opus")?1:a.includes("haiku")?2:3};return r(n)-r(i)});return await qo({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function Vn(){let t=await Xo();o.info(`Verify key (${Jo(t)}) qua ${Qt}/v1/models...`);let e=await Qo(t);o.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await Zo(e);return{apiKey:t,baseUrl:Qt,model:n}}import{input as ts,password as es,select as ns}from"@inquirer/prompts";var is="https://llm.nal.vn",Bn=1e4;function U(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function rs(){return await es({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function os(t=is){return(await ts({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 ss(t,e){let n=new AbortController,i=setTimeout(()=>n.abort(),Bn);try{let r=await fetch(`${t}/v1/models`,{method:"GET",headers:{Authorization:`Bearer ${e}`,Accept:"application/json"},signal:n.signal});if(r.status===401||r.status===403)throw new Error(`API key invalid (HTTP ${r.status}).`);if(r.status===404)throw new Error(`Endpoint /v1/models kh\xF4ng t\u1ED3n t\u1EA1i tr\xEAn ${t}.`);if(!r.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${r.status}).`);let a=((await r.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(r){if(r.name==="AbortError"){let c=t.includes("nal.vn")||t.includes("nal-vn")?"\n Hint: llm.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://llm.nal.vn`.":`
12
- Hint: check m\u1EA1ng / firewall / VPN, ho\u1EB7c base URL c\xF3 \u0111\xFAng kh\xF4ng.`;throw new Error(`Connect ${t} timeout sau ${Bn/1e3}s.${c}`)}let s=r.message||String(r);if(s.toLowerCase().includes("fetch failed")||s.includes("ENOTFOUND")){let c=t.includes("nal.vn")?" (llm.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 r}finally{clearTimeout(i)}}async function as(t){let e=t.filter(i=>i.toLowerCase().includes("claude"));if(e.length===1){let i=e[0];return o.info(`Auto-pick model: ${i} (ch\u1EC9 1 claude alias tr\xEAn endpoint)`),i}let n=e.length>0?e:t;return await ns({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(i=>({name:i,value:i}))})}async function Kn(){let t=await rs(),e=await os();o.info(`Verify key (${U(t)}) qua ${e}/v1/models...`);let n=await ss(e,t);o.success(`Endpoint OK \u2014 ${n.length} models available`);let i=await as(n);return{apiKey:t,baseUrl:e,model:i}}import{promises as cs}from"fs";import{join as us}from"path";var De=384;function ls(t){return us(t,".claude","settings.json")}async function ps(t){if(!await d(t))return{};try{return await w(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 ms(t,e){let{env:n,...i}=t,r={...i,model:e};if(n){let{ANTHROPIC_BASE_URL:s,ANTHROPIC_AUTH_TOKEN:a,ANTHROPIC_API_KEY:c,...u}=n;Object.keys(u).length>0&&(r.env=u)}return r}function ds(t,e,n,i){let{ANTHROPIC_API_KEY:r,...s}=t.env||{};return{...t,env:{...s,ANTHROPIC_BASE_URL:n,ANTHROPIC_AUTH_TOKEN:e},model:i}}function gs(t,e,n,i){let{ANTHROPIC_AUTH_TOKEN:r,...s}=t.env||{};return{...t,env:{...s,ANTHROPIC_BASE_URL:n,ANTHROPIC_API_KEY:e},model:i}}function fs(t,e){let n=e.env||{},i=typeof e.model=="string"?e.model:void 0;return{...t,env:{...t.env||{},...n},...i?{model:i}:{}}}async function nt(t,e){let n=ls(t),i=await ps(n),r;switch(e.provider){case"subscription":r=ms(i,e.model);break;case"llmlite":r=ds(i,e.apiKey,e.baseUrl,e.model);break;case"anthropic":r=gs(i,e.apiKey,e.baseUrl,e.model);break;case"use-global":r=fs(i,e.sourceSettings);break}await R(n,r,De);try{await cs.chmod(n,De)}catch{}return{path:n,mode:De}}var N="sonnet";function Wn(t){o.warn(`\u{1F512} ${t} key \u0111\xE3 l\u01B0u PLAINTEXT v\xE0o .claude/settings.json.
20
+ `)};function U(t){return ds({text:t,spinner:"dots",isEnabled:process.stdout.isTTY??!1}).start()}function Mt(t){let e=Date.now(),n=U(`${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 er=6e4,gs="ok";function Xe(){let t=Ye("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 Qe(){o.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=Ye("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 fs(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 nr(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 Ze(){let t=Ye("claude",["--print",gs],{encoding:"utf8",timeout:er,stdio:["ignore","pipe","pipe"]});if(t.signal==="SIGTERM"||t.status===143||t.error?.code==="ETIMEDOUT")return{ok:!1,reason:"timeout",detail:`claude --print > ${er/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=fs(`${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 rr}from"child_process";import{platform as hs}from"os";function tt(){let t=hs();return t==="darwin"||t==="linux"||t==="win32"?t:"unsupported"}var ks=5e3,ws=/(\d+\.\d+\.\d+)/;function ys(){let e=tt()==="win32"?"where":"which",n=rr(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 vs(){let t=rr("claude",["--version"],{encoding:"utf8",timeout:ks});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return ws.exec(e)?.[1]??null}var et=null;function Lt(){if(et!==null)return et;let t=ys();return t?(et={installed:!0,version:vs(),path:t},et):(et={installed:!1,version:null,path:null},et)}function oe(){et=null}import{spawnSync as bs}from"child_process";var ir=300*1e3,or="@anthropic-ai/claude-code",nt=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallClaudeCodeError",this.reason=e,this.exitCode=r}};function xs(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new nt("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${or} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new nt("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new nt("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function sr(){o.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=bs("npm",["install","-g",or],{stdio:["inherit","inherit","pipe"],timeout:ir,encoding:"utf8"});if(t.signal==="SIGTERM")throw new nt("timeout",`npm install timeout sau ${ir/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),xs(t.status,t.stderr||"");oe();let e=Lt();if(!e.installed||!e.path)throw new nt("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 Ss}from"fs";import{homedir as As}from"os";import{join as Cs}from"path";import{select as ar}from"@inquirer/prompts";function Ps(){return Cs(As(),".claude","settings.json")}function tn(){let t=Ps(),e;try{e=Ss(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 cr(t=tn()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await ar({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 ar({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. LLMLite API key (llm.nal.vn \u2014 NAL gateway, key sk-...)",value:"llmlite"},{name:"3. Anthropic API key tr\u1EF1c ti\u1EBFp (console.anthropic.com, key sk-ant-...)",value:"anthropic"}]})}import{password as $s,select as Es}from"@inquirer/prompts";var se="https://api.anthropic.com",Rs="2023-06-01",lr=1e4;function ae(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function Ts(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 _s(){return await $s({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:Ts})}async function ce(t){let e=new AbortController,n=setTimeout(()=>e.abort(),lr);try{let r=await fetch(`${se}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":Rs,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 ${se} timeout sau ${lr/1e3}s.`):r}finally{clearTimeout(n)}}async function Is(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 Es({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function ur(){let t=await _s();o.info(`Verify key (${ae(t)}) qua ${se}/v1/models...`);let e=await ce(t);o.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await Is(e);return{apiKey:t,baseUrl:se,model:n}}import{input as Os,password as Ns,select as js}from"@inquirer/prompts";var Ms="https://llm.nal.vn",pr=1e4;function D(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function Ls(){return await Ns({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function Gs(t=Ms){return(await Os({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 Hs(t,e){let n=new AbortController,r=setTimeout(()=>n.abort(),pr);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: llm.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://llm.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 ${pr/1e3}s.${c}`)}let s=i.message||String(i);if(s.toLowerCase().includes("fetch failed")||s.includes("ENOTFOUND")){let c=t.includes("nal.vn")?" (llm.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 Us(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 js({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function mr(){let t=await Ls(),e=await Gs();o.info(`Verify key (${D(t)}) qua ${e}/v1/models...`);let n=await Hs(e,t);o.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await Us(n);return{apiKey:t,baseUrl:e,model:r}}k();import{promises as Ds}from"fs";import{join as Fs}from"path";var en=384;function Vs(t){return Fs(t,".claude","settings.json")}async function Ks(t){if(!await d(t))return{};try{return await w(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 Bs(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 Ws(t,e,n,r){let{ANTHROPIC_API_KEY:i,...s}=t.env||{};return{...t,env:{...s,ANTHROPIC_BASE_URL:n,ANTHROPIC_AUTH_TOKEN:e},model:r}}function zs(t,e,n,r){let{ANTHROPIC_AUTH_TOKEN:i,...s}=t.env||{};return{...t,env:{...s,ANTHROPIC_BASE_URL:n,ANTHROPIC_API_KEY:e},model:r}}function qs(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 rt(t,e){let n=Vs(t),r=await Ks(n),i;switch(e.provider){case"subscription":i=Bs(r,e.model);break;case"llmlite":i=Ws(r,e.apiKey,e.baseUrl,e.model);break;case"anthropic":i=zs(r,e.apiKey,e.baseUrl,e.model);break;case"use-global":i=qs(r,e.sourceSettings);break}await A(n,i,en);try{await Ds.chmod(n,en)}catch{}return{path:n,mode:en}}var j="sonnet";function dr(t){o.warn(`\u{1F512} ${t} key \u0111\xE3 l\u01B0u PLAINTEXT v\xE0o .claude/settings.json.
13
23
  File n\xE0y \u0111\u01B0\u1EE3c gitignore t\u1EEB Avatar v1.7.0, nh\u01B0ng workspace c\u0169 c\xF3 th\u1EC3 CH\u01AFA.
14
24
  Ki\u1EC3m tra: grep '.claude/settings.json' .gitignore
15
- N\u1EBFu ch\u01B0a c\xF3 \u2192 TH\xCAM NGAY, tr\xE1nh leak key khi commit/push.`)}async function Zt(t){try{o.info("Setup AI provider cho workspace...");let e=Tt();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."),Hn(),Xt(),e=Tt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=Ue();switch(await Dn(n)){case"subscription":{let r=Ge();if(r.state!=="authenticated"&&(Me(),r=Ge()),r.state==="authenticated"&&r.subscriptionType)return await nt(t.workspacePath,{provider:"subscription",model:N}),await k("ai_setup",`provider=subscription,result=ok,plan=${r.subscriptionType},probe=skipped`),o.success(`AI ready \xB7 Subscription (${r.subscriptionType}) \xB7 model=${N}`),{ok:!0,provider:"subscription",model:N};o.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let s=He();if(!s.ok&&s.reason==="auth-expired"&&(o.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),Me(),s=He()),!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 nt(t.workspacePath,{provider:"subscription",model:N}),await k("ai_setup",`provider=subscription,result=ok,probe=${s.reason}-soft-pass`),o.success(`AI ready \xB7 Subscription (probe ${s.reason}, soft-pass) \xB7 model=${N}`),{ok:!0,provider:"subscription",model:N};if(!s.ok){let a=s.reason??"unknown";return await k("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 ${jn(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await nt(t.workspacePath,{provider:"subscription",model:N}),await k("ai_setup","provider=subscription,result=ok"),o.success(`AI ready \xB7 Subscription \xB7 model=${N}`),{ok:!0,provider:"subscription",model:N}}case"llmlite":{let r=await Kn();return await nt(t.workspacePath,{provider:"llmlite",apiKey:r.apiKey,baseUrl:r.baseUrl,model:r.model}),await k("ai_setup",`provider=llmlite,result=ok,model=${r.model},base=${r.baseUrl}`),o.success(`AI ready \xB7 LLMLite \xB7 model=${r.model} \xB7 ${r.baseUrl}`),Wn("LLMLite"),{ok:!0,provider:"llmlite",model:r.model}}case"anthropic":{let r=await Vn();return await nt(t.workspacePath,{provider:"anthropic",apiKey:r.apiKey,baseUrl:r.baseUrl,model:r.model}),await k("ai_setup",`provider=anthropic,result=ok,model=${r.model}`),o.success(`AI ready \xB7 Anthropic Direct \xB7 model=${r.model} \xB7 ${r.baseUrl}`),Wn("Anthropic Direct API"),{ok:!0,provider:"anthropic",model:r.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 nt(t.workspacePath,{provider:"use-global",sourceSettings:n.rawSettings}),await k("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 k("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}import{spawnSync as hs}from"child_process";var Fe=1e4,qn=3e4,Jn=5,Ve="say ok",zn="2023-06-01";async function ks(t,e,n){o.info(`Testing LLMLite provider: ${t} (key: ${U(e)})`);let i=new AbortController,r=setTimeout(()=>i.abort(),Fe);try{let s=await fetch(`${t}/v1/models`,{headers:{Authorization:`Bearer ${e}`},signal:i.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(h=>typeof h.id=="string"?h.id:null).filter(h=>h!==null);if(o.success(`Connectivity OK \xB7 ${c.length} models available`),c.length>0){let h=c.slice(0,5).join(", "),S=c.length>5?` ...+${c.length-5} more`:"";o.dim(` Models: ${h}${S}`)}o.info(`Testing chat completion v\u1EDBi model "${n}"...`);let u=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:Ve}],max_tokens:Jn}),signal:i.signal});if(!u.ok){let h=(await u.text()).slice(0,200);throw new Error(`Chat completion fail (HTTP ${u.status}). ${h}`)}let l=await u.json(),m=typeof l.choices?.[0]?.message?.content=="string"?l.choices[0].message.content:"(empty response)",g=l.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 ${Fe/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):s}finally{clearTimeout(r)}}function ws(){o.info("Testing Subscription provider qua `claude --print`...");let t=hs("claude",["--print",Ve],{encoding:"utf8",timeout:qn});if(t.signal==="SIGTERM")throw new Error(`Timeout ${qn/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 ys(t,e,n){o.info(`Testing Anthropic Direct provider: ${t} (key: ${U(e)})`);let i=new AbortController,r=setTimeout(()=>i.abort(),Fe);try{let s=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":zn},signal:i.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 u=await fetch(`${t}/v1/messages`,{method:"POST",headers:{"x-api-key":e,"anthropic-version":zn,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:Jn,messages:[{role:"user",content:Ve}]}),signal:i.signal});if(!u.ok){let g=(await u.text()).slice(0,200);throw new Error(`Message endpoint fail (HTTP ${u.status}): ${g}`)}let m=((await u.json()).content||[]).map(g=>typeof g.text=="string"?g.text:"").join("").trim().slice(0,100);o.success(`Response: "${m}"`)}finally{clearTimeout(r)}}async function Yn(t){let e=t.env||{},n=typeof e.ANTHROPIC_BASE_URL=="string"?e.ANTHROPIC_BASE_URL:void 0,i=typeof e.ANTHROPIC_AUTH_TOKEN=="string"?e.ANTHROPIC_AUTH_TOKEN:void 0,r=typeof e.ANTHROPIC_API_KEY=="string"?e.ANTHROPIC_API_KEY:void 0,s=typeof t.model=="string"?t.model:"default";return r&&n?(await ys(n,r,s),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&i?(await ks(n,i,s),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(ws(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function te(){let t=process.cwd(),e=Yt(t);return e||(o.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
25
+ N\u1EBFu ch\u01B0a c\xF3 \u2192 TH\xCAM NGAY, tr\xE1nh leak key khi commit/push.`)}async function le(t){try{o.info("Setup AI provider cho workspace...");let e=Lt();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."),sr(),oe(),e=Lt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=tn();switch(await cr(n)){case"subscription":{let i=Xe();if(i.state!=="authenticated"&&(Qe(),i=Xe()),i.state==="authenticated"&&i.subscriptionType)return await rt(t.workspacePath,{provider:"subscription",model:j}),await y("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=Ze();if(!s.ok&&s.reason==="auth-expired"&&(o.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),Qe(),s=Ze()),!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 rt(t.workspacePath,{provider:"subscription",model:j}),await y("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 y("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 ${nr(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await rt(t.workspacePath,{provider:"subscription",model:j}),await y("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 mr();return await rt(t.workspacePath,{provider:"llmlite",apiKey:i.apiKey,baseUrl:i.baseUrl,model:i.model}),await y("ai_setup",`provider=llmlite,result=ok,model=${i.model},base=${i.baseUrl}`),o.success(`AI ready \xB7 LLMLite \xB7 model=${i.model} \xB7 ${i.baseUrl}`),dr("LLMLite"),{ok:!0,provider:"llmlite",model:i.model}}case"anthropic":{let i=await ur();return await rt(t.workspacePath,{provider:"anthropic",apiKey:i.apiKey,baseUrl:i.baseUrl,model:i.model}),await y("ai_setup",`provider=anthropic,result=ok,model=${i.model}`),o.success(`AI ready \xB7 Anthropic Direct \xB7 model=${i.model} \xB7 ${i.baseUrl}`),dr("Anthropic Direct API"),{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 rt(t.workspacePath,{provider:"use-global",sourceSettings:n.rawSettings}),await y("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 y("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}import{spawnSync as Js}from"child_process";var nn=1e4,gr=3e4,hr=5,rn="say ok",fr="2023-06-01";async function Ys(t,e,n){o.info(`Testing LLMLite provider: ${t} (key: ${D(e)})`);let r=new AbortController,i=setTimeout(()=>r.abort(),nn);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(h=>typeof h.id=="string"?h.id:null).filter(h=>h!==null);if(o.success(`Connectivity OK \xB7 ${c.length} models available`),c.length>0){let h=c.slice(0,5).join(", "),x=c.length>5?` ...+${c.length-5} more`:"";o.dim(` Models: ${h}${x}`)}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:rn}],max_tokens:hr}),signal:r.signal});if(!l.ok){let h=(await l.text()).slice(0,200);throw new Error(`Chat completion fail (HTTP ${l.status}). ${h}`)}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 ${nn/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):s}finally{clearTimeout(i)}}function Xs(){o.info("Testing Subscription provider qua `claude --print`...");let t=Js("claude",["--print",rn],{encoding:"utf8",timeout:gr});if(t.signal==="SIGTERM")throw new Error(`Timeout ${gr/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 Qs(t,e,n){o.info(`Testing Anthropic Direct provider: ${t} (key: ${D(e)})`);let r=new AbortController,i=setTimeout(()=>r.abort(),nn);try{let s=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":fr},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":fr,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:hr,messages:[{role:"user",content:rn}]}),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 kr(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 Qs(n,i,s),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&r?(await Ys(n,r,s),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(Xs(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function ue(){let t=process.cwd(),e=wt(t);return e||(o.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
16
26
  Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
17
27
  B\u1EA1n \u0111ang \u1EDF: ${t}
18
- 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 Be(t){let e=Xn(t,".claude","settings.json");if(!await d(e))return{};try{return await w(e)}catch{return{}}}async function xs(){let t=await te();await Zt({workspacePath:t})}async function Ss(){let t=await te(),e=await Be(t),n=e.env||{},i=typeof n.ANTHROPIC_BASE_URL=="string"?n.ANTHROPIC_BASE_URL:void 0,r=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,u;s?(c=i?.includes("api.anthropic.com")||s.startsWith("sk-ant-")?"Anthropic Direct":"Custom (API key)",u=U(s)):i&&r?(c="LLMLite",u=U(r)):r?(c="Custom",u=U(r)):(c="Subscription (default)",u="(kh\xF4ng set \u2014 d\xF9ng subscription auth)"),o.info(`Project: ${t}`),o.info(`Provider: ${c}${i?` (${i})`:""}`),o.info(`Model: ${a??"(default \u2014 Claude Code ch\u1ECDn)"}`),o.info(`Token: ${u}`)}async function Cs(){let t=await te(),e=await Be(t);try{let n=await Yn(e);o.success(`\u2713 ${n.message}`)}catch(n){o.error(`Test fail: ${n.message}`),process.exit(1)}}async function As(t){let e=await te(),n=Xn(e,".claude","settings.json"),i=await Be(e);if(!t.yes&&!await vs({message:"X\xF3a AI config (v\u1EC1 d\xF9ng Claude Code Subscription default)?",default:!1})){o.dim("\u0110\xE3 h\u1EE7y.");return}let{env:r,...s}=i,a={...s};if(r){let{ANTHROPIC_BASE_URL:c,ANTHROPIC_AUTH_TOKEN:u,ANTHROPIC_API_KEY:l,...m}=r;Object.keys(m).length>0&&(a.env=m)}Object.keys(a).length===0?(await bs.unlink(n).catch(()=>{}),o.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await R(n,a,384),o.success("\u0110\xE3 reset env block trong .claude/settings.json"))}function Qn(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 xs()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await Ss()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await Cs()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await As(n)})}import{input as Es}from"@inquirer/prompts";import{spawnSync as Ps}from"child_process";import{existsSync as Ke}from"fs";import{join as ee}from"path";function $s(t){let e=ee(t,"src",".git"),n=ee(t,".git"),i=ee(t,".claude");if(!Ke(n))throw new Error(`Kh\xF4ng ph\u1EA3i workspace root: ${t}
19
- Ch\u1EA1y 'avatar commit' trong workspace dir (c\xF3 .git v\xE0 .claude/).`);if(!Ke(i))throw new Error(`Kh\xF4ng th\u1EA5y .claude/ trong ${t}.
20
- Ch\u1EA1y 'avatar commit' trong Avatar workspace, kh\xF4ng ph\u1EA3i project b\xECnh th\u01B0\u1EDDng.`);if(!Ke(e))throw new Error(`Kh\xF4ng th\u1EA5y src/.git trong ${t}.
21
- Workspace thi\u1EBFu submodule src/. Ch\u1EA1y 'avatar init' l\u1EA1i?`)}function j(t,e){let n=Ps("git",["-C",t,...e],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(n.status!==0){let i=(n.stderr||"").trim();throw new Error(`git ${e.join(" ")} (in ${t}) failed:
22
- ${i}`)}return(n.stdout||"").trim()}function Zn(t){return j(t,["status","--porcelain"]).length>0}async function Rs(t,e){let n=ee(t,"src");if(!Zn(n))return o.dim("src/: nothing to commit (clean)"),{};o.info("Committing src/ ..."),j(n,["add","."]),j(n,["commit","-m",e.message]);let i=j(n,["rev-parse","HEAD"]);o.success(`src/ committed: ${i.slice(0,7)}`);let r=!1;return e.push&&(o.info("Pushing src/ ..."),j(n,["push"]),o.success("src/ pushed"),r=!0),{sha:i,pushed:r}}async function Ts(t,e){if(!Zn(t))return o.dim("workspace: nothing to commit (clean)"),{};o.info("Committing workspace root ..."),j(t,["add","."]),j(t,["commit","-m",e.message]);let n=j(t,["rev-parse","HEAD"]);o.success(`workspace committed: ${n.slice(0,7)}`);let i=!1;return e.push&&(o.info("Pushing workspace ..."),j(t,["push"]),o.success("workspace pushed"),i=!0),{sha:n,pushed:i}}async function ti(t){$s(t.workspaceRoot);let e={target:t.target,skipped:[]};if(t.target==="src"||t.target==="all"){let n=await Rs(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 Ts(t.workspaceRoot,t.options);e.workspaceCommitSha=n.sha,e.workspacePushed=n.pushed,n.sha||e.skipped?.push("workspace")}return e}function ei(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 _s(n)})}async function _s(t){try{let e=t.message??await Es({message:"Commit message:",validate:i=>i.trim().length>0?!0:"Message b\u1EAFt bu\u1ED9c"}),n=await ti({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 Ye}from"child_process";import{promises as Xe}from"fs";import{join as it}from"path";import ia from"boxen";import{join as Bs}from"path";import{promises as Gs}from"fs";import{join as si}from"path";import{promises as Is}from"fs";import ni,{join as We}from"path";async function Os(t,e){let i=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!i?.[2])return!0;let r=i[2];if(r.startsWith("/"))return o.warn(`Pack hook reject: absolute path "${r}" \u2014 ch\u1EC9 accept relative path t\u1EDBi workspace.`),!1;let s=We(t,r),a=ni.resolve(t),c=ni.resolve(s);return!c.startsWith(`${a}/`)&&c!==a?(o.warn(`Pack hook reject: path "${r}" resolve ra ngo\xE0i workspace (path traversal).`),!1):await d(s)}function qe(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 Et(t,e){let n=new Set,i=[];for(let r of[...t,...e]){let s=typeof r=="string"?r:JSON.stringify(r);n.has(s)||(n.add(s),i.push(r))}return i}function Ns(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function js(t){if(typeof t!="object"||t===null)return[];let e=t,n=Array.isArray(e.hooks)?e.hooks:[],i=[];for(let r of n)if(typeof r=="object"&&r!==null){let s=r.command;typeof s=="string"&&i.push(s)}return i}function ii(t){let e=t.map((s,a)=>{let c=js(s),u=c.some(l=>l.includes("${CLAUDE_PROJECT_DIR}"));return{index:a,entry:s,commands:c,hasVar:u}}),n=new Map;for(let s of e){let a=s.commands.map(Ns).sort().join("|");if(!a)continue;let c=n.get(a)??[];c.push(s),n.set(a,c)}let i=new Set;for(let[,s]of n)if(!(s.length<2||!s.some(c=>c.hasVar)))for(let c of s)c.hasVar||i.add(c.index);return i.size===0?{entries:t,droppedCount:0}:{entries:t.filter((s,a)=>!i.has(a)),droppedCount:i.size}}function Ls(t,e){let n=[],i={...e},r=0;for(let s of Object.keys(i)){let a=i[s]||[],{entries:c,droppedCount:u}=ii(a);u>0&&(i[s]=c,r+=u,n.includes(s)||n.push(s))}for(let[s,a]of Object.entries(t)){let c=i[s]||[],u=Et(c,a),{entries:l,droppedCount:m}=ii(u);m>0&&(r+=m),l.length!==c.length&&(n.includes(s)||n.push(s)),i[s]=l}return{merged:i,touchedEvents:n,migratedCount:r}}async function ne(t){let e=We(t,".claude","pack","templates","settings.json.tpl"),n=We(t,".claude","settings.json");if(!await d(e))return{action:"no-pack-template",changes:[]};let i;try{let l=await M(e);i=JSON.parse(l)}catch(l){throw new Error(`Pack settings template kh\xF4ng parse \u0111\u01B0\u1EE3c JSON: ${l.message}. Path: ${e}`)}let r={},s=!1;if(await d(n)){s=!0;try{r=await w(n)}catch(l){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${l.message}. Manual fix tr\u01B0\u1EDBc khi sync.`)}}let a=[],c={...r};if(i.statusLine&&!r.statusLine&&(await Os(t,i.statusLine.command)?(c.statusLine=i.statusLine,a.push("statusLine added")):a.push(`statusLine SKIPPED (file ref '${i.statusLine.command}' kh\xF4ng t\u1ED3n t\u1EA1i)`)),typeof i.includeCoAuthoredBy=="boolean"&&typeof r.includeCoAuthoredBy!="boolean"&&(c.includeCoAuthoredBy=i.includeCoAuthoredBy,a.push("includeCoAuthoredBy added")),i.model&&!r.model&&(c.model=i.model,a.push("model added")),i.env){let l={...r.env||{}},m=!1;for(let[g,h]of Object.entries(i.env))g in l||(l[g]=h,m=!0);m&&(c.env=l,a.push("env vars added from pack"))}if(i.permissions){let l=r.permissions?.allow||[],m=r.permissions?.deny||[],g=i.permissions.allow||[],h=i.permissions.deny||[],S=Et(l,g),C=Et(m,h);(S.length!==l.length||C.length!==m.length)&&(c.permissions={allow:S,deny:C},a.push(`permissions union (+${S.length-l.length} allow, +${C.length-m.length} deny)`))}if(i.hooks){let l=r.hooks||{},{merged:m,touchedEvents:g,migratedCount:h}=Ls(i.hooks,l);(g.length>0||h>0)&&(c.hooks=m,g.length>0&&a.push(`hooks added for events: ${g.join(", ")}`),h>0&&a.push(`migrated ${h} 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 u;return s&&(u=qe(n),await Is.copyFile(n,u)),await R(n,c),{action:"merged",backupPath:u,changes:a}}function Ms(t,e){return si(t,".claude","pack","features",e,"feature.json")}async function _t(t,e){let n=Ms(t,e);return await d(n)?await w(n):null}var Hs=[".claude","settings.json"];function ai(t){return si(t,...Hs)}function ie(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function ci(t){let e=ai(t);if(!await d(e))return{settings:{},existed:!1};try{return{settings:await w(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 ui(t,e,n){let i=ai(t),r;return n&&(r=qe(i),await Gs.copyFile(i,r)),await R(i,e),r}function ri(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function oi(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function Us(t,e){let n=new Set;for(let s of e)oi(s)&&n.add(ri(s));if(n.size===0)return{migratedUser:t,droppedCount:0};let i=0;return{migratedUser:t.filter(s=>oi(s)?!0:n.has(ri(s))?(i++,!1):!0),droppedCount:i}}async function re(t,e){let{settings:n,existed:i}=await ci(t),r={...n},s=[],a=e.settings.hooks??{};if(Object.keys(a).length>0){let l=n.hooks??{},m={...l},g=[],h=0;for(let[S,C]of Object.entries(a)){let E=l[S]??[],{migratedUser:lt,droppedCount:A}=Us(E,C);A>0&&(h+=A);let I=new Set(lt.map(ie)),_=C.filter(pt=>!I.has(ie(pt)));(_.length>0||A>0)&&(m[S]=[...lt,..._],g.push(S))}g.length>0&&(r.hooks=m,s.push(`hooks added: ${g.join(", ")}`),h>0&&s.push(`migrated ${h} stale relative-path entries \u2192 \\{CLAUDE_PROJECT_DIR\\} version`))}let c=e.settings.permissions?.deny??[];if(c.length>0){let l=n.permissions?.deny??[],m=Et(l,c);m.length!==l.length&&(r.permissions={...n.permissions,deny:m},s.push(`deny +${m.length-l.length}`))}return s.length===0?{action:"no-change",changes:[]}:{action:"enabled",backupPath:await ui(t,r,i),changes:s}}async function li(t,e){let{settings:n,existed:i}=await ci(t);if(!i)return{action:"no-change",changes:[]};let{hooks:r,permissions:s,...a}=n,c={...a},u=[],l=e.settings.hooks??{},m=n.hooks??{},g={},h=[];for(let[A,I]of Object.entries(m)){let _=l[A];if(!_){g[A]=I;continue}let pt=new Set(_.map(ie)),mt=I.filter(qt=>!pt.has(ie(qt)));mt.length!==I.length&&h.push(A),mt.length>0&&(g[A]=mt)}Object.keys(g).length>0&&(c.hooks=g),h.length>0&&u.push(`hooks removed: ${h.join(", ")}`);let S=new Set(e.settings.permissions?.deny??[]),C=n.permissions?.deny??[],E=S.size>0?C.filter(A=>!S.has(A)):C;if(n.permissions){let{deny:A,...I}=n.permissions,_={...I,...E.length>0?{deny:E}:{}};Object.keys(_).length>0&&(c.permissions=_)}return E.length!==C.length&&u.push(`deny -${C.length-E.length}`),u.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await ui(t,c,i),changes:u}}import{join as Ds}from"path";var Fs=".claude/avatar-features.json";function pi(){return{features:{}}}function mi(t){return Ds(t,Fs)}async function ft(t){let e=mi(t);if(!await d(e))return pi();try{return{features:(await w(e)).features??{}}}catch{return pi()}}async function Vs(t,e){await R(mi(t),e)}async function oe(t,e,n){let i=await ft(t);return i.features[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await Vs(t,i),i}async function ht(t){let e=await ft(t);return Object.entries(e.features).filter(([,n])=>n.enabled).map(([n])=>n)}function Ks(t,e){let n=e.settings.hooks??{},i=t.hooks??{};for(let[r,s]of Object.entries(n)){let a=i[r]??[],c=new Set(a.flatMap(u=>(u.hooks??[]).map(l=>l.command??"")));for(let u of s)for(let l of u.hooks??[])if(l.command&&!c.has(l.command))return!1}return!0}async function di(t){let e=await ht(t);if(e.length===0)return[];let n=Bs(t,".claude","settings.json"),i={};if(await d(n))try{i=await w(n)}catch{i={}}let r=[];for(let s of e){let a=await _t(t,s);if(!a){r.push({name:`Feature: ${s}`,status:"warn",detail:"state=enabled nh\u01B0ng pack thi\u1EBFu manifest (version skew). Ch\u1EA1y 'avatar sync'.",fixable:!1});continue}Ks(i,a)?r.push({name:`Feature: ${s}`,status:"ok",detail:`enabled, hook active (v${a.version})`,fixable:!1}):r.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 re(t,a)}})}return r}import{join as ze}from"path";import{simpleGit as Ws}from"simple-git";function y(t=process.cwd()){return Ws({baseDir:t,binary:"git"})}async function kt(t=process.cwd()){return await d(ze(t,".git"))}async function gi(t,e,n=process.cwd()){await y(n).subModule(["add",t,e])}async function It(t,e,n=process.cwd()){let i=ze(n,t);await y(i).fetch(["--tags"]),await y(i).checkout(e)}async function se(t,e,n=process.cwd()){let i=ze(n,t);await y(i).fetch(["origin"]),await y(i).checkout(["-B",e,`origin/${e}`])}async function D(t=process.cwd()){return(await y(t).tags()).all}async function ae(t=process.cwd()){try{return(await y(t).raw(["describe","--tags","--exact-match","HEAD"])).trim()||null}catch{return null}}async function wt(t=process.cwd()){return(await y(t).revparse(["HEAD"])).trim()}import{promises as yi}from"fs";import{join as F}from"path";import{existsSync as zs}from"fs";import{dirname as hi,join as Ot}from"path";import{fileURLToPath as Js}from"url";var qs=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g;function fi(t,e){return t.replace(qs,(n,i)=>{let r=e[i];return r===void 0?n:String(r)})}var Ys=hi(Js(import.meta.url)),ki=Zs(Ys),Xs=Ot(ki,"src","templates"),Qs=Ot(ki,"src","hooks");function Zs(t){let e=t;for(;;){if(zs(Ot(e,"package.json")))return e;let n=hi(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function ta(t){return await M(Ot(Xs,`${t}.tpl`))}async function ce(t,e){let n=await ta(t);return fi(n,e)}async function wi(t){return await M(Ot(Qs,`${t}.sh.tpl`))}async function ea(t){if(!await d(t))return null;let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`,i=n,r=1;for(;await d(i);)if(i=`${n}-${r}`,r++,r>5)throw new Error(`Could not find free backup name for ${t}`);return await yi.rename(t,i),i}async function bi(t,e,n){let i=await ea(t);return await gt(t,e,n),i}var na=["state","_pending","_backup"];async function vi(t){let e=F(t,".claude");await P(e);for(let n of na){let i=F(e,n);await P(i),await gt(F(i,".gitkeep"),"")}}async function xi(t,e){return[]}async function Je(t,e){let n=await ce("CLAUDE.md",e);return await bi(F(t,"CLAUDE.md"),n)}async function Si(t,e){let n=await ce("settings.json",e);return await bi(F(t,".claude","settings.json"),n)}async function Ci(t){let e=F(t,".gitignore"),n=await ce("gitignore",{}),i="# Avatar \u2014 git-ignored entries injected on `avatar init`",r="";if(await d(e)&&(r=await yi.readFile(e,"utf8"),r.includes(i)))return;let s=r.endsWith(`
23
- `)||r.length===0?"":`
24
- `;await gt(e,`${r}${s}
25
- ${n}`)}async function Nt(t,e){let n=await wi(e),i=F(t,"hooks");await P(i);let r=F(i,e);await gt(r,n,493)}function Ai(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 ra(process.cwd());oa(n),e.fix&&await sa(n)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function ra(t){let e=[],n=process.versions.node,[i,r]=n.split(".").map(O=>Number.parseInt(O,10)),s=(i??0)>18||(i??0)===18&&(r??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 X();a?Q(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=it(t,".claude","pack"),u=it(t,"CLAUDE.md"),l=it(t,".git","hooks","post-merge"),[m,g,h,S]=await Promise.all([kt(t),d(c),d(u),d(l)]);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:h?"ok":"warn",detail:h?"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 Nt(it(t,".git"),"post-merge")}});let C=it(t,".gitignore"),E="";if(m){let O=!1;await d(C)&&(E=await Xe.readFile(C,"utf8"),O=E.includes(".claude/_pending/")),e.push({name:".gitignore Avatar entries",status:O?"ok":g?"fail":"warn",detail:O?"c\xF3 .claude/_pending/, .claude/_backup/":"thi\u1EBFu entries",fixable:!1});let J=E.includes(".claude/settings.json")||E.includes("/.claude/settings.json")||E.includes(".claude/*.json");e.push({name:"\u{1F512} settings.json gitignored",status:J?"ok":"fail",detail:J?"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:!J,fix:J?void 0:async()=>{await Xe.appendFile(C,`
28
+ 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 on(t){let e=wr(t,".claude","settings.json");if(!await d(e))return{};try{return await w(e)}catch{return{}}}async function ea(){let t=await ue();await le({workspacePath:t})}async function na(){let t=await ue(),e=await on(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=D(s)):r&&i?(c="LLMLite",l=D(i)):i?(c="Custom",l=D(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 ra(){let t=await ue(),e=await on(t);try{let n=await kr(e);o.success(`\u2713 ${n.message}`)}catch(n){o.error(`Test fail: ${n.message}`),process.exit(1)}}async function ia(t){let e=await ue(),n=wr(e,".claude","settings.json"),r=await on(e);if(!t.yes&&!await ta({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 Zs.unlink(n).catch(()=>{}),o.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await A(n,a,384),o.success("\u0110\xE3 reset env block trong .claude/settings.json"))}function yr(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 ea()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await na()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await ra()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await ia(n)})}import{input as la}from"@inquirer/prompts";import{spawnSync as oa}from"child_process";import{existsSync as sn}from"fs";import{join as pe}from"path";function sa(t){let e=pe(t,"src",".git"),n=pe(t,".git"),r=pe(t,".claude");if(!sn(n))throw new Error(`Kh\xF4ng ph\u1EA3i workspace root: ${t}
29
+ Ch\u1EA1y 'avatar commit' trong workspace dir (c\xF3 .git v\xE0 .claude/).`);if(!sn(r))throw new Error(`Kh\xF4ng th\u1EA5y .claude/ trong ${t}.
30
+ Ch\u1EA1y 'avatar commit' trong Avatar workspace, kh\xF4ng ph\u1EA3i project b\xECnh th\u01B0\u1EDDng.`);if(!sn(e))throw new Error(`Kh\xF4ng th\u1EA5y src/.git trong ${t}.
31
+ Workspace thi\u1EBFu submodule src/. Ch\u1EA1y 'avatar init' l\u1EA1i?`)}function M(t,e){let n=oa("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:
32
+ ${r}`)}return(n.stdout||"").trim()}function vr(t){return M(t,["status","--porcelain"]).length>0}async function aa(t,e){let n=pe(t,"src");if(!vr(n))return o.dim("src/: nothing to commit (clean)"),{};o.info("Committing src/ ..."),M(n,["add","."]),M(n,["commit","-m",e.message]);let r=M(n,["rev-parse","HEAD"]);o.success(`src/ committed: ${r.slice(0,7)}`);let i=!1;return e.push&&(o.info("Pushing src/ ..."),M(n,["push"]),o.success("src/ pushed"),i=!0),{sha:r,pushed:i}}async function ca(t,e){if(!vr(t))return o.dim("workspace: nothing to commit (clean)"),{};o.info("Committing workspace root ..."),M(t,["add","."]),M(t,["commit","-m",e.message]);let n=M(t,["rev-parse","HEAD"]);o.success(`workspace committed: ${n.slice(0,7)}`);let r=!1;return e.push&&(o.info("Pushing workspace ..."),M(t,["push"]),o.success("workspace pushed"),r=!0),{sha:n,pushed:r}}async function br(t){sa(t.workspaceRoot);let e={target:t.target,skipped:[]};if(t.target==="src"||t.target==="all"){let n=await aa(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 ca(t.workspaceRoot,t.options);e.workspaceCommitSha=n.sha,e.workspacePushed=n.pushed,n.sha||e.skipped?.push("workspace")}return e}function xr(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 ua(n)})}async function ua(t){try{let e=t.message??await la({message:"Commit message:",validate:r=>r.trim().length>0?!0:"Message b\u1EAFt bu\u1ED9c"}),n=await br({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 pn}from"child_process";import{promises as mn}from"fs";import{join as it}from"path";import Ma from"boxen";import{join as Sa}from"path";k();import{promises as ha}from"fs";import{join as $r}from"path";k();import{promises as pa}from"fs";import Sr,{join as an}from"path";async function ma(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=an(t,i),a=Sr.resolve(t),c=Sr.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 cn(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 Gt(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 da(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function ga(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 Ar(t){let e=t.map((s,a)=>{let c=ga(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(da).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 fa(t,e){let n=[],r={...e},i=0;for(let s of Object.keys(r)){let a=r[s]||[],{entries:c,droppedCount:l}=Ar(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=Gt(c,a),{entries:u,droppedCount:m}=Ar(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 me(t){let e=an(t,".claude","pack","templates","settings.json.tpl"),n=an(t,".claude","settings.json");if(!await d(e))return{action:"no-pack-template",changes:[]};let r;try{let u=await H(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 w(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 ma(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){let u={...i.env||{}},m=!1;for(let[g,h]of Object.entries(r.env))g in u||(u[g]=h,m=!0);m&&(c.env=u,a.push("env vars added from pack"))}if(r.permissions){let u=i.permissions?.allow||[],m=i.permissions?.deny||[],g=r.permissions.allow||[],h=r.permissions.deny||[],x=Gt(u,g),C=Gt(m,h);(x.length!==u.length||C.length!==m.length)&&(c.permissions={allow:x,deny:C},a.push(`permissions union (+${x.length-u.length} allow, +${C.length-m.length} deny)`))}if(r.hooks){let u=i.hooks||{},{merged:m,touchedEvents:g,migratedCount:h}=fa(r.hooks,u);(g.length>0||h>0)&&(c.hooks=m,g.length>0&&a.push(`hooks added for events: ${g.join(", ")}`),h>0&&a.push(`migrated ${h} 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=cn(n),await pa.copyFile(n,l)),await A(n,c),{action:"merged",backupPath:l,changes:a}}function ka(t,e){return $r(t,".claude","pack","features",e,"feature.json")}async function Ht(t,e){let n=ka(t,e);return await d(n)?await w(n):null}var wa=[".claude","settings.json"];function Er(t){return $r(t,...wa)}function de(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function Rr(t){let e=Er(t);if(!await d(e))return{settings:{},existed:!1};try{return{settings:await w(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 Tr(t,e,n){let r=Er(t),i;return n&&(i=cn(r),await ha.copyFile(r,i)),await A(r,e),i}function Cr(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function Pr(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function ya(t,e){let n=new Set;for(let s of e)Pr(s)&&n.add(Cr(s));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(s=>Pr(s)?!0:n.has(Cr(s))?(r++,!1):!0),droppedCount:r}}async function ge(t,e){let{settings:n,existed:r}=await Rr(t),i={...n},s=[],a=e.settings.hooks??{};if(Object.keys(a).length>0){let u=n.hooks??{},m={...u},g=[],h=0;for(let[x,C]of Object.entries(a)){let T=u[x]??[],{migratedUser:dt,droppedCount:P}=ya(T,C);P>0&&(h+=P);let O=new Set(dt.map(de)),I=C.filter(gt=>!O.has(de(gt)));(I.length>0||P>0)&&(m[x]=[...dt,...I],g.push(x))}g.length>0&&(i.hooks=m,s.push(`hooks added: ${g.join(", ")}`),h>0&&s.push(`migrated ${h} 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=Gt(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 Tr(t,i,r),changes:s}}async function _r(t,e){let{settings:n,existed:r}=await Rr(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={},h=[];for(let[P,O]of Object.entries(m)){let I=u[P];if(!I){g[P]=O;continue}let gt=new Set(I.map(de)),ft=O.filter(ne=>!gt.has(de(ne)));ft.length!==O.length&&h.push(P),ft.length>0&&(g[P]=ft)}Object.keys(g).length>0&&(c.hooks=g),h.length>0&&l.push(`hooks removed: ${h.join(", ")}`);let x=new Set(e.settings.permissions?.deny??[]),C=n.permissions?.deny??[],T=x.size>0?C.filter(P=>!x.has(P)):C;if(n.permissions){let{deny:P,...O}=n.permissions,I={...O,...T.length>0?{deny:T}:{}};Object.keys(I).length>0&&(c.permissions=I)}return T.length!==C.length&&l.push(`deny -${C.length-T.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await Tr(t,c,r),changes:l}}k();import{join as va}from"path";var ba=".claude/avatar-features.json";function Ir(){return{features:{}}}function Or(t){return va(t,ba)}async function yt(t){let e=Or(t);if(!await d(e))return Ir();try{return{features:(await w(e)).features??{}}}catch{return Ir()}}async function xa(t,e){await A(Or(t),e)}async function fe(t,e,n){let r=await yt(t);return r.features[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await xa(t,r),r}async function vt(t){let e=await yt(t);return Object.entries(e.features).filter(([,n])=>n.enabled).map(([n])=>n)}k();function Aa(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 Nr(t){let e=await vt(t);if(e.length===0)return[];let n=Sa(t,".claude","settings.json"),r={};if(await d(n))try{r=await w(n)}catch{r={}}let i=[];for(let s of e){let a=await Ht(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}Aa(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 ge(t,a)}})}return i}k();k();import{join as ln}from"path";import{simpleGit as Ca}from"simple-git";function v(t=process.cwd()){return Ca({baseDir:t,binary:"git"})}async function bt(t=process.cwd()){return await d(ln(t,".git"))}async function jr(t,e,n=process.cwd()){await v(n).subModule(["add",t,e])}async function Ut(t,e,n=process.cwd()){let r=ln(n,t);await v(r).fetch(["--tags"]),await v(r).checkout(e)}async function he(t,e,n=process.cwd()){let r=ln(n,t);await v(r).fetch(["origin"]),await v(r).checkout(["-B",e,`origin/${e}`])}async function F(t=process.cwd()){return(await v(t).tags()).all}async function ke(t=process.cwd()){try{return(await v(t).raw(["describe","--tags","--exact-match","HEAD"])).trim()||null}catch{return null}}async function xt(t=process.cwd()){return(await v(t).revparse(["HEAD"])).trim()}k();import{promises as Ur}from"fs";import{join as V}from"path";k();import{existsSync as $a}from"fs";import{dirname as Lr,join as Dt}from"path";import{fileURLToPath as Ea}from"url";var Pa=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g;function Mr(t,e){return t.replace(Pa,(n,r)=>{let i=e[r];return i===void 0?n:String(i)})}var Ra=Lr(Ea(import.meta.url)),Gr=Ia(Ra),Ta=Dt(Gr,"src","templates"),_a=Dt(Gr,"src","hooks");function Ia(t){let e=t;for(;;){if($a(Dt(e,"package.json")))return e;let n=Lr(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function Oa(t){return await H(Dt(Ta,`${t}.tpl`))}async function we(t,e){let n=await Oa(t);return Mr(n,e)}async function Hr(t){return await H(Dt(_a,`${t}.sh.tpl`))}async function Na(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 Ur.rename(t,r),r}async function Dr(t,e,n){let r=await Na(t);return await kt(t,e,n),r}var ja=["state","_pending","_backup"];async function Fr(t){let e=V(t,".claude");await $(e);for(let n of ja){let r=V(e,n);await $(r),await kt(V(r,".gitkeep"),"")}}async function Vr(t,e){return[]}async function un(t,e){let n=await we("CLAUDE.md",e);return await Dr(V(t,"CLAUDE.md"),n)}async function Kr(t,e){let n=await we("settings.json",e);return await Dr(V(t,".claude","settings.json"),n)}async function Br(t){let e=V(t,".gitignore"),n=await we("gitignore",{}),r="# Avatar \u2014 git-ignored entries injected on `avatar init`",i="";if(await d(e)&&(i=await Ur.readFile(e,"utf8"),i.includes(r)))return;let s=i.endsWith(`
33
+ `)||i.length===0?"":`
34
+ `;await kt(e,`${i}${s}
35
+ ${n}`)}async function Ft(t,e){let n=await Hr(e),r=V(t,"hooks");await $(r);let i=V(r,e);await kt(i,n,493)}function Wr(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 La(process.cwd());Ga(n),e.fix&&await Ha(n)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function La(t){let e=[],n=process.versions.node,[r,i]=n.split(".").map(N=>Number.parseInt(N,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 Q();a?Z(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=it(t,".claude","pack"),l=it(t,"CLAUDE.md"),u=it(t,".git","hooks","post-merge"),[m,g,h,x]=await Promise.all([bt(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:h?"ok":"warn",detail:h?"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:x?"ok":"fail",detail:x?"installed":"missing \u2014 fixable",fixable:!x,fix:x?void 0:async()=>{await Ft(it(t,".git"),"post-merge")}});let C=it(t,".gitignore"),T="";if(m){let N=!1;await d(C)&&(T=await mn.readFile(C,"utf8"),N=T.includes(".claude/_pending/")),e.push({name:".gitignore Avatar entries",status:N?"ok":g?"fail":"warn",detail:N?"c\xF3 .claude/_pending/, .claude/_backup/":"thi\u1EBFu entries",fixable:!1});let Y=T.includes(".claude/settings.json")||T.includes("/.claude/settings.json")||T.includes(".claude/*.json");e.push({name:"\u{1F512} settings.json gitignored",status:Y?"ok":"fail",detail:Y?"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:!Y,fix:Y?void 0:async()=>{await mn.appendFile(C,`
26
36
  # Avatar v1.7.0 \u2014 Security: settings.json ch\u1EE9a raw API key, KH\xD4NG commit.
27
37
  .claude/settings.json
28
38
  .claude/settings.json.backup-*
29
- `)}})}let lt=Ye("which",["python"]),A=Ye("which",["python3"]),I=lt.status===0,_=A.status===0;_&&!I?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}):I?e.push({name:"Python binary",status:"ok",detail:`python: ${lt.stdout.toString().trim()}`,fixable:!1}):_&&e.push({name:"Python binary",status:"ok",detail:`python3: ${A.stdout.toString().trim()}`,fixable:!1});let pt=it(t,".claude","settings.json");if(await d(pt))try{let O=await Xe.readFile(pt,"utf8"),J=JSON.parse(O);if(J.statusLine?.command){let Rn=J.statusLine.command.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(Rn?.[2]){let At=Rn[2],vo=At.startsWith("/")?At:it(t,At),Tn=await d(vo);e.push({name:"statusLine command",status:Tn?"ok":"fail",detail:Tn?`ref OK: ${At}`:`BROKEN: settings.json ref '${At}' nh\u01B0ng file kh\xF4ng t\u1ED3n t\u1EA1i. Strip field statusLine ho\u1EB7c fix path.`,fixable:!1})}}}catch{}let mt=Ye("which",["claude"]),qt=mt.status===0;if(e.push({name:"Claude Code CLI",status:qt?"ok":"warn",detail:qt?mt.stdout.toString().trim():"kh\xF4ng t\xECm th\u1EA5y 'claude' tr\xEAn PATH",fixable:!1}),g){let O=await di(t);e.push(...O)}return e}function oa(t){let e=[p.bold("Avatar Doctor"),"\u2500".repeat(48)],n=0,i=0,r=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:(i+=1,s.fixable&&(r+=1))}e.push("\u2500".repeat(48)),e.push(`${n} checks passed, ${i} issue${i===1?"":"s"}${r>0?` (${r} fixable \u2014 ch\u1EA1y 'avatar doctor --fix')`:""}`),process.stdout.write(`${ia(e.join(`
39
+ `)}})}let dt=pn("which",["python"]),P=pn("which",["python3"]),O=dt.status===0,I=P.status===0;I&&!O?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 ${P.stdout.toString().trim()} ~/.local/bin/python`,fixable:!1}):O?e.push({name:"Python binary",status:"ok",detail:`python: ${dt.stdout.toString().trim()}`,fixable:!1}):I&&e.push({name:"Python binary",status:"ok",detail:`python3: ${P.stdout.toString().trim()}`,fixable:!1});let gt=it(t,".claude","settings.json");if(await d(gt))try{let N=await mn.readFile(gt,"utf8"),Y=JSON.parse(N);if(Y.statusLine?.command){let Jn=Y.statusLine.command.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(Jn?.[2]){let Ot=Jn[2],ts=Ot.startsWith("/")?Ot:it(t,Ot),Yn=await d(ts);e.push({name:"statusLine command",status:Yn?"ok":"fail",detail:Yn?`ref OK: ${Ot}`:`BROKEN: settings.json ref '${Ot}' nh\u01B0ng file kh\xF4ng t\u1ED3n t\u1EA1i. Strip field statusLine ho\u1EB7c fix path.`,fixable:!1})}}}catch{}let ft=pn("which",["claude"]),ne=ft.status===0;if(e.push({name:"Claude Code CLI",status:ne?"ok":"warn",detail:ne?ft.stdout.toString().trim():"kh\xF4ng t\xECm th\u1EA5y 'claude' tr\xEAn PATH",fixable:!1}),g){let N=await Nr(t);e.push(...N)}return e}function Ga(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(`${Ma(e.join(`
30
40
  `),{padding:1,borderStyle:"round"})}
31
- `)}async function sa(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(i){o.error(`Failed to fix ${n.name}: ${i instanceof Error?i.message:String(i)}`)}e===0&&o.dim("Kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 fix t\u1EF1 \u0111\u1ED9ng.")}import{resolve as pa}from"path";import{checkbox as ma,confirm as da}from"@inquirer/prompts";import{promises as aa}from"fs";import{join as Qe}from"path";var ca=[".claude","pack","features"];function Pi(t){return Qe(t,...ca)}async function Ze(t){let e=Pi(t);if(!await d(e))return[];let n=await aa.readdir(e,{withFileTypes:!0}),i=[];for(let r of n)r.isDirectory()&&await d(Qe(e,r.name,"feature.json"))&&i.push(r.name);return i.sort()}async function $i(t){let e=Qe(Pi(t),"defaults.json");if(!await d(e))return[];try{return(await w(e)).defaultFeatures??[]}catch{return[]}}async function jt(t){let[e,n]=await Promise.all([Ze(t),ht(t)]);return{available:e,enabled:n}}function Ri(t,e){let n=new Set(e);return t.map(i=>({name:n.has(i)?`${i} (installed)`:i,value:i,checked:n.has(i)}))}function Ti(t){return t.map(e=>({name:e,value:e,checked:!1}))}function Ei(t){return[...t]}import{spawnSync as ua}from"child_process";function la(t){if(!t)return;let e=ua(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 _i(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 rt(t,e,n={}){let i=await _t(t,e);if(!i)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;la(i.requires?.runtime);let r=await re(t,i);return _i(e,r),await oe(t,e,{enabled:!0,version:i.version}),!0}async function tn(t,e){let n=await _t(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 ft(t)).features[e]?.version??"unknown";return await oe(t,e,{enabled:!1,version:s}),!1}let i=await li(t,n);return _i(e,i),await oe(t,e,{enabled:!1,version:n.version}),!0}function Lt(t){return pa(t.target??process.cwd())}async function Ii(t,e,n,i){if(n.all===!0||!process.stdout.isTTY)return Ei(t);let s=i==="add"?Ri(t,e):Ti(t);return await ma({message:i==="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 ga(t){let e=Lt(t),{available:n,enabled:i}=await jt(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 r=await Ii(n,i,t,"add");if(r.length===0){o.dim("Kh\xF4ng ch\u1ECDn feature n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}for(let s of r)await rt(e,s)}async function fa(t){let e=Lt(t),{enabled:n}=await jt(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 i=await Ii(n,[],t,"remove");if(i.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 da({message:`G\u1EE1 ${i.length} feature: ${i.join(", ")}?`,default:!0})){o.dim("H\u1EE7y \u2014 kh\xF4ng g\u1EE1 feature n\xE0o.");return}for(let s of i)await tn(e,s)}async function ha(t){let e=Lt(t),n=await Ze(e),i=await ft(e);if(n.length===0&&Object.keys(i.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 r=[...new Set([...n,...Object.keys(i.features)])].sort();o.info("Features:"),console.log(` ${"NAME".padEnd(20)} ${"AVAILABLE".padEnd(10)} ${"ENABLED".padEnd(8)} VERSION`);for(let s of r){let a=n.includes(s)?"yes":"no",c=i.features[s],u=c?.enabled?"yes":"no",l=c?.version??"-";console.log(` ${s.padEnd(20)} ${a.padEnd(10)} ${u.padEnd(8)} ${l}`)}}function Oi(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,i)=>{let r=Lt(i);await rt(r,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,i)=>{let r=Lt(i);await tn(r,n)}),e.command("list").description("Li\u1EC7t k\xEA feature: available | enabled | version").option("--target <path>","Workspace root (default: cwd)").action(async n=>{await ha(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 ga(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 fa(n)})}import{spawnSync as Qa}from"child_process";import{promises as Za}from"fs";import{join as Qi}from"path";import{spawnSync as ji}from"child_process";import{existsSync as ka}from"fs";import{join as Ni}from"path";var wa=120*1e3,ya=300*1e3,V=class extends Error{operation;reason;exitCode;stderr;constructor(e,n,i,r=null,s){super(i),this.name="GitnexusOperationError",this.operation=e,this.reason=n,this.exitCode=r,this.stderr=s}};function Li(t,e,n,i){if(n==="SIGTERM")return new V(t,"timeout",`gitnexus ${t} timeout. Check m\u1EA1ng / repo size.`,null,i);let r=i.toLowerCase();return r.includes("eacces")||r.includes("permission denied")?new V(t,"permission",`gitnexus ${t} fail (permission). Check write access ~/.claude ho\u1EB7c cwd.`,e,i):new V(t,"non-zero-exit",`gitnexus ${t} exit ${e??"null"}. Xem log ph\xEDa tr\xEAn.`,e,i)}function ue(t,e){return t.split(`
41
+ `)}async function Ha(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 Ka}from"path";import{checkbox as Ba,confirm as Wa}from"@inquirer/prompts";import{promises as Ua}from"fs";import{join as dn}from"path";k();var Da=[".claude","pack","features"];function zr(t){return dn(t,...Da)}async function gn(t){let e=zr(t);if(!await d(e))return[];let n=await Ua.readdir(e,{withFileTypes:!0}),r=[];for(let i of n)i.isDirectory()&&await d(dn(e,i.name,"feature.json"))&&r.push(i.name);return r.sort()}async function qr(t){let e=dn(zr(t),"defaults.json");if(!await d(e))return[];try{return(await w(e)).defaultFeatures??[]}catch{return[]}}async function Vt(t){let[e,n]=await Promise.all([gn(t),vt(t)]);return{available:e,enabled:n}}function Jr(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 Yr(t){return t.map(e=>({name:e,value:e,checked:!1}))}function Xr(t){return[...t]}import{spawnSync as Fa}from"child_process";function Va(t){if(!t)return;let e=Fa(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 Qr(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 ot(t,e,n={}){let r=await Ht(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;Va(r.requires?.runtime);let i=await ge(t,r);return Qr(e,i),await fe(t,e,{enabled:!0,version:r.version}),!0}async function fn(t,e){let n=await Ht(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 yt(t)).features[e]?.version??"unknown";return await fe(t,e,{enabled:!1,version:s}),!1}let r=await _r(t,n);return Qr(e,r),await fe(t,e,{enabled:!1,version:n.version}),!0}function Kt(t){return Ka(t.target??process.cwd())}async function Zr(t,e,n,r){if(n.all===!0||!process.stdout.isTTY)return Xr(t);let s=r==="add"?Jr(t,e):Yr(t);return await Ba({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 za(t){let e=Kt(t),{available:n,enabled:r}=await Vt(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 Zr(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 ot(e,s)}async function qa(t){let e=Kt(t),{enabled:n}=await Vt(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 Zr(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 Wa({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 fn(e,s)}async function Ja(t){let e=Kt(t),n=await gn(e),r=await yt(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 ti(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=Kt(r);await ot(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=Kt(r);await fn(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 Ja(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 za(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 qa(n)})}k();import{spawnSync as _c}from"child_process";import{promises as Ic}from"fs";import{join as yi}from"path";import{spawnSync as ni}from"child_process";import{existsSync as Ya}from"fs";import{join as ei}from"path";var Xa=120*1e3,Qa=300*1e3,K=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 ri(t,e,n,r){if(n==="SIGTERM")return new K(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 K(t,"permission",`gitnexus ${t} fail (permission). Check write access ~/.claude ho\u1EB7c cwd.`,e,r):new K(t,"non-zero-exit",`gitnexus ${t} exit ${e??"null"}. Xem log ph\xEDa tr\xEAn.`,e,r)}function ye(t,e){return t.split(`
32
42
  `).slice(-e).join(`
33
- `)}function Gi(){let t=Rt("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=ji("gitnexus",["setup"],{stdio:["ignore","pipe","pipe"],timeout:wa,encoding:"utf8"});if(e.status!==0||e.signal==="SIGTERM"){t.fail("GitNexus setup failed");let n=(e.stderr||"").trim(),i=(e.stdout||"").trim();throw n?process.stderr.write(`${ue(n,30)}
34
- `):i&&process.stderr.write(`${ue(i,30)}
35
- `),Li("setup",e.status,e.signal,n)}t.succeed("GitNexus setup OK (global skills installed)")}function le(t){let e=Rt(`Analyze workspace ${t} (1-3 ph\xFAt)`),n=ji("gitnexus",["analyze","."],{cwd:t,stdio:["ignore","pipe","pipe"],timeout:ya,encoding:"utf8"});if(n.status!==0||n.signal==="SIGTERM"){e.fail("Analyze failed");let r=(n.stderr||"").trim(),s=(n.stdout||"").trim();throw r?process.stderr.write(`${ue(r,30)}
36
- `):s&&process.stderr.write(`${ue(s,30)}
37
- `),Li("analyze",n.status,n.signal,r)}let i=Ni(t,".gitnexus","meta.json");if(!ka(i))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new V("analyze","missing-output",`gitnexus analyze xong nh\u01B0ng kh\xF4ng th\u1EA5y ${i}. Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus fail silent.`);e.succeed(`Analyze OK (index t\u1EA1i ${Ni(t,".gitnexus")})`)}import{confirm as qa}from"@inquirer/prompts";import za from"boxen";import{spawnSync as Mi}from"child_process";var ba=5e3,va=/(\d+\.\d+\.\d+)/;function xa(){let e=Z()==="win32"?"where":"which",n=Mi(e,["gitnexus"],{encoding:"utf8"});if(n.error||n.status!==0)return null;let i=(n.stdout||"").trim();return i?i.split(/\r?\n/)[0].trim():null}function Sa(){let t=Mi("gitnexus",["--version"],{encoding:"utf8",timeout:ba});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return va.exec(e)?.[1]??null}var ot=null;function Gt(){if(ot!==null)return ot;let t=xa();return t?(ot={installed:!0,version:Sa(),path:t},ot):(ot={installed:!1,version:null,path:null},ot)}function pe(){ot=null}import{spawnSync as Ca}from"child_process";var Hi=300*1e3,Ui="gitnexus",L=class extends Error{reason;exitCode;constructor(e,n,i=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=i}};function Aa(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new L("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${Ui} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new L("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new L("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function Di(){o.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Ca("npm",["install","-g",Ui],{stdio:["inherit","inherit","pipe"],timeout:Hi,encoding:"utf8"});if(t.signal==="SIGTERM")throw new L("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),Aa(t.status,t.stderr||"");pe();let e=Gt();if(!e.installed||!e.path)throw new L("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 Od,select as Pa}from"@inquirer/prompts";var b=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 Pa({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}import{promises as Bi}from"fs";import{homedir as $a}from"os";import{join as Ra}from"path";var Fi=384,Vi={command:"gitnexus",args:["mcp"]};function Ta(){return Ra($a(),".claude","mcp_servers.json")}function Ea(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 _a(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await Bi.copyFile(t,n),n}async function Ki(){let t=Ta(),e={},n=!1;if(await d(t)){n=!0;try{e=await w(t)}catch(a){throw new Error(`MCP config corrupted (${t}): ${a.message}. Backup + x\xF3a file \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}let i=e.mcp_servers?.gitnexus;if(i&&Ea(i,Vi))return o.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let r;n&&(r=await _a(t),o.dim(`Backup ${t} \u2192 ${r}`));let s={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:Vi}};await R(t,s,Fi);try{await Bi.chmod(t,Fi)}catch{}return o.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:r}}import{spawnSync as Ma}from"child_process";import{existsSync as Ha}from"fs";import{join as Yi}from"path";import{confirm as Ua}from"@inquirer/prompts";var Ia=[/^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 Wi(t){return t?Ia.some(e=>e.test(t)):!1}import{promises as Mt}from"fs";import{homedir as Oa}from"os";import{dirname as Na,join as qi}from"path";var ja=qi(Oa(),".gitnexus"),Ht=qi(ja,"config.json");async function La(){try{let t=await Mt.readFile(Ht,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function Ga(t){await Mt.mkdir(Na(Ht),{recursive:!0});let e=`${Ht}.tmp-${process.pid}`;await Mt.writeFile(e,t,{mode:384}),await Mt.rename(e,Ht)}async function zi(t){let n={...await La(),apiKey:t.apiKey,baseUrl:t.baseUrl,model:t.model,isReasoningModel:t.isReasoningModel??!1};await Ga(JSON.stringify(n,null,2)),await Mt.chmod(Ht,384).catch(()=>{})}var Da=900*1e3,Fa="nal-claude",Va="claude-sonnet-4-5";function Ba(t){let e=t.replace(/\/+$/,"");return e.endsWith("/v1")?`${e}/`:e.endsWith("/v1/")?e:`${e}/v1/`}async function Ka(t){let e=Yi(t,".claude","settings.json");if(!await d(e))return null;try{let n=await w(e),i=n.env||{},r=typeof i.ANTHROPIC_BASE_URL=="string"?i.ANTHROPIC_BASE_URL:null;if(!r)return null;let s=typeof n.model=="string"?n.model:"",a=typeof i.ANTHROPIC_MODEL=="string"?i.ANTHROPIC_MODEL:"",c=s.length>0?s:a,u=typeof i.ANTHROPIC_API_KEY=="string"?i.ANTHROPIC_API_KEY:null;if(u)return{provider:"anthropic",apiKey:u,baseUrl:Ba(r),model:c.length>0?c:Va};let l=typeof i.ANTHROPIC_AUTH_TOKEN=="string"?i.ANTHROPIC_AUTH_TOKEN:null;return l?{provider:"llmlite",apiKey:l,baseUrl:r,model:c.length>0?c:Fa}:null}catch{return null}}async function Wa(t,e){return await Ua({message:`Generate wiki cho workspace? (~$0.50 qua ${t} model=${e}, 2-5 ph\xFAt). Continue?`,default:!0})}function Ji(t,e){return t.split(`
43
+ `)}function ii(){let t=Mt("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=ni("gitnexus",["setup"],{stdio:["ignore","pipe","pipe"],timeout:Xa,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(`${ye(n,30)}
44
+ `):r&&process.stderr.write(`${ye(r,30)}
45
+ `),ri("setup",e.status,e.signal,n)}t.succeed("GitNexus setup OK (global skills installed)")}function ve(t){let e=Mt(`Analyze workspace ${t} (1-3 ph\xFAt)`),n=ni("gitnexus",["analyze","."],{cwd:t,stdio:["ignore","pipe","pipe"],timeout:Qa,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(`${ye(i,30)}
46
+ `):s&&process.stderr.write(`${ye(s,30)}
47
+ `),ri("analyze",n.status,n.signal,i)}let r=ei(t,".gitnexus","meta.json");if(!Ya(r))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new K("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 ${ei(t,".gitnexus")})`)}import{confirm as Pc}from"@inquirer/prompts";import $c from"boxen";import{spawnSync as oi}from"child_process";var Za=5e3,tc=/(\d+\.\d+\.\d+)/;function ec(){let e=tt()==="win32"?"where":"which",n=oi(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 nc(){let t=oi("gitnexus",["--version"],{encoding:"utf8",timeout:Za});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return tc.exec(e)?.[1]??null}var st=null;function Bt(){if(st!==null)return st;let t=ec();return t?(st={installed:!0,version:nc(),path:t},st):(st={installed:!1,version:null,path:null},st)}function be(){st=null}import{spawnSync as rc}from"child_process";var si=300*1e3,ai="gitnexus",L=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=r}};function ic(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new L("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 L("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new L("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function ci(){o.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=rc("npm",["install","-g",ai],{stdio:["inherit","inherit","pipe"],timeout:si,encoding:"utf8"});if(t.signal==="SIGTERM")throw new L("timeout",`npm install timeout sau ${si/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),ic(t.status,t.stderr||"");be();let e=Bt();if(!e.installed||!e.path)throw new L("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 Wg,select as oc}from"@inquirer/prompts";var b=class extends Error{constructor(e){super(e),this.name="UserAbortedRecoveryError"}};async function R(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 oc({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}k();import{promises as pi}from"fs";import{homedir as sc}from"os";import{join as ac}from"path";var li=384,ui={command:"gitnexus",args:["mcp"]};function cc(){return ac(sc(),".claude","mcp_servers.json")}function lc(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 uc(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await pi.copyFile(t,n),n}async function mi(){let t=cc(),e={},n=!1;if(await d(t)){n=!0;try{e=await w(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&&lc(r,ui))return o.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let i;n&&(i=await uc(t),o.dim(`Backup ${t} \u2192 ${i}`));let s={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:ui}};await A(t,s,li);try{await pi.chmod(t,li)}catch{}return o.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:i}}import{spawnSync as kc}from"child_process";import{existsSync as wc}from"fs";import{join as ki}from"path";import{confirm as yc}from"@inquirer/prompts";var pc=[/^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 di(t){return t?pc.some(e=>e.test(t)):!1}k();import{promises as Wt}from"fs";import{homedir as mc}from"os";import{dirname as dc,join as gi}from"path";var gc=gi(mc(),".gitnexus"),zt=gi(gc,"config.json");async function fc(){try{let t=await Wt.readFile(zt,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function hc(t){await Wt.mkdir(dc(zt),{recursive:!0});let e=`${zt}.tmp-${process.pid}`;await Wt.writeFile(e,t,{mode:384}),await Wt.rename(e,zt)}async function fi(t){let n={...await fc(),apiKey:t.apiKey,baseUrl:t.baseUrl,model:t.model,isReasoningModel:t.isReasoningModel??!1};await hc(JSON.stringify(n,null,2)),await Wt.chmod(zt,384).catch(()=>{})}var vc=900*1e3,bc="nal-claude",xc="claude-sonnet-4-5";function Sc(t){let e=t.replace(/\/+$/,"");return e.endsWith("/v1")?`${e}/`:e.endsWith("/v1/")?e:`${e}/v1/`}async function Ac(t){let e=ki(t,".claude","settings.json");if(!await d(e))return null;try{let n=await w(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=typeof r.ANTHROPIC_API_KEY=="string"?r.ANTHROPIC_API_KEY:null;if(l)return{provider:"anthropic",apiKey:l,baseUrl:Sc(i),model:c.length>0?c:xc};let u=typeof r.ANTHROPIC_AUTH_TOKEN=="string"?r.ANTHROPIC_AUTH_TOKEN:null;return u?{provider:"llmlite",apiKey:u,baseUrl:i,model:c.length>0?c:bc}:null}catch{return null}}async function Cc(t,e){return await yc({message:`Generate wiki cho workspace? (~$0.50 qua ${t} model=${e}, 2-5 ph\xFAt). Continue?`,default:!0})}function hi(t,e){return t.split(`
38
48
  `).slice(-e).join(`
39
- `)}async function Xi(t){let e=await Ka(t);if(!e)return o.warn("Subscription mode (OAuth, kh\xF4ng c\xF3 API key trong settings.json) \u2192 skip wiki gen."),o.dim(`\u0110\u1EC3 gen wiki sau, ch\u1EA1y manual:
40
- gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await Wa(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 i=Wi(e.model);await zi({apiKey:e.apiKey,baseUrl:e.baseUrl,model:e.model,isReasoningModel:i});let r=["wiki",".","--base-url",e.baseUrl,"--model",e.model];i&&r.push("--reasoning-model");let s=Rt(`Generating wiki via ${e.baseUrl} (${e.provider}) model=${e.model}${i?" [reasoning]":""}`),a=Ma("gitnexus",r,{cwd:t,stdio:["ignore","pipe","pipe"],timeout:Da,encoding:"utf8"});if(a.status!==0||a.signal==="SIGTERM"){let u=a.signal==="SIGTERM"?"timeout":"non-zero-exit";s.fail(`Wiki gen ${u} (exit ${a.status??"null"})`);let l=(a.stderr||"").trim(),m=(a.stdout||"").trim();return l?process.stderr.write(`${Ji(l,30)}
41
- `):m&&process.stderr.write(`${Ji(m,30)}
42
- `),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${u} (exit ${a.status??"null"})`}}let c=Yi(t,".gitnexus","wiki","index.html");return Ha(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 Ja(){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(`${za(t.join(`
49
+ `)}async function wi(t){let e=await Ac(t);if(!e)return o.warn("Subscription mode (OAuth, kh\xF4ng c\xF3 API key trong settings.json) \u2192 skip wiki gen."),o.dim(`\u0110\u1EC3 gen wiki sau, ch\u1EA1y manual:
50
+ gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await Cc(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=di(e.model);await fi({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=Mt(`Generating wiki via ${e.baseUrl} (${e.provider}) model=${e.model}${r?" [reasoning]":""}`),a=kc("gitnexus",i,{cwd:t,stdio:["ignore","pipe","pipe"],timeout:vc,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(`${hi(u,30)}
51
+ `):m&&process.stderr.write(`${hi(m,30)}
52
+ `),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${l} (exit ${a.status??"null"})`}}let c=ki(t,".gitnexus","wiki","index.html");return wc(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 Ec(){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(`${$c(t.join(`
43
53
  `),{padding:1,borderStyle:"round",borderColor:"cyan"})}
44
- `),await qa({message:"C\xE0i GitNexus global?",default:!0})}async function Ya(){for(;;)try{return Di(),!0}catch(t){let e=t instanceof Error?t.message:String(t),n=t instanceof L&&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.",i=await T({taskName:"C\xE0i GitNexus qua npm",reason:e,allowSkip:!0,hint:n});if(i==="abort")throw new b("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i GitNexus.");if(i==="skip")return!1}}async function Xa(t){for(;;)try{return le(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),i=e instanceof V&&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.",r=await T({taskName:"GitNexus analyze workspace",reason:n,allowSkip:!0,hint:i});if(r==="abort")throw new b("User abort t\u1EA1i b\u01B0\u1EDBc GitNexus analyze.");if(r==="skip")return!1}}async function me(t){let e={ok:!1,installed:!1,analyzed:!1,wikiGenerated:!1,mcpRegistered:!1};try{o.info("=== Phase 10: GitNexus Setup ===");let n=Gt();if(!n.installed){if(!await Ja())return await k("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 Ya())return await k("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(pe(),n=Gt(),!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 Xa(t.workspacePath))return await k("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 r=await Xi(t.workspacePath);e.wikiGenerated=r.ran,r.skipped&&r.reason==="fail"&&o.warn(`Wiki gen fail (workspace v\u1EABn OK): ${r.detail??"unknown"}`);try{let s=await Ki();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 k("gitnexus_setup",`result=ok,analyzed=${e.analyzed},wiki=${e.wikiGenerated},mcp=${e.mcpRegistered}`),o.success("GitNexus ready"),e}catch(n){if(n instanceof b)throw n;let i=n instanceof Error?n.message:String(n);return o.warn(`GitNexus setup th\u1EA5t b\u1EA1i: ${i}`),o.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup sau qua `avatar gitnexus install`."),await k("gitnexus_setup",`result=failed,error=${i.slice(0,200)}`),e.reason=i,e}}function en(){let t=process.cwd(),e=Yt(t);return e||(o.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
54
+ `),await Pc({message:"C\xE0i GitNexus global?",default:!0})}async function Rc(){for(;;)try{return ci(),!0}catch(t){let e=t instanceof Error?t.message:String(t),n=t instanceof L&&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 R({taskName:"C\xE0i GitNexus qua npm",reason:e,allowSkip:!0,hint:n});if(r==="abort")throw new b("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i GitNexus.");if(r==="skip")return!1}}async function Tc(t){for(;;)try{return ve(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),r=e instanceof K&&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 R({taskName:"GitNexus analyze workspace",reason:n,allowSkip:!0,hint:r});if(i==="abort")throw new b("User abort t\u1EA1i b\u01B0\u1EDBc GitNexus analyze.");if(i==="skip")return!1}}async function xe(t){let e={ok:!1,installed:!1,analyzed:!1,wikiGenerated:!1,mcpRegistered:!1};try{o.info("=== Phase 10: GitNexus Setup ===");let n=Bt();if(!n.installed){if(!await Ec())return await y("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 Rc())return await y("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(be(),n=Bt(),!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{ii()}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 Tc(t.workspacePath))return await y("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 wi(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 mi();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 y("gitnexus_setup",`result=ok,analyzed=${e.analyzed},wiki=${e.wikiGenerated},mcp=${e.mcpRegistered}`),o.success("GitNexus ready"),e}catch(n){if(n instanceof b)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 y("gitnexus_setup",`result=failed,error=${r.slice(0,200)}`),e.reason=r,e}}function hn(){let t=process.cwd(),e=wt(t);return e||(o.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
45
55
  Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
46
56
  B\u1EA1n \u0111ang \u1EDF: ${t}
47
- 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 tc(){let t=en(),e=await me({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 ec(){let t=en(),e=Qi(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 w(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 r=Qa("git",["rev-parse","HEAD"],{cwd:t,encoding:"utf8"});r.status===0&&(r.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 i=Qi(t,".gitnexus","wiki","index.html");if(await d(i)){let r=await Za.stat(i);o.info(`Wiki: ${r.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 nc(){let t=en();try{le(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 Zi(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 tc()}),e.command("status").description("Show index info + staleness warning").action(async()=>{await ec()}),e.command("analyze").description("Re-run analyze refresh index (no wiki)").action(async()=>{await nc()})}import{select as Qu}from"@inquirer/prompts";import nn from"chalk";var de=[" \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"],ge=[[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 ic(t){let n=Math.max(0,Math.min(1,t))*(ge.length-1),i=Math.floor(n),r=Math.min(ge.length-1,i+1),s=n-i,a=ge[i],c=ge[r];return[rn(a[0],c[0],s),rn(a[1],c[1],s),rn(a[2],c[2],s)]}function on(t){if(!((process.stdout.isTTY??!1)&&nn.level>0))return[...de,...t?.tagline?["",t.tagline]:[]].join(`
48
- `);let i=de.map((r,s)=>{let a=de.length===1?0:s/(de.length-1),[c,u,l]=ic(a);return nn.rgb(c,u,l).bold(r)});return t?.tagline&&(i.push(""),i.push(nn.dim(t.tagline))),i.join(`
49
- `)}function yt(t){process.stdout.write(`
50
- ${on(t)}
57
+ 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 Oc(){let t=hn(),e=await xe({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 Nc(){let t=hn(),e=yi(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 w(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=_c("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=yi(t,".gitnexus","wiki","index.html");if(await d(r)){let i=await Ic.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 jc(){let t=hn();try{ve(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 vi(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 Oc()}),e.command("status").description("Show index info + staleness warning").action(async()=>{await Nc()}),e.command("analyze").description("Re-run analyze refresh index (no wiki)").action(async()=>{await jc()})}import{select as up}from"@inquirer/prompts";import kn from"chalk";var Se=[" \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"],Ae=[[217,79,30],[200,70,80],[170,70,140],[125,88,217]];function wn(t,e,n){return Math.round(t+(e-t)*n)}function Mc(t){let n=Math.max(0,Math.min(1,t))*(Ae.length-1),r=Math.floor(n),i=Math.min(Ae.length-1,r+1),s=n-r,a=Ae[r],c=Ae[i];return[wn(a[0],c[0],s),wn(a[1],c[1],s),wn(a[2],c[2],s)]}function yn(t){if(!((process.stdout.isTTY??!1)&&kn.level>0))return[...Se,...t?.tagline?["",t.tagline]:[]].join(`
58
+ `);let r=Se.map((i,s)=>{let a=Se.length===1?0:s/(Se.length-1),[c,l,u]=Mc(a);return kn.rgb(c,l,u).bold(i)});return t?.tagline&&(r.push(""),r.push(kn.dim(t.tagline))),r.join(`
59
+ `)}function St(t){process.stdout.write(`
60
+ ${yn(t)}
51
61
 
52
- `)}import{spawnSync as ir}from"child_process";import{input as hc,select as kc}from"@inquirer/prompts";import{spawnSync as cn}from"child_process";import{promises as ac}from"fs";import{basename as cc,join as er}from"path";import{confirm as uc,select as lc}from"@inquirer/prompts";import{spawnSync as rc}from"child_process";var sn=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 tr(t){let e=`${t.org}/${t.name}`,n=["repo","create",e,`--${t.visibility}`,"--source",t.folder,"--remote","origin","--push"],i=rc("gh",n,{stdio:"inherit"});if(i.status!==0)throw i.status===1?new sn(e):new Error(`gh repo create th\u1EA5t b\u1EA1i (exit ${i.status})`);return{sshUrl:`git@github.com:${e}.git`,httpsUrl:`https://github.com/${e}.git`}}import{spawnSync as oc}from"child_process";function fe(){let t=oc("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 sc=/^[a-zA-Z0-9._-]{1,100}$/,an=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 he(t){if(!sc.test(t))throw new an(t)}function ke(t){if(t!=="private"&&t!=="public")throw new Error(`Visibility ph\u1EA3i l\xE0 "private" ho\u1EB7c "public", nh\u1EADn: "${t}"`)}function bt(t){he(t.name),ke(t.visibility);let e=t.org??fe();o.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=tr({folder:t.folder,org:e,name:t.name,visibility:t.visibility});return o.success(`\u0110\xE3 t\u1EA1o: ${n.sshUrl}`),n}function pc(){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 mc(t){let e=er(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-${pc()}`,i=er(t,n);return await ac.rename(e,i),o.success(`Backup .git \u2192 ${n}`),i}function dc(t){let e=cn("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)"),cn("git",["-C",t,"add","-A"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});let n=cn("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 nr(t){let e=cc(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await uc({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
62
+ `)}import{spawnSync as Ai}from"child_process";import{input as Jc,select as Yc}from"@inquirer/prompts";import{spawnSync as xn}from"child_process";import{promises as Uc}from"fs";import{basename as Dc,join as xi}from"path";import{confirm as Fc,select as Vc}from"@inquirer/prompts";import{spawnSync as Lc}from"child_process";var vn=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 bi(t){let e=`${t.org}/${t.name}`,n=["repo","create",e,`--${t.visibility}`,"--source",t.folder,"--remote","origin","--push"],r=Lc("gh",n,{stdio:"inherit"});if(r.status!==0)throw r.status===1?new vn(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 Gc}from"child_process";function Ce(){let t=Gc("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 Hc=/^[a-zA-Z0-9._-]{1,100}$/,bn=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 Pe(t){if(!Hc.test(t))throw new bn(t)}function $e(t){if(t!=="private"&&t!=="public")throw new Error(`Visibility ph\u1EA3i l\xE0 "private" ho\u1EB7c "public", nh\u1EADn: "${t}"`)}function At(t){Pe(t.name),$e(t.visibility);let e=t.org??Ce();o.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=bi({folder:t.folder,org:e,name:t.name,visibility:t.visibility});return o.success(`\u0110\xE3 t\u1EA1o: ${n.sshUrl}`),n}k();function Kc(){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 Bc(t){let e=xi(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-${Kc()}`,r=xi(t,n);return await Uc.rename(e,r),o.success(`Backup .git \u2192 ${n}`),r}function Wc(t){let e=xn("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)"),xn("git",["-C",t,"add","-A"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});let n=xn("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 Si(t){let e=Dc(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await Fc({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
53
63
  1. Backup .git \u2192 .git.backup-{timestamp}
54
64
  2. Git init m\u1EDBi (branch main, initial commit)
55
65
  3. T\u1EA1o GitHub repo m\u1EDBi '${n}' d\u01B0\u1EDBi account c\u1EE7a b\u1EA1n
56
- Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let i=t.visibility??(t.autoYes?"private":await lc({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),r=await mc(t.folderPath);return dc(t.folderPath),{newRemoteUrl:bt({folder:t.folderPath,name:n,visibility:i,org:t.org}).httpsUrl,backupPath:r}}import{spawnSync as gc}from"child_process";var un=5e3;function fc(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 vt(t){let e=gc("git",["ls-remote","--exit-code",t,"HEAD"],{encoding:"utf8",timeout:un,stdio:["ignore","pipe","pipe"]});if(e.error){let r=e.error;return r.code==="ENOENT"?{ok:!1,reason:"network",detail:"git binary kh\xF4ng t\xECm th\u1EA5y \u2014 c\xE0i git r\u1ED3i retry"}:r.code==="ETIMEDOUT"?{ok:!1,reason:"timeout",detail:`git ls-remote > ${un/1e3}s`}:{ok:!1,reason:"unknown",detail:r.message.slice(0,300)}}if(e.status===0)return{ok:!0};if(e.signal==="SIGTERM")return{ok:!1,reason:"timeout",detail:`git ls-remote > ${un/1e3}s`};let n=(e.stderr||"").trim();return{ok:!1,reason:fc(n),detail:n.slice(0,300)}}var Ut=class extends Error{constructor(e){super(e),this.name="RemoteAccessAbortedError"}};function wc(){let t=ir("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function yc(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=ir("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 bc(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 vc(t){let e=t.trim();return e?/^https?:\/\/[\w.@/-]+$/.test(e)||/^git@[\w.-]+:[\w./-]+\.git$/.test(e)||/^[\w.-]+\/[\w.-]+$/.test(e):!1}async function we(t){let e=t.url,n=t.initialReason,i=t.initialDetail;for(;;){let r=wc();o.warn(`Kh\xF4ng truy c\u1EADp \u0111\u01B0\u1EE3c ${e}`),o.dim(` L\xFD do: ${n}${i?` \u2014 ${i.slice(0,150)}`:""}`),o.info(bc(n,e,r)),r&&o.dim(` gh CLI hi\u1EC7n \u0111ang login: ${r}`);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 kc({message:"C\xE1ch x\u1EED l\xFD?",choices:s});if(a==="abort")throw new Ut(`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 u=await nr({folderPath:t.folderPath,visibility:t.defaultVisibility});return o.success(`Folder \u0111\xE3 reset. Backup t\u1EA1i: ${u.backupPath}`),o.success(`Remote m\u1EDBi: ${u.newRemoteUrl}`),{resolvedUrl:u.newRemoteUrl}}catch(u){o.warn(`Reset folder th\u1EA5t b\u1EA1i: ${u.message}`);continue}}a==="re-input-url"&&(e=(await hc({message:"URL git remote (https://github.com/owner/repo.git ho\u1EB7c git@github.com:owner/repo.git):",default:e,validate:l=>vc(l)||"URL kh\xF4ng \u0111\xFAng format git remote"})).trim()),a==="switch"&&yc(),o.info(`Verify remote l\u1EA1i: ${e}...`);let c=vt(e);if(c.ok)return o.success(`Remote accessible: ${e}`),{resolvedUrl:e};n=c.reason??"unknown",i=c.detail}}import{readdirSync as Gc}from"fs";import{select as Mc}from"@inquirer/prompts";import{simpleGit as cr}from"simple-git";import{existsSync as xc,statSync as Sc}from"fs";import{join as Cc}from"path";function ye(t){let e=Cc(t,".git");if(!xc(e))return!1;let n=Sc(e);return n.isDirectory()||n.isFile()}import{simpleGit as Ac}from"simple-git";var rr="chore: initial commit";async function Dt(t){let e=Ac({baseDir:t});await e.checkIsRepo().catch(()=>!1)||await e.init();try{await e.branch(["-M","main"])}catch{}await e.add(".");let i=await e.status();(await e.raw(["rev-list","-n","1","--all"]).catch(()=>"")).trim()||(i.files.length===0?await e.commit(rr,void 0,{"--allow-empty":null}):await e.commit(rr))}import{existsSync as Pc}from"fs";import{join as $c}from"path";var Rc={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 or(t){let e=[];for(let[n,i]of Object.entries(Rc))i.some(r=>Pc($c(t,r)))&&e.push(n);return e.length>0?e:["generic"]}import{readFileSync as Tc}from"fs";import{dirname as Ec,join as Ft}from"path";import{fileURLToPath as _c}from"url";var be=Ec(_c(import.meta.url)),Ic=[Ft(be,"templates","gitignore"),Ft(be,"..","templates","gitignore"),Ft(be,"..","..","src","templates","gitignore"),Ft(be,"..","src","templates","gitignore")],Vt="# === avatar ===",st="# === /avatar ===";function Oc(t){for(let e of Ic)try{return Tc(Ft(e,`${t}.txt`),"utf8")}catch{}throw new Error(`Kh\xF4ng t\xECm th\u1EA5y template gitignore cho stack "${t}"`)}function sr(t){let n=["generic",...t.filter(i=>i!=="generic")].map(i=>`# --- ${i} ---
57
- ${Oc(i).trim()}`);return[Vt,...n,st,""].join(`
58
- `)}import{existsSync as Nc,readFileSync as jc,writeFileSync as ln}from"fs";import{join as Lc}from"path";function ar(t,e){let n=Lc(t,".gitignore");if(!Nc(n)){ln(n,e,"utf8");return}let i=jc(n,"utf8"),r=i.indexOf(Vt),s=i.indexOf(st);if(r!==-1&&s!==-1&&s>r){let a=i.slice(0,r),c=i.slice(s+st.length);ln(n,`${a.trimEnd()}
66
+ Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let r=t.visibility??(t.autoYes?"private":await Vc({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),i=await Bc(t.folderPath);return Wc(t.folderPath),{newRemoteUrl:At({folder:t.folderPath,name:n,visibility:r,org:t.org}).httpsUrl,backupPath:i}}import{spawnSync as zc}from"child_process";var Sn=5e3;function qc(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 Ct(t){let e=zc("git",["ls-remote","--exit-code",t,"HEAD"],{encoding:"utf8",timeout:Sn,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 > ${Sn/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 > ${Sn/1e3}s`};let n=(e.stderr||"").trim();return{ok:!1,reason:qc(n),detail:n.slice(0,300)}}var qt=class extends Error{constructor(e){super(e),this.name="RemoteAccessAbortedError"}};function Xc(){let t=Ai("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function Qc(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=Ai("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 Zc(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 tl(t){let e=t.trim();return e?/^https?:\/\/[\w.@/-]+$/.test(e)||/^git@[\w.-]+:[\w./-]+\.git$/.test(e)||/^[\w.-]+\/[\w.-]+$/.test(e):!1}async function Ee(t){let e=t.url,n=t.initialReason,r=t.initialDetail;for(;;){let i=Xc();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(Zc(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 Yc({message:"C\xE1ch x\u1EED l\xFD?",choices:s});if(a==="abort")throw new qt(`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 Si({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 Jc({message:"URL git remote (https://github.com/owner/repo.git ho\u1EB7c git@github.com:owner/repo.git):",default:e,validate:u=>tl(u)||"URL kh\xF4ng \u0111\xFAng format git remote"})).trim()),a==="switch"&&Qc(),o.info(`Verify remote l\u1EA1i: ${e}...`);let c=Ct(e);if(c.ok)return o.success(`Remote accessible: ${e}`),{resolvedUrl:e};n=c.reason??"unknown",r=c.detail}}import{readdirSync as hl}from"fs";import{select as kl}from"@inquirer/prompts";import{simpleGit as Ri}from"simple-git";import{existsSync as el,statSync as nl}from"fs";import{join as rl}from"path";function Re(t){let e=rl(t,".git");if(!el(e))return!1;let n=nl(e);return n.isDirectory()||n.isFile()}import{simpleGit as il}from"simple-git";var Ci="chore: initial commit";async function Jt(t){let e=il({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(Ci,void 0,{"--allow-empty":null}):await e.commit(Ci))}import{existsSync as ol}from"fs";import{join as sl}from"path";var al={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 Pi(t){let e=[];for(let[n,r]of Object.entries(al))r.some(i=>ol(sl(t,i)))&&e.push(n);return e.length>0?e:["generic"]}import{readFileSync as cl}from"fs";import{dirname as ll,join as Yt}from"path";import{fileURLToPath as ul}from"url";var Te=ll(ul(import.meta.url)),pl=[Yt(Te,"templates","gitignore"),Yt(Te,"..","templates","gitignore"),Yt(Te,"..","..","src","templates","gitignore"),Yt(Te,"..","src","templates","gitignore")],Xt="# === avatar ===",at="# === /avatar ===";function ml(t){for(let e of pl)try{return cl(Yt(e,`${t}.txt`),"utf8")}catch{}throw new Error(`Kh\xF4ng t\xECm th\u1EA5y template gitignore cho stack "${t}"`)}function $i(t){let n=["generic",...t.filter(r=>r!=="generic")].map(r=>`# --- ${r} ---
67
+ ${ml(r).trim()}`);return[Xt,...n,at,""].join(`
68
+ `)}import{existsSync as dl,readFileSync as gl,writeFileSync as An}from"fs";import{join as fl}from"path";function Ei(t,e){let n=fl(t,".gitignore");if(!dl(n)){An(n,e,"utf8");return}let r=gl(n,"utf8"),i=r.indexOf(Xt),s=r.indexOf(at);if(i!==-1&&s!==-1&&s>i){let a=r.slice(0,i),c=r.slice(s+at.length);An(n,`${a.trimEnd()}
59
69
 
60
- ${e}${c.trimStart()}`,"utf8");return}ln(n,`${i.trimEnd()}
70
+ ${e}${c.trimStart()}`,"utf8");return}An(n,`${r.trimEnd()}
61
71
 
62
- ${e}`,"utf8")}var Bt=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};async function Hc(t){return ye(t)?(await cr({baseDir:t}).status()).isClean()?"clean":"dirty":Gc(t).filter(s=>s!==".git").length===0?"empty":"untracked-only"}async function Uc(t,e){return e.presetStrategy?e.presetStrategy:e.autoYes?"stash":t==="empty"||t==="clean"?"commit-all":await Mc({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 Dc(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 Fc(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 Vc(t){try{let n=(await t.revparse(["--abbrev-ref","HEAD"])).trim();return n==="HEAD"?"main":n}catch{return"main"}}async function ve(t){let e=or(t);o.info(`Tech stack: ${e.join(", ")}`),ar(t,sr(e)),o.success(".gitignore \u0111\xE3 ghi (Avatar block)")}async function Bc(t,e){let n=cr({baseDir:t});switch(e){case"skip":throw new Bt("Init aborted. Commit th\u1EE7 c\xF4ng changes hi\u1EC7n t\u1EA1i r\u1ED3i ch\u1EA1y l\u1EA1i `avatar init`.");case"stash":{let i=`avatar-init-backup-${Date.now()}`;ye(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 Dc(n,i);try{await ve(t),await Dt(t)}finally{a&&await Fc(n,i)}break}case"commit-all":{await ve(t),await Dt(t);break}case"branch":{ye(t)||(await n.init(),await n.branch(["-M","main"]));let r=await Vc(n);try{await n.checkoutLocalBranch("avatar/init")}catch{await n.checkout("avatar/init")}await ve(t),await Dt(t);try{await n.checkout(r),o.info(`Avatar init committed \u1EDF branch 'avatar/init'. Switch back v\u1EC1 '${r}'. Merge khi s\u1EB5n s\xE0ng: git merge avatar/init`)}catch{o.warn(`Kh\xF4ng switch v\u1EC1 '${r}' \u0111\u01B0\u1EE3c \u2014 \u1EDF l\u1EA1i branch 'avatar/init'. Switch tay sau.`)}break}}}async function pn(t,e={}){let n=await Hc(t);if(o.info(`Folder state: ${n}`),n==="empty"||n==="clean"){await ve(t),n==="empty"&&await Dt(t),await k("bootstrap",`state=${n},strategy=auto`);return}let i=await Uc(n,e);await Bc(t,i),await k("bootstrap",`state=${n},strategy=${i}`)}import{join as fr}from"path";import{spawnSync as dn}from"child_process";import{confirm as Kc,select as Wc}from"@inquirer/prompts";import qc from"boxen";function lr(t){return t.match(/github\.com[/:]([^/]+\/[^/]+?)(?:\.git)?$/)?.[1]??null}function ur(t){return dn("gh",["api",`repos/${t}`],{stdio:"ignore"}).status===0}function mn(){let t=dn("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function zc(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=dn("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 Jc(t){if(await Kc({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 Yc(t,e,n){let i=[`${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(`${qc(i.join(`
72
+ ${e}`,"utf8")}var Qt=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};async function wl(t){return Re(t)?(await Ri({baseDir:t}).status()).isClean()?"clean":"dirty":hl(t).filter(s=>s!==".git").length===0?"empty":"untracked-only"}async function yl(t,e){return e.presetStrategy?e.presetStrategy:e.autoYes?"stash":t==="empty"||t==="clean"?"commit-all":await kl({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 vl(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 xl(t){try{let n=(await t.revparse(["--abbrev-ref","HEAD"])).trim();return n==="HEAD"?"main":n}catch{return"main"}}async function _e(t){let e=Pi(t);o.info(`Tech stack: ${e.join(", ")}`),Ei(t,$i(e)),o.success(".gitignore \u0111\xE3 ghi (Avatar block)")}async function Sl(t,e){let n=Ri({baseDir:t});switch(e){case"skip":throw new Qt("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()}`;Re(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 vl(n,r);try{await _e(t),await Jt(t)}finally{a&&await bl(n,r)}break}case"commit-all":{await _e(t),await Jt(t);break}case"branch":{Re(t)||(await n.init(),await n.branch(["-M","main"]));let i=await xl(n);try{await n.checkoutLocalBranch("avatar/init")}catch{await n.checkout("avatar/init")}await _e(t),await Jt(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 Cn(t,e={}){let n=await wl(t);if(o.info(`Folder state: ${n}`),n==="empty"||n==="clean"){await _e(t),n==="empty"&&await Jt(t),await y("bootstrap",`state=${n},strategy=auto`);return}let r=await yl(n,e);await Sl(t,r),await y("bootstrap",`state=${n},strategy=${r}`)}import{join as Mi}from"path";import{spawnSync as $n}from"child_process";import{confirm as Al,select as Cl}from"@inquirer/prompts";import Pl from"boxen";function _i(t){return t.match(/github\.com[/:]([^/]+\/[^/]+?)(?:\.git)?$/)?.[1]??null}function Ti(t){return $n("gh",["api",`repos/${t}`],{stdio:"ignore"}).status===0}function Pn(){let t=$n("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function $l(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=$n("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 El(t){if(await Al({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 Rl(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(`${Pl(r.join(`
63
73
  `),{padding:1,borderColor:"red",borderStyle:"round"})}
64
- `)}function Xc(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(`
65
- `)}async function pr(t){if(ur(t.repoSlug))return!0;let e=mn();for(Yc(t.repoSlug,e,t.ssoEmail??null),await Jc(Xc(t.repoSlug,e,t.ssoEmail??null));;){let n=mn(),r=await Wc({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(r==="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(r==="switch-account"&&zc(),o.info("Ki\u1EC3m tra access..."),ur(t.repoSlug)){let s=mn();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 Qc=/^v?(\d+)\.(\d+)\.(\d+)(?:-([0-9A-Za-z.-]+))?$/;function Zc(t){let e=t.match(Qc);if(!e)return null;let[,n,i,r,s]=e;return{raw:t,major:Number.parseInt(n??"0",10),minor:Number.parseInt(i??"0",10),patch:Number.parseInt(r??"0",10),prerelease:s??null}}function B(t,e=!1){let n=t.map(Zc).filter(i=>i!==null).filter(i=>e||i.prerelease===null);return n.length===0?null:(n.sort((i,r)=>i.major!==r.major?i.major-r.major:i.minor!==r.minor?i.minor-r.minor:i.patch-r.patch),n[n.length-1]?.raw??null)}var mr="git@github.com:nalvn/team-ai-pack.git",tu=/^(git@github\.com:|https:\/\/github\.com\/)[A-Za-z0-9._-]+\/[A-Za-z0-9._-]+\.git$/,gn=class extends Error{url;constructor(e){super(`AVATAR_TEAM_PACK_REPO_URL kh\xF4ng h\u1EE3p l\u1EC7: ${e}
74
+ `)}function Tl(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(`
75
+ `)}async function Ii(t){if(Ti(t.repoSlug))return!0;let e=Pn();for(Rl(t.repoSlug,e,t.ssoEmail??null),await El(Tl(t.repoSlug,e,t.ssoEmail??null));;){let n=Pn(),i=await Cl({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"&&$l(),o.info("Ki\u1EC3m tra access..."),Ti(t.repoSlug)){let s=Pn();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 _l=/^v?(\d+)\.(\d+)\.(\d+)(?:-([0-9A-Za-z.-]+))?$/;function Il(t){let e=t.match(_l);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 B(t,e=!1){let n=t.map(Il).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 Oi="git@github.com:nalvn/team-ai-pack.git",Ol=/^(git@github\.com:|https:\/\/github\.com\/)[A-Za-z0-9._-]+\/[A-Za-z0-9._-]+\.git$/,En=class extends Error{url;constructor(e){super(`AVATAR_TEAM_PACK_REPO_URL kh\xF4ng h\u1EE3p l\u1EC7: ${e}
66
76
  Y\xEAu c\u1EA7u: GitHub URL c\xF3 .git suffix, vd:
67
77
  git@github.com:owner/repo.git
68
78
  https://github.com/owner/repo.git
69
- L\xFD do block: ng\u0103n file:// (filesystem exfil) + URL malicious (git hook RCE).`),this.name="InvalidTeamPackUrlError",this.url=e}};function dr(t){if(!tu.test(t))throw new gn(t)}function fn(){let t=process.env.AVATAR_TEAM_PACK_REPO_URL;return t?(dr(t),t):(dr(mr),mr)}var Gf=fn(),$=".claude/pack",at=class extends Error{constructor(e){super(e),this.name="TeamPackAccessAbortedError"}},gr="main";async function hr(t,e,n,i=!1){let r=fn(),s=lr(r);if(s&&!await pr({repoSlug:s,ssoEmail:n}))throw new at("User ch\u1ECDn t\u1EA1m ng\u01B0ng. Ch\u1EA1y l\u1EA1i 'avatar init' sau khi \u0111\u01B0\u1EE3c add v\xE0o org.");try{await gi(r,$,t)}catch(l){let m=l instanceof Error?l.message:String(l);throw(m.includes("Repository not found")||m.includes("not found"))&&o.error(`Repo team-ai-pack kh\xF4ng t\u1ED3n t\u1EA1i: ${r}
79
+ L\xFD do block: ng\u0103n file:// (filesystem exfil) + URL malicious (git hook RCE).`),this.name="InvalidTeamPackUrlError",this.url=e}};function Ni(t){if(!Ol.test(t))throw new En(t)}function Rn(){let t=process.env.AVATAR_TEAM_PACK_REPO_URL;return t?(Ni(t),t):(Ni(Oi),Oi)}var Yh=Rn(),E=".claude/pack",ct=class extends Error{constructor(e){super(e),this.name="TeamPackAccessAbortedError"}},ji="main";async function Li(t,e,n,r=!1){let i=Rn(),s=_i(i);if(s&&!await Ii({repoSlug:s,ssoEmail:n}))throw new ct("User ch\u1ECDn t\u1EA1m ng\u01B0ng. Ch\u1EA1y l\u1EA1i 'avatar init' sau khi \u0111\u01B0\u1EE3c add v\xE0o org.");try{await jr(i,E,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}
70
80
  C\xE1ch fix:
71
81
  1. T\u1EA1o repo: gh repo create <owner>/team-ai-pack --private --add-readme
72
82
  2. Ho\u1EB7c override URL: export AVATAR_TEAM_PACK_REPO_URL=<url-repo-c\u1EE7a-b\u1EA1n>
73
- 3. Ho\u1EB7c d\xF9ng flag --skip-team-pack`),l}if(e)return await It($,e,t),{pinnedTag:e};if(i)return await se($,gr,t),{pinnedTag:`${gr} (HEAD)`};let a=fr(t,$),c=await D(a),u=B(c);return u&&await It($,u,t),{pinnedTag:u}}async function xe(t){let e=fr(t,$),n=await ae(e);return n||(await wt(e)).slice(0,7)}import{basename as Mu,join as Hu,resolve as Ee}from"path";import{input as Wt,select as Uu}from"@inquirer/prompts";import{spawnSync as kr}from"child_process";import{select as eu}from"@inquirer/prompts";function nu(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 iu(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=kr("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 ru(){o.info("M\u1EDF trang GitHub Settings \u2192 SSH Keys..."),kr("open",["https://github.com/settings/keys"],{stdio:"ignore"}).status!==0&&o.info("URL: https://github.com/settings/keys")}async function ou(){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 Se(t,e,n,i=!1){for(;;)try{return{pinnedTag:(await hr(t,e,n,i)).pinnedTag,skipped:!1}}catch(r){if(r instanceof at)throw r;let s=r instanceof Error?r.message:String(r);if(nu(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 ou();if(c==="abort")throw new b("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"){iu();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."),ru();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 b("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}}}import{spawnSync as su}from"child_process";function Ce(){let t=su("gh",["auth","status"],{stdio:"ignore"});return t.error&&t.error.code==="ENOENT"?"not-installed":t.status===0?"authenticated":"not-authenticated"}import{spawnSync as au}from"child_process";function cu(t){let e=Z();return au(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function wr(){let t=Z(),e=t==="darwin"?["brew"]:t==="win32"?["winget"]:t==="linux"?["apt","dnf","pacman"]:[];for(let n of e)if(cu(n))return n;return null}import{spawnSync as uu}from"child_process";var lu={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 yr(t){let e=lu[t];o.info(`\u0110ang c\xE0i gh CLI qua ${t}...`);let n=uu(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 pu}from"child_process";function br(){if(pu("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 mu}from"child_process";function vr(){o.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp GitHub qua gh CLI (browser s\u1EBD m\u1EDF)...");let t=mu("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 ct(t){for(;Ce()==="not-installed";){o.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=wr();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 b("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{yr(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 b("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;Ce()==="not-authenticated";){o.warn("Ch\u01B0a \u0111\u0103ng nh\u1EADp GitHub.");try{vr()}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 b("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(Ce()!=="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 b("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(o.success("gh CLI s\u1EB5n s\xE0ng"),br(),t){let e=vt(t);return e.ok?(o.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await we({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 du}from"fs";import{dirname as gu,resolve as fu}from"path";import{fileURLToPath as hu}from"url";var xt=null;function G(){if(xt!==null)return xt;let t=gu(hu(import.meta.url));for(let e=0;e<5;e++){let n=fu(t,...Array(e).fill(".."),"package.json");try{let i=du(n,"utf8"),r=JSON.parse(i);if(r.name==="@nalvietnam/avatar-cli"&&typeof r.version=="string")return xt=r.version,xt}catch{}}return xt="unknown",xt}function xr(t){let i=(t.trim().replace(/\/+$/,"").split(/[/:]/).pop()??"").replace(/\.git$/,"");return i?`avatar-${i.replace(/^avatar-/,"")}-workspace`:"avatar-client-workspace"}function ku(t){return t?`
83
+ 3. Ho\u1EB7c d\xF9ng flag --skip-team-pack`),u}if(e)return await Ut(E,e,t),{pinnedTag:e};if(r)return await he(E,ji,t),{pinnedTag:`${ji} (HEAD)`};let a=Mi(t,E),c=await F(a),l=B(c);return l&&await Ut(E,l,t),{pinnedTag:l}}async function Ie(t){let e=Mi(t,E),n=await ke(e);return n||(await xt(e)).slice(0,7)}import{basename as Yu,join as Xu,resolve as Ve}from"path";import{input as ee,select as Qu}from"@inquirer/prompts";import{spawnSync as Gi}from"child_process";import{select as Nl}from"@inquirer/prompts";function jl(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 Ml(){o.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=Gi("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 Ll(){o.info("M\u1EDF trang GitHub Settings \u2192 SSH Keys..."),Gi("open",["https://github.com/settings/keys"],{stdio:"ignore"}).status!==0&&o.info("URL: https://github.com/settings/keys")}async function Gl(){return await Nl({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 Oe(t,e,n,r=!1){for(;;)try{return{pinnedTag:(await Li(t,e,n,r)).pinnedTag,skipped:!1}}catch(i){if(i instanceof ct)throw i;let s=i instanceof Error?i.message:String(i);if(jl(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 Gl();if(c==="abort")throw new b("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"){Ml();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."),Ll();continue}continue}let a=await R({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 b("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 Hl}from"child_process";function Ne(){let t=Hl("gh",["auth","status"],{stdio:"ignore"});return t.error&&t.error.code==="ENOENT"?"not-installed":t.status===0?"authenticated":"not-authenticated"}import{spawnSync as Ul}from"child_process";function Dl(t){let e=tt();return Ul(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function Hi(){let t=tt(),e=t==="darwin"?["brew"]:t==="win32"?["winget"]:t==="linux"?["apt","dnf","pacman"]:[];for(let n of e)if(Dl(n))return n;return null}import{spawnSync as Fl}from"child_process";var Vl={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 Ui(t){let e=Vl[t];o.info(`\u0110ang c\xE0i gh CLI qua ${t}...`);let n=Fl(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 Kl}from"child_process";function Di(){if(Kl("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 Bl}from"child_process";function Fi(){o.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp GitHub qua gh CLI (browser s\u1EBD m\u1EDF)...");let t=Bl("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 lt(t){for(;Ne()==="not-installed";){o.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=Hi();if(!e){if(await R({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 b("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{Ui(e)}catch(n){if(await R({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 b("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;Ne()==="not-authenticated";){o.warn("Ch\u01B0a \u0111\u0103ng nh\u1EADp GitHub.");try{Fi()}catch(e){if(await R({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 b("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(Ne()!=="authenticated"&&await R({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 b("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(o.success("gh CLI s\u1EB5n s\xE0ng"),Di(),t){let e=Ct(t);return e.ok?(o.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await Ee({url:t,initialReason:e.reason??"unknown",initialDetail:e.detail})).resolvedUrl}}return{}}function Tn(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 Wl}from"fs";import{dirname as zl,resolve as ql}from"path";import{fileURLToPath as Jl}from"url";var Pt=null;function G(){if(Pt!==null)return Pt;let t=zl(Jl(import.meta.url));for(let e=0;e<5;e++){let n=ql(t,...Array(e).fill(".."),"package.json");try{let r=Wl(n,"utf8"),i=JSON.parse(r);if(i.name==="@nalvietnam/avatar-cli"&&typeof i.version=="string")return Pt=i.version,Pt}catch{}}return Pt="unknown",Pt}function Vi(t){let r=(t.trim().replace(/\/+$/,"").split(/[/:]/).pop()??"").replace(/\.git$/,"");return r?`avatar-${r.replace(/^avatar-/,"")}-workspace`:"avatar-client-workspace"}function Yl(t){return t?`
74
84
  ### \u{1F9E0} CODEBASE INTELLIGENCE \u2014 GitNexus
75
85
 
76
86
  Workspace c\xF3 GitNexus index t\u1EA1i \`.gitnexus/\` cung c\u1EA5p architectural awareness
@@ -98,46 +108,50 @@ Khi user c\u1EA7n regenerate wiki sau refactor l\u1EDBn \u2014 ch\u1EA1y:
98
108
  \`\`\`bash
99
109
  gitnexus wiki . --api-key <key> --base-url <url>
100
110
  \`\`\`
101
- `:""}function kn(t){return{projectName:t.projectName,projectDescription:t.projectDescription,teamOwner:t.teamOwner,avatarVersion:G(),packVersion:t.packVersion,lastScan:new Date().toISOString(),mode:t.mode,gitnexusSection:ku(t.gitnexusReady??!1)}}import{join as Pe,relative as vu,resolve as Pr}from"path";import{input as Rr,select as xu}from"@inquirer/prompts";import Su from"boxen";import wu from"boxen";var Sr=[{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 Cr(){let t=Math.max(...Sr.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:"),i=Sr.map(a=>` ${p.cyan(a.cmd.padEnd(t))} ${p.dim(a.desc)}`),r=p.dim("Catalog \u0111\u1EA7y \u0111\u1EE7 46 commands: cat .claude/pack/scripts/commands_data.yaml"),s=[e,n,"",...i,"",r].join(`
102
- `);return wu(s,{padding:1,borderStyle:"round",borderColor:"cyan"})}import{readdir as yu}from"fs/promises";import{join as bu}from"path";async function Ae(t){if(!await d(t))return!0;try{return(await yu(t)).filter(i=>!i.startsWith(".")&&i!=="Thumbs.db").length===0}catch{return!1}}async function Ar(t,e,n=10){for(let i=2;i<n;i++){let r=bu(t,`${e}-${i}`);if(await Ae(r))return r}return null}function $r(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}".
103
- Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=Pr(Pe(t,e)),i=Pr(t);if(!n.startsWith(`${i}/`)&&n!==i)throw new Error(`T\xEAn workspace "${e}" resolve ra ngo\xE0i parent dir (path traversal). Block.`)}async function Kt(t,e,n){$r(t,e);let i=Pe(t,e);if(await Ae(i))return i;for(o.warn(`Workspace path "${i}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let r=await Ar(t,e);if(n&&r)return o.info(`--force: d\xF9ng ${r}`),r;let s=[];r&&s.push({name:`D\xF9ng "${r}" (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 xu({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:s});if(a==="abort")throw new b("User abort t\u1EA1i b\u01B0\u1EDBc resolve workspace path. Ch\u1EA1y l\u1EA1i v\u1EDBi --workspace-name kh\xE1c.");if(a==="use-alt"&&r)return r;let c=await Rr({message:"T\xEAn workspace m\u1EDBi:",validate:l=>{let m=l.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}});$r(t,c.trim());let u=Pe(t,c.trim());if(await Ae(u))return u;o.warn(`"${u}" c\u0169ng \u0111\xE3 c\xF3 n\u1ED9i dung. Th\u1EED t\xEAn kh\xE1c.`)}}async function $e(t){return await Rr({message:"Team owner email:",default:t})}async function Tr(t,e){if(e){o.warn("Skip commit (--no-commit). Ch\u1EA1y 'git status' + commit th\u1EE7 c\xF4ng sau.");return}let n=y(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 Cu(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 Au(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 Er(t,e,n=null,i=null){let r=[`${p.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${vu(process.cwd(),t)||t}`,` ${p.dim(`(flow: ${e})`)}`,Cu(n),Au(i),"",` ${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(`${Su(r.join(`
111
+ `:""}function _n(t){return{projectName:t.projectName,projectDescription:t.projectDescription,teamOwner:t.teamOwner,avatarVersion:G(),packVersion:t.packVersion,lastScan:new Date().toISOString(),mode:t.mode,gitnexusSection:Yl(t.gitnexusReady??!1)}}k();import{join as Me,relative as tu,resolve as zi}from"path";import{input as Ji,select as eu}from"@inquirer/prompts";import nu from"boxen";import Xl from"boxen";var Ki=[{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 Bi(){let t=Math.max(...Ki.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=Ki.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(`
112
+ `);return Xl(s,{padding:1,borderStyle:"round",borderColor:"cyan"})}k();import{readdir as Ql}from"fs/promises";import{join as Zl}from"path";async function je(t){if(!await d(t))return!0;try{return(await Ql(t)).filter(r=>!r.startsWith(".")&&r!=="Thumbs.db").length===0}catch{return!1}}async function Wi(t,e,n=10){for(let r=2;r<n;r++){let i=Zl(t,`${e}-${r}`);if(await je(i))return i}return null}function qi(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}".
113
+ Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=zi(Me(t,e)),r=zi(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 Zt(t,e,n){qi(t,e);let r=Me(t,e);if(await je(r))return r;for(o.warn(`Workspace path "${r}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let i=await Wi(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 eu({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:s});if(a==="abort")throw new b("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 Ji({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}});qi(t,c.trim());let l=Me(t,c.trim());if(await je(l))return l;o.warn(`"${l}" c\u0169ng \u0111\xE3 c\xF3 n\u1ED9i dung. Th\u1EED t\xEAn kh\xE1c.`)}}async function Le(t){return await Ji({message:"Team owner email:",default:t})}async function Yi(t,e){if(e){o.warn("Skip commit (--no-commit). Ch\u1EA1y 'git status' + commit th\u1EE7 c\xF4ng sau.");return}let n=v(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 ru(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 iu(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 Xi(t,e,n=null,r=null){let i=[`${p.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${tu(process.cwd(),t)||t}`,` ${p.dim(`(flow: ${e})`)}`,ru(n),iu(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(`${nu(i.join(`
104
114
  `),{padding:1,borderStyle:"round"})}
105
- `);let s=Pe(t,$);await d(s)&&process.stdout.write(`
106
- ${Cr()}
107
- `)}import{join as Ct}from"path";import{basename as ju}from"path";import{confirm as Lr,input as Gr,select as vn}from"@inquirer/prompts";import{spawnSync as ut}from"child_process";var St=class extends Error{reason;fullName;stderr;constructor(e,n,i,r){super(i),this.name="CreateWorkspaceRemoteError",this.reason=e,this.fullName=n,this.stderr=r}};function Pu(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 $u(t){return ut("gh",["repo","view",t,"--json","name"],{stdio:"ignore"}).status===0}function Ru(t,e){return t.toLowerCase()===e.toLowerCase()?{ok:!0}:ut("gh",["api",`orgs/${t}/members/${e}`,"--silent"],{stdio:"ignore"}).status===0?{ok:!0}:ut("gh",["api",`orgs/${t}`,"--silent"],{stdio:"ignore"}).status!==0?ut("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 _r(t){he(t.workspaceName),ke(t.visibility),await ct();let e=fe(),n=t.org??e,i=Ru(n,e);if(!i.ok)throw new Error(`Kh\xF4ng th\u1EC3 t\u1EA1o repo d\u01B0\u1EDBi '${n}/': ${i.reason}`);let r=`${n}/${t.workspaceName}`;if($u(r))throw new St("repo-exists",r,`Repo '${r}' \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: ${r} (${t.visibility})...`);let s=ut("gh",["repo","create",r,`--${t.visibility}`,"--source",t.workspacePath,"--remote","origin","--push"],{stdio:["ignore","pipe","pipe"],encoding:"utf8"});if(s.status!==0){let u=(s.stderr||"").trim(),l=(s.stdout||"").trim(),m=[u,l].filter(Boolean).join(`
108
- `),g=Pu(m);throw m&&process.stderr.write(`
115
+ `);let s=Me(t,E);await d(s)&&process.stdout.write(`
116
+ ${Bi()}
117
+ `)}import{join as It}from"path";import{basename as zu}from"path";import{confirm as Fn,input as wo,select as Dn}from"@inquirer/prompts";import{spawnSync as ut}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 ou(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 su(t){return ut("gh",["repo","view",t,"--json","name"],{stdio:"ignore"}).status===0}function au(t,e){return t.toLowerCase()===e.toLowerCase()?{ok:!0}:ut("gh",["api",`orgs/${t}/members/${e}`,"--silent"],{stdio:"ignore"}).status===0?{ok:!0}:ut("gh",["api",`orgs/${t}`,"--silent"],{stdio:"ignore"}).status!==0?ut("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 Qi(t){Pe(t.workspaceName),$e(t.visibility),await lt();let e=Ce(),n=t.org??e,r=au(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(su(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=ut("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(`
118
+ `),g=ou(m);throw m&&process.stderr.write(`
109
119
  ${m}
110
120
 
111
- `),new St(g,r,`T\u1EA1o workspace remote th\u1EA5t b\u1EA1i (exit ${s.status}): ${g}.`,m)}let a=`git@github.com:${r}.git`,c=`https://github.com/${r}.git`;return o.success(`Workspace remote: ${a}`),{sshUrl:a,httpsUrl:c}}function Ir(t){let e=`git@github.com:${t.fullName}.git`,n=`https://github.com/${t.fullName}.git`;return ut("git",["-C",t.workspacePath,"remote","add","origin",e],{encoding:"utf8",stdio:["ignore","pipe","pipe"]}).status!==0&&ut("git",["-C",t.workspacePath,"remote","set-url","origin",e],{stdio:"ignore"}),o.success(`Linked existing remote: ${e}`),{sshUrl:e,httpsUrl:n}}import{confirm as Tu}from"@inquirer/prompts";var Eu={"prompt-scoring":"prompt scoring"};async function Or(t,e={}){let n=await $i(t);if(n.length!==0)for(let i of n){let r=Eu[i]??i,s;if(e.autoYes?s=!0:s=await Tu({message:`Do you want to setup ${r} (y/n)`,default:!0}),!s){o.dim(` Skip feature '${i}' (user opt-out).`);continue}await rt(t,i,{silent:!0})}}import{promises as Re}from"fs";import{dirname as wn,join as jr,relative as yn}from"path";import{promises as _u}from"fs";function Iu(){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 Nr(t){let e=`${t}.backup-${Iu()}`;return await _u.rename(t,e),e}var bn=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function Ou(t){try{return(await Re.lstat(t)).isSymbolicLink()}catch{return!1}}async function Nu(t,e,n){let i=yn(wn(e),e)||e;if(!await d(t))return{dir:i,action:"source-missing"};if(await d(e))if(await Ou(e))await Re.unlink(e);else if(n){let s=await Nr(e),a=yn(wn(e),t);return await Re.symlink(a,e),{dir:i,action:"backed-up-and-linked",backupPath:s}}else return{dir:i,action:"skipped-conflict"};let r=yn(wn(e),t);return await Re.symlink(r,e),{dir:i,action:"created"}}async function Te(t,e,n){let i=[];for(let r of bn){let s=jr(t,r),a=jr(e,r);i.push(await Nu(s,a,n))}return i}async function Mr(t,e){let i=(await y(t).getRemotes(!0)).find(u=>u.name==="origin");if(i?.refs.push)return o.success(`Folder \u0111\xE3 c\xF3 remote origin: ${i.refs.push}`),i.refs.push;if(!(e.createRemote??await Lr({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 ct();let s=e.repoVisibility??await vn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),a=await Gr({message:"T\xEAn repo:",default:ju(t)});return bt({folder:t,name:a,visibility:s,org:e.repoOrg}).sshUrl}async function xn(t){await P(t.workspacePath),await y(t.workspacePath).init();let e=H(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await y(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 Se(t.workspacePath,t.packVersion,t.ssoEmail,t.packLatest===!0&&!t.packVersion)).pinnedTag??"HEAD",e.succeed(`Pin team-ai-pack v\xE0o ${n}`)),await Sn({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 Sn(t){let e=kn({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client"});await vi(t.workspacePath),await xi(t.workspacePath,e),await Je(t.workspacePath,e),await Si(t.workspacePath,e),await Ci(t.workspacePath),await P(Ct(t.workspacePath,"notes")),await P(Ct(t.workspacePath,"scripts")),await Nt(Ct(t.workspacePath,".git"),"post-merge"),await Nt(Ct(t.workspacePath,".git","modules","src"),"pre-push"),o.success("C\xE0i post-merge (workspace) + pre-push (src/)"),await Lu(t.workspacePath,t.autoYes),await k("init",`flow=${t.flow},workspace=${t.workspaceName}`),await Tr(t.workspacePath,t.skipCommit),await Gu(t);let n=null;t.aiSkip?o.dim("B\u1ECF qua AI setup (--ai-skip). Setup sau qua: avatar ai setup"):n=await Zt({workspacePath:t.workspacePath});let i=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)."):i=await me({workspacePath:t.workspacePath}),i?.ok){let s=kn({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client",gitnexusReady:!0});await Je(t.workspacePath,s),o.dim("Updated CLAUDE.md v\u1EDBi GitNexus section")}await Er(t.workspacePath,t.flow,n,i)}async function Lu(t,e){let n=Ct(t,$);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 i=Ct(t,".claude");o.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings merge)...");try{let r=await Te(n,i,!1),s=r.filter(u=>u.action==="created"||u.action==="updated").length,a=r.filter(u=>u.action==="source-missing").length;o.success(` \u2713 Symlinks: ${s} created${a>0?`, ${a} source-missing (pack thi\u1EBFu dir)`:""}`);let c=await ne(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 Or(t,{autoYes:e})}catch(r){o.warn(`Auto-sync pack fail: ${r instanceof Error?r.message:r}. Ch\u1EA1y \`avatar sync\` th\u1EE7 c\xF4ng \u0111\u1EC3 retry.`)}}async function Gu(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 Lr({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 vn({message:"Workspace visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh, an to\xE0n)",value:"private"},{name:"public",value:"public"}]}));for(;;)try{await _r({workspacePath:t.workspacePath,workspaceName:t.workspaceName,visibility:n,org:t.repoOrg});return}catch(i){if(i instanceof St&&i.reason==="repo-exists"){let s=i.fullName,a=await vn({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 b("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"){Ir({workspacePath:t.workspacePath,fullName:s});return}let c=await Gr({message:"T\xEAn workspace m\u1EDBi (s\u1EBD t\u1EA1o repo new):",validate:u=>u.trim().length>0?!0:"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng"});t.workspaceName=c.trim();continue}let r=await T({taskName:"T\u1EA1o workspace remote tr\xEAn GitHub",reason:i instanceof Error?i.message:String(i),allowSkip:!0,hint:"Tip: sai org? Pass --repo-org=<your-gh-user>. Ho\u1EB7c switch gh: gh auth login."});if(r==="abort")throw new b("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA1o workspace remote.");if(r==="skip"){o.warn("Workspace v\u1EABn s\u1EB5n s\xE0ng local-only. Setup remote sau khi c\u1EA7n.");return}}}async function Hr(t,e){let n=t.clientRepo??await Wt({message:"URL git c\u1EE7a repo:",validate:m=>m.length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),{resolvedRemoteUrl:i}=await ct(n),r=i??n,s=t.teamOwner??await $e(e),a=xr(r),c=t.workspaceName??await Wt({message:"T\xEAn workspace:",default:a}),u=Ee(t.workspaceParent??"."),l=await Kt(u,c,t.force);await xn({workspacePath:l,workspaceName:c,srcRemoteUrl:r,teamOwner:s,skipTeamPack:t.skipTeamPack,description:t.description??`Avatar workspace cho ${r}`,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 Ur(t,e){let n=Ee(t.folderPath??await Wt({message:"\u0110\u01B0\u1EDDng d\u1EABn folder hi\u1EC7n c\xF3:",validate:l=>l.length>0?!0:"Path b\u1EAFt bu\u1ED9c"}));await pn(n,{presetStrategy:hn(t),autoYes:t.yes});let i=await Mr(n,t);if(i){let l=vt(i);l.ok||(o.warn(`Remote ${i} kh\xF4ng accessible (${l.reason??"unknown"}).`),i=(await we({url:i,initialReason:l.reason??"unknown",initialDetail:l.detail,folderPath:n,defaultVisibility:t.repoVisibility})).resolvedUrl)}let r=t.teamOwner??await $e(e),s=t.workspaceName??`${Mu(n)}-avatar-workspace`,a=t.workspaceName??await Wt({message:"T\xEAn workspace:",default:s}),c=Ee(t.workspaceParent??"."),u=await Kt(c,a,t.force);await xn({workspacePath:u,workspaceName:a,srcRemoteUrl:i??n,teamOwner:r,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 Dr(t,e){await ct();let n=t.workspaceName??await Wt({message:"T\xEAn d\u1EF1 \xE1n:",validate:m=>m.length>0?!0:"T\xEAn b\u1EAFt bu\u1ED9c"}),i=t.repoVisibility??await Uu({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),r=t.teamOwner??await $e(e),s=Ee(t.workspaceParent??"."),a=await Kt(s,n,t.force),c=Hu(a,"src");await P(a),await P(c),await pn(c,{autoYes:!0});let u=bt({folder:c,name:n,visibility:i,org:t.repoOrg});await y(a).init();let l=H(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await y(a).subModule(["add",u.sshUrl,"src"]);let m="HEAD";t.skipTeamPack?l.succeed("Skip team-ai-pack (--skip-team-pack)"):(l.stop(),m=(await Se(a,t.packVersion,e,t.latest===!0&&!t.packVersion)).pinnedTag??"HEAD",l.succeed(`Pin team-ai-pack v\xE0o ${m}`)),await Sn({workspacePath:a,workspaceName:n,teamOwner:r,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 l.fail("Init workspace th\u1EA5t b\u1EA1i"),m}}import Ju from"boxen";import Yu from"open";var An="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",Du="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",Cn="nal.vn",Fu=["openid","email","profile"],Vu="https://oauth2.googleapis.com/device/code",Bu="https://oauth2.googleapis.com/token",Ku="https://oauth2.googleapis.com/revoke";async function Fr(){let t=new URLSearchParams({client_id:An,scope:Fu.join(" ")}),e=await fetch(Vu,{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 Vr(t){let e=new URLSearchParams({client_id:An,client_secret:Du,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(Bu,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e});if(n.ok)return await n.json();let i="";try{i=(await n.json()).error??""}catch{i=""}if(i==="authorization_pending"||i==="slow_down")return null;throw i==="access_denied"?new Error("User t\u1EEB ch\u1ED1i quy\u1EC1n truy c\u1EADp"):i==="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: ${i||n.status}`)}function Br(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 i=n.replace(/-/g,"+").replace(/_/g,"/"),r=Buffer.from(i,"base64").toString("utf8");return JSON.parse(r)}var Wu=new Set(["https://accounts.google.com","accounts.google.com"]),qu=60;function zu(t){if(!Wu.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==An)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+qu<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!==Cn)throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u @${Cn}). Nh\u1EADn: ${t.email}`);if(!t.email_verified)throw new Error("Email ch\u01B0a \u0111\u01B0\u1EE3c Google verify")}var Kr=zu;function Wr(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 qr(t){let e=new URLSearchParams({token:t});await fetch(Ku,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e}).catch(()=>{})}function zr(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.searchParams.set("hd",Cn),e.toString()}function Jr(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 Pn(e)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Pn(t){if(yt({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await In(),await k("login_reset");else{let g=await X();if(g&&!Q(g)){o.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=H("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await Fr(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let i=zr(n),r=[`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(`
112
- `);process.stdout.write(`${Ju(r,{padding:1,borderStyle:"round"})}
113
- `),Yu(i).catch(()=>{o.dim("(Kh\xF4ng m\u1EDF \u0111\u01B0\u1EE3c browser t\u1EF1 \u0111\u1ED9ng \u2014 copy URL \u1EDF tr\xEAn)")});let s=H("\u0110ang ch\u1EDD x\xE1c nh\u1EADn trong browser..."),a=n.interval*1e3,c=Date.now()+n.expires_in*1e3,u=null;for(;Date.now()<c;){await Xu(a);try{if(u=await Vr(n.device_code),u)break}catch(g){throw s.fail("X\xE1c th\u1EF1c th\u1EA5t b\u1EA1i"),g}}u||(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 l=Br(u.id_token);try{Kr(l)}catch(g){throw await qr(u.access_token),g}let m=Wr(u,l);await _n(m),await k("login",m.email),o.success(`X\xE1c th\u1EF1c th\xE0nh c\xF4ng: ${m.email}`),o.success(`Verify hosted domain: ${l.hd} \u2713`),o.success(`L\u01B0u credential v\xE0o ${Y} (chmod 600)`)}function Xu(t){return new Promise(e=>setTimeout(e,t))}function Yr(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 Zu(e)}catch(n){n instanceof Bt&&(o.dim(n.message),process.exit(0)),n instanceof at&&(o.dim(n.message),process.exit(0)),n instanceof Ut&&(o.dim(n.message),process.exit(0)),n instanceof b&&(o.dim(n.message),process.exit(0)),o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Zu(t){t.yes||yt({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 X();for(;!e||Q(e);){o.info("Ch\u01B0a \u0111\u0103ng nh\u1EADp \u2014 ch\u1EA1y login flow tr\u01B0\u1EDBc khi init...");try{await Pn({})}catch(r){o.warn(`Login fail: ${r.message}`)}if(e=await X(),e&&!Q(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 b("User abort t\u1EA1i b\u01B0\u1EDBc login. Ch\u1EA1y 'avatar login' tay r\u1ED3i 'avatar init' l\u1EA1i.")}switch(t.projectStatus??await tl()){case"existing-remote":await Hr(t,e.email);break;case"existing-folder":await Ur(t,e.email);break;case"new-project":await Dr(t,e.email);break}}async function tl(){return await Qu({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 x(t,e){return()=>{process.stdout.write(`${p.yellow("\u23F3")} ${p.bold(`avatar ${t}`)} \u2014 ch\u01B0a implement \u1EDF milestone hi\u1EC7n t\u1EA1i.
121
+ `),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 Zi(t){let e=`git@github.com:${t.fullName}.git`,n=`https://github.com/${t.fullName}.git`;return ut("git",["-C",t.workspacePath,"remote","add","origin",e],{encoding:"utf8",stdio:["ignore","pipe","pipe"]}).status!==0&&ut("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 cu}from"@inquirer/prompts";var lu={"prompt-scoring":"prompt scoring"};async function to(t,e={}){let n=await qr(t);if(n.length!==0)for(let r of n){let i=lu[r]??r,s;if(e.autoYes?s=!0:s=await cu({message:`Do you want to setup ${i} (y/n)`,default:!0}),!s){o.dim(` Skip feature '${r}' (user opt-out).`);continue}await ot(t,r,{silent:!0})}}import{promises as Ge}from"fs";import{dirname as In,join as no,relative as On}from"path";import{promises as uu}from"fs";function pu(){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 eo(t){let e=`${t}.backup-${pu()}`;return await uu.rename(t,e),e}k();var Nn=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function mu(t){try{return(await Ge.lstat(t)).isSymbolicLink()}catch{return!1}}async function du(t,e,n){let r=On(In(e),e)||e;if(!await d(t))return{dir:r,action:"source-missing"};if(await d(e))if(await mu(e))await Ge.unlink(e);else if(n){let s=await eo(e),a=On(In(e),t);return await Ge.symlink(a,e),{dir:r,action:"backed-up-and-linked",backupPath:s}}else return{dir:r,action:"skipped-conflict"};let i=On(In(e),t);return await Ge.symlink(i,e),{dir:r,action:"created"}}async function He(t,e,n){let r=[];for(let i of Nn){let s=no(t,i),a=no(e,i);r.push(await du(s,a,n))}return r}import{confirm as _t,password as Hu}from"@inquirer/prompts";import{spawnSync as gu}from"child_process";import{existsSync as fu,readFileSync as hu}from"fs";import{homedir as ku}from"os";import{join as wu}from"path";function jn(t,e){try{let n=gu(t,e,{encoding:"utf8",timeout:5e3});return{ok:n.status===0,stdout:(n.stdout??"").trim()}}catch{return{ok:!1,stdout:""}}}async function Mn(){let t=jn("direnv",["--version"]);if(t.ok&&t.stdout){let n=jn("which",["direnv"]);return{installed:!0,version:t.stdout,binaryPath:n.ok?n.stdout:null,shellHookInZshrc:ro()}}let e=jn("which",["direnv"]);return e.ok&&e.stdout?{installed:!0,version:null,binaryPath:e.stdout,shellHookInZshrc:ro()}:{installed:!1,version:null,binaryPath:null,shellHookInZshrc:!1}}function ro(){let t=wu(ku(),".zshrc");if(!fu(t))return!1;try{let e=hu(t,"utf8");return/\bdirenv\s+hook\s+(zsh|bash|sh)\b/.test(e)}catch{return!1}}import{spawnSync as io}from"child_process";import{appendFileSync as yu,existsSync as vu,readFileSync as bu}from"fs";import{homedir as xu,platform as Su}from"os";import{join as Au}from"path";var pt="https://direnv.net/docs/installation.html";function Ue(t){return io("which",[t],{encoding:"utf8",timeout:3e3}).status===0}function De(t,e){try{let n=io(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 oo(){let t=Su();if(t==="darwin"){if(Ue("brew")){let e=De("brew",["install","direnv"]);return{success:e.ok,method:"brew",message:e.ok?"Installed via Homebrew":`brew install failed: ${e.output.slice(0,500)}`,manualInstructionsUrl:pt}}return{success:!1,method:null,message:"macOS detected but Homebrew not found. Install brew first: https://brew.sh",manualInstructionsUrl:pt}}if(t==="linux"){if(Ue("apt-get")){let e=De("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:pt}}if(Ue("dnf")){let e=De("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:pt}}if(Ue("pacman")){let e=De("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:pt}}return{success:!1,method:null,message:"Linux detected but no supported package manager (apt-get/dnf/pacman) found.",manualInstructionsUrl:pt}}return{success:!1,method:null,message:`Unsupported platform: ${t}. Install direnv manually.`,manualInstructionsUrl:pt}}function so(){let t=Au(xu(),".zshrc"),e='eval "$(direnv hook zsh)"',n="# direnv shell integration (added by avatar-cli)",r="";if(vu(t))try{r=bu(t,"utf8")}catch{}if(/\bdirenv\s+hook\s+(zsh|bash|sh)\b/.test(r))return{modified:!1,zshrcPath:t};let i=`
122
+ ${n}
123
+ ${e}
124
+ `;try{return yu(t,i,"utf8"),{modified:!0,zshrcPath:t}}catch{return{modified:!1,zshrcPath:t}}}Hn();Gn();async function mt(){let t=wt(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 fo(t){return t.length<=8?"****":`${t.slice(0,4)}...${t.slice(-4)}`}async function Un(t){o.info(`Workspace: ${t}`);let e=await Mn();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 _t({message:"C\xE0i direnv ngay? (qua brew/apt/dnf/pacman t\xF9y OS)",default:!0})){o.info("\u0110ang c\xE0i direnv...");let s=await oo();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 _t({message:'Add eval "$(direnv hook zsh)" v\xE0o ~/.zshrc?',default:!0})){let s=so();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 Tt(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 _t({message:"Migrate sang .envrc (direnv) ngay?",default:!0}))){let s=await Fe(t);o.success(`\u2713 Migrated ${s.migratedKeys.length} secrets: ${s.migratedKeys.join(", ")}`),s.settingsJsonBackupPath&&o.dim(` settings.json backup: ${s.settingsJsonBackupPath}`)}Rt(t).has("ANTHROPIC_API_KEY")||await _t({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: ${Et(t)}`),o.dim(" View: avatar secrets list")}async function Uu(){let t=await mt();await Un(t)}async function ho(t,e,n={}){let r=await Hu({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 (${ae(r)}) qua Anthropic /v1/models...`);try{let s=await ce(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 _t({message:"V\u1EABn ghi key (b\u1ECF qua validation)?",default:!1}))return}}let i=await te(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 Du(t){let e=await mt();await ho(e,t,{validateAnthropic:t==="ANTHROPIC_API_KEY"})}async function Fu(t){let e=await mt(),r=Rt(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(fo(r)),o.dim(`(masked. \u0110\u1EC3 in raw: 'avatar secrets get ${t} | cat')`)):console.log(r)}async function Vu(){let t=await mt(),e=Rt(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 ${Et(t)}:`);for(let[n,r]of[...e.entries()].sort())console.log(` ${n.padEnd(30)} = ${fo(r)}`)}async function Ku(t){let e=await mt();if(!Rt(e).has(t)){o.warn(`Secret '${t}' kh\xF4ng t\u1ED3n t\u1EA1i \u2014 no-op.`);return}if(!await _t({message:`X\xF3a secret '${t}' kh\u1ECFi .envrc?`,default:!1})){o.dim("H\u1EE7y.");return}let i=await te(e,{[t]:null});o.success(`\u2713 Removed ${t} (.envrc: ${i.varsBefore} \u2192 ${i.varsAfter} vars)`)}async function Bu(){let t=await mt();o.info(`Workspace: ${t}
125
+ `);let e=await Mn();o.info(`direnv installed: ${e.installed?`\u2713 ${e.version??""}`:"\u2717"}`),o.info(`direnv shell hook (~/.zshrc): ${e.shellHookInZshrc?"\u2713":"\u2717"}`);let n=Et(t),r=Rt(t);o.info(`.envrc: ${r.size>0?`\u2713 ${r.size} secrets at ${n}`:"\u2717 (no Avatar block)"}`);let i=await Tt(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 (${ae(s)})...`);try{let a=await ce(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 Wu(){let t=await mt(),e=await Tt(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 Fe(t);o.success(`\u2713 Migrated ${n.migratedKeys.length} secrets to .envrc`),n.settingsJsonBackupPath&&o.dim(` Backup: ${n.settingsJsonBackupPath}`)}function ko(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(Uu)),e.command("set <NAME>").description("Set/update one secret in .envrc (hidden input prompt)").action(r(Du)),e.command("get <NAME>").description("Print one secret value (masked on TTY, raw when piped)").action(r(Fu)),e.command("list").description("List Avatar-managed secret names (values masked)").action(n(Vu)),e.command("rm <NAME>").description("Remove one secret from .envrc").action(r(Ku)),e.command("check").description("Health: direnv installed, .envrc exists, Anthropic key validates").action(n(Bu)),e.command("migrate").description("Migrate plaintext secrets from settings.json \u2192 .envrc").action(n(Wu))}async function yo(t,e){let r=(await v(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 Fn({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 lt();let s=e.repoVisibility??await Dn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),a=await wo({message:"T\xEAn repo:",default:zu(t)});return At({folder:t,name:a,visibility:s,org:e.repoOrg}).sshUrl}async function Vn(t){await $(t.workspacePath),await v(t.workspacePath).init();let e=U(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await v(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 Oe(t.workspacePath,t.packVersion,t.ssoEmail,t.packLatest===!0&&!t.packVersion)).pinnedTag??"HEAD",e.succeed(`Pin team-ai-pack v\xE0o ${n}`)),await Kn({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 Kn(t){let e=_n({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client"});await Fr(t.workspacePath),await Vr(t.workspacePath,e),await un(t.workspacePath,e),await Kr(t.workspacePath,e),await Br(t.workspacePath),await $(It(t.workspacePath,"notes")),await $(It(t.workspacePath,"scripts")),await Ft(It(t.workspacePath,".git"),"post-merge"),await Ft(It(t.workspacePath,".git","modules","src"),"pre-push"),o.success("C\xE0i post-merge (workspace) + pre-push (src/)"),await qu(t.workspacePath,t.autoYes),await y("init",`flow=${t.flow},workspace=${t.workspaceName}`),await Yi(t.workspacePath,t.skipCommit),await Ju(t);let n=null;t.aiSkip?o.dim("B\u1ECF qua AI setup (--ai-skip). Setup sau qua: avatar ai setup"):n=await le({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 xe({workspacePath:t.workspacePath}),r?.ok){let s=_n({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:t.packVersion,mode:"client",gitnexusReady:!0});await un(t.workspacePath,s),o.dim("Updated CLAUDE.md v\u1EDBi GitNexus section")}await Xi(t.workspacePath,t.flow,n,r)}async function qu(t,e){let n=It(t,E);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=It(t,".claude");o.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings merge)...");try{let i=await He(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 me(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}if(await to(t,{autoYes:e}),e)try{let{detectPlaintextSecretsInSettings:l,migrateSecretsFromSettingsToEnvrc:u}=await Promise.resolve().then(()=>(Hn(),go)),m=await l(t);m.keys.length>0&&(o.info(`Auto-migrating ${m.keys.length} plaintext secrets \u2192 .envrc...`),await u(t))}catch(l){o.warn(`Auto-migrate secrets fail: ${l instanceof Error?l.message:l}.`)}else try{await Fn({message:"Setup per-project secrets (direnv .envrc) ngay? (recommend cho security)",default:!0})&&await Un(t)}catch(l){o.warn(`Secrets setup fail: ${l instanceof Error?l.message:l}. Ch\u1EA1y 'avatar secrets setup' th\u1EE7 c\xF4ng sau.`)}}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 Ju(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 Fn({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 Dn({message:"Workspace visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh, an to\xE0n)",value:"private"},{name:"public",value:"public"}]}));for(;;)try{await Qi({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 Dn({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 b("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"){Zi({workspacePath:t.workspacePath,fullName:s});return}let c=await wo({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 R({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 b("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 vo(t,e){let n=t.clientRepo??await ee({message:"URL git c\u1EE7a repo:",validate:m=>m.length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),{resolvedRemoteUrl:r}=await lt(n),i=r??n,s=t.teamOwner??await Le(e),a=Vi(i),c=t.workspaceName??await ee({message:"T\xEAn workspace:",default:a}),l=Ve(t.workspaceParent??"."),u=await Zt(l,c,t.force);await Vn({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 bo(t,e){let n=Ve(t.folderPath??await ee({message:"\u0110\u01B0\u1EDDng d\u1EABn folder hi\u1EC7n c\xF3:",validate:u=>u.length>0?!0:"Path b\u1EAFt bu\u1ED9c"}));await Cn(n,{presetStrategy:Tn(t),autoYes:t.yes});let r=await yo(n,t);if(r){let u=Ct(r);u.ok||(o.warn(`Remote ${r} kh\xF4ng accessible (${u.reason??"unknown"}).`),r=(await Ee({url:r,initialReason:u.reason??"unknown",initialDetail:u.detail,folderPath:n,defaultVisibility:t.repoVisibility})).resolvedUrl)}let i=t.teamOwner??await Le(e),s=t.workspaceName??`${Yu(n)}-avatar-workspace`,a=t.workspaceName??await ee({message:"T\xEAn workspace:",default:s}),c=Ve(t.workspaceParent??"."),l=await Zt(c,a,t.force);await Vn({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 xo(t,e){await lt();let n=t.workspaceName??await ee({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 Le(e),s=Ve(t.workspaceParent??"."),a=await Zt(s,n,t.force),c=Xu(a,"src");await $(a),await $(c),await Cn(c,{autoYes:!0});let l=At({folder:c,name:n,visibility:r,org:t.repoOrg});await v(a).init();let u=U(t.skipTeamPack?"Add submodule src/...":"Add submodule src/ + team-ai-pack...");try{await v(a).subModule(["add",l.sshUrl,"src"]);let m="HEAD";t.skipTeamPack?u.succeed("Skip team-ai-pack (--skip-team-pack)"):(u.stop(),m=(await Oe(a,t.packVersion,e,t.latest===!0&&!t.packVersion)).pinnedTag??"HEAD",u.succeed(`Pin team-ai-pack v\xE0o ${m}`)),await Kn({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 ap from"boxen";import cp from"open";var Wn="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",Zu="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",Bn="nal.vn",tp=["openid","email","profile"],ep="https://oauth2.googleapis.com/device/code",np="https://oauth2.googleapis.com/token",rp="https://oauth2.googleapis.com/revoke";async function So(){let t=new URLSearchParams({client_id:Wn,scope:tp.join(" ")}),e=await fetch(ep,{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 Ao(t){let e=new URLSearchParams({client_id:Wn,client_secret:Zu,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(np,{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 Co(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 ip=new Set(["https://accounts.google.com","accounts.google.com"]),op=60;function sp(t){if(!ip.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==Wn)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+op<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!==Bn)throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u @${Bn}). Nh\u1EADn: ${t.email}`);if(!t.email_verified)throw new Error("Email ch\u01B0a \u0111\u01B0\u1EE3c Google verify")}var Po=sp;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 Eo(t){let e=new URLSearchParams({token:t});await fetch(rp,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e}).catch(()=>{})}function Ro(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.searchParams.set("hd",Bn),e.toString()}function To(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(St({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await Zn(),await y("login_reset");else{let g=await Q();if(g&&!Z(g)){o.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=U("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await So(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let r=Ro(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(`
126
+ `);process.stdout.write(`${ap(i,{padding:1,borderStyle:"round"})}
127
+ `),cp(r).catch(()=>{o.dim("(Kh\xF4ng m\u1EDF \u0111\u01B0\u1EE3c browser t\u1EF1 \u0111\u1ED9ng \u2014 copy URL \u1EDF tr\xEAn)")});let s=U("\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 lp(a);try{if(l=await Ao(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=Co(l.id_token);try{Po(u)}catch(g){throw await Eo(l.access_token),g}let m=$o(l,u);await Qn(m),await y("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 ${X} (chmod 600)`)}function lp(t){return new Promise(e=>setTimeout(e,t))}function _o(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 pp(e)}catch(n){n instanceof Qt&&(o.dim(n.message),process.exit(0)),n instanceof ct&&(o.dim(n.message),process.exit(0)),n instanceof qt&&(o.dim(n.message),process.exit(0)),n instanceof b&&(o.dim(n.message),process.exit(0)),o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function pp(t){t.yes||St({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 Q();for(;!e||Z(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 Q(),e&&!Z(e))break;if(await R({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 b("User abort t\u1EA1i b\u01B0\u1EDBc login. Ch\u1EA1y 'avatar login' tay r\u1ED3i 'avatar init' l\u1EA1i.")}switch(t.projectStatus??await mp()){case"existing-remote":await vo(t,e.email);break;case"existing-folder":await bo(t,e.email);break;case"new-project":await xo(t,e.email);break}}async function mp(){return await up({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.
114
128
  `),e&&process.stdout.write(` D\u1EF1 ki\u1EBFn: ${p.cyan(e)}
115
129
  `),process.stdout.write(` Spec \u0111\xE3 c\xF3 trong avatar-cli-implementation_4.html.
116
- `),process.exit(0)}}function Xr(t){t.command("mcp-run <tool-id>",{hidden:!0}).description("[internal] Spawn MCP v\u1EDBi secrets injected (M09)").action(x("mcp-run","Milestone 09"))}import{join as el}from"path";import Qr from"boxen";var nl=".claude/pack";function Zr(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 i=await il(process.cwd(),n.fetch===!0);n.json?process.stdout.write(`${JSON.stringify(i,null,2)}
117
- `):rl(i)}catch(i){o.error(i instanceof Error?i.message:String(i)),process.exit(1)}})}async function il(t,e){let n=el(t,nl);if(!await d(n)||!await kt(n))return{installed:!1,currentTag:null,latestTag:null,upToDate:null,fetched:!1};let i=await xe(t).catch(()=>null),r=!1;if(e)try{await y(n).fetch(["--tags","origin"]),r=!0}catch{r=!1}let s=await D(n).catch(()=>[]),a=B(s);return{installed:!0,currentTag:i,latestTag:a,upToDate:i&&a?i===a:null,fetched:r}}function rl(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(`${Qr(s.join(`
130
+ `),process.exit(0)}}function Io(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 dp}from"path";import Oo from"boxen";var gp=".claude/pack";function No(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 fp(process.cwd(),n.fetch===!0);n.json?process.stdout.write(`${JSON.stringify(r,null,2)}
131
+ `):hp(r)}catch(r){o.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function fp(t,e){let n=dp(t,gp);if(!await d(n)||!await bt(n))return{installed:!1,currentTag:null,latestTag:null,upToDate:null,fetched:!1};let r=await Ie(t).catch(()=>null),i=!1;if(e)try{await v(n).fetch(["--tags","origin"]),i=!0}catch{i=!1}let s=await F(n).catch(()=>[]),a=B(s);return{installed:!0,currentTag:r,latestTag:a,upToDate:r&&a?r===a:null,fetched:i}}function hp(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(`${Oo(s.join(`
118
132
  `),{padding:1,borderStyle:"round"})}
119
- `);return}let e=t.currentTag??p.yellow("(unknown)"),n=t.latestTag??p.dim(t.fetched?"(no tags)":"(kh\xF4ng fetch)"),i;t.upToDate===!0?i=p.green("\u2713 \u0110ang d\xF9ng tag m\u1EDBi nh\u1EA5t"):t.upToDate===!1?i=`${p.yellow("\u26A0 C\xF3 version m\u1EDBi")} ${p.dim("\u2192 avatar sync")}`:i=p.dim("(kh\xF4ng so s\xE1nh \u0111\u01B0\u1EE3c)");let r=[`${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:")} ${i}`];process.stdout.write(`${Qr(r.join(`
133
+ `);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(`${Oo(i.join(`
120
134
  `),{padding:1,borderStyle:"round"})}
121
- `)}function to(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(x("restore","Milestone 08"))}function eo(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(x("review","Milestone 08"))}function no(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(x("scan","Milestone 06"))}function io(t){let e=t.command("secrets").description("Qu\u1EA3n l\xFD secrets trong OS keychain (M09)");e.command("list").description("Li\u1EC7t k\xEA secrets \u0111\xE3 set (ch\u1EC9 t\xEAn, kh\xF4ng value)").action(x("secrets list","Milestone 09")),e.command("set <service> <name>").description("Set/update secret (prompt \u1EA9n)").action(x("secrets set","Milestone 09")),e.command("get <service> <name>").description("L\u1EA5y secret, copy clipboard, auto-x\xF3a sau 30s").action(x("secrets get","Milestone 09")),e.command("rm <service> <name>").description("X\xF3a secret kh\u1ECFi keychain").action(x("secrets rm","Milestone 09")),e.command("check").description("Verify m\u1ECDi secret required b\u1EDFi MCP \u0111\xE3 enabled").action(x("secrets check","Milestone 09"))}import{promises as cl}from"fs";import{join as _e}from"path";import ul from"boxen";import{promises as ol}from"fs";import{join as sl}from"path";var al="_backup";async function ro(t){let e=sl(t,".claude",al);return await d(e)?(await ol.readdir(e,{withFileTypes:!0})).filter(i=>i.isDirectory()).map(i=>i.name).sort().reverse():[]}function oo(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 ll(process.cwd());e.json?process.stdout.write(`${JSON.stringify(n,null,2)}
122
- `):dl(n)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function ll(t){let e=t.split("/").filter(Boolean).pop()??"unknown",n=_e(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 r=_e(n,"_pending"),[s,a,c,u,l]=await Promise.all([(async()=>await kt(_e(n,"pack"))?xe(t).catch(()=>null):null)(),(async()=>await d(r)?(await cl.readdir(r)).filter(g=>g.endsWith(".diff.md")).length:0)(),ro(t).then(m=>m.length).catch(()=>0),pl(n).catch(()=>"(read error)"),jt(t).catch(()=>({available:[],enabled:[]}))]);return{projectName:e,cliVersion:G(),packVersion:s,pendingCount:a,backupCount:c,techStackSummary:u,hasAvatar:!0,featuresEnabled:l.enabled,featuresAvailableCount:l.available.length}}async function pl(t){let e=_e(t,"project","tech-stack.md");return await d(e)?(await M(e)).split(`
123
- `).find(r=>r.trim()&&!r.startsWith("#")&&!r.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function ml(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 dl(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:")} ${ml(t)}`];process.stdout.write(`${ul(e.join(`
135
+ `)}function jo(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 Lo(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{promises as vp}from"fs";import{join as Ke}from"path";import bp from"boxen";k();k();import{promises as kp}from"fs";import{join as wp}from"path";var yp="_backup";async function Go(t){let e=wp(t,".claude",yp);return await d(e)?(await kp.readdir(e,{withFileTypes:!0})).filter(r=>r.isDirectory()).map(r=>r.name).sort().reverse():[]}function Ho(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)}
136
+ `):Cp(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=Ke(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=Ke(n,"_pending"),[s,a,c,l,u]=await Promise.all([(async()=>await bt(Ke(n,"pack"))?Ie(t).catch(()=>null):null)(),(async()=>await d(i)?(await vp.readdir(i)).filter(g=>g.endsWith(".diff.md")).length:0)(),Go(t).then(m=>m.length).catch(()=>0),Sp(n).catch(()=>"(read error)"),Vt(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 Sp(t){let e=Ke(t,"project","tech-stack.md");return await d(e)?(await H(e)).split(`
137
+ `).find(i=>i.trim()&&!i.startsWith("#")&&!i.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function Ap(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 Cp(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:")} ${Ap(t)}`];process.stdout.write(`${bp(e.join(`
124
138
  `),{padding:1,borderStyle:"round"})}
125
- `)}import{join as uo}from"path";import{join as so}from"path";async function gl(t,e,n){let i=so(t,n),r=so(e,n);if(!await d(i))return"source-missing";if(!await d(r))return"needs-creation";let{promises:s}=await import("fs");return(await s.lstat(r)).isSymbolicLink()?"already-linked":"conflict-real-dir"}async function fl(t,e,n){try{return(await y(t).raw(["log","--oneline",`${e}..${n}`])).split(`
126
- `).map(r=>r.trim()).filter(r=>r.length>0)}catch{return[]}}async function ao(t,e,n){let i=await ae(t),r=await wt(t),s=i??r.slice(0,7),a=await D(t),c=n??B(a)??"HEAD",u=await fl(t,r,c),l=[];for(let m of bn)l.push({dir:m,status:await gl(t,e,m)});return{currentVersion:s,targetVersion:c,commitsBehind:u,mountDirStatuses:l}}async function co(t){let e=await ht(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 rt(t,n)}}var Ie="main";async function hl(t){let e=process.cwd(),n=uo(e,".claude"),i=uo(e,$);await d(i)||(o.error(`team-ai-pack submodule ch\u01B0a \u0111\u01B0\u1EE3c kh\u1EDFi t\u1EA1o \u1EDF ${$}/.
127
- 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 y(i).fetch(["--tags","origin"])}catch(u){o.warn(`Kh\xF4ng fetch \u0111\u01B0\u1EE3c tags t\u1EEB origin (${u instanceof Error?u.message:u}). S\u1EBD d\xF9ng tag local hi\u1EC7n c\xF3.`)}let r=t.latest===!0&&!t.version,s=await D(i),a;if(r)a=`${Ie} (HEAD)`;else{let u=t.version??B(s);u||(o.error(`Kh\xF4ng t\xECm th\u1EA5y stable SemVer tag (vMAJOR.MINOR.PATCH) trong team-ai-pack submodule.
139
+ `)}k();import{join as Vo}from"path";k();import{join as Uo}from"path";async function Pp(t,e,n){let r=Uo(t,n),i=Uo(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 $p(t,e,n){try{return(await v(t).raw(["log","--oneline",`${e}..${n}`])).split(`
140
+ `).map(i=>i.trim()).filter(i=>i.length>0)}catch{return[]}}async function Do(t,e,n){let r=await ke(t),i=await xt(t),s=r??i.slice(0,7),a=await F(t),c=n??B(a)??"HEAD",l=await $p(t,i,c),u=[];for(let m of Nn)u.push({dir:m,status:await Pp(t,e,m)});return{currentVersion:s,targetVersion:c,commitsBehind:l,mountDirStatuses:u}}async function Fo(t){let e=await vt(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 ot(t,n)}}var Be="main";async function Ep(t){let e=process.cwd(),n=Vo(e,".claude"),r=Vo(e,E);await d(r)||(o.error(`team-ai-pack submodule ch\u01B0a \u0111\u01B0\u1EE3c kh\u1EDFi t\u1EA1o \u1EDF ${E}/.
141
+ 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 v(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 F(r),a;if(i)a=`${Be} (HEAD)`;else{let l=t.version??B(s);l||(o.error(`Kh\xF4ng t\xECm th\u1EA5y stable SemVer tag (vMAJOR.MINOR.PATCH) trong team-ai-pack submodule.
128
142
  Tags hi\u1EC7n c\xF3: ${s.length>0?s.join(", "):"(none)"}
129
- Pass --version <tag> r\xF5 r\xE0ng, ho\u1EB7c d\xF9ng --latest \u0111\u1EC3 pull HEAD branch ${Ie}.`),process.exit(1)),a=u}if(t.dryRun){let u=await ao(i,n,a);if(o.info(`Pack version hi\u1EC7n t\u1EA1i: ${u.currentVersion}`),o.info(`Target version: ${u.targetVersion}`),u.commitsBehind.length===0)o.info("\u0110\xE3 \u1EDF target version, kh\xF4ng c\xF3 commit m\u1EDBi.");else{o.info(`Commits c\u1EA7n pull (${u.commitsBehind.length}):`);for(let l of u.commitsBehind.slice(0,20))console.log(` ${l}`);u.commitsBehind.length>20&&console.log(` ... v\xE0 ${u.commitsBehind.length-20} commits kh\xE1c`)}o.info(`
130
- Mount dir statuses:`);for(let l of u.mountDirStatuses)console.log(` ${l.dir.padEnd(12)} ${l.status}`);o.info(`
131
- Dry-run done. Kh\xF4ng apply thay \u0111\u1ED5i. B\u1ECF --dry-run \u0111\u1EC3 th\u1EF1c thi.`);return}if(r){o.info(`Pulling HEAD c\u1EE7a branch ${Ie} (bleeding-edge mode)...`),await se($,Ie,e);let u=await wt(i);o.dim(` HEAD = ${u.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 It($,a,e);o.info("Creating symlink farm...");let c=await Te(i,n,t.force===!0);kl(c,t.force===!0),o.info("Merging pack settings.json template into project settings.json...");try{let u=await ne(e);switch(u.action){case"merged":o.success(` \u2713 settings.json merged (${u.changes.join("; ")}). Backup: ${u.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(u){o.warn(` ! Merge settings.json fail: ${u instanceof Error?u.message:u}. Symlinks \u0111\xE3 t\u1EA1o OK, hooks c\xF3 th\u1EC3 ch\u01B0a active. Manual merge n\u1EBFu c\u1EA7n.`)}try{await co(e)}catch(u){o.warn(` ! Re-apply features fail: ${u instanceof Error?u.message:u}. Ch\u1EA1y 'avatar feature enable <name>' th\u1EE7 c\xF4ng n\u1EBFu c\u1EA7n.`)}o.success(`Synced team-ai-pack to ${a}.`)}function kl(t,e){for(let i of t)switch(i.action){case"created":o.info(` \u2713 ${i.dir} \u2192 symlinked (new)`);break;case"updated":o.info(` \u2713 ${i.dir} \u2192 symlink refreshed`);break;case"backed-up-and-linked":o.info(` \u2713 ${i.dir} \u2192 symlinked (backup: ${i.backupPath})`);break;case"source-missing":o.warn(` - ${i.dir} \u2192 pack kh\xF4ng c\xF3 dir n\xE0y, skip`);break;case"skipped-conflict":o.warn(` ! ${i.dir} \u2192 CONFLICT: existing real dir. D\xF9ng --force \u0111\u1EC3 backup + override (s\u1EBD gi\u1EEF data \u1EDF ${i.dir}.backup-<timestamp>/).`);break}let n=t.filter(i=>i.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 lo(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(hl)}function po(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(x("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(x("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(x("tools remove","Milestone 09"))}import{relative as Al}from"path";import{confirm as Pl}from"@inquirer/prompts";import $l from"boxen";import{cp as Oe,mkdir as mo,writeFile as wl}from"fs/promises";import{homedir as yl}from"os";import{basename as bl,join as K}from"path";var vl=K(yl(),".avatar","uninstall-backups");async function go(t,e,n){let i=bl(t),r=new Date().toISOString().replace(/[:.]/g,"-"),s=K(vl,`${i}-${r}`);if(await mo(s,{recursive:!0,mode:448}),e.claudeDir&&await Oe(e.claudeDir,K(s,".claude"),{recursive:!0}),e.claudeMd&&await Oe(e.claudeMd,K(s,"CLAUDE.md")),e.postMergeHook||e.prePushHook){let c=K(s,"hooks");await mo(c,{recursive:!0}),e.postMergeHook&&await Oe(e.postMergeHook,K(c,"post-merge")),e.prePushHook&&await Oe(e.prePushHook,K(c,"pre-push"))}let a={projectName:i,projectPath:t,timestamp:r,avatarVersion:n,artifacts:{claudeDir:!!e.claudeDir,claudeMd:!!e.claudeMd,postMergeHook:!!e.postMergeHook,prePushHook:!!e.prePushHook}};return await wl(K(s,"manifest.json"),JSON.stringify(a,null,2),"utf8"),s}import{existsSync as xl}from"fs";import{join as W}from"path";function q(t){return xl(t)?t:null}function fo(t){let e=q(W(t,".claude")),n=q(W(t,"CLAUDE.md")),i=q(W(t,".git","hooks","post-merge")),r=q(W(t,".git","modules","src","hooks","pre-push")),s=q(W(t,".gitignore")),a=q(W(t,".gitmodules")),c=q(W(t,"notes")),u=q(W(t,"scripts"));return{hasAnyArtifact:!!(e||n||i||r),claudeDir:e,claudeMd:n,postMergeHook:i,prePushHook:r,gitignorePath:s,gitmodulesPath:a,notesDir:c,scriptsDir:u}}import{readFile as ho,rm as z,writeFile as ko}from"fs/promises";async function wo(t,e){if(t.claudeDir)if(e.keepSubmodule){let{readdir:n}=await import("fs/promises"),{join:i}=await import("path"),r=await n(t.claudeDir);for(let s of r)s!=="pack"&&await z(i(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 Sl(t.gitignorePath),t.gitmodulesPath&&!e.keepSubmodule&&await Cl(t.gitmodulesPath,".claude/pack");for(let n of[t.notesDir,t.scriptsDir]){if(!n)continue;let{readdir:i}=await import("fs/promises");(await i(n)).length===0&&await z(n,{recursive:!0,force:!0})}}async function Sl(t){let e=await ho(t,"utf8"),n=e.indexOf(Vt),i=e.indexOf(st);if(n===-1||i===-1)return;let r=e.slice(0,n),s=e.slice(i+st.length),a=`${r.trimEnd()}
132
- ${s.trimStart()}`.trim();a.length===0?await z(t,{force:!0}):await ko(t,`${a}
133
- `,"utf8")}async function Cl(t,e){let i=(await ho(t,"utf8")).split(`
134
- `),r=[],s=!1;for(let c of i){if(c.trim().startsWith("[submodule")&&c.includes(e)){s=!0;continue}s&&c.trim().startsWith("[submodule")&&(s=!1),s||r.push(c)}let a=r.join(`
135
- `).trim();a.length===0?await z(t,{force:!0}):await ko(t,`${a}
136
- `,"utf8")}function yo(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 Rl(e)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Rl(t){let e=process.cwd(),n=fo(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(Tl(e,n,t),t.dryRun){o.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await Pl({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){o.info("\u0110\xE3 h\u1EE7y.");return}let i=null;t.noBackup||(i=await go(e,n,G()),o.success(`Backup t\u1EA1o t\u1EA1i: ${i}`)),await wo(n,{keepSubmodule:t.keepSubmodule,keepHooks:t.keepHooks}),await k("uninstall",`project=${e},backup=${i??"skipped"}`),El(i)}function Tl(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")} ${Al(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 El(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(`${$l(e.join(`
143
+ Pass --version <tag> r\xF5 r\xE0ng, ho\u1EB7c d\xF9ng --latest \u0111\u1EC3 pull HEAD branch ${Be}.`),process.exit(1)),a=l}if(t.dryRun){let l=await Do(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(`
144
+ Mount dir statuses:`);for(let u of l.mountDirStatuses)console.log(` ${u.dir.padEnd(12)} ${u.status}`);o.info(`
145
+ 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 ${Be} (bleeding-edge mode)...`),await he(E,Be,e);let l=await xt(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 Ut(E,a,e);o.info("Creating symlink farm...");let c=await He(r,n,t.force===!0);Rp(c,t.force===!0),o.info("Merging pack settings.json template into project settings.json...");try{let l=await me(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 Fo(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 Rp(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 Ko(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(Ep)}function Bo(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 Lp}from"path";import{confirm as Gp}from"@inquirer/prompts";import Hp from"boxen";import{cp as We,mkdir as Wo,writeFile as Tp}from"fs/promises";import{homedir as _p}from"os";import{basename as Ip,join as W}from"path";var Op=W(_p(),".avatar","uninstall-backups");async function zo(t,e,n){let r=Ip(t),i=new Date().toISOString().replace(/[:.]/g,"-"),s=W(Op,`${r}-${i}`);if(await Wo(s,{recursive:!0,mode:448}),e.claudeDir&&await We(e.claudeDir,W(s,".claude"),{recursive:!0}),e.claudeMd&&await We(e.claudeMd,W(s,"CLAUDE.md")),e.postMergeHook||e.prePushHook){let c=W(s,"hooks");await Wo(c,{recursive:!0}),e.postMergeHook&&await We(e.postMergeHook,W(c,"post-merge")),e.prePushHook&&await We(e.prePushHook,W(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 Tp(W(s,"manifest.json"),JSON.stringify(a,null,2),"utf8"),s}import{existsSync as Np}from"fs";import{join as z}from"path";function q(t){return Np(t)?t:null}function qo(t){let e=q(z(t,".claude")),n=q(z(t,"CLAUDE.md")),r=q(z(t,".git","hooks","post-merge")),i=q(z(t,".git","modules","src","hooks","pre-push")),s=q(z(t,".gitignore")),a=q(z(t,".gitmodules")),c=q(z(t,"notes")),l=q(z(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 Jo,rm as J,writeFile as Yo}from"fs/promises";async function Xo(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 J(r(t.claudeDir,s),{recursive:!0,force:!0})}else await J(t.claudeDir,{recursive:!0,force:!0});t.claudeMd&&await J(t.claudeMd,{force:!0}),e.keepHooks||(t.postMergeHook&&await J(t.postMergeHook,{force:!0}),t.prePushHook&&await J(t.prePushHook,{force:!0})),t.gitignorePath&&await jp(t.gitignorePath),t.gitmodulesPath&&!e.keepSubmodule&&await Mp(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 J(n,{recursive:!0,force:!0})}}async function jp(t){let e=await Jo(t,"utf8"),n=e.indexOf(Xt),r=e.indexOf(at);if(n===-1||r===-1)return;let i=e.slice(0,n),s=e.slice(r+at.length),a=`${i.trimEnd()}
146
+ ${s.trimStart()}`.trim();a.length===0?await J(t,{force:!0}):await Yo(t,`${a}
147
+ `,"utf8")}async function Mp(t,e){let r=(await Jo(t,"utf8")).split(`
148
+ `),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(`
149
+ `).trim();a.length===0?await J(t,{force:!0}):await Yo(t,`${a}
150
+ `,"utf8")}function Qo(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 Up(e)}catch(n){o.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Up(t){let e=process.cwd(),n=qo(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(Dp(e,n,t),t.dryRun){o.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await Gp({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){o.info("\u0110\xE3 h\u1EE7y.");return}let r=null;t.noBackup||(r=await zo(e,n,G()),o.success(`Backup t\u1EA1o t\u1EA1i: ${r}`)),await Xo(n,{keepSubmodule:t.keepSubmodule,keepHooks:t.keepHooks}),await y("uninstall",`project=${e},backup=${r??"skipped"}`),Fp(r)}function Dp(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")} ${Lp(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 Fp(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(`${Hp(e.join(`
137
151
  `),{padding:1,borderStyle:"round"})}
138
- `)}var $n=G(),v=new _l;v.name("avatar").description("AI harness CLI for NAL Vietnam engineering").version($n,"-v, --version","Hi\u1EC3n th\u1ECB phi\xEAn b\u1EA3n Avatar CLI").addHelpText("beforeAll",()=>`
139
- ${on({tagline:`v${$n} \xB7 AI harness CLI for NAL Vietnam`})}
152
+ `)}var qn=G(),S=new Vp;S.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",()=>`
153
+ ${yn({tagline:`v${qn} \xB7 AI harness CLI for NAL Vietnam`})}
140
154
 
141
- `);var Il=process.argv.includes("-v")||process.argv.includes("--version");Il&&(yt({tagline:`v${$n} \xB7 AI harness CLI for NAL Vietnam`}),process.exit(0));Jr(v);Yr(v);lo(v);no(v);eo(v);oo(v);Ai(v);to(v);ei(v);po(v);io(v);Xr(v);Qn(v);Zi(v);Oi(v);Zr(v);yo(v);v.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}
155
+ `);var Kp=process.argv.includes("-v")||process.argv.includes("--version");Kp&&(St({tagline:`v${qn} \xB7 AI harness CLI for NAL Vietnam`}),process.exit(0));To(S);_o(S);Ko(S);Lo(S);Mo(S);Ho(S);Wr(S);jo(S);xr(S);Bo(S);ko(S);Io(S);yr(S);vi(S);ti(S);No(S);Qo(S);S.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}
142
156
  `),process.exit(1)});
143
157
  //# sourceMappingURL=index.js.map