@nalvietnam/avatar-cli 3.0.0-beta.13 → 3.0.0-beta.15

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