@nalvietnam/avatar-cli 3.0.0-beta.4 → 3.0.0-beta.5

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,6 +1,6 @@
1
1
  // @nalvietnam/avatar-cli — built with tsup
2
- var xi=Object.defineProperty;var R=(t,e)=>()=>(t&&(e=t(t=0)),e);var Lt=(t,e)=>{for(var n in e)xi(t,n,{get:e[n],enumerable:!0})};import{spawn as Ci,spawnSync as Ai}from"child_process";import{createHash as pu}from"crypto";import{constants as Si,promises as G,createReadStream as du}from"fs";import{dirname as $n,join as fu,relative as hu}from"path";import{Readable as Pi}from"stream";async function m(t){try{return await G.access(t,Si.F_OK),!0}catch{return!1}}async function x(t){await G.mkdir(t,{recursive:!0})}async function I(t){return await G.readFile(t,"utf8")}async function y(t){return JSON.parse(await I(t))}async function ut(t,e,n){await x($n(t));let r=`${t}.tmp-${process.pid}-${Date.now()}`;await G.writeFile(r,e,"utf8"),n!==void 0&&await G.chmod(r,n),await G.rename(r,t)}async function S(t,e,n){await ut(t,`${JSON.stringify(e,null,2)}
3
- `,n)}async function pt(t){await G.rm(t,{recursive:!0,force:!0})}async function Rn(t,e,n=Ti){let r=new AbortController,o=setTimeout(()=>r.abort(),n);await x($n(e));let{createWriteStream:s}=await import("fs");try{let a=await fetch(t,{signal:r.signal});if(!a.ok)throw new Error(`Download fail (${a.status}) t\u1EEB ${t.slice(0,80)}...`);if(!a.body)throw new Error("Response kh\xF4ng c\xF3 body \u0111\u1EC3 t\u1EA3i.");let c=s(e);await new Promise((l,p)=>{let d=Pi.fromWeb(a.body);d.on("error",p),c.on("error",p),c.on("finish",()=>l()),d.pipe(c)})}catch(a){throw await G.rm(e,{force:!0}).catch(()=>{}),a instanceof Error&&a.name==="AbortError"?new Error(`T\u1EA3i file qu\xE1 ${n/1e3}s (signed URL c\xF3 th\u1EC3 \u0111\xE3 h\u1EBFt h\u1EA1n).`):a}finally{clearTimeout(o)}}function Ei(){return Gt!==null||(Gt=Ai("tar",["--version"],{stdio:"ignore"}).status===0),Gt}async function _n(t,e){if(!Ei())throw new Error("Kh\xF4ng t\xECm th\u1EA5y l\u1EC7nh `tar` tr\xEAn h\u1EC7 th\u1ED1ng. C\u1EA7n \u0111\u1EC3 gi\u1EA3i n\xE9n pack.\n Windows: c\xE0i qua WSL2, Git Bash, ho\u1EB7c MSYS2.\n macOS/Linux: tar th\u01B0\u1EDDng c\xF3 s\u1EB5n \u2014 ki\u1EC3m tra PATH.");await x(e),await new Promise((n,r)=>{let o=Ci("tar",["-xzf",t,"-C",e],{stdio:["ignore","ignore","pipe"]}),s="";o.stderr.on("data",a=>{s+=a.toString()}),o.on("error",r),o.on("close",a=>{a===0?n():r(new Error(`tar extract exit ${a}: ${s.trim()}`))})})}var Ti,Gt,k=R(()=>{"use strict";Ti=3e5;Gt=null});import{join as bu}from"path";import{simpleGit as $i}from"simple-git";function jt(t=process.cwd()){return $i({baseDir:t,binary:"git"})}var In=R(()=>{"use strict";k()});function Nn(t,e){return t.replace(Ri,(n,r)=>{let o=e[r];return o===void 0?n:String(o)})}var Ri,On=R(()=>{"use strict";Ri=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g});import{existsSync as _i}from"fs";import{dirname as Mn,join as vt}from"path";import{fileURLToPath as Ii}from"url";function Li(t){let e=t;for(;;){if(_i(vt(e,"package.json")))return e;let n=Mn(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function Gi(t){return await I(vt(Oi,`${t}.tpl`))}async function Ae(t,e){let n=await Gi(t);return Nn(n,e)}async function Gn(t){return await I(vt(Mi,`${t}.sh.tpl`))}var Ni,Ln,Oi,Mi,jn=R(()=>{"use strict";k();On();Ni=Mn(Ii(import.meta.url)),Ln=Li(Ni),Oi=vt(Ln,"src","templates"),Mi=vt(Ln,"src","hooks")});var Kn={};Lt(Kn,{AVATAR_MANAGED_PATHS:()=>Un,backupIfExists:()=>Dn,createClaudeDirTree:()=>Se,installGitHook:()=>Re,writeClaudeGitignore:()=>$e,writeProjectKnowledgeFiles:()=>Pe,writeProjectSettings:()=>Ee,writeRootClaudeMd:()=>Te});import{promises as ji}from"fs";import{join as j}from"path";async function Dn(t){if(!await m(t))return null;let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`,r=n,o=1;for(;await m(r);)if(r=`${n}-${o}`,o++,o>5)throw new Error(`Could not find free backup name for ${t}`);return await ji.rename(t,r),r}async function Hn(t,e,n){let r=await Dn(t);return await ut(t,e,n),r}async function Se(t){let e=j(t,".claude");await x(e);for(let n of Ui){let r=j(e,n);await x(r),await ut(j(r,".gitkeep"),"")}}async function Pe(t,e){return[]}async function Te(t,e){let n=await Ae("CLAUDE.md",e);return await Hn(j(t,"CLAUDE.md"),n)}async function Ee(t,e){let n=await Ae("settings.json",e);return await Hn(j(t,".claude","settings.json"),n)}async function $e(t){let e=j(t,".claude",".gitignore");await ut(e,Di)}async function Re(t,e){let n=await Gn(e),r=j(t,"hooks");await x(r);let o=j(r,e);await ut(o,n,493)}var Un,Ui,Di,xt=R(()=>{"use strict";k();jn();Un=[".claude","CLAUDE.md"];Ui=["state","_pending","_backup"];Di=`# Avatar \u2014 b\u1EA3o v\u1EC7 file nh\u1EA1y c\u1EA3m trong .claude/.
2
+ var _i=Object.defineProperty;var _=(t,e)=>()=>(t&&(e=t(t=0)),e);var X=(t,e)=>{for(var n in e)_i(t,n,{get:e[n],enumerable:!0})};var Dn={};X(Dn,{computeFileSha256:()=>Ki,copyDirRecursive:()=>Se,downloadFile:()=>Te,ensureDir:()=>v,extractTarballToDir:()=>Pe,pathExists:()=>m,readJson:()=>y,readText:()=>N,relativeFromCwd:()=>Ui,removeRecursive:()=>Z,writeJsonAtomic:()=>C,writeTextAtomic:()=>Q});import{spawn as Ii,spawnSync as Ni}from"child_process";import{createHash as Oi}from"crypto";import{constants as Mi,promises as I,createReadStream as Li}from"fs";import{dirname as Un,join as jn,relative as Gi}from"path";import{Readable as ji}from"stream";async function m(t){try{return await I.access(t,Mi.F_OK),!0}catch{return!1}}async function v(t){await I.mkdir(t,{recursive:!0})}async function N(t){return await I.readFile(t,"utf8")}async function y(t){return JSON.parse(await N(t))}async function Q(t,e,n){await v(Un(t));let r=`${t}.tmp-${process.pid}-${Date.now()}`;await I.writeFile(r,e,"utf8"),n!==void 0&&await I.chmod(r,n),await I.rename(r,t)}async function C(t,e,n){await Q(t,`${JSON.stringify(e,null,2)}
3
+ `,n)}async function Se(t,e,n=[]){await v(e);let r=await I.readdir(t,{withFileTypes:!0});for(let o of r){if(n.includes(o.name))continue;let i=jn(t,o.name),a=jn(e,o.name);if(o.isDirectory())await Se(i,a,n);else if(o.isSymbolicLink()){let c=await I.readlink(i);await I.symlink(c,a)}else await I.copyFile(i,a)}}async function Z(t){await I.rm(t,{recursive:!0,force:!0})}function Ui(t){return Gi(process.cwd(),t)}async function Te(t,e,n=Di){let r=new AbortController,o=setTimeout(()=>r.abort(),n);await v(Un(e));let{createWriteStream:i}=await import("fs");try{let a=await fetch(t,{signal:r.signal});if(!a.ok)throw new Error(`Download fail (${a.status}) t\u1EEB ${t.slice(0,80)}...`);if(!a.body)throw new Error("Response kh\xF4ng c\xF3 body \u0111\u1EC3 t\u1EA3i.");let c=i(e);await new Promise((l,p)=>{let d=ji.fromWeb(a.body);d.on("error",p),c.on("error",p),c.on("finish",()=>l()),d.pipe(c)})}catch(a){throw await I.rm(e,{force:!0}).catch(()=>{}),a instanceof Error&&a.name==="AbortError"?new Error(`T\u1EA3i file qu\xE1 ${n/1e3}s (signed URL c\xF3 th\u1EC3 \u0111\xE3 h\u1EBFt h\u1EA1n).`):a}finally{clearTimeout(o)}}function Hi(){return Ut!==null||(Ut=Ni("tar",["--version"],{stdio:"ignore"}).status===0),Ut}async function Pe(t,e){if(!Hi())throw new Error("Kh\xF4ng t\xECm th\u1EA5y l\u1EC7nh `tar` tr\xEAn h\u1EC7 th\u1ED1ng. C\u1EA7n \u0111\u1EC3 gi\u1EA3i n\xE9n pack.\n Windows: c\xE0i qua WSL2, Git Bash, ho\u1EB7c MSYS2.\n macOS/Linux: tar th\u01B0\u1EDDng c\xF3 s\u1EB5n \u2014 ki\u1EC3m tra PATH.");await v(e),await new Promise((n,r)=>{let o=Ii("tar",["-xzf",t,"-C",e],{stdio:["ignore","ignore","pipe"]}),i="";o.stderr.on("data",a=>{i+=a.toString()}),o.on("error",r),o.on("close",a=>{a===0?n():r(new Error(`tar extract exit ${a}: ${i.trim()}`))})})}async function Ki(t){return await new Promise((e,n)=>{let r=Oi("sha256"),o=Li(t);o.on("data",i=>r.update(i)),o.on("end",()=>e(r.digest("hex"))),o.on("error",n)})}var Di,Ut,k=_(()=>{"use strict";Di=3e5;Ut=null});var Hn={};X(Hn,{addSubmodule:()=>Wi,checkoutBranchHeadInSubmodule:()=>zi,checkoutTagInSubmodule:()=>qi,currentBranch:()=>Bi,currentCommitSha:()=>Xi,git:()=>$,isGitRepo:()=>Vi,listTags:()=>Ji,tagAtHead:()=>Yi,workingTreeIsDirty:()=>Qi});import{join as Ee}from"path";import{simpleGit as Fi}from"simple-git";function $(t=process.cwd()){return Fi({baseDir:t,binary:"git"})}async function Vi(t=process.cwd()){return await m(Ee(t,".git"))}async function Bi(t=process.cwd()){return(await $(t).revparse(["--abbrev-ref","HEAD"])).trim()}async function Wi(t,e,n=process.cwd()){await $(n).subModule(["add",t,e])}async function qi(t,e,n=process.cwd()){let r=Ee(n,t);await $(r).fetch(["--tags"]),await $(r).checkout(e)}async function zi(t,e,n=process.cwd()){let r=Ee(n,t);await $(r).fetch(["origin"]),await $(r).checkout(["-B",e,`origin/${e}`])}async function Ji(t=process.cwd()){return(await $(t).tags()).all}async function Yi(t=process.cwd()){try{return(await $(t).raw(["describe","--tags","--exact-match","HEAD"])).trim()||null}catch{return null}}async function Xi(t=process.cwd()){return(await $(t).revparse(["HEAD"])).trim()}async function Qi(t=process.cwd()){return!(await $(t).status()).isClean()}var $e=_(()=>{"use strict";k()});function Kn(t,e){return t.replace(Zi,(n,r)=>{let o=e[r];return o===void 0?n:String(o)})}var Zi,Fn=_(()=>{"use strict";Zi=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g});import{existsSync as ts}from"fs";import{dirname as Vn,join as Ct}from"path";import{fileURLToPath as es}from"url";function is(t){let e=t;for(;;){if(ts(Ct(e,"package.json")))return e;let n=Vn(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function ss(t){return await N(Ct(rs,`${t}.tpl`))}async function Re(t,e){let n=await ss(t);return Kn(n,e)}async function Wn(t){return await N(Ct(os,`${t}.sh.tpl`))}var ns,Bn,rs,os,qn=_(()=>{"use strict";k();Fn();ns=Vn(es(import.meta.url)),Bn=is(ns),rs=Ct(Bn,"src","templates"),os=Ct(Bn,"src","hooks")});var Xn={};X(Xn,{AVATAR_MANAGED_PATHS:()=>zn,backupIfExists:()=>Jn,createClaudeDirTree:()=>_e,installGitHook:()=>Le,writeClaudeGitignore:()=>Me,writeProjectKnowledgeFiles:()=>Ie,writeProjectSettings:()=>Oe,writeRootClaudeMd:()=>Ne});import{promises as as}from"fs";import{join as U}from"path";async function Jn(t){if(!await m(t))return null;let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`,r=n,o=1;for(;await m(r);)if(r=`${n}-${o}`,o++,o>5)throw new Error(`Could not find free backup name for ${t}`);return await as.rename(t,r),r}async function Yn(t,e,n){let r=await Jn(t);return await Q(t,e,n),r}async function _e(t){let e=U(t,".claude");await v(e);for(let n of cs){let r=U(e,n);await v(r),await Q(U(r,".gitkeep"),"")}}async function Ie(t,e){return[]}async function Ne(t,e){let n=await Re("CLAUDE.md",e);return await Yn(U(t,"CLAUDE.md"),n)}async function Oe(t,e){let n=await Re("settings.json",e);return await Yn(U(t,".claude","settings.json"),n)}async function Me(t){let e=U(t,".claude",".gitignore");await Q(e,ls)}async function Le(t,e){let n=await Wn(e),r=U(t,"hooks");await v(r);let o=U(r,e);await Q(o,n,493)}var zn,cs,ls,At=_(()=>{"use strict";k();qn();zn=[".claude","CLAUDE.md"];cs=["state","_pending","_backup"];ls=`# Avatar \u2014 b\u1EA3o v\u1EC7 file nh\u1EA1y c\u1EA3m trong .claude/.
4
4
  # Workspace root KH\xD4NG git; file n\xE0y ph\xF2ng tr\u01B0\u1EDDng h\u1EE3p .claude/ b\u1ECB track nh\u1EA7m.
5
5
  settings.json
6
6
  settings.json.backup-*
@@ -9,61 +9,61 @@ settings.json.backup-*
9
9
  _pending/
10
10
  _backup/
11
11
  state/session-*.json
12
- `});import{join as Hi}from"path";function Fn(t){return Hi(t,Ki)}async function Vn(t){let e=Fn(t);if(!await m(e))return[];try{let n=await y(e);return Array.isArray(n.repos)?n.repos:[]}catch{return[]}}async function Bn(t,e){let r=(await Vn(t)).filter(o=>o.name!==e.name);r.push(e),await S(Fn(t),{repos:r})}async function Wn(t,e){return(await Vn(t)).some(r=>r.name===e)}var Ki,qn=R(()=>{"use strict";k();Ki=".claude/repos.json"});var Jn={};Lt(Jn,{cloneCodeRepoIntoSrc:()=>Ne,folderDirtyStatus:()=>Oe,inferRepoNameFromUrl:()=>Ie,readFolderRemoteUrl:()=>Me,validateRepoName:()=>zn});import{join as _e}from"path";function zn(t){return!t||t.trim().length===0?"T\xEAn repo kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng.":/[/\\]/.test(t)||t.includes("..")?"T\xEAn repo kh\xF4ng \u0111\u01B0\u1EE3c ch\u1EE9a '/', '\\', '..'.":t==="pack"?"T\xEAn 'pack' b\u1ECB gi\u1EEF cho team-ai-pack.":null}function Ie(t){return(t.trim().replace(/\/+$/,"").split(/[/:]/).pop()??"").replace(/\.git$/,"")||"repo"}async function Ne(t){let e=zn(t.name);if(e)throw new Error(e);if(await Wn(t.workspaceRoot,t.name))throw new Error(`Repo "${t.name}" \u0111\xE3 c\xF3 trong workspace. Ch\u1ECDn t\xEAn kh\xE1c ho\u1EB7c g\u1EE1 tr\u01B0\u1EDBc.`);let n=_e(t.workspaceRoot,"src");await x(n);let r=_e(n,t.name);if(await m(r))throw new Error(`Th\u01B0 m\u1EE5c src/${t.name} \u0111\xE3 t\u1ED3n t\u1EA1i. Ch\u1ECDn t\xEAn kh\xE1c.`);await jt(n).clone(t.url,t.name);let o=_e(r,".git");return await Re(o,"pre-push").catch(()=>{}),await Bn(t.workspaceRoot,{name:t.name,path:`src/${t.name}`,url:t.url,addedAt:new Date().toISOString()}),{name:t.name,path:r,url:t.url}}async function Oe(t){try{let e=await jt(t).status();return e.isClean()?"":e.files.map(n=>`${n.path}`).join(", ")}catch{return""}}async function Me(t){try{let n=(await jt(t).getRemotes(!0)).find(r=>r.name==="origin");return n?.refs.fetch??n?.refs.push??null}catch{return null}}var Ut=R(()=>{"use strict";k();In();xt();qn()});import u from"chalk";import Fi from"ora";function mt(t){return Fi({text:t,spinner:"dots",isEnabled:process.stdout.isTTY??!1}).start()}function Ct(t){let e=Date.now(),n=mt(`${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")}`},o=setInterval(()=>{n.text=`${t} (${r()})`},1e3);return{succeed:s=>{clearInterval(o),n.succeed(`${s} (${r()})`)},fail:s=>{clearInterval(o),n.fail(`${s} (${r()})`)},stop:()=>{clearInterval(o),n.stop()}}}var i,f=R(()=>{"use strict";i={info:t=>process.stdout.write(`${u.blue("\u2139")} ${t}
12
+ `});import{join as us}from"path";function Qn(t){return us(t,ps)}async function Zn(t){let e=Qn(t);if(!await m(e))return[];try{let n=await y(e);return Array.isArray(n.repos)?n.repos:[]}catch{return[]}}async function tr(t,e){let r=(await Zn(t)).filter(o=>o.name!==e.name);r.push(e),await C(Qn(t),{repos:r})}async function er(t,e){return(await Zn(t)).some(r=>r.name===e)}var ps,nr=_(()=>{"use strict";k();ps=".claude/repos.json"});var or={};X(or,{cloneCodeRepoIntoSrc:()=>Ue,folderDirtyStatus:()=>De,inferRepoNameFromUrl:()=>je,readFolderRemoteUrl:()=>He,validateRepoName:()=>rr});import{join as Ge}from"path";function rr(t){return!t||t.trim().length===0?"T\xEAn repo kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng.":/[/\\]/.test(t)||t.includes("..")?"T\xEAn repo kh\xF4ng \u0111\u01B0\u1EE3c ch\u1EE9a '/', '\\', '..'.":t==="pack"?"T\xEAn 'pack' b\u1ECB gi\u1EEF cho team-ai-pack.":null}function je(t){return(t.trim().replace(/\/+$/,"").split(/[/:]/).pop()??"").replace(/\.git$/,"")||"repo"}async function Ue(t){let e=rr(t.name);if(e)throw new Error(e);if(await er(t.workspaceRoot,t.name))throw new Error(`Repo "${t.name}" \u0111\xE3 c\xF3 trong workspace. Ch\u1ECDn t\xEAn kh\xE1c ho\u1EB7c g\u1EE1 tr\u01B0\u1EDBc.`);let n=Ge(t.workspaceRoot,"src");await v(n);let r=Ge(n,t.name);if(await m(r))throw new Error(`Th\u01B0 m\u1EE5c src/${t.name} \u0111\xE3 t\u1ED3n t\u1EA1i. Ch\u1ECDn t\xEAn kh\xE1c.`);await $(n).clone(t.url,t.name);let o=Ge(r,".git");return await Le(o,"pre-push").catch(()=>{}),await tr(t.workspaceRoot,{name:t.name,path:`src/${t.name}`,url:t.url,addedAt:new Date().toISOString()}),{name:t.name,path:r,url:t.url}}async function De(t){try{let e=await $(t).status();return e.isClean()?"":e.files.map(n=>`${n.path}`).join(", ")}catch{return""}}async function He(t){try{let n=(await $(t).getRemotes(!0)).find(r=>r.name==="origin");return n?.refs.fetch??n?.refs.push??null}catch{return null}}var Dt=_(()=>{"use strict";k();$e();At();nr()});import u from"chalk";import ms from"ora";function gt(t){return ms({text:t,spinner:"dots",isEnabled:process.stdout.isTTY??!1}).start()}function St(t){let e=Date.now(),n=gt(`${t} (0:00)`),r=()=>{let i=Math.floor((Date.now()-e)/1e3),a=Math.floor(i/60),c=i%60;return`${a}:${String(c).padStart(2,"0")}`},o=setInterval(()=>{n.text=`${t} (${r()})`},1e3);return{succeed:i=>{clearInterval(o),n.succeed(`${i} (${r()})`)},fail:i=>{clearInterval(o),n.fail(`${i} (${r()})`)},stop:()=>{clearInterval(o),n.stop()}}}var s,f=_(()=>{"use strict";s={info:t=>process.stdout.write(`${u.blue("\u2139")} ${t}
13
13
  `),success:t=>process.stdout.write(`${u.green("\u2713")} ${t}
14
14
  `),warn:t=>process.stdout.write(`${u.yellow("\u26A0")} ${t}
15
15
  `),error:t=>process.stderr.write(`${u.red("\u2717")} ${t}
16
16
  `),dim:t=>process.stdout.write(`${u.dim(t)}
17
17
  `),plain:t=>process.stdout.write(`${t}
18
- `)}});import{spawnSync as ss}from"child_process";function Kt(){let t=ss("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 Ue=R(()=>{"use strict"});var xr={};Lt(xr,{createEmptyGithubRepoAndGetUrl:()=>zs});import{spawnSync as qs}from"child_process";async function zs(t,e,n){let o=`${n??Kt()}/${t}`;i.info(`T\u1EA1o GitHub repo r\u1ED7ng ${o} (${e})...`);let s=qs("gh",["repo","create",o,`--${e}`,"--add-readme"],{stdio:"inherit"});if(s.status!==0)throw new Error(`gh repo create th\u1EA5t b\u1EA1i (exit ${s.status}). Repo c\xF3 th\u1EC3 \u0111\xE3 t\u1ED3n t\u1EA1i.`);return i.success(`\u0110\xE3 t\u1EA1o: git@github.com:${o}.git`),`git@github.com:${o}.git`}var Cr=R(()=>{"use strict";Ue();f()});var Qe={};Lt(Qe,{HOSTED_DOMAIN:()=>Wt,SCOPES:()=>Er,buildUserConfig:()=>Je,buildVerificationUrl:()=>Xe,decodeIdToken:()=>qe,pollForToken:()=>We,refreshAccessToken:()=>sa,requestDeviceCode:()=>Be,revokeToken:()=>Ye,verifyHostedDomain:()=>ze,verifyIdTokenClaims:()=>Rr});async function Be(){let t=new URLSearchParams({client_id:qt,scope:Er.join(" ")}),e=await fetch(na,{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 We(t){let e=new URLSearchParams({client_id:qt,client_secret:Tr,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch($r,{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 qe(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,"/"),o=Buffer.from(r,"base64").toString("utf8");return JSON.parse(o)}function Rr(t){if(!oa.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==qt)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+ia<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!==Wt)throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u @${Wt}). Nh\u1EADn: ${t.email}`);if(!t.email_verified)throw new Error("Email ch\u01B0a \u0111\u01B0\u1EE3c Google verify")}function Je(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 sa(t){let e=new URLSearchParams({client_id:qt,client_secret:Tr,refresh_token:t,grant_type:"refresh_token"}),n=await fetch($r,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e});if(!n.ok){let r=await n.text();throw new Error(`Refresh token failed (${n.status}): ${r}`)}return await n.json()}async function Ye(t){let e=new URLSearchParams({token:t});await fetch(ra,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e}).catch(()=>{})}function Xe(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.searchParams.set("hd",Wt),e.toString()}var qt,Tr,Wt,Er,na,$r,ra,oa,ia,ze,zt=R(()=>{"use strict";qt="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",Tr="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",Wt="nal.vn",Er=["openid","email","profile"],na="https://oauth2.googleapis.com/device/code",$r="https://oauth2.googleapis.com/token",ra="https://oauth2.googleapis.com/revoke";oa=new Set(["https://accounts.google.com","accounts.google.com"]),ia=60;ze=Rr});import{Command as su}from"commander";Ut();import{resolve as Js}from"path";import{confirm as Ve,input as Bt,select as Ar}from"@inquirer/prompts";Ut();k();f();import{spawnSync as Wi}from"child_process";import{join as qi}from"path";import{confirm as zi,input as Ge,select as Ji}from"@inquirer/prompts";import{spawnSync as Vi}from"child_process";var Le=5e3;function Bi(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 dt(t){let e=Vi("git",["ls-remote","--exit-code",t,"HEAD"],{encoding:"utf8",timeout:Le,stdio:["ignore","pipe","pipe"]});if(e.error){let o=e.error;return o.code==="ENOENT"?{ok:!1,reason:"network",detail:"git binary kh\xF4ng t\xECm th\u1EA5y \u2014 c\xE0i git r\u1ED3i retry"}:o.code==="ETIMEDOUT"?{ok:!1,reason:"timeout",detail:`git ls-remote > ${Le/1e3}s`}:{ok:!1,reason:"unknown",detail:o.message.slice(0,300)}}if(e.status===0)return{ok:!0};if(e.signal==="SIGTERM")return{ok:!1,reason:"timeout",detail:`git ls-remote > ${Le/1e3}s`};let n=(e.stderr||"").trim();return{ok:!1,reason:Bi(n),detail:n.slice(0,300)}}function Yi(t,e,n){switch(i.warn(`\u26D4 Kh\xF4ng clone \u0111\u01B0\u1EE3c ${t}`),e){case"not-found":i.dim(" \u2192 Repo kh\xF4ng t\u1ED3n t\u1EA1i (sai URL, ho\u1EB7c repo ch\u01B0a \u0111\u01B0\u1EE3c t\u1EA1o tr\xEAn GitHub).");break;case"no-access":i.dim(` \u2192 Kh\xF4ng c\xF3 quy\u1EC1n truy c\u1EADp (repo private + sai account), HO\u1EB6C repo kh\xF4ng t\u1ED3n t\u1EA1i.
19
- Ki\u1EC3m tra: \u0111ang login \u0111\xFAng GitHub account ch\u01B0a? (gh auth status)`);break;case"network":i.dim(" \u2192 L\u1ED7i m\u1EA1ng (m\u1EA5t k\u1EBFt n\u1ED1i / DNS / timeout).");break;default:i.dim(` \u2192 L\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh${n?`: ${n}`:""}.`)}}function Yn(){i.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=Wi("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&i.warn(`gh auth login exit ${t.status}. C\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}async function Xn(t){let e=[{name:"Th\u1EED l\u1EA1i (retry)",value:"retry"},{name:"Nh\u1EADp URL repo kh\xE1c",value:"new-url"}];return t==="no-access"&&e.push({name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch-account"}),e.push({name:"B\u1ECF qua repo n\xE0y",value:"skip"}),await Ji({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}async function Qn(t){let e=t.url,n=t.name;for(;;){let r=dt(e);if(!r.ok){Yi(e,r.reason??"unknown",r.detail);let o=await Xn(r.reason??"unknown");if(o==="skip")return i.dim(`B\u1ECF qua repo ${n}.`),{cloned:null,skipped:!0};if(o==="switch-account"){Yn();continue}if(o==="new-url"){if(e=await Ge({message:"URL repo kh\xE1c:",validate:a=>a.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),await zi({message:`\u0110\u1ED5i t\xEAn th\u01B0 m\u1EE5c trong src/ theo URL m\u1EDBi? (hi\u1EC7n: ${n})`,default:!1})){let{inferRepoNameFromUrl:a}=await Promise.resolve().then(()=>(Ut(),Jn));n=await Ge({message:"T\xEAn th\u01B0 m\u1EE5c:",default:a(e)})}continue}continue}try{return{cloned:await Ne({workspaceRoot:t.workspaceRoot,url:e,name:n}),skipped:!1}}catch(o){let s=o instanceof Error?o.message:String(o);i.warn(`Clone fail: ${s}`),await pt(qi(t.workspaceRoot,"src",n)).catch(()=>{});let a=await Xn("unknown");if(a==="skip")return{cloned:null,skipped:!0};if(a==="switch-account"){Yn();continue}if(a==="new-url"){e=await Ge({message:"URL repo kh\xE1c:",validate:c=>c.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"});continue}}}}import{spawnSync as Zn}from"child_process";import{platform as Xi}from"os";function Y(){let t=Xi();return t==="darwin"||t==="linux"||t==="win32"?t:"unsupported"}var Qi=5e3,Zi=/(\d+\.\d+\.\d+)/;function ts(){let e=Y()==="win32"?"where":"which",n=Zn(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 es(){let t=Zn("gitnexus",["--version"],{encoding:"utf8",timeout:Qi});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return Zi.exec(e)?.[1]??null}var X=null;function Q(){if(X!==null)return X;let t=ts();return t?(X={installed:!0,version:es(),path:t},X):(X={installed:!1,version:null,path:null},X)}function Dt(){X=null}import{spawnSync as ns}from"child_process";function Ht(){let t=ns("gh",["auth","status"],{stdio:"ignore"});return t.error&&t.error.code==="ENOENT"?"not-installed":t.status===0?"authenticated":"not-authenticated"}import{spawnSync as rs}from"child_process";function os(t){let e=Y();return rs(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function tr(){let t=Y(),e=t==="darwin"?["brew"]:t==="win32"?["winget"]:t==="linux"?["apt","dnf","pacman"]:[];for(let n of e)if(os(n))return n;return null}import{spawnSync as ar}from"child_process";import{input as fs,select as hs}from"@inquirer/prompts";import{spawnSync as He}from"child_process";import{promises as cs}from"fs";import{basename as ls,join as ir}from"path";import{confirm as us,select as ps}from"@inquirer/prompts";import{spawnSync as is}from"child_process";var je=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 er(t){let e=`${t.org}/${t.name}`,n=["repo","create",e,`--${t.visibility}`,"--source",t.folder,"--remote","origin","--push"],r=is("gh",n,{stdio:"inherit"});if(r.status!==0)throw r.status===1?new je(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`}}Ue();f();var as=/^[a-zA-Z0-9._-]{1,100}$/,De=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 nr(t){if(!as.test(t))throw new De(t)}function rr(t){if(t!=="private"&&t!=="public")throw new Error(`Visibility ph\u1EA3i l\xE0 "private" ho\u1EB7c "public", nh\u1EADn: "${t}"`)}function or(t){nr(t.name),rr(t.visibility);let e=t.org??Kt();i.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=er({folder:t.folder,org:e,name:t.name,visibility:t.visibility});return i.success(`\u0110\xE3 t\u1EA1o: ${n.sshUrl}`),n}k();f();function ms(){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 ds(t){let e=ir(t,".git");if(!await m(e))throw new Error(`.git kh\xF4ng t\u1ED3n t\u1EA1i \u1EDF ${t} \u2014 kh\xF4ng c\u1EA7n reset.`);let n=`.git.backup-${ms()}`,r=ir(t,n);return await cs.rename(e,r),i.success(`Backup .git \u2192 ${n}`),r}function gs(t){let e=He("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}`);i.success("Git init m\u1EDBi (branch main)"),He("git",["-C",t,"add","-A"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});let n=He("git",["-C",t,"commit","-m","chore: initial commit from existing folder"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});n.status!==0?i.warn(`git commit th\u1EA5t b\u1EA1i (folder c\xF3 th\u1EC3 r\u1ED7ng): ${(n.stderr||"").slice(0,200)}`):i.success("Initial commit")}async function sr(t){let e=ls(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await us({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
18
+ `)}});import{spawnSync as Es}from"child_process";function ur(t){let e=`${t.org}/${t.name}`,n=["repo","create",e,`--${t.visibility}`,"--source",t.folder,"--remote","origin","--push"],r=Es("gh",n,{stdio:"inherit"});if(r.status!==0)throw r.status===1?new Ve(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`}}var Ve,pr=_(()=>{"use strict";Ve=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"}}});import{spawnSync as $s}from"child_process";function Ft(){let t=$s("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 Be=_(()=>{"use strict"});function mr(t){if(!Rs.test(t))throw new We(t)}function dr(t){if(t!=="private"&&t!=="public")throw new Error(`Visibility ph\u1EA3i l\xE0 "private" ho\u1EB7c "public", nh\u1EADn: "${t}"`)}var Rs,We,gr=_(()=>{"use strict";Rs=/^[a-zA-Z0-9._-]{1,100}$/,We=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"}}});var fr={};X(fr,{createGithubRemoteFromFolder:()=>qe});function qe(t){mr(t.name),dr(t.visibility);let e=t.org??Ft();s.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=ur({folder:t.folder,org:e,name:t.name,visibility:t.visibility});return s.success(`\u0110\xE3 t\u1EA1o: ${n.sshUrl}`),n}var ze=_(()=>{"use strict";pr();Be();f();gr()});var Ir={};X(Ir,{createEmptyGithubRepoAndGetUrl:()=>wa});import{spawnSync as ha}from"child_process";async function wa(t,e,n){let o=`${n??Ft()}/${t}`;s.info(`T\u1EA1o GitHub repo r\u1ED7ng ${o} (${e})...`);let i=ha("gh",["repo","create",o,`--${e}`,"--add-readme"],{stdio:"inherit"});if(i.status!==0)throw new Error(`gh repo create th\u1EA5t b\u1EA1i (exit ${i.status}). Repo c\xF3 th\u1EC3 \u0111\xE3 t\u1ED3n t\u1EA1i.`);return s.success(`\u0110\xE3 t\u1EA1o: git@github.com:${o}.git`),`git@github.com:${o}.git`}var Nr=_(()=>{"use strict";Be();f()});var cn={};X(cn,{HOSTED_DOMAIN:()=>qt,SCOPES:()=>Gr,buildUserConfig:()=>on,buildVerificationUrl:()=>an,decodeIdToken:()=>nn,pollForToken:()=>en,refreshAccessToken:()=>Ra,requestDeviceCode:()=>tn,revokeToken:()=>sn,verifyHostedDomain:()=>rn,verifyIdTokenClaims:()=>Ur});async function tn(){let t=new URLSearchParams({client_id:zt,scope:Gr.join(" ")}),e=await fetch(Ta,{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 en(t){let e=new URLSearchParams({client_id:zt,client_secret:Lr,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(jr,{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 nn(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,"/"),o=Buffer.from(r,"base64").toString("utf8");return JSON.parse(o)}function Ur(t){if(!Ea.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==zt)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+$a<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!==qt)throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u @${qt}). Nh\u1EADn: ${t.email}`);if(!t.email_verified)throw new Error("Email ch\u01B0a \u0111\u01B0\u1EE3c Google verify")}function on(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 Ra(t){let e=new URLSearchParams({client_id:zt,client_secret:Lr,refresh_token:t,grant_type:"refresh_token"}),n=await fetch(jr,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e});if(!n.ok){let r=await n.text();throw new Error(`Refresh token failed (${n.status}): ${r}`)}return await n.json()}async function sn(t){let e=new URLSearchParams({token:t});await fetch(Pa,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e}).catch(()=>{})}function an(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.searchParams.set("hd",qt),e.toString()}var zt,Lr,qt,Gr,Ta,jr,Pa,Ea,$a,rn,Jt=_(()=>{"use strict";zt="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",Lr="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",qt="nal.vn",Gr=["openid","email","profile"],Ta="https://oauth2.googleapis.com/device/code",jr="https://oauth2.googleapis.com/token",Pa="https://oauth2.googleapis.com/revoke";Ea=new Set(["https://accounts.google.com","accounts.google.com"]),$a=60;rn=Ur});import{Command as Ru}from"commander";Dt();import{resolve as ka}from"path";import{confirm as Wt,input as $t,select as Ze}from"@inquirer/prompts";Dt();k();f();import{spawnSync as fs}from"child_process";import{join as hs}from"path";import{confirm as ws,input as Fe,select as ks}from"@inquirer/prompts";import{spawnSync as ds}from"child_process";var Ke=5e3;function gs(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 ft(t){let e=ds("git",["ls-remote","--exit-code",t,"HEAD"],{encoding:"utf8",timeout:Ke,stdio:["ignore","pipe","pipe"]});if(e.error){let o=e.error;return o.code==="ENOENT"?{ok:!1,reason:"network",detail:"git binary kh\xF4ng t\xECm th\u1EA5y \u2014 c\xE0i git r\u1ED3i retry"}:o.code==="ETIMEDOUT"?{ok:!1,reason:"timeout",detail:`git ls-remote > ${Ke/1e3}s`}:{ok:!1,reason:"unknown",detail:o.message.slice(0,300)}}if(e.status===0)return{ok:!0};if(e.signal==="SIGTERM")return{ok:!1,reason:"timeout",detail:`git ls-remote > ${Ke/1e3}s`};let n=(e.stderr||"").trim();return{ok:!1,reason:gs(n),detail:n.slice(0,300)}}function ys(t,e,n){switch(s.warn(`\u26D4 Kh\xF4ng clone \u0111\u01B0\u1EE3c ${t}`),e){case"not-found":s.dim(" \u2192 Repo kh\xF4ng t\u1ED3n t\u1EA1i (sai URL, ho\u1EB7c repo ch\u01B0a \u0111\u01B0\u1EE3c t\u1EA1o tr\xEAn GitHub).");break;case"no-access":s.dim(` \u2192 Kh\xF4ng c\xF3 quy\u1EC1n truy c\u1EADp (repo private + sai account), HO\u1EB6C repo kh\xF4ng t\u1ED3n t\u1EA1i.
19
+ Ki\u1EC3m tra: \u0111ang login \u0111\xFAng GitHub account ch\u01B0a? (gh auth status)`);break;case"network":s.dim(" \u2192 L\u1ED7i m\u1EA1ng (m\u1EA5t k\u1EBFt n\u1ED1i / DNS / timeout).");break;default:s.dim(` \u2192 L\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh${n?`: ${n}`:""}.`)}}function ir(){s.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=fs("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&s.warn(`gh auth login exit ${t.status}. C\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}async function sr(t){let e=[{name:"Th\u1EED l\u1EA1i (retry)",value:"retry"},{name:"Nh\u1EADp URL repo kh\xE1c",value:"new-url"}];return t==="no-access"&&e.push({name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch-account"}),e.push({name:"B\u1ECF qua repo n\xE0y",value:"skip"}),await ks({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}async function ar(t){let e=t.url,n=t.name;for(;;){let r=ft(e);if(!r.ok){ys(e,r.reason??"unknown",r.detail);let o=await sr(r.reason??"unknown");if(o==="skip")return s.dim(`B\u1ECF qua repo ${n}.`),{cloned:null,skipped:!0};if(o==="switch-account"){ir();continue}if(o==="new-url"){if(e=await Fe({message:"URL repo kh\xE1c:",validate:a=>a.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),await ws({message:`\u0110\u1ED5i t\xEAn th\u01B0 m\u1EE5c trong src/ theo URL m\u1EDBi? (hi\u1EC7n: ${n})`,default:!1})){let{inferRepoNameFromUrl:a}=await Promise.resolve().then(()=>(Dt(),or));n=await Fe({message:"T\xEAn th\u01B0 m\u1EE5c:",default:a(e)})}continue}continue}try{return{cloned:await Ue({workspaceRoot:t.workspaceRoot,url:e,name:n}),skipped:!1}}catch(o){let i=o instanceof Error?o.message:String(o);s.warn(`Clone fail: ${i}`),await Z(hs(t.workspaceRoot,"src",n)).catch(()=>{});let a=await sr("unknown");if(a==="skip")return{cloned:null,skipped:!0};if(a==="switch-account"){ir();continue}if(a==="new-url"){e=await Fe({message:"URL repo kh\xE1c:",validate:c=>c.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"});continue}}}}import{spawnSync as cr}from"child_process";import{platform as bs}from"os";function tt(){let t=bs();return t==="darwin"||t==="linux"||t==="win32"?t:"unsupported"}var vs=5e3,xs=/(\d+\.\d+\.\d+)/;function Cs(){let e=tt()==="win32"?"where":"which",n=cr(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 As(){let t=cr("gitnexus",["--version"],{encoding:"utf8",timeout:vs});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return xs.exec(e)?.[1]??null}var et=null;function nt(){if(et!==null)return et;let t=Cs();return t?(et={installed:!0,version:As(),path:t},et):(et={installed:!1,version:null,path:null},et)}function Ht(){et=null}import{spawnSync as Ss}from"child_process";function Kt(){let t=Ss("gh",["auth","status"],{stdio:"ignore"});return t.error&&t.error.code==="ENOENT"?"not-installed":t.status===0?"authenticated":"not-authenticated"}import{spawnSync as Ts}from"child_process";function Ps(t){let e=tt();return Ts(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function lr(){let t=tt(),e=t==="darwin"?["brew"]:t==="win32"?["winget"]:t==="linux"?["apt","dnf","pacman"]:[];for(let n of e)if(Ps(n))return n;return null}import{spawnSync as kr}from"child_process";import{input as js,select as Us}from"@inquirer/prompts";ze();k();f();import{spawnSync as Je}from"child_process";import{promises as _s}from"fs";import{basename as Is,join as hr}from"path";import{confirm as Ns,select as Os}from"@inquirer/prompts";function Ms(){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 Ls(t){let e=hr(t,".git");if(!await m(e))throw new Error(`.git kh\xF4ng t\u1ED3n t\u1EA1i \u1EDF ${t} \u2014 kh\xF4ng c\u1EA7n reset.`);let n=`.git.backup-${Ms()}`,r=hr(t,n);return await _s.rename(e,r),s.success(`Backup .git \u2192 ${n}`),r}function Gs(t){let e=Je("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}`);s.success("Git init m\u1EDBi (branch main)"),Je("git",["-C",t,"add","-A"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});let n=Je("git",["-C",t,"commit","-m","chore: initial commit from existing folder"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});n.status!==0?s.warn(`git commit th\u1EA5t b\u1EA1i (folder c\xF3 th\u1EC3 r\u1ED7ng): ${(n.stderr||"").slice(0,200)}`):s.success("Initial commit")}async function wr(t){let e=Is(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await Ns({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
20
20
  1. Backup .git \u2192 .git.backup-{timestamp}
21
21
  2. Git init m\u1EDBi (branch main, initial commit)
22
22
  3. T\u1EA1o GitHub repo m\u1EDBi '${n}' d\u01B0\u1EDBi account c\u1EE7a b\u1EA1n
23
- Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let r=t.visibility??(t.autoYes?"private":await ps({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),o=await ds(t.folderPath);return gs(t.folderPath),{newRemoteUrl:or({folder:t.folderPath,name:n,visibility:r,org:t.org}).httpsUrl,backupPath:o}}f();var At=class extends Error{constructor(e){super(e),this.name="RemoteAccessAbortedError"}};function ws(){let t=ar("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function ks(){i.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=ar("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&i.warn(`gh auth login exit ${t.status}. B\u1EA1n c\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function ys(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 bs(t){let e=t.trim();return e?/^https?:\/\/[\w.@/-]+$/.test(e)||/^git@[\w.-]+:[\w./-]+\.git$/.test(e)||/^[\w.-]+\/[\w.-]+$/.test(e):!1}async function cr(t){let e=t.url,n=t.initialReason,r=t.initialDetail;for(;;){let o=ws();i.warn(`Kh\xF4ng truy c\u1EADp \u0111\u01B0\u1EE3c ${e}`),i.dim(` L\xFD do: ${n}${r?` \u2014 ${r.slice(0,150)}`:""}`),i.info(ys(n,e,o)),o&&i.dim(` gh CLI hi\u1EC7n \u0111ang login: ${o}`);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 hs({message:"C\xE1ch x\u1EED l\xFD?",choices:s});if(a==="abort")throw new At(`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){i.warn("Reset folder c\u1EA7n folderPath \u2014 internal error.");continue}try{let l=await sr({folderPath:t.folderPath,visibility:t.defaultVisibility});return i.success(`Folder \u0111\xE3 reset. Backup t\u1EA1i: ${l.backupPath}`),i.success(`Remote m\u1EDBi: ${l.newRemoteUrl}`),{resolvedUrl:l.newRemoteUrl}}catch(l){i.warn(`Reset folder th\u1EA5t b\u1EA1i: ${l.message}`);continue}}a==="re-input-url"&&(e=(await fs({message:"URL git remote (https://github.com/owner/repo.git ho\u1EB7c git@github.com:owner/repo.git):",default:e,validate:p=>bs(p)||"URL kh\xF4ng \u0111\xFAng format git remote"})).trim()),a==="switch"&&ks(),i.info(`Verify remote l\u1EA1i: ${e}...`);let c=dt(e);if(c.ok)return i.success(`Remote accessible: ${e}`),{resolvedUrl:e};n=c.reason??"unknown",r=c.detail}}f();import{spawnSync as vs}from"child_process";var xs={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 lr(t){let e=xs[t];i.info(`\u0110ang c\xE0i gh CLI qua ${t}...`);let n=vs(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.`);i.success("\u0110\xE3 c\xE0i gh CLI")}f();import{input as Dp,select as Cs}from"@inquirer/prompts";var C=class extends Error{constructor(e){super(e),this.name="UserAbortedRecoveryError"}};async function _(t){i.warn(`${t.taskName} th\u1EA5t b\u1EA1i: ${t.reason}`),t.hint&&i.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 Cs({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}f();import{spawnSync as As}from"child_process";function ur(){if(As("gh",["auth","setup-git"],{stdio:"ignore"}).status!==0){i.warn("gh auth setup-git fail (non-fatal). N\u1EBFu git clone l\u1ED7i 128 \u2192 ch\u1EA1y th\u1EE7 c\xF4ng.");return}i.dim("Git credential helper \u0111\xE3 link v\u1EDBi gh token.")}f();f();import{spawnSync as Ss}from"child_process";function pr(){i.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp GitHub qua gh CLI (browser s\u1EBD m\u1EDF)...");let t=Ss("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.`);i.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp GitHub")}async function mr(t){for(;Ht()==="not-installed";){i.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=tr();if(!e){if(await _({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 C("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{lr(e)}catch(n){if(await _({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 C("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;Ht()==="not-authenticated";){i.warn("Ch\u01B0a \u0111\u0103ng nh\u1EADp GitHub.");try{pr()}catch(e){if(await _({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 C("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(Ht()!=="authenticated"&&await _({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 C("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(i.success("gh CLI s\u1EB5n s\xE0ng"),ur(),t){let e=dt(t);return e.ok?(i.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await cr({url:t,initialReason:e.reason??"unknown",initialDetail:e.detail})).resolvedUrl}}return{}}import{existsSync as Ke}from"fs";import{dirname as Ps,join as Fe}from"path";var Ts=5;function Es(t){let e=Ke(Fe(t,".claude")),n=Ke(Fe(t,"CLAUDE.md")),r=Ke(Fe(t,"src"));return e&&n&&r}function gt(t){let e=t;for(let n=0;n<Ts;n++){if(Es(e))return e;let r=Ps(e);if(r===e)return null;e=r}return null}f();import{spawnSync as gr}from"child_process";import{existsSync as $s}from"fs";import{join as dr}from"path";var Rs=120*1e3,_s=300*1e3,U=class extends Error{operation;reason;exitCode;stderr;constructor(e,n,r,o=null,s){super(r),this.name="GitnexusOperationError",this.operation=e,this.reason=n,this.exitCode=o,this.stderr=s}};function fr(t,e,n,r){if(n==="SIGTERM")return new U(t,"timeout",`gitnexus ${t} timeout. Check m\u1EA1ng / repo size.`,null,r);let o=r.toLowerCase();return o.includes("eacces")||o.includes("permission denied")?new U(t,"permission",`gitnexus ${t} fail (permission). Check write access ~/.claude ho\u1EB7c cwd.`,e,r):new U(t,"non-zero-exit",`gitnexus ${t} exit ${e??"null"}. Xem log ph\xEDa tr\xEAn.`,e,r)}function Ft(t,e){return t.split(`
23
+ Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let r=t.visibility??(t.autoYes?"private":await Os({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),o=await Ls(t.folderPath);return Gs(t.folderPath),{newRemoteUrl:qe({folder:t.folderPath,name:n,visibility:r,org:t.org}).httpsUrl,backupPath:o}}f();var Tt=class extends Error{constructor(e){super(e),this.name="RemoteAccessAbortedError"}};function Ds(){let t=kr("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function Hs(){s.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=kr("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&s.warn(`gh auth login exit ${t.status}. B\u1EA1n c\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function Ks(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 Fs(t){let e=t.trim();return e?/^https?:\/\/[\w.@/-]+$/.test(e)||/^git@[\w.-]+:[\w./-]+\.git$/.test(e)||/^[\w.-]+\/[\w.-]+$/.test(e):!1}async function yr(t){let e=t.url,n=t.initialReason,r=t.initialDetail;for(;;){let o=Ds();s.warn(`Kh\xF4ng truy c\u1EADp \u0111\u01B0\u1EE3c ${e}`),s.dim(` L\xFD do: ${n}${r?` \u2014 ${r.slice(0,150)}`:""}`),s.info(Ks(n,e,o)),o&&s.dim(` gh CLI hi\u1EC7n \u0111ang login: ${o}`);let i=[{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&&i.push({name:"Reset folder & t\u1EA1o repo M\u1EDAI d\u01B0\u1EDBi account c\u1EE7a t\xF4i (backup .git c\u0169)",value:"reset-folder"}),i.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i 'avatar init' sau",value:"abort"});let a=await Us({message:"C\xE1ch x\u1EED l\xFD?",choices:i});if(a==="abort")throw new Tt(`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){s.warn("Reset folder c\u1EA7n folderPath \u2014 internal error.");continue}try{let l=await wr({folderPath:t.folderPath,visibility:t.defaultVisibility});return s.success(`Folder \u0111\xE3 reset. Backup t\u1EA1i: ${l.backupPath}`),s.success(`Remote m\u1EDBi: ${l.newRemoteUrl}`),{resolvedUrl:l.newRemoteUrl}}catch(l){s.warn(`Reset folder th\u1EA5t b\u1EA1i: ${l.message}`);continue}}a==="re-input-url"&&(e=(await js({message:"URL git remote (https://github.com/owner/repo.git ho\u1EB7c git@github.com:owner/repo.git):",default:e,validate:p=>Fs(p)||"URL kh\xF4ng \u0111\xFAng format git remote"})).trim()),a==="switch"&&Hs(),s.info(`Verify remote l\u1EA1i: ${e}...`);let c=ft(e);if(c.ok)return s.success(`Remote accessible: ${e}`),{resolvedUrl:e};n=c.reason??"unknown",r=c.detail}}f();import{spawnSync as Vs}from"child_process";var Bs={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 br(t){let e=Bs[t];s.info(`\u0110ang c\xE0i gh CLI qua ${t}...`);let n=Vs(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.`);s.success("\u0110\xE3 c\xE0i gh CLI")}f();import{input as nm,select as Ws}from"@inquirer/prompts";var A=class extends Error{constructor(e){super(e),this.name="UserAbortedRecoveryError"}};async function O(t){s.warn(`${t.taskName} th\u1EA5t b\u1EA1i: ${t.reason}`),t.hint&&s.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 Ws({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}f();import{spawnSync as qs}from"child_process";function vr(){if(qs("gh",["auth","setup-git"],{stdio:"ignore"}).status!==0){s.warn("gh auth setup-git fail (non-fatal). N\u1EBFu git clone l\u1ED7i 128 \u2192 ch\u1EA1y th\u1EE7 c\xF4ng.");return}s.dim("Git credential helper \u0111\xE3 link v\u1EDBi gh token.")}f();f();import{spawnSync as zs}from"child_process";function xr(){s.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp GitHub qua gh CLI (browser s\u1EBD m\u1EDF)...");let t=zs("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.`);s.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp GitHub")}async function Ye(t){for(;Kt()==="not-installed";){s.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=lr();if(!e){if(await O({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 A("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{br(e)}catch(n){if(await O({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 A("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;Kt()==="not-authenticated";){s.warn("Ch\u01B0a \u0111\u0103ng nh\u1EADp GitHub.");try{xr()}catch(e){if(await O({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 A("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(Kt()!=="authenticated"&&await O({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 A("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(s.success("gh CLI s\u1EB5n s\xE0ng"),vr(),t){let e=ft(t);return e.ok?(s.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await yr({url:t,initialReason:e.reason??"unknown",initialDetail:e.detail})).resolvedUrl}}return{}}import{existsSync as Xe}from"fs";import{dirname as Js,join as Qe}from"path";var Ys=5;function Xs(t){let e=Xe(Qe(t,".claude")),n=Xe(Qe(t,"CLAUDE.md")),r=Xe(Qe(t,"src"));return e&&n&&r}function ht(t){let e=t;for(let n=0;n<Ys;n++){if(Xs(e))return e;let r=Js(e);if(r===e)return null;e=r}return null}f();import{spawnSync as Ar}from"child_process";import{existsSync as Qs}from"fs";import{join as Cr}from"path";var Zs=120*1e3,ta=300*1e3,D=class extends Error{operation;reason;exitCode;stderr;constructor(e,n,r,o=null,i){super(r),this.name="GitnexusOperationError",this.operation=e,this.reason=n,this.exitCode=o,this.stderr=i}};function Sr(t,e,n,r){if(n==="SIGTERM")return new D(t,"timeout",`gitnexus ${t} timeout. Check m\u1EA1ng / repo size.`,null,r);let o=r.toLowerCase();return o.includes("eacces")||o.includes("permission denied")?new D(t,"permission",`gitnexus ${t} fail (permission). Check write access ~/.claude ho\u1EB7c cwd.`,e,r):new D(t,"non-zero-exit",`gitnexus ${t} exit ${e??"null"}. Xem log ph\xEDa tr\xEAn.`,e,r)}function Vt(t,e){return t.split(`
24
24
  `).slice(-e).join(`
25
- `)}function hr(){let t=Ct("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=gr("gitnexus",["setup"],{stdio:["ignore","pipe","pipe"],timeout:Rs,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(`${Ft(n,30)}
26
- `):r&&process.stderr.write(`${Ft(r,30)}
27
- `),fr("setup",e.status,e.signal,n)}t.succeed("GitNexus setup OK (global skills installed)")}function ft(t){let e=Ct(`Analyze workspace ${t} (1-3 ph\xFAt)`),n=gr("gitnexus",["analyze","."],{cwd:t,stdio:["ignore","pipe","pipe"],timeout:_s,encoding:"utf8"});if(n.status!==0||n.signal==="SIGTERM"){e.fail("Analyze failed");let o=(n.stderr||"").trim(),s=(n.stdout||"").trim();throw o?process.stderr.write(`${Ft(o,30)}
28
- `):s&&process.stderr.write(`${Ft(s,30)}
29
- `),fr("analyze",n.status,n.signal,o)}let r=dr(t,".gitnexus","meta.json");if(!$s(r))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new U("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 ${dr(t,".gitnexus")})`)}import{spawnSync as js}from"child_process";import{existsSync as Us}from"fs";import{join as vr}from"path";import{confirm as Ds}from"@inquirer/prompts";var Is=[/^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 wr(t){return t?Is.some(e=>e.test(t)):!1}k();f();import{promises as St}from"fs";import{homedir as Ns}from"os";import{dirname as Os,join as kr}from"path";var Ms=kr(Ns(),".gitnexus"),Pt=kr(Ms,"config.json");async function Ls(){try{let t=await St.readFile(Pt,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function Gs(t){await St.mkdir(Os(Pt),{recursive:!0});let e=`${Pt}.tmp-${process.pid}`;await St.writeFile(e,t,{mode:384}),await St.rename(e,Pt)}async function yr(t){let n={...await Ls(),apiKey:t.apiKey,baseUrl:t.baseUrl,model:t.model,isReasoningModel:t.isReasoningModel??!1};await Gs(JSON.stringify(n,null,2)),await St.chmod(Pt,384).catch(()=>{})}var Hs=900*1e3,Ks="nal-claude",Fs="claude-sonnet-4-5";function Vs(t){let e=t.replace(/\/+$/,"");return e.endsWith("/v1")?`${e}/`:e.endsWith("/v1/")?e:`${e}/v1/`}async function Bs(t){let e=vr(t,".claude","settings.json");if(!await m(e))return null;try{let n=await y(e),r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:null;if(!o)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=h=>{let v=process.env[h];if(typeof v=="string"&&v.length>0)return v;let A=r[h];return typeof A=="string"&&A.length>0?A:null},p=typeof n.avatarProvider=="string"?n.avatarProvider:null,d=!1;try{d=new URL(o).hostname==="api.anthropic.com"}catch{d=!1}if((p==="anthropic"?"anthropic":p==="llmlite"?"llmlite":d?"anthropic":"llmlite")==="anthropic"){let h=l("ANTHROPIC_API_KEY");if(h)return{provider:"anthropic",apiKey:h,baseUrl:Vs(o),model:c.length>0?c:Fs}}else{let h=l("ANTHROPIC_AUTH_TOKEN");if(h)return{provider:"llmlite",apiKey:h,baseUrl:o,model:c.length>0?c:Ks}}return null}catch{return null}}async function Ws(t,e){return await Ds({message:`Generate wiki cho workspace? (~$0.50 qua ${t} model=${e}, 2-5 ph\xFAt). Continue?`,default:!0})}function br(t,e){return t.split(`
25
+ `)}function Tr(){let t=St("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=Ar("gitnexus",["setup"],{stdio:["ignore","pipe","pipe"],timeout:Zs,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(`${Vt(n,30)}
26
+ `):r&&process.stderr.write(`${Vt(r,30)}
27
+ `),Sr("setup",e.status,e.signal,n)}t.succeed("GitNexus setup OK (global skills installed)")}function wt(t){let e=St(`Analyze workspace ${t} (1-3 ph\xFAt)`),n=Ar("gitnexus",["analyze","."],{cwd:t,stdio:["ignore","pipe","pipe"],timeout:ta,encoding:"utf8"});if(n.status!==0||n.signal==="SIGTERM"){e.fail("Analyze failed");let o=(n.stderr||"").trim(),i=(n.stdout||"").trim();throw o?process.stderr.write(`${Vt(o,30)}
28
+ `):i&&process.stderr.write(`${Vt(i,30)}
29
+ `),Sr("analyze",n.status,n.signal,o)}let r=Cr(t,".gitnexus","meta.json");if(!Qs(r))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new D("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 ${Cr(t,".gitnexus")})`)}import{spawnSync as aa}from"child_process";import{existsSync as ca}from"fs";import{join as _r}from"path";import{confirm as la}from"@inquirer/prompts";var ea=[/^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 Pr(t){return t?ea.some(e=>e.test(t)):!1}k();f();import{promises as Pt}from"fs";import{homedir as na}from"os";import{dirname as ra,join as Er}from"path";var oa=Er(na(),".gitnexus"),Et=Er(oa,"config.json");async function ia(){try{let t=await Pt.readFile(Et,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function sa(t){await Pt.mkdir(ra(Et),{recursive:!0});let e=`${Et}.tmp-${process.pid}`;await Pt.writeFile(e,t,{mode:384}),await Pt.rename(e,Et)}async function $r(t){let n={...await ia(),apiKey:t.apiKey,baseUrl:t.baseUrl,model:t.model,isReasoningModel:t.isReasoningModel??!1};await sa(JSON.stringify(n,null,2)),await Pt.chmod(Et,384).catch(()=>{})}var ua=900*1e3,pa="nal-claude",ma="claude-sonnet-4-5";function da(t){let e=t.replace(/\/+$/,"");return e.endsWith("/v1")?`${e}/`:e.endsWith("/v1/")?e:`${e}/v1/`}async function ga(t){let e=_r(t,".claude","settings.json");if(!await m(e))return null;try{let n=await y(e),r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:null;if(!o)return null;let i=typeof n.model=="string"?n.model:"",a=typeof r.ANTHROPIC_MODEL=="string"?r.ANTHROPIC_MODEL:"",c=i.length>0?i:a,l=h=>{let x=process.env[h];if(typeof x=="string"&&x.length>0)return x;let S=r[h];return typeof S=="string"&&S.length>0?S:null},p=typeof n.avatarProvider=="string"?n.avatarProvider:null,d=!1;try{d=new URL(o).hostname==="api.anthropic.com"}catch{d=!1}if((p==="anthropic"?"anthropic":p==="llmlite"?"llmlite":d?"anthropic":"llmlite")==="anthropic"){let h=l("ANTHROPIC_API_KEY");if(h)return{provider:"anthropic",apiKey:h,baseUrl:da(o),model:c.length>0?c:ma}}else{let h=l("ANTHROPIC_AUTH_TOKEN");if(h)return{provider:"llmlite",apiKey:h,baseUrl:o,model:c.length>0?c:pa}}return null}catch{return null}}async function fa(t,e){return await la({message:`Generate wiki cho workspace? (~$0.50 qua ${t} model=${e}, 2-5 ph\xFAt). Continue?`,default:!0})}function Rr(t,e){return t.split(`
30
30
  `).slice(-e).join(`
31
- `)}async function Vt(t,e=t){let n=await Bs(t);if(!n)return i.warn(`Kh\xF4ng resolve \u0111\u01B0\u1EE3c API key cho wiki gen.
31
+ `)}async function Bt(t,e=t){let n=await ga(t);if(!n)return s.warn(`Kh\xF4ng resolve \u0111\u01B0\u1EE3c API key cho wiki gen.
32
32
  \u0110\xE3 ki\u1EC3m tra: process.env \u2192 .envrc Avatar block \u2192 settings.json.env
33
33
  Nguy\xEAn nh\xE2n c\xF3 th\u1EC3:
34
34
  \u2022 Subscription mode (OAuth, kh\xF4ng c\xF3 key)
35
35
  \u2022 Key ch\u01B0a setup \u2014 ch\u1EA1y 'avatar ai setup' ho\u1EB7c 'avatar secrets set ANTHROPIC_API_KEY'
36
- \u2022 settings.json.env.ANTHROPIC_BASE_URL b\u1ECB thi\u1EBFu (provider ch\u01B0a configured)`),i.dim(`\u0110\u1EC3 gen wiki sau khi \u0111\xE3 c\xF3 key, ch\u1EA1y manual:
37
- gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await Ws(n.baseUrl,n.model))return i.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 o=wr(n.model);await yr({apiKey:n.apiKey,baseUrl:n.baseUrl,model:n.model,isReasoningModel:o});let s=["wiki",".","--base-url",n.baseUrl,"--model",n.model];o&&s.push("--reasoning-model");let a=Ct(`Generating wiki via ${n.baseUrl} (${n.provider}) model=${n.model}${o?" [reasoning]":""}`),c=js("gitnexus",s,{cwd:e,stdio:["ignore","pipe","pipe"],timeout:Hs,encoding:"utf8"});if(c.status!==0||c.signal==="SIGTERM"){let p=c.signal==="SIGTERM"?"timeout":"non-zero-exit";a.fail(`Wiki gen ${p} (exit ${c.status??"null"})`);let d=(c.stderr||"").trim(),g=(c.stdout||"").trim();return d?process.stderr.write(`${br(d,30)}
38
- `):g&&process.stderr.write(`${br(g,30)}
39
- `),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${p} (exit ${c.status??"null"})`}}let l=vr(e,".gitnexus","wiki","index.html");return Us(l)?(a.succeed(`Wiki ready: ${l}`),{ran:!0,skipped:!1,wikiPath:l}):(a.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 ${l}`})}f();function Sr(t){t.command("add").description("Th\xEAm t\xE0i nguy\xEAn v\xE0o workspace").command("repo").description("Clone 1 repo code v\xE0o src/ (repo c\xF3 s\u1EB5n / th\u01B0 m\u1EE5c / d\u1EF1 \xE1n m\u1EDBi)").option("--url <url>","URL git repo c\xF3 s\u1EB5n (b\u1ECF qua prompt ngu\u1ED3n)").option("--name <name>","T\xEAn th\u01B0 m\u1EE5c trong src/ (m\u1EB7c \u0111\u1ECBnh suy t\u1EEB URL)").option("--yes","Auto-confirm prompt").action(async n=>{try{await Xs(n)}catch(r){i.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}function Ys(){let t=gt(process.cwd());return t||(i.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
40
- C\u1EA7n .claude/ + CLAUDE.md + src/. Ch\u1EA1y 'avatar init' tr\u01B0\u1EDBc.`),process.exit(1)),t}async function Xs(t){let e=Ys(),n=!0,r=t.url,o=t.name;for(;n;){let s=await Qs(r),a=o??await Zs(s);i.info(`Clone ${s} \u2192 src/${a} ...`);let{cloned:c,skipped:l}=await Qn({workspaceRoot:e,url:s,name:a});if(l||!c?i.dim(`B\u1ECF qua repo ${a}.`):(i.success(`\u2713 \u0110\xE3 clone src/${c.name}`),await ta(e,c.path,t.yes),i.success(`\u2713 Done repo: ${c.name}`)),r=void 0,o=void 0,t.yes)break;n=await Ve({message:"Add repo kh\xE1c n\u1EEFa?",default:!1})}}async function Qs(t){if(t)return t;let e=await Ar({message:"Ngu\u1ED3n repo?",choices:[{name:"1. Repo c\xF3 s\u1EB5n (\u0111i\u1EC1n URL git)",value:"url"},{name:"2. Th\u01B0 m\u1EE5c \u0111\xE3 c\xF3 tr\xEAn m\xE1y (clone t\u1EEB remote c\u1EE7a n\xF3)",value:"folder"},{name:"3. D\u1EF1 \xE1n m\u1EDBi ho\xE0n to\xE0n (t\u1EA1o repo GitHub)",value:"new"}]});if(e==="url")return await Bt({message:"URL git repo:",validate:s=>s.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"});if(e==="folder"){let s=Js(await Bt({message:"\u0110\u01B0\u1EDDng d\u1EABn folder:",validate:l=>l.trim().length>0?!0:"Path b\u1EAFt bu\u1ED9c"})),a=await Oe(s);if(a&&(i.warn(`\u26A0 Folder c\xF3 thay \u0111\u1ED5i ch\u01B0a commit: ${a.slice(0,120)}`),i.warn(" Clone l\u1EA5y b\u1EA3n REMOTE \u2192 c\xE1c thay \u0111\u1ED5i local n\xE0y s\u1EBD KH\xD4NG c\xF3 trong src/<name>."),!await Ve({message:"Ti\u1EBFp t\u1EE5c clone t\u1EEB remote (b\u1ECF thay \u0111\u1ED5i local ch\u01B0a push)?",default:!1})))throw new Error("H\u1EE7y. Commit + push thay \u0111\u1ED5i trong folder g\u1ED1c r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar add repo'.");let c=await Me(s);if(!c)throw new Error(`Folder ${s} kh\xF4ng c\xF3 remote origin \u2192 kh\xF4ng clone \u0111\u01B0\u1EE3c. T\u1EA1o remote (gh repo create) r\u1ED3i th\u1EED l\u1EA1i.`);return c}await mr();let n=await Bt({message:"T\xEAn repo m\u1EDBi:",validate:s=>s.trim().length>0?!0:"T\xEAn b\u1EAFt bu\u1ED9c"}),r=await Ar({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),{createEmptyGithubRepoAndGetUrl:o}=await Promise.resolve().then(()=>(Cr(),xr));return await o(n.trim(),r)}async function Zs(t){let e=Ie(t);return await Bt({message:"T\xEAn th\u01B0 m\u1EE5c trong src/:",default:e})}async function ta(t,e,n){if(!Q().installed){i.dim("GitNexus ch\u01B0a c\xE0i \u2014 skip index. C\xE0i qua 'avatar gitnexus install'.");return}if(n||await Ve({message:"Index repo n\xE0y b\u1EB1ng GitNexus?",default:!0})){try{ft(e),i.success(` \u2713 GitNexus indexed src/${e.split("/").pop()}`)}catch(s){i.warn(` ! GitNexus index fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${s instanceof Error?s.message:s}`);return}try{let s=await Vt(t,e);s.ran?i.success(` \u2713 GitNexus wiki t\u1EA1o cho src/${e.split("/").pop()}`):s.reason==="subscription-mode"&&i.dim(" (Subscription mode \u2014 kh\xF4ng c\xF3 API key cho wiki, skip.)")}catch(s){i.warn(` ! Wiki fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${s instanceof Error?s.message:s}`)}}}k();import{promises as Xa}from"fs";import{join as Yr}from"path";import{confirm as Qa}from"@inquirer/prompts";import{promises as Ir}from"fs";import{homedir as aa}from"os";import{join as Et}from"path";import{z as w}from"zod";var Pr=w.object({email:w.string().email(),name:w.string(),access_token:w.string().min(1),refresh_token:w.string().min(1),expires_at:w.string().datetime(),id_token:w.string().min(1)}),ea=w.object({installed_tools:w.record(w.string(),w.object({version:w.string().optional(),installed_at:w.string().datetime(),install_method:w.string()})).default({}),tool_inputs:w.record(w.string(),w.unknown()).default({})}),Fm=w.object({$schema:w.string().optional(),includeCoAuthoredBy:w.boolean().optional(),env:w.record(w.string(),w.string()).default({}),permissions:w.object({allow:w.array(w.string()).default([]),deny:w.array(w.string()).default([])}).partial().optional(),hooks:w.record(w.string(),w.array(w.unknown())).optional(),statusLine:w.object({type:w.string(),command:w.string(),padding:w.number().optional()}).optional()}),Vm=w.enum(["internal","client","library"]);k();var $t=Et(aa(),".avatar"),Z=Et($t,"config.json"),Ym=Et($t,"state.json"),Ze=Et($t,"audit.log"),Xm=Et($t,"backups"),ca=384;async function tn(){await x($t)}async function D(){if(!await m(Z))return null;let t=await y(Z),e=Pr.safeParse(t);return e.success?e.data:null}async function en(t){await tn(),await S(Z,t,ca)}async function _r(){if(await m(Z)){let{promises:t}=await import("fs");await t.unlink(Z)}}function tt(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}var Tt=class extends Error{constructor(e){super(e),this.name="NoValidTokenError"}};async function la(t){let{decodeIdToken:e}=await Promise.resolve().then(()=>(zt(),Qe));try{let n=e(t),r=Math.floor(Date.now()/1e3);return n.exp-60<r}catch{return!0}}async function Jt(){let t=await D();if(!t)throw new Tt("Ch\u01B0a \u0111\u0103ng nh\u1EADp. Ch\u1EA1y 'avatar login' tr\u01B0\u1EDBc.");let{refreshAccessToken:e,decodeIdToken:n,verifyIdTokenClaims:r}=await Promise.resolve().then(()=>(zt(),Qe));if(!await la(t.id_token))return r(n(t.id_token)),t.id_token;let o;try{o=await e(t.refresh_token)}catch(a){throw new Tt(`Token h\u1EBFt h\u1EA1n v\xE0 refresh th\u1EA5t b\u1EA1i (${a instanceof Error?a.message:a}). Ch\u1EA1y 'avatar login' l\u1EA1i.`)}if(!o.id_token)throw new Tt("Refresh kh\xF4ng tr\u1EA3 id_token m\u1EDBi (id_token c\u0169 \u0111\xE3 h\u1EBFt h\u1EA1n). Ch\u1EA1y 'avatar login' l\u1EA1i.");r(n(o.id_token));let s={...t,access_token:o.access_token,id_token:o.id_token,expires_at:new Date(Date.now()+o.expires_in*1e3).toISOString()};return await en(s),o.id_token}async function ua(){try{await Ir.chmod(Ze,384)}catch{}}async function b(t,e){await tn();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
41
- `;await Ir.appendFile(Ze,r,{encoding:"utf8",mode:384}),await ua()}f();import{spawnSync as nn}from"child_process";var Nr=6e4,pa="ok";function rn(){let t=nn("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 on(){i.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=nn("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.`);i.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp Claude Code")}function ma(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 Or(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 sn(){let t=nn("claude",["--print",pa],{encoding:"utf8",timeout:Nr,stdio:["ignore","pipe","pipe"]});if(t.signal==="SIGTERM"||t.status===143||t.error?.code==="ETIMEDOUT")return{ok:!1,reason:"timeout",detail:`claude --print > ${Nr/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 o=r.trim(),s=n.toLowerCase();if(o.length>20&&!s.includes("error")&&!s.includes("limit")&&!s.includes("quota")&&!s.includes("401"))return i.warn(`claude --print exit=${t.status} nh\u01B0ng c\xF3 response (${o.length} chars). Accept v\u1EDBi caution.`),{ok:!0};let a=ma(`${n}
42
- ${r}`);return a==="unknown"&&(i.warn(`[debug] claude --print exit=${t.status} signal=${t.signal??"none"}`),n.trim()&&i.warn(`[debug] stderr: ${n.slice(0,500)}`),r.trim()&&i.warn(`[debug] stdout: ${r.slice(0,300)}`)),{ok:!1,reason:a,detail:n.slice(0,500)||r.slice(0,500)}}import{spawnSync as Mr}from"child_process";var da=5e3,ga=/(\d+\.\d+\.\d+)/;function fa(){let e=Y()==="win32"?"where":"which",n=Mr(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 ha(){let t=Mr("claude",["--version"],{encoding:"utf8",timeout:da});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return ga.exec(e)?.[1]??null}var et=null;function Rt(){if(et!==null)return et;let t=fa();return t?(et={installed:!0,version:ha(),path:t},et):(et={installed:!1,version:null,path:null},et)}function Yt(){et=null}import{spawnSync as wa}from"child_process";f();var Lr=300*1e3,Gr="@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 ka(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 ${Gr} 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 jr(){i.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=wa("npm",["install","-g",Gr],{stdio:["inherit","inherit","pipe"],timeout:Lr,encoding:"utf8"});if(t.signal==="SIGTERM")throw new nt("timeout",`npm install timeout sau ${Lr/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),ka(t.status,t.stderr||"");Yt();let e=Rt();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 i.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 ya}from"fs";import{homedir as ba}from"os";import{join as va}from"path";import{select as Ur}from"@inquirer/prompts";function xa(){return va(ba(),".claude","settings.json")}function an(){let t=xa(),e;try{e=ya(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||{},o=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:!!o,baseUrl:o,hasToken:s,model:a,rawSettings:n}}async function Dr(t=an()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await Ur({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 Ur({message:"Ch\u1ECDn provider cho AI tools:",choices:[{name:"1. Claude Code Subscription (d\xF9ng quota c\xE1 nh\xE2n Anthropic, OAuth login)",value:"subscription"},{name:"2. LLM key NAL (ai.nal.vn \u2014 gateway n\u1ED9i b\u1ED9, key sk-...)",value:"llmlite"},{name:"3. Anthropic API key tr\u1EF1c ti\u1EBFp (console.anthropic.com, key sk-ant-...)",value:"anthropic"}]})}f();import{password as Ca,select as Aa}from"@inquirer/prompts";var Xt="https://api.anthropic.com",Sa="2023-06-01",Hr=1e4;function Pa(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function Ta(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 Ea(){return await Ca({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:Ta})}async function $a(t){let e=new AbortController,n=setTimeout(()=>e.abort(),Hr);try{let r=await fetch(`${Xt}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":Sa,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 ${Xt} timeout sau ${Hr/1e3}s.`):r}finally{clearTimeout(n)}}async function Ra(t){if(t.length===1){let n=t[0];return i.info(`Auto-pick model: ${n} (ch\u1EC9 1 model available)`),n}let e=[...t].sort((n,r)=>{let o=s=>{let a=s.toLowerCase();return a.includes("sonnet")?0:a.includes("opus")?1:a.includes("haiku")?2:3};return o(n)-o(r)});return await Aa({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function Kr(){let t=await Ea();i.info(`Verify key (${Pa(t)}) qua ${Xt}/v1/models...`);let e=await $a(t);i.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await Ra(e);return{apiKey:t,baseUrl:Xt,model:n}}f();import{input as _a,password as Ia,select as Na}from"@inquirer/prompts";var Oa="https://ai.nal.vn",Fr=1e4;function H(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function Ma(){return await Ia({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function La(t=Oa){return(await _a({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 Ga(t,e){let n=new AbortController,r=setTimeout(()=>n.abort(),Fr);try{let o=await fetch(`${t}/v1/models`,{method:"GET",headers:{Authorization:`Bearer ${e}`,Accept:"application/json"},signal:n.signal});if(o.status===401||o.status===403)throw new Error(`API key invalid (HTTP ${o.status}).`);if(o.status===404)throw new Error(`Endpoint /v1/models kh\xF4ng t\u1ED3n t\u1EA1i tr\xEAn ${t}.`);if(!o.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${o.status}).`);let a=((await o.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(o){if(o.name==="AbortError"){let c=t.includes("nal.vn")||t.includes("nal-vn")?"\n Hint: ai.nal.vn l\xE0 internal endpoint NAL \u2014 ki\u1EC3m tra VPN NAL \u0111\xE3 b\u1EADt ch\u01B0a, ho\u1EB7c tcp test b\u1EB1ng `curl -v https://ai.nal.vn`.":`
43
- Hint: check m\u1EA1ng / firewall / VPN, ho\u1EB7c base URL c\xF3 \u0111\xFAng kh\xF4ng.`;throw new Error(`Connect ${t} timeout sau ${Fr/1e3}s.${c}`)}let s=o.message||String(o);if(s.toLowerCase().includes("fetch failed")||s.includes("ENOTFOUND")){let c=t.includes("nal.vn")?" (ai.nal.vn c\u1EA7n VPN NAL \u2014 ki\u1EC3m tra VPN \u0111\xE3 b\u1EADt ch\u01B0a)":"";throw new Error(`Network error khi connect ${t}: ${s}${c}`)}throw o}finally{clearTimeout(r)}}async function ja(t){let e=t.filter(r=>r.toLowerCase().includes("claude"));if(e.length===1){let r=e[0];return i.info(`Auto-pick model: ${r} (ch\u1EC9 1 claude alias tr\xEAn endpoint)`),r}let n=e.length>0?e:t;return await Na({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function Vr(){let t=await Ma(),e=await La();i.info(`Verify key (${H(t)}) qua ${e}/v1/models...`);let n=await Ga(e,t);i.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await ja(n);return{apiKey:t,baseUrl:e,model:r}}f();k();import{promises as Ua}from"fs";import{join as Da}from"path";var cn=384;function Ha(t){return Da(t,".claude","settings.json")}async function Ka(t){if(!await m(t))return{};try{return await y(t)}catch(e){throw new Error(`Kh\xF4ng parse \u0111\u01B0\u1EE3c ${t} (JSON l\u1ED7i): ${e.message}. Backup file r\u1ED3i x\xF3a \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}function Fa(t,e){let{env:n,...r}=t,o={...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&&(o.env=l)}return o}function Br(t){if(!t)return t;let{ANTHROPIC_AUTH_TOKEN:e,ANTHROPIC_API_KEY:n,...r}=t;return r}function Va(t,e,n,r,o){let a={...Br(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_AUTH_TOKEN=e),{...t,env:a,model:r,avatarProvider:"llmlite"}}function Ba(t,e,n,r,o){let a={...Br(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_API_KEY=e),{...t,env:a,model:r,avatarProvider:"anthropic"}}function Wa(t,e){let n=e.env||{},r=typeof e.model=="string"?e.model:void 0;return{...t,env:{...t.env||{},...n},...r?{model:r}:{}}}async function rt(t,e){let n=Ha(t),r=await Ka(n),o;switch(e.provider){case"subscription":o=Fa(r,e.model);break;case"llmlite":o=Va(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"anthropic":o=Ba(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"use-global":o=Wa(r,e.sourceSettings);break}await S(n,o,cn);try{await Ua.chmod(n,cn)}catch{}return{path:n,mode:cn}}var O="sonnet";async function Qt(t){try{i.info("Setup AI provider cho workspace...");let e=Rt();if(e.installed)i.success(`Claude Code \u0111\xE3 c\xF3${e.version?` v${e.version}`:""}`);else if(i.info("Ch\u01B0a c\xF3 Claude Code \u2014 s\u1EBD t\u1EF1 c\xE0i qua npm."),jr(),Yt(),e=Rt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=an();switch(await Dr(n)){case"subscription":{let o=rn();if(o.state!=="authenticated"&&(on(),o=rn()),o.state==="authenticated"&&o.subscriptionType)return await rt(t.workspacePath,{provider:"subscription",model:O}),await b("ai_setup",`provider=subscription,result=ok,plan=${o.subscriptionType},probe=skipped`),i.success(`AI ready \xB7 Subscription (${o.subscriptionType}) \xB7 model=${O}`),{ok:!0,provider:"subscription",model:O};i.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let s=sn();if(!s.ok&&s.reason==="auth-expired"&&(i.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),on(),s=sn()),!s.ok&&(s.reason==="timeout"||s.reason==="unknown"))return i.warn(`Probe verify ${s.reason} \u2014 accept trust auth status. Ti\u1EBFp t\u1EE5c.`),s.detail?.trim()&&i.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),await rt(t.workspacePath,{provider:"subscription",model:O}),await b("ai_setup",`provider=subscription,result=ok,probe=${s.reason}-soft-pass`),i.success(`AI ready \xB7 Subscription (probe ${s.reason}, soft-pass) \xB7 model=${O}`),{ok:!0,provider:"subscription",model:O};if(!s.ok){let a=s.reason??"unknown";return await b("ai_setup",`provider=subscription,result=no-quota,reason=${a}`),i.warn(`Subscription verify th\u1EA5t b\u1EA1i (${a}).`),s.detail?.trim()&&i.warn(` Chi ti\u1EBFt: ${s.detail.slice(0,200)}`),i.warn(` \u2192 ${Or(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await rt(t.workspacePath,{provider:"subscription",model:O}),await b("ai_setup","provider=subscription,result=ok"),i.success(`AI ready \xB7 Subscription \xB7 model=${O}`),{ok:!0,provider:"subscription",model:O}}case"llmlite":{let o=await Vr();return await rt(t.workspacePath,{provider:"llmlite",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:!1}),await b("ai_setup",`provider=llmlite,result=ok,model=${o.model},base=${o.baseUrl},storage=settings.json`),i.success(`AI ready \xB7 LLMLite (NAL) \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"llmlite",model:o.model}}case"anthropic":{let o=await Kr();return await rt(t.workspacePath,{provider:"anthropic",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:!1}),await b("ai_setup",`provider=anthropic,result=ok,model=${o.model},storage=settings.json`),i.success(`AI ready \xB7 Anthropic Direct \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"anthropic",model:o.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 b("ai_setup","provider=use-global,result=ok"),i.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 i.warn(`AI setup th\u1EA5t b\u1EA1i: ${n}`),i.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup AI sau qua: avatar ai setup"),await b("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}f();import{spawnSync as qa}from"child_process";f();var ln=1e4,Wr=3e4,zr=5,un="say ok",qr="2023-06-01";async function za(t,e,n){i.info(`Testing LLMLite provider: ${t} (key: ${H(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),ln);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(i.success(`Connectivity OK \xB7 ${c.length} models available`),c.length>0){let h=c.slice(0,5).join(", "),v=c.length>5?` ...+${c.length-5} more`:"";i.dim(` Models: ${h}${v}`)}i.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:un}],max_tokens:zr}),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 p=await l.json(),d=typeof p.choices?.[0]?.message?.content=="string"?p.choices[0].message.content:"(empty response)",g=p.usage?.total_tokens??"?";i.success(`Response: "${String(d).trim().slice(0,100)}"`),i.dim(` Tokens used: ${g}`)}catch(s){throw s.name==="AbortError"?new Error(`Timeout ${ln/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):s}finally{clearTimeout(o)}}function Ja(){i.info("Testing Subscription provider qua `claude --print`...");let t=qa("claude",["--print",un],{encoding:"utf8",timeout:Wr});if(t.signal==="SIGTERM")throw new Error(`Timeout ${Wr/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)}`)}i.success(`Response: "${(t.stdout||"").trim().slice(0,100)}"`)}async function Ya(t,e,n){i.info(`Testing Anthropic Direct provider: ${t} (key: ${H(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),ln);try{let s=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":qr},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);i.success(`Connectivity OK \xB7 ${c.length} models available`),i.info(`Testing message v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/messages`,{method:"POST",headers:{"x-api-key":e,"anthropic-version":qr,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:zr,messages:[{role:"user",content:un}]}),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 d=((await l.json()).content||[]).map(g=>typeof g.text=="string"?g.text:"").join("").trim().slice(0,100);i.success(`Response: "${d}"`)}finally{clearTimeout(o)}}async function Jr(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,o=typeof e.ANTHROPIC_API_KEY=="string"?e.ANTHROPIC_API_KEY:void 0,s=typeof t.model=="string"?t.model:"default";return o&&n?(await Ya(n,o,s),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&r?(await za(n,r,s),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(Ja(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function Zt(){let t=process.cwd(),e=gt(t);return e||(i.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
36
+ \u2022 settings.json.env.ANTHROPIC_BASE_URL b\u1ECB thi\u1EBFu (provider ch\u01B0a configured)`),s.dim(`\u0110\u1EC3 gen wiki sau khi \u0111\xE3 c\xF3 key, ch\u1EA1y manual:
37
+ gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await fa(n.baseUrl,n.model))return s.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 o=Pr(n.model);await $r({apiKey:n.apiKey,baseUrl:n.baseUrl,model:n.model,isReasoningModel:o});let i=["wiki",".","--base-url",n.baseUrl,"--model",n.model];o&&i.push("--reasoning-model");let a=St(`Generating wiki via ${n.baseUrl} (${n.provider}) model=${n.model}${o?" [reasoning]":""}`),c=aa("gitnexus",i,{cwd:e,stdio:["ignore","pipe","pipe"],timeout:ua,encoding:"utf8"});if(c.status!==0||c.signal==="SIGTERM"){let p=c.signal==="SIGTERM"?"timeout":"non-zero-exit";a.fail(`Wiki gen ${p} (exit ${c.status??"null"})`);let d=(c.stderr||"").trim(),g=(c.stdout||"").trim();return d?process.stderr.write(`${Rr(d,30)}
38
+ `):g&&process.stderr.write(`${Rr(g,30)}
39
+ `),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${p} (exit ${c.status??"null"})`}}let l=_r(e,".gitnexus","wiki","index.html");return ca(l)?(a.succeed(`Wiki ready: ${l}`),{ran:!0,skipped:!1,wikiPath:l}):(a.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 ${l}`})}f();function Or(t){t.command("add").description("Th\xEAm t\xE0i nguy\xEAn v\xE0o workspace").command("repo").description("Clone 1 repo code v\xE0o src/ (repo c\xF3 s\u1EB5n / th\u01B0 m\u1EE5c / d\u1EF1 \xE1n m\u1EDBi)").option("--url <url>","URL git repo c\xF3 s\u1EB5n (b\u1ECF qua prompt ngu\u1ED3n)").option("--name <name>","T\xEAn th\u01B0 m\u1EE5c trong src/ (m\u1EB7c \u0111\u1ECBnh suy t\u1EEB URL)").option("--yes","Auto-confirm prompt").action(async n=>{try{await ba(n)}catch(r){s.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}function ya(){let t=ht(process.cwd());return t||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
40
+ C\u1EA7n .claude/ + CLAUDE.md + src/. Ch\u1EA1y 'avatar init' tr\u01B0\u1EDBc.`),process.exit(1)),t}async function ba(t){let e=ya(),n=!0,r=t.url,o=t.name;for(;n;){let i=await va(r),a=o??await Ca(i);s.info(`Clone ${i} \u2192 src/${a} ...`);let{cloned:c,skipped:l}=await ar({workspaceRoot:e,url:i,name:a});if(l||!c?s.dim(`B\u1ECF qua repo ${a}.`):(s.success(`\u2713 \u0110\xE3 clone src/${c.name}`),await Aa(e,c.path,t.yes),s.success(`\u2713 Done repo: ${c.name}`)),r=void 0,o=void 0,t.yes)break;n=await Wt({message:"Add repo kh\xE1c n\u1EEFa?",default:!1})}}async function va(t){if(t)return t;let e=await Ze({message:"Ngu\u1ED3n repo?",choices:[{name:"1. Repo c\xF3 s\u1EB5n (\u0111i\u1EC1n URL git)",value:"url"},{name:"2. Th\u01B0 m\u1EE5c \u0111\xE3 c\xF3 tr\xEAn m\xE1y (clone t\u1EEB remote c\u1EE7a n\xF3)",value:"folder"},{name:"3. D\u1EF1 \xE1n m\u1EDBi ho\xE0n to\xE0n (t\u1EA1o repo GitHub)",value:"new"}]});if(e==="url")return await $t({message:"URL git repo:",validate:i=>i.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"});if(e==="folder"){let i=ka(await $t({message:"\u0110\u01B0\u1EDDng d\u1EABn folder:",validate:l=>l.trim().length>0?!0:"Path b\u1EAFt bu\u1ED9c"})),a=await De(i);if(a&&(s.warn(`\u26A0 Folder c\xF3 thay \u0111\u1ED5i ch\u01B0a commit: ${a.slice(0,120)}`),s.warn(" Clone l\u1EA5y b\u1EA3n REMOTE \u2192 c\xE1c thay \u0111\u1ED5i local n\xE0y s\u1EBD KH\xD4NG c\xF3 trong src/<name>."),!await Wt({message:"Ti\u1EBFp t\u1EE5c clone t\u1EEB remote (b\u1ECF thay \u0111\u1ED5i local ch\u01B0a push)?",default:!1})))throw new Error("H\u1EE7y. Commit + push thay \u0111\u1ED5i trong folder g\u1ED1c r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar add repo'.");let c=await He(i);return c||await xa(i)}await Ye();let n=await $t({message:"T\xEAn repo m\u1EDBi:",validate:i=>i.trim().length>0?!0:"T\xEAn b\u1EAFt bu\u1ED9c"}),r=await Ze({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),{createEmptyGithubRepoAndGetUrl:o}=await Promise.resolve().then(()=>(Nr(),Ir));return await o(n.trim(),r)}async function xa(t){let{basename:e}=await import("path");if(s.warn(`Folder ${t} ch\u01B0a c\xF3 remote origin.`),!await Wt({message:"T\u1EA1o GitHub repo cho folder n\xE0y + push code l\xEAn?",default:!0}))throw new Error("H\u1EE7y. C\u1EA7n remote \u0111\u1EC3 add repo. T\u1EF1 t\u1EA1o remote (gh repo create) r\u1ED3i ch\u1EA1y l\u1EA1i, ho\u1EB7c ch\u1ECDn ngu\u1ED3n kh\xE1c.");await Ye();let{git:r}=await Promise.resolve().then(()=>($e(),Hn)),{pathExists:o}=await Promise.resolve().then(()=>(k(),Dn)),i=r(t);await o(`${t}/.git`)||(s.info("Folder ch\u01B0a git init \u2014 \u0111ang init..."),await i.init()),s.info("Commit code hi\u1EC7n t\u1EA1i trong folder (baseline cho push)..."),await i.add("."),await i.commit("chore: initial commit (avatar add repo)").catch(()=>{});let a=await $t({message:"T\xEAn repo GitHub:",default:e(t)}),c=await Ze({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),{createGithubRemoteFromFolder:l}=await Promise.resolve().then(()=>(ze(),fr)),p=l({folder:t,name:a.trim(),visibility:c});return s.success(`\u2713 \u0110\xE3 t\u1EA1o remote + push: ${p.sshUrl}`),p.sshUrl}async function Ca(t){let e=je(t);return await $t({message:"T\xEAn th\u01B0 m\u1EE5c trong src/:",default:e})}async function Aa(t,e,n){if(!nt().installed){s.dim("GitNexus ch\u01B0a c\xE0i \u2014 skip index. C\xE0i qua 'avatar gitnexus install'.");return}if(n||await Wt({message:"Index repo n\xE0y b\u1EB1ng GitNexus?",default:!0})){try{wt(e),s.success(` \u2713 GitNexus indexed src/${e.split("/").pop()}`)}catch(i){s.warn(` ! GitNexus index fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${i instanceof Error?i.message:i}`);return}try{let i=await Bt(t,e);i.ran?s.success(` \u2713 GitNexus wiki t\u1EA1o cho src/${e.split("/").pop()}`):i.reason==="subscription-mode"&&s.dim(" (Subscription mode \u2014 kh\xF4ng c\xF3 API key cho wiki, skip.)")}catch(i){s.warn(` ! Wiki fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${i instanceof Error?i.message:i}`)}}}k();import{promises as vc}from"fs";import{join as io}from"path";import{confirm as xc}from"@inquirer/prompts";import{promises as Hr}from"fs";import{homedir as _a}from"os";import{join as _t}from"path";import{z as w}from"zod";var Mr=w.object({email:w.string().email(),name:w.string(),access_token:w.string().min(1),refresh_token:w.string().min(1),expires_at:w.string().datetime(),id_token:w.string().min(1)}),Sa=w.object({installed_tools:w.record(w.string(),w.object({version:w.string().optional(),installed_at:w.string().datetime(),install_method:w.string()})).default({}),tool_inputs:w.record(w.string(),w.unknown()).default({})}),id=w.object({$schema:w.string().optional(),includeCoAuthoredBy:w.boolean().optional(),env:w.record(w.string(),w.string()).default({}),permissions:w.object({allow:w.array(w.string()).default([]),deny:w.array(w.string()).default([])}).partial().optional(),hooks:w.record(w.string(),w.array(w.unknown())).optional(),statusLine:w.object({type:w.string(),command:w.string(),padding:w.number().optional()}).optional()}),sd=w.enum(["internal","client","library"]);k();var It=_t(_a(),".avatar"),rt=_t(It,"config.json"),md=_t(It,"state.json"),ln=_t(It,"audit.log"),dd=_t(It,"backups"),Ia=384;async function un(){await v(It)}async function H(){if(!await m(rt))return null;let t=await y(rt),e=Mr.safeParse(t);return e.success?e.data:null}async function pn(t){await un(),await C(rt,t,Ia)}async function Dr(){if(await m(rt)){let{promises:t}=await import("fs");await t.unlink(rt)}}function ot(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}var Rt=class extends Error{constructor(e){super(e),this.name="NoValidTokenError"}};async function Na(t){let{decodeIdToken:e}=await Promise.resolve().then(()=>(Jt(),cn));try{let n=e(t),r=Math.floor(Date.now()/1e3);return n.exp-60<r}catch{return!0}}async function Yt(){let t=await H();if(!t)throw new Rt("Ch\u01B0a \u0111\u0103ng nh\u1EADp. Ch\u1EA1y 'avatar login' tr\u01B0\u1EDBc.");let{refreshAccessToken:e,decodeIdToken:n,verifyIdTokenClaims:r}=await Promise.resolve().then(()=>(Jt(),cn));if(!await Na(t.id_token))return r(n(t.id_token)),t.id_token;let o;try{o=await e(t.refresh_token)}catch(a){throw new Rt(`Token h\u1EBFt h\u1EA1n v\xE0 refresh th\u1EA5t b\u1EA1i (${a instanceof Error?a.message:a}). Ch\u1EA1y 'avatar login' l\u1EA1i.`)}if(!o.id_token)throw new Rt("Refresh kh\xF4ng tr\u1EA3 id_token m\u1EDBi (id_token c\u0169 \u0111\xE3 h\u1EBFt h\u1EA1n). Ch\u1EA1y 'avatar login' l\u1EA1i.");r(n(o.id_token));let i={...t,access_token:o.access_token,id_token:o.id_token,expires_at:new Date(Date.now()+o.expires_in*1e3).toISOString()};return await pn(i),o.id_token}async function Oa(){try{await Hr.chmod(ln,384)}catch{}}async function b(t,e){await un();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
41
+ `;await Hr.appendFile(ln,r,{encoding:"utf8",mode:384}),await Oa()}f();import{spawnSync as mn}from"child_process";var Kr=6e4,Ma="ok";function dn(){let t=mn("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 gn(){s.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=mn("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.`);s.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp Claude Code")}function La(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 Fr(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 fn(){let t=mn("claude",["--print",Ma],{encoding:"utf8",timeout:Kr,stdio:["ignore","pipe","pipe"]});if(t.signal==="SIGTERM"||t.status===143||t.error?.code==="ETIMEDOUT")return{ok:!1,reason:"timeout",detail:`claude --print > ${Kr/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 o=r.trim(),i=n.toLowerCase();if(o.length>20&&!i.includes("error")&&!i.includes("limit")&&!i.includes("quota")&&!i.includes("401"))return s.warn(`claude --print exit=${t.status} nh\u01B0ng c\xF3 response (${o.length} chars). Accept v\u1EDBi caution.`),{ok:!0};let a=La(`${n}
42
+ ${r}`);return a==="unknown"&&(s.warn(`[debug] claude --print exit=${t.status} signal=${t.signal??"none"}`),n.trim()&&s.warn(`[debug] stderr: ${n.slice(0,500)}`),r.trim()&&s.warn(`[debug] stdout: ${r.slice(0,300)}`)),{ok:!1,reason:a,detail:n.slice(0,500)||r.slice(0,500)}}import{spawnSync as Vr}from"child_process";var Ga=5e3,ja=/(\d+\.\d+\.\d+)/;function Ua(){let e=tt()==="win32"?"where":"which",n=Vr(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 Da(){let t=Vr("claude",["--version"],{encoding:"utf8",timeout:Ga});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return ja.exec(e)?.[1]??null}var it=null;function Nt(){if(it!==null)return it;let t=Ua();return t?(it={installed:!0,version:Da(),path:t},it):(it={installed:!1,version:null,path:null},it)}function Xt(){it=null}import{spawnSync as Ha}from"child_process";f();var Br=300*1e3,Wr="@anthropic-ai/claude-code",st=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallClaudeCodeError",this.reason=e,this.exitCode=r}};function Ka(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new st("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${Wr} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new st("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new st("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function qr(){s.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Ha("npm",["install","-g",Wr],{stdio:["inherit","inherit","pipe"],timeout:Br,encoding:"utf8"});if(t.signal==="SIGTERM")throw new st("timeout",`npm install timeout sau ${Br/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),Ka(t.status,t.stderr||"");Xt();let e=Nt();if(!e.installed||!e.path)throw new st("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 s.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 Fa}from"fs";import{homedir as Va}from"os";import{join as Ba}from"path";import{select as zr}from"@inquirer/prompts";function Wa(){return Ba(Va(),".claude","settings.json")}function hn(){let t=Wa(),e;try{e=Fa(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||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:void 0,i=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:!!o,baseUrl:o,hasToken:i,model:a,rawSettings:n}}async function Jr(t=hn()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await zr({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 zr({message:"Ch\u1ECDn provider cho AI tools:",choices:[{name:"1. Claude Code Subscription (d\xF9ng quota c\xE1 nh\xE2n Anthropic, OAuth login)",value:"subscription"},{name:"2. LLM key NAL (ai.nal.vn \u2014 gateway n\u1ED9i b\u1ED9, key sk-...)",value:"llmlite"},{name:"3. Anthropic API key tr\u1EF1c ti\u1EBFp (console.anthropic.com, key sk-ant-...)",value:"anthropic"}]})}f();import{password as qa,select as za}from"@inquirer/prompts";var Qt="https://api.anthropic.com",Ja="2023-06-01",Yr=1e4;function Ya(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function Xa(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 Qa(){return await qa({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:Xa})}async function Za(t){let e=new AbortController,n=setTimeout(()=>e.abort(),Yr);try{let r=await fetch(`${Qt}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":Ja,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 i=((await r.json()).data||[]).map(a=>typeof a.id=="string"?a.id:null).filter(a=>a!==null);if(i.length===0)throw new Error("Anthropic tr\u1EA3 v\u1EC1 list r\u1ED7ng. Li\xEAn h\u1EC7 support ho\u1EB7c check account.");return i}catch(r){throw r.name==="AbortError"?new Error(`Connect ${Qt} timeout sau ${Yr/1e3}s.`):r}finally{clearTimeout(n)}}async function tc(t){if(t.length===1){let n=t[0];return s.info(`Auto-pick model: ${n} (ch\u1EC9 1 model available)`),n}let e=[...t].sort((n,r)=>{let o=i=>{let a=i.toLowerCase();return a.includes("sonnet")?0:a.includes("opus")?1:a.includes("haiku")?2:3};return o(n)-o(r)});return await za({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function Xr(){let t=await Qa();s.info(`Verify key (${Ya(t)}) qua ${Qt}/v1/models...`);let e=await Za(t);s.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await tc(e);return{apiKey:t,baseUrl:Qt,model:n}}f();import{input as ec,password as nc,select as rc}from"@inquirer/prompts";var oc="https://ai.nal.vn",Qr=1e4;function K(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function ic(){return await nc({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function sc(t=oc){return(await ec({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 ac(t,e){let n=new AbortController,r=setTimeout(()=>n.abort(),Qr);try{let o=await fetch(`${t}/v1/models`,{method:"GET",headers:{Authorization:`Bearer ${e}`,Accept:"application/json"},signal:n.signal});if(o.status===401||o.status===403)throw new Error(`API key invalid (HTTP ${o.status}).`);if(o.status===404)throw new Error(`Endpoint /v1/models kh\xF4ng t\u1ED3n t\u1EA1i tr\xEAn ${t}.`);if(!o.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${o.status}).`);let a=((await o.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(o){if(o.name==="AbortError"){let c=t.includes("nal.vn")||t.includes("nal-vn")?"\n Hint: ai.nal.vn l\xE0 internal endpoint NAL \u2014 ki\u1EC3m tra VPN NAL \u0111\xE3 b\u1EADt ch\u01B0a, ho\u1EB7c tcp test b\u1EB1ng `curl -v https://ai.nal.vn`.":`
43
+ Hint: check m\u1EA1ng / firewall / VPN, ho\u1EB7c base URL c\xF3 \u0111\xFAng kh\xF4ng.`;throw new Error(`Connect ${t} timeout sau ${Qr/1e3}s.${c}`)}let i=o.message||String(o);if(i.toLowerCase().includes("fetch failed")||i.includes("ENOTFOUND")){let c=t.includes("nal.vn")?" (ai.nal.vn c\u1EA7n VPN NAL \u2014 ki\u1EC3m tra VPN \u0111\xE3 b\u1EADt ch\u01B0a)":"";throw new Error(`Network error khi connect ${t}: ${i}${c}`)}throw o}finally{clearTimeout(r)}}async function cc(t){let e=t.filter(r=>r.toLowerCase().includes("claude"));if(e.length===1){let r=e[0];return s.info(`Auto-pick model: ${r} (ch\u1EC9 1 claude alias tr\xEAn endpoint)`),r}let n=e.length>0?e:t;return await rc({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function Zr(){let t=await ic(),e=await sc();s.info(`Verify key (${K(t)}) qua ${e}/v1/models...`);let n=await ac(e,t);s.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await cc(n);return{apiKey:t,baseUrl:e,model:r}}f();k();import{promises as lc}from"fs";import{join as uc}from"path";var wn=384;function pc(t){return uc(t,".claude","settings.json")}async function mc(t){if(!await m(t))return{};try{return await y(t)}catch(e){throw new Error(`Kh\xF4ng parse \u0111\u01B0\u1EE3c ${t} (JSON l\u1ED7i): ${e.message}. Backup file r\u1ED3i x\xF3a \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}function dc(t,e){let{env:n,...r}=t,o={...r,model:e};if(n){let{ANTHROPIC_BASE_URL:i,ANTHROPIC_AUTH_TOKEN:a,ANTHROPIC_API_KEY:c,...l}=n;Object.keys(l).length>0&&(o.env=l)}return o}function to(t){if(!t)return t;let{ANTHROPIC_AUTH_TOKEN:e,ANTHROPIC_API_KEY:n,...r}=t;return r}function gc(t,e,n,r,o){let a={...to(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_AUTH_TOKEN=e),{...t,env:a,model:r,avatarProvider:"llmlite"}}function fc(t,e,n,r,o){let a={...to(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_API_KEY=e),{...t,env:a,model:r,avatarProvider:"anthropic"}}function hc(t,e){let n=e.env||{},r=typeof e.model=="string"?e.model:void 0;return{...t,env:{...t.env||{},...n},...r?{model:r}:{}}}async function at(t,e){let n=pc(t),r=await mc(n),o;switch(e.provider){case"subscription":o=dc(r,e.model);break;case"llmlite":o=gc(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"anthropic":o=fc(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"use-global":o=hc(r,e.sourceSettings);break}await C(n,o,wn);try{await lc.chmod(n,wn)}catch{}return{path:n,mode:wn}}var L="sonnet";async function Zt(t){try{s.info("Setup AI provider cho workspace...");let e=Nt();if(e.installed)s.success(`Claude Code \u0111\xE3 c\xF3${e.version?` v${e.version}`:""}`);else if(s.info("Ch\u01B0a c\xF3 Claude Code \u2014 s\u1EBD t\u1EF1 c\xE0i qua npm."),qr(),Xt(),e=Nt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=hn();switch(await Jr(n)){case"subscription":{let o=dn();if(o.state!=="authenticated"&&(gn(),o=dn()),o.state==="authenticated"&&o.subscriptionType)return await at(t.workspacePath,{provider:"subscription",model:L}),await b("ai_setup",`provider=subscription,result=ok,plan=${o.subscriptionType},probe=skipped`),s.success(`AI ready \xB7 Subscription (${o.subscriptionType}) \xB7 model=${L}`),{ok:!0,provider:"subscription",model:L};s.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let i=fn();if(!i.ok&&i.reason==="auth-expired"&&(s.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),gn(),i=fn()),!i.ok&&(i.reason==="timeout"||i.reason==="unknown"))return s.warn(`Probe verify ${i.reason} \u2014 accept trust auth status. Ti\u1EBFp t\u1EE5c.`),i.detail?.trim()&&s.warn(` Chi ti\u1EBFt: ${i.detail.slice(0,200)}`),await at(t.workspacePath,{provider:"subscription",model:L}),await b("ai_setup",`provider=subscription,result=ok,probe=${i.reason}-soft-pass`),s.success(`AI ready \xB7 Subscription (probe ${i.reason}, soft-pass) \xB7 model=${L}`),{ok:!0,provider:"subscription",model:L};if(!i.ok){let a=i.reason??"unknown";return await b("ai_setup",`provider=subscription,result=no-quota,reason=${a}`),s.warn(`Subscription verify th\u1EA5t b\u1EA1i (${a}).`),i.detail?.trim()&&s.warn(` Chi ti\u1EBFt: ${i.detail.slice(0,200)}`),s.warn(` \u2192 ${Fr(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await at(t.workspacePath,{provider:"subscription",model:L}),await b("ai_setup","provider=subscription,result=ok"),s.success(`AI ready \xB7 Subscription \xB7 model=${L}`),{ok:!0,provider:"subscription",model:L}}case"llmlite":{let o=await Zr();return await at(t.workspacePath,{provider:"llmlite",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:!1}),await b("ai_setup",`provider=llmlite,result=ok,model=${o.model},base=${o.baseUrl},storage=settings.json`),s.success(`AI ready \xB7 LLMLite (NAL) \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"llmlite",model:o.model}}case"anthropic":{let o=await Xr();return await at(t.workspacePath,{provider:"anthropic",apiKey:o.apiKey,baseUrl:o.baseUrl,model:o.model,skipApiKey:!1}),await b("ai_setup",`provider=anthropic,result=ok,model=${o.model},storage=settings.json`),s.success(`AI ready \xB7 Anthropic Direct \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"anthropic",model:o.model}}case"use-global":{if(!n.rawSettings)throw new Error("use-global ch\u1ECDn nh\u01B0ng kh\xF4ng \u0111\u1ECDc \u0111\u01B0\u1EE3c global settings.");return await at(t.workspacePath,{provider:"use-global",sourceSettings:n.rawSettings}),await b("ai_setup","provider=use-global,result=ok"),s.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 s.warn(`AI setup th\u1EA5t b\u1EA1i: ${n}`),s.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup AI sau qua: avatar ai setup"),await b("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}f();import{spawnSync as wc}from"child_process";f();var kn=1e4,eo=3e4,ro=5,yn="say ok",no="2023-06-01";async function kc(t,e,n){s.info(`Testing LLMLite provider: ${t} (key: ${K(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),kn);try{let i=await fetch(`${t}/v1/models`,{headers:{Authorization:`Bearer ${e}`},signal:r.signal});if(i.status===401||i.status===403)throw new Error(`API key invalid (HTTP ${i.status}). Re-run: avatar ai setup`);if(!i.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${i.status}).`);let c=((await i.json()).data||[]).map(h=>typeof h.id=="string"?h.id:null).filter(h=>h!==null);if(s.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`:"";s.dim(` Models: ${h}${x}`)}s.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:yn}],max_tokens:ro}),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 p=await l.json(),d=typeof p.choices?.[0]?.message?.content=="string"?p.choices[0].message.content:"(empty response)",g=p.usage?.total_tokens??"?";s.success(`Response: "${String(d).trim().slice(0,100)}"`),s.dim(` Tokens used: ${g}`)}catch(i){throw i.name==="AbortError"?new Error(`Timeout ${kn/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):i}finally{clearTimeout(o)}}function yc(){s.info("Testing Subscription provider qua `claude --print`...");let t=wc("claude",["--print",yn],{encoding:"utf8",timeout:eo});if(t.signal==="SIGTERM")throw new Error(`Timeout ${eo/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)}`)}s.success(`Response: "${(t.stdout||"").trim().slice(0,100)}"`)}async function bc(t,e,n){s.info(`Testing Anthropic Direct provider: ${t} (key: ${K(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),kn);try{let i=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":no},signal:r.signal});if(i.status===401||i.status===403)throw new Error(`API key invalid (HTTP ${i.status}). Re-run: avatar ai setup`);if(!i.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${i.status}).`);let c=((await i.json()).data||[]).map(g=>typeof g.id=="string"?g.id:null).filter(g=>g!==null);s.success(`Connectivity OK \xB7 ${c.length} models available`),s.info(`Testing message v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/messages`,{method:"POST",headers:{"x-api-key":e,"anthropic-version":no,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:ro,messages:[{role:"user",content:yn}]}),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 d=((await l.json()).content||[]).map(g=>typeof g.text=="string"?g.text:"").join("").trim().slice(0,100);s.success(`Response: "${d}"`)}finally{clearTimeout(o)}}async function oo(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,o=typeof e.ANTHROPIC_API_KEY=="string"?e.ANTHROPIC_API_KEY:void 0,i=typeof t.model=="string"?t.model:"default";return o&&n?(await bc(n,o,i),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&r?(await kc(n,r,i),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(yc(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function te(){let t=process.cwd(),e=ht(t);return e||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
44
44
  Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
45
45
  B\u1EA1n \u0111ang \u1EDF: ${t}
46
- Cd v\xE0o workspace dir (vd /path/to/<project>-workspace) r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&i.dim(`Detected workspace root: ${e}`),e}async function pn(t){let e=Yr(t,".claude","settings.json");if(!await m(e))return{};try{return await y(e)}catch{return{}}}async function Za(){let t=await Zt();await Qt({workspacePath:t})}async function tc(){let t=await Zt(),e=await pn(t),n=e.env||{},r=typeof n.ANTHROPIC_BASE_URL=="string"?n.ANTHROPIC_BASE_URL:void 0,o=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=H(s)):r&&o?(c="LLMLite",l=H(o)):o?(c="Custom",l=H(o)):(c="Subscription (default)",l="(kh\xF4ng set \u2014 d\xF9ng subscription auth)"),i.info(`Project: ${t}`),i.info(`Provider: ${c}${r?` (${r})`:""}`),i.info(`Model: ${a??"(default \u2014 Claude Code ch\u1ECDn)"}`),i.info(`Token: ${l}`)}async function ec(){let t=await Zt(),e=await pn(t);try{let n=await Jr(e);i.success(`\u2713 ${n.message}`)}catch(n){i.error(`Test fail: ${n.message}`),process.exit(1)}}async function nc(t){let e=await Zt(),n=Yr(e,".claude","settings.json"),r=await pn(e);if(!t.yes&&!await Qa({message:"X\xF3a AI config (v\u1EC1 d\xF9ng Claude Code Subscription default)?",default:!1})){i.dim("\u0110\xE3 h\u1EE7y.");return}let{env:o,...s}=r,a={...s};if(o){let{ANTHROPIC_BASE_URL:c,ANTHROPIC_AUTH_TOKEN:l,ANTHROPIC_API_KEY:p,...d}=o;Object.keys(d).length>0&&(a.env=d)}Object.keys(a).length===0?(await Xa.unlink(n).catch(()=>{}),i.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await S(n,a,384),i.success("\u0110\xE3 reset env block trong .claude/settings.json"))}function Xr(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 Za()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await tc()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await ec()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await nc(n)})}import{spawnSync as gn}from"child_process";import{promises as uo}from"fs";import{join as Nt}from"path";import wc from"boxen";import{join as fc}from"path";k();import{promises as cc}from"fs";import{join as no}from"path";k();f();import{promises as rc}from"fs";import Qr,{join as mn}from"path";async function oc(t,e){let r=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!r?.[2])return!0;let o=r[2];if(o.startsWith("/"))return i.warn(`Pack hook reject: absolute path "${o}" \u2014 ch\u1EC9 accept relative path t\u1EDBi workspace.`),!1;let s=mn(t,o),a=Qr.resolve(t),c=Qr.resolve(s);return!c.startsWith(`${a}/`)&&c!==a?(i.warn(`Pack hook reject: path "${o}" resolve ra ngo\xE0i workspace (path traversal).`),!1):await m(s)}function dn(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 _t(t,e){let n=new Set,r=[];for(let o of[...t,...e]){let s=typeof o=="string"?o:JSON.stringify(o);n.has(s)||(n.add(s),r.push(o))}return r}function ic(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function sc(t){if(typeof t!="object"||t===null)return[];let e=t,n=Array.isArray(e.hooks)?e.hooks:[],r=[];for(let o of n)if(typeof o=="object"&&o!==null){let s=o.command;typeof s=="string"&&r.push(s)}return r}function Zr(t){let e=t.map((s,a)=>{let c=sc(s),l=c.some(p=>p.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(ic).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 ac(t,e){let n=[],r={...e},o=0;for(let s of Object.keys(r)){let a=r[s]||[],{entries:c,droppedCount:l}=Zr(a);l>0&&(r[s]=c,o+=l,n.includes(s)||n.push(s))}for(let[s,a]of Object.entries(t)){let c=r[s]||[],l=_t(c,a),{entries:p,droppedCount:d}=Zr(l);d>0&&(o+=d),p.length!==c.length&&(n.includes(s)||n.push(s)),r[s]=p}return{merged:r,touchedEvents:n,migratedCount:o}}async function te(t){let e=mn(t,".claude","pack","templates","settings.json.tpl"),n=mn(t,".claude","settings.json");if(!await m(e))return{action:"no-pack-template",changes:[]};let r;try{let p=await I(e);r=JSON.parse(p)}catch(p){throw new Error(`Pack settings template kh\xF4ng parse \u0111\u01B0\u1EE3c JSON: ${p.message}. Path: ${e}`)}let o={},s=!1;if(await m(n)){s=!0;try{o=await y(n)}catch(p){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${p.message}. Manual fix tr\u01B0\u1EDBc khi sync.`)}}let a=[],c={...o};if(r.statusLine&&!o.statusLine&&(await oc(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 o.includeCoAuthoredBy!="boolean"&&(c.includeCoAuthoredBy=r.includeCoAuthoredBy,a.push("includeCoAuthoredBy added")),r.model&&!o.model&&(c.model=r.model,a.push("model added")),r.env||o.env){let p={...o.env||{}},d=!1;for(let[g,h]of Object.entries(p))typeof h=="string"&&h.includes("llm.nal.vn")&&(p[g]=h.replace(/llm\.nal\.vn/g,"ai.nal.vn"),d=!0);if(d&&a.push("rewrote legacy llm.nal.vn \u2192 ai.nal.vn in env vars"),r.env)for(let[g,h]of Object.entries(r.env))g in p||(p[g]=h,d=!0);d&&(c.env=p,a.some(g=>g.includes("env vars"))||a.push("env vars added from pack"))}if(r.permissions){let p=o.permissions?.allow||[],d=o.permissions?.deny||[],g=r.permissions.allow||[],h=r.permissions.deny||[],v=_t(p,g),A=_t(d,h);(v.length!==p.length||A.length!==d.length)&&(c.permissions={allow:v,deny:A},a.push(`permissions union (+${v.length-p.length} allow, +${A.length-d.length} deny)`))}if(r.hooks){let p=o.hooks||{},{merged:d,touchedEvents:g,migratedCount:h}=ac(r.hooks,p);(g.length>0||h>0)&&(c.hooks=d,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=dn(n),await rc.copyFile(n,l)),await S(n,c),{action:"merged",backupPath:l,changes:a}}function lc(t,e){return no(t,".claude","pack","tools",e,"tool.json")}async function It(t,e){let n=lc(t,e);return await m(n)?await y(n):null}var uc=[".claude","settings.json"];function ro(t){return no(t,...uc)}function ee(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function oo(t){let e=ro(t);if(!await m(e))return{settings:{},existed:!1};try{return{settings:await y(e),existed:!0}}catch(n){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${n.message}. Manual fix tr\u01B0\u1EDBc khi enable/disable tool.`)}}async function io(t,e,n){let r=ro(t),o;return n&&(o=dn(r),await cc.copyFile(r,o)),await S(r,e),o}function to(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function eo(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function pc(t,e){let n=new Set;for(let s of e)eo(s)&&n.add(to(s));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(s=>eo(s)?!0:n.has(to(s))?(r++,!1):!0),droppedCount:r}}async function ne(t,e){let{settings:n,existed:r}=await oo(t),o={...n},s=[],a=e.settings.hooks??{};if(Object.keys(a).length>0){let p=n.hooks??{},d={...p},g=[],h=0;for(let[v,A]of Object.entries(a)){let N=p[v]??[],{migratedUser:ct,droppedCount:$}=pc(N,A);$>0&&(h+=$);let E=new Set(ct.map(ee)),P=A.filter(z=>!E.has(ee(z)));(P.length>0||$>0)&&(d[v]=[...ct,...P],g.push(v))}g.length>0&&(o.hooks=d,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 p=n.permissions?.deny??[],d=_t(p,c);d.length!==p.length&&(o.permissions={...n.permissions,deny:d},s.push(`deny +${d.length-p.length}`))}return s.length===0?{action:"no-change",changes:[]}:{action:"enabled",backupPath:await io(t,o,r),changes:s}}async function so(t,e){let{settings:n,existed:r}=await oo(t);if(!r)return{action:"no-change",changes:[]};let{hooks:o,permissions:s,...a}=n,c={...a},l=[],p=e.settings.hooks??{},d=n.hooks??{},g={},h=[];for(let[$,E]of Object.entries(d)){let P=p[$];if(!P){g[$]=E;continue}let z=new Set(P.map(ee)),lt=E.filter(J=>!z.has(ee(J)));lt.length!==E.length&&h.push($),lt.length>0&&(g[$]=lt)}Object.keys(g).length>0&&(c.hooks=g),h.length>0&&l.push(`hooks removed: ${h.join(", ")}`);let v=new Set(e.settings.permissions?.deny??[]),A=n.permissions?.deny??[],N=v.size>0?A.filter($=>!v.has($)):A;if(n.permissions){let{deny:$,...E}=n.permissions,P={...E,...N.length>0?{deny:N}:{}};Object.keys(P).length>0&&(c.permissions=P)}return N.length!==A.length&&l.push(`deny -${A.length-N.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await io(t,c,r),changes:l}}k();k();import{join as mc}from"path";var dc=".claude/avatar-tools.json";function ao(){return{tools:{}}}function co(t){return mc(t,dc)}async function ht(t){let e=co(t);if(!await m(e))return ao();try{return{tools:(await y(e)).tools??{}}}catch{return ao()}}async function gc(t,e){await S(co(t),e)}async function re(t,e,n){let r=await ht(t);return r.tools[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await gc(t,r),r}async function wt(t){let e=await ht(t);return Object.entries(e.tools).filter(([,n])=>n.enabled).map(([n])=>n)}function hc(t,e){let n=e.settings.hooks??{},r=t.hooks??{};for(let[o,s]of Object.entries(n)){let a=r[o]??[],c=new Set(a.flatMap(l=>(l.hooks??[]).map(p=>p.command??"")));for(let l of s)for(let p of l.hooks??[])if(p.command&&!c.has(p.command))return!1}return!0}async function lo(t){let e=await wt(t);if(e.length===0)return[];let n=fc(t,".claude","settings.json"),r={};if(await m(n))try{r=await y(n)}catch{r={}}let o=[];for(let s of e){let a=await It(t,s);if(!a){o.push({name:`Tool: ${s}`,status:"warn",detail:"state=enabled nh\u01B0ng pack thi\u1EBFu manifest (version skew). Ch\u1EA1y 'avatar sync'.",fixable:!1});continue}hc(r,a)?o.push({name:`Tool: ${s}`,status:"ok",detail:`enabled, hook active (v${a.version})`,fixable:!1}):o.push({name:`Tool: ${s}`,status:"fail",detail:"state=enabled nh\u01B0ng hook thi\u1EBFu trong settings.json \u2014 fixable (re-apply)",fixable:!0,fix:async()=>{await ne(t,a)}})}return o}k();f();function po(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 kc(process.cwd());yc(n),e.fix&&await bc(n)}catch(n){i.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function kc(t){let e=[],n=process.versions.node,[r,o]=n.split(".").map(E=>Number.parseInt(E,10)),s=(r??0)>18||(r??0)===18&&(o??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 D();a?tt(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=Nt(t,".claude","pack"),l=Nt(t,"CLAUDE.md"),[p,d]=await Promise.all([m(c),m(l)]);if(e.push({name:"team-ai-pack installation",status:p?"ok":"warn",detail:p?c:"Avatar ch\u01B0a init \u2014 ch\u1EA1y 'avatar init'",fixable:!1}),e.push({name:"CLAUDE.md",status:d?"ok":"warn",detail:d?"t\u1ED3n t\u1EA1i \u1EDF project root":"thi\u1EBFu \u2014 ch\u1EA1y 'avatar init'",fixable:!1}),p){let E=Nt(t,".claude",".gitignore"),P=!1;await m(E)&&(P=(await uo.readFile(E,"utf8")).includes("settings.json")),e.push({name:"\u{1F512} settings.json gitignored (.claude/.gitignore)",status:P?"ok":"fail",detail:P?"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 b\u1EA3o v\u1EC7",fixable:!P,fix:P?void 0:async()=>{let{writeClaudeGitignore:z}=await Promise.resolve().then(()=>(xt(),Kn));await z(t)}})}let g=gn("which",["python"]),h=gn("which",["python3"]),v=g.status===0,A=h.status===0;A&&!v?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 ${h.stdout.toString().trim()} ~/.local/bin/python`,fixable:!1}):v?e.push({name:"Python binary",status:"ok",detail:`python: ${g.stdout.toString().trim()}`,fixable:!1}):A&&e.push({name:"Python binary",status:"ok",detail:`python3: ${h.stdout.toString().trim()}`,fixable:!1});let N=Nt(t,".claude","settings.json");if(await m(N))try{let E=await uo.readFile(N,"utf8"),P=JSON.parse(E);if(P.statusLine?.command){let lt=P.statusLine.command.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(lt?.[2]){let J=lt[2],vi=J.startsWith("/")?J:Nt(t,J),En=await m(vi);e.push({name:"statusLine command",status:En?"ok":"fail",detail:En?`ref OK: ${J}`:`BROKEN: settings.json ref '${J}' nh\u01B0ng file kh\xF4ng t\u1ED3n t\u1EA1i. Strip field statusLine ho\u1EB7c fix path.`,fixable:!1})}}}catch{}let ct=gn("which",["claude"]),$=ct.status===0;if(e.push({name:"Claude Code CLI",status:$?"ok":"warn",detail:$?ct.stdout.toString().trim():"kh\xF4ng t\xECm th\u1EA5y 'claude' tr\xEAn PATH",fixable:!1}),p){let E=await lo(t);e.push(...E)}return e}function yc(t){let e=[u.bold("Avatar Doctor"),"\u2500".repeat(48)],n=0,r=0,o=0;for(let s of t){let a=s.status==="ok"?u.green("\u2713"):s.status==="warn"?u.yellow("\u26A0"):u.red("\u2717");e.push(`${a} ${s.name.padEnd(28)} ${u.dim(s.detail)}`),s.status==="ok"?n+=1:(r+=1,s.fixable&&(o+=1))}e.push("\u2500".repeat(48)),e.push(`${n} checks passed, ${r} issue${r===1?"":"s"}${o>0?` (${o} fixable \u2014 ch\u1EA1y 'avatar doctor --fix')`:""}`),process.stdout.write(`${wc(e.join(`
46
+ Cd v\xE0o workspace dir (vd /path/to/<project>-workspace) r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&s.dim(`Detected workspace root: ${e}`),e}async function bn(t){let e=io(t,".claude","settings.json");if(!await m(e))return{};try{return await y(e)}catch{return{}}}async function Cc(){let t=await te();await Zt({workspacePath:t})}async function Ac(){let t=await te(),e=await bn(t),n=e.env||{},r=typeof n.ANTHROPIC_BASE_URL=="string"?n.ANTHROPIC_BASE_URL:void 0,o=typeof n.ANTHROPIC_AUTH_TOKEN=="string"?n.ANTHROPIC_AUTH_TOKEN:void 0,i=typeof n.ANTHROPIC_API_KEY=="string"?n.ANTHROPIC_API_KEY:void 0,a=typeof e.model=="string"?e.model:void 0,c,l;i?(c=r?.includes("api.anthropic.com")||i.startsWith("sk-ant-")?"Anthropic Direct":"Custom (API key)",l=K(i)):r&&o?(c="LLMLite",l=K(o)):o?(c="Custom",l=K(o)):(c="Subscription (default)",l="(kh\xF4ng set \u2014 d\xF9ng subscription auth)"),s.info(`Project: ${t}`),s.info(`Provider: ${c}${r?` (${r})`:""}`),s.info(`Model: ${a??"(default \u2014 Claude Code ch\u1ECDn)"}`),s.info(`Token: ${l}`)}async function Sc(){let t=await te(),e=await bn(t);try{let n=await oo(e);s.success(`\u2713 ${n.message}`)}catch(n){s.error(`Test fail: ${n.message}`),process.exit(1)}}async function Tc(t){let e=await te(),n=io(e,".claude","settings.json"),r=await bn(e);if(!t.yes&&!await xc({message:"X\xF3a AI config (v\u1EC1 d\xF9ng Claude Code Subscription default)?",default:!1})){s.dim("\u0110\xE3 h\u1EE7y.");return}let{env:o,...i}=r,a={...i};if(o){let{ANTHROPIC_BASE_URL:c,ANTHROPIC_AUTH_TOKEN:l,ANTHROPIC_API_KEY:p,...d}=o;Object.keys(d).length>0&&(a.env=d)}Object.keys(a).length===0?(await vc.unlink(n).catch(()=>{}),s.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await C(n,a,384),s.success("\u0110\xE3 reset env block trong .claude/settings.json"))}function so(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 Cc()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await Ac()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await Sc()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await Tc(n)})}import{spawnSync as Cn}from"child_process";import{promises as bo}from"fs";import{join as Lt}from"path";import Hc from"boxen";import{join as Uc}from"path";k();import{promises as Ic}from"fs";import{join as po}from"path";k();f();import{promises as Pc}from"fs";import ao,{join as vn}from"path";async function Ec(t,e){let r=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!r?.[2])return!0;let o=r[2];if(o.startsWith("/"))return s.warn(`Pack hook reject: absolute path "${o}" \u2014 ch\u1EC9 accept relative path t\u1EDBi workspace.`),!1;let i=vn(t,o),a=ao.resolve(t),c=ao.resolve(i);return!c.startsWith(`${a}/`)&&c!==a?(s.warn(`Pack hook reject: path "${o}" resolve ra ngo\xE0i workspace (path traversal).`),!1):await m(i)}function xn(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 Ot(t,e){let n=new Set,r=[];for(let o of[...t,...e]){let i=typeof o=="string"?o:JSON.stringify(o);n.has(i)||(n.add(i),r.push(o))}return r}function $c(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function Rc(t){if(typeof t!="object"||t===null)return[];let e=t,n=Array.isArray(e.hooks)?e.hooks:[],r=[];for(let o of n)if(typeof o=="object"&&o!==null){let i=o.command;typeof i=="string"&&r.push(i)}return r}function co(t){let e=t.map((i,a)=>{let c=Rc(i),l=c.some(p=>p.includes("${CLAUDE_PROJECT_DIR}"));return{index:a,entry:i,commands:c,hasVar:l}}),n=new Map;for(let i of e){let a=i.commands.map($c).sort().join("|");if(!a)continue;let c=n.get(a)??[];c.push(i),n.set(a,c)}let r=new Set;for(let[,i]of n)if(!(i.length<2||!i.some(c=>c.hasVar)))for(let c of i)c.hasVar||r.add(c.index);return r.size===0?{entries:t,droppedCount:0}:{entries:t.filter((i,a)=>!r.has(a)),droppedCount:r.size}}function _c(t,e){let n=[],r={...e},o=0;for(let i of Object.keys(r)){let a=r[i]||[],{entries:c,droppedCount:l}=co(a);l>0&&(r[i]=c,o+=l,n.includes(i)||n.push(i))}for(let[i,a]of Object.entries(t)){let c=r[i]||[],l=Ot(c,a),{entries:p,droppedCount:d}=co(l);d>0&&(o+=d),p.length!==c.length&&(n.includes(i)||n.push(i)),r[i]=p}return{merged:r,touchedEvents:n,migratedCount:o}}async function ee(t){let e=vn(t,".claude","pack","templates","settings.json.tpl"),n=vn(t,".claude","settings.json");if(!await m(e))return{action:"no-pack-template",changes:[]};let r;try{let p=await N(e);r=JSON.parse(p)}catch(p){throw new Error(`Pack settings template kh\xF4ng parse \u0111\u01B0\u1EE3c JSON: ${p.message}. Path: ${e}`)}let o={},i=!1;if(await m(n)){i=!0;try{o=await y(n)}catch(p){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${p.message}. Manual fix tr\u01B0\u1EDBc khi sync.`)}}let a=[],c={...o};if(r.statusLine&&!o.statusLine&&(await Ec(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 o.includeCoAuthoredBy!="boolean"&&(c.includeCoAuthoredBy=r.includeCoAuthoredBy,a.push("includeCoAuthoredBy added")),r.model&&!o.model&&(c.model=r.model,a.push("model added")),r.env||o.env){let p={...o.env||{}},d=!1;for(let[g,h]of Object.entries(p))typeof h=="string"&&h.includes("llm.nal.vn")&&(p[g]=h.replace(/llm\.nal\.vn/g,"ai.nal.vn"),d=!0);if(d&&a.push("rewrote legacy llm.nal.vn \u2192 ai.nal.vn in env vars"),r.env)for(let[g,h]of Object.entries(r.env))g in p||(p[g]=h,d=!0);d&&(c.env=p,a.some(g=>g.includes("env vars"))||a.push("env vars added from pack"))}if(r.permissions){let p=o.permissions?.allow||[],d=o.permissions?.deny||[],g=r.permissions.allow||[],h=r.permissions.deny||[],x=Ot(p,g),S=Ot(d,h);(x.length!==p.length||S.length!==d.length)&&(c.permissions={allow:x,deny:S},a.push(`permissions union (+${x.length-p.length} allow, +${S.length-d.length} deny)`))}if(r.hooks){let p=o.hooks||{},{merged:d,touchedEvents:g,migratedCount:h}=_c(r.hooks,p);(g.length>0||h>0)&&(c.hooks=d,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 i&&(l=xn(n),await Pc.copyFile(n,l)),await C(n,c),{action:"merged",backupPath:l,changes:a}}function Nc(t,e){return po(t,".claude","pack","tools",e,"tool.json")}async function Mt(t,e){let n=Nc(t,e);return await m(n)?await y(n):null}var Oc=[".claude","settings.json"];function mo(t){return po(t,...Oc)}function ne(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function go(t){let e=mo(t);if(!await m(e))return{settings:{},existed:!1};try{return{settings:await y(e),existed:!0}}catch(n){throw new Error(`Project settings.json kh\xF4ng parse \u0111\u01B0\u1EE3c: ${n.message}. Manual fix tr\u01B0\u1EDBc khi enable/disable tool.`)}}async function fo(t,e,n){let r=mo(t),o;return n&&(o=xn(r),await Ic.copyFile(r,o)),await C(r,e),o}function lo(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function uo(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function Mc(t,e){let n=new Set;for(let i of e)uo(i)&&n.add(lo(i));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(i=>uo(i)?!0:n.has(lo(i))?(r++,!1):!0),droppedCount:r}}async function re(t,e){let{settings:n,existed:r}=await go(t),o={...n},i=[],a=e.settings.hooks??{};if(Object.keys(a).length>0){let p=n.hooks??{},d={...p},g=[],h=0;for(let[x,S]of Object.entries(a)){let M=p[x]??[],{migratedUser:mt,droppedCount:R}=Mc(M,S);R>0&&(h+=R);let E=new Set(mt.map(ne)),T=S.filter(J=>!E.has(ne(J)));(T.length>0||R>0)&&(d[x]=[...mt,...T],g.push(x))}g.length>0&&(o.hooks=d,i.push(`hooks added: ${g.join(", ")}`),h>0&&i.push(`migrated ${h} stale relative-path entries \u2192 \\{CLAUDE_PROJECT_DIR\\} version`))}let c=e.settings.permissions?.deny??[];if(c.length>0){let p=n.permissions?.deny??[],d=Ot(p,c);d.length!==p.length&&(o.permissions={...n.permissions,deny:d},i.push(`deny +${d.length-p.length}`))}return i.length===0?{action:"no-change",changes:[]}:{action:"enabled",backupPath:await fo(t,o,r),changes:i}}async function ho(t,e){let{settings:n,existed:r}=await go(t);if(!r)return{action:"no-change",changes:[]};let{hooks:o,permissions:i,...a}=n,c={...a},l=[],p=e.settings.hooks??{},d=n.hooks??{},g={},h=[];for(let[R,E]of Object.entries(d)){let T=p[R];if(!T){g[R]=E;continue}let J=new Set(T.map(ne)),dt=E.filter(Y=>!J.has(ne(Y)));dt.length!==E.length&&h.push(R),dt.length>0&&(g[R]=dt)}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??[]),S=n.permissions?.deny??[],M=x.size>0?S.filter(R=>!x.has(R)):S;if(n.permissions){let{deny:R,...E}=n.permissions,T={...E,...M.length>0?{deny:M}:{}};Object.keys(T).length>0&&(c.permissions=T)}return M.length!==S.length&&l.push(`deny -${S.length-M.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await fo(t,c,r),changes:l}}k();k();import{join as Lc}from"path";var Gc=".claude/avatar-tools.json";function wo(){return{tools:{}}}function ko(t){return Lc(t,Gc)}async function kt(t){let e=ko(t);if(!await m(e))return wo();try{return{tools:(await y(e)).tools??{}}}catch{return wo()}}async function jc(t,e){await C(ko(t),e)}async function oe(t,e,n){let r=await kt(t);return r.tools[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await jc(t,r),r}async function yt(t){let e=await kt(t);return Object.entries(e.tools).filter(([,n])=>n.enabled).map(([n])=>n)}function Dc(t,e){let n=e.settings.hooks??{},r=t.hooks??{};for(let[o,i]of Object.entries(n)){let a=r[o]??[],c=new Set(a.flatMap(l=>(l.hooks??[]).map(p=>p.command??"")));for(let l of i)for(let p of l.hooks??[])if(p.command&&!c.has(p.command))return!1}return!0}async function yo(t){let e=await yt(t);if(e.length===0)return[];let n=Uc(t,".claude","settings.json"),r={};if(await m(n))try{r=await y(n)}catch{r={}}let o=[];for(let i of e){let a=await Mt(t,i);if(!a){o.push({name:`Tool: ${i}`,status:"warn",detail:"state=enabled nh\u01B0ng pack thi\u1EBFu manifest (version skew). Ch\u1EA1y 'avatar sync'.",fixable:!1});continue}Dc(r,a)?o.push({name:`Tool: ${i}`,status:"ok",detail:`enabled, hook active (v${a.version})`,fixable:!1}):o.push({name:`Tool: ${i}`,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 o}k();f();function vo(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 Kc(process.cwd());Fc(n),e.fix&&await Vc(n)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Kc(t){let e=[],n=process.versions.node,[r,o]=n.split(".").map(E=>Number.parseInt(E,10)),i=(r??0)>18||(r??0)===18&&(o??0)>=17;e.push({name:"Node.js version",status:i?"ok":"fail",detail:`v${n}${i?"":" (c\u1EA7n >= 18.17)"}`,fixable:!1});let a=await H();a?ot(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=Lt(t,".claude","pack"),l=Lt(t,"CLAUDE.md"),[p,d]=await Promise.all([m(c),m(l)]);if(e.push({name:"team-ai-pack installation",status:p?"ok":"warn",detail:p?c:"Avatar ch\u01B0a init \u2014 ch\u1EA1y 'avatar init'",fixable:!1}),e.push({name:"CLAUDE.md",status:d?"ok":"warn",detail:d?"t\u1ED3n t\u1EA1i \u1EDF project root":"thi\u1EBFu \u2014 ch\u1EA1y 'avatar init'",fixable:!1}),p){let E=Lt(t,".claude",".gitignore"),T=!1;await m(E)&&(T=(await bo.readFile(E,"utf8")).includes("settings.json")),e.push({name:"\u{1F512} settings.json gitignored (.claude/.gitignore)",status:T?"ok":"fail",detail:T?"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 b\u1EA3o v\u1EC7",fixable:!T,fix:T?void 0:async()=>{let{writeClaudeGitignore:J}=await Promise.resolve().then(()=>(At(),Xn));await J(t)}})}let g=Cn("which",["python"]),h=Cn("which",["python3"]),x=g.status===0,S=h.status===0;S&&!x?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 ${h.stdout.toString().trim()} ~/.local/bin/python`,fixable:!1}):x?e.push({name:"Python binary",status:"ok",detail:`python: ${g.stdout.toString().trim()}`,fixable:!1}):S&&e.push({name:"Python binary",status:"ok",detail:`python3: ${h.stdout.toString().trim()}`,fixable:!1});let M=Lt(t,".claude","settings.json");if(await m(M))try{let E=await bo.readFile(M,"utf8"),T=JSON.parse(E);if(T.statusLine?.command){let dt=T.statusLine.command.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(dt?.[2]){let Y=dt[2],Ri=Y.startsWith("/")?Y:Lt(t,Y),Gn=await m(Ri);e.push({name:"statusLine command",status:Gn?"ok":"fail",detail:Gn?`ref OK: ${Y}`:`BROKEN: settings.json ref '${Y}' nh\u01B0ng file kh\xF4ng t\u1ED3n t\u1EA1i. Strip field statusLine ho\u1EB7c fix path.`,fixable:!1})}}}catch{}let mt=Cn("which",["claude"]),R=mt.status===0;if(e.push({name:"Claude Code CLI",status:R?"ok":"warn",detail:R?mt.stdout.toString().trim():"kh\xF4ng t\xECm th\u1EA5y 'claude' tr\xEAn PATH",fixable:!1}),p){let E=await yo(t);e.push(...E)}return e}function Fc(t){let e=[u.bold("Avatar Doctor"),"\u2500".repeat(48)],n=0,r=0,o=0;for(let i of t){let a=i.status==="ok"?u.green("\u2713"):i.status==="warn"?u.yellow("\u26A0"):u.red("\u2717");e.push(`${a} ${i.name.padEnd(28)} ${u.dim(i.detail)}`),i.status==="ok"?n+=1:(r+=1,i.fixable&&(o+=1))}e.push("\u2500".repeat(48)),e.push(`${n} checks passed, ${r} issue${r===1?"":"s"}${o>0?` (${o} fixable \u2014 ch\u1EA1y 'avatar doctor --fix')`:""}`),process.stdout.write(`${Hc(e.join(`
47
47
  `),{padding:1,borderStyle:"round"})}
48
- `)}async function bc(t){let e=0;for(let n of t)if(n.fixable&&n.fix)try{await n.fix(),i.success(`Fixed: ${n.name}`),e+=1}catch(r){i.error(`Failed to fix ${n.name}: ${r instanceof Error?r.message:String(r)}`)}e===0&&i.dim("Kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 fix t\u1EF1 \u0111\u1ED9ng.")}k();import{spawnSync as Nc}from"child_process";import{promises as Oc}from"fs";import{join as bo}from"path";import{confirm as Ec}from"@inquirer/prompts";import $c from"boxen";import{spawnSync as vc}from"child_process";f();var mo=300*1e3,go="gitnexus",M=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=r}};function xc(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new M("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${go} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new M("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new M("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function fo(){i.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=vc("npm",["install","-g",go],{stdio:["inherit","inherit","pipe"],timeout:mo,encoding:"utf8"});if(t.signal==="SIGTERM")throw new M("timeout",`npm install timeout sau ${mo/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),xc(t.status,t.stderr||"");Dt();let e=Q();if(!e.installed||!e.path)throw new M("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 i.success(`\u0110\xE3 c\xE0i GitNexus${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}k();f();import{promises as ko}from"fs";import{homedir as Cc}from"os";import{join as Ac}from"path";var ho=384,wo={command:"gitnexus",args:["mcp"]};function Sc(){return Ac(Cc(),".claude","mcp_servers.json")}function Pc(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 Tc(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await ko.copyFile(t,n),n}async function yo(){let t=Sc(),e={},n=!1;if(await m(t)){n=!0;try{e=await y(t)}catch(a){throw new Error(`MCP config corrupted (${t}): ${a.message}. Backup + x\xF3a file \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}let r=e.mcp_servers?.gitnexus;if(r&&Pc(r,wo))return i.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let o;n&&(o=await Tc(t),i.dim(`Backup ${t} \u2192 ${o}`));let s={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:wo}};await S(t,s,ho);try{await ko.chmod(t,ho)}catch{}return i.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:o}}f();async function Rc(){let t=[u.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: ${u.cyan("npm install -g gitnexus")} (global)`];return process.stdout.write(`${$c(t.join(`
48
+ `)}async function Vc(t){let e=0;for(let n of t)if(n.fixable&&n.fix)try{await n.fix(),s.success(`Fixed: ${n.name}`),e+=1}catch(r){s.error(`Failed to fix ${n.name}: ${r instanceof Error?r.message:String(r)}`)}e===0&&s.dim("Kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 fix t\u1EF1 \u0111\u1ED9ng.")}k();import{spawnSync as rl}from"child_process";import{promises as ol}from"fs";import{join as $o}from"path";import{confirm as Qc}from"@inquirer/prompts";import Zc from"boxen";import{spawnSync as Bc}from"child_process";f();var xo=300*1e3,Co="gitnexus",G=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=r}};function Wc(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new G("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${Co} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new G("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new G("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function Ao(){s.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Bc("npm",["install","-g",Co],{stdio:["inherit","inherit","pipe"],timeout:xo,encoding:"utf8"});if(t.signal==="SIGTERM")throw new G("timeout",`npm install timeout sau ${xo/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),Wc(t.status,t.stderr||"");Ht();let e=nt();if(!e.installed||!e.path)throw new G("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 s.success(`\u0110\xE3 c\xE0i GitNexus${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}k();f();import{promises as Po}from"fs";import{homedir as qc}from"os";import{join as zc}from"path";var So=384,To={command:"gitnexus",args:["mcp"]};function Jc(){return zc(qc(),".claude","mcp_servers.json")}function Yc(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 Xc(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await Po.copyFile(t,n),n}async function Eo(){let t=Jc(),e={},n=!1;if(await m(t)){n=!0;try{e=await y(t)}catch(a){throw new Error(`MCP config corrupted (${t}): ${a.message}. Backup + x\xF3a file \u0111\u1EC3 Avatar t\u1EA1o l\u1EA1i.`)}}let r=e.mcp_servers?.gitnexus;if(r&&Yc(r,To))return s.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let o;n&&(o=await Xc(t),s.dim(`Backup ${t} \u2192 ${o}`));let i={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:To}};await C(t,i,So);try{await Po.chmod(t,So)}catch{}return s.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:o}}f();async function tl(){let t=[u.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: ${u.cyan("npm install -g gitnexus")} (global)`];return process.stdout.write(`${Zc(t.join(`
49
49
  `),{padding:1,borderStyle:"round",borderColor:"cyan"})}
50
- `),await Ec({message:"C\xE0i GitNexus global?",default:!0})}async function _c(){for(;;)try{return fo(),!0}catch(t){let e=t instanceof Error?t.message:String(t),n=t instanceof M&&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 _({taskName:"C\xE0i GitNexus qua npm",reason:e,allowSkip:!0,hint:n});if(r==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i GitNexus.");if(r==="skip")return!1}}async function Ic(t){for(;;)try{return ft(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),r=e instanceof U&&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.",o=await _({taskName:"GitNexus analyze workspace",reason:n,allowSkip:!0,hint:r});if(o==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc GitNexus analyze.");if(o==="skip")return!1}}async function oe(t){let e={ok:!1,installed:!1,analyzed:!1,wikiGenerated:!1,mcpRegistered:!1};try{i.info("=== Phase 10: GitNexus Setup ===");let n=Q();if(!n.installed){if(!await Rc())return await b("gitnexus_setup","result=skipped,reason=user-declined"),i.dim("Skip GitNexus. C\xE0i sau qua `avatar gitnexus install`."),e.reason="user-declined",e;if(!await _c())return await b("gitnexus_setup","result=skipped,reason=install-skipped"),i.dim("Skip GitNexus install. Workspace OK kh\xF4ng c\xF3 codebase intelligence."),e.reason="install-skipped",e;if(Dt(),n=Q(),!n.installed)throw new Error("C\xE0i xong nh\u01B0ng kh\xF4ng detect \u0111\u01B0\u1EE3c binary (PATH issue).")}e.installed=!0,i.success(`GitNexus available${n.version?` v${n.version}`:""}`);try{hr()}catch(r){i.warn(`gitnexus setup fail: ${r.message}`),i.dim("Skip global skills install. Workspace v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c.")}if(t.skipAnalyze)i.dim("GitNexus: c\xE0i + MCP xong. Index s\u1EBD ch\u1EA1y per-repo khi `avatar add repo`.");else{if(!await Ic(t.workspacePath))return await b("gitnexus_setup","result=skipped,reason=analyze-skipped"),i.dim("Skip analyze. GitNexus installed nh\u01B0ng ch\u01B0a index."),e.reason="analyze-skipped",e;e.analyzed=!0;let o=await Vt(t.workspacePath);e.wikiGenerated=o.ran,o.skipped&&o.reason==="fail"&&i.warn(`Wiki gen fail (workspace v\u1EABn OK): ${o.detail??"unknown"}`)}try{let r=await yo();e.mcpRegistered=!0,r.wasUpdated||i.dim("MCP server gitnexus \u0111\xE3 registered tr\u01B0\u1EDBc \u0111\xF3.")}catch(r){i.warn(`MCP server register fail: ${r.message}`),i.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 b("gitnexus_setup",`result=ok,analyzed=${e.analyzed},wiki=${e.wikiGenerated},mcp=${e.mcpRegistered}`),i.success("GitNexus ready"),e}catch(n){if(n instanceof C)throw n;let r=n instanceof Error?n.message:String(n);return i.warn(`GitNexus setup th\u1EA5t b\u1EA1i: ${r}`),i.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup sau qua `avatar gitnexus install`."),await b("gitnexus_setup",`result=failed,error=${r.slice(0,200)}`),e.reason=r,e}}f();function fn(){let t=process.cwd(),e=gt(t);return e||(i.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
50
+ `),await Qc({message:"C\xE0i GitNexus global?",default:!0})}async function el(){for(;;)try{return Ao(),!0}catch(t){let e=t instanceof Error?t.message:String(t),n=t instanceof G&&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 O({taskName:"C\xE0i GitNexus qua npm",reason:e,allowSkip:!0,hint:n});if(r==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i GitNexus.");if(r==="skip")return!1}}async function nl(t){for(;;)try{return wt(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),r=e instanceof D&&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.",o=await O({taskName:"GitNexus analyze workspace",reason:n,allowSkip:!0,hint:r});if(o==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc GitNexus analyze.");if(o==="skip")return!1}}async function ie(t){let e={ok:!1,installed:!1,analyzed:!1,wikiGenerated:!1,mcpRegistered:!1};try{s.info("=== Phase 10: GitNexus Setup ===");let n=nt();if(!n.installed){if(!await tl())return await b("gitnexus_setup","result=skipped,reason=user-declined"),s.dim("Skip GitNexus. C\xE0i sau qua `avatar gitnexus install`."),e.reason="user-declined",e;if(!await el())return await b("gitnexus_setup","result=skipped,reason=install-skipped"),s.dim("Skip GitNexus install. Workspace OK kh\xF4ng c\xF3 codebase intelligence."),e.reason="install-skipped",e;if(Ht(),n=nt(),!n.installed)throw new Error("C\xE0i xong nh\u01B0ng kh\xF4ng detect \u0111\u01B0\u1EE3c binary (PATH issue).")}e.installed=!0,s.success(`GitNexus available${n.version?` v${n.version}`:""}`);try{Tr()}catch(r){s.warn(`gitnexus setup fail: ${r.message}`),s.dim("Skip global skills install. Workspace v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c.")}if(t.skipAnalyze)s.dim("GitNexus: c\xE0i + MCP xong. Index s\u1EBD ch\u1EA1y per-repo khi `avatar add repo`.");else{if(!await nl(t.workspacePath))return await b("gitnexus_setup","result=skipped,reason=analyze-skipped"),s.dim("Skip analyze. GitNexus installed nh\u01B0ng ch\u01B0a index."),e.reason="analyze-skipped",e;e.analyzed=!0;let o=await Bt(t.workspacePath);e.wikiGenerated=o.ran,o.skipped&&o.reason==="fail"&&s.warn(`Wiki gen fail (workspace v\u1EABn OK): ${o.detail??"unknown"}`)}try{let r=await Eo();e.mcpRegistered=!0,r.wasUpdated||s.dim("MCP server gitnexus \u0111\xE3 registered tr\u01B0\u1EDBc \u0111\xF3.")}catch(r){s.warn(`MCP server register fail: ${r.message}`),s.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 b("gitnexus_setup",`result=ok,analyzed=${e.analyzed},wiki=${e.wikiGenerated},mcp=${e.mcpRegistered}`),s.success("GitNexus ready"),e}catch(n){if(n instanceof A)throw n;let r=n instanceof Error?n.message:String(n);return s.warn(`GitNexus setup th\u1EA5t b\u1EA1i: ${r}`),s.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup sau qua `avatar gitnexus install`."),await b("gitnexus_setup",`result=failed,error=${r.slice(0,200)}`),e.reason=r,e}}f();function An(){let t=process.cwd(),e=ht(t);return e||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
51
51
  Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
52
52
  B\u1EA1n \u0111ang \u1EDF: ${t}
53
- Cd v\xE0o workspace dir r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&i.dim(`Detected workspace root: ${e}`),e}async function Mc(){let t=fn(),e=await oe({workspacePath:t});e.ok?(i.success("GitNexus setup complete"),i.dim("Update CLAUDE.md \u0111\u1EC3 re-render section GitNexus: re-run avatar init ho\u1EB7c ch\u1EC9nh tay.")):i.warn(`Setup kh\xF4ng complete: ${e.reason??"unknown"}`)}async function Lc(){let t=fn(),e=bo(t,".gitnexus","meta.json");if(!await m(e)){i.warn(`Ch\u01B0a c\xF3 ${e}. Ch\u1EA1y: avatar gitnexus install`);return}try{let n=await y(e);if(i.info(`Project: ${t}`),i.info(`Last commit: ${n.lastCommit?.slice(0,7)??"(unknown)"}`),i.info(`Indexed at: ${n.indexedAt??"(unknown)"}`),n.stats&&i.info(`Stats: ${n.stats.files??"?"} files \xB7 ${n.stats.nodes??"?"} nodes \xB7 ${n.stats.edges??"?"} edges`),n.lastCommit){let o=Nc("git",["rev-parse","HEAD"],{cwd:t,encoding:"utf8"});o.status===0&&(o.stdout.trim()!==n.lastCommit?i.warn("\u26A0 Index stale \u2014 HEAD \u0111\xE3 ti\u1EBFn t\u1EEB lastCommit. Ch\u1EA1y: avatar gitnexus analyze"):i.dim("Index fresh (HEAD === lastCommit)"))}let r=bo(t,".gitnexus","wiki","index.html");if(await m(r)){let o=await Oc.stat(r);i.info(`Wiki: ${o.mtime.toISOString()}`)}else i.dim("Wiki: ch\u01B0a generate (ch\u1EA1y gitnexus wiki manual)")}catch(n){i.error(`Read meta.json fail: ${n.message}`),process.exit(1)}}async function Gc(){let t=fn();try{ft(t),i.success("Index refreshed. Ch\u1EA1y `avatar gitnexus status` xem chi ti\u1EBFt.")}catch(e){i.error(`Analyze fail: ${e.message}`),process.exit(1)}}function vo(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 Mc()}),e.command("status").description("Show index info + staleness warning").action(async()=>{await Lc()}),e.command("analyze").description("Re-run analyze refresh index (no wiki)").action(async()=>{await Gc()})}import{resolve as vl}from"path";import{input as xl}from"@inquirer/prompts";import hn from"chalk";var ie=[" \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"],se=[[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 jc(t){let n=Math.max(0,Math.min(1,t))*(se.length-1),r=Math.floor(n),o=Math.min(se.length-1,r+1),s=n-r,a=se[r],c=se[o];return[wn(a[0],c[0],s),wn(a[1],c[1],s),wn(a[2],c[2],s)]}function kn(t){if(!((process.stdout.isTTY??!1)&&hn.level>0))return[...ie,...t?.tagline?["",t.tagline]:[]].join(`
54
- `);let r=ie.map((o,s)=>{let a=ie.length===1?0:s/(ie.length-1),[c,l,p]=jc(a);return hn.rgb(c,l,p).bold(o)});return t?.tagline&&(r.push(""),r.push(hn.dim(t.tagline))),r.join(`
55
- `)}function kt(t){process.stdout.write(`
56
- ${kn(t)}
53
+ Cd v\xE0o workspace dir r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&s.dim(`Detected workspace root: ${e}`),e}async function il(){let t=An(),e=await ie({workspacePath:t});e.ok?(s.success("GitNexus setup complete"),s.dim("Update CLAUDE.md \u0111\u1EC3 re-render section GitNexus: re-run avatar init ho\u1EB7c ch\u1EC9nh tay.")):s.warn(`Setup kh\xF4ng complete: ${e.reason??"unknown"}`)}async function sl(){let t=An(),e=$o(t,".gitnexus","meta.json");if(!await m(e)){s.warn(`Ch\u01B0a c\xF3 ${e}. Ch\u1EA1y: avatar gitnexus install`);return}try{let n=await y(e);if(s.info(`Project: ${t}`),s.info(`Last commit: ${n.lastCommit?.slice(0,7)??"(unknown)"}`),s.info(`Indexed at: ${n.indexedAt??"(unknown)"}`),n.stats&&s.info(`Stats: ${n.stats.files??"?"} files \xB7 ${n.stats.nodes??"?"} nodes \xB7 ${n.stats.edges??"?"} edges`),n.lastCommit){let o=rl("git",["rev-parse","HEAD"],{cwd:t,encoding:"utf8"});o.status===0&&(o.stdout.trim()!==n.lastCommit?s.warn("\u26A0 Index stale \u2014 HEAD \u0111\xE3 ti\u1EBFn t\u1EEB lastCommit. Ch\u1EA1y: avatar gitnexus analyze"):s.dim("Index fresh (HEAD === lastCommit)"))}let r=$o(t,".gitnexus","wiki","index.html");if(await m(r)){let o=await ol.stat(r);s.info(`Wiki: ${o.mtime.toISOString()}`)}else s.dim("Wiki: ch\u01B0a generate (ch\u1EA1y gitnexus wiki manual)")}catch(n){s.error(`Read meta.json fail: ${n.message}`),process.exit(1)}}async function al(){let t=An();try{wt(t),s.success("Index refreshed. Ch\u1EA1y `avatar gitnexus status` xem chi ti\u1EBFt.")}catch(e){s.error(`Analyze fail: ${e.message}`),process.exit(1)}}function Ro(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 il()}),e.command("status").description("Show index info + staleness warning").action(async()=>{await sl()}),e.command("analyze").description("Re-run analyze refresh index (no wiki)").action(async()=>{await al()})}import{resolve as Bl}from"path";import{input as Wl}from"@inquirer/prompts";import Sn 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 Tn(t,e,n){return Math.round(t+(e-t)*n)}function cl(t){let n=Math.max(0,Math.min(1,t))*(ae.length-1),r=Math.floor(n),o=Math.min(ae.length-1,r+1),i=n-r,a=ae[r],c=ae[o];return[Tn(a[0],c[0],i),Tn(a[1],c[1],i),Tn(a[2],c[2],i)]}function Pn(t){if(!((process.stdout.isTTY??!1)&&Sn.level>0))return[...se,...t?.tagline?["",t.tagline]:[]].join(`
54
+ `);let r=se.map((o,i)=>{let a=se.length===1?0:i/(se.length-1),[c,l,p]=cl(a);return Sn.rgb(c,l,p).bold(o)});return t?.tagline&&(r.push(""),r.push(Sn.dim(t.tagline))),r.join(`
55
+ `)}function bt(t){process.stdout.write(`
56
+ ${Pn(t)}
57
57
 
58
- `)}import{readdirSync as Gf}from"fs";import{select as Uf}from"@inquirer/prompts";import{simpleGit as Hf}from"simple-git";import{existsSync as rf,statSync as of}from"fs";import{join as af}from"path";import{simpleGit as uf}from"simple-git";import{existsSync as df}from"fs";import{join as ff}from"path";import{readFileSync as kf}from"fs";import{dirname as Uc,join as ae}from"path";import{fileURLToPath as Dc}from"url";var ce=Uc(Dc(import.meta.url)),vf=[ae(ce,"templates","gitignore"),ae(ce,"..","templates","gitignore"),ae(ce,"..","..","src","templates","gitignore"),ae(ce,"..","src","templates","gitignore")],yn="# === avatar ===",le="# === /avatar ===";f();import{existsSync as Af,readFileSync as Sf,writeFileSync as Pf}from"fs";import{join as Ef}from"path";var ue=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};import{join as Po}from"path";k();import{join as xo}from"path";var Hc=".avatar-pack-manifest.json",Kc=".pack-version";function Co(t){return xo(t,Hc)}async function pe(t,e){await S(Co(t),e)}async function Fc(t){let e=Co(t);if(!await m(e))return null;try{return await y(e)}catch{return null}}async function me(t){let e=await Fc(t);if(e?.version)return e.version;let n=xo(t,Kc);if(await m(n)){let r=(await I(n)).trim();if(r)return r}return null}k();import{promises as de}from"fs";import{join as Ao}from"path";var Vc="https://qpdbkewtpkkbcrptnlos.supabase.co/functions/v1/get-pack";function Bc(){return process.env.AVATAR_GET_PACK_ENDPOINT??Vc}var it=class extends Error{constructor(e){super(e),this.name="InvalidIdTokenError"}},st=class extends Error{constructor(e){super(e),this.name="PackNetworkError"}},ot=class extends Error{constructor(e){super(e),this.name="PackParseError"}};async function ge(t,e){let n;try{n=await fetch(Bc(),{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({id_token:t,version:e})})}catch(o){throw new st(`Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Supabase get-pack: ${o instanceof Error?o.message:o}`)}if(n.status===401||n.status===403){let o=await So(n);throw new it(`Supabase t\u1EEB ch\u1ED1i (${n.status}): ${o}. Ch\u1EA1y 'avatar login' v\u1EDBi t\xE0i kho\u1EA3n @nal.vn.`)}if(n.status>=500)throw new st(`Supabase get-pack l\u1ED7i server (${n.status}). Th\u1EED l\u1EA1i sau.`);if(!n.ok){let o=await So(n);throw new ot(`get-pack tr\u1EA3 ${n.status}: ${o}`)}let r;try{r=await n.json()}catch{throw new ot("get-pack tr\u1EA3 response kh\xF4ng ph\u1EA3i JSON.")}if(r.error)throw new ot(`get-pack b\xE1o l\u1ED7i: ${r.error}`);if(!r.url||!r.object)throw new ot("get-pack thi\u1EBFu url/object trong response.");return{url:r.url,object:r.object,expiresIn:r.expires_in??0,requestedBy:r.requested_by??""}}async function fe(t,e){let n=Ao(e,"..");await x(n);let r=Ao(n,`.pack-download-${process.pid}.tar.gz`),o=`${e}.new-${process.pid}`;try{try{await Rn(t,r)}catch(c){throw new st(`T\u1EA3i tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}await pt(o);try{await _n(r,o)}catch(c){throw new ot(`Gi\u1EA3i n\xE9n tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}let s=`${e}.old-${process.pid}`,a=await m(e);a&&await de.rename(e,s);try{await de.rename(o,e)}catch(c){throw a&&await de.rename(s,e).catch(()=>{}),c}a&&await pt(s).catch(()=>{})}finally{await de.rm(r,{force:!0}).catch(()=>{}),await pt(o).catch(()=>{})}}async function So(t){try{let e=await t.json();return e.error??JSON.stringify(e).slice(0,200)}catch{return(await t.text().catch(()=>"")).slice(0,200)}}var K=".claude/pack",yt=class extends Error{constructor(e){super(e),this.name="TeamPackAccessAbortedError"}};async function To(t,e){let n=await Jt(),r=await ge(n,e),o=Po(t,K);await fe(r.url,o);let s=e??Wc(r.object);return await pe(o,{version:s,downloadedObject:r.object,extractedAt:new Date().toISOString()}),{pinnedTag:s}}function Wc(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function he(t){let e=Po(t,K);return await me(e)??"(unknown)"}f();k();import{join as ke,relative as Yc,resolve as _o}from"path";import{input as No,select as Xc}from"@inquirer/prompts";import Qc from"boxen";f();import qc from"boxen";var Eo=[{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 tool"},{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 tool 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 $o(){let t=Math.max(...Eo.map(a=>a.cmd.length)),e=u.bold("\u{1F3AF} Slash commands t\u1EEB team-ai-pack"),n=u.dim("G\xF5 trong Claude Code session \u0111\u1EC3 g\u1ECDi capability c\u1EE7a pack:"),r=Eo.map(a=>` ${u.cyan(a.cmd.padEnd(t))} ${u.dim(a.desc)}`),o=u.dim("Catalog \u0111\u1EA7y \u0111\u1EE7 46 commands: cat .claude/pack/scripts/commands_data.yaml"),s=[e,n,"",...r,"",o].join(`
59
- `);return qc(s,{padding:1,borderStyle:"round",borderColor:"cyan"})}f();k();xt();import{readdir as zc}from"fs/promises";import{join as Jc}from"path";async function we(t){if(!await m(t))return!0;try{return(await zc(t)).filter(r=>!r.startsWith(".")&&r!=="Thumbs.db").length===0}catch{return!1}}async function Ro(t,e,n=10){for(let r=2;r<n;r++){let o=Jc(t,`${e}-${r}`);if(await we(o))return o}return null}function Io(t,e){if(/[\/\\]/.test(e)||e==="."||e===".."||e.includes(".."))throw new Error(`T\xEAn workspace kh\xF4ng an to\xE0n (ch\u1EE9a path separator ho\u1EB7c '..'): "${e}".
60
- Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=_o(ke(t,e)),r=_o(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 Oo(t,e,n){Io(t,e);let r=ke(t,e);if(await we(r))return r;for(i.warn(`Workspace path "${r}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let o=await Ro(t,e);if(n&&o)return i.info(`--force: d\xF9ng ${o}`),o;let s=[];o&&s.push({name:`D\xF9ng "${o}" (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 Xc({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:s});if(a==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc resolve workspace path. Ch\u1EA1y l\u1EA1i v\u1EDBi --workspace-name kh\xE1c.");if(a==="use-alt"&&o)return o;let c=await No({message:"T\xEAn workspace m\u1EDBi:",validate:p=>{let d=p.trim();return d.length===0?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng":/[\/\\]/.test(d)||d==="."||d===".."||d.includes("..")?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c ch\u1EE9a '/', '\\', '..' (path traversal)":!0}});Io(t,c.trim());let l=ke(t,c.trim());if(await we(l))return l;i.warn(`"${l}" c\u0169ng \u0111\xE3 c\xF3 n\u1ED9i dung. Th\u1EED t\xEAn kh\xE1c.`)}}async function Mo(t){return await No({message:"Team owner email:",default:t})}async function Lo(t){try{await x(t)}catch(e){let n=e;throw n.code==="EACCES"||n.code==="EPERM"?new Error(`Kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace "${t}": kh\xF4ng c\xF3 quy\u1EC1n ghi.`):n.code==="ENAMETOOLONG"?new Error(`\u0110\u01B0\u1EDDng d\u1EABn workspace qu\xE1 d\xE0i: "${t}".`):n.code==="ENOSPC"?new Error("H\u1EBFt dung l\u01B0\u1EE3ng \u0111\u0129a, kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace."):new Error(`Kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace "${t}": ${n.message}`)}}function Zc(t){if(t===null)return` ${u.yellow("AI:")} skipped \xB7 ${u.cyan("avatar ai setup")} \u0111\u1EC3 config sau`;if(t.ok){let e=t.model?` \xB7 model=${t.model}`:"";return` ${u.green("AI:")} ready \xB7 ${t.provider}${e}`}return` ${u.yellow("AI:")} failed (${t.reason.slice(0,60)}) \xB7 th\u1EED ${u.cyan("avatar ai setup")}`}function tl(t){if(t===null)return` ${u.yellow("GitNexus:")} skipped \xB7 ${u.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"),` ${u.green("GitNexus:")} ${e.join(" \xB7 ")}`}return` ${u.yellow("GitNexus:")} skipped (${(t.reason??"unknown").slice(0,40)}) \xB7 th\u1EED ${u.cyan("avatar gitnexus install")}`}async function Go(t,e=null,n=null){let r=[`${u.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${Yc(process.cwd(),t)||t}`,Zc(e),tl(n),"",` ${u.cyan(`cd ${t}`)}`,` ${u.cyan("avatar add repo")} Th\xEAm repo code v\xE0o src/`,` ${u.cyan("claude")} M\u1EDF Claude Code \u1EDF workspace root`,"",` ${u.cyan("avatar sync")} C\u1EADp nh\u1EADt team-ai-pack m\u1EDBi`,` ${u.cyan("avatar uninstall")} G\u1EE1 Avatar (gi\u1EEF code)`];process.stdout.write(`${Qc(r.join(`
58
+ `)}import{readdirSync as Zf}from"fs";import{select as eh}from"@inquirer/prompts";import{simpleGit as rh}from"simple-git";import{existsSync as bf,statSync as vf}from"fs";import{join as Cf}from"path";import{simpleGit as Tf}from"simple-git";import{existsSync as $f}from"fs";import{join as _f}from"path";import{readFileSync as Of}from"fs";import{dirname as ll,join as ce}from"path";import{fileURLToPath as ul}from"url";var le=ll(ul(import.meta.url)),Gf=[ce(le,"templates","gitignore"),ce(le,"..","templates","gitignore"),ce(le,"..","..","src","templates","gitignore"),ce(le,"..","src","templates","gitignore")],En="# === avatar ===",ue="# === /avatar ===";f();import{existsSync as Df,readFileSync as Hf,writeFileSync as Kf}from"fs";import{join as Vf}from"path";var pe=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};import{join as Mo}from"path";k();import{join as _o}from"path";var pl=".avatar-pack-manifest.json",ml=".pack-version";function Io(t){return _o(t,pl)}async function me(t,e){await C(Io(t),e)}async function dl(t){let e=Io(t);if(!await m(e))return null;try{return await y(e)}catch{return null}}async function de(t){let e=await dl(t);if(e?.version)return e.version;let n=_o(t,ml);if(await m(n)){let r=(await N(n)).trim();if(r)return r}return null}k();import{promises as ge}from"fs";import{join as No}from"path";var gl="https://qpdbkewtpkkbcrptnlos.supabase.co/functions/v1/get-pack";function fl(){return process.env.AVATAR_GET_PACK_ENDPOINT??gl}var lt=class extends Error{constructor(e){super(e),this.name="InvalidIdTokenError"}},ut=class extends Error{constructor(e){super(e),this.name="PackNetworkError"}},ct=class extends Error{constructor(e){super(e),this.name="PackParseError"}};async function fe(t,e){let n;try{n=await fetch(fl(),{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({id_token:t,version:e})})}catch(o){throw new ut(`Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Supabase get-pack: ${o instanceof Error?o.message:o}`)}if(n.status===401||n.status===403){let o=await Oo(n);throw new lt(`Supabase t\u1EEB ch\u1ED1i (${n.status}): ${o}. Ch\u1EA1y 'avatar login' v\u1EDBi t\xE0i kho\u1EA3n @nal.vn.`)}if(n.status>=500)throw new ut(`Supabase get-pack l\u1ED7i server (${n.status}). Th\u1EED l\u1EA1i sau.`);if(!n.ok){let o=await Oo(n);throw new ct(`get-pack tr\u1EA3 ${n.status}: ${o}`)}let r;try{r=await n.json()}catch{throw new ct("get-pack tr\u1EA3 response kh\xF4ng ph\u1EA3i JSON.")}if(r.error)throw new ct(`get-pack b\xE1o l\u1ED7i: ${r.error}`);if(!r.url||!r.object)throw new ct("get-pack thi\u1EBFu url/object trong response.");return{url:r.url,object:r.object,expiresIn:r.expires_in??0,requestedBy:r.requested_by??""}}async function he(t,e){let n=No(e,"..");await v(n);let r=No(n,`.pack-download-${process.pid}.tar.gz`),o=`${e}.new-${process.pid}`;try{try{await Te(t,r)}catch(c){throw new ut(`T\u1EA3i tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}await Z(o);try{await Pe(r,o)}catch(c){throw new ct(`Gi\u1EA3i n\xE9n tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}let i=`${e}.old-${process.pid}`,a=await m(e);a&&await ge.rename(e,i);try{await ge.rename(o,e)}catch(c){throw a&&await ge.rename(i,e).catch(()=>{}),c}a&&await Z(i).catch(()=>{})}finally{await ge.rm(r,{force:!0}).catch(()=>{}),await Z(o).catch(()=>{})}}async function Oo(t){try{let e=await t.json();return e.error??JSON.stringify(e).slice(0,200)}catch{return(await t.text().catch(()=>"")).slice(0,200)}}var F=".claude/pack",vt=class extends Error{constructor(e){super(e),this.name="TeamPackAccessAbortedError"}};async function Lo(t,e){let n=await Yt(),r=await fe(n,e),o=Mo(t,F);await he(r.url,o);let i=e??hl(r.object);return await me(o,{version:i,downloadedObject:r.object,extractedAt:new Date().toISOString()}),{pinnedTag:i}}function hl(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function we(t){let e=Mo(t,F);return await de(e)??"(unknown)"}f();k();import{join as ye,relative as bl,resolve as Do}from"path";import{input as Ko,select as vl}from"@inquirer/prompts";import xl from"boxen";f();import wl from"boxen";var Go=[{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 tool"},{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 tool 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 jo(){let t=Math.max(...Go.map(a=>a.cmd.length)),e=u.bold("\u{1F3AF} Slash commands t\u1EEB team-ai-pack"),n=u.dim("G\xF5 trong Claude Code session \u0111\u1EC3 g\u1ECDi capability c\u1EE7a pack:"),r=Go.map(a=>` ${u.cyan(a.cmd.padEnd(t))} ${u.dim(a.desc)}`),o=u.dim("Catalog \u0111\u1EA7y \u0111\u1EE7 46 commands: cat .claude/pack/scripts/commands_data.yaml"),i=[e,n,"",...r,"",o].join(`
59
+ `);return wl(i,{padding:1,borderStyle:"round",borderColor:"cyan"})}f();k();At();import{readdir as kl}from"fs/promises";import{join as yl}from"path";async function ke(t){if(!await m(t))return!0;try{return(await kl(t)).filter(r=>!r.startsWith(".")&&r!=="Thumbs.db").length===0}catch{return!1}}async function Uo(t,e,n=10){for(let r=2;r<n;r++){let o=yl(t,`${e}-${r}`);if(await ke(o))return o}return null}function Ho(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}".
60
+ Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=Do(ye(t,e)),r=Do(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 Fo(t,e,n){Ho(t,e);let r=ye(t,e);if(await ke(r))return r;for(s.warn(`Workspace path "${r}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let o=await Uo(t,e);if(n&&o)return s.info(`--force: d\xF9ng ${o}`),o;let i=[];o&&i.push({name:`D\xF9ng "${o}" (suggest)`,value:"use-alt"}),i.push({name:"Nh\u1EADp t\xEAn workspace kh\xE1c (manual)",value:"manual"}),i.push({name:"T\u1EA1m ng\u01B0ng init",value:"abort"});let a=await vl({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:i});if(a==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc resolve workspace path. Ch\u1EA1y l\u1EA1i v\u1EDBi --workspace-name kh\xE1c.");if(a==="use-alt"&&o)return o;let c=await Ko({message:"T\xEAn workspace m\u1EDBi:",validate:p=>{let d=p.trim();return d.length===0?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng":/[\/\\]/.test(d)||d==="."||d===".."||d.includes("..")?"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c ch\u1EE9a '/', '\\', '..' (path traversal)":!0}});Ho(t,c.trim());let l=ye(t,c.trim());if(await ke(l))return l;s.warn(`"${l}" c\u0169ng \u0111\xE3 c\xF3 n\u1ED9i dung. Th\u1EED t\xEAn kh\xE1c.`)}}async function Vo(t){return await Ko({message:"Team owner email:",default:t})}async function Bo(t){try{await v(t)}catch(e){let n=e;throw n.code==="EACCES"||n.code==="EPERM"?new Error(`Kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace "${t}": kh\xF4ng c\xF3 quy\u1EC1n ghi.`):n.code==="ENAMETOOLONG"?new Error(`\u0110\u01B0\u1EDDng d\u1EABn workspace qu\xE1 d\xE0i: "${t}".`):n.code==="ENOSPC"?new Error("H\u1EBFt dung l\u01B0\u1EE3ng \u0111\u0129a, kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace."):new Error(`Kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace "${t}": ${n.message}`)}}function Cl(t){if(t===null)return` ${u.yellow("AI:")} skipped \xB7 ${u.cyan("avatar ai setup")} \u0111\u1EC3 config sau`;if(t.ok){let e=t.model?` \xB7 model=${t.model}`:"";return` ${u.green("AI:")} ready \xB7 ${t.provider}${e}`}return` ${u.yellow("AI:")} failed (${t.reason.slice(0,60)}) \xB7 th\u1EED ${u.cyan("avatar ai setup")}`}function Al(t){if(t===null)return` ${u.yellow("GitNexus:")} skipped \xB7 ${u.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"),` ${u.green("GitNexus:")} ${e.join(" \xB7 ")}`}return` ${u.yellow("GitNexus:")} skipped (${(t.reason??"unknown").slice(0,40)}) \xB7 th\u1EED ${u.cyan("avatar gitnexus install")}`}async function Wo(t,e=null,n=null){let r=[`${u.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${bl(process.cwd(),t)||t}`,Cl(e),Al(n),"",` ${u.cyan(`cd ${t}`)}`,` ${u.cyan("avatar add repo")} Th\xEAm repo code v\xE0o src/`,` ${u.cyan("claude")} M\u1EDF Claude Code \u1EDF workspace root`,"",` ${u.cyan("avatar sync")} C\u1EADp nh\u1EADt team-ai-pack m\u1EDBi`,` ${u.cyan("avatar uninstall")} G\u1EE1 Avatar (gi\u1EEF code)`];process.stdout.write(`${xl(r.join(`
61
61
  `),{padding:1,borderStyle:"round"})}
62
- `);let o=ke(t,K);await m(o)&&process.stdout.write(`
63
- ${$o()}
64
- `)}import el from"boxen";import nl from"open";zt();f();function jo(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 bn(e)}catch(n){i.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function bn(t){if(kt({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await _r(),await b("login_reset");else{let g=await D();if(g&&!tt(g)){i.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=mt("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await Be(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let r=Xe(n),o=[`1. Truy c\u1EADp: ${u.cyan(n.verification_url)}`,`2. Nh\u1EADp code: ${u.bold.yellow(n.user_code)}`,"",`Ho\u1EB7c Avatar t\u1EF1 m\u1EDF browser, click ${u.green("Allow")}...`].join(`
65
- `);process.stdout.write(`${el(o,{padding:1,borderStyle:"round"})}
66
- `),nl(r).catch(()=>{i.dim("(Kh\xF4ng m\u1EDF \u0111\u01B0\u1EE3c browser t\u1EF1 \u0111\u1ED9ng \u2014 copy URL \u1EDF tr\xEAn)")});let s=mt("\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 rl(a);try{if(l=await We(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 p=qe(l.id_token);try{ze(p)}catch(g){throw await Ye(l.access_token),g}let d=Je(l,p);await en(d),await b("login",d.email),i.success(`X\xE1c th\u1EF1c th\xE0nh c\xF4ng: ${d.email}`),i.success(`Verify hosted domain: ${p.hd} \u2713`),i.success(`L\u01B0u credential v\xE0o ${Z} (chmod 600)`)}function rl(t){return new Promise(e=>setTimeout(e,t))}import{join as ve}from"path";f();var Uo=3;async function Do(t,e,n,r=!1){let o=0;for(;;)try{return{pinnedTag:(await To(t,e)).pinnedTag,skipped:!1}}catch(s){if(s instanceof yt)throw s;if(s instanceof it){i.error(s.message);let l=await _({taskName:"T\u1EA3i team-ai-pack (x\xE1c th\u1EF1c NAL)",reason:s.message,allowSkip:!0,hint:"Ch\u1EA1y 'avatar login' v\u1EDBi t\xE0i kho\u1EA3n @nal.vn r\u1ED3i th\u1EED l\u1EA1i. Ho\u1EB7c skip + 'avatar sync' sau."});if(l==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc x\xE1c th\u1EF1c t\u1EA3i pack.");if(l==="skip")return i.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};continue}if(s instanceof st&&o<Uo){o+=1,i.warn(`T\u1EA3i pack l\u1ED7i m\u1EA1ng (l\u1EA7n ${o}/${Uo}): ${s.message}`),await ol(1e3*o);continue}let a=s instanceof Error?s.message:String(s),c=await _({taskName:"T\u1EA3i team-ai-pack tarball",reason:a,allowSkip:!0,hint:"Network glitch? Retry th\u01B0\u1EDDng work. Skip \u2192 d\xF9ng `avatar sync` sau."});if(c==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA3i team-ai-pack.");if(c==="skip")return i.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};o=0}}function ol(t){return new Promise(e=>setTimeout(e,t))}k();xt();import{confirm as ll}from"@inquirer/prompts";k();import{promises as il}from"fs";import{join as vn}from"path";var sl=[".claude","pack","tools"];function Ho(t){return vn(t,...sl)}async function xn(t){let e=Ho(t);if(!await m(e))return[];let n=await il.readdir(e,{withFileTypes:!0}),r=[];for(let o of n)o.isDirectory()&&await m(vn(e,o.name,"tool.json"))&&r.push(o.name);return r.sort()}async function Ko(t){let e=vn(Ho(t),"defaults.json");if(!await m(e))return[];try{return(await y(e)).defaultTools??[]}catch{return[]}}async function Ot(t){let[e,n]=await Promise.all([xn(t),wt(t)]);return{available:e,enabled:n}}f();import{spawnSync as al}from"child_process";f();function cl(t){if(!t)return;let e=al(t,["--version"],{stdio:"ignore"});(e.status!==0||e.error)&&i.warn(`Tool 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 Fo(t,e){switch(e.action){case"enabled":i.success(`Tool '${t}' enabled (${e.changes.join("; ")}).`),e.backupPath&&i.dim(` Backup: ${e.backupPath}`);break;case"disabled":i.success(`Tool '${t}' disabled (${e.changes.join("; ")}).`),e.backupPath&&i.dim(` Backup: ${e.backupPath}`);break;case"no-change":i.dim(`Tool '${t}': settings.json \u0111\xE3 \u0111\xFAng tr\u1EA1ng th\xE1i, kh\xF4ng thay \u0111\u1ED5i.`);break;case"no-manifest":break}}async function at(t,e,n={}){let r=await It(t,e);if(!r)return n.silent||i.warn(`Pack version hi\u1EC7n t\u1EA1i ch\u01B0a h\u1ED7 tr\u1EE3 tool '${e}' (thi\u1EBFu .claude/pack/tools/${e}/tool.json). Ch\u1EA1y 'avatar sync' \u0111\u1EC3 pull pack m\u1EDBi, ho\u1EB7c ki\u1EC3m t\xEAn tool.`),!1;cl(r.requires?.runtime);let o=await ne(t,r);return Fo(e,o),await re(t,e,{enabled:!0,version:r.version}),!0}async function Cn(t,e){let n=await It(t,e);if(!n){i.warn(`Kh\xF4ng t\xECm th\u1EA5y manifest tool '${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 ht(t)).tools[e]?.version??"unknown";return await re(t,e,{enabled:!1,version:s}),!1}let r=await so(t,n);return Fo(e,r),await re(t,e,{enabled:!1,version:n.version}),!0}var ul={"prompt-scoring":"prompt scoring"};async function Vo(t,e={}){let n=await Ko(t);if(n.length!==0)for(let r of n){let o=ul[r]??r,s;if(e.autoYes?s=!0:s=await ll({message:`Do you want to setup ${o} (y/n)`,default:!0}),!s){i.dim(` Skip tool '${r}' (user opt-out).`);continue}await at(t,r,{silent:!0})}}import{promises as ye}from"fs";import{dirname as An,join as Wo,relative as Sn}from"path";import{promises as pl}from"fs";function ml(){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 Bo(t){let e=`${t}.backup-${ml()}`;return await pl.rename(t,e),e}k();var Pn=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function dl(t){try{return(await ye.lstat(t)).isSymbolicLink()}catch{return!1}}async function gl(t,e,n){let r=Sn(An(e),e)||e;if(!await m(t))return{dir:r,action:"source-missing"};if(await m(e))if(await dl(e))await ye.unlink(e);else if(n){let s=await Bo(e),a=Sn(An(e),t);return await ye.symlink(a,e),{dir:r,action:"backed-up-and-linked",backupPath:s}}else return{dir:r,action:"skipped-conflict"};let o=Sn(An(e),t);return await ye.symlink(o,e),{dir:r,action:"created"}}async function be(t,e,n){let r=[];for(let o of Pn){let s=Wo(t,o),a=Wo(e,o);r.push(await gl(s,a,n))}return r}f();import{readFileSync as fl}from"fs";import{dirname as hl,resolve as wl}from"path";import{fileURLToPath as kl}from"url";var bt=null;function L(){if(bt!==null)return bt;let t=hl(kl(import.meta.url));for(let e=0;e<5;e++){let n=wl(t,...Array(e).fill(".."),"package.json");try{let r=fl(n,"utf8"),o=JSON.parse(r);if(o.name==="@nalvietnam/avatar-cli"&&typeof o.version=="string")return bt=o.version,bt}catch{}}return bt="unknown",bt}function yl(t){return t?`
62
+ `);let o=ye(t,F);await m(o)&&process.stdout.write(`
63
+ ${jo()}
64
+ `)}import Sl from"boxen";import Tl from"open";Jt();f();function qo(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 $n(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function $n(t){if(bt({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await Dr(),await b("login_reset");else{let g=await H();if(g&&!ot(g)){s.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=gt("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await tn(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let r=an(n),o=[`1. Truy c\u1EADp: ${u.cyan(n.verification_url)}`,`2. Nh\u1EADp code: ${u.bold.yellow(n.user_code)}`,"",`Ho\u1EB7c Avatar t\u1EF1 m\u1EDF browser, click ${u.green("Allow")}...`].join(`
65
+ `);process.stdout.write(`${Sl(o,{padding:1,borderStyle:"round"})}
66
+ `),Tl(r).catch(()=>{s.dim("(Kh\xF4ng m\u1EDF \u0111\u01B0\u1EE3c browser t\u1EF1 \u0111\u1ED9ng \u2014 copy URL \u1EDF tr\xEAn)")});let i=gt("\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 Pl(a);try{if(l=await en(n.device_code),l)break}catch(g){throw i.fail("X\xE1c th\u1EF1c th\u1EA5t b\u1EA1i"),g}}l||(i.fail("H\u1EBFt h\u1EA1n ch\u1EDD (5 ph\xFAt). Ch\u1EA1y l\u1EA1i 'avatar login'."),process.exit(1)),i.succeed("\u0110\xE3 nh\u1EADn token t\u1EEB Google");let p=nn(l.id_token);try{rn(p)}catch(g){throw await sn(l.access_token),g}let d=on(l,p);await pn(d),await b("login",d.email),s.success(`X\xE1c th\u1EF1c th\xE0nh c\xF4ng: ${d.email}`),s.success(`Verify hosted domain: ${p.hd} \u2713`),s.success(`L\u01B0u credential v\xE0o ${rt} (chmod 600)`)}function Pl(t){return new Promise(e=>setTimeout(e,t))}import{join as xe}from"path";f();var zo=3;async function Jo(t,e,n,r=!1){let o=0;for(;;)try{return{pinnedTag:(await Lo(t,e)).pinnedTag,skipped:!1}}catch(i){if(i instanceof vt)throw i;if(i instanceof lt){s.error(i.message);let l=await O({taskName:"T\u1EA3i team-ai-pack (x\xE1c th\u1EF1c NAL)",reason:i.message,allowSkip:!0,hint:"Ch\u1EA1y 'avatar login' v\u1EDBi t\xE0i kho\u1EA3n @nal.vn r\u1ED3i th\u1EED l\u1EA1i. Ho\u1EB7c skip + 'avatar sync' sau."});if(l==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc x\xE1c th\u1EF1c t\u1EA3i pack.");if(l==="skip")return s.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};continue}if(i instanceof ut&&o<zo){o+=1,s.warn(`T\u1EA3i pack l\u1ED7i m\u1EA1ng (l\u1EA7n ${o}/${zo}): ${i.message}`),await El(1e3*o);continue}let a=i instanceof Error?i.message:String(i),c=await O({taskName:"T\u1EA3i team-ai-pack tarball",reason:a,allowSkip:!0,hint:"Network glitch? Retry th\u01B0\u1EDDng work. Skip \u2192 d\xF9ng `avatar sync` sau."});if(c==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA3i team-ai-pack.");if(c==="skip")return s.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};o=0}}function El(t){return new Promise(e=>setTimeout(e,t))}k();At();import{confirm as Nl}from"@inquirer/prompts";k();import{promises as $l}from"fs";import{join as Rn}from"path";var Rl=[".claude","pack","tools"];function Yo(t){return Rn(t,...Rl)}async function _n(t){let e=Yo(t);if(!await m(e))return[];let n=await $l.readdir(e,{withFileTypes:!0}),r=[];for(let o of n)o.isDirectory()&&await m(Rn(e,o.name,"tool.json"))&&r.push(o.name);return r.sort()}async function Xo(t){let e=Rn(Yo(t),"defaults.json");if(!await m(e))return[];try{return(await y(e)).defaultTools??[]}catch{return[]}}async function Gt(t){let[e,n]=await Promise.all([_n(t),yt(t)]);return{available:e,enabled:n}}f();import{spawnSync as _l}from"child_process";f();function Il(t){if(!t)return;let e=_l(t,["--version"],{stdio:"ignore"});(e.status!==0||e.error)&&s.warn(`Tool 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 Qo(t,e){switch(e.action){case"enabled":s.success(`Tool '${t}' enabled (${e.changes.join("; ")}).`),e.backupPath&&s.dim(` Backup: ${e.backupPath}`);break;case"disabled":s.success(`Tool '${t}' disabled (${e.changes.join("; ")}).`),e.backupPath&&s.dim(` Backup: ${e.backupPath}`);break;case"no-change":s.dim(`Tool '${t}': settings.json \u0111\xE3 \u0111\xFAng tr\u1EA1ng th\xE1i, kh\xF4ng thay \u0111\u1ED5i.`);break;case"no-manifest":break}}async function pt(t,e,n={}){let r=await Mt(t,e);if(!r)return n.silent||s.warn(`Pack version hi\u1EC7n t\u1EA1i ch\u01B0a h\u1ED7 tr\u1EE3 tool '${e}' (thi\u1EBFu .claude/pack/tools/${e}/tool.json). Ch\u1EA1y 'avatar sync' \u0111\u1EC3 pull pack m\u1EDBi, ho\u1EB7c ki\u1EC3m t\xEAn tool.`),!1;Il(r.requires?.runtime);let o=await re(t,r);return Qo(e,o),await oe(t,e,{enabled:!0,version:r.version}),!0}async function In(t,e){let n=await Mt(t,e);if(!n){s.warn(`Kh\xF4ng t\xECm th\u1EA5y manifest tool '${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 i=(await kt(t)).tools[e]?.version??"unknown";return await oe(t,e,{enabled:!1,version:i}),!1}let r=await ho(t,n);return Qo(e,r),await oe(t,e,{enabled:!1,version:n.version}),!0}var Ol={"prompt-scoring":"prompt scoring"};async function Zo(t,e={}){let n=await Xo(t);if(n.length!==0)for(let r of n){let o=Ol[r]??r,i;if(e.autoYes?i=!0:i=await Nl({message:`Do you want to setup ${o} (y/n)`,default:!0}),!i){s.dim(` Skip tool '${r}' (user opt-out).`);continue}await pt(t,r,{silent:!0})}}import{promises as be}from"fs";import{dirname as Nn,join as ei,relative as On}from"path";import{promises as Ml}from"fs";function Ll(){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 ti(t){let e=`${t}.backup-${Ll()}`;return await Ml.rename(t,e),e}k();var Mn=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function Gl(t){try{return(await be.lstat(t)).isSymbolicLink()}catch{return!1}}async function jl(t,e,n){let r=On(Nn(e),e)||e;if(!await m(t))return{dir:r,action:"source-missing"};if(await m(e))if(await Gl(e))await be.unlink(e);else if(n){let i=await ti(e),a=On(Nn(e),t);return await be.symlink(a,e),{dir:r,action:"backed-up-and-linked",backupPath:i}}else return{dir:r,action:"skipped-conflict"};let o=On(Nn(e),t);return await be.symlink(o,e),{dir:r,action:"created"}}async function ve(t,e,n){let r=[];for(let o of Mn){let i=ei(t,o),a=ei(e,o);r.push(await jl(i,a,n))}return r}f();import{readFileSync as Ul}from"fs";import{dirname as Dl,resolve as Hl}from"path";import{fileURLToPath as Kl}from"url";var xt=null;function j(){if(xt!==null)return xt;let t=Dl(Kl(import.meta.url));for(let e=0;e<5;e++){let n=Hl(t,...Array(e).fill(".."),"package.json");try{let r=Ul(n,"utf8"),o=JSON.parse(r);if(o.name==="@nalvietnam/avatar-cli"&&typeof o.version=="string")return xt=o.version,xt}catch{}}return xt="unknown",xt}function Fl(t){return t?`
67
67
  ### \u{1F9E0} CODEBASE INTELLIGENCE \u2014 GitNexus
68
68
 
69
69
  Workspace c\xF3 GitNexus index t\u1EA1i \`.gitnexus/\` cung c\u1EA5p architectural awareness
@@ -91,31 +91,31 @@ Khi user c\u1EA7n regenerate wiki sau refactor l\u1EDBn \u2014 ch\u1EA1y:
91
91
  \`\`\`bash
92
92
  gitnexus wiki . --api-key <key> --base-url <url>
93
93
  \`\`\`
94
- `:""}function qo(t){return{projectName:t.projectName,projectDescription:t.projectDescription,teamOwner:t.teamOwner,avatarVersion:L(),packVersion:t.packVersion,lastScan:new Date().toISOString(),gitnexusSection:yl(t.gitnexusReady??!1)}}async function zo(t){await Lo(t.workspacePath),await x(ve(t.workspacePath,"src")),await x(ve(t.workspacePath,"notes"));let e="(skipped)";if(t.skipTeamPack)i.dim("Skip team-ai-pack (--skip-team-pack).");else{mt("T\u1EA3i team-ai-pack t\u1EEB Supabase...").stop();let c=await Do(t.workspacePath,t.packVersion);e=c.pinnedTag??"(skipped)",c.skipped||i.success(`team-ai-pack: ${e}`)}let n=qo({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:e});await Se(t.workspacePath),await Pe(t.workspacePath,n),await Te(t.workspacePath,n),await Ee(t.workspacePath,n),await $e(t.workspacePath),await bl(t.workspacePath,t.autoYes),await b("init",`workspace=${t.workspaceName}`);let r=null;t.aiSkip?i.dim("B\u1ECF qua AI setup (--ai-skip). Setup sau: avatar ai setup"):r=await Qt({workspacePath:t.workspacePath});let o=null;t.aiSkip||t.gitnexusSkip?i.dim(t.gitnexusSkip?"B\u1ECF qua GitNexus (--gitnexus-skip). Setup sau: avatar gitnexus install":"B\u1ECF qua GitNexus (auto-skip do --ai-skip)."):o=await oe({workspacePath:t.workspacePath,skipAnalyze:!0}),await Go(t.workspacePath,r,o)}async function bl(t,e){let n=ve(t,K);if(!await m(n)){i.dim("Pack ch\u01B0a c\xE0i (skip auto-sync). Ch\u1EA1y `avatar sync` sau.");return}let r=ve(t,".claude");i.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings)...");try{let o=await be(n,r,!1),s=o.filter(l=>l.action==="created"||l.action==="updated").length,a=o.filter(l=>l.action==="source-missing").length;i.success(` \u2713 Symlinks: ${s} created${a>0?`, ${a} source-missing`:""}`);let c=await te(t);c.action==="merged"&&i.success(` \u2713 settings.json merged (${c.changes.join("; ")})`),await Vo(t,{autoYes:e})}catch(o){i.warn(`Auto-sync pack fail: ${o instanceof Error?o.message:o}. Ch\u1EA1y \`avatar sync\` \u0111\u1EC3 retry.`)}}function Jo(t){t.command("init").description("Kh\u1EDFi t\u1EA1o Avatar workspace (m\xF4i tr\u01B0\u1EDDng) \u2014 repo c\xE0i sau b\u1EB1ng 'avatar add repo'").argument("[workspace-name]","T\xEAn workspace (b\u1EAFt bu\u1ED9c, h\u1ECFi n\u1EBFu thi\u1EBFu)").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 (vd v0.2.0-beta.0)").option("--team-owner <email>","Email team owner (b\u1ECF qua prompt)").option("--description <text>","M\xF4 t\u1EA3 1 d\xF2ng c\u1EE7a workspace").option("--skip-team-pack","B\u1ECF qua t\u1EA3i 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("--ai-skip","B\u1ECF qua phase AI setup (ch\u1EA1y 'avatar ai setup' sau)").option("--gitnexus-skip","B\u1ECF qua phase GitNexus setup").action(async(e,n)=>{try{await Cl(e,n)}catch(r){(r instanceof ue||r instanceof yt||r instanceof At||r instanceof C)&&(i.dim(r.message),process.exit(0)),i.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function Cl(t,e){e.yes||kt({tagline:"Kh\u1EDFi t\u1EA1o Avatar workspace"});let n=await D();for(;!n||tt(n);){i.info("Ch\u01B0a \u0111\u0103ng nh\u1EADp (ho\u1EB7c token h\u1EBFt h\u1EA1n) \u2014 ch\u1EA1y login tr\u01B0\u1EDBc...");try{await bn({})}catch(c){i.warn(`Login fail: ${c.message}`)}if(n=await D(),n&&!tt(n))break;throw new C("C\u1EA7n \u0111\u0103ng nh\u1EADp @nal.vn \u0111\u1EC3 t\u1EA3i team-ai-pack. Ch\u1EA1y 'avatar login' r\u1ED3i 'avatar init' l\u1EA1i.")}let r=t??await xl({message:"T\xEAn workspace:",validate:c=>c.trim().length>0?!0:"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng"}),o=e.teamOwner??await Mo(n.email),s=vl(e.workspaceParent??"."),a=await Oo(s,r.trim(),e.force);await zo({workspacePath:a,workspaceName:r.trim(),teamOwner:o,description:e.description??`Avatar workspace: ${r.trim()}`,packVersion:e.packVersion,autoYes:e.yes,skipTeamPack:e.skipTeamPack,aiSkip:e.aiSkip,gitnexusSkip:e.gitnexusSkip})}f();function F(t,e){return()=>{process.stdout.write(`${u.yellow("\u23F3")} ${u.bold(`avatar ${t}`)} \u2014 ch\u01B0a implement \u1EDF milestone hi\u1EC7n t\u1EA1i.
94
+ `:""}function ni(t){return{projectName:t.projectName,projectDescription:t.projectDescription,teamOwner:t.teamOwner,avatarVersion:j(),packVersion:t.packVersion,lastScan:new Date().toISOString(),gitnexusSection:Fl(t.gitnexusReady??!1)}}async function ri(t){await Bo(t.workspacePath),await v(xe(t.workspacePath,"src")),await v(xe(t.workspacePath,"notes"));let e="(skipped)";if(t.skipTeamPack)s.dim("Skip team-ai-pack (--skip-team-pack).");else{gt("T\u1EA3i team-ai-pack t\u1EEB Supabase...").stop();let c=await Jo(t.workspacePath,t.packVersion);e=c.pinnedTag??"(skipped)",c.skipped||s.success(`team-ai-pack: ${e}`)}let n=ni({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:e});await _e(t.workspacePath),await Ie(t.workspacePath,n),await Ne(t.workspacePath,n),await Oe(t.workspacePath,n),await Me(t.workspacePath),await Vl(t.workspacePath,t.autoYes),await b("init",`workspace=${t.workspaceName}`);let r=null;t.aiSkip?s.dim("B\u1ECF qua AI setup (--ai-skip). Setup sau: avatar ai setup"):r=await Zt({workspacePath:t.workspacePath});let o=null;t.aiSkip||t.gitnexusSkip?s.dim(t.gitnexusSkip?"B\u1ECF qua GitNexus (--gitnexus-skip). Setup sau: avatar gitnexus install":"B\u1ECF qua GitNexus (auto-skip do --ai-skip)."):o=await ie({workspacePath:t.workspacePath,skipAnalyze:!0}),await Wo(t.workspacePath,r,o)}async function Vl(t,e){let n=xe(t,F);if(!await m(n)){s.dim("Pack ch\u01B0a c\xE0i (skip auto-sync). Ch\u1EA1y `avatar sync` sau.");return}let r=xe(t,".claude");s.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings)...");try{let o=await ve(n,r,!1),i=o.filter(l=>l.action==="created"||l.action==="updated").length,a=o.filter(l=>l.action==="source-missing").length;s.success(` \u2713 Symlinks: ${i} created${a>0?`, ${a} source-missing`:""}`);let c=await ee(t);c.action==="merged"&&s.success(` \u2713 settings.json merged (${c.changes.join("; ")})`),await Zo(t,{autoYes:e})}catch(o){s.warn(`Auto-sync pack fail: ${o instanceof Error?o.message:o}. Ch\u1EA1y \`avatar sync\` \u0111\u1EC3 retry.`)}}function oi(t){t.command("init").description("Kh\u1EDFi t\u1EA1o Avatar workspace (m\xF4i tr\u01B0\u1EDDng) \u2014 repo c\xE0i sau b\u1EB1ng 'avatar add repo'").argument("[workspace-name]","T\xEAn workspace (b\u1EAFt bu\u1ED9c, h\u1ECFi n\u1EBFu thi\u1EBFu)").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 (vd v0.2.0-beta.0)").option("--team-owner <email>","Email team owner (b\u1ECF qua prompt)").option("--description <text>","M\xF4 t\u1EA3 1 d\xF2ng c\u1EE7a workspace").option("--skip-team-pack","B\u1ECF qua t\u1EA3i 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("--ai-skip","B\u1ECF qua phase AI setup (ch\u1EA1y 'avatar ai setup' sau)").option("--gitnexus-skip","B\u1ECF qua phase GitNexus setup").action(async(e,n)=>{try{await ql(e,n)}catch(r){(r instanceof pe||r instanceof vt||r instanceof Tt||r instanceof A)&&(s.dim(r.message),process.exit(0)),s.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function ql(t,e){e.yes||bt({tagline:"Kh\u1EDFi t\u1EA1o Avatar workspace"});let n=await H();for(;!n||ot(n);){s.info("Ch\u01B0a \u0111\u0103ng nh\u1EADp (ho\u1EB7c token h\u1EBFt h\u1EA1n) \u2014 ch\u1EA1y login tr\u01B0\u1EDBc...");try{await $n({})}catch(c){s.warn(`Login fail: ${c.message}`)}if(n=await H(),n&&!ot(n))break;throw new A("C\u1EA7n \u0111\u0103ng nh\u1EADp @nal.vn \u0111\u1EC3 t\u1EA3i team-ai-pack. Ch\u1EA1y 'avatar login' r\u1ED3i 'avatar init' l\u1EA1i.")}let r=t??await Wl({message:"T\xEAn workspace:",validate:c=>c.trim().length>0?!0:"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng"}),o=e.teamOwner??await Vo(n.email),i=Bl(e.workspaceParent??"."),a=await Fo(i,r.trim(),e.force);await ri({workspacePath:a,workspaceName:r.trim(),teamOwner:o,description:e.description??`Avatar workspace: ${r.trim()}`,packVersion:e.packVersion,autoYes:e.yes,skipTeamPack:e.skipTeamPack,aiSkip:e.aiSkip,gitnexusSkip:e.gitnexusSkip})}f();function V(t,e){return()=>{process.stdout.write(`${u.yellow("\u23F3")} ${u.bold(`avatar ${t}`)} \u2014 ch\u01B0a implement \u1EDF milestone hi\u1EC7n t\u1EA1i.
95
95
  `),e&&process.stdout.write(` D\u1EF1 ki\u1EBFn: ${u.cyan(e)}
96
96
  `),process.stdout.write(` Spec \u0111\xE3 c\xF3 trong avatar-cli-implementation_4.html.
97
- `),process.exit(0)}}function Yo(t){t.command("mcp-run <tool-id>",{hidden:!0}).description("[internal] Spawn MCP v\u1EDBi secrets injected (M09)").action(F("mcp-run","Milestone 09"))}k();import{join as Al}from"path";import Xo from"boxen";f();var Sl=".claude/pack";function Qo(t){t.command("pack").description("Qu\u1EA3n l\xFD team-ai-pack (status, ...)").command("status").description("Hi\u1EC3n th\u1ECB version pack \u0111ang c\xE0i").option("--json","Output JSON cho script").action(async n=>{try{let r=await Pl(process.cwd());n.json?process.stdout.write(`${JSON.stringify(r,null,2)}
98
- `):Tl(r)}catch(r){i.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function Pl(t){let e=Al(t,Sl);return await m(e)?{installed:!0,currentVersion:await he(t).catch(()=>null)}:{installed:!1,currentVersion:null}}function Tl(t){if(!t.installed){let r=[`${u.bold("team-ai-pack")} \xB7 ${u.yellow("ch\u01B0a c\xE0i")}`,"\u2500".repeat(48),u.dim("Ch\u1EA1y `avatar init` ho\u1EB7c `avatar sync` \u0111\u1EC3 c\xE0i pack.")];process.stdout.write(`${Xo(r.join(`
97
+ `),process.exit(0)}}function ii(t){t.command("mcp-run <tool-id>",{hidden:!0}).description("[internal] Spawn MCP v\u1EDBi secrets injected (M09)").action(V("mcp-run","Milestone 09"))}k();import{join as zl}from"path";import si from"boxen";f();var Jl=".claude/pack";function ai(t){t.command("pack").description("Qu\u1EA3n l\xFD team-ai-pack (status, ...)").command("status").description("Hi\u1EC3n th\u1ECB version pack \u0111ang c\xE0i").option("--json","Output JSON cho script").action(async n=>{try{let r=await Yl(process.cwd());n.json?process.stdout.write(`${JSON.stringify(r,null,2)}
98
+ `):Xl(r)}catch(r){s.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function Yl(t){let e=zl(t,Jl);return await m(e)?{installed:!0,currentVersion:await we(t).catch(()=>null)}:{installed:!1,currentVersion:null}}function Xl(t){if(!t.installed){let r=[`${u.bold("team-ai-pack")} \xB7 ${u.yellow("ch\u01B0a c\xE0i")}`,"\u2500".repeat(48),u.dim("Ch\u1EA1y `avatar init` ho\u1EB7c `avatar sync` \u0111\u1EC3 c\xE0i pack.")];process.stdout.write(`${si(r.join(`
99
99
  `),{padding:1,borderStyle:"round"})}
100
- `);return}let e=t.currentVersion??u.yellow("(unknown)"),n=[`${u.bold("team-ai-pack status")}`,"\u2500".repeat(48),`${u.dim("Version hi\u1EC7n t\u1EA1i:")} ${e}`,"",u.dim("C\u1EADp nh\u1EADt m\u1EDBi nh\u1EA5t: avatar sync")];process.stdout.write(`${Xo(n.join(`
100
+ `);return}let e=t.currentVersion??u.yellow("(unknown)"),n=[`${u.bold("team-ai-pack status")}`,"\u2500".repeat(48),`${u.dim("Version hi\u1EC7n t\u1EA1i:")} ${e}`,"",u.dim("C\u1EADp nh\u1EADt m\u1EDBi nh\u1EA5t: avatar sync")];process.stdout.write(`${si(n.join(`
101
101
  `),{padding:1,borderStyle:"round"})}
102
- `)}function Zo(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(F("restore","Milestone 08"))}function ti(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(F("review","Milestone 08"))}function ei(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(F("scan","Milestone 06"))}import{promises as _l}from"fs";import{join as xe}from"path";import Il from"boxen";k();k();import{promises as El}from"fs";import{join as $l}from"path";var Rl="_backup";async function ni(t){let e=$l(t,".claude",Rl);return await m(e)?(await El.readdir(e,{withFileTypes:!0})).filter(r=>r.isDirectory()).map(r=>r.name).sort().reverse():[]}f();function ri(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 Nl(process.cwd());e.json?process.stdout.write(`${JSON.stringify(n,null,2)}
103
- `):Ll(n)}catch(n){i.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Nl(t){let e=t.split("/").filter(Boolean).pop()??"unknown",n=xe(t,".claude");if(!await m(n))return{projectName:e,cliVersion:L(),packVersion:null,pendingCount:0,backupCount:0,techStackSummary:"(Avatar ch\u01B0a init)",hasAvatar:!1,toolsEnabled:[],toolsAvailableCount:0};let o=xe(n,"_pending"),[s,a,c,l,p]=await Promise.all([(async()=>await m(xe(n,"pack"))?he(t).catch(()=>null):null)(),(async()=>await m(o)?(await _l.readdir(o)).filter(g=>g.endsWith(".diff.md")).length:0)(),ni(t).then(d=>d.length).catch(()=>0),Ol(n).catch(()=>"(read error)"),Ot(t).catch(()=>({available:[],enabled:[]}))]);return{projectName:e,cliVersion:L(),packVersion:s,pendingCount:a,backupCount:c,techStackSummary:l,hasAvatar:!0,toolsEnabled:p.enabled,toolsAvailableCount:p.available.length}}async function Ol(t){let e=xe(t,"project","tech-stack.md");return await m(e)?(await I(e)).split(`
104
- `).find(o=>o.trim()&&!o.startsWith("#")&&!o.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function Ml(t){return t.toolsEnabled.length>0?`${t.toolsEnabled.join(", ")} (${t.toolsEnabled.length}/${t.toolsAvailableCount} available)`:t.toolsAvailableCount>0?`none enabled (${t.toolsAvailableCount} available \u2014 avatar tool list)`:"none"}function Ll(t){let e=[`${u.bold("Avatar Status")} \xB7 ${u.cyan(t.projectName)}`,"\u2500".repeat(48),`${u.dim("CLI version:")} ${t.cliVersion}`,`${u.dim("Pack version:")} ${t.packVersion??u.yellow("not installed")}`,`${u.dim("Pending changes:")} ${t.pendingCount}${t.pendingCount>0?u.dim(" (avatar review)"):""}`,`${u.dim("Backups:")} ${t.backupCount}`,`${u.dim("Tech stack:")} ${t.techStackSummary}`,`${u.dim("Tools:")} ${Ml(t)}`];process.stdout.write(`${Il(e.join(`
102
+ `)}function ci(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(V("restore","Milestone 08"))}function li(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(V("review","Milestone 08"))}function ui(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(V("scan","Milestone 06"))}import{promises as eu}from"fs";import{join as Ce}from"path";import nu from"boxen";k();k();import{promises as Ql}from"fs";import{join as Zl}from"path";var tu="_backup";async function pi(t){let e=Zl(t,".claude",tu);return await m(e)?(await Ql.readdir(e,{withFileTypes:!0})).filter(r=>r.isDirectory()).map(r=>r.name).sort().reverse():[]}f();function mi(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 ru(process.cwd());e.json?process.stdout.write(`${JSON.stringify(n,null,2)}
103
+ `):su(n)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function ru(t){let e=t.split("/").filter(Boolean).pop()??"unknown",n=Ce(t,".claude");if(!await m(n))return{projectName:e,cliVersion:j(),packVersion:null,pendingCount:0,backupCount:0,techStackSummary:"(Avatar ch\u01B0a init)",hasAvatar:!1,toolsEnabled:[],toolsAvailableCount:0};let o=Ce(n,"_pending"),[i,a,c,l,p]=await Promise.all([(async()=>await m(Ce(n,"pack"))?we(t).catch(()=>null):null)(),(async()=>await m(o)?(await eu.readdir(o)).filter(g=>g.endsWith(".diff.md")).length:0)(),pi(t).then(d=>d.length).catch(()=>0),ou(n).catch(()=>"(read error)"),Gt(t).catch(()=>({available:[],enabled:[]}))]);return{projectName:e,cliVersion:j(),packVersion:i,pendingCount:a,backupCount:c,techStackSummary:l,hasAvatar:!0,toolsEnabled:p.enabled,toolsAvailableCount:p.available.length}}async function ou(t){let e=Ce(t,"project","tech-stack.md");return await m(e)?(await N(e)).split(`
104
+ `).find(o=>o.trim()&&!o.startsWith("#")&&!o.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function iu(t){return t.toolsEnabled.length>0?`${t.toolsEnabled.join(", ")} (${t.toolsEnabled.length}/${t.toolsAvailableCount} available)`:t.toolsAvailableCount>0?`none enabled (${t.toolsAvailableCount} available \u2014 avatar tool list)`:"none"}function su(t){let e=[`${u.bold("Avatar Status")} \xB7 ${u.cyan(t.projectName)}`,"\u2500".repeat(48),`${u.dim("CLI version:")} ${t.cliVersion}`,`${u.dim("Pack version:")} ${t.packVersion??u.yellow("not installed")}`,`${u.dim("Pending changes:")} ${t.pendingCount}${t.pendingCount>0?u.dim(" (avatar review)"):""}`,`${u.dim("Backups:")} ${t.backupCount}`,`${u.dim("Tech stack:")} ${t.techStackSummary}`,`${u.dim("Tools:")} ${iu(t)}`];process.stdout.write(`${nu(e.join(`
105
105
  `),{padding:1,borderStyle:"round"})}
106
- `)}k();import{join as ai}from"path";k();import{join as oi}from"path";async function Gl(t,e,n){let r=oi(t,n),o=oi(e,n);if(!await m(r))return"source-missing";if(!await m(o))return"needs-creation";let{promises:s}=await import("fs");return(await s.lstat(o)).isSymbolicLink()?"already-linked":"conflict-real-dir"}async function ii(t,e,n){let r=await me(t)??"(ch\u01B0a c\xE0i)",o=n??"stable m\u1EDBi nh\u1EA5t (server resolve)",s=[];for(let a of Pn)s.push({dir:a,status:await Gl(t,e,a)});return{currentVersion:r,targetVersion:o,mountDirStatuses:s}}f();async function si(t){let e=await wt(t);if(e.length!==0){i.info(`Re-apply ${e.length} tool(s) \u0111ang b\u1EADt v\xE0o settings.json...`);for(let n of e)await at(t,n)}}f();function jl(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function Ul(t){let e=process.cwd(),n=ai(e,".claude"),r=ai(e,K);if(await m(n)||(i.error(`Kh\xF4ng th\u1EA5y .claude/ \u2014 kh\xF4ng ph\u1EA3i Avatar workspace.
107
- Ch\u1EA1y 'avatar init' \u0111\u1EC3 kh\u1EDFi t\u1EA1o.`),process.exit(1)),t.dryRun){let c=await ii(r,n,t.version);i.info(`Pack version hi\u1EC7n t\u1EA1i: ${c.currentVersion}`),i.info(`Target version: ${c.targetVersion}`),i.info(`
108
- Mount dir statuses:`);for(let l of c.mountDirStatuses)console.log(` ${l.dir.padEnd(12)} ${l.status}`);i.info(`
109
- Dry-run done. Kh\xF4ng apply. B\u1ECF --dry-run \u0111\u1EC3 th\u1EF1c thi.`);return}let o;try{o=await Jt()}catch(c){i.error(c instanceof Error?c.message:String(c)),process.exit(1)}i.info(t.version?`T\u1EA3i team-ai-pack ${t.version} t\u1EEB Supabase...`:"T\u1EA3i team-ai-pack (stable m\u1EDBi nh\u1EA5t) t\u1EEB Supabase...");let s;try{let c=await ge(o,t.version);await fe(c.url,r),s=t.version??jl(c.object),await pe(r,{version:s,downloadedObject:c.object,extractedAt:new Date().toISOString()}),i.success(`\u0110\xE3 c\xE0i pack ${s}`)}catch(c){c instanceof it?i.error(`${c.message}`):i.error(`T\u1EA3i pack th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`),process.exit(1)}i.info("T\u1EA1o symlink farm...");let a=await be(r,n,t.force===!0);Dl(a,t.force===!0),i.info("Merge pack settings.json template v\xE0o project settings.json...");try{let c=await te(e);switch(c.action){case"merged":i.success(` \u2713 settings.json merged (${c.changes.join("; ")})`);break;case"no-change":i.info(" - settings.json \u0111\xE3 sync.");break;case"no-pack-template":i.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip.");break}}catch(c){i.warn(` ! Merge settings.json fail: ${c instanceof Error?c.message:c}.`)}try{await si(e)}catch(c){i.warn(` ! Re-apply tools fail: ${c instanceof Error?c.message:c}.`)}i.success(`Synced team-ai-pack \u2192 ${s}.`)}function Dl(t,e){for(let r of t)switch(r.action){case"created":i.info(` \u2713 ${r.dir} \u2192 symlinked (new)`);break;case"updated":i.info(` \u2713 ${r.dir} \u2192 symlink refreshed`);break;case"backed-up-and-linked":i.info(` \u2713 ${r.dir} \u2192 symlinked (backup: ${r.backupPath})`);break;case"source-missing":i.warn(` - ${r.dir} \u2192 pack kh\xF4ng c\xF3 dir n\xE0y, skip`);break;case"skipped-conflict":i.warn(` ! ${r.dir} \u2192 CONFLICT: existing real dir. D\xF9ng --force \u0111\u1EC3 backup + override.`);break}let n=t.filter(r=>r.action==="skipped-conflict").length;n>0&&!e&&i.warn(`${n} mount dir(s) skip do conflict. Ch\u1EA1y l\u1EA1i v\u1EDBi --force.`)}function ci(t){t.command("sync").description("T\u1EA3i team-ai-pack m\u1EDBi nh\u1EA5t (tarball Supabase) + t\u1EA1o symlink farm").option("--force","Override .claude/<dir>/ n\u1EBFu l\xE0 real dir (backup tr\u01B0\u1EDBc)").option("--version <tag>","Pin version c\u1EE5 th\u1EC3 (vd: v0.2.0 ho\u1EB7c v0.2.0-beta.0)").option("--dry-run","Hi\u1EC3n th\u1ECB preview, kh\xF4ng apply").action(Ul)}import{resolve as Hl}from"path";import{checkbox as Kl,confirm as Fl}from"@inquirer/prompts";f();function li(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 ui(t){return t.map(e=>({name:e,value:e,checked:!1}))}function pi(t){return[...t]}function Mt(t){return Hl(t.target??process.cwd())}async function mi(t,e,n,r){if(n.all===!0||!process.stdout.isTTY)return pi(t);let s=r==="add"?li(t,e):ui(t);return await Kl({message:r==="add"?"Ch\u1ECDn tool \u0111\u1EC3 c\xE0i (space ch\u1ECDn, enter x\xE1c nh\u1EADn):":"Ch\u1ECDn tool \u0111\u1EC3 g\u1EE1 (space ch\u1ECDn, enter x\xE1c nh\u1EADn):",choices:s})}async function Vl(t){let e=Mt(t),{available:n,enabled:r}=await Ot(e);if(n.length===0){i.dim("Kh\xF4ng c\xF3 tool available (pack ch\u01B0a c\xF3 tools/, ho\u1EB7c ch\u01B0a sync).");return}let o=await mi(n,r,t,"add");if(o.length===0){i.dim("Kh\xF4ng ch\u1ECDn tool n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}for(let s of o)await at(e,s)}async function Bl(t){let e=Mt(t),{enabled:n}=await Ot(e);if(n.length===0){i.dim("Kh\xF4ng c\xF3 tool n\xE0o \u0111ang b\u1EADt \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}let r=await mi(n,[],t,"remove");if(r.length===0){i.dim("Kh\xF4ng ch\u1ECDn tool n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}if(t.yes!==!0&&t.all!==!0&&process.stdout.isTTY&&!await Fl({message:`G\u1EE1 ${r.length} tool: ${r.join(", ")}?`,default:!0})){i.dim("H\u1EE7y \u2014 kh\xF4ng g\u1EE1 tool n\xE0o.");return}for(let s of r)await Cn(e,s)}async function Wl(t){let e=Mt(t),n=await xn(e),r=await ht(e);if(n.length===0&&Object.keys(r.tools).length===0){i.dim("Kh\xF4ng c\xF3 tool n\xE0o (pack ch\u01B0a c\xF3 tools/, ho\u1EB7c ch\u01B0a sync). Ch\u1EA1y 'avatar sync'.");return}let o=[...new Set([...n,...Object.keys(r.tools)])].sort();i.info("Tools:"),console.log(` ${"NAME".padEnd(20)} ${"AVAILABLE".padEnd(10)} ${"ENABLED".padEnd(8)} VERSION`);for(let s of o){let a=n.includes(s)?"yes":"no",c=r.tools[s],l=c?.enabled?"yes":"no",p=c?.version??"-";console.log(` ${s.padEnd(20)} ${a.padEnd(10)} ${l.padEnd(8)} ${p}`)}}function di(t){let e=t.command("tools").description("Qu\u1EA3n l\xFD tool toggle (prompt-scoring, ...) b\u1EADt/t\u1EAFt nguy\xEAn kh\u1ED1i");e.command("enable <name>").description("B\u1EADt tool: merge manifest v\xE0o settings.json + ghi state").option("--target <path>","Workspace root (default: cwd)").action(async(n,r)=>{let o=Mt(r);await at(o,n)||process.exit(1)}),e.command("disable <name>").description("T\u1EAFt tool: 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 o=Mt(r);await Cn(o,n)}),e.command("list").description("Li\u1EC7t k\xEA tool: available | enabled | version").option("--target <path>","Workspace root (default: cwd)").action(async n=>{await Wl(n)}),e.command("add").description("Ch\u1ECDn (multi-select) tool available \u0111\u1EC3 c\xE0i").option("--target <path>","Workspace root (default: cwd)").option("--all","C\xE0i h\u1EBFt tool available (b\u1ECF qua prompt; auto khi non-TTY)").option("--yes","B\u1ECF x\xE1c nh\u1EADn").action(async n=>{await Vl(n)}),e.command("remove").description("Ch\u1ECDn (multi-select) tool \u0111ang b\u1EADt \u0111\u1EC3 g\u1EE1").option("--target <path>","Workspace root (default: cwd)").option("--all","G\u1EE1 h\u1EBFt tool \u0111ang b\u1EADt (b\u1ECF qua prompt; auto khi non-TTY)").option("--yes","B\u1ECF x\xE1c nh\u1EADn").action(async n=>{await Bl(n)})}import{relative as tu}from"path";import{confirm as eu}from"@inquirer/prompts";import nu from"boxen";import{cp as Ce,mkdir as gi,writeFile as ql}from"fs/promises";import{homedir as zl}from"os";import{basename as Jl,join as V}from"path";var Yl=V(zl(),".avatar","uninstall-backups");async function fi(t,e,n){let r=Jl(t),o=new Date().toISOString().replace(/[:.]/g,"-"),s=V(Yl,`${r}-${o}`);if(await gi(s,{recursive:!0,mode:448}),e.claudeDir&&await Ce(e.claudeDir,V(s,".claude"),{recursive:!0}),e.claudeMd&&await Ce(e.claudeMd,V(s,"CLAUDE.md")),e.postMergeHook||e.prePushHook){let c=V(s,"hooks");await gi(c,{recursive:!0}),e.postMergeHook&&await Ce(e.postMergeHook,V(c,"post-merge")),e.prePushHook&&await Ce(e.prePushHook,V(c,"pre-push"))}let a={projectName:r,projectPath:t,timestamp:o,avatarVersion:n,artifacts:{claudeDir:!!e.claudeDir,claudeMd:!!e.claudeMd,postMergeHook:!!e.postMergeHook,prePushHook:!!e.prePushHook}};return await ql(V(s,"manifest.json"),JSON.stringify(a,null,2),"utf8"),s}import{existsSync as Xl}from"fs";import{join as B}from"path";function W(t){return Xl(t)?t:null}function hi(t){let e=W(B(t,".claude")),n=W(B(t,"CLAUDE.md")),r=W(B(t,".git","hooks","post-merge")),o=W(B(t,".git","modules","src","hooks","pre-push")),s=W(B(t,".gitignore")),a=W(B(t,".gitmodules")),c=W(B(t,"notes")),l=W(B(t,"scripts"));return{hasAnyArtifact:!!(e||n||r||o),claudeDir:e,claudeMd:n,postMergeHook:r,prePushHook:o,gitignorePath:s,gitmodulesPath:a,notesDir:c,scriptsDir:l}}import{readFile as wi,rm as q,writeFile as ki}from"fs/promises";async function yi(t,e){if(t.claudeDir)if(e.keepSubmodule){let{readdir:n}=await import("fs/promises"),{join:r}=await import("path"),o=await n(t.claudeDir);for(let s of o)s!=="pack"&&await q(r(t.claudeDir,s),{recursive:!0,force:!0})}else await q(t.claudeDir,{recursive:!0,force:!0});t.claudeMd&&await q(t.claudeMd,{force:!0}),e.keepHooks||(t.postMergeHook&&await q(t.postMergeHook,{force:!0}),t.prePushHook&&await q(t.prePushHook,{force:!0})),t.gitignorePath&&await Ql(t.gitignorePath),t.gitmodulesPath&&!e.keepSubmodule&&await Zl(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 q(n,{recursive:!0,force:!0})}}async function Ql(t){let e=await wi(t,"utf8"),n=e.indexOf(yn),r=e.indexOf(le);if(n===-1||r===-1)return;let o=e.slice(0,n),s=e.slice(r+le.length),a=`${o.trimEnd()}
110
- ${s.trimStart()}`.trim();a.length===0?await q(t,{force:!0}):await ki(t,`${a}
111
- `,"utf8")}async function Zl(t,e){let r=(await wi(t,"utf8")).split(`
112
- `),o=[],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||o.push(c)}let a=o.join(`
113
- `).trim();a.length===0?await q(t,{force:!0}):await ki(t,`${a}
114
- `,"utf8")}f();function bi(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 ru(e)}catch(n){i.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function ru(t){let e=process.cwd(),n=hi(e);if(!n.hasAnyArtifact){i.info("Project ch\u01B0a c\xE0i Avatar \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}if(ou(e,n,t),t.dryRun){i.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await eu({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){i.info("\u0110\xE3 h\u1EE7y.");return}let r=null;t.noBackup||(r=await fi(e,n,L()),i.success(`Backup t\u1EA1o t\u1EA1i: ${r}`)),await yi(n,{keepSubmodule:t.keepSubmodule,keepHooks:t.keepHooks}),await b("uninstall",`project=${e},backup=${r??"skipped"}`),iu(r)}function ou(t,e,n){i.info(`Project: ${t}`),i.plain(""),i.plain("C\xE1c artifact s\u1EBD g\u1EE1:"),e.claudeDir&&i.plain(` ${u.red("\u2717")} ${tu(t,e.claudeDir)||".claude/"}`),e.claudeMd&&i.plain(` ${u.red("\u2717")} CLAUDE.md`),e.postMergeHook&&!n.keepHooks&&i.plain(` ${u.red("\u2717")} .git/hooks/post-merge`),e.prePushHook&&!n.keepHooks&&i.plain(` ${u.red("\u2717")} .git/modules/src/hooks/pre-push`),e.gitignorePath&&i.plain(` ${u.yellow("\u270E")} .gitignore (g\u1EE1 Avatar block)`),e.gitmodulesPath&&!n.keepSubmodule&&i.plain(` ${u.yellow("\u270E")} .gitmodules (g\u1EE1 entry .claude/pack)`),i.plain(""),i.plain("Kh\xF4ng \u0111\u1EE5ng:"),i.plain(` ${u.green("\u2713")} src/ (code kh\xE1ch)`),i.plain(` ${u.green("\u2713")} Git history`),i.plain(` ${u.green("\u2713")} ~/.avatar/config.json (token SSO)`),i.plain(` ${u.green("\u2713")} Secrets trong keychain`),i.plain("")}function iu(t){let e=[`${u.green("\u2713")} Avatar \u0111\xE3 \u0111\u01B0\u1EE3c g\u1EE1 kh\u1ECFi project`];t&&(e.push(""),e.push(` ${u.dim("Backup:")} ${t}`),e.push(` ${u.dim("Restore:")} ${u.cyan(`cp -r "${t}"/* .`)}`)),process.stdout.write(`${nu(e.join(`
106
+ `)}k();import{join as hi}from"path";k();import{join as di}from"path";async function au(t,e,n){let r=di(t,n),o=di(e,n);if(!await m(r))return"source-missing";if(!await m(o))return"needs-creation";let{promises:i}=await import("fs");return(await i.lstat(o)).isSymbolicLink()?"already-linked":"conflict-real-dir"}async function gi(t,e,n){let r=await de(t)??"(ch\u01B0a c\xE0i)",o=n??"stable m\u1EDBi nh\u1EA5t (server resolve)",i=[];for(let a of Mn)i.push({dir:a,status:await au(t,e,a)});return{currentVersion:r,targetVersion:o,mountDirStatuses:i}}f();async function fi(t){let e=await yt(t);if(e.length!==0){s.info(`Re-apply ${e.length} tool(s) \u0111ang b\u1EADt v\xE0o settings.json...`);for(let n of e)await pt(t,n)}}f();function cu(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function lu(t){let e=process.cwd(),n=hi(e,".claude"),r=hi(e,F);if(await m(n)||(s.error(`Kh\xF4ng th\u1EA5y .claude/ \u2014 kh\xF4ng ph\u1EA3i Avatar workspace.
107
+ Ch\u1EA1y 'avatar init' \u0111\u1EC3 kh\u1EDFi t\u1EA1o.`),process.exit(1)),t.dryRun){let c=await gi(r,n,t.version);s.info(`Pack version hi\u1EC7n t\u1EA1i: ${c.currentVersion}`),s.info(`Target version: ${c.targetVersion}`),s.info(`
108
+ Mount dir statuses:`);for(let l of c.mountDirStatuses)console.log(` ${l.dir.padEnd(12)} ${l.status}`);s.info(`
109
+ Dry-run done. Kh\xF4ng apply. B\u1ECF --dry-run \u0111\u1EC3 th\u1EF1c thi.`);return}let o;try{o=await Yt()}catch(c){s.error(c instanceof Error?c.message:String(c)),process.exit(1)}s.info(t.version?`T\u1EA3i team-ai-pack ${t.version} t\u1EEB Supabase...`:"T\u1EA3i team-ai-pack (stable m\u1EDBi nh\u1EA5t) t\u1EEB Supabase...");let i;try{let c=await fe(o,t.version);await he(c.url,r),i=t.version??cu(c.object),await me(r,{version:i,downloadedObject:c.object,extractedAt:new Date().toISOString()}),s.success(`\u0110\xE3 c\xE0i pack ${i}`)}catch(c){c instanceof lt?s.error(`${c.message}`):s.error(`T\u1EA3i pack th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`),process.exit(1)}s.info("T\u1EA1o symlink farm...");let a=await ve(r,n,t.force===!0);uu(a,t.force===!0),s.info("Merge pack settings.json template v\xE0o project settings.json...");try{let c=await ee(e);switch(c.action){case"merged":s.success(` \u2713 settings.json merged (${c.changes.join("; ")})`);break;case"no-change":s.info(" - settings.json \u0111\xE3 sync.");break;case"no-pack-template":s.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip.");break}}catch(c){s.warn(` ! Merge settings.json fail: ${c instanceof Error?c.message:c}.`)}try{await fi(e)}catch(c){s.warn(` ! Re-apply tools fail: ${c instanceof Error?c.message:c}.`)}s.success(`Synced team-ai-pack \u2192 ${i}.`)}function uu(t,e){for(let r of t)switch(r.action){case"created":s.info(` \u2713 ${r.dir} \u2192 symlinked (new)`);break;case"updated":s.info(` \u2713 ${r.dir} \u2192 symlink refreshed`);break;case"backed-up-and-linked":s.info(` \u2713 ${r.dir} \u2192 symlinked (backup: ${r.backupPath})`);break;case"source-missing":s.warn(` - ${r.dir} \u2192 pack kh\xF4ng c\xF3 dir n\xE0y, skip`);break;case"skipped-conflict":s.warn(` ! ${r.dir} \u2192 CONFLICT: existing real dir. D\xF9ng --force \u0111\u1EC3 backup + override.`);break}let n=t.filter(r=>r.action==="skipped-conflict").length;n>0&&!e&&s.warn(`${n} mount dir(s) skip do conflict. Ch\u1EA1y l\u1EA1i v\u1EDBi --force.`)}function wi(t){t.command("sync").description("T\u1EA3i team-ai-pack m\u1EDBi nh\u1EA5t (tarball Supabase) + t\u1EA1o symlink farm").option("--force","Override .claude/<dir>/ n\u1EBFu l\xE0 real dir (backup tr\u01B0\u1EDBc)").option("--version <tag>","Pin version c\u1EE5 th\u1EC3 (vd: v0.2.0 ho\u1EB7c v0.2.0-beta.0)").option("--dry-run","Hi\u1EC3n th\u1ECB preview, kh\xF4ng apply").action(lu)}import{resolve as pu}from"path";import{checkbox as mu,confirm as du}from"@inquirer/prompts";f();function ki(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 yi(t){return t.map(e=>({name:e,value:e,checked:!1}))}function bi(t){return[...t]}function jt(t){return pu(t.target??process.cwd())}async function vi(t,e,n,r){if(n.all===!0||!process.stdout.isTTY)return bi(t);let i=r==="add"?ki(t,e):yi(t);return await mu({message:r==="add"?"Ch\u1ECDn tool \u0111\u1EC3 c\xE0i (space ch\u1ECDn, enter x\xE1c nh\u1EADn):":"Ch\u1ECDn tool \u0111\u1EC3 g\u1EE1 (space ch\u1ECDn, enter x\xE1c nh\u1EADn):",choices:i})}async function gu(t){let e=jt(t),{available:n,enabled:r}=await Gt(e);if(n.length===0){s.dim("Kh\xF4ng c\xF3 tool available (pack ch\u01B0a c\xF3 tools/, ho\u1EB7c ch\u01B0a sync).");return}let o=await vi(n,r,t,"add");if(o.length===0){s.dim("Kh\xF4ng ch\u1ECDn tool n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}for(let i of o)await pt(e,i)}async function fu(t){let e=jt(t),{enabled:n}=await Gt(e);if(n.length===0){s.dim("Kh\xF4ng c\xF3 tool n\xE0o \u0111ang b\u1EADt \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}let r=await vi(n,[],t,"remove");if(r.length===0){s.dim("Kh\xF4ng ch\u1ECDn tool n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}if(t.yes!==!0&&t.all!==!0&&process.stdout.isTTY&&!await du({message:`G\u1EE1 ${r.length} tool: ${r.join(", ")}?`,default:!0})){s.dim("H\u1EE7y \u2014 kh\xF4ng g\u1EE1 tool n\xE0o.");return}for(let i of r)await In(e,i)}async function hu(t){let e=jt(t),n=await _n(e),r=await kt(e);if(n.length===0&&Object.keys(r.tools).length===0){s.dim("Kh\xF4ng c\xF3 tool n\xE0o (pack ch\u01B0a c\xF3 tools/, ho\u1EB7c ch\u01B0a sync). Ch\u1EA1y 'avatar sync'.");return}let o=[...new Set([...n,...Object.keys(r.tools)])].sort();s.info("Tools:"),console.log(` ${"NAME".padEnd(20)} ${"AVAILABLE".padEnd(10)} ${"ENABLED".padEnd(8)} VERSION`);for(let i of o){let a=n.includes(i)?"yes":"no",c=r.tools[i],l=c?.enabled?"yes":"no",p=c?.version??"-";console.log(` ${i.padEnd(20)} ${a.padEnd(10)} ${l.padEnd(8)} ${p}`)}}function xi(t){let e=t.command("tools").description("Qu\u1EA3n l\xFD tool toggle (prompt-scoring, ...) b\u1EADt/t\u1EAFt nguy\xEAn kh\u1ED1i");e.command("enable <name>").description("B\u1EADt tool: merge manifest v\xE0o settings.json + ghi state").option("--target <path>","Workspace root (default: cwd)").action(async(n,r)=>{let o=jt(r);await pt(o,n)||process.exit(1)}),e.command("disable <name>").description("T\u1EAFt tool: 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 o=jt(r);await In(o,n)}),e.command("list").description("Li\u1EC7t k\xEA tool: available | enabled | version").option("--target <path>","Workspace root (default: cwd)").action(async n=>{await hu(n)}),e.command("add").description("Ch\u1ECDn (multi-select) tool available \u0111\u1EC3 c\xE0i").option("--target <path>","Workspace root (default: cwd)").option("--all","C\xE0i h\u1EBFt tool available (b\u1ECF qua prompt; auto khi non-TTY)").option("--yes","B\u1ECF x\xE1c nh\u1EADn").action(async n=>{await gu(n)}),e.command("remove").description("Ch\u1ECDn (multi-select) tool \u0111ang b\u1EADt \u0111\u1EC3 g\u1EE1").option("--target <path>","Workspace root (default: cwd)").option("--all","G\u1EE1 h\u1EBFt tool \u0111ang b\u1EADt (b\u1ECF qua prompt; auto khi non-TTY)").option("--yes","B\u1ECF x\xE1c nh\u1EADn").action(async n=>{await fu(n)})}import{relative as Au}from"path";import{confirm as Su}from"@inquirer/prompts";import Tu from"boxen";import{cp as Ae,mkdir as Ci,writeFile as wu}from"fs/promises";import{homedir as ku}from"os";import{basename as yu,join as B}from"path";var bu=B(ku(),".avatar","uninstall-backups");async function Ai(t,e,n){let r=yu(t),o=new Date().toISOString().replace(/[:.]/g,"-"),i=B(bu,`${r}-${o}`);if(await Ci(i,{recursive:!0,mode:448}),e.claudeDir&&await Ae(e.claudeDir,B(i,".claude"),{recursive:!0}),e.claudeMd&&await Ae(e.claudeMd,B(i,"CLAUDE.md")),e.postMergeHook||e.prePushHook){let c=B(i,"hooks");await Ci(c,{recursive:!0}),e.postMergeHook&&await Ae(e.postMergeHook,B(c,"post-merge")),e.prePushHook&&await Ae(e.prePushHook,B(c,"pre-push"))}let a={projectName:r,projectPath:t,timestamp:o,avatarVersion:n,artifacts:{claudeDir:!!e.claudeDir,claudeMd:!!e.claudeMd,postMergeHook:!!e.postMergeHook,prePushHook:!!e.prePushHook}};return await wu(B(i,"manifest.json"),JSON.stringify(a,null,2),"utf8"),i}import{existsSync as vu}from"fs";import{join as W}from"path";function q(t){return vu(t)?t:null}function Si(t){let e=q(W(t,".claude")),n=q(W(t,"CLAUDE.md")),r=q(W(t,".git","hooks","post-merge")),o=q(W(t,".git","modules","src","hooks","pre-push")),i=q(W(t,".gitignore")),a=q(W(t,".gitmodules")),c=q(W(t,"notes")),l=q(W(t,"scripts"));return{hasAnyArtifact:!!(e||n||r||o),claudeDir:e,claudeMd:n,postMergeHook:r,prePushHook:o,gitignorePath:i,gitmodulesPath:a,notesDir:c,scriptsDir:l}}import{readFile as Ti,rm as z,writeFile as Pi}from"fs/promises";async function Ei(t,e){if(t.claudeDir)if(e.keepSubmodule){let{readdir:n}=await import("fs/promises"),{join:r}=await import("path"),o=await n(t.claudeDir);for(let i of o)i!=="pack"&&await z(r(t.claudeDir,i),{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 xu(t.gitignorePath),t.gitmodulesPath&&!e.keepSubmodule&&await Cu(t.gitmodulesPath,".claude/pack");for(let n of[t.notesDir,t.scriptsDir]){if(!n)continue;let{readdir:r}=await import("fs/promises");(await r(n)).length===0&&await z(n,{recursive:!0,force:!0})}}async function xu(t){let e=await Ti(t,"utf8"),n=e.indexOf(En),r=e.indexOf(ue);if(n===-1||r===-1)return;let o=e.slice(0,n),i=e.slice(r+ue.length),a=`${o.trimEnd()}
110
+ ${i.trimStart()}`.trim();a.length===0?await z(t,{force:!0}):await Pi(t,`${a}
111
+ `,"utf8")}async function Cu(t,e){let r=(await Ti(t,"utf8")).split(`
112
+ `),o=[],i=!1;for(let c of r){if(c.trim().startsWith("[submodule")&&c.includes(e)){i=!0;continue}i&&c.trim().startsWith("[submodule")&&(i=!1),i||o.push(c)}let a=o.join(`
113
+ `).trim();a.length===0?await z(t,{force:!0}):await Pi(t,`${a}
114
+ `,"utf8")}f();function $i(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 Pu(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Pu(t){let e=process.cwd(),n=Si(e);if(!n.hasAnyArtifact){s.info("Project ch\u01B0a c\xE0i Avatar \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}if(Eu(e,n,t),t.dryRun){s.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await Su({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){s.info("\u0110\xE3 h\u1EE7y.");return}let r=null;t.noBackup||(r=await Ai(e,n,j()),s.success(`Backup t\u1EA1o t\u1EA1i: ${r}`)),await Ei(n,{keepSubmodule:t.keepSubmodule,keepHooks:t.keepHooks}),await b("uninstall",`project=${e},backup=${r??"skipped"}`),$u(r)}function Eu(t,e,n){s.info(`Project: ${t}`),s.plain(""),s.plain("C\xE1c artifact s\u1EBD g\u1EE1:"),e.claudeDir&&s.plain(` ${u.red("\u2717")} ${Au(t,e.claudeDir)||".claude/"}`),e.claudeMd&&s.plain(` ${u.red("\u2717")} CLAUDE.md`),e.postMergeHook&&!n.keepHooks&&s.plain(` ${u.red("\u2717")} .git/hooks/post-merge`),e.prePushHook&&!n.keepHooks&&s.plain(` ${u.red("\u2717")} .git/modules/src/hooks/pre-push`),e.gitignorePath&&s.plain(` ${u.yellow("\u270E")} .gitignore (g\u1EE1 Avatar block)`),e.gitmodulesPath&&!n.keepSubmodule&&s.plain(` ${u.yellow("\u270E")} .gitmodules (g\u1EE1 entry .claude/pack)`),s.plain(""),s.plain("Kh\xF4ng \u0111\u1EE5ng:"),s.plain(` ${u.green("\u2713")} src/ (code kh\xE1ch)`),s.plain(` ${u.green("\u2713")} Git history`),s.plain(` ${u.green("\u2713")} ~/.avatar/config.json (token SSO)`),s.plain(` ${u.green("\u2713")} Secrets trong keychain`),s.plain("")}function $u(t){let e=[`${u.green("\u2713")} Avatar \u0111\xE3 \u0111\u01B0\u1EE3c g\u1EE1 kh\u1ECFi project`];t&&(e.push(""),e.push(` ${u.dim("Backup:")} ${t}`),e.push(` ${u.dim("Restore:")} ${u.cyan(`cp -r "${t}"/* .`)}`)),process.stdout.write(`${Tu(e.join(`
115
115
  `),{padding:1,borderStyle:"round"})}
116
- `)}var Tn=L(),T=new su;T.name("avatar").description("AI harness CLI for NAL Vietnam engineering").version(Tn,"-v, --version","Hi\u1EC3n th\u1ECB phi\xEAn b\u1EA3n Avatar CLI").addHelpText("beforeAll",()=>`
117
- ${kn({tagline:`v${Tn} \xB7 AI harness CLI for NAL Vietnam`})}
116
+ `)}var Ln=j(),P=new Ru;P.name("avatar").description("AI harness CLI for NAL Vietnam engineering").version(Ln,"-v, --version","Hi\u1EC3n th\u1ECB phi\xEAn b\u1EA3n Avatar CLI").addHelpText("beforeAll",()=>`
117
+ ${Pn({tagline:`v${Ln} \xB7 AI harness CLI for NAL Vietnam`})}
118
118
 
119
- `);var au=process.argv.includes("-v")||process.argv.includes("--version");au&&(kt({tagline:`v${Tn} \xB7 AI harness CLI for NAL Vietnam`}),process.exit(0));jo(T);Jo(T);ci(T);ei(T);ti(T);ri(T);po(T);Zo(T);di(T);Yo(T);Xr(T);vo(T);Qo(T);Sr(T);bi(T);T.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}
119
+ `);var _u=process.argv.includes("-v")||process.argv.includes("--version");_u&&(bt({tagline:`v${Ln} \xB7 AI harness CLI for NAL Vietnam`}),process.exit(0));qo(P);oi(P);wi(P);ui(P);li(P);mi(P);vo(P);ci(P);xi(P);ii(P);so(P);Ro(P);ai(P);Or(P);$i(P);P.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}
120
120
  `),process.exit(1)});
121
121
  //# sourceMappingURL=index.js.map