@nalvietnam/avatar-cli 3.3.1 → 3.5.0-beta.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,8 +1,10 @@
1
1
  // @nalvietnam/avatar-cli — built with tsup
2
- var as=Object.defineProperty;var I=(t,e)=>()=>(t&&(e=t(t=0)),e);var rt=(t,e)=>{for(var n in e)as(t,n,{get:e[n],enumerable:!0})};var yt={};rt(yt,{computeFileSha256:()=>ys,copyDirRecursive:()=>Be,downloadFile:()=>We,ensureDir:()=>x,extractTarballToDir:()=>qe,pathExists:()=>m,readJson:()=>b,readText:()=>O,relativeFromCwd:()=>gs,removeRecursive:()=>ot,writeJsonAtomic:()=>S,writeTextAtomic:()=>F});import{spawn as cr,spawnSync as cs}from"child_process";import{createHash as ls}from"crypto";import{constants as us,promises as N,createReadStream as ps}from"fs";import{dirname as lr,join as Ve,relative as ur,sep as ms}from"path";import{Readable as ds}from"stream";async function m(t){try{return await N.access(t,us.F_OK),!0}catch{return!1}}async function x(t){await N.mkdir(t,{recursive:!0})}async function O(t){return await N.readFile(t,"utf8")}async function b(t){return JSON.parse(await O(t))}async function F(t,e,n){await x(lr(t));let r=`${t}.tmp-${process.pid}-${Date.now()}`,o=!1;try{await N.writeFile(r,e,"utf8"),n!==void 0&&await N.chmod(r,n),await N.rename(r,t),o=!0}finally{o||await N.rm(r,{force:!0}).catch(()=>{})}}async function S(t,e,n){await F(t,`${JSON.stringify(e,null,2)}
3
- `,n)}async function Be(t,e,n=[]){await x(e);let r=await N.readdir(t,{withFileTypes:!0});for(let o of r){if(n.includes(o.name))continue;let i=Ve(t,o.name),a=Ve(e,o.name);if(o.isDirectory())await Be(i,a,n);else if(o.isSymbolicLink()){let c=await N.readlink(i);await N.symlink(c,a)}else await N.copyFile(i,a)}}async function ot(t){await N.rm(t,{recursive:!0,force:!0})}function gs(t){return ur(process.cwd(),t)}async function We(t,e,n=fs){let r=new AbortController,o=setTimeout(()=>r.abort(),n);await x(lr(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=ds.fromWeb(a.body);d.on("error",u),c.on("error",u),c.on("finish",()=>l()),d.pipe(c)})}catch(a){throw await N.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 hs(){return ee!==null||(ee=cs("tar",["--version"],{stdio:"ignore"}).status===0),ee}async function ws(t){return await new Promise((e,n)=>{let r=cr("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=Ve(e,t),r=ur(e,n);return r!==".."&&!r.startsWith(`..${ms}`)}async function qe(t,e){if(!hs())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 ws(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=cr("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 ys(t){return await new Promise((e,n)=>{let r=ls("sha256"),o=ps(t);o.on("data",i=>r.update(i)),o.on("end",()=>e(r.digest("hex"))),o.on("error",n)})}var fs,ee,k=I(()=>{"use strict";fs=3e5;ee=null});var pr={};rt(pr,{addSubmodule:()=>As,checkoutBranchHeadInSubmodule:()=>Ss,checkoutTagInSubmodule:()=>Cs,currentBranch:()=>xs,currentCommitSha:()=>Es,git:()=>R,isGitRepo:()=>vs,listTags:()=>Ts,tagAtHead:()=>Ps,workingTreeIsDirty:()=>$s});import{join as ze}from"path";import{simpleGit as bs}from"simple-git";function R(t=process.cwd()){return bs({baseDir:t,binary:"git"})}async function vs(t=process.cwd()){return await m(ze(t,".git"))}async function xs(t=process.cwd()){return(await R(t).revparse(["--abbrev-ref","HEAD"])).trim()}async function As(t,e,n=process.cwd()){await R(n).subModule(["add",t,e])}async function Cs(t,e,n=process.cwd()){let r=ze(n,t);await R(r).fetch(["--tags"]),await R(r).checkout(e)}async function Ss(t,e,n=process.cwd()){let r=ze(n,t);await R(r).fetch(["origin"]),await R(r).checkout(["-B",e,`origin/${e}`])}async function Ts(t=process.cwd()){return(await R(t).tags()).all}async function Ps(t=process.cwd()){try{return(await R(t).raw(["describe","--tags","--exact-match","HEAD"])).trim()||null}catch{return null}}async function Es(t=process.cwd()){return(await R(t).revparse(["HEAD"])).trim()}async function $s(t=process.cwd()){return!(await R(t).status()).isClean()}var Je=I(()=>{"use strict";k()});function mr(t,e){return t.replace(Rs,(n,r)=>{let o=e[r];return o===void 0?n:String(o)})}var Rs,dr=I(()=>{"use strict";Rs=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g});import{existsSync as _s}from"fs";import{dirname as gr,join as Nt}from"path";import{fileURLToPath as Is}from"url";function Ms(t){let e=t;for(;;){if(_s(Nt(e,"package.json")))return e;let n=gr(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function Ls(t){return await O(Nt(Os,`${t}.tpl`))}async function Ye(t,e){let n=await Ls(t);return mr(n,e)}async function hr(t){return await O(Nt(Gs,`${t}.sh.tpl`))}var Ns,fr,Os,Gs,wr=I(()=>{"use strict";k();dr();Ns=gr(Is(import.meta.url)),fr=Ms(Ns),Os=Nt(fr,"src","templates"),Gs=Nt(fr,"src","hooks")});var xr={};rt(xr,{AVATAR_MANAGED_PATHS:()=>kr,AVATAR_WORKSPACE_MARKER_RELATIVE:()=>vr,backupIfExists:()=>yr,createClaudeDirTree:()=>Xe,installGitHook:()=>ne,writeAvatarWorkspaceMarker:()=>Qe,writeClaudeGitignore:()=>nn,writeProjectKnowledgeFiles:()=>Ze,writeProjectSettings:()=>en,writeRootClaudeMd:()=>tn});import{promises as js}from"fs";import{join as j}from"path";async function yr(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 js.rename(t,r),r}async function br(t,e,n){let r=await yr(t);return await F(t,e,n),r}async function Xe(t){let e=j(t,".claude");await x(e);for(let n of Us){let r=j(e,n);await x(r),await F(j(r,".gitkeep"),"")}}async function Qe(t,e){let n=j(t,vr),r=`${JSON.stringify({avatarWorkspace:!0,avatarVersion:e.avatarVersion,workspaceName:e.workspaceName,createdAt:new Date().toISOString()},null,2)}
5
- `;await F(n,r)}async function Ze(t,e){return[]}async function tn(t,e){let n=await Ye("CLAUDE.md",e);return await br(j(t,"CLAUDE.md"),n)}async function en(t,e){let n=await Ye("settings.json",e);return await br(j(t,".claude","settings.json"),n)}async function nn(t){let e=j(t,".claude",".gitignore");await F(e,Ds)}async function ne(t,e){let n=await hr(e),r=j(t,"hooks");await x(r);let o=j(r,e);await F(o,n,493)}var kr,Us,vr,Ds,Ot=I(()=>{"use strict";k();wr();kr=[".claude","CLAUDE.md"];Us=["state","_pending","_backup"];vr=".claude/avatar.json";Ds=`# Avatar \u2014 b\u1EA3o v\u1EC7 file nh\u1EA1y c\u1EA3m trong .claude/.
2
+ var Ss=Object.defineProperty;var _=(t,e)=>()=>(t&&(e=t(t=0)),e);var V=(t,e)=>{for(var n in e)Ss(t,n,{get:e[n],enumerable:!0})};var yt={};V(yt,{computeFileSha256:()=>Gs,copyDirRecursive:()=>qe,downloadFile:()=>ze,ensureDir:()=>A,extractTarballToDir:()=>Je,pathExists:()=>m,readJson:()=>b,readText:()=>M,relativeFromCwd:()=>Is,removeRecursive:()=>ot,writeJsonAtomic:()=>C,writeTextAtomic:()=>U});import{spawn as gr,spawnSync as Ts}from"child_process";import{createHash as Es}from"crypto";import{constants as Ps,promises as O,createReadStream as Rs}from"fs";import{dirname as fr,join as We,relative as hr,sep as $s}from"path";import{Readable as _s}from"stream";async function m(t){try{return await O.access(t,Ps.F_OK),!0}catch{return!1}}async function A(t){await O.mkdir(t,{recursive:!0})}async function M(t){return await O.readFile(t,"utf8")}async function b(t){return JSON.parse(await M(t))}async function U(t,e,n){await A(fr(t));let r=`${t}.tmp-${process.pid}-${Date.now()}`,o=!1;try{await O.writeFile(r,e,"utf8"),n!==void 0&&await O.chmod(r,n),await O.rename(r,t),o=!0}finally{o||await O.rm(r,{force:!0}).catch(()=>{})}}async function C(t,e,n){await U(t,`${JSON.stringify(e,null,2)}
3
+ `,n)}async function qe(t,e,n=[]){await A(e);let r=await O.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 O.readlink(i);await O.symlink(c,a)}else await O.copyFile(i,a)}}async function ot(t){await O.rm(t,{recursive:!0,force:!0})}function Is(t){return hr(process.cwd(),t)}async function ze(t,e,n=Os){let r=new AbortController,o=setTimeout(()=>r.abort(),n);await A(fr(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 O.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 re!==null||(re=Ts("tar",["--version"],{stdio:"ignore"}).status===0),re}async function Ms(t){return await new Promise((e,n)=>{let r=gr("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=hr(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 A(e);let r=(await Ms(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=gr("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 Os,re,w=_(()=>{"use strict";Os=3e5;re=null});var wr={};V(wr,{addSubmodule:()=>Hs,checkoutBranchHeadInSubmodule:()=>Ks,checkoutTagInSubmodule:()=>Fs,currentBranch:()=>js,currentCommitSha:()=>Ws,git:()=>$,isGitRepo:()=>Us,listTags:()=>Vs,tagAtHead:()=>Bs,workingTreeIsDirty:()=>qs});import{join as Ye}from"path";import{simpleGit as Ds}from"simple-git";function $(t=process.cwd()){return Ds({baseDir:t,binary:"git"})}async function Us(t=process.cwd()){return await m(Ye(t,".git"))}async function js(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 Fs(t,e,n=process.cwd()){let r=Ye(n,t);await $(r).fetch(["--tags"]),await $(r).checkout(e)}async function Ks(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 kr(t,e){return t.replace(zs,(n,r)=>{let o=e[r];return o===void 0?n:String(o)})}var zs,yr=_(()=>{"use strict";zs=/\{\{\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*\}\}/g});import{existsSync as Js}from"fs";import{dirname as br,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=br(e);if(n===e)throw new Error(`Cannot locate package root from ${t}`);e=n}}async function ea(t){return await M(Nt(Qs,`${t}.tpl`))}async function oe(t,e){let n=await ea(t);return kr(n,e)}async function xr(t){return await M(Nt(Zs,`${t}.sh.tpl`))}var Xs,vr,Qs,Zs,Ar=_(()=>{"use strict";w();yr();Xs=br(Ys(import.meta.url)),vr=ta(Xs),Qs=Nt(vr,"src","templates"),Zs=Nt(vr,"src","hooks")});var Er={};V(Er,{MANAGED_END_PREFIX:()=>Tr,MANAGED_START_PREFIX:()=>Sr,mergeClaudeMdManagedBlock:()=>na});function Cr(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=Cr(n,Sr),o=Cr(n,Tr);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 Sr,Tr,Pr=_(()=>{"use strict";Sr="<!-- AVATAR:MANAGED:START",Tr="<!-- AVATAR:MANAGED:END"});var Nr={};V(Nr,{AVATAR_MANAGED_PATHS:()=>$r,AVATAR_WORKSPACE_MARKER_RELATIVE:()=>Or,backupIfExists:()=>_r,createClaudeDirTree:()=>Qe,installGitHook:()=>ie,mergeRootClaudeMdManaged:()=>nn,writeAvatarWorkspaceMarker:()=>Ze,writeClaudeGitignore:()=>on,writeProjectKnowledgeFiles:()=>tn,writeProjectSettings:()=>rn,writeRootClaudeMd:()=>en});import{promises as Rr}from"fs";import{join as G}from"path";async function _r(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 Rr.rename(t,r),r}async function Ir(t,e,n){let r=await _r(t);return await U(t,e,n),r}async function Qe(t){let e=G(t,".claude");await A(e);for(let n of ra){let r=G(e,n);await A(r),await U(G(r,".gitkeep"),"")}}async function Ze(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 U(n,r)}async function tn(t,e){return[]}async function en(t,e){let n=await oe("CLAUDE.md",e);return await Ir(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 Rr.readFile(n,"utf8"),o=await oe("CLAUDE.md",e),{mergeClaudeMdManagedBlock:i}=await Promise.resolve().then(()=>(Pr(),Er)),a=i(r,o);return await U(n,a.content),{outcome:a.outcome}}async function rn(t,e){let n=await oe("settings.json",e);return await Ir(G(t,".claude","settings.json"),n)}async function on(t){let e=G(t,".claude",".gitignore");await U(e,oa)}async function ie(t,e){let n=await xr(e),r=G(t,"hooks");await A(r);let o=G(r,e);await U(o,n,493)}var $r,ra,Or,oa,bt=_(()=>{"use strict";w();Ar();$r=[".claude","CLAUDE.md"];ra=["state","_pending","_backup"];Or=".claude/avatar.json";oa=`# Avatar \u2014 b\u1EA3o v\u1EC7 file nh\u1EA1y c\u1EA3m trong .claude/.
6
8
  # Workspace root KH\xD4NG git; file n\xE0y ph\xF2ng tr\u01B0\u1EDDng h\u1EE3p .claude/ b\u1ECB track nh\u1EA7m.
7
9
  settings.json
8
10
  settings.json.backup-*
@@ -11,79 +13,79 @@ settings.json.backup-*
11
13
  _pending/
12
14
  _backup/
13
15
  state/session-*.json
14
- `});var it={};rt(it,{REPOS_MANIFEST_RELATIVE:()=>Ar,addRepoToManifest:()=>oe,readReposManifest:()=>re,removeRepoFromManifest:()=>Ks,repoNameExists:()=>ie});import{join as Hs}from"path";function rn(t){return Hs(t,Ar)}async function re(t){let e=rn(t);if(!await m(e))return[];try{let n=await b(e);return Array.isArray(n.repos)?n.repos:[]}catch{return[]}}async function oe(t,e){let r=(await re(t)).filter(o=>o.name!==e.name);r.push(e),await S(rn(t),{repos:r})}async function Ks(t,e){let n=await re(t);await S(rn(t),{repos:n.filter(r=>r.name!==e)})}async function ie(t,e){return(await re(t)).some(r=>r.name===e)}var Ar,V=I(()=>{"use strict";k();Ar=".claude/repos.json"});var pn={};rt(pn,{RepoAlreadyInWorkspaceError:()=>at,cloneCodeRepoIntoSrc:()=>an,folderDirtyStatus:()=>ln,inferRepoNameFromUrl:()=>sn,readFolderRemoteUrl:()=>un,scaffoldNewRepoInSrc:()=>cn,validateRepoName:()=>on});import{join as st}from"path";function on(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 sn(t){return(t.trim().replace(/\/+$/,"").split(/[/:]/).pop()??"").replace(/\.git$/,"")||"repo"}async function an(t){let e=on(t.name);if(e)throw new at(e);let n=st(t.workspaceRoot,"src");await x(n);let r=st(n,t.name),o=await m(r),i=await ie(t.workspaceRoot,t.name);if(o)throw new at(`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(()=>(V(),it));await c(t.workspaceRoot,t.name)}await R(n).clone(t.url,t.name);let a=st(r,".git");return await ne(a,"pre-push").catch(()=>{}),await oe(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 cn(t){let e=on(t.name);if(e)throw new Error(e);if(await ie(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=st(t.workspaceRoot,"src");await x(n);let r=st(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(r);await o.init();let{writeTextAtomic:i}=await Promise.resolve().then(()=>(k(),yt));return await i(st(r,"README.md"),`# ${t.name}
15
- `),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 ne(st(r,".git"),"pre-push").catch(()=>{}),await oe(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 ln(t){try{let e=await R(t).status();return e.isClean()?"":e.files.map(n=>`${n.path}`).join(", ")}catch{return""}}async function un(t){try{let n=(await R(t).getRemotes(!0)).find(r=>r.name==="origin");return n?.refs.fetch??n?.refs.push??null}catch{return null}}var at,Gt=I(()=>{"use strict";k();Je();Ot();V();at=class extends Error{constructor(e){super(e),this.name="RepoAlreadyInWorkspaceError"}}});import p from"chalk";import Fs from"ora";function B(t){return Fs({text:t,spinner:"dots",isEnabled:process.stdout.isTTY??!1}).start()}function Mt(t){let e=Date.now(),n=B(`${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=I(()=>{"use strict";s={info:t=>process.stdout.write(`${p.blue("\u2139")} ${t}
16
+ `});var it={};V(it,{REPOS_MANIFEST_RELATIVE:()=>Mr,addRepoToManifest:()=>ae,readReposManifest:()=>se,removeRepoFromManifest:()=>sa,repoNameExists:()=>ce});import{join as ia}from"path";function sn(t){return ia(t,Mr)}async function se(t){let e=sn(t);if(!await m(e))return[];try{let n=await b(e);return Array.isArray(n.repos)?n.repos:[]}catch{return[]}}async function ae(t,e){let r=(await se(t)).filter(o=>o.name!==e.name);r.push(e),await C(sn(t),{repos:r})}async function sa(t,e){let n=await se(t);await C(sn(t),{repos:n.filter(r=>r.name!==e)})}async function ce(t,e){return(await se(t)).some(r=>r.name===e)}var Mr,B=_(()=>{"use strict";w();Mr=".claude/repos.json"});var dn={};V(dn,{RepoAlreadyInWorkspaceError:()=>at,cloneCodeRepoIntoSrc:()=>ln,folderDirtyStatus:()=>pn,inferRepoNameFromUrl:()=>cn,readFolderRemoteUrl:()=>mn,scaffoldNewRepoInSrc:()=>un,validateRepoName:()=>an});import{join as st}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 at(e);let n=st(t.workspaceRoot,"src");await A(n);let r=st(n,t.name),o=await m(r),i=await ce(t.workspaceRoot,t.name);if(o)throw new at(`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(()=>(B(),it));await c(t.workspaceRoot,t.name)}await $(n).clone(t.url,t.name);let a=st(r,".git");return await ie(a,"pre-push").catch(()=>{}),await ae(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 ce(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=st(t.workspaceRoot,"src");await A(n);let r=st(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 A(r);let o=$(r);await o.init();let{writeTextAtomic:i}=await Promise.resolve().then(()=>(w(),yt));return await i(st(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 ie(st(r,".git"),"pre-push").catch(()=>{}),await ae(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 at,Mt=_(()=>{"use strict";w();Xe();bt();B();at=class extends Error{constructor(e){super(e),this.name="RepoAlreadyInWorkspaceError"}}});import p from"chalk";import aa from"ora";function W(t){return aa({text:t,spinner:"dots",isEnabled:process.stdout.isTTY??!1}).start()}function Lt(t){let e=Date.now(),n=W(`${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
18
  `),success:t=>process.stdout.write(`${p.green("\u2713")} ${t}
17
19
  `),warn:t=>process.stdout.write(`${p.yellow("\u26A0")} ${t}
18
20
  `),error:t=>process.stderr.write(`${p.red("\u2717")} ${t}
19
21
  `),dim:t=>process.stdout.write(`${p.dim(t)}
20
22
  `),plain:t=>process.stdout.write(`${t}
21
- `)}});import{spawnSync as Qs}from"child_process";function ae(){let t=Qs("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 dn=I(()=>{"use strict"});import{spawnSync as Zs}from"child_process";function gn(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 Er(t,e,n){let o=`${n??ae()}/${t}`;s.info(`T\u1EA1o GitHub repo r\u1ED7ng ${o} (${e})...`);let i=Zs("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}
22
- `),new Lt(gn(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 Lt,fn=I(()=>{"use strict";dn();f();Lt=class extends Error{reason;constructor(e,n){super(n),this.name="RepoCreateError",this.reason=e}}});import{spawnSync as da}from"child_process";import{existsSync as ga,statSync as fa}from"fs";function Or(t){let e=`${t.org}/${t.name}`;if(!ga(t.folder)||!fa(t.folder).isDirectory())throw new xt("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=da("gh",n,{stdio:["ignore","pipe","pipe"],encoding:"utf8"});if(r.status!==0){let o=(r.stderr||"").trim();o&&process.stderr.write(`${o}
23
- `);let i=gn(o);throw i==="name-exists"?new hn(e):i==="no-permission"?new xt("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 xt("network",`L\u1ED7i m\u1EA1ng khi t\u1EA1o repo ${e}.`):new xt("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 hn,xt,Gr=I(()=>{"use strict";fn();hn=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"}},xt=class extends Error{reason;constructor(e,n){super(n),this.name="GhRepoCreateError",this.reason=e}}});function Mr(t){if(!ha.test(t))throw new wn(t)}function Lr(t){if(t!=="private"&&t!=="public")throw new Error(`Visibility ph\u1EA3i l\xE0 "private" ho\u1EB7c "public", nh\u1EADn: "${t}"`)}var ha,wn,jr=I(()=>{"use strict";ha=/^[a-zA-Z0-9._-]{1,100}$/,wn=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 Ur={};rt(Ur,{createGithubRemoteFromFolder:()=>kn});function kn(t){Mr(t.name),Lr(t.visibility);let e=t.org??ae();s.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=Or({folder:t.folder,org:e,name:t.name,visibility:t.visibility});return s.success(`\u0110\xE3 t\u1EA1o: ${n.sshUrl}`),n}var yn=I(()=>{"use strict";Gr();dn();f();jr()});var Rn={};rt(Rn,{ALLOWED_HOSTED_DOMAINS:()=>me,HOSTED_DOMAIN:()=>uc,SCOPES:()=>co,buildUserConfig:()=>En,buildVerificationUrl:()=>$n,decodeIdToken:()=>Tn,pollForToken:()=>Sn,refreshAccessToken:()=>fc,requestDeviceCode:()=>Cn,revokeToken:()=>Ft,verifyHostedDomain:()=>Pn,verifyIdTokenClaims:()=>uo});async function Cn(){let t=new URLSearchParams({client_id:de,scope:co.join(" ")}),e=await fetch(pc,{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 Sn(t){let e=new URLSearchParams({client_id:de,client_secret:ao,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(lo,{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 Tn(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 uo(t){if(!dc.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==de)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+gc<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||!me.includes(t.hd))throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u ${me.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 En(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 fc(t){let e=new URLSearchParams({client_id:de,client_secret:ao,refresh_token:t,grant_type:"refresh_token"}),n=await fetch(lo,{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 Ft(t){let e=new URLSearchParams({token:t}),n=new AbortController,r=setTimeout(()=>n.abort(),1e4);await fetch(mc,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e,signal:n.signal}).catch(()=>{}).finally(()=>clearTimeout(r))}function $n(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.toString()}var de,ao,me,uc,co,pc,lo,mc,dc,gc,Pn,Vt=I(()=>{"use strict";de="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",ao="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",me=["nal.vn","nal-software.com"],uc=me[0],co=["openid","email","profile"],pc="https://oauth2.googleapis.com/device/code",lo="https://oauth2.googleapis.com/token",mc="https://oauth2.googleapis.com/revoke";dc=new Set(["https://accounts.google.com","accounts.google.com"]),gc=60;Pn=uo});import{Command as kp}from"commander";Gt();import{resolve as no}from"path";import{confirm as Kt,input as Ht,select as xn}from"@inquirer/prompts";Gt();k();f();import{join as qs}from"path";import{confirm as zs,input as se,select as Tr}from"@inquirer/prompts";f();import{spawnSync as Vs}from"child_process";function bt(){s.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp GitHub qua gh CLI (browser s\u1EBD m\u1EDF)...");let t=Vs("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 Bs}from"child_process";var mn=5e3;function Ws(t){let e=t.toLowerCase();return e.includes("authentication")||e.includes("could not read username")||e.includes("permission denied")||e.includes("403")||e.includes("access denied")||e.includes("repository not found")?"no-access":e.includes("404")||e.includes("does not exist")?"not-found":e.includes("could not resolve host")||e.includes("network")||e.includes("connection refused")||e.includes("connection timed out")?"network":"unknown"}function vt(t){let e=Bs("git",["ls-remote","--exit-code",t,"HEAD"],{encoding:"utf8",timeout:mn,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 > ${mn/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 > ${mn/1e3}s`};let n=(e.stderr||"").trim();return{ok:!1,reason:Ws(n),detail:n.slice(0,300)}}function Js(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.
24
- 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 Cr(){s.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");try{bt()}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 Sr(t){let e=[{name:"Th\u1EED l\u1EA1i (retry)",value:"retry"},{name:"Nh\u1EADp URL repo kh\xE1c",value:"new-url"}];return t==="no-access"&&e.push({name:"Switch GitHub account (gh auth login \u2014 m\u1EDF browser)",value:"switch-account"}),e.push({name:"B\u1ECF qua repo n\xE0y",value:"skip"}),await Tr({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}async function Pr(t){let e=t.url,n=t.name;for(;;){let r=vt(e);if(!r.ok){Js(e,r.reason??"unknown",r.detail);let o=await Sr(r.reason??"unknown");if(o==="skip")return s.dim(`B\u1ECF qua repo ${n}.`),{cloned:null,skipped:!0};if(o==="switch-account"){Cr();continue}if(o==="new-url"){if(e=await se({message:"URL repo kh\xE1c:",validate:a=>a.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}),await zs({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(()=>(Gt(),pn));n=await se({message:"T\xEAn th\u01B0 m\u1EE5c:",default:a(e)})}continue}continue}try{return{cloned:await an({workspaceRoot:t.workspaceRoot,url:e,name:n}),skipped:!1}}catch(o){let i=o instanceof Error?o.message:String(o);if(o instanceof at){let c=await Ys(t.workspaceRoot,n);if(c&&Xs(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 Tr({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 se({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 ot(qs(t.workspaceRoot,"src",n)).catch(()=>{});let a=await Sr("unknown");if(a==="skip")return{cloned:null,skipped:!0};if(a==="switch-account"){Cr();continue}a==="new-url"&&(e=await se({message:"URL repo kh\xE1c:",validate:c=>c.trim().length>0?!0:"URL b\u1EAFt bu\u1ED9c"}))}}}async function Ys(t,e){let{readReposManifest:n}=await Promise.resolve().then(()=>(V(),it));return(await n(t)).find(o=>o.name===e)?.url??null}function Xs(t,e){let n=r=>r.trim().toLowerCase().replace(/^git@/,"").replace(/^https?:\/\//,"").replace(/:/,"/").replace(/\.git$/,"").replace(/\/+$/,"");return n(t)===n(e)}fn();import{input as ea,select as na}from"@inquirer/prompts";f();import{input as hm,select as ta}from"@inquirer/prompts";var A=class extends Error{constructor(e){super(e),this.name="UserAbortedRecoveryError"}};async function G(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 ta({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}f();function ra(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 oa(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 na({message:"C\xE1ch x\u1EED l\xFD?",choices:e})}async function $r(t,e){let n=t;for(;;)try{return await Er(n,e)}catch(r){let o=r instanceof Lt?r.reason:"unknown",i=await _r(o,n);if(i.action==="retry")continue;n=i.newName}}async function Rr(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 _r(o,n);if(i.action==="retry")continue;n=i.newName}}async function _r(t,e){ra(t);let n=await oa(t);if(n==="abort")throw new A("User h\u1EE7y t\u1EA1i b\u01B0\u1EDBc t\u1EA1o repo GitHub.");return n==="switch-account"?(bt(),{action:"retry",newName:e}):n==="rename"?{action:"rename",newName:(await ea({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 Ir}from"child_process";import{platform as ia}from"os";function ct(){let t=ia();return t==="darwin"||t==="linux"||t==="win32"?t:"unsupported"}var sa=5e3,aa=/(\d+\.\d+\.\d+)/;function ca(){let e=ct()==="win32"?"where":"which",n=Ir(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 la(){let t=Ir("gitnexus",["--version"],{encoding:"utf8",timeout:sa});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return aa.exec(e)?.[1]??null}var lt=null;function ut(){if(lt!==null)return lt;let t=ca();return t?(lt={installed:!0,version:la(),path:t},lt):(lt={installed:!1,version:null,path:null},lt)}function ce(){lt=null}import{spawnSync as ua}from"child_process";function le(){let t=ua("gh",["auth","status"],{stdio:"ignore"});return t.error&&t.error.code==="ENOENT"?"not-installed":t.status===0?"authenticated":"not-authenticated"}import{spawnSync as pa}from"child_process";function ma(t){let e=ct();return pa(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function Nr(){let t=ct(),e=t==="darwin"?["brew"]:t==="win32"?["winget"]:t==="linux"?["apt","dnf","pacman"]:[];for(let n of e)if(ma(n))return n;return null}import{spawnSync as Kr}from"child_process";import{input as Ca,select as Sa}from"@inquirer/prompts";yn();k();f();import{spawnSync as bn}from"child_process";import{promises as wa}from"fs";import{basename as ka,join as Dr}from"path";import{confirm as ya,select as ba}from"@inquirer/prompts";function va(){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 xa(t){let e=Dr(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-${va()}`,r=Dr(t,n);return await wa.rename(e,r),s.success(`Backup .git \u2192 ${n}`),r}function Aa(t){let e=bn("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=bn("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=bn("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 Hr(t){let e=ka(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await ya({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
23
+ `)}});import{spawnSync as ha}from"child_process";function ue(){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 jr(t,e,n){let o=`${n??ue()}/${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 Oa}from"fs";function Wr(t){let e=`${t.org}/${t.name}`;if(!Ia(t.folder)||!Oa(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,qr=_(()=>{"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 zr(t){if(!Na.test(t))throw new yn(t)}function Jr(t){if(t!=="private"&&t!=="public")throw new Error(`Visibility ph\u1EA3i l\xE0 "private" ho\u1EB7c "public", nh\u1EADn: "${t}"`)}var Na,yn,Yr=_(()=>{"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 Xr={};V(Xr,{createGithubRemoteFromFolder:()=>bn});function bn(t){zr(t.name),Jr(t.visibility);let e=t.org??ue();s.info(`T\u1EA1o GitHub repo ${e}/${t.name} (${t.visibility})...`);let n=Wr({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";qr();fn();f();Yr()});var Ln={};V(Ln,{ALLOWED_HOSTED_DOMAINS:()=>fe,HOSTED_DOMAIN:()=>Dc,SCOPES:()=>Eo,buildUserConfig:()=>Nn,buildVerificationUrl:()=>Mn,decodeIdToken:()=>In,pollForToken:()=>_n,refreshAccessToken:()=>Kc,requestDeviceCode:()=>$n,revokeToken:()=>Kt,verifyHostedDomain:()=>On,verifyIdTokenClaims:()=>Ro});async function $n(){let t=new URLSearchParams({client_id:he,scope:Eo.join(" ")}),e=await fetch(Uc,{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 _n(t){let e=new URLSearchParams({client_id:he,client_secret:To,device_code:t,grant_type:"urn:ietf:params:oauth:grant-type:device_code"}),n=await fetch(Po,{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 In(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 Ro(t){if(!Hc.has(t.iss))throw new Error(`id_token issuer kh\xF4ng h\u1EE3p l\u1EC7: ${t.iss} (expect: accounts.google.com)`);if(t.aud!==he)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+Fc<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||!fe.includes(t.hd))throw new Error(`Email kh\xF4ng thu\u1ED9c workspace NAL (y\xEAu c\u1EA7u ${fe.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 Nn(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 Kc(t){let e=new URLSearchParams({client_id:he,client_secret:To,refresh_token:t,grant_type:"refresh_token"}),n=await fetch(Po,{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 Kt(t){let e=new URLSearchParams({token:t}),n=new AbortController,r=setTimeout(()=>n.abort(),1e4);await fetch(jc,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:e,signal:n.signal}).catch(()=>{}).finally(()=>clearTimeout(r))}function Mn(t){let e=new URL(t.verification_url);return e.searchParams.set("user_code",t.user_code),e.toString()}var he,To,fe,Dc,Eo,Uc,Po,jc,Hc,Fc,On,Vt=_(()=>{"use strict";he="1014766441755-i4jimivh5rd7vt8phuhmepmt45sovtph.apps.googleusercontent.com",To="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",fe=["nal.vn","nal-software.com"],Dc=fe[0],Eo=["openid","email","profile"],Uc="https://oauth2.googleapis.com/device/code",Po="https://oauth2.googleapis.com/token",jc="https://oauth2.googleapis.com/revoke";Hc=new Set(["https://accounts.google.com","accounts.google.com"]),Fc=60;On=Ro});import{Command as Fp}from"commander";Mt();import{resolve as fo}from"path";import{confirm as Ft,input as Ht,select as Cn}from"@inquirer/prompts";Mt();w();f();import{join as pa}from"path";import{confirm as ma,input as le,select as Dr}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 Lr(){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 Gr(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 Ur(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 Gr(r.reason??"unknown");if(o==="skip")return s.dim(`B\u1ECF qua repo ${n}.`),{cloned:null,skipped:!0};if(o==="switch-account"){Lr();continue}if(o==="new-url"){if(e=await le({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(()=>(Mt(),dn));n=await le({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 at){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 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 le({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 ot(pa(t.workspaceRoot,"src",n)).catch(()=>{});let a=await Gr("unknown");if(a==="skip")return{cloned:null,skipped:!0};if(a==="switch-account"){Lr();continue}a==="new-url"&&(e=await le({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(()=>(B(),it));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 jm,select as ka}from"@inquirer/prompts";var S=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 Hr(t,e){let n=t;for(;;)try{return await jr(n,e)}catch(r){let o=r instanceof Gt?r.reason:"unknown",i=await Kr(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 Kr(o,n);if(i.action==="retry")continue;n=i.newName}}async function Kr(t,e){va(t);let n=await xa(t);if(n==="abort")throw new S("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 Vr}from"child_process";import{platform as Aa}from"os";function ct(){let t=Aa();return t==="darwin"||t==="linux"||t==="win32"?t:"unsupported"}var Ca=5e3,Sa=/(\d+\.\d+\.\d+)/;function Ta(){let e=ct()==="win32"?"where":"which",n=Vr(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=Vr("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 lt=null;function ut(){if(lt!==null)return lt;let t=Ta();return t?(lt={installed:!0,version:Ea(),path:t},lt):(lt={installed:!1,version:null,path:null},lt)}function pe(){lt=null}import{spawnSync as Pa}from"child_process";function me(){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=ct();return Ra(e==="win32"?"where":"command",e==="win32"?[t]:["-v",t],{shell:e!=="win32",stdio:"ignore"}).status===0}function Br(){let t=ct(),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 to}from"child_process";import{input as Fa,select as Ka}from"@inquirer/prompts";vn();w();f();import{spawnSync as xn}from"child_process";import{promises as Ma}from"fs";import{basename as La,join as Qr}from"path";import{confirm as Ga,select as Da}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 ja(t){let e=Qr(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=Qr(t,n);return await Ma.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 Zr(t){let e=La(t.folderPath),n=t.repoName??e;if(!t.autoYes&&!await Ga({message:`Folder '${e}' s\u1EBD \u0111\u01B0\u1EE3c reset:
25
27
  1. Backup .git \u2192 .git.backup-{timestamp}
26
28
  2. Git init m\u1EDBi (branch main, initial commit)
27
29
  3. T\u1EA1o GitHub repo m\u1EDBi '${n}' d\u01B0\u1EDBi account c\u1EE7a b\u1EA1n
28
- Ti\u1EBFp t\u1EE5c?`,default:!0}))throw new Error("User abort reset folder.");let r=t.visibility??(t.autoYes?"private":await ba({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),o=await xa(t.folderPath);return Aa(t.folderPath),{newRemoteUrl:kn({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 Ta(){let t=Kr("gh",["api","user","--jq",".login"],{encoding:"utf8",stdio:["ignore","pipe","pipe"]});return t.status!==0?null:t.stdout.trim()||null}function Pa(){s.info("M\u1EDF browser \u0111\u1EC3 switch GitHub account...");let t=Kr("gh",["auth","login","--web"],{stdio:"inherit"});t.status!==0&&s.warn(`gh auth login exit ${t.status}. B\u1EA1n c\xF3 th\u1EC3 ch\u1EA1y tay r\u1ED3i quay l\u1EA1i retry.`)}function Ea(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 $a(t){let e=t.trim();return e?/^https?:\/\/[\w.@/-]+$/.test(e)||/^git@[\w.-]+:[\w./-]+\.git$/.test(e)||/^[\w.-]+\/[\w.-]+$/.test(e):!1}async function Fr(t){let e=t.url,n=t.initialReason,r=t.initialDetail;for(;;){let o=Ta();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(Ea(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 Sa({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 Hr({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 Ca({message:"URL git remote (https://github.com/owner/repo.git ho\u1EB7c git@github.com:owner/repo.git):",default:e,validate:u=>$a(u)||"URL kh\xF4ng \u0111\xFAng format git remote"})).trim()),a==="switch"&&Pa(),s.info(`Verify remote l\u1EA1i: ${e}...`);let c=vt(e);if(c.ok)return s.success(`Remote accessible: ${e}`),{resolvedUrl:e};n=c.reason??"unknown",r=c.detail}}f();import{spawnSync as Ra}from"child_process";var _a={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 Vr(t){let e=_a[t];s.info(`\u0110ang c\xE0i gh CLI qua ${t}...`);let n=Ra(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 Ia}from"child_process";function Br(){if(Ia("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 Wr(t){for(;le()==="not-installed";){s.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=Nr();if(!e){if(await G({taskName:"Ph\xE1t hi\u1EC7n package manager",reason:"Kh\xF4ng t\xECm th\u1EA5y brew/apt/dnf/pacman/winget tr\xEAn m\xE1y.",allowSkip:!1,hint:"C\xE0i gh CLI tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry."})==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{Vr(e)}catch(n){if(await G({taskName:`C\xE0i gh CLI qua ${e}`,reason:n.message,allowSkip:!1,hint:"C\xE0i tay (https://cli.github.com) r\u1ED3i ch\u1ECDn Retry, ho\u1EB7c Abort."})==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;le()==="not-authenticated";){s.warn("Ch\u01B0a \u0111\u0103ng nh\u1EADp GitHub.");try{bt()}catch(e){if(await G({taskName:"\u0110\u0103ng nh\u1EADp GitHub qua gh",reason:e.message,allowSkip:!1,hint:"Th\u1EED l\u1EA1i \u2014 ch\u1ECDn c\xE1ch login kh\xE1c (browser vs token) khi gh prompt."})==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(le()!=="authenticated"&&await G({taskName:"Verify gh auth",reason:"Sau gh auth login v\u1EABn b\xE1o not-authenticated.",allowSkip:!1,hint:"C\xF3 th\u1EC3 user cancel browser. Th\u1EED l\u1EA1i ho\u1EB7c abort."})==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(s.success("gh CLI s\u1EB5n s\xE0ng"),Br(),t){let e=vt(t);return e.ok?(s.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await Fr({url:t,initialReason:e.reason??"unknown",initialDetail:e.detail})).resolvedUrl}}return{}}import{existsSync as At}from"fs";import{dirname as Na,join as Ct}from"path";var Oa=5;function Ga(t){if(At(Ct(t,".claude","avatar.json")))return!0;let e=At(Ct(t,".claude")),n=At(Ct(t,"CLAUDE.md")),r=At(Ct(t,"src"));if(!(e&&n&&r))return!1;let o=At(Ct(t,".claude","repos.json")),i=At(Ct(t,".claude","pack"));return o||i}function St(t){let e=t;for(let n=0;n<Oa;n++){if(Ga(e))return e;let r=Na(e);if(r===e)return null;e=r}return null}f();import{spawnSync as qr}from"child_process";import{appendFileSync as Ma,existsSync as zr,readFileSync as La,writeFileSync as ja}from"fs";import{join as vn}from"path";var Ua=120*1e3,Da=300*1e3,W=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 Jr(t,e,n,r){if(n==="SIGTERM")return new W(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 W(t,"permission",`gitnexus ${t} fail (permission). Check write access ~/.claude ho\u1EB7c cwd.`,e,r):new W(t,"non-zero-exit",`gitnexus ${t} exit ${e??"null"}. Xem log ph\xEDa tr\xEAn.`,e,r)}function ue(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 Da({message:"Visibility cho repo m\u1EDBi?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]})),o=await ja(t.folderPath);return Ha(t.folderPath),{newRemoteUrl:bn({folder:t.folderPath,name:n,visibility:r,org:t.org}).httpsUrl,backupPath:o}}f();var Dt=class extends Error{constructor(e){super(e),this.name="RemoteAccessAbortedError"}};function Va(){let t=to("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=to("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 eo(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 Ka({message:"C\xE1ch x\u1EED l\xFD?",choices:i});if(a==="abort")throw new Dt(`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 Zr({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 Fa({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 no(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 ro(){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 oo(t){for(;me()==="not-installed";){s.warn("gh CLI ch\u01B0a c\xE0i. Avatar s\u1EBD t\u1EF1 c\xE0i.");let e=Br();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 S("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.");continue}try{no(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 S("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i gh CLI.")}}for(;me()==="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 S("User abort t\u1EA1i b\u01B0\u1EDBc gh auth login.");continue}if(me()!=="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 S("User abort t\u1EA1i b\u01B0\u1EDBc verify gh auth.")}if(s.success("gh CLI s\u1EB5n s\xE0ng"),ro(),t){let e=xt(t);return e.ok?(s.success(`Remote accessible: ${t}`),{resolvedRemoteUrl:t}):{resolvedRemoteUrl:(await eo({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 io}from"child_process";import{appendFileSync as tc,existsSync as so,readFileSync as ec,writeFileSync as nc}from"fs";import{join as An}from"path";var rc=120*1e3,oc=300*1e3,q=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 ao(t,e,n,r){if(n==="SIGTERM")return new q(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 q(t,"permission",`gitnexus ${t} fail (permission). Check write access ~/.claude ho\u1EB7c cwd.`,e,r):new q(t,"non-zero-exit",`gitnexus ${t} exit ${e??"null"}. Xem log ph\xEDa tr\xEAn.`,e,r)}function de(t,e){return t.split(`
29
31
  `).slice(-e).join(`
30
- `)}function Yr(){let t=Mt("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=qr("gitnexus",["setup"],{stdio:["ignore","pipe","pipe"],timeout:Ua,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(`${ue(n,30)}
31
- `):r&&process.stderr.write(`${ue(r,30)}
32
- `),Jr("setup",e.status,e.signal,n)}t.succeed("GitNexus setup OK (global skills installed)")}function Tt(t){let e=Mt(`Analyze ${t} (1-3 ph\xFAt)`),n=qr("gitnexus",["analyze",".","--index-only"],{cwd:t,stdio:["ignore","pipe","pipe"],timeout:Da,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(`${ue(o,30)}
33
- `):i&&process.stderr.write(`${ue(i,30)}
34
- `),Jr("analyze",n.status,n.signal,o)}let r=vn(t,".gitnexus","meta.json");if(!zr(r))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new W("analyze","missing-output",`gitnexus analyze xong nh\u01B0ng kh\xF4ng th\u1EA5y ${r}. Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus fail silent.`);Ha(t),e.succeed(`Analyze OK (index t\u1EA1i ${vn(t,".gitnexus")}, \u0111\xE3 gitignore)`)}function Ha(t){let e=vn(t,".gitignore"),n=".gitnexus/";try{let r="";if(zr(e)&&(r=La(e,"utf8"),/^\.gitnexus\/?\s*$/m.test(r)))return;let i=`${r.length>0&&!r.endsWith(`
32
+ `)}function co(){let t=Lt("Setup GitNexus global skills (~/.claude/skills/gitnexus-*)"),e=io("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(`${de(n,30)}
33
+ `):r&&process.stderr.write(`${de(r,30)}
34
+ `),ao("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=io("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(`${de(o,30)}
35
+ `):i&&process.stderr.write(`${de(i,30)}
36
+ `),ao("analyze",n.status,n.signal,o)}let r=An(t,".gitnexus","meta.json");if(!so(r))throw e.fail("Analyze exit 0 nh\u01B0ng kh\xF4ng th\u1EA5y meta.json"),new q("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(so(e)&&(r=ec(e,"utf8"),/^\.gitnexus\/?\s*$/m.test(r)))return;let i=`${r.length>0&&!r.endsWith(`
35
37
  `)?`
36
38
  `:""}
37
39
  # GitNexus index (Avatar) \u2014 database local, kh\xF4ng commit.
38
40
  ${n}
39
- `;r.length>0?Ma(e,i):ja(e,`# GitNexus index (Avatar) \u2014 database local, kh\xF4ng commit.
41
+ `;r.length>0?tc(e,i):nc(e,`# GitNexus index (Avatar) \u2014 database local, kh\xF4ng commit.
40
42
  ${n}
41
- `)}catch{}}import{spawnSync as za}from"child_process";import{existsSync as Ja}from"fs";import{join as eo}from"path";import{confirm as Ya}from"@inquirer/prompts";var Ka=[/^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 Xr(t){return t?Ka.some(e=>e.test(t)):!1}k();f();import{promises as Ut}from"fs";import{homedir as Fa}from"os";import{dirname as Va,join as Qr}from"path";var Ba=Qr(Fa(),".gitnexus"),Dt=Qr(Ba,"config.json");async function Wa(){try{let t=await Ut.readFile(Dt,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function qa(t){await Ut.mkdir(Va(Dt),{recursive:!0});let e=`${Dt}.tmp-${process.pid}`;await Ut.writeFile(e,t,{mode:384}),await Ut.rename(e,Dt)}async function Zr(t){let n={...await Wa(),apiKey:t.apiKey,baseUrl:t.baseUrl,model:t.model,isReasoningModel:t.isReasoningModel??!1};await qa(JSON.stringify(n,null,2)),await Ut.chmod(Dt,384).catch(()=>{})}var Xa=900*1e3,Qa="nal-claude",Za="claude-sonnet-4-5";function tc(t){let e=t.replace(/\/+$/,"");return e.endsWith("/v1")?`${e}/`:e.endsWith("/v1/")?e:`${e}/v1/`}async function ec(t){let e=eo(t,".claude","settings.json");if(!await m(e))return null;try{let n=await b(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 y=process.env[h];if(typeof y=="string"&&y.length>0)return y;let C=r[h];return typeof C=="string"&&C.length>0?C: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:tc(o),model:c.length>0?c:Za}}else{let h=l("ANTHROPIC_AUTH_TOKEN");if(h)return{provider:"llmlite",apiKey:h,baseUrl:o,model:c.length>0?c:Qa}}return null}catch{return null}}async function nc(t,e){return await Ya({message:`Generate wiki cho workspace? (~$0.50 qua ${t} model=${e}, 2-5 ph\xFAt). Continue?`,default:!0})}function to(t,e){return t.split(`
43
+ `)}catch{}}import{spawnSync as mc}from"child_process";import{existsSync as dc}from"fs";import{join as go}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 lo(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 uo}from"path";var lc=uo(ac(),".gitnexus"),jt=uo(lc,"config.json");async function uc(){try{let t=await Ut.readFile(jt,"utf8"),e=JSON.parse(t);return typeof e=="object"&&e!==null?e:{}}catch{return{}}}async function pc(t){await Ut.mkdir(cc(jt),{recursive:!0});let e=`${jt}.tmp-${process.pid}`;await Ut.writeFile(e,t,{mode:384}),await Ut.rename(e,jt)}async function po(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(jt,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=go(t,".claude","settings.json");if(!await m(e))return null;try{let n=await b(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 y=process.env[h];if(typeof y=="string"&&y.length>0)return y;let x=r[h];return typeof x=="string"&&x.length>0?x: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 mo(t,e){return t.split(`
42
44
  `).slice(-e).join(`
43
- `)}async function pe(t,e=t){let n=await ec(t);if(!n)return s.warn(`Kh\xF4ng resolve \u0111\u01B0\u1EE3c API key cho wiki gen.
45
+ `)}async function ge(t,e=t){let n=await yc(t);if(!n)return s.warn(`Kh\xF4ng resolve \u0111\u01B0\u1EE3c API key cho wiki gen.
44
46
  \u0110\xE3 ki\u1EC3m tra: process.env \u2192 .envrc Avatar block \u2192 settings.json.env
45
47
  Nguy\xEAn nh\xE2n c\xF3 th\u1EC3:
46
48
  \u2022 Subscription mode (OAuth, kh\xF4ng c\xF3 key)
47
49
  \u2022 Key ch\u01B0a setup \u2014 ch\u1EA1y 'avatar ai setup' ho\u1EB7c 'avatar secrets set ANTHROPIC_API_KEY'
48
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:
49
- gitnexus wiki . --api-key <key> --base-url <url> --model <model>`),{ran:!1,skipped:!0,reason:"subscription-mode"};if(!await nc(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=Xr(n.model);await Zr({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=Mt(`Generating wiki via ${n.baseUrl} (${n.provider}) model=${n.model}${o?" [reasoning]":""}`),c=za("gitnexus",i,{cwd:e,stdio:["ignore","pipe","pipe"],timeout:Xa,encoding:"utf8"});if(c.status!==0||c.signal==="SIGTERM"){let u=c.signal==="SIGTERM"?"timeout":"non-zero-exit";a.fail(`Wiki gen ${u} (exit ${c.status??"null"})`);let d=(c.stderr||"").trim(),g=(c.stdout||"").trim();return d?process.stderr.write(`${to(d,30)}
50
- `):g&&process.stderr.write(`${to(g,30)}
51
- `),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${u} (exit ${c.status??"null"})`}}let l=eo(e,".gitnexus","wiki","index.html");return Ja(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 oo(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 rc(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 oc(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 ic()}catch(o){s.error(o instanceof Error?o.message:String(o)),process.exit(1)}})}function io(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 An(){let t=St(process.cwd());return t||(s.error(`Kh\xF4ng t\xECm th\u1EA5y Avatar workspace t\u1EEB th\u01B0 m\u1EE5c hi\u1EC7n t\u1EA1i.
52
- C\u1EA7n .claude/ + CLAUDE.md + src/. Ch\u1EA1y 'avatar init' tr\u01B0\u1EDBc.`),process.exit(1)),t}async function rc(t){let e=An();await Wr();let n=!0,r=t.url,o=t.name;for(;n;){let i=await sc(r),a=null,c=null;if(i.mode==="new"){let l=o??await ro(i.url),u=await cn({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 ro(i.url);s.info(`Clone ${i.url} \u2192 src/${l} ...`);let{cloned:u,skipped:d}=await Pr({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 cc(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 oc(t,e){let n=An(),{validateRepoName:r}=await Promise.resolve().then(()=>(Gt(),pn)),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(()=>(V(),it)),{removeRecursive:c}=await Promise.resolve().then(()=>(k(),yt)),{join:l,relative:u,sep:d}=await import("path"),h=(await i(n)).find(T=>T.name===t),y=l(n,"src",t),C=l(n,"src"),_=u(C,y);(_===""||_===".."||_.startsWith(`..${d}`)||_.startsWith("/"))&&(s.error(`\u0110\u01B0\u1EDDng d\u1EABn repo escape kh\u1ECFi workspace/src \u2014 t\u1EEB ch\u1ED1i x\xF3a: ${y}`),process.exit(1));let{pathExists:K}=await Promise.resolve().then(()=>(k(),yt)),P=await K(y);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(y),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 ic(){let t=An(),{readReposManifest:e}=await Promise.resolve().then(()=>(V(),it)),{pathExists:n}=await Promise.resolve().then(()=>(k(),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 sc(t){if(t)return{url:t,mode:"clone"};let e=await xn({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=no(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(no(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 ln(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 un(a);return l?{url:l,mode:"clone"}:{url:await ac(a),mode:"clone"}}let n=await Ht({message:"T\xEAn repo m\u1EDBi:",validate:io}),r=await xn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]});return{url:await $r(n.trim(),r),mode:"new"}}async function ac(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(()=>(Je(),pr)),{pathExists:o}=await Promise.resolve().then(()=>(k(),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:io}),c=await xn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]}),{createGithubRemoteFromFolder:l}=await Promise.resolve().then(()=>(yn(),Ur)),u=await Rr(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 ro(t){let e=sn(t);return await Ht({message:"T\xEAn th\u01B0 m\u1EE5c trong src/:",default:e})}async function cc(t,e,n){if(!ut().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{Tt(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 pe(t,e);i.ran?s.success(` \u2713 GitNexus wiki t\u1EA1o cho src/${e.split("/").pop()}`):i.reason==="subscription-mode"&&s.dim(" (Subscription mode \u2014 kh\xF4ng c\xF3 API key cho wiki, skip.)")}catch(i){s.warn(` ! Wiki fail (repo v\u1EABn d\xF9ng \u0111\u01B0\u1EE3c): ${i instanceof Error?i.message:i}`)}}}k();import{promises as sl}from"fs";import{join as Ro}from"path";import{confirm as al}from"@inquirer/prompts";import{promises as po}from"fs";import{homedir as hc}from"os";import{join as Wt}from"path";import{z as w}from"zod";var so=w.object({email:w.string().email(),name:w.string(),access_token:w.string().min(1),refresh_token:w.string().min(1),expires_at:w.string().datetime(),id_token:w.string().min(1)}),lc=w.object({installed_tools:w.record(w.string(),w.object({version:w.string().optional(),installed_at:w.string().datetime(),install_method:w.string()})).default({}),tool_inputs:w.record(w.string(),w.unknown()).default({})}),eg=w.object({$schema:w.string().optional(),includeCoAuthoredBy:w.boolean().optional(),env:w.record(w.string(),w.string()).default({}),permissions:w.object({allow:w.array(w.string()).default([]),deny:w.array(w.string()).default([])}).partial().optional(),hooks:w.record(w.string(),w.array(w.unknown())).optional(),statusLine:w.object({type:w.string(),command:w.string(),padding:w.number().optional()}).optional()}),ng=w.enum(["internal","client","library"]);k();var qt=Wt(hc(),".avatar"),pt=Wt(qt,"config.json"),cg=Wt(qt,"state.json"),_n=Wt(qt,"audit.log"),lg=Wt(qt,"backups"),wc=384;async function In(){await x(qt)}async function L(){if(!await m(pt))return null;let t=await b(pt),e=so.safeParse(t);return e.success?e.data:null}async function Nn(t){await In(),await S(pt,t,wc)}async function ge(){if(await m(pt)){let{promises:t}=await import("fs");await t.unlink(pt)}}function mt(t){let e=Date.parse(t.expires_at);return Number.isNaN(e)||e-Date.now()<6e4}var Bt=class extends Error{constructor(e){super(e),this.name="NoValidTokenError"}};async function kc(t){let{decodeIdToken:e}=await Promise.resolve().then(()=>(Vt(),Rn));try{let n=e(t),r=Math.floor(Date.now()/1e3);return n.exp-60<r}catch{return!0}}async function fe(){let t=await L();if(!t)throw new Bt("Ch\u01B0a \u0111\u0103ng nh\u1EADp. Ch\u1EA1y 'avatar login' tr\u01B0\u1EDBc.");let{refreshAccessToken:e,decodeIdToken:n,verifyIdTokenClaims:r}=await Promise.resolve().then(()=>(Vt(),Rn));if(!await kc(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 Bt(`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 Bt("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 Nn(i),o.id_token}async function yc(){try{await po.chmod(_n,384)}catch{}}async function v(t,e){await In();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
53
- `;await po.appendFile(_n,r,{encoding:"utf8",mode:384}),await yc()}f();import{spawnSync as On}from"child_process";var mo=6e4,bc="ok";function Gn(){let t=On("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 Mn(){s.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=On("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 vc(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 go(t){switch(t){case"auth-expired":return"Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n/b\u1ECB revoke. Ch\u1EA1y: `claude auth logout && claude auth login`.";case"credit_balance_too_low":case"insufficient_quota":return"H\u1EBFt quota subscription. Upgrade plan ho\u1EB7c d\xF9ng LLMLite (avatar ai setup \u2192 ch\u1ECDn LLMLite).";case"invalid_api_key":return"API key invalid. Re-login: `claude auth login`.";case"rate_limit":return"B\u1ECB rate limit t\u1EA1m th\u1EDDi. Ch\u1EDD v\xE0i ph\xFAt r\u1ED3i ch\u1EA1y `avatar ai setup`.";case"timeout":return"Timeout 60s: (1) m\u1EA1ng VN ch\u1EADm \u2014 th\u1EED VPN, (2) Anthropic API spike \u2014 retry v\xE0i ph\xFAt, (3) token v\u1EABn auth nh\u01B0ng quota h\u1EBFt \xE2m th\u1EA7m \u2014 check t\u1EA1i claude.ai/settings/usage.";default:return"L\u1ED7i ch\u01B0a bi\u1EBFt. Xem stderr \u1EDF tr\xEAn + ch\u1EA1y `claude --print ok` tay \u0111\u1EC3 debug."}}function Ln(){let t=On("claude",["--print",bc],{encoding:"utf8",timeout:mo,stdio:["ignore","pipe","pipe"]});if(t.signal==="SIGTERM"||t.status===143||t.error?.code==="ETIMEDOUT")return{ok:!1,reason:"timeout",detail:`claude --print > ${mo/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=vc(`${n}
54
- ${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 fo}from"child_process";var xc=5e3,Ac=/(\d+\.\d+\.\d+)/;function Cc(){let e=ct()==="win32"?"where":"which",n=fo(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 Sc(){let t=fo("claude",["--version"],{encoding:"utf8",timeout:xc});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return Ac.exec(e)?.[1]??null}var dt=null;function zt(){if(dt!==null)return dt;let t=Cc();return t?(dt={installed:!0,version:Sc(),path:t},dt):(dt={installed:!1,version:null,path:null},dt)}function he(){dt=null}import{spawnSync as Tc}from"child_process";f();var ho=300*1e3,wo="@anthropic-ai/claude-code",gt=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallClaudeCodeError",this.reason=e,this.exitCode=r}};function Pc(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new gt("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${wo} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new gt("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new gt("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function ko(){s.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Tc("npm",["install","-g",wo],{stdio:["inherit","inherit","pipe"],timeout:ho,encoding:"utf8"});if(t.signal==="SIGTERM")throw new gt("timeout",`npm install timeout sau ${ho/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),Pc(t.status,t.stderr||"");he();let e=zt();if(!e.installed||!e.path)throw new gt("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 Ec}from"fs";import{homedir as $c}from"os";import{join as Rc}from"path";import{select as yo}from"@inquirer/prompts";function _c(){return Rc($c(),".claude","settings.json")}function jn(){let t=_c(),e;try{e=Ec(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 bo(t=jn()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await yo({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 yo({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 Ic,select as Nc}from"@inquirer/prompts";var we="https://api.anthropic.com",Oc="2023-06-01",vo=1e4;function Gc(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function Mc(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 Lc(){return await Ic({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:Mc})}async function jc(t){let e=new AbortController,n=setTimeout(()=>e.abort(),vo);try{let r=await fetch(`${we}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":Oc,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 ${we} timeout sau ${vo/1e3}s.`):r}finally{clearTimeout(n)}}async function Uc(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 Nc({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function xo(){let t=await Lc();s.info(`Verify key (${Gc(t)}) qua ${we}/v1/models...`);let e=await jc(t);s.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await Uc(e);return{apiKey:t,baseUrl:we,model:n}}f();import{input as Dc,password as Hc,select as Kc}from"@inquirer/prompts";var Fc="https://ai.nal.vn",Ao=1e4;function q(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function Vc(){return await Hc({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function Bc(t=Fc){return(await Dc({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 Wc(t,e){let n=new AbortController,r=setTimeout(()=>n.abort(),Ao);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 ${Ao/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 qc(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 Kc({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function Co(){let t=await Vc(),e=await Bc();s.info(`Verify key (${q(t)}) qua ${e}/v1/models...`);let n=await Wc(e,t);s.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await qc(n);return{apiKey:t,baseUrl:e,model:r}}f();k();import{promises as zc}from"fs";import{join as Jc}from"path";var Un=384;function Yc(t){return Jc(t,".claude","settings.json")}async function Xc(t){if(!await m(t))return{};try{return await b(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 Qc(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 So(t){if(!t)return t;let{ANTHROPIC_AUTH_TOKEN:e,ANTHROPIC_API_KEY:n,...r}=t;return r}function Zc(t,e,n,r,o){let a={...So(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_AUTH_TOKEN=e),{...t,env:a,model:r,avatarProvider:"llmlite"}}function tl(t,e,n,r,o){let a={...So(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_API_KEY=e),{...t,env:a,model:r,avatarProvider:"anthropic"}}function el(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 ft(t,e){let n=Yc(t),r=await Xc(n),o;switch(e.provider){case"subscription":o=Qc(r,e.model);break;case"llmlite":o=Zc(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"anthropic":o=tl(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"use-global":o=el(r,e.sourceSettings);break}await S(n,o,Un);try{await zc.chmod(n,Un)}catch{}return{path:n,mode:Un}}var U="sonnet";async function ke(t){try{s.info("Setup AI provider cho workspace...");let e=zt();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."),ko(),he(),e=zt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=jn();switch(await bo(n)){case"subscription":{let o=Gn();if(o.state!=="authenticated"&&(Mn(),o=Gn()),o.state==="authenticated"&&o.subscriptionType)return await ft(t.workspacePath,{provider:"subscription",model:U}),await v("ai_setup",`provider=subscription,result=ok,plan=${o.subscriptionType},probe=skipped`),s.success(`AI ready \xB7 Subscription (${o.subscriptionType}) \xB7 model=${U}`),{ok:!0,provider:"subscription",model:U};s.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let i=Ln();if(!i.ok&&i.reason==="auth-expired"&&(s.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),Mn(),i=Ln()),!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 ft(t.workspacePath,{provider:"subscription",model:U}),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=${U}`),{ok:!0,provider:"subscription",model:U};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 ${go(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await ft(t.workspacePath,{provider:"subscription",model:U}),await v("ai_setup","provider=subscription,result=ok"),s.success(`AI ready \xB7 Subscription \xB7 model=${U}`),{ok:!0,provider:"subscription",model:U}}case"llmlite":{let o=await Co();return await ft(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}}case"anthropic":{let o=await xo();return await ft(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}}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 ft(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 nl}from"child_process";f();var Dn=1e4,To=3e4,Eo=5,Hn="say ok",Po="2023-06-01";async function rl(t,e,n){s.info(`Testing LLMLite provider: ${t} (key: ${q(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),Dn);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(", "),y=c.length>5?` ...+${c.length-5} more`:"";s.dim(` Models: ${h}${y}`)}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:Hn}],max_tokens:Eo}),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 ${Dn/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):i}finally{clearTimeout(o)}}function ol(){s.info("Testing Subscription provider qua `claude --print`...");let t=nl("claude",["--print",Hn],{encoding:"utf8",timeout:To});if(t.signal==="SIGTERM")throw new Error(`Timeout ${To/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 il(t,e,n){s.info(`Testing Anthropic Direct provider: ${t} (key: ${q(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),Dn);try{let i=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":Po},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":Po,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:Eo,messages:[{role:"user",content:Hn}]}),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 $o(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 il(n,o,i),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&r?(await rl(n,r,i),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(ol(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function ye(){let t=process.cwd(),e=St(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 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=lo(n.model);await po({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(`${mo(d,30)}
52
+ `):g&&process.stderr.write(`${mo(g,30)}
53
+ `),{ran:!1,skipped:!0,reason:"fail",detail:`Wiki gen ${u} (exit ${c.status??"null"})`}}let l=go(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 wo(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 ko(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 oo();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 ho(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 ho(i.url);s.info(`Clone ${i.url} \u2192 src/${l} ...`);let{cloned:u,skipped:d}=await Ur({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 Ft({message:"Add repo kh\xE1c n\u1EEFa?",default:!1})}}async function xc(t,e){let n=Sn(),{validateRepoName:r}=await Promise.resolve().then(()=>(Mt(),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(()=>(B(),it)),{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),y=l(n,"src",t),x=l(n,"src"),I=u(x,y);(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: ${y}`),process.exit(1));let{pathExists:K}=await Promise.resolve().then(()=>(w(),yt)),E=await K(y);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 Ft({message:`X\xF3a lu\xF4n folder src/${t}?`,default:!1})?(await c(y),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(()=>(B(),it)),{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=fo(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(fo(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 Ft({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:ko}),r=await Cn({message:"Visibility?",choices:[{name:"private (m\u1EB7c \u0111\u1ECBnh)",value:"private"},{name:"public",value:"public"}]});return{url:await Hr(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 Ft({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(),wr)),{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:ko}),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(),Xr)),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 ho(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(!ut().installed){s.dim("GitNexus ch\u01B0a c\xE0i \u2014 skip index. C\xE0i qua 'avatar gitnexus install'.");return}if(n||await Ft({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 ge(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}`)}}}import{promises as Sl}from"fs";import{join as qo}from"path";import{confirm as Tl}from"@inquirer/prompts";w();import{promises as Nc}from"fs";w();import{promises as Ec}from"fs";import{join as Pc}from"path";var Tn=384;function En(t){return Pc(t,".claude","settings.json")}async function Rc(t){if(!await m(t))return{};try{return await b(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 $c(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 yo(t){if(!t)return t;let{ANTHROPIC_AUTH_TOKEN:e,ANTHROPIC_API_KEY:n,...r}=t;return r}function _c(t,e,n,r,o){let a={...yo(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_AUTH_TOKEN=e),{...t,env:a,model:r,avatarProvider:"llmlite"}}function Ic(t,e,n,r,o){let a={...yo(t.env),ANTHROPIC_BASE_URL:n};return o||(a.ANTHROPIC_API_KEY=e),{...t,env:a,model:r,avatarProvider:"anthropic"}}function Oc(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 pt(t,e){let n=En(t),r=await Rc(n),o;switch(e.provider){case"subscription":o=$c(r,e.model);break;case"llmlite":o=_c(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"anthropic":o=Ic(r,e.apiKey,e.baseUrl,e.model,e.skipApiKey===!0);break;case"use-global":o=Oc(r,e.sourceSettings);break}await C(n,o,Tn);try{await Ec.chmod(n,Tn)}catch{}return{path:n,mode:Tn}}var bo=384,vo="claude-opus-4-8",xo="claude-sonnet-4-6",Ao="claude-haiku-4-5";function Pn(t,e){return t.some(n=>n.toLowerCase().includes(e))}function Rn(t,e){return t.find(n=>n.toLowerCase().includes(e))}function Mc(t){let e=Pn(t,"opus"),n=Pn(t,"sonnet"),r=Pn(t,"haiku"),o=Rn(t,"opus")??vo,i=Rn(t,"sonnet")??xo,a=Rn(t,"haiku")??Ao;if(e&&n)return{model:"opusplan",defaultOpus:o,defaultSonnet:i,defaultHaiku:r?a:i,opusplan:!0,reason:r?"\u0110\u1EE7 Opus + Sonnet + Haiku \u2192 b\u1EADt opusplan (Opus plan / Sonnet execute).":"\u0110\u1EE7 Opus + Sonnet (thi\u1EBFu Haiku) \u2192 b\u1EADt opusplan; background task fallback Haiku\u2192Sonnet."};if(!e&&n)return{model:i,opusplan:!1,reason:`\u26A0 Gateway/provider thi\u1EBFu Opus \u2192 d\xF9ng ${i} \u0111\u01A1n. opusplan KH\xD4NG kh\u1EA3 d\u1EE5ng (c\u1EA7n c\u1EA3 Opus + Sonnet).`};if(e&&!n)return{model:o,opusplan:!1,reason:`\u26A0 Gateway/provider thi\u1EBFu Sonnet \u2192 d\xF9ng ${o} \u0111\u01A1n (\u0111\u1EAFt h\u01A1n). opusplan KH\xD4NG kh\u1EA3 d\u1EE5ng.`};let c=t[0]??a;return{model:c,opusplan:!1,reason:`\u26A0 Kh\xF4ng c\xF3 Opus l\u1EABn Sonnet \u2192 d\xF9ng ${c}. opusplan KH\xD4NG kh\u1EA3 d\u1EE5ng.`}}async function Lc(t){return await m(t)?await b(t):{}}async function Co(t,e){let n=En(t),r=await Lc(n);if(e.mode==="savecost"&&r.model==="opusplan")return{decision:{model:"opusplan",opusplan:!0,reason:"Workspace \u0111\xE3 b\u1EADt opusplan t\u1EEB tr\u01B0\u1EDBc \u2014 kh\xF4ng c\u1EA7n l\xE0m g\xEC."},changed:!1};let o=e.availableModels??[vo,xo,Ao],i=Mc(o);if(r.model=i.model,i.opusplan){let a=r.env??{};i.defaultOpus&&(a.ANTHROPIC_DEFAULT_OPUS_MODEL=i.defaultOpus),i.defaultSonnet&&(a.ANTHROPIC_DEFAULT_SONNET_MODEL=i.defaultSonnet),i.defaultHaiku&&(a.ANTHROPIC_DEFAULT_HAIKU_MODEL=i.defaultHaiku),r.env=a}await C(n,r,bo);try{await Nc.chmod(n,bo)}catch{}return{decision:i,changed:!0}}w();import{promises as $o}from"fs";import{homedir as Vc}from"os";import{join as Wt}from"path";import{z as k}from"zod";var So=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)}),Gc=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({})}),Og=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()}),Ng=k.enum(["internal","client","library"]);w();var qt=Wt(Vc(),".avatar"),mt=Wt(qt,"config.json"),jg=Wt(qt,"state.json"),Gn=Wt(qt,"audit.log"),Hg=Wt(qt,"backups"),Bc=384;async function Dn(){await A(qt)}async function D(){if(!await m(mt))return null;let t=await b(mt),e=So.safeParse(t);return e.success?e.data:null}async function Un(t){await Dn(),await C(mt,t,Bc)}async function we(){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 Bt=class extends Error{constructor(e){super(e),this.name="NoValidTokenError"}};async function Wc(t){let{decodeIdToken:e}=await Promise.resolve().then(()=>(Vt(),Ln));try{let n=e(t),r=Math.floor(Date.now()/1e3);return n.exp-60<r}catch{return!0}}async function ke(){let t=await D();if(!t)throw new Bt("Ch\u01B0a \u0111\u0103ng nh\u1EADp. Ch\u1EA1y 'avatar login' tr\u01B0\u1EDBc.");let{refreshAccessToken:e,decodeIdToken:n,verifyIdTokenClaims:r}=await Promise.resolve().then(()=>(Vt(),Ln));if(!await Wc(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 Bt(`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 Bt("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 Un(i),o.id_token}async function qc(){try{await $o.chmod(Gn,384)}catch{}}async function v(t,e){await Dn();let n={timestamp:new Date().toISOString(),action:t,...e?{detail:e}:{}},r=`${JSON.stringify(n)}
55
+ `;await $o.appendFile(Gn,r,{encoding:"utf8",mode:384}),await qc()}f();import{spawnSync as jn}from"child_process";var _o=6e4,zc="ok";function Hn(){let t=jn("claude",["auth","status"],{encoding:"utf8"});if(t.error||t.status!==0)return{state:"not-authenticated"};let e=(t.stdout||"").trim();if(!e.startsWith("{"))return{state:"authenticated"};try{let n=JSON.parse(e);return n.loggedIn!==!0?{state:"not-authenticated"}:{state:"authenticated",email:n.email,subscriptionType:n.subscriptionType,apiProvider:n.apiProvider}}catch{return{state:"authenticated"}}}function Fn(){s.info("Kh\u1EDFi \u0111\u1ED9ng \u0111\u0103ng nh\u1EADp Claude Code (browser s\u1EBD m\u1EDF)...");let t=jn("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 Jc(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 Io(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 Kn(){let t=jn("claude",["--print",zc],{encoding:"utf8",timeout:_o,stdio:["ignore","pipe","pipe"]});if(t.signal==="SIGTERM"||t.status===143||t.error?.code==="ETIMEDOUT")return{ok:!1,reason:"timeout",detail:`claude --print > ${_o/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=Jc(`${n}
56
+ ${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 Oo}from"child_process";var Yc=5e3,Xc=/(\d+\.\d+\.\d+)/;function Qc(){let e=ct()==="win32"?"where":"which",n=Oo(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 Zc(){let t=Oo("claude",["--version"],{encoding:"utf8",timeout:Yc});if(t.error||t.status!==0)return null;let e=(t.stdout||"").trim();return Xc.exec(e)?.[1]??null}var gt=null;function zt(){if(gt!==null)return gt;let t=Qc();return t?(gt={installed:!0,version:Zc(),path:t},gt):(gt={installed:!1,version:null,path:null},gt)}function ye(){gt=null}import{spawnSync as tl}from"child_process";f();var No=300*1e3,Mo="@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 el(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 ${Mo} 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 Lo(){s.info("\u0110ang c\xE0i Claude Code qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=tl("npm",["install","-g",Mo],{stdio:["inherit","inherit","pipe"],timeout:No,encoding:"utf8"});if(t.signal==="SIGTERM")throw new ft("timeout",`npm install timeout sau ${No/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),el(t.status,t.stderr||"");ye();let e=zt();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 nl}from"fs";import{homedir as rl}from"os";import{join as ol}from"path";import{select as Go}from"@inquirer/prompts";function il(){return ol(rl(),".claude","settings.json")}function Vn(){let t=il(),e;try{e=nl(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 Do(t=Vn()){return t.exists&&t.hasBaseUrl&&t.hasToken&&await Go({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 Go({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 al}from"@inquirer/prompts";var be="https://api.anthropic.com",cl="2023-06-01",Uo=1e4;function ll(t){return t.length<=12?"sk-ant-***":`${t.slice(0,7)}...${t.slice(-4)}`}function ul(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 pl(){return await sl({message:"Anthropic API key (sk-ant-..., \u1EA9n input):",mask:"*",validate:ul})}async function ml(t){let e=new AbortController,n=setTimeout(()=>e.abort(),Uo);try{let r=await fetch(`${be}/v1/models`,{method:"GET",headers:{"x-api-key":t,"anthropic-version":cl,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 ${be} timeout sau ${Uo/1e3}s.`):r}finally{clearTimeout(n)}}async function dl(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 al({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:e.map(n=>({name:n,value:n}))})}async function jo(){let t=await pl();s.info(`Verify key (${ll(t)}) qua ${be}/v1/models...`);let e=await ml(t);s.success(`Endpoint OK \u2014 ${e.length} models available`);let n=await dl(e);return{apiKey:t,baseUrl:be,model:n}}f();import{input as gl,password as fl,select as hl}from"@inquirer/prompts";var wl="https://ai.nal.vn",Ho=1e4;function z(t){return t.length<=8?"sk-***":`${t.slice(0,3)}...${t.slice(-4)}`}async function kl(){return await fl({message:"LLMLite API key (\u1EA9n input):",mask:"*",validate:t=>t.trim().length>0?!0:"API key b\u1EAFt bu\u1ED9c"})}async function yl(t=wl){return(await gl({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 Bn(t,e){let n=new AbortController,r=setTimeout(()=>n.abort(),Ho);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`.":`
57
+ Hint: check m\u1EA1ng / firewall / VPN, ho\u1EB7c base URL c\xF3 \u0111\xFAng kh\xF4ng.`;throw new Error(`Connect ${t} timeout sau ${Ho/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 bl(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 hl({message:"Ch\u1ECDn model m\u1EB7c \u0111\u1ECBnh cho project:",choices:n.map(r=>({name:r,value:r}))})}async function Fo(){let t=await kl(),e=await yl();s.info(`Verify key (${z(t)}) qua ${e}/v1/models...`);let n=await Bn(e,t);s.success(`Endpoint OK \u2014 ${n.length} models available`);let r=await bl(n);return{apiKey:t,baseUrl:e,model:r}}f();var j="sonnet";async function ve(t){try{s.info("Setup AI provider cho workspace...");let e=zt();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."),Lo(),ye(),e=zt(),!e.installed)throw new Error("C\xE0i Claude Code xong nh\u01B0ng v\u1EABn kh\xF4ng detect \u0111\u01B0\u1EE3c binary.");let n=Vn();switch(await Do(n)){case"subscription":{let o=Hn();if(o.state!=="authenticated"&&(Fn(),o=Hn()),o.state==="authenticated"&&o.subscriptionType)return await pt(t.workspacePath,{provider:"subscription",model:j}),await v("ai_setup",`provider=subscription,result=ok,plan=${o.subscriptionType},probe=skipped`),s.success(`AI ready \xB7 Subscription (${o.subscriptionType}) \xB7 model=${j}`),{ok:!0,provider:"subscription",model:j};s.dim("Auth status kh\xF4ng tr\u1EA3 subscriptionType \u2014 verify quota (30-60s)...");let i=Kn();if(!i.ok&&i.reason==="auth-expired"&&(s.warn("Token Claude Code \u0111\xE3 h\u1EBFt h\u1EA1n. T\u1EF1 \u0111\u1ED9ng re-login..."),Fn(),i=Kn()),!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 pt(t.workspacePath,{provider:"subscription",model:j}),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=${j}`),{ok:!0,provider:"subscription",model:j};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 ${Io(a)}`),{ok:!1,reason:`subscription-${a}`,phase:"quota"}}return await pt(t.workspacePath,{provider:"subscription",model:j}),await v("ai_setup","provider=subscription,result=ok"),s.success(`AI ready \xB7 Subscription \xB7 model=${j}`),{ok:!0,provider:"subscription",model:j}}case"llmlite":{let o=await Fo();return await pt(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}}case"anthropic":{let o=await jo();return await pt(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}}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 pt(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 vl}from"child_process";f();var Wn=1e4,Ko=3e4,Bo=5,qn="say ok",Vo="2023-06-01";async function xl(t,e,n){s.info(`Testing LLMLite provider: ${t} (key: ${z(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),Wn);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(", "),y=c.length>5?` ...+${c.length-5} more`:"";s.dim(` Models: ${h}${y}`)}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:qn}],max_tokens:Bo}),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 ${Wn/1e3}s. Check m\u1EA1ng / endpoint ${t}.`):i}finally{clearTimeout(o)}}function Al(){s.info("Testing Subscription provider qua `claude --print`...");let t=vl("claude",["--print",qn],{encoding:"utf8",timeout:Ko});if(t.signal==="SIGTERM")throw new Error(`Timeout ${Ko/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 Cl(t,e,n){s.info(`Testing Anthropic Direct provider: ${t} (key: ${z(e)})`);let r=new AbortController,o=setTimeout(()=>r.abort(),Wn);try{let i=await fetch(`${t}/v1/models`,{headers:{"x-api-key":e,"anthropic-version":Vo},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":Vo,"Content-Type":"application/json"},body:JSON.stringify({model:n,max_tokens:Bo,messages:[{role:"user",content:qn}]}),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 Wo(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 Cl(n,o,i),{ok:!0,provider:"anthropic",message:"Anthropic Direct provider working"}):n&&r?(await xl(n,r,i),{ok:!0,provider:"llmlite",message:"LLMLite provider working"}):(Al(),{ok:!0,provider:"subscription",message:"Subscription provider working"})}async function El(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;if(!(!n||!r))try{return await Bn(n,r)}catch(o){s.warn(`Kh\xF4ng query \u0111\u01B0\u1EE3c model list t\u1EEB gateway: ${o.message}`),s.dim(" \u2192 D\xF9ng known-set 3 model chu\u1EA9n (Opus/Sonnet/Haiku). Verify b\u1EB1ng 'avatar ai status'.");return}}async function zo(t,e,n){let r=await El(e),{decision:o,changed:i}=await Co(t,{mode:n,availableModels:r});if(!i){s.info(o.reason);return}o.opusplan?(s.success(`\u2713 B\u1EADt opusplan \xB7 model=opusplan \xB7 ${o.reason}`),s.dim(` Pin: opus=${o.defaultOpus} \xB7 sonnet=${o.defaultSonnet} \xB7 haiku=${o.defaultHaiku}`)):(s.warn(`Kh\xF4ng b\u1EADt opusplan \xB7 model=${o.model}`),s.warn(` ${o.reason}`))}async function Jt(){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.
56
58
  Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
57
59
  B\u1EA1n \u0111ang \u1EDF: ${t}
58
- 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 Kn(t){let e=Ro(t,".claude","settings.json");if(!await m(e))return{};try{return await b(e)}catch{return{}}}async function cl(){let t=await ye();await ke({workspacePath:t})}async function ll(){let t=await ye(),e=await Kn(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=q(i)):r&&o?(c="LLMLite",l=q(o)):o?(c="Custom",l=q(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 ul(){let t=await ye(),e=await Kn(t);try{let n=await $o(e);s.success(`\u2713 ${n.message}`)}catch(n){s.error(`Test fail: ${n.message}`),process.exit(1)}}async function pl(t){let e=await ye(),n=Ro(e,".claude","settings.json"),r=await Kn(e);if(!t.yes&&!await al({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 sl.unlink(n).catch(()=>{}),s.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await S(n,a,384),s.success("\u0110\xE3 reset env block trong .claude/settings.json"))}function _o(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 cl()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await ll()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await ul()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await pl(n)})}import{spawnSync as Bn}from"child_process";import{promises as Vo}from"fs";import{join as Xt}from"path";import Tl from"boxen";import{join as Cl}from"path";k();import{promises as wl}from"fs";import{join as Mo}from"path";k();f();import{promises as ml}from"fs";import Io,{join as Fn}from"path";async function dl(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=Fn(t,o),a=Io.resolve(t),c=Io.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 Vn(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 Jt(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 gl(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function fl(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 No(t){let e=t.map((i,a)=>{let c=fl(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(gl).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 hl(t,e){let n=[],r={...e},o=0;for(let i of Object.keys(r)){let a=r[i]||[],{entries:c,droppedCount:l}=No(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=Jt(c,a),{entries:u,droppedCount:d}=No(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 be(t){let e=Fn(t,".claude","pack","templates","settings.json.tpl"),n=Fn(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 b(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 dl(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,y]of Object.entries(u))typeof y=="string"&&y.includes("llm.nal.vn")&&(u[h]=y.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,y]of Object.entries(r.env))h in u||(u[h]=y,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||[],y=Jt(u,g),C=Jt(d,h);(y.length!==u.length||C.length!==d.length)&&(c.permissions={allow:y,deny:C},a.push(`permissions union (+${y.length-u.length} allow, +${C.length-d.length} deny)`))}if(r.hooks){let u=o.hooks||{},{merged:d,touchedEvents:g,migratedCount:h}=hl(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=Vn(n),await ml.copyFile(n,l)),await S(n,c),{action:"merged",backupPath:l,changes:a}}function kl(t,e){return Mo(t,".claude","pack","tools",e,"tool.json")}async function Yt(t,e){let n=kl(t,e);if(!await m(n))return null;try{return await b(n)}catch{return null}}var yl=[".claude","settings.json"];function Lo(t){return Mo(t,...yl)}function ve(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function jo(t){let e=Lo(t);if(!await m(e))return{settings:{},existed:!1};try{return{settings:await b(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 Uo(t,e,n){let r=Lo(t),o;return n&&(o=Vn(r),await wl.copyFile(r,o)),await S(r,e),o}function Oo(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function Go(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function bl(t,e){let n=new Set;for(let i of e)Go(i)&&n.add(Oo(i));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(i=>Go(i)?!0:n.has(Oo(i))?(r++,!1):!0),droppedCount:r}}async function xe(t,e){let{settings:n,existed:r}=await jo(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[y,C]of Object.entries(a)){let _=u[y]??[],{migratedUser:K,droppedCount:P}=bl(_,C);P>0&&(h+=P);let T=new Set(K.map(ve)),$=C.filter(et=>!T.has(ve(et)));($.length>0||P>0)&&(d[y]=[...K,...$],g.push(y))}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=Jt(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 Uo(t,o,r),changes:i}}async function Do(t,e){let{settings:n,existed:r}=await jo(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 $=u[P];if(!$){g[P]=T;continue}let et=new Set($.map(ve)),kt=T.filter(nt=>!et.has(ve(nt)));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 y=new Set(e.settings.permissions?.deny??[]),C=n.permissions?.deny??[],_=y.size>0?C.filter(P=>!y.has(P)):C;if(n.permissions){let{deny:P,...T}=n.permissions,$={...T,..._.length>0?{deny:_}:{}};Object.keys($).length>0&&(c.permissions=$)}return _.length!==C.length&&l.push(`deny -${C.length-_.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await Uo(t,c,r),changes:l}}k();k();import{join as vl}from"path";var xl=".claude/avatar-tools.json";function Ho(){return{tools:{}}}function Ko(t){return vl(t,xl)}async function Pt(t){let e=Ko(t);if(!await m(e))return Ho();try{return{tools:(await b(e)).tools??{}}}catch{return Ho()}}async function Al(t,e){await S(Ko(t),e)}async function Ae(t,e,n){let r=await Pt(t);return r.tools[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await Al(t,r),r}async function Et(t){let e=await Pt(t);return Object.entries(e.tools).filter(([,n])=>n.enabled).map(([n])=>n)}function Sl(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 Fo(t){let e=await Et(t);if(e.length===0)return[];let n=Cl(t,".claude","settings.json"),r={};if(await m(n))try{r=await b(n)}catch{r={}}let o=[];for(let i of e){let a=await Yt(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}Sl(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 xe(t,a)}})}return o}k();f();function Bo(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 Pl(process.cwd());El(n),e.fix&&await $l(n)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Pl(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 L();a?mt(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"),$=!1;await m(T)&&($=(await Vo.readFile(T,"utf8")).includes("settings.json")),e.push({name:"\u{1F512} settings.json gitignored (.claude/.gitignore)",status:$?"ok":"fail",detail:$?"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:!$,fix:$?void 0:async()=>{let{writeClaudeGitignore:et}=await Promise.resolve().then(()=>(Ot(),xr));await et(t)}})}let g=Bn("which",["python"]),h=Bn("which",["python3"]),y=g.status===0,C=h.status===0;C&&!y?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}):y?e.push({name:"Python binary",status:"ok",detail:`python: ${g.stdout.toString().trim()}`,fixable:!1}):C&&e.push({name:"Python binary",status:"ok",detail:`python3: ${h.stdout.toString().trim()}`,fixable:!1});let _=Xt(t,".claude","settings.json");if(await m(_))try{let T=await Vo.readFile(_,"utf8"),$=JSON.parse(T);if($.statusLine?.command){let kt=$.statusLine.command.trim().match(/^(node|python|python3|bash|sh)\s+([^\s]+)/);if(kt?.[2]){let nt=kt[2],ss=nt.startsWith("/")?nt:Xt(t,nt),ar=await m(ss);e.push({name:"statusLine command",status:ar?"ok":"fail",detail:ar?`ref OK: ${nt}`:`BROKEN: settings.json ref '${nt}' nh\u01B0ng file kh\xF4ng t\u1ED3n t\u1EA1i. Strip field statusLine ho\u1EB7c fix path.`,fixable:!1})}}}catch{}let K=Bn("which",["claude"]),P=K.status===0;if(e.push({name:"Claude Code CLI",status:P?"ok":"warn",detail:P?K.stdout.toString().trim():"kh\xF4ng t\xECm th\u1EA5y 'claude' tr\xEAn PATH",fixable:!1}),u){let T=await Fo(t);e.push(...T)}return e}function El(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(`${Tl(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 Yt(t){let e=qo(t,".claude","settings.json");if(!await m(e))return{};try{return await b(e)}catch{return{}}}async function Pl(){let t=await Jt();if((await ve({workspacePath:t})).ok){let n=await Yt(t);await zo(t,n,"force")}}async function Rl(){let t=await Jt(),e=await Yt(t);if(Object.keys(e).length===0){s.warn("Ch\u01B0a c\xF3 .claude/settings.json \u2014 ch\u1EA1y 'avatar ai setup' tr\u01B0\u1EDBc.");return}await zo(t,e,"savecost")}async function $l(){let t=await Jt(),e=await Yt(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=z(i)):r&&o?(c="LLMLite",l=z(o)):o?(c="Custom",l=z(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 _l(){let t=await Jt(),e=await Yt(t);try{let n=await Wo(e);s.success(`\u2713 ${n.message}`)}catch(n){s.error(`Test fail: ${n.message}`),process.exit(1)}}async function Il(t){let e=await Jt(),n=qo(e,".claude","settings.json"),r=await Yt(e);if(!t.yes&&!await Tl({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,model:i,...a}=r,c={...a};if(i!==void 0&&i!=="opusplan"&&(c.model=i),o){let{ANTHROPIC_BASE_URL:l,ANTHROPIC_AUTH_TOKEN:u,ANTHROPIC_API_KEY:d,ANTHROPIC_DEFAULT_OPUS_MODEL:g,ANTHROPIC_DEFAULT_SONNET_MODEL:h,ANTHROPIC_DEFAULT_HAIKU_MODEL:y,...x}=o;Object.keys(x).length>0&&(c.env=x)}Object.keys(c).length===0?(await Sl.unlink(n).catch(()=>{}),s.success("\u0110\xE3 x\xF3a .claude/settings.json (clean state)")):(await C(n,c,384),s.success("\u0110\xE3 reset env block + g\u1EE1 opusplan trong .claude/settings.json"))}function Jo(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 Pl()}),e.command("status").description("Show AI config hi\u1EC7n t\u1EA1i (mask token)").action(async()=>{await $l()}),e.command("test").description("Verify AI provider qua cheap prompt").action(async()=>{await _l()}),e.command("reset").description("X\xF3a env.ANTHROPIC_* + g\u1EE1 opusplan kh\u1ECFi settings.json (v\u1EC1 Subscription default)").option("--yes","Skip confirm").action(async n=>{await Il(n)}),e.command("savecost").description("B\u1EADt opusplan (Opus plan / Sonnet execute) n\u1EBFu ch\u01B0a c\xF3 \u2014 ti\u1EBFt ki\u1EC7m, cho workspace c\u0169").action(async()=>{await Rl()})}import{spawnSync as Yn}from"child_process";import{promises as ci}from"fs";import{join as Zt}from"path";import ql from"boxen";import{join as Bl}from"path";w();import{promises as Dl}from"fs";import{join as ti}from"path";w();f();import{promises as Ol}from"fs";import Yo,{join as zn}from"path";async function Nl(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=zn(t,o),a=Yo.resolve(t),c=Yo.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 Jn(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 Xt(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 Ml(t){return t.replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()}function Ll(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 Xo(t){let e=t.map((i,a)=>{let c=Ll(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(Ml).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 Gl(t,e){let n=[],r={...e},o=0;for(let i of Object.keys(r)){let a=r[i]||[],{entries:c,droppedCount:l}=Xo(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=Xt(c,a),{entries:u,droppedCount:d}=Xo(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 xe(t){let e=zn(t,".claude","pack","templates","settings.json.tpl"),n=zn(t,".claude","settings.json");if(!await m(e))return{action:"no-pack-template",changes:[]};let r;try{let u=await M(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 b(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 Nl(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,y]of Object.entries(u))typeof y=="string"&&y.includes("llm.nal.vn")&&(u[h]=y.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,y]of Object.entries(r.env))h in u||(u[h]=y,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||[],y=Xt(u,g),x=Xt(d,h);(y.length!==u.length||x.length!==d.length)&&(c.permissions={allow:y,deny:x},a.push(`permissions union (+${y.length-u.length} allow, +${x.length-d.length} deny)`))}if(r.hooks){let u=o.hooks||{},{merged:d,touchedEvents:g,migratedCount:h}=Gl(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=Jn(n),await Ol.copyFile(n,l)),await C(n,c),{action:"merged",backupPath:l,changes:a}}function Ul(t,e){return ti(t,".claude","pack","tools",e,"tool.json")}async function Qt(t,e){let n=Ul(t,e);if(!await m(n))return null;try{return await b(n)}catch{return null}}var jl=[".claude","settings.json"];function ei(t){return ti(t,...jl)}function Ae(t){let e=(t.hooks??[]).map(n=>n.command??"").sort().join("\0");return`${t.matcher??""}${e}`}async function ni(t){let e=ei(t);if(!await m(e))return{settings:{},existed:!1};try{return{settings:await b(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 ri(t,e,n){let r=ei(t),o;return n&&(o=Jn(r),await Dl.copyFile(r,o)),await C(r,e),o}function Qo(t){let e=(t.hooks??[]).map(n=>(n.command??"").replace(/\$\{CLAUDE_PROJECT_DIR\}\//g,"").trim()).sort().join("|");return`${t.matcher??""}::${e}`}function Zo(t){return(t.hooks??[]).some(e=>(e.command??"").includes("${CLAUDE_PROJECT_DIR}"))}function Hl(t,e){let n=new Set;for(let i of e)Zo(i)&&n.add(Qo(i));if(n.size===0)return{migratedUser:t,droppedCount:0};let r=0;return{migratedUser:t.filter(i=>Zo(i)?!0:n.has(Qo(i))?(r++,!1):!0),droppedCount:r}}async function Ce(t,e){let{settings:n,existed:r}=await ni(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[y,x]of Object.entries(a)){let I=u[y]??[],{migratedUser:K,droppedCount:E}=Hl(I,x);E>0&&(h+=E);let T=new Set(K.map(Ae)),R=x.filter(nt=>!T.has(Ae(nt)));(R.length>0||E>0)&&(d[y]=[...K,...R],g.push(y))}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=Xt(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 ri(t,o,r),changes:i}}async function oi(t,e){let{settings:n,existed:r}=await ni(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 nt=new Set(R.map(Ae)),kt=T.filter(rt=>!nt.has(Ae(rt)));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 y=new Set(e.settings.permissions?.deny??[]),x=n.permissions?.deny??[],I=y.size>0?x.filter(E=>!y.has(E)):x;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!==x.length&&l.push(`deny -${x.length-I.length}`),l.length===0?{action:"no-change",changes:[]}:{action:"disabled",backupPath:await ri(t,c,r),changes:l}}w();w();import{join as Fl}from"path";var Kl=".claude/avatar-tools.json";function ii(){return{tools:{}}}function si(t){return Fl(t,Kl)}async function Pt(t){let e=si(t);if(!await m(e))return ii();try{return{tools:(await b(e)).tools??{}}}catch{return ii()}}async function Vl(t,e){await C(si(t),e)}async function Se(t,e,n){let r=await Pt(t);return r.tools[e]={enabled:n.enabled,version:n.version,appliedAt:new Date().toISOString()},await Vl(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 Wl(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 ai(t){let e=await Rt(t);if(e.length===0)return[];let n=Bl(t,".claude","settings.json"),r={};if(await m(n))try{r=await b(n)}catch{r={}}let o=[];for(let i of e){let a=await Qt(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}Wl(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 Ce(t,a)}})}return o}w();f();function li(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 zl(process.cwd());Jl(n),e.fix&&await Yl(n)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function zl(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 D();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=Zt(t,".claude","pack"),l=Zt(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=Zt(t,".claude",".gitignore"),R=!1;await m(T)&&(R=(await ci.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:nt}=await Promise.resolve().then(()=>(bt(),Nr));await nt(t)}})}let g=Yn("which",["python"]),h=Yn("which",["python3"]),y=g.status===0,x=h.status===0;x&&!y?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}):y?e.push({name:"Python binary",status:"ok",detail:`python: ${g.stdout.toString().trim()}`,fixable:!1}):x&&e.push({name:"Python binary",status:"ok",detail:`python3: ${h.stdout.toString().trim()}`,fixable:!1});let I=Zt(t,".claude","settings.json");if(await m(I))try{let T=await ci.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 rt=kt[2],Cs=rt.startsWith("/")?rt:Zt(t,rt),dr=await m(Cs);e.push({name:"statusLine command",status:dr?"ok":"fail",detail:dr?`ref OK: ${rt}`:`BROKEN: settings.json ref '${rt}' nh\u01B0ng file kh\xF4ng t\u1ED3n t\u1EA1i. Strip field statusLine ho\u1EB7c fix path.`,fixable:!1})}}}catch{}let K=Yn("which",["claude"]),E=K.status===0;if(e.push({name:"Claude Code CLI",status:E?"ok":"warn",detail:E?K.stdout.toString().trim():"kh\xF4ng t\xECm th\u1EA5y 'claude' tr\xEAn PATH",fixable:!1}),u){let T=await ai(t);e.push(...T)}return e}function Jl(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(`${ql(e.join(`
59
61
  `),{padding:1,borderStyle:"round"})}
60
- `)}async function $l(t){let e=0;for(let n of t)if(n.fixable&&n.fix)try{await n.fix(),s.success(`Fixed: ${n.name}`),e+=1}catch(r){s.error(`Failed to fix ${n.name}: ${r instanceof Error?r.message:String(r)}`)}e===0&&s.dim("Kh\xF4ng c\xF3 g\xEC \u0111\u1EC3 fix t\u1EF1 \u0111\u1ED9ng.")}k();import{promises as Zo}from"fs";import{join as Qt}from"path";import{confirm as Ll}from"@inquirer/prompts";import jl from"boxen";import{spawnSync as Rl}from"child_process";f();var Wo=300*1e3,qo="gitnexus",D=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=r}};function _l(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new D("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${qo} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new D("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new D("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function zo(){s.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Rl("npm",["install","-g",qo],{stdio:["inherit","inherit","pipe"],timeout:Wo,encoding:"utf8"});if(t.signal==="SIGTERM")throw new D("timeout",`npm install timeout sau ${Wo/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),_l(t.status,t.stderr||"");ce();let e=ut();if(!e.installed||!e.path)throw new D("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `gitnexus` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return s.success(`\u0110\xE3 c\xE0i GitNexus${e.version?` v${e.version}`:""} t\u1EA1i ${e.path}`),{version:e.version,path:e.path}}k();f();import{promises as Xo}from"fs";import{homedir as Il}from"os";import{join as Nl}from"path";var Jo=384,Yo={command:"gitnexus",args:["mcp"]};function Ol(){return Nl(Il(),".claude","mcp_servers.json")}function Gl(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 Ml(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await Xo.copyFile(t,n),n}async function Qo(){let t=Ol(),e={},n=!1;if(await m(t)){n=!0;try{e=await b(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&&Gl(r,Yo))return s.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let o;n&&(o=await Ml(t),s.dim(`Backup ${t} \u2192 ${o}`));let i={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:Yo}};await S(t,i,Jo);try{await Xo.chmod(t,Jo)}catch{}return s.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:o}}f();async function Ul(){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(`${jl(t.join(`
62
+ `)}async function Yl(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 wi}from"fs";import{join as te}from"path";import{confirm as ou}from"@inquirer/prompts";import iu from"boxen";import{spawnSync as Xl}from"child_process";f();var ui=300*1e3,pi="gitnexus",H=class extends Error{reason;exitCode;constructor(e,n,r=null){super(n),this.name="InstallGitnexusError",this.reason=e,this.exitCode=r}};function Ql(t,e){let n=e.toLowerCase();return n.includes("eacces")||n.includes("permission denied")?new H("permission-denied",`npm install -g c\u1EA7n quy\u1EC1n. Th\u1EED: sudo npm install -g ${pi} ho\u1EB7c fix npm prefix (npm config set prefix ~/.npm-global).`,t):n.includes("enospc")||n.includes("no space")?new H("disk-full","\u0110\u0129a \u0111\u1EA7y. Free disk space r\u1ED3i th\u1EED l\u1EA1i.",t):new H("generic",`npm install th\u1EA5t b\u1EA1i (exit ${t??"null"}). Xem log npm ph\xEDa tr\xEAn.`,t)}function mi(){s.info("\u0110ang c\xE0i GitNexus qua npm (c\xF3 th\u1EC3 m\u1EA5t 1-2 ph\xFAt)...");let t=Xl("npm",["install","-g",pi],{stdio:["inherit","inherit","pipe"],timeout:ui,encoding:"utf8"});if(t.signal==="SIGTERM")throw new H("timeout",`npm install timeout sau ${ui/1e3}s. Check m\u1EA1ng r\u1ED3i th\u1EED l\u1EA1i.`,null);if(t.status!==0)throw t.stderr&&process.stderr.write(t.stderr),Ql(t.status,t.stderr||"");pe();let e=ut();if(!e.installed||!e.path)throw new H("binary-not-in-path","npm c\xE0i xong nh\u01B0ng `gitnexus` kh\xF4ng trong PATH. Reload shell (source ~/.zshrc) ho\u1EB7c th\xEAm npm global bin v\xE0o PATH.",null);return 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 fi}from"fs";import{homedir as Zl}from"os";import{join as tu}from"path";var di=384,gi={command:"gitnexus",args:["mcp"]};function eu(){return tu(Zl(),".claude","mcp_servers.json")}function nu(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 ru(t){let e=new Date().toISOString().replace(/[:.]/g,"-"),n=`${t}.avatar-backup-${e}`;return await fi.copyFile(t,n),n}async function hi(){let t=eu(),e={},n=!1;if(await m(t)){n=!0;try{e=await b(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&&nu(r,gi))return s.dim(`MCP entry gitnexus \u0111\xE3 \u0111\xFAng t\u1EA1i ${t} (no-op)`),{path:t,wasUpdated:!1};let o;n&&(o=await ru(t),s.dim(`Backup ${t} \u2192 ${o}`));let i={...e,mcp_servers:{...e.mcp_servers||{},gitnexus:gi}};await C(t,i,di);try{await fi.chmod(t,di)}catch{}return s.success(`Registered MCP server: gitnexus \u2192 ${t}`),{path:t,wasUpdated:!0,backup:o}}f();async function su(){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(`${iu(t.join(`
61
63
  `),{padding:1,borderStyle:"round",borderColor:"cyan"})}
62
- `),await Ll({message:"C\xE0i GitNexus global?",default:!0})}async function Dl(){for(;;)try{return zo(),!0}catch(t){let e=t instanceof Error?t.message:String(t),n=t instanceof D&&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 G({taskName:"C\xE0i GitNexus qua npm",reason:e,allowSkip:!0,hint:n});if(r==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i GitNexus.");if(r==="skip")return!1}}async function Hl(t){for(;;)try{return Tt(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),r=e instanceof W&&e.reason==="missing-output"?"Repo c\xF3 th\u1EC3 empty ho\u1EB7c gitnexus version mismatch. Check `gitnexus --version`.":"Network glitch? Retry th\u01B0\u1EDDng work.",o=await G({taskName:"GitNexus analyze workspace",reason:n,allowSkip:!0,hint:r});if(o==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc GitNexus analyze.");if(o==="skip")return!1}}async function Ce(t){let e={ok:!1,installed:!1,analyzed:!1,wikiGenerated:!1,mcpRegistered:!1};try{s.info("=== Phase 10: GitNexus Setup ===");let n=ut();if(!n.installed){if(!await Ul())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 Dl())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(ce(),n=ut(),!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{Yr()}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 Hl(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 pe(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 Qo();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 A)throw n;let r=n instanceof Error?n.message:String(n);return s.warn(`GitNexus setup th\u1EA5t b\u1EA1i: ${r}`),s.dim("Workspace v\u1EABn s\u1EB5n s\xE0ng. Setup sau qua `avatar gitnexus install`."),await v("gitnexus_setup",`result=failed,error=${r.slice(0,200)}`),e.reason=r,e}}f();function Wn(){let t=process.cwd(),e=St(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 ou({message:"C\xE0i GitNexus global?",default:!0})}async function au(){for(;;)try{return mi(),!0}catch(t){let e=t instanceof Error?t.message:String(t),n=t instanceof H&&t.reason==="permission-denied"?"Th\u1EED l\u1EA1i v\u1EDBi sudo, ho\u1EB7c fix npm prefix: npm config set prefix ~/.npm-global":"Check log npm ph\xEDa tr\xEAn + th\u1EED l\u1EA1i.",r=await L({taskName:"C\xE0i GitNexus qua npm",reason:e,allowSkip:!0,hint:n});if(r==="abort")throw new S("User abort t\u1EA1i b\u01B0\u1EDBc c\xE0i GitNexus.");if(r==="skip")return!1}}async function cu(t){for(;;)try{return Et(t),!0}catch(e){let n=e instanceof Error?e.message:String(e),r=e instanceof q&&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 S("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=ut();if(!n.installed){if(!await su())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 au())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(pe(),n=ut(),!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{co()}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 cu(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 ge(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 hi();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 S)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 Xn(){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.
63
65
  Avatar workspace c\u1EA7n c\xF3: .claude/ + CLAUDE.md + src/ (ho\u1EB7c .gitmodules).
64
66
  B\u1EA1n \u0111ang \u1EDF: ${t}
65
- 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 Kl(){let t=Wn(),e=await Ce({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 Fl(t){try{let e=Qt(t,".git","HEAD"),n=(await Zo.readFile(e,"utf8")).trim();if(n.startsWith("ref:")){let r=n.slice(4).trim();try{return(await Zo.readFile(Qt(t,".git",r),"utf8")).trim()}catch{return null}}return n}catch{return null}}async function Vl(){let t=Wn(),{readReposManifest:e}=await Promise.resolve().then(()=>(V(),it)),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 b(i),c=!1;if(a.lastCommit){let u=await Fl(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(`
66
- Stale \u2192 'avatar gitnexus analyze <name>' \u0111\u1EC3 c\u1EADp nh\u1EADt.`)}async function Bl(t,e){let n=Wn(),{readReposManifest:r}=await Promise.resolve().then(()=>(V(),it)),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{Tt(l),a+=1}catch(u){s.warn(` ! ${c} fail: ${u.message}`)}}s.success(`Index xong ${a}/${i.length} repo. Xem: avatar gitnexus status`)}function ti(t){let e=t.command("gitnexus").description("Qu\u1EA3n l\xFD GitNexus code intelligence (M10)");e.command("install").description("C\xE0i + setup GitNexus cho workspace hi\u1EC7n t\u1EA1i").action(async()=>{await Kl()}),e.command("status").description("Show index info + staleness warning").action(async()=>{await Vl()}),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 Bl(n,r)})}import{resolve as Iu}from"path";import{input as Nu}from"@inquirer/prompts";import qn from"chalk";var Se=[" \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2557 \u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2557 \u2588\u2588\u2588\u2588\u2588\u2588\u2557 ","\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u255A\u2550\u2550\u2588\u2588\u2554\u2550\u2550\u255D\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557","\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551\u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2588\u2588\u2588\u2588\u2588\u2551\u2588\u2588\u2588\u2588\u2588\u2588\u2554\u255D","\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551\u255A\u2588\u2588\u2557 \u2588\u2588\u2554\u255D\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2551\u2588\u2588\u2554\u2550\u2550\u2588\u2588\u2557","\u2588\u2588\u2551 \u2588\u2588\u2551 \u255A\u2588\u2588\u2588\u2588\u2554\u255D \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551 \u2588\u2588\u2551\u2588\u2588\u2551 \u2588\u2588\u2551","\u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u2550\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D \u255A\u2550\u255D\u255A\u2550\u255D \u255A\u2550\u255D"],Te=[[217,79,30],[200,70,80],[170,70,140],[125,88,217]];function zn(t,e,n){return Math.round(t+(e-t)*n)}function Wl(t){let n=Math.max(0,Math.min(1,t))*(Te.length-1),r=Math.floor(n),o=Math.min(Te.length-1,r+1),i=n-r,a=Te[r],c=Te[o];return[zn(a[0],c[0],i),zn(a[1],c[1],i),zn(a[2],c[2],i)]}function Jn(t){if(!((process.stdout.isTTY??!1)&&qn.level>0))return[...Se,...t?.tagline?["",t.tagline]:[]].join(`
67
- `);let r=Se.map((o,i)=>{let a=Se.length===1?0:i/(Se.length-1),[c,l,u]=Wl(a);return qn.rgb(c,l,u).bold(o)});return t?.tagline&&(r.push(""),r.push(qn.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 lu(){let t=Xn(),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 uu(t){try{let e=te(t,".git","HEAD"),n=(await wi.readFile(e,"utf8")).trim();if(n.startsWith("ref:")){let r=n.slice(4).trim();try{return(await wi.readFile(te(t,".git",r),"utf8")).trim()}catch{return null}}return n}catch{return null}}async function pu(){let t=Xn(),{readReposManifest:e}=await Promise.resolve().then(()=>(B(),it)),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=te(t,"src",r.name),i=te(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 b(i),c=!1;if(a.lastCommit){let u=await uu(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 mu(t,e){let n=Xn(),{readReposManifest:r}=await Promise.resolve().then(()=>(B(),it)),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=te(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 ki(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 lu()}),e.command("status").description("Show index info + staleness warning").action(async()=>{await pu()}),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 mu(n,r)})}import{resolve as Zu}from"path";import{input as tp}from"@inquirer/prompts";import Qn 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 Zn(t,e,n){return Math.round(t+(e-t)*n)}function du(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[Zn(a[0],c[0],i),Zn(a[1],c[1],i),Zn(a[2],c[2],i)]}function tr(t){if(!((process.stdout.isTTY??!1)&&Qn.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]=du(a);return Qn.rgb(c,l,u).bold(o)});return t?.tagline&&(r.push(""),r.push(Qn.dim(t.tagline))),r.join(`
68
70
  `)}function $t(t){process.stdout.write(`
69
- ${Jn(t)}
71
+ ${tr(t)}
70
72
 
71
- `)}import{readdirSync as zh}from"fs";import{select as Yh}from"@inquirer/prompts";import{simpleGit as Qh}from"simple-git";import{existsSync as fh,statSync as hh}from"fs";import{join as kh}from"path";import{simpleGit as vh}from"simple-git";import{existsSync as Ch}from"fs";import{join as Th}from"path";import{readFileSync as $h}from"fs";import{dirname as ql,join as Pe}from"path";import{fileURLToPath as zl}from"url";var Ee=ql(zl(import.meta.url)),Ih=[Pe(Ee,"templates","gitignore"),Pe(Ee,"..","templates","gitignore"),Pe(Ee,"..","..","src","templates","gitignore"),Pe(Ee,"..","src","templates","gitignore")],Yn="# === avatar ===",$e="# === /avatar ===";f();import{existsSync as Gh,readFileSync as Mh,writeFileSync as Lh}from"fs";import{join as Uh}from"path";var Re=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};import{join as si}from"path";k();import{join as ei}from"path";var Jl=".avatar-pack-manifest.json",Yl=".pack-version";function ni(t){return ei(t,Jl)}async function _e(t,e){await S(ni(t),e)}async function Xl(t){let e=ni(t);if(!await m(e))return null;try{return await b(e)}catch{return null}}async function Ie(t){let e=await Xl(t);if(e?.version)return e.version;let n=ei(t,Yl);if(await m(n)){let r=(await O(n)).trim();if(r)return r}return null}k();import{promises as Ne}from"fs";import{join as ri}from"path";var Ql="https://qpdbkewtpkkbcrptnlos.supabase.co/functions/v1/get-pack",oi=2e4;function Zl(){let t=process.env.AVATAR_GET_PACK_ENDPOINT??Ql,e;try{e=new URL(t)}catch{throw new H(`AVATAR_GET_PACK_ENDPOINT kh\xF4ng ph\u1EA3i URL h\u1EE3p l\u1EC7: ${t}`)}if(e.protocol!=="https:")throw new H(`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 ht=class extends Error{constructor(e){super(e),this.name="InvalidIdTokenError"}},z=class extends Error{constructor(e){super(e),this.name="PackNetworkError"}},H=class extends Error{constructor(e){super(e),this.name="PackParseError"}};async function Oe(t,e){let n,r=new AbortController,o=setTimeout(()=>r.abort(),oi);try{n=await fetch(Zl(),{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 z(`get-pack qu\xE1 ${oi/1e3}s kh\xF4ng ph\u1EA3n h\u1ED3i. Th\u1EED l\u1EA1i.`):new z(`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 ii(n);throw new ht(`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 z(`Supabase get-pack l\u1ED7i server (${n.status}). Th\u1EED l\u1EA1i sau.`);if(!n.ok){let a=await ii(n);throw new H(`get-pack tr\u1EA3 ${n.status}: ${a}`)}let i;try{i=await n.json()}catch{throw new H("get-pack tr\u1EA3 response kh\xF4ng ph\u1EA3i JSON.")}if(i.error)throw new H(`get-pack b\xE1o l\u1ED7i: ${i.error}`);if(!i.url||!i.object)throw new H("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=ri(e,"..");await x(n);let r=ri(n,`.pack-download-${process.pid}.tar.gz`),o=`${e}.new-${process.pid}`;try{try{await We(t,r)}catch(c){throw new z(`T\u1EA3i tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}await ot(o);try{await qe(r,o)}catch(c){throw new H(`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 Ne.rename(e,i);try{await Ne.rename(o,e)}catch(c){throw a&&await Ne.rename(i,e).catch(()=>{}),c}a&&await ot(i).catch(()=>{})}finally{await Ne.rm(r,{force:!0}).catch(()=>{}),await ot(o).catch(()=>{})}}async function ii(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 J=".claude/pack",Rt=class extends Error{constructor(e){super(e),this.name="TeamPackAccessAbortedError"}};async function ai(t,e){let n=await fe(),r=await Oe(n,e),o=si(t,J);await Ge(r.url,o);let i=e??tu(r.object);return await _e(o,{version:i,downloadedObject:r.object,extractedAt:new Date().toISOString()}),{pinnedTag:i}}function tu(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function Me(t){let e=si(t,J);return await Ie(e)??"(unknown)"}f();k();import{join as je,relative as au,resolve as fi}from"path";import{input as wi,select as cu}from"@inquirer/prompts";import lu from"boxen";f();import eu from"boxen";var ci=[{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 li(){let t=Math.max(...ci.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=ci.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(`
72
- `);return eu(i,{padding:1,borderStyle:"round",borderColor:"cyan"})}f();import nu from"boxen";var ui=[{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"}],Xn=[125,88,217],ru=[205,95,130],pi=[150,120,200];function mi(){return(process.stdout.isTTY??!1)&&p.level>0}function _t(t,e,n=!1){if(!mi())return e;let r=p.rgb(t[0],t[1],t[2]);return n?r.bold(e):r(e)}function Qn(t){return mi()?p.dim(t):t}function ou(){let t=[];return t.push(`${_t(Xn,"\u25C6",!0)} ${_t(Xn,"AVATAR",!0)} ${Qn("\u2014 l\xF5i \u0111i\u1EC1u ph\u1ED1i")}`),t.push(_t(pi,"\u2502")),ui.forEach((e,n)=>{let r=n===ui.length-1,o=_t(pi,r?"\u2514\u2500\u2500":"\u251C\u2500\u2500");t.push(`${o} ${_t(ru,e.name,!0)} ${Qn(e.role)}`)}),t.join(`
73
- `)}function di(){let t=_t(Xn,"Ki\u1EBFn tr\xFAc Avatar \u2014 8 kh\u1ED1i quanh m\u1ED9t l\xF5i",!0),e=Qn("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 Aw}from"fs";import{select as Sw}from"@inquirer/prompts";import{simpleGit as Ew}from"simple-git";import{existsSync as Vh,statSync as Bh}from"fs";import{join as qh}from"path";import{simpleGit as Yh}from"simple-git";import{existsSync as Zh}from"fs";import{join as ew}from"path";import{readFileSync as ow}from"fs";import{dirname as gu,join as Re}from"path";import{fileURLToPath as fu}from"url";var $e=gu(fu(import.meta.url)),aw=[Re($e,"templates","gitignore"),Re($e,"..","templates","gitignore"),Re($e,"..","..","src","templates","gitignore"),Re($e,"..","src","templates","gitignore")],er="# === avatar ===",_e="# === /avatar ===";f();import{existsSync as uw,readFileSync as pw,writeFileSync as mw}from"fs";import{join as gw}from"path";var Ie=class extends Error{constructor(e){super(e),this.name="InitAbortedByUserError"}};import{join as Ci}from"path";w();import{join as yi}from"path";var hu=".avatar-pack-manifest.json",wu=".pack-version";function bi(t){return yi(t,hu)}async function Oe(t,e){await C(bi(t),e)}async function ku(t){let e=bi(t);if(!await m(e))return null;try{return await b(e)}catch{return null}}async function Ne(t){let e=await ku(t);if(e?.version)return e.version;let n=yi(t,wu);if(await m(n)){let r=(await M(n)).trim();if(r)return r}return null}w();import{promises as Me}from"fs";import{join as vi}from"path";var yu="https://qpdbkewtpkkbcrptnlos.supabase.co/functions/v1/get-pack",xi=2e4;function bu(){let t=process.env.AVATAR_GET_PACK_ENDPOINT??yu,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 ht=class extends Error{constructor(e){super(e),this.name="InvalidIdTokenError"}},J=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(),xi);try{n=await fetch(bu(),{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 J(`get-pack qu\xE1 ${xi/1e3}s kh\xF4ng ph\u1EA3n h\u1ED3i. Th\u1EED l\u1EA1i.`):new J(`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 Ai(n);throw new ht(`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 J(`Supabase get-pack l\u1ED7i server (${n.status}). Th\u1EED l\u1EA1i sau.`);if(!n.ok){let a=await Ai(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=vi(e,"..");await A(n);let r=vi(n,`.pack-download-${process.pid}.tar.gz`),o=`${e}.new-${process.pid}`;try{try{await ze(t,r)}catch(c){throw new J(`T\u1EA3i tarball th\u1EA5t b\u1EA1i: ${c instanceof Error?c.message:c}`)}await ot(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 Me.rename(e,i);try{await Me.rename(o,e)}catch(c){throw a&&await Me.rename(i,e).catch(()=>{}),c}a&&await ot(i).catch(()=>{})}finally{await Me.rm(r,{force:!0}).catch(()=>{}),await ot(o).catch(()=>{})}}async function Ai(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 Y=".claude/pack",_t=class extends Error{constructor(e){super(e),this.name="TeamPackAccessAbortedError"}};async function Si(t,e){let n=await ke(),r=await Le(n,e),o=Ci(t,Y);await Ge(r.url,o);let i=e??vu(r.object);return await Oe(o,{version:i,downloadedObject:r.object,extractedAt:new Date().toISOString()}),{pinnedTag:i}}function vu(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function De(t){let e=Ci(t,Y);return await Ne(e)??"(unknown)"}f();w();import{join as je,relative as Pu,resolve as Oi}from"path";import{input as Mi,select as Ru}from"@inquirer/prompts";import $u from"boxen";f();import xu from"boxen";var Ti=[{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 Ei(){let t=Math.max(...Ti.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=Ti.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 xu(i,{padding:1,borderStyle:"round",borderColor:"cyan"})}f();import Au from"boxen";var Pi=[{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"}],nr=[125,88,217],Cu=[205,95,130],Ri=[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 rr(t){return $i()?p.dim(t):t}function Su(){let t=[];return t.push(`${It(nr,"\u25C6",!0)} ${It(nr,"AVATAR",!0)} ${rr("\u2014 l\xF5i \u0111i\u1EC1u ph\u1ED1i")}`),t.push(It(Ri,"\u2502")),Pi.forEach((e,n)=>{let r=n===Pi.length-1,o=It(Ri,r?"\u2514\u2500\u2500":"\u251C\u2500\u2500");t.push(`${o} ${It(Cu,e.name,!0)} ${rr(e.role)}`)}),t.join(`
75
+ `)}function _i(){let t=It(nr,"Ki\u1EBFn tr\xFAc Avatar \u2014 8 kh\u1ED1i quanh m\u1ED9t l\xF5i",!0),e=rr("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}
74
76
 
75
- ${ou()}
77
+ ${Su()}
76
78
 
77
79
  ${e}`;process.stdout.write(`
78
- ${nu(n,{padding:1,borderStyle:"round",borderColor:"magenta",textAlignment:"left"})}
79
- `)}f();k();Ot();import{readdir as iu}from"fs/promises";import{join as su}from"path";async function Le(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 gi(t,e,n=10){for(let r=2;r<n;r++){let o=su(t,`${e}-${r}`);if(await Le(o))return o}return null}function hi(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}".
80
- Y\xEAu c\u1EA7u: ch\u1EC9 ch\u1EEF/s\u1ED1/dash/underscore/dot, kh\xF4ng '/', '\\', '..'.`);let n=fi(je(t,e)),r=fi(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 ki(t,e,n){hi(t,e);let r=je(t,e);if(await Le(r))return r;for(s.warn(`Workspace path "${r}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let o=await gi(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 cu({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:i});if(a==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc resolve workspace path. Ch\u1EA1y l\u1EA1i v\u1EDBi --workspace-name kh\xE1c.");if(a==="use-alt"&&o)return o;let c=await wi({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}});hi(t,c.trim());let l=je(t,c.trim());if(await Le(l))return l;s.warn(`"${l}" c\u0169ng \u0111\xE3 c\xF3 n\u1ED9i dung. Th\u1EED t\xEAn kh\xE1c.`)}}async function yi(t){return await wi({message:"Team owner email:",default:t})}async function bi(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 uu(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 pu(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 vi(t,e=null,n=null){let r=[`${p.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${au(process.cwd(),t)||t}`,uu(e),pu(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
+ ${Au(n,{padding:1,borderStyle:"round",borderColor:"magenta",textAlignment:"left"})}
81
+ `)}f();w();bt();import{readdir as Tu}from"fs/promises";import{join as Eu}from"path";async function Ue(t){if(!await m(t))return!0;try{return(await Tu(t)).filter(r=>!r.startsWith(".")&&r!=="Thumbs.db").length===0}catch{return!1}}async function Ii(t,e,n=10){for(let r=2;r<n;r++){let o=Eu(t,`${e}-${r}`);if(await Ue(o))return o}return null}function Ni(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(je(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 Li(t,e,n){Ni(t,e);let r=je(t,e);if(await Ue(r))return r;for(s.warn(`Workspace path "${r}" \u0111\xE3 c\xF3 n\u1ED9i dung.`);;){let o=await Ii(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 Ru({message:"C\xE1ch x\u1EED l\xFD workspace path conflict?",choices:i});if(a==="abort")throw new S("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 Mi({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}});Ni(t,c.trim());let l=je(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 Gi(t){return await Mi({message:"Team owner email:",default:t})}async function Di(t){try{await A(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 _u(t){if(t===null)return` ${p.yellow("AI:")} skipped \xB7 ${p.cyan("avatar ai setup")} \u0111\u1EC3 config sau`;if(t.ok){let e=t.model?` \xB7 model=${t.model}`:"";return` ${p.green("AI:")} ready \xB7 ${t.provider}${e}`}return` ${p.yellow("AI:")} failed (${t.reason.slice(0,60)}) \xB7 th\u1EED ${p.cyan("avatar ai setup")}`}function Iu(t){if(t===null)return` ${p.yellow("GitNexus:")} skipped \xB7 ${p.cyan("avatar gitnexus install")} \u0111\u1EC3 setup sau`;if(t.ok){let e=["ready"];return t.analyzed&&e.push("indexed"),t.wikiGenerated&&e.push("wiki"),t.mcpRegistered&&e.push("mcp"),` ${p.green("GitNexus:")} ${e.join(" \xB7 ")}`}return` ${p.yellow("GitNexus:")} skipped (${(t.reason??"unknown").slice(0,40)}) \xB7 th\u1EED ${p.cyan("avatar gitnexus install")}`}async function Ui(t,e=null,n=null){let r=[`${p.green("\u2713")} Workspace s\u1EB5n s\xE0ng: ${Pu(process.cwd(),t)||t}`,_u(e),Iu(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(`${$u(r.join(`
81
83
  `),{padding:1,borderStyle:"round"})}
82
- `);let o=je(t,J);await m(o)&&process.stdout.write(`
83
- ${li()}
84
- `),di()}import mu from"boxen";import du from"open";Vt();f();function xi(t){t.command("login").description("\u0110\u0103ng nh\u1EADp Google SSO (workspace @nal.vn)").option("--reset","X\xF3a credential c\u0169 v\xE0 \u0111\u0103ng nh\u1EADp l\u1EA1i").action(async e=>{try{await Zn(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Zn(t){if($t({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await ge(),await v("login_reset");else{let g=await L();if(g&&!mt(g)){s.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=B("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await Cn(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let r=$n(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(`
85
- `);process.stdout.write(`${mu(o,{padding:1,borderStyle:"round"})}
86
- `),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=B("\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 gu(a);try{if(l=await Sn(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=Tn(l.id_token);try{Pn(u)}catch(g){throw await Ft(l.access_token),g}let d=En(l,u);await Nn(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 ${pt} (chmod 600)`)}function gu(t){return new Promise(e=>setTimeout(e,t))}import{join as He}from"path";f();var Ai=3;async function Ci(t,e,n,r=!1){let o=0;for(;;)try{return{pinnedTag:(await ai(t,e)).pinnedTag,skipped:!1}}catch(i){if(i instanceof Rt)throw i;if(i instanceof ht){s.error(i.message);let l=await G({taskName:"T\u1EA3i team-ai-pack (x\xE1c th\u1EF1c NAL)",reason:i.message,allowSkip:!0,hint:"Ch\u1EA1y 'avatar login' v\u1EDBi t\xE0i kho\u1EA3n @nal.vn r\u1ED3i th\u1EED l\u1EA1i. Ho\u1EB7c skip + 'avatar sync' sau."});if(l==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc x\xE1c th\u1EF1c t\u1EA3i pack.");if(l==="skip")return s.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};continue}if(i instanceof z&&o<Ai){o+=1,s.warn(`T\u1EA3i pack l\u1ED7i m\u1EA1ng (l\u1EA7n ${o}/${Ai}): ${i.message}`),await fu(1e3*o);continue}let a=i instanceof Error?i.message:String(i),c=await G({taskName:"T\u1EA3i team-ai-pack tarball",reason:a,allowSkip:!0,hint:"Network glitch? Retry th\u01B0\u1EDDng work. Skip \u2192 d\xF9ng `avatar sync` sau."});if(c==="abort")throw new A("User abort t\u1EA1i b\u01B0\u1EDBc t\u1EA3i team-ai-pack.");if(c==="skip")return s.warn("Skip team-ai-pack. Workspace d\xF9ng \u0111\u01B0\u1EE3c nh\u01B0ng ch\u01B0a c\xF3 pack. Pull sau qua `avatar sync`."),{pinnedTag:null,skipped:!0};o=0}}function fu(t){return new Promise(e=>setTimeout(e,t))}k();Ot();import{readFileSync as hu}from"fs";import{dirname as wu,resolve as ku}from"path";import{fileURLToPath as yu}from"url";var It=null;function M(){if(It!==null)return It;let t=wu(yu(import.meta.url));for(let e=0;e<5;e++){let n=ku(t,...Array(e).fill(".."),"package.json");try{let r=hu(n,"utf8"),o=JSON.parse(r);if(o.name==="@nalvietnam/avatar-cli"&&typeof o.version=="string")return It=o.version,It}catch{}}return It="unknown",It}import{confirm as Cu}from"@inquirer/prompts";k();import{promises as bu}from"fs";import{join as tr}from"path";var vu=[".claude","pack","tools"];function Si(t){return tr(t,...vu)}async function er(t){let e=Si(t);if(!await m(e))return[];let n=await bu.readdir(e,{withFileTypes:!0}),r=[];for(let o of n)o.isDirectory()&&await m(tr(e,o.name,"tool.json"))&&r.push(o.name);return r.sort()}async function Ti(t){let e=tr(Si(t),"defaults.json");if(!await m(e))return[];try{return(await b(e)).defaultTools??[]}catch{return[]}}async function Zt(t){let[e,n]=await Promise.all([er(t),Et(t)]);return{available:e,enabled:n}}f();import{spawnSync as xu}from"child_process";f();function Au(t){if(!t)return;let e=xu(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 Pi(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 wt(t,e,n={}){let r=await Yt(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;Au(r.requires?.runtime);let o=await xe(t,r);return Pi(e,o),await Ae(t,e,{enabled:!0,version:r.version}),!0}async function nr(t,e){let n=await Yt(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 Ae(t,e,{enabled:!1,version:i}),!1}let r=await Do(t,n);return Pi(e,r),await Ae(t,e,{enabled:!1,version:n.version}),!0}var Su={"prompt-scoring":"prompt scoring"};async function Ei(t,e={}){let n=await Ti(t);if(n.length!==0)for(let r of n){let o=Su[r]??r,i;if(e.autoYes?i=!0:i=await Cu({message:`Do you want to setup ${o} (y/n)`,default:!0}),!i){s.dim(` Skip tool '${r}' (user opt-out).`);continue}await wt(t,r,{silent:!0})}}import{promises as Ue}from"fs";import{dirname as rr,join as Ri,relative as or}from"path";import{promises as Tu}from"fs";function Pu(){let t=new Date,e=n=>n.toString().padStart(2,"0");return`${t.getFullYear()}-${e(t.getMonth()+1)}-${e(t.getDate())}-${e(t.getHours())}${e(t.getMinutes())}`}async function $i(t){let e=`${t}.backup-${Pu()}`;return await Tu.rename(t,e),e}k();var ir=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function Eu(t){try{return(await Ue.lstat(t)).isSymbolicLink()}catch{return!1}}async function $u(t,e,n){let r=or(rr(e),e)||e;if(!await m(t))return{dir:r,action:"source-missing"};if(await m(e))if(await Eu(e))await Ue.unlink(e);else if(n){let i=await $i(e),a=or(rr(e),t);return await Ue.symlink(a,e),{dir:r,action:"backed-up-and-linked",backupPath:i}}else return{dir:r,action:"skipped-conflict"};let o=or(rr(e),t);return await Ue.symlink(o,e),{dir:r,action:"created"}}async function De(t,e,n){let r=[];for(let o of ir){let i=Ri(t,o),a=Ri(e,o);r.push(await $u(i,a,n))}return r}f();function Ru(t){return t?`
84
+ `);let o=je(t,Y);await m(o)&&process.stdout.write(`
85
+ ${Ei()}
86
+ `),_i()}import Ou from"boxen";import Nu from"open";Vt();f();function ji(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 or(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function or(t){if($t({tagline:"\u0110\u0103ng nh\u1EADp Google SSO \xB7 workspace @nal.vn"}),t.reset)await we(),await v("login_reset");else{let g=await D();if(g&&!dt(g)){s.success(`\u0110\xE3 \u0111\u0103ng nh\u1EADp: ${g.email}`);return}}let e=W("\u0110ang y\xEAu c\u1EA7u device code t\u1EEB Google..."),n;try{n=await $n(),e.succeed("Nh\u1EADn device code")}catch(g){throw e.fail("Kh\xF4ng k\u1EBFt n\u1ED1i \u0111\u01B0\u1EE3c Google"),g}let r=Mn(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(`${Ou(o,{padding:1,borderStyle:"round"})}
88
+ `),Nu(r).catch(()=>{s.dim("(Kh\xF4ng m\u1EDF \u0111\u01B0\u1EE3c browser t\u1EF1 \u0111\u1ED9ng \u2014 copy URL \u1EDF tr\xEAn)")});let i=W("\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 Mu(a);try{if(l=await _n(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=In(l.id_token);try{On(u)}catch(g){throw await Kt(l.access_token),g}let d=Nn(l,u);await Un(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 Mu(t){return new Promise(e=>setTimeout(e,t))}import{join as Ke}from"path";f();var Hi=3;async function Fi(t,e,n,r=!1){let o=0;for(;;)try{return{pinnedTag:(await Si(t,e)).pinnedTag,skipped:!1}}catch(i){if(i instanceof _t)throw i;if(i instanceof ht){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 S("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 J&&o<Hi){o+=1,s.warn(`T\u1EA3i pack l\u1ED7i m\u1EA1ng (l\u1EA7n ${o}/${Hi}): ${i.message}`),await Lu(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 S("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 Lu(t){return new Promise(e=>setTimeout(e,t))}w();bt();import{readFileSync as Gu}from"fs";import{dirname as Du,resolve as Uu}from"path";import{fileURLToPath as ju}from"url";var Ot=null;function N(){if(Ot!==null)return Ot;let t=Du(ju(import.meta.url));for(let e=0;e<5;e++){let n=Uu(t,...Array(e).fill(".."),"package.json");try{let r=Gu(n,"utf8"),o=JSON.parse(r);if(o.name==="@nalvietnam/avatar-cli"&&typeof o.version=="string")return Ot=o.version,Ot}catch{}}return Ot="unknown",Ot}import{confirm as Bu}from"@inquirer/prompts";w();import{promises as Hu}from"fs";import{join as ir}from"path";var Fu=[".claude","pack","tools"];function Ki(t){return ir(t,...Fu)}async function sr(t){let e=Ki(t);if(!await m(e))return[];let n=await Hu.readdir(e,{withFileTypes:!0}),r=[];for(let o of n)o.isDirectory()&&await m(ir(e,o.name,"tool.json"))&&r.push(o.name);return r.sort()}async function Vi(t){let e=ir(Ki(t),"defaults.json");if(!await m(e))return[];try{return(await b(e)).defaultTools??[]}catch{return[]}}async function ee(t){let[e,n]=await Promise.all([sr(t),Rt(t)]);return{available:e,enabled:n}}f();import{spawnSync as Ku}from"child_process";f();function Vu(t){if(!t)return;let e=Ku(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 Bi(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 wt(t,e,n={}){let r=await Qt(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;Vu(r.requires?.runtime);let o=await Ce(t,r);return Bi(e,o),await Se(t,e,{enabled:!0,version:r.version}),!0}async function ar(t,e){let n=await Qt(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 Se(t,e,{enabled:!1,version:i}),!1}let r=await oi(t,n);return Bi(e,r),await Se(t,e,{enabled:!1,version:n.version}),!0}var Wu={"prompt-scoring":"prompt scoring"};async function Wi(t,e={}){let n=await Vi(t);if(n.length!==0)for(let r of n){let o=Wu[r]??r,i;if(e.autoYes?i=!0:i=await Bu({message:`Do you want to setup ${o} (y/n)`,default:!0}),!i){s.dim(` Skip tool '${r}' (user opt-out).`);continue}await wt(t,r,{silent:!0})}}import{promises as He}from"fs";import{dirname as cr,join as zi,relative as lr}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 ur=["skills","agents","commands","hooks","workflows","scripts","knowledge"];async function Ju(t){try{return(await He.lstat(t)).isSymbolicLink()}catch{return!1}}async function Yu(t,e,n){let r=lr(cr(e),e)||e;if(!await m(t))return{dir:r,action:"source-missing"};if(await m(e))if(await Ju(e))await He.unlink(e);else if(n){let i=await qi(e),a=lr(cr(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=lr(cr(e),t);return await He.symlink(o,e),{dir:r,action:"created"}}async function Fe(t,e,n){let r=[];for(let o of ur){let i=zi(t,o),a=zi(e,o);r.push(await Yu(i,a,n))}return r}f();function Xu(t){return t?`
87
89
  ### \u{1F9E0} CODEBASE INTELLIGENCE \u2014 GitNexus
88
90
 
89
91
  Workspace c\xF3 GitNexus index t\u1EA1i \`.gitnexus/\` cung c\u1EA5p architectural awareness
@@ -111,31 +113,31 @@ Khi user c\u1EA7n regenerate wiki sau refactor l\u1EDBn \u2014 ch\u1EA1y:
111
113
  \`\`\`bash
112
114
  gitnexus wiki . --api-key <key> --base-url <url>
113
115
  \`\`\`
114
- `:""}function _i(t){return{projectName:t.projectName,projectDescription:t.projectDescription,teamOwner:t.teamOwner,avatarVersion:M(),packVersion:t.packVersion,lastScan:new Date().toISOString(),gitnexusSection:Ru(t.gitnexusReady??!1)}}async function Ii(t){await bi(t.workspacePath),await x(He(t.workspacePath,"src")),await x(He(t.workspacePath,"notes"));let e="(skipped)";if(t.skipTeamPack)s.dim("Skip team-ai-pack (--skip-team-pack).");else{B("T\u1EA3i team-ai-pack t\u1EEB Supabase...").stop();let c=await Ci(t.workspacePath,t.packVersion);e=c.pinnedTag??"(skipped)",c.skipped||s.success(`team-ai-pack: ${e}`)}let n=_i({projectName:t.workspaceName,projectDescription:t.description,teamOwner:t.teamOwner,packVersion:e});await Xe(t.workspacePath),await Qe(t.workspacePath,{avatarVersion:M(),workspaceName:t.workspaceName}),await Ze(t.workspacePath,n),await tn(t.workspacePath,n),await en(t.workspacePath,n),await nn(t.workspacePath),await _u(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 ke({workspacePath:t.workspacePath});let o=null;t.aiSkip||t.gitnexusSkip?s.dim(t.gitnexusSkip?"B\u1ECF qua GitNexus (--gitnexus-skip). Setup sau: avatar gitnexus install":"B\u1ECF qua GitNexus (auto-skip do --ai-skip)."):o=await Ce({workspacePath:t.workspacePath,skipAnalyze:!0}),await vi(t.workspacePath,r,o)}async function _u(t,e){let n=He(t,J);if(!await m(n)){s.dim("Pack ch\u01B0a c\xE0i (skip auto-sync). Ch\u1EA1y `avatar sync` sau.");return}let r=He(t,".claude");s.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings)...");let o=!1;try{let i=await De(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 be(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 Ei(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 Ni(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 Ou(e,n)}catch(r){(r instanceof Re||r instanceof Rt||r instanceof jt||r instanceof A)&&(s.dim(r.message),process.exit(0)),s.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function Ou(t,e){e.yes||$t({tagline:"Kh\u1EDFi t\u1EA1o Avatar workspace"});let n=await L();for(;!n||mt(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 Zn({})}catch(c){throw c instanceof A||s.error(`Login th\u1EA5t b\u1EA1i: ${c.message}`),c}if(n=await L(),n&&!mt(n))break;throw new A("C\u1EA7n \u0111\u0103ng nh\u1EADp @nal.vn \u0111\u1EC3 t\u1EA3i team-ai-pack. Ch\u1EA1y 'avatar login' r\u1ED3i 'avatar init' l\u1EA1i.")}let r=t??await Nu({message:"T\xEAn workspace:",validate:c=>c.trim().length>0?!0:"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng"}),o=e.teamOwner??await yi(n.email),i=Iu(e.workspaceParent??"."),a=await ki(i,r.trim(),e.force);await Ii({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})}Vt();f();function Oi(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 Gu(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Gu(t={}){let e=await L();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=B("Revoke token v\u1EDBi Google...");await Ft(e.refresh_token),r.succeed("\u0110\xE3 revoke token v\u1EDBi Google")}await ge(),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 Y(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 Ji(t){return{projectName:t.projectName,projectDescription:t.projectDescription,teamOwner:t.teamOwner,avatarVersion:N(),packVersion:t.packVersion,lastScan:new Date().toISOString(),gitnexusSection:Xu(t.gitnexusReady??!1)}}async function Yi(t){await Di(t.workspacePath),await A(Ke(t.workspacePath,"src")),await A(Ke(t.workspacePath,"notes"));let e="(skipped)";if(t.skipTeamPack)s.dim("Skip team-ai-pack (--skip-team-pack).");else{W("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 Qu(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 ve({workspacePath:t.workspacePath});let o=null;t.aiSkip||t.gitnexusSkip?s.dim(t.gitnexusSkip?"B\u1ECF qua GitNexus (--gitnexus-skip). Setup sau: avatar gitnexus install":"B\u1ECF qua GitNexus (auto-skip do --ai-skip)."):o=await Te({workspacePath:t.workspacePath,skipAnalyze:!0}),await Ui(t.workspacePath,r,o)}async function Qu(t,e){let n=Ke(t,Y);if(!await m(n)){s.dim("Pack ch\u01B0a c\xE0i (skip auto-sync). Ch\u1EA1y `avatar sync` sau.");return}let r=Ke(t,".claude");s.info("Auto-sync pack content v\xE0o .claude/ (symlinks + settings)...");let o=!1;try{let i=await Fe(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 xe(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 ep(e,n)}catch(r){(r instanceof Ie||r instanceof _t||r instanceof Dt||r instanceof S)&&(s.dim(r.message),process.exit(0)),s.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function ep(t,e){e.yes||$t({tagline:"Kh\u1EDFi t\u1EA1o Avatar workspace"});let n=await D();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 or({})}catch(c){throw c instanceof S||s.error(`Login th\u1EA5t b\u1EA1i: ${c.message}`),c}if(n=await D(),n&&!dt(n))break;throw new S("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 tp({message:"T\xEAn workspace:",validate:c=>c.trim().length>0?!0:"T\xEAn kh\xF4ng \u0111\u01B0\u1EE3c r\u1ED7ng"}),o=e.teamOwner??await Gi(n.email),i=Zu(e.workspaceParent??"."),a=await Li(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})}Vt();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 np(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function np(t={}){let e=await D();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=W("Revoke token v\u1EDBi Google...");await Kt(e.refresh_token),r.succeed("\u0110\xE3 revoke token v\u1EDBi Google")}await we(),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 X(t,e){return()=>{process.stdout.write(`${p.yellow("\u23F3")} ${p.bold(`avatar ${t}`)} \u2014 ch\u01B0a implement \u1EDF milestone hi\u1EC7n t\u1EA1i.
115
117
  `),e&&process.stdout.write(` D\u1EF1 ki\u1EBFn: ${p.cyan(e)}
116
118
  `),process.stdout.write(` Spec \u0111\xE3 c\xF3 trong avatar-cli-implementation_4.html.
117
- `),process.exit(0)}}function Gi(t){t.command("mcp-run <tool-id>",{hidden:!0}).description("[internal] Spawn MCP v\u1EDBi secrets injected (M09)").action(Y("mcp-run","Milestone 09"))}k();import{join as Mu}from"path";import Mi from"boxen";f();var Lu=".claude/pack";function Li(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 ju(process.cwd());n.json?process.stdout.write(`${JSON.stringify(r,null,2)}
118
- `):Uu(r)}catch(r){s.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function ju(t){let e=Mu(t,Lu);return await m(e)?{installed:!0,currentVersion:await Me(t).catch(()=>null)}:{installed:!1,currentVersion:null}}function Uu(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(`${Mi(r.join(`
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(X("mcp-run","Milestone 09"))}w();import{join as rp}from"path";import ts from"boxen";f();var op=".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 ip(process.cwd());n.json?process.stdout.write(`${JSON.stringify(r,null,2)}
120
+ `):sp(r)}catch(r){s.error(r instanceof Error?r.message:String(r)),process.exit(1)}})}async function ip(t){let e=rp(t,op);return await m(e)?{installed:!0,currentVersion:await De(t).catch(()=>null)}:{installed:!1,currentVersion:null}}function sp(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
121
  `),{padding:1,borderStyle:"round"})}
120
- `);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(`${Mi(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(`${ts(n.join(`
121
123
  `),{padding:1,borderStyle:"round"})}
122
- `)}function ji(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(Y("restore","Milestone 08"))}function Ui(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(Y("review","Milestone 08"))}function Di(t){t.command("scan").description("Ch\u1EA1y project scanner v\xE0 \u0111\u1EC1 xu\u1EA5t knowledge update (M06)").option("--incremental","Ch\u1EC9 scan c\xE1c file thay \u0111\u1ED5i t\u1EEB commit cu\u1ED1i").option("--full","Scan to\xE0n b\u1ED9 d\u1EF1 \xE1n (default)").option("--scanners <list>","tech-stack,conventions,architecture,domain,git-pattern").option("--quiet","Ch\u1EA1y ng\u1EA7m, \xEDt output (d\xF9ng cho git hook)").action(Y("scan","Milestone 06"))}import{promises as Fu}from"fs";import{join as Ke}from"path";import Vu from"boxen";k();k();import{promises as Du}from"fs";import{join as Hu}from"path";var Ku="_backup";async function Hi(t){let e=Hu(t,".claude",Ku);return await m(e)?(await Du.readdir(e,{withFileTypes:!0})).filter(r=>r.isDirectory()).map(r=>r.name).sort().reverse():[]}f();function Ki(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 Bu(process.cwd());e.json?process.stdout.write(`${JSON.stringify(n,null,2)}
123
- `):zu(n)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Bu(t){let e=t.split("/").filter(Boolean).pop()??"unknown",n=Ke(t,".claude");if(!await m(n))return{projectName:e,cliVersion:M(),packVersion:null,pendingCount:0,backupCount:0,techStackSummary:"(Avatar ch\u01B0a init)",hasAvatar:!1,toolsEnabled:[],toolsAvailableCount:0};let o=Ke(n,"_pending"),[i,a,c,l,u]=await Promise.all([(async()=>await m(Ke(n,"pack"))?Me(t).catch(()=>null):null)(),(async()=>await m(o)?(await Fu.readdir(o)).filter(g=>g.endsWith(".diff.md")).length:0)(),Hi(t).then(d=>d.length).catch(()=>0),Wu(n).catch(()=>"(read error)"),Zt(t).catch(()=>({available:[],enabled:[]}))]);return{projectName:e,cliVersion:M(),packVersion:i,pendingCount:a,backupCount:c,techStackSummary:l,hasAvatar:!0,toolsEnabled:u.enabled,toolsAvailableCount:u.available.length}}async function Wu(t){let e=Ke(t,"project","tech-stack.md");return await m(e)?(await O(e)).split(`
124
- `).find(o=>o.trim()&&!o.startsWith("#")&&!o.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function qu(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 zu(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:")} ${qu(t)}`];process.stdout.write(`${Vu(e.join(`
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(X("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(X("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(X("scan","Milestone 06"))}import{promises as up}from"fs";import{join as Ve}from"path";import pp from"boxen";w();w();import{promises as ap}from"fs";import{join as cp}from"path";var lp="_backup";async function is(t){let e=cp(t,".claude",lp);return await m(e)?(await ap.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 mp(process.cwd());e.json?process.stdout.write(`${JSON.stringify(n,null,2)}
125
+ `):fp(n)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function mp(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"))?De(t).catch(()=>null):null)(),(async()=>await m(o)?(await up.readdir(o)).filter(g=>g.endsWith(".diff.md")).length:0)(),is(t).then(d=>d.length).catch(()=>0),dp(n).catch(()=>"(read error)"),ee(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 dp(t){let e=Ve(t,"project","tech-stack.md");return await m(e)?(await M(e)).split(`
126
+ `).find(o=>o.trim()&&!o.startsWith("#")&&!o.startsWith(">"))?.trim()??"(empty)":"(no tech-stack.md)"}function gp(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 fp(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:")} ${gp(t)}`];process.stdout.write(`${pp(e.join(`
125
127
  `),{padding:1,borderStyle:"round"})}
126
- `)}k();import{join as Wi}from"path";k();import{join as Fi}from"path";async function Ju(t,e,n){let r=Fi(t,n),o=Fi(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 Vi(t,e,n){let r=await Ie(t)??"(ch\u01B0a c\xE0i)",o=n??"stable m\u1EDBi nh\u1EA5t (server resolve)",i=[];for(let a of ir)i.push({dir:a,status:await Ju(t,e,a)});return{currentVersion:r,targetVersion:o,mountDirStatuses:i}}f();async function Bi(t){let e=await Et(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 wt(t,n)}}f();function Yu(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function Xu(t){let e=process.cwd(),n=Wi(e,".claude"),r=Wi(e,J);if(await m(n)||(s.error(`Kh\xF4ng th\u1EA5y .claude/ \u2014 kh\xF4ng ph\u1EA3i Avatar workspace.
127
- Ch\u1EA1y 'avatar init' \u0111\u1EC3 kh\u1EDFi t\u1EA1o.`),process.exit(1)),t.dryRun){let c=await Vi(r,n,t.version);s.info(`Pack version hi\u1EC7n t\u1EA1i: ${c.currentVersion}`),s.info(`Target version: ${c.targetVersion}`),s.info(`
128
+ `)}w();import{basename as wp,join as pr}from"path";import{readFile as kp}from"fs/promises";bt();w();import{join as as}from"path";async function hp(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 ur)i.push({dir:a,status:await hp(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 wt(t,n)}}f();function yp(t){return t.replace(/^pack-/,"").replace(/\.tar\.gz$/,"")}async function bp(t,e){let n=wp(t)||"workspace",r="";try{r=(await kp(e,"utf8")).match(/^-\s*Team owner:\s*(.+)$/m)?.[1]?.trim()??""}catch{}return{projectName:n,projectDescription:"",teamOwner:r}}async function vp(t){let e=process.cwd(),n=pr(e,".claude"),r=pr(e,Y);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
130
  Mount dir statuses:`);for(let l of c.mountDirStatuses)console.log(` ${l.dir.padEnd(12)} ${l.status}`);s.info(`
129
- Dry-run done. Kh\xF4ng apply. B\u1ECF --dry-run \u0111\u1EC3 th\u1EF1c thi.`);return}let o;try{o=await fe()}catch(c){s.error(c instanceof Error?c.message:String(c)),process.exit(1)}s.info(t.version?`T\u1EA3i team-ai-pack ${t.version} t\u1EEB Supabase...`:"T\u1EA3i team-ai-pack (stable m\u1EDBi nh\u1EA5t) t\u1EEB Supabase...");let i;try{let c=await Oe(o,t.version);await Ge(c.url,r),i=t.version??Yu(c.object),await _e(r,{version:i,downloadedObject:c.object,extractedAt:new Date().toISOString()}),s.success(`\u0110\xE3 c\xE0i pack ${i}`)}catch(c){c instanceof ht?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 De(r,n,t.force===!0);Qu(a,t.force===!0),s.info("Merge pack settings.json template v\xE0o project settings.json...");try{let c=await be(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 Bi(e)}catch(c){s.warn(` ! Re-apply tools fail: ${c instanceof Error?c.message:c}.`)}s.success(`Synced team-ai-pack \u2192 ${i}.`)}function Qu(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 qi(t){t.command("sync").description("T\u1EA3i team-ai-pack m\u1EDBi nh\u1EA5t (tarball Supabase) + t\u1EA1o symlink farm").option("--force","Override .claude/<dir>/ n\u1EBFu l\xE0 real dir (backup tr\u01B0\u1EDBc)").option("--version <tag>","Pin version c\u1EE5 th\u1EC3 (vd: v0.2.0 ho\u1EB7c v0.2.0-beta.0)").option("--dry-run","Hi\u1EC3n th\u1ECB preview, kh\xF4ng apply").action(Xu)}import{resolve as Zu}from"path";import{checkbox as tp,confirm as ep}from"@inquirer/prompts";f();function zi(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 Ji(t){return t.map(e=>({name:e,value:e,checked:!1}))}function Yi(t){return[...t]}function te(t){return Zu(t.target??process.cwd())}async function Xi(t,e,n,r){if(n.all===!0||!process.stdout.isTTY)return Yi(t);let i=r==="add"?zi(t,e):Ji(t);return await tp({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 np(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 Xi(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 wt(e,i)}async function rp(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 Xi(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 ep({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 nr(e,i)}async function op(t){let e=te(t),n=await er(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 Qi(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 wt(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 nr(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 op(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 np(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 rp(n)})}import{relative as mp}from"path";import{confirm as dp}from"@inquirer/prompts";import gp from"boxen";import{cp as Fe,mkdir as Zi,writeFile as ip}from"fs/promises";import{homedir as sp}from"os";import{basename as ap,join as X}from"path";var cp=X(sp(),".avatar","uninstall-backups");async function ts(t,e,n){let r=ap(t),o=new Date().toISOString().replace(/[:.]/g,"-"),i=X(cp,`${r}-${o}`);if(await Zi(i,{recursive:!0,mode:448}),e.claudeDir&&await Fe(e.claudeDir,X(i,".claude"),{recursive:!0}),e.claudeMd&&await Fe(e.claudeMd,X(i,"CLAUDE.md")),e.postMergeHook||e.prePushHook){let c=X(i,"hooks");await Zi(c,{recursive:!0}),e.postMergeHook&&await Fe(e.postMergeHook,X(c,"post-merge")),e.prePushHook&&await Fe(e.prePushHook,X(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 ip(X(i,"manifest.json"),JSON.stringify(a,null,2),"utf8"),i}import{existsSync as lp}from"fs";import{join as Q}from"path";function Z(t){return lp(t)?t:null}function es(t){let e=Z(Q(t,".claude")),n=Z(Q(t,"CLAUDE.md")),r=Z(Q(t,".git","hooks","post-merge")),o=Z(Q(t,".git","modules","src","hooks","pre-push")),i=Z(Q(t,".gitignore")),a=Z(Q(t,".gitmodules")),c=Z(Q(t,"notes")),l=Z(Q(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 ns,rm as tt,writeFile as rs}from"fs/promises";async function os(t,e){if(t.claudeDir)if(e.keepSubmodule){let{readdir:n}=await import("fs/promises"),{join:r}=await import("path"),o=await n(t.claudeDir);for(let i of o)i!=="pack"&&await tt(r(t.claudeDir,i),{recursive:!0,force:!0})}else await tt(t.claudeDir,{recursive:!0,force:!0});t.claudeMd&&await tt(t.claudeMd,{force:!0}),e.keepHooks||(t.postMergeHook&&await tt(t.postMergeHook,{force:!0}),t.prePushHook&&await tt(t.prePushHook,{force:!0})),t.gitignorePath&&await up(t.gitignorePath),t.gitmodulesPath&&!e.keepSubmodule&&await pp(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 tt(n,{recursive:!0,force:!0})}}async function up(t){let e=await ns(t,"utf8"),n=e.indexOf(Yn),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()}
130
- ${i.trimStart()}`.trim();a.length===0?await tt(t,{force:!0}):await rs(t,`${a}
131
- `,"utf8")}async function pp(t,e){let r=(await ns(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 ke()}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??yp(c.object),await Oe(r,{version:i,downloadedObject:c.object,extractedAt:new Date().toISOString()}),s.success(`\u0110\xE3 c\xE0i pack ${i}`)}catch(c){c instanceof ht?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 Fe(r,n,t.force===!0);xp(a,t.force===!0),s.info("Merge pack settings.json template v\xE0o project settings.json...");try{let c=await xe(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 bp(e,pr(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 xp(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(vp)}import{resolve as Ap}from"path";import{checkbox as Cp,confirm as Sp}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 ne(t){return Ap(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 Cp({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 Tp(t){let e=ne(t),{available:n,enabled:r}=await ee(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 wt(e,i)}async function Ep(t){let e=ne(t),{enabled:n}=await ee(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 Sp({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 ar(e,i)}async function Pp(t){let e=ne(t),n=await sr(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=ne(r);await wt(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=ne(r);await ar(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 Pp(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 Tp(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 Ep(n)})}import{relative as Lp}from"path";import{confirm as Gp}from"@inquirer/prompts";import Dp from"boxen";import{cp as Be,mkdir as hs,writeFile as Rp}from"fs/promises";import{homedir as $p}from"os";import{basename as _p,join as Q}from"path";var Ip=Q($p(),".avatar","uninstall-backups");async function ws(t,e,n){let r=_p(t),o=new Date().toISOString().replace(/[:.]/g,"-"),i=Q(Ip,`${r}-${o}`);if(await hs(i,{recursive:!0,mode:448}),e.claudeDir&&await Be(e.claudeDir,Q(i,".claude"),{recursive:!0}),e.claudeMd&&await Be(e.claudeMd,Q(i,"CLAUDE.md")),e.postMergeHook||e.prePushHook){let c=Q(i,"hooks");await hs(c,{recursive:!0}),e.postMergeHook&&await Be(e.postMergeHook,Q(c,"post-merge")),e.prePushHook&&await Be(e.prePushHook,Q(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 Rp(Q(i,"manifest.json"),JSON.stringify(a,null,2),"utf8"),i}import{existsSync as Op}from"fs";import{join as Z}from"path";function tt(t){return Op(t)?t:null}function ks(t){let e=tt(Z(t,".claude")),n=tt(Z(t,"CLAUDE.md")),r=tt(Z(t,".git","hooks","post-merge")),o=tt(Z(t,".git","modules","src","hooks","pre-push")),i=tt(Z(t,".gitignore")),a=tt(Z(t,".gitmodules")),c=tt(Z(t,"notes")),l=tt(Z(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 et,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 et(r(t.claudeDir,i),{recursive:!0,force:!0})}else await et(t.claudeDir,{recursive:!0,force:!0});t.claudeMd&&await et(t.claudeMd,{force:!0}),e.keepHooks||(t.postMergeHook&&await et(t.postMergeHook,{force:!0}),t.prePushHook&&await et(t.prePushHook,{force:!0})),t.gitignorePath&&await Np(t.gitignorePath),t.gitmodulesPath&&!e.keepSubmodule&&await Mp(t.gitmodulesPath,".claude/pack");for(let n of[t.notesDir,t.scriptsDir]){if(!n)continue;let{readdir:r}=await import("fs/promises");(await r(n)).length===0&&await et(n,{recursive:!0,force:!0})}}async function Np(t){let e=await ys(t,"utf8"),n=e.indexOf(er),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 et(t,{force:!0}):await bs(t,`${a}
133
+ `,"utf8")}async function Mp(t,e){let r=(await ys(t,"utf8")).split(`
132
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(`
133
- `).trim();a.length===0?await tt(t,{force:!0}):await rs(t,`${a}
134
- `,"utf8")}f();function is(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 fp(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function fp(t){let e=process.cwd(),n=es(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(hp(e,n,t),t.dryRun){s.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await dp({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){s.info("\u0110\xE3 h\u1EE7y.");return}let r=null;t.noBackup||(r=await ts(e,n,M()),s.success(`Backup t\u1EA1o t\u1EA1i: ${r}`)),await os(n,{keepSubmodule:t.keepSubmodule,keepHooks:t.keepHooks}),await v("uninstall",`project=${e},backup=${r??"skipped"}`),wp(r)}function hp(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")} ${mp(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(`${gp(e.join(`
135
+ `).trim();a.length===0?await et(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 Up(e)}catch(n){s.error(n instanceof Error?n.message:String(n)),process.exit(1)}})}async function Up(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(jp(e,n,t),t.dryRun){s.dim("--dry-run: k\u1EBFt th\xFAc, kh\xF4ng x\xF3a.");return}if(!t.yes&&!await Gp({message:"Ti\u1EBFp t\u1EE5c g\u1EE1 Avatar?",default:!1})){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"}`),Hp(r)}function jp(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")} ${Lp(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 Hp(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(`${Dp(e.join(`
135
137
  `),{padding:1,borderStyle:"round"})}
136
- `)}var sr=M(),E=new kp;E.name("avatar").description("AI harness CLI for NAL Vietnam engineering").version(sr,"-v, --version","Hi\u1EC3n th\u1ECB phi\xEAn b\u1EA3n Avatar CLI").addHelpText("beforeAll",()=>`
137
- ${Jn({tagline:`v${sr} \xB7 AI harness CLI for NAL Vietnam`})}
138
+ `)}var mr=N(),P=new Fp;P.name("avatar").description("AI harness CLI for NAL Vietnam engineering").version(mr,"-v, --version","Hi\u1EC3n th\u1ECB phi\xEAn b\u1EA3n Avatar CLI").addHelpText("beforeAll",()=>`
139
+ ${tr({tagline:`v${mr} \xB7 AI harness CLI for NAL Vietnam`})}
138
140
 
139
- `);var yp=process.argv.includes("-v")||process.argv.includes("--version");yp&&($t({tagline:`v${sr} \xB7 AI harness CLI for NAL Vietnam`}),process.exit(0));xi(E);Oi(E);Ni(E);qi(E);Di(E);Ui(E);Ki(E);Bo(E);ji(E);Qi(E);Gi(E);_o(E);ti(E);Li(E);oo(E);is(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}
141
+ `);var As=process.argv[2],Kp=As==="-v"||As==="--version";Kp&&($t({tagline:`v${mr} \xB7 AI harness CLI for NAL Vietnam`}),process.exit(0));ji(P);Qi(P);Xi(P);us(P);os(P);rs(P);ss(P);li(P);ns(P);fs(P);Zi(P);Jo(P);ki(P);es(P);wo(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}
140
142
  `),process.exit(1)});
141
143
  //# sourceMappingURL=index.js.map