@nalvietnam/avatar-cli 3.9.0 → 3.11.0

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