@nalvietnam/avatar-cli 3.5.0 → 3.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -1,10 +1,10 @@
1
1
  // @nalvietnam/avatar-cli — built with tsup
2
- var Ss=Object.defineProperty;var _=(t,e)=>()=>(t&&(e=t(t=0)),e);var B=(t,e)=>{for(var n in e)Ss(t,n,{get:e[n],enumerable:!0})};var yt={};B(yt,{computeFileSha256:()=>Gs,copyDirRecursive:()=>qe,downloadFile:()=>ze,ensureDir:()=>x,extractTarballToDir:()=>Je,pathExists:()=>m,readJson:()=>y,readText:()=>O,relativeFromCwd:()=>Is,removeRecursive:()=>it,writeJsonAtomic:()=>A,writeTextAtomic:()=>U});import{spawn as fr,spawnSync as Ts}from"child_process";import{createHash as Es}from"crypto";import{constants as Ps,promises as M,createReadStream as Rs}from"fs";import{dirname as hr,join as We,relative as wr,sep as $s}from"path";import{Readable as _s}from"stream";async function m(t){try{return await M.access(t,Ps.F_OK),!0}catch{return!1}}async function x(t){await M.mkdir(t,{recursive:!0})}async function O(t){return await M.readFile(t,"utf8")}async function y(t){return JSON.parse(await O(t))}async function U(t,e,n){await x(hr(t));let r=`${t}.tmp-${process.pid}-${Date.now()}`,o=!1;try{await M.writeFile(r,e,"utf8"),n!==void 0&&await M.chmod(r,n),await M.rename(r,t),o=!0}finally{o||await M.rm(r,{force:!0}).catch(()=>{})}}async function A(t,e,n){await U(t,`${JSON.stringify(e,null,2)}
3
- `,n)}async function qe(t,e,n=[]){await x(e);let r=await M.readdir(t,{withFileTypes:!0});for(let o of r){if(n.includes(o.name))continue;let i=We(t,o.name),a=We(e,o.name);if(o.isDirectory())await qe(i,a,n);else if(o.isSymbolicLink()){let c=await M.readlink(i);await M.symlink(c,a)}else await M.copyFile(i,a)}}async function it(t){await M.rm(t,{recursive:!0,force:!0})}function Is(t){return wr(process.cwd(),t)}async function ze(t,e,n=Ms){let r=new AbortController,o=setTimeout(()=>r.abort(),n);await x(hr(e));let{createWriteStream:i}=await import("fs");try{let a=await fetch(t,{signal:r.signal});if(!a.ok)throw new Error(`Download fail (${a.status}) t\u1EEB ${t.slice(0,80)}...`);if(!a.body)throw new Error("Response kh\xF4ng c\xF3 body \u0111\u1EC3 t\u1EA3i.");let c=i(e);await new Promise((l,u)=>{let d=_s.fromWeb(a.body);d.on("error",u),c.on("error",u),c.on("finish",()=>l()),d.pipe(c)})}catch(a){throw await M.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 Ns(){return ee!==null||(ee=Ts("tar",["--version"],{stdio:"ignore"}).status===0),ee}async function Os(t){return await new Promise((e,n)=>{let r=fr("tar",["-tzf",t],{stdio:["ignore","pipe","pipe"]}),o="",i="";r.stdout.on("data",a=>{o+=a.toString()}),r.stderr.on("data",a=>{i+=a.toString()}),r.on("error",n),r.on("close",a=>{a===0?e(o.split(`
4
- `).filter(c=>c.trim().length>0)):n(new Error(`tar list exit ${a}: ${i.trim()}`))})})}function Ls(t,e){if(t.startsWith("/")||t.startsWith("~"))return!1;let n=We(e,t),r=wr(e,n);return r!==".."&&!r.startsWith(`..${$s}`)}async function Je(t,e){if(!Ns())throw new Error("Kh\xF4ng t\xECm th\u1EA5y l\u1EC7nh `tar` tr\xEAn h\u1EC7 th\u1ED1ng. C\u1EA7n \u0111\u1EC3 gi\u1EA3i n\xE9n pack.\n Windows: c\xE0i qua WSL2, Git Bash, ho\u1EB7c MSYS2.\n macOS/Linux: tar th\u01B0\u1EDDng c\xF3 s\u1EB5n \u2014 ki\u1EC3m tra PATH.");await x(e);let r=(await Os(t)).find(o=>!Ls(o,e));if(r)throw new Error(`Tarball ch\u1EE9a path kh\xF4ng an to\xE0n (path-traversal): "${r}". T\u1EEB ch\u1ED1i gi\u1EA3i n\xE9n \u0111\u1EC3 tr\xE1nh ghi \u0111\xE8 file ngo\xE0i th\u01B0 m\u1EE5c \u0111\xEDch.`);await new Promise((o,i)=>{let a=fr("tar",["-xzf",t,"-C",e],{stdio:["ignore","ignore","pipe"]}),c="";a.stderr.on("data",l=>{c+=l.toString()}),a.on("error",i),a.on("close",l=>{l===0?o():i(new Error(`tar extract exit ${l}: ${c.trim()}`))})})}async function Gs(t){return await new Promise((e,n)=>{let r=Es("sha256"),o=Rs(t);o.on("data",i=>r.update(i)),o.on("end",()=>e(r.digest("hex"))),o.on("error",n)})}var Ms,ee,w=_(()=>{"use strict";Ms=3e5;ee=null});var kr={};B(kr,{addSubmodule:()=>Hs,checkoutBranchHeadInSubmodule:()=>Fs,checkoutTagInSubmodule:()=>Ks,currentBranch:()=>Ds,currentCommitSha:()=>Ws,git:()=>$,isGitRepo:()=>Us,listTags:()=>Vs,tagAtHead:()=>Bs,workingTreeIsDirty:()=>qs});import{join as Ye}from"path";import{simpleGit as js}from"simple-git";function $(t=process.cwd()){return js({baseDir:t,binary:"git"})}async function Us(t=process.cwd()){return await m(Ye(t,".git"))}async function Ds(t=process.cwd()){return(await $(t).revparse(["--abbrev-ref","HEAD"])).trim()}async function Hs(t,e,n=process.cwd()){await $(n).subModule(["add",t,e])}async function Ks(t,e,n=process.cwd()){let r=Ye(n,t);await $(r).fetch(["--tags"]),await $(r).checkout(e)}async function Fs(t,e,n=process.cwd()){let r=Ye(n,t);await $(r).fetch(["origin"]),await $(r).checkout(["-B",e,`origin/${e}`])}async function Vs(t=process.cwd()){return(await $(t).tags()).all}async function Bs(t=process.cwd()){try{return(await $(t).raw(["describe","--tags","--exact-match","HEAD"])).trim()||null}catch{return null}}async function Ws(t=process.cwd()){return(await $(t).revparse(["HEAD"])).trim()}async function qs(t=process.cwd()){return!(await $(t).status()).isClean()}var Xe=_(()=>{"use strict";w()});function yr(t,e){return t.replace(zs,(n,r)=>{let o=e[r];return o===void 0?n:String(o)})}var zs,br=_(()=>{"use strict";zs=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g});import{existsSync as Js}from"fs";import{dirname as vr,join as Nt}from"path";import{fileURLToPath as Ys}from"url";function ta(t){let e=t;for(;;){if(Js(Nt(e,"package.json")))return e;let n=vr(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function ea(t){return await O(Nt(Qs,`${t}.tpl`))}async function ne(t,e){let n=await ea(t);return yr(n,e)}async function Ar(t){return await O(Nt(Zs,`${t}.sh.tpl`))}var Xs,xr,Qs,Zs,Cr=_(()=>{"use strict";w();br();Xs=vr(Ys(import.meta.url)),xr=ta(Xs),Qs=Nt(xr,"src","templates"),Zs=Nt(xr,"src","hooks")});var Pr={};B(Pr,{MANAGED_END_PREFIX:()=>Er,MANAGED_START_PREFIX:()=>Tr,mergeClaudeMdManagedBlock:()=>na});function Sr(t,e){return t.findIndex(n=>n.trimStart().startsWith(e))}function na(t,e){if(t===void 0||t.trim()==="")return{content:e,outcome:"created"};let n=t.split(`
5
- `),r=Sr(n,Tr),o=Sr(n,Er);if(r!==-1&&o!==-1&&r<o){let i=n.slice(0,r),a=n.slice(o+1);return{content:[...i,e.trimEnd(),...a].join(`
6
- `),outcome:"replaced-block"}}return{content:e,outcome:"replaced-whole"}}var Tr,Er,Rr=_(()=>{"use strict";Tr="<!-- AVATAR:MANAGED:START",Er="<!-- AVATAR:MANAGED:END"});var Or={};B(Or,{AVATAR_MANAGED_PATHS:()=>_r,AVATAR_WORKSPACE_MARKER_RELATIVE:()=>Nr,backupIfExists:()=>Ir,createClaudeDirTree:()=>Qe,installGitHook:()=>re,mergeRootClaudeMdManaged:()=>nn,writeAvatarWorkspaceMarker:()=>Ze,writeClaudeGitignore:()=>on,writeProjectKnowledgeFiles:()=>tn,writeProjectSettings:()=>rn,writeRootClaudeMd:()=>en});import{promises as $r}from"fs";import{join as G}from"path";async function Ir(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 $r.rename(t,r),r}async function Mr(t,e,n){let r=await Ir(t);return await U(t,e,n),r}async function Qe(t){let e=G(t,".claude");await x(e);for(let n of ra){let r=G(e,n);await x(r),await U(G(r,".gitkeep"),"")}}async function Ze(t,e){let n=G(t,Nr),r=`${JSON.stringify({avatarWorkspace:!0,avatarVersion:e.avatarVersion,workspaceName:e.workspaceName,createdAt:new Date().toISOString()},null,2)}
7
- `;await U(n,r)}async function tn(t,e){return[]}async function en(t,e){let n=await ne("CLAUDE.md",e);return await Mr(G(t,"CLAUDE.md"),n)}async function nn(t,e){let n=G(t,"CLAUDE.md");if(!await m(n))return{outcome:"skipped-no-file"};let r=await $r.readFile(n,"utf8"),o=await ne("CLAUDE.md",e),{mergeClaudeMdManagedBlock:i}=await Promise.resolve().then(()=>(Rr(),Pr)),a=i(r,o);return await U(n,a.content),{outcome:a.outcome}}async function rn(t,e){let n=await ne("settings.json",e);return await Mr(G(t,".claude","settings.json"),n)}async function on(t){let e=G(t,".claude",".gitignore");await U(e,oa)}async function re(t,e){let n=await Ar(e),r=G(t,"hooks");await x(r);let o=G(r,e);await U(o,n,493)}var _r,ra,Nr,oa,bt=_(()=>{"use strict";w();Cr();_r=[".claude","CLAUDE.md"];ra=["state","_pending","_backup"];Nr=".claude/avatar.json";oa=`# Avatar \u2014 b\u1EA3o v\u1EC7 file nh\u1EA1y c\u1EA3m trong .claude/.
2
+ var _s=Object.defineProperty;var _=(t,e)=>()=>(t&&(e=t(t=0)),e);var W=(t,e)=>{for(var n in e)_s(t,n,{get:e[n],enumerable:!0})};var yt={};W(yt,{computeFileSha256:()=>Vs,copyDirRecursive:()=>ze,downloadFile:()=>Je,ensureDir:()=>x,extractTarballToDir:()=>Ye,pathExists:()=>m,readJson:()=>y,readText:()=>O,relativeFromCwd:()=>Us,removeRecursive:()=>st,writeJsonAtomic:()=>A,writeTextAtomic:()=>j});import{spawn as hr,spawnSync as Is}from"child_process";import{createHash as Ns}from"crypto";import{constants as Ms,promises as M,createReadStream as Os}from"fs";import{dirname as wr,join as qe,relative as kr,sep as Ls}from"path";import{Readable as Gs}from"stream";async function m(t){try{return await M.access(t,Ms.F_OK),!0}catch{return!1}}async function x(t){await M.mkdir(t,{recursive:!0})}async function O(t){return await M.readFile(t,"utf8")}async function y(t){return JSON.parse(await O(t))}async function j(t,e,n){await x(wr(t));let r=`${t}.tmp-${process.pid}-${Date.now()}`,o=!1;try{await M.writeFile(r,e,"utf8"),n!==void 0&&await M.chmod(r,n),await M.rename(r,t),o=!0}finally{o||await M.rm(r,{force:!0}).catch(()=>{})}}async function A(t,e,n){await j(t,`${JSON.stringify(e,null,2)}
3
+ `,n)}async function ze(t,e,n=[]){await x(e);let r=await M.readdir(t,{withFileTypes:!0});for(let o of r){if(n.includes(o.name))continue;let i=qe(t,o.name),a=qe(e,o.name);if(o.isDirectory())await ze(i,a,n);else if(o.isSymbolicLink()){let c=await M.readlink(i);await M.symlink(c,a)}else await M.copyFile(i,a)}}async function st(t){await M.rm(t,{recursive:!0,force:!0})}function Us(t){return kr(process.cwd(),t)}async function Je(t,e,n=js){let r=new AbortController,o=setTimeout(()=>r.abort(),n);await x(wr(e));let{createWriteStream:i}=await import("fs");try{let a=await fetch(t,{signal:r.signal});if(!a.ok)throw new Error(`Download fail (${a.status}) t\u1EEB ${t.slice(0,80)}...`);if(!a.body)throw new Error("Response kh\xF4ng c\xF3 body \u0111\u1EC3 t\u1EA3i.");let c=i(e);await new Promise((l,u)=>{let d=Gs.fromWeb(a.body);d.on("error",u),c.on("error",u),c.on("finish",()=>l()),d.pipe(c)})}catch(a){throw await M.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 Ds(){return ne!==null||(ne=Is("tar",["--version"],{stdio:"ignore"}).status===0),ne}async function Hs(t){return await new Promise((e,n)=>{let r=hr("tar",["-tzf",t],{stdio:["ignore","pipe","pipe"]}),o="",i="";r.stdout.on("data",a=>{o+=a.toString()}),r.stderr.on("data",a=>{i+=a.toString()}),r.on("error",n),r.on("close",a=>{a===0?e(o.split(`
4
+ `).filter(c=>c.trim().length>0)):n(new Error(`tar list exit ${a}: ${i.trim()}`))})})}function Ks(t,e){if(t.startsWith("/")||t.startsWith("~"))return!1;let n=qe(e,t),r=kr(e,n);return r!==".."&&!r.startsWith(`..${Ls}`)}async function Ye(t,e){if(!Ds())throw new Error("Kh\xF4ng t\xECm th\u1EA5y l\u1EC7nh `tar` tr\xEAn h\u1EC7 th\u1ED1ng. C\u1EA7n \u0111\u1EC3 gi\u1EA3i n\xE9n pack.\n Windows: c\xE0i qua WSL2, Git Bash, ho\u1EB7c MSYS2.\n macOS/Linux: tar th\u01B0\u1EDDng c\xF3 s\u1EB5n \u2014 ki\u1EC3m tra PATH.");await x(e);let r=(await Hs(t)).find(o=>!Ks(o,e));if(r)throw new Error(`Tarball ch\u1EE9a path kh\xF4ng an to\xE0n (path-traversal): "${r}". T\u1EEB ch\u1ED1i gi\u1EA3i n\xE9n \u0111\u1EC3 tr\xE1nh ghi \u0111\xE8 file ngo\xE0i th\u01B0 m\u1EE5c \u0111\xEDch.`);await new Promise((o,i)=>{let a=hr("tar",["-xzf",t,"-C",e],{stdio:["ignore","ignore","pipe"]}),c="";a.stderr.on("data",l=>{c+=l.toString()}),a.on("error",i),a.on("close",l=>{l===0?o():i(new Error(`tar extract exit ${l}: ${c.trim()}`))})})}async function Vs(t){return await new Promise((e,n)=>{let r=Ns("sha256"),o=Os(t);o.on("data",i=>r.update(i)),o.on("end",()=>e(r.digest("hex"))),o.on("error",n)})}var js,ne,w=_(()=>{"use strict";js=3e5;ne=null});var yr={};W(yr,{addSubmodule:()=>qs,checkoutBranchHeadInSubmodule:()=>Js,checkoutTagInSubmodule:()=>zs,currentBranch:()=>Ws,currentCommitSha:()=>Qs,git:()=>$,isGitRepo:()=>Bs,listTags:()=>Ys,tagAtHead:()=>Xs,workingTreeIsDirty:()=>Zs});import{join as Xe}from"path";import{simpleGit as Fs}from"simple-git";function $(t=process.cwd()){return Fs({baseDir:t,binary:"git"})}async function Bs(t=process.cwd()){return await m(Xe(t,".git"))}async function Ws(t=process.cwd()){return(await $(t).revparse(["--abbrev-ref","HEAD"])).trim()}async function qs(t,e,n=process.cwd()){await $(n).subModule(["add",t,e])}async function zs(t,e,n=process.cwd()){let r=Xe(n,t);await $(r).fetch(["--tags"]),await $(r).checkout(e)}async function Js(t,e,n=process.cwd()){let r=Xe(n,t);await $(r).fetch(["origin"]),await $(r).checkout(["-B",e,`origin/${e}`])}async function Ys(t=process.cwd()){return(await $(t).tags()).all}async function Xs(t=process.cwd()){try{return(await $(t).raw(["describe","--tags","--exact-match","HEAD"])).trim()||null}catch{return null}}async function Qs(t=process.cwd()){return(await $(t).revparse(["HEAD"])).trim()}async function Zs(t=process.cwd()){return!(await $(t).status()).isClean()}var Qe=_(()=>{"use strict";w()});function br(t,e){return t.replace(ta,(n,r)=>{let o=e[r];return o===void 0?n:String(o)})}var ta,vr=_(()=>{"use strict";ta=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g});import{existsSync as ea}from"fs";import{dirname as xr,join as Mt}from"path";import{fileURLToPath as na}from"url";function sa(t){let e=t;for(;;){if(ea(Mt(e,"package.json")))return e;let n=xr(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function aa(t){return await O(Mt(oa,`${t}.tpl`))}async function re(t,e){let n=await aa(t);return br(n,e)}async function Cr(t){return await O(Mt(ia,`${t}.sh.tpl`))}var ra,Ar,oa,ia,Sr=_(()=>{"use strict";w();vr();ra=xr(na(import.meta.url)),Ar=sa(ra),oa=Mt(Ar,"src","templates"),ia=Mt(Ar,"src","hooks")});var Rr={};W(Rr,{MANAGED_END_PREFIX:()=>Pr,MANAGED_START_PREFIX:()=>Er,mergeClaudeMdManagedBlock:()=>ca});function Tr(t,e){return t.findIndex(n=>n.trimStart().startsWith(e))}function ca(t,e){if(t===void 0||t.trim()==="")return{content:e,outcome:"created"};let n=t.split(`
5
+ `),r=Tr(n,Er),o=Tr(n,Pr);if(r!==-1&&o!==-1&&r<o){let i=n.slice(0,r),a=n.slice(o+1);return{content:[...i,e.trimEnd(),...a].join(`
6
+ `),outcome:"replaced-block"}}return{content:e,outcome:"replaced-whole"}}var Er,Pr,$r=_(()=>{"use strict";Er="<!-- AVATAR:MANAGED:START",Pr="<!-- AVATAR:MANAGED:END"});var Lr={};W(Lr,{AVATAR_MANAGED_PATHS:()=>Ir,AVATAR_WORKSPACE_MARKER_RELATIVE:()=>Or,backupIfExists:()=>Nr,createClaudeDirTree:()=>Ze,installGitHook:()=>oe,mergeRootClaudeMdManaged:()=>rn,writeAvatarWorkspaceMarker:()=>tn,writeClaudeGitignore:()=>sn,writeProjectKnowledgeFiles:()=>en,writeProjectSettings:()=>on,writeRootClaudeMd:()=>nn});import{promises as _r}from"fs";import{join as G}from"path";async function Nr(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 _r.rename(t,r),r}async function Mr(t,e,n){let r=await Nr(t);return await j(t,e,n),r}async function Ze(t){let e=G(t,".claude");await x(e);for(let n of la){let r=G(e,n);await x(r),await j(G(r,".gitkeep"),"")}}async function tn(t,e){let n=G(t,Or),r=`${JSON.stringify({avatarWorkspace:!0,avatarVersion:e.avatarVersion,workspaceName:e.workspaceName,createdAt:new Date().toISOString()},null,2)}
7
+ `;await j(n,r)}async function en(t,e){return[]}async function nn(t,e){let n=await re("CLAUDE.md",e);return await Mr(G(t,"CLAUDE.md"),n)}async function rn(t,e){let n=G(t,"CLAUDE.md");if(!await m(n))return{outcome:"skipped-no-file"};let r=await _r.readFile(n,"utf8"),o=await re("CLAUDE.md",e),{mergeClaudeMdManagedBlock:i}=await Promise.resolve().then(()=>($r(),Rr)),a=i(r,o);return await j(n,a.content),{outcome:a.outcome}}async function on(t,e){let n=await re("settings.json",e);return await Mr(G(t,".claude","settings.json"),n)}async function sn(t){let e=G(t,".claude",".gitignore");await j(e,ua)}async function oe(t,e){let n=await Cr(e),r=G(t,"hooks");await x(r);let o=G(r,e);await j(o,n,493)}var Ir,la,Or,ua,bt=_(()=>{"use strict";w();Sr();Ir=[".claude","CLAUDE.md"];la=["state","_pending","_backup"];Or=".claude/avatar.json";ua=`# Avatar \u2014 b\u1EA3o v\u1EC7 file nh\u1EA1y c\u1EA3m trong .claude/.
8
8
  # Workspace root KH\xD4NG git; file n\xE0y ph\xF2ng tr\u01B0\u1EDDng h\u1EE3p .claude/ b\u1ECB track nh\u1EA7m.
9
9
  settings.json
10
10
  settings.json.backup-*
@@ -13,79 +13,79 @@ settings.json.backup-*
13
13
  _pending/
14
14
  _backup/
15
15
  state/session-*.json
16
- `});var st={};B(st,{REPOS_MANIFEST_RELATIVE:()=>Lr,addRepoToManifest:()=>ie,readReposManifest:()=>oe,removeRepoFromManifest:()=>sa,repoNameExists:()=>se});import{join as ia}from"path";function sn(t){return ia(t,Lr)}async function oe(t){let e=sn(t);if(!await m(e))return[];try{let n=await y(e);return Array.isArray(n.repos)?n.repos:[]}catch{return[]}}async function ie(t,e){let r=(await oe(t)).filter(o=>o.name!==e.name);r.push(e),await A(sn(t),{repos:r})}async function sa(t,e){let n=await oe(t);await A(sn(t),{repos:n.filter(r=>r.name!==e)})}async function se(t,e){return(await oe(t)).some(r=>r.name===e)}var Lr,W=_(()=>{"use strict";w();Lr=".claude/repos.json"});var dn={};B(dn,{RepoAlreadyInWorkspaceError:()=>ct,cloneCodeRepoIntoSrc:()=>ln,folderDirtyStatus:()=>pn,inferRepoNameFromUrl:()=>cn,readFolderRemoteUrl:()=>mn,scaffoldNewRepoInSrc:()=>un,validateRepoName:()=>an});import{join as at}from"path";function an(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 cn(t){return(t.trim().replace(/\/+$/,"").split(/[/:]/).pop()??"").replace(/\.git$/,"")||"repo"}async function ln(t){let e=an(t.name);if(e)throw new ct(e);let n=at(t.workspaceRoot,"src");await x(n);let r=at(n,t.name),o=await m(r),i=await se(t.workspaceRoot,t.name);if(o)throw new ct(`Th\u01B0 m\u1EE5c src/${t.name} \u0111\xE3 t\u1ED3n t\u1EA1i. Ch\u1ECDn t\xEAn kh\xE1c ho\u1EB7c g\u1EE1 tr\u01B0\u1EDBc (avatar remove repo).`);if(i){let{removeRepoFromManifest:c}=await Promise.resolve().then(()=>(W(),st));await c(t.workspaceRoot,t.name)}await $(n).clone(t.url,t.name);let a=at(r,".git");return await re(a,"pre-push").catch(()=>{}),await ie(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 un(t){let e=an(t.name);if(e)throw new Error(e);if(await se(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=at(t.workspaceRoot,"src");await x(n);let r=at(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 x(r);let o=$(r);await o.init();let{writeTextAtomic:i}=await Promise.resolve().then(()=>(w(),yt));return await i(at(r,"README.md"),`# ${t.name}
17
- `),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 re(at(r,".git"),"pre-push").catch(()=>{}),await ie(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 pn(t){try{let e=await $(t).status();return e.isClean()?"":e.files.map(n=>`${n.path}`).join(", ")}catch{return""}}async function mn(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 ct,Ot=_(()=>{"use strict";w();Xe();bt();W();ct=class extends Error{constructor(e){super(e),this.name="RepoAlreadyInWorkspaceError"}}});import p from"chalk";import aa from"ora";function q(t){return aa({text:t,spinner:"dots",isEnabled:process.stdout.isTTY??!1}).start()}function Lt(t){let e=Date.now(),n=q(`${t} (0:00)`),r=()=>{let i=Math.floor((Date.now()-e)/1e3),a=Math.floor(i/60),c=i%60;return`${a}:${String(c).padStart(2,"0")}`},o=setInterval(()=>{n.text=`${t} (${r()})`},1e3);return{succeed:i=>{clearInterval(o),n.succeed(`${i} (${r()})`)},fail:i=>{clearInterval(o),n.fail(`${i} (${r()})`)},stop:()=>{clearInterval(o),n.stop()}}}var s,f=_(()=>{"use strict";s={info:t=>process.stdout.write(`${p.blue("\u2139")} ${t}
16
+ `});var at={};W(at,{REPOS_MANIFEST_RELATIVE:()=>Gr,addRepoToManifest:()=>se,readReposManifest:()=>ie,removeRepoFromManifest:()=>ma,repoNameExists:()=>ae});import{join as pa}from"path";function an(t){return pa(t,Gr)}async function ie(t){let e=an(t);if(!await m(e))return[];try{let n=await y(e);return Array.isArray(n.repos)?n.repos:[]}catch{return[]}}async function se(t,e){let r=(await ie(t)).filter(o=>o.name!==e.name);r.push(e),await A(an(t),{repos:r})}async function ma(t,e){let n=await ie(t);await A(an(t),{repos:n.filter(r=>r.name!==e)})}async function ae(t,e){return(await ie(t)).some(r=>r.name===e)}var Gr,q=_(()=>{"use strict";w();Gr=".claude/repos.json"});var gn={};W(gn,{RepoAlreadyInWorkspaceError:()=>lt,cloneCodeRepoIntoSrc:()=>un,folderDirtyStatus:()=>mn,inferRepoNameFromUrl:()=>ln,readFolderRemoteUrl:()=>dn,scaffoldNewRepoInSrc:()=>pn,validateRepoName:()=>cn});import{join as ct}from"path";function cn(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 ln(t){return(t.trim().replace(/\/+$/,"").split(/[/:]/).pop()??"").replace(/\.git$/,"")||"repo"}async function un(t){let e=cn(t.name);if(e)throw new lt(e);let n=ct(t.workspaceRoot,"src");await x(n);let r=ct(n,t.name),o=await m(r),i=await ae(t.workspaceRoot,t.name);if(o)throw new lt(`Th\u01B0 m\u1EE5c src/${t.name} \u0111\xE3 t\u1ED3n t\u1EA1i. Ch\u1ECDn t\xEAn kh\xE1c ho\u1EB7c g\u1EE1 tr\u01B0\u1EDBc (avatar remove repo).`);if(i){let{removeRepoFromManifest:c}=await Promise.resolve().then(()=>(q(),at));await c(t.workspaceRoot,t.name)}await $(n).clone(t.url,t.name);let a=ct(r,".git");return await oe(a,"pre-push").catch(()=>{}),await se(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 pn(t){let e=cn(t.name);if(e)throw new Error(e);if(await ae(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=ct(t.workspaceRoot,"src");await x(n);let r=ct(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 x(r);let o=$(r);await o.init();let{writeTextAtomic:i}=await Promise.resolve().then(()=>(w(),yt));return await i(ct(r,"README.md"),`# ${t.name}
17
+ `),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 oe(ct(r,".git"),"pre-push").catch(()=>{}),await se(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 mn(t){try{let e=await $(t).status();return e.isClean()?"":e.files.map(n=>`${n.path}`).join(", ")}catch{return""}}async function dn(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 lt,Ot=_(()=>{"use strict";w();Qe();bt();q();lt=class extends Error{constructor(e){super(e),this.name="RepoAlreadyInWorkspaceError"}}});import p from"chalk";import da from"ora";function z(t){return da({text:t,spinner:"dots",isEnabled:process.stdout.isTTY??!1}).start()}function Lt(t){let e=Date.now(),n=z(`${t} (0:00)`),r=()=>{let i=Math.floor((Date.now()-e)/1e3),a=Math.floor(i/60),c=i%60;return`${a}:${String(c).padStart(2,"0")}`},o=setInterval(()=>{n.text=`${t} (${r()})`},1e3);return{succeed:i=>{clearInterval(o),n.succeed(`${i} (${r()})`)},fail:i=>{clearInterval(o),n.fail(`${i} (${r()})`)},stop:()=>{clearInterval(o),n.stop()}}}var s,f=_(()=>{"use strict";s={info:t=>process.stdout.write(`${p.blue("\u2139")} ${t}
18
18
  `),success:t=>process.stdout.write(`${p.green("\u2713")} ${t}
19
19
  `),warn:t=>process.stdout.write(`${p.yellow("\u26A0")} ${t}
20
20
  `),error:t=>process.stderr.write(`${p.red("\u2717")} ${t}
21
21
  `),dim:t=>process.stdout.write(`${p.dim(t)}
22
22
  `),plain:t=>process.stdout.write(`${t}
23
- `)}});import{spawnSync as ha}from"child_process";function ce(){let t=ha("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"],timeout:1e4});if(t.signal==="SIGTERM"||t.error?.code==="ETIMEDOUT")throw new Error("Qu\xE1 th\u1EDDi gian l\u1EA5y GitHub username (10s). Ki\u1EC3m tra m\u1EA1ng / k\u1EBFt n\u1ED1i GitHub r\u1ED3i th\u1EED l\u1EA1i.");if(t.status!==0)throw new Error(`Kh\xF4ng l\u1EA5y \u0111\u01B0\u1EE3c GitHub username: ${t.stderr?.trim()}`);return t.stdout.trim()}var fn=_(()=>{"use strict"});import{spawnSync as wa}from"child_process";function hn(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 Hr(t,e,n){let o=`${n??ce()}/${t}`;s.info(`T\u1EA1o GitHub repo r\u1ED7ng ${o} (${e})...`);let i=wa("gh",["repo","create",o,`--${e}`],{stdio:["ignore","pipe","pipe"],encoding:"utf8"});if(i.status!==0){let a=(i.stderr||"").trim();throw a&&process.stderr.write(`${a}
24
- `),new Gt(hn(a),`T\u1EA1o repo ${o} th\u1EA5t b\u1EA1i.`)}return s.success(`\u0110\xE3 t\u1EA1o: git@github.com:${o}.git`),`git@github.com:${o}.git`}var Gt,wn=_(()=>{"use strict";fn();f();Gt=class extends Error{reason;constructor(e,n){super(n),this.name="RepoCreateError",this.reason=e}}});import{spawnSync as _a}from"child_process";import{existsSync as Ia,statSync as Ma}from"fs";function qr(t){let e=`${t.org}/${t.name}`;if(!Ia(t.folder)||!Ma(t.folder).isDirectory())throw new At("missing-folder",`Folder ngu\u1ED3n kh\xF4ng t\u1ED3n t\u1EA1i: ${t.folder}. Ki\u1EC3m tra l\u1EA1i \u0111\u01B0\u1EDDng d\u1EABn.`);let n=["repo","create",e,`--${t.visibility}`,"--source",t.folder,"--remote","origin","--push"],r=_a("gh",n,{stdio:["ignore","pipe","pipe"],encoding:"utf8"});if(r.status!==0){let o=(r.stderr||"").trim();o&&process.stderr.write(`${o}
25
- `);let i=hn(o);throw i==="name-exists"?new kn(e):i==="no-permission"?new At("no-permission",`Kh\xF4ng c\xF3 quy\u1EC1n t\u1EA1o repo ${e}. Ki\u1EC3m tra \u0111\u0103ng nh\u1EADp \u0111\xFAng account/org ch\u01B0a (gh auth status).`):i==="network"?new At("network",`L\u1ED7i m\u1EA1ng khi t\u1EA1o repo ${e}.`):new At("unknown",`gh repo create th\u1EA5t b\u1EA1i (exit ${r.status}).`)}return{sshUrl:`git@github.com:${e}.git`,httpsUrl:`https://github.com/${e}.git`}}var kn,At,zr=_(()=>{"use strict";wn();kn=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"}},At=class extends Error{reason;constructor(e,n){super(n),this.name="GhRepoCreateError",this.reason=e}}});function Jr(t){if(!Na.test(t))throw new yn(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 Na,yn,Xr=_(()=>{"use strict";Na=/^[a-zA-Z0-9._-]{1,100}$/,yn=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 Qr={};B(Qr,{createGithubRemoteFromFolder:()=>bn});function bn(t){Jr(t.name),Yr(t.visibility);let e=t.org??ce();s.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=qr({folder:t.folder,org:e,name:t.name,visibility:t.visibility});return s.success(`\u0110\xE3 t\u1EA1o: ${n.sshUrl}`),n}var vn=_(()=>{"use strict";zr();fn();f();Xr()});var Un={};B(Un,{ALLOWED_HOSTED_DOMAINS:()=>ke,HOSTED_DOMAIN:()=>rl,SCOPES:()=>Ho,buildUserConfig:()=>Gn,buildVerificationUrl:()=>jn,decodeIdToken:()=>On,pollForToken:()=>Nn,refreshAccessToken:()=>cl,requestDeviceCode:()=>Mn,revokeToken:()=>Bt,verifyHostedDomain:()=>Ln,verifyIdTokenClaims:()=>Fo});async function Mn(){let t=new URLSearchParams({client_id:ye,scope:Ho.join(" ")}),e=await fetch(ol,{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:ye,client_secret:Do,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(Ko,{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 On(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 Fo(t){if(!sl.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==ye)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+al<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||!ke.includes(t.hd))throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u ${ke.map(n=>`@${n}`).join(" ho\u1EB7c ")}). Nh\u1EADn: ${t.email}${t.hd?` (hd=${t.hd})`:" (kh\xF4ng ph\u1EA3i t\xE0i kho\u1EA3n Workspace)"}`);if(!t.email_verified)throw new Error("Email ch\u01B0a \u0111\u01B0\u1EE3c Google verify")}function Gn(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 cl(t){let e=new URLSearchParams({client_id:ye,client_secret:Do,refresh_token:t,grant_type:"refresh_token"}),n=await fetch(Ko,{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 Bt(t){let e=new URLSearchParams({token:t}),n=new AbortController,r=setTimeout(()=>n.abort(),1e4);await fetch(il,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e,signal:n.signal}).catch(()=>{}).finally(()=>clearTimeout(r))}function jn(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.toString()}var ye,Do,ke,rl,Ho,ol,Ko,il,sl,al,Ln,Wt=_(()=>{"use strict";ye="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",Do="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",ke=["nal.vn","nal-software.com"],rl=ke[0],Ho=["openid","email","profile"],ol="https://oauth2.googleapis.com/device/code",Ko="https://oauth2.googleapis.com/token",il="https://oauth2.googleapis.com/revoke";sl=new Set(["https://accounts.google.com","accounts.google.com"]),al=60;Ln=Fo});import{Command as qp}from"commander";Ot();import{resolve as ho}from"path";import{confirm as Kt,input as Ht,select as Cn}from"@inquirer/prompts";Ot();w();f();import{join as pa}from"path";import{confirm as ma,input as ae,select as Ur}from"@inquirer/prompts";f();import{spawnSync as ca}from"child_process";function vt(){s.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp GitHub qua gh CLI (browser s\u1EBD m\u1EDF)...");let t=ca("gh",["auth","login","--hostname","github.com","--web","--git-protocol","ssh"],{stdio:"inherit"});if(t.status!==0)throw new Error(`gh auth login th\u1EA5t b\u1EA1i (exit ${t.status}). Th\u1EED 'gh auth login' tay.`);s.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp GitHub")}import{spawnSync as la}from"child_process";var gn=5e3;function ua(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 xt(t){let e=la("git",["ls-remote","--exit-code",t,"HEAD"],{encoding:"utf8",timeout:gn,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 > ${gn/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 > ${gn/1e3}s`};let n=(e.stderr||"").trim();return{ok:!1,reason:ua(n),detail:n.slice(0,300)}}function da(t,e,n){switch(s.warn(`\u26D4 Kh\xF4ng clone \u0111\u01B0\u1EE3c ${t}`),e){case"not-found":s.dim(" \u2192 Repo kh\xF4ng t\u1ED3n t\u1EA1i (sai URL, ho\u1EB7c repo ch\u01B0a \u0111\u01B0\u1EE3c t\u1EA1o tr\xEAn GitHub).");break;case"no-access":s.dim(` \u2192 Kh\xF4ng c\xF3 quy\u1EC1n truy c\u1EADp (repo private + sai account), HO\u1EB6C repo kh\xF4ng t\u1ED3n t\u1EA1i.
26
- Ki\u1EC3m tra: \u0111ang login \u0111\xFAng GitHub account ch\u01B0a? (gh auth status)`);break;case"network":s.dim(" \u2192 L\u1ED7i m\u1EA1ng (m\u1EA5t k\u1EBFt n\u1ED1i / DNS / timeout).");break;default:s.dim(` \u2192 L\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh${n?`: ${n}`:""}.`)}}function Gr(){s.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");try{vt()}catch(t){s.warn(`${t instanceof Error?t.message:t}. C\xF3 th\u1EC3 ch\u1EA1y 'gh auth login' tay r\u1ED3i retry.`)}}async function jr(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 Ur({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}async function Dr(t){let e=t.url,n=t.name;for(;;){let r=xt(e);if(!r.ok){da(e,r.reason??"unknown",r.detail);let o=await jr(r.reason??"unknown");if(o==="skip")return s.dim(`B\u1ECF qua repo ${n}.`),{cloned:null,skipped:!0};if(o==="switch-account"){Gr();continue}if(o==="new-url"){if(e=await ae({message:"URL repo kh\xE1c:",validate:a=>a.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),await ma({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(()=>(Ot(),dn));n=await ae({message:"T\xEAn th\u01B0 m\u1EE5c:",default:a(e)})}continue}continue}try{return{cloned:await ln({workspaceRoot:t.workspaceRoot,url:e,name:n}),skipped:!1}}catch(o){let i=o instanceof Error?o.message:String(o);if(o instanceof ct){let c=await ga(t.workspaceRoot,n);if(c&&fa(c,e))return s.warn(`\u26A0 Repo n\xE0y \u0111\xE3 c\xF3 trong workspace (src/${n}). D\xF9ng tr\u1EF1c ti\u1EBFp, kh\xF4ng add l\u1EA1i.`),s.dim(` Mu\u1ED1n thay b\u1EB1ng b\u1EA3n m\u1EDBi? G\u1EE1 tr\u01B0\u1EDBc: avatar remove repo ${n}`),{cloned:null,skipped:!0};if(s.warn(`\u26A0 ${i}`),await Ur({message:"C\xE1ch x\u1EED l\xFD?",choices:[{name:"\u0110\u1ED5i t\xEAn th\u01B0 m\u1EE5c kh\xE1c",value:"rename"},{name:"B\u1ECF qua repo n\xE0y",value:"skip"}]})==="skip")return{cloned:null,skipped:!0};n=await ae({message:"T\xEAn th\u01B0 m\u1EE5c m\u1EDBi trong src/:",validate:u=>u.trim().length>0?!0:"T\xEAn b\u1EAFt bu\u1ED9c"});continue}s.warn(`Clone fail: ${i}`),await it(pa(t.workspaceRoot,"src",n)).catch(()=>{});let a=await jr("unknown");if(a==="skip")return{cloned:null,skipped:!0};if(a==="switch-account"){Gr();continue}a==="new-url"&&(e=await ae({message:"URL repo kh\xE1c:",validate:c=>c.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}))}}}async function ga(t,e){let{readReposManifest:n}=await Promise.resolve().then(()=>(W(),st));return(await n(t)).find(o=>o.name===e)?.url??null}function fa(t,e){let n=r=>r.trim().toLowerCase().replace(/^git@/,"").replace(/^https?:\/\//,"").replace(/:/,"/").replace(/\.git$/,"").replace(/\/+$/,"");return n(t)===n(e)}wn();import{input as ya,select as ba}from"@inquirer/prompts";f();import{input as Bm,select as ka}from"@inquirer/prompts";var C=class extends Error{constructor(e){super(e),this.name="UserAbortedRecoveryError"}};async function L(t){s.warn(`${t.taskName} th\u1EA5t b\u1EA1i: ${t.reason}`),t.hint&&s.info(t.hint);let e=[{name:"Th\u1EED l\u1EA1i (Retry)",value:"retry"}];return t.allowSkip&&e.push({name:"B\u1ECF qua b\u01B0\u1EDBc n\xE0y v\xE0 ti\u1EBFp t\u1EE5c (Skip)",value:"skip"}),e.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i sau (Abort)",value:"abort"}),await ka({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}f();function va(t){switch(t){case"name-exists":s.warn("\u26D4 Repo t\xEAn n\xE0y \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub."),s.dim(" \u2192 \u0110\u1ED5i t\xEAn kh\xE1c, ho\u1EB7c d\xF9ng account/org kh\xE1c.");break;case"no-permission":s.warn("\u26D4 Kh\xF4ng c\xF3 quy\u1EC1n t\u1EA1o repo (sai account / org)."),s.dim(" \u2192 Ki\u1EC3m tra: \u0111ang login \u0111\xFAng GitHub account ch\u01B0a? (gh auth status)");break;case"network":s.warn("\u26D4 L\u1ED7i m\u1EA1ng khi t\u1EA1o repo.");break;default:s.warn("\u26D4 T\u1EA1o repo th\u1EA5t b\u1EA1i (l\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh)."),s.dim(" \u2192 Ki\u1EC3m tra gh auth status + th\u1EED l\u1EA1i.")}}async function xa(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 ba({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}async function Kr(t,e){let n=t;for(;;)try{return await Hr(n,e)}catch(r){let o=r instanceof Gt?r.reason:"unknown",i=await Vr(o,n);if(i.action==="retry")continue;n=i.newName}}async function Fr(t,e){let n=t;for(;;)try{return e(n)}catch(r){let o=r instanceof Error&&r.name==="RepoAlreadyExistsError"?"name-exists":"unknown",i=await Vr(o,n);if(i.action==="retry")continue;n=i.newName}}async function Vr(t,e){va(t);let n=await xa(t);if(n==="abort")throw new C("User h\u1EE7y t\u1EA1i b\u01B0\u1EDBc t\u1EA1o repo GitHub.");return n==="switch-account"?(vt(),{action:"retry",newName:e}):n==="rename"?{action:"rename",newName:(await ya({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 Br}from"child_process";import{platform as Aa}from"os";function lt(){let t=Aa();return t==="darwin"||t==="linux"||t==="win32"?t:"unsupported"}var Ca=5e3,Sa=/(\d+\.\d+\.\d+)/;function Ta(){let e=lt()==="win32"?"where":"which",n=Br(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 Ea(){let t=Br("gitnexus",["--version"],{encoding:"utf8",timeout:Ca});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return Sa.exec(e)?.[1]??null}var ut=null;function pt(){if(ut!==null)return ut;let t=Ta();return t?(ut={installed:!0,version:Ea(),path:t},ut):(ut={installed:!1,version:null,path:null},ut)}function le(){ut=null}import{spawnSync as Pa}from"child_process";function ue(){let t=Pa("gh",["auth","status"],{stdio:"ignore"});return t.error&&t.error.code==="ENOENT"?"not-installed":t.status===0?"authenticated":"not-authenticated"}import{spawnSync as Ra}from"child_process";function $a(t){let e=lt();return Ra(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function Wr(){let t=lt(),e=t==="darwin"?["brew"]:t==="win32"?["winget"]:t==="linux"?["apt","dnf","pacman"]:[];for(let n of e)if($a(n))return n;return null}import{spawnSync as eo}from"child_process";import{input as Ka,select as Fa}from"@inquirer/prompts";vn();w();f();import{spawnSync as xn}from"child_process";import{promises as Oa}from"fs";import{basename as La,join as Zr}from"path";import{confirm as Ga,select as ja}from"@inquirer/prompts";function Ua(){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 Da(t){let e=Zr(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-${Ua()}`,r=Zr(t,n);return await Oa.rename(e,r),s.success(`Backup .git \u2192 ${n}`),r}function Ha(t){let e=xn("git",["-C",t,"init","-b","main"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(e.status!==0)throw new Error(`git init th\u1EA5t b\u1EA1i: ${e.stderr||e.stdout}`);s.success("Git init m\u1EDBi (branch main)");let n=xn("git",["-C",t,"add","-A"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});n.status!==0&&s.warn(`git add -A th\u1EA5t b\u1EA1i: ${(n.stderr||"").slice(0,200)}`);let r=xn("git",["-C",t,"commit","-m","chore: initial commit from existing folder"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});r.status!==0?s.warn(`git commit th\u1EA5t b\u1EA1i (folder c\xF3 th\u1EC3 r\u1ED7ng): ${(r.stderr||"").slice(0,200)}`):s.success("Initial commit")}async function to(t){let e=La(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await Ga({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
23
+ `)}});import{spawnSync as xa}from"child_process";function le(){let t=xa("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"],timeout:1e4});if(t.signal==="SIGTERM"||t.error?.code==="ETIMEDOUT")throw new Error("Qu\xE1 th\u1EDDi gian l\u1EA5y GitHub username (10s). Ki\u1EC3m tra m\u1EA1ng / k\u1EBFt n\u1ED1i GitHub r\u1ED3i th\u1EED l\u1EA1i.");if(t.status!==0)throw new Error(`Kh\xF4ng l\u1EA5y \u0111\u01B0\u1EE3c GitHub username: ${t.stderr?.trim()}`);return t.stdout.trim()}var hn=_(()=>{"use strict"});import{spawnSync as Aa}from"child_process";function wn(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 Kr(t,e,n){let o=`${n??le()}/${t}`;s.info(`T\u1EA1o GitHub repo r\u1ED7ng ${o} (${e})...`);let i=Aa("gh",["repo","create",o,`--${e}`],{stdio:["ignore","pipe","pipe"],encoding:"utf8"});if(i.status!==0){let a=(i.stderr||"").trim();throw a&&process.stderr.write(`${a}
24
+ `),new Gt(wn(a),`T\u1EA1o repo ${o} th\u1EA5t b\u1EA1i.`)}return s.success(`\u0110\xE3 t\u1EA1o: git@github.com:${o}.git`),`git@github.com:${o}.git`}var Gt,kn=_(()=>{"use strict";hn();f();Gt=class extends Error{reason;constructor(e,n){super(n),this.name="RepoCreateError",this.reason=e}}});import{spawnSync as Ga}from"child_process";import{existsSync as Ua,statSync as ja}from"fs";function zr(t){let e=`${t.org}/${t.name}`;if(!Ua(t.folder)||!ja(t.folder).isDirectory())throw new At("missing-folder",`Folder ngu\u1ED3n kh\xF4ng t\u1ED3n t\u1EA1i: ${t.folder}. Ki\u1EC3m tra l\u1EA1i \u0111\u01B0\u1EDDng d\u1EABn.`);let n=["repo","create",e,`--${t.visibility}`,"--source",t.folder,"--remote","origin","--push"],r=Ga("gh",n,{stdio:["ignore","pipe","pipe"],encoding:"utf8"});if(r.status!==0){let o=(r.stderr||"").trim();o&&process.stderr.write(`${o}
25
+ `);let i=wn(o);throw i==="name-exists"?new yn(e):i==="no-permission"?new At("no-permission",`Kh\xF4ng c\xF3 quy\u1EC1n t\u1EA1o repo ${e}. Ki\u1EC3m tra \u0111\u0103ng nh\u1EADp \u0111\xFAng account/org ch\u01B0a (gh auth status).`):i==="network"?new At("network",`L\u1ED7i m\u1EA1ng khi t\u1EA1o repo ${e}.`):new At("unknown",`gh repo create th\u1EA5t b\u1EA1i (exit ${r.status}).`)}return{sshUrl:`git@github.com:${e}.git`,httpsUrl:`https://github.com/${e}.git`}}var yn,At,Jr=_(()=>{"use strict";kn();yn=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"}},At=class extends Error{reason;constructor(e,n){super(n),this.name="GhRepoCreateError",this.reason=e}}});function Yr(t){if(!Da.test(t))throw new bn(t)}function Xr(t){if(t!=="private"&&t!=="public")throw new Error(`Visibility ph\u1EA3i l\xE0 "private" ho\u1EB7c "public", nh\u1EADn: "${t}"`)}var Da,bn,Qr=_(()=>{"use strict";Da=/^[a-zA-Z0-9._-]{1,100}$/,bn=class extends Error{constructor(e){super(`T\xEAn repo "${e}" kh\xF4ng h\u1EE3p l\u1EC7. Ch\u1EC9 d\xF9ng ch\u1EEF/s\u1ED1/d\u1EA5u ch\u1EA5m/g\u1EA1ch/underscore, d\xE0i 1-100 k\xFD t\u1EF1.`),this.name="InvalidRepoNameError"}}});var Zr={};W(Zr,{createGithubRemoteFromFolder:()=>vn});function vn(t){Yr(t.name),Xr(t.visibility);let e=t.org??le();s.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=zr({folder:t.folder,org:e,name:t.name,visibility:t.visibility});return s.success(`\u0110\xE3 t\u1EA1o: ${n.sshUrl}`),n}var xn=_(()=>{"use strict";Jr();hn();f();Qr()});var Dn={};W(Dn,{ALLOWED_HOSTED_DOMAINS:()=>ye,HOSTED_DOMAIN:()=>ll,SCOPES:()=>Ko,buildUserConfig:()=>Un,buildVerificationUrl:()=>jn,decodeIdToken:()=>Ln,pollForToken:()=>On,refreshAccessToken:()=>gl,requestDeviceCode:()=>Mn,revokeToken:()=>Bt,verifyHostedDomain:()=>Gn,verifyIdTokenClaims:()=>Fo});async function Mn(){let t=new URLSearchParams({client_id:be,scope:Ko.join(" ")}),e=await fetch(ul,{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 On(t){let e=new URLSearchParams({client_id:be,client_secret:Ho,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(Vo,{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 Fo(t){if(!ml.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==be)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+dl<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||!ye.includes(t.hd))throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u ${ye.map(n=>`@${n}`).join(" ho\u1EB7c ")}). Nh\u1EADn: ${t.email}${t.hd?` (hd=${t.hd})`:" (kh\xF4ng ph\u1EA3i t\xE0i kho\u1EA3n Workspace)"}`);if(!t.email_verified)throw new Error("Email ch\u01B0a \u0111\u01B0\u1EE3c Google verify")}function Un(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 gl(t){let e=new URLSearchParams({client_id:be,client_secret:Ho,refresh_token:t,grant_type:"refresh_token"}),n=await fetch(Vo,{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 Bt(t){let e=new URLSearchParams({token:t}),n=new AbortController,r=setTimeout(()=>n.abort(),1e4);await fetch(pl,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e,signal:n.signal}).catch(()=>{}).finally(()=>clearTimeout(r))}function jn(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.toString()}var be,Ho,ye,ll,Ko,ul,Vo,pl,ml,dl,Gn,Wt=_(()=>{"use strict";be="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",Ho="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",ye=["nal.vn","nal-software.com"],ll=ye[0],Ko=["openid","email","profile"],ul="https://oauth2.googleapis.com/device/code",Vo="https://oauth2.googleapis.com/token",pl="https://oauth2.googleapis.com/revoke";ml=new Set(["https://accounts.google.com","accounts.google.com"]),dl=60;Gn=Fo});import{Command as nm}from"commander";Ot();import{resolve as wo}from"path";import{confirm as Kt,input as Ht,select as Sn}from"@inquirer/prompts";Ot();w();f();import{join as wa}from"path";import{confirm as ka,input as ce,select as Dr}from"@inquirer/prompts";f();import{spawnSync as ga}from"child_process";function vt(){s.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp GitHub qua gh CLI (browser s\u1EBD m\u1EDF)...");let t=ga("gh",["auth","login","--hostname","github.com","--web","--git-protocol","ssh"],{stdio:"inherit"});if(t.status!==0)throw new Error(`gh auth login th\u1EA5t b\u1EA1i (exit ${t.status}). Th\u1EED 'gh auth login' tay.`);s.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp GitHub")}import{spawnSync as fa}from"child_process";var fn=5e3;function ha(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 xt(t){let e=fa("git",["ls-remote","--exit-code",t,"HEAD"],{encoding:"utf8",timeout:fn,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 > ${fn/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 > ${fn/1e3}s`};let n=(e.stderr||"").trim();return{ok:!1,reason:ha(n),detail:n.slice(0,300)}}function ya(t,e,n){switch(s.warn(`\u26D4 Kh\xF4ng clone \u0111\u01B0\u1EE3c ${t}`),e){case"not-found":s.dim(" \u2192 Repo kh\xF4ng t\u1ED3n t\u1EA1i (sai URL, ho\u1EB7c repo ch\u01B0a \u0111\u01B0\u1EE3c t\u1EA1o tr\xEAn GitHub).");break;case"no-access":s.dim(` \u2192 Kh\xF4ng c\xF3 quy\u1EC1n truy c\u1EADp (repo private + sai account), HO\u1EB6C repo kh\xF4ng t\u1ED3n t\u1EA1i.
26
+ Ki\u1EC3m tra: \u0111ang login \u0111\xFAng GitHub account ch\u01B0a? (gh auth status)`);break;case"network":s.dim(" \u2192 L\u1ED7i m\u1EA1ng (m\u1EA5t k\u1EBFt n\u1ED1i / DNS / timeout).");break;default:s.dim(` \u2192 L\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh${n?`: ${n}`:""}.`)}}function Ur(){s.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");try{vt()}catch(t){s.warn(`${t instanceof Error?t.message:t}. C\xF3 th\u1EC3 ch\u1EA1y 'gh auth login' tay r\u1ED3i retry.`)}}async function jr(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 Dr({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}async function Hr(t){let e=t.url,n=t.name;for(;;){let r=xt(e);if(!r.ok){ya(e,r.reason??"unknown",r.detail);let o=await jr(r.reason??"unknown");if(o==="skip")return s.dim(`B\u1ECF qua repo ${n}.`),{cloned:null,skipped:!0};if(o==="switch-account"){Ur();continue}if(o==="new-url"){if(e=await ce({message:"URL repo kh\xE1c:",validate:a=>a.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),await ka({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(()=>(Ot(),gn));n=await ce({message:"T\xEAn th\u01B0 m\u1EE5c:",default:a(e)})}continue}continue}try{return{cloned:await un({workspaceRoot:t.workspaceRoot,url:e,name:n}),skipped:!1}}catch(o){let i=o instanceof Error?o.message:String(o);if(o instanceof lt){let c=await ba(t.workspaceRoot,n);if(c&&va(c,e))return s.warn(`\u26A0 Repo n\xE0y \u0111\xE3 c\xF3 trong workspace (src/${n}). D\xF9ng tr\u1EF1c ti\u1EBFp, kh\xF4ng add l\u1EA1i.`),s.dim(` Mu\u1ED1n thay b\u1EB1ng b\u1EA3n m\u1EDBi? G\u1EE1 tr\u01B0\u1EDBc: avatar remove repo ${n}`),{cloned:null,skipped:!0};if(s.warn(`\u26A0 ${i}`),await Dr({message:"C\xE1ch x\u1EED l\xFD?",choices:[{name:"\u0110\u1ED5i t\xEAn th\u01B0 m\u1EE5c kh\xE1c",value:"rename"},{name:"B\u1ECF qua repo n\xE0y",value:"skip"}]})==="skip")return{cloned:null,skipped:!0};n=await ce({message:"T\xEAn th\u01B0 m\u1EE5c m\u1EDBi trong src/:",validate:u=>u.trim().length>0?!0:"T\xEAn b\u1EAFt bu\u1ED9c"});continue}s.warn(`Clone fail: ${i}`),await st(wa(t.workspaceRoot,"src",n)).catch(()=>{});let a=await jr("unknown");if(a==="skip")return{cloned:null,skipped:!0};if(a==="switch-account"){Ur();continue}a==="new-url"&&(e=await ce({message:"URL repo kh\xE1c:",validate:c=>c.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}))}}}async function ba(t,e){let{readReposManifest:n}=await Promise.resolve().then(()=>(q(),at));return(await n(t)).find(o=>o.name===e)?.url??null}function va(t,e){let n=r=>r.trim().toLowerCase().replace(/^git@/,"").replace(/^https?:\/\//,"").replace(/:/,"/").replace(/\.git$/,"").replace(/\/+$/,"");return n(t)===n(e)}kn();import{input as Sa,select as Ta}from"@inquirer/prompts";f();import{input as td,select as Ca}from"@inquirer/prompts";var C=class extends Error{constructor(e){super(e),this.name="UserAbortedRecoveryError"}};async function L(t){s.warn(`${t.taskName} th\u1EA5t b\u1EA1i: ${t.reason}`),t.hint&&s.info(t.hint);let e=[{name:"Th\u1EED l\u1EA1i (Retry)",value:"retry"}];return t.allowSkip&&e.push({name:"B\u1ECF qua b\u01B0\u1EDBc n\xE0y v\xE0 ti\u1EBFp t\u1EE5c (Skip)",value:"skip"}),e.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i sau (Abort)",value:"abort"}),await Ca({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}f();function Ea(t){switch(t){case"name-exists":s.warn("\u26D4 Repo t\xEAn n\xE0y \u0111\xE3 t\u1ED3n t\u1EA1i tr\xEAn GitHub."),s.dim(" \u2192 \u0110\u1ED5i t\xEAn kh\xE1c, ho\u1EB7c d\xF9ng account/org kh\xE1c.");break;case"no-permission":s.warn("\u26D4 Kh\xF4ng c\xF3 quy\u1EC1n t\u1EA1o repo (sai account / org)."),s.dim(" \u2192 Ki\u1EC3m tra: \u0111ang login \u0111\xFAng GitHub account ch\u01B0a? (gh auth status)");break;case"network":s.warn("\u26D4 L\u1ED7i m\u1EA1ng khi t\u1EA1o repo.");break;default:s.warn("\u26D4 T\u1EA1o repo th\u1EA5t b\u1EA1i (l\u1ED7i kh\xF4ng x\xE1c \u0111\u1ECBnh)."),s.dim(" \u2192 Ki\u1EC3m tra gh auth status + th\u1EED l\u1EA1i.")}}async function Pa(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 Ta({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}async function Vr(t,e){let n=t;for(;;)try{return await Kr(n,e)}catch(r){let o=r instanceof Gt?r.reason:"unknown",i=await Br(o,n);if(i.action==="retry")continue;n=i.newName}}async function Fr(t,e){let n=t;for(;;)try{return e(n)}catch(r){let o=r instanceof Error&&r.name==="RepoAlreadyExistsError"?"name-exists":"unknown",i=await Br(o,n);if(i.action==="retry")continue;n=i.newName}}async function Br(t,e){Ea(t);let n=await Pa(t);if(n==="abort")throw new C("User h\u1EE7y t\u1EA1i b\u01B0\u1EDBc t\u1EA1o repo GitHub.");return n==="switch-account"?(vt(),{action:"retry",newName:e}):n==="rename"?{action:"rename",newName:(await Sa({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 Wr}from"child_process";import{platform as Ra}from"os";function D(){let t=Ra();return t==="darwin"||t==="linux"||t==="win32"?t:"unsupported"}var $a=5e3,_a=/(\d+\.\d+\.\d+)/;function Ia(){let e=D()==="win32"?"where":"which",n=Wr(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 Na(){let t=Wr("gitnexus",["--version"],{encoding:"utf8",timeout:$a});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return _a.exec(e)?.[1]??null}var ut=null;function pt(){if(ut!==null)return ut;let t=Ia();return t?(ut={installed:!0,version:Na(),path:t},ut):(ut={installed:!1,version:null,path:null},ut)}function ue(){ut=null}import{spawnSync as Ma}from"child_process";function pe(){let t=Ma("gh",["auth","status"],{stdio:"ignore"});return t.error&&t.error.code==="ENOENT"?"not-installed":t.status===0?"authenticated":"not-authenticated"}import{spawnSync as Oa}from"child_process";function La(t){let e=D();return Oa(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function qr(){let t=D(),e=t==="darwin"?["brew"]:t==="win32"?["winget"]:t==="linux"?["apt","dnf","pacman"]:[];for(let n of e)if(La(n))return n;return null}import{spawnSync as no}from"child_process";import{input as za,select as Ja}from"@inquirer/prompts";xn();w();f();import{spawnSync as An}from"child_process";import{promises as Ha}from"fs";import{basename as Ka,join as to}from"path";import{confirm as Va,select as Fa}from"@inquirer/prompts";function Ba(){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 Wa(t){let e=to(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-${Ba()}`,r=to(t,n);return await Ha.rename(e,r),s.success(`Backup .git \u2192 ${n}`),r}function qa(t){let e=An("git",["-C",t,"init","-b","main"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});if(e.status!==0)throw new Error(`git init th\u1EA5t b\u1EA1i: ${e.stderr||e.stdout}`);s.success("Git init m\u1EDBi (branch main)");let n=An("git",["-C",t,"add","-A"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});n.status!==0&&s.warn(`git add -A th\u1EA5t b\u1EA1i: ${(n.stderr||"").slice(0,200)}`);let r=An("git",["-C",t,"commit","-m","chore: initial commit from existing folder"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});r.status!==0?s.warn(`git commit th\u1EA5t b\u1EA1i (folder c\xF3 th\u1EC3 r\u1ED7ng): ${(r.stderr||"").slice(0,200)}`):s.success("Initial commit")}async function eo(t){let e=Ka(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await Va({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
27
27
  1. Backup .git \u2192 .git.backup-{timestamp}
28
28
  2. Git init m\u1EDBi (branch main, initial commit)
29
29
  3. T\u1EA1o GitHub repo m\u1EDBi '${n}' d\u01B0\u1EDBi account c\u1EE7a b\u1EA1n
30
- Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let r=t.visibility??(t.autoYes?"private":await ja({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),o=await Da(t.folderPath);return Ha(t.folderPath),{newRemoteUrl:bn({folder:t.folderPath,name:n,visibility:r,org:t.org}).httpsUrl,backupPath:o}}f();var jt=class extends Error{constructor(e){super(e),this.name="RemoteAccessAbortedError"}};function Va(){let t=eo("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function Ba(){s.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=eo("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&s.warn(`gh auth login exit ${t.status}. B\u1EA1n c\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function Wa(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 qa(t){let e=t.trim();return e?/^https?:\/\/[\w.@/-]+$/.test(e)||/^git@[\w.-]+:[\w./-]+\.git$/.test(e)||/^[\w.-]+\/[\w.-]+$/.test(e):!1}async function no(t){let e=t.url,n=t.initialReason,r=t.initialDetail;for(;;){let o=Va();s.warn(`Kh\xF4ng truy c\u1EADp \u0111\u01B0\u1EE3c ${e}`),s.dim(` L\xFD do: ${n}${r?` \u2014 ${r.slice(0,150)}`:""}`),s.info(Wa(n,e,o)),o&&s.dim(` gh CLI hi\u1EC7n \u0111ang login: ${o}`);let i=[{name:"Nh\u1EADp l\u1EA1i URL \u0111\xFAng (recommended khi URL sai ch\xEDnh t\u1EA3)",value:"re-input-url"},{name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch"},{name:"T\xF4i v\u1EEBa fix (accept invite / s\u1EEDa permission) \u2014 retry verify",value:"retry"}];t.folderPath&&i.push({name:"Reset folder & t\u1EA1o repo M\u1EDAI d\u01B0\u1EDBi account c\u1EE7a t\xF4i (backup .git c\u0169)",value:"reset-folder"}),i.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i 'avatar init' sau",value:"abort"});let a=await Fa({message:"C\xE1ch x\u1EED l\xFD?",choices:i});if(a==="abort")throw new jt(`User ch\u1ECDn t\u1EA1m ng\u01B0ng. Fix access ${e} r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar init'.`);if(a==="reset-folder"){if(!t.folderPath){s.warn("Reset folder c\u1EA7n folderPath \u2014 internal error.");continue}try{let l=await to({folderPath:t.folderPath,visibility:t.defaultVisibility});return s.success(`Folder \u0111\xE3 reset. Backup t\u1EA1i: ${l.backupPath}`),s.success(`Remote m\u1EDBi: ${l.newRemoteUrl}`),{resolvedUrl:l.newRemoteUrl}}catch(l){s.warn(`Reset folder th\u1EA5t b\u1EA1i: ${l.message}`);continue}}a==="re-input-url"&&(e=(await Ka({message:"URL git remote (https://github.com/owner/repo.git ho\u1EB7c git@github.com:owner/repo.git):",default:e,validate:u=>qa(u)||"URL kh\xF4ng \u0111\xFAng format git remote"})).trim()),a==="switch"&&Ba(),s.info(`Verify remote l\u1EA1i: ${e}...`);let c=xt(e);if(c.ok)return s.success(`Remote accessible: ${e}`),{resolvedUrl:e};n=c.reason??"unknown",r=c.detail}}f();import{spawnSync as za}from"child_process";var Ja={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 ro(t){let e=Ja[t];s.info(`\u0110ang c\xE0i gh CLI qua ${t}...`);let n=za(e.cmd,e.args,{stdio:"inherit"});if(n.status!==0)throw new Error(`C\xE0i gh CLI th\u1EA5t b\u1EA1i qua ${t} (exit ${n.status}). C\xE0i tay r\u1ED3i ch\u1EA1y l\u1EA1i.`);s.success("\u0110\xE3 c\xE0i gh CLI")}f();import{spawnSync as Ya}from"child_process";function oo(){if(Ya("gh",["auth","setup-git"],{stdio:"ignore"}).status!==0){s.warn("gh auth setup-git fail (non-fatal). N\u1EBFu git clone l\u1ED7i 128 \u2192 ch\u1EA1y th\u1EE7 c\xF4ng.");return}s.dim("Git credential helper \u0111\xE3 link v\u1EDBi gh token.")}f();async function io(t){for(;ue()==="not-installed";){s.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=Wr();if(!e){if(await L({taskName:"Ph\xE1t hi\u1EC7n package manager",reason:"Kh\xF4ng t\xECm th\u1EA5y brew/apt/dnf/pacman/winget tr\xEAn m\xE1y.",allowSkip:!1,hint:"C\xE0i gh CLI tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry."})==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{ro(e)}catch(n){if(await L({taskName:`C\xE0i gh CLI qua ${e}`,reason:n.message,allowSkip:!1,hint:"C\xE0i tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry, ho\u1EB7c Abort."})==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;ue()==="not-authenticated";){s.warn("Ch\u01B0a \u0111\u0103ng nh\u1EADp GitHub.");try{vt()}catch(e){if(await L({taskName:"\u0110\u0103ng nh\u1EADp GitHub qua gh",reason:e.message,allowSkip:!1,hint:"Th\u1EED l\u1EA1i \u2014 ch\u1ECDn c\xE1ch login kh\xE1c (browser vs token) khi gh prompt."})==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(ue()!=="authenticated"&&await L({taskName:"Verify gh auth",reason:"Sau gh auth login v\u1EABn b\xE1o not-authenticated.",allowSkip:!1,hint:"C\xF3 th\u1EC3 user cancel browser. Th\u1EED l\u1EA1i ho\u1EB7c abort."})==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(s.success("gh CLI s\u1EB5n s\xE0ng"),oo(),t){let e=xt(t);return e.ok?(s.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await no({url:t,initialReason:e.reason??"unknown",initialDetail:e.detail})).resolvedUrl}}return{}}import{existsSync as Ct}from"fs";import{dirname as Xa,join as St}from"path";var Qa=5;function Za(t){if(Ct(St(t,".claude","avatar.json")))return!0;let e=Ct(St(t,".claude")),n=Ct(St(t,"CLAUDE.md")),r=Ct(St(t,"src"));if(!(e&&n&&r))return!1;let o=Ct(St(t,".claude","repos.json")),i=Ct(St(t,".claude","pack"));return o||i}function Tt(t){let e=t;for(let n=0;n<Qa;n++){if(Za(e))return e;let r=Xa(e);if(r===e)return null;e=r}return null}f();import{spawnSync as so}from"child_process";import{appendFileSync as tc,existsSync as ao,readFileSync as ec,writeFileSync as nc}from"fs";import{join as An}from"path";var rc=120*1e3,oc=300*1e3,z=class extends Error{operation;reason;exitCode;stderr;constructor(e,n,r,o=null,i){super(r),this.name="GitnexusOperationError",this.operation=e,this.reason=n,this.exitCode=o,this.stderr=i}};function co(t,e,n,r){if(n==="SIGTERM")return new z(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 z(t,"permission",`gitnexus ${t} fail (permission). Check write access ~/.claude ho\u1EB7c cwd.`,e,r):new z(t,"non-zero-exit",`gitnexus ${t} exit ${e??"null"}. Xem log ph\xEDa tr\xEAn.`,e,r)}function pe(t,e){return t.split(`
30
+ Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let r=t.visibility??(t.autoYes?"private":await Fa({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),o=await Wa(t.folderPath);return qa(t.folderPath),{newRemoteUrl:vn({folder:t.folderPath,name:n,visibility:r,org:t.org}).httpsUrl,backupPath:o}}f();var Ut=class extends Error{constructor(e){super(e),this.name="RemoteAccessAbortedError"}};function Ya(){let t=no("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function Xa(){s.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=no("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&s.warn(`gh auth login exit ${t.status}. B\u1EA1n c\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function Qa(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 Za(t){let e=t.trim();return e?/^https?:\/\/[\w.@/-]+$/.test(e)||/^git@[\w.-]+:[\w./-]+\.git$/.test(e)||/^[\w.-]+\/[\w.-]+$/.test(e):!1}async function ro(t){let e=t.url,n=t.initialReason,r=t.initialDetail;for(;;){let o=Ya();s.warn(`Kh\xF4ng truy c\u1EADp \u0111\u01B0\u1EE3c ${e}`),s.dim(` L\xFD do: ${n}${r?` \u2014 ${r.slice(0,150)}`:""}`),s.info(Qa(n,e,o)),o&&s.dim(` gh CLI hi\u1EC7n \u0111ang login: ${o}`);let i=[{name:"Nh\u1EADp l\u1EA1i URL \u0111\xFAng (recommended khi URL sai ch\xEDnh t\u1EA3)",value:"re-input-url"},{name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch"},{name:"T\xF4i v\u1EEBa fix (accept invite / s\u1EEDa permission) \u2014 retry verify",value:"retry"}];t.folderPath&&i.push({name:"Reset folder & t\u1EA1o repo M\u1EDAI d\u01B0\u1EDBi account c\u1EE7a t\xF4i (backup .git c\u0169)",value:"reset-folder"}),i.push({name:"T\u1EA1m ng\u01B0ng init \u2014 ch\u1EA1y l\u1EA1i 'avatar init' sau",value:"abort"});let a=await Ja({message:"C\xE1ch x\u1EED l\xFD?",choices:i});if(a==="abort")throw new Ut(`User ch\u1ECDn t\u1EA1m ng\u01B0ng. Fix access ${e} r\u1ED3i ch\u1EA1y l\u1EA1i 'avatar init'.`);if(a==="reset-folder"){if(!t.folderPath){s.warn("Reset folder c\u1EA7n folderPath \u2014 internal error.");continue}try{let l=await eo({folderPath:t.folderPath,visibility:t.defaultVisibility});return s.success(`Folder \u0111\xE3 reset. Backup t\u1EA1i: ${l.backupPath}`),s.success(`Remote m\u1EDBi: ${l.newRemoteUrl}`),{resolvedUrl:l.newRemoteUrl}}catch(l){s.warn(`Reset folder th\u1EA5t b\u1EA1i: ${l.message}`);continue}}a==="re-input-url"&&(e=(await za({message:"URL git remote (https://github.com/owner/repo.git ho\u1EB7c git@github.com:owner/repo.git):",default:e,validate:u=>Za(u)||"URL kh\xF4ng \u0111\xFAng format git remote"})).trim()),a==="switch"&&Xa(),s.info(`Verify remote l\u1EA1i: ${e}...`);let c=xt(e);if(c.ok)return s.success(`Remote accessible: ${e}`),{resolvedUrl:e};n=c.reason??"unknown",r=c.detail}}f();import{spawnSync as tc}from"child_process";var ec={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 oo(t){let e=ec[t];s.info(`\u0110ang c\xE0i gh CLI qua ${t}...`);let n=tc(e.cmd,e.args,{stdio:"inherit"});if(n.status!==0)throw new Error(`C\xE0i gh CLI th\u1EA5t b\u1EA1i qua ${t} (exit ${n.status}). C\xE0i tay r\u1ED3i ch\u1EA1y l\u1EA1i.`);s.success("\u0110\xE3 c\xE0i gh CLI")}f();import{spawnSync as nc}from"child_process";function io(){if(nc("gh",["auth","setup-git"],{stdio:"ignore"}).status!==0){s.warn("gh auth setup-git fail (non-fatal). N\u1EBFu git clone l\u1ED7i 128 \u2192 ch\u1EA1y th\u1EE7 c\xF4ng.");return}s.dim("Git credential helper \u0111\xE3 link v\u1EDBi gh token.")}f();async function so(t){for(;pe()==="not-installed";){s.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=qr();if(!e){if(await L({taskName:"Ph\xE1t hi\u1EC7n package manager",reason:"Kh\xF4ng t\xECm th\u1EA5y brew/apt/dnf/pacman/winget tr\xEAn m\xE1y.",allowSkip:!1,hint:"C\xE0i gh CLI tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry."})==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{oo(e)}catch(n){if(await L({taskName:`C\xE0i gh CLI qua ${e}`,reason:n.message,allowSkip:!1,hint:"C\xE0i tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry, ho\u1EB7c Abort."})==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;pe()==="not-authenticated";){s.warn("Ch\u01B0a \u0111\u0103ng nh\u1EADp GitHub.");try{vt()}catch(e){if(await L({taskName:"\u0110\u0103ng nh\u1EADp GitHub qua gh",reason:e.message,allowSkip:!1,hint:"Th\u1EED l\u1EA1i \u2014 ch\u1ECDn c\xE1ch login kh\xE1c (browser vs token) khi gh prompt."})==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(pe()!=="authenticated"&&await L({taskName:"Verify gh auth",reason:"Sau gh auth login v\u1EABn b\xE1o not-authenticated.",allowSkip:!1,hint:"C\xF3 th\u1EC3 user cancel browser. Th\u1EED l\u1EA1i ho\u1EB7c abort."})==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(s.success("gh CLI s\u1EB5n s\xE0ng"),io(),t){let e=xt(t);return e.ok?(s.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await ro({url:t,initialReason:e.reason??"unknown",initialDetail:e.detail})).resolvedUrl}}return{}}import{existsSync as Ct}from"fs";import{dirname as rc,join as St}from"path";var oc=5;function ic(t){if(Ct(St(t,".claude","avatar.json")))return!0;let e=Ct(St(t,".claude")),n=Ct(St(t,"CLAUDE.md")),r=Ct(St(t,"src"));if(!(e&&n&&r))return!1;let o=Ct(St(t,".claude","repos.json")),i=Ct(St(t,".claude","pack"));return o||i}function Tt(t){let e=t;for(let n=0;n<oc;n++){if(ic(e))return e;let r=rc(e);if(r===e)return null;e=r}return null}f();import{spawnSync as ao}from"child_process";import{appendFileSync as sc,existsSync as co,readFileSync as ac,writeFileSync as cc}from"fs";import{join as Cn}from"path";var lc=120*1e3,uc=300*1e3,J=class extends Error{operation;reason;exitCode;stderr;constructor(e,n,r,o=null,i){super(r),this.name="GitnexusOperationError",this.operation=e,this.reason=n,this.exitCode=o,this.stderr=i}};function lo(t,e,n,r){if(n==="SIGTERM")return new J(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 J(t,"permission",`gitnexus ${t} fail (permission). Check write access ~/.claude ho\u1EB7c cwd.`,e,r):new J(t,"non-zero-exit",`gitnexus ${t} exit ${e??"null"}. Xem log ph\xEDa tr\xEAn.`,e,r)}function me(t,e){return t.split(`
31
31
  `).slice(-e).join(`
32
- `)}function lo(){let t=Lt("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=so("gitnexus",["setup"],{stdio:["ignore","pipe","pipe"],timeout:rc,encoding:"utf8"});if(e.status!==0||e.signal==="SIGTERM"){t.fail("GitNexus setup failed");let n=(e.stderr||"").trim(),r=(e.stdout||"").trim();throw n?process.stderr.write(`${pe(n,30)}
33
- `):r&&process.stderr.write(`${pe(r,30)}
34
- `),co("setup",e.status,e.signal,n)}t.succeed("GitNexus setup OK (global skills installed)")}function Et(t){let e=Lt(`Analyze ${t} (1-3 ph\xFAt)`),n=so("gitnexus",["analyze",".","--index-only"],{cwd:t,stdio:["ignore","pipe","pipe"],timeout:oc,encoding:"utf8"});if(n.status!==0||n.signal==="SIGTERM"){e.fail("Analyze failed");let o=(n.stderr||"").trim(),i=(n.stdout||"").trim();throw o?process.stderr.write(`${pe(o,30)}
35
- `):i&&process.stderr.write(`${pe(i,30)}
36
- `),co("analyze",n.status,n.signal,o)}let r=An(t,".gitnexus","meta.json");if(!ao(r))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new z("analyze","missing-output",`gitnexus analyze xong nh\u01B0ng kh\xF4ng th\u1EA5y ${r}. Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus fail silent.`);ic(t),e.succeed(`Analyze OK (index t\u1EA1i ${An(t,".gitnexus")}, \u0111\xE3 gitignore)`)}function ic(t){let e=An(t,".gitignore"),n=".gitnexus/";try{let r="";if(ao(e)&&(r=ec(e,"utf8"),/^\.gitnexus\/?\s*$/m.test(r)))return;let i=`${r.length>0&&!r.endsWith(`
32
+ `)}function uo(){let t=Lt("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=ao("gitnexus",["setup"],{stdio:["ignore","pipe","pipe"],timeout:lc,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(`${me(n,30)}
33
+ `):r&&process.stderr.write(`${me(r,30)}
34
+ `),lo("setup",e.status,e.signal,n)}t.succeed("GitNexus setup OK (global skills installed)")}function Et(t){let e=Lt(`Analyze ${t} (1-3 ph\xFAt)`),n=ao("gitnexus",["analyze",".","--index-only"],{cwd:t,stdio:["ignore","pipe","pipe"],timeout:uc,encoding:"utf8"});if(n.status!==0||n.signal==="SIGTERM"){e.fail("Analyze failed");let o=(n.stderr||"").trim(),i=(n.stdout||"").trim();throw o?process.stderr.write(`${me(o,30)}
35
+ `):i&&process.stderr.write(`${me(i,30)}
36
+ `),lo("analyze",n.status,n.signal,o)}let r=Cn(t,".gitnexus","meta.json");if(!co(r))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new J("analyze","missing-output",`gitnexus analyze xong nh\u01B0ng kh\xF4ng th\u1EA5y ${r}. Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus fail silent.`);pc(t),e.succeed(`Analyze OK (index t\u1EA1i ${Cn(t,".gitnexus")}, \u0111\xE3 gitignore)`)}function pc(t){let e=Cn(t,".gitignore"),n=".gitnexus/";try{let r="";if(co(e)&&(r=ac(e,"utf8"),/^\.gitnexus\/?\s*$/m.test(r)))return;let i=`${r.length>0&&!r.endsWith(`
37
37
  `)?`
38
38
  `:""}
39
39
  # GitNexus index (Avatar) \u2014 database local, kh\xF4ng commit.
40
40
  ${n}
41
- `;r.length>0?tc(e,i):nc(e,`# GitNexus index (Avatar) \u2014 database local, kh\xF4ng commit.
41
+ `;r.length>0?sc(e,i):cc(e,`# GitNexus index (Avatar) \u2014 database local, kh\xF4ng commit.
42
42
  ${n}
43
- `)}catch{}}import{spawnSync as mc}from"child_process";import{existsSync as dc}from"fs";import{join as fo}from"path";import{confirm as gc}from"@inquirer/prompts";var sc=[/^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 uo(t){return t?sc.some(e=>e.test(t)):!1}w();f();import{promises as Ut}from"fs";import{homedir as ac}from"os";import{dirname as cc,join as po}from"path";var lc=po(ac(),".gitnexus"),Dt=po(lc,"config.json");async function uc(){try{let t=await Ut.readFile(Dt,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function pc(t){await Ut.mkdir(cc(Dt),{recursive:!0});let e=`${Dt}.tmp-${process.pid}`;await Ut.writeFile(e,t,{mode:384}),await Ut.rename(e,Dt)}async function mo(t){let n={...await uc(),apiKey:t.apiKey,baseUrl:t.baseUrl,model:t.model,isReasoningModel:t.isReasoningModel??!1};await pc(JSON.stringify(n,null,2)),await Ut.chmod(Dt,384).catch(()=>{})}var fc=900*1e3,hc="nal-claude",wc="claude-sonnet-4-5";function kc(t){let e=t.replace(/\/+$/,"");return e.endsWith("/v1")?`${e}/`:e.endsWith("/v1/")?e:`${e}/v1/`}async function yc(t){let e=fo(t,".claude","settings.json");if(!await m(e))return null;try{let n=await y(e),r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:null;if(!o)return null;let i=typeof n.model=="string"?n.model:"",a=typeof r.ANTHROPIC_MODEL=="string"?r.ANTHROPIC_MODEL:"",c=i.length>0?i:a,l=h=>{let b=process.env[h];if(typeof b=="string"&&b.length>0)return b;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:kc(o),model:c.length>0?c:wc}}else{let h=l("ANTHROPIC_AUTH_TOKEN");if(h)return{provider:"llmlite",apiKey:h,baseUrl:o,model:c.length>0?c:hc}}return null}catch{return null}}async function bc(t,e){return await gc({message:`Generate wiki cho workspace? (~$0.50 qua ${t} model=${e}, 2-5 ph\xFAt). Continue?`,default:!0})}function go(t,e){return t.split(`
43
+ `)}catch{}}import{spawnSync as kc}from"child_process";import{existsSync as yc}from"fs";import{join as ho}from"path";import{confirm as bc}from"@inquirer/prompts";var mc=[/^claude-(opus|sonnet)-4/i,/^claude-(opus|sonnet|haiku)-[5-9]/i,/^o1(-|$)/i,/^o3(-|$)/i,/^o4(-|$)/i,/nal-claude-(opus|sonnet)-?4/i];function po(t){return t?mc.some(e=>e.test(t)):!1}w();f();import{promises as jt}from"fs";import{homedir as dc}from"os";import{dirname as gc,join as mo}from"path";var fc=mo(dc(),".gitnexus"),Dt=mo(fc,"config.json");async function hc(){try{let t=await jt.readFile(Dt,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function wc(t){await jt.mkdir(gc(Dt),{recursive:!0});let e=`${Dt}.tmp-${process.pid}`;await jt.writeFile(e,t,{mode:384}),await jt.rename(e,Dt)}async function go(t){let n={...await hc(),apiKey:t.apiKey,baseUrl:t.baseUrl,model:t.model,isReasoningModel:t.isReasoningModel??!1};await wc(JSON.stringify(n,null,2)),await jt.chmod(Dt,384).catch(()=>{})}var vc=900*1e3,xc="nal-claude",Ac="claude-sonnet-4-5";function Cc(t){let e=t.replace(/\/+$/,"");return e.endsWith("/v1")?`${e}/`:e.endsWith("/v1/")?e:`${e}/v1/`}async function Sc(t){let e=ho(t,".claude","settings.json");if(!await m(e))return null;try{let n=await y(e),r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:null;if(!o)return null;let i=typeof n.model=="string"?n.model:"",a=typeof r.ANTHROPIC_MODEL=="string"?r.ANTHROPIC_MODEL:"",c=i.length>0?i:a,l=h=>{let b=process.env[h];if(typeof b=="string"&&b.length>0)return b;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:Cc(o),model:c.length>0?c:Ac}}else{let h=l("ANTHROPIC_AUTH_TOKEN");if(h)return{provider:"llmlite",apiKey:h,baseUrl:o,model:c.length>0?c:xc}}return null}catch{return null}}async function Tc(t,e){return await bc({message:`Generate wiki cho workspace? (~$0.50 qua ${t} model=${e}, 2-5 ph\xFAt). Continue?`,default:!0})}function fo(t,e){return t.split(`
44
44
  `).slice(-e).join(`
45
- `)}async function me(t,e=t){let n=await yc(t);if(!n)return s.warn(`Kh\xF4ng resolve \u0111\u01B0\u1EE3c API key cho wiki gen.
45
+ `)}async function de(t,e=t){let n=await Sc(t);if(!n)return s.warn(`Kh\xF4ng resolve \u0111\u01B0\u1EE3c API key cho wiki gen.
46
46
  \u0110\xE3 ki\u1EC3m tra: process.env \u2192 .envrc Avatar block \u2192 settings.json.env
47
47
  Nguy\xEAn nh\xE2n c\xF3 th\u1EC3:
48
48
  \u2022 Subscription mode (OAuth, kh\xF4ng c\xF3 key)
49
49
  \u2022 Key ch\u01B0a setup \u2014 ch\u1EA1y 'avatar ai setup' ho\u1EB7c 'avatar secrets set ANTHROPIC_API_KEY'
50
50
  \u2022 settings.json.env.ANTHROPIC_BASE_URL b\u1ECB thi\u1EBFu (provider ch\u01B0a configured)`),s.dim(`\u0110\u1EC3 gen wiki sau khi \u0111\xE3 c\xF3 key, ch\u1EA1y manual:
51
- gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await bc(n.baseUrl,n.model))return s.dim("User decline wiki gen \u2014 workspace OK kh\xF4ng c\xF3 wiki. Ch\u1EA1y `gitnexus wiki` manual sau khi c\u1EA7n."),{ran:!1,skipped:!0,reason:"user-declined"};let o=uo(n.model);await mo({apiKey:n.apiKey,baseUrl:n.baseUrl,model:n.model,isReasoningModel:o});let i=["wiki",".","--base-url",n.baseUrl,"--model",n.model];o&&i.push("--reasoning-model");let a=Lt(`Generating wiki via ${n.baseUrl} (${n.provider}) model=${n.model}${o?" [reasoning]":""}`),c=mc("gitnexus",i,{cwd:e,stdio:["ignore","pipe","pipe"],timeout:fc,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(`${go(d,30)}
52
- `):g&&process.stderr.write(`${go(g,30)}
53
- `),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${u} (exit ${c.status??"null"})`}}let l=fo(e,".gitnexus","wiki","index.html");return dc(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 ko(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 o=>{try{await vc(o)}catch(i){s.error(i instanceof Error?i.message:String(i)),process.exit(1)}}),t.command("remove").description("G\u1EE1 t\xE0i nguy\xEAn kh\u1ECFi workspace").command("repo <name>").description("G\u1EE1 repo kh\u1ECFi src/ (x\xF3a kh\u1ECFi repos.json + t\xF9y ch\u1ECDn x\xF3a folder)").option("--keep-files","Ch\u1EC9 g\u1EE1 kh\u1ECFi manifest, GI\u1EEE folder src/<name>").action(async(o,i)=>{try{await xc(o,i)}catch(a){s.error(a instanceof Error?a.message:String(a)),process.exit(1)}}),t.command("list").description("Li\u1EC7t k\xEA t\xE0i nguy\xEAn trong workspace").command("repo").description("Li\u1EC7t k\xEA repo trong src/").action(async()=>{try{await Ac()}catch(o){s.error(o instanceof Error?o.message:String(o)),process.exit(1)}})}function yo(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 Sn(){let t=Tt(process.cwd());return t||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
54
- C\u1EA7n .claude/ + CLAUDE.md + src/. Ch\u1EA1y 'avatar init' tr\u01B0\u1EDBc.`),process.exit(1)),t}async function vc(t){let e=Sn();await io();let n=!0,r=t.url,o=t.name;for(;n;){let i=await Cc(r),a=null,c=null;if(i.mode==="new"){let l=o??await wo(i.url),u=await un({workspaceRoot:e,url:i.url,name:l});a=u.path,c=u.name,s.success(`\u2713 T\u1EA1o src/${u.name} (repo m\u1EDBi)`)}else{let l=o??await wo(i.url);s.info(`Clone ${i.url} \u2192 src/${l} ...`);let{cloned:u,skipped:d}=await Dr({workspaceRoot:e,url:i.url,name:l});d||!u?s.dim(`B\u1ECF qua repo ${l}.`):(a=u.path,c=u.name,s.success(`\u2713 \u0110\xE3 clone src/${u.name}`))}if(a&&c&&(await Tc(e,a,t.yes),s.success(`\u2713 Done repo: ${c}`)),r=void 0,o=void 0,t.yes)break;n=await Kt({message:"Add repo kh\xE1c n\u1EEFa?",default:!1})}}async function xc(t,e){let n=Sn(),{validateRepoName:r}=await Promise.resolve().then(()=>(Ot(),dn)),o=r(t);o&&(s.error(`T\xEAn repo kh\xF4ng h\u1EE3p l\u1EC7: ${o}`),process.exit(1));let{readReposManifest:i,removeRepoFromManifest:a}=await Promise.resolve().then(()=>(W(),st)),{removeRecursive:c}=await Promise.resolve().then(()=>(w(),yt)),{join:l,relative:u,sep:d}=await import("path"),h=(await i(n)).find(T=>T.name===t),b=l(n,"src",t),S=l(n,"src"),I=u(S,b);(I===""||I===".."||I.startsWith(`..${d}`)||I.startsWith("/"))&&(s.error(`\u0110\u01B0\u1EDDng d\u1EABn repo escape kh\u1ECFi workspace/src \u2014 t\u1EEB ch\u1ED1i x\xF3a: ${b}`),process.exit(1));let{pathExists:V}=await Promise.resolve().then(()=>(w(),yt)),E=await V(b);if(!h&&!E){s.warn(`Repo "${t}" kh\xF4ng c\xF3 trong workspace (c\u1EA3 manifest l\u1EABn src/).`);return}await a(n,t),s.success(`\u2713 G\u1EE1 "${t}" kh\u1ECFi repos.json`),E&&!e.keepFiles?(s.warn(`\u26A0 Folder src/${t} ch\u1EE9a code. X\xF3a = m\u1EA5t data n\u1EBFu ch\u01B0a push remote.`),await Kt({message:`X\xF3a lu\xF4n folder src/${t}?`,default:!1})?(await c(b),s.success(`\u2713 \u0110\xE3 x\xF3a src/${t}`)):s.dim(`Gi\u1EEF folder src/${t}. (\u0110\xE3 g\u1EE1 kh\u1ECFi manifest \u2014 ch\u1EA1y 'avatar add repo' \u0111\u1EC3 add l\u1EA1i.)`)):E&&e.keepFiles&&s.dim(`Gi\u1EEF folder src/${t} (--keep-files).`)}async function Ac(){let t=Sn(),{readReposManifest:e}=await Promise.resolve().then(()=>(W(),st)),{pathExists:n}=await Promise.resolve().then(()=>(w(),yt)),{join:r}=await import("path"),o=await e(t);if(o.length===0){s.dim("Ch\u01B0a c\xF3 repo n\xE0o. Th\xEAm b\u1EB1ng 'avatar add repo'.");return}s.info(`Repo trong workspace (${o.length}):`);for(let i of o){let c=await n(r(t,"src",i.name))?"\u2713":"\u26A0 folder thi\u1EBFu";console.log(` ${c} ${i.name.padEnd(28)} ${i.url}`)}}async function Cc(t){if(t)return{url:t,mode:"clone"};let e=await Cn({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 Ht({message:"URL git repo:",validate:a=>a.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),mode:"clone"};if(e==="folder"){let{statSync:i}=await import("fs"),a=ho(await Ht({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 i(ho(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 pn(a);if(c&&(s.warn(`\u26A0 Folder c\xF3 thay \u0111\u1ED5i ch\u01B0a commit: ${c.slice(0,120)}`),s.warn(" Clone l\u1EA5y b\u1EA3n REMOTE \u2192 c\xE1c thay \u0111\u1ED5i local n\xE0y s\u1EBD KH\xD4NG c\xF3 trong src/<name>."),!await Kt({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 mn(a);return l?{url:l,mode:"clone"}:{url:await Sc(a),mode:"clone"}}let n=await Ht({message:"T\xEAn repo m\u1EDBi:",validate:yo}),r=await Cn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]});return{url:await Kr(n.trim(),r),mode:"new"}}async function Sc(t){let{basename:e}=await import("path");if(s.warn(`Folder ${t} ch\u01B0a c\xF3 remote origin.`),!await Kt({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(()=>(Xe(),kr)),{pathExists:o}=await Promise.resolve().then(()=>(w(),yt)),i=r(t);await o(`${t}/.git`)||(s.info("Folder ch\u01B0a git init \u2014 \u0111ang init..."),await i.init()),s.info("Commit code hi\u1EC7n t\u1EA1i trong folder (baseline cho push)..."),await i.add("."),await i.commit("chore: initial commit (avatar add repo)").catch(()=>{});let a=await Ht({message:"T\xEAn repo GitHub:",default:e(t),validate:yo}),c=await Cn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),{createGithubRemoteFromFolder:l}=await Promise.resolve().then(()=>(vn(),Qr)),u=await Fr(a.trim(),d=>l({folder:t,name:d,visibility:c}).sshUrl);return s.success(`\u2713 \u0110\xE3 t\u1EA1o remote + push: ${u}`),u}async function wo(t){let e=cn(t);return await Ht({message:"T\xEAn th\u01B0 m\u1EE5c trong src/:",default:e})}async function Tc(t,e,n){if(!pt().installed){s.dim("GitNexus ch\u01B0a c\xE0i \u2014 skip index. C\xE0i qua 'avatar gitnexus install'.");return}if(n||await Kt({message:"Index repo n\xE0y b\u1EB1ng GitNexus?",default:!0})){try{Et(e),s.success(` \u2713 GitNexus indexed src/${e.split("/").pop()}`)}catch(i){s.warn(` ! GitNexus index fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${i instanceof Error?i.message:i}`);return}try{let i=await me(t,e);i.ran?s.success(` \u2713 GitNexus wiki t\u1EA1o cho src/${e.split("/").pop()}`):i.reason==="subscription-mode"&&s.dim(" (Subscription mode \u2014 kh\xF4ng c\xF3 API key cho wiki, skip.)")}catch(i){s.warn(` ! Wiki fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${i instanceof Error?i.message:i}`)}}}w();import{promises as Bl}from"fs";import{join as si}from"path";import{confirm as Wl}from"@inquirer/prompts";import{confirm as tl}from"@inquirer/prompts";w();import{join as Pn}from"path";import{select as Nc}from"@inquirer/prompts";function bo(t){return{ANTHROPIC_DEFAULT_OPUS_MODEL:t.hard,ANTHROPIC_DEFAULT_SONNET_MODEL:t.standard,ANTHROPIC_DEFAULT_HAIKU_MODEL:t.fast}}f();import{input as Ec,password as Pc,select as Rc}from"@inquirer/prompts";var $c="https://ai.nal.vn",vo=1e4;function J(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function _c(){return await Pc({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function Ic(t=$c){return(await Ec({message:"LLMLite base URL:",default:t,validate:n=>/^https?:\/\//.test(n)?!0:"Ph\u1EA3i l\xE0 URL h\u1EE3p l\u1EC7 (http/https)"})).replace(/\/+$/,"")}async function Tn(t,e){let n=new AbortController,r=setTimeout(()=>n.abort(),vo);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`.":`
55
- Hint: check m\u1EA1ng / firewall / VPN, ho\u1EB7c base URL c\xF3 \u0111\xFAng kh\xF4ng.`;throw new Error(`Connect ${t} timeout sau ${vo/1e3}s.${c}`)}let i=o.message||String(o);if(i.toLowerCase().includes("fetch failed")||i.includes("ENOTFOUND")){let c=t.includes("nal.vn")?" (ai.nal.vn c\u1EA7n VPN NAL \u2014 ki\u1EC3m tra VPN \u0111\xE3 b\u1EADt ch\u01B0a)":"";throw new Error(`Network error khi connect ${t}: ${i}${c}`)}throw o}finally{clearTimeout(r)}}async function Mc(t){let e=t.filter(r=>r.toLowerCase().includes("claude"));if(e.length===1){let r=e[0];return s.info(`Auto-pick model: ${r} (ch\u1EC9 1 claude alias tr\xEAn endpoint)`),r}let n=e.length>0?e:t;return await Rc({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function xo(){let t=await _c(),e=await Ic();s.info(`Verify key (${J(t)}) qua ${e}/v1/models...`);let n=await Tn(e,t);s.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await Mc(n);return{apiKey:t,baseUrl:e,model:r,availableModels:n}}f();var Oc=".claude/state/ai-orchestrator-model-map.json",Lc=["opus","sonnet","haiku"];function So(t){return Pn(t,Oc)}async function Gc(t){let e=So(t);if(!await m(e))return null;try{let r=(await y(e)).map;return r&&typeof r.hard=="string"&&typeof r.standard=="string"&&typeof r.fast=="string"?{hard:r.hard,standard:r.standard,fast:r.fast}:null}catch{return null}}async function Ao(t){let e=Pn(t,".claude","settings.json");if(!await m(e))return{provider:"subscription",baseUrl:null,token:null};try{let n=await y(e),r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:null,i=typeof n.avatarProvider=="string"?n.avatarProvider:null;if(!o)return{provider:"subscription",baseUrl:null,token:null};let a=d=>{let g=process.env[d];if(typeof g=="string"&&g.length>0)return g;let h=r[d];return typeof h=="string"&&h.length>0?h:null},c=!1;try{c=new URL(o).hostname==="api.anthropic.com"}catch{c=!1}let l=i==="anthropic"?"anthropic":i==="llmlite"?"llmlite":c?"anthropic":"llmlite",u=a(l==="anthropic"?"ANTHROPIC_API_KEY":"ANTHROPIC_AUTH_TOKEN");return{provider:l,baseUrl:o,token:u}}catch{return{provider:"subscription",baseUrl:null,token:null}}}function Rn(t){let e=t.filter(n=>n.toLowerCase().includes("claude"));return e.length>0?e:t}async function jc(t){if(t.provider==="subscription"||!t.baseUrl||!t.token)return{models:[...Lc],source:"alias"};let e=await Tn(t.baseUrl,t.token);return{models:Rn(e),source:"/v1/models"}}async function En(t,e){return await Nc({message:`Tier ${t} \u2192 ch\u1ECDn model:`,choices:e.map(n=>({name:n,value:n}))})}async function Uc(t,e,n,r){let o={map:e,provider:n,source:r,updatedAt:Math.floor(Date.now()/1e3)};await A(So(t),o)}async function Co(t,e,n){if(n==="subscription")return;let r=bo(e),o=Pn(t,".claude","settings.json");try{if(!await m(o))return;let i=await y(o);i.env={...i.env||{},...r},await A(o,i),s.dim("ai-orchestrator: set env slot tier (OPUS=hard, SONNET=standard, HAIKU=fast) \u2014 alias spawn expand \u0111\xFAng model \u0111\xE3 map.")}catch(i){s.warn(`ai-orchestrator: kh\xF4ng ghi \u0111\u01B0\u1EE3c tier-slot env (${i.message}) \u2014 route c\xF3 th\u1EC3 ch\u1EA1y sai model. B\u1ECF qua, tool v\u1EABn b\u1EADt.`)}}async function To(t,e={}){if(!e.forceRefresh){let a=await Gc(t);if(a){s.dim("ai-orchestrator: state map \u0111\xE3 t\u1ED3n t\u1EA1i \u2014 gi\u1EEF map c\u0169 (kh\xF4ng h\u1ECFi l\u1EA1i).");let c=await Ao(t);return await Co(t,a,c.provider),!0}}if(!process.stdout.isTTY)return s.warn("ai-orchestrator c\u1EA7n interactive setup (ch\u1ECDn model cho 3 tier) \u2014 b\u1ECF qua trong m\xF4i tr\u01B0\u1EDDng non-TTY. Ch\u1EA1y 'avatar tools enable ai-orchestrator' trong terminal th\u1EADt \u0111\u1EC3 c\xE0i."),!1;let n=await Ao(t),r,o;try{let a=await jc(n);r=a.models,o=a.source}catch(a){return s.error(`ai-orchestrator: kh\xF4ng l\u1EA5y \u0111\u01B0\u1EE3c danh s\xE1ch model (${a.message}). Kh\xF4ng b\u1EADt tool. Ki\u1EC3m tra VPN/gateway r\u1ED3i th\u1EED l\u1EA1i.`),!1}if(r.length===1)return s.warn(`ai-orchestrator: key ch\u1EC9 c\xF3 1 model (${r[0]}) \u2014 route v\xF4 ngh\u0129a khi ch\u1EC9 1 model. KH\xD4NG b\u1EADt tool.`),!1;s.info(`ai-orchestrator: ${r.length} model kh\u1EA3 d\u1EE5ng (${n.provider}). Ch\u1ECDn model cho m\u1ED7i lo\u1EA1i vi\u1EC7c (Kh\xF3/V\u1EEBa/D\u1EC5). C\xF3 th\u1EC3 ch\u1ECDn tr\xF9ng.`);let i;try{let a=await En("Vi\u1EC7c kh\xF3 (model m\u1EA1nh)",r),c=await En("Vi\u1EC7c v\u1EEBa (model c\xE2n b\u1EB1ng)",r),l=await En("Vi\u1EC7c d\u1EC5 (model r\u1EBB)",r);i={hard:a,standard:c,fast:l}}catch{return s.dim("ai-orchestrator: h\u1EE7y ch\u1ECDn model \u2014 kh\xF4ng b\u1EADt tool."),!1}try{await Uc(t,i,n.provider,o)}catch(a){return s.error(`ai-orchestrator: ghi state map th\u1EA5t b\u1EA1i (${a.message}). Kh\xF4ng b\u1EADt tool.`),!1}return await Co(t,i,n.provider),s.success(`ai-orchestrator: \u0111\xE3 map tier \u2192 model. Kh\xF3=${i.hard} \xB7 V\u1EEBa=${i.standard} \xB7 D\u1EC5=${i.fast}`),!0}f();import{spawnSync as Qc}from"child_process";w();import{promises as Bc}from"fs";import{join as _o}from"path";w();f();import{promises as Dc}from"fs";import Eo,{join as $n}from"path";async function Hc(t,e){let r=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!r?.[2])return!0;let o=r[2];if(o.startsWith("/"))return s.warn(`Pack hook reject: absolute path "${o}" \u2014 ch\u1EC9 accept relative path t\u1EDBi workspace.`),!1;let i=$n(t,o),a=Eo.resolve(t),c=Eo.resolve(i);return!c.startsWith(`${a}/`)&&c!==a?(s.warn(`Pack hook reject: path "${o}" resolve ra ngo\xE0i workspace (path traversal).`),!1):await m(i)}function _n(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")}${String(e.getSeconds()).padStart(2,"0")}`;return`${t}.backup-${n}`}function Ft(t,e){let n=new Set,r=[];for(let o of[...t,...e]){let i=typeof o=="string"?o:JSON.stringify(o);n.has(i)||(n.add(i),r.push(o))}return r}function Kc(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function Fc(t){if(typeof t!="object"||t===null)return[];let e=t,n=Array.isArray(e.hooks)?e.hooks:[],r=[];for(let o of n)if(typeof o=="object"&&o!==null){let i=o.command;typeof i=="string"&&r.push(i)}return r}function Po(t){let e=t.map((i,a)=>{let c=Fc(i),l=c.some(u=>u.includes("${CLAUDE_PROJECT_DIR}"));return{index:a,entry:i,commands:c,hasVar:l}}),n=new Map;for(let i of e){let a=i.commands.map(Kc).sort().join("|");if(!a)continue;let c=n.get(a)??[];c.push(i),n.set(a,c)}let r=new Set;for(let[,i]of n)if(!(i.length<2||!i.some(c=>c.hasVar)))for(let c of i)c.hasVar||r.add(c.index);return r.size===0?{entries:t,droppedCount:0}:{entries:t.filter((i,a)=>!r.has(a)),droppedCount:r.size}}function Vc(t,e){let n=[],r={...e},o=0;for(let i of Object.keys(r)){let a=r[i]||[],{entries:c,droppedCount:l}=Po(a);l>0&&(r[i]=c,o+=l,n.includes(i)||n.push(i))}for(let[i,a]of Object.entries(t)){let c=r[i]||[],l=Ft(c,a),{entries:u,droppedCount:d}=Po(l);d>0&&(o+=d),u.length!==c.length&&(n.includes(i)||n.push(i)),r[i]=u}return{merged:r,touchedEvents:n,migratedCount:o}}async function de(t){let e=$n(t,".claude","pack","templates","settings.json.tpl"),n=$n(t,".claude","settings.json");if(!await m(e))return{action:"no-pack-template",changes:[]};let r;try{let u=await O(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={},i=!1;if(await m(n)){i=!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(typeof o.disableWorkflows!="boolean"&&(c.disableWorkflows=!0,a.push("disableWorkflows=true (Avatar t\u1EAFt Dynamic Workflows)")),r.statusLine&&!o.statusLine&&(await Hc(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,g=!1;for(let[h,b]of Object.entries(u))typeof b=="string"&&b.includes("llm.nal.vn")&&(u[h]=b.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[h,b]of Object.entries(r.env))h in u||(u[h]=b,g=!0);g&&a.push("env vars added from pack"),(d||g)&&(c.env=u)}if(r.permissions){let u=o.permissions?.allow||[],d=o.permissions?.deny||[],g=r.permissions.allow||[],h=r.permissions.deny||[],b=Ft(u,g),S=Ft(d,h);(b.length!==u.length||S.length!==d.length)&&(c.permissions={allow:b,deny:S},a.push(`permissions union (+${b.length-u.length} allow, +${S.length-d.length} deny)`))}if(r.hooks){let u=o.hooks||{},{merged:d,touchedEvents:g,migratedCount:h}=Vc(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 i&&(l=_n(n),await Dc.copyFile(n,l)),await A(n,c),{action:"merged",backupPath:l,changes:a}}function Wc(t,e){return _o(t,".claude","pack","tools",e,"tool.json")}async function Vt(t,e){let n=Wc(t,e);if(!await m(n))return null;try{return await y(n)}catch{return null}}var qc=[".claude","settings.json"];function Io(t){return _o(t,...qc)}function ge(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function Mo(t){let e=Io(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 No(t,e,n){let r=Io(t),o;return n&&(o=_n(r),await Bc.copyFile(r,o)),await A(r,e),o}function Ro(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function $o(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function zc(t,e){let n=new Set;for(let i of e)$o(i)&&n.add(Ro(i));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(i=>$o(i)?!0:n.has(Ro(i))?(r++,!1):!0),droppedCount:r}}async function fe(t,e){let{settings:n,existed:r}=await Mo(t),o={...n},i=[],a=e.settings.hooks??{};if(Object.keys(a).length>0){let u=n.hooks??{},d={...u},g=[],h=0;for(let[b,S]of Object.entries(a)){let I=u[b]??[],{migratedUser:V,droppedCount:E}=zc(I,S);E>0&&(h+=E);let T=new Set(V.map(ge)),R=S.filter(rt=>!T.has(ge(rt)));(R.length>0||E>0)&&(d[b]=[...V,...R],g.push(b))}g.length>0&&(o.hooks=d,i.push(`hooks added: ${g.join(", ")}`),h>0&&i.push(`migrated ${h} stale relative-path entries \u2192 \\{CLAUDE_PROJECT_DIR\\} version`))}let c=e.settings.permissions?.deny??[];if(c.length>0){let u=n.permissions?.deny??[],d=Ft(u,c);d.length!==u.length&&(o.permissions={...n.permissions,deny:d},i.push(`deny +${d.length-u.length}`))}return i.length===0?{action:"no-change",changes:[]}:{action:"enabled",backupPath:await No(t,o,r),changes:i}}async function Oo(t,e){let{settings:n,existed:r}=await Mo(t);if(!r)return{action:"no-change",changes:[]};let{hooks:o,permissions:i,...a}=n,c={...a},l=[],u=e.settings.hooks??{},d=n.hooks??{},g={},h=[];for(let[E,T]of Object.entries(d)){let R=u[E];if(!R){g[E]=T;continue}let rt=new Set(R.map(ge)),kt=T.filter(ot=>!rt.has(ge(ot)));kt.length!==T.length&&h.push(E),kt.length>0&&(g[E]=kt)}Object.keys(g).length>0&&(c.hooks=g),h.length>0&&l.push(`hooks removed: ${h.join(", ")}`);let b=new Set(e.settings.permissions?.deny??[]),S=n.permissions?.deny??[],I=b.size>0?S.filter(E=>!b.has(E)):S;if(n.permissions){let{deny:E,...T}=n.permissions,R={...T,...I.length>0?{deny:I}:{}};Object.keys(R).length>0&&(c.permissions=R)}return I.length!==S.length&&l.push(`deny -${S.length-I.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await No(t,c,r),changes:l}}f();w();import{join as Jc}from"path";var Yc=".claude/avatar-tools.json";function Lo(){return{tools:{}}}function Go(t){return Jc(t,Yc)}async function Pt(t){let e=Go(t);if(!await m(e))return Lo();try{return{tools:(await y(e)).tools??{}}}catch{return Lo()}}async function Xc(t,e){await A(Go(t),e)}async function he(t,e,n){let r=await Pt(t);return r.tools[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await Xc(t,r),r}async function Rt(t){let e=await Pt(t);return Object.entries(e.tools).filter(([,n])=>n.enabled).map(([n])=>n)}function Zc(t){if(!t)return;let e=Qc(t,["--version"],{stdio:"ignore"});(e.status!==0||e.error)&&s.warn(`Tool y\xEAu c\u1EA7u runtime '${t}' nh\u01B0ng kh\xF4ng t\xECm th\u1EA5y tr\xEAn PATH. Hook c\xF3 th\u1EC3 fail khi ch\u1EA1y. C\xE0i '${t}' r\u1ED3i th\u1EED l\u1EA1i.`)}function jo(t,e){switch(e.action){case"enabled":s.success(`Tool '${t}' enabled (${e.changes.join("; ")}).`),e.backupPath&&s.dim(` Backup: ${e.backupPath}`);break;case"disabled":s.success(`Tool '${t}' disabled (${e.changes.join("; ")}).`),e.backupPath&&s.dim(` Backup: ${e.backupPath}`);break;case"no-change":s.dim(`Tool '${t}': settings.json \u0111\xE3 \u0111\xFAng tr\u1EA1ng th\xE1i, kh\xF4ng thay \u0111\u1ED5i.`);break;case"no-manifest":break}}async function D(t,e,n={}){let r=await Vt(t,e);if(!r)return n.silent||s.warn(`Pack version hi\u1EC7n t\u1EA1i ch\u01B0a h\u1ED7 tr\u1EE3 tool '${e}' (thi\u1EBFu .claude/pack/tools/${e}/tool.json). Ch\u1EA1y 'avatar sync' \u0111\u1EC3 pull pack m\u1EDBi, ho\u1EB7c ki\u1EC3m t\xEAn tool.`),!1;if(Zc(r.requires?.runtime),e==="ai-orchestrator"&&!await To(t,{forceRefresh:n.forceRefresh}))return!1;let o=await fe(t,r);return jo(e,o),await he(t,e,{enabled:!0,version:r.version}),!0}async function In(t,e){let n=await Vt(t,e);if(!n){s.warn(`Kh\xF4ng t\xECm th\u1EA5y manifest tool '${e}' \u0111\u1EC3 bi\u1EBFt entry n\xE0o c\u1EA7n r\xFAt. N\u1EBFu pack \u0111\xE3 \u0111\u1ED5i, state v\u1EABn \u0111\u01B0\u1EE3c set disabled nh\u01B0ng settings.json c\xF3 th\u1EC3 c\xF2n s\xF3t entry.`);let i=(await Pt(t)).tools[e]?.version??"unknown";return await he(t,e,{enabled:!1,version:i}),!1}let r=await Oo(t,n);return jo(e,r),await he(t,e,{enabled:!1,version:n.version}),!0}var el="ai-orchestrator";async function we(t,e,n={}){if(!e||!e.ok)return;let r;if(e.provider==="subscription")r=3;else if(e.provider==="llmlite"||e.provider==="anthropic")r=Rn(e.availableModels??[]).length;else return;if(r<=1){s.dim(`AI Orchestrator: key ch\u1EC9 c\xF3 ${r} model ph\xE2n lo\u1EA1i \u0111\u01B0\u1EE3c \u2014 b\u1ECF qua (c\u1EA7n >1 model).`);return}if(n.autoYes){s.dim("AI Orchestrator: b\u1ECF qua trong ch\u1EBF \u0111\u1ED9 non-interactive \u2014 b\u1EADt sau b\u1EB1ng 'avatar tools enable ai-orchestrator'.");return}let o=n.forceRefresh?"AI Orchestrator: ch\u1ECDn l\u1EA1i model cho m\u1ED7i lo\u1EA1i vi\u1EC7c? (key c\xF3 th\u1EC3 \u0111\xE3 \u0111\u1ED5i)":"B\u1EADt AI Orchestrator? (t\u1EF1 ch\u1ECDn model theo \u0111\u1ED9 kh\xF3 vi\u1EC7c: Vi\u1EC7c kh\xF3 \u2192 model m\u1EA1nh, Vi\u1EC7c d\u1EC5 \u2192 model r\u1EBB)";if(!await tl({message:o,default:!0})){s.dim(" B\u1ECF qua AI Orchestrator (user opt-out).");return}await D(t,el,{forceRefresh:n.forceRefresh})}import{promises as Vo}from"fs";import{homedir as ll}from"os";import{join as zt}from"path";import{z as k}from"zod";var Uo=k.object({email:k.string().email(),name:k.string(),access_token:k.string().min(1),refresh_token:k.string().min(1),expires_at:k.string().datetime(),id_token:k.string().min(1)}),nl=k.object({installed_tools:k.record(k.string(),k.object({version:k.string().optional(),installed_at:k.string().datetime(),install_method:k.string()})).default({}),tool_inputs:k.record(k.string(),k.unknown()).default({})}),mf=k.object({$schema:k.string().optional(),includeCoAuthoredBy:k.boolean().optional(),env:k.record(k.string(),k.string()).default({}),permissions:k.object({allow:k.array(k.string()).default([]),deny:k.array(k.string()).default([])}).partial().optional(),hooks:k.record(k.string(),k.array(k.unknown())).optional(),statusLine:k.object({type:k.string(),command:k.string(),padding:k.number().optional()}).optional()}),df=k.enum(["internal","client","library"]);w();var Jt=zt(ll(),".avatar"),mt=zt(Jt,"config.json"),yf=zt(Jt,"state.json"),Dn=zt(Jt,"audit.log"),bf=zt(Jt,"backups"),ul=384;async function Hn(){await x(Jt)}async function j(){if(!await m(mt))return null;let t=await y(mt),e=Uo.safeParse(t);return e.success?e.data:null}async function Kn(t){await Hn(),await A(mt,t,ul)}async function be(){if(await m(mt)){let{promises:t}=await import("fs");await t.unlink(mt)}}function dt(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}var qt=class extends Error{constructor(e){super(e),this.name="NoValidTokenError"}};async function pl(t){let{decodeIdToken:e}=await Promise.resolve().then(()=>(Wt(),Un));try{let n=e(t),r=Math.floor(Date.now()/1e3);return n.exp-60<r}catch{return!0}}async function ve(){let t=await j();if(!t)throw new qt("Ch\u01B0a \u0111\u0103ng nh\u1EADp. Ch\u1EA1y 'avatar login' tr\u01B0\u1EDBc.");let{refreshAccessToken:e,decodeIdToken:n,verifyIdTokenClaims:r}=await Promise.resolve().then(()=>(Wt(),Un));if(!await pl(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 qt(`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 qt("Refresh kh\xF4ng tr\u1EA3 id_token m\u1EDBi (id_token c\u0169 \u0111\xE3 h\u1EBFt h\u1EA1n). Ch\u1EA1y 'avatar login' l\u1EA1i.");r(n(o.id_token));let i={...t,access_token:o.access_token,id_token:o.id_token,expires_at:new Date(Date.now()+o.expires_in*1e3).toISOString()};return await Kn(i),o.id_token}async function ml(){try{await Vo.chmod(Dn,384)}catch{}}async function v(t,e){await Hn();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
56
- `;await Vo.appendFile(Dn,r,{encoding:"utf8",mode:384}),await ml()}f();import{spawnSync as Fn}from"child_process";var Bo=6e4,dl="ok";function Vn(){let t=Fn("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(){s.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=Fn("claude",["auth","login"],{stdio:"inherit"});if(t.status!==0)throw new Error(`claude auth login th\u1EA5t b\u1EA1i (exit ${t.status}). Th\u1EED 'claude auth login' tay r\u1ED3i ch\u1EA1y l\u1EA1i.`);s.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp Claude Code")}function gl(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 Wo(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 Wn(){let t=Fn("claude",["--print",dl],{encoding:"utf8",timeout:Bo,stdio:["ignore","pipe","pipe"]});if(t.signal==="SIGTERM"||t.status===143||t.error?.code==="ETIMEDOUT")return{ok:!1,reason:"timeout",detail:`claude --print > ${Bo/1e3}s (m\u1EA1ng ch\u1EADm / API rate limit / token revoked kh\xF4ng return error)`};let n=t.stderr||"",r=t.stdout||"";if(t.status===0)return{ok:!0};let o=r.trim(),i=n.toLowerCase();if(o.length>20&&!i.includes("error")&&!i.includes("limit")&&!i.includes("quota")&&!i.includes("401"))return s.warn(`claude --print exit=${t.status} nh\u01B0ng c\xF3 response (${o.length} chars). Accept v\u1EDBi caution.`),{ok:!0};let a=gl(`${n}
57
- ${r}`);return a==="unknown"&&(s.warn(`[debug] claude --print exit=${t.status} signal=${t.signal??"none"}`),n.trim()&&s.warn(`[debug] stderr: ${n.slice(0,500)}`),r.trim()&&s.warn(`[debug] stdout: ${r.slice(0,300)}`)),{ok:!1,reason:a,detail:n.slice(0,500)||r.slice(0,500)}}import{spawnSync as qo}from"child_process";var fl=5e3,hl=/(\d+\.\d+\.\d+)/;function wl(){let e=lt()==="win32"?"where":"which",n=qo(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 kl(){let t=qo("claude",["--version"],{encoding:"utf8",timeout:fl});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return hl.exec(e)?.[1]??null}var gt=null;function Yt(){if(gt!==null)return gt;let t=wl();return t?(gt={installed:!0,version:kl(),path:t},gt):(gt={installed:!1,version:null,path:null},gt)}function xe(){gt=null}import{spawnSync as yl}from"child_process";f();var zo=300*1e3,Jo="@anthropic-ai/claude-code",ft=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallClaudeCodeError",this.reason=e,this.exitCode=r}};function bl(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new ft("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${Jo} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new ft("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new ft("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function Yo(){s.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=yl("npm",["install","-g",Jo],{stdio:["inherit","inherit","pipe"],timeout:zo,encoding:"utf8"});if(t.signal==="SIGTERM")throw new ft("timeout",`npm install timeout sau ${zo/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),bl(t.status,t.stderr||"");xe();let e=Yt();if(!e.installed||!e.path)throw new ft("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `claude` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return s.success(`\u0110\xE3 c\xE0i Claude Code${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}import{readFileSync as vl}from"fs";import{homedir as xl}from"os";import{join as Al}from"path";import{select as Xo}from"@inquirer/prompts";function Cl(){return Al(xl(),".claude","settings.json")}function qn(){let t=Cl(),e;try{e=vl(t,"utf8")}catch{return{exists:!1,hasBaseUrl:!1,hasToken:!1}}let n;try{n=JSON.parse(e)}catch{return{exists:!0,hasBaseUrl:!1,hasToken:!1}}let r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:void 0,i=typeof r.ANTHROPIC_AUTH_TOKEN=="string"&&r.ANTHROPIC_AUTH_TOKEN.length>0,a=typeof n.model=="string"?n.model:void 0;return{exists:!0,hasBaseUrl:!!o,baseUrl:o,hasToken:i,model:a,rawSettings:n}}async function Qo(t=qn()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await Xo({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 Xo({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 Sl,select as Tl}from"@inquirer/prompts";var Ae="https://api.anthropic.com",El="2023-06-01",Zo=1e4;function Pl(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function Rl(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 $l(){return await Sl({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:Rl})}async function _l(t){let e=new AbortController,n=setTimeout(()=>e.abort(),Zo);try{let r=await fetch(`${Ae}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":El,Accept:"application/json"},signal:e.signal});if(r.status===401)throw new Error("API key invalid (HTTP 401). Check key tr\xEAn console.anthropic.com.");if(r.status===403)throw new Error("API key b\u1ECB reject (HTTP 403). Key c\xF3 th\u1EC3 \u0111\xE3 b\u1ECB revoke ho\u1EB7c thi\u1EBFu permission.");if(r.status===429)throw new Error("Rate limit (HTTP 429). Ch\u1EDD v\xE0i gi\xE2y r\u1ED3i th\u1EED l\u1EA1i.");if(!r.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${r.status}).`);let i=((await r.json()).data||[]).map(a=>typeof a.id=="string"?a.id:null).filter(a=>a!==null);if(i.length===0)throw new Error("Anthropic tr\u1EA3 v\u1EC1 list r\u1ED7ng. Li\xEAn h\u1EC7 support ho\u1EB7c check account.");return i}catch(r){throw r.name==="AbortError"?new Error(`Connect ${Ae} timeout sau ${Zo/1e3}s.`):r}finally{clearTimeout(n)}}async function Il(t){if(t.length===1){let n=t[0];return s.info(`Auto-pick model: ${n} (ch\u1EC9 1 model available)`),n}let e=[...t].sort((n,r)=>{let o=i=>{let a=i.toLowerCase();return a.includes("sonnet")?0:a.includes("opus")?1:a.includes("haiku")?2:3};return o(n)-o(r)});return await Tl({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function ti(){let t=await $l();s.info(`Verify key (${Pl(t)}) qua ${Ae}/v1/models...`);let e=await _l(t);s.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await Il(e);return{apiKey:t,baseUrl:Ae,model:n,availableModels:e}}f();w();import{promises as Ml}from"fs";import{join as Nl}from"path";var zn=384;function Ol(t){return Nl(t,".claude","settings.json")}async function Ll(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 Gl(t,e){let{env:n,...r}=t,o={...r,model:e};if(n){let{ANTHROPIC_BASE_URL:i,ANTHROPIC_AUTH_TOKEN:a,ANTHROPIC_API_KEY:c,...l}=n;Object.keys(l).length>0&&(o.env=l)}return o}function ei(t){if(!t)return t;let{ANTHROPIC_AUTH_TOKEN:e,ANTHROPIC_API_KEY:n,...r}=t;return r}function jl(t,e,n,r,o){let a={...ei(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_AUTH_TOKEN=e),{...t,env:a,model:r,avatarProvider:"llmlite"}}function Ul(t,e,n,r,o){let a={...ei(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_API_KEY=e),{...t,env:a,model:r,avatarProvider:"anthropic"}}function Dl(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 ht(t,e){let n=Ol(t),r=await Ll(n),o;switch(e.provider){case"subscription":o=Gl(r,e.model);break;case"llmlite":o=jl(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"anthropic":o=Ul(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"use-global":o=Dl(r,e.sourceSettings);break}await A(n,o,zn);try{await Ml.chmod(n,zn)}catch{}return{path:n,mode:zn}}var H="sonnet";async function Ce(t){try{s.info("Setup AI provider cho workspace...");let e=Yt();if(e.installed)s.success(`Claude Code \u0111\xE3 c\xF3${e.version?` v${e.version}`:""}`);else if(s.info("Ch\u01B0a c\xF3 Claude Code \u2014 s\u1EBD t\u1EF1 c\xE0i qua npm."),Yo(),xe(),e=Yt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=qn();switch(await Qo(n)){case"subscription":{let o=Vn();if(o.state!=="authenticated"&&(Bn(),o=Vn()),o.state==="authenticated"&&o.subscriptionType)return await ht(t.workspacePath,{provider:"subscription",model:H}),await v("ai_setup",`provider=subscription,result=ok,plan=${o.subscriptionType},probe=skipped`),s.success(`AI ready \xB7 Subscription (${o.subscriptionType}) \xB7 model=${H}`),{ok:!0,provider:"subscription",model:H};s.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let i=Wn();if(!i.ok&&i.reason==="auth-expired"&&(s.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),Bn(),i=Wn()),!i.ok&&(i.reason==="timeout"||i.reason==="unknown"))return s.warn(`Probe verify ${i.reason} \u2014 accept trust auth status. Ti\u1EBFp t\u1EE5c.`),i.detail?.trim()&&s.warn(` Chi ti\u1EBFt: ${i.detail.slice(0,200)}`),await ht(t.workspacePath,{provider:"subscription",model:H}),await v("ai_setup",`provider=subscription,result=ok,probe=${i.reason}-soft-pass`),s.success(`AI ready \xB7 Subscription (probe ${i.reason}, soft-pass) \xB7 model=${H}`),{ok:!0,provider:"subscription",model:H};if(!i.ok){let a=i.reason??"unknown";return await v("ai_setup",`provider=subscription,result=no-quota,reason=${a}`),s.warn(`Subscription verify th\u1EA5t b\u1EA1i (${a}).`),i.detail?.trim()&&s.warn(` Chi ti\u1EBFt: ${i.detail.slice(0,200)}`),s.warn(` \u2192 ${Wo(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await ht(t.workspacePath,{provider:"subscription",model:H}),await v("ai_setup","provider=subscription,result=ok"),s.success(`AI ready \xB7 Subscription \xB7 model=${H}`),{ok:!0,provider:"subscription",model:H}}case"llmlite":{let o=await xo();return await ht(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`),s.success(`AI ready \xB7 LLMLite (NAL) \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"llmlite",model:o.model,availableModels:o.availableModels}}case"anthropic":{let o=await ti();return await ht(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`),s.success(`AI ready \xB7 Anthropic Direct \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"anthropic",model:o.model,availableModels:o.availableModels}}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 ht(t.workspacePath,{provider:"use-global",sourceSettings:n.rawSettings}),await v("ai_setup","provider=use-global,result=ok"),s.success(`AI ready \xB7 Copy t\u1EEB global config (${n.baseUrl??"subscription"})`),{ok:!0,provider:"use-global",model:n.model}}}}catch(e){let n=e instanceof Error?e.message:String(e);return s.warn(`AI setup th\u1EA5t b\u1EA1i: ${n}`),s.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup AI sau qua: avatar ai setup"),await v("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}f();import{spawnSync as Hl}from"child_process";f();var Jn=1e4,ni=3e4,oi=5,Yn="say ok",ri="2023-06-01";async function Kl(t,e,n){s.info(`Testing LLMLite provider: ${t} (key: ${J(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),Jn);try{let i=await fetch(`${t}/v1/models`,{headers:{Authorization:`Bearer ${e}`},signal:r.signal});if(i.status===401||i.status===403)throw new Error(`API key invalid (HTTP ${i.status}). Re-run: avatar ai setup`);if(!i.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${i.status}).`);let c=((await i.json()).data||[]).map(h=>typeof h.id=="string"?h.id:null).filter(h=>h!==null);if(s.success(`Connectivity OK \xB7 ${c.length} models available`),c.length>0){let h=c.slice(0,5).join(", "),b=c.length>5?` ...+${c.length-5} more`:"";s.dim(` Models: ${h}${b}`)}s.info(`Testing chat completion v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/chat/completions`,{method:"POST",headers:{Authorization:`Bearer ${e}`,"Content-Type":"application/json"},body:JSON.stringify({model:n,messages:[{role:"user",content:Yn}],max_tokens:oi}),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??"?";s.success(`Response: "${String(d).trim().slice(0,100)}"`),s.dim(` Tokens used: ${g}`)}catch(i){throw i.name==="AbortError"?new Error(`Timeout ${Jn/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):i}finally{clearTimeout(o)}}function Fl(){s.info("Testing Subscription provider qua `claude --print`...");let t=Hl("claude",["--print",Yn],{encoding:"utf8",timeout:ni});if(t.signal==="SIGTERM")throw new Error(`Timeout ${ni/1e3}s. Check m\u1EA1ng / endpoint.`);if(t.status!==0){let e=(t.stderr||"").toLowerCase();throw e.includes("401")||e.includes("invalid authentication")||e.includes("unauthorized")?new Error("Token Claude Code stale (401). Fix: `claude auth logout && claude auth login`."):new Error(`Test fail (exit ${t.status}). Stderr: ${(t.stderr||"").slice(0,200)}`)}s.success(`Response: "${(t.stdout||"").trim().slice(0,100)}"`)}async function Vl(t,e,n){s.info(`Testing Anthropic Direct provider: ${t} (key: ${J(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),Jn);try{let i=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":ri},signal:r.signal});if(i.status===401||i.status===403)throw new Error(`API key invalid (HTTP ${i.status}). Re-run: avatar ai setup`);if(!i.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${i.status}).`);let c=((await i.json()).data||[]).map(g=>typeof g.id=="string"?g.id:null).filter(g=>g!==null);s.success(`Connectivity OK \xB7 ${c.length} models available`),s.info(`Testing message v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/messages`,{method:"POST",headers:{"x-api-key":e,"anthropic-version":ri,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:oi,messages:[{role:"user",content:Yn}]}),signal:r.signal});if(!l.ok){let g=(await l.text()).slice(0,200);throw new Error(`Message endpoint fail (HTTP ${l.status}): ${g}`)}let d=((await l.json()).content||[]).map(g=>typeof g.text=="string"?g.text:"").join("").trim().slice(0,100);s.success(`Response: "${d}"`)}finally{clearTimeout(o)}}async function ii(t){let e=t.env||{},n=typeof e.ANTHROPIC_BASE_URL=="string"?e.ANTHROPIC_BASE_URL:void 0,r=typeof e.ANTHROPIC_AUTH_TOKEN=="string"?e.ANTHROPIC_AUTH_TOKEN:void 0,o=typeof e.ANTHROPIC_API_KEY=="string"?e.ANTHROPIC_API_KEY:void 0,i=typeof t.model=="string"?t.model:"default";return o&&n?(await Vl(n,o,i),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&r?(await Kl(n,r,i),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(Fl(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function Se(){let t=process.cwd(),e=Tt(t);return e||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
51
+ gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await Tc(n.baseUrl,n.model))return s.dim("User decline wiki gen \u2014 workspace OK kh\xF4ng c\xF3 wiki. Ch\u1EA1y `gitnexus wiki` manual sau khi c\u1EA7n."),{ran:!1,skipped:!0,reason:"user-declined"};let o=po(n.model);await go({apiKey:n.apiKey,baseUrl:n.baseUrl,model:n.model,isReasoningModel:o});let i=["wiki",".","--base-url",n.baseUrl,"--model",n.model];o&&i.push("--reasoning-model");let a=Lt(`Generating wiki via ${n.baseUrl} (${n.provider}) model=${n.model}${o?" [reasoning]":""}`),c=kc("gitnexus",i,{cwd:e,stdio:["ignore","pipe","pipe"],timeout:vc,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(`${fo(d,30)}
52
+ `):g&&process.stderr.write(`${fo(g,30)}
53
+ `),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${u} (exit ${c.status??"null"})`}}let l=ho(e,".gitnexus","wiki","index.html");return yc(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 yo(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 o=>{try{await Ec(o)}catch(i){s.error(i instanceof Error?i.message:String(i)),process.exit(1)}}),t.command("remove").description("G\u1EE1 t\xE0i nguy\xEAn kh\u1ECFi workspace").command("repo <name>").description("G\u1EE1 repo kh\u1ECFi src/ (x\xF3a kh\u1ECFi repos.json + t\xF9y ch\u1ECDn x\xF3a folder)").option("--keep-files","Ch\u1EC9 g\u1EE1 kh\u1ECFi manifest, GI\u1EEE folder src/<name>").action(async(o,i)=>{try{await Pc(o,i)}catch(a){s.error(a instanceof Error?a.message:String(a)),process.exit(1)}}),t.command("list").description("Li\u1EC7t k\xEA t\xE0i nguy\xEAn trong workspace").command("repo").description("Li\u1EC7t k\xEA repo trong src/").action(async()=>{try{await Rc()}catch(o){s.error(o instanceof Error?o.message:String(o)),process.exit(1)}})}function bo(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 Tn(){let t=Tt(process.cwd());return t||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
54
+ C\u1EA7n .claude/ + CLAUDE.md + src/. Ch\u1EA1y 'avatar init' tr\u01B0\u1EDBc.`),process.exit(1)),t}async function Ec(t){let e=Tn();await so();let n=!0,r=t.url,o=t.name;for(;n;){let i=await $c(r),a=null,c=null;if(i.mode==="new"){let l=o??await ko(i.url),u=await pn({workspaceRoot:e,url:i.url,name:l});a=u.path,c=u.name,s.success(`\u2713 T\u1EA1o src/${u.name} (repo m\u1EDBi)`)}else{let l=o??await ko(i.url);s.info(`Clone ${i.url} \u2192 src/${l} ...`);let{cloned:u,skipped:d}=await Hr({workspaceRoot:e,url:i.url,name:l});d||!u?s.dim(`B\u1ECF qua repo ${l}.`):(a=u.path,c=u.name,s.success(`\u2713 \u0110\xE3 clone src/${u.name}`))}if(a&&c&&(await Ic(e,a,t.yes),s.success(`\u2713 Done repo: ${c}`)),r=void 0,o=void 0,t.yes)break;n=await Kt({message:"Add repo kh\xE1c n\u1EEFa?",default:!1})}}async function Pc(t,e){let n=Tn(),{validateRepoName:r}=await Promise.resolve().then(()=>(Ot(),gn)),o=r(t);o&&(s.error(`T\xEAn repo kh\xF4ng h\u1EE3p l\u1EC7: ${o}`),process.exit(1));let{readReposManifest:i,removeRepoFromManifest:a}=await Promise.resolve().then(()=>(q(),at)),{removeRecursive:c}=await Promise.resolve().then(()=>(w(),yt)),{join:l,relative:u,sep:d}=await import("path"),h=(await i(n)).find(T=>T.name===t),b=l(n,"src",t),S=l(n,"src"),N=u(S,b);(N===""||N===".."||N.startsWith(`..${d}`)||N.startsWith("/"))&&(s.error(`\u0110\u01B0\u1EDDng d\u1EABn repo escape kh\u1ECFi workspace/src \u2014 t\u1EEB ch\u1ED1i x\xF3a: ${b}`),process.exit(1));let{pathExists:B}=await Promise.resolve().then(()=>(w(),yt)),P=await B(b);if(!h&&!P){s.warn(`Repo "${t}" kh\xF4ng c\xF3 trong workspace (c\u1EA3 manifest l\u1EABn src/).`);return}await a(n,t),s.success(`\u2713 G\u1EE1 "${t}" kh\u1ECFi repos.json`),P&&!e.keepFiles?(s.warn(`\u26A0 Folder src/${t} ch\u1EE9a code. X\xF3a = m\u1EA5t data n\u1EBFu ch\u01B0a push remote.`),await Kt({message:`X\xF3a lu\xF4n folder src/${t}?`,default:!1})?(await c(b),s.success(`\u2713 \u0110\xE3 x\xF3a src/${t}`)):s.dim(`Gi\u1EEF folder src/${t}. (\u0110\xE3 g\u1EE1 kh\u1ECFi manifest \u2014 ch\u1EA1y 'avatar add repo' \u0111\u1EC3 add l\u1EA1i.)`)):P&&e.keepFiles&&s.dim(`Gi\u1EEF folder src/${t} (--keep-files).`)}async function Rc(){let t=Tn(),{readReposManifest:e}=await Promise.resolve().then(()=>(q(),at)),{pathExists:n}=await Promise.resolve().then(()=>(w(),yt)),{join:r}=await import("path"),o=await e(t);if(o.length===0){s.dim("Ch\u01B0a c\xF3 repo n\xE0o. Th\xEAm b\u1EB1ng 'avatar add repo'.");return}s.info(`Repo trong workspace (${o.length}):`);for(let i of o){let c=await n(r(t,"src",i.name))?"\u2713":"\u26A0 folder thi\u1EBFu";console.log(` ${c} ${i.name.padEnd(28)} ${i.url}`)}}async function $c(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 Ht({message:"URL git repo:",validate:a=>a.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),mode:"clone"};if(e==="folder"){let{statSync:i}=await import("fs"),a=wo(await Ht({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 i(wo(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 mn(a);if(c&&(s.warn(`\u26A0 Folder c\xF3 thay \u0111\u1ED5i ch\u01B0a commit: ${c.slice(0,120)}`),s.warn(" Clone l\u1EA5y b\u1EA3n REMOTE \u2192 c\xE1c thay \u0111\u1ED5i local n\xE0y s\u1EBD KH\xD4NG c\xF3 trong src/<name>."),!await Kt({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 dn(a);return l?{url:l,mode:"clone"}:{url:await _c(a),mode:"clone"}}let n=await Ht({message:"T\xEAn repo m\u1EDBi:",validate:bo}),r=await Sn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]});return{url:await Vr(n.trim(),r),mode:"new"}}async function _c(t){let{basename:e}=await import("path");if(s.warn(`Folder ${t} ch\u01B0a c\xF3 remote origin.`),!await Kt({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(()=>(Qe(),yr)),{pathExists:o}=await Promise.resolve().then(()=>(w(),yt)),i=r(t);await o(`${t}/.git`)||(s.info("Folder ch\u01B0a git init \u2014 \u0111ang init..."),await i.init()),s.info("Commit code hi\u1EC7n t\u1EA1i trong folder (baseline cho push)..."),await i.add("."),await i.commit("chore: initial commit (avatar add repo)").catch(()=>{});let a=await Ht({message:"T\xEAn repo GitHub:",default:e(t),validate:bo}),c=await Sn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),{createGithubRemoteFromFolder:l}=await Promise.resolve().then(()=>(xn(),Zr)),u=await Fr(a.trim(),d=>l({folder:t,name:d,visibility:c}).sshUrl);return s.success(`\u2713 \u0110\xE3 t\u1EA1o remote + push: ${u}`),u}async function ko(t){let e=ln(t);return await Ht({message:"T\xEAn th\u01B0 m\u1EE5c trong src/:",default:e})}async function Ic(t,e,n){if(!pt().installed){s.dim("GitNexus ch\u01B0a c\xE0i \u2014 skip index. C\xE0i qua 'avatar gitnexus install'.");return}if(n||await Kt({message:"Index repo n\xE0y b\u1EB1ng GitNexus?",default:!0})){try{Et(e),s.success(` \u2713 GitNexus indexed src/${e.split("/").pop()}`)}catch(i){s.warn(` ! GitNexus index fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${i instanceof Error?i.message:i}`);return}try{let i=await de(t,e);i.ran?s.success(` \u2713 GitNexus wiki t\u1EA1o cho src/${e.split("/").pop()}`):i.reason==="subscription-mode"&&s.dim(" (Subscription mode \u2014 kh\xF4ng c\xF3 API key cho wiki, skip.)")}catch(i){s.warn(` ! Wiki fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${i instanceof Error?i.message:i}`)}}}w();import{promises as Xl}from"fs";import{join as ai}from"path";import{confirm as Ql}from"@inquirer/prompts";import{confirm as sl}from"@inquirer/prompts";w();import{join as Rn}from"path";import{select as Dc}from"@inquirer/prompts";function vo(t){return{ANTHROPIC_DEFAULT_OPUS_MODEL:t.hard,ANTHROPIC_DEFAULT_SONNET_MODEL:t.standard,ANTHROPIC_DEFAULT_HAIKU_MODEL:t.fast}}f();import{input as Nc,password as Mc,select as Oc}from"@inquirer/prompts";var Lc="https://ai.nal.vn",xo=1e4;function Y(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function Gc(){return await Mc({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function Uc(t=Lc){return(await Nc({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 En(t,e){let n=new AbortController,r=setTimeout(()=>n.abort(),xo);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`.":`
55
+ Hint: check m\u1EA1ng / firewall / VPN, ho\u1EB7c base URL c\xF3 \u0111\xFAng kh\xF4ng.`;throw new Error(`Connect ${t} timeout sau ${xo/1e3}s.${c}`)}let i=o.message||String(o);if(i.toLowerCase().includes("fetch failed")||i.includes("ENOTFOUND")){let c=t.includes("nal.vn")?" (ai.nal.vn c\u1EA7n VPN NAL \u2014 ki\u1EC3m tra VPN \u0111\xE3 b\u1EADt ch\u01B0a)":"";throw new Error(`Network error khi connect ${t}: ${i}${c}`)}throw o}finally{clearTimeout(r)}}async function jc(t){let e=t.filter(r=>r.toLowerCase().includes("claude"));if(e.length===1){let r=e[0];return s.info(`Auto-pick model: ${r} (ch\u1EC9 1 claude alias tr\xEAn endpoint)`),r}let n=e.length>0?e:t;return await Oc({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function Ao(){let t=await Gc(),e=await Uc();s.info(`Verify key (${Y(t)}) qua ${e}/v1/models...`);let n=await En(e,t);s.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await jc(n);return{apiKey:t,baseUrl:e,model:r,availableModels:n}}f();var Hc=".claude/state/ai-orchestrator-model-map.json",Kc=["opus","sonnet","haiku"];function To(t){return Rn(t,Hc)}async function Vc(t){let e=To(t);if(!await m(e))return null;try{let r=(await y(e)).map;return r&&typeof r.hard=="string"&&typeof r.standard=="string"&&typeof r.fast=="string"?{hard:r.hard,standard:r.standard,fast:r.fast}:null}catch{return null}}async function Co(t){let e=Rn(t,".claude","settings.json");if(!await m(e))return{provider:"subscription",baseUrl:null,token:null};try{let n=await y(e),r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:null,i=typeof n.avatarProvider=="string"?n.avatarProvider:null;if(!o)return{provider:"subscription",baseUrl:null,token:null};let a=d=>{let g=process.env[d];if(typeof g=="string"&&g.length>0)return g;let h=r[d];return typeof h=="string"&&h.length>0?h:null},c=!1;try{c=new URL(o).hostname==="api.anthropic.com"}catch{c=!1}let l=i==="anthropic"?"anthropic":i==="llmlite"?"llmlite":c?"anthropic":"llmlite",u=a(l==="anthropic"?"ANTHROPIC_API_KEY":"ANTHROPIC_AUTH_TOKEN");return{provider:l,baseUrl:o,token:u}}catch{return{provider:"subscription",baseUrl:null,token:null}}}function $n(t){let e=t.filter(n=>n.toLowerCase().includes("claude"));return e.length>0?e:t}async function Fc(t){if(t.provider==="subscription"||!t.baseUrl||!t.token)return{models:[...Kc],source:"alias"};let e=await En(t.baseUrl,t.token);return{models:$n(e),source:"/v1/models"}}async function Pn(t,e){return await Dc({message:`Tier ${t} \u2192 ch\u1ECDn model:`,choices:e.map(n=>({name:n,value:n}))})}async function Bc(t,e,n,r){let o={map:e,provider:n,source:r,updatedAt:Math.floor(Date.now()/1e3)};await A(To(t),o)}async function So(t,e,n){if(n==="subscription")return;let r=vo(e),o=Rn(t,".claude","settings.json");try{if(!await m(o))return;let i=await y(o);i.env={...i.env||{},...r},await A(o,i),s.dim("ai-orchestrator: set env slot tier (OPUS=hard, SONNET=standard, HAIKU=fast) \u2014 alias spawn expand \u0111\xFAng model \u0111\xE3 map.")}catch(i){s.warn(`ai-orchestrator: kh\xF4ng ghi \u0111\u01B0\u1EE3c tier-slot env (${i.message}) \u2014 route c\xF3 th\u1EC3 ch\u1EA1y sai model. B\u1ECF qua, tool v\u1EABn b\u1EADt.`)}}async function Eo(t,e={}){if(!e.forceRefresh){let a=await Vc(t);if(a){s.dim("ai-orchestrator: state map \u0111\xE3 t\u1ED3n t\u1EA1i \u2014 gi\u1EEF map c\u0169 (kh\xF4ng h\u1ECFi l\u1EA1i).");let c=await Co(t);return await So(t,a,c.provider),!0}}if(!process.stdout.isTTY)return s.warn("ai-orchestrator c\u1EA7n interactive setup (ch\u1ECDn model cho 3 tier) \u2014 b\u1ECF qua trong m\xF4i tr\u01B0\u1EDDng non-TTY. Ch\u1EA1y 'avatar tools enable ai-orchestrator' trong terminal th\u1EADt \u0111\u1EC3 c\xE0i."),!1;let n=await Co(t),r,o;try{let a=await Fc(n);r=a.models,o=a.source}catch(a){return s.error(`ai-orchestrator: kh\xF4ng l\u1EA5y \u0111\u01B0\u1EE3c danh s\xE1ch model (${a.message}). Kh\xF4ng b\u1EADt tool. Ki\u1EC3m tra VPN/gateway r\u1ED3i th\u1EED l\u1EA1i.`),!1}if(r.length===1)return s.warn(`ai-orchestrator: key ch\u1EC9 c\xF3 1 model (${r[0]}) \u2014 route v\xF4 ngh\u0129a khi ch\u1EC9 1 model. KH\xD4NG b\u1EADt tool.`),!1;s.info(`ai-orchestrator: ${r.length} model kh\u1EA3 d\u1EE5ng (${n.provider}). Ch\u1ECDn model cho m\u1ED7i lo\u1EA1i vi\u1EC7c (Kh\xF3/V\u1EEBa/D\u1EC5). C\xF3 th\u1EC3 ch\u1ECDn tr\xF9ng.`);let i;try{let a=await Pn("Vi\u1EC7c kh\xF3 (model m\u1EA1nh)",r),c=await Pn("Vi\u1EC7c v\u1EEBa (model c\xE2n b\u1EB1ng)",r),l=await Pn("Vi\u1EC7c d\u1EC5 (model r\u1EBB)",r);i={hard:a,standard:c,fast:l}}catch{return s.dim("ai-orchestrator: h\u1EE7y ch\u1ECDn model \u2014 kh\xF4ng b\u1EADt tool."),!1}try{await Bc(t,i,n.provider,o)}catch(a){return s.error(`ai-orchestrator: ghi state map th\u1EA5t b\u1EA1i (${a.message}). Kh\xF4ng b\u1EADt tool.`),!1}return await So(t,i,n.provider),s.success(`ai-orchestrator: \u0111\xE3 map tier \u2192 model. Kh\xF3=${i.hard} \xB7 V\u1EEBa=${i.standard} \xB7 D\u1EC5=${i.fast}`),!0}f();import{spawnSync as ol}from"child_process";w();import{promises as Xc}from"fs";import{join as Io}from"path";w();f();import{promises as Wc}from"fs";import Po,{join as _n}from"path";async function qc(t,e){let r=e.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(!r?.[2])return!0;let o=r[2];if(o.startsWith("/"))return s.warn(`Pack hook reject: absolute path "${o}" \u2014 ch\u1EC9 accept relative path t\u1EDBi workspace.`),!1;let i=_n(t,o),a=Po.resolve(t),c=Po.resolve(i);return!c.startsWith(`${a}/`)&&c!==a?(s.warn(`Pack hook reject: path "${o}" resolve ra ngo\xE0i workspace (path traversal).`),!1):await m(i)}function In(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")}${String(e.getSeconds()).padStart(2,"0")}`;return`${t}.backup-${n}`}function Vt(t,e){let n=new Set,r=[];for(let o of[...t,...e]){let i=typeof o=="string"?o:JSON.stringify(o);n.has(i)||(n.add(i),r.push(o))}return r}function zc(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 i=o.command;typeof i=="string"&&r.push(i)}return r}function Ro(t){let e=t.map((i,a)=>{let c=Jc(i),l=c.some(u=>u.includes("${CLAUDE_PROJECT_DIR}"));return{index:a,entry:i,commands:c,hasVar:l}}),n=new Map;for(let i of e){let a=i.commands.map(zc).sort().join("|");if(!a)continue;let c=n.get(a)??[];c.push(i),n.set(a,c)}let r=new Set;for(let[,i]of n)if(!(i.length<2||!i.some(c=>c.hasVar)))for(let c of i)c.hasVar||r.add(c.index);return r.size===0?{entries:t,droppedCount:0}:{entries:t.filter((i,a)=>!r.has(a)),droppedCount:r.size}}function Yc(t,e){let n=[],r={...e},o=0;for(let i of Object.keys(r)){let a=r[i]||[],{entries:c,droppedCount:l}=Ro(a);l>0&&(r[i]=c,o+=l,n.includes(i)||n.push(i))}for(let[i,a]of Object.entries(t)){let c=r[i]||[],l=Vt(c,a),{entries:u,droppedCount:d}=Ro(l);d>0&&(o+=d),u.length!==c.length&&(n.includes(i)||n.push(i)),r[i]=u}return{merged:r,touchedEvents:n,migratedCount:o}}async function ge(t){let e=_n(t,".claude","pack","templates","settings.json.tpl"),n=_n(t,".claude","settings.json");if(!await m(e))return{action:"no-pack-template",changes:[]};let r;try{let u=await O(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={},i=!1;if(await m(n)){i=!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(typeof o.disableWorkflows!="boolean"&&(c.disableWorkflows=!0,a.push("disableWorkflows=true (Avatar t\u1EAFt Dynamic Workflows)")),r.statusLine&&!o.statusLine&&(await qc(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,g=!1;for(let[h,b]of Object.entries(u))typeof b=="string"&&b.includes("llm.nal.vn")&&(u[h]=b.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[h,b]of Object.entries(r.env))h in u||(u[h]=b,g=!0);g&&a.push("env vars added from pack"),(d||g)&&(c.env=u)}if(r.permissions){let u=o.permissions?.allow||[],d=o.permissions?.deny||[],g=r.permissions.allow||[],h=r.permissions.deny||[],b=Vt(u,g),S=Vt(d,h);(b.length!==u.length||S.length!==d.length)&&(c.permissions={allow:b,deny:S},a.push(`permissions union (+${b.length-u.length} allow, +${S.length-d.length} deny)`))}if(r.hooks){let u=o.hooks||{},{merged:d,touchedEvents:g,migratedCount:h}=Yc(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 i&&(l=In(n),await Wc.copyFile(n,l)),await A(n,c),{action:"merged",backupPath:l,changes:a}}function Qc(t,e){return Io(t,".claude","pack","tools",e,"tool.json")}async function Ft(t,e){let n=Qc(t,e);if(!await m(n))return null;try{return await y(n)}catch{return null}}var Zc=[".claude","settings.json"];function No(t){return Io(t,...Zc)}function fe(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function Mo(t){let e=No(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 Oo(t,e,n){let r=No(t),o;return n&&(o=In(r),await Xc.copyFile(r,o)),await A(r,e),o}function $o(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function _o(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function tl(t,e){let n=new Set;for(let i of e)_o(i)&&n.add($o(i));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(i=>_o(i)?!0:n.has($o(i))?(r++,!1):!0),droppedCount:r}}async function he(t,e){let{settings:n,existed:r}=await Mo(t),o={...n},i=[],a=e.settings.hooks??{};if(Object.keys(a).length>0){let u=n.hooks??{},d={...u},g=[],h=0;for(let[b,S]of Object.entries(a)){let N=u[b]??[],{migratedUser:B,droppedCount:P}=tl(N,S);P>0&&(h+=P);let T=new Set(B.map(fe)),R=S.filter(ot=>!T.has(fe(ot)));(R.length>0||P>0)&&(d[b]=[...B,...R],g.push(b))}g.length>0&&(o.hooks=d,i.push(`hooks added: ${g.join(", ")}`),h>0&&i.push(`migrated ${h} stale relative-path entries \u2192 \\{CLAUDE_PROJECT_DIR\\} version`))}let c=e.settings.permissions?.deny??[];if(c.length>0){let u=n.permissions?.deny??[],d=Vt(u,c);d.length!==u.length&&(o.permissions={...n.permissions,deny:d},i.push(`deny +${d.length-u.length}`))}return i.length===0?{action:"no-change",changes:[]}:{action:"enabled",backupPath:await Oo(t,o,r),changes:i}}async function Lo(t,e){let{settings:n,existed:r}=await Mo(t);if(!r)return{action:"no-change",changes:[]};let{hooks:o,permissions:i,...a}=n,c={...a},l=[],u=e.settings.hooks??{},d=n.hooks??{},g={},h=[];for(let[P,T]of Object.entries(d)){let R=u[P];if(!R){g[P]=T;continue}let ot=new Set(R.map(fe)),kt=T.filter(it=>!ot.has(fe(it)));kt.length!==T.length&&h.push(P),kt.length>0&&(g[P]=kt)}Object.keys(g).length>0&&(c.hooks=g),h.length>0&&l.push(`hooks removed: ${h.join(", ")}`);let b=new Set(e.settings.permissions?.deny??[]),S=n.permissions?.deny??[],N=b.size>0?S.filter(P=>!b.has(P)):S;if(n.permissions){let{deny:P,...T}=n.permissions,R={...T,...N.length>0?{deny:N}:{}};Object.keys(R).length>0&&(c.permissions=R)}return N.length!==S.length&&l.push(`deny -${S.length-N.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await Oo(t,c,r),changes:l}}f();w();import{join as el}from"path";var nl=".claude/avatar-tools.json";function Go(){return{tools:{}}}function Uo(t){return el(t,nl)}async function Pt(t){let e=Uo(t);if(!await m(e))return Go();try{return{tools:(await y(e)).tools??{}}}catch{return Go()}}async function rl(t,e){await A(Uo(t),e)}async function we(t,e,n){let r=await Pt(t);return r.tools[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await rl(t,r),r}async function Rt(t){let e=await Pt(t);return Object.entries(e.tools).filter(([,n])=>n.enabled).map(([n])=>n)}function il(t){if(!t)return;let e=ol(t,["--version"],{stdio:"ignore"});(e.status!==0||e.error)&&s.warn(`Tool y\xEAu c\u1EA7u runtime '${t}' nh\u01B0ng kh\xF4ng t\xECm th\u1EA5y tr\xEAn PATH. Hook c\xF3 th\u1EC3 fail khi ch\u1EA1y. C\xE0i '${t}' r\u1ED3i th\u1EED l\u1EA1i.`)}function jo(t,e){switch(e.action){case"enabled":s.success(`Tool '${t}' enabled (${e.changes.join("; ")}).`),e.backupPath&&s.dim(` Backup: ${e.backupPath}`);break;case"disabled":s.success(`Tool '${t}' disabled (${e.changes.join("; ")}).`),e.backupPath&&s.dim(` Backup: ${e.backupPath}`);break;case"no-change":s.dim(`Tool '${t}': settings.json \u0111\xE3 \u0111\xFAng tr\u1EA1ng th\xE1i, kh\xF4ng thay \u0111\u1ED5i.`);break;case"no-manifest":break}}async function H(t,e,n={}){let r=await Ft(t,e);if(!r)return n.silent||s.warn(`Pack version hi\u1EC7n t\u1EA1i ch\u01B0a h\u1ED7 tr\u1EE3 tool '${e}' (thi\u1EBFu .claude/pack/tools/${e}/tool.json). Ch\u1EA1y 'avatar sync' \u0111\u1EC3 pull pack m\u1EDBi, ho\u1EB7c ki\u1EC3m t\xEAn tool.`),!1;if(il(r.requires?.runtime),e==="ai-orchestrator"&&!await Eo(t,{forceRefresh:n.forceRefresh}))return!1;let o=await he(t,r);return jo(e,o),await we(t,e,{enabled:!0,version:r.version}),!0}async function Nn(t,e){let n=await Ft(t,e);if(!n){s.warn(`Kh\xF4ng t\xECm th\u1EA5y manifest tool '${e}' \u0111\u1EC3 bi\u1EBFt entry n\xE0o c\u1EA7n r\xFAt. N\u1EBFu pack \u0111\xE3 \u0111\u1ED5i, state v\u1EABn \u0111\u01B0\u1EE3c set disabled nh\u01B0ng settings.json c\xF3 th\u1EC3 c\xF2n s\xF3t entry.`);let i=(await Pt(t)).tools[e]?.version??"unknown";return await we(t,e,{enabled:!1,version:i}),!1}let r=await Lo(t,n);return jo(e,r),await we(t,e,{enabled:!1,version:n.version}),!0}var al="ai-orchestrator";async function ke(t,e,n={}){if(!e||!e.ok)return;let r;if(e.provider==="subscription")r=3;else if(e.provider==="llmlite"||e.provider==="anthropic")r=$n(e.availableModels??[]).length;else return;if(r<=1){s.dim(`AI Orchestrator: key ch\u1EC9 c\xF3 ${r} model ph\xE2n lo\u1EA1i \u0111\u01B0\u1EE3c \u2014 b\u1ECF qua (c\u1EA7n >1 model).`);return}if(n.autoYes){s.dim("AI Orchestrator: b\u1ECF qua trong ch\u1EBF \u0111\u1ED9 non-interactive \u2014 b\u1EADt sau b\u1EB1ng 'avatar tools enable ai-orchestrator'.");return}let o=n.forceRefresh?"AI Orchestrator: ch\u1ECDn l\u1EA1i model cho m\u1ED7i lo\u1EA1i vi\u1EC7c? (key c\xF3 th\u1EC3 \u0111\xE3 \u0111\u1ED5i)":"B\u1EADt AI Orchestrator? (t\u1EF1 ch\u1ECDn model theo \u0111\u1ED9 kh\xF3 vi\u1EC7c: Vi\u1EC7c kh\xF3 \u2192 model m\u1EA1nh, Vi\u1EC7c d\u1EC5 \u2192 model r\u1EBB)";if(!await sl({message:o,default:!0})){s.dim(" B\u1ECF qua AI Orchestrator (user opt-out).");return}await H(t,al,{forceRefresh:n.forceRefresh})}import{promises as Bo}from"fs";import{homedir as fl}from"os";import{join as zt}from"path";import{z as k}from"zod";var Do=k.object({email:k.string().email(),name:k.string(),access_token:k.string().min(1),refresh_token:k.string().min(1),expires_at:k.string().datetime(),id_token:k.string().min(1)}),cl=k.object({installed_tools:k.record(k.string(),k.object({version:k.string().optional(),installed_at:k.string().datetime(),install_method:k.string()})).default({}),tool_inputs:k.record(k.string(),k.unknown()).default({})}),vf=k.object({$schema:k.string().optional(),includeCoAuthoredBy:k.boolean().optional(),env:k.record(k.string(),k.string()).default({}),permissions:k.object({allow:k.array(k.string()).default([]),deny:k.array(k.string()).default([])}).partial().optional(),hooks:k.record(k.string(),k.array(k.unknown())).optional(),statusLine:k.object({type:k.string(),command:k.string(),padding:k.number().optional()}).optional()}),xf=k.enum(["internal","client","library"]);w();var Jt=zt(fl(),".avatar"),mt=zt(Jt,"config.json"),Pf=zt(Jt,"state.json"),Hn=zt(Jt,"audit.log"),Rf=zt(Jt,"backups"),hl=384;async function Kn(){await x(Jt)}async function U(){if(!await m(mt))return null;let t=await y(mt),e=Do.safeParse(t);return e.success?e.data:null}async function Vn(t){await Kn(),await A(mt,t,hl)}async function ve(){if(await m(mt)){let{promises:t}=await import("fs");await t.unlink(mt)}}function dt(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}var qt=class extends Error{constructor(e){super(e),this.name="NoValidTokenError"}};async function wl(t){let{decodeIdToken:e}=await Promise.resolve().then(()=>(Wt(),Dn));try{let n=e(t),r=Math.floor(Date.now()/1e3);return n.exp-60<r}catch{return!0}}async function xe(){let t=await U();if(!t)throw new qt("Ch\u01B0a \u0111\u0103ng nh\u1EADp. Ch\u1EA1y 'avatar login' tr\u01B0\u1EDBc.");let{refreshAccessToken:e,decodeIdToken:n,verifyIdTokenClaims:r}=await Promise.resolve().then(()=>(Wt(),Dn));if(!await wl(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 qt(`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 qt("Refresh kh\xF4ng tr\u1EA3 id_token m\u1EDBi (id_token c\u0169 \u0111\xE3 h\u1EBFt h\u1EA1n). Ch\u1EA1y 'avatar login' l\u1EA1i.");r(n(o.id_token));let i={...t,access_token:o.access_token,id_token:o.id_token,expires_at:new Date(Date.now()+o.expires_in*1e3).toISOString()};return await Vn(i),o.id_token}async function kl(){try{await Bo.chmod(Hn,384)}catch{}}async function v(t,e){await Kn();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
56
+ `;await Bo.appendFile(Hn,r,{encoding:"utf8",mode:384}),await kl()}f();import{spawnSync as Fn}from"child_process";var Wo=6e4,yl="ok";function Bn(){let t=Fn("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 Wn(){s.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=Fn("claude",["auth","login"],{stdio:"inherit"});if(t.status!==0)throw new Error(`claude auth login th\u1EA5t b\u1EA1i (exit ${t.status}). Th\u1EED 'claude auth login' tay r\u1ED3i ch\u1EA1y l\u1EA1i.`);s.success("\u0110\xE3 \u0111\u0103ng nh\u1EADp Claude Code")}function bl(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 qo(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 qn(){let t=Fn("claude",["--print",yl],{encoding:"utf8",timeout:Wo,stdio:["ignore","pipe","pipe"]});if(t.signal==="SIGTERM"||t.status===143||t.error?.code==="ETIMEDOUT")return{ok:!1,reason:"timeout",detail:`claude --print > ${Wo/1e3}s (m\u1EA1ng ch\u1EADm / API rate limit / token revoked kh\xF4ng return error)`};let n=t.stderr||"",r=t.stdout||"";if(t.status===0)return{ok:!0};let o=r.trim(),i=n.toLowerCase();if(o.length>20&&!i.includes("error")&&!i.includes("limit")&&!i.includes("quota")&&!i.includes("401"))return s.warn(`claude --print exit=${t.status} nh\u01B0ng c\xF3 response (${o.length} chars). Accept v\u1EDBi caution.`),{ok:!0};let a=bl(`${n}
57
+ ${r}`);return a==="unknown"&&(s.warn(`[debug] claude --print exit=${t.status} signal=${t.signal??"none"}`),n.trim()&&s.warn(`[debug] stderr: ${n.slice(0,500)}`),r.trim()&&s.warn(`[debug] stdout: ${r.slice(0,300)}`)),{ok:!1,reason:a,detail:n.slice(0,500)||r.slice(0,500)}}import{spawnSync as zo}from"child_process";var vl=5e3,xl=/(\d+\.\d+\.\d+)/;function Al(){let e=D()==="win32"?"where":"which",n=zo(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 Cl(){let t=zo("claude",["--version"],{encoding:"utf8",timeout:vl});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return xl.exec(e)?.[1]??null}var gt=null;function Yt(){if(gt!==null)return gt;let t=Al();return t?(gt={installed:!0,version:Cl(),path:t},gt):(gt={installed:!1,version:null,path:null},gt)}function Ae(){gt=null}import{spawnSync as Sl}from"child_process";f();var Jo=300*1e3,Yo="@anthropic-ai/claude-code",ft=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallClaudeCodeError",this.reason=e,this.exitCode=r}};function Tl(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new ft("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${Yo} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new ft("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new ft("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function Xo(){s.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Sl("npm",["install","-g",Yo],{stdio:["inherit","inherit","pipe"],timeout:Jo,encoding:"utf8"});if(t.signal==="SIGTERM")throw new ft("timeout",`npm install timeout sau ${Jo/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||"");Ae();let e=Yt();if(!e.installed||!e.path)throw new ft("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `claude` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return s.success(`\u0110\xE3 c\xE0i Claude Code${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}import{readFileSync as El}from"fs";import{homedir as Pl}from"os";import{join as Rl}from"path";import{select as Qo}from"@inquirer/prompts";function $l(){return Rl(Pl(),".claude","settings.json")}function zn(){let t=$l(),e;try{e=El(t,"utf8")}catch{return{exists:!1,hasBaseUrl:!1,hasToken:!1}}let n;try{n=JSON.parse(e)}catch{return{exists:!0,hasBaseUrl:!1,hasToken:!1}}let r=n.env||{},o=typeof r.ANTHROPIC_BASE_URL=="string"?r.ANTHROPIC_BASE_URL:void 0,i=typeof r.ANTHROPIC_AUTH_TOKEN=="string"&&r.ANTHROPIC_AUTH_TOKEN.length>0,a=typeof n.model=="string"?n.model:void 0;return{exists:!0,hasBaseUrl:!!o,baseUrl:o,hasToken:i,model:a,rawSettings:n}}async function Zo(t=zn()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await Qo({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 Qo({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 _l,select as Il}from"@inquirer/prompts";var Ce="https://api.anthropic.com",Nl="2023-06-01",ti=1e4;function Ml(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function Ol(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 Ll(){return await _l({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:Ol})}async function Gl(t){let e=new AbortController,n=setTimeout(()=>e.abort(),ti);try{let r=await fetch(`${Ce}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":Nl,Accept:"application/json"},signal:e.signal});if(r.status===401)throw new Error("API key invalid (HTTP 401). Check key tr\xEAn console.anthropic.com.");if(r.status===403)throw new Error("API key b\u1ECB reject (HTTP 403). Key c\xF3 th\u1EC3 \u0111\xE3 b\u1ECB revoke ho\u1EB7c thi\u1EBFu permission.");if(r.status===429)throw new Error("Rate limit (HTTP 429). Ch\u1EDD v\xE0i gi\xE2y r\u1ED3i th\u1EED l\u1EA1i.");if(!r.ok)throw new Error(`Fetch models th\u1EA5t b\u1EA1i (HTTP ${r.status}).`);let i=((await r.json()).data||[]).map(a=>typeof a.id=="string"?a.id:null).filter(a=>a!==null);if(i.length===0)throw new Error("Anthropic tr\u1EA3 v\u1EC1 list r\u1ED7ng. Li\xEAn h\u1EC7 support ho\u1EB7c check account.");return i}catch(r){throw r.name==="AbortError"?new Error(`Connect ${Ce} timeout sau ${ti/1e3}s.`):r}finally{clearTimeout(n)}}async function Ul(t){if(t.length===1){let n=t[0];return s.info(`Auto-pick model: ${n} (ch\u1EC9 1 model available)`),n}let e=[...t].sort((n,r)=>{let o=i=>{let a=i.toLowerCase();return a.includes("sonnet")?0:a.includes("opus")?1:a.includes("haiku")?2:3};return o(n)-o(r)});return await Il({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function ei(){let t=await Ll();s.info(`Verify key (${Ml(t)}) qua ${Ce}/v1/models...`);let e=await Gl(t);s.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await Ul(e);return{apiKey:t,baseUrl:Ce,model:n,availableModels:e}}f();w();import{promises as jl}from"fs";import{join as Dl}from"path";var Jn=384;function Hl(t){return Dl(t,".claude","settings.json")}async function Kl(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 Vl(t,e){let{env:n,...r}=t,o={...r,model:e};if(n){let{ANTHROPIC_BASE_URL:i,ANTHROPIC_AUTH_TOKEN:a,ANTHROPIC_API_KEY:c,...l}=n;Object.keys(l).length>0&&(o.env=l)}return o}function ni(t){if(!t)return t;let{ANTHROPIC_AUTH_TOKEN:e,ANTHROPIC_API_KEY:n,...r}=t;return r}function Fl(t,e,n,r,o){let a={...ni(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_AUTH_TOKEN=e),{...t,env:a,model:r,avatarProvider:"llmlite"}}function Bl(t,e,n,r,o){let a={...ni(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_API_KEY=e),{...t,env:a,model:r,avatarProvider:"anthropic"}}function Wl(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 ht(t,e){let n=Hl(t),r=await Kl(n),o;switch(e.provider){case"subscription":o=Vl(r,e.model);break;case"llmlite":o=Fl(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"anthropic":o=Bl(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"use-global":o=Wl(r,e.sourceSettings);break}await A(n,o,Jn);try{await jl.chmod(n,Jn)}catch{}return{path:n,mode:Jn}}var K="sonnet";async function Se(t){try{s.info("Setup AI provider cho workspace...");let e=Yt();if(e.installed)s.success(`Claude Code \u0111\xE3 c\xF3${e.version?` v${e.version}`:""}`);else if(s.info("Ch\u01B0a c\xF3 Claude Code \u2014 s\u1EBD t\u1EF1 c\xE0i qua npm."),Xo(),Ae(),e=Yt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=zn();switch(await Zo(n)){case"subscription":{let o=Bn();if(o.state!=="authenticated"&&(Wn(),o=Bn()),o.state==="authenticated"&&o.subscriptionType)return await ht(t.workspacePath,{provider:"subscription",model:K}),await v("ai_setup",`provider=subscription,result=ok,plan=${o.subscriptionType},probe=skipped`),s.success(`AI ready \xB7 Subscription (${o.subscriptionType}) \xB7 model=${K}`),{ok:!0,provider:"subscription",model:K};s.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let i=qn();if(!i.ok&&i.reason==="auth-expired"&&(s.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),Wn(),i=qn()),!i.ok&&(i.reason==="timeout"||i.reason==="unknown"))return s.warn(`Probe verify ${i.reason} \u2014 accept trust auth status. Ti\u1EBFp t\u1EE5c.`),i.detail?.trim()&&s.warn(` Chi ti\u1EBFt: ${i.detail.slice(0,200)}`),await ht(t.workspacePath,{provider:"subscription",model:K}),await v("ai_setup",`provider=subscription,result=ok,probe=${i.reason}-soft-pass`),s.success(`AI ready \xB7 Subscription (probe ${i.reason}, soft-pass) \xB7 model=${K}`),{ok:!0,provider:"subscription",model:K};if(!i.ok){let a=i.reason??"unknown";return await v("ai_setup",`provider=subscription,result=no-quota,reason=${a}`),s.warn(`Subscription verify th\u1EA5t b\u1EA1i (${a}).`),i.detail?.trim()&&s.warn(` Chi ti\u1EBFt: ${i.detail.slice(0,200)}`),s.warn(` \u2192 ${qo(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await ht(t.workspacePath,{provider:"subscription",model:K}),await v("ai_setup","provider=subscription,result=ok"),s.success(`AI ready \xB7 Subscription \xB7 model=${K}`),{ok:!0,provider:"subscription",model:K}}case"llmlite":{let o=await Ao();return await ht(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`),s.success(`AI ready \xB7 LLMLite (NAL) \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"llmlite",model:o.model,availableModels:o.availableModels}}case"anthropic":{let o=await ei();return await ht(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`),s.success(`AI ready \xB7 Anthropic Direct \xB7 model=${o.model} \xB7 ${o.baseUrl}`),{ok:!0,provider:"anthropic",model:o.model,availableModels:o.availableModels}}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 ht(t.workspacePath,{provider:"use-global",sourceSettings:n.rawSettings}),await v("ai_setup","provider=use-global,result=ok"),s.success(`AI ready \xB7 Copy t\u1EEB global config (${n.baseUrl??"subscription"})`),{ok:!0,provider:"use-global",model:n.model}}}}catch(e){let n=e instanceof Error?e.message:String(e);return s.warn(`AI setup th\u1EA5t b\u1EA1i: ${n}`),s.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup AI sau qua: avatar ai setup"),await v("ai_setup",`result=failed,error=${n.slice(0,200)}`),{ok:!1,reason:n}}}f();import{spawnSync as ql}from"child_process";f();var Yn=1e4,ri=3e4,ii=5,Xn="say ok",oi="2023-06-01";async function zl(t,e,n){s.info(`Testing LLMLite provider: ${t} (key: ${Y(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),Yn);try{let i=await fetch(`${t}/v1/models`,{headers:{Authorization:`Bearer ${e}`},signal:r.signal});if(i.status===401||i.status===403)throw new Error(`API key invalid (HTTP ${i.status}). Re-run: avatar ai setup`);if(!i.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${i.status}).`);let c=((await i.json()).data||[]).map(h=>typeof h.id=="string"?h.id:null).filter(h=>h!==null);if(s.success(`Connectivity OK \xB7 ${c.length} models available`),c.length>0){let h=c.slice(0,5).join(", "),b=c.length>5?` ...+${c.length-5} more`:"";s.dim(` Models: ${h}${b}`)}s.info(`Testing chat completion v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/chat/completions`,{method:"POST",headers:{Authorization:`Bearer ${e}`,"Content-Type":"application/json"},body:JSON.stringify({model:n,messages:[{role:"user",content:Xn}],max_tokens:ii}),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??"?";s.success(`Response: "${String(d).trim().slice(0,100)}"`),s.dim(` Tokens used: ${g}`)}catch(i){throw i.name==="AbortError"?new Error(`Timeout ${Yn/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):i}finally{clearTimeout(o)}}function Jl(){s.info("Testing Subscription provider qua `claude --print`...");let t=ql("claude",["--print",Xn],{encoding:"utf8",timeout:ri});if(t.signal==="SIGTERM")throw new Error(`Timeout ${ri/1e3}s. Check m\u1EA1ng / endpoint.`);if(t.status!==0){let e=(t.stderr||"").toLowerCase();throw e.includes("401")||e.includes("invalid authentication")||e.includes("unauthorized")?new Error("Token Claude Code stale (401). Fix: `claude auth logout && claude auth login`."):new Error(`Test fail (exit ${t.status}). Stderr: ${(t.stderr||"").slice(0,200)}`)}s.success(`Response: "${(t.stdout||"").trim().slice(0,100)}"`)}async function Yl(t,e,n){s.info(`Testing Anthropic Direct provider: ${t} (key: ${Y(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),Yn);try{let i=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":oi},signal:r.signal});if(i.status===401||i.status===403)throw new Error(`API key invalid (HTTP ${i.status}). Re-run: avatar ai setup`);if(!i.ok)throw new Error(`Endpoint /v1/models l\u1ED7i (HTTP ${i.status}).`);let c=((await i.json()).data||[]).map(g=>typeof g.id=="string"?g.id:null).filter(g=>g!==null);s.success(`Connectivity OK \xB7 ${c.length} models available`),s.info(`Testing message v\u1EDBi model "${n}"...`);let l=await fetch(`${t}/v1/messages`,{method:"POST",headers:{"x-api-key":e,"anthropic-version":oi,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:ii,messages:[{role:"user",content:Xn}]}),signal:r.signal});if(!l.ok){let g=(await l.text()).slice(0,200);throw new Error(`Message endpoint fail (HTTP ${l.status}): ${g}`)}let d=((await l.json()).content||[]).map(g=>typeof g.text=="string"?g.text:"").join("").trim().slice(0,100);s.success(`Response: "${d}"`)}finally{clearTimeout(o)}}async function si(t){let e=t.env||{},n=typeof e.ANTHROPIC_BASE_URL=="string"?e.ANTHROPIC_BASE_URL:void 0,r=typeof e.ANTHROPIC_AUTH_TOKEN=="string"?e.ANTHROPIC_AUTH_TOKEN:void 0,o=typeof e.ANTHROPIC_API_KEY=="string"?e.ANTHROPIC_API_KEY:void 0,i=typeof t.model=="string"?t.model:"default";return o&&n?(await Yl(n,o,i),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&r?(await zl(n,r,i),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(Jl(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function Te(){let t=process.cwd(),e=Tt(t);return e||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
58
58
  Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
59
59
  B\u1EA1n \u0111ang \u1EDF: ${t}
60
- Cd v\xE0o workspace dir (vd /path/to/<project>-workspace) r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&s.dim(`Detected workspace root: ${e}`),e}async function Xn(t){let e=si(t,".claude","settings.json");if(!await m(e))return{};try{return await y(e)}catch{return{}}}async function ql(){let t=await Se(),e=await Ce({workspacePath:t});try{await we(t,e,{forceRefresh:!0})}catch(n){s.warn(`Setup AI Orchestrator fail: ${n instanceof Error?n.message:n}`)}}async function zl(){let t=await Se(),e=await Xn(t),n=e.env||{},r=typeof n.ANTHROPIC_BASE_URL=="string"?n.ANTHROPIC_BASE_URL:void 0,o=typeof n.ANTHROPIC_AUTH_TOKEN=="string"?n.ANTHROPIC_AUTH_TOKEN:void 0,i=typeof n.ANTHROPIC_API_KEY=="string"?n.ANTHROPIC_API_KEY:void 0,a=typeof e.model=="string"?e.model:void 0,c,l;i?(c=r?.includes("api.anthropic.com")||i.startsWith("sk-ant-")?"Anthropic Direct":"Custom (API key)",l=J(i)):r&&o?(c="LLMLite",l=J(o)):o?(c="Custom",l=J(o)):(c="Subscription (default)",l="(kh\xF4ng set \u2014 d\xF9ng subscription auth)"),s.info(`Project: ${t}`),s.info(`Provider: ${c}${r?` (${r})`:""}`),s.info(`Model: ${a??"(default \u2014 Claude Code ch\u1ECDn)"}`),s.info(`Token: ${l}`)}async function Jl(){let t=await Se(),e=await Xn(t);try{let n=await ii(e);s.success(`\u2713 ${n.message}`)}catch(n){s.error(`Test fail: ${n.message}`),process.exit(1)}}async function Yl(t){let e=await Se(),n=si(e,".claude","settings.json"),r=await Xn(e);if(!t.yes&&!await Wl({message:"X\xF3a AI config (v\u1EC1 d\xF9ng Claude Code Subscription default)?",default:!1})){s.dim("\u0110\xE3 h\u1EE7y.");return}let{env:o,...i}=r,a={...i};if(o){let{ANTHROPIC_BASE_URL:c,ANTHROPIC_AUTH_TOKEN:l,ANTHROPIC_API_KEY:u,...d}=o;Object.keys(d).length>0&&(a.env=d)}Object.keys(a).length===0?(await Bl.unlink(n).catch(()=>{}),s.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await A(n,a,384),s.success("\u0110\xE3 reset env block trong .claude/settings.json"))}function ai(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 ql()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await zl()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await Jl()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await Yl(n)})}import{spawnSync as Qn}from"child_process";import{promises as li}from"fs";import{join as Xt}from"path";import Zl from"boxen";import{join as Xl}from"path";w();function Ql(t,e){let n=e.settings.hooks??{},r=t.hooks??{};for(let[o,i]of Object.entries(n)){let a=r[o]??[],c=new Set(a.flatMap(l=>(l.hooks??[]).map(u=>u.command??"")));for(let l of i)for(let u of l.hooks??[])if(u.command&&!c.has(u.command))return!1}return!0}async function ci(t){let e=await Rt(t);if(e.length===0)return[];let n=Xl(t,".claude","settings.json"),r={};if(await m(n))try{r=await y(n)}catch{r={}}let o=[];for(let i of e){let a=await Vt(t,i);if(!a){o.push({name:`Tool: ${i}`,status:"warn",detail:"state=enabled nh\u01B0ng pack thi\u1EBFu manifest (version skew). Ch\u1EA1y 'avatar sync'.",fixable:!1});continue}Ql(r,a)?o.push({name:`Tool: ${i}`,status:"ok",detail:`enabled, hook active (v${a.version})`,fixable:!1}):o.push({name:`Tool: ${i}`,status:"fail",detail:"state=enabled nh\u01B0ng hook thi\u1EBFu trong settings.json \u2014 fixable (re-apply)",fixable:!0,fix:async()=>{await fe(t,a)}})}return o}w();f();function ui(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 tu(process.cwd());eu(n),e.fix&&await nu(n)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function tu(t){let e=[],n=process.versions.node,[r,o]=n.split(".").map(T=>Number.parseInt(T,10)),i=(r??0)>18||(r??0)===18&&(o??0)>=17;e.push({name:"Node.js version",status:i?"ok":"fail",detail:`v${n}${i?"":" (c\u1EA7n >= 18.17)"}`,fixable:!1});let a=await j();a?dt(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=Xt(t,".claude","pack"),l=Xt(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 T=Xt(t,".claude",".gitignore"),R=!1;await m(T)&&(R=(await li.readFile(T,"utf8")).includes("settings.json")),e.push({name:"\u{1F512} settings.json gitignored (.claude/.gitignore)",status:R?"ok":"fail",detail:R?"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:!R,fix:R?void 0:async()=>{let{writeClaudeGitignore:rt}=await Promise.resolve().then(()=>(bt(),Or));await rt(t)}})}let g=Qn("which",["python"]),h=Qn("which",["python3"]),b=g.status===0,S=h.status===0;S&&!b?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}):b?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 I=Xt(t,".claude","settings.json");if(await m(I))try{let T=await li.readFile(I,"utf8"),R=JSON.parse(T);if(R.statusLine?.command){let kt=R.statusLine.command.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(kt?.[2]){let ot=kt[2],Cs=ot.startsWith("/")?ot:Xt(t,ot),gr=await m(Cs);e.push({name:"statusLine command",status:gr?"ok":"fail",detail:gr?`ref OK: ${ot}`:`BROKEN: settings.json ref '${ot}' nh\u01B0ng file kh\xF4ng t\u1ED3n t\u1EA1i. Strip field statusLine ho\u1EB7c fix path.`,fixable:!1})}}}catch{}let V=Qn("which",["claude"]),E=V.status===0;if(e.push({name:"Claude Code CLI",status:E?"ok":"warn",detail:E?V.stdout.toString().trim():"kh\xF4ng t\xECm th\u1EA5y 'claude' tr\xEAn PATH",fixable:!1}),u){let T=await ci(t);e.push(...T)}return e}function eu(t){let e=[p.bold("Avatar Doctor"),"\u2500".repeat(48)],n=0,r=0,o=0;for(let i of t){let a=i.status==="ok"?p.green("\u2713"):i.status==="warn"?p.yellow("\u26A0"):p.red("\u2717");e.push(`${a} ${i.name.padEnd(28)} ${p.dim(i.detail)}`),i.status==="ok"?n+=1:(r+=1,i.fixable&&(o+=1))}e.push("\u2500".repeat(48)),e.push(`${n} checks passed, ${r} issue${r===1?"":"s"}${o>0?` (${o} fixable \u2014 ch\u1EA1y 'avatar doctor --fix')`:""}`),process.stdout.write(`${Zl(e.join(`
60
+ Cd v\xE0o workspace dir (vd /path/to/<project>-workspace) r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&s.dim(`Detected workspace root: ${e}`),e}async function Qn(t){let e=ai(t,".claude","settings.json");if(!await m(e))return{};try{return await y(e)}catch{return{}}}async function Zl(){let t=await Te(),e=await Se({workspacePath:t});try{await ke(t,e,{forceRefresh:!0})}catch(n){s.warn(`Setup AI Orchestrator fail: ${n instanceof Error?n.message:n}`)}}async function tu(){let t=await Te(),e=await Qn(t),n=e.env||{},r=typeof n.ANTHROPIC_BASE_URL=="string"?n.ANTHROPIC_BASE_URL:void 0,o=typeof n.ANTHROPIC_AUTH_TOKEN=="string"?n.ANTHROPIC_AUTH_TOKEN:void 0,i=typeof n.ANTHROPIC_API_KEY=="string"?n.ANTHROPIC_API_KEY:void 0,a=typeof e.model=="string"?e.model:void 0,c,l;i?(c=r?.includes("api.anthropic.com")||i.startsWith("sk-ant-")?"Anthropic Direct":"Custom (API key)",l=Y(i)):r&&o?(c="LLMLite",l=Y(o)):o?(c="Custom",l=Y(o)):(c="Subscription (default)",l="(kh\xF4ng set \u2014 d\xF9ng subscription auth)"),s.info(`Project: ${t}`),s.info(`Provider: ${c}${r?` (${r})`:""}`),s.info(`Model: ${a??"(default \u2014 Claude Code ch\u1ECDn)"}`),s.info(`Token: ${l}`)}async function eu(){let t=await Te(),e=await Qn(t);try{let n=await si(e);s.success(`\u2713 ${n.message}`)}catch(n){s.error(`Test fail: ${n.message}`),process.exit(1)}}async function nu(t){let e=await Te(),n=ai(e,".claude","settings.json"),r=await Qn(e);if(!t.yes&&!await Ql({message:"X\xF3a AI config (v\u1EC1 d\xF9ng Claude Code Subscription default)?",default:!1})){s.dim("\u0110\xE3 h\u1EE7y.");return}let{env:o,...i}=r,a={...i};if(o){let{ANTHROPIC_BASE_URL:c,ANTHROPIC_AUTH_TOKEN:l,ANTHROPIC_API_KEY:u,...d}=o;Object.keys(d).length>0&&(a.env=d)}Object.keys(a).length===0?(await Xl.unlink(n).catch(()=>{}),s.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await A(n,a,384),s.success("\u0110\xE3 reset env block trong .claude/settings.json"))}function ci(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 Zl()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await tu()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await eu()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await nu(n)})}import{spawnSync as Zn}from"child_process";import{promises as ui}from"fs";import{join as Xt}from"path";import iu from"boxen";import{join as ru}from"path";w();function ou(t,e){let n=e.settings.hooks??{},r=t.hooks??{};for(let[o,i]of Object.entries(n)){let a=r[o]??[],c=new Set(a.flatMap(l=>(l.hooks??[]).map(u=>u.command??"")));for(let l of i)for(let u of l.hooks??[])if(u.command&&!c.has(u.command))return!1}return!0}async function li(t){let e=await Rt(t);if(e.length===0)return[];let n=ru(t,".claude","settings.json"),r={};if(await m(n))try{r=await y(n)}catch{r={}}let o=[];for(let i of e){let a=await Ft(t,i);if(!a){o.push({name:`Tool: ${i}`,status:"warn",detail:"state=enabled nh\u01B0ng pack thi\u1EBFu manifest (version skew). Ch\u1EA1y 'avatar sync'.",fixable:!1});continue}ou(r,a)?o.push({name:`Tool: ${i}`,status:"ok",detail:`enabled, hook active (v${a.version})`,fixable:!1}):o.push({name:`Tool: ${i}`,status:"fail",detail:"state=enabled nh\u01B0ng hook thi\u1EBFu trong settings.json \u2014 fixable (re-apply)",fixable:!0,fix:async()=>{await he(t,a)}})}return o}w();f();function pi(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 su(process.cwd());au(n),e.fix&&await cu(n)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function su(t){let e=[],n=process.versions.node,[r,o]=n.split(".").map(T=>Number.parseInt(T,10)),i=(r??0)>18||(r??0)===18&&(o??0)>=17;e.push({name:"Node.js version",status:i?"ok":"fail",detail:`v${n}${i?"":" (c\u1EA7n >= 18.17)"}`,fixable:!1});let a=await U();a?dt(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=Xt(t,".claude","pack"),l=Xt(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 T=Xt(t,".claude",".gitignore"),R=!1;await m(T)&&(R=(await ui.readFile(T,"utf8")).includes("settings.json")),e.push({name:"\u{1F512} settings.json gitignored (.claude/.gitignore)",status:R?"ok":"fail",detail:R?"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:!R,fix:R?void 0:async()=>{let{writeClaudeGitignore:ot}=await Promise.resolve().then(()=>(bt(),Lr));await ot(t)}})}let g=Zn("which",["python"]),h=Zn("which",["python3"]),b=g.status===0,S=h.status===0;S&&!b?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}):b?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 N=Xt(t,".claude","settings.json");if(await m(N))try{let T=await ui.readFile(N,"utf8"),R=JSON.parse(T);if(R.statusLine?.command){let kt=R.statusLine.command.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(kt?.[2]){let it=kt[2],$s=it.startsWith("/")?it:Xt(t,it),fr=await m($s);e.push({name:"statusLine command",status:fr?"ok":"fail",detail:fr?`ref OK: ${it}`:`BROKEN: settings.json ref '${it}' nh\u01B0ng file kh\xF4ng t\u1ED3n t\u1EA1i. Strip field statusLine ho\u1EB7c fix path.`,fixable:!1})}}}catch{}let B=Zn("which",["claude"]),P=B.status===0;if(e.push({name:"Claude Code CLI",status:P?"ok":"warn",detail:P?B.stdout.toString().trim():"kh\xF4ng t\xECm th\u1EA5y 'claude' tr\xEAn PATH",fixable:!1}),u){let T=await li(t);e.push(...T)}return e}function au(t){let e=[p.bold("Avatar Doctor"),"\u2500".repeat(48)],n=0,r=0,o=0;for(let i of t){let a=i.status==="ok"?p.green("\u2713"):i.status==="warn"?p.yellow("\u26A0"):p.red("\u2717");e.push(`${a} ${i.name.padEnd(28)} ${p.dim(i.detail)}`),i.status==="ok"?n+=1:(r+=1,i.fixable&&(o+=1))}e.push("\u2500".repeat(48)),e.push(`${n} checks passed, ${r} issue${r===1?"":"s"}${o>0?` (${o} fixable \u2014 ch\u1EA1y 'avatar doctor --fix')`:""}`),process.stdout.write(`${iu(e.join(`
61
61
  `),{padding:1,borderStyle:"round"})}
62
- `)}async function nu(t){let e=0;for(let n of t)if(n.fixable&&n.fix)try{await n.fix(),s.success(`Fixed: ${n.name}`),e+=1}catch(r){s.error(`Failed to fix ${n.name}: ${r instanceof Error?r.message:String(r)}`)}e===0&&s.dim("Kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 fix t\u1EF1 \u0111\u1ED9ng.")}w();import{promises as ki}from"fs";import{join as Qt}from"path";import{confirm as uu}from"@inquirer/prompts";import pu from"boxen";import{spawnSync as ru}from"child_process";f();var pi=300*1e3,mi="gitnexus",K=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=r}};function ou(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new K("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${mi} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new K("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new K("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function di(){s.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=ru("npm",["install","-g",mi],{stdio:["inherit","inherit","pipe"],timeout:pi,encoding:"utf8"});if(t.signal==="SIGTERM")throw new K("timeout",`npm install timeout sau ${pi/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),ou(t.status,t.stderr||"");le();let e=pt();if(!e.installed||!e.path)throw new K("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `gitnexus` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return s.success(`\u0110\xE3 c\xE0i GitNexus${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}w();f();import{promises as hi}from"fs";import{homedir as iu}from"os";import{join as su}from"path";var gi=384,fi={command:"gitnexus",args:["mcp"]};function au(){return su(iu(),".claude","mcp_servers.json")}function cu(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 lu(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await hi.copyFile(t,n),n}async function wi(){let t=au(),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&&cu(r,fi))return s.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let o;n&&(o=await lu(t),s.dim(`Backup ${t} \u2192 ${o}`));let i={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:fi}};await A(t,i,gi);try{await hi.chmod(t,gi)}catch{}return s.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:o}}f();async function mu(){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(`${pu(t.join(`
62
+ `)}async function cu(t){let e=0;for(let n of t)if(n.fixable&&n.fix)try{await n.fix(),s.success(`Fixed: ${n.name}`),e+=1}catch(r){s.error(`Failed to fix ${n.name}: ${r instanceof Error?r.message:String(r)}`)}e===0&&s.dim("Kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 fix t\u1EF1 \u0111\u1ED9ng.")}w();import{promises as yi}from"fs";import{join as Qt}from"path";import{confirm as hu}from"@inquirer/prompts";import wu from"boxen";import{spawnSync as lu}from"child_process";f();var mi=300*1e3,di="gitnexus",V=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=r}};function uu(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new V("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${di} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new V("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new V("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function gi(){s.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=lu("npm",["install","-g",di],{stdio:["inherit","inherit","pipe"],timeout:mi,encoding:"utf8"});if(t.signal==="SIGTERM")throw new V("timeout",`npm install timeout sau ${mi/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),uu(t.status,t.stderr||"");ue();let e=pt();if(!e.installed||!e.path)throw new V("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `gitnexus` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return s.success(`\u0110\xE3 c\xE0i GitNexus${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}w();f();import{promises as wi}from"fs";import{homedir as pu}from"os";import{join as mu}from"path";var fi=384,hi={command:"gitnexus",args:["mcp"]};function du(){return mu(pu(),".claude","mcp_servers.json")}function gu(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 fu(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await wi.copyFile(t,n),n}async function ki(){let t=du(),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&&gu(r,hi))return s.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let o;n&&(o=await fu(t),s.dim(`Backup ${t} \u2192 ${o}`));let i={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:hi}};await A(t,i,fi);try{await wi.chmod(t,fi)}catch{}return s.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:o}}f();async function ku(){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(`${wu(t.join(`
63
63
  `),{padding:1,borderStyle:"round",borderColor:"cyan"})}
64
- `),await uu({message:"C\xE0i GitNexus global?",default:!0})}async function du(){for(;;)try{return di(),!0}catch(t){let e=t instanceof Error?t.message:String(t),n=t instanceof K&&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 L({taskName:"C\xE0i GitNexus qua npm",reason:e,allowSkip:!0,hint:n});if(r==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i GitNexus.");if(r==="skip")return!1}}async function gu(t){for(;;)try{return Et(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),r=e instanceof z&&e.reason==="missing-output"?"Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus version mismatch. Check `gitnexus --version`.":"Network glitch? Retry th\u01B0\u1EDDng work.",o=await L({taskName:"GitNexus analyze workspace",reason:n,allowSkip:!0,hint:r});if(o==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc GitNexus analyze.");if(o==="skip")return!1}}async function Te(t){let e={ok:!1,installed:!1,analyzed:!1,wikiGenerated:!1,mcpRegistered:!1};try{s.info("=== Phase 10: GitNexus Setup ===");let n=pt();if(!n.installed){if(!await mu())return await v("gitnexus_setup","result=skipped,reason=user-declined"),s.dim("Skip GitNexus. C\xE0i sau qua `avatar gitnexus install`."),e.reason="user-declined",e;if(!await du())return await v("gitnexus_setup","result=skipped,reason=install-skipped"),s.dim("Skip GitNexus install. Workspace OK kh\xF4ng c\xF3 codebase intelligence."),e.reason="install-skipped",e;if(le(),n=pt(),!n.installed)throw new Error("C\xE0i xong nh\u01B0ng kh\xF4ng detect \u0111\u01B0\u1EE3c binary (PATH issue).")}e.installed=!0,s.success(`GitNexus available${n.version?` v${n.version}`:""}`);try{lo()}catch(r){s.warn(`gitnexus setup fail: ${r.message}`),s.dim("Skip global skills install. Workspace v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c.")}if(t.skipAnalyze)s.dim("GitNexus: c\xE0i + MCP xong. Index s\u1EBD ch\u1EA1y per-repo khi `avatar add repo`.");else{if(!await gu(t.workspacePath))return await v("gitnexus_setup","result=skipped,reason=analyze-skipped"),s.dim("Skip analyze. GitNexus installed nh\u01B0ng ch\u01B0a index."),e.reason="analyze-skipped",e;e.analyzed=!0;let o=await me(t.workspacePath);e.wikiGenerated=o.ran,o.skipped&&o.reason==="fail"&&s.warn(`Wiki gen fail (workspace v\u1EABn OK): ${o.detail??"unknown"}`)}try{let r=await wi();e.mcpRegistered=!0,r.wasUpdated||s.dim("MCP server gitnexus \u0111\xE3 registered tr\u01B0\u1EDBc \u0111\xF3.")}catch(r){s.warn(`MCP server register fail: ${r.message}`),s.dim("Workspace OK nh\u01B0ng Claude Code kh\xF4ng t\u1EF1 attach MCP server. Manual add v\xE0o ~/.claude/mcp_servers.json.")}return e.ok=!0,await v("gitnexus_setup",`result=ok,analyzed=${e.analyzed},wiki=${e.wikiGenerated},mcp=${e.mcpRegistered}`),s.success("GitNexus ready"),e}catch(n){if(n instanceof C)throw n;let r=n instanceof Error?n.message:String(n);return s.warn(`GitNexus setup th\u1EA5t b\u1EA1i: ${r}`),s.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup sau qua `avatar gitnexus install`."),await v("gitnexus_setup",`result=failed,error=${r.slice(0,200)}`),e.reason=r,e}}f();function Zn(){let t=process.cwd(),e=Tt(t);return e||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
64
+ `),await hu({message:"C\xE0i GitNexus global?",default:!0})}async function yu(){for(;;)try{return gi(),!0}catch(t){let e=t instanceof Error?t.message:String(t),n=t instanceof V&&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 L({taskName:"C\xE0i GitNexus qua npm",reason:e,allowSkip:!0,hint:n});if(r==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i GitNexus.");if(r==="skip")return!1}}async function bu(t){for(;;)try{return Et(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),r=e instanceof J&&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 L({taskName:"GitNexus analyze workspace",reason:n,allowSkip:!0,hint:r});if(o==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc GitNexus analyze.");if(o==="skip")return!1}}async function Ee(t){let e={ok:!1,installed:!1,analyzed:!1,wikiGenerated:!1,mcpRegistered:!1};try{s.info("=== Phase 10: GitNexus Setup ===");let n=pt();if(!n.installed){if(!await ku())return await v("gitnexus_setup","result=skipped,reason=user-declined"),s.dim("Skip GitNexus. C\xE0i sau qua `avatar gitnexus install`."),e.reason="user-declined",e;if(!await yu())return await v("gitnexus_setup","result=skipped,reason=install-skipped"),s.dim("Skip GitNexus install. Workspace OK kh\xF4ng c\xF3 codebase intelligence."),e.reason="install-skipped",e;if(ue(),n=pt(),!n.installed)throw new Error("C\xE0i xong nh\u01B0ng kh\xF4ng detect \u0111\u01B0\u1EE3c binary (PATH issue).")}e.installed=!0,s.success(`GitNexus available${n.version?` v${n.version}`:""}`);try{uo()}catch(r){s.warn(`gitnexus setup fail: ${r.message}`),s.dim("Skip global skills install. Workspace v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c.")}if(t.skipAnalyze)s.dim("GitNexus: c\xE0i + MCP xong. Index s\u1EBD ch\u1EA1y per-repo khi `avatar add repo`.");else{if(!await bu(t.workspacePath))return await v("gitnexus_setup","result=skipped,reason=analyze-skipped"),s.dim("Skip analyze. GitNexus installed nh\u01B0ng ch\u01B0a index."),e.reason="analyze-skipped",e;e.analyzed=!0;let o=await de(t.workspacePath);e.wikiGenerated=o.ran,o.skipped&&o.reason==="fail"&&s.warn(`Wiki gen fail (workspace v\u1EABn OK): ${o.detail??"unknown"}`)}try{let r=await ki();e.mcpRegistered=!0,r.wasUpdated||s.dim("MCP server gitnexus \u0111\xE3 registered tr\u01B0\u1EDBc \u0111\xF3.")}catch(r){s.warn(`MCP server register fail: ${r.message}`),s.dim("Workspace OK nh\u01B0ng Claude Code kh\xF4ng t\u1EF1 attach MCP server. Manual add v\xE0o ~/.claude/mcp_servers.json.")}return e.ok=!0,await v("gitnexus_setup",`result=ok,analyzed=${e.analyzed},wiki=${e.wikiGenerated},mcp=${e.mcpRegistered}`),s.success("GitNexus ready"),e}catch(n){if(n instanceof C)throw n;let r=n instanceof Error?n.message:String(n);return s.warn(`GitNexus setup th\u1EA5t b\u1EA1i: ${r}`),s.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup sau qua `avatar gitnexus install`."),await v("gitnexus_setup",`result=failed,error=${r.slice(0,200)}`),e.reason=r,e}}f();function tr(){let t=process.cwd(),e=Tt(t);return e||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
65
65
  Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
66
66
  B\u1EA1n \u0111ang \u1EDF: ${t}
67
- Cd v\xE0o workspace dir r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&s.dim(`Detected workspace root: ${e}`),e}async function fu(){let t=Zn(),e=await Te({workspacePath:t,skipAnalyze:!0});e.ok||e.installed?(s.success("GitNexus c\xE0i + MCP \u0111\u0103ng k\xFD xong."),s.dim("Index repo: avatar gitnexus analyze <name> | --all | (menu)")):s.warn(`Setup kh\xF4ng complete: ${e.reason??"unknown"}`)}async function hu(t){try{let e=Qt(t,".git","HEAD"),n=(await ki.readFile(e,"utf8")).trim();if(n.startsWith("ref:")){let r=n.slice(4).trim();try{return(await ki.readFile(Qt(t,".git",r),"utf8")).trim()}catch{return null}}return n}catch{return null}}async function wu(){let t=Zn(),{readReposManifest:e}=await Promise.resolve().then(()=>(W(),st)),n=await e(t);if(n.length===0){s.dim("Ch\u01B0a c\xF3 repo n\xE0o trong workspace. Th\xEAm b\u1EB1ng 'avatar add repo'.");return}s.info(`GitNexus index status (${n.length} repo):`);for(let r of n){let o=Qt(t,"src",r.name),i=Qt(o,".gitnexus","meta.json");if(!await m(i)){console.log(` \u2717 ${r.name.padEnd(28)} ch\u01B0a index \u2014 'avatar gitnexus analyze ${r.name}'`);continue}try{let a=await y(i),c=!1;if(a.lastCommit){let u=await hu(o);u&&(c=u!==a.lastCommit)}console.log(` ${c?"\u26A0 stale":"\u2713 fresh"} ${r.name.padEnd(28)} indexed ${a.indexedAt?.slice(0,10)??"?"}`)}catch{console.log(` ? ${r.name.padEnd(28)} (\u0111\u1ECDc meta.json fail)`)}}s.dim(`
68
- Stale \u2192 'avatar gitnexus analyze <name>' \u0111\u1EC3 c\u1EADp nh\u1EADt.`)}async function ku(t,e){let n=Zn(),{readReposManifest:r}=await Promise.resolve().then(()=>(W(),st)),o=await r(n);if(o.length===0){s.warn("Ch\u01B0a c\xF3 repo n\xE0o trong workspace. Th\xEAm b\u1EB1ng 'avatar add repo'.");return}let i;if(e.all)i=o.map(c=>c.name);else if(t)o.some(c=>c.name===t)||(s.error(`Repo "${t}" kh\xF4ng c\xF3 trong workspace. Xem: avatar list repo`),process.exit(1)),i=[t];else{let{select:c}=await import("@inquirer/prompts"),l=await c({message:"Index repo n\xE0o?",choices:[...o.map(u=>({name:u.name,value:u.name})),{name:"\u2500\u2500 T\u1EA5t c\u1EA3 \u2500\u2500",value:"__all__"}]});i=l==="__all__"?o.map(u=>u.name):[l]}let a=0;for(let c of i){let l=Qt(n,"src",c);try{Et(l),a+=1}catch(u){s.warn(` ! ${c} fail: ${u.message}`)}}s.success(`Index xong ${a}/${i.length} repo. Xem: avatar gitnexus status`)}function yi(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 fu()}),e.command("status").description("Show index info + staleness warning").action(async()=>{await wu()}),e.command("analyze [name]").description("Index 1 repo trong src/ (t\xEAn), --all cho t\u1EA5t c\u1EA3, kh\xF4ng t\xEAn \u2192 menu").option("--all","Index m\u1ECDi repo trong src/").action(async(n,r)=>{await ku(n,r)})}import{resolve as op}from"path";import{input as ip}from"@inquirer/prompts";import tr from"chalk";var Ee=[" \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 er(t,e,n){return Math.round(t+(e-t)*n)}function yu(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),i=n-r,a=Pe[r],c=Pe[o];return[er(a[0],c[0],i),er(a[1],c[1],i),er(a[2],c[2],i)]}function nr(t){if(!((process.stdout.isTTY??!1)&&tr.level>0))return[...Ee,...t?.tagline?["",t.tagline]:[]].join(`
69
- `);let r=Ee.map((o,i)=>{let a=Ee.length===1?0:i/(Ee.length-1),[c,l,u]=yu(a);return tr.rgb(c,l,u).bold(o)});return t?.tagline&&(r.push(""),r.push(tr.dim(t.tagline))),r.join(`
67
+ Cd v\xE0o workspace dir r\u1ED3i ch\u1EA1y l\u1EA1i.`),process.exit(1)),e!==t&&s.dim(`Detected workspace root: ${e}`),e}async function vu(){let t=tr(),e=await Ee({workspacePath:t,skipAnalyze:!0});e.ok||e.installed?(s.success("GitNexus c\xE0i + MCP \u0111\u0103ng k\xFD xong."),s.dim("Index repo: avatar gitnexus analyze <name> | --all | (menu)")):s.warn(`Setup kh\xF4ng complete: ${e.reason??"unknown"}`)}async function xu(t){try{let e=Qt(t,".git","HEAD"),n=(await yi.readFile(e,"utf8")).trim();if(n.startsWith("ref:")){let r=n.slice(4).trim();try{return(await yi.readFile(Qt(t,".git",r),"utf8")).trim()}catch{return null}}return n}catch{return null}}async function Au(){let t=tr(),{readReposManifest:e}=await Promise.resolve().then(()=>(q(),at)),n=await e(t);if(n.length===0){s.dim("Ch\u01B0a c\xF3 repo n\xE0o trong workspace. Th\xEAm b\u1EB1ng 'avatar add repo'.");return}s.info(`GitNexus index status (${n.length} repo):`);for(let r of n){let o=Qt(t,"src",r.name),i=Qt(o,".gitnexus","meta.json");if(!await m(i)){console.log(` \u2717 ${r.name.padEnd(28)} ch\u01B0a index \u2014 'avatar gitnexus analyze ${r.name}'`);continue}try{let a=await y(i),c=!1;if(a.lastCommit){let u=await xu(o);u&&(c=u!==a.lastCommit)}console.log(` ${c?"\u26A0 stale":"\u2713 fresh"} ${r.name.padEnd(28)} indexed ${a.indexedAt?.slice(0,10)??"?"}`)}catch{console.log(` ? ${r.name.padEnd(28)} (\u0111\u1ECDc meta.json fail)`)}}s.dim(`
68
+ Stale \u2192 'avatar gitnexus analyze <name>' \u0111\u1EC3 c\u1EADp nh\u1EADt.`)}async function Cu(t,e){let n=tr(),{readReposManifest:r}=await Promise.resolve().then(()=>(q(),at)),o=await r(n);if(o.length===0){s.warn("Ch\u01B0a c\xF3 repo n\xE0o trong workspace. Th\xEAm b\u1EB1ng 'avatar add repo'.");return}let i;if(e.all)i=o.map(c=>c.name);else if(t)o.some(c=>c.name===t)||(s.error(`Repo "${t}" kh\xF4ng c\xF3 trong workspace. Xem: avatar list repo`),process.exit(1)),i=[t];else{let{select:c}=await import("@inquirer/prompts"),l=await c({message:"Index repo n\xE0o?",choices:[...o.map(u=>({name:u.name,value:u.name})),{name:"\u2500\u2500 T\u1EA5t c\u1EA3 \u2500\u2500",value:"__all__"}]});i=l==="__all__"?o.map(u=>u.name):[l]}let a=0;for(let c of i){let l=Qt(n,"src",c);try{Et(l),a+=1}catch(u){s.warn(` ! ${c} fail: ${u.message}`)}}s.success(`Index xong ${a}/${i.length} repo. Xem: avatar gitnexus status`)}function bi(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 vu()}),e.command("status").description("Show index info + staleness warning").action(async()=>{await Au()}),e.command("analyze [name]").description("Index 1 repo trong src/ (t\xEAn), --all cho t\u1EA5t c\u1EA3, kh\xF4ng t\xEAn \u2192 menu").option("--all","Index m\u1ECDi repo trong src/").action(async(n,r)=>{await Cu(n,r)})}import{resolve as up}from"path";import{input as pp}from"@inquirer/prompts";import er from"chalk";var Pe=[" \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"],Re=[[217,79,30],[200,70,80],[170,70,140],[125,88,217]];function nr(t,e,n){return Math.round(t+(e-t)*n)}function Su(t){let n=Math.max(0,Math.min(1,t))*(Re.length-1),r=Math.floor(n),o=Math.min(Re.length-1,r+1),i=n-r,a=Re[r],c=Re[o];return[nr(a[0],c[0],i),nr(a[1],c[1],i),nr(a[2],c[2],i)]}function rr(t){if(!((process.stdout.isTTY??!1)&&er.level>0))return[...Pe,...t?.tagline?["",t.tagline]:[]].join(`
69
+ `);let r=Pe.map((o,i)=>{let a=Pe.length===1?0:i/(Pe.length-1),[c,l,u]=Su(a);return er.rgb(c,l,u).bold(o)});return t?.tagline&&(r.push(""),r.push(er.dim(t.tagline))),r.join(`
70
70
  `)}function $t(t){process.stdout.write(`
71
- ${nr(t)}
71
+ ${rr(t)}
72
72
 
73
- `)}import{readdirSync as Fw}from"fs";import{select as Bw}from"@inquirer/prompts";import{simpleGit as qw}from"simple-git";import{existsSync as uw,statSync as pw}from"fs";import{join as dw}from"path";import{simpleGit as hw}from"simple-git";import{existsSync as yw}from"fs";import{join as vw}from"path";import{readFileSync as Cw}from"fs";import{dirname as bu,join as Re}from"path";import{fileURLToPath as vu}from"url";var $e=bu(vu(import.meta.url)),Ew=[Re($e,"templates","gitignore"),Re($e,"..","templates","gitignore"),Re($e,"..","..","src","templates","gitignore"),Re($e,"..","src","templates","gitignore")],rr="# === avatar ===",_e="# === /avatar ===";f();import{existsSync as $w,readFileSync as _w,writeFileSync as Iw}from"fs";import{join as Nw}from"path";var Ie=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};import{join as Si}from"path";w();import{join as bi}from"path";var xu=".avatar-pack-manifest.json",Au=".pack-version";function vi(t){return bi(t,xu)}async function Me(t,e){await A(vi(t),e)}async function Cu(t){let e=vi(t);if(!await m(e))return null;try{return await y(e)}catch{return null}}async function Ne(t){let e=await Cu(t);if(e?.version)return e.version;let n=bi(t,Au);if(await m(n)){let r=(await O(n)).trim();if(r)return r}return null}w();import{promises as Oe}from"fs";import{join as xi}from"path";var Su="https://qpdbkewtpkkbcrptnlos.supabase.co/functions/v1/get-pack",Ai=2e4;function Tu(){let t=process.env.AVATAR_GET_PACK_ENDPOINT??Su,e;try{e=new URL(t)}catch{throw new F(`AVATAR_GET_PACK_ENDPOINT kh\xF4ng ph\u1EA3i URL h\u1EE3p l\u1EC7: ${t}`)}if(e.protocol!=="https:")throw new F(`AVATAR_GET_PACK_ENDPOINT ph\u1EA3i d\xF9ng HTTPS (nh\u1EADn: ${e.protocol}). T\u1EEB ch\u1ED1i g\u1EEDi token qua k\u1EBFt n\u1ED1i kh\xF4ng m\xE3 h\xF3a.`);return t}var wt=class extends Error{constructor(e){super(e),this.name="InvalidIdTokenError"}},Y=class extends Error{constructor(e){super(e),this.name="PackNetworkError"}},F=class extends Error{constructor(e){super(e),this.name="PackParseError"}};async function Le(t,e){let n,r=new AbortController,o=setTimeout(()=>r.abort(),Ai);try{n=await fetch(Tu(),{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t}`},body:JSON.stringify({id_token:t,version:e}),signal:r.signal})}catch(a){throw a instanceof Error&&a.name==="AbortError"?new Y(`get-pack qu\xE1 ${Ai/1e3}s kh\xF4ng ph\u1EA3n h\u1ED3i. Th\u1EED l\u1EA1i.`):new Y(`Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Supabase get-pack: ${a instanceof Error?a.message:a}`)}finally{clearTimeout(o)}if(n.status===401||n.status===403){let a=await Ci(n);throw new wt(`Supabase t\u1EEB ch\u1ED1i (${n.status}): ${a}. Ch\u1EA1y 'avatar login' v\u1EDBi t\xE0i kho\u1EA3n @nal.vn.`)}if(n.status>=500)throw new Y(`Supabase get-pack l\u1ED7i server (${n.status}). Th\u1EED l\u1EA1i sau.`);if(!n.ok){let a=await Ci(n);throw new F(`get-pack tr\u1EA3 ${n.status}: ${a}`)}let i;try{i=await n.json()}catch{throw new F("get-pack tr\u1EA3 response kh\xF4ng ph\u1EA3i JSON.")}if(i.error)throw new F(`get-pack b\xE1o l\u1ED7i: ${i.error}`);if(!i.url||!i.object)throw new F("get-pack thi\u1EBFu url/object trong response.");return{url:i.url,object:i.object,expiresIn:i.expires_in??0,requestedBy:i.requested_by??""}}async function Ge(t,e){let n=xi(e,"..");await x(n);let r=xi(n,`.pack-download-${process.pid}.tar.gz`),o=`${e}.new-${process.pid}`;try{try{await ze(t,r)}catch(c){throw new Y(`T\u1EA3i tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}await it(o);try{await Je(r,o)}catch(c){throw new F(`Gi\u1EA3i n\xE9n tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}let i=`${e}.old-${process.pid}`,a=await m(e);a&&await Oe.rename(e,i);try{await Oe.rename(o,e)}catch(c){throw a&&await Oe.rename(i,e).catch(()=>{}),c}a&&await it(i).catch(()=>{})}finally{await Oe.rm(r,{force:!0}).catch(()=>{}),await it(o).catch(()=>{})}}async function Ci(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 X=".claude/pack",_t=class extends Error{constructor(e){super(e),this.name="TeamPackAccessAbortedError"}};async function Ti(t,e){let n=await ve(),r=await Le(n,e),o=Si(t,X);await Ge(r.url,o);let i=e??Eu(r.object);return await Me(o,{version:i,downloadedObject:r.object,extractedAt:new Date().toISOString()}),{pinnedTag:i}}function Eu(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function je(t){let e=Si(t,X);return await Ne(e)??"(unknown)"}f();w();import{join as De,relative as Nu,resolve as Ni}from"path";import{input as Li,select as Ou}from"@inquirer/prompts";import Lu from"boxen";f();import Pu from"boxen";var Ei=[{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 Pi(){let t=Math.max(...Ei.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=Ei.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"),i=[e,n,"",...r,"",o].join(`
74
- `);return Pu(i,{padding:1,borderStyle:"round",borderColor:"cyan"})}f();import Ru from"boxen";var Ri=[{name:"SKILLS",role:"k\u1EF9 n\u0103ng t\xE1i d\xF9ng \u2014 Claude g\u1ECDi khi c\u1EA7n"},{name:"AGENTS",role:"subagent chuy\xEAn tr\xE1ch \u2014 skills \u0111i\u1EC1u ph\u1ED1i"},{name:"WORKFLOW",role:"orchestrate nhi\u1EC1u agent song song"},{name:"TOOLS",role:"hook g\xE1c c\u1ED5ng m\u1ECDi thao t\xE1c"},{name:"SECURITY",role:"ch\u1EB7n push/secret r\u1EE7i ro tr\u01B0\u1EDBc khi x\u1EA3y ra"},{name:"LOG TRACKING",role:"ghi nh\u1EADn m\u1ECDi h\xE0nh \u0111\u1ED9ng \u0111\u1EC3 truy v\u1EBFt"},{name:"AUTO SUPPORT",role:"t\u1EF1 x\u1EED l\xFD / nh\u1EAFc \u0111\xFAng l\xFAc c\u1EA7n"},{name:"KNOWLEDGE",role:"s\u1ED5 kinh nghi\u1EC7m nu\xF4i c\u1EA3 h\u1EC7 th\u1ED1ng"}],or=[125,88,217],$u=[205,95,130],$i=[150,120,200];function _i(){return(process.stdout.isTTY??!1)&&p.level>0}function It(t,e,n=!1){if(!_i())return e;let r=p.rgb(t[0],t[1],t[2]);return n?r.bold(e):r(e)}function ir(t){return _i()?p.dim(t):t}function _u(){let t=[];return t.push(`${It(or,"\u25C6",!0)} ${It(or,"AVATAR",!0)} ${ir("\u2014 l\xF5i \u0111i\u1EC1u ph\u1ED1i")}`),t.push(It($i,"\u2502")),Ri.forEach((e,n)=>{let r=n===Ri.length-1,o=It($i,r?"\u2514\u2500\u2500":"\u251C\u2500\u2500");t.push(`${o} ${It($u,e.name,!0)} ${ir(e.role)}`)}),t.join(`
75
- `)}function Ii(){let t=It(or,"Ki\u1EBFn tr\xFAc Avatar \u2014 8 kh\u1ED1i quanh m\u1ED9t l\xF5i",!0),e=ir("M\u1ECDi kh\u1ED1i chia s\u1EBB c\xF9ng m\u1ED9t l\xF5i, h\u1EE3p th\xE0nh 1 h\u1EC7 th\u1ED1ng cho c\u1EA3 \u0111\u1ED9i."),n=`${t}
73
+ `)}import{readdirSync as Qw}from"fs";import{select as tk}from"@inquirer/prompts";import{simpleGit as nk}from"simple-git";import{existsSync as yw,statSync as bw}from"fs";import{join as xw}from"path";import{simpleGit as Sw}from"simple-git";import{existsSync as Pw}from"fs";import{join as $w}from"path";import{readFileSync as Nw}from"fs";import{dirname as Tu,join as $e}from"path";import{fileURLToPath as Eu}from"url";var _e=Tu(Eu(import.meta.url)),Lw=[$e(_e,"templates","gitignore"),$e(_e,"..","templates","gitignore"),$e(_e,"..","..","src","templates","gitignore"),$e(_e,"..","src","templates","gitignore")],or="# === avatar ===",Ie="# === /avatar ===";f();import{existsSync as jw,readFileSync as Dw,writeFileSync as Hw}from"fs";import{join as Vw}from"path";var Ne=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};import{join as Ti}from"path";w();import{join as vi}from"path";var Pu=".avatar-pack-manifest.json",Ru=".pack-version";function xi(t){return vi(t,Pu)}async function Me(t,e){await A(xi(t),e)}async function $u(t){let e=xi(t);if(!await m(e))return null;try{return await y(e)}catch{return null}}async function Oe(t){let e=await $u(t);if(e?.version)return e.version;let n=vi(t,Ru);if(await m(n)){let r=(await O(n)).trim();if(r)return r}return null}w();import{promises as Le}from"fs";import{join as Ai}from"path";var _u="https://qpdbkewtpkkbcrptnlos.supabase.co/functions/v1/get-pack",Ci=2e4;function Iu(){let t=process.env.AVATAR_GET_PACK_ENDPOINT??_u,e;try{e=new URL(t)}catch{throw new F(`AVATAR_GET_PACK_ENDPOINT kh\xF4ng ph\u1EA3i URL h\u1EE3p l\u1EC7: ${t}`)}if(e.protocol!=="https:")throw new F(`AVATAR_GET_PACK_ENDPOINT ph\u1EA3i d\xF9ng HTTPS (nh\u1EADn: ${e.protocol}). T\u1EEB ch\u1ED1i g\u1EEDi token qua k\u1EBFt n\u1ED1i kh\xF4ng m\xE3 h\xF3a.`);return t}var wt=class extends Error{constructor(e){super(e),this.name="InvalidIdTokenError"}},X=class extends Error{constructor(e){super(e),this.name="PackNetworkError"}},F=class extends Error{constructor(e){super(e),this.name="PackParseError"}};async function Ge(t,e){let n,r=new AbortController,o=setTimeout(()=>r.abort(),Ci);try{n=await fetch(Iu(),{method:"POST",headers:{"Content-Type":"application/json",Authorization:`Bearer ${t}`},body:JSON.stringify({id_token:t,version:e}),signal:r.signal})}catch(a){throw a instanceof Error&&a.name==="AbortError"?new X(`get-pack qu\xE1 ${Ci/1e3}s kh\xF4ng ph\u1EA3n h\u1ED3i. Th\u1EED l\u1EA1i.`):new X(`Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Supabase get-pack: ${a instanceof Error?a.message:a}`)}finally{clearTimeout(o)}if(n.status===401||n.status===403){let a=await Si(n);throw new wt(`Supabase t\u1EEB ch\u1ED1i (${n.status}): ${a}. Ch\u1EA1y 'avatar login' v\u1EDBi t\xE0i kho\u1EA3n @nal.vn.`)}if(n.status>=500)throw new X(`Supabase get-pack l\u1ED7i server (${n.status}). Th\u1EED l\u1EA1i sau.`);if(!n.ok){let a=await Si(n);throw new F(`get-pack tr\u1EA3 ${n.status}: ${a}`)}let i;try{i=await n.json()}catch{throw new F("get-pack tr\u1EA3 response kh\xF4ng ph\u1EA3i JSON.")}if(i.error)throw new F(`get-pack b\xE1o l\u1ED7i: ${i.error}`);if(!i.url||!i.object)throw new F("get-pack thi\u1EBFu url/object trong response.");return{url:i.url,object:i.object,expiresIn:i.expires_in??0,requestedBy:i.requested_by??""}}async function Ue(t,e){let n=Ai(e,"..");await x(n);let r=Ai(n,`.pack-download-${process.pid}.tar.gz`),o=`${e}.new-${process.pid}`;try{try{await Je(t,r)}catch(c){throw new X(`T\u1EA3i tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}await st(o);try{await Ye(r,o)}catch(c){throw new F(`Gi\u1EA3i n\xE9n tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}let i=`${e}.old-${process.pid}`,a=await m(e);a&&await Le.rename(e,i);try{await Le.rename(o,e)}catch(c){throw a&&await Le.rename(i,e).catch(()=>{}),c}a&&await st(i).catch(()=>{})}finally{await Le.rm(r,{force:!0}).catch(()=>{}),await st(o).catch(()=>{})}}async function Si(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 Q=".claude/pack",_t=class extends Error{constructor(e){super(e),this.name="TeamPackAccessAbortedError"}};async function Ei(t,e){let n=await xe(),r=await Ge(n,e),o=Ti(t,Q);await Ue(r.url,o);let i=e??Nu(r.object);return await Me(o,{version:i,downloadedObject:r.object,extractedAt:new Date().toISOString()}),{pinnedTag:i}}function Nu(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function je(t){let e=Ti(t,Q);return await Oe(e)??"(unknown)"}f();w();import{join as He,relative as Du,resolve as Oi}from"path";import{input as Gi,select as Hu}from"@inquirer/prompts";import Ku from"boxen";f();import Mu from"boxen";var Pi=[{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 Ri(){let t=Math.max(...Pi.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=Pi.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"),i=[e,n,"",...r,"",o].join(`
74
+ `);return Mu(i,{padding:1,borderStyle:"round",borderColor:"cyan"})}f();import Ou from"boxen";var $i=[{name:"SKILLS",role:"k\u1EF9 n\u0103ng t\xE1i d\xF9ng \u2014 Claude g\u1ECDi khi c\u1EA7n"},{name:"AGENTS",role:"subagent chuy\xEAn tr\xE1ch \u2014 skills \u0111i\u1EC1u ph\u1ED1i"},{name:"WORKFLOW",role:"orchestrate nhi\u1EC1u agent song song"},{name:"TOOLS",role:"hook g\xE1c c\u1ED5ng m\u1ECDi thao t\xE1c"},{name:"SECURITY",role:"ch\u1EB7n push/secret r\u1EE7i ro tr\u01B0\u1EDBc khi x\u1EA3y ra"},{name:"LOG TRACKING",role:"ghi nh\u1EADn m\u1ECDi h\xE0nh \u0111\u1ED9ng \u0111\u1EC3 truy v\u1EBFt"},{name:"AUTO SUPPORT",role:"t\u1EF1 x\u1EED l\xFD / nh\u1EAFc \u0111\xFAng l\xFAc c\u1EA7n"},{name:"KNOWLEDGE",role:"s\u1ED5 kinh nghi\u1EC7m nu\xF4i c\u1EA3 h\u1EC7 th\u1ED1ng"}],ir=[125,88,217],Lu=[205,95,130],_i=[150,120,200];function Ii(){return(process.stdout.isTTY??!1)&&p.level>0}function It(t,e,n=!1){if(!Ii())return e;let r=p.rgb(t[0],t[1],t[2]);return n?r.bold(e):r(e)}function sr(t){return Ii()?p.dim(t):t}function Gu(){let t=[];return t.push(`${It(ir,"\u25C6",!0)} ${It(ir,"AVATAR",!0)} ${sr("\u2014 l\xF5i \u0111i\u1EC1u ph\u1ED1i")}`),t.push(It(_i,"\u2502")),$i.forEach((e,n)=>{let r=n===$i.length-1,o=It(_i,r?"\u2514\u2500\u2500":"\u251C\u2500\u2500");t.push(`${o} ${It(Lu,e.name,!0)} ${sr(e.role)}`)}),t.join(`
75
+ `)}function Ni(){let t=It(ir,"Ki\u1EBFn tr\xFAc Avatar \u2014 8 kh\u1ED1i quanh m\u1ED9t l\xF5i",!0),e=sr("M\u1ECDi kh\u1ED1i chia s\u1EBB c\xF9ng m\u1ED9t l\xF5i, h\u1EE3p th\xE0nh 1 h\u1EC7 th\u1ED1ng cho c\u1EA3 \u0111\u1ED9i."),n=`${t}
76
76
 
77
- ${_u()}
77
+ ${Gu()}
78
78
 
79
79
  ${e}`;process.stdout.write(`
80
- ${Ru(n,{padding:1,borderStyle:"round",borderColor:"magenta",textAlignment:"left"})}
81
- `)}f();w();bt();import{readdir as Iu}from"fs/promises";import{join as Mu}from"path";async function Ue(t){if(!await m(t))return!0;try{return(await Iu(t)).filter(r=>!r.startsWith(".")&&r!=="Thumbs.db").length===0}catch{return!1}}async function Mi(t,e,n=10){for(let r=2;r<n;r++){let o=Mu(t,`${e}-${r}`);if(await Ue(o))return o}return null}function Oi(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}".
82
- Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=Ni(De(t,e)),r=Ni(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 Gi(t,e,n){Oi(t,e);let r=De(t,e);if(await Ue(r))return r;for(s.warn(`Workspace path "${r}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let o=await Mi(t,e);if(n&&o)return s.info(`--force: d\xF9ng ${o}`),o;let i=[];o&&i.push({name:`D\xF9ng "${o}" (suggest)`,value:"use-alt"}),i.push({name:"Nh\u1EADp t\xEAn workspace kh\xE1c (manual)",value:"manual"}),i.push({name:"T\u1EA1m ng\u01B0ng init",value:"abort"});let a=await Ou({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:i});if(a==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc resolve workspace path. Ch\u1EA1y l\u1EA1i v\u1EDBi --workspace-name kh\xE1c.");if(a==="use-alt"&&o)return o;let c=await Li({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}});Oi(t,c.trim());let l=De(t,c.trim());if(await Ue(l))return l;s.warn(`"${l}" c\u0169ng \u0111\xE3 c\xF3 n\u1ED9i dung. Th\u1EED t\xEAn kh\xE1c.`)}}async function ji(t){return await Li({message:"Team owner email:",default:t})}async function Ui(t){try{await x(t)}catch(e){let n=e;throw n.code==="EACCES"||n.code==="EPERM"?new Error(`Kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace "${t}": kh\xF4ng c\xF3 quy\u1EC1n ghi.`):n.code==="ENAMETOOLONG"?new Error(`\u0110\u01B0\u1EDDng d\u1EABn workspace qu\xE1 d\xE0i: "${t}".`):n.code==="ENOSPC"?new Error("H\u1EBFt dung l\u01B0\u1EE3ng \u0111\u0129a, kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace."):new Error(`Kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace "${t}": ${n.message}`)}}function Gu(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 ju(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 Di(t,e=null,n=null){let r=[`${p.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${Nu(process.cwd(),t)||t}`,Gu(e),ju(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(`${Lu(r.join(`
80
+ ${Ou(n,{padding:1,borderStyle:"round",borderColor:"magenta",textAlignment:"left"})}
81
+ `)}f();w();bt();import{readdir as Uu}from"fs/promises";import{join as ju}from"path";async function De(t){if(!await m(t))return!0;try{return(await Uu(t)).filter(r=>!r.startsWith(".")&&r!=="Thumbs.db").length===0}catch{return!1}}async function Mi(t,e,n=10){for(let r=2;r<n;r++){let o=ju(t,`${e}-${r}`);if(await De(o))return o}return null}function Li(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}".
82
+ Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=Oi(He(t,e)),r=Oi(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 Ui(t,e,n){Li(t,e);let r=He(t,e);if(await De(r))return r;for(s.warn(`Workspace path "${r}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let o=await Mi(t,e);if(n&&o)return s.info(`--force: d\xF9ng ${o}`),o;let i=[];o&&i.push({name:`D\xF9ng "${o}" (suggest)`,value:"use-alt"}),i.push({name:"Nh\u1EADp t\xEAn workspace kh\xE1c (manual)",value:"manual"}),i.push({name:"T\u1EA1m ng\u01B0ng init",value:"abort"});let a=await Hu({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:i});if(a==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc resolve workspace path. Ch\u1EA1y l\u1EA1i v\u1EDBi --workspace-name kh\xE1c.");if(a==="use-alt"&&o)return o;let c=await Gi({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}});Li(t,c.trim());let l=He(t,c.trim());if(await De(l))return l;s.warn(`"${l}" c\u0169ng \u0111\xE3 c\xF3 n\u1ED9i dung. Th\u1EED t\xEAn kh\xE1c.`)}}async function ji(t){return await Gi({message:"Team owner email:",default:t})}async function Di(t){try{await x(t)}catch(e){let n=e;throw n.code==="EACCES"||n.code==="EPERM"?new Error(`Kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace "${t}": kh\xF4ng c\xF3 quy\u1EC1n ghi.`):n.code==="ENAMETOOLONG"?new Error(`\u0110\u01B0\u1EDDng d\u1EABn workspace qu\xE1 d\xE0i: "${t}".`):n.code==="ENOSPC"?new Error("H\u1EBFt dung l\u01B0\u1EE3ng \u0111\u0129a, kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace."):new Error(`Kh\xF4ng t\u1EA1o \u0111\u01B0\u1EE3c workspace "${t}": ${n.message}`)}}function Vu(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 Fu(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 Hi(t,e=null,n=null){let r=[`${p.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${Du(process.cwd(),t)||t}`,Vu(e),Fu(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(`${Ku(r.join(`
83
83
  `),{padding:1,borderStyle:"round"})}
84
- `);let o=De(t,X);await m(o)&&process.stdout.write(`
85
- ${Pi()}
86
- `),Ii()}import Uu from"boxen";import Du from"open";Wt();f();function Hi(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 sr(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function sr(t){if($t({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await be(),await v("login_reset");else{let g=await j();if(g&&!dt(g)){s.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=q("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await Mn(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let r=jn(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(`
87
- `);process.stdout.write(`${Uu(o,{padding:1,borderStyle:"round"})}
88
- `),Du(r).catch(()=>{s.dim("(Kh\xF4ng m\u1EDF \u0111\u01B0\u1EE3c browser t\u1EF1 \u0111\u1ED9ng \u2014 copy URL \u1EDF tr\xEAn)")});let i=q("\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 Hu(a);try{if(l=await Nn(n.device_code),l)break}catch(g){throw i.fail("X\xE1c th\u1EF1c th\u1EA5t b\u1EA1i"),g}}l||(i.fail("H\u1EBFt h\u1EA1n ch\u1EDD (5 ph\xFAt). Ch\u1EA1y l\u1EA1i 'avatar login'."),process.exit(1)),i.succeed("\u0110\xE3 nh\u1EADn token t\u1EEB Google");let u=On(l.id_token);try{Ln(u)}catch(g){throw await Bt(l.access_token),g}let d=Gn(l,u);await Kn(d),await v("login",d.email),s.success(`X\xE1c th\u1EF1c th\xE0nh c\xF4ng: ${d.email}`),s.success(`Verify hosted domain: ${u.hd} \u2713`),s.success(`L\u01B0u credential v\xE0o ${mt} (chmod 600)`)}function Hu(t){return new Promise(e=>setTimeout(e,t))}import{join as Fe}from"path";f();var Ki=3;async function Fi(t,e,n,r=!1){let o=0;for(;;)try{return{pinnedTag:(await Ti(t,e)).pinnedTag,skipped:!1}}catch(i){if(i instanceof _t)throw i;if(i instanceof wt){s.error(i.message);let l=await L({taskName:"T\u1EA3i team-ai-pack (x\xE1c th\u1EF1c NAL)",reason:i.message,allowSkip:!0,hint:"Ch\u1EA1y 'avatar login' v\u1EDBi t\xE0i kho\u1EA3n @nal.vn r\u1ED3i th\u1EED l\u1EA1i. Ho\u1EB7c skip + 'avatar sync' sau."});if(l==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc x\xE1c th\u1EF1c t\u1EA3i pack.");if(l==="skip")return s.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};continue}if(i instanceof Y&&o<Ki){o+=1,s.warn(`T\u1EA3i pack l\u1ED7i m\u1EA1ng (l\u1EA7n ${o}/${Ki}): ${i.message}`),await Ku(1e3*o);continue}let a=i instanceof Error?i.message:String(i),c=await L({taskName:"T\u1EA3i team-ai-pack tarball",reason:a,allowSkip:!0,hint:"Network glitch? Retry th\u01B0\u1EDDng work. Skip \u2192 d\xF9ng `avatar sync` sau."});if(c==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA3i team-ai-pack.");if(c==="skip")return s.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};o=0}}function Ku(t){return new Promise(e=>setTimeout(e,t))}w();bt();import{readFileSync as Fu}from"fs";import{dirname as Vu,resolve as Bu}from"path";import{fileURLToPath as Wu}from"url";var Mt=null;function N(){if(Mt!==null)return Mt;let t=Vu(Wu(import.meta.url));for(let e=0;e<5;e++){let n=Bu(t,...Array(e).fill(".."),"package.json");try{let r=Fu(n,"utf8"),o=JSON.parse(r);if(o.name==="@nalvietnam/avatar-cli"&&typeof o.version=="string")return Mt=o.version,Mt}catch{}}return Mt="unknown",Mt}import{confirm as Ju}from"@inquirer/prompts";w();import{promises as qu}from"fs";import{join as ar}from"path";var zu=[".claude","pack","tools"];function Vi(t){return ar(t,...zu)}async function cr(t){let e=Vi(t);if(!await m(e))return[];let n=await qu.readdir(e,{withFileTypes:!0}),r=[];for(let o of n)o.isDirectory()&&await m(ar(e,o.name,"tool.json"))&&r.push(o.name);return r.sort()}async function Bi(t){let e=ar(Vi(t),"defaults.json");if(!await m(e))return[];try{return(await y(e)).defaultTools??[]}catch{return[]}}async function Zt(t){let[e,n]=await Promise.all([cr(t),Rt(t)]);return{available:e,enabled:n}}f();var Yu={"prompt-scoring":"prompt scoring"},Xu=new Set(["ai-orchestrator"]);async function Wi(t,e={}){let r=(await Bi(t)).filter(o=>!Xu.has(o));if(r.length!==0)for(let o of r){let i=Yu[o]??o,a;if(e.autoYes?a=!0:a=await Ju({message:`Do you want to setup ${i} (y/n)`,default:!0}),!a){s.dim(` Skip tool '${o}' (user opt-out).`);continue}await D(t,o,{silent:!0})}}import{promises as He}from"fs";import{dirname as lr,join as zi,relative as ur}from"path";import{promises as Qu}from"fs";function Zu(){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 qi(t){let e=`${t}.backup-${Zu()}`;return await Qu.rename(t,e),e}w();var pr=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function tp(t){try{return(await He.lstat(t)).isSymbolicLink()}catch{return!1}}async function ep(t,e,n){let r=ur(lr(e),e)||e;if(!await m(t))return{dir:r,action:"source-missing"};if(await m(e))if(await tp(e))await He.unlink(e);else if(n){let i=await qi(e),a=ur(lr(e),t);return await He.symlink(a,e),{dir:r,action:"backed-up-and-linked",backupPath:i}}else return{dir:r,action:"skipped-conflict"};let o=ur(lr(e),t);return await He.symlink(o,e),{dir:r,action:"created"}}async function Ke(t,e,n){let r=[];for(let o of pr){let i=zi(t,o),a=zi(e,o);r.push(await ep(i,a,n))}return r}f();function np(t){return t?`
84
+ `);let o=He(t,Q);await m(o)&&process.stdout.write(`
85
+ ${Ri()}
86
+ `),Ni()}import Bu from"boxen";import Wu from"open";Wt();f();function Ki(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 ar(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function ar(t){if($t({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await ve(),await v("login_reset");else{let g=await U();if(g&&!dt(g)){s.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=z("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await Mn(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let r=jn(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(`
87
+ `);process.stdout.write(`${Bu(o,{padding:1,borderStyle:"round"})}
88
+ `),Wu(r).catch(()=>{s.dim("(Kh\xF4ng m\u1EDF \u0111\u01B0\u1EE3c browser t\u1EF1 \u0111\u1ED9ng \u2014 copy URL \u1EDF tr\xEAn)")});let i=z("\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 qu(a);try{if(l=await On(n.device_code),l)break}catch(g){throw i.fail("X\xE1c th\u1EF1c th\u1EA5t b\u1EA1i"),g}}l||(i.fail("H\u1EBFt h\u1EA1n ch\u1EDD (5 ph\xFAt). Ch\u1EA1y l\u1EA1i 'avatar login'."),process.exit(1)),i.succeed("\u0110\xE3 nh\u1EADn token t\u1EEB Google");let u=Ln(l.id_token);try{Gn(u)}catch(g){throw await Bt(l.access_token),g}let d=Un(l,u);await Vn(d),await v("login",d.email),s.success(`X\xE1c th\u1EF1c th\xE0nh c\xF4ng: ${d.email}`),s.success(`Verify hosted domain: ${u.hd} \u2713`),s.success(`L\u01B0u credential v\xE0o ${mt} (chmod 600)`)}function qu(t){return new Promise(e=>setTimeout(e,t))}import{join as Fe}from"path";f();var Vi=3;async function Fi(t,e,n,r=!1){let o=0;for(;;)try{return{pinnedTag:(await Ei(t,e)).pinnedTag,skipped:!1}}catch(i){if(i instanceof _t)throw i;if(i instanceof wt){s.error(i.message);let l=await L({taskName:"T\u1EA3i team-ai-pack (x\xE1c th\u1EF1c NAL)",reason:i.message,allowSkip:!0,hint:"Ch\u1EA1y 'avatar login' v\u1EDBi t\xE0i kho\u1EA3n @nal.vn r\u1ED3i th\u1EED l\u1EA1i. Ho\u1EB7c skip + 'avatar sync' sau."});if(l==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc x\xE1c th\u1EF1c t\u1EA3i pack.");if(l==="skip")return s.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};continue}if(i instanceof X&&o<Vi){o+=1,s.warn(`T\u1EA3i pack l\u1ED7i m\u1EA1ng (l\u1EA7n ${o}/${Vi}): ${i.message}`),await zu(1e3*o);continue}let a=i instanceof Error?i.message:String(i),c=await L({taskName:"T\u1EA3i team-ai-pack tarball",reason:a,allowSkip:!0,hint:"Network glitch? Retry th\u01B0\u1EDDng work. Skip \u2192 d\xF9ng `avatar sync` sau."});if(c==="abort")throw new C("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA3i team-ai-pack.");if(c==="skip")return s.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};o=0}}function zu(t){return new Promise(e=>setTimeout(e,t))}w();bt();import{readFileSync as Ju}from"fs";import{dirname as Yu,resolve as Xu}from"path";import{fileURLToPath as Qu}from"url";var Nt=null;function I(){if(Nt!==null)return Nt;let t=Yu(Qu(import.meta.url));for(let e=0;e<5;e++){let n=Xu(t,...Array(e).fill(".."),"package.json");try{let r=Ju(n,"utf8"),o=JSON.parse(r);if(o.name==="@nalvietnam/avatar-cli"&&typeof o.version=="string")return Nt=o.version,Nt}catch{}}return Nt="unknown",Nt}import{confirm as ep}from"@inquirer/prompts";w();import{promises as Zu}from"fs";import{join as cr}from"path";var tp=[".claude","pack","tools"];function Bi(t){return cr(t,...tp)}async function lr(t){let e=Bi(t);if(!await m(e))return[];let n=await Zu.readdir(e,{withFileTypes:!0}),r=[];for(let o of n)o.isDirectory()&&await m(cr(e,o.name,"tool.json"))&&r.push(o.name);return r.sort()}async function Wi(t){let e=cr(Bi(t),"defaults.json");if(!await m(e))return[];try{return(await y(e)).defaultTools??[]}catch{return[]}}async function Zt(t){let[e,n]=await Promise.all([lr(t),Rt(t)]);return{available:e,enabled:n}}f();var np={"prompt-scoring":"prompt scoring"},rp=new Set(["ai-orchestrator"]);async function qi(t,e={}){let r=(await Wi(t)).filter(o=>!rp.has(o));if(r.length!==0)for(let o of r){let i=np[o]??o,a;if(e.autoYes?a=!0:a=await ep({message:`Do you want to setup ${i} (y/n)`,default:!0}),!a){s.dim(` Skip tool '${o}' (user opt-out).`);continue}await H(t,o,{silent:!0})}}import{promises as Ke}from"fs";import{dirname as ur,join as Ji,relative as pr}from"path";import{promises as op}from"fs";function ip(){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 zi(t){let e=`${t}.backup-${ip()}`;return await op.rename(t,e),e}w();var mr=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function sp(t){try{return(await Ke.lstat(t)).isSymbolicLink()}catch{return!1}}async function ap(t,e,n){let r=pr(ur(e),e)||e;if(!await m(t))return{dir:r,action:"source-missing"};if(await m(e))if(await sp(e))await Ke.unlink(e);else if(n){let i=await zi(e),a=pr(ur(e),t);return await Ke.symlink(a,e),{dir:r,action:"backed-up-and-linked",backupPath:i}}else return{dir:r,action:"skipped-conflict"};let o=pr(ur(e),t);return await Ke.symlink(o,e),{dir:r,action:"created"}}async function Ve(t,e,n){let r=[];for(let o of mr){let i=Ji(t,o),a=Ji(e,o);r.push(await ap(i,a,n))}return r}f();function cp(t){return t?`
89
89
  ### \u{1F9E0} CODEBASE INTELLIGENCE \u2014 GitNexus
90
90
 
91
91
  Workspace c\xF3 GitNexus index t\u1EA1i \`.gitnexus/\` cung c\u1EA5p architectural awareness
@@ -113,31 +113,32 @@ Khi user c\u1EA7n regenerate wiki sau refactor l\u1EDBn \u2014 ch\u1EA1y:
113
113
  \`\`\`bash
114
114
  gitnexus wiki . --api-key <key> --base-url <url>
115
115
  \`\`\`
116
- `:""}function Ji(t){return{projectName:t.projectName,projectDescription:t.projectDescription,teamOwner:t.teamOwner,avatarVersion:N(),packVersion:t.packVersion,lastScan:new Date().toISOString(),gitnexusSection:np(t.gitnexusReady??!1)}}async function Yi(t){await Ui(t.workspacePath),await x(Fe(t.workspacePath,"src")),await x(Fe(t.workspacePath,"notes"));let e="(skipped)";if(t.skipTeamPack)s.dim("Skip team-ai-pack (--skip-team-pack).");else{q("T\u1EA3i team-ai-pack t\u1EEB Supabase...").stop();let c=await Fi(t.workspacePath,t.packVersion);e=c.pinnedTag??"(skipped)",c.skipped||s.success(`team-ai-pack: ${e}`)}let n=Ji({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:e});await Qe(t.workspacePath),await Ze(t.workspacePath,{avatarVersion:N(),workspaceName:t.workspaceName}),await tn(t.workspacePath,n),await en(t.workspacePath,n),await rn(t.workspacePath,n),await on(t.workspacePath),await rp(t.workspacePath,t.autoYes),await v("init",`workspace=${t.workspaceName}`);let r=null;t.aiSkip?s.dim("B\u1ECF qua AI setup (--ai-skip). Setup sau: avatar ai setup"):r=await Ce({workspacePath:t.workspacePath});try{await we(t.workspacePath,r,{autoYes:t.autoYes})}catch(a){s.warn(` \u2717 Setup AI Orchestrator fail: ${a instanceof Error?a.message:a}`)}let o=null;t.aiSkip||t.gitnexusSkip?s.dim(t.gitnexusSkip?"B\u1ECF qua GitNexus (--gitnexus-skip). Setup sau: avatar gitnexus install":"B\u1ECF qua GitNexus (auto-skip do --ai-skip)."):o=await Te({workspacePath:t.workspacePath,skipAnalyze:!0}),await Di(t.workspacePath,r,o)}async function rp(t,e){let n=Fe(t,X);if(!await m(n)){s.dim("Pack ch\u01B0a c\xE0i (skip auto-sync). Ch\u1EA1y `avatar sync` sau.");return}let r=Fe(t,".claude");s.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings)...");let o=!1;try{let i=await Ke(n,r,!1),a=i.filter(l=>l.action==="created"||l.action==="updated").length,c=i.filter(l=>l.action==="source-missing").length;s.success(` \u2713 Symlinks: ${a} created${c>0?`, ${c} source-missing`:""}`)}catch(i){o=!0,s.warn(` \u2717 Symlink farm fail: ${i instanceof Error?i.message:i}`)}try{let i=await de(t);i.action==="merged"&&s.success(` \u2713 settings.json merged (${i.changes.join("; ")})`)}catch(i){o=!0,s.warn(` \u2717 Merge settings fail: ${i instanceof Error?i.message:i}`)}try{await Wi(t,{autoYes:e})}catch(i){o=!0,s.warn(` \u2717 Setup default tools fail: ${i instanceof Error?i.message:i}`)}o&&(s.warn("\u26A0 Auto-sync pack CH\u01AFA ho\xE0n t\u1EA5t \u2014 workspace c\xF3 th\u1EC3 thi\u1EBFu symlinks/settings."),s.warn(" Ch\u1EA1y `avatar sync` \u0111\u1EC3 ho\xE0n t\u1EA5t setup."))}function Xi(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 sp(e,n)}catch(r){(r instanceof Ie||r instanceof _t||r instanceof jt||r instanceof C)&&(s.dim(r.message),process.exit(0)),s.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function sp(t,e){e.yes||$t({tagline:"Kh\u1EDFi t\u1EA1o Avatar workspace"});let n=await j();for(;!n||dt(n);){s.info("Ch\u01B0a \u0111\u0103ng nh\u1EADp (ho\u1EB7c token h\u1EBFt h\u1EA1n) \u2014 ch\u1EA1y login tr\u01B0\u1EDBc...");try{await sr({})}catch(c){throw c instanceof C||s.error(`Login th\u1EA5t b\u1EA1i: ${c.message}`),c}if(n=await j(),n&&!dt(n))break;throw new C("C\u1EA7n \u0111\u0103ng nh\u1EADp @nal.vn \u0111\u1EC3 t\u1EA3i team-ai-pack. Ch\u1EA1y 'avatar login' r\u1ED3i 'avatar init' l\u1EA1i.")}let r=t??await ip({message:"T\xEAn workspace:",validate:c=>c.trim().length>0?!0:"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng"}),o=e.teamOwner??await ji(n.email),i=op(e.workspaceParent??"."),a=await Gi(i,r.trim(),e.force);await Yi({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})}Wt();f();function Qi(t){t.command("logout").description("\u0110\u0103ng xu\u1EA5t \u2014 revoke token Google + x\xF3a credential local").option("--keep-remote","Ch\u1EC9 x\xF3a credential local, KH\xD4NG revoke v\u1EDBi Google").action(async e=>{try{await ap(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function ap(t={}){let e=await j();if(!e){s.dim("Ch\u01B0a \u0111\u0103ng nh\u1EADp \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 \u0111\u0103ng xu\u1EA5t.");return}let n=e.email;if(!t.keepRemote){let r=q("Revoke token v\u1EDBi Google...");await Bt(e.refresh_token),r.succeed("\u0110\xE3 revoke token v\u1EDBi Google")}await be(),await v("logout",n),s.success(`\u0110\xE3 \u0111\u0103ng xu\u1EA5t: ${n}`),s.dim("\u0110\u0103ng nh\u1EADp l\u1EA1i: avatar login")}f();function Q(t,e){return()=>{process.stdout.write(`${p.yellow("\u23F3")} ${p.bold(`avatar ${t}`)} \u2014 ch\u01B0a implement \u1EDF milestone hi\u1EC7n t\u1EA1i.
116
+ `:""}function Yi(t){return{projectName:t.projectName,projectDescription:t.projectDescription,teamOwner:t.teamOwner,avatarVersion:I(),packVersion:t.packVersion,lastScan:new Date().toISOString(),gitnexusSection:cp(t.gitnexusReady??!1)}}async function Xi(t){await Di(t.workspacePath),await x(Fe(t.workspacePath,"src")),await x(Fe(t.workspacePath,"notes"));let e="(skipped)";if(t.skipTeamPack)s.dim("Skip team-ai-pack (--skip-team-pack).");else{z("T\u1EA3i team-ai-pack t\u1EEB Supabase...").stop();let c=await Fi(t.workspacePath,t.packVersion);e=c.pinnedTag??"(skipped)",c.skipped||s.success(`team-ai-pack: ${e}`)}let n=Yi({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:e});await Ze(t.workspacePath),await tn(t.workspacePath,{avatarVersion:I(),workspaceName:t.workspaceName}),await en(t.workspacePath,n),await nn(t.workspacePath,n),await on(t.workspacePath,n),await sn(t.workspacePath),await lp(t.workspacePath,t.autoYes),await v("init",`workspace=${t.workspaceName}`);let r=null;t.aiSkip?s.dim("B\u1ECF qua AI setup (--ai-skip). Setup sau: avatar ai setup"):r=await Se({workspacePath:t.workspacePath});try{await ke(t.workspacePath,r,{autoYes:t.autoYes})}catch(a){s.warn(` \u2717 Setup AI Orchestrator fail: ${a instanceof Error?a.message:a}`)}let o=null;t.aiSkip||t.gitnexusSkip?s.dim(t.gitnexusSkip?"B\u1ECF qua GitNexus (--gitnexus-skip). Setup sau: avatar gitnexus install":"B\u1ECF qua GitNexus (auto-skip do --ai-skip)."):o=await Ee({workspacePath:t.workspacePath,skipAnalyze:!0}),await Hi(t.workspacePath,r,o)}async function lp(t,e){let n=Fe(t,Q);if(!await m(n)){s.dim("Pack ch\u01B0a c\xE0i (skip auto-sync). Ch\u1EA1y `avatar sync` sau.");return}let r=Fe(t,".claude");s.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings)...");let o=!1;try{let i=await Ve(n,r,!1),a=i.filter(l=>l.action==="created"||l.action==="updated").length,c=i.filter(l=>l.action==="source-missing").length;s.success(` \u2713 Symlinks: ${a} created${c>0?`, ${c} source-missing`:""}`)}catch(i){o=!0,s.warn(` \u2717 Symlink farm fail: ${i instanceof Error?i.message:i}`)}try{let i=await ge(t);i.action==="merged"&&s.success(` \u2713 settings.json merged (${i.changes.join("; ")})`)}catch(i){o=!0,s.warn(` \u2717 Merge settings fail: ${i instanceof Error?i.message:i}`)}try{await qi(t,{autoYes:e})}catch(i){o=!0,s.warn(` \u2717 Setup default tools fail: ${i instanceof Error?i.message:i}`)}o&&(s.warn("\u26A0 Auto-sync pack CH\u01AFA ho\xE0n t\u1EA5t \u2014 workspace c\xF3 th\u1EC3 thi\u1EBFu symlinks/settings."),s.warn(" Ch\u1EA1y `avatar sync` \u0111\u1EC3 ho\xE0n t\u1EA5t setup."))}function Qi(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 mp(e,n)}catch(r){(r instanceof Ne||r instanceof _t||r instanceof Ut||r instanceof C)&&(s.dim(r.message),process.exit(0)),s.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function mp(t,e){e.yes||$t({tagline:"Kh\u1EDFi t\u1EA1o Avatar workspace"});let n=await U();for(;!n||dt(n);){s.info("Ch\u01B0a \u0111\u0103ng nh\u1EADp (ho\u1EB7c token h\u1EBFt h\u1EA1n) \u2014 ch\u1EA1y login tr\u01B0\u1EDBc...");try{await ar({})}catch(c){throw c instanceof C||s.error(`Login th\u1EA5t b\u1EA1i: ${c.message}`),c}if(n=await U(),n&&!dt(n))break;throw new C("C\u1EA7n \u0111\u0103ng nh\u1EADp @nal.vn \u0111\u1EC3 t\u1EA3i team-ai-pack. Ch\u1EA1y 'avatar login' r\u1ED3i 'avatar init' l\u1EA1i.")}let r=t??await pp({message:"T\xEAn workspace:",validate:c=>c.trim().length>0?!0:"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng"}),o=e.teamOwner??await ji(n.email),i=up(e.workspaceParent??"."),a=await Ui(i,r.trim(),e.force);await Xi({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})}Wt();f();function Zi(t){t.command("logout").description("\u0110\u0103ng xu\u1EA5t \u2014 revoke token Google + x\xF3a credential local").option("--keep-remote","Ch\u1EC9 x\xF3a credential local, KH\xD4NG revoke v\u1EDBi Google").action(async e=>{try{await dp(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function dp(t={}){let e=await U();if(!e){s.dim("Ch\u01B0a \u0111\u0103ng nh\u1EADp \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 \u0111\u0103ng xu\u1EA5t.");return}let n=e.email;if(!t.keepRemote){let r=z("Revoke token v\u1EDBi Google...");await Bt(e.refresh_token),r.succeed("\u0110\xE3 revoke token v\u1EDBi Google")}await ve(),await v("logout",n),s.success(`\u0110\xE3 \u0111\u0103ng xu\u1EA5t: ${n}`),s.dim("\u0110\u0103ng nh\u1EADp l\u1EA1i: avatar login")}f();function Z(t,e){return()=>{process.stdout.write(`${p.yellow("\u23F3")} ${p.bold(`avatar ${t}`)} \u2014 ch\u01B0a implement \u1EDF milestone hi\u1EC7n t\u1EA1i.
117
117
  `),e&&process.stdout.write(` D\u1EF1 ki\u1EBFn: ${p.cyan(e)}
118
118
  `),process.stdout.write(` Spec \u0111\xE3 c\xF3 trong avatar-cli-implementation_4.html.
119
- `),process.exit(0)}}function Zi(t){t.command("mcp-run <tool-id>",{hidden:!0}).description("[internal] Spawn MCP v\u1EDBi secrets injected (M09)").action(Q("mcp-run","Milestone 09"))}w();import{join as cp}from"path";import ts from"boxen";f();var lp=".claude/pack";function es(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 up(process.cwd());n.json?process.stdout.write(`${JSON.stringify(r,null,2)}
120
- `):pp(r)}catch(r){s.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function up(t){let e=cp(t,lp);return await m(e)?{installed:!0,currentVersion:await je(t).catch(()=>null)}:{installed:!1,currentVersion:null}}function pp(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(`${ts(r.join(`
119
+ `),process.exit(0)}}function ts(t){t.command("mcp-run <tool-id>",{hidden:!0}).description("[internal] Spawn MCP v\u1EDBi secrets injected (M09)").action(Z("mcp-run","Milestone 09"))}w();import{join as gp}from"path";import es from"boxen";f();var fp=".claude/pack";function ns(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 hp(process.cwd());n.json?process.stdout.write(`${JSON.stringify(r,null,2)}
120
+ `):wp(r)}catch(r){s.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function hp(t){let e=gp(t,fp);return await m(e)?{installed:!0,currentVersion:await je(t).catch(()=>null)}:{installed:!1,currentVersion:null}}function wp(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(`${es(r.join(`
121
121
  `),{padding:1,borderStyle:"round"})}
122
- `);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(`${ts(n.join(`
122
+ `);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(`${es(n.join(`
123
123
  `),{padding:1,borderStyle:"round"})}
124
- `)}function ns(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(Q("restore","Milestone 08"))}function rs(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(Q("review","Milestone 08"))}function os(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(Q("scan","Milestone 06"))}import{promises as fp}from"fs";import{join as Ve}from"path";import hp from"boxen";w();w();import{promises as mp}from"fs";import{join as dp}from"path";var gp="_backup";async function is(t){let e=dp(t,".claude",gp);return await m(e)?(await mp.readdir(e,{withFileTypes:!0})).filter(r=>r.isDirectory()).map(r=>r.name).sort().reverse():[]}f();function ss(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 wp(process.cwd());e.json?process.stdout.write(`${JSON.stringify(n,null,2)}
125
- `):bp(n)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function wp(t){let e=t.split("/").filter(Boolean).pop()??"unknown",n=Ve(t,".claude");if(!await m(n))return{projectName:e,cliVersion:N(),packVersion:null,pendingCount:0,backupCount:0,techStackSummary:"(Avatar ch\u01B0a init)",hasAvatar:!1,toolsEnabled:[],toolsAvailableCount:0};let o=Ve(n,"_pending"),[i,a,c,l,u]=await Promise.all([(async()=>await m(Ve(n,"pack"))?je(t).catch(()=>null):null)(),(async()=>await m(o)?(await fp.readdir(o)).filter(g=>g.endsWith(".diff.md")).length:0)(),is(t).then(d=>d.length).catch(()=>0),kp(n).catch(()=>"(read error)"),Zt(t).catch(()=>({available:[],enabled:[]}))]);return{projectName:e,cliVersion:N(),packVersion:i,pendingCount:a,backupCount:c,techStackSummary:l,hasAvatar:!0,toolsEnabled:u.enabled,toolsAvailableCount:u.available.length}}async function kp(t){let e=Ve(t,"project","tech-stack.md");return await m(e)?(await O(e)).split(`
126
- `).find(o=>o.trim()&&!o.startsWith("#")&&!o.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function yp(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 bp(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:")} ${yp(t)}`];process.stdout.write(`${hp(e.join(`
124
+ `)}function rs(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(Z("restore","Milestone 08"))}function os(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(Z("review","Milestone 08"))}function is(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(Z("scan","Milestone 06"))}import{promises as vp}from"fs";import{join as Be}from"path";import xp from"boxen";w();w();import{promises as kp}from"fs";import{join as yp}from"path";var bp="_backup";async function ss(t){let e=yp(t,".claude",bp);return await m(e)?(await kp.readdir(e,{withFileTypes:!0})).filter(r=>r.isDirectory()).map(r=>r.name).sort().reverse():[]}f();function as(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 Ap(process.cwd());e.json?process.stdout.write(`${JSON.stringify(n,null,2)}
125
+ `):Tp(n)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Ap(t){let e=t.split("/").filter(Boolean).pop()??"unknown",n=Be(t,".claude");if(!await m(n))return{projectName:e,cliVersion:I(),packVersion:null,pendingCount:0,backupCount:0,techStackSummary:"(Avatar ch\u01B0a init)",hasAvatar:!1,toolsEnabled:[],toolsAvailableCount:0};let o=Be(n,"_pending"),[i,a,c,l,u]=await Promise.all([(async()=>await m(Be(n,"pack"))?je(t).catch(()=>null):null)(),(async()=>await m(o)?(await vp.readdir(o)).filter(g=>g.endsWith(".diff.md")).length:0)(),ss(t).then(d=>d.length).catch(()=>0),Cp(n).catch(()=>"(read error)"),Zt(t).catch(()=>({available:[],enabled:[]}))]);return{projectName:e,cliVersion:I(),packVersion:i,pendingCount:a,backupCount:c,techStackSummary:l,hasAvatar:!0,toolsEnabled:u.enabled,toolsAvailableCount:u.available.length}}async function Cp(t){let e=Be(t,"project","tech-stack.md");return await m(e)?(await O(e)).split(`
126
+ `).find(o=>o.trim()&&!o.startsWith("#")&&!o.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function Sp(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 Tp(t){let e=[`${p.bold("Avatar Status")} \xB7 ${p.cyan(t.projectName)}`,"\u2500".repeat(48),`${p.dim("CLI version:")} ${t.cliVersion}`,`${p.dim("Pack version:")} ${t.packVersion??p.yellow("not installed")}`,`${p.dim("Pending changes:")} ${t.pendingCount}${t.pendingCount>0?p.dim(" (avatar review)"):""}`,`${p.dim("Backups:")} ${t.backupCount}`,`${p.dim("Tech stack:")} ${t.techStackSummary}`,`${p.dim("Tools:")} ${Sp(t)}`];process.stdout.write(`${xp(e.join(`
127
127
  `),{padding:1,borderStyle:"round"})}
128
- `)}w();import{basename as xp,join as mr}from"path";import{readFile as Ap}from"fs/promises";bt();w();import{join as as}from"path";async function vp(t,e,n){let r=as(t,n),o=as(e,n);if(!await m(r))return"source-missing";if(!await m(o))return"needs-creation";let{promises:i}=await import("fs");return(await i.lstat(o)).isSymbolicLink()?"already-linked":"conflict-real-dir"}async function cs(t,e,n){let r=await Ne(t)??"(ch\u01B0a c\xE0i)",o=n??"stable m\u1EDBi nh\u1EA5t (server resolve)",i=[];for(let a of pr)i.push({dir:a,status:await vp(t,e,a)});return{currentVersion:r,targetVersion:o,mountDirStatuses:i}}f();async function ls(t){let e=await Rt(t);if(e.length!==0){s.info(`Re-apply ${e.length} tool(s) \u0111ang b\u1EADt v\xE0o settings.json...`);for(let n of e)await D(t,n)}}f();function Cp(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function Sp(t,e){let n=xp(t)||"workspace",r="";try{r=(await Ap(e,"utf8")).match(/^-\s*Team owner:\s*(.+)$/m)?.[1]?.trim()??""}catch{}return{projectName:n,projectDescription:"",teamOwner:r}}async function Tp(t){let e=process.cwd(),n=mr(e,".claude"),r=mr(e,X);if(await m(n)||(s.error(`Kh\xF4ng th\u1EA5y .claude/ \u2014 kh\xF4ng ph\u1EA3i Avatar workspace.
129
- Ch\u1EA1y 'avatar init' \u0111\u1EC3 kh\u1EDFi t\u1EA1o.`),process.exit(1)),t.dryRun){let c=await cs(r,n,t.packVersion);s.info(`Pack version hi\u1EC7n t\u1EA1i: ${c.currentVersion}`),s.info(`Target version: ${c.targetVersion}`),s.info(`
128
+ `)}w();import{basename as Pp,join as dr}from"path";import{readFile as Rp}from"fs/promises";bt();w();import{join as cs}from"path";async function Ep(t,e,n){let r=cs(t,n),o=cs(e,n);if(!await m(r))return"source-missing";if(!await m(o))return"needs-creation";let{promises:i}=await import("fs");return(await i.lstat(o)).isSymbolicLink()?"already-linked":"conflict-real-dir"}async function ls(t,e,n){let r=await Oe(t)??"(ch\u01B0a c\xE0i)",o=n??"stable m\u1EDBi nh\u1EA5t (server resolve)",i=[];for(let a of mr)i.push({dir:a,status:await Ep(t,e,a)});return{currentVersion:r,targetVersion:o,mountDirStatuses:i}}f();async function us(t){let e=await Rt(t);if(e.length!==0){s.info(`Re-apply ${e.length} tool(s) \u0111ang b\u1EADt v\xE0o settings.json...`);for(let n of e)await H(t,n)}}f();function $p(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function _p(t,e){let n=Pp(t)||"workspace",r="";try{r=(await Rp(e,"utf8")).match(/^-\s*Team owner:\s*(.+)$/m)?.[1]?.trim()??""}catch{}return{projectName:n,projectDescription:"",teamOwner:r}}async function Ip(t){let e=process.cwd(),n=dr(e,".claude"),r=dr(e,Q);if(await m(n)||(s.error(`Kh\xF4ng th\u1EA5y .claude/ \u2014 kh\xF4ng ph\u1EA3i Avatar workspace.
129
+ Ch\u1EA1y 'avatar init' \u0111\u1EC3 kh\u1EDFi t\u1EA1o.`),process.exit(1)),t.dryRun){let c=await ls(r,n,t.packVersion);s.info(`Pack version hi\u1EC7n t\u1EA1i: ${c.currentVersion}`),s.info(`Target version: ${c.targetVersion}`),s.info(`
130
130
  Mount dir statuses:`);for(let l of c.mountDirStatuses)console.log(` ${l.dir.padEnd(12)} ${l.status}`);s.info(`
131
- Dry-run done. Kh\xF4ng apply. B\u1ECF --dry-run \u0111\u1EC3 th\u1EF1c thi.`);return}let o;try{o=await ve()}catch(c){s.error(c instanceof Error?c.message:String(c)),process.exit(1)}s.info(t.packVersion?`T\u1EA3i team-ai-pack ${t.packVersion} t\u1EEB Supabase...`:"T\u1EA3i team-ai-pack (stable m\u1EDBi nh\u1EA5t) t\u1EEB Supabase...");let i;try{let c=await Le(o,t.packVersion);await Ge(c.url,r),i=t.packVersion??Cp(c.object),await Me(r,{version:i,downloadedObject:c.object,extractedAt:new Date().toISOString()}),s.success(`\u0110\xE3 c\xE0i pack ${i}`)}catch(c){c instanceof wt?s.error(`${c.message}`):s.error(`T\u1EA3i pack th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`),process.exit(1)}s.info("T\u1EA1o symlink farm...");let a=await Ke(r,n,t.force===!0);Ep(a,t.force===!0),s.info("Merge pack settings.json template v\xE0o project settings.json...");try{let c=await de(e);switch(c.action){case"merged":s.success(` \u2713 settings.json merged (${c.changes.join("; ")})`);break;case"no-change":s.info(" - settings.json \u0111\xE3 sync.");break;case"no-pack-template":s.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip.");break}}catch(c){s.warn(` ! Merge settings.json fail: ${c instanceof Error?c.message:c}.`)}try{await ls(e)}catch(c){s.warn(` ! Re-apply tools fail: ${c instanceof Error?c.message:c}.`)}s.info("C\u1EADp nh\u1EADt managed block trong CLAUDE.md...");try{let c=await Sp(e,mr(e,"CLAUDE.md")),l={projectName:c.projectName,projectDescription:c.projectDescription,teamOwner:c.teamOwner,avatarVersion:N(),packVersion:i,lastScan:new Date().toISOString(),gitnexusSection:""};switch((await nn(e,l)).outcome){case"replaced-block":s.success(" \u2713 CLAUDE.md managed block \u0111\xE3 c\u1EADp nh\u1EADt (gi\u1EEF context user ngo\xE0i block).");break;case"replaced-whole":s.success(" \u2713 CLAUDE.md (ch\u01B0a c\xF3 marker) \u2192 thay ho\xE0n to\xE0n b\u1EB1ng b\u1EA3n Avatar m\u1EDBi.");break;case"skipped-no-file":s.dim(" - Ch\u01B0a c\xF3 CLAUDE.md (ch\u1EA1y 'avatar init' \u0111\u1EC3 t\u1EA1o), skip.");break}}catch(c){s.warn(` ! C\u1EADp nh\u1EADt CLAUDE.md fail: ${c instanceof Error?c.message:c}.`)}s.success(`Synced team-ai-pack \u2192 ${i}.`)}function Ep(t,e){for(let r of t)switch(r.action){case"created":s.info(` \u2713 ${r.dir} \u2192 symlinked (new)`);break;case"updated":s.info(` \u2713 ${r.dir} \u2192 symlink refreshed`);break;case"backed-up-and-linked":s.info(` \u2713 ${r.dir} \u2192 symlinked (backup: ${r.backupPath})`);break;case"source-missing":s.warn(` - ${r.dir} \u2192 pack kh\xF4ng c\xF3 dir n\xE0y, skip`);break;case"skipped-conflict":s.warn(` ! ${r.dir} \u2192 CONFLICT: existing real dir. D\xF9ng --force \u0111\u1EC3 backup + override.`);break}let n=t.filter(r=>r.action==="skipped-conflict").length;n>0&&!e&&s.warn(`${n} mount dir(s) skip do conflict. Ch\u1EA1y l\u1EA1i v\u1EDBi --force.`)}function us(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("--pack-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(Tp)}import{resolve as Pp}from"path";import{checkbox as Rp,confirm as $p}from"@inquirer/prompts";f();function ps(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 ms(t){return t.map(e=>({name:e,value:e,checked:!1}))}function ds(t){return[...t]}function te(t){return Pp(t.target??process.cwd())}async function gs(t,e,n,r){if(n.all===!0||!process.stdout.isTTY)return ds(t);let i=r==="add"?ps(t,e):ms(t);return await Rp({message:r==="add"?"Ch\u1ECDn tool \u0111\u1EC3 c\xE0i (space ch\u1ECDn, enter x\xE1c nh\u1EADn):":"Ch\u1ECDn tool \u0111\u1EC3 g\u1EE1 (space ch\u1ECDn, enter x\xE1c nh\u1EADn):",choices:i})}async function _p(t){let e=te(t),{available:n,enabled:r}=await Zt(e);if(n.length===0){s.dim("Kh\xF4ng c\xF3 tool available (pack ch\u01B0a c\xF3 tools/, ho\u1EB7c ch\u01B0a sync).");return}let o=await gs(n,r,t,"add");if(o.length===0){s.dim("Kh\xF4ng ch\u1ECDn tool n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}for(let i of o)await D(e,i)}async function Ip(t){let e=te(t),{enabled:n}=await Zt(e);if(n.length===0){s.dim("Kh\xF4ng c\xF3 tool n\xE0o \u0111ang b\u1EADt \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}let r=await gs(n,[],t,"remove");if(r.length===0){s.dim("Kh\xF4ng ch\u1ECDn tool n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}if(t.yes!==!0&&t.all!==!0&&process.stdout.isTTY&&!await $p({message:`G\u1EE1 ${r.length} tool: ${r.join(", ")}?`,default:!0})){s.dim("H\u1EE7y \u2014 kh\xF4ng g\u1EE1 tool n\xE0o.");return}for(let i of r)await In(e,i)}async function Mp(t){let e=te(t),n=await cr(e),r=await Pt(e);if(n.length===0&&Object.keys(r.tools).length===0){s.dim("Kh\xF4ng c\xF3 tool n\xE0o (pack ch\u01B0a c\xF3 tools/, ho\u1EB7c ch\u01B0a sync). Ch\u1EA1y 'avatar sync'.");return}let o=[...new Set([...n,...Object.keys(r.tools)])].sort();s.info("Tools:"),console.log(` ${"NAME".padEnd(20)} ${"AVAILABLE".padEnd(10)} ${"ENABLED".padEnd(8)} VERSION`);for(let i of o){let a=n.includes(i)?"yes":"no",c=r.tools[i],l=c?.enabled?"yes":"no",u=c?.version??"-";console.log(` ${i.padEnd(20)} ${a.padEnd(10)} ${l.padEnd(8)} ${u}`)}}function fs(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=te(r);await D(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=te(r);await In(o,n)}),e.command("list").description("Li\u1EC7t k\xEA tool: available | enabled | version").option("--target <path>","Workspace root (default: cwd)").action(async n=>{await Mp(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 _p(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 Ip(n)})}import{relative as Hp}from"path";import{confirm as Kp}from"@inquirer/prompts";import Fp from"boxen";import{cp as Be,mkdir as hs,writeFile as Np}from"fs/promises";import{homedir as Op}from"os";import{basename as Lp,join as Z}from"path";var Gp=Z(Op(),".avatar","uninstall-backups");async function ws(t,e,n){let r=Lp(t),o=new Date().toISOString().replace(/[:.]/g,"-"),i=Z(Gp,`${r}-${o}`);if(await hs(i,{recursive:!0,mode:448}),e.claudeDir&&await Be(e.claudeDir,Z(i,".claude"),{recursive:!0}),e.claudeMd&&await Be(e.claudeMd,Z(i,"CLAUDE.md")),e.postMergeHook||e.prePushHook){let c=Z(i,"hooks");await hs(c,{recursive:!0}),e.postMergeHook&&await Be(e.postMergeHook,Z(c,"post-merge")),e.prePushHook&&await Be(e.prePushHook,Z(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 Np(Z(i,"manifest.json"),JSON.stringify(a,null,2),"utf8"),i}import{existsSync as jp}from"fs";import{join as tt}from"path";function et(t){return jp(t)?t:null}function ks(t){let e=et(tt(t,".claude")),n=et(tt(t,"CLAUDE.md")),r=et(tt(t,".git","hooks","post-merge")),o=et(tt(t,".git","modules","src","hooks","pre-push")),i=et(tt(t,".gitignore")),a=et(tt(t,".gitmodules")),c=et(tt(t,"notes")),l=et(tt(t,"scripts"));return{hasAnyArtifact:!!(e||n||r||o),claudeDir:e,claudeMd:n,postMergeHook:r,prePushHook:o,gitignorePath:i,gitmodulesPath:a,notesDir:c,scriptsDir:l}}import{readFile as ys,rm as nt,writeFile as bs}from"fs/promises";async function vs(t,e){if(t.claudeDir)if(e.keepSubmodule){let{readdir:n}=await import("fs/promises"),{join:r}=await import("path"),o=await n(t.claudeDir);for(let i of o)i!=="pack"&&await nt(r(t.claudeDir,i),{recursive:!0,force:!0})}else await nt(t.claudeDir,{recursive:!0,force:!0});t.claudeMd&&await nt(t.claudeMd,{force:!0}),e.keepHooks||(t.postMergeHook&&await nt(t.postMergeHook,{force:!0}),t.prePushHook&&await nt(t.prePushHook,{force:!0})),t.gitignorePath&&await Up(t.gitignorePath),t.gitmodulesPath&&!e.keepSubmodule&&await Dp(t.gitmodulesPath,".claude/pack");for(let n of[t.notesDir,t.scriptsDir]){if(!n)continue;let{readdir:r}=await import("fs/promises");(await r(n)).length===0&&await nt(n,{recursive:!0,force:!0})}}async function Up(t){let e=await ys(t,"utf8"),n=e.indexOf(rr),r=e.indexOf(_e);if(n===-1||r===-1)return;let o=e.slice(0,n),i=e.slice(r+_e.length),a=`${o.trimEnd()}
132
- ${i.trimStart()}`.trim();a.length===0?await nt(t,{force:!0}):await bs(t,`${a}
133
- `,"utf8")}async function Dp(t,e){let r=(await ys(t,"utf8")).split(`
131
+ Dry-run done. Kh\xF4ng apply. B\u1ECF --dry-run \u0111\u1EC3 th\u1EF1c thi.`);return}let o;try{o=await xe()}catch(c){s.error(c instanceof Error?c.message:String(c)),process.exit(1)}s.info(t.packVersion?`T\u1EA3i team-ai-pack ${t.packVersion} t\u1EEB Supabase...`:"T\u1EA3i team-ai-pack (stable m\u1EDBi nh\u1EA5t) t\u1EEB Supabase...");let i;try{let c=await Ge(o,t.packVersion);await Ue(c.url,r),i=t.packVersion??$p(c.object),await Me(r,{version:i,downloadedObject:c.object,extractedAt:new Date().toISOString()}),s.success(`\u0110\xE3 c\xE0i pack ${i}`)}catch(c){c instanceof wt?s.error(`${c.message}`):s.error(`T\u1EA3i pack th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`),process.exit(1)}s.info("T\u1EA1o symlink farm...");let a=await Ve(r,n,t.force===!0);Np(a,t.force===!0),s.info("Merge pack settings.json template v\xE0o project settings.json...");try{let c=await ge(e);switch(c.action){case"merged":s.success(` \u2713 settings.json merged (${c.changes.join("; ")})`);break;case"no-change":s.info(" - settings.json \u0111\xE3 sync.");break;case"no-pack-template":s.dim(" - Pack kh\xF4ng c\xF3 templates/settings.json.tpl, skip.");break}}catch(c){s.warn(` ! Merge settings.json fail: ${c instanceof Error?c.message:c}.`)}try{await us(e)}catch(c){s.warn(` ! Re-apply tools fail: ${c instanceof Error?c.message:c}.`)}s.info("C\u1EADp nh\u1EADt managed block trong CLAUDE.md...");try{let c=await _p(e,dr(e,"CLAUDE.md")),l={projectName:c.projectName,projectDescription:c.projectDescription,teamOwner:c.teamOwner,avatarVersion:I(),packVersion:i,lastScan:new Date().toISOString(),gitnexusSection:""};switch((await rn(e,l)).outcome){case"replaced-block":s.success(" \u2713 CLAUDE.md managed block \u0111\xE3 c\u1EADp nh\u1EADt (gi\u1EEF context user ngo\xE0i block).");break;case"replaced-whole":s.success(" \u2713 CLAUDE.md (ch\u01B0a c\xF3 marker) \u2192 thay ho\xE0n to\xE0n b\u1EB1ng b\u1EA3n Avatar m\u1EDBi.");break;case"skipped-no-file":s.dim(" - Ch\u01B0a c\xF3 CLAUDE.md (ch\u1EA1y 'avatar init' \u0111\u1EC3 t\u1EA1o), skip.");break}}catch(c){s.warn(` ! C\u1EADp nh\u1EADt CLAUDE.md fail: ${c instanceof Error?c.message:c}.`)}s.success(`Synced team-ai-pack \u2192 ${i}.`)}function Np(t,e){for(let r of t)switch(r.action){case"created":s.info(` \u2713 ${r.dir} \u2192 symlinked (new)`);break;case"updated":s.info(` \u2713 ${r.dir} \u2192 symlink refreshed`);break;case"backed-up-and-linked":s.info(` \u2713 ${r.dir} \u2192 symlinked (backup: ${r.backupPath})`);break;case"source-missing":s.warn(` - ${r.dir} \u2192 pack kh\xF4ng c\xF3 dir n\xE0y, skip`);break;case"skipped-conflict":s.warn(` ! ${r.dir} \u2192 CONFLICT: existing real dir. D\xF9ng --force \u0111\u1EC3 backup + override.`);break}let n=t.filter(r=>r.action==="skipped-conflict").length;n>0&&!e&&s.warn(`${n} mount dir(s) skip do conflict. Ch\u1EA1y l\u1EA1i v\u1EDBi --force.`)}function ps(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("--pack-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(Ip)}import{resolve as Mp}from"path";import{checkbox as Op,confirm as Lp}from"@inquirer/prompts";f();function ms(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 ds(t){return t.map(e=>({name:e,value:e,checked:!1}))}function gs(t){return[...t]}function te(t){return Mp(t.target??process.cwd())}async function fs(t,e,n,r){if(n.all===!0||!process.stdout.isTTY)return gs(t);let i=r==="add"?ms(t,e):ds(t);return await Op({message:r==="add"?"Ch\u1ECDn tool \u0111\u1EC3 c\xE0i (space ch\u1ECDn, enter x\xE1c nh\u1EADn):":"Ch\u1ECDn tool \u0111\u1EC3 g\u1EE1 (space ch\u1ECDn, enter x\xE1c nh\u1EADn):",choices:i})}async function Gp(t){let e=te(t),{available:n,enabled:r}=await Zt(e);if(n.length===0){s.dim("Kh\xF4ng c\xF3 tool available (pack ch\u01B0a c\xF3 tools/, ho\u1EB7c ch\u01B0a sync).");return}let o=await fs(n,r,t,"add");if(o.length===0){s.dim("Kh\xF4ng ch\u1ECDn tool n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}for(let i of o)await H(e,i)}async function Up(t){let e=te(t),{enabled:n}=await Zt(e);if(n.length===0){s.dim("Kh\xF4ng c\xF3 tool n\xE0o \u0111ang b\u1EADt \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}let r=await fs(n,[],t,"remove");if(r.length===0){s.dim("Kh\xF4ng ch\u1ECDn tool n\xE0o \u2014 h\u1EE7y, kh\xF4ng thay \u0111\u1ED5i.");return}if(t.yes!==!0&&t.all!==!0&&process.stdout.isTTY&&!await Lp({message:`G\u1EE1 ${r.length} tool: ${r.join(", ")}?`,default:!0})){s.dim("H\u1EE7y \u2014 kh\xF4ng g\u1EE1 tool n\xE0o.");return}for(let i of r)await Nn(e,i)}async function jp(t){let e=te(t),n=await lr(e),r=await Pt(e);if(n.length===0&&Object.keys(r.tools).length===0){s.dim("Kh\xF4ng c\xF3 tool n\xE0o (pack ch\u01B0a c\xF3 tools/, ho\u1EB7c ch\u01B0a sync). Ch\u1EA1y 'avatar sync'.");return}let o=[...new Set([...n,...Object.keys(r.tools)])].sort();s.info("Tools:"),console.log(` ${"NAME".padEnd(20)} ${"AVAILABLE".padEnd(10)} ${"ENABLED".padEnd(8)} VERSION`);for(let i of o){let a=n.includes(i)?"yes":"no",c=r.tools[i],l=c?.enabled?"yes":"no",u=c?.version??"-";console.log(` ${i.padEnd(20)} ${a.padEnd(10)} ${l.padEnd(8)} ${u}`)}}function hs(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=te(r);await H(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=te(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 jp(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 Gp(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 Up(n)})}import{relative as qp}from"path";import{confirm as zp}from"@inquirer/prompts";import Jp from"boxen";import{cp as We,mkdir as ws,writeFile as Dp}from"fs/promises";import{homedir as Hp}from"os";import{basename as Kp,join as tt}from"path";var Vp=tt(Hp(),".avatar","uninstall-backups");async function ks(t,e,n){let r=Kp(t),o=new Date().toISOString().replace(/[:.]/g,"-"),i=tt(Vp,`${r}-${o}`);if(await ws(i,{recursive:!0,mode:448}),e.claudeDir&&await We(e.claudeDir,tt(i,".claude"),{recursive:!0}),e.claudeMd&&await We(e.claudeMd,tt(i,"CLAUDE.md")),e.postMergeHook||e.prePushHook){let c=tt(i,"hooks");await ws(c,{recursive:!0}),e.postMergeHook&&await We(e.postMergeHook,tt(c,"post-merge")),e.prePushHook&&await We(e.prePushHook,tt(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 Dp(tt(i,"manifest.json"),JSON.stringify(a,null,2),"utf8"),i}import{existsSync as Fp}from"fs";import{join as et}from"path";function nt(t){return Fp(t)?t:null}function ys(t){let e=nt(et(t,".claude")),n=nt(et(t,"CLAUDE.md")),r=nt(et(t,".git","hooks","post-merge")),o=nt(et(t,".git","modules","src","hooks","pre-push")),i=nt(et(t,".gitignore")),a=nt(et(t,".gitmodules")),c=nt(et(t,"notes")),l=nt(et(t,"scripts"));return{hasAnyArtifact:!!(e||n||r||o),claudeDir:e,claudeMd:n,postMergeHook:r,prePushHook:o,gitignorePath:i,gitmodulesPath:a,notesDir:c,scriptsDir:l}}import{readFile as bs,rm as rt,writeFile as vs}from"fs/promises";async function xs(t,e){if(t.claudeDir)if(e.keepSubmodule){let{readdir:n}=await import("fs/promises"),{join:r}=await import("path"),o=await n(t.claudeDir);for(let i of o)i!=="pack"&&await rt(r(t.claudeDir,i),{recursive:!0,force:!0})}else await rt(t.claudeDir,{recursive:!0,force:!0});t.claudeMd&&await rt(t.claudeMd,{force:!0}),e.keepHooks||(t.postMergeHook&&await rt(t.postMergeHook,{force:!0}),t.prePushHook&&await rt(t.prePushHook,{force:!0})),t.gitignorePath&&await Bp(t.gitignorePath),t.gitmodulesPath&&!e.keepSubmodule&&await Wp(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 rt(n,{recursive:!0,force:!0})}}async function Bp(t){let e=await bs(t,"utf8"),n=e.indexOf(or),r=e.indexOf(Ie);if(n===-1||r===-1)return;let o=e.slice(0,n),i=e.slice(r+Ie.length),a=`${o.trimEnd()}
132
+ ${i.trimStart()}`.trim();a.length===0?await rt(t,{force:!0}):await vs(t,`${a}
133
+ `,"utf8")}async function Wp(t,e){let r=(await bs(t,"utf8")).split(`
134
134
  `),o=[],i=!1;for(let c of r){if(c.trim().startsWith("[submodule")&&c.includes(e)){i=!0;continue}i&&c.trim().startsWith("[")&&(i=!1),i||o.push(c)}let a=o.join(`
135
- `).trim();a.length===0?await nt(t,{force:!0}):await bs(t,`${a}
136
- `,"utf8")}f();function xs(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 Vp(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Vp(t){let e=process.cwd(),n=ks(e);if(!n.hasAnyArtifact){s.info("Project ch\u01B0a c\xE0i Avatar \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}if(Bp(e,n,t),t.dryRun){s.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await Kp({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){s.info("\u0110\xE3 h\u1EE7y.");return}let r=null;t.noBackup||(r=await ws(e,n,N()),s.success(`Backup t\u1EA1o t\u1EA1i: ${r}`)),await vs(n,{keepSubmodule:t.keepSubmodule,keepHooks:t.keepHooks}),await v("uninstall",`project=${e},backup=${r??"skipped"}`),Wp(r)}function Bp(t,e,n){s.info(`Project: ${t}`),s.plain(""),s.plain("C\xE1c artifact s\u1EBD g\u1EE1:"),e.claudeDir&&s.plain(` ${p.red("\u2717")} ${Hp(t,e.claudeDir)||".claude/"}`),e.claudeMd&&s.plain(` ${p.red("\u2717")} CLAUDE.md`),e.postMergeHook&&!n.keepHooks&&s.plain(` ${p.red("\u2717")} .git/hooks/post-merge`),e.prePushHook&&!n.keepHooks&&s.plain(` ${p.red("\u2717")} .git/modules/src/hooks/pre-push`),e.gitignorePath&&s.plain(` ${p.yellow("\u270E")} .gitignore (g\u1EE1 Avatar block)`),e.gitmodulesPath&&!n.keepSubmodule&&s.plain(` ${p.yellow("\u270E")} .gitmodules (g\u1EE1 entry .claude/pack)`),s.plain(""),s.plain("Kh\xF4ng \u0111\u1EE5ng:"),s.plain(` ${p.green("\u2713")} src/ (code kh\xE1ch)`),s.plain(` ${p.green("\u2713")} Git history`),s.plain(` ${p.green("\u2713")} ~/.avatar/config.json (token SSO)`),s.plain(` ${p.green("\u2713")} Secrets trong keychain`),s.plain("")}function Wp(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(`${Fp(e.join(`
135
+ `).trim();a.length===0?await rt(t,{force:!0}):await vs(t,`${a}
136
+ `,"utf8")}f();function As(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 Yp(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Yp(t){let e=process.cwd(),n=ys(e);if(!n.hasAnyArtifact){s.info("Project ch\u01B0a c\xE0i Avatar \u2014 kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 g\u1EE1.");return}if(Xp(e,n,t),t.dryRun){s.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await zp({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){s.info("\u0110\xE3 h\u1EE7y.");return}let r=null;t.noBackup||(r=await ks(e,n,I()),s.success(`Backup t\u1EA1o t\u1EA1i: ${r}`)),await xs(n,{keepSubmodule:t.keepSubmodule,keepHooks:t.keepHooks}),await v("uninstall",`project=${e},backup=${r??"skipped"}`),Qp(r)}function Xp(t,e,n){s.info(`Project: ${t}`),s.plain(""),s.plain("C\xE1c artifact s\u1EBD g\u1EE1:"),e.claudeDir&&s.plain(` ${p.red("\u2717")} ${qp(t,e.claudeDir)||".claude/"}`),e.claudeMd&&s.plain(` ${p.red("\u2717")} CLAUDE.md`),e.postMergeHook&&!n.keepHooks&&s.plain(` ${p.red("\u2717")} .git/hooks/post-merge`),e.prePushHook&&!n.keepHooks&&s.plain(` ${p.red("\u2717")} .git/modules/src/hooks/pre-push`),e.gitignorePath&&s.plain(` ${p.yellow("\u270E")} .gitignore (g\u1EE1 Avatar block)`),e.gitmodulesPath&&!n.keepSubmodule&&s.plain(` ${p.yellow("\u270E")} .gitmodules (g\u1EE1 entry .claude/pack)`),s.plain(""),s.plain("Kh\xF4ng \u0111\u1EE5ng:"),s.plain(` ${p.green("\u2713")} src/ (code kh\xE1ch)`),s.plain(` ${p.green("\u2713")} Git history`),s.plain(` ${p.green("\u2713")} ~/.avatar/config.json (token SSO)`),s.plain(` ${p.green("\u2713")} Secrets trong keychain`),s.plain("")}function Qp(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(`${Jp(e.join(`
137
137
  `),{padding:1,borderStyle:"round"})}
138
- `)}var dr=N(),P=new qp;P.name("avatar").description("AI harness CLI for NAL Vietnam engineering").version(dr,"-v, --version","Hi\u1EC3n th\u1ECB phi\xEAn b\u1EA3n Avatar CLI").addHelpText("beforeAll",()=>`
139
- ${nr({tagline:`v${dr} \xB7 AI harness CLI for NAL Vietnam`})}
138
+ `)}import{spawnSync as Cs}from"child_process";var ee="@nalvietnam/avatar-cli";function Ss(){return D()==="win32"}function Zp(t){return["view",`${ee}@${t}`,"version"]}function tm(t){return["install","-g",`${ee}@${t}`]}function em(t){let n=t.split(`
139
+ `).map(r=>r.trim()).filter(Boolean).at(-1)??"";return/^\d+\.\d+\.\d+(-[\w.]+)?$/.test(n)?n:null}function Ts(t){let e=Cs("npm",Zp(t),{encoding:"utf-8",shell:Ss()});return e.status!==0||!e.stdout?null:em(e.stdout)}function Es(t){return Cs("npm",tm(t),{stdio:"inherit",shell:Ss()}).status??1}f();function Ps(t){t.command("update").description("Update avatar CLI l\xEAn b\u1EA3n m\u1EDBi nh\u1EA5t (npm install -g)").option("--tag <dist-tag>","npm dist-tag mu\u1ED1n theo (latest | next)","latest").action(async e=>{let n=I(),r=Ts(e.tag);if(r||(s.error(`Kh\xF4ng l\u1EA5y \u0111\u01B0\u1EE3c version cho tag '${e.tag}' t\u1EEB npm registry. Ki\u1EC3m tra m\u1EA1ng/npm r\u1ED3i th\u1EED l\u1EA1i, ho\u1EB7c ch\u1EA1y tay: npm i -g ${ee}@${e.tag}`),process.exit(1)),r===n){s.success(`avatar CLI \u0111\xE3 \u1EDF b\u1EA3n m\u1EDBi nh\u1EA5t c\u1EE7a tag '${e.tag}' (v${n}).`);return}s.info(`Update avatar CLI: v${n} \u2192 v${r} (tag '${e.tag}')...`);let o=Es(e.tag);o!==0&&(s.error(`npm install th\u1EA5t b\u1EA1i (exit ${o}). Th\u1EED ch\u1EA1y tay: npm i -g ${ee}@${e.tag}`),process.exit(o)),s.success(`\u0110\xE3 update avatar CLI \u2192 v${r}. M\u1EDF terminal m\u1EDBi n\u1EBFu version ch\u01B0a \u0111\u1ED5i.`)})}var gr=I(),E=new nm;E.name("avatar").description("AI harness CLI for NAL Vietnam engineering").version(gr,"-v, --version","Hi\u1EC3n th\u1ECB phi\xEAn b\u1EA3n Avatar CLI").addHelpText("beforeAll",()=>`
140
+ ${rr({tagline:`v${gr} \xB7 AI harness CLI for NAL Vietnam`})}
140
141
 
141
- `);var As=process.argv[2],zp=As==="-v"||As==="--version";zp&&($t({tagline:`v${dr} \xB7 AI harness CLI for NAL Vietnam`}),process.exit(0));Hi(P);Qi(P);Xi(P);us(P);os(P);rs(P);ss(P);ui(P);ns(P);fs(P);Zi(P);ai(P);yi(P);es(P);ko(P);xs(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}
142
+ `);var Rs=process.argv[2],rm=Rs==="-v"||Rs==="--version";rm&&($t({tagline:`v${gr} \xB7 AI harness CLI for NAL Vietnam`}),process.exit(0));Ki(E);Zi(E);Qi(E);ps(E);is(E);os(E);as(E);pi(E);rs(E);hs(E);ts(E);ci(E);bi(E);ns(E);yo(E);As(E);Ps(E);E.parseAsync(process.argv).catch(t=>{let e=t instanceof Error?t.message:String(t);process.stderr.write(`\u2717 L\u1ED7i kh\xF4ng x\u1EED l\xFD \u0111\u01B0\u1EE3c: ${e}
142
143
  `),process.exit(1)});
143
144
  //# sourceMappingURL=index.js.map