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