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