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