@nalvietnam/avatar-cli 3.0.0-beta.6 → 3.0.0-beta.8

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